I’m trying to use C++ Template ‘mixins’ to create some new VCL components with shared additional functionality. Example…
template <class T> class Mixin : public T
{
private:
typedef T inherited;
// ...additional methods
public:
Mixin(TComponent *owner) : inherited(owner)
{
// .. do stuff here
};
};
Used like this:
class MyLabel : public Mixin<TLabel>
{
....
}
class MyEdit : public Mixin<TEdit>
{
....
}
Now, everything compiles fine, and the mixin stuff seems to work – until I try and save the component to a stream using TStream->WriteComponent, where the inherited properties (eg TLabel.Width/Height/etc.) don’t get written. This is even with a ‘null’ mixin like the one shown above.
My code works fine when just deriving classes directly from TForm, TEdit, etc – and the class is correctly registered with the streaming system.
The quick/simple answer is: no; when dealing with a template, the compiler won’t generate the proper descriptors to make streaming working. However, since this has come up before, I peeked under the cover to find out what’s missing. And what I found is that it’s almost there. So here’s a little more information.
Upfront the compiler will never treat a template-based type as a Delphi. For example, do something like this:
… and you’ll see the error
“Error E2242 test.cpp 53: __classid requires Delphi style class type (i.e. class marked __declspec(delphiclass) or derived from System::TObject) in function testing()”
This basically says the compiler does not consider this type/class as compatible with Delphi-classes [i.e. those that derive from TObject]. Internally there’s just a flag on the symbol that says whether the type is delphi-compatible or not. And I noticed that I could trick the compiler into marking the type as delphi-style if I forced it to walk up the hierarchy.. which is something it has to do if I create an instance of the object. So, with this hack the error goes away:
But much nicer was actually to use the __declspec(delphiclass) directly on the template, as in:
So now that the compiler treats the type as a delphi-style class without hacks, I peeked a little more and found the issue you’re probably running into: Delphi classes have the TTypeData.PropCount field – http://docwiki.embarcadero.com/VCL/en/TypInfo.TTypeData – which is a sum of the class’ properties, including those of its base classes. Due to the way the various pieces of information are computed, the compiler writes out a ‘0’ for that field when a template is involved:(
You can see this by printing out the PropCount, as in:
When run the above prints:
IOW, Mixin shows up with ‘0’ published properties while its base type has 3:(
I suspect the streaming system relies on this count and that’s why inherited properties are not being written out in your setup.
I considered tweaking the generated descriptors at runtime but since we write them to _TEXT it’s bound to trigger DEP.
I’ll look at the logic that computes the PropCount to see if there’s some way to get it to compute the correct number. If time allows, please do open a QC for this: now that I’ve peek underneath, I believe it would not require much effort to get this working as expected.
Cheers,
Bruneau
PS: In my sample I even had the Mixin publish a property and the compiler generated the correct descriptor for that property; however, the total count was still zero.