Customize Windows Form Scrollbar

2020-02-01 04:25发布

I have searched the world wide web without a proper answer.

In my Windows Form application, i want to change the width of a scrollbar that belongs to a FlowLayoutPanel.

The Scrollbar is added "automatically" since the content of the Flow Layout Panel is larger the Form.

From what I've found on the web, it seems to be tricky.

Is there any solution to this?

Cheers!

2条回答
家丑人穷心不美
2楼-- · 2020-02-01 04:37

Cody Gray's answer is 100% correct, though I wanted to add more reference material on the subject matter.

Background

The way that Windows Forms creates scrollbars is through using the window styles WS_HSCROLL and WS_VSCROLL. Respectively, these styles are responsible for enabling the horizontal and vertical scrollbars for a given HWND. An HWND is a native resource handle to a "window", which in .NET lingo corresponds to a Control.

Thinking from a Windows API perspective, we must set window styles at the time we create the HWND. This is done through a call to CreateWindow, CreateWindowEx, or SetWindowLong. Naturally, we may begin to think about using P/Invoke to help us out, but this would be quite a burden as it means we would need to re-implement Windows Forms from scratch.

Fortunately, Windows Forms exposes a property, CreateParams, that can be overridden to specify the exact window styles, among other Control creation parameters. This property is in turn consumed by the .NET framework so that it can create the HWND with the appropriate styles when the Control is instantiated.

Customizing the Scrollbar

Replacing the Windows API functionality for a scrollbar is actually simpler than it may seem; however, this isn't obvious (well for me anyways, I had to sift through .NET source to find the answer). To do this, we must choose the appropriate Control to inherit from to create our own custom ScrollableControl. If we observe the source code for System.Windows.Forms.ScrollableControl, we see the following styles are used:

CreateParams cp = base.CreateParams;

if (HScroll || HorizontalScroll.Visible) {
    cp.Style |= NativeMethods.WS_HSCROLL;
}
else {
    cp.Style &= (~NativeMethods.WS_HSCROLL);
}
if (VScroll || VerticalScroll.Visible) {
    cp.Style |= NativeMethods.WS_VSCROLL;
}
else {
    cp.Style &= (~NativeMethods.WS_VSCROLL);
}

So, in short, when we extend from ScrollableControl, the native horizontal and vertical scrollbars are enabled based on its internal logic. We could access the ScrollableControl's window handle and then call SetWindowLong to hide the scrollbars; however, we would need to keep track of all places that ScrollableControl interacts with the Windows API. In fact, there is an internal function Control.UpdateStylesCore() that is called based on whether or not the scrollbars should be shown. This function effectively reapplies the windows styles above, and it would likely be best not to fight with it. It would be a much cleaner approach to steer away from the Windows API and extend directly from Control. We can then provide whatever API's we desire to have.

This means we would be looking at re-implementing:

  1. Updating scrollbars for mouse wheel events.
  2. Updating scrollbars based on clicking the custom scrollbar buttons and tracks/thumbs.
  3. Updating scrollbars based on adding/removing/moving/resizing child controls.
  4. Creating auto-scroll margins.
  5. Constraints on the client area that affects when scrollbars are visible.
  6. And so on...

Alternatively, a simple approach might be to create a new UserControl. This would allow us to use the Visual Studio designer to simplify configuring our scrollbar buttons and tracks.

Whichever path is taken, it will be necessary to see how ScrollableControl works internally in order to provide a comfortable user-experience.

查看更多
别忘想泡老子
3楼-- · 2020-02-01 04:43

No, there's no way to change the width of a scrollbar displayed on a single control (although there is a system-wide setting that will affect all scrollbars in all applications).

The ugly truth is that the lowly scrollbar control is far more complicated than it looks. Basically, the scrollbars on the FlowLayoutPanel are drawn by Windows itself (rather than the .NET Framework) because of the WS_HSCROLL and/or WS_VSCROLL window styles that are set for the control behind the scenes. The FlowLayoutPanel doesn't provide any facility to change or modify how these built-in scrollbars are drawn. Unlike other more advanced modifications in WinForms, there are no such messages we can send to the control's window procedure. And to make matters worse, the scrollbars are drawn in the non-client area of the FlowLayoutPanel, which means we can't just override its Paint event and handle drawing the scrollbars ourselves.

Unfortunately, if you really want to customize your scrollbars, you're going to have to hide the built-in scrollbars and roll your own. It's not quite as difficult as it sounds, though, if you're up for it. This article on CodeProject provides a good walk-through on creating your own skinnable scrollbar as a user control and use it as a replacement in the container control of your choice.

查看更多
登录 后发表回答