Pages

5.05.2015

How to Create Components on Runtime

Creating components on runtime on exe
The good thing about programming is that you can do almost anything you wish. Even create components out of thin air!

Today we are going to learn how to create Buttons, Labels, Edits right at the time when the exe is running.

Basics

Before getting our hands dirty, let's get our basics straight. Before this article, LazPlanet had another article quite similar to this: How to clone your forms into many. Be sure to check that out if you're interested.

If you are scared that the code might be difficult, don't worry. Creating components are just dead simple. We create an instance of class of the component and then more gracefully give some properties to it. That's it! It's just that simple.

If you want to add some event to it, it's easy as well.

Let's see this sample code for creating buttons:

var

  btn:Tbutton;

begin

  btn:=Tbutton.Create(nil);

  // we say that place button inside Form1

  btn.parent := Form1;



  // set properties for new button

  btn.top := 10;

  btn.Left := 10;

  btn.Caption := 'Click me!';

end;


So, we have 3 steps to creating a component on the fly (or in the runtime):
1. Create an instance of the component class
btn:=Tbutton.Create(nil);

2. Set its parent
btn.parent := Form1;

3. Set properties to it (just like we always do in the Object Inspector->Properties):
  btn.top := 10;

  btn.Left := 10;

  btn.Caption := 'Click me!';

Additionally if you want to have an event attached to it, just point a procedure to the event:

btn.OnClick := @btnClickEvent;

And btnClickEvent might be like this:

procedure TForm1.btnClickEvent(Sender: TObject);

begin

  ShowMessage('Button clicked!');

end;

I said it was easy! Right!

Tiny project

Let's get a tiny project running before the actual sample project.

Start Lazarus.

Create a new Application Project (Project->New Project->Application->OK).

Draw a TButton on the form. It would be named "Button1".

Double click on it and enter:
procedure TForm1.Button1Click(Sender: TObject);

var

 btn: TButton;

begin

 btn := TButton.Create(Button1);

 btn.parent := Form1;



 // set properties for new button

 btn.top := (random(300));

 btn.Left := (random(300));

 btn.Caption := inttostr(random(9999));



 // set the click event to this procedure

 // so that new buttons also have 'cloning'

 // on its click event

 btn.OnClick := @Button1Click;

end;

Now run the project (F9 or Run->Run).

Creating components on runtime on exe


Now click the Button several times. You will see the button get 'cloned'.

Creating components on runtime on exe

Real Project

Now that we got our simple project working, we can focus on the real project. This project will be easy as well. We will just structure it a bit better, that's all.

Start Lazarus.

Create a new Application Project (Project->New Project->Application->OK).

Draw 3 Tbuttons side by side. Set caption like below:
button1 -> Create Buttons
button2 -> Create TEdits
button3 -> Create Shapes

Resize the form if you need to.

Place 3 buttons on your form

Now Draw 1 TPanel below the first Tbutton. While you have it selected change some properties from the Object Inspector. Set its BevelOuter to bvLowered. Also, empty its Caption, so that there is no "Panel 1" text in the middle of the panel.

lazarus, Set properties of TPanel

Now right click the TPanel and select Copy. Then right click the form and choose Paste. The Panel will be copied with the name Panel2. Position it under the second Tbutton.

Again Copy the last TPanel, select the form and paste. The new Tpanel will be automatically named Panel3. Position it under the third Tbutton.

lazarus, Form layout is done

Double click the button1 and enter the code below:

procedure TForm1.Button1Click(Sender: TObject);

var

  btn:TButton;

begin

  // we create an instance of a 'virtual' Tbutton

  btn := TButton.Create(nil);

  // we place the button inside Panel1

  btn.parent := Panel1;



  // position the button randomly & set caption

  btn.top := (random( Panel1.Height - btn.Height ));

  btn.Left := (random( Panel1.Width - btn.Width ));

  btn.Caption := inttostr(random(9999));



  // set the click event

  btn.OnClick := @WhatToDo;

end;

Now that we are in code view, let's do another thing. Paste the procedure somewhere under the implementation clause. If you are unsure, place it before the last line that says "end." (not end semicolon).

procedure TForm1.WhatToDo(Sender: TObject);

begin

  ShowMessage('Click event works!');

end;

Now put your cursor on the procedure name and press Ctrl+Shift+C.

WhatToDo is just a test to see if the click event works on the 'virtually' created buttons. You can do whatever you want.

If you want to use any properties of the clicked button, use it with (Sender as TButton).PropertyName. But this only works for TButton. We want to reuse our function to be used with other type of components as well. So we would use another solution... (Sender as TControl).PropertyName.

**How to use property when type of Sender component is unknown

Simple. Use (Sender as TControl).PropertyName

For example, if you want to get the Caption of the clicked component use (Sender as TControl).Caption, like as follows:

procedure TForm1.WhatToDo(Sender: TObject);

begin

    ShowMessage('Clicked on ... '+(Sender as TControl).Caption);

end;

If you want to get the type of the sender in a string, use Sender.ClassName:

procedure TForm1.WhatToDo(Sender: TObject);

begin

    ShowMessage('You clicked on a '+Sender.ClassName);

end;

Seems like a lot of discussion. But don't worry, we are all set.

Now back to our tutorial.

Go to form view if our are not (press F12). Go ahead and double click button2 and enter:

procedure TForm1.Button2Click(Sender: TObject);

var

  edt: TEdit;

begin

    // we create an instance of a 'virtual' Tbutton

  edt := TEdit.Create(nil);

    // we place the tedit inside Panel1

  edt.parent := Panel2;



  // position the tedit randomly & set caption

    edt.top := (random( Panel2.Height - edt.Height ));

  edt.Left := (random( Panel2.Width - edt.Width ));

  edt.Text := inttostr(random(9999));



  // set the click event

  edt.OnClick := @WhatToDo;

end;

Go to form view (F12). Now again double click button3 and enter:

procedure TForm1.Button3Click(Sender: TObject);

var

  shp: TShape;

begin

  // we create an instance of a 'virtual' Tbutton

  shp := TShape.Create(nil);

  // we place the tedit inside Panel1

  shp.parent := Panel3;



  // position the tedit randomly

    shp.top := random( Panel3.Height - shp.Height );

  shp.Left := random( Panel3.Width - shp.Width );

    // we make the shape random

  shp.Shape := TShapeType(random( 8 ));

    // we make its color random

    shp.Brush.Color := RGBToColor(

                                  random(255),

                                  random(255),

                                  random(255)

                                  ) ;



  // set the click event

  shp.OnClick := @WhatToDo;

end;

Here, the command is like before. We only give our shapes a random shape and we give each one a random color. So each shape is different than other.

Now Run the project to see your hard work! (Press F9 or Run-> Run.)


Now click the buttons to test!


Download Sample Code ZIP

You can download the above example tutorial project's source code from here.
Or here
Size: 588 KB
The package contains compiled executable EXE file.

5 comments:

  1. hi
    shp := TShape.Create(nil);

    and now how can you destroy the shape ?
    thank you for this article

    ReplyDelete
  2. @Thierry Dumez

    Hello. Nice to hear from a reader.

    You can free the TShape with:
    shp.free;

    Or if you want to reuse the shp variable, it is better to use:
    FreeAndNil(shp);

    Regards

    ReplyDelete
  3. Thanks for the information! But...I can't seem to compile it. I receive "unit1.pas(34,3) Fatal: Syntax error, "identifier" expected but "<" found".

    Any thoughts?

    Thanks!

    ReplyDelete
  4. I learned something! I compared your zipped code with mine and, maybe because the "copy procedure" exercise didn't work, I noticed that I had no type declaration for the WhatToDo procedure.

    On to the next tutorial!!

    Thanks for your time and efforts for these tutorials!

    ReplyDelete
  5. About my first comment above... I was on the wrong page. I was referring to the tutorial that incorporated the "Apple-tab-span" classes. That is the one I could not successfully compile. Sorry for the confusion.

    ReplyDelete