{ /*************************************************************************** designer.pp - Lazarus IDE unit -------------------------------- Initial Revision : Sat May 10 23:15:32 CST 1999 ***************************************************************************/ *************************************************************************** * * * This source is free software; you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * * the Free Software Foundation; either version 2 of the License, or * * (at your option) any later version. * * * * This code is distributed in the hope that it will be useful, but * * WITHOUT ANY WARRANTY; without even the implied warranty of * * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * * General Public License for more details. * * * * A copy of the GNU General Public License is available on the World * * Wide Web at . You can also * * obtain it by writing to the Free Software Foundation, * * Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. * * * *************************************************************************** } unit Designer; {$mode objfpc}{$H+} interface {off $DEFINE VerboseDesigner} {off $DEFINE VerboseDesignerDraw} {off $DEFINE VerboseDesignerSelect} uses // FCL + LCL Types, Classes, SysUtils, Math, LCLProc, LCLType, LResources, LCLIntf, LMessages, InterfaceBase, Forms, Controls, GraphType, Graphics, Dialogs, ExtCtrls, Menus, ClipBrd, TypInfo, // IDEIntf IDEDialogs, PropEdits, PropEditUtils, ComponentEditors, MenuIntf, IDEImagesIntf, FormEditingIntf, // IDE LazarusIDEStrConsts, EnvironmentOpts, IDECommands, ComponentReg, NonControlDesigner, FrameDesigner, AlignCompsDlg, SizeCompsDlg, ScaleCompsDlg, TabOrderDlg, AnchorEditor, DesignerProcs, CustomFormEditor, AskCompNameDlg, ControlSelection, ChangeClassDialog, EditorOptions; type TDesigner = class; TOnGetSelectedComponentClass = procedure(Sender: TObject; var RegisteredComponent: TRegisteredComponent) of object; TOnSetDesigning = procedure(Sender: TObject; Component: TComponent; Value: boolean) of object; TOnPasteComponent = procedure(Sender: TObject; LookupRoot: TComponent; TxtCompStream: TStream; Parent: TWinControl; var NewComponent: TComponent) of object; TOnPersistentDeleted = procedure(Sender: TObject; APersistent: TPersistent) of object; TOnGetNonVisualCompIcon = procedure(Sender: TObject; AComponent: TComponent; var Icon: TCustomBitmap) of object; TOnRenameComponent = procedure(Designer: TDesigner; AComponent: TComponent; const NewName: string) of object; TOnProcessCommand = procedure(Sender: TObject; Command: word; var Handled: boolean) of object; TDesignerFlag = ( dfHasSized, dfDuringPaintControl, dfShowEditorHints, dfShowComponentCaptions, dfDestroyingForm, dfDeleting, dfNeedPainting ); TDesignerFlags = set of TDesignerFlag; { TDesigner } TDesigner = class(TComponentEditorDesigner) private FDesignerPopupMenu: TPopupMenu; FDefaultFormBounds: TRect; FLastFormBounds: TRect; FDefaultFormBoundsValid: boolean; FFlags: TDesignerFlags; FGridColor: TColor; FLookupRoot: TComponent; FMediator: TDesignerMediator; FOnActivated: TNotifyEvent; FOnCloseQuery: TNotifyEvent; FOnShowObjectInspector: TNotifyEvent; FOnShowAnchorEditor: TNotifyEvent; FOnShowTabOrderEditor: TNotifyEvent; FOnPersistentDeleted: TOnPersistentDeleted; FOnGetNonVisualCompIcon: TOnGetNonVisualCompIcon; FOnGetSelectedComponentClass: TOnGetSelectedComponentClass; FOnModified: TNotifyEvent; FOnPasteComponent: TOnPasteComponent; FOnProcessCommand: TOnProcessCommand; FOnPropertiesChanged: TNotifyEvent; FOnRenameComponent: TOnRenameComponent; FOnSaveAsXML: TNotifyEvent; FOnSetDesigning: TOnSetDesigning; FOnShowOptions: TNotifyEvent; FOnComponentAdded: TNotifyEvent; FOnViewLFM: TNotifyEvent; FShiftState: TShiftState; FTheFormEditor: TCustomFormEditor; FPopupMenuComponentEditor: TBaseComponentEditor; //hint stuff FHintTimer: TTimer; FHintWIndow: THintWindow; // component drawing FDDC: TDesignerDeviceContext; FSurface: TBitmap; procedure DrawNonVisualComponent(AComponent: TComponent); function GetGridColor: TColor; function GetGridSizeX: integer; function GetGridSizeY: integer; function GetIsControl: Boolean; function GetShowBorderSpacing: boolean; function GetShowComponentCaptions: boolean; function GetShowEditorHints: boolean; function GetShowGrid: boolean; function GetSnapToGrid: boolean; procedure HintTimer(Sender : TObject); procedure InvalidateWithParent(AComponent: TComponent); procedure SetDefaultFormBounds(const AValue: TRect); procedure SetGridColor(const AValue: TColor); procedure SetGridSizeX(const AValue: integer); procedure SetGridSizeY(const AValue: integer); procedure SetIsControl(Value: Boolean); procedure SetMediator(const AValue: TDesignerMediator); procedure SetPopupMenuComponentEditor(const AValue: TBaseComponentEditor); procedure SetShowBorderSpacing(const AValue: boolean); procedure SetShowComponentCaptions(const AValue: boolean); procedure SetShowEditorHints(const AValue: boolean); procedure SetShowGrid(const AValue: boolean); procedure SetSnapToGrid(const AValue: boolean); protected MouseDownComponent: TComponent; MouseDownSender: TComponent; MouseDownPos: TPoint; MouseDownShift: TShiftState; MouseUpPos: TPoint; LastMouseMovePos: TPoint; LastFormCursor: TCursor; DeletingPersistent: TList; IgnoreDeletingPersistent: TList; LastPaintSender: TControl; // event handlers for designed components function PaintControl(Sender: TControl; TheMessage: TLMPaint): Boolean; function SizeControl(Sender: TControl; TheMessage: TLMSize): Boolean; function MoveControl(Sender: TControl; TheMessage: TLMMove): Boolean; procedure MouseDownOnControl(Sender: TControl; var TheMessage: TLMMouse); procedure MouseMoveOnControl(Sender: TControl; var TheMessage: TLMMouse); procedure MouseUpOnControl(Sender: TControl; var TheMessage: TLMMouse); procedure KeyDown(Sender: TControl; var TheMessage: TLMKEY); procedure KeyUp(Sender: TControl; var TheMessage: TLMKEY); function HandleSetCursor(var TheMessage: TLMessage): boolean; procedure HandlePopupMenu(Sender: TControl; var Message: TLMContextMenu); procedure GetMouseMsgShift(TheMessage: TLMMouse; var Shift: TShiftState; var Button: TMouseButton); // procedures for working with components and persistents function GetDesignControl(AControl: TControl): TControl; function DoDeleteSelectedPersistents: boolean; procedure DoSelectAll; procedure DoDeletePersistent(APersistent: TPersistent; FreeIt: boolean); procedure MarkPersistentForDeletion(APersistent: TPersistent); function PersistentIsMarkedForDeletion(APersistent: TPersistent): boolean; function GetSelectedComponentClass: TRegisteredComponent; procedure NudgePosition(DiffX, DiffY: Integer); procedure NudgeSize(DiffX, DiffY: Integer); procedure NudgeSelection(DiffX, DiffY: Integer); overload; procedure NudgeSelection(SelectNext: Boolean); overload; procedure SelectParentOfSelection; function DoCopySelectionToClipboard: boolean; function GetPasteParent: TWinControl; procedure DoModified; function DoPasteSelectionFromClipboard(PasteFlags: TComponentPasteSelectionFlags ): boolean; function DoInsertFromStream(s: TStream; PasteParent: TWinControl; PasteFlags: TComponentPasteSelectionFlags): Boolean; procedure DoShowAnchorEditor; procedure DoShowTabOrderEditor; procedure DoShowChangeClassDialog; procedure DoShowObjectInspector; procedure DoOrderMoveSelectionToFront; procedure DoOrderMoveSelectionToBack; procedure DoOrderForwardSelectionOne; procedure DoOrderBackSelectionOne; procedure GiveComponentsNames; procedure NotifyPersistentAdded(APersistent: TPersistent); function ComponentClassAtPos(const AClass: TComponentClass; const APos: TPoint; const UseRootAsDefault, IgnoreHidden: boolean): TComponent; procedure SetTempCursor(ARoot: TWinControl; ACursor: TCursor); // popup menu procedure BuildPopupMenu; procedure DesignerPopupMenuPopup(Sender: TObject); procedure OnComponentEditorVerbMenuItemClick(Sender: TObject); procedure OnAlignPopupMenuClick(Sender: TObject); procedure OnMirrorHorizontalPopupMenuClick(Sender: TObject); procedure OnMirrorVerticalPopupMenuClick(Sender: TObject); procedure OnScalePopupMenuClick(Sender: TObject); procedure OnSizePopupMenuClick(Sender: TObject); procedure OnAnchorEditorMenuClick(Sender: TObject); procedure OnTabOrderMenuClick(Sender: TObject); procedure OnOrderMoveToFrontMenuClick(Sender: TObject); procedure OnOrderMoveToBackMenuClick(Sender: TObject); procedure OnOrderForwardOneMenuClick(Sender: TObject); procedure OnOrderBackOneMenuClick(Sender: TObject); procedure OnCopyMenuClick(Sender: TObject); procedure OnCutMenuClick(Sender: TObject); procedure OnPasteMenuClick(Sender: TObject); procedure OnDeleteSelectionMenuClick(Sender: TObject); procedure OnSelectAllMenuClick(Sender: TObject); procedure OnChangeClassMenuClick(Sender: TObject); procedure OnChangeParentMenuClick(Sender: TObject); procedure OnSnapToGridOptionMenuClick(Sender: TObject); procedure OnShowOptionsMenuItemClick(Sender: TObject); procedure OnSnapToGuideLinesOptionMenuClick(Sender: TObject); procedure OnViewLFMMenuClick(Sender: TObject); procedure OnSaveAsXMLMenuClick(Sender: TObject); procedure OnCenterFormMenuClick(Sender: TObject); // hook function GetPropertyEditorHook: TPropertyEditorHook; override; function OnFormActivated: boolean; function OnFormCloseQuery: boolean; property PopupMenuComponentEditor: TBaseComponentEditor read FPopupMenuComponentEditor write SetPopupMenuComponentEditor; public ControlSelection : TControlSelection; DDC: TDesignerDeviceContext; constructor Create(TheDesignerForm: TCustomForm; AControlSelection: TControlSelection); procedure FreeDesigner(FreeComponent: boolean); destructor Destroy; override; procedure Modified; override; procedure SelectOnlyThisComponent(AComponent: TComponent); override; function CopySelection: boolean; override; function CutSelection: boolean; override; function CanPaste: Boolean; override; function PasteSelection(PasteFlags: TComponentPasteSelectionFlags): boolean; override; function DeleteSelection: boolean; override; function CopySelectionToStream(AllComponentsStream: TStream): boolean; override; function InsertFromStream(s: TStream; Parent: TWinControl; PasteFlags: TComponentPasteSelectionFlags): Boolean; override; function InvokeComponentEditor(AComponent: TComponent; MenuIndex: integer): boolean; override; procedure DoProcessCommand(Sender: TObject; var Command: word; var Handled: boolean); function NonVisualComponentLeftTop(AComponent: TComponent): TPoint; function NonVisualComponentAtPos(X, Y: integer): TComponent; procedure MoveNonVisualComponentIntoForm(AComponent: TComponent); procedure MoveNonVisualComponentsIntoForm; function WinControlAtPos(x,y: integer; UseRootAsDefault, IgnoreHidden: boolean): TWinControl; function ControlAtPos(x,y: integer; UseRootAsDefault, IgnoreHidden: boolean): TControl; function ComponentAtPos(x,y: integer; UseRootAsDefault, IgnoreHidden: boolean): TComponent; function GetDesignedComponent(AComponent: TComponent): TComponent; function GetComponentEditorForSelection: TBaseComponentEditor; function GetShiftState: TShiftState; override; procedure AddComponentEditorMenuItems(AComponentEditor: TBaseComponentEditor; ClearOldOnes: boolean); function IsDesignMsg(Sender: TControl; var TheMessage: TLMessage): Boolean; override; function UniqueName(const BaseName: string): string; override; Procedure RemovePersistentAndChilds(APersistent: TPersistent); procedure Notification(AComponent: TComponent; Operation: TOperation); override; procedure ValidateRename(AComponent: TComponent; const CurName, NewName: string); override; function CreateUniqueComponentName(const AClassName: string): string; override; procedure PaintGrid; override; procedure PaintClientGrid(AWinControl: TWinControl; aDDC: TDesignerDeviceContext); procedure DrawNonVisualComponents(aDDC: TDesignerDeviceContext); procedure DrawDesignerItems(OnlyIfNeeded: boolean); override; procedure CheckFormBounds; procedure DoPaintDesignerItems; function ComponentIsIcon(AComponent: TComponent): boolean; function GetParentFormRelativeClientOrigin(AComponent: TComponent): TPoint; public property Flags: TDesignerFlags read FFlags; property GridSizeX: integer read GetGridSizeX write SetGridSizeX; property GridSizeY: integer read GetGridSizeY write SetGridSizeY; property GridColor: TColor read GetGridColor write SetGridColor; property IsControl: Boolean read GetIsControl write SetIsControl; property LookupRoot: TComponent read FLookupRoot; property Mediator: TDesignerMediator read FMediator write SetMediator; property OnActivated: TNotifyEvent read FOnActivated write FOnActivated; property OnCloseQuery: TNotifyEvent read FOnCloseQuery write FOnCloseQuery; property OnPersistentDeleted: TOnPersistentDeleted read FOnPersistentDeleted write FOnPersistentDeleted; property OnGetNonVisualCompIcon: TOnGetNonVisualCompIcon read FOnGetNonVisualCompIcon write FOnGetNonVisualCompIcon; property OnGetSelectedComponentClass: TOnGetSelectedComponentClass read FOnGetSelectedComponentClass write FOnGetSelectedComponentClass; property OnProcessCommand: TOnProcessCommand read FOnProcessCommand write FOnProcessCommand; property OnModified: TNotifyEvent read FOnModified write FOnModified; property OnPasteComponent: TOnPasteComponent read FOnPasteComponent write FOnPasteComponent; property OnPropertiesChanged: TNotifyEvent read FOnPropertiesChanged write FOnPropertiesChanged; property OnRenameComponent: TOnRenameComponent read FOnRenameComponent write FOnRenameComponent; property OnSetDesigning: TOnSetDesigning read FOnSetDesigning write FOnSetDesigning; property OnComponentAdded: TNotifyEvent read FOnComponentAdded write FOnComponentAdded; property OnShowOptions: TNotifyEvent read FOnShowOptions write FOnShowOptions; property OnViewLFM: TNotifyEvent read FOnViewLFM write FOnViewLFM; property OnSaveAsXML: TNotifyEvent read FOnSaveAsXML write FOnSaveAsXML; property OnShowObjectInspector: TNotifyEvent read FOnShowObjectInspector write FOnShowObjectInspector; property OnShowAnchorEditor: TNotifyEvent read FOnShowAnchorEditor write FOnShowAnchorEditor; property OnShowTabOrderEditor: TNotifyEvent read FOnShowTabOrderEditor write FOnShowTabOrderEditor; property ShowGrid: boolean read GetShowGrid write SetShowGrid; property ShowBorderSpacing: boolean read GetShowBorderSpacing write SetShowBorderSpacing; property ShowEditorHints: boolean read GetShowEditorHints write SetShowEditorHints; property ShowComponentCaptions: boolean read GetShowComponentCaptions write SetShowComponentCaptions; property SnapToGrid: boolean read GetSnapToGrid write SetSnapToGrid; property TheFormEditor: TCustomFormEditor read FTheFormEditor write FTheFormEditor; property DefaultFormBounds: TRect read FDefaultFormBounds write SetDefaultFormBounds; property DefaultFormBoundsValid: boolean read FDefaultFormBoundsValid write FDefaultFormBoundsValid; end; const DesignerMenuRootName = 'Designer'; var DesignerMenuAlign: TIDEMenuCommand; DesignerMenuMirrorHorizontal: TIDEMenuCommand; DesignerMenuMirrorVertical: TIDEMenuCommand; DesignerMenuScale: TIDEMenuCommand; DesignerMenuSize: TIDEMenuCommand; DesignerMenuAnchorEditor: TIDEMenuCommand; DesignerMenuTabOrder: TIDEMenuCommand; DesignerMenuOrderMoveToFront: TIDEMenuCommand; DesignerMenuOrderMoveToBack: TIDEMenuCommand; DesignerMenuOrderForwardOne: TIDEMenuCommand; DesignerMenuOrderBackOne: TIDEMenuCommand; DesignerMenuCut: TIDEMenuCommand; DesignerMenuCopy: TIDEMenuCommand; DesignerMenuPaste: TIDEMenuCommand; DesignerMenuDeleteSelection: TIDEMenuCommand; DesignerMenuSelectAll: TIDEMenuCommand; DesignerMenuChangeClass: TIDEMenuCommand; DesignerMenuChangeParent: TIDEMenuSection; DesignerMenuViewLFM: TIDEMenuCommand; DesignerMenuSaveAsXML: TIDEMenuCommand; DesignerMenuCenterForm: TIDEMenuCommand; DesignerMenuSnapToGridOption: TIDEMenuCommand; DesignerMenuSnapToGuideLinesOption: TIDEMenuCommand; DesignerMenuShowOptions: TIDEMenuCommand; procedure RegisterStandardDesignerMenuItems; implementation type TCustomFormAccess = class(TCustomForm); TControlAccess = class(TControl); TComponentAccess = class(TComponent); { TComponentSearch } TComponentSearch = class(TComponent) public Best: TComponent; BestLevel: integer; BestIsNonVisual: boolean; Level: integer; AtPos: TPoint; MinClass: TComponentClass; IgnoreHidden: boolean; OnlyNonVisual: boolean; Mediator: TDesignerMediator; Root: TComponent; procedure Gather(Child: TComponent); procedure Search(ARoot: TComponent); end; { TComponentSearch } procedure TComponentSearch.Gather(Child: TComponent); var Control: TControl; ChildBounds: TRect; OldRoot: TComponent; IsNonVisual: Boolean; begin if Assigned(Best) and BestIsNonVisual and (BestLevel < Level) then exit; {$IFDEF VerboseDesignerSelect} DebugLn(['TComponentSearch.Gather ',DbgSName(Child),' ',dbgs(AtPos),' MinClass=',DbgSName(MinClass)]); {$ENDIF} // check if child is at position if Child is TControl then begin Control := TControl(Child); if IgnoreHidden and (csNoDesignVisible in Control.ControlStyle) then exit; if csNoDesignSelectable in Control.ControlStyle then exit; end else Control := nil; ChildBounds := GetParentFormRelativeBounds(Child); {$IFDEF VerboseDesignerSelect} DebugLn(['TComponentSearch.Gather PtInRect=',PtInRect(ChildBounds, AtPos),' ChildBounds=',dbgs(ChildBounds)]); {$ENDIF} if not PtInRect(ChildBounds, AtPos) then Exit; if Assigned(Mediator) then IsNonVisual := Mediator.ComponentIsIcon(Child) else IsNonVisual := DesignerProcs.ComponentIsNonVisual(Child); if Child.InheritsFrom(MinClass) and (IsNonVisual or not OnlyNonVisual) then begin Best := Child; BestIsNonVisual := IsNonVisual; BestLevel := Level; {$IFDEF VerboseDesignerSelect} DebugLn(['TComponentSearch.Gather Best=',DbgSName(Best)]); {$ENDIF} end; // search in children if (csInline in Child.ComponentState) or (Assigned(Control) and not (csOwnedChildrenNotSelectable in Control.ControlStyle)) then begin {$IFDEF VerboseDesignerSelect} DebugLn(['TComponentSearch.Gather search in children of ',DbgSName(Child)]); {$ENDIF} OldRoot := Root; try inc(Level); if csInline in Child.ComponentState then Root := Child; {$IFDEF VerboseDesignerSelect} DebugLn(['TComponentSearch.Gather Root=',DbgSName(Root)]); {$ENDIF} TComponentAccess(Child).GetChildren(@Gather, Root); finally dec(Level); Root := OldRoot; end; {$IFDEF VerboseDesignerSelect} DebugLn(['TComponentSearch.Gather searched in children of ',DbgSName(Child)]); {$ENDIF} end; end; procedure TComponentSearch.Search(ARoot: TComponent); begin Root := ARoot; Level := 1; TComponentAccess(Root).GetChildren(@Gather, Root); Level := 0; end; const mk_lbutton = 1; mk_rbutton = 2; mk_shift = 4; mk_control = 8; mk_mbutton = $10; procedure RegisterStandardDesignerMenuItems; begin DesignerMenuRoot:=RegisterIDEMenuRoot(DesignerMenuRootName); // register the dynamic section for the component editor DesignerMenuSectionComponentEditor:=RegisterIDEMenuSection(DesignerMenuRoot, 'Component editor section'); // register the custom dynamic section DesignerMenuSectionCustomDynamic:=RegisterIDEMenuSection(DesignerMenuRoot, 'Custom dynamic section'); // register align section DesignerMenuSectionAlign:=RegisterIDEMenuSection(DesignerMenuRoot, 'Align section'); DesignerMenuAlign:=RegisterIDEMenuCommand(DesignerMenuSectionAlign, 'Align',fdmAlignMenu, nil, nil, nil, 'align'); DesignerMenuMirrorHorizontal:=RegisterIDEMenuCommand(DesignerMenuSectionAlign, 'Mirror horizontal',fdmMirrorHorizontal, nil, nil, nil, 'mirror_horizontal'); DesignerMenuMirrorVertical:=RegisterIDEMenuCommand(DesignerMenuSectionAlign, 'Mirror vertical',fdmMirrorVertical, nil, nil, nil, 'mirror_vertical'); DesignerMenuScale:=RegisterIDEMenuCommand(DesignerMenuSectionAlign, 'Scale',fdmScaleMenu, nil, nil, nil, 'scale'); DesignerMenuSize:=RegisterIDEMenuCommand(DesignerMenuSectionAlign, 'Size',fdmSizeMenu, nil, nil, nil, 'size'); // register tab and z-order section DesignerMenuSectionOrder:=RegisterIDEMenuSection(DesignerMenuRoot,'Order section'); DesignerMenuAnchorEditor:=RegisterIDEMenuCommand(DesignerMenuSectionOrder, 'Anchor Editor',lisMenuViewAnchorEditor, nil, nil, nil, 'menu_view_anchor_editor'); DesignerMenuTabOrder:=RegisterIDEMenuCommand(DesignerMenuSectionOrder, 'Tab order',lisMenuViewTabOrder, nil, nil, nil, 'tab_order'); DesignerMenuSectionZOrder:=RegisterIDESubMenu(DesignerMenuSectionOrder, 'ZOrder section', fdmZOrder); DesignerMenuOrderMoveToFront:=RegisterIDEMenuCommand(DesignerMenuSectionZOrder, 'Move to z order front',fdmOrderMoveTofront, nil, nil, nil, 'Order_move_front'); DesignerMenuOrderMoveToBack:=RegisterIDEMenuCommand(DesignerMenuSectionZOrder, 'Move to z order back',fdmOrderMoveToBack, nil, nil, nil, 'Order_move_back'); DesignerMenuOrderForwardOne:=RegisterIDEMenuCommand(DesignerMenuSectionZOrder, 'Move z order forward one',fdmOrderForwardOne, nil, nil, nil, 'Order_forward_one'); DesignerMenuOrderBackOne:=RegisterIDEMenuCommand(DesignerMenuSectionZOrder, 'Move z order backwards one',fdmOrderBackOne, nil, nil, nil, 'Order_back_one'); // register clipboard section DesignerMenuSectionClipboard:=RegisterIDEMenuSection(DesignerMenuRoot, 'Clipboard section'); DesignerMenuCut:=RegisterIDEMenuCommand(DesignerMenuSectionClipboard, 'Cut',lisMenuCut, nil, nil, nil, 'laz_cut'); DesignerMenuCopy:=RegisterIDEMenuCommand(DesignerMenuSectionClipboard, 'Copy',lisMenuCopy, nil, nil, nil, 'laz_copy'); DesignerMenuPaste:=RegisterIDEMenuCommand(DesignerMenuSectionClipboard, 'Paste',lisMenuPaste, nil, nil, nil, 'laz_paste'); DesignerMenuDeleteSelection:=RegisterIDEMenuCommand(DesignerMenuSectionClipboard, 'Delete Selection',fdmDeleteSelection, nil, nil, nil, 'delete_selection'); DesignerMenuSelectAll:=RegisterIDEMenuCommand(DesignerMenuSectionClipboard, 'Select All',fdmSelectAll, nil, nil, nil, 'menu_select_all'); // register miscellaneous section DesignerMenuSectionMisc:=RegisterIDEMenuSection(DesignerMenuRoot, 'Miscellaneous section'); DesignerMenuChangeClass:=RegisterIDEMenuCommand(DesignerMenuSectionMisc, 'Change class',lisChangeClass); DesignerMenuChangeParent:=RegisterIDEMenuSection(DesignerMenuSectionMisc, 'Change parent'); DesignerMenuChangeParent.ChildsAsSubMenu:=true; DesignerMenuChangeParent.Caption:=lisChangeParent; DesignerMenuViewLFM:=RegisterIDEMenuCommand(DesignerMenuSectionMisc, 'View LFM',lisViewSourceLfm); DesignerMenuSaveAsXML:=RegisterIDEMenuCommand(DesignerMenuSectionMisc, 'Save as XML',fdmSaveFormAsXML); DesignerMenuCenterForm:=RegisterIDEMenuCommand(DesignerMenuSectionMisc, 'Center form', lisCenterForm); // register options section DesignerMenuSectionOptions:=RegisterIDEMenuSection(DesignerMenuRoot, 'Options section'); DesignerMenuSnapToGridOption:=RegisterIDEMenuCommand(DesignerMenuSectionOptions, 'Snap to grid',fdmSnapToGridOption); DesignerMenuSnapToGuideLinesOption:=RegisterIDEMenuCommand(DesignerMenuSectionOptions, 'Snap to guide lines',fdmSnapToGuideLinesOption); DesignerMenuShowOptions:=RegisterIDEMenuCommand(DesignerMenuSectionOptions, 'Show options',dlgFROpts, nil, nil, nil, 'menu_environment_options'); end; constructor TDesigner.Create(TheDesignerForm: TCustomForm; AControlSelection: TControlSelection); begin inherited Create; FForm := TheDesignerForm; if FForm is TNonControlDesignerForm then begin FLookupRoot := TNonControlDesignerForm(FForm).LookupRoot; Mediator:=TNonControlDesignerForm(FForm).Mediator; end else if FForm is TFrameDesignerForm then FLookupRoot := TFrameDesignerForm(FForm).LookupRoot else FLookupRoot := FForm; ControlSelection := AControlSelection; FFlags := []; FGridColor := clGray; FHintTimer := TTimer.Create(nil); FHintTimer.Interval := 500; FHintTimer.Enabled := False; FHintTimer.OnTimer := @HintTimer; FHintWindow := THintWindow.Create(nil); FHIntWindow.Visible := False; FHintWindow.HideInterval := 4000; FHintWindow.AutoHide := True; DDC:=TDesignerDeviceContext.Create; LastFormCursor := crDefault; DeletingPersistent:=TList.Create; IgnoreDeletingPersistent:=TList.Create; FPopupMenuComponentEditor := nil; end; procedure TDesigner.FreeDesigner(FreeComponent: boolean); var i: Integer; begin Include(FFlags, dfDestroyingForm); if FLookupRoot is TComponent then begin // unselect if TheControlSelection.LookupRoot = FLookupRoot then begin TheControlSelection.BeginUpdate; TheControlSelection.Clear; TheControlSelection.EndUpdate; end; if GlobalDesignHook.LookupRoot = FLookupRoot then GlobalDesignHook.LookupRoot := nil; if FreeComponent then begin // tell hooks about deleting for i := FLookupRoot.ComponentCount - 1 downto 0 do GlobalDesignHook.PersistentDeleting(FLookupRoot.Components[i]); GlobalDesignHook.PersistentDeleting(FLookupRoot); end; // delete if Form <> nil then Form.Designer := nil; if Mediator<>nil then Mediator.Designer:=nil; // free or hide the form TheFormEditor.DeleteComponent(FLookupRoot,FreeComponent); FMediator:=nil; end; Free; end; destructor TDesigner.Destroy; begin PopupMenuComponentEditor := nil; FreeAndNil(FDesignerPopupMenu); FreeAndNil(FHintWIndow); FreeAndNil(FHintTimer); FreeAndNil(DDC); FreeAndNil(DeletingPersistent); FreeAndNil(IgnoreDeletingPersistent); inherited Destroy; end; procedure TDesigner.NudgePosition(DiffX, DiffY : Integer); begin {$IFDEF VerboseDesigner} DebugLn('[TDesigner.NudgePosition]'); {$ENDIF} if (ControlSelection.SelectionForm<>Form) or ControlSelection.LookupRootSelected then exit; ControlSelection.MoveSelection(DiffX, DiffY, False); Modified; end; procedure TDesigner.NudgeSize(DiffX, DiffY: Integer); begin {$IFDEF VerboseDesigner} DebugLn('[TDesigner.NudgeSize]'); {$ENDIF} if (ControlSelection.SelectionForm<>Form) or ControlSelection.LookupRootSelected then exit; ControlSelection.SizeSelection(DiffX, DiffY); Modified; end; function ComponentsSortByLeft(Item1, Item2: Pointer): Integer; var Comp1: TComponent absolute Item1; Comp2: TComponent absolute Item2; L1, L2: Integer; begin L1 := GetComponentLeft(Comp1); L2 := GetComponentLeft(Comp2); if L1 < L2 then Result := -1 else if L1 > L2 then Result := 1 else Result := 0; end; function ComponentsSortByTop(Item1, Item2: Pointer): Integer; var Comp1: TComponent absolute Item1; Comp2: TComponent absolute Item2; T1, T2: Integer; begin T1 := GetComponentTop(Comp1); T2 := GetComponentTop(Comp2); if T1 < T2 then Result := -1 else if T1 > T2 then Result := 1 else Result := 0; end; procedure TDesigner.NudgeSelection(DiffX, DiffY: Integer); const Delta = 50; // radius for searching components var List: TFPList; Coord, Test: TPoint; Current, AComponent: TComponent; i: integer; begin if (ControlSelection.SelectionForm <> Form) or (ControlSelection.SelectionForm.ComponentCount = 0) or ControlSelection.LookupRootSelected or (ControlSelection.Count <> 1) then Exit; if not ControlSelection[0].IsTComponent then Exit; // create a list of components at the similar top/left Current := TComponent(ControlSelection[0].Persistent); AComponent := nil; List := TFPList.Create; try Coord := GetParentFormRelativeClientOrigin(Current); if DiffX <> 0 then begin for i := 0 to ControlSelection.SelectionForm.ComponentCount - 1 do begin AComponent := ControlSelection.SelectionForm.Components[i]; if (AComponent = Current) or ComponentIsInvisible(AComponent) then Continue; Test := GetParentFormRelativeClientOrigin(AComponent); if (Abs(Test.Y - Coord.Y) <= Delta) and (Sign(Test.X - Coord.X) = Sign(DiffX)) then List.Add(AComponent); end; if List.Count > 0 then begin List.Sort(@ComponentsSortByLeft); if DiffX > 0 then AComponent := TComponent(List[0]) else AComponent := TComponent(List[List.Count - 1]); end else AComponent := nil; end else if DiffY <> 0 then begin for i := 0 to ControlSelection.SelectionForm.ComponentCount - 1 do begin AComponent := ControlSelection.SelectionForm.Components[i]; if (AComponent = Current) or ComponentIsInvisible(AComponent) then Continue; Test := GetParentFormRelativeClientOrigin(AComponent); if (Abs(Test.X - Coord.X) <= Delta) and (Sign(Test.Y - Coord.Y) = Sign(DiffY)) then List.Add(AComponent); end; if List.Count > 0 then begin List.Sort(@ComponentsSortByTop); if DiffY > 0 then AComponent := TComponent(List[0]) else AComponent := TComponent(List[List.Count - 1]); end else AComponent := nil; end; finally List.Free; end; if AComponent <> nil then begin ControlSelection.AssignPersistent(AComponent); Modified; end; end; procedure TDesigner.NudgeSelection(SelectNext: Boolean); function StepIndex(Index: Integer): Integer; begin Result := Index; if SelectNext then Inc(Result) else Dec(Result); if Result >= ControlSelection.SelectionForm.ComponentCount then Result := 0 else if Result < 0 then Result := ControlSelection.SelectionForm.ComponentCount - 1; end; var Index, StartIndex: Integer; AComponent: TComponent; begin if (ControlSelection.SelectionForm <> Form) or (ControlSelection.SelectionForm.ComponentCount = 0) then Exit; if (ControlSelection.Count = 1) and ControlSelection[0].IsTComponent then Index := TComponent(ControlSelection[0].Persistent).ComponentIndex else Index := -1; Index := StepIndex(Index); StartIndex := Index; AComponent := nil; while AComponent = nil do begin AComponent := ControlSelection.SelectionForm.Components[Index]; if ComponentIsInvisible(AComponent) then begin AComponent := nil; Index := StepIndex(Index); if Index = StartIndex then break; end; end; if AComponent <> nil then begin ControlSelection.AssignPersistent(AComponent); Modified; end; end; procedure TDesigner.SelectParentOfSelection; function ParentComponent(AComponent: TComponent): TComponent; begin Result := AComponent.GetParentComponent; if (Result = nil) and ComponentIsIcon(AComponent) then Result := AComponent.Owner; end; var i: Integer; begin // resizing or moving if dfHasSized in FFlags then begin ControlSelection.RestoreBounds; ControlSelection.ActiveGrabber := nil; if ControlSelection.RubberbandActive then ControlSelection.RubberbandActive := False; LastMouseMovePos.X := -1; Exclude(FFlags, dfHasSized); MouseDownComponent := nil; MouseDownSender := nil; Exit; end; if ControlSelection.OnlyInvisiblePersistentsSelected then Exit; if ControlSelection.LookupRootSelected then begin SelectOnlyThisComponent(FLookupRoot); Exit; end; // if not component moving then select parent i := ControlSelection.Count - 1; while (i >= 0) and (ControlSelection[i].ParentInSelection or not ControlSelection[i].IsTComponent or (ParentComponent(TComponent(ControlSelection[i].Persistent)) = nil)) do Dec(i); if i >= 0 then SelectOnlyThisComponent(ParentComponent(TComponent(ControlSelection[i].Persistent))); end; function TDesigner.CopySelectionToStream(AllComponentsStream: TStream): boolean; function UnselectDistinctControls: boolean; var i: Integer; AParent, CurParent: TWinControl; begin Result:=false; AParent:=nil; i:=0; while iCurParent) then begin ControlSelection.Delete(i); continue; end; end; inc(i); end; Result:=true; end; var i: Integer; BinCompStream: TMemoryStream; TxtCompStream: TMemoryStream; CurComponent: TComponent; DestroyDriver: Boolean; Writer: TWriter; begin Result:=false; if (ControlSelection.Count=0) then exit; // Because controls will be pasted on a single parent, // unselect all controls, that do not have the same parent if not UnselectDistinctControls then exit; for i:=0 to ControlSelection.Count-1 do begin if not ControlSelection[i].IsTComponent then continue; BinCompStream:=TMemoryStream.Create; TxtCompStream:=TMemoryStream.Create; try // write component binary stream try CurComponent:=TComponent(ControlSelection[i].Persistent); DestroyDriver:=false; Writer := CreateLRSWriter(BinCompStream,DestroyDriver); try Writer.OnWriteMethodProperty:=@BaseFormEditor1.WriteMethodPropertyEvent; Writer.Root:=FLookupRoot; Writer.WriteComponent(CurComponent); finally if DestroyDriver then Writer.Driver.Free; Writer.Destroy; end; except on E: Exception do begin MessageDlg(lisUnableToStreamSelectedComponents, Format(lisThereWasAnErrorDuringWritingTheSelectedComponent, [ CurComponent.Name, CurComponent.ClassName, #13, E.Message]), mtError,[mbOk],0); exit; end; end; BinCompStream.Position:=0; // convert binary to text stream try LRSObjectBinaryToText(BinCompStream,TxtCompStream); except on E: Exception do begin MessageDlg(lisUnableConvertBinaryStreamToText, Format(lisThereWasAnErrorWhileConvertingTheBinaryStreamOfThe, [ CurComponent.Name, CurComponent.ClassName, #13, E.Message]), mtError,[mbOk],0); exit; end; end; // add text stream to the all stream TxtCompStream.Position:=0; AllComponentsStream.CopyFrom(TxtCompStream,TxtCompStream.Size); finally BinCompStream.Free; TxtCompStream.Free; end; end; Result:=true; end; function TDesigner.InsertFromStream(s: TStream; Parent: TWinControl; PasteFlags: TComponentPasteSelectionFlags): Boolean; begin Result:=DoInsertFromStream(s,Parent,PasteFlags); end; function TDesigner.DoCopySelectionToClipboard: boolean; var AllComponentsStream: TMemoryStream; AllComponentText: string; begin Result := false; if ControlSelection.Count = 0 then exit; if ControlSelection.OnlyInvisiblePersistentsSelected then exit; AllComponentsStream:=TMemoryStream.Create; try // copy components to stream if not CopySelectionToStream(AllComponentsStream) then exit; SetLength(AllComponentText,AllComponentsStream.Size); if AllComponentText<>'' then begin AllComponentsStream.Position:=0; AllComponentsStream.Read(AllComponentText[1],length(AllComponentText)); end; // copy to clipboard try ClipBoard.AsText:=AllComponentText; except on E: Exception do begin MessageDlg(lisUnableCopyComponentsToClipboard, Format(lisThereWasAnErrorWhileCopyingTheComponentStreamToCli, [#13, E.Message]), mtError,[mbOk],0); exit; end; end; finally AllComponentsStream.Free; end; Result:=true; end; function TDesigner.GetPasteParent: TWinControl; var i: Integer; begin Result:=nil; for i:=0 to ControlSelection.Count-1 do begin if (ControlSelection[i].IsTWinControl) and (csAcceptsControls in TWinControl(ControlSelection[i].Persistent).ControlStyle) and (not ControlSelection[i].ParentInSelection) then begin Result:=TWinControl(ControlSelection[i].Persistent); if GetLookupRootForComponent(Result)<>FLookupRoot then Result:=nil; break; end; end; if (Result=nil) and (FLookupRoot is TWinControl) then Result:=TWinControl(FLookupRoot); end; procedure TDesigner.DoModified; begin if Assigned(OnModified) then OnModified(Self) end; function TDesigner.DoPasteSelectionFromClipboard( PasteFlags: TComponentPasteSelectionFlags): boolean; var AllComponentText: string; CurTextCompStream: TMemoryStream; begin Result:=false; if not CanPaste then exit; // read component stream from clipboard AllComponentText:=ClipBoard.AsText; if AllComponentText='' then exit; CurTextCompStream:=TMemoryStream.Create; try CurTextCompStream.Write(AllComponentText[1],length(AllComponentText)); CurTextCompStream.Position:=0; if not DoInsertFromStream(CurTextCompStream,nil,PasteFlags) then exit; finally CurTextCompStream.Free; end; Result:=true; end; function TDesigner.DoInsertFromStream(s: TStream; PasteParent: TWinControl; PasteFlags: TComponentPasteSelectionFlags): Boolean; var AllComponentText: string; StartPos: Integer; EndPos: Integer; CurTextCompStream: TStream; NewSelection: TControlSelection; l: Integer; procedure FindUniquePosition(AComponent: TComponent); var OverlappedComponent: TComponent; P: TPoint; AControl: TControl; AParent: TWinControl; i: Integer; OverlappedControl: TControl; begin if AComponent is TControl then begin AControl:=TControl(AComponent); AParent:=AControl.Parent; if AParent=nil then exit; P:=Point(AControl.Left,AControl.Top); i:=AParent.ControlCount-1; while i>=0 do begin OverlappedControl:=AParent.Controls[i]; if (OverlappedControl<>AComponent) and (OverlappedControl.Left=P.X) and (OverlappedControl.Top=P.Y) then begin inc(P.X,NonVisualCompWidth); inc(P.Y,NonVisualCompWidth); if (P.X>AParent.ClientWidth-AControl.Width) or (P.Y>AParent.ClientHeight-AControl.Height) then break; i:=AParent.ControlCount-1; end else dec(i); end; P.x:=Max(0,Min(P.x,AParent.ClientWidth-AControl.Width)); P.y:=Max(0,Min(P.y,AParent.ClientHeight-AControl.Height)); AControl.SetBounds(P.x,P.y,AControl.Width,AControl.Height); end else begin P:=GetParentFormRelativeTopLeft(AComponent); repeat OverlappedComponent:=NonVisualComponentAtPos(P.x,P.y); if (OverlappedComponent=nil) then break; inc(P.X,NonVisualCompWidth); inc(P.Y,NonVisualCompWidth); if (P.X+NonVisualCompWidth>Form.ClientWidth) or (P.Y+NonVisualCompWidth>Form.ClientHeight) then break; until false; AComponent.DesignInfo := LeftTopToDesignInfo( SmallInt(Max(0, Min(P.x, Form.ClientWidth - NonVisualCompWidth))), SmallInt(Max(0, Min(P.y, Form.ClientHeight - NonVisualCompWidth)))); end; end; function PasteComponent(TextCompStream: TStream): boolean; var NewComponent: TComponent; begin Result:=false; TextCompStream.Position:=0; if Assigned(FOnPasteComponent) then begin NewComponent:=nil; // create component and add to LookupRoot FOnPasteComponent(Self,FLookupRoot,TextCompStream, PasteParent,NewComponent); if NewComponent=nil then exit; // add new component to new selection NewSelection.Add(NewComponent); // set new nice bounds if cpsfFindUniquePositions in PasteFlags then FindUniquePosition(NewComponent); // finish adding component NotifyPersistentAdded(NewComponent); Modified; end; Result:=true; end; begin Result:=false; //debugln('TDesigner.DoInsertFromStream A'); if (cpsfReplace in PasteFlags) and (not DeleteSelection) then exit; //debugln('TDesigner.DoInsertFromStream B s.Size=',dbgs(s.Size),' S.Position=',dbgs(S.Position)); if PasteParent=nil then PasteParent:=GetPasteParent; NewSelection:=TControlSelection.Create; try Form.DisableAutoSizing; try // read component stream from clipboard if (s.Size<=S.Position) then begin debugln('TDesigner.DoInsertFromStream Stream Empty s.Size=',dbgs(s.Size),' S.Position=',dbgs(S.Position)); exit; end; l:=s.Size-s.Position; SetLength(AllComponentText,l); s.Read(AllComponentText[1],length(AllComponentText)); StartPos:=1; EndPos:=StartPos; // read till 'end' while EndPos<=length(AllComponentText) do begin //debugln('TDesigner.DoInsertFromStream C'); if (AllComponentText[EndPos] in ['e','E']) and (EndPos>1) and (AllComponentText[EndPos-1] in [#10,#13]) and (CompareText(copy(AllComponentText,EndPos,3),'END')=0) and ((EndPos+3>length(AllComponentText)) or (AllComponentText[EndPos+3] in [#10,#13])) then begin inc(EndPos,4); while (EndPos<=length(AllComponentText)) and (AllComponentText[EndPos] in [' ',#10,#13]) do inc(EndPos); // extract text for the current component {$IFDEF VerboseDesigner} DebugLn('TDesigner.DoInsertFromStream=============================='); DebugLn(copy(AllComponentText,StartPos,EndPos-StartPos)); DebugLn('TDesigner.DoInsertFromStream=============================='); {$ENDIF} CurTextCompStream:=TMemoryStream.Create; try CurTextCompStream.Write(AllComponentText[StartPos],EndPos-StartPos); CurTextCompStream.Position:=0; // create component from stream if not PasteComponent(CurTextCompStream) then exit; finally CurTextCompStream.Free; end; StartPos:=EndPos; end else begin inc(EndPos); end; end; finally Form.EnableAutoSizing; end; finally if NewSelection.Count>0 then ControlSelection.Assign(NewSelection); NewSelection.Free; end; Result:=true; end; procedure TDesigner.DoShowAnchorEditor; begin if Assigned(FOnShowAnchorEditor) then FOnShowAnchorEditor(Self); end; procedure TDesigner.DoShowTabOrderEditor; begin if Assigned(FOnShowTabOrderEditor) then FOnShowTabOrderEditor(Self); end; procedure TDesigner.DoShowChangeClassDialog; begin if (ControlSelection.Count=1) and (not ControlSelection.LookupRootSelected) then ShowChangeClassDialog(Self,ControlSelection[0].Persistent); end; procedure TDesigner.DoShowObjectInspector; begin if Assigned(FOnShowObjectInspector) then OnShowObjectInspector(Self); end; procedure TDesigner.DoOrderMoveSelectionToFront; begin if ControlSelection.Count <> 1 then Exit; if not ControlSelection[0].IsTControl then Exit; TControl(ControlSelection[0].Persistent).BringToFront; Modified; end; procedure TDesigner.DoOrderMoveSelectionToBack; begin if ControlSelection.Count <> 1 then Exit; if not ControlSelection[0].IsTControl then Exit; TControl(ControlSelection[0].Persistent).SendToBack; Modified; end; procedure TDesigner.DoOrderForwardSelectionOne; var Control: TControl; Parent: TWinControl; begin if ControlSelection.Count <> 1 then Exit; if not ControlSelection[0].IsTControl then Exit; Control := TControl(ControlSelection[0].Persistent); Parent := Control.Parent; if Parent = nil then Exit; Parent.SetControlIndex(Control, Parent.GetControlIndex(Control) + 1); Modified; end; procedure TDesigner.DoOrderBackSelectionOne; var Control: TControl; Parent: TWinControl; begin if ControlSelection.Count <> 1 then Exit; if not ControlSelection[0].IsTControl then Exit; Control := TControl(ControlSelection[0].Persistent); Parent := Control.Parent; if Parent = nil then Exit; Parent.SetControlIndex(Control, Parent.GetControlIndex(Control) - 1); Modified; end; procedure TDesigner.GiveComponentsNames; var i: Integer; CurComponent: TComponent; begin if LookupRoot=nil then exit; for i:=0 to LookupRoot.ComponentCount-1 do begin CurComponent:=LookupRoot.Components[i]; if CurComponent.Name='' then CurComponent.Name:=UniqueName(CurComponent.ClassName); end; end; procedure TDesigner.NotifyPersistentAdded(APersistent: TPersistent); begin try GiveComponentsNames; GlobalDesignHook.PersistentAdded(APersistent,false); except on E: Exception do MessageDlg('Error:',E.Message,mtError,[mbOk],0); end; end; procedure TDesigner.SelectOnlyThisComponent(AComponent: TComponent); begin ControlSelection.AssignPersistent(AComponent); end; function TDesigner.CopySelection: boolean; begin Result := DoCopySelectionToClipboard; end; function TDesigner.CutSelection: boolean; begin Result := DoCopySelectionToClipboard and DoDeleteSelectedPersistents; end; function TDesigner.CanPaste: Boolean; begin Result:=(Form<>nil) and (FLookupRoot<>nil) and (not (csDestroying in FLookupRoot.ComponentState)); end; function TDesigner.PasteSelection( PasteFlags: TComponentPasteSelectionFlags): boolean; begin Result:=DoPasteSelectionFromClipboard(PasteFlags); end; function TDesigner.DeleteSelection: boolean; begin Result:=DoDeleteSelectedPersistents; end; function TDesigner.InvokeComponentEditor(AComponent: TComponent; MenuIndex: integer): boolean; var CompEditor: TBaseComponentEditor; begin Result:=false; DebugLn('TDesigner.InvokeComponentEditor A ',AComponent.Name,':',AComponent.ClassName); CompEditor:=TheFormEditor.GetComponentEditor(AComponent); if CompEditor=nil then begin DebugLn('TDesigner.InvokeComponentEditor', ' WARNING: no component editor found for ', AComponent.Name,':',AComponent.ClassName); exit; end; DebugLn('TDesigner.InvokeComponentEditor B ',CompEditor.ClassName); try CompEditor.Edit; Result:=true; except on E: Exception do begin DebugLn('TDesigner.InvokeComponentEditor ERROR: ',E.Message); MessageDlg(Format(lisErrorIn, [CompEditor.ClassName]), Format(lisTheComponentEditorOfClassHasCreatedTheError, ['"', CompEditor.ClassName, '"', #13, '"', E.Message, '"']), mtError,[mbOk],0); end; end; try CompEditor.Free; except on E: Exception do begin DebugLn('TDesigner.InvokeComponentEditor ERROR freeing component editor: ',E.Message); end; end; end; procedure TDesigner.DoProcessCommand(Sender: TObject; var Command: word; var Handled: boolean); begin if Assigned(OnProcessCommand) and (Command <> ecNone) then begin OnProcessCommand(Self,Command,Handled); Handled := Handled or (Command = ecNone); end; if Handled then Exit; case Command of ecDesignerSelectParent : SelectParentOfSelection; ecDesignerCopy : CopySelection; ecDesignerCut : CutSelection; ecDesignerPaste : PasteSelection([cpsfFindUniquePositions]); ecDesignerMoveToFront : DoOrderMoveSelectionToFront; ecDesignerMoveToBack : DoOrderMoveSelectionToBack; ecDesignerForwardOne : DoOrderForwardSelectionOne; ecDesignerBackOne : DoOrderBackSelectionOne; else Exit; end; Handled := True; end; function TDesigner.NonVisualComponentLeftTop(AComponent: TComponent): TPoint; var ParentForm: TPoint; begin Result.X := LeftFromDesignInfo(AComponent.DesignInfo); Result.Y := TopFromDesignInfo(AComponent.DesignInfo); // convert to lookuproot coords if (AComponent.Owner <> FLookupRoot) then begin ParentForm:=GetParentFormRelativeClientOrigin(AComponent.Owner); inc(Result.X,ParentForm.X); inc(Result.Y,ParentForm.Y); end; end; procedure TDesigner.InvalidateWithParent(AComponent: TComponent); begin {$IFDEF VerboseDesigner} DebugLn('TDesigner.INVALIDATEWITHPARENT ',AComponent.Name,':',AComponent.ClassName); {$ENDIF} if AComponent is TControl then begin if TControl(AComponent).Parent<>nil then TControl(AComponent).Parent.Invalidate else TControl(AComponent).Invalidate; end else begin FForm.Invalidate; end; end; procedure TDesigner.SetDefaultFormBounds(const AValue: TRect); begin FDefaultFormBounds:=AValue; end; procedure TDesigner.SetGridColor(const AValue: TColor); begin if GridColor=AValue then exit; EnvironmentOptions.GridColor:=AValue; Form.Invalidate; end; procedure TDesigner.SetShowBorderSpacing(const AValue: boolean); begin if ShowBorderSpacing=AValue then exit; EnvironmentOptions.ShowBorderSpacing:=AValue; Form.Invalidate; end; procedure TDesigner.SetShowComponentCaptions(const AValue: boolean); begin if AValue=ShowComponentCaptions then exit; if AValue then Include(FFlags, dfShowComponentCaptions) else Exclude(FFlags, dfShowComponentCaptions); Form.Invalidate; end; function TDesigner.PaintControl(Sender: TControl; TheMessage: TLMPaint): Boolean; var OldDuringPaintControl: boolean; begin Result:=true; {$IFDEF VerboseDsgnPaintMsg} writeln('*** TDesigner.PaintControl A ',Sender.Name,':',Sender.ClassName, ' DC=',DbgS(TheMessage.DC)); {$ENDIF} // Set flag OldDuringPaintControl:=dfDuringPaintControl in FFlags; Include(FFlags,dfDuringPaintControl); // send the Paint message to the control, so that it paints itself //writeln('TDesigner.PaintControl B ',Sender.Name); Sender.Dispatch(TheMessage); //writeln('TDesigner.PaintControl C ',Sender.Name,' DC=',DbgS(TheMessage.DC)); // paint the Designer stuff if TheMessage.DC <> 0 then begin Include(FFlags,dfNeedPainting); if Sender is TWinControl then DDC.SetDC(Form, TWinControl(Sender), TheMessage.DC) else if Sender <> nil then DDC.SetDC(Form, Sender.Parent, TheMessage.DC) else DDC.SetDC(Form, nil, TheMessage.DC); {$IFDEF VerboseDesignerDraw} writeln('TDesigner.PaintControl D ',Sender.Name,':',Sender.ClassName, ' DC=',DbgS(DDC.DC,8), {' FormOrigin=',DDC.FormOrigin.X,',',DDC.FormOrigin.Y,} ' DCOrigin=',DDC.DCOrigin.X,',',DDC.DCOrigin.Y, ' FormClientOrigin=',DDC.FormClientOrigin.X,',',DDC.FormClientOrigin.Y ); {$ENDIF} if LastPaintSender=Sender then begin //writeln('NOTE: TDesigner.PaintControl E control painted twice: ', // Sender.Name,':',Sender.ClassName,' DC=',DbgS(TheMessage.DC)); //RaiseException(''); end; LastPaintSender:=Sender; if IsDesignerDC(Form.Handle, TheMessage.DC) then DoPaintDesignerItems else begin // client grid if (Sender is TWinControl) and (csAcceptsControls in Sender.ControlStyle) then PaintClientGrid(TWinControl(Sender),DDC); if (WidgetSet.GetLCLCapability(lcCanDrawOutsideOnPaint) <> 0) and not EnvironmentOptions.DesignerPaintLazy then DoPaintDesignerItems; end; // clean up DDC.Clear; end; //writeln('TDesigner.PaintControl END ',Sender.Name); if not OldDuringPaintControl then Exclude(FFlags,dfDuringPaintControl); end; function TDesigner.HandleSetCursor(var TheMessage: TLMessage): boolean; begin Result := Lo(TheMessage.LParam) = HTCLIENT; if Result then begin SetTempCursor(Form, LastFormCursor); TheMessage.Result := 1; end; end; procedure TDesigner.HandlePopupMenu(Sender: TControl; var Message: TLMContextMenu); var PopupPos: TPoint; begin if Message.XPos = -1 then begin PopupMenuComponentEditor := GetComponentEditorForSelection; BuildPopupMenu; with ControlSelection do PopupPos := Point(Left + Width, Top); with Form.ClientToScreen(PopupPos) do FDesignerPopupMenu.Popup(X, Y); end; Message.Result := 1; end; procedure TDesigner.GetMouseMsgShift(TheMessage: TLMMouse; var Shift: TShiftState; var Button: TMouseButton); begin Shift := []; if (TheMessage.Keys and MK_Shift) = MK_Shift then Include(Shift, ssShift); if (TheMessage.Keys and MK_Control) = MK_Control then Include(Shift, ssCtrl); if GetKeyState(VK_MENU) < 0 then Include(Shift, ssAlt); if (GetKeyState(VK_LWIN) < 0) or (GetKeyState(VK_RWIN) < 0) then Include(Shift, ssMeta); case TheMessage.Msg of LM_LBUTTONUP,LM_LBUTTONDBLCLK,LM_LBUTTONTRIPLECLK,LM_LBUTTONQUADCLK: begin Include(Shift, ssLeft); Button := mbLeft; end; LM_MBUTTONUP,LM_MBUTTONDBLCLK,LM_MBUTTONTRIPLECLK,LM_MBUTTONQUADCLK: begin Include(Shift, ssMiddle); Button := mbMiddle; end; LM_RBUTTONUP,LM_RBUTTONDBLCLK,LM_RBUTTONTRIPLECLK,LM_RBUTTONQUADCLK: begin Include(Shift, ssRight); Button := mbRight; end; else if (TheMessage.Keys and MK_MButton) <> 0 then begin Include(Shift, ssMiddle); Button := mbMiddle; end; if (TheMessage.Keys and MK_RButton) <> 0 then begin Include(Shift, ssRight); Button := mbRight; end; if (TheMessage.Keys and MK_LButton) <> 0 then begin Include(Shift, ssLeft); Button := mbLeft; end; if (TheMessage.Keys and MK_XBUTTON1) <> 0 then begin Include(Shift, ssExtra1); Button := mbExtra1; end; if (TheMessage.Keys and MK_XBUTTON2) <> 0 then begin Include(Shift, ssExtra2); Button := mbExtra2; end; end; case TheMessage.Msg of LM_LBUTTONDBLCLK,LM_MBUTTONDBLCLK,LM_RBUTTONDBLCLK,LM_XBUTTONDBLCLK: Include(Shift, ssDouble); LM_LBUTTONTRIPLECLK,LM_MBUTTONTRIPLECLK,LM_RBUTTONTRIPLECLK,LM_XBUTTONTRIPLECLK: Include(Shift, ssTriple); LM_LBUTTONQUADCLK,LM_MBUTTONQUADCLK,LM_RBUTTONQUADCLK,LM_XBUTTONQUADCLK: Include(Shift, ssQuad); end; end; function TDesigner.GetDesignControl(AControl: TControl): TControl; // checks if AControl is designable. // if not check Owner. // AControl can be a TNonControlDesignerForm var OwnerControl: TControl; AComponent: TComponent; begin Result:=AControl; if (Result=nil) or (Result=LookupRoot) or (Result.Owner=LookupRoot) then exit; if Result=Form then exit; if (Result.Owner is TControl) then begin OwnerControl:=TControl(Result.Owner); if (not (csOwnedChildrenNotSelectable in OwnerControl.ControlStyle)) then exit; Result:=GetDesignControl(OwnerControl); end else begin AComponent:=GetDesignedComponent(AControl); if AComponent is TControl then Result:=TControl(AComponent) else Result:=nil; end; end; function TDesigner.SizeControl(Sender: TControl; TheMessage: TLMSize): Boolean; begin Result := True; Sender.Dispatch(TheMessage); if ControlSelection.SelectionForm = Form then begin ControlSelection.CheckForLCLChanges(True); end; end; function TDesigner.MoveControl(Sender: TControl; TheMessage: TLMMove): Boolean; begin Result := True; Sender.Dispatch(TheMessage); //debugln('*** TDesigner.MoveControl A ',Sender.Name,':',Sender.ClassName,' ',ControlSelection.SelectionForm=Form,' ',not ControlSelection.IsResizing,' ',ControlSelection.IsSelected(Sender)); if ControlSelection.SelectionForm = Form then begin if not ControlSelection.CheckForLCLChanges(True) and (Sender = Form) and ControlSelection.LookupRootSelected then begin // the selected form was moved (nothing else has changed) // ControlSelection does not need an update, but properties like // Form.Left/Top have to be updated in the OI OnPropertiesChanged(Self); end; end; end; procedure TDesigner.MouseDownOnControl(Sender: TControl; var TheMessage: TLMMouse); var CompIndex:integer; SelectedCompClass: TRegisteredComponent; NonVisualComp: TComponent; ParentForm: TCustomForm; Shift: TShiftState; DesignSender: TControl; Button: TMouseButton; Handled: Boolean; begin FHintTimer.Enabled := False; FHintWindow.Visible := False; Exclude(FFLags, dfHasSized); SetCaptureControl(nil); DesignSender := GetDesignControl(Sender); ParentForm := GetParentForm(DesignSender); //DebugLn(['TDesigner.MouseDownOnControl DesignSender=',dbgsName(DesignSender),' ParentForm=',dbgsName(ParentForm)]); if (ParentForm = nil) then exit; MouseDownPos := GetFormRelativeMousePosition(Form); LastMouseMovePos := MouseDownPos; MouseDownComponent := nil; MouseDownSender := nil; MouseDownComponent := ComponentAtPos(MouseDownPos.X, MouseDownPos.Y, True, True); if (MouseDownComponent = nil) then exit; if ComponentIsIcon(MouseDownComponent) then begin NonVisualComp := MouseDownComponent; MoveNonVisualComponentIntoForm(NonVisualComp); end; MouseDownSender := DesignSender; GetMouseMsgShift(TheMessage,Shift,Button); MouseDownShift:=Shift; {$IFDEF VerboseDesigner} DebugLn('************************************************************'); DbgOut('MouseDownOnControl'); DbgOut(' Sender=',dbgsName(Sender),' DesignSender=',dbgsName(DesignSender)); //write(' Msg=',TheMessage.Pos.X,',',TheMessage.Pos.Y); //write(' Mouse=',MouseDownPos.X,',',MouseDownPos.Y); //writeln(''); if (TheMessage.Keys and MK_Shift) = MK_Shift then DbgOut(' Shift down') else DbgOut(' No Shift down'); if (TheMessage.Keys and MK_Control) = MK_Control then DebugLn(', CTRL down') else DebugLn(', No CTRL down'); {$ENDIF} if (MouseDownComponent <> nil) and (MouseDownComponent is TControl) then begin with TControl(MouseDownComponent).ScreenToClient(Form.ClientToScreen(MouseDownPos)) do if TControl(MouseDownComponent).Perform(CM_DESIGNHITTEST, TheMessage.Keys, Longint(SmallPoint(X, Y))) > 0 then begin TControlAccess(MouseDownComponent).MouseDown(Button, Shift, X, Y); Exit; end; end; if Mediator<>nil then begin Handled:=false; Mediator.MouseDown(Button,Shift,MouseDownPos,Handled); if Handled then exit; end; SelectedCompClass := GetSelectedComponentClass; if Button=mbLeft then begin // left button // -> check if a grabber was activated ControlSelection.ActiveGrabber:= ControlSelection.GrabberAtPos(MouseDownPos.X, MouseDownPos.Y); SetCaptureControl(ParentForm); if SelectedCompClass = nil then begin // selection mode if ControlSelection.ActiveGrabber=nil then begin // no grabber resizing CompIndex:=ControlSelection.IndexOf(MouseDownComponent); if ssCtrl in Shift then begin // child selection end else begin if (ssShift in Shift) then begin // shift key pressed (multiselection) if CompIndex<0 then begin // not selected // add component to selection if (ControlSelection.SelectionForm<>nil) and (ControlSelection.SelectionForm<>Form) then begin MessageDlg(lisInvalidMultiselection, fdInvalidMultiselectionText, mtInformation,[mbOk],0); end else begin ControlSelection.Add(MouseDownComponent); end; end else begin // remove from multiselection ControlSelection.Delete(CompIndex); end; end else begin // no shift key (single selection or keeping multiselection) if (CompIndex<0) then begin // select only this component ControlSelection.AssignPersistent(MouseDownComponent); end else // sync with the interface ControlSelection.UpdateBounds; end; end; end else begin // mouse down on grabber -> begin sizing // grabber is already activated // the sizing is handled in mousemove and mouseup end; end else begin // add component mode -> handled in mousemove and mouseup // but check if we pressed mouse on the form which is not selected if (ControlSelection.SelectionForm <> Form) then ControlSelection.AssignPersistent(MouseDownComponent); end; end else begin // not left button ControlSelection.ActiveGrabber := nil; if (Button = mbRight) and EnvironmentOptions.RightClickSelects and (ControlSelection.SelectionForm <> Form) then ControlSelection.AssignPersistent(MouseDownComponent); end; if not ControlSelection.OnlyVisualComponentsSelected and ShowComponentCaptions then Form.Invalidate; {$IFDEF VerboseDesigner} DebugLn('[TDesigner.MouseDownOnControl] END'); {$ENDIF} end; procedure TDesigner.MouseUpOnControl(Sender : TControl; var TheMessage:TLMMouse); var NewLeft, NewTop, NewWidth, NewHeight: Integer; Button: TMouseButton; Shift: TShiftState; SenderParentForm: TCustomForm; RubberBandWasActive: boolean; ParentClientOrigin, PopupPos: TPoint; SelectedCompClass: TRegisteredComponent; SelectionChanged, NewRubberbandSelection: boolean; DesignSender: TControl; procedure AddComponent; var NewParent: TComponent; NewParentControl: TWinControl; NewComponent: TComponent; NewComponentClass: TComponentClass; NewName: String; DisableAutoSize: Boolean; NewControl: TControl; begin if MouseDownComponent=nil then exit; // add a new component ControlSelection.RubberbandActive:=false; ControlSelection.Clear; NewComponentClass := SelectedCompClass.GetCreationClass; // find a parent for the new component NewParent := FLookupRoot; if Mediator<>nil then begin NewParent:=MouseDownComponent; while (NewParent<>nil) and (not Mediator.ParentAcceptsChild(NewParent,NewComponentClass)) do NewParent:=NewParent.GetParentComponent; if NewParent=nil then NewParent:=FLookupRoot; end else if (FLookupRoot is TCustomForm) or (FLookupRoot is TCustomFrame) then begin if MouseDownComponent is TWinControl then NewParentControl := TWinControl(MouseDownComponent) else NewParentControl := WinControlAtPos(MouseDownPos.X, MouseUpPos.X, true, true); while (NewParentControl <> nil) and ((not (csAcceptsControls in NewParentControl.ControlStyle)) or (NewComponentClass.InheritsFrom(TControl) and not NewParentControl.CheckChildClassAllowed(NewComponentClass, False)) or (csInline in NewParentControl.ComponentState) or // Because of TWriter, you can not put a control onto an csInline control (e.g. on a frame). ((NewParentControl.Owner <> FLookupRoot) and (NewParentControl <> FLookupRoot))) do begin NewParentControl := NewParentControl.Parent; end; NewParent := NewParentControl; end; if not Assigned(NewParent) then exit; if not PropertyEditorHook.BeforeAddPersistent(Self, SelectedCompClass.ComponentClass,NewParent) then begin DebugLn('TDesigner.AddComponent ', SelectedCompClass.ComponentClass.ClassName,' not possible'); exit; end; // calculate initial bounds NewLeft:=Min(MouseDownPos.X,MouseUpPos.X); NewTop:=Min(MouseDownPos.Y,MouseUpPos.Y); if (Mediator<>nil) then begin ParentClientOrigin:=Mediator.GetComponentOriginOnForm(NewParent); DebugLn(['AddComponent ParentClientOrigin=',dbgs(ParentClientOrigin)]); // adjust left,top to parent origin dec(NewLeft,ParentClientOrigin.X); dec(NewTop,ParentClientOrigin.Y); end else if SelectedCompClass.ComponentClass.InheritsFrom(TControl) then begin ParentClientOrigin:=GetParentFormRelativeClientOrigin(NewParent); // adjust left,top to parent origin dec(NewLeft,ParentClientOrigin.X); dec(NewTop,ParentClientOrigin.Y); end; NewWidth:=Abs(MouseUpPos.X-MouseDownPos.X); NewHeight:=Abs(MouseUpPos.Y-MouseDownPos.Y); if Abs(NewWidth+NewHeight)<7 then begin // this very small component is probably only a wag, take default size NewWidth:=0; NewHeight:=0; end; //DebugLn(['AddComponent ',dbgsName(NewComponentClass)]); if NewComponentClass = nil then exit; // check circles if TheFormEditor.ClassDependsOnComponent(NewComponentClass, LookupRoot) then begin IDEMessageDialog(lisInvalidCircle, Format(lisIsAThisCircleDependencyIsNotAllowed, [dbgsName(LookupRoot), dbgsName(NewComponentClass), #13]), mtError,[mbOk],''); exit; end; // create component and component interface DebugLn(['AddComponent ',DbgSName(NewComponentClass),' Parent=',DbgSName(NewParent),' ',NewLeft,',',NewTop,',',NewWidth,',',NewHeight]); DisableAutoSize:=true; NewComponent := TheFormEditor.CreateComponent( NewParent,NewComponentClass,'', NewLeft,NewTop,NewWidth,NewHeight,DisableAutoSize); if NewComponent=nil then exit; if DisableAutoSize and (NewComponent is TControl) then TControl(NewComponent).EnableAutoSizing; TheFormEditor.FixupReferences(NewComponent); // e.g. frame references a datamodule // modified Modified; // set initial properties if NewComponent is TControl then begin NewControl:=TControl(NewComponent); //debugln(['AddComponent ',DbgSName(Self),' Bounds=',dbgs(NewControl.BoundsRect),' BaseBounds=',dbgs(NewControl.BaseBounds),' BaseParentClientSize=',dbgs(NewControl.BaseParentClientSize)]); NewControl.Visible:=true; if csSetCaption in NewControl.ControlStyle then NewControl.Caption:=NewComponent.Name; end; if Assigned(FOnSetDesigning) then FOnSetDesigning(Self,NewComponent,True); if EnvironmentOptions.CreateComponentFocusNameProperty then begin // ask user for name NewName:=NewComponent.Name; ShowComponentNameDialog(LookupRoot,NewComponent,NewName); NewComponent.Name:=NewName; end; // tell IDE about the new component (e.g. add it to the source) NotifyPersistentAdded(NewComponent); // creation completed // -> select new component SelectOnlyThisComponent(NewComponent); if Assigned(FOnComponentAdded) then // this resets the component palette to the selection tool FOnComponentAdded(Self); {$IFDEF VerboseDesigner} DebugLn('NEW COMPONENT ADDED: Form.ComponentCount=',DbgS(Form.ComponentCount), ' NewComponent.Owner.Name=',NewComponent.Owner.Name); {$ENDIF} end; procedure RubberbandSelect; var MaxParentComponent: TComponent; begin if (ssShift in Shift) and (ControlSelection.SelectionForm<>nil) and (ControlSelection.SelectionForm<>Form) then begin MessageDlg(lisInvalidMultiselection, fdInvalidMultiselectionText, mtInformation,[mbOk],0); exit; end; ControlSelection.BeginUpdate; // check if start new selection or add/remove: NewRubberbandSelection:= (not (ssShift in Shift)) or (ControlSelection.SelectionForm<>Form); // update non visual components MoveNonVisualComponentsIntoForm; // if user press the Control key, then component candidates are only // children of the control, where the mouse started if (ssCtrl in shift) then begin if MouseDownComponent=Form then MaxParentComponent:=FLookupRoot else MaxParentComponent:=MouseDownComponent; end else MaxParentComponent:=FLookupRoot; SelectionChanged:=false; ControlSelection.SelectWithRubberBand( FLookupRoot,Mediator,NewRubberbandSelection,ssShift in Shift, SelectionChanged,MaxParentComponent); if ControlSelection.Count=0 then begin ControlSelection.Add(FLookupRoot); SelectionChanged:=true; end; ControlSelection.RubberbandActive:=false; ControlSelection.EndUpdate; {$IFDEF VerboseDesigner} DebugLn('RubberbandSelect ',DbgS(ControlSelection.Grabbers[0])); {$ENDIF} Form.Invalidate; end; procedure PointSelect; begin if not (ssShift in Shift) then begin // select only the mouse down component ControlSelection.AssignPersistent(MouseDownComponent); if (ssDouble in MouseDownShift) and (ControlSelection.SelectionForm = Form) then begin // Double Click -> invoke 'Edit' of the component editor FShiftState := Shift; InvokeComponentEditor(MouseDownComponent, -1); FShiftState := []; end; end; end; procedure DisableRubberBand; begin if ControlSelection.RubberbandActive then ControlSelection.RubberbandActive := False; end; var Handled: Boolean; begin FHintTimer.Enabled := False; FHintWindow.Visible := False; SetCaptureControl(nil); // check if the message is for the designed form // and there was a mouse down before DesignSender:=GetDesignControl(Sender); SenderParentForm:=GetParentForm(DesignSender); //DebugLn(['TDesigner.MouseUpOnControl DesignSender=',dbgsName(DesignSender),' SenderParentForm=',dbgsName(SenderParentForm),' ',TheMessage.XPos,',',TheMessage.YPos]); if (MouseDownComponent=nil) or (SenderParentForm=nil) or (SenderParentForm<>Form) or ((ControlSelection.SelectionForm<>nil) and (ControlSelection.SelectionForm<>Form)) then begin MouseDownComponent:=nil; MouseDownSender:=nil; exit; end; ControlSelection.ActiveGrabber:=nil; RubberBandWasActive:=ControlSelection.RubberBandActive; SelectedCompClass:=GetSelectedComponentClass; GetMouseMsgShift(TheMessage,Shift,Button); MouseUpPos:=GetFormRelativeMousePosition(Form); {$IFDEF VerboseDesigner} DebugLn('************************************************************'); DbgOut('MouseUpOnControl'); DbgOut(' Sender=',dbgsName(Sender),' DesignSender=',dbgsName(DesignSender)); //write(' Msg=',TheMessage.Pos.X,',',TheMessage.Pos.Y); DebugLn(''); {$ENDIF} if (MouseDownComponent <> nil) and (MouseDownComponent is TControl) then begin with TControl(MouseDownComponent).ScreenToClient(Form.ClientToScreen(MouseUpPos)) do if TControl(MouseDownComponent).Perform(CM_DESIGNHITTEST, TheMessage.Keys, Longint(SmallPoint(X, Y))) > 0 then begin TControlAccess(MouseDownComponent).MouseUp(Button, Shift, X, Y); Exit; end; end; if Mediator<>nil then begin Handled:=false; Mediator.MouseUp(Button,Shift,MouseUpPos,Handled); if Handled then exit; end; if Button=mbLeft then begin if SelectedCompClass = nil then begin // layout mode (selection, moving and resizing) if not (dfHasSized in FFlags) then begin // new selection if RubberBandWasActive then begin // rubberband selection RubberbandSelect; end else begin // point selection PointSelect; end; end else ControlSelection.UpdateBounds; end else begin // create new a component on the form AddComponent; end; end else if Button=mbRight then begin // right click -> popup menu DisableRubberBand; if EnvironmentOptions.RightClickSelects and (not ControlSelection.IsSelected(MouseDownComponent)) and (Shift - [ssRight] = []) then PointSelect; PopupMenuComponentEditor := GetComponentEditorForSelection; BuildPopupMenu; PopupPos := Form.ClientToScreen(MouseUpPos); FDesignerPopupMenu.Popup(PopupPos.X, PopupPos.Y); end; DisableRubberBand; LastMouseMovePos.X:=-1; if (not ControlSelection.OnlyVisualComponentsSelected and ShowComponentCaptions) or (dfHasSized in FFlags) then Form.Invalidate; Exclude(FFlags,dfHasSized); MouseDownComponent:=nil; MouseDownSender:=nil; {$IFDEF VerboseDesigner} DebugLn('[TDesigner.MouseLeftUpOnControl] END'); {$ENDIF} end; procedure TDesigner.MouseMoveOnControl(Sender: TControl; var TheMessage: TLMMouse); var Button: TMouseButton; Shift : TShiftState; SenderParentForm:TCustomForm; OldMouseMovePos: TPoint; Grabber: TGrabber; ACursor: TCursor; SelectedCompClass: TRegisteredComponent; CurSnappedMousePos, OldSnappedMousePos: TPoint; DesignSender: TControl; Handled: Boolean; MouseMoveComponent: TComponent; begin GetMouseMsgShift(TheMessage, Shift, Button); if [dfShowEditorHints] * FFlags <> [] then begin FHintTimer.Enabled := False; // hide hint FHintTimer.Enabled := Shift * [ssLeft, ssRight, ssMiddle] = []; if not (dfHasSized in FFlags) then FHintWindow.Visible := False; end; DesignSender := GetDesignControl(Sender); //DebugLn('TDesigner.MouseMoveOnControl Sender=',dbgsName(Sender),' ',dbgsName(DesignSender)); SenderParentForm := GetParentForm(DesignSender); if (SenderParentForm = nil) or (SenderParentForm <> Form) then Exit; OldMouseMovePos := LastMouseMovePos; LastMouseMovePos := GetFormRelativeMousePosition(Form); if (OldMouseMovePos.X = LastMouseMovePos.X) and (OldMouseMovePos.Y = LastMouseMovePos.Y) then Exit; MouseMoveComponent := MouseDownComponent; if MouseMoveComponent = nil then MouseMoveComponent := ComponentAtPos(LastMouseMovePos.X, LastMouseMovePos.Y, True, True); if (MouseMoveComponent <> nil) and (MouseMoveComponent is TControl) then begin with TControl(MouseMoveComponent).ScreenToClient(Form.ClientToScreen(LastMouseMovePos)) do if TControl(MouseMoveComponent).Perform(CM_DESIGNHITTEST, TheMessage.Keys, Longint(SmallPoint(X, Y))) > 0 then begin TControlAccess(MouseMoveComponent).MouseMove(Shift, X, Y); Exit; end; end; if Mediator <> nil then begin Handled := False; Mediator.MouseMove(Shift, LastMouseMovePos, Handled); if Handled then Exit; end; if ControlSelection.SelectionForm = Form then Grabber := ControlSelection.GrabberAtPos(LastMouseMovePos.X, LastMouseMovePos.Y) else Grabber := nil; if MouseDownComponent = nil then begin if Grabber = nil then ACursor := crDefault else ACursor := Grabber.Cursor; if ACursor <> LastFormCursor then begin LastFormCursor := ACursor; SetTempCursor(Form, ACursor); end; Exit; end; if (ControlSelection.SelectionForm = nil) or (ControlSelection.SelectionForm = Form) then begin if Button = mbLeft then // left button pressed begin if (ControlSelection.ActiveGrabber <> nil) then // grabber active => resizing begin // grabber moving -> size selection if not ControlSelection.LookupRootSelected then // if not current form is selected then resize selection begin if not (dfHasSized in FFlags) then begin ControlSelection.SaveBounds; Include(FFlags, dfHasSized); end; // skip snapping when Alt is pressed if not (ssAlt in Shift) then begin OldSnappedMousePos := ControlSelection.SnapGrabberMousePos(OldMouseMovePos); CurSnappedMousePos := ControlSelection.SnapGrabberMousePos(LastMouseMovePos); end else begin OldSnappedMousePos := OldMouseMovePos; CurSnappedMousePos := LastMouseMovePos; end; ControlSelection.SizeSelection( CurSnappedMousePos.X - OldSnappedMousePos.X, CurSnappedMousePos.Y - OldSnappedMousePos.Y); DoModified; end; end else begin // no grabber active => moving SelectedCompClass := GetSelectedComponentClass; if (not ControlSelection.RubberBandActive) and (SelectedCompClass=nil) and ((Shift=[ssLeft]) or (Shift=[ssAlt, ssLeft])) and (ControlSelection.Count>=1) and (not ControlSelection.LookupRootSelected) then begin // move selection if not (dfHasSized in FFlags) then begin ControlSelection.SaveBounds; Include(FFlags, dfHasSized); end; //debugln('TDesigner.MouseMoveOnControl Move MouseDownComponent=',dbgsName(MouseDownComponent),' OldMouseMovePos=',dbgs(OldMouseMovePos),' MouseMovePos',dbgs(LastMouseMovePos),' MouseDownPos=',dbgs(MouseDownPos)); if ((ssAlt in Shift) and ControlSelection.MoveSelection(LastMouseMovePos.X - MouseDownPos.X, LastMouseMovePos.Y - MouseDownPos.Y, True)) or ControlSelection.MoveSelectionWithSnapping(LastMouseMovePos.X - MouseDownPos.X, LastMouseMovePos.Y - MouseDownPos.Y) then DoModified; end else begin // rubberband sizing (selection or creation) ControlSelection.RubberBandBounds := Rect(MouseDownPos.X, MouseDownPos.Y, LastMouseMovePos.X, LastMouseMovePos.Y); if SelectedCompClass = nil then ControlSelection.RubberbandType := rbtSelection else ControlSelection.RubberbandType := rbtCreating; ControlSelection.RubberBandActive := True; end; end; end else ControlSelection.ActiveGrabber:=nil; end; if [dfShowEditorHints, dfHasSized] * FFlags = [dfShowEditorHints, dfHasSized] then HintTimer(Self); end; { -----------------------------K E Y D O W N ------------------------------- } { Handles the keydown messages. DEL deletes the selected controls, CTRL-ARROR moves the selection up one, SHIFT-ARROW resizes, etc. } Procedure TDesigner.KeyDown(Sender : TControl; var TheMessage: TLMKEY); var Shift: TShiftState; Command: word; Handled: boolean; procedure Nudge(x, y: integer); begin if (ssCtrl in Shift) then begin if ssShift in Shift then begin x := x * GetGridSizeX; y := y * GetGridSizeY; end; NudgePosition(x, y) end else if (ssShift in Shift) then NudgeSize(x, y) else if (Shift = []) then NudgeSelection(x, y); end; begin {$IFDEF VerboseDesigner} DebugLn(['TDesigner.KEYDOWN ',TheMessage.CharCode,' ',TheMessage.KeyData]); {$ENDIF} Shift := KeyDataToShiftState(TheMessage.KeyData); Handled := False; if Mediator<>nil then Mediator.KeyDown(Sender,TheMessage.CharCode,Shift); Command := FTheFormEditor.TranslateKeyToDesignerCommand( TheMessage.CharCode, Shift); //DebugLn(['TDesigner.KEYDOWN Command=',dbgs(Command),' ',TheMessage.CharCode,' ',dbgs(Shift)]); DoProcessCommand(Self, Command, Handled); //DebugLn(['TDesigner.KeyDown Command=',Command,' Handled=',Handled,' TheMessage.CharCode=',TheMessage.CharCode]); if not Handled then begin Handled := True; case TheMessage.CharCode of VK_DELETE: if not ControlSelection.OnlyInvisiblePersistentsSelected then DoDeleteSelectedPersistents; VK_UP: Nudge(0,-1); VK_DOWN: Nudge(0,1); VK_RIGHT: Nudge(1,0); VK_LEFT: Nudge(-1,0); VK_TAB: if Shift = [ssShift] then NudgeSelection(False) else if Shift = [] then NudgeSelection(True) else Handled := False; VK_RETURN: if Shift = [] then DoShowObjectInspector else Handled := False; VK_A: if Shift = [ssCtrl] then DoSelectAll else Handled := False; else Handled := False; end; end; if Handled then TheMessage.CharCode := 0; end; {------------------------------------K E Y U P --------------------------------} Procedure TDesigner.KeyUp(Sender : TControl; var TheMessage: TLMKEY); var Shift: TShiftState; Begin {$IFDEF VerboseDesigner} //Writeln('TDesigner.KEYUP ',TheMessage.CharCode,' ',TheMessage.KeyData); {$ENDIF} if Mediator<>nil then begin Shift := KeyDataToShiftState(TheMessage.KeyData); Mediator.KeyUp(Sender,TheMessage.CharCode,Shift); end; end; function TDesigner.DoDeleteSelectedPersistents: boolean; var i: integer; APersistent: TPersistent; AncestorRoot: TComponent; AComponent: TComponent; begin Result:=true; if (ControlSelection.Count=0) or (ControlSelection.SelectionForm<>Form) then exit; Result:=false; // check if a component is the lookup root (can not be deleted) if (ControlSelection.LookupRootSelected) then begin if ControlSelection.Count>1 then MessageDlg(lisInvalidDelete, lisTheRootComponentCanNotBeDeleted, mtInformation, [mbOk],0); exit; end; // check if a selected component is inherited (can not be deleted) for i:=0 to ControlSelection.Count-1 do begin if not ControlSelection[i].IsTComponent then continue; AncestorRoot:=TheFormEditor.GetAncestorLookupRoot( TComponent(ControlSelection[i].Persistent)); if AncestorRoot<>nil then begin MessageDlg(lisInvalidDelete, Format(lisTheComponentIsInheritedFromToDeleteAnInheritedComp, [dbgsName( ControlSelection[i].Persistent), dbgsName(AncestorRoot), #13]), mtInformation, [mbOk],0); exit; end; end; // check if a selected component is not owned by lookuproot (can not be deleted) for i:=0 to ControlSelection.Count-1 do begin if not ControlSelection[i].IsTComponent then continue; AComponent:=TComponent(ControlSelection[i].Persistent); if AComponent.Owner<>FLookupRoot then begin MessageDlg(lisInvalidDelete, Format(lisTheComponentCanNotBeDeletedBecauseItIsNotOwnedBy, [dbgsName( ControlSelection[i].Persistent), dbgsName(FLookupRoot)]), mtInformation, [mbOk],0); exit; end; end; // mark selected components for deletion for i:=0 to ControlSelection.Count-1 do MarkPersistentForDeletion(ControlSelection[i].Persistent); // clear selection by selecting the LookupRoot SelectOnlyThisComponent(FLookupRoot); // delete marked components Include(FFlags,dfDeleting); try if DeletingPersistent.Count=0 then exit; while DeletingPersistent.Count>0 do begin APersistent:=TPersistent(DeletingPersistent[DeletingPersistent.Count-1]); //debugln(['TDesigner.DoDeleteSelectedComponents A ',dbgsName(APersistent),' ',(APersistent is TComponent) and (TheFormEditor.FindComponent(TComponent(APersistent))<>nil)]); RemovePersistentAndChilds(APersistent); //writeln('TDesigner.DoDeleteSelectedComponents B ',DeletingPersistent.IndexOf(AComponent)); end; IgnoreDeletingPersistent.Clear; finally Exclude(FFlags,dfDeleting); Modified; end; Result:=true; end; procedure TDesigner.DoSelectAll; begin ControlSelection.BeginUpdate; ControlSelection.Clear; ControlSelection.SelectAll(FLookupRoot); ControlSelection.EndUpdate; Form.Invalidate; end; procedure TDesigner.DoDeletePersistent(APersistent: TPersistent; FreeIt: boolean); var Hook: TPropertyEditorHook; AComponent: TComponent; AForm: TCustomForm; begin if APersistent=nil then exit; try //debugln(['TDesigner.DoDeletePersistent A ',dbgsName(APersistent),' FreeIt=',FreeIt]); PopupMenuComponentEditor:=nil; // unselect component ControlSelection.Remove(APersistent); if (APersistent is TComponent) then begin AComponent:=TComponent(APersistent); if csDestroying in AComponent.ComponentState then FreeIt:=false; end; AForm:=GetDesignerForm(APersistent); if AForm=nil then begin // has no designer // -> do not call handlers and simply get rid of the rubbish if FreeIt then begin //debugln('TDesigner.DoDeletePersistent UNKNOWN in formeditor: ',dbgsName(APersistent)); APersistent.Free; end; exit; end; // call component deleting handlers Hook:=GetPropertyEditorHook; if Hook<>nil then Hook.PersistentDeleting(APersistent); // delete component if APersistent is TComponent then TheFormEditor.DeleteComponent(TComponent(APersistent),FreeIt) else if FreeIt then APersistent.Free; finally // unmark component DeletingPersistent.Remove(APersistent); IgnoreDeletingPersistent.Remove(APersistent); end; // call ComponentDeleted handler if Assigned(FOnPersistentDeleted) then FOnPersistentDeleted(Self,APersistent); end; procedure TDesigner.MarkPersistentForDeletion(APersistent: TPersistent); begin if (not PersistentIsMarkedForDeletion(APersistent)) then DeletingPersistent.Add(APersistent); end; function TDesigner.PersistentIsMarkedForDeletion(APersistent: TPersistent ): boolean; begin Result:=(DeletingPersistent.IndexOf(APersistent)>=0); end; function TDesigner.GetSelectedComponentClass: TRegisteredComponent; begin Result:=nil; if Assigned(FOnGetSelectedComponentClass) then FOnGetSelectedComponentClass(Self,Result); end; function TDesigner.IsDesignMsg(Sender: TControl; var TheMessage: TLMessage): Boolean; begin Result := false; if csDesigning in Sender.ComponentState then begin Result:=true; case TheMessage.Msg of LM_PAINT: Result := PaintControl(Sender, TLMPaint(TheMessage)); CN_KEYDOWN,CN_SYSKEYDOWN: KeyDown(Sender,TLMKey(TheMessage)); CN_KEYUP,CN_SYSKEYUP: KeyUP(Sender,TLMKey(TheMessage)); LM_LBUTTONDOWN, LM_RBUTTONDOWN, LM_LBUTTONDBLCLK: MouseDownOnControl(Sender,TLMMouse(TheMessage)); LM_LBUTTONUP, LM_RBUTTONUP: MouseUpOnControl(Sender, TLMMouse(TheMessage)); LM_MOUSEMOVE: MouseMoveOnControl(Sender, TLMMouse(TheMessage)); LM_SIZE: Result:=SizeControl(Sender, TLMSize(TheMessage)); LM_MOVE: Result:=MoveControl(Sender, TLMMove(TheMessage)); LM_ACTIVATE: Result:=OnFormActivated; LM_CLOSEQUERY: Result:=OnFormCloseQuery; LM_SETCURSOR: Result:=HandleSetCursor(TheMessage); LM_CONTEXTMENU: HandlePopupMenu(Sender, TLMContextMenu(TheMessage)); else Result:=false; end; end else begin if (TheMessage.Msg=LM_PAINT) or (TheMessage.Msg=CN_KEYDOWN) or (TheMessage.Msg=CN_KEYUP) or (TheMessage.Msg=LM_LBUTTONDOWN) or (TheMessage.Msg=LM_RBUTTONDOWN) or (TheMessage.Msg=LM_LBUTTONDBLCLK) or (TheMessage.Msg=LM_LBUTTONUP) or (TheMessage.Msg=LM_RBUTTONUP) or (TheMessage.Msg=LM_MOUSEMOVE) or (TheMessage.Msg=LM_SIZE) or (TheMessage.Msg=LM_MOVE) or (TheMessage.Msg=LM_ACTIVATE) or (TheMessage.Msg=LM_CLOSEQUERY) or (TheMessage.Msg=LM_SETCURSOR) then DebugLn(['TDesigner.IsDesignMsg NOT DESIGNING? ',dbgsName(Sender),' TheMessage.Msg=',GetMessageName(TheMessage.Msg)]); end; end; function TDesigner.UniqueName(const BaseName: string): string; begin Result:=TheFormEditor.CreateUniqueComponentName(BaseName,LookupRoot); end; procedure TDesigner.Modified; Begin ControlSelection.SaveBounds; DoModified; inherited Modified; end; Procedure TDesigner.RemovePersistentAndChilds(APersistent: TPersistent); var i: integer; AWinControl: TWinControl; ChildControl: TControl; Begin if APersistent=nil then exit; {$IFDEF VerboseDesigner} DebugLn('[TDesigner.RemovePersistentAndChilds] START ',dbgsName(APersistent),' ',DbgS(APersistent)); {$ENDIF} if (APersistent=FLookupRoot) or (APersistent=Form) or (IgnoreDeletingPersistent.IndexOf(APersistent)>=0) then exit; // remove all child controls owned by the LookupRoot if (APersistent is TWinControl) then begin AWinControl:=TWinControl(APersistent); i:=AWinControl.ControlCount-1; while (i>=0) do begin ChildControl:=AWinControl.Controls[i]; // if (GetLookupRootForComponent(ChildControl)=FLookupRoot) if (ChildControl.Owner=FLookupRoot) and (IgnoreDeletingPersistent.IndexOf(ChildControl)<0) then begin //Debugln(['[TDesigner.RemoveComponentAndChilds] B ',dbgsName(APersistent),' Child=',dbgsName(ChildControl),' i=',i,' ',TheFormEditor.FindComponent(ChildControl)<>nil]); RemovePersistentAndChilds(ChildControl); // the component list of the form has changed // -> restart the search i:=AWinControl.ControlCount-1; end else dec(i); end; end; // remove component {$IFDEF VerboseDesigner} DebugLn('[TDesigner.RemovePersistentAndChilds] DoDeletePersistent ',dbgsName(APersistent)); {$ENDIF} DoDeletePersistent(APersistent,true); end; procedure TDesigner.Notification(AComponent: TComponent; Operation: TOperation); begin if Operation = opInsert then begin {$IFDEF VerboseDesigner} DebugLn('opInsert ',dbgsName(AComponent),' ',DbgS(AComponent)); {$ENDIF} if dfDeleting in FFlags then begin // a component has auto created a new component during deletion // -> ignore the new component IgnoreDeletingPersistent.Add(AComponent); end; end else if Operation = opRemove then begin {$IFDEF VerboseDesigner} DebugLn('[TDesigner.Notification] opRemove ',dbgsName(AComponent)); {$ENDIF} DoDeletePersistent(AComponent,false); end; end; procedure TDesigner.PaintGrid; begin // This is normally done in PaintControls if FLookupRoot<>FForm then begin // this is a special designer form -> lets draw itself TCustomFormAccess(FForm).Paint; end; end; procedure TDesigner.PaintClientGrid(AWinControl: TWinControl; aDDC: TDesignerDeviceContext); var Clip: integer; Count: integer; i: integer; CurControl: TControl; begin if (AWinControl=nil) or (not (csAcceptsControls in AWinControl.ControlStyle)) or ((not ShowGrid) and (not ShowBorderSpacing)) then exit; aDDC.BeginPainting; try // exclude all child control areas Count:=AWinControl.ControlCount; for i := 0 to Count - 1 do begin with AWinControl.Controls[I] do begin if (Visible or ((csDesigning in ComponentState) and not (csNoDesignVisible in ControlStyle))) then begin Clip := ExcludeClipRect(aDDC.DC, Left, Top, Left + Width, Top + Height); if Clip = NullRegion then exit; end; end; end; // paint points if ShowGrid then begin ADDC.Canvas.Pen.Color := GridColor; ADDC.Canvas.Pen.Width := 1; ADDC.Canvas.Pen.Style := psSolid; DrawGrid(ADDC.Canvas.Handle, AWinControl.ClientRect, GridSizeX, GridSizeY); end; if ShowBorderSpacing then begin aDDC.Canvas.Brush.Color := clRed; for i := 0 to Count - 1 do begin CurControl := AWinControl.Controls[i]; if csNoDesignSelectable in CurControl.ControlStyle then Continue; aDDC.Canvas.FrameRect( CurControl.Left-CurControl.BorderSpacing.GetSpace(akLeft), CurControl.Top-CurControl.BorderSpacing.GetSpace(akTop), CurControl.Left+CurControl.Width+CurControl.BorderSpacing.GetSpace(akRight)-1, CurControl.Top+CurControl.Height+CurControl.BorderSpacing.GetSpace(akBottom)-1 ); end; end; finally aDDC.EndPainting; end; end; procedure TDesigner.ValidateRename(AComponent: TComponent; const CurName, NewName: string); begin // check if component is initialized if (CurName='') or (NewName='') or ((AComponent<>nil) and (csDestroying in AComponent.ComponentState)) then exit; // check if component is the LookupRoot if AComponent=nil then AComponent:=FLookupRoot; // consistency check if CurName<>AComponent.Name then DebugLn('WARNING: TDesigner.ValidateRename: OldComponentName="',CurName,'" <> AComponent=',dbgsName(AComponent)); if Assigned(OnRenameComponent) then OnRenameComponent(Self,AComponent,NewName); end; function TDesigner.GetShiftState: TShiftState; begin Result:=FShiftState; end; function TDesigner.CreateUniqueComponentName(const AClassName: string): string; begin Result:=TheFormEditor.CreateUniqueComponentName(AClassName,FLookupRoot); end; procedure TDesigner.OnComponentEditorVerbMenuItemClick(Sender: TObject); var Verb: integer; VerbCaption: string; AMenuItem: TMenuItem; begin if (PopupMenuComponentEditor=nil) or (Sender=nil) then exit; //DebugLn(['TDesigner.OnComponentEditorVerbMenuItemClick Sender=',dbgsName(Sender)]); if Sender is TMenuItem then AMenuItem:=TMenuItem(Sender) else if Sender is TIDEMenuCommand then AMenuItem:=TIDEMenuCommand(Sender).MenuItem else exit; Verb:=PopupMenuComponentEditor.GetVerbCount-1; VerbCaption:=AMenuItem.Caption; while (Verb>=0) and (VerbCaption<>PopupMenuComponentEditor.GetVerb(Verb)) do dec(Verb); if Verb<0 then exit; try PopupMenuComponentEditor.ExecuteVerb(Verb); except on E: Exception do begin DebugLn('TDesigner.OnComponentEditorVerbMenuItemClick ERROR: ',E.Message); MessageDlg(Format(lisErrorIn, [PopupMenuComponentEditor.ClassName]), Format(lisTheComponentEditorOfClassInvokedWithVerbHasCreated, ['"', PopupMenuComponentEditor.ClassName, '"', #13, IntToStr(Verb), '"', VerbCaption, '"', #13, #13, '"', E.Message, '"']), mtError,[mbOk],0); end; end; end; procedure TDesigner.OnDeleteSelectionMenuClick(Sender: TObject); begin DoDeleteSelectedPersistents; end; procedure TDesigner.OnSelectAllMenuClick(Sender: TObject); begin DoSelectAll; end; procedure TDesigner.OnChangeClassMenuClick(Sender: TObject); begin DoShowChangeClassDialog; end; procedure TDesigner.OnChangeParentMenuClick(Sender: TObject); var Item: TIDEMenuCommand; NewParentName: String; i: Integer; CurControl: TControl; NewParent: TWinControl; begin if not (Sender is TIDEMenuCommand) then Exit; Item := TIDEMenuCommand(Sender); NewParentName := Item.Caption; if SysUtils.CompareText(LookupRoot.Name, NewParentName) = 0 then NewParent := TWinControl(LookupRoot) else NewParent := TWinControl(LookupRoot.FindComponent(NewParentName)); if (NewParent=nil) or (not (NewParent is TWinControl)) then Exit; Form.DisableAlign; try i := ControlSelection.Count - 1; while (i >= 0) do begin if i < ControlSelection.Count then begin if ControlSelection[i].IsTControl then begin CurControl := TControl(ControlSelection[i].Persistent); CurControl.Parent := NewParent; end; end; dec(i); end; finally if Form <> nil then Form.EnableAlign; ControlSelection.DoChange(True); // request updates since control hierarchi change end; end; procedure TDesigner.OnSnapToGridOptionMenuClick(Sender: TObject); begin EnvironmentOptions.SnapToGrid := not EnvironmentOptions.SnapToGrid; end; procedure TDesigner.OnShowOptionsMenuItemClick(Sender: TObject); begin if Assigned(OnShowOptions) then OnShowOptions(Self); end; procedure TDesigner.OnSnapToGuideLinesOptionMenuClick(Sender: TObject); begin EnvironmentOptions.SnapToGuideLines := not EnvironmentOptions.SnapToGuideLines; end; procedure TDesigner.OnViewLFMMenuClick(Sender: TObject); begin if Assigned(OnViewLFM) then OnViewLFM(Self); end; procedure TDesigner.OnSaveAsXMLMenuClick(Sender: TObject); begin if Assigned(OnSaveAsXML) then OnSaveAsXML(Self); end; procedure TDesigner.OnCenterFormMenuClick(Sender: TObject); var NewLeft: Integer; NewTop: Integer; begin if Form=nil then exit; NewLeft:=Max(30,(Screen.Width-Form.Width) div 2); NewTop:=Max(30,(Screen.Height-Form.Height) div 2); Form.SetBounds(NewLeft,NewTop,Form.Width,Form.Height); end; procedure TDesigner.OnCopyMenuClick(Sender: TObject); begin CopySelection; end; procedure TDesigner.OnCutMenuClick(Sender: TObject); begin CutSelection; end; procedure TDesigner.OnPasteMenuClick(Sender: TObject); begin PasteSelection([cpsfFindUniquePositions]); end; procedure TDesigner.OnAnchorEditorMenuClick(Sender: TObject); begin DoShowAnchorEditor; end; procedure TDesigner.OnTabOrderMenuClick(Sender: TObject); begin DoShowTabOrderEditor; end; function TDesigner.GetGridColor: TColor; begin Result:=EnvironmentOptions.GridColor; end; function TDesigner.GetShowBorderSpacing: boolean; begin Result:=EnvironmentOptions.ShowBorderSpacing; end; function TDesigner.GetShowComponentCaptions: boolean; begin Result:=dfShowComponentCaptions in FFlags; end; function TDesigner.GetShowGrid: boolean; begin Result:=EnvironmentOptions.ShowGrid; end; function TDesigner.GetGridSizeX: integer; begin Result:=EnvironmentOptions.GridSizeX; if Result<2 then Result:=2; end; function TDesigner.GetGridSizeY: integer; begin Result:=EnvironmentOptions.GridSizeY; if Result<2 then Result:=2; end; function TDesigner.GetIsControl: Boolean; Begin Result := True; end; function TDesigner.GetShowEditorHints: boolean; begin Result:=dfShowEditorHints in FFlags; end; function TDesigner.GetSnapToGrid: boolean; begin Result := EnvironmentOptions.SnapToGrid; end; procedure TDesigner.SetShowGrid(const AValue: boolean); begin if ShowGrid=AValue then exit; EnvironmentOptions.ShowGrid:=AValue; Form.Invalidate; end; procedure TDesigner.SetGridSizeX(const AValue: integer); begin if GridSizeX=AValue then exit; EnvironmentOptions.GridSizeX:=AValue; end; procedure TDesigner.SetGridSizeY(const AValue: integer); begin if GridSizeY=AValue then exit; EnvironmentOptions.GridSizeY:=AValue; end; procedure TDesigner.SetIsControl(Value: Boolean); begin end; procedure TDesigner.SetMediator(const AValue: TDesignerMediator); begin if Mediator=AValue then exit; if Mediator<>nil then Mediator.Designer:=nil; FMediator:=AValue; if Mediator<>nil then Mediator.Designer:=Self; end; procedure TDesigner.SetPopupMenuComponentEditor(const AValue: TBaseComponentEditor); begin if FPopupMenuComponentEditor <> AValue then begin FPopupMenuComponentEditor.Free; FPopupMenuComponentEditor := AValue; end; end; procedure TDesigner.SetShowEditorHints(const AValue: boolean); begin if AValue = ShowEditorHints then Exit; if AValue then Include(FFlags, dfShowEditorHints) else Exclude(FFlags, dfShowEditorHints); end; procedure TDesigner.DrawNonVisualComponent(AComponent: TComponent); var Icon: TBitmap; ItemLeft, ItemTop, ItemRight, ItemBottom: integer; Diff, ItemLeftTop: TPoint; OwnerRect, IconRect, TextRect: TRect; TextSize: TSize; IsSelected: Boolean; RGN: HRGN; begin if (AComponent is TControl) and (csNoDesignVisible in TControl(AComponent).ControlStyle) then exit; // draw children if (AComponent.Owner=nil) then begin FDDC.BeginPainting; TComponentAccess(AComponent).GetChildren(@DrawNonVisualComponent, AComponent); FDDC.EndPainting; end else if (csInline in AComponent.ComponentState) then begin if AComponent is TControl then begin // clip to client area FDDC.BeginPainting; FDDC.Canvas.SaveHandleState; OwnerRect := TControl(AComponent).ClientRect; Diff := GetParentFormRelativeClientOrigin(AComponent); OffsetRect(OwnerRect, Diff.X, Diff.Y); with OwnerRect do RGN := CreateRectRGN(Left, Top, Right, Bottom); SelectClipRGN(FDDC.DC, RGN); DeleteObject(RGN); end; TComponentAccess(AComponent).GetChildren(@DrawNonVisualComponent, AComponent); if AComponent is TControl then begin FDDC.Canvas.RestoreHandleState; FDDC.EndPainting; end; end else TComponentAccess(AComponent).GetChildren(@DrawNonVisualComponent, AComponent.Owner); if not ComponentIsIcon(AComponent) or (AComponent.Owner = nil) then Exit; // actual draw Diff := FDDC.FormOrigin; //DebugLn(['FDDC.FormOrigin - ', Diff.X, ' : ' ,Diff.Y]); // non-visual component ItemLeftTop := NonVisualComponentLeftTop(AComponent); ItemLeft := ItemLeftTop.X - Diff.X; ItemTop := ItemLeftTop.Y - Diff.Y; ItemRight := ItemLeft + NonVisualCompWidth; ItemBottom := ItemTop + NonVisualCompWidth; if not FDDC.RectVisible(ItemLeft, ItemTop, ItemRight, ItemBottom) then Exit; IsSelected := ControlSelection.IsSelected(AComponent); if FSurface = nil then begin FSurface := TBitmap.Create; FSurface.SetSize(NonVisualCompWidth, NonVisualCompWidth); FSurface.Canvas.Brush.Color := clBtnFace; FSurface.Canvas.Pen.Width := 1; end; IconRect := Rect(0, 0, NonVisualCompWidth, NonVisualCompWidth); FSurface.Canvas.Frame3D(IconRect, 1, bvRaised); FSurface.Canvas.FillRect(IconRect); if NonVisualCompBorder > 1 then InflateRect(IconRect, -NonVisualCompBorder + 1, -NonVisualCompBorder + 1); // draw component Name if ShowComponentCaptions and (((GetKeyState(VK_LBUTTON) and $80) = 0) or not IsSelected) then begin // workarounds gtk2 problem with DrawText on gc with GDK_INCLUDE_INFERIORS // it uses pango drawing and this for some reason does not take subwindow_mode // into account Icon := TBitmap.Create; try TextSize := FDDC.Canvas.TextExtent(AComponent.Name); Icon.SetSize(TextSize.cx, TextSize.cy); TextRect := Rect(0, 0, TextSize.cx, TextSize.cy); if FDDC.Form <> nil then Icon.Canvas.Brush.Color := FDDC.Form.Brush.Color else Icon.Canvas.Brush.Color := clBtnFace; Icon.Canvas.FillRect(TextRect); DrawText(Icon.Canvas.Handle, PChar(AComponent.Name), -1, TextRect, DT_CENTER or DT_VCENTER or DT_SINGLELINE or DT_NOCLIP); FDDC.Canvas.Draw( (ItemLeft + ItemRight - TextSize.cx) div 2, ItemBottom + NonVisualCompBorder + 2, Icon); finally Icon.Free; end; end; // draw component icon if Assigned(FOnGetNonVisualCompIcon) then begin Icon := nil; FOnGetNonVisualCompIcon(Self, AComponent, Icon); if Icon <> nil then begin inc(IconRect.Left, (NonVisualCompIconWidth - Icon.Width) div 2); inc(IconRect.Top, (NonVisualCompIconWidth - Icon.Height) div 2); IconRect.Right := IconRect.Left + Icon.Width; IconRect.Bottom := IconRect.Top + Icon.Height; FSurface.Canvas.StretchDraw(IconRect, Icon); end; end; FDDC.Canvas.Draw(ItemLeft, ItemTop, FSurface); if (ControlSelection.Count > 1) and IsSelected then ControlSelection.DrawMarkerAt(FDDC, ItemLeft, ItemTop, NonVisualCompWidth, NonVisualCompWidth); end; procedure TDesigner.DrawNonVisualComponents(aDDC: TDesignerDeviceContext); begin FSurface := nil; FDDC := aDDC; DrawNonVisualComponent(FLookupRoot); FDDC := nil; if FSurface <> nil then FSurface.Free; end; procedure TDesigner.DrawDesignerItems(OnlyIfNeeded: boolean); var DesignerDC: HDC; begin if WidgetSet.GetLCLCapability(lcCanDrawOutsideOnPaint) = 0 then Exit; if OnlyIfNeeded and (not (dfNeedPainting in FFlags)) then exit; Exclude(FFlags,dfNeedPainting); if (Form=nil) or (not Form.HandleAllocated) then exit; //writeln('TDesigner.DrawDesignerItems B painting'); DesignerDC := GetDesignerDC(Form.Handle); DDC.SetDC(Form, Form, DesignerDC); DDC.BeginPainting; DoPaintDesignerItems; DDC.EndPainting; DDC.Clear; ReleaseDesignerDC(Form.Handle, DesignerDC); end; procedure TDesigner.CheckFormBounds; // check if the Form was moved or resized // Note: During form loading the window manager can resize and position // the Form. Such initial changes are ignored, by waiting and comparing // not before the IDE becomes idle. When the IDE becomes the first time // idle, the form bounds are stored and used as default. // After that any change of the Form Bounds is treated as a user move // and thus calls Modified. var NewFormBounds: TRect; begin NewFormBounds:=Form.BoundsRect; if FDefaultFormBoundsValid then begin if (not CompareRect(@NewFormBounds,@FLastFormBounds)) and (not CompareRect(@NewFormBounds,@FDefaultFormBounds)) then begin //debugln('TDesigner.CheckFormBounds'); Modified; if ControlSelection.SelectionForm=Form then begin ControlSelection.CheckForLCLChanges(true); end; end; end else begin FDefaultFormBoundsValid:=true; FDefaultFormBounds:=NewFormBounds; end; FLastFormBounds:=NewFormBounds; end; procedure TDesigner.DoPaintDesignerItems; begin // marker (multi selection markers) if (ControlSelection.SelectionForm = Form) and (ControlSelection.Count > 1) then begin ControlSelection.DrawMarkers(DDC); end; // non visual component icons DrawNonVisualComponents(DDC); // guidelines and grabbers if (ControlSelection.SelectionForm=Form) then begin if EnvironmentOptions.ShowGuideLines then ControlSelection.DrawGuideLines(DDC); ControlSelection.DrawGrabbers(DDC); end; // rubberband if ControlSelection.RubberBandActive and ((ControlSelection.SelectionForm = Form) or (ControlSelection.SelectionForm = nil)) then begin ControlSelection.DrawRubberBand(DDC); end; end; function TDesigner.ComponentIsIcon(AComponent: TComponent): boolean; begin Result:=DesignerProcs.ComponentIsNonVisual(AComponent); if Result and (Mediator<>nil) then Result:=Mediator.ComponentIsIcon(AComponent); end; function TDesigner.GetParentFormRelativeClientOrigin(AComponent: TComponent): TPoint; var CurClientArea: TRect; ScrollOffset: TPoint; begin if Mediator<>nil then begin Result:=Mediator.GetComponentOriginOnForm(AComponent); Mediator.GetClientArea(AComponent,CurClientArea,ScrollOffset); inc(Result.X,CurClientArea.Left+ScrollOffset.X); inc(Result.Y,CurClientArea.Top+ScrollOffset.Y); end else begin Result:=DesignerProcs.GetParentFormRelativeClientOrigin(AComponent); end; end; function TDesigner.GetDesignedComponent(AComponent: TComponent): TComponent; begin Result:=AComponent; if AComponent=Form then begin Result:=FLookupRoot; end else begin while (Result<>nil) and (Result<>FLookupRoot) and (Result.Owner<>FLookupRoot) and (Result is TControl) do Result:=TControl(Result).Parent; end; end; function TDesigner.GetComponentEditorForSelection: TBaseComponentEditor; begin Result := nil; if (ControlSelection.Count <> 1) or (ControlSelection.SelectionForm <> Form) or (not ControlSelection[0].IsTComponent) then Exit; Result := TheFormEditor.GetComponentEditor(TComponent(ControlSelection[0].Persistent)); end; procedure TDesigner.AddComponentEditorMenuItems( AComponentEditor: TBaseComponentEditor; ClearOldOnes: boolean); var VerbCount, i: integer; NewMenuCmd: TIDEMenuCommand; begin if ClearOldOnes then DesignerMenuSectionComponentEditor.Clear; if (AComponentEditor = nil) or (DesignerMenuSectionComponentEditor = nil) then Exit; VerbCount := AComponentEditor.GetVerbCount; for i := 0 to VerbCount - 1 do begin NewMenuCmd:=RegisterIDEMenuCommand(DesignerMenuSectionComponentEditor, 'ComponentEditorVerMenuItem' + IntToStr(i), AComponentEditor.GetVerb(i), @OnComponentEditorVerbMenuItemClick); if NewMenuCmd.MenuItem<>nil then AComponentEditor.PrepareItem(i, NewMenuCmd.MenuItem); end; end; function TDesigner.NonVisualComponentAtPos(X, Y: integer): TComponent; var s: TComponentSearch; begin s := TComponentSearch.Create(nil); try s.MinClass := TComponent; s.AtPos := Point(X,Y); s.IgnoreHidden := true; s.OnlyNonVisual := true; s.Search(FLookupRoot); s.Mediator := Mediator; Result := s.Best; finally s.Free; end; end; procedure TDesigner.MoveNonVisualComponentIntoForm(AComponent: TComponent); var X, Y: SmallInt; begin DesignInfoToLeftTop(AComponent.DesignInfo, X, Y); AComponent.DesignInfo := LeftTopToDesignInfo(X, Y); end; procedure TDesigner.MoveNonVisualComponentsIntoForm; var i: Integer; AComponent: TComponent; begin for i:=0 to FLookupRoot.ComponentCount-1 do begin AComponent:=FLookupRoot.Components[i]; if ComponentIsIcon(AComponent) then begin MoveNonVisualComponentIntoForm(AComponent); end; end; end; function TDesigner.ComponentClassAtPos(const AClass: TComponentClass; const APos: TPoint; const UseRootAsDefault, IgnoreHidden: boolean): TComponent; var s: TComponentSearch; MediatorFlags: TDMCompAtPosFlags; begin if Mediator <> nil then begin MediatorFlags := []; if IgnoreHidden then Include(MediatorFlags, dmcapfOnlyVisible); Result := Mediator.ComponentAtPos(APos,AClass,MediatorFlags); end else begin s := TComponentSearch.Create(nil); try s.AtPos := APos; s.MinClass := AClass; s.IgnoreHidden := IgnoreHidden; s.Search(FLookupRoot); s.Mediator := Mediator; Result := s.Best; finally s.Free; end; end; if (Result = nil) and UseRootAsDefault and (FLookupRoot.InheritsFrom(AClass)) then Result := LookupRoot; end; procedure TDesigner.SetTempCursor(ARoot: TWinControl; ACursor: TCursor); procedure Traverse(ARoot: TWinControl); var i: integer; begin for i := 0 to ARoot.ControlCount - 1 do begin ARoot.Controls[i].SetTempCursor(ACursor); if ARoot.Controls[i] is TWinControl then Traverse(TWinControl(ARoot.Controls[i])); end; end; begin Traverse(ARoot); ARoot.SetTempCursor(ACursor); end; function TDesigner.WinControlAtPos(x, y: integer; UseRootAsDefault, IgnoreHidden: boolean): TWinControl; begin Result := TWinControl(ComponentClassAtPos(TWinControl, Point(x,y), UseRootAsDefault, IgnoreHidden)); end; function TDesigner.ControlAtPos(x, y: integer; UseRootAsDefault, IgnoreHidden: boolean): TControl; begin Result := TControl(ComponentClassAtPos(TControl, Point(x,y), UseRootAsDefault, IgnoreHidden)); end; function TDesigner.ComponentAtPos(x, y: integer; UseRootAsDefault, IgnoreHidden: boolean): TComponent; begin Result := ComponentClassAtPos(TComponent, Point(x,y), UseRootAsDefault, IgnoreHidden); end; procedure TDesigner.BuildPopupMenu; begin if FDesignerPopupMenu = nil then begin FDesignerPopupMenu:=TPopupMenu.Create(nil); with FDesignerPopupMenu do begin Name := 'DesignerPopupmenu'; OnPopup := @DesignerPopupMenuPopup; Images := IDEImages.Images_16; end; end; // assign the root TMenuItem to the registered menu root. // This will automatically create all registered items {$IFDEF VerboseMenuIntf} DesignerPopupMenu.Items.WriteDebugReport('TSourceNotebook.BuildPopupMenu '); DesignerMenuRoot.ConsistencyCheck; {$ENDIF} DesignerMenuRoot.MenuItem := FDesignerPopupMenu.Items; DesignerMenuAlign.OnClick := @OnAlignPopupMenuClick; DesignerMenuMirrorHorizontal.OnClick := @OnMirrorHorizontalPopupMenuClick; DesignerMenuMirrorVertical.OnClick := @OnMirrorVerticalPopupMenuClick; DesignerMenuScale.OnClick := @OnScalePopupMenuClick; DesignerMenuSize.OnClick := @OnSizePopupMenuClick; DesignerMenuAnchorEditor.OnClick:=@OnAnchorEditorMenuClick; DesignerMenuTabOrder.OnClick:=@OnTabOrderMenuClick; DesignerMenuOrderMoveToFront.OnClick := @OnOrderMoveToFrontMenuClick; DesignerMenuOrderMoveToFront.MenuItem.ShortCut := EditorOpts.KeyMap.CommandToShortCut(ecDesignerMoveToFront); DesignerMenuOrderMoveToBack.OnClick := @OnOrderMoveToBackMenuClick; DesignerMenuOrderMoveToBack.MenuItem.ShortCut := EditorOpts.KeyMap.CommandToShortCut(ecDesignerMoveToBack); DesignerMenuOrderForwardOne.OnClick := @OnOrderForwardOneMenuClick; DesignerMenuOrderForwardOne.MenuItem.ShortCut := EditorOpts.KeyMap.CommandToShortCut(ecDesignerForwardOne); DesignerMenuOrderBackOne.OnClick := @OnOrderBackOneMenuClick; DesignerMenuOrderBackOne.MenuItem.ShortCut := EditorOpts.KeyMap.CommandToShortCut(ecDesignerBackOne); DesignerMenuCut.OnClick:=@OnCutMenuClick; DesignerMenuCopy.OnClick:=@OnCopyMenuClick; DesignerMenuPaste.OnClick:=@OnPasteMenuClick; DesignerMenuDeleteSelection.OnClick:=@OnDeleteSelectionMenuClick; DesignerMenuSelectAll.OnClick:=@OnSelectAllMenuClick; DesignerMenuChangeClass.OnClick:=@OnChangeClassMenuClick; DesignerMenuViewLFM.OnClick:=@OnViewLFMMenuClick; DesignerMenuSaveAsXML.OnClick:=@OnSaveAsXMLMenuClick; DesignerMenuCenterForm.OnClick:=@OnCenterFormMenuClick; DesignerMenuSnapToGridOption.OnClick:=@OnSnapToGridOptionMenuClick; DesignerMenuSnapToGridOption.ShowAlwaysCheckable:=true; DesignerMenuSnapToGuideLinesOption.OnClick:=@OnSnapToGuideLinesOptionMenuClick; DesignerMenuSnapToGuideLinesOption.ShowAlwaysCheckable:=true; DesignerMenuShowOptions.OnClick:=@OnShowOptionsMenuItemClick; end; procedure TDesigner.DesignerPopupMenuPopup(Sender: TObject); var ControlSelIsNotEmpty, LookupRootIsSelected, OnlyNonVisualsAreSelected, CompsAreSelected: boolean; MultiCompsAreSelected: boolean; OneControlSelected: Boolean; SelectionVisible: Boolean; procedure UpdateChangeParentMenu; var Candidates: TFPList; i: Integer; Candidate: TWinControl; j: Integer; CurSelected: TSelectedControl; Item: TIDEMenuItem; begin Candidates:=TFPList.Create; if ControlSelIsNotEmpty and (not OnlyNonVisualsAreSelected) and (not LookupRootIsSelected) and (LookupRoot is TWinControl) then begin for i := 0 to LookupRoot.ComponentCount - 1 do begin if not (LookupRoot.Components[i] is TWinControl) then continue; Candidate:=TWinControl(LookupRoot.Components[i]); if not (csAcceptsControls in Candidate.ControlStyle) then continue; j:=ControlSelection.Count-1; while j>=0 do begin CurSelected:=ControlSelection[j]; if CurSelected.IsTControl then begin if CurSelected.Persistent=Candidate then break; if CurSelected.IsTWinControl and TWinControl(CurSelected.Persistent).IsParentOf(Candidate) then break; end; dec(j); end; if j<0 then Candidates.Add(Candidate); end; Candidates.Add(LookupRoot); end; DesignerMenuChangeParent.Visible:=Candidates.Count>0; DesignerMenuChangeParent.Clear; for i:=0 to Candidates.Count-1 do begin Item:=TIDEMenuCommand.Create(DesignerMenuChangeParent.Name+'_'+IntToStr(i)); DesignerMenuChangeParent.AddLast(Item); Item.Caption:=TWinControl(Candidates[i]).Name; Item.OnClick:=@OnChangeParentMenuClick; end; Candidates.Free; end; begin ControlSelIsNotEmpty:=(ControlSelection.Count>0) and (ControlSelection.SelectionForm=Form); LookupRootIsSelected:=ControlSelection.LookupRootSelected; OnlyNonVisualsAreSelected := ControlSelection.OnlyNonVisualPersistentsSelected; SelectionVisible:=not ControlSelection.OnlyInvisiblePersistentsSelected; CompsAreSelected:=ControlSelIsNotEmpty and SelectionVisible and not LookupRootIsSelected; OneControlSelected := ControlSelIsNotEmpty and ControlSelection[0].IsTControl; MultiCompsAreSelected := CompsAreSelected and (ControlSelection.Count>1); AddComponentEditorMenuItems(PopupMenuComponentEditor,true); DesignerMenuAlign.Enabled := CompsAreSelected and not OnlyNonVisualsAreSelected; DesignerMenuMirrorHorizontal.Enabled := MultiCompsAreSelected and not OnlyNonVisualsAreSelected; DesignerMenuMirrorVertical.Enabled := MultiCompsAreSelected and not OnlyNonVisualsAreSelected; DesignerMenuScale.Enabled := CompsAreSelected and not OnlyNonVisualsAreSelected; DesignerMenuSize.Enabled := CompsAreSelected and not OnlyNonVisualsAreSelected; DesignerMenuAnchorEditor.Enabled := (FLookupRoot is TWinControl) and (TWinControl(FLookupRoot).ControlCount > 0); DesignerMenuTabOrder.Enabled := (FLookupRoot is TWinControl) and (TWinControl(FLookupRoot).ControlCount > 0); DesignerMenuSectionZOrder.Enabled := CompsAreSelected and not OnlyNonVisualsAreSelected; DesignerMenuOrderMoveToFront.Enabled := OneControlSelected and not OnlyNonVisualsAreSelected; DesignerMenuOrderMoveToBack.Enabled := OneControlSelected and not OnlyNonVisualsAreSelected; DesignerMenuOrderForwardOne.Enabled := OneControlSelected and not OnlyNonVisualsAreSelected; DesignerMenuOrderBackOne.Enabled := OneControlSelected and not OnlyNonVisualsAreSelected; DesignerMenuCut.Enabled := CompsAreSelected; DesignerMenuCopy.Enabled := CompsAreSelected; DesignerMenuPaste.Enabled := CanPaste; DesignerMenuDeleteSelection.Enabled := CompsAreSelected; DesignerMenuChangeClass.Enabled := CompsAreSelected and (ControlSelection.Count = 1); UpdateChangeParentMenu; DesignerMenuSnapToGridOption.Checked := EnvironmentOptions.SnapToGrid; DesignerMenuSnapToGuideLinesOption.Checked := EnvironmentOptions.SnapToGuideLines; end; procedure TDesigner.OnAlignPopupMenuClick(Sender: TObject); var HorizAlignment, VertAlignment: TComponentAlignment; HorizAlignID, VertAlignID: integer; begin if ShowAlignComponentsDialog(HorizAlignID,VertAlignID)=mrOk then begin case HorizAlignID of 0: HorizAlignment:=csaNone; 1: HorizAlignment:=csaSides1; 2: HorizAlignment:=csaCenters; 3: HorizAlignment:=csaSides2; 4: HorizAlignment:=csaCenterInWindow; 5: HorizAlignment:=csaSpaceEqually; 6: HorizAlignment:=csaSide1SpaceEqually; 7: HorizAlignment:=csaSide2SpaceEqually; end; case VertAlignID of 0: VertAlignment:=csaNone; 1: VertAlignment:=csaSides1; 2: VertAlignment:=csaCenters; 3: VertAlignment:=csaSides2; 4: VertAlignment:=csaCenterInWindow; 5: VertAlignment:=csaSpaceEqually; 6: VertAlignment:=csaSide1SpaceEqually; 7: VertAlignment:=csaSide2SpaceEqually; end; ControlSelection.AlignComponents(HorizAlignment,VertAlignment); Modified; end; end; procedure TDesigner.OnMirrorHorizontalPopupMenuClick(Sender: TObject); begin ControlSelection.MirrorHorizontal; Modified; end; procedure TDesigner.OnMirrorVerticalPopupMenuClick(Sender: TObject); begin ControlSelection.MirrorVertical; Modified; end; procedure TDesigner.OnScalePopupMenuClick(Sender: TObject); var ScaleInPercent: integer; begin if ShowScaleComponentsDialog(ScaleInPercent)=mrOk then begin ControlSelection.ScaleComponents(ScaleInPercent); Modified; end; end; procedure TDesigner.OnSizePopupMenuClick(Sender: TObject); var HorizSizing, VertSizing: TComponentSizing; HorizSizingID, VertSizingID: integer; AWidth, AHeight: integer; begin if ShowSizeComponentsDialog(HorizSizingID,AWidth,VertSizingID,AHeight) = mrOk then begin case HorizSizingID of 0: HorizSizing:=cssNone; 1: HorizSizing:=cssShrinkToSmallest; 2: HorizSizing:=cssGrowToLargest; 3: HorizSizing:=cssFixed; end; case VertSizingID of 0: VertSizing:=cssNone; 1: VertSizing:=cssShrinkToSmallest; 2: VertSizing:=cssGrowToLargest; 3: VertSizing:=cssFixed; end; ControlSelection.SizeComponents(HorizSizing,AWidth,VertSizing,AHeight); Modified; end; end; procedure TDesigner.OnOrderMoveToFrontMenuClick(Sender: TObject); begin DoOrderMoveSelectionToFront; end; procedure TDesigner.OnOrderMoveToBackMenuClick(Sender: TObject); begin DoOrderMoveSelectionToBack; end; procedure TDesigner.OnOrderForwardOneMenuClick(Sender: TObject); begin DoOrderForwardSelectionOne; end; procedure TDesigner.OnOrderBackOneMenuClick(Sender: TObject); begin DoOrderBackSelectionOne; end; procedure TDesigner.HintTimer(Sender: TObject); function GetComponentHintText(AComponent: TComponent): String; const HintNameStr = '%s: %s'; HintPositionStr = 'Position: %d, %d'; HintSizeStr = 'Size: %d x %d'; HintTabStr = 'TabStop: %s; TabOrder: %d'; var AControl: TControl absolute AComponent; AWinControl: TWinControl absolute AComponent; AComponentEditor: TBaseComponentEditor; S: String; begin // component name and classname Result := Format(HintNameStr, [AComponent.Name, AComponent.ClassName]); // component position Result := Result + LineEnding + Format(HintPositionStr, [GetComponentLeft(AComponent), GetComponentTop(AComponent)]); if AComponent is TControl then // more info for controls begin // size Result := Result + '; ' + Format(HintSizeStr, [AControl.Width, AControl.Height]); // and TabStop, TabOrder for TWinControl if (AComponent is TWinControl) and not (AComponent = Form) then Result := Result + LineEnding + Format(HintTabStr, [BoolToStr(AWinControl.TabStop, True), AWinControl.TabOrder]); end; AComponentEditor := TheFormEditor.GetComponentEditor(AComponent); if Assigned(AComponentEditor) then begin S := AComponentEditor.GetCustomHint; if S <> '' then Result := Result + LineEnding + S; AComponentEditor.Free; end; end; function GetSelectionSizeHintText: String; begin Result := Format('%d x %d', [ControlSelection.Width, ControlSelection.Height]); end; function GetSelectionPosHintText: String; function ParentComponent(AComponent: TComponent): TComponent; begin Result := AComponent.GetParentComponent; if (Result = nil) and ComponentIsIcon(AComponent) then Result := AComponent.Owner; end; var BaseParent, TestParent: TComponent; BaseFound: Boolean; i: integer; P: TPoint; begin BaseFound := ControlSelection[0].IsTComponent; // search for one parent of our selection if BaseFound then begin BaseParent := ParentComponent(TComponent(ControlSelection[0].Persistent)); BaseFound := BaseParent is TWinControl; if BaseFound then begin for i := 1 to ControlSelection.Count - 1 do begin if ControlSelection[0].IsTComponent then TestParent := ParentComponent(TComponent(ControlSelection[0].Persistent)) else TestParent := nil; if TestParent <> BaseParent then begin BaseFound := False; Break; end; end; end; end; P := Point(ControlSelection.Left, ControlSelection.Top); if BaseFound then P := TWinControl(BaseParent).ScreenToClient(Form.ClientToScreen(P)); Result := Format('%d, %d', [P.X, P.Y]); end; var Rect: TRect; AHint: String; Position, ClientPos: TPoint; AWinControl: TWinControl; AComponent: TComponent; begin FHintTimer.Enabled := False; if [dfShowEditorHints]*FFlags=[] then exit; Position := Mouse.CursorPos; if not (dfHasSized in FFlags) then begin AWinControl := FindLCLWindow(Position); if not (Assigned(AWinControl)) then Exit; if GetDesignerForm(AWinControl) <> Form then exit; // search a component at the position ClientPos := Form.ScreenToClient(Position); AComponent := ComponentAtPos(ClientPos.X,ClientPos.Y,true,true); if not Assigned(AComponent) then AComponent := AWinControl; AComponent := GetDesignedComponent(AComponent); if AComponent = nil then exit; AHint := GetComponentHintText(AComponent); end else begin // components are either resize or move if (ControlSelection.LookupRoot <> Form) or (ControlSelection.Count = 0) then Exit; if ControlSelection.ActiveGrabber <> nil then AHint := GetSelectionSizeHintText else AHint := GetSelectionPosHintText; end; Rect := FHintWindow.CalcHintRect(0, AHint, nil); //no maxwidth Rect.Left := Position.X + 15; Rect.Top := Position.Y + 15; Rect.Right := Rect.Left + Rect.Right; Rect.Bottom := Rect.Top + Rect.Bottom; FHintWindow.ActivateHint(Rect, AHint); end; procedure TDesigner.SetSnapToGrid(const AValue: boolean); begin if SnapToGrid=AValue then exit; EnvironmentOptions.SnapToGrid:=AValue; end; function TDesigner.OnFormActivated: boolean; begin //the form was activated. if Assigned(FOnActivated) then FOnActivated(Self); Result:=true; end; function TDesigner.OnFormCloseQuery: boolean; begin if Assigned(FOnCloseQuery) then FOnCloseQuery(Self); Result:=true; end; function TDesigner.GetPropertyEditorHook: TPropertyEditorHook; begin Result:=TheFormEditor.PropertyEditorHook; end; end.