Working with a pretty gnarly problem: I'm designing a custom Win32 control, and I cannot get a scrollbar to appear in it.
I'm not going to post any code just yet: it's complicated, and I want to approach this from a general, theoretical angle first to try to resolve it. Hoping that someone who knows a lot about Win32 stuff comes along.
This is a project I've been working on for some time, and the last time I got it up it had working scrollbars. What I have now isn't that much different, but for some reason the scrollbars just won't appear.
It's actually a fairly simple control, basically a listbox with multiple columns. I want the scrollbar (horizontal only) to appear when needed, when there are more items in the control than will appear in the visible viewport.
I tried using SetScrollInfo() to set the scrollbar to a state where the scrollbar should appear (page size > total range), but nothing.
I've set the WS_HSCROLL style on creation, and I've confirmed that this style is making it through to the CREATESTRUCT, so Windows apparently knows that I want a scrollbar.
This is a purely custom control, not based on any existing Windows control; I'm registering the class and then creating it. I've tried it both in a dialog and as a stand-alone child window (using CreateWindowEx() ); doesn't work either way.
One specific question I have is the role of DefWindowProc() in a custom control like this. With standard Win32 controls, this function can take care of practically anything that needs doing, since it knows all the details of the window class. But in my case, it knows almost nothing. I'm wondering what the role of this function is in cases like this where the window class is unknown to Windows. Does it have anything to do with the creation of scrollbars? (Meaning the non-child-control type of scrollbars, the ones intrinsic to windows.)
I could create a scrollbar "by hand" (by using CreateWindowEx() ), but I don't want to do this. It's so much easier to have Windows take of all that for you and just forward the scrollbar messages to you. (I tried this, and it works, but like I say I don't want to do it that way.)
I think the earlier version was run under XP, and I'm now using Windows 7-64, so that may be the problem ...
Quote from: NoCforMe on August 27, 2022, 02:39:36 PM
One specific question I have is the role of DefWindowProc() in a custom control like this. With standard Win32 controls, this function can take care of practically anything that needs doing, since it knows all the details of the window class. But in my case, it knows almost nothing. I'm wondering what the role of this function is in cases like this where the window class is unknown to Windows. Does it have anything to do with the creation of scrollbars? (Meaning the non-child-control type of scrollbars, the ones intrinsic to windows.)
The
DefWindowProc() handles the default window messages for any window, no matter if it is a common control or a custom control or any kind of other window.
Without the relevant parts of code one can just guess what's wrong.
Assuming you call the
SetScrollInfo in the WM_SIZE with the correct values set in the
SCROLLINFO the scrollbar should appear to be visible if necessary.
Little testbed attached, pure Masm32 SDK code and less than 100 lines. It adds a scrollbar to an edit control.
You might wish to add a subclass for the edit, and catch the scrollbar messages there. The scrollbar is a child of the edit control, so dealing with it in the main WndProc will probably not work.
Maybe you missed the part where I said I don't want to create any child scrollbars; I want to use the "intrinsic" Win32 scrollbars, which aren't actually separate controls (windows) but, as Raymond Chen puts it (https://devblogs.microsoft.com/oldnewthing/20040510-00/?p=39413), "decorations" to a control.
If you are not using one of the predefined registered controls, there might be no "intrinsic" Win32 scrollbar. So you might have to add it... but I might be wrong, too.
Well, there kinda sorta is (an intrinsic scrollbar). Ackshooly, I know that there is, because I got it to work earlier. Though I don't know how or why.
Ackshooly, this is getting verrry interesting. After reading Greenhorn's reply, I started experimenting with the window proc of the control, specifically the sequence of message processing. Since he mentioned WM_SIZE as a place to call SetScrollInfo() from, I tried various sequences involving WM_SIZE.
Let me tell you, WM_SIZE is a goddamn can of worms. It's quite mysterious to me what is actually going on with that message. Yeah, I know what MSDN (https://docs.microsoft.com/en-us/windows/win32/winmsg/wm-size) has to say about it: it is "sent to a window after its size has changed".
There are a number of values sent depending on what kind of resizing is supposed to have occurred. When I captured this message, I found only two of them were being sent to my control: SIZE_RESTORED (0) and SIZE_MINIMIZED (1). I don't understand the latter, since so far as I know the control was never minimized (unless it started out life in this state). SIZE_RESTORED basically means "neither maximized nor minimized", therefore "normal" (I'm guessing).
In any case, I'm not sure I even need to mess around with WM_SIZE, since my controls are intended to be fixed-size and not resizeable. There are other ways of getting the actual control size (like GetClientRect(), or through the CREATESTRUCT passed with WM_CREATE). So maybe I don't even need to bother with this message?
I read a discussion of this online (https://groups.google.com/g/comp.os.ms-windows.programmer.win32/c/Ch6R1XCAzlI) where a programmer basically said that it's almost impossible to tell when the "real" WM_SIZE comes along, and that he had given up on this and put SetScrollInfo() in his WM_PAINT handler instead.
Oh, I almost forgot to mention the other can-o'worms aspect: setting the scroll info can then generate even more WM_SIZE messages, because if the scrollbar appears or disappears the size of the client rectangle changes, which can lead to a cascade of recursive (or at least nested) messages. So this makes it seem like WM_SIZE would be the last place you'd want to put SetScrollInfo().
Anyhow, I experimented with the placement of a WM_SIZE handler, with wildly divergent results. I've attached 3 small executables that show what happens. What's strange is that even if the WM_SIZE handler goes straight to default processing (via DefWindowProc() ), the results are very different depending on when this message is handled.
In one case (first program, mcLB_SBtest1), where I look for WM_SIZE first thing (and use DefWindowProc() to handle it), I get a blank control but I get a non-operational but perfectly visible scroll bar! Weird, huh? The second program, mcLB_SBtest2, shows the control normally (and operational) but shows a "ghost" scrollbar. In this case, the scrollbar isn't visible, but you can tell it's (sort of) there by clicking and dragging in the area where it should be, at the bottom of the control; I forward all the WM_HSCROLL messages to the parent window which displays them in the status window at the bottom. Only 2 values of this message ever come through: SB_ENDSCROLL and SB_THUMBTRACK. (The position value of the latter never changes from 0 even if you "drag" the scrollbar thumb, which of course you cannot see.) This is probably because the scrollbar isn't properly set, even though I call SetScrollbarInfo() with values that should make it both visible and active. I suspect that the trick here, if it is even possible to make this work, is to call that function at the right time ...
Try out these testbed programs to see what happens. Explanations are displayed.
So I'm really scratching my head now. Any ideas?
One question this raises for me is this: what is the proper sequence of handling messages in a window proc? This is something I've never read anything about, but given that there are obviously interactions between the different message (as proven by my little testbed programs), it seems important.
The messages I need to handle are:
- WM_CREATE
- WM_DESTROY
- WM_PAINT
- WM_ERASEBKGND
- WM_LBUTTONDOWN
- WM_LBUTTONDBLCLK
- WM_HSCROLL
That's pretty much it.
Typically you call the
DefWindowProc for messages your WndProc does NOT handle. This function proceeds the default actions for window messages.
Quote from: NoCforMe on August 28, 2022, 09:25:17 AM
One question this raises for me is this: what is the proper sequence of handling messages in a window proc? This is something I've never read anything about, but given that there are obviously interactions between the different message (as proven by my little testbed programs), it seems important.
The messages I need to handle are:
- WM_CREATE
- WM_DESTROY
- WM_PAINT
- WM_ERASEBKGND
- WM_LBUTTONDOWN
- WM_LBUTTONDBLCLK
- WM_HSCROLL
That's pretty much it.
Handle these messages and return the appropriate return value for each message. The sequence doesn't matter.
For all messages you don't handle call
DefWindowProc.
https://docs.microsoft.com/en-us/windows/win32/winmsg/using-window-procedures
Quote from: NoCforMe on August 28, 2022, 08:30:33 AM
In any case, I'm not sure I even need to mess around with WM_SIZE, since my controls are intended to be fixed-size and not resizeable. There are other ways of getting the actual control size (like GetClientRect(), or through the CREATESTRUCT passed with WM_CREATE). So maybe I don't even need to bother with this message?
The WM_SIZE is called at least once after creation and before displaying, so you can place it there, even if the size never changes.
Apply this to the columns of your control ... How to Scroll Text (https://docs.microsoft.com/en-us/windows/win32/controls/scroll-text-in-scroll-bars)
Quote from: Greenhorn on August 28, 2022, 11:01:35 AM
Quote from: NoCforMe on August 28, 2022, 09:25:17 AM
One question this raises for me is this: what is the proper sequence of handling messages in a window proc?
Handle these messages and return the appropriate return value for each message. The sequence doesn't matter.
Except that it does matter, as shown by the programs I posted here.
Quote from: jj2007 on August 28, 2022, 07:04:17 AM
If you are not using one of the predefined registered controls, there might be no "intrinsic" Win32 scrollbar. So you might have to add it... but I might be wrong, too.
At this point I cannot say definitively that you are wrong, only that you
might be wrong. Here's a static control with an "intrinsic" scrollbar. (Only thing is, it doesn't work at all. But it sure looks purty, don't it?) Interesting thing is, this is the only scrollbar I could get to show with the thumb correctly sized (about 1/10 of the scroll range), so we're halfway there ... but no cigar.
I tried this with an edit control: it may not be possible to get to work, since you can use the
ES_AUTOHSCROLL and
ES_AUTOVSCROLL styles to have the control automatically scroll, and that built-in code probably won't work with an outside attempt to interfere with it.
Try attached .exe.
Progress!
I now have a custom control with a working intrinsic scrollbar. .exe attached. It can be done.
It doesn't work correctly--the scrolling and painting code is still screwed up under construction, but it works: you can click on the scrollbar or move the thumb and it does stuff. Proof of concept at least.
(I knew it should work because I had a leftover testbed of an earlier version of the control but no source code. So what I did was fire up Olly Debug and print out the disassembly of the control's window proc. The problem was just in my SetScrollInfo() code. And I'm still trying to figure out how to figure "page size" vs. "range" to get it to work correctly. Small steps ...)
@Greenhorn, you're actually right, for the most part: it doesn't matter what order you look for messages in your message handler. (It does matter in part of my code, but that's because I'm using a scheme that allows me to have multiple controls of the same class using a lookup table; I have to treat WM_CREATE (and WM_NCCREATE) specially because the table hasn't been constructed before those messages are received.)
Quote from: NoCforMe on August 28, 2022, 08:26:46 PM
@Greenhorn, you're actually right, for the most part: it doesn't matter what order you look for messages in your message handler. (It does matter in part of my code, but that's because I'm using a scheme that allows me to have multiple controls of the same class using a lookup table; I have to treat WM_CREATE (and WM_NCCREATE) specially because the table hasn't been constructed before those messages are received.)
A lookup table ? To keep the data assigned to a control ? That's how MFC is doin' it.
You can do it easier by using the window's extra bytes to store a pointer to a structure. Custom Controls (http://www.catch22.net/tuts/win32/custom-controls)
WM_NCCREATE is preferable used because if something went wrong on creation - e.g. allocating memory for the control's data structure - you can abort the window creation by returning FALSE.
Quote from: Greenhorn on August 29, 2022, 01:03:59 AM
Quote from: NoCforMe on August 28, 2022, 08:26:46 PM
@Greenhorn, you're actually right, for the most part: it doesn't matter what order you look for messages in your message handler. (It does matter in part of my code, but that's because I'm using a scheme that allows me to have multiple controls of the same class using a lookup table; I have to treat WM_CREATE (and WM_NCCREATE) specially because the table hasn't been constructed before those messages are received.)
A lookup table ? To keep the data assigned to a control ? That's how MFC is doin' it.
You can do it easier by using the window's extra bytes to store a pointer to a structure. Custom Controls (http://www.catch22.net/tuts/win32/custom-controls)
Mmm, six of one, half a dozen of the other. I allocate a fixed-size table (to handle a maximum # of controls) and make a new entry in it for each WM_CREATEd control. For my purposes, works fine. I can see how using the window's extra bytes would better generalize the control, and if I were making a production control I'd do it that way. Easy to change if needed.
Of course, if I were in production I'd also have a better version-control system than the current one here at NoCforMe Laboratories, GmbH, which is basically a 1-level "undo" (get last backup from external drive) ...
Quote
WM_NCCREATE is preferable used because if something went wrong on creation - e.g. allocating memory for the control's data structure - you can abort the window creation by returning FALSE.
You can abort creation handling WM_CREATE as well by returning -1.
More Progress!
The little control is getting better every day. It is now actually usable. Scrolling works correctly*. In this little testbed you can actually enter items into the second box and see them appear (and the scrollbar appear at the appropriate time). You can also click on any item to select it, and double-click on any item; see the response in the status bar.
Still a lot more work to go, but I'm pretty happy at this point. (My idea is to use this in a browse-for-folder dialog that actually works the way I want it to.)
* Dragging the thumb is not implemented yet, but every other way of scrolling works (I've set "scroll by lines" to do the same thing as "scroll by pages"). Any Day Now ...
Quote from: NoCforMe on August 30, 2022, 01:44:22 PM
More Progress!
Still a lot more work to go, but I'm pretty happy at this point. (My idea is to use this in a browse-for-folder dialog ...)
I'm on the porch at the moment. (iPad post) I'll take a look at it when I go back inside.
If you're happy with it, it's all that matters.
Well, it's getting movement. Which means the control is receiving messages for it. That's a start at least. Are you also going to use a vertical scroll bar once you have mastered the 'mechanics' of how they work and how to properly implement them?
No, no vertical scrollbar. This control is meant to mimic the behavior of the listview controls used in the GetOpenFilename() and GetSaveFilename() dialogs, which only use a horizontal scrollbar. (I could certainly put in a v-bar if I wanted to.) No need for a vertical one since the height of the control is fixed.
It's more than a "start"--this is fully-functional scrolling. Notice how the content of the control scrolls? I've pretty much mastered their operation (except for thumb-tracking; that's for V. 2.0). it's not that complicated: Windows does most of the heavy lifting for you.
Quote from: NoCforMe on August 30, 2022, 03:25:56 PM
It's more than a "start"--this is fully-functional scrolling.
More than what I could do with one. I've given up on them many moons ago.
Quote from: Swordfish on August 30, 2022, 03:43:28 PM
Quote from: NoCforMe on August 30, 2022, 03:25:56 PM
It's more than a "start"--this is fully-functional scrolling.
More than what I could do with one. I've given up on them many moons ago.
You just need to understand the relationship between the "range" (the difference betwixt min and max positions) and the "page" size. After that everything falls into place.
Here's a good explanation (https://www.codeproject.com/Articles/1042516/Custom-Controls-in-Win-API-Scrolling).
Wait a minute. First you said...Quote
* Dragging the thumb is not implemented yet, but...
From this post (http://masm32.com/board/index.php?topic=10298.msg112696#msg112696), But now you say:Quote from: NoCforMe on August 30, 2022, 03:25:56 PM
It's more than a "start"--this is fully-functional scrolling.
but also from that same post:Quote (except for thumb-tracking; that's for V. 2.0)
So yes, it is only started. The only thing seem to work are the arrow buttons "<" and ">" from what I can see. Not fully functioning unless the thumb tracking also works. :biggrin: Sorry to nitpick. I'm sure when it's really finished (meaning no further adjustments needed) it will be fine.
The "page" scrolling function also works (clicking in the bar in the space between the thumb and the buttons); this is normally implemented as "scroll-by-page", and clicking on the button is "scroll-by-line". I've made them work the same (scroll by one column), the same as the listview in the GetOpenFilename() dialog.
So the only thing that's not implemented is thumb tracking. So yes, technically not complete, but pretty close. (Oh, and there's no keyboard interface yet.)
Quote from: NoCforMe on August 31, 2022, 05:53:27 AM
So yes, technically not complete, but pretty close.
:biggrin: Ok, semantics aside I think you're doing a good job. You're more persistent than I would be. After the first fail or two I usually give up.
So let me pose a li'l challenge to you (when you've got the time and inclination, of course): write a little something that incorporates a working scrollbar in some control. You can do it if you try.
Does writing an editor that uses rich edit and its embedded scroll bar count? :tongue:
If not, then I'll pass thanks.
I still may, one day...
I remember the first attempt back in maybe 2006-7 I got the scroll bar into the window but didn't do anything.
The thumb button moved up and down (vertical SB) but no reaction to the action. I played with it for a few daze. :undecided:
You have to keep in mind that the scrollbar is only a graphical representation of what's supposed to be displayed in your window. The scrollbar itself doesn't scroll your contents: that's on you, usually in your WM_PAINT handler. It merely sets a starting point for the display of whatever's in the window, text, graphics, whatever. Once you understand that, it's not that hard.
Once you figure out how SetScrollInfo() (https://docs.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-setscrollinfo) works, you're in business.
Quote from: Swordfish on August 31, 2022, 07:08:03 AM
Does writing an editor that uses rich edit and its embedded scroll bar count? :tongue:
No, doesn't count, because the RichEdit control handles the scrollbar itself, no need for intervention :tongue:
Quote from: jj2007 on August 31, 2022, 07:52:00 AM
Quote from: Swordfish on August 31, 2022, 07:08:03 AM
Does writing an editor that uses rich edit and its embedded scroll bar count? :tongue:
No, doesn't count, because the RichEdit control handles the scrollbar itself, no need for intervention :tongue:
:biggrin:
Same thing with a plain old edit control that uses the ES_AUTOHSCROLL or ES_AUTOVSCROLL styles, which bring up scrollbars automagically as needed.
Quote from: NoCforMe on August 31, 2022, 07:33:46 AM
Once you figure out how SetScrollInfo() works, you're in business.
I'll look at it one of these daze. Haven't really had the need, but would be nice to know though. And progress bars too :tongue:
We've got bad thunderstorms here internet acting flaky. So if I disappear for a while....
I had this reply on the clipboard, the internet went down. :sad: It's up now but for how long?
List and Combo boxes will set scroll bars as well if you use the right styles with them. :biggrin:
OK, something new: You can add images (icons or bitmaps) to each item. Check it out. (Be sure to get the attached image files.)
The first 3 are .ICOs, the last one's a .BMP. They must be 16x16. Later could accept different sizes. Programming interface isn't too bad; to add an item (with or without image) you pass a small structure to the SendMessage() call:
MLB_ITEM STRUCT
txtPtr DD ?
imageHandle DD ?
flag DD ?
MLB_ITEM ENDS
; Item-image flags:
MLBF_IMAGE_ICON EQU 4000H
MLBF_IMAGE_BITMAP EQU 8000H
If there's an image wanted you pass its handle, otherwise you set this member to NULL. The user (caller) also needs to set the flag to the type of image file; the control handles it appropriately depending on the type.
One thing I discovered is that DrawIcon() didn't work too well for some reason; I ended up using DrawIconEx(), with size params of 0 (the files are all 16x16) and the DI_NORMAL flag. (For bitmaps you have to go through that whole CreateCompatibleDC/SelectObject/BitBlt rigamarole.)
This is looking more like "NoCforMe's custom control workshop" than the current topic title. :thumbsup:
Looks fine indeed.
Yeah, I changed the subject line to reflect how things are going.
Pretty soon I'll be looking for volunteers to actually use the thing. Complete package coming soon, including (gasp!) documentation. (Wait: programmers are like electricians: they're never supposed to document anything! That's the job of that illiterate slob over there in the corner.)
Hobby programmers like me are like DIYers. They never read instructions. :tongue:
Just kidding of course, except for the DIYers. Sounds like you could possibly write a tutorial on using scroll bars, list boxes and the like.
Quote from: NoCforMe on August 31, 2022, 01:01:38 PM
OK, something new: You can add images (icons or bitmaps) to each item. Check it out. (Be sure to get the attached image files.)
:thumbsup:
OK, if anybody still cares, here's the latest and greatest. @Swordfish, notice that scrolling is now fully implemented, including thumb-tracking. (Wasn't hard to get working; I just wasn't tracking it properly in my paint handler.) Also notice that long items now are drawn with ellipsis (...) at the end if they're wider than their "cell". This is due to using
DrawText() rather than
ExtTextOut() (see below for some gory details).
It's getting close; I'm not ready to release the code into the wild yet, because I don't want to have to go chasing versions all over the place as I continue to tweak things and go through my todo list:
- Proper memory management (halfway there, but still not robust)
- Keyboard interface (none currently)
- Maybe make it resizeable?
- Take Greenhorn's suggestion and store instance data pointer in the control's "extra bytes"
Anyhow, play around with it as usual. Next time I'll explain the programming interface for adding items and solicit comments on it.
Regarding drawing text: I learnt something about this. I was using
ExtTextOut(), which works OK, except I wanted to add text ellipsis (...) for items that don't fit in their cell. So I tried
DrawText(), which has this capability. At first I couldn't get anything at all to show. I was using the following flags:
DT_END_ELLIPSIS or DT_LEFT or DT_TOP or DT_NOPREFIX
Nothing. I searched online and found a couple discussions of why this function doesn't work sometimes, but no solutions that made it work for me.
Then I went back to the documentation and noticed some additional flags. One of them,
DT_SINGLELINE, says it "Displays text on a single line". I hadn't figured I needed this--isn't this the default?--but when I added it, things started to work. Way to go, Micro$oft, for incomplete explanations of your API!
Moral of the story: if things don't work, after consulting the oracle (the Intertubes), it pays to experiment.
Quote from: NoCforMe on September 02, 2022, 05:37:19 AM
@Swordfish, notice that scrolling is now fully implemented, including thumb-tracking.
Okay. I gotta go take the dog for a walk, brb.
Okeedokee, I'm back.
The example looks fine. Nice job. There is some noticeable flickering of the 'items' while scrolling though.
Not a show stopper by any means. Just something for your todo list.
Yes, thumb-scrolling isn't as smooth as I'd like. Icing on the cake for later. (I consider that in the "chrome" category: chrome bumpers, not the carburetor or fuel pump.)
Yeah, nothing to halt progress over. You're really making good strides with it. :thumbsup:
OK, here are the gory details of adding an item to my mcListbox. (Pronounced like an Irish or Scottish name.) From my documentation:
Adding Items to mcListbox ControlsItems are added to a mcListbox control by means of the
MLBM_ADDITEM message. Items are always added to the end of the list. An item can also have an image (icon or bitmap) attached to it.
To add an item, create a
MLB_ITEM structure:
typedef struct {
DWORD txtPtr;
DWORD imageHandle;
DWORD flag;
} MLB_ITEM;
Put a pointer to the text to be added in the
txtPtr member. Pass a pointer to this structure via the
wParam parameter to
SendMessage.
If images are to be added to items, you need to first enable this functionality through the
MLBM_SETAPPEARANCE message:
MLBAPPEARANCE mlba;
mlba.flag = MLBF_SETIMAGESYES;
SendMessage (ctrlHandle, MLBM_SETAPPEARANCE, &mlba, 0);
After that you can pass a handle to an image that has been loaded through the
imageHandle member of the
MLB_ITEM structure. You must also set the type of image by setting a flag value in the
flag member:
- MLBF_IMAGE_ICON: Icon (.ico) image
- MLBF_IMAGE_BITMAP: Bitmap (.bmp) image
If you have enabled image display with the MLBF_SETIMAGESYES flag, and you do
not want an image displayed for an item, then set the
imageHandle member to
NULL.
================================================================
So, any comments? As a programmer, would you be offended at all by this interface? Because there are more than two parms, they can't just be passed in wParam and lParam to
SendMessage(), therefore the need for the structure.
No, it seems the best way to pass multiple arguments. It's much better than fetching the variables you need discreetly in the code that follows. That's exactly what one of the purposes for using structures. In'nt it?
I'll have a quarter pounder, a McListBox, and a large order of fries to go btw.
Quote from: swordfish on September 02, 2022, 07:10:52 AM
It's much better than fetching the variables you need discreetly in the code that follows.
How would you even do that? (And I think the word you're looking for is "discretely". No need for discretion here.)
Quote from: NoCforMe on September 02, 2022, 07:24:33 AM
How would you even do that?
Not sure really. I just know that passing pointer to the structure, as an arg, is a perfect use of it.Quote
(And I think the word you're looking for is "discretely". No need for discretion here.)
discreet as in discreet values for instance, or discreet variable rather than an argument.Maybe not the right word though. :mrgreen:
Parental discretion advised though, in either case.I know what I'm thinking even though I don't always have the right words in mind to articulate my thoughts. But I try.
Discrete: separate.
Discreet: tactful, using discretion.
One of the common homophone mix-ups. (Reign/rein, principle/principal, etc.)