 |
BorlandTalk.com Borland discussion newsgroups
|
| View previous topic :: View next topic |
| Author |
Message |
Maurizio Ferreira Guest
|
Posted: Thu Jun 30, 2005 8:51 am Post subject: typecasting interfaces |
|
|
I have a series of frames, each implementing a common interface
type
ImyInterf = interface
procedure doit;
end;
tframe1 = class (tframe,ImyInterf)
....
procedure doit;
end;
tframe2 = class (tframe,ImyInterf)
....
procedure doit;
end;
Now I'd like to iterate on the form components to activate every frame
:
procedure TMainForm.ActivateFrame;
var I : integer;
F : ImyInterf;
begin
for I := 0 to ComponentCount -1 do begin
if Components[i] is tFrame then begin
F := Components[i] as ImyInterf; // ---- Error
F := ImyInterf(Components[i]); // ---- Error
F.doit; // --- this is what i'd like to call
end;
end;
end;
However The compiler (d7) complains in the above statements:
The first gives the error:
operator not applicable to this operand type
the second the error
incompatible types ImyInterf and tcomponent
What can I do ?
Regards
Maurizio
|
|
| Back to top |
|
 |
Skybuck Flying Guest
|
Posted: Thu Sep 08, 2005 2:41 pm Post subject: Re: typecasting interfaces |
|
|
Hi,
Interesting problem.
The solution is simple.. at least for me since I have been coding in delphi
for many years... but I never use interfaces though I am still able to solve
your problem =D
The problem is with these two lines:
| Quote: | if Components[i] is tFrame then begin
|
Tframe does not contain any interfaces.
| Quote: | F := Components[i] as ImyInterf; // ---- Error
|
Tcomponent does not implement any interfaces therefore it's not allowed to
use the "as" operator in combination with an interface type as above. This
is only allowed if the class implements an interface as described in the
delphi help:
"
Using the as operator with interfaces
Classes that implement interfaces can use the as operator for dynamic
binding on the interface. In the following example,
procedure PaintObjects(P: TInterfacedObject)
var
X: IPaint;
begin
X := P as IPaint;
{ statements }
end;
the variable P of type TInterfacedObject, can be assigned to the variable X,
which is an IPaint interface reference. Dynamic binding makes this
assignment possible. For this assignment, the compiler generates code to
call the QueryInterface method of P's IInterface interface. This is because
the compiler cannot tell from P's declared type whether P's instance
actually supports IPaint. At runtime, P either resolves to an IPaint
reference or an exception is raised. In either case, assigning P to X will
not generate a compile-time error as it would if P was of a class type that
did not implement IInterface.
When you use the as operator for dynamic binding on an interface, you should
be aware of the following requirements:
Explicitly declaring IInterface: Although all interfaces derive from
IInterface, it is not sufficient, if you want to use the as operator, for a
class to simply implement the methods of IInterface. This is true even if it
also implements the interfaces it explicitly declares. The class must
explicitly declare IInterface in its interface list.
Using an IID: Interfaces can use an identifier that is based on a GUID
(globally unique identifier). GUIDs that are used to identify interfaces are
referred to as interface identifiers (IIDs). If you are using the as
operator with an interface, it must have an associated IID. To create a new
GUID in your source code you can use the Ctrl+Shift+G editor shortcut key.
"
The solution is to create a special version of Tframe which does contain an
interface like so:
TmyInterfacedFrame = class(Tframe, ImyInterface)
etc
This will be an abstract class where all other frames are to be derived
from.
Here is a complete example:
// unit1:
unit Unit1;
interface
uses
Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
Dialogs, StdCtrls, Unit3, Unit2;
type
TForm1 = class(TForm)
Frame21: TFrame2;
Frame31: TFrame3;
Button1: TButton;
procedure Button1Click(Sender: TObject);
private
{ Private declarations }
public
{ Public declarations }
end;
var
Form1: TForm1;
implementation
{$R *.dfm}
uses
unit4;
procedure TForm1.Button1Click(Sender: TObject);
var I : integer;
F : ImyInterface;
begin
for I := 0 to ComponentCount -1 do
begin
if Components[i] is TmyInterfacedFrame then
begin
F := Components[i] as TmyInterfacedFrame;
F.doit;
end;
end;
end;
end.
// unit2:
unit Unit2;
interface
uses
Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
Dialogs, StdCtrls, Unit4;
type
TFrame2 = class(TmyInterfacedFrame)
Memo1: TMemo;
private
{ Private declarations }
public
{ Public declarations }
procedure DoIt; override;
end;
implementation
{$R *.dfm}
procedure TFrame2.DoIt;
begin
ShowMessage('do it 2');
end;
end.
// unit3
unit Unit3;
interface
uses
Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
Dialogs, StdCtrls, Unit4;
type
TFrame3 = class(TmyInterfacedFrame)
Button1: TButton;
private
{ Private declarations }
public
{ Public declarations }
procedure DoIt; override;
end;
implementation
{$R *.dfm}
procedure TFrame3.DoIt;
begin
ShowMessage('do it 3');
end;
end.
unit Unit4;
interface
uses
Forms;
type
ImyInterface = interface
procedure DoIt;
end;
TmyInterfacedFrame = class(Tframe, ImyInterface)
procedure DoIt; virtual; abstract;
end;
implementation
end.
Note:
DoIt is now a virtual method.
All frames which are derived from TmyInterfaceFrame need to override the
"DoIt" method otherwise the virtual abstract method will be called which has
no implementation so then an EAbstractError exception is raised.
Bye,
Skybuck.
|
|
| Back to top |
|
 |
|
|
You cannot post new topics in this forum You cannot reply to topics in this forum You cannot edit your posts in this forum You cannot delete your posts in this forum You cannot vote in polls in this forum
|
|