lazarus/components/sparta/mdi/source/sparta_multiplyresizer.pas
hnb d53cfa9c28 Merged revision(s) 51414, 51448-51450, 51468, 51473, 51477, 51480, 51483, 51486, 51511, 51543-51544, 51638, 52714, 52725-52727 from branches/free-sparta:
sparta: initial commit of "compilable" new sparta package - smart form editor. !NOTE: not for daily usage.
........
sparta: Initial conception for package for MDI: sparta_MDI. Base for sparta_DockedFormEditor package.
........
sparta: Generics.Collections library ( sync with https://github.com/dathox/generics.collections SHA fda586932bd80ef58c08f8ebf5a24316ca4ccca5)
........
sparta: smart form editor adjustment for new sparta_MDI
........
sparta: new class "TFormImpl" for MDI solution (created from TDesignedFormImpl). 
........
sparta: 
-MDI form container "TFormContainer"
-New IResizeFrame interface to handle MDI form moving
-New frame TfrFormBackgroundForMDI
........
sparta: sparta_MDI package modifications:
-new class TMultiplyResizer to menage MDI desktop
-more generic resizer: TAbstractResizer. Base for IDE resizer and TMultiplyResizer
-more advanced IResizeFrame interface


........
sparta: 
-DockedFormEditor adjustment for latest changes in mdi package
-small changes in mdi (visibility of methods).
-OnModified method for IResizeFrame

........
sparta: MDI
-simulate MDI forms order for TMultiplyResizer
-property DesignedForm: IDesignedForm for IResizeFrame
........
sparta:
-IMPORTANT! pixel perfect form resizing (fix for problems for controls with align alLeft, alRight etc on design form).
-Fix problem for windows: wrong design design window width (a little bigger than designed size) TFormImpl.SetRealBounds -> AdjustSize
........
sparta: mdi bug fix for AV in TMultiplyResizer
........
Fix compilation for FPC 3.0 (TRect changes in FPC 3.1 trunk)
........
sparta: Cannot resize the docked form designer, issue #29380 patch from Anthony Walter. Thanks!
........
sparta ToolsAPI: Delphi compatible ToolsAPI/DesignIDE interface at XE2 level (proxy for IDEIntf). Initial commit (no functionality yet), just interfaces and classes without implementation:

designeditors.pas:
-TComponentEditor

designintf.pas:
-Interfaces: IEventInfo, IClass, IActivatable, IDesignObject, IDesignPersistent, IDesignerSelections, IDesigner60, IDesigner70, IDesigner80, IDesigner100, IDesigner, IComponentEditor
-TBaseComponentEditor
-RegisterComponentEditor

designmenus.pas:
-Interfaces: IMenuItems, IMenu, IMainMenu, IPopupMenu, IMenuItem

 
 


........
when form is removed we need to remove all handlers located in collections FFormsStack and FForms. Necessary to avoid AV.
........
sparta: more correct and simpler calculation of form border for Windows
........
sparta: 
  * Fix for loop error for resize. Highly visible problem for docked forms/frames with Align=alClient. 
  * New THookFrame class as new meta class for Frames.
........
updated lpl
........

git-svn-id: trunk@52728 -
2016-07-20 10:40:03 +00:00

209 lines
5.1 KiB
ObjectPascal

unit sparta_MultiplyResizer;
{$mode delphi}{$H+}
interface
uses
Forms, Classes, SysUtils, Controls, Generics.Collections, LMessages,
sparta_AbstractResizer, sparta_InterfacesMDI, sparta_BasicResizeFrame;
type
{ TMultiplyResizer }
{ TResizerRec }
TResizerRec = class
public
Frame: TBasicResizeFrame;
Idx: Integer;
constructor Create(AFrame: TBasicResizeFrame);
destructor Destroy; override;
end;
TMultiplyResizer = class(TAbstractResizer)
private class var
FAllForms: TDictionary<IDesignedForm, TMultiplyResizer>;
class constructor Create;
class destructor Destroy;
class procedure OnUserInputHandler(Sender: TObject; Msg: Cardinal);
private
FFormsStack: TList<IDesignedForm>;
FForms: TObjectDictionary<IDesignedForm, TResizerRec>;
protected
// only allow to set prevously added DesignedForms by AddDesignedForm
//procedure SetDesignedForm(const AValue: IDesignedForm); override;
procedure RemoveFormEvent(Sender: TObject; Form: TCustomForm);
protected { IResizer }
//procedure TryBoundSizerToDesignedForm(Sender: TObject); override;
function GetActiveResizeFrame: IResizeFrame; override;
function GetActiveDesignedForm: IDesignedForm; override;
public
constructor Create(AParent: TWinControl; AResizerFrameClass: TResizerFrameClass); override;
destructor Destroy; override;
procedure AddDesignedForm(const AForm: IDesignedForm);
end;
implementation
{ TResizerRec }
constructor TResizerRec.Create(AFrame: TBasicResizeFrame);
begin
Frame := AFrame;
end;
destructor TResizerRec.Destroy;
begin
//Frame.Free; // free by owner
inherited Destroy;
end;
{ TMultiplyResizer }
class constructor TMultiplyResizer.Create;
begin
Application.AddOnUserInputHandler(OnUserInputHandler);
FAllForms := TDictionary<IDesignedForm, TMultiplyResizer>.Create;
end;
class destructor TMultiplyResizer.Destroy;
begin
Application.RemoveOnUserInputHandler(OnUserInputHandler);
FAllForms.Free;
end;
class procedure TMultiplyResizer.OnUserInputHandler(Sender: TObject;
Msg: Cardinal);
var
LCtrl: TControl;
LActiveFrame: TBasicResizeFrame = nil;
LResizer: TMultiplyResizer = nil;
LResizerRec, LLastResizerRec: TResizerRec;
tmp: Integer;
begin
if (Msg = LM_LBUTTONDOWN) or (Msg = LM_RBUTTONDOWN) or (Msg = LM_MBUTTONDOWN) then
begin
LCtrl := FindDragTarget(Mouse.CursorPos, True);
// find dedicated TMultiplyResizer and Frame
if LCtrl <> nil then
repeat
if LCtrl is TBasicResizeFrame then
LActiveFrame := TBasicResizeFrame(LCtrl);
LCtrl := LCtrl.Parent;
if (LCtrl <> nil) and (LCtrl.Owner is TMultiplyResizer) then
begin
LResizer := TMultiplyResizer(LCtrl.Owner);
Break;
end;
until (LCtrl = nil);
// frame to activate
if Assigned(LActiveFrame) and Assigned(LResizer) then
begin
LResizerRec := LResizer.FForms[LActiveFrame.DesignedForm];
LLastResizerRec := LResizer.FForms[LResizer.FFormsStack.Last];
// already on top
if LResizerRec = LLastResizerRec then
Exit;
LResizer.FFormsStack.Exchange(LResizerRec.Idx, LLastResizerRec.Idx);
tmp := LLastResizerRec.Idx;
LLastResizerRec.Idx := LResizerRec.Idx;
LResizerRec.Idx := tmp;
// show!
LActiveFrame.BringToFront;
end;
end;
end;
procedure TMultiplyResizer.RemoveFormEvent(Sender: TObject; Form: TCustomForm);
var
LForm: IDesignedForm;
begin
if Supports(Form, IDesignedForm, LForm) then
begin
FFormsStack.Remove(LForm);
FForms.Remove(LForm);
end;
end;
function TMultiplyResizer.GetActiveResizeFrame: IResizeFrame;
var
LForm: IDesignedForm;
begin
LForm := GetActiveDesignedForm;
if LForm = nil then
Result := nil
else
Result := FForms[LForm].Frame;
end;
function TMultiplyResizer.GetActiveDesignedForm: IDesignedForm;
begin
if FFormsStack.Count = 0 then
Result := nil
else
Result := FFormsStack.Last;
end;
constructor TMultiplyResizer.Create(AParent: TWinControl;
AResizerFrameClass: TResizerFrameClass);
begin
inherited Create(AParent, AResizerFrameClass);
FForms := TObjectDictionary<IDesignedForm, TResizerRec>.Create([doOwnsValues]);
FFormsStack := TList<IDesignedForm>.Create;
end;
destructor TMultiplyResizer.Destroy;
begin
FFormsStack.Free;
FForms.Free;
inherited Destroy;
end;
procedure TMultiplyResizer.AddDesignedForm(const AForm: IDesignedForm);
var
LFrame: TBasicResizeFrame;
LResizerRec: TResizerRec;
begin
if AForm = nil then
Exit;
LFrame := CreateResizeFrame;
AForm.BeginUpdate;
AForm.Form.Parent := LFrame.pClient;
{$IFNDEF WINDOWS}
AForm.Form.BorderStyle := bsNone;
{$ENDIF}
// for big forms (bigger than screen resolution) we need to refresh Real* values
AForm.RealWidth := AForm.Width;
AForm.RealHeight := AForm.Height;
AForm.EndUpdate;
AForm.OnChangeHackedBounds := TryBoundSizerToDesignedForm;
LFrame.DesignedForm := AForm;
LResizerRec := TResizerRec.Create(LFrame);
FForms.Add(AForm, LResizerRec);
LResizerRec.Idx := FFormsStack.Add(AForm);
// when form is removed we need to remove all handlers located in FFormsStack
// and FForms
Screen.AddHandlerRemoveForm(RemoveFormEvent);
end;
end.