<p>Am 01.05.2017 18:50 schrieb "Tony Whyman via Lazarus" <<a href="mailto:lazarus@lists.lazarus-ide.org">lazarus@lists.lazarus-ide.org</a>>:<br>
><br>
><br>
><br>
> On 01/05/17 16:33, Sven Barth via Lazarus wrote:<br>
>><br>
>> Would you please elaborate on these and which FPC version you targeted?<br>
>><br>
>> Regards,<br>
>> Sven<br>
>><br>
>><br>
>><br>
> I am working with FPC 3.0.x and an example of the solution I used follows. Note that with FPC, there is no problem calling a constructor from a placeholder, while with delphi I found it necessary to add an explicit coercion. I also wanted to return the interface provided by the object. Again with FPC, this is easy. With Delphi, I found it necessary to use QueryInterface to do the same.<br>
><br>
> The difference between the two seems to be that FPC waits until the generic is instantiated before type checking while Delphi performs type checking when compiling the generic itself - or thereabouts. IMHO, FPC is much superior to Delphi in this area.</p>
<p>Your assumption about the way FPC and Delphi do things is indeed correct.</p>
<p>Comments of mine follow inside your example.</p>
<p>> {$IFDEF FPC}<br>
> TOutputBlockItemGroup<_TItem,_IItem> = class(TOutputBlockItem)<br>
> {$ELSE}<br>
> TOutputBlockItemGroup<_TItem: TOutputBlockItem; _IItem: IUnknown> = class(TOutputBlockItem)<br>
> {$ENDIF}</p>
<p>You should be able to use the second variant with FPC as well. It would allow the compiler to do more type checking when parsing the generic instead of when specializing it.</p>
<p>> public<br>
> function GetItem(index: integer): _IItem;<br>
> function Find(ItemType: byte): _IItem;<br>
> property Items[index: integer]: _IItem read getItem; default;<br>
> end;<br>
><br>
> ....<br>
><br>
> {$IFDEF FPC}<br>
> { TOutputBlockItemGroup }<br>
><br>
> function TOutputBlockItemGroup<_TItem,_IItem>.GetItem(index: integer): _IItem;<br>
> var P: POutputBlockItemData;<br>
> begin<br>
> P := inherited getItem(index);<br>
> Result := _TItem.Create(self.Owner,P);<br>
> end;<br>
><br>
> function TOutputBlockItemGroup<_TItem,_IItem>.Find(ItemType: byte): _IItem;<br>
> var P: POutputBlockItemData;<br>
> begin<br>
> P := inherited Find(ItemType);<br>
> Result := _TItem.Create(self.Owner,P);<br>
> end;<br>
><br>
><br>
> {$ELSE}<br>
><br>
> { TOutputBlockItemGroup }<br>
><br>
> function TOutputBlockItemGroup<_TItem,_IItem>.GetItem(index: integer): _IItem;<br>
> var P: POutputBlockItemData;<br>
> Obj: TOutputBlockItem;<br>
> begin<br>
> P := inherited getItem(index);<br>
> Obj := TOutputBlockItemClass(_TItem).Create(self.Owner,P);</p>
<p>Is this typecast really needed?</p>
<p>> if Obj.QueryInterface(GetTypeData(TypeInfo(_IItem))^.Guid,Result) <> 0 then<br>
> IBError(ibxeInterfaceNotSupported,[GuidToString(GetTypeData(TypeInfo(_IItem))^.Guid)]);<br>
> end;</p>
<p>"Result := Obj;" does not work? What about "Result := Obj as _IItem;"?</p>
<p>> function TOutputBlockItemGroup<_TItem,_IItem>.Find(ItemType: byte): _IItem;<br>
> var P: POutputBlockItemData;<br>
> Obj: TOutputBlockItem;<br>
> begin<br>
> P := inherited Find(ItemType);<br>
> Obj := TOutputBlockItemClass(_TItem).Create(self.Owner,P);<br>
> if Obj.QueryInterface(GetTypeData(TypeInfo(_IItem))^.Guid,Result) <> 0 then<br>
> IBError(ibxeInterfaceNotSupported,[GuidToString(GetTypeData(TypeInfo(_IItem))^.Guid)]);<br>
> end;</p>
<p>Regards,<br>
Sven</p>