LCL Tips
Deutsch (de) │ English (en) │ français (fr) │ русский (ru) │ 中文(中国大陆) (zh_CN) │
Creating a GUI by code
It is possible to create theGUI (Graphical User Interface) code completely by pascal code in Lazarus. Everything accessible from the IDE is also accessible by code. The example program and unit files below (codegui.lpr and mainform.pas) give you a template you can adapt. The most important part is not forgetting to set theParent property of the components. The creation of controls inside the form is best done in the constructor of the form:
Main program file:
programcodedgui;{$MODE DELPHI}{$H+}usesInterfaces,Forms,StdCtrls,MainForm;varMyForm:TMyForm;beginApplication.Initialize;Application.CreateForm(TMyForm,MyForm);Application.Run;end.
And a unit containing a form:
unitmainform;{$MODE DELPHI}{$H+}interfaceusesForms,StdCtrls;typeTMyForm=class(TForm)publicMyButton:TButton;procedureButtonClick(ASender:TObject);constructorCreate(AOwner:TComponent);override;end;implementationprocedureTMyForm.ButtonClick(ASender:TObject);beginClose;end;constructorTMyForm.Create(AOwner:TComponent);begininheritedCreate(AOwner);//Hint: FormCreate() is called BEFORE Create() !//so You can also put this code into FormCreate()//(This is not the case when creating components ..)Position:=poScreenCenter;Height:=400;Width:=400;VertScrollBar.Visible:=False;HorzScrollBar.Visible:=False;MyButton:=TButton.Create(Self);withMyButtondobeginHeight:=30;Left:=100;Top:=100;Width:=100;Caption:='Close';OnClick:=ButtonClick;Parent:=Self;end;// Add other component creation and property setting code hereend;end.
Create controls manually without overhead
Set the Parent as last
For Delphians: Contrary to Delphi the LCL (Lazarus Component Library) allows you to set nearly all properties in any order. For example under Delphi you cannot position a control if it has no parent. The LCL allows this and this feature can be used to reduce overhead.
withTButton.Create(Form1)dobegin// 1. creating a button sets the default size// 2. change position. No side effects, because Parent=nilSetBounds(10,10,Width,Height);// 3. change size depending on theme. Not yet, because Parent=nilAutoSize:=true;// 4. changing size because of AutoSize=true. Not yet, because Parent=nilCaption:='Ok';// 5. Set Parent. Now all the above takes place, but in a single action.Parent:=Form1;end;
When a control has a Parent, then all properties take effect immediately. Without a Parent many properties do nothing more than store the value. And as soon as the Parent is set every property is applied. This is especially true for grand children:
GroupBox1:=TGroupBox.Create(Self);withGroupBox1dobeginwithTButton1.Create(Self)dobeginAutoSize:=true;Caption:='Click me';Parent:=GroupBox1;end;Parent:=Form1;end;Form1.Show;
Autosizing starts only after every parent is set up and the form becomes visible.
Avoid early Handle creation
As soon as the Handle of a TWinControl is created, every change of a property changes the visual thing (called the widget). Even if a control is not visible, when it has a Handle, changes are still expensive.
Use SetBounds instead of Left, Top, Width, Height
Instead of
withButton1dobeginLeft:=10;Top:=10;Width:=100;Height:=25;end;
Use
withButton1dobeginSetBounds(10,10,100,25);end;
Left, Top, Width, Height are calling SetBounds. And every change of position or size invokes recalculation of all sibling controls and maybe recursively the parent and/or the grandchild controls.
DisableAlign / EnableAlign
When positioning many controls, it is a good idea to disable the recalculation of all auto sizing, aligning, anchoring.
DisableAlign;tryListBox1.Width:=ClientWidthdiv3;ListBox2.Width:=ClientWidthdiv3;ListBox3.Width:=ClientWidthdiv3;finallyEnableAlign;end;
Note: Every DisableAlign call needs an EnableAlign call. For example if you call DisableAlign two times, you must call EnableAlign twice as well.
For Delphians: This works recursively. That means DisableAlign stops aligning in all child and grandchild controls.
Creating a non-rectangular window or control
One can easily create non-rectangular windows or controls in Lazarus. For this, one can simply call TWinControl.SetShape, with the visible region as a parameter. Note that this will work for both windows and controls, as both TCustomForm and TCustomControl descend from TWinControl. One can also call the LCLIntf routine SetWindowRgn, which is completely equivalent to calling the SetShape method.
Using SetWindowRgn the code will be similar to this:
usesLCLIntf,LCLType;procedureTForm1.FormCreate(Sender:TObject);varMyRegion:HRGN;beginMyRegion:=CreateRectRgn(0,0,100,100);SetWindowRgn(Handle,MyRegion,True);DeleteObject(MyRegion);end;
An equivalent code, which uses a higher level TRegion object available in Lazarus 0.9.31+ (note that the previous way of doing things is still supported and will be in the future too), is:
usesGraphics;procedureTForm1.FormCreate(Sender:TObject);varMyRegion:TRegion;beginMyRegion:=TRegion.Create;tryMyRegion.AddRectangle(0,0,100,100);Self.SetShape(MyRegion);finallyMyRegion.Free;end;end;
The result of this operation in a window in macOS using the Qt widgetset can be seen here:
Note that SetShape can also accept a TBitmap to describe the transparent region.
See also:
Documentation entries:
- http://lazarus-ccr.sourceforge.net/docs/lcl/lclintf/setwindowrgn.html
- http://lazarus-ccr.sourceforge.net/docs/lcl/controls/twincontrol.setshape.html
Limitations:
In Gtk2 a region can only be set after a window is realized. Calling SetWindowRgn in the OnShow event handler doesn't work, the only way seems to be to call it from a timer set with interval 1, for example. Enable the timer in Form.OnShow and disable it in it's OnTimer handler.
Simulating Mouse and Keyboard input
It is very easy to simulate mouse and keyboard input in the LCL, just use the routines from the unit LCLMessageGlue like all widgetset interfaces do. This unit has routines like:
unitLCLMessageGlue;functionLCLSendMouseMoveMsg(constTarget:TControl;XPos,YPos:smallint;ShiftState:TShiftState=[]):PtrInt;functionLCLSendMouseDownMsg(constTarget:TControl;XPos,YPos:smallint;Button:TMouseButton;ShiftState:TShiftState=[]):PtrInt;functionLCLSendMouseUpMsg(constTarget:TControl;XPos,YPos:smallint;Button:TMouseButton;ShiftState:TShiftState=[]):PtrInt;functionLCLSendMouseWheelMsg(constTarget:TControl;XPos,YPos,WheelDelta:smallint;ShiftState:TShiftState=[]):PtrInt;functionLCLSendCaptureChangedMsg(constTarget:TControl):PtrInt;functionLCLSendSelectionChangedMsg(constTarget:TControl):PtrInt;functionLCLSendClickedMsg(constTarget:TControl):PtrInt;functionLCLSendMouseEnterMsg(constTarget:TControl):PtrInt;functionLCLSendMouseLeaveMsg(constTarget:TControl):PtrInt;...functionLCLSendKeyDownEvent(constTarget:TControl;varCharCode:word;KeyData:PtrInt;BeforeEvent,IsSysKey:boolean):PtrInt;functionLCLSendKeyUpEvent(constTarget:TControl;varCharCode:word;KeyData:PtrInt;BeforeEvent,IsSysKey:boolean):PtrInt;
Showing the Virtual Keyboard in Smartphones/tablets
To show the virtual keyboard when a widget receives focus just add the csRequiresKeyboardInput ControlStyle:
constructorTMyTextEditor.Create(AOwner:TComponent);begininheritedCreate(AOwner);ControlStyle:=ControlStyle+[csRequiresKeyboardInput];...end;
Iterating through all child controls of a TWinControl
This is very easy, just use a loop to iterate over TWinControl.ControlCount and TWinControl.Controls[Index]. The index is zero based.
procedureTWinControl.WriteLayoutDebugReport(constPrefix:string);vari:Integer;begininheritedWriteLayoutDebugReport(Prefix);fori:=0toControlCount-1doControls[i].WriteLayoutDebugReport(Prefix+' ');end;
Responding only once to an event
Sometimes one wants an event handler to be called just once. Say, for example, that you want to do something only the first time a form is activated, once the form and all its controls are fully loaded (which might not be the case when OnCreate is fired).
The solution is easy: simply set the corresponding property to Nil at the beginning of the handler:
procedureTMyForm.FormActivate(constSender:TObject);beginOnActivate:=Nil;{... rest of the code here ...}end;
and the event handler won't be called again.