Проблема рисования полосы прокрутки с утолщенной границей окна не-клиента

Я пытаюсь нарисовать цветную рамку вокруг клиентской области пользовательского элемента управления с полосами прокрутки. Для этого я установил 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 Самые левые пиксели вертикальной полосы прокрутки, по-видимому, не окрашены операционной системой.

Некоторые наблюдения:

  1. Используя полный inherited вместо простого DefaltHandler(Message) делает проблему намного хуже. В этом случае желтая область полностью закроет полосы прокрутки после того, как форма будет временно перемещена за пределы экрана и после операции сжатия-увеличения формы, закрывающей элемент управления.

Скриншот

  1. Реализация соответствующего ответа на WM_NCHITTEST сообщение заставляет элемент управления вести себя лучше, но не решает проблему рисования полосы прокрутки.

  2. Мне известно, как нарисовать пользовательскую границу внутри не клиентской области элемента управления с полосами прокрутки?; ответы на этот вопрос все страдают от вопросов, описанных выше.

Я использую Delphi 2009 и Windows 7 Home Premium, 64-битную версию с поддержкой Aero.

0 ответов

Другие вопросы по тегам