In Delphi XE2, we have use
{$ifdef Win32}
{$ifdef Win64}
to identify which platform we are in.
Is there any predefined conditional that may identify VCL and FMX?
In Delphi XE2, we have use
{$ifdef Win32}
{$ifdef Win64}
to identify which platform we are in.
Is there any predefined conditional that may identify VCL and FMX?
As others says, there is not a conditional directive to determine if your application is VCL or FireMonkey. I think the most reliable way to determine if your app is FireMonkey or VCL is using a function instead of a conditional directive.
Something like
Uses
Rtti;
function IsVCLApp:Boolean;
begin
Result:= CompareText(TRttiContext.Create.GetType(TApplication.ClassInfo).QualifiedName,'Vcl.Forms.TApplication')=0;
end;
function IsFireMonkeyApp:Boolean;
begin
Result:= CompareText(TRttiContext.Create.GetType(TApplication.ClassInfo).QualifiedName,'FMX.Forms.TApplication')=0;
end;
Forms
or Vcl.Forms
the code will work ok. And finally about your last comment that's depend of the kind of application, this answer suggests use a function as alternative to the use of a conditional directive
. –
Agonizing Forms
and not Vcl.Forms
. If you explicitly use Vcl.Forms
or Fmx.Forms
then you've already decided on the platform in the uses clause for the unit, and thus already have a conditional way to check the target widgetset. –
Lighterman IsFireMonkeyApp
to Result := {$IF DECLARED(TFmxObject)}True{$ELSE}False{$IFEND};
and it would have the exact same behavior. –
Lighterman Result := GetClass('TFmxObject') <> nil;
, as this does not require to have FMX.Types is scope. –
Ratiocinate Although not documented you can have VCL and Firemonkey in the same application.
There is no compiler define.
If you're building something that needs to be both VCL and Firemonkey I would recommend separation of the units.
A possible way:
Mixing UI code from two different frameworks in the same unit is not a good idea. It will link in the other library when it's not needed.
As others says, there is not a conditional directive to determine if your application is VCL or FireMonkey. I think the most reliable way to determine if your app is FireMonkey or VCL is using a function instead of a conditional directive.
Something like
Uses
Rtti;
function IsVCLApp:Boolean;
begin
Result:= CompareText(TRttiContext.Create.GetType(TApplication.ClassInfo).QualifiedName,'Vcl.Forms.TApplication')=0;
end;
function IsFireMonkeyApp:Boolean;
begin
Result:= CompareText(TRttiContext.Create.GetType(TApplication.ClassInfo).QualifiedName,'FMX.Forms.TApplication')=0;
end;
Forms
or Vcl.Forms
the code will work ok. And finally about your last comment that's depend of the kind of application, this answer suggests use a function as alternative to the use of a conditional directive
. –
Agonizing Forms
and not Vcl.Forms
. If you explicitly use Vcl.Forms
or Fmx.Forms
then you've already decided on the platform in the uses clause for the unit, and thus already have a conditional way to check the target widgetset. –
Lighterman IsFireMonkeyApp
to Result := {$IF DECLARED(TFmxObject)}True{$ELSE}False{$IFEND};
and it would have the exact same behavior. –
Lighterman Result := GetClass('TFmxObject') <> nil;
, as this does not require to have FMX.Types is scope. –
Ratiocinate There is no compiler directive because technically there is no such thing as a firemonkey application or a vcl application. Only applications which make use of these technologies. An application can use fxm or vcl or both or neither (eg. a console app). This is a bit like asking if it is an SQL application. You can of course programatically check the ancestry of individual forms to see which framework they inherit from. Again, inside a unit that has no associated form, this has no meaning.
There does not seem to be a compiler define specifically for VCL/FireMonkey. You would need to create your own.
A list of predefined conditionals can be found in the documentation.
GUI without Firemonkey
as is shown here #7442631 –
Agonizing Abbrevia supports both the VCL and CLX using this kind of split:
QAbUnit1.pas:
{$DEFINE UsingCLX}
unit QAbUnit1;
{$I AbUnit1.pas}
AbUnit1.pas:
{$IFNDEF UsingCLX}
{$DEFINE UsingVCL}
unit AbUnit1;
{$ENDIF}
type
...
TMyWidget = class({$IFDEF UsingVCL}TWinControl{$ENDIF}
{$IFDEF UsingCLX}TWidgetControl{$ENDIF})
...
end;
end.
To add FireMonkey support, I'd add a file like this:
FmxAbUnit1.pas:
{$DEFINE UsingFMX}
unit FmxAbUnit1;
{$I AbUnit1.pas}
{$ENDIF}
and then make whatever conditional changes I need to AbUnit1.pas.
It's not a nice clean split like Robert's suggestion, but the advantage is that all of your editing occurs in a single file, and the conditional define is handled automatically, so it doesn't need to appear in the project options. Who ever uses your library just includes the appropriate unit to decide which one they want to use. You could probably take advantage of unit scoping too, by naming the files Fmx.AbUnit1.pas
and Vcl.AbUnit1.pas
, but I think Embarcadero discourages that.
Try this snippet:
{$IF Declared(FMX)}
// FMX code here. To test this approach you may use {$MESSAGE FATAL 'FMX'}
{$ELSEIF Declared(VCL)}
// VCL code here. To test this approach you may use {$MESSAGE FATAL 'VCL'}
{$IFEND}
It tests if corresponding namespace was declared using IF compilation directive
As FMX isn't mutually exclusive with VCL, there may be a need to add following branches:
{$ELSEIF Declared(FMX) and Declared(VCL)}
// FMX+VCL code here. To test: {$MESSAGE FATAL 'FMX+VCL'}
{$ELSE}
// no GUI frameworks code here. To test: {$MESSAGE FATAL 'no GUI frameworks'}
© 2022 - 2024 — McMap. All rights reserved.