lazarus/lcl/interfaces/win32/win32wsforms.pp
Ondrej Pokorny c82b6089b1 comment
2022-11-16 00:50:18 +01:00

983 lines
32 KiB
ObjectPascal

{ $Id$}
{
*****************************************************************************
* Win32WSForms.pp *
* --------------- *
* *
* *
*****************************************************************************
*****************************************************************************
This file is part of the Lazarus Component Library (LCL)
See the file COPYING.modifiedLGPL.txt, included in this distribution,
for details about the license.
*****************************************************************************
}
unit Win32WSForms;
{$mode objfpc}{$H+}
interface
uses
////////////////////////////////////////////////////
// I M P O R T A N T
////////////////////////////////////////////////////
// To get as little as posible circles,
// uncomment only when needed for registration
////////////////////////////////////////////////////
Forms, Controls, LCLType, Classes,
////////////////////////////////////////////////////
WSForms, WSProc, WSLCLClasses, Windows, SysUtils, Win32Extra,
InterfaceBase, Graphics, Win32Int, Win32Proc, Win32WSControls;
type
{ TWin32WSScrollingWinControl }
TWin32WSScrollingWinControl = class(TWSScrollingWinControl)
published
end;
{ TWin32WSScrollBox }
TWin32WSScrollBox = class(TWSScrollBox)
published
class function CreateHandle(const AWinControl: TWinControl;
const AParams: TCreateParams): HWND; override;
end;
{ TWin32WSCustomFrame }
TWin32WSCustomFrame = class(TWSCustomFrame)
published
end;
{ TWin32WSFrame }
TWin32WSFrame = class(TWSFrame)
published
end;
{ TWin32WSCustomForm }
TWin32WSCustomForm = class(TWSCustomForm)
published
class function CreateHandle(const AWinControl: TWinControl;
const AParams: TCreateParams): HWND; override;
class function GetDefaultDoubleBuffered: Boolean; override;
class procedure SetAllowDropFiles(const AForm: TCustomForm; AValue: Boolean); override;
class procedure SetAlphaBlend(const ACustomForm: TCustomForm; const AlphaBlend: Boolean;
const Alpha: Byte); override;
class procedure SetBorderIcons(const AForm: TCustomForm;
const ABorderIcons: TBorderIcons); override;
class procedure SetBounds(const AWinControl: TWinControl; const ALeft, ATop,
AWidth, AHeight: Integer); override;
class procedure SetFormBorderStyle(const AForm: TCustomForm;
const AFormBorderStyle: TFormBorderStyle); override;
class procedure SetFormStyle(const AForm: TCustomform; const AFormStyle, AOldFormStyle: TFormStyle); override;
class procedure SetIcon(const AForm: TCustomForm; const Small, Big: HICON); override;
class procedure ShowModal(const ACustomForm: TCustomForm); override;
class procedure SetRealPopupParent(const ACustomForm: TCustomForm;
const APopupParent: TCustomForm); override;
class procedure SetShowInTaskbar(const AForm: TCustomForm; const AValue: TShowInTaskbar); override;
class procedure ShowHide(const AWinControl: TWinControl); override;
{mdi support}
class function ActiveMDIChild(const AForm: TCustomForm): TCustomForm; override;
class function Cascade(const AForm: TCustomForm): Boolean; override;
class function GetClientHandle(const AForm: TCustomForm): HWND; override;
class function GetMDIChildren(const AForm: TCustomForm; AIndex: Integer): TCustomForm; override;
class function Next(const AForm: TCustomForm): Boolean; override;
class function Previous(const AForm: TCustomForm): Boolean; override;
class function Tile(const AForm: TCustomForm): Boolean; override;
class function ArrangeIcons(const AForm: TCustomForm): Boolean; override;
class function MDIChildCount(const AForm: TCustomForm): Integer; override;
end;
{ TWin32WSForm }
TWin32WSForm = class(TWSForm)
published
end;
{ TWin32WSHintWindow }
TWin32WSHintWindow = class(TWSHintWindow)
published
class function CreateHandle(const AWinControl: TWinControl;
const AParams: TCreateParams): HWND; override;
class procedure ShowHide(const AWinControl: TWinControl); override;
end;
{ TWin32WSScreen }
TWin32WSScreen = class(TWSScreen)
published
end;
{ TWin32WSApplicationProperties }
TWin32WSApplicationProperties = class(TWSApplicationProperties)
published
end;
procedure AdjustFormClientToWindowSize(const AForm: TCustomForm; var ioSize: TSize); overload;
implementation
type
TWinControlAccess = class(TWinControl)
end;
{ TWin32WSScrollBox }
class function TWin32WSScrollBox.CreateHandle(const AWinControl: TWinControl;
const AParams: TCreateParams): HWND;
{$IFDEF NewScrollingLayer}
procedure CreateScrollingLayer(ParentH: HWND);
var
Params: TCreateWindowExParams;
begin
// general initialization of Params
with Params do
begin
Flags := WS_CHILD or WS_CLIPSIBLINGS or WS_CLIPCHILDREN;
FlagsEx := 0;
Window := HWND(nil);
Buddy := HWND(nil);
Parent := ParentH;
SubClassWndProc := @WindowProc;
WindowTitle := nil;
StrCaption := 'TWin32WSScrollBox.CreateHandle ScrollLayer';
Height := 50;
Left := 0;
//Parent := AWinControl.Parent;
Top := 0;
Width := 50;
Flags := Flags or WS_VISIBLE;
FlagsEx := FlagsEx or WS_EX_CONTROLPARENT;
end;
// customization of Params
with Params do
begin
pClassName := @ClsName[0];
SubClassWndProc := nil;
end;
// create window
with Params do
begin
MenuHandle := HMENU(nil);
Window := CreateWindowEx(FlagsEx, pClassName, WindowTitle, Flags,
Left, Top, Width, Height, Parent, MenuHandle, HInstance, Nil);
if Window = 0 then
begin
raise exception.create('failed to create win32 sub control, error: '+IntToStr(GetLastError()));
end;
end;
with Params do
begin
if Window <> HWND(Nil) then
begin
// some controls (combobox) immediately send a message upon setting font
{WindowInfo := AllocWindowInfo(Window);
if GetWindowInfo(Parent)^.needParentPaint then
WindowInfo^.needParentPaint := true;
WindowInfo^.WinControl := AWinControl;
if SubClassWndProc <> nil then
WindowInfo^.DefWndProc := Windows.WNDPROC(SetWindowLongPtrW(
Window, GWL_WNDPROC, PtrInt(SubClassWndProc)));
lhFont := FDefaultFont;
Windows.SendMessage(Window, WM_SETFONT, WPARAM(lhFont), 0);}
end;
end;
Result := Params.Window;
end;
{$ENDIF}
var
Params: TCreateWindowExParams;
begin
// general initialization of Params
PrepareCreateWindow(AWinControl, AParams, Params);
// customization of Params
with Params do
begin
//TODO: Make control respond to user scroll request
if TScrollBox(AWinControl).BorderStyle = bsSingle then
FlagsEx := FlagsEx or WS_EX_CLIENTEDGE;
pClassName := @ClsName[0];
Flags := Flags or WS_HSCROLL or WS_VSCROLL;
SubClassWndProc := nil;
end;
// create window
FinishCreateWindow(AWinControl, Params, false);
Result := Params.Window;
{$IFDEF NewScrollingLayer}
CreateScrollingLayer(Result);
{$ENDIF}
end;
{ TWin32WSCustomForm }
function GetDesigningBorderStyle(const AForm: TCustomForm): TFormBorderStyle;
begin
if csDesigning in AForm.ComponentState then
Result := bsSizeable
else
Result := AForm.BorderStyle;
end;
function CalcBorderStyleFlags(const AForm: TCustomForm): DWORD;
begin
Result := WS_CLIPCHILDREN or WS_CLIPSIBLINGS;
case GetDesigningBorderStyle(AForm) of
bsSizeable, bsSizeToolWin:
Result := Result or (WS_OVERLAPPED or WS_THICKFRAME or WS_CAPTION);
bsSingle, bsToolWindow:
Result := Result or (WS_OVERLAPPED or WS_BORDER or WS_CAPTION);
bsDialog:
Result := Result or (WS_POPUP or WS_BORDER or WS_CAPTION);
bsNone:
if (AForm.Parent = nil) and (AForm.ParentWindow = 0) then
Result := Result or WS_POPUP;
end;
end;
function CalcBorderStyleFlagsEx(const AForm: TCustomForm): DWORD;
begin
Result := 0;
case GetDesigningBorderStyle(AForm) of
bsDialog:
Result := WS_EX_DLGMODALFRAME or WS_EX_WINDOWEDGE;
bsToolWindow, bsSizeToolWin:
Result := WS_EX_TOOLWINDOW;
end;
end;
function CalcBorderIconsFlags(const AForm: TCustomForm): DWORD;
var
BorderIcons: TBorderIcons;
begin
Result := 0;
BorderIcons := AForm.BorderIcons;
if (biSystemMenu in BorderIcons) or (csDesigning in AForm.ComponentState) then
Result := Result or WS_SYSMENU;
if GetDesigningBorderStyle(AForm) in [bsNone, bsSingle, bsSizeable] then
begin
if biMinimize in BorderIcons then
Result := Result or WS_MINIMIZEBOX;
if biMaximize in BorderIcons then
Result := Result or WS_MAXIMIZEBOX;
end;
end;
function CalcBorderIconsFlagsEx(const AForm: TCustomForm): DWORD;
var
BorderIcons: TBorderIcons;
begin
Result := 0;
BorderIcons := AForm.BorderIcons;
if GetDesigningBorderStyle(AForm) in [bsSingle, bsSizeable, bsDialog] then
begin
if biHelp in BorderIcons then
Result := Result or WS_EX_CONTEXTHELP;
end;
end;
procedure CalcFormWindowFlags(const AForm: TCustomForm; var Flags, FlagsEx: DWORD);
begin
// clear all styles which can be set by border style and icons
Flags := Flags and not (WS_POPUP or WS_BORDER or WS_CAPTION or WS_THICKFRAME or
WS_DLGFRAME or WS_MINIMIZEBOX or WS_MAXIMIZEBOX or WS_SYSMENU);
FlagsEx := FlagsEx and not (WS_EX_DLGMODALFRAME or WS_EX_WINDOWEDGE or
WS_EX_TOOLWINDOW or WS_EX_CONTEXTHELP);
// set border style flags
Flags := Flags or CalcBorderStyleFlags(AForm);
FlagsEx := FlagsEx or CalcBorderStyleFlagsEx(AForm);
if (AForm.FormStyle in fsAllStayOnTop) and not (csDesigning in AForm.ComponentState) then
FlagsEx := FlagsEx or WS_EX_TOPMOST;
Flags := Flags or CalcBorderIconsFlags(AForm);
FlagsEx := FlagsEx or CalcBorderIconsFlagsEx(AForm);
end;
procedure AdjustFormClientToWindowSize(const AForm: TCustomForm; var ioSize: TSize);
var
xNonClientDPI: LCLType.UINT;
begin
if AForm.HandleAllocated then
AdjustFormClientToWindowSize(AForm.Handle, ioSize)
else // default handling
AdjustFormClientToWindowSize(AForm.Menu<>nil,
CalcBorderStyleFlags(AForm) or CalcBorderIconsFlags(AForm),
CalcBorderStyleFlagsEx(AForm) or CalcBorderIconsFlagsEx(AForm),
ScreenInfo.PixelsPerInchX, ioSize);
end;
function CustomFormWndProc(Window: HWnd; Msg: UInt; WParam: Windows.WParam; LParam: Windows.LParam): LResult; stdcall;
procedure LCLFormSizeToWin32Size(AForm: TCustomForm; var AWidth, AHeight: Integer);
var
Size: TSize;
begin
Size := TSize.Create(AWidth, AHeight);
AdjustFormClientToWindowSize(AForm, Size);
AWidth := Size.Width;
AHeight := Size.Height;
end;
procedure SetMinMaxInfo(WinControl: TWinControl; var MinMaxInfo: TMINMAXINFO);
procedure SetWin32SizePoint(AWidth, AHeight: integer; var pt: TPoint);
var
IntfWidth, IntfHeight: integer;
begin
// 0 means no constraint
if (AWidth = 0) and (AHeight = 0) then exit;
IntfWidth := AWidth;
IntfHeight := AHeight;
LCLFormSizeToWin32Size(TCustomForm(WinControl), IntfWidth, IntfHeight);
if AWidth > 0 then
pt.X := IntfWidth;
if AHeight > 0 then
pt.Y := IntfHeight;
end;
begin
with WinControl.Constraints do
begin
SetWin32SizePoint(MinWidth, MinHeight, MinMaxInfo.ptMinTrackSize);
SetWin32SizePoint(MaxWidth, MaxHeight, MinMaxInfo.ptMaxTrackSize);
end;
end;
procedure CallWindowPosChanging;
var
WP: PWindowPos;
begin
if not LockWindowPosChanging then
Exit;
WP := PWindowPos(LParam);
if (WP^.flags and SWP_NOMOVE)<>0 then
Exit;
WP^.x := LockWindowPosChangingXY.X;
WP^.y := LockWindowPosChangingXY.Y;
end;
var
Info: PWin32WindowInfo;
WinControl: TWinControl;
begin
Info := GetWin32WindowInfo(Window);
WinControl := Info^.WinControl;
case Msg of
WM_WINDOWPOSCHANGING:
CallWindowPosChanging;
WM_GETMINMAXINFO:
begin
SetMinMaxInfo(WinControl, PMINMAXINFO(LParam)^);
Exit(CallDefaultWindowProc(Window, Msg, WParam, LParam));
end;
WM_SHOWWINDOW:
begin
// this happens when parent window is being minized/restored
// an example of parent window can be an Application.Handle window if MainFormOnTaskBar = False
case LParam of
SW_PARENTCLOSING:
begin
if IsIconic(Window) then
Info^.RestoreState := SW_SHOWMINNOACTIVE
else
if IsZoomed(Window) then
Info^.RestoreState := SW_SHOWMAXIMIZED
else
Info^.RestoreState := SW_SHOWNOACTIVATE;
end;
SW_PARENTOPENING:
begin
if (Info^.RestoreState <> 0) and WinControl.Visible then
begin
Windows.ShowWindowAsync(Window, Info^.RestoreState);
Info^.RestoreState := 0;
Exit(CallDefaultWindowProc(Window, Msg, WParam, LParam));
end;
end;
end;
end;
end;
Result := WindowProc(Window, Msg, WParam, LParam);
end;
class function TWin32WSCustomForm.CreateHandle(const AWinControl: TWinControl;
const AParams: TCreateParams): HWND;
var
Params: TCreateWindowExParams;
lForm: TCustomForm absolute AWinControl;
Bounds: TRect;
SystemMenu: HMenu;
MaximizeForm: Boolean = False;
lSize: TSize;
begin
// general initialization of Params
PrepareCreateWindow(AWinControl, AParams, Params);
// customization of Params
with Params do
begin
if (Parent = 0) then
begin
// Leave Parent at 0 if this is a standalone form.
case lForm.EffectiveShowInTaskBar of
stDefault:
begin
if not Application.MainFormOnTaskBar then
Parent := Win32WidgetSet.AppHandle
else
if (AWinControl <> Application.MainForm) then
begin
if Assigned(Application.MainForm) and Application.MainForm.HandleAllocated then
Parent := Application.MainFormHandle
else
Parent := Win32WidgetSet.AppHandle;
end;
end;
stNever:
begin
Parent := Win32WidgetSet.AppHandle;
FlagsEx := FlagsEx and not WS_EX_APPWINDOW;
end;
end;
end;
if (not (csDesigning in lForm.ComponentState)) and
(lForm.FormStyle=fsMDIChild) and
(lForm <> Application.MainForm) and
Assigned(Application.MainForm) and
(Application.MainForm.FormStyle=fsMDIForm) then
begin
Parent := Win32WidgetSet.MDIClientHandle;
if Parent <> 0 then
begin
Flags := Flags or WS_CHILD;
FlagsEx := FlagsEx or WS_EX_MDICHILD;
// If there is already a maximized MDI child, we'll need to maximize the new one too
if Assigned(Application.MainForm) and Assigned(Application.MainForm.ActiveMDIChild) then
MaximizeForm := Application.MainForm.ActiveMDIChild.WindowState=wsMaximized;
end;
end;
CalcFormWindowFlags(lForm, Flags, FlagsEx);
pClassName := @ClsName[0];
WindowTitle := StrCaption;
Bounds := lForm.BoundsRect;
lSize := Bounds.Size;
AdjustFormClientToWindowSize(lForm, lSize);
Bounds.Size := lSize;
if (lForm.Position in [poDefault, poDefaultPosOnly]) and not (csDesigning in lForm.ComponentState) then
begin
Left := CW_USEDEFAULT;
Top := CW_USEDEFAULT;
end
else
begin
Left := Bounds.Left;
Top := Bounds.Top;
end;
if (lForm.Position in [poDefault, poDefaultSizeOnly]) and not (csDesigning in lForm.ComponentState) then
begin
Width := CW_USEDEFAULT;
Height := CW_USEDEFAULT;
end
else
begin
Width := Bounds.Right - Bounds.Left;
Height := Bounds.Bottom - Bounds.Top;
end;
SubClassWndProc := @CustomFormWndProc;
// mantis #26206: Layered windows are only supported for top-level windows.
// After Windows 8 it is supported for child windows too.
if not (csDesigning in lForm.ComponentState) and lForm.AlphaBlend
and ((WindowsVersion >= wv8) or (Parent = 0)) then
FlagsEx := FlagsEx or WS_EX_LAYERED;
end;
SetStdBiDiModeParams(AWinControl, Params);
// create window
FinishCreateWindow(AWinControl, Params, False);
if (not (csDesigning in lForm.ComponentState)) and
(lForm.FormStyle=fsMDIChild) and
(lForm <> Application.MainForm) and
Assigned(Application.MainForm) and
(Application.MainForm.FormStyle=fsMDIForm) then
begin
// Force a resize event to align children
GetWindowRect(Params.Window, Bounds);
lForm.BoundsRect := Bounds;
// New MDI forms are always activated
SendMessage(Win32WidgetSet.MDIClientHandle, WM_MDIACTIVATE, Params.Window, 0);
// Maximize the form if there was already a maximized MDI child
if MaximizeForm then
lForm.WindowState := wsMaximized;
end;
Result := Params.Window;
// remove system menu items for bsDialog
if (lForm.BorderStyle = bsDialog) and not (csDesigning in lForm.ComponentState) then
begin
SystemMenu := GetSystemMenu(Result, False);
DeleteMenu(SystemMenu, SC_RESTORE, MF_BYCOMMAND);
DeleteMenu(SystemMenu, SC_SIZE, MF_BYCOMMAND);
DeleteMenu(SystemMenu, SC_MINIMIZE, MF_BYCOMMAND);
DeleteMenu(SystemMenu, SC_MAXIMIZE, MF_BYCOMMAND);
DeleteMenu(SystemMenu, 1, MF_BYPOSITION); // remove the separator between move and close
end;
// Beginning with Windows 2000 the UI in an application may hide focus
// rectangles and accelerator key indication. According to msdn we need to
// initialize all root windows with this message
if WindowsVersion >= wv2000 then
Windows.SendMessage(Result, WM_CHANGEUISTATE,
MakeWParam(UIS_INITIALIZE, UISF_HIDEFOCUS or UISF_HIDEACCEL), 0)
end;
class function TWin32WSCustomForm.GetDefaultDoubleBuffered: Boolean;
begin
Result := GetSystemMetrics(SM_REMOTESESSION)=0;
end;
class procedure TWin32WSCustomForm.SetAllowDropFiles(const AForm: TCustomForm;
AValue: Boolean);
begin
DragAcceptFiles(AForm.Handle, AValue);
end;
class procedure TWin32WSCustomForm.SetBorderIcons(const AForm: TCustomForm;
const ABorderIcons: TBorderIcons);
var
ExStyle, NewStyle: DWORD;
begin
UpdateWindowStyle(AForm.Handle, CalcBorderIconsFlags(AForm),
WS_SYSMENU or WS_MINIMIZEBOX or WS_MAXIMIZEBOX);
ExStyle := GetWindowLong(AForm.Handle, GWL_EXSTYLE);
NewStyle := (ExStyle and not WS_EX_CONTEXTHELP) or CalcBorderIconsFlagsEx(AForm);
if ExStyle <> NewStyle then
begin
SetWindowLongPtrW(AForm.Handle, GWL_EXSTYLE, NewStyle);
Windows.RedrawWindow(AForm.Handle, nil, 0, RDW_FRAME or RDW_ERASE or RDW_INVALIDATE or RDW_NOCHILDREN);
end;
end;
class procedure TWin32WSCustomForm.SetFormBorderStyle(const AForm: TCustomForm;
const AFormBorderStyle: TFormBorderStyle);
begin
RecreateWnd(AForm);
end;
function EnumStayOnTopProc(Handle: HWND; Param: LPARAM): WINBOOL; stdcall;
var
list: TList absolute Param;
lWindowInfo: PWin32WindowInfo;
lWinControl: TWinControl;
begin
Result := True;
lWindowInfo := GetWin32WindowInfo(Handle);
if (lWindowInfo <> nil) then
begin
lWinControl := lWindowInfo^.WinControl;
if Assigned(lWinControl) and
(lWinControl is TCustomForm) and
(TCustomForm(lWinControl).FormStyle in fsAllStayOnTop) and
not (csDesigning in lWinControl.ComponentState) then
list.Add(Pointer(Handle));
end;
end;
procedure EnumStayOnTop(Window: THandle; dstlist: TList);
begin
EnumThreadWindows(GetWindowThreadProcessId(Window, nil),
@EnumStayOnTopProc, LPARAM(dstlist));
end;
class procedure TWin32WSCustomForm.SetFormStyle(const AForm: TCustomform;
const AFormStyle, AOldFormStyle: TFormStyle);
const
WindowPosFlags = SWP_NOMOVE or SWP_NOSIZE or SWP_NOACTIVATE or SWP_NOOWNERZORDER;
begin
// Some changes don't require RecreateWnd
if (AOldFormStyle in fsAllStayOnTop) and (AFormStyle in fsAllStayOnTop) then
Exit;
// From normal to StayOnTop
if (AOldFormStyle = fsNormal) and (AFormStyle in fsAllStayOnTop) then
begin
if not (csDesigning in AForm.ComponentState) then
SetWindowPos(AForm.Handle, HWND_TOPMOST, 0, 0, 0, 0, WindowPosFlags)
// From StayOnTop to normal
end
else
if (AOldFormStyle in fsAllStayOnTop) and (AFormStyle = fsNormal) then
begin
// NOTE:
// see bug report #16573 and #38790
// if a window changes from HWND_TOPMOST to HWND_NOTOPMOST
// other TOP most windows also change their state to Non-topmost!
// the page http://msdn.microsoft.com/en-us/library/ms633545(VS.85).aspx, says:
// "When a topmost window is made non-topmost, its owners and its owned windows are also made non-topmost windows"
// Is it possible, that Application window, makes all other forms, non-top most?
// It's also possible to make a list of "topmost forms" and re-enable their state
// after changing the style of the window (so recreation can be avoided).
if not (csDesigning in AForm.ComponentState) then
RecreateWnd(AForm);
end
else
RecreateWnd(AForm);
end;
class procedure TWin32WSCustomForm.SetBounds(const AWinControl: TWinControl;
const ALeft, ATop, AWidth, AHeight: Integer);
var
AForm: TCustomForm absolute AWinControl;
CurRect: Windows.RECT = (Left: 0; Top: 0; Right: 0; Bottom: 0);
lSize: Windows.SIZE;
L, T, W, H: Integer;
Attempt: 0..1; // 2 attempts
begin
// Problem:
// When setting the ClientRect, the main menu may change height (the menu lines may change).
// After the first attempt to set bounds, they can be wrong because the number of the lines changed and
// it is not possible to determine the menu line count for the target rectangle.
// Also, when handling the WM_DPICHANGED message, Windows don't update the menu bar height, therefore
// wrong menu bar height is used for the calculation. The menu bar height gets updated during the first
// TWin32WSWinControl.SetBounds() call.
// -> Therefore a second attempt is needed to get the correct height.
for Attempt := Low(Attempt) to High(Attempt) do
begin
// the LCL defines the size of a form without border, win32 with.
// -> adjust size according to BorderStyle
lSize := TSize.Create(AWidth, AHeight);
AdjustFormClientToWindowSize(AForm, lSize);
L := ALeft;
T := ATop;
W := lSize.Width;
H := lSize.Height;
// we are calling setbounds in TWinControl.Initialize
// if position is default it will be changed to designed. We do not want this.
if wcfInitializing in TWinControlAccess(AWinControl).FWinControlFlags then
begin
if GetWindowRect(AForm.Handle, CurRect) then
begin
if AForm.Position in [poDefault, poDefaultPosOnly] then
begin
L := CurRect.Left;
T := CurRect.Top;
end;
if AForm.Position in [poDefault, poDefaultSizeOnly] then
begin
W := CurRect.Right - CurRect.Left;
H := CurRect.Bottom - CurRect.Top;
end;
end;
end;
// rect adjusted, pass to inherited to do real work
TWin32WSWinControl.SetBounds(AWinControl, L, T, W, H);
if (Attempt=High(Attempt)) // last one, no need to call GetClientRect
or not GetClientRect(AWinControl, CurRect) // not available
or ((CurRect.Width=AWidth) and (CurRect.Height=AHeight)) then // or correct size -> break
break;
end;
end;
class procedure TWin32WSCustomForm.SetIcon(const AForm: TCustomForm; const Small, Big: HICON);
var
Wnd: HWND;
begin
if not WSCheckHandleAllocated(AForm, 'SetIcon') then
Exit;
Wnd := AForm.Handle;
SendMessage(Wnd, WM_SETICON, ICON_SMALL, LPARAM(Small));
SetClassLongPtr(Wnd, GCL_HICONSM, LONG_PTR(Small));
SendMessage(Wnd, WM_SETICON, ICON_BIG, LPARAM(Big));
SetClassLongPtr(Wnd, GCL_HICON, LONG_PTR(Big));
// for some reason sometimes frame does not invalidate itself. lets ask it to invalidate always
Windows.RedrawWindow(Wnd, nil, 0,
RDW_INVALIDATE or RDW_FRAME or RDW_NOCHILDREN or RDW_ERASE);
end;
class procedure TWin32WSCustomForm.SetRealPopupParent(
const ACustomForm: TCustomForm; const APopupParent: TCustomForm);
begin
// changing parent is not possible without handle recreation
RecreateWnd(ACustomForm);
end;
class procedure TWin32WSCustomForm.SetShowInTaskbar(const AForm: TCustomForm;
const AValue: TShowInTaskbar);
var
OldStyle, NewStyle: DWord;
Visible, Active: Boolean;
begin
if not WSCheckHandleAllocated(AForm, 'SetShowInTaskbar') then
Exit;
if Assigned(Application) and (AForm = Application.MainForm) then
Exit;
OldStyle := GetWindowLong(AForm.Handle, GWL_EXSTYLE);
if AValue = stAlways then
NewStyle := OldStyle or WS_EX_APPWINDOW
else
NewStyle := OldStyle and not WS_EX_APPWINDOW;
if OldStyle = NewStyle then exit;
// to apply this changes we need either to hide window or recreate it. Hide is
// less difficult
Visible := IsWindowVisible(AForm.Handle);
Active := GetForegroundWindow = AForm.Handle;
if Visible then
ShowWindow(AForm.Handle, SW_HIDE);
SetWindowLongPtrW(AForm.Handle, GWL_EXSTYLE, NewStyle);
// now we need to restore window visibility with saving focus
if Visible then
if Active then
ShowWindow(AForm.Handle, SW_SHOW)
else
ShowWindow(AForm.Handle, SW_SHOWNA);
end;
class procedure TWin32WSCustomForm.ShowHide(const AWinControl: TWinControl);
const
WindowStateToFlags: array[TWindowState] of DWord = (
{ wsNormal } SW_SHOWNORMAL, // to restore from minimzed/maximized we need to use SW_SHOWNORMAL instead of SW_SHOW
{ wsMinimized } SW_SHOWMINIMIZED,
{ wsMaximized } SW_SHOWMAXIMIZED,
{ wsFullScreen } SW_SHOWMAXIMIZED // win32 has no fullscreen window state
);
var
Flags: DWord;
begin
if AWinControl.HandleObjectShouldBeVisible then
begin
Flags := WindowStateToFlags[TCustomForm(AWinControl).WindowState];
Windows.ShowWindow(AWinControl.Handle, Flags);
{ ShowWindow does not send WM_SHOWWINDOW when creating overlapped maximized window }
{ TODO: multiple WM_SHOWWINDOW when maximizing after initial show? }
if Flags = SW_SHOWMAXIMIZED then
Windows.SendMessage(AWinControl.Handle, WM_SHOWWINDOW, 1, 0);
end
else
if fsModal in TCustomForm(AWinControl).FormState then
Windows.SetWindowPos(AWinControl.Handle, 0, 0, 0, 0, 0,
SWP_NOSIZE or SWP_NOMOVE or SWP_NOZORDER or SWP_NOACTIVATE or SWP_HIDEWINDOW)
else
Windows.ShowWindow(AWinControl.Handle, SW_HIDE);
end;
class function TWin32WSCustomForm.ActiveMDIChild(const AForm: TCustomForm): TCustomForm;
var
ActiveChildHWND: HWND;
PInfo: PWin32WindowInfo;
begin
if (AForm.FormStyle=fsMDIForm) and (Application.MainForm=AForm) then
begin
ActiveChildHWND := SendMessage(Win32WidgetSet.MDIClientHandle, WM_MDIGETACTIVE, 0, 0);
if ActiveChildHWND=0 then Exit(nil);
PInfo := GetWin32WindowInfo(ActiveChildHWND);
if not (PInfo^.WinControl is TCustomForm) then Exit(nil);
Result := TCustomForm(PInfo^.WinControl);
end else
Result := nil;
end;
class function TWin32WSCustomForm.Cascade(const AForm: TCustomForm): Boolean;
begin
if (AForm.FormStyle=fsMDIForm) and (Application.MainForm=AForm) then
begin
SendMessage(Win32WidgetSet.MDIClientHandle, WM_MDICASCADE, 0, 0);
Result := True;
end else
Result := False;
end;
class function TWin32WSCustomForm.GetClientHandle(const AForm: TCustomForm): HWND;
begin
if AForm.FormStyle=fsMDIForm then
Result := Win32WidgetSet.MDIClientHandle
else
Result := 0;
end;
class function TWin32WSCustomForm.GetMDIChildren(const AForm: TCustomForm;
AIndex: Integer): TCustomForm;
var
ChildHWND: HWND;
PInfo: PWin32WindowInfo;
Index: Integer;
begin
Index := 0;
Result := nil;
if (AForm.FormStyle=fsMDIForm) and (Application.MainForm=AForm) then
begin
ChildHWND := GetWindow(Win32WidgetSet.MDIClientHandle, GW_CHILD);
while ChildHWND <> 0 do
begin
if (GetWindowLong(ChildHWND, GWL_EXSTYLE) and WS_EX_MDICHILD) <> 0 then
begin
PInfo := GetWin32WindowInfo(ChildHWND);
if (PInfo^.WinControl is TCustomForm) and not (csDestroying in PInfo^.WinControl.ComponentState) then
begin
if Index=AIndex then Exit(TCustomForm(PInfo^.WinControl));
Inc(Index);
end;
end;
ChildHWND := GetWindow(ChildHWND, GW_HWNDNEXT);
end;
end;
end;
class function TWin32WSCustomForm.Next(const AForm: TCustomForm): Boolean;
begin
if (AForm.FormStyle=fsMDIForm) and (Application.MainForm=AForm) then
begin
SendMessage(Win32WidgetSet.MDIClientHandle, WM_MDINEXT, 0, 1);
Result := True;
end else
Result := False;
end;
class function TWin32WSCustomForm.Previous(const AForm: TCustomForm): Boolean;
begin
if (AForm.FormStyle=fsMDIForm) and (Application.MainForm=AForm) then
begin
SendMessage(Win32WidgetSet.MDIClientHandle, WM_MDINEXT, 0, 0);
Result := True;
end else
Result := False;
end;
class function TWin32WSCustomForm.Tile(const AForm: TCustomForm): Boolean;
begin
if (AForm.FormStyle=fsMDIForm) and (Application.MainForm=AForm) then
begin
SendMessage(Win32WidgetSet.MDIClientHandle, WM_MDITILE, MDITILE_HORIZONTAL, 0);
Result := True;
end else
Result := False;
end;
class function TWin32WSCustomForm.ArrangeIcons(const AForm: TCustomForm): Boolean;
begin
if (AForm.FormStyle=fsMDIForm) and (Application.MainForm=AForm) then
begin
SendMessage(Win32WidgetSet.MDIClientHandle, WM_MDIICONARRANGE, 0, 0);
Result := True;
end else
Result := False;
end;
class function TWin32WSCustomForm.MDIChildCount(const AForm: TCustomForm): Integer;
var
ChildHWND: HWND;
PInfo: PWin32WindowInfo;
begin
Result := 0;
if (AForm.FormStyle=fsMDIForm) and (Application.MainForm=AForm) then
begin
ChildHWND := GetWindow(Win32WidgetSet.MDIClientHandle, GW_CHILD);
while ChildHWND <> 0 do
begin
if (GetWindowLong(ChildHWND, GWL_EXSTYLE) and WS_EX_MDICHILD) <> 0 then
begin
PInfo := GetWin32WindowInfo(ChildHWND);
if (PInfo^.WinControl is TCustomForm) and not (csDestroying in PInfo^.WinControl.ComponentState) then
Inc(Result);
end;
ChildHWND := GetWindow(ChildHWND, GW_HWNDNEXT);
end;
end;
end;
class procedure TWin32WSCustomForm.ShowModal(const ACustomForm: TCustomForm);
var
Parent: HWND;
begin
Parent := GetParent(ACustomForm.Handle);
if (Parent <> 0) and (GetWindowLong(Parent, GWL_EXSTYLE) and WS_EX_TOPMOST <> 0) then
SetWindowPos(ACustomForm.Handle, HWND_TOPMOST, 0, 0, 0, 0, SWP_NOMOVE or SWP_NOSIZE or SWP_NOACTIVATE)
else
BringWindowToTop(ACustomForm.Handle);
end;
class procedure TWin32WSCustomForm.SetAlphaBlend(const ACustomForm: TCustomForm;
const AlphaBlend: Boolean; const Alpha: Byte);
var
Style: DWord;
begin
if not WSCheckHandleAllocated(ACustomForm, 'SetAlphaBlend') then
Exit;
Style := GetWindowLong(ACustomForm.Handle, GWL_EXSTYLE);
if AlphaBlend then
begin
if (Style and WS_EX_LAYERED) = 0 then
SetWindowLongPtrW(ACustomForm.Handle, GWL_EXSTYLE, Style or WS_EX_LAYERED);
Win32Extra.SetLayeredWindowAttributes(ACustomForm.Handle, 0, Alpha, LWA_ALPHA);
end
else
begin
if (Style and WS_EX_LAYERED) <> 0 then
SetWindowLongPtrW(ACustomForm.Handle, GWL_EXSTYLE, Style and not WS_EX_LAYERED);
RedrawWindow(ACustomForm.Handle, nil, 0, RDW_ERASE or RDW_INVALIDATE or RDW_FRAME or RDW_ALLCHILDREN);
end;
end;
{ TWin32WSHintWindow }
class function TWin32WSHintWindow.CreateHandle(const AWinControl: TWinControl;
const AParams: TCreateParams): HWND;
var
Params: TCreateWindowExParams;
begin
// general initialization of Params
PrepareCreateWindow(AWinControl, AParams, Params);
// customization of Params
with Params do
begin
pClassName := @ClsHintName[0];
WindowTitle := StrCaption;
Flags := WS_POPUP;
FlagsEx := FlagsEx or WS_EX_TOOLWINDOW;
Left := LongInt(CW_USEDEFAULT);
Top := LongInt(CW_USEDEFAULT);
Width := LongInt(CW_USEDEFAULT);
Height := LongInt(CW_USEDEFAULT);
end;
// create window
FinishCreateWindow(AWinControl, Params, false);
Result := Params.Window;
end;
class procedure TWin32WSHintWindow.ShowHide(const AWinControl: TWinControl);
begin
if AWinControl.HandleObjectShouldBeVisible then
Windows.SetWindowPos(AWinControl.Handle, HWND_TOPMOST, 0, 0, 0, 0, SWP_SHOWWINDOW or SWP_NOMOVE or SWP_NOSIZE or SWP_NOACTIVATE or SWP_NOOWNERZORDER)
else
Windows.ShowWindow(AWinControl.Handle, SW_HIDE);
end;
end.