Title: Delegation, Events and Methods
Question: How do you delegate actions and operations in Delphi ?
How do you use a stand-alone procedure as an event handler ?
Answer:
The basic types of reuse mechanisms are
(see Gamma, Helm, Johnson and Vlissides,
"Design Patterns",
Addision-Wesley, 1995)
Inheritance :
reuse by subclassing, lets you define the
implementation of one class in terms of another's
Aggregation (or Composition):
reuse by assembling or composing objects to get
more complex functionality
Delegation :
reuse by connecting methods : a object that
receives a request, delegates operations to
another object
(special kind of composition, two or more objects
are involved to produce a behavior)
The difference between inheritance and delegation
is the same as between to "be" and to "have",
for instance a rectangular windows-class "is" a
subclass of a general window, or a window can
"have" access to a class that knows how to
draw a rectangular window.
Delegation is used for example in the "State"
and "Strategy" design patterns, which change
the behavior of an object by changing the
objects to which it delegates requests.
In delphi, inheritance is a feature of the
language, and its obvious how to compose
objects. How does delegation work in delphi ?
Delegation within a class is done by defining
objects for a state or a strategy (see "State"
and "Stategy" design pattern).
Delegation between different classes is done by
method pointers, variables that point to methods.
I want to show this by an example :
Let us suppose that you want to display the
progress of a calculation in a progress-form.
To do this, you can use messages or method pointers.
Using messages, you can send messages with
SendMessage(ProgressForm.Handle,WM_ProgressMsg,Progress,0);
The progress-form can receive this message and
respond to it with WndProc or own procedures like
procedure WMProgress(var msg : TMessage); message WM_Progress;
This is the "windows-way". The "delphi-way"
is to use method pointers : Delphi uses them to
manage and to hide the processing of messages :
if a window-message like WM_Activate is received,
the corresponding event OnActivate is triggered.
An event encapsulates the response for a certain message,
and is implemented by a method pointer.
Instead of using SendMessage, we can define a
"ProgressMessage" event with a method pointer :
TProgressMsg = procedure(Msg : string;Progress : integer) of object;
constructor TProgressForm.Create(LCID : integer;CompressClass : TCompressClass);
begin
inherited create(nil);
fCompressClass:=CompressClass;
fCompressClass.ProgressMsg:=ProgressMsg;
...
end;
Now we can delegate the process of displaying
the progress to the progress-form :
procedure TCompressClass.SendProgressMsg(Msg: string;Progress: integer);
begin
if Assigned(fProgressMsg) then
fProgressMsg(Msg,Progress);
end;
The calling class knows, *when* the action is
triggered, the class that rents the method knows
exactly *what* to do in this case.
procedure TProgressForm.ProgressMsg(Msg : string;Progress : integer);
var s : string;
k : integer;
begin
StepLabel.Caption:=Format(sProgress+' %s',[IntToStr(Progress)])+'%';
StepLabel.Refresh;
end;
...
Delegating or Event handling is done with
Method Pointers. They allow you to change and
to extend an object's behavior assigning new methods to the pointers.
With method pointers, actions can be delegated
to other classes.
(- Events page of the Object-Inspector)
Properties allow to change an object's
state the properties.
(- Properties page of the Object-Inspector)
Normal properties encapsulate elementary
blocks of states (access to object-data),
"event properties" encapsulate elementary
blocks of behavior (access to message-handling).
All method variables, including the event
properties, are defined in the unit System
(former in SysUtils) as follows :
TMethod = record
Code, Data: Pointer;
end;
If you know that, you can do tricks like
accessing methods by the method-name with
(see Artikel 2644)
Method.Code := Class.MethodAddress(MethodName);
or using a Stand-alone procedure as an event handler
Normally an event handler for the OnClick event would
look like this :
procedure TForm1.ClassProc(Sender: TObject);
begin
...
end;
Because a Stand-alone procedure has no relation
to a certain class, we have to add an additional
Data parameter of type pointer (for methods, the
first parameter is always Self (passed in EAX),
and the first parameter explicitly declared is
in fact the second parameter (passed in EDX)):
procedure StandAloneProc(Data: Pointer; Sender: TObject);
begin
...
end;
procedure TForm1.FormCreate(Sender: TObject);
var
Event: TNotifyEvent;
begin
TMethod(Event).Code := @StandAloneProc;
TMethod(Event).Data := nil; // or Button1
Button1.OnClick := Event;
end;