5

我已将表单更改为无边框表单,我只是将BorderStyle属性更改为bsNone,但现在我的应用程序丢失了 windows 锚点和一些命令,例如

WIN + ↑ (对齐客户端)
WIN + ↓ (最小化表单)
WIN + →(对齐表单右)
WIN + ←(对齐表单左)

我尝试在 中设置BorderStyle: bsSizeable和使用以下代码FormCreate,但这不起作用:

procedure TfrmBase.FormCreate(Sender: TObject);
begin
  SetWindowLong(Handle
               ,GWL_STYLE
               ,GetWindowLong(Handle, GWL_STYLE)
                AND (NOT WS_CAPTION)
                AND (NOT WS_THICKFRAME)
               );


  Refresh;
  FormColor := oLauncher.oCor;
end;

结果:

我的表格

上图是我想要的,但是我已经提到的 Windows 命令不起作用

有什么方法可以设置BorderStyle: bsNone并且不会丢失这些命令?

已编辑

如果我使用WS_THICKFRAME我的表单返回一个小的顶部边框并且 windows 命令运行良好,但我不想要那个顶部边框。

我的表格2

已编辑 2

我非常接近预期的结果,但还有一点问题......

我把这个放在我的FormCreate

SetWindowLong(Handle
             ,GWL_STYLE
             ,GetWindowLong(Handle, GWL_STYLE)
              AND (NOT WS_CAPTION)
              );

我创建了方法

private
   procedure WmNCCalcSize(var Msg: TWMNCCalcSize); message WM_NCCALCSIZE;

接着

procedure TfrmBase.WmNCCalcSize(var Msg: TWMNCCalcSize);
begin
  inherited;
  if Msg.CalcValidRects then
  begin
    InflateRect(Msg.CalcSize_Params.rgrc[0], 0, 6);
    Msg.Result := 0;
  end;
end;

我在这里得到了这个方法

现在边框已经消失了,但是当我的表单失去焦点时,顶部/底部边框再次显示......

我怎样才能避免这种情况?

在此处输入图像描述


解决了

我离开了边界BorderStyle: bsSizeable,然后我做到了:

private
  procedure WmNCCalcSize(var Msg: TWMNCCalcSize); message WM_NCCALCSIZE;
[...]
procedure TfrmBase.WmNCCalcSize(var Msg: TWMNCCalcSize);
var
  R: TRect;
begin
  if not Msg.CalcValidRects then
    R := PRect(Msg.CalcSize_Params)^;
  inherited;
  if Msg.CalcValidRects then
    Msg.CalcSize_Params.rgrc0 := Msg.CalcSize_Params.rgrc1
  else
    PRect(Msg.CalcSize_Params)^ := R;

  Msg.Result := 0;
end;

procedure TfrmBase.FormCreate(Sender: TObject);
begin
  BorderStyle := bsNone;
  SetWindowLong(Handle
               ,GWL_STYLE
               ,WS_CLIPCHILDREN or WS_OVERLAPPEDWINDOW
               );
end;

procedure TfrmBase.FormShow(Sender: TObject);
begin
  Width := (Width - 1);
end;

GitHub 上的解决方案

我在这里创建了一个存储库

4

1 回答 1

2

您提到的一些命令是与窗口大小相关的系统命令。这需要厚框架,没有它“WIN + 右”和“WIN + 左”将不起作用。此外,您需要最小化框和最大化框才能使 WIN + 向上/向下命令起作用。

最好是从头开始并包含您需要的样式,否则 VCL 可能会干扰。如果有可能重新创建您的表单,请将样式设置为CreateWnd覆盖。

procedure TForm1.FormCreate(Sender: TObject);
begin
  BorderStyle := bsNone;
  SetWindowLong(Handle, GWL_STYLE, WS_CLIPCHILDREN or WS_OVERLAPPEDWINDOW);
end;


然后是你不想要的框架。在问题的编辑中,您膨胀客户矩形以摆脱它。不要猜测框架的宽度/高度,如下所示。

procedure TForm1.WMNCCalcSize(var Message: TWMNCCalcSize);
var
  R: TRect;
begin
  if not Message.CalcValidRects then
    R := PRect(Message.CalcSize_Params)^;
  inherited;
  if Message.CalcValidRects then
    Message.CalcSize_Params.rgrc0 := Message.CalcSize_Params.rgrc1
  else
    PRect(Message.CalcSize_Params)^ := R;
  Message.Result := 0;
end;

此时必须阅读消息的文档,参数在不同阶段具有不同的含义等。


上面留下了一个完全没有任何非客户区的窗口。客户矩形等于窗口矩形。虽然标题不可见,但您可以通过按 Alt+Space 来激活系统菜单。问题是,系统坚持要绘制激活状态。现在它在客户区画了一个框架!!

通过拦截去掉它WM_NCACTIVATE,你还需要它根据激活状态来绘制你的标题:

procedure TForm1.WMNCActivate(var Message: TWMNCActivate);
begin
  if Message.Active then
    // draw active caption
  else
    // draw incactive caption

  // don't call inherited
end;


您可能必须处理一些故障,弄乱窗户会产生后果。例如,在我的测试中,最小化的表单在 alt+tab 对话框中没有关联的图标。



以下是我的完整测试单元。

unit Unit1;

interface

uses
  Winapi.Windows, Winapi.Messages, System.SysUtils, System.Variants,
  System.Classes, Vcl.Graphics, Vcl.Controls, Vcl.Forms, Vcl.Dialogs;

type
  TForm1 = class(TForm)
    procedure FormCreate(Sender: TObject);
  protected
    procedure WMNCActivate(var Message: TWMNCActivate); message WM_NCACTIVATE;
    procedure WMNCCalcSize(var Message: TWMNCCalcSize); message WM_NCCALCSIZE;
  end;

var
  Form1: TForm1;

implementation

{$R *.dfm}

procedure TForm1.FormCreate(Sender: TObject);
begin
  BorderStyle := bsNone;
  SetWindowLong(Handle, GWL_STYLE, WS_CLIPCHILDREN or WS_OVERLAPPEDWINDOW);
end;

procedure TForm1.WMNCActivate(var Message: TWMNCActivate);
begin
  if Message.Active then
    // draw active caption
  else
    // draw incactive caption

  // don't call inherited
end;

procedure TForm1.WMNCCalcSize(var Message: TWMNCCalcSize);
var
  R: TRect;
begin
  if not Message.CalcValidRects then
    R := PRect(Message.CalcSize_Params)^;
  inherited;
  if Message.CalcValidRects then
    Message.CalcSize_Params.rgrc0 := Message.CalcSize_Params.rgrc1
  else
    PRect(Message.CalcSize_Params)^ := R;
  Message.Result := 0;
end;

end.
于 2019-02-13T15:39:13.950 回答