我正在尝试使用滚动条在自定义控件的客户区周围绘制彩色边框。为此,我将BorderWidth
设置为正整数并响应WM_NCPAINT
消息。这听起来像混合VCL和Win32,但BorderWidth
属性只会导致WM_NCCALCSIZE
消息的适当处理。
以下代码为SSCCE:
unit Unit6;
interface
uses
Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
Dialogs;
type
TSample = class(TCustomControl)
protected
procedure Paint; override;
procedure CreateParams(var Params: TCreateParams); override;
procedure WMNCPaint(var Message: TWMNCPaint); message WM_NCPAINT;
published
property BorderWidth;
end;
TForm6 = class(TForm)
procedure FormCreate(Sender: TObject);
private
{ Private declarations }
public
{ Public declarations }
end;
var
Form6: TForm6;
implementation
{$R *.dfm}
{ TSample }
procedure TSample.CreateParams(var Params: TCreateParams);
begin
inherited;
Params.Style := Params.Style or WS_VSCROLL or WS_HSCROLL;
end;
procedure TSample.Paint;
begin
inherited;
Canvas.Brush.Color := clWhite;
Canvas.FillRect(ClientRect);
end;
procedure TSample.WMNCPaint(var Message: TWMNCPaint);
var
dc: HDC;
R: TRect;
begin
DefaultHandler(Message);
dc := GetWindowDC(Handle);
try
Brush.Color := clYellow;
GetWindowRect(Handle, R);
with R do
R := Rect(0, 0, Right - Left, Bottom - Top);
ExcludeClipRect(dc, BorderWidth, BorderWidth,
R.Right - BorderWidth, R.Bottom - BorderWidth);
FillRect(dc, R, Brush.Handle);
finally
ReleaseDC(Handle, dc);
end;
end;
procedure TForm6.FormCreate(Sender: TObject);
begin
with TSample.Create(self) do
begin
Parent := Self;
SetBounds(10, 10, 500, 100);
BorderWidth := 10;
end;
end;
end.
结果如下:
除了右下方之外,这看起来很完美。通过做一些事情可以很容易地解决这个问题。我故意不画这个区域,因为它与我试图描述的实际问题无关。所以请忽略那个方格。
现在,我可以通过拖动窗体的右边框来调整窗体的大小。我首先将其缩小,以便隐藏样本控制窗口的垂直滚动条。然后我慢慢放大表格,使样品控制再次完全可见。然后它看起来像这样:
在这里你可以看到问题:垂直滚动条的最左边的〜BorderSize
像素似乎没有被操作系统绘制。
一些观察结果:
inherited
代替仅仅DefaltHandler(Message)
会使问题更严重。在这种情况下,黄色区域将在窗体暂时移出屏幕后以及在控制遮挡形式收缩 - 生长操作之后完全遮挡滚动条。
对WM_NCHITTEST
消息实施匹配响应会使控件的行为更好,但不会解决滚动条绘制问题。
我知道How to draw a custom border inside the non client area of a control with scroll bars?;这个Q的答案都有上述问题。
我正在使用Delphi 2009和Windows 7家庭高级版,64位,启用Aero。