Componentes.Terceros.DevExp.../internal/x.44/1/ExpressSkins Library/Sources/dxSkinsForm.pas
2009-06-29 12:09:02 +00:00

3832 lines
118 KiB
ObjectPascal

{********************************************************************}
{ }
{ Developer Express Visual Component Library }
{ ExpressSkins Library }
{ }
{ Copyright (c) 2006-2009 Developer Express Inc. }
{ ALL RIGHTS RESERVED }
{ }
{ The entire contents of this file is protected by U.S. and }
{ International Copyright Laws. Unauthorized reproduction, }
{ reverse-engineering, and distribution of all or any portion of }
{ the code contained in this file is strictly prohibited and may }
{ result in severe civil and criminal penalties and will be }
{ prosecuted to the maximum extent possible under the law. }
{ }
{ RESTRICTIONS }
{ }
{ THIS SOURCE CODE AND ALL RESULTING INTERMEDIATE FILES }
{ (DCU, OBJ, DLL, ETC.) ARE CONFIDENTIAL AND PROPRIETARY TRADE }
{ SECRETS OF DEVELOPER EXPRESS INC. THE REGISTERED DEVELOPER IS }
{ LICENSED TO DISTRIBUTE THE EXPRESSSKINS AND ALL ACCOMPANYING }
{ VCL CONTROLS AS PART OF AN EXECUTABLE PROGRAM ONLY. }
{ }
{ THE SOURCE CODE CONTAINED WITHIN THIS FILE AND ALL RELATED }
{ FILES OR ANY PORTION OF ITS CONTENTS SHALL AT NO TIME BE }
{ COPIED, TRANSFERRED, SOLD, DISTRIBUTED, OR OTHERWISE MADE }
{ AVAILABLE TO OTHER INDIVIDUALS WITHOUT EXPRESS WRITTEN CONSENT }
{ AND PERMISSION FROM DEVELOPER EXPRESS INC. }
{ }
{ CONSULT THE END USER LICENSE AGREEMENT FOR INFORMATION ON }
{ ADDITIONAL RESTRICTIONS. }
{ }
{********************************************************************}
unit dxSkinsForm;
interface
{$I cxVer.inc}
uses
Types, Windows, Classes, SysUtils, Messages, Forms, Graphics, Controls,
MultiMon, cxDWMAPI, ShellApi, cxLookAndFeelPainters, cxClasses, StdCtrls,
cxGraphics, cxControls, cxGeometry, dxSkinsLookAndFeelPainter, dxSkinsCore,
cxScrollBar, ExtCtrls, FlatSB, cxLookAndFeels, dxUxTheme, dxSkinInfo, dxCore,
cxContainer, Math;
const
dxSkinFormTextOffset = 5;
dxSkinIconSpacing = 2;
WM_POSTREDRAW: Cardinal = WM_DX + 1;
WM_CHILDCHANGED: Cardinal = WM_DX + 2;
WM_POSTCHECKRGN: Cardinal = WM_DX + 4;
WM_POSTCREATE: Cardinal = WM_DX + 3;
WM_POSTMSGFORMINIT: Cardinal = WM_DX + 5;
WM_HASOWNSKINENGINE: Cardinal = WM_DX + 6;
IsDesigning: Boolean = False;
type
TdxSkinForm = class;
TdxSkinFormController = class;
TdxSkinFormNonClientAreaInfo = class;
TdxSkinFormPainter = class;
TdxSkinFormCorner = (sfcLeftTop, sfcRightTop, sfcLeftBottom, sfcRightBottom);
TdxScrollAreaElement = (saeHorzScroll, saeVertScroll, saeSizeGrip);
TdxScrollAreaElements = set of TdxScrollAreaElement;
TdxSkinFormScrollBar = saeHorzScroll..saeVertScroll;
TdxSkinFormEvent = procedure(Sender: TObject; AForm: TCustomForm;
var ASkinName: string; var UseSkin: Boolean) of object;
TdxSkinControlEvent = procedure(Sender: TObject; AControl: TWinControl;
var UseSkin: Boolean) of object;
{ TdxSkinController }
TdxSkinController = class(TcxLookAndFeelController)
private
FOnSkinControl: TdxSkinControlEvent;
FOnSkinForm: TdxSkinFormEvent;
function GetUseSkins: Boolean;
procedure SetUseSkins(Value: Boolean);
protected
procedure Changed;
function DoSkinControl(AControl: TWinControl): Boolean; virtual;
function DoSkinForm(AForm: TCustomForm): TdxSkinLookAndFeelPainterClass; virtual;
function DoSkinFormEx(AForm: TCustomForm;
var ASkinName: string; var AUseSkin: Boolean): TdxSkinLookAndFeelPainterClass; virtual;
procedure Loaded; override;
procedure MasterLookAndFeelChanged(Sender: TcxLookAndFeel; AChangedValues: TcxLookAndFeelValues); override;
procedure MasterLookAndFeelDestroying(Sender: TcxLookAndFeel); override;
public
constructor Create(AOwner: TComponent); override;
destructor Destroy; override;
procedure Refresh;
class function GetFormSkin(AForm: TCustomForm; var ASkinName: string): Boolean;
published
property Kind;
property NativeStyle;
property SkinName;
property UseSkins: Boolean read GetUseSkins write SetUseSkins default True;
property OnSkinForm: TdxSkinFormEvent read FOnSkinForm write FOnSkinForm;
property OnSkinControl: TdxSkinControlEvent read FOnSkinControl write FOnSkinControl;
end;
{ TdxSkinWinController }
TdxSkinWinController = class(TcxIUnknownObject, IcxMouseTrackingCaller)
private
FHandle: HWND;
FLookAndFeelPainter: TdxSkinLookAndFeelPainterClass;
FMaster: TdxSkinWinController;
FProcInstance: Pointer;
FSavedWndProc: TWndMethod;
FSavedWndProcPtr: Pointer;
FWinControl: TWinControl;
function GetHasVirtualChilds: Boolean;
function GetIsHooked: Boolean;
function GetIsMDIClient: Boolean;
function GetLookAndFeelPainter: TdxSkinLookAndFeelPainterClass;
procedure SetHandle(AHandle: HWND);
protected
function GetIsSkinUsed: Boolean; virtual;
function GetMaster(AHandle: HWND): TdxSkinWinController; virtual;
function GetUseSkinForControl: Boolean; virtual;
function IsHookAvailable: Boolean; virtual;
procedure DefWndProc(var AMessage); virtual;
procedure HookWndProc; virtual;
procedure InitializePainter; virtual;
procedure MasterDestroyed; virtual;
procedure RedrawWindow(AUpdateNow: Boolean);
procedure UnHookWndProc; virtual;
procedure WndProc(var AMessage: TMessage); virtual;
{ IcxMouseTrackingCaller }
procedure MouseLeave; virtual;
public
constructor Create(AHandle: HWND); virtual;
destructor Destroy; override;
class function IsMDIChildWindow(AHandle: HWND): Boolean; virtual;
class function IsMDIClientWindow(AHandle: HWND): Boolean; virtual;
class function IsMessageDlgWindow(AHandle: HWND): Boolean; virtual;
class function IsSkinActive(AHandle: HWND): Boolean;
class procedure FinalizeEngine(AHandle: HWND);
class procedure InitializeEngine(AHandle: HWND);
function GetSkinName(var ASkinName: string): Boolean;
procedure Refresh; virtual;
procedure Update; virtual;
property Handle: HWND read FHandle write SetHandle;
property HasVirtualChilds: Boolean read GetHasVirtualChilds;
property IsHooked: Boolean read GetIsHooked;
property IsMDIClient: Boolean read GetIsMDIClient;
property IsSkinUsed: Boolean read GetIsSkinUsed;
property LookAndFeelPainter: TdxSkinLookAndFeelPainterClass read GetLookAndFeelPainter;
property Master: TdxSkinWinController read FMaster;
property ProcInstance: Pointer read FProcInstance;
property WinControl: TWinControl read FWinControl;
end;
TdxSkinWinControllerClass = class of TdxSkinWinController;
TdxSkinGetControllerClassForWindowProc = function (AWnd: HWND): TdxSkinWinControllerClass;
{ TdxSkinFormController }
TdxSkinFormController = class(TdxSkinWinController)
private
FForceRedraw: Boolean;
FHasRegion: Boolean;
FLockRedrawCount: Integer;
FPainter: TdxSkinFormPainter;
FSkinForm: TdxSkinForm;
FViewInfo: TdxSkinFormNonClientAreaInfo;
function GetForm: TCustomForm;
protected
function GetIsSkinUsed: Boolean; override;
function IsHookAvailable: Boolean; override;
function NeedForceRgnUpdate(ASizeType: Integer): Boolean;
procedure CalculateViewInfo; virtual;
procedure CheckWindowRgn(AForceRgn: Boolean);
procedure DefWndProc(var AMessage); override;
procedure DrawWindowBackground(DC: HDC); virtual;
procedure DrawWindowBorder(DC: HDC = 0); virtual;
procedure FlushWindowRegion; virtual;
procedure InitializeMessageForm; virtual;
procedure InitializePainter; override;
procedure InvalidateBorders;
function HandleInternalMessages(var AMessage: TMessage): Boolean; virtual;
function HandleWindowMessage(var AMessage: TMessage): Boolean; virtual;
function RefreshOnMouseEvent(AForceRefresh: Boolean = False): Boolean;
procedure RefreshOnSystemMenuShown;
procedure LockRedraw;
procedure MouseLeave; override;
procedure UnlockRedraw;
function WMNCCreate(var Message: TMessage): Boolean; virtual;
procedure WMContextMenu(var Message: TWMContextMenu); virtual;
procedure WMDestroy(var Message: TMessage); virtual;
procedure WMEraseBkgnd(var Message: TWMEraseBkgnd); virtual;
procedure WMNCActivate(var Message: TWMNCActivate); virtual;
procedure WMNCButtonDown(var Message: TWMNCHitMessage); virtual;
procedure WMNCCalcSize(var Message: TWMNCCALCSIZE); virtual;
procedure WMNCHitTest(var Message: TWMNCHitTest); virtual;
procedure WMNCLButtonUp(var Message: TWMNCHitMessage); virtual;
procedure WMNCMouseMove(var Message: TWMNCHitMessage);
procedure WMNCPaint(var Message: TWMNCPaint); virtual;
procedure WMPrint(var Message: TWMPrint); virtual;
procedure WMScroll(var Message: TWMScroll); virtual;
procedure WMSetText(var Message: TWMSetText); virtual;
procedure WMSize(var Message: TWMSize); virtual;
procedure WMSysCommand(var Message: TWMSysCommand); virtual;
procedure WMWindowPosChanged(var Message: TWMWindowPosChanged); virtual;
procedure WndProc(var AMessage: TMessage); override;
public
constructor Create(AHandle: HWND); override;
constructor CreateEx(ASkinForm: TdxSkinForm);
destructor Destroy; override;
procedure RefreshController;
procedure Update; override;
property ForceRedraw: Boolean read FForceRedraw write FForceRedraw;
property Form: TCustomForm read GetForm;
property HasRegion: Boolean read FHasRegion write FHasRegion;
property Painter: TdxSkinFormPainter read FPainter;
property SkinForm: TdxSkinForm read FSkinForm;
property ViewInfo: TdxSkinFormNonClientAreaInfo read FViewInfo;
end;
{ TdxSkinFormNonClientAreaInfo }
TdxSkinFormNonClientAreaInfo = class(TObject)
private
FClientRect: TRect;
FContentOffset: Integer;
FController: TdxSkinFormController;
FExStyle: Integer;
FSizeType: Integer;
FStyle: Integer;
FThemeActive: Boolean;
FThemeActiveAssigned: Boolean;
FWindowRect: TRect;
function GetBorderBounds(ASide: TcxBorder): TRect;
function GetButtonPressed: Boolean;
function GetCaptionBounds: TRect;
function GetCaptionButtonRect(const ACaptionRect: TRect): TRect;
function GetCaptionContentOffset: TRect;
function GetCaptionIconSize: Integer;
function GetCaptionTextColor: TColor;
function GetClientRectOnClient: TRect;
function GetHandle: HWND;
function GetHasBorder: Boolean;
function GetHasCaption: Boolean;
function GetHasCaptionTextShadow: Boolean;
function GetHasMenu: Boolean;
function GetIconBounds(AIcon: TdxSkinFormIcon): TRect;
function GetIconState(AIcon: TdxSkinFormIcon): TdxSkinElementState;
function GetIsAlphaBlendUsed: Boolean;
function GetIsDialog: Boolean;
function GetIsIconic: Boolean;
function GetIsSizeBox: Boolean;
function GetIsZoomed: Boolean;
function GetNativeBorderWidth: Boolean;
function GetNeedCheckNonClientSize: Boolean;
function GetScrollAreaBounds(AItem: TdxScrollAreaElement): TRect;
function GetScrollBarInfo(AScrollBar: TdxSkinFormScrollBar): TScrollBarInfo;
function GetScrollBarPartBounds(AScrollBar: TdxSkinFormScrollBar; APart: TcxScrollBarPart): TRect;
function GetScrollBarPartState(AScrollBar: TdxSkinFormScrollBar; APart: TcxScrollBarPart): TcxButtonState;
function GetSizeArea(ASide: TcxBorder): TRect;
function GetSizeCorners(ACorner: TdxSkinFormCorner): TRect;
function GetSkinBorderWidth(ASide: TcxBorder): Integer;
function GetThemeActive: Boolean;
function GetToolWindow: Boolean;
procedure SetActive(AActive: Boolean);
procedure SetSizeType(AValue: Integer);
procedure SetUpdateRgn(ARgn: HRGN);
protected
FActive: Boolean;
FBorderBounds: array[TcxBorder] of TRect;
FBorderWidths: TRect;
FBoundsNoBorders: TRect;
FCaption: string;
FCaptionBounds: TRect;
FCaptionFont: TFont;
FCaptionSufix: string;
FCaptionTextColor: array[Boolean] of TColor;
FCaptionTextShadowColor: TColor;
FClientEdgeSize: TSize;
FHasClientEdge: Boolean;
FHasMenu: Boolean;
FIconBounds: array[TdxSkinFormIcon] of TRect;
FIconPressed: TdxSkinFormIcon;
FIcons: TdxSkinFormIcons;
FIconState: array[TdxSkinFormIcon] of TdxSkinElementState;
FIsMDIChild: Boolean;
FIsMDIClient: Boolean;
FMenuSeparatorBounds: TRect;
FPainter: TcxCustomLookAndFeelPainterClass;
FPainterInfo: TdxSkinLookAndFeelPainterInfo;
FScrollAreaBounds: array[TdxScrollAreaElement] of TRect;
FScrollAreaElements: TdxScrollAreaElements;
FScrollBarPartBounds: array[TdxSkinFormScrollBar, TcxScrollBarPart] of TRect;
FScrollBarPartState: array[TdxSkinFormScrollBar, TcxScrollBarPart] of TcxButtonState;
FScrollBarsInfo: array[TdxSkinFormScrollBar] of TScrollBarInfo;
FSizeFrame: TSize;
FSysMenuIcon: HICON;
FSystemSizeFrame: TSize;
FTrackedScrollBar: TdxScrollAreaElement;
FTrackIcon: TdxSkinFormIcon;
FUpdateRgn: HRGN;
FWindowBounds: TRect;
procedure CalculateBordersInfo; virtual;
procedure CalculateBorderWidths; virtual;
function CalculateCaptionHeight: Integer; virtual;
procedure CalculateCaptionIconsInfo; virtual;
procedure CalculateFontInfo;
procedure CalculateFrameSizes; virtual;
function CalculateMargins: TRect; virtual;
procedure CalculateScrollArea; virtual;
procedure CalculateScrollBarPartInfo(AScrollBar: TdxSkinFormScrollBar;
Pos1, Pos2: Integer; APart: TcxScrollBarPart);
procedure CalculateScrollBarPartsInfo; virtual;
function CalculateSystemBorderWidth: Integer; virtual;
procedure CalculateWindowInfo; virtual;
function GetBorderRect(ASide: TcxBorder; const ABounds, AWidths: TRect): TRect;
function GetCaption: string;
function GetIcons: TdxSkinFormIcons; virtual;
function GetMDIClientDrawRgn: HRGN; virtual;
function GetSysMenuIcon: HICON; virtual;
function InternalGetScrollBarInfo(AScrollBar: TdxSkinFormScrollBar): TScrollBarInfo;
procedure UpdateCaption(const ANewText: string);
function UpdateCaptionIconStates: Boolean;
function UpdateCaptionSufix: Boolean;
function UpdateIconPressed(AReset: Boolean = False): TdxSkinFormIcon;
procedure UpdateSystemMenu;
public
constructor Create(AController: TdxSkinFormController); virtual;
destructor Destroy; override;
procedure Calculate(AUpdateRgn: HRGN); virtual;
function ClientToScreen(const P: TPoint): TPoint; overload;
function ClientToScreen(const R: TRect): TRect; overload;
function CreateDrawRgn: HRGN; virtual;
function GetHitTest(AHitPoint: TPoint; AHitTest: Integer = 0): Integer;
function GetIconHitTest: TdxSkinFormIcon;
function GetIconHitTestFromHitTest(AHitTest: Integer): TdxSkinFormIcon;
function GetScrollBarHitTest(var AScrollBar: TdxSkinFormScrollBar; var APart: TcxScrollBarPart): Boolean;
function ScreenToClient(const P: TSmallPoint): TPoint; overload;
function ScreenToClient(const P: TPoint): TPoint; overload;
function ScreenToClient(const R: TRect): TRect; overload;
procedure UpdateFormCaption;
property Active: Boolean read FActive write SetActive;
property BorderBounds[ASide: TcxBorder]: TRect read GetBorderBounds;
property BorderWidths: TRect read FBorderWidths;
property BoundsNoBorders: TRect read FBoundsNoBorders;
property ButtonPressed: Boolean read GetButtonPressed;
property Caption: string read GetCaption;
property CaptionBounds: TRect read FCaptionBounds;
property CaptionFont: TFont read FCaptionFont;
property CaptionTextColor: TColor read GetCaptionTextColor;
property CaptionTextShadowColor: TColor read FCaptionTextShadowColor write FCaptionTextShadowColor;
property ClientEdgeSize: TSize read FClientEdgeSize;
property ClientRect: TRect read FClientRect;
property ClientRectOnClient: TRect read GetClientRectOnClient;
property ContentOffset: Integer read FContentOffset;
property Controller: TdxSkinFormController read FController;
property ExStyle: Integer read FExStyle;
property Handle: HWND read GetHandle;
property HasBorder: Boolean read GetHasBorder;
property HasCaption: Boolean read GetHasCaption;
property HasCaptionTextShadow: Boolean read GetHasCaptionTextShadow;
property HasClientEdge: Boolean read FHasClientEdge;
property HasMenu: Boolean read GetHasMenu;
property IconBounds[AIcon: TdxSkinFormIcon]: TRect read GetIconBounds;
property IconPressed: TdxSkinFormIcon read FIconPressed write FIconPressed;
property Icons: TdxSkinFormIcons read FIcons;
property IconState[AIcon: TdxSkinFormIcon]: TdxSkinElementState read GetIconState;
property IsAlphaBlendUsed: Boolean read GetIsAlphaBlendUsed;
property IsDialog: Boolean read GetIsDialog;
property IsIconic: Boolean read GetIsIconic;
property IsMDIChild: Boolean read FIsMDIChild;
property IsMDIClient: Boolean read FIsMDIClient;
property IsSizebox: Boolean read GetIsSizeBox;
property IsZoomed: Boolean read GetIsZoomed;
property MenuSeparatorBounds: TRect read FMenuSeparatorBounds;
property NativeBorderWidth: Boolean read GetNativeBorderWidth;
property NeedCheckNonClientSize: Boolean read GetNeedCheckNonClientSize;
property Painter: TcxCustomLookAndFeelPainterClass read FPainter;
property PainterInfo: TdxSkinLookAndFeelPainterInfo read FPainterInfo;
property ScrollAreaBounds[AItem: TdxScrollAreaElement]: TRect read GetScrollAreaBounds;
property ScrollAreaElements: TdxScrollAreaElements read FScrollAreaElements;
property ScrollBarInfo[AScrollBar: TdxSkinFormScrollBar]: TScrollBarInfo read GetScrollBarInfo;
property ScrollBarPartBounds[AScrollBar: TdxSkinFormScrollBar; APart: TcxScrollBarPart]: TRect read GetScrollBarPartBounds;
property ScrollBarPartState[AScrollBar: TdxSkinFormScrollBar; APart: TcxScrollBarPart]: TcxButtonState read GetScrollBarPartState;
property SizeArea[ASide: TcxBorder]: TRect read GetSizeArea;
property SizeCorners[ACorner: TdxSkinFormCorner]: TRect read GetSizeCorners;
property SizeFrame: TSize read FSizeFrame;
property SizeType: Integer read FSizeType write SetSizeType;
property SkinBorderWidth[ASide: TcxBorder]: Integer read GetSkinBorderWidth;
property Style: Integer read FStyle;
property SysMenuIcon: HICON read FSysMenuIcon;
property SystemSizeFrame: TSize read FSystemSizeFrame;
property ThemeActive: Boolean read GetThemeActive;
property ThemeActiveAssigned: Boolean read FThemeActiveAssigned write FThemeActiveAssigned;
property ToolWindow: Boolean read GetToolWindow;
property TrackedScrollBar: TdxScrollAreaElement read FTrackedScrollBar write FTrackedScrollBar;
property TrackIcon: TdxSkinFormIcon read FTrackIcon write FTrackIcon;
property UpdateRgn: HRGN read FUpdateRgn write SetUpdateRgn;
property WindowBounds: TRect read FWindowBounds;
property WindowRect: TRect read FWindowRect;
end;
{ TdxSkinFormPainter }
TdxSkinFormPainter = class(TObject)
private
FBaseCanvas: TCanvas;
FBordersCache: array[TcxBorder] of TdxSkinElementCache;
FCanvas: TcxCanvas;
FDC: HDC;
FIconsCache: array[TdxSkinFormIcon] of TdxSkinElementCache;
FNeedRelease: Boolean;
FPainter: TcxCustomLookAndFeelPainterClass;
FPainterInfo: TdxSkinLookAndFeelPainterInfo;
FViewInfo: TdxSkinFormNonClientAreaInfo;
function GetActive: Boolean;
function GetIconElement(AIcon: TdxSkinFormIcon): TdxSkinElement;
function GetIsBordersThin: Boolean;
protected
procedure CreateCacheInfos;
procedure DrawBackground(DC: HDC; const R: TRect); virtual;
procedure DrawCaptionText(DC: HDC; R: TRect; const AText: string); virtual;
procedure DrawScrollAreaElements(DC: HDC); virtual;
procedure DrawScrollBar(DC: HDC; AScrollBar: TdxSkinFormScrollBar; const R: TRect); virtual;
procedure DrawSizeGrip(DC: HDC; const R: TRect);
procedure DrawWindowCaption(DC: HDC; const R: TRect; AElement: TdxSkinElement); virtual;
procedure DrawWindowCaptionBackground(DC: HDC; const R: TRect; AElement: TdxSkinElement); virtual;
procedure DrawWindowCaptionText(DC: HDC; R: TRect); virtual;
procedure DrawWindowIcon(DC: HDC; const R: TRect; AIcon: TdxSkinFormIcon;
AElement: TdxSkinElement); virtual;
procedure FreeCacheInfos;
procedure InternalDrawBorder(DC: HDC; const R: TRect; ASide: TcxBorder;
AFillBackground: Boolean);
procedure InternalDrawBorders;
procedure InternalDrawCaption(const R: TRect; AElement: TdxSkinElement);
procedure InternalDrawThinBorders;
public
constructor Create(AViewInfo: TdxSkinFormNonClientAreaInfo); virtual;
destructor Destroy; override;
procedure BeginPaint(ADestDC: HDC = 0);
procedure DrawContentOffsetArea; virtual;
procedure DrawMDIClientEdgeArea; virtual;
procedure DrawMenuSeparator; virtual;
procedure DrawWindowBackground; virtual;
procedure DrawWindowBorder; virtual;
procedure EndPaint;
function IsRectVisible(const R: TRect): Boolean;
function SelectDC(DC: HDC): Integer;
property Active: Boolean read GetActive;
property Canvas: TcxCanvas read FCanvas;
property IconElements[AIcon: TdxSkinFormIcon]: TdxSkinElement read GetIconElement;
property IsBordersThin: Boolean read GetIsBordersThin;
property Painter: TcxCustomLookAndFeelPainterClass read FPainter;
property PainterInfo: TdxSkinLookAndFeelPainterInfo read FPainterInfo;
property ViewInfo: TdxSkinFormNonClientAreaInfo read FViewInfo;
end;
{ TdxSkinCustomControlViewInfo }
TdxSkinCustomControlViewInfo = class(TObject)
private
FController: TdxSkinWinController;
function GetClientRect: TRect;
function GetIsEnabled: Boolean;
function GetIsFocused: Boolean;
function GetIsMouseAtControl: Boolean;
public
constructor Create(AController: TdxSkinWinController); virtual;
property ClientRect: TRect read GetClientRect;
property Controller: TdxSkinWinController read FController;
property IsEnabled: Boolean read GetIsEnabled;
property IsFocused: Boolean read GetIsFocused;
property IsMouseAtControl: Boolean read GetIsMouseAtControl;
end;
TdxSkinCustomControlViewInfoClass = class of TdxSkinCustomControlViewInfo;
{ TdxSkinForm }
TdxSkinForm = class(TForm)
private
FController: TdxSkinFormController;
protected
procedure CreateWindowHandle(const Params: TCreateParams); override;
procedure DefaultWndProc(var Message: TMessage);
procedure DestroyWindowHandle; override;
procedure FinalizeController;
procedure InitializeController;
procedure WndProc(var Message: TMessage); override;
property Controller: TdxSkinFormController read FController;
end;
{ TdxSkinFormHelper }
TdxSkinFormHelper = class(TObject)
class function GetActiveMDIChild(AHandle: HWND): TCustomForm;
class function GetContentOffset(AHandle: HWND): Integer;
class function GetForm(AHandle: HWND): TCustomForm;
class function GetZoomedMDIChild(AHandle: HWND): TCustomForm;
class function HasClientEdge(AHandle: HWND): Boolean;
class function IsAlphaBlendUsed(AHandle: HWND): Boolean;
end;
{ TdxSkinButtonViewInfo }
TdxSkinButtonViewInfo = class(TdxSkinCustomControlViewInfo)
private
FActive: Boolean;
FCaption: string;
FState: TcxButtonState;
function GetIsDefault: Boolean;
function GetIsPressed: Boolean;
function GetWordWrap: Boolean;
procedure SetActive(AActive: Boolean);
procedure SetState(AState: TcxButtonState);
protected
procedure CMFocusChanged(var Message: TCMFocusChanged);
procedure UpdateButtonState;
public
constructor Create(AController: TdxSkinWinController); override;
property Active: Boolean read FActive write SetActive;
property Caption: string read FCaption;
property IsDefault: Boolean read GetIsDefault;
property IsPressed: Boolean read GetIsPressed;
property State: TcxButtonState read FState write SetState;
property WordWrap: Boolean read GetWordWrap;
end;
{ TdxSkinCustomPainter }
TdxSkinCustomControlPainter = class(TObject)
private
FCanvas: TCanvas;
FcxCanvas: TcxCanvas;
FDC: HDC;
FNeedRelease: Boolean;
FViewInfo: TdxSkinCustomControlViewInfo;
function GetController: TdxSkinWinController;
function GetPainter: TcxCustomLookAndFeelPainterClass;
protected
procedure BeginPaint(DC: HDC = 0);
procedure EndPaint;
//
property NeedRelease: Boolean read FNeedRelease;
public
constructor Create(AViewInfo: TdxSkinCustomControlViewInfo);
destructor Destroy; override;
procedure DrawBackground;
procedure DrawButton(const ACaption: string; const R: TRect;
AState: TcxButtonState; AWordWrap: Boolean; AFont: TFont);
procedure DrawFocus(const R: TRect);
property Canvas: TcxCanvas read FcxCanvas;
property Controller: TdxSkinWinController read GetController;
property Painter: TcxCustomLookAndFeelPainterClass read GetPainter;
property ViewInfo: TdxSkinCustomControlViewInfo read FViewInfo;
end;
{ TdxSkinCustomController }
TdxSkinCustomController = class(TdxSkinWinController)
private
FPainter: TdxSkinCustomControlPainter;
FViewInfo: TdxSkinCustomControlViewInfo;
protected
class function GetViewInfoClass: TdxSkinCustomControlViewInfoClass; virtual;
function GetMaster(AHandle: HWND): TdxSkinWinController; override;
procedure InitializePainter; override;
procedure WndProc(var AMessage: TMessage); override;
// Messages
function WMEraseBk(var AMessage: TWMEraseBkgnd): Boolean; virtual;
function WMPaint(var AMessage: TWMPaint): Boolean; virtual;
public
constructor Create(AHandle: HWND); override;
destructor Destroy; override;
procedure DrawBackground(DC: HDC = 0);
procedure DrawContent(DC: HDC = 0); virtual;
property Painter: TdxSkinCustomControlPainter read FPainter;
property ViewInfo: TdxSkinCustomControlViewInfo read FViewInfo;
end;
{ TdxSkinButtonController }
TdxSkinButtonController = class(TdxSkinCustomController)
protected
function GetViewInfo: TdxSkinButtonViewInfo;
class function GetViewInfoClass: TdxSkinCustomControlViewInfoClass; override;
procedure MouseLeave; override;
procedure WndProc(var AMessage: TMessage); override;
public
procedure DrawContent(DC: HDC = 0); override;
property ViewInfo: TdxSkinButtonViewInfo read GetViewInfo;
end;
{ TdxSkinPanelController }
TdxSkinPanelController = class(TdxSkinCustomController)
private
FPainting: Boolean;
protected
function WMEraseBk(var AMessage: TWMEraseBkgnd): Boolean; override;
function WMPaint(var AMessage: TWMPaint): Boolean; override;
function WMPrintClient(var AMessage: TWMPrintClient): Boolean; virtual;
procedure InternalDrawBackground(APanel: TCustomPanel; const R: TRect);
procedure WndProc(var AMessage: TMessage); override;
public
procedure DrawContent(DC: HDC = 0); override;
end;
{ TdxSkinFrameController }
TdxSkinFrameController = class(TdxSkinFormController)
private
function GetIsTransparent: Boolean;
protected
function GetMaster(AHandle: HWND): TdxSkinWinController; override;
procedure DrawWindowBackground(DC: HDC); override;
procedure InitializePainter; override;
//
property IsTransparent: Boolean read GetIsTransparent;
end;
var
dxSkinControllersList: TList;
dxSkinGetControllerClassForWindowProc: TdxSkinGetControllerClassForWindowProc;
function dxSkinGetControllerClassForWindow(AWnd: HWND): TdxSkinWinControllerClass;
implementation
const
SC_TITLEDBLCLICK = 61490;
{$IFNDEF DELPHI7}
ICON_SMALL2 = 2;
WM_NCMOUSELEAVE = $02A2;
WM_NCMOUSEHOVER = $02A0;
{$ENDIF}
WM_NCUAHDRAWCAPTION = $00AE;
WM_NCUAHDRAWFRAME = $00AF;
WM_SYNCPAINT = $0088;
// hittests
CornerHitTests: array[TdxSkinFormCorner] of DWORD =
(HTTOPLEFT, HTTOPRIGHT, HTBOTTOMLEFT, HTBOTTOMRIGHT);
ResizeHitTests: array[TcxBorder] of DWORD =
(HTLEFT, HTTOP, HTRIGHT, HTBOTTOM);
IconsHitTest: array[TdxSkinFormIcon] of DWORD =
(HTSYSMENU, HTHELP, HTMINBUTTON, HTMAXBUTTON, HTMAXBUTTON, HTCLOSE);
IconCommand: array[TdxSkinFormIcon] of Integer =
(SC_DEFAULT, SC_CONTEXTHELP, SC_MINIMIZE, SC_MAXIMIZE, SC_RESTORE, SC_CLOSE);
sdxMDICaptionFormat = '%s - [%s]';
const
CaptionFlags = DT_VCENTER or DT_SINGLELINE or DT_EDITCONTROL or
DT_END_ELLIPSIS or DT_NOPREFIX;
FrameStates: array[Boolean] of TdxSkinElementState = (esActiveDisabled, esActive);
type
{ TdxSkinWinControllerHelper }
TdxSkinWinControllerHelper = class(TObject)
private
FHandle: HWND;
protected
procedure InternalInitializeEngine(AHandle: HWND);
procedure WndProc(var AMsg: TMessage);
public
constructor Create;
destructor Destroy; override;
procedure ChildChanged(AHandle: HWND);
procedure FinalizeEngine(AHandle: HWND);
procedure InitializeEngine(AHandle: HWND);
property Handle: HWND read FHandle;
end;
var
FormControllers: TcxObjectList;
SkinHelper: TdxSkinWinControllerHelper;
WndProcHookHandle: HHOOK;
type
TControlAccess = class(TControl);
TCustomFormAccess = class(TCustomForm);
TCustomFrameAccess = class(TCustomFrame);
TCustomLabelAccess = class(TCustomLabel);
TCustomPanelAccess = class(TCustomPanel);
function GetControllerByControl(AControl: TWinControl): TdxSkinWinController;
var
I: Integer;
begin
Result := nil;
if AControl = nil then Exit;
for I := 0 to FormControllers.Count - 1 do
if TdxSkinWinController(FormControllers.Items[I]).WinControl = AControl then
begin
Result := TdxSkinWinController(FormControllers.Items[I]);
Break;
end;
end;
function GetControllerByHandle(AHandle: HWND): TdxSkinWinController;
var
I: Integer;
begin
Result := nil;
for I := 0 to FormControllers.Count - 1 do
if TdxSkinWinController(FormControllers.Items[I]).Handle = AHandle then
begin
Result := TdxSkinWinController(FormControllers.Items[I]);
Break;
end;
end;
function GetWindowCaption(AWnd: HWND): string;
var
AControl: TControl;
L: Integer;
begin
AControl := FindControl(AWnd);
if AControl <> nil then // todo: bug with Delphi2007
Result := TControlAccess(AControl).Caption
else
begin
L := SendMessage(AWnd, WM_GETTEXTLENGTH, 0, 0);
SetLength(Result, L);
if L > 0 then
SendMessage(AWnd, WM_GETTEXT, L + 1, Integer(@Result[1]));
end;
end;
function GetWindowClass(AWnd: HWND): string;
var
AClassName: array[Byte] of Char;
begin
if GetClassName(AWnd, @AClassName[0], 256) > 0 then
Result := AClassName
else
Result := '';
end;
procedure RefreshControllers;
var
AIndex: Integer;
begin
if csDestroying in Application.ComponentState then
Exit;
for AIndex := 0 to FormControllers.Count - 1 do
with TdxSkinWinController(FormControllers[AIndex]) do
begin
FLookAndFeelPainter := nil;
if IsMDIClient then
Continue;
if FormControllers[AIndex] is TdxSkinFormController then
TdxSkinFormController(FormControllers[AIndex]).RefreshController;
end;
for AIndex := 0 to FormControllers.Count - 1 do
TdxSkinWinController(FormControllers[AIndex]).Update;
end;
//
function dxSkinGetControllerClassForWindow(AWnd: HWND): TdxSkinWinControllerClass;
var
AControl: TControl;
begin
Result := nil;
if TdxSkinWinController.IsMDIClientWindow(AWnd) then
Result := TdxSkinFormController
else
begin
AControl := FindControl(AWnd);
if AControl <> nil then
begin
if AControl is TCustomForm then
Result := TdxSkinFormController;
if AControl is TCustomFrame then
Result := TdxSkinFrameController;
if SameText(AControl.ClassName, 'TButton') then
Result := TdxSkinButtonController;
if SameText(AControl.ClassName, 'TPanel') then
Result := TdxSkinPanelController;
end;
end;
end;
function dxSkinsWndProcHook(Code: Integer; wParam: WParam; lParam: LParam): LRESULT; stdcall;
var
AMsg: PCWPStruct;
begin
AMsg := PCWPStruct(lParam);
Result := CallNextHookEx(WndProcHookHandle, Code, wParam, lParam);
if IsDesigning then Exit;
case AMsg.message of
WM_CHILDACTIVATE, WM_MDIACTIVATE:
SkinHelper.ChildChanged(AMsg.hwnd);
WM_CREATE, WM_MDICREATE:
SkinHelper.InitializeEngine(AMsg.hwnd);
WM_DESTROY, WM_MDIDESTROY:
if AMsg.wParam = 0 then
SkinHelper.FinalizeEngine(AMsg.hwnd);
end;
end;
var
SetScrollInfo: function(hWnd: HWND; BarFlag: Integer;
const ScrollInfo: TScrollInfo; Redraw: BOOL): Integer; stdcall;
SetScrollPos: function (hWnd: HWND; nBar, nPos: Integer;
bRedraw: BOOL): Integer stdcall;
function My_SetScrollPos(hWnd: HWND; nBar, nPos: Integer;
bRedraw: BOOL): Integer; stdcall;
begin
bRedraw := bRedraw and not TdxSkinFormController.IsSkinActive(hWnd);
Result := SetScrollPos(hWnd, nBar, nPos, bRedraw);
end;
function My_SetScrollInfo(hWnd: HWND; BarFlag: Integer;
const ScrollInfo: TScrollInfo; Redraw: BOOL): Integer; stdcall;
begin
Redraw := Redraw and not TdxSkinFormController.IsSkinActive(hWnd);
Result := SetScrollInfo(hWnd, BarFlag, ScrollInfo, Redraw);
end;
procedure RegisterAssistants;
begin
SetScrollPos := FlatSB_SetScrollPos;
SetScrollInfo := FlatSB_SetScrollInfo;
FlatSB_SetScrollPos := My_SetScrollPos;
FlatSB_SetScrollInfo := My_SetScrollInfo;
dxSkinControllersList := TList.Create;
SkinHelper := TdxSkinWinControllerHelper.Create;
SetHook(WndProcHookHandle, WH_CALLWNDPROC, dxSkinsWndProcHook);
WM_POSTREDRAW := RegisterWindowMessage('WM_POSTREDRAW');
WM_CHILDCHANGED := RegisterWindowMessage('WM_CHILDCHANGED');
WM_POSTCREATE := RegisterWindowMessage('WM_POSTCREATE');
WM_POSTCHECKRGN := RegisterWindowMessage('WM_POSTCHECKRGN');
WM_POSTMSGFORMINIT := RegisterWindowMessage('WM_POSTMSGFORMINIT');
WM_HASOWNSKINENGINE := RegisterWindowMessage('WM_HASOWNSKINENGINE');
FormControllers := TcxObjectList.Create;
end;
procedure UnregisterAssistants;
begin
FlatSB_SetScrollPos := SetScrollPos;
FlatSB_SetScrollInfo := SetScrollInfo;
ReleaseHook(WndProcHookHandle);
FormControllers.Clear; // destroy all active controllers
FreeAndNil(dxSkinControllersList);
FreeAndNil(FormControllers);
FreeAndNil(SkinHelper);
end;
{ TdxSkinCustomControlViewInfo }
constructor TdxSkinCustomControlViewInfo.Create(AController: TdxSkinWinController);
begin
FController := AController;
end;
function TdxSkinCustomControlViewInfo.GetClientRect: TRect;
begin
Result := cxGetWindowRect(Controller.Handle);
OffsetRect(Result, -Result.Left, -Result.Top);
end;
function TdxSkinCustomControlViewInfo.GetIsEnabled: Boolean;
begin
Result := IsWindowEnabled(Controller.Handle);
end;
function TdxSkinCustomControlViewInfo.GetIsFocused: Boolean;
begin
Result := GetFocus = Controller.Handle;
end;
function TdxSkinCustomControlViewInfo.GetIsMouseAtControl: Boolean;
begin
Result := WindowFromPoint(GetMouseCursorPos) = FController.Handle;
end;
{ TdxSkinButtonViewInfo }
constructor TdxSkinButtonViewInfo.Create(AController: TdxSkinWinController);
begin
inherited Create(AController);
FCaption := GetWindowCaption(Controller.Handle);
UpdateButtonState;
end;
procedure TdxSkinButtonViewInfo.CMFocusChanged(var Message: TCMFocusChanged);
begin
if Message.Sender is TButton then
Active := Message.Sender = Controller.WinControl
else
Active := IsDefault;
end;
function TdxSkinButtonViewInfo.GetIsDefault: Boolean;
begin
Result := (Controller.WinControl as TButton).Default;
end;
function TdxSkinButtonViewInfo.GetIsPressed: Boolean;
begin
Result := SendMessage(Controller.Handle, BM_GETSTATE, 0, 0) and BST_PUSHED <> 0;
end;
function TdxSkinButtonViewInfo.GetWordWrap: Boolean;
begin
Result := GetWindowLong(Controller.Handle, GWL_STYLE) and BS_MULTILINE <> 0;
end;
procedure TdxSkinButtonViewInfo.SetActive(AActive: Boolean);
begin
if AActive <> FActive then
begin
FActive := AActive;
UpdateButtonState;
end;
end;
procedure TdxSkinButtonViewInfo.SetState(AState: TcxButtonState);
var
ANewState: TcxButtonState;
begin
if IsEnabled then
ANewState := AState
else
ANewState := cxbsDisabled;
if ANewState <> FState then
begin
FState := ANewState;
if Controller.IsSkinUsed then
Controller.RedrawWindow(True);
end;
end;
procedure TdxSkinButtonViewInfo.UpdateButtonState;
const
DefaultStatesMap: array[Boolean] of TcxButtonState = (cxbsNormal, cxbsDefault);
var
AState: TcxButtonState;
begin
AState := DefaultStatesMap[Active];
if IsMouseAtControl then
AState := cxbsHot;
if IsPressed then
AState := cxbsPressed;
State := AState;
end;
{ TdxSkinForm }
procedure TdxSkinForm.CreateWindowHandle(const Params: TCreateParams);
begin
inherited CreateWindowHandle(Params);
InitializeController;
end;
procedure TdxSkinForm.DestroyWindowHandle;
begin
FinalizeController;
inherited DestroyWindowHandle;
end;
procedure TdxSkinForm.DefaultWndProc(var Message: TMessage);
begin
inherited WndProc(Message);
end;
procedure TdxSkinForm.FinalizeController;
begin
if Controller <> nil then
begin
FormControllers.Remove(Controller);
FreeAndNil(FController);
end;
end;
procedure TdxSkinForm.InitializeController;
begin
FController := TdxSkinFormController.CreateEx(Self);
FormControllers.Add(Controller);
Controller.InitializePainter;
Controller.Update;
PostMessage(Handle, WM_POSTCHECKRGN, 0, 0);
end;
procedure TdxSkinForm.WndProc(var Message: TMessage);
begin
if Message.Msg = WM_HASOWNSKINENGINE then
Message.Result := 1
else
if (Controller = nil) or (Controller.Handle = 0) then
DefaultWndProc(Message)
else
Controller.WndProc(Message);
end;
{ TdxSkinPanelController }
procedure TdxSkinPanelController.DrawContent(DC: HDC = 0);
const
Alignments: array[TAlignment] of Longint = (DT_LEFT, DT_RIGHT, DT_CENTER);
var
AControl: TControl;
R: TRect;
begin
AControl := WinControl;
if Assigned(AControl) and (AControl is TCustomPanel) then
begin
Painter.BeginPaint(DC);
try
R := ViewInfo.ClientRect;
InternalDrawBackground(TCustomPanel(AControl), R);
with TCustomPanelAccess(AControl) do
begin
Painter.Canvas.Brush.Style := bsClear;
Painter.Canvas.Font := Font;
cxDrawText(Painter.Canvas.Handle, Caption, R,
DrawTextBiDiModeFlags(DT_EXPANDTABS or DT_SINGLELINE or DT_VCENTER or
Alignments[Alignment]));
Painter.Canvas.Brush.Style := bsSolid;
PaintControls(Painter.Canvas.Handle, nil);
end;
finally
Painter.EndPaint;
end;
end;
end;
procedure TdxSkinPanelController.InternalDrawBackground(APanel: TCustomPanel;
const R: TRect);
begin
with TCustomPanelAccess(APanel) do
begin
if csParentBackground in ControlStyle then
cxDrawTransparentControlBackground(APanel, Painter.Canvas, R, False)
else
if Color = clBtnFace then
Painter.Painter.DrawPanelContent(Painter.Canvas, R, False)
else
Painter.Painter.DrawPanelBackground(Painter.Canvas, APanel, R, Color);
if (BevelOuter <> bvNone) or (BevelInner <> bvNone) then
Painter.Painter.DrawPanelBorders(Painter.Canvas, R);
end;
end;
function TdxSkinPanelController.WMEraseBk(var AMessage: TWMEraseBkgnd): Boolean;
begin
Result := False;
end;
function TdxSkinPanelController.WMPaint(var AMessage: TWMPaint): Boolean;
begin
Result := False;
if not FPainting then
begin
FPainting := True;
try
Result := inherited WMPaint(AMessage);
finally
FPainting := False;
end;
end;
end;
function TdxSkinPanelController.WMPrintClient(var AMessage: TWMPrintClient): Boolean;
begin
Result := (AMessage.Result <> 1) and
((AMessage.Flags and PRF_CHECKVISIBLE = 0) or IsWindowVisible(Handle)) and
WMPaint(TWMPaint(AMessage));
end;
procedure TdxSkinPanelController.WndProc(var AMessage: TMessage);
var
AHandled: Boolean;
begin
AHandled := False;
if IsSkinUsed then
begin
case AMessage.Msg of
CM_COLORCHANGED:
RedrawWindow(False);
WM_PRINTCLIENT:
AHandled := WMPrintClient(TWMPrintClient(AMessage));
end;
end;
if not AHandled then
inherited WndProc(AMessage);
end;
{ TdxSkinFrameController }
procedure TdxSkinFrameController.DrawWindowBackground(DC: HDC);
begin
if not IsTransparent then
inherited DrawWindowBackground(DC)
else
begin
Painter.BeginPaint(DC);
try
cxDrawTransparentControlBackground(
WinControl, Painter.Canvas, ViewInfo.BoundsNoBorders, False);
finally
Painter.EndPaint;
end;
end;
end;
function TdxSkinFrameController.GetIsTransparent: Boolean;
begin
Result := Assigned(WinControl) and (csParentBackground in WinControl.ControlStyle);
end;
function TdxSkinFrameController.GetMaster(AHandle: HWND): TdxSkinWinController;
var
AParent: TCustomForm;
begin
Result := nil;
if Assigned(WinControl) then
begin
AParent := GetParentForm(WinControl);
if Assigned(AParent) and GetUseSkinForControl then
Result := GetControllerByHandle(AParent.Handle)
end;
end;
procedure TdxSkinFrameController.InitializePainter;
begin
// nothing to do
end;
{ TdxSkinButtonPainter }
constructor TdxSkinCustomControlPainter.Create(AViewInfo: TdxSkinCustomControlViewInfo);
begin
FViewInfo := AViewInfo;
FCanvas := TCanvas.Create;
FcxCanvas := TcxCanvas.Create(FCanvas);
end;
destructor TdxSkinCustomControlPainter.Destroy;
begin
FViewInfo := nil;
FcxCanvas.Free;
FCanvas.Free;
inherited Destroy;
end;
function TdxSkinCustomControlPainter.GetController: TdxSkinWinController;
begin
Result := ViewInfo.Controller;
end;
function TdxSkinCustomControlPainter.GetPainter: TcxCustomLookAndFeelPainterClass;
begin
Result := Controller.LookAndFeelPainter;
end;
procedure TdxSkinCustomControlPainter.BeginPaint(DC: HDC = 0);
const
Flags = DCX_CACHE or DCX_CLIPSIBLINGS or DCX_WINDOW or DCX_VALIDATE;
begin
FNeedRelease := DC = 0;
if NeedRelease then
DC := GetDCEx(Controller.Handle, 0, Flags);
FDC := DC;
Canvas.Canvas.Handle := FDC;
end;
procedure TdxSkinCustomControlPainter.EndPaint;
begin
Canvas.Canvas.Handle := 0;
if NeedRelease then
ReleaseDC(Controller.Handle, FDC);
end;
procedure TdxSkinCustomControlPainter.DrawBackground;
var
AControl: TWinControl;
begin
AControl := Controller.WinControl;
if AControl <> nil then
cxDrawTransparentControlBackground(AControl, Canvas, ViewInfo.ClientRect);
end;
procedure TdxSkinCustomControlPainter.DrawButton(const ACaption: string;
const R: TRect; AState: TcxButtonState; AWordWrap: Boolean; AFont: TFont);
procedure DrawButtonText(const R: TRect; AFlags: Integer; ATextColor: TColor);
begin
Canvas.Font := AFont;
Canvas.Font.Color := ATextColor;
Canvas.DrawText(ACaption, R, AFlags, True);
end;
const
TextAlignFlagsMap: array[Boolean] of Integer = (cxSingleLine, cxWordBreak);
var
AFlags: Integer;
begin
Painter.DrawButton(Canvas, R, '', AState, False, clDefault, clDefault, False);
if Length(ACaption) > 0 then
begin
Canvas.Brush.Style := bsClear;
AFlags := cxAlignCenter or cxShowPrefix or TextAlignFlagsMap[AWordWrap];
if AState = cxbsDisabled then
DrawButtonText(R, AFlags, clBtnHighlight);
DrawButtonText(R, AFlags, Painter.ButtonSymbolColor(AState));
end;
end;
procedure TdxSkinCustomControlPainter.DrawFocus(const R: TRect);
begin
Canvas.Brush.Style := bsSolid;
Canvas.DrawFocusRect(R);
end;
{ TdxSkinController }
constructor TdxSkinController.Create(AOwner: TComponent);
begin
inherited Create(AOwner);
if dxSkinControllersList <> nil then
dxSkinControllersList.Add(Self);
Changed;
end;
destructor TdxSkinController.Destroy;
begin
Changed;
if dxSkinControllersList <> nil then
dxSkinControllersList.Remove(Self);
inherited Destroy;
end;
procedure TdxSkinController.Refresh;
begin
Changed;
end;
class function TdxSkinController.GetFormSkin(AForm: TCustomForm; var ASkinName: string): Boolean;
var
AController: TdxSkinWinController;
begin
AController := GetControllerByHandle(AForm.Handle);
Result := Assigned(AController) and AController.GetSkinName(ASkinName);
end;
procedure TdxSkinController.Changed;
begin
RefreshControllers;
end;
function TdxSkinController.DoSkinControl(AControl: TWinControl): Boolean;
var
AUseSkin: Boolean;
begin
Result := AControl <> nil;
if Result then
begin
AUseSkin := True;
if Assigned(OnSkinControl) then
OnSkinControl(Self, AControl, AUseSkin);
Result := AUseSkin
end
end;
function TdxSkinController.DoSkinForm(
AForm: TCustomForm): TdxSkinLookAndFeelPainterClass;
var
ASkinName: string;
AUseSkin: Boolean;
begin
if (AForm <> nil) and (SendMessage(AForm.Handle, dxWMGetSkinnedMessage, 0, 0) = 1) then
begin
ASkinName := '';
AUseSkin := False;
end
else
begin
ASkinName := SkinName;
AUseSkin := UseSkins and not NativeStyle;
end;
Result := DoSkinFormEx(AForm, ASkinName, AUseSkin);
end;
function TdxSkinController.DoSkinFormEx(AForm: TCustomForm;
var ASkinName: string; var AUseSkin: Boolean): TdxSkinLookAndFeelPainterClass;
var
APainter: TcxCustomLookAndFeelPainterClass;
begin
Result := nil;
if AForm = nil then Exit;
if Assigned(OnSkinForm) then
OnSkinForm(Self, AForm, ASkinName, AUseSkin);
if AUseSkin and (ASkinName <> '') and GetExtendedStylePainters.GetPainterByName(
ASkinName, APainter) and (APainter.InheritsFrom(TdxSkinLookAndFeelPainter)) then
Result := TdxSkinLookAndFeelPainterClass(APainter);
end;
procedure TdxSkinController.Loaded;
begin
inherited Loaded;
Changed;
end;
procedure TdxSkinController.MasterLookAndFeelChanged(
Sender: TcxLookAndFeel; AChangedValues: TcxLookAndFeelValues);
begin
inherited MasterLookAndFeelChanged(Sender, AChangedValues);
Changed;
end;
procedure TdxSkinController.MasterLookAndFeelDestroying(
Sender: TcxLookAndFeel);
begin
inherited MasterLookAndFeelDestroying(Sender);
Changed;
end;
function TdxSkinController.GetUseSkins: Boolean;
begin
Result := cxUseSkins;
end;
procedure TdxSkinController.SetUseSkins(Value: Boolean);
begin
if Value <> UseSkins then
begin
cxLookAndFeels.cxUseSkins := Value;
RootLookAndFeel.Refresh;
Changed;
end;
end;
{ TdxSkinButtonController }
function TdxSkinButtonController.GetViewInfo: TdxSkinButtonViewInfo;
begin
Result := TdxSkinButtonViewInfo(inherited ViewInfo);
end;
class function TdxSkinButtonController.GetViewInfoClass: TdxSkinCustomControlViewInfoClass;
begin
Result := TdxSkinButtonViewInfo;
end;
procedure TdxSkinButtonController.DrawContent(DC: HDC = 0);
begin
Painter.BeginPaint(DC);
try
Painter.DrawBackground;
with TdxSkinButtonViewInfo(ViewInfo) do
begin
Painter.DrawButton(Caption, ClientRect, State, WordWrap,
TControlAccess(WinControl).Font);
if IsFocused then
Painter.DrawFocus(cxRectInflate(ClientRect, -4, -4));
end;
finally
Painter.EndPaint;
end;
end;
procedure TdxSkinButtonController.MouseLeave;
begin
inherited MouseLeave;
ViewInfo.UpdateButtonState;
end;
procedure TdxSkinButtonController.WndProc(var AMessage: TMessage);
const
AStates: array[Boolean] of TcxButtonState = (cxbsDisabled, cxbsNormal);
begin
if AMessage.Msg = CM_FOCUSCHANGED then
ViewInfo.CMFocusChanged(TCMFocusChanged(AMessage));
inherited WndProc(AMessage);
case AMessage.Msg of
WM_ENABLE, BM_SETSTATE, WM_MOUSELEAVE,
WM_MOUSEMOVE, CM_MOUSELEAVE, WM_MOUSEHOVER:
ViewInfo.UpdateButtonState;
WM_UPDATEUISTATE:
if IsSkinUsed and (AMessage.WParamLo in [UIS_SET, UIS_CLEAR]) then
DrawContent;
WM_SETTEXT:
begin
ViewInfo.FCaption := TWMSetText(AMessage).Text;
if IsSkinUsed then
DrawContent;
end;
end;
end;
{ TdxSkinFormNonClientAreaInfo }
constructor TdxSkinFormNonClientAreaInfo.Create(AController: TdxSkinFormController);
begin
FController := AController;
FCaptionFont := TFont.Create;
FTrackedScrollBar := saeSizeGrip;
UpdateFormCaption;
FActive := True;
end;
destructor TdxSkinFormNonClientAreaInfo.Destroy;
begin
UpdateRgn := 0;
FreeAndNil(FCaptionFont);
inherited Destroy;
end;
procedure TdxSkinFormNonClientAreaInfo.Calculate(AUpdateRgn: HRGN);
begin
FIsMDIChild := TdxSkinFormController.IsMDIChildWindow(Handle);
FIsMDIClient := TdxSkinFormController.IsMDIClientWindow(Handle);
FHasMenu := IsMenu(GetMenu(Handle)) and not FIsMDIChild;
FHasClientEdge := TdxSkinFormHelper.HasClientEdge(Handle);
FPainter := Controller.LookAndFeelPainter;
GetExtendedStylePainters.GetPainterData(Painter, FPainterInfo);
CalculateWindowInfo;
CalculateFrameSizes;
FContentOffset := TdxSkinFormHelper.GetContentOffset(Handle);
FWindowBounds := cxRectOffset(WindowRect, -WindowRect.Left, -WindowRect.Top);
FBoundsNOBorders := cxRectOffset(ClientRect, cxPointInvert(WindowRect.TopLeft));
FIcons := GetIcons;
FSysMenuIcon := GetSysMenuIcon;
CalculateFontInfo;
CalculateBorderWidths;
FMenuSeparatorBounds := cxRectSetBottom(FBoundsNOBorders, FBoundsNOBorders.Top, 1);
CalculateBordersInfo;
CalculateCaptionIconsInfo;
CalculateScrollArea;
end;
function TdxSkinFormNonClientAreaInfo.ClientToScreen(const P: TPoint): TPoint;
begin
Result := cxPointOffset(P, WindowRect.TopLeft);
end;
function TdxSkinFormNonClientAreaInfo.ClientToScreen(const R: TRect): TRect;
begin
Result := cxRectOffset(R, WindowRect.TopLeft);
end;
function TdxSkinFormNonClientAreaInfo.CreateDrawRgn: HRGN;
var
ARgn: HRgn;
ASide: TcxBorder;
AItem: TdxScrollAreaElement;
begin
Result := CreateRectRgnIndirect(cxNullRect);
for ASide := Low(TcxBorder) to High(TcxBorder) do
begin
if cxRectIsEmpty(BorderBounds[ASide]) then Continue;
ARgn := CreateRectRgnIndirect(ClientToScreen(BorderBounds[ASide]));
CombineRgn(Result, Result, ARgn, RGN_OR);
DeleteObject(ARgn);
end;
if IsMDIClient then
begin
ARgn := GetMDIClientDrawRgn;
CombineRgn(Result, Result, ARgn, RGN_OR);
DeleteObject(ARgn);
end;
if HasMenu then
begin
ARgn := CreateRectRgnIndirect(ClientToScreen(MenuSeparatorBounds));
CombineRgn(Result, Result, ARgn, RGN_OR);
DeleteObject(ARgn);
end;
for AItem := Low(TdxScrollAreaElement) to High(TdxScrollAreaElement) do
if AItem in FScrollAreaElements then
begin
ARgn := CreateRectRgnIndirect(ClientToScreen(ScrollAreaBounds[AItem]));
CombineRgn(Result, Result, ARgn, RGN_OR);
DeleteObject(ARgn);
end;
end;
function TdxSkinFormNonClientAreaInfo.GetHitTest(AHitPoint: TPoint; AHitTest: Integer = 0): Integer;
var
ASide: TcxBorder;
AIcon: TdxSkinFormIcon;
ACorner: TdxSkinFormCorner;
begin
Result := AHitTest;
AHitPoint := ScreenToClient(AHitPoint);
if IsSizebox and not IsZoomed then
begin
for ACorner := sfcLeftTop to sfcRightBottom do
if (Result = HTNOWHERE) and PtInRect(SizeCorners[ACorner], AHitPoint) then
Result := CornerHitTests[ACorner];
for ASide := bLeft to bBottom do
if (Result = HTNOWHERE) and PtInRect(SizeArea[ASide], AHitPoint) then
Result := ResizeHitTests[ASide];
end;
if (Result = HTNOWHERE) and PtInRect(BorderBounds[bTop], AHitPoint) then
begin
Result := HTCAPTION;
// correct 'restore' hittest
IconsHitTest[sfiRestore] := HTMAXBUTTON;
if IsIconic then
IconsHitTest[sfiRestore] := HTMINBUTTON;
//
for AIcon := sfiMenu to sfiClose do
begin
if PtInRect(IconBounds[AIcon], AHitPoint) then
Result := IconsHitTest[AIcon];
end;
end;
end;
function TdxSkinFormNonClientAreaInfo.GetIconHitTest: TdxSkinFormIcon;
begin
Result := GetIconHitTestFromHitTest(GetHitTest(GetMouseCursorPos));
end;
function TdxSkinFormNonClientAreaInfo.GetIconHitTestFromHitTest(AHitTest: Integer): TdxSkinFormIcon;
const
MinBtnIconsMap: array[Boolean] of TdxSkinFormIcon = (sfiMinimize, sfiRestore);
MaxBtnIconsMap: array[Boolean] of TdxSkinFormIcon = (sfiMaximize, sfiRestore);
begin
Result := sfiMenu;
case AHitTest of
HTHELP:
Result := sfiHelp;
HTMINBUTTON:
Result := MinBtnIconsMap[IsIconic];
HTMAXBUTTON:
Result := MaxBtnIconsMap[IsZoomed];
HTCLOSE:
Result := sfiClose;
end;
end;
function TdxSkinFormNonClientAreaInfo.GetScrollBarHitTest(
var AScrollBar: TdxSkinFormScrollBar; var APart: TcxScrollBarPart): Boolean;
var
APoint: TPoint;
AItem: TdxSkinFormScrollBar;
APartItem: TcxScrollBarPart;
begin
Result := False;
APoint := ScreenToClient(GetMouseCursorPos);
for AItem := saeHorzScroll to saeVertScroll do
begin
if not (AItem in ScrollAreaElements) or
not PtInRect(ScrollAreaBounds[AItem], APoint) then Continue;
AScrollBar := AItem;
for APartItem := sbpLineUp to sbpPageDown do
begin
Result := (ScrollBarPartState[AItem, APartItem] <> cxbsDefault) and
PtInRect(ScrollBarPartBounds[AItem, APartItem], APoint);
if Result then
begin
APart := APartItem;
Exit;
end;
end;
end;
end;
function TdxSkinFormNonClientAreaInfo.ScreenToClient(const P: TSmallPoint): TPoint;
begin
Result := ScreenToClient(SmallPointToPoint(P));
end;
function TdxSkinFormNonClientAreaInfo.ScreenToClient(const P: TPoint): TPoint;
begin
Result := cxPointOffset(P, cxPointInvert(WindowRect.TopLeft));
end;
function TdxSkinFormNonClientAreaInfo.ScreenToClient(const R: TRect): TRect;
begin
Result := cxRectOffset(R, cxPointInvert(WindowRect.TopLeft));
end;
procedure TdxSkinFormNonClientAreaInfo.CalculateBordersInfo;
var
ASide: TcxBorder;
begin
for ASide := Low(TcxBorder) to High(TcxBorder) do
FBorderBounds[ASide] := GetBorderRect(ASide, WindowBounds, BorderWidths);
if IsZoomed and not NativeBorderWidth and IsSizeBox then
OffsetRect(FBorderBounds[bTop], 0, SystemSizeFrame.cy - SizeFrame.cy);
end;
procedure TdxSkinFormNonClientAreaInfo.CalculateBorderWidths;
var
ASystemBordersWidth: Integer;
begin
FBorderWidths := cxNullRect;
if IsIconic then
FBorderWidths.Top := cxRectHeight(WindowRect)
else
begin
if HasBorder then
begin
if NativeBorderWidth then
begin
ASystemBordersWidth := CalculateSystemBorderWidth;
Inc(FBorderWidths.Left, ASystemBordersWidth);
Inc(FBorderWidths.Right, ASystemBordersWidth);
Inc(FBorderWidths.Top, ASystemBordersWidth);
Inc(FBorderWidths.Bottom, ASystemBordersWidth);
end
else
begin
FBorderWidths := PainterInfo.FormBorderWidths[not ToolWindow];
if ToolWindow then
Inc(FBorderWidths.Top, SizeFrame.cy);
end;
end;
Inc(FBorderWidths.Top, CalculateCaptionHeight);
end;
end;
function TdxSkinFormNonClientAreaInfo.GetCaptionButtonRect(
const ACaptionRect: TRect): TRect;
begin
Result := BorderBounds[bTop];
with GetCaptionContentOffset do
begin
if NativeBorderWidth and IsSizeBox then
Inc(Result.Top, SystemSizeFrame.cy)
else
Inc(Result.Top, Top);
Dec(Result.Bottom, Bottom);
Result.Right := ACaptionRect.Right;
Result.Left := Result.Right - cxRectHeight(Result);
end;
end;
function TdxSkinFormNonClientAreaInfo.CalculateCaptionHeight: Integer;
const
CaptionMetric: array[Boolean] of Integer = (SM_CYCAPTION, SM_CYSMCAPTION);
begin
if not HasCaption then
Result := 0
else
if NativeBorderWidth or ToolWindow then
Result := GetSystemMetrics(CaptionMetric[ToolWindow])
else
Result := SizeFrame.cy + 2 * GetSystemMetrics(SM_CYEDGE) +
Max(GetSystemMetrics(SM_CYSMICON), cxTextHeight(FCaptionFont));
end;
procedure TdxSkinFormNonClientAreaInfo.CalculateCaptionIconsInfo;
const
RestoreAndMaximize = [sfiRestore, sfiMaximize];
var
AIcon: TdxSkinFormIcon;
AIconSize: Integer;
ARealIcons: TdxSkinFormIcons;
R: TRect;
begin
FillChar(FIconBounds, SizeOf(FIconBounds), 0);
AIconSize := GetCaptionIconSize;
FCaptionBounds := GetCaptionBounds;
R := FCaptionBounds;
R.Top := (R.Top + R.Bottom - AIconSize) div 2;
R.Bottom := R.Top + AIconSize;
if sfiMenu in Icons then
begin
FIconBounds[sfiMenu] := cxRectCenter(cxRectSetWidth(R, AIconSize),
AIconSize, AIconSize);
FCaptionBounds.Left := FIconBounds[sfiMenu].Right + dxSkinFormTextOffset;
end
else
FCaptionBounds.Left := cxTextOffset + dxSkinFormTextOffset;
R := GetCaptionButtonRect(FCaptionBounds);
ARealIcons := Icons;
if RestoreAndMaximize * Icons = RestoreAndMaximize then
ARealIcons := ARealIcons - [sfiRestore] + [sfiMinimize];
for AIcon := sfiClose downto sfiHelp do
begin
if not (AIcon in ARealIcons) then Continue;
FIconBounds[AIcon] := R;
OffsetRect(R, -(R.Right - R.Left + dxSkinIconSpacing), 0);
FCaptionBounds.Right := FIconBounds[AIcon].Left - dxSkinFormTextOffset;
end;
if RestoreAndMaximize * Icons = RestoreAndMaximize then
FIconBounds[sfiRestore] := FIconBounds[sfiMinimize];
UpdateCaptionIconStates;
end;
procedure TdxSkinFormNonClientAreaInfo.CalculateFontInfo;
var
AMetrics: TNonClientMetrics;
begin
AMetrics.cbSize := SizeOf(AMetrics);
SystemParametersInfo(SPI_GETNONCLIENTMETRICS, 0, @AMetrics, 0);
if ToolWindow then
FCaptionFont.Handle := CreateFontIndirect(AMetrics.lfSmCaptionFont)
else
begin
FCaptionFont.Handle := CreateFontIndirect(AMetrics.lfCaptionFont);
if PainterInfo <> nil then
FCaptionFont.Size := FCaptionFont.Size + (PainterInfo.FormCaptionDelta - 1);
end;
FCaptionTextShadowColor := clBtnShadow;
FCaptionTextColor[True] := GetSysColor(COLOR_CAPTIONTEXT);
FCaptionTextColor[False] := GetSysColor(COLOR_INACTIVECAPTIONTEXT);
if PainterInfo <> nil then
begin
FCaptionTextColor[True] := PainterInfo.FormFrames[True, bTop].TextColor;
if PainterInfo.FormInactiveColor <> nil then
FCaptionTextColor[False] := PainterInfo.FormInactiveColor.Value;
if PainterInfo.FormTextShadowColor <> nil then
FCaptionTextShadowColor := PainterInfo.FormTextShadowColor.Value;
end;
end;
procedure TdxSkinFormNonClientAreaInfo.CalculateFrameSizes;
begin
FClientEdgeSize := Size(GetSystemMetrics(SM_CXEDGE), GetSystemMetrics(SM_CYEDGE));
FSystemSizeFrame := Size(GetSystemMetrics(SM_CXSIZEFRAME),
GetSystemMetrics(SM_CYSIZEFRAME));
if NativeBorderWidth then
FSizeFrame := SystemSizeFrame
else
FSizeFrame := Size(4, 4);
end;
function TdxSkinFormNonClientAreaInfo.CalculateMargins: TRect;
begin
Result := BorderWidths;
Inc(Result.Bottom, ContentOffset);
Inc(Result.Left, ContentOffset);
Inc(Result.Right, ContentOffset);
Inc(Result.Top, ContentOffset);
if saeVertScroll in ScrollAreaElements then
Inc(Result.Right, GetSystemMetrics(SM_CXVSCROLL));
if saeHorzScroll in ScrollAreaElements then
Inc(Result.Bottom, GetSystemMetrics(SM_CYHSCROLL));
if IsZoomed and IsSizebox then
begin
Inc(Result.Left, SystemSizeFrame.cx - SizeFrame.cx);
Inc(Result.Right, SystemSizeFrame.cx - SizeFrame.cx);
Inc(Result.Top, SystemSizeFrame.cy - SizeFrame.cy);
Inc(Result.Bottom, SystemSizeFrame.cy - SizeFrame.cy);
if saeVertScroll in ScrollAreaElements then
Inc(Result.Right, SizeFrame.cx - BorderWidths.Right);
if saeHorzScroll in ScrollAreaElements then
Inc(Result.Bottom, SizeFrame.cy - BorderWidths.Bottom);
end;
end;
procedure TdxSkinFormNonClientAreaInfo.CalculateScrollArea;
const
BothScrollBars = [saeHorzScroll, saeVertScroll];
begin
FScrollAreaElements := [];
FillChar(FScrollAreaBounds, SizeOf(FScrollAreaBounds), 0);
if UsecxScrollBars then
begin
FScrollBarsInfo[saeHorzScroll] := InternalGetScrollBarInfo(saeHorzScroll);
FScrollBarsInfo[saeVertScroll] := InternalGetScrollBarInfo(saeVertScroll);
if Style and WS_HSCROLL = WS_HSCROLL then
begin
Include(FScrollAreaElements, saeHorzScroll);
FScrollAreaBounds[saeHorzScroll] := cxRectSetTop(BoundsNOBorders,
BoundsNOBorders.Bottom, GetSystemMetrics(SM_CYHSCROLL));
end;
if Style and WS_VSCROLL = WS_VSCROLL then
begin
Include(FScrollAreaElements, saeVertScroll);
FScrollAreaBounds[saeVertScroll] := cxRectSetLeft(BoundsNOBorders,
BoundsNOBorders.Right, GetSystemMetrics(SM_CXVSCROLL));
end;
if BothScrollBars * ScrollAreaElements = BothScrollBars then
begin
Include(FScrollAreaElements, saeSizeGrip);
FScrollAreaBounds[saeHorzScroll].Right := FScrollAreaBounds[saeVertScroll].Left;
FScrollAreaBounds[saeVertScroll].Bottom := FScrollAreaBounds[saeHorzScroll].Top;
FScrollAreaBounds[saeSizeGrip] := cxRect(BoundsNOBorders.BottomRight,
Point(FScrollAreaBounds[saeVertScroll].Right, FScrollAreaBounds[saeHorzScroll].Bottom));
end;
CalculateScrollBarPartsInfo;
end;
end;
procedure TdxSkinFormNonClientAreaInfo.CalculateScrollBarPartInfo(
AScrollBar: TdxSkinFormScrollBar; Pos1, Pos2: Integer; APart: TcxScrollBarPart);
var
P: TPoint;
AState: Integer;
const
Part2StateIndex: array[TcxScrollBarPart] of Integer = (0, 1, 5, 3, 2, 4);
begin
AState := ScrollBarInfo[AScrollBar].rgState[Part2StateIndex[APart]];
FScrollBarPartState[AScrollBar, APart] := cxbsDefault;
if AState and STATE_SYSTEM_INVISIBLE = STATE_SYSTEM_INVISIBLE then Exit;
FScrollBarPartState[AScrollBar, APart] := cxbsNormal;
FScrollBarPartBounds[AScrollBar, APart] := ScrollAreaBounds[AScrollBar];
P := ScreenToClient(GetMouseCursorPos);
if (Pos1 <> Pos2) and (Pos1 <> -1) then
begin
if AScrollBar = saeHorzScroll then
FScrollBarPartBounds[AScrollBar, APart] :=
cxRectSetXPos(FScrollBarPartBounds[AScrollBar, APart], Pos1, Pos2)
else
FScrollBarPartBounds[AScrollBar, APart] :=
cxRectSetYPos(FScrollBarPartBounds[AScrollBar, APart], Pos1, Pos2);
end;
if ((APart = sbpThumbnail) and (AScrollBar = TrackedScrollBar)) or
(AState and STATE_SYSTEM_PRESSED = STATE_SYSTEM_PRESSED)
then
FScrollBarPartState[AScrollBar, APart] := cxbsPressed
else
if AState and STATE_SYSTEM_UNAVAILABLE = STATE_SYSTEM_UNAVAILABLE then
FScrollBarPartState[AScrollBar, APart] := cxbsDisabled
else
if (AState and STATE_SYSTEM_HOTTRACKED = STATE_SYSTEM_HOTTRACKED) or
PtInRect(FScrollBarPartBounds[AScrollBar, APart], P) then
begin
FScrollBarPartState[AScrollBar, APart] := cxbsHot;
if ButtonPressed and PtInRect(FScrollBarPartBounds[AScrollBar, APart], P) then
FScrollBarPartState[AScrollBar, APart] := cxbsPressed;
end;
end;
procedure TdxSkinFormNonClientAreaInfo.CalculateScrollBarPartsInfo;
var
R: TRect;
AScrollBar: TdxSkinFormScrollBar;
begin
for AScrollBar := saeHorzScroll to saeVertScroll do
begin
if not (AScrollBar in ScrollAreaElements) then Continue;
R := ScrollAreaBounds[AScrollBar];
if AScrollBar = saeVertScroll then
R := cxRectSetXPos(R, R.Top, R.Bottom);
with ScrollBarInfo[AScrollBar] do
begin
CalculateScrollBarPartInfo(AScrollBar, R.Left,
R.Left + dxyLineButton, sbpLineUp);
CalculateScrollBarPartInfo(AScrollBar, R.Right - dxyLineButton,
R.Right, sbpLineDown);
CalculateScrollBarPartInfo(AScrollBar, R.Left + xyThumbTop,
R.Left + xyThumbBottom, sbpThumbnail);
CalculateScrollBarPartInfo(AScrollBar, R.Left + dxyLineButton,
R.Left + xyThumbTop, sbpPageUp);
CalculateScrollBarPartInfo(AScrollBar, R.Left + xyThumbBottom,
R.Right - dxyLineButton, sbpPageDown);
end
end;
end;
function TdxSkinFormNonClientAreaInfo.CalculateSystemBorderWidth: Integer;
begin
Result := ClientRect.Left - WindowRect.Left - ContentOffset;
end;
procedure TdxSkinFormNonClientAreaInfo.CalculateWindowInfo;
var
APoint: TPoint;
begin
APoint := cxNullPoint;
FStyle := GetWindowLong(Handle, GWL_STYLE);
FExStyle := GetWindowLong(Handle, GWL_EXSTYLE);
FClientRect := cxGetClientRect(Handle);
FWindowRect := cxGetWindowRect(Handle);
Windows.ClientToScreen(Handle, APoint);
OffsetRect(FClientRect, APoint.X, APoint.Y);
end;
function TdxSkinFormNonClientAreaInfo.GetBorderRect(
ASide: TcxBorder; const ABounds, AWidths: TRect): TRect;
begin
Result := ABounds;
case ASide of
bLeft:
Result.Right := Result.Left + AWidths.Left;
bTop:
Result.Bottom := Result.Top + AWidths.Top;
bRight:
Result.Left := Result.Right - AWidths.Right;
bBottom:
Result.Top := Result.Bottom - AWidths.Bottom;
end;
end;
function TdxSkinFormNonClientAreaInfo.GetCaption: string;
begin
if FCaptionSufix = '' then
Result := FCaption
else
Result := Format(sdxMDICaptionFormat, [FCaption, FCaptionSufix]);
end;
function TdxSkinFormNonClientAreaInfo.GetIcons: TdxSkinFormIcons;
const
MaximizeMenuIcons: array[Boolean] of TdxSkinFormIcon = (sfiMaximize, sfiRestore);
MimizeMenuIcons: array[Boolean] of TdxSkinFormIcon = (sfiMinimize, sfiRestore);
SysMenuIcons: array[Boolean] of TdxSkinFormIcons =
([sfiMenu, sfiClose], [sfiClose]);
begin
Result := [];
if Style and WS_SYSMENU = WS_SYSMENU then
Result := SysMenuIcons[ToolWindow or IsDialog];
if Style and WS_MINIMIZEBOX = WS_MINIMIZEBOX then
Include(Result, MimizeMenuIcons[IsIconic]);
if Style and WS_MAXIMIZEBOX = WS_MAXIMIZEBOX then
Include(Result, MaximizeMenuIcons[IsZoomed]);
if IsDialog and (ExStyle and WS_EX_CONTEXTHELP = WS_EX_CONTEXTHELP) then
Include(Result, sfiHelp);
end;
function TdxSkinFormNonClientAreaInfo.GetMDIClientDrawRgn: HRGN;
var
ARgn: HRGN;
R: TRect;
begin
R := WindowRect;
Result := CreateRectRgnIndirect(R);
InflateRect(R, -ClientEdgeSize.cx, -ClientEdgeSize.cy);
ARgn := CreateRectRgnIndirect(R);
CombineRgn(Result, Result, ARgn, RGN_XOR);
DeleteObject(ARgn);
end;
function TdxSkinFormNonClientAreaInfo.GetSysMenuIcon: HIcon;
var
wParam: Integer;
begin
wParam := Byte(not ToolWindow);
if IsWinXPOrLater then
wParam := ICON_SMALL2;
Result := DefWindowProc(Handle, WM_GETICON, wParam, 0);
end;
function TdxSkinFormNonClientAreaInfo.InternalGetScrollBarInfo(
AScrollBar: TdxSkinFormScrollBar): TScrollBarInfo;
const
ScrollBarOBJIDs: array[Boolean] of DWORD = (OBJID_HSCROLL, OBJID_VSCROLL);
var
AScrollInfo: TScrollInfo;
begin
FillChar(Result, SizeOf(TScrollBarInfo), 0);
if AScrollBar <> saeSizeGrip then
begin
Result.cbSize := SizeOf(TScrollBarInfo);
if TrackedScrollBar <> AScrollBar then
cxGetScrollBarInfo(Handle, Integer(ScrollBarOBJIDs[AScrollBar = saeVertScroll]), Result)
else
begin
FillChar(AScrollInfo, SizeOf(AScrollInfo), 0);
AScrollInfo.cbSize := SizeOf(AScrollInfo);
AScrollInfo.fMask := SIF_TRACKPOS or SIF_POS;
GetScrollInfo(Handle, Integer(AScrollBar), AScrollInfo);
Controller.LockRedraw;
try
SetScrollPos(Handle, Integer(AScrollBar), AScrollInfo.nTrackPos, False);
cxGetScrollBarInfo(Handle, Integer(ScrollBarOBJIDs[AScrollBar = saeVertScroll]), Result);
SetScrollPos(Handle, Integer(AScrollBar), AScrollInfo.nPos, False);
finally
Controller.UnlockRedraw;
end;
end;
end;
end;
procedure TdxSkinFormNonClientAreaInfo.UpdateCaption(const ANewText: string);
begin
FCaption := ANewText;
if IsMDIChild then
SkinHelper.ChildChanged(Handle);
UpdateCaptionSufix;
end;
function TdxSkinFormNonClientAreaInfo.UpdateCaptionIconStates: Boolean;
var
AHitTest, AIcon: TdxSkinFormIcon;
APrevStates: array[TdxSkinFormIcon] of TdxSkinElementState;
begin
Result := False;
Move(FIconState, APrevStates, SizeOf(FIconState));
FillChar(FIconState, SizeOf(FIconState), 0);
AHitTest := GetIconHitTest;
TrackIcon := IconPressed;
for AIcon := sfiMenu to sfiClose do
begin
if not Active then
FIconState[AIcon] := esActiveDisabled
else
if ((AIcon = sfiMinimize) and (Style and WS_MINIMIZEBOX <> WS_MINIMIZEBOX)) or
((AIcon = sfiMaximize) and (Style and WS_MAXIMIZEBOX <> WS_MAXIMIZEBOX))
then
FIconState[AIcon] := esDisabled;
if (AHitTest <> AIcon) or (FIconState[AIcon] = esDisabled) then Continue;
if ButtonPressed and (IconPressed = AIcon) then
FIconState[AIcon] := esPressed
else
if IconPressed = sfiMenu then
begin
FIconState[AIcon] := esHot;
TrackIcon := AIcon;
end;
end;
for AIcon := sfiMenu to sfiClose do
Result := Result or (FIconState[AIcon] <> APrevStates[AIcon]);
end;
function TdxSkinFormNonClientAreaInfo.UpdateCaptionSufix: Boolean;
var
AMDIChild: TCustomForm;
ATempSufix: string;
begin
AMDIChild := TdxSkinFormHelper.GetZoomedMDIChild(Handle);
if Assigned(AMDIChild) then
ATempSufix := AMDIChild.Caption
else
ATempSufix := '';
Result := ATempSufix <> FCaptionSufix;
if Result then
FCaptionSufix := ATempSufix;
end;
function TdxSkinFormNonClientAreaInfo.UpdateIconPressed(
AReset: Boolean = False): TdxSkinFormIcon;
begin
Result := FIconPressed;
FIconPressed := GetIconHitTest;
if AReset then
begin
if Result <> FIconPressed then
Result := sfiMenu;
FIconPressed := sfiMenu;
end;
end;
procedure TdxSkinFormNonClientAreaInfo.UpdateSystemMenu;
var
ASysMenu: THandle;
begin
if (IsWin2K or NativeBorderWidth) and not IsMDIChild then
begin
GetSystemMenu(Handle, True); // Win2k redrawing bug
if IsDialog and HasBorder and (Style and WS_THICKFRAME = 0) then
begin
ASysMenu := GetSystemMenu(Handle, False);
DeleteMenu(ASysMenu, SC_TASKLIST, MF_BYCOMMAND);
DeleteMenu(ASysMenu, 7, MF_BYPOSITION);
DeleteMenu(ASysMenu, 5, MF_BYPOSITION);
DeleteMenu(ASysMenu, SC_MAXIMIZE, MF_BYCOMMAND);
DeleteMenu(ASysMenu, SC_MINIMIZE, MF_BYCOMMAND);
DeleteMenu(ASysMenu, SC_SIZE, MF_BYCOMMAND);
DeleteMenu(ASysMenu, SC_RESTORE, MF_BYCOMMAND);
SetMenuDefaultItem(ASysMenu, SC_CLOSE, MF_BYCOMMAND);
end;
end;
end;
function TdxSkinFormNonClientAreaInfo.GetBorderBounds(ASide: TcxBorder): TRect;
begin
Result := FBorderBounds[ASide];
end;
function TdxSkinFormNonClientAreaInfo.GetButtonPressed: Boolean;
begin
Result := GetMouseKeys and MK_LBUTTON = MK_LBUTTON;
end;
function TdxSkinFormNonClientAreaInfo.GetCaptionBounds: TRect;
var
AFrame: TSize;
begin
AFrame := Size(GetSystemMetrics(SM_CXEDGE), GetSystemMetrics(SM_CYEDGE));
Result := BorderBounds[bTop];
Inc(Result.Top, Byte(ToolWindow) + 1);
if IsSizeBox and not NativeBorderWidth and IsZoomed then
InflateRect(Result, -SystemSizeFrame.cx, -AFrame.cy)
else
InflateRect(Result, -SizeFrame.cx, -AFrame.cy);
if IsSizeBox then
begin
if IsZoomed then
Inc(Result.Top, SizeFrame.cy)
else
if NativeBorderWidth then
Inc(Result.Top, SystemSizeFrame.cy - 4);
end;
end;
function TdxSkinFormNonClientAreaInfo.GetCaptionContentOffset: TRect;
begin
if PainterInfo.FormFrames[not ToolWindow, bTop] <> nil then
Result := PainterInfo.FormFrames[not ToolWindow, bTop].ContentOffset.Rect
else
Result := cxNullRect;
end;
function TdxSkinFormNonClientAreaInfo.GetCaptionIconSize: Integer;
begin
if ToolWindow then
Result := GetSystemMetrics(SM_CYSMCAPTION) - 2 * GetSystemMetrics(SM_CYEDGE)
else
Result := GetSystemMetrics(SM_CYSMICON);
end;
function TdxSkinFormNonClientAreaInfo.GetCaptionTextColor: TColor;
begin
Result := FCaptionTextColor[Active];
end;
function TdxSkinFormNonClientAreaInfo.GetHandle: HWND;
begin
Result := Controller.Handle;
end;
function TdxSkinFormNonClientAreaInfo.GetClientRectOnClient: TRect;
begin
Result := cxRectOffset(ClientRect, cxPointInvert(ClientRect.TopLeft));
end;
function TdxSkinFormNonClientAreaInfo.GetHasBorder: Boolean;
begin
Result := Style and WS_BORDER = WS_BORDER;
end;
function TdxSkinFormNonClientAreaInfo.GetHasCaption: Boolean;
begin
Result := Style and WS_CAPTION = WS_CAPTION;
end;
function TdxSkinFormNonClientAreaInfo.GetHasCaptionTextShadow: Boolean;
begin
Result := Active and not ToolWindow and (CaptionTextShadowColor <> clNone) and
(CaptionTextShadowColor <> clDefault);
end;
function TdxSkinFormNonClientAreaInfo.GetHasMenu: Boolean;
begin
Result := FHasMenu;
end;
function TdxSkinFormNonClientAreaInfo.GetIconBounds(
AIcon: TdxSkinFormIcon): TRect;
begin
Result := FIconBounds[AIcon];
end;
function TdxSkinFormNonClientAreaInfo.GetIconState(
AIcon: TdxSkinFormIcon): TdxSkinElementState;
begin
Result := FIconState[AIcon];
end;
function TdxSkinFormNonClientAreaInfo.GetIsAlphaBlendUsed: Boolean;
begin
Result := TdxSkinFormHelper.IsAlphaBlendUsed(Handle);
end;
function TdxSkinFormNonClientAreaInfo.GetIsDialog: Boolean;
begin
Result := ExStyle and WS_EX_DLGMODALFRAME = WS_EX_DLGMODALFRAME;
end;
function TdxSkinFormNonClientAreaInfo.GetIsIconic: Boolean;
begin
Result := Style and WS_ICONIC = WS_ICONIC;
end;
function TdxSkinFormNonClientAreaInfo.GetIsSizeBox: Boolean;
begin
Result := Style and WS_SIZEBOX = WS_SIZEBOX;
end;
function TdxSkinFormNonClientAreaInfo.GetIsZoomed: Boolean;
begin
Result := Style and WS_MAXIMIZE = WS_MAXIMIZE;
end;
function TdxSkinFormNonClientAreaInfo.GetNativeBorderWidth: Boolean;
begin
Result := FHasMenu or FIsMDIClient or not ThemeActive;
end;
function TdxSkinFormNonClientAreaInfo.GetNeedCheckNonClientSize: Boolean;
begin
Result := HasBorder and not (NativeBorderWidth or
IsZoomed and TdxSkinFormController.IsMDIChildWindow(Handle));
end;
function TdxSkinFormNonClientAreaInfo.GetScrollAreaBounds(
AItem: TdxScrollAreaElement): TRect;
begin
Result := FScrollAreaBounds[AItem];
end;
function TdxSkinFormNonClientAreaInfo.GetScrollBarInfo(
AScrollBar: TdxSkinFormScrollBar): TScrollBarInfo;
begin
Result := FScrollBarsInfo[AScrollBar];
end;
function TdxSkinFormNonClientAreaInfo.GetScrollBarPartBounds(
AScrollBar: TdxSkinFormScrollBar; APart: TcxScrollBarPart): TRect;
begin
Result := FScrollBarPartBounds[AScrollBar, APart];
end;
function TdxSkinFormNonClientAreaInfo.GetScrollBarPartState(
AScrollBar: TdxSkinFormScrollBar; APart: TcxScrollBarPart): TcxButtonState;
begin
Result := FScrollBarPartState[AScrollBar, APart];
end;
function TdxSkinFormNonClientAreaInfo.GetSizeArea(ASide: TcxBorder): TRect;
begin
Result := WindowBounds;
if ASide in [bLeft, bRight] then
Inc(Result.Top, BorderWidths.Top);
if NativeBorderWidth then
with SizeFrame do
Result := GetBorderRect(ASide, Result, Rect(cx, cy, cx, cy))
else
Result := GetBorderRect(ASide, Result, Rect(2, 2, 2, 2))
end;
function TdxSkinFormNonClientAreaInfo.GetSizeCorners(
ACorner: TdxSkinFormCorner): TRect;
var
ASize: TSize;
begin
Result := WindowBounds;
ASize := SizeFrame;
if not NativeBorderWidth then
ASize := Size(2, 2);
if ACorner in [sfcLeftTop, sfcLeftBottom] then
Result.Right := Result.Left + ASize.cx
else
Result.Left := Result.Right - ASize.cx;
if ACorner in [sfcLeftTop, sfcRightTop] then
Result.Bottom := Result.Top + BorderWidths.Top
else
Result.Top := Result.Bottom - ASize.cy;
end;
function TdxSkinFormNonClientAreaInfo.GetSkinBorderWidth(ASide: TcxBorder): Integer;
var
AElement: TdxSkinElement;
begin
Result := 0;
AElement := PainterInfo.FormFrames[not ToolWindow, ASide];
if Assigned(AElement) then
begin
if ASide in [bLeft, bRight] then
Result := AElement.Size.cx
else
Result := AElement.Size.cy;
end;
end;
function TdxSkinFormNonClientAreaInfo.GetThemeActive: Boolean;
begin
if not ThemeActiveAssigned then
begin
FThemeActive := IsThemeActive;
ThemeActiveAssigned := True;
end;
Result := FThemeActive;
end;
function TdxSkinFormNonClientAreaInfo.GetToolWindow: Boolean;
begin
Result := ExStyle and WS_EX_TOOLWINDOW = WS_EX_TOOLWINDOW;
end;
procedure TdxSkinFormNonClientAreaInfo.SetActive(AActive: Boolean);
begin
if FActive <> AActive then
begin
FActive := AActive;
UpdateCaptionIconStates;
end;
end;
procedure TdxSkinFormNonClientAreaInfo.SetSizeType(AValue: Integer);
begin
if AValue <> FSizeType then
begin
if IsMDIChild and (SizeType = SIZE_MAXIMIZED) then
UpdateSystemMenu;
FSizeType := AValue;
end;
end;
procedure TdxSkinFormNonClientAreaInfo.SetUpdateRgn(ARgn: HRGN);
begin
if FUpdateRgn <> 0 then
DeleteObject(FUpdateRgn);
FUpdateRgn := ARgn;
if FUpdateRgn <> 0 then
OffsetRgn(FUpdateRgn, -WindowRect.Left, -WindowRect.Top);
end;
procedure TdxSkinFormNonClientAreaInfo.UpdateFormCaption;
begin
UpdateCaption(GetWindowCaption(Handle));
end;
{ TdxSkinFormPainter }
constructor TdxSkinFormPainter.Create(AViewInfo: TdxSkinFormNonClientAreaInfo);
begin
FViewInfo := AViewInfo;
FBaseCanvas := TCanvas.Create;
FCanvas := TcxCanvas.Create(FBaseCanvas);
CreateCacheInfos;
end;
destructor TdxSkinFormPainter.Destroy;
begin
FreeCacheInfos;
FCanvas.Free;
FBaseCanvas.Free;
inherited Destroy;
end;
procedure TdxSkinFormPainter.CreateCacheInfos;
var
AIcon: TdxSkinFormIcon;
ASide: TcxBorder;
begin
for ASide := Low(TcxBorder) to High(TcxBorder) do
FBordersCache[ASide] := TdxSkinElementCache.Create;
for AIcon := Low(TdxSkinFormIcon) to High(TdxSkinFormIcon) do
FIconsCache[AIcon] := TdxSkinElementCache.Create;
end;
procedure TdxSkinFormPainter.FreeCacheInfos;
var
AIcon: TdxSkinFormIcon;
ASide: TcxBorder;
begin
for ASide := Low(TcxBorder) to High(TcxBorder) do
FBordersCache[ASide].Free;
for AIcon := Low(TdxSkinFormIcon) to High(TdxSkinFormIcon) do
FIconsCache[AIcon].Free;
end;
procedure TdxSkinFormPainter.BeginPaint(ADestDC: HDC = 0);
const
Flags = DCX_CACHE or DCX_CLIPSIBLINGS or DCX_WINDOW or DCX_VALIDATE;
begin
FPainter := ViewInfo.Painter;
FPainterInfo := ViewInfo.PainterInfo;
FDC := ADestDC;
FNeedRelease := ADestDC = 0;
if FNeedRelease then
FDC := GetDCEx(ViewInfo.Handle, 0, Flags);
FBaseCanvas.Handle := FDC;
end;
procedure TdxSkinFormPainter.DrawContentOffsetArea;
var
R: TRect;
begin
if (ViewInfo.ContentOffset > 0) and not ViewInfo.IsIconic then
begin
Canvas.SaveClipRegion;
try
R := cxRectContent(ViewInfo.WindowBounds, ViewInfo.CalculateMargins);
Canvas.ExcludeClipRect(R);
dxSkinCheckSkinElement(PainterInfo.FormContent).Draw(Canvas.Handle,
cxRectInflate(R, ViewInfo.ContentOffset, ViewInfo.ContentOffset));
finally
Canvas.RestoreClipRegion;
end;
end;
end;
procedure TdxSkinFormPainter.DrawMDIClientEdgeArea;
begin
if ViewInfo.IsMDIClient and ViewInfo.HasClientEdge then
begin
Canvas.SaveClipRegion;
try
Canvas.ExcludeClipRect(cxRectInflate(ViewInfo.WindowBounds,
-ViewInfo.ClientEdgeSize.cx, -ViewInfo.ClientEdgeSize.cy));
dxSkinCheckSkinElement(PainterInfo.FormContent).Draw(Canvas.Handle,
ViewInfo.WindowBounds);
finally
Canvas.RestoreClipRegion;
end;
end;
end;
procedure TdxSkinFormPainter.DrawMenuSeparator;
begin
if ViewInfo.HasMenu then
FillRect(Canvas.Handle, ViewInfo.MenuSeparatorBounds, GetSysColorBrush(COLOR_MENU));
end;
procedure TdxSkinFormPainter.DrawWindowBackground;
begin
if PainterInfo <> nil then
dxSkinCheckSkinElement(PainterInfo.FormContent).Draw(
Canvas.Handle, ViewInfo.ClientRectOnClient);
end;
procedure TdxSkinFormPainter.DrawWindowBorder;
begin
DrawMDIClientEdgeArea;
DrawContentOffsetArea;
InternalDrawCaption(ViewInfo.BorderBounds[bTop],
PainterInfo.FormFrames[not ViewInfo.ToolWindow, bTop]);
InternalDrawBorders;
if not ViewInfo.IsIconic then
begin
DrawMenuSeparator;
if ViewInfo.ScrollAreaElements <> [] then
DrawScrollAreaElements(Canvas.Handle);
end;
end;
procedure TdxSkinFormPainter.EndPaint;
begin
if FNeedRelease then
ReleaseDC(ViewInfo.Handle, FDC);
FBaseCanvas.Handle := 0;
end;
procedure TdxSkinFormPainter.DrawBackground(DC: HDC; const R: TRect);
begin
dxSkinCheckSkinElement(PainterInfo.FormContent).Draw(DC, R);
end;
procedure TdxSkinFormPainter.DrawCaptionText(DC: HDC; R: TRect; const AText: string);
begin
{$IFDEF DELPHI12}
cxDrawText(DC, AText, R, CaptionFlags);
{$ELSE}
if not IsWinNT then
cxDrawText(DC, AText, R, CaptionFlags)
else
DrawTextW(DC, PWideChar(dxStringToWideString(AText)), -1, R, CaptionFlags);
{$ENDIF}
end;
procedure TdxSkinFormPainter.DrawScrollAreaElements(DC: HDC);
var
AElement: TdxScrollAreaElement;
begin
for AElement := Low(TdxScrollAreaElement) to High(TdxScrollAreaElement) do
begin
if AElement in ViewInfo.ScrollAreaElements then
begin
if AElement = saeSizeGrip then
DrawSizeGrip(DC, ViewInfo.ScrollAreaBounds[AElement])
else
DrawScrollBar(DC, AElement, ViewInfo.ScrollAreaBounds[AElement]);
end;
end;
end;
procedure TdxSkinFormPainter.DrawScrollBar(
DC: HDC; AScrollBar: TdxSkinFormScrollBar; const R: TRect);
var
AMemBitmap: HBITMAP;
AMemDC: HDC;
APart: TcxScrollBarPart;
begin
AMemDC := CreateCompatibleDC(0);
try
SelectDC(AMemDC);
AMemBitmap := CreateCompatibleBitmap(DC, cxRectWidth(R), cxRectHeight(R));
AMemBitmap := SelectObject(AMemDC, AMemBitmap);
SetWindowOrgEx(AMemDC, R.Left, R.Top, nil);
DrawBackground(Canvas.Handle, R);
Painter.DrawScrollBarBackground(Canvas, R, AScrollBar = saeHorzScroll);
for APart := Low(TcxScrollBarPart) to High(TcxScrollBarPart) do
with ViewInfo do
begin
if ScrollBarPartState[AScrollBar, APart] = cxbsDefault then Continue;
Painter.DrawScrollBarPart(Canvas, AScrollBar = saeHorzScroll,
ScrollBarPartBounds[AScrollBar, APart], APart, ScrollBarPartState[AScrollBar, APart]);
end;
cxBitBlt(DC, AMemDC, R, R.TopLeft, SRCCOPY);
AMemBitmap := SelectObject(AMemDC, AMemBitmap);
DeleteObject(AMemBitmap);
finally
DeleteDC(AMemDC)
end;
end;
procedure TdxSkinFormPainter.DrawSizeGrip(DC: HDC; const R: TRect);
begin
DrawBackground(DC, R);
Painter.DrawSizeGrip(Canvas, R, clNone);
end;
procedure TdxSkinFormPainter.DrawWindowCaption(DC: HDC; const R: TRect;
AElement: TdxSkinElement);
var
AIcon: TdxSkinFormIcon;
begin
DrawWindowCaptionBackground(DC, R, AElement);
DrawWindowCaptionText(DC, ViewInfo.CaptionBounds);
for AIcon := Low(TdxSkinFormIcon) to High(TdxSkinFormIcon) do
begin
if AIcon in ViewInfo.Icons then
begin
DrawWindowIcon(DC, ViewInfo.IconBounds[AIcon], AIcon,
PainterInfo.FormIcons[not ViewInfo.ToolWindow, AIcon]);
end;
end;
end;
procedure TdxSkinFormPainter.DrawWindowCaptionBackground(DC: HDC;
const R: TRect; AElement: TdxSkinElement);
var
R1: TRect;
begin
R1 := R;
if ViewInfo.IsIconic then
begin
InternalDrawBorder(DC, R1, bBottom, ViewInfo.IsAlphaBlendUsed);
Dec(R1.Bottom);
end;
FBordersCache[bTop].DrawEx(DC, dxSkinCheckSkinElement(AElement), R1,
FrameStates[ViewInfo.Active]);
end;
procedure TdxSkinFormPainter.DrawWindowCaptionText(DC: HDC; R: TRect);
var
APrevColor: TColor;
APrevFont: HFONT;
APrevTransparent: Integer;
begin
if Length(ViewInfo.Caption) > 0 then
begin
APrevFont := SelectObject(DC, ViewInfo.CaptionFont.Handle);
APrevTransparent := SetBkMode(DC, Transparent);
if ViewInfo.HasCaptionTextShadow then
begin
APrevColor := SetTextColor(DC, ColorToRGB(ViewInfo.CaptionTextShadowColor));
cxDrawText(DC, ViewInfo.Caption, R, CaptionFlags);
SetTextColor(DC, APrevColor);
end;
OffsetRect(R, -1, -1);
APrevColor := SetTextColor(DC, ColorToRGB(ViewInfo.CaptionTextColor));
DrawCaptionText(DC, R, ViewInfo.Caption);
SetBkMode(DC, APrevTransparent);
SelectObject(DC, APrevFont);
SetTextColor(DC, APrevColor);
end;
end;
procedure TdxSkinFormPainter.DrawWindowIcon(DC: HDC; const R: TRect;
AIcon: TdxSkinFormIcon; AElement: TdxSkinElement);
begin
if ((AIcon = sfiMenu) and (ViewInfo.SysMenuIcon = 0)) or
((AIcon <> sfiMenu) and (AElement = nil)) then Exit;
if RectVisible(DC, R) then
begin
if AElement = nil then
DrawIconEx(DC, R.Left, R.Top, ViewInfo.SysMenuIcon, R.Right - R.Left,
R.Bottom - R.Top, 0, 0, DI_NORMAL)
else
begin
FIconsCache[AIcon].CheckCacheState(AElement, R, ViewInfo.IconState[AIcon], 0);
FIconsCache[AIcon].Draw(DC, R);
end;
end;
end;
procedure TdxSkinFormPainter.InternalDrawBorder(DC: HDC; const R: TRect;
ASide: TcxBorder; AFillBackground: Boolean);
function GetBorderRect(const R: TRect): TRect;
begin
Result := R;
if IsBordersThin then
case ASide of
bLeft:
Result.Right := Result.Left + ViewInfo.GetSkinBorderWidth(ASide);
bRight:
Result.Left := Result.Right - ViewInfo.GetSkinBorderWidth(ASide);
bBottom:
Result.Top := Result.Bottom - ViewInfo.GetSkinBorderWidth(ASide);
end;
end;
procedure DrawBorderElement(DC: HDC; const R: TRect);
begin
FBordersCache[ASide].DrawEx(DC,
PainterInfo.FormFrames[not ViewInfo.ToolWindow, ASide], R,
FrameStates[Active]);
end;
var
ACachedDC: HDC;
AMemBmp: HBitmap;
R1: TRect;
begin
if not AFillBackground then
DrawBorderElement(DC, R)
else
begin
R1 := R;
OffsetRect(R1, -R1.Left, -R1.Top);
ACachedDC := CreateCompatibleDC(Canvas.Handle);
AMemBmp := CreateCompatibleBitmap(Canvas.Handle, R1.Right, R1.Bottom);
AMemBmp := SelectObject(ACachedDC, AMemBmp);
dxSkinCheckSkinElement(PainterInfo.FormContent).Draw(ACachedDC, R1, 0, FrameStates[Active]);
DrawBorderElement(ACachedDC, GetBorderRect(R1));
BitBlt(DC, R.Left, R.Top, R1.Right, R1.Bottom, ACachedDC, 0, 0, SRCCOPY);
AMemBMP := SelectObject(ACachedDC, AMemBmp);
DeleteObject(AMemBmp);
DeleteDC(ACachedDC);
end;
end;
procedure TdxSkinFormPainter.InternalDrawBorders;
var
AIsAlphaBlendUsed: Boolean;
ASide: TcxBorder;
R: TRect;
begin
if IsBordersThin then
begin
InternalDrawThinBorders;
Exit;
end;
AIsAlphaBlendUsed := ViewInfo.IsAlphaBlendUsed;
for ASide := Low(TcxBorder) to High(TcxBorder) do
begin
R := ViewInfo.BorderBounds[ASide];
if (ASide <> bTop) and IsRectVisible(R) and not cxRectIsEmpty(R) then
begin
if ASide in [bLeft, bRight] then
begin
R.Top := ViewInfo.BorderBounds[bTop].Bottom;
R.Bottom := ViewInfo.BorderBounds[bBottom].Top;
end;
InternalDrawBorder(Canvas.Handle, R, ASide, AIsAlphaBlendUsed);
end;
end;
end;
procedure TdxSkinFormPainter.InternalDrawCaption(const R: TRect;
AElement: TdxSkinElement);
var
ACachedDC: HDC;
AMemBmp: HBitmap;
begin
ACachedDC := CreateCompatibleDC(Canvas.Handle);
AMemBmp := CreateCompatibleBitmap(Canvas.Handle, R.Right, R.Bottom);
AMemBmp := SelectObject(ACachedDC, AMemBmp);
DrawWindowCaption(ACachedDC, R, AElement);
BitBlt(Canvas.Handle, 0, 0, R.Right, R.Bottom, ACachedDC, 0, 0, SRCCOPY);
AMemBMP := SelectObject(ACachedDC, AMemBmp);
DeleteObject(AMemBmp);
DeleteDC(ACachedDC);
end;
procedure TdxSkinFormPainter.InternalDrawThinBorders;
var
ASide: TcxBorder;
R: TRect;
begin
InternalDrawBorder(Canvas.Handle, ViewInfo.BorderBounds[bBottom], bBottom, True);
for ASide := Low(TcxBorder) to High(TcxBorder) do
if ASide in [bLeft, bRight] then
begin
R := ViewInfo.BorderBounds[ASide];
if IsRectVisible(R) and not cxRectIsEmpty(R) then
begin
R.Top := ViewInfo.BorderBounds[bTop].Bottom;
R.Bottom := ViewInfo.WindowBounds.Bottom - ViewInfo.SkinBorderWidth[bBottom];
InternalDrawBorder(Canvas.Handle, R, ASide, True);
end;
end;
end;
function TdxSkinFormPainter.IsRectVisible(const R: TRect): Boolean;
begin
with FViewInfo do
Result := (FUpdateRgn = 0) or RectInRegion(FUpdateRgn, R);
if Result then
Result := Canvas.RectVisible(R);
end;
function TdxSkinFormPainter.SelectDC(DC: HDC): Integer;
begin
Result := Canvas.Handle;
Canvas.Canvas.Handle := DC;
end;
function TdxSkinFormPainter.GetActive: Boolean;
begin
Result := ViewInfo.Active;
end;
function TdxSkinFormPainter.GetIconElement(
AIcon: TdxSkinFormIcon): TdxSkinElement;
begin
Result := FPainterInfo.FormIcons[ViewInfo.ToolWindow, AIcon];
end;
function TdxSkinFormPainter.GetIsBordersThin: Boolean;
var
ASide: TcxBorder;
begin
Result := False;
for ASide := Low(TcxBorder) to High(TcxBorder) do
Result := Result or (ViewInfo.SkinBorderWidth[ASide] < 3);
end;
{ TdxSkinFormController }
constructor TdxSkinFormController.Create(AHandle: HWND);
begin
inherited Create(AHandle);
FViewInfo := TdxSkinFormNonClientAreaInfo.Create(Self);
FPainter := TdxSkinFormPainter.Create(FViewInfo);
if IsMessageDlgWindow(AHandle) then
PostMessage(AHandle, WM_POSTMSGFORMINIT, 0, 0);
end;
constructor TdxSkinFormController.CreateEx(ASkinForm: TdxSkinForm);
begin
FSkinForm := ASkinForm;
Create(ASkinForm.Handle);
end;
destructor TdxSkinFormController.Destroy;
begin
cxClearObjectLinks(Self);
FreeAndNil(FPainter);
FreeAndNil(FViewInfo);
inherited Destroy;
end;
procedure TdxSkinFormController.RefreshController;
var
ASkinController: TdxSkinController;
ASkinName: string;
AUseSkin: Boolean;
I: Integer;
begin
ViewInfo.UpdateFormCaption;
for I := dxSkinControllersList.Count - 1 downto 0 do
begin
ASkinController := TdxSkinController(dxSkinControllersList[I]);
if csDestroying in ASkinController.ComponentState then
Continue;
if FLookAndFeelPainter = nil then
FLookAndFeelPainter := ASkinController.DoSkinForm(Form)
else
begin
FLookAndFeelPainter := nil;
if SendMessage(Form.Handle, dxWMGetSkinnedMessage, 0, 0) <> 1 then
begin
AUseSkin := GetSkinName(ASkinName);
FLookAndFeelPainter := ASkinController.DoSkinFormEx(Form, ASkinName, AUseSkin);
end;
end;
end;
end;
procedure TdxSkinFormController.Update;
begin
inherited Update;
if (Handle <> 0) and HasRegion and not IsSkinUsed then
begin
HasRegion := False;
SetWindowRgn(Handle, 0, False);
end;
end;
procedure TdxSkinFormController.DefWndProc(var AMessage);
begin
if SkinForm = nil then
inherited DefWndProc(AMessage)
else
SkinForm.DefaultWndProc(TMessage(AMessage));
end;
procedure TdxSkinFormController.DrawWindowBackground(DC: HDC);
begin
Painter.BeginPaint(DC);
try
Painter.DrawWindowBackground;
finally
Painter.EndPaint;
end;
end;
procedure TdxSkinFormController.DrawWindowBorder(DC: HDC = 0);
begin
Painter.BeginPaint(DC);
try
Painter.DrawWindowBorder;
finally
Painter.EndPaint;
end;
end;
procedure TdxSkinFormController.CalculateViewInfo;
begin
ViewInfo.Calculate(0);
end;
procedure TdxSkinFormController.CheckWindowRgn(AForceRgn: Boolean);
var
R: TRect;
begin
if AForceRgn then
begin
HasRegion := True;
R := cxGetWindowRect(Handle);
OffsetRect(R, -R.Left, -R.Top);
SetWindowRgn(Handle, CreateRectRgnIndirect(R), True);
end
else
if IsZoomed(Handle) and HasRegion then
begin
LockRedraw;
FlushWindowRegion;
UnlockRedraw;
end;
end;
procedure TdxSkinFormController.FlushWindowRegion;
begin
if HasRegion then
begin
HasRegion := False;
SetWindowRgn(Handle, 0, False);
PostMessage(Handle, WM_POSTREDRAW, 0, 0);
end;
end;
procedure TdxSkinFormController.InitializeMessageForm;
var
I: Integer;
AForm: TCustomForm;
begin
AForm := Form;
for I := AForm.ControlCount - 1 downto 0 do
begin
if AForm.Controls[I] is TCustomLabel then
TCustomLabelAccess(AForm.Controls[I]).Transparent := True;
end;
end;
procedure TdxSkinFormController.InitializePainter;
var
ASkinName: string;
AUseSkin: Boolean;
I: Integer;
begin
if Form = nil then
Exit;
for I := 0 to dxSkinControllersList.Count - 1 do
with TdxSkinController(dxSkinControllersList[I]) do
begin
if I = 0 then
begin
FLookAndFeelPainter := DoSkinForm(Form);
AUseSkin := FLookAndFeelPainter <> nil;
end
else
begin
AUseSkin := GetFormSkin(Form, ASkinName);
FLookAndFeelPainter := DoSkinFormEx(Form, ASkinName, AUseSkin);
end;
if FLookAndFeelPainter <> nil then
Break;
end;
end;
procedure TdxSkinFormController.InvalidateBorders;
begin
CalculateViewInfo;
DrawWindowBorder;
end;
function TdxSkinFormController.HandleInternalMessages(var AMessage: TMessage): Boolean;
const
WindowState: array[Boolean] of Integer = (0, SIZE_MAXIMIZED);
begin
Result := True;
if AMessage.Msg = WM_CHILDCHANGED then
begin
if ViewInfo.UpdateCaptionSufix then
DrawWindowBorder;
end
else
if (dxWMSetSkinnedMessage > 0) and (AMessage.Msg = dxWMSetSkinnedMessage) then
RefreshController
else
if AMessage.Msg = WM_POSTREDRAW then
RedrawWindow(False)
else
if AMessage.Msg = WM_POSTCHECKRGN then
begin
ViewInfo.SizeType := WindowState[IsZoomed(Handle)];
CheckWindowRgn(NeedForceRgnUpdate(ViewInfo.SizeType));
ViewInfo.UpdateSystemMenu;
end
else
Result := False;
end;
function TdxSkinFormController.HandleWindowMessage(
var AMessage: TMessage): Boolean;
begin
Result := True;
CheckWindowRgn(False);
if ForceRedraw then
begin
DrawWindowBorder;
ForceRedraw := False;
end;
case AMessage.Msg of
WM_CONTEXTMENU:
WMContextMenu(TWMContextMenu(AMessage));
WM_NCCREATE:
Result := WMNCCreate(AMessage);
WM_NCCALCSIZE:
WMNCCalcSize(TWMNCCalcSize(AMessage));
WM_NCMOUSEMOVE:
WMNCMouseMove(TWMNCHitMessage(AMessage));
WM_NCACTIVATE:
WMNCActivate(TWMNCActivate(AMessage));
WM_NCUAHDRAWFRAME, WM_NCUAHDRAWCAPTION, WM_SYNCPAINT:
DrawWindowBorder;
WM_NCLBUTTONDOWN, WM_NCLBUTTONDBLCLK:
WMNCButtonDown(TWMNCHitMessage(AMessage));
WM_NCLBUTTONUP:
WMNCLButtonUp(TWMNCHitMessage(AMessage));
WM_NCPAINT:
WMNCPaint(TWMNCPaint(AMessage));
WM_NCHITTEST:
WMNCHitTest(TWMNCHitTest(AMessage));
WM_ERASEBKGND:
WMEraseBkgnd(TWMEraseBkgnd(AMessage));
WM_WINDOWPOSCHANGED:
WMWindowPosChanged(TWMWindowPosMsg(AMessage));
WM_SIZE:
WMSize(TWMSize(AMessage));
WM_PRINT:
WMPrint(TWMPrint(AMessage));
WM_SYSCOMMAND:
WMSysCommand(TWMSysCommand(AMessage));
WM_DESTROY, WM_MDIDESTROY:
WMDestroy(AMessage);
WM_VSCROLL, WM_HSCROLL:
WMScroll(TWMScroll(AMessage));
WM_ACTIVATE, WM_MOUSEACTIVATE:
begin
DefWndProc(AMessage);
DrawWindowBorder;
end;
$3F:
begin
DefWndProc(AMessage);
Refresh;
end;
else
Result := HandleInternalMessages(AMessage);
end;
end;
function TdxSkinFormController.IsHookAvailable: Boolean;
begin
Result := inherited IsHookAvailable and (SkinForm = nil);
end;
procedure TdxSkinFormController.LockRedraw;
begin
Inc(FLockRedrawCount);
if FLockRedrawCount = 1 then
DefWindowProc(Handle, WM_SETREDRAW, 0, 0);
end;
function TdxSkinFormController.RefreshOnMouseEvent(
AForceRefresh: Boolean = False): Boolean;
var
APart: TcxScrollBarPart;
AScrollBar: TdxSkinFormScrollBar;
begin
with ViewInfo do
begin
Result := UpdateCaptionIconStates;
if TrackIcon <> sfiMenu then
BeginMouseTracking(nil, ClientToScreen(IconBounds[TrackIcon]), Self)
else
if GetScrollBarHitTest(AScrollBar, APart) then
begin
Result := not IsMouseTracking(Self);
BeginMouseTracking(nil, ClientToScreen(ScrollBarPartBounds[AScrollBar, APart]), Self)
end
else
begin
Result := Result or IsMouseTracking(Self);
EndMouseTracking(Self);
end;
end;
if Result or AForceRefresh then
InvalidateBorders;
end;
procedure TdxSkinFormController.RefreshOnSystemMenuShown;
begin
if IsWin2K or ViewInfo.NativeBorderWidth then
PostMessage(Handle, WM_POSTREDRAW, 0, 0);
end;
procedure TdxSkinFormController.UnlockRedraw;
begin
Dec(FLockRedrawCount);
if FLockRedrawCount = 0 then
DefWindowProc(Handle, WM_SETREDRAW, 1, 0);
end;
procedure TdxSkinFormController.WMContextMenu(var Message: TWMContextMenu);
begin
if ViewInfo.GetHitTest(SmallPointToPoint(Message.Pos)) <> HTNOWHERE then
RefreshOnSystemMenuShown;
DefWndProc(Message);
end;
procedure TdxSkinFormController.WMDestroy(var Message: TMessage);
begin
DefWndProc(Message);
if Message.WParam = 0 then
begin
FLookAndFeelPainter := nil;
UnHookWndProc;
end;
end;
procedure TdxSkinFormController.WMEraseBkgnd(var Message: TWMEraseBkgnd);
var
ASaveIndex: Integer;
begin
if Message.DC <> 0 then
begin
ASaveIndex := SaveDC(Message.DC);
try
DrawWindowBackground(Message.DC);
Message.Result := 1;
finally
RestoreDC(Message.DC, ASaveIndex);
end;
end;
end;
procedure TdxSkinFormController.WMNCActivate(var Message: TWMNCActivate);
var
AFlags: DWORD;
AMDIChild: TCustomForm;
begin
ViewInfo.Active := Message.Active;
AMDIChild := TdxSkinFormHelper.GetActiveMDIChild(Handle);
if Assigned(AMDIChild) then
AMDIChild.Perform(WM_NCACTIVATE, TMessage(Message).WParam, 0);
if IsMDIChildWindow(Handle) then
begin
AFlags := GetWindowLong(Handle, GWL_STYLE);
SetWindowLong(Handle, GWL_STYLE, AFlags and not WS_VISIBLE);
Message.Result := DefWindowProc(Handle, WM_NCACTIVATE, TMessage(Message).WParam, 0);
SetWindowLong(Handle, GWL_STYLE, AFlags);
end
else
Message.Result := 1;
DrawWindowBorder;
end;
procedure TdxSkinFormController.WMNCButtonDown(var Message: TWMNCHitMessage);
var
ALink: TcxObjectLink;
ALocked: Boolean;
begin
ALink := cxAddObjectLink(Self);
try
ForceRedraw := True;
ViewInfo.UpdateIconPressed;
ViewInfo.UpdateCaptionIconStates;
ALocked := Message.HitTest in [HTHSCROLL, HTVSCROLL];
if ALocked then
LockRedraw;
if ViewInfo.IconPressed = sfiMenu then
DefWndProc(Message);
if Assigned(ALink.Ref) then
begin
if ALocked then
UnlockRedraw;
RefreshOnMouseEvent(True);
DrawWindowBorder;
end;
finally
cxRemoveObjectLink(ALink);
end;
Message.Result := 0;
end;
function TdxSkinFormController.WMNCCreate(var Message: TMessage): Boolean;
begin
Result := False;
// note: D2007 perform this message, when BorderStyle changed
FlushWindowRegion;
CalculateViewInfo;
end;
procedure TdxSkinFormController.WMNCCalcSize(var Message: TWMNCCalcSize);
var
R: TRect;
begin
R := Message.CalcSize_Params^.rgrc[0];
DefWndProc(Message);
CalculateViewInfo;
if Message.CalcValidRects and ViewInfo.NeedCheckNonClientSize then
Message.CalcSize_Params^.rgrc[0] := cxRectContent(R, ViewInfo.CalculateMargins);
end;
procedure TdxSkinFormController.WMNCHitTest(var Message: TWMNCHitTest);
begin
with Message do
begin
Result := ViewInfo.GetHitTest(SmallPointToPoint(Pos), Result);
if (Result = HTNOWHERE) or (Result = HTSYSMENU) then
DefWndProc(Message);
end;
end;
procedure TdxSkinFormController.WMNCLButtonUp(var Message: TWMNCHitMessage);
var
ADownIcon: TdxSkinFormIcon;
begin
ADownIcon := ViewInfo.UpdateIconPressed(True);
RefreshOnMouseEvent(True);
if ADownIcon in [sfiMenu, sfiHelp] then
DefWndProc(Message)
else
SendMessage(Handle, WM_SYSCOMMAND, IconCommand[ADownIcon], 0);
end;
procedure TdxSkinFormController.WMNCMouseMove(var Message: TWMNCHitMessage);
begin
if not RefreshOnMouseEvent then
begin
Message.HitTest := 0;
DefWndProc(Message);
DrawWindowBorder;
end;
end;
procedure TdxSkinFormController.WMNCPaint(var Message: TWMNCPaint);
var
AFrameRgn, AWindowRgn: HRgn;
begin
InvalidateBorders;
if ViewInfo.HasMenu or ViewInfo.IsMDIClient then
begin
AFrameRgn := ViewInfo.CreateDrawRgn;
AWindowRgn := CreateRectRgnIndirect(ViewInfo.WindowRect);
CombineRgn(AWindowRgn, AWindowRgn, AFrameRgn, RGN_XOR);
DeleteObject(AFrameRgn);
if Message.RGN <> 1 then
begin
CombineRgn(AWindowRgn, AWindowRgn, Message.RGN, RGN_AND);
DeleteObject(Message.RGN);
end;
Message.RGN := AWindowRgn;
DefWndProc(Message);
DeleteObject(AWindowRgn);
end;
Message.RGN := 1;
Message.Result := 0;
end;
procedure TdxSkinFormController.WMPrint(var Message: TWMPrint);
begin
DefWndProc(Message);
if (Message.Flags and PRF_CHECKVISIBLE = 0) or IsWindowVisible(Handle) then
begin
if Message.Flags and PRF_NONCLIENT <> 0 then
DrawWindowBorder(Message.DC);
end;
end;
procedure TdxSkinFormController.WMScroll(var Message: TWMScroll);
begin
if Message.ScrollCode = SB_THUMBTRACK then
ViewInfo.TrackedScrollBar := TdxScrollAreaElement(Message.Msg - WM_HSCROLL)
else
ViewInfo.TrackedScrollBar := saeSizeGrip;
DefWndProc(Message);
UnlockRedraw;
try
RedrawWindow(True);
finally
LockRedraw;
end;
end;
procedure TdxSkinFormController.WMSetText(var Message: TWMSetText);
begin
DefWndProc(Message);
ViewInfo.UpdateFormCaption;
Refresh;
end;
procedure TdxSkinFormController.WMSize(var Message: TWMSize);
begin
DefWndProc(Message);
ViewInfo.SizeType := Message.SizeType;
CheckWindowRgn(NeedForceRgnUpdate(Message.SizeType));
end;
procedure TdxSkinFormController.WMWindowPosChanged(var Message: TWMWindowPosChanged);
begin
DefWndProc(Message);
if (Message.WindowPos^.flags and SWP_NOSIZE = 0) or
(Message.WindowPos^.flags and SWP_NOMOVE = 0)
then
CalculateViewInfo;
end;
procedure TdxSkinFormController.WMSysCommand(var Message: TWMSysCommand);
var
ALink: TcxObjectLink;
begin
ALink := cxAddObjectLink(Self);
try
if Message.CmdType = SC_KEYMENU then
RefreshOnSystemMenuShown;
DefWndProc(Message);
if Assigned(ALink.Ref) then
DrawWindowBorder;
finally
cxRemoveObjectLink(ALink);
end;
end;
procedure TdxSkinFormController.MouseLeave;
begin
UnlockRedraw;
RefreshOnMouseEvent(True);
UpdateWindow(Handle);
LockRedraw;
end;
function TdxSkinFormController.GetForm: TCustomForm;
begin
if SkinForm = nil then
Result := TdxSkinFormHelper.GetForm(Handle)
else
Result := SkinForm;
end;
function TdxSkinFormController.GetIsSkinUsed: Boolean;
begin
Result := inherited GetIsSkinUsed and (ViewInfo <> nil);
end;
function TdxSkinFormController.NeedForceRgnUpdate(ASizeType: Integer): Boolean;
var
AForm: TCustomForm;
begin
AForm := Form;
if AForm = nil then
Result := ASizeType <> SIZE_MAXIMIZED
else
begin
Result := AForm.BorderStyle <> bsNone;
if Result then
Result := (ASizeType <> SIZE_MAXIMIZED) or (AForm.Parent <> nil);
end;
end;
procedure TdxSkinFormController.WndProc(var AMessage: TMessage);
begin
case AMessage.Msg of
WM_SETTEXT:
WMSetText(TWMSetText(AMessage));
WM_THEMECHANGED:
ViewInfo.ThemeActiveAssigned := False;
else
if AMessage.Msg = WM_POSTMSGFORMINIT then
InitializeMessageForm
else
if not (IsSkinUsed and HandleWindowMessage(AMessage)) then
DefWndProc(AMessage);
end;
end;
{ TdxSkinFormHelper }
class function TdxSkinFormHelper.GetActiveMDIChild(AHandle: HWND): TCustomForm;
var
ACustomForm: TCustomFormAccess;
begin
Result := nil;
ACustomForm := TCustomFormAccess(GetForm(AHandle));
if (ACustomForm <> nil) and (ACustomForm.FormStyle = fsMDIForm) then
if ACustomForm.ActiveMDIChild <> nil then
Result := ACustomForm.ActiveMDIChild;
end;
class function TdxSkinFormHelper.GetContentOffset(AHandle: HWND): Integer;
var
ACustomForm: TCustomFormAccess;
begin
ACustomForm := TCustomFormAccess(GetForm(AHandle));
if ACustomForm = nil then
Result := 0
else
Result := ACustomForm.BorderWidth;
end;
class function TdxSkinFormHelper.GetForm(AHandle: HWND): TCustomForm;
var
AControl: TWinControl;
begin
AControl := FindControl(AHandle);
if AControl is TCustomForm then
Result := TCustomForm(AControl)
else
Result := nil;
end;
class function TdxSkinFormHelper.GetZoomedMDIChild(AHandle: HWND): TCustomForm;
var
AActiveChild: TCustomForm;
begin
AActiveChild := GetActiveMDIChild(AHandle);
if Assigned(AActiveChild) and IsZoomed(AActiveChild.Handle) then
Result := AActiveChild
else
Result := nil;
end;
class function TdxSkinFormHelper.HasClientEdge(AHandle: HWND): Boolean;
begin
Result := GetWindowLong(AHandle, GWL_EXSTYLE) and WS_EX_CLIENTEDGE <> 0;
end;
class function TdxSkinFormHelper.IsAlphaBlendUsed(AHandle: HWND): Boolean;
begin
Result := GetWindowLong(AHandle, GWL_EXSTYLE) and WS_EX_LAYERED <> 0;
AHandle := GetParent(AHandle);
if TdxSkinWinController.IsMDIClientWindow(AHandle) then
Result := GetWindowLong(GetParent(AHandle), GWL_EXSTYLE) and WS_EX_LAYERED <> 0;
end;
{ TdxSkinWinController }
constructor TdxSkinWinController.Create(AHandle: HWND);
begin
FProcInstance := Classes.MakeObjectInstance(WndProc);
Handle := AHandle;
end;
destructor TdxSkinWinController.Destroy;
begin
MasterDestroyed;
EndMouseTracking(Self);
Handle := 0;
Classes.FreeObjectInstance(FProcInstance);
inherited Destroy;
end;
class procedure TdxSkinWinController.InitializeEngine(AHandle: HWND);
var
ANewController: TdxSkinWinController;
begin
ANewController := GetControllerByHandle(AHandle);
if ANewController = nil then
begin
ANewController := GetControllerByControl(FindControl(AHandle));
if ANewController <> nil then
ANewController.Handle := AHandle
else
begin
ANewController := Self.Create(AHandle);
ANewController.FMaster := ANewController.GetMaster(AHandle);
FormControllers.Add(ANewController);
end;
ANewController.InitializePainter;
ANewController.Update;
end;
end;
class procedure TdxSkinWinController.FinalizeEngine(AHandle: HWND);
var
AController: TdxSkinWinController;
begin
AController := GetControllerByHandle(AHandle);
if Assigned(AController) then
begin
FormControllers.Remove(AController);
AController.Free;
end;
end;
class function TdxSkinWinController.IsMDIChildWindow(AHandle: HWND): Boolean;
var
AControl: TWinControl;
begin
AControl := FindControl(AHandle);
Result := (AControl is TCustomForm) and
(TCustomFormAccess(AControl).FormStyle = fsMDIChild);
end;
class function TdxSkinWinController.IsMDIClientWindow(AHandle: HWND): Boolean;
begin
Result := AnsiSameText(GetWindowClass(AHandle), 'MDICLIENT');
end;
class function TdxSkinWinController.IsSkinActive(AHandle: HWND): Boolean;
var
AController: TdxSkinWinController;
begin
AController := GetControllerByHandle(AHandle);
Result := Assigned(AController) and AController.IsSkinUsed;
end;
class function TdxSkinWinController.IsMessageDlgWindow(AHandle: HWND): Boolean;
var
AWindowClass: string;
begin
AWindowClass := GetWindowClass(AHandle);
Result := SameText(AWindowClass, 'TMessageForm') or SameText(AWindowClass, 'TForm');
end;
function TdxSkinWinController.GetHasVirtualChilds: Boolean;
var
AControl: TWinControl;
I: Integer;
begin
AControl := WinControl;
Result := False;
if Assigned(AControl) then
for I := 0 to AControl.ControlCount - 1 do
begin
Result := not (AControl.Controls[I] is TWinControl);
if Result then
Break;
end;
end;
function TdxSkinWinController.GetIsHooked: Boolean;
begin
Result := (Handle <> 0) and (Assigned(FSavedWndProcPtr) or Assigned(FSavedWndProc));
end;
function TdxSkinWinController.GetIsMDIClient: Boolean;
begin
Result := Assigned(FMaster);
end;
function TdxSkinWinController.GetLookAndFeelPainter: TdxSkinLookAndFeelPainterClass;
begin
Result := FLookAndFeelPainter;
if Master <> nil then
Result := Master.LookAndFeelPainter;
end;
procedure TdxSkinWinController.SetHandle(AHandle: HWND);
begin
UnHookWndProc;
FHandle := AHandle;
FWinControl := FindControl(Handle);
Update;
end;
function TdxSkinWinController.GetIsSkinUsed: Boolean;
begin
Result := LookAndFeelPainter <> nil;
end;
function TdxSkinWinController.GetMaster(AHandle: HWND): TdxSkinWinController;
begin
Result := nil;
if IsMDIClientWindow(AHandle) then
Result := GetControllerByHandle(GetParent(AHandle));
end;
function TdxSkinWinController.GetUseSkinForControl: Boolean;
var
AControl: TWinControl;
AIndex: Integer;
ASkinController: TdxSkinController;
AUseSkin: Boolean;
begin
AControl := WinControl;
AUseSkin := cxUseSkins and (AControl <> nil);
if Assigned(AControl) then
for AIndex := dxSkinControllersList.Count - 1 downto 0 do
begin
ASkinController := TdxSkinController(dxSkinControllersList[AIndex]);
if (csDestroying in ASkinController.ComponentState) then
Continue;
AUseSkin := ASkinController.DoSkinControl(AControl);
if AUseSkin then
Break;
end;
Result := AUseSkin;
end;
function TdxSkinWinController.IsHookAvailable: Boolean;
begin
Result := IsSkinUsed;
end;
procedure TdxSkinWinController.DefWndProc(var AMessage);
begin
if FSavedWndProcPtr <> nil then
with TMessage(AMessage) do
Result := CallWindowProc(FSavedWndProcPtr, Handle, Msg, wParam, lParam)
else
if Assigned(FSavedWndProc) then
FSavedWndProc(TMessage(AMessage));
end;
procedure TdxSkinWinController.HookWndProc;
begin
UnHookWndProc;
if (Handle <> 0) and IsHookAvailable then
begin
if FWinControl = nil then
begin
FSavedWndProcPtr := Pointer(GetWindowLong(Handle, GWL_WNDPROC));
SetWindowLong(Handle, GWL_WNDPROC, Integer(FProcInstance));
end
else
begin
FSavedWndProc := FWinControl.WindowProc;
FWinControl.WindowProc := WndProc;
end;
end;
end;
procedure TdxSkinWinController.InitializePainter;
begin
// nothing todo
end;
procedure TdxSkinWinController.MasterDestroyed;
var
AController: TdxSkinWinController;
I: Integer;
begin
for I := 0 to FormControllers.Count - 1 do
begin
AController := TdxSkinWinController(FormControllers[I]);
if AController.FMaster = Self then
AController.FMaster := nil;
end;
end;
procedure TdxSkinWinController.RedrawWindow(AUpdateNow: Boolean);
var
AFlags: Integer;
const
DefaultFlags = RDW_ERASE or RDW_INVALIDATE or RDW_FRAME or RDW_ALLCHILDREN;
begin
if Handle = 0 then Exit;
AFlags := DefaultFlags;
if AUpdateNow then
AFlags := AFlags or RDW_UPDATENOW;
Windows.RedrawWindow(Handle, nil, 0, AFlags);
end;
procedure TdxSkinWinController.UnHookWndProc;
begin
if IsHooked then
begin
if FWinControl = nil then
SetWindowLong(Handle, GWL_WNDPROC, Integer(FSavedWndProcPtr))
else
FWinControl.WindowProc := FSavedWndProc;
FSavedWndProcPtr := nil;
FSavedWndProc := nil;
end;
end;
function TdxSkinWinController.GetSkinName(var ASkinName: string): Boolean;
begin
Result := GetExtendedStylePainters.GetNameByPainter(FLookAndFeelPainter, ASkinName);
end;
procedure TdxSkinWinController.Refresh;
const
Flags = SWP_FRAMECHANGED or SWP_NOACTIVATE or SWP_NOMOVE or SWP_NOSIZE or
SWP_NOZORDER;
begin
if Handle <> 0 then
SetWindowPos(Handle, 0, 0, 0, 0, 0, Flags);
end;
procedure TdxSkinWinController.Update;
begin
HookWndProc;
Refresh;
RedrawWindow(HasVirtualChilds);
end;
procedure TdxSkinWinController.WndProc(var AMessage: TMessage);
begin
DefWndProc(AMessage);
end;
procedure TdxSkinWinController.MouseLeave;
begin
end;
{ TdxSkinCustomController }
constructor TdxSkinCustomController.Create(AHandle: HWND);
begin
inherited Create(AHandle);
FViewInfo := GetViewInfoClass.Create(Self);
FPainter := TdxSkinCustomControlPainter.Create(FViewInfo);
end;
destructor TdxSkinCustomController.Destroy;
begin
FreeAndNil(FViewInfo);
FreeAndNil(FPainter);
inherited Destroy;
end;
procedure TdxSkinCustomController.DrawBackground(DC: HDC = 0);
begin
Painter.BeginPaint(DC);
try
Painter.DrawBackground;
finally
Painter.EndPaint;
end;
end;
procedure TdxSkinCustomController.DrawContent(DC: HDC = 0);
begin
end;
class function TdxSkinCustomController.GetViewInfoClass: TdxSkinCustomControlViewInfoClass;
begin
Result := TdxSkinCustomControlViewInfo;
end;
function TdxSkinCustomController.GetMaster(AHandle: HWND): TdxSkinWinController;
var
AParent: TCustomForm;
begin
Result := nil;
if Assigned(WinControl) then
begin
AParent := GetParentForm(WinControl);
if Assigned(AParent) and GetUseSkinForControl then
Result := GetControllerByHandle(AParent.Handle)
end;
end;
procedure TdxSkinCustomController.InitializePainter;
begin
// nothing todo
end;
function TdxSkinCustomController.WMEraseBk(var AMessage: TWMEraseBkgnd): Boolean;
begin
AMessage.Result := 1;
Result := True;
end;
function TdxSkinCustomController.WMPaint(var AMessage: TWMPaint): Boolean;
var
APaintStruct: TPaintStruct;
ADC: HDC;
begin
Result := True;
if AMessage.DC <> 0 then
DrawContent(AMessage.DC)
else
begin
ADC := BeginPaint(Handle, APaintStruct);
try
DrawContent(ADC);
finally
EndPaint(Handle, APaintStruct);
end;
end;
end;
procedure TdxSkinCustomController.WndProc(var AMessage: TMessage);
var
AHandled: Boolean;
begin
AHandled := False;
if IsSkinUsed then
begin
case AMessage.Msg of
WM_ERASEBKGND:
AHandled := WMEraseBk(TWMEraseBkgnd(AMessage));
WM_PAINT:
AHandled := WMPaint(TWMPaint(AMessage));
end;
end;
if not AHandled then
inherited WndProc(AMessage);
end;
{ TdxSkinWinControllerHelper }
constructor TdxSkinWinControllerHelper.Create;
begin
FHandle := {$IFDEF DELPHI6}Classes.{$ENDIF}AllocateHWnd(WndProc);
end;
destructor TdxSkinWinControllerHelper.Destroy;
begin
{$IFDEF DELPHI6}Classes.{$ENDIF}DeallocateHWnd(FHandle);
inherited Destroy;
end;
procedure TdxSkinWinControllerHelper.ChildChanged(AHandle: HWND);
begin
AHandle := GetParent(AHandle);
if (AHandle <> 0) and AnsiSameText(GetWindowClass(AHandle), 'MDICLIENT') then
begin
AHandle := GetParent(AHandle);
if AHandle <> 0 then
SendMessage(AHandle, WM_CHILDCHANGED, 0, 0);
end;
end;
procedure TdxSkinWinControllerHelper.FinalizeEngine(AHandle: HWND);
begin
TdxSkinWinController.FinalizeEngine(AHandle);
SkinHelper.ChildChanged(AHandle);
end;
procedure TdxSkinWinControllerHelper.InitializeEngine(AHandle: HWND);
begin
if SendMessage(AHandle, WM_HASOWNSKINENGINE, 0, 0) = 0 then
begin
if {$IFNDEF DELPHI12} IsWindowUnicode(AHandle) or {$ENDIF} // note: try to detect TntControls
TdxSkinFormController.IsMDIClientWindow(AHandle)
then
PostMessage(SkinHelper.Handle, WM_POSTCREATE, 0, AHandle)
else
InternalInitializeEngine(AHandle);
end;
end;
procedure TdxSkinWinControllerHelper.InternalInitializeEngine(AHandle: HWND);
function IsAnotherApplicatonWindow(AWnd: HWND): Boolean;
var
AProcessId: Cardinal;
begin
GetWindowThreadProcessId(AWnd, @AProcessId);
Result := (AWnd = 0) or (AProcessId <> GetCurrentProcessId);
end;
function GetSkinClassForWindow(AWnd: HWND): TdxSkinWinControllerClass;
begin
Result := nil;
if not IsAnotherApplicatonWindow(AWnd) then
Result := dxSkinGetControllerClassForWindowProc(AWnd);
end;
var
ASkinClass: TdxSkinWinControllerClass;
begin
ASkinClass := GetSkinClassForWindow(AHandle);
if ASkinClass <> nil then
begin
ASkinClass.InitializeEngine(AHandle);
PostMessage(AHandle, WM_POSTCHECKRGN, 0, 0);
end;
end;
procedure TdxSkinWinControllerHelper.WndProc(var AMsg: TMessage);
begin
if AMsg.Msg = WM_POSTCREATE then
InternalInitializeEngine(AMsg.LParam)
else
AMsg.Result := DefWindowProc(Handle, AMsg.Msg, AMsg.WParam, AMsg.LParam);
end;
initialization
dxSkinGetControllerClassForWindowProc := dxSkinGetControllerClassForWindow;
RegisterAssistants;
finalization
UnregisterAssistants;
end.