{ +-------------------------------------------------------------+ } { | | } { | GM-Software | } { | =========== | } { | | } { | Project: All Projects | } { | | } { | Description: Union for multiple types like variant. | } { | | } { | | } { | Copyright (C) - 2020 - Gerrit Moeller. | } { | | } { | Source code distributed under MIT license. | } { | | } { | See: https://www.gm-software.de | } { | | } { +-------------------------------------------------------------+ } {$INCLUDE GMCompilerSettings.inc} unit GMUnionValue; interface uses GMStrDef, Variants; type EGMUnionValueType = (uvtUnassigned, uvtNull, uvtString, uvtBoolean, uvtInt16, uvtInt32, uvtInt64, uvtDouble, uvtDateTime, uvtPointer); // // TStrType is meant to be a string type like: UnicodeString, WideString, AnsiString, RawByteString, Utf8String or AnsiString(CP_XXXX) // RGMGenericUnionValue<TStrType> = packed record private StringValue: TStrType; class operator Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): Boolean; class operator Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): Byte; class operator Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): SmallInt; class operator Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): Word; class operator Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): LongInt; class operator Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): LongWord; class operator Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): Int64; class operator Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): QWord; class operator Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): Pointer; class operator Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): Single; class operator Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): Double; class operator Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): TDateTime; class operator Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): UnicodeString; class operator Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): WideString; class operator Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): AnsiString; class operator Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): Utf8String; class operator Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): Variant; class operator Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): OleVariant; class operator Implicit(AValue: EGMUnionValueType): RGMGenericUnionValue<TStrType>; class operator Implicit(AValue: Boolean): RGMGenericUnionValue<TStrType>; class operator Implicit(AValue: SmallInt): RGMGenericUnionValue<TStrType>; class operator Implicit(AValue: LongInt): RGMGenericUnionValue<TStrType>; class operator Implicit(AValue: Int64): RGMGenericUnionValue<TStrType>; class operator Implicit(AValue: Double): RGMGenericUnionValue<TStrType>; class operator Implicit(AValue: TDateTime): RGMGenericUnionValue<TStrType>; class operator Implicit(AValue: Pointer): RGMGenericUnionValue<TStrType>; class operator Implicit(AValue: UnicodeString): RGMGenericUnionValue<TStrType>; class operator Implicit(AValue: WideString): RGMGenericUnionValue<TStrType>; class operator Implicit(AValue: AnsiString): RGMGenericUnionValue<TStrType>; class operator Implicit(AValue: Utf8String): RGMGenericUnionValue<TStrType>; class operator Implicit(AValue: Variant): RGMGenericUnionValue<TStrType>; class operator Implicit(AValue: OleVariant): RGMGenericUnionValue<TStrType>; public class operator Initialize(var AUnionValue: RGMGenericUnionValue<TStrType>); //class operator Finalize(var AUnionValue: RGMGenericUnionValue<TStrType>); //class operator Copy(constref ASrc: RGMGenericUnionValue<TStrType>; var ADst: RGMGenericUnionValue<TStrType>); class operator Equal(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): Boolean; class operator Equal(const AUnionValue: RGMGenericUnionValue<TStrType>; AValue: Int64): Boolean; class operator Equal(const AUnionValue: RGMGenericUnionValue<TStrType>; AValue: LongInt): Boolean; class operator Equal(const AUnionValue: RGMGenericUnionValue<TStrType>; AValue: Pointer): Boolean; class operator Equal(const AUnionValue: RGMGenericUnionValue<TStrType>; AValue: Double): Boolean; //class operator Equal(const AUnionValue: RGMGenericUnionValue<TStrType>; AValue: TDateTime): Boolean; class operator Equal(const AUnionValue: RGMGenericUnionValue<TStrType>; AValue: UnicodeString): Boolean; class operator Equal(const AUnionValue: RGMGenericUnionValue<TStrType>; AValue: WideString): Boolean; class operator Equal(const AUnionValue: RGMGenericUnionValue<TStrType>; AValue: AnsiString): Boolean; class operator Equal(const AUnionValue: RGMGenericUnionValue<TStrType>; AValue: Utf8String): Boolean; class operator NotEqual(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): Boolean; class operator Add(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): RGMGenericUnionValue<TStrType>; class operator Multiply(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): RGMGenericUnionValue<TStrType>; class operator Divide(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): RGMGenericUnionValue<TStrType>; class operator IntDivide(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): RGMGenericUnionValue<TStrType>; class operator Modulus(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): RGMGenericUnionValue<TStrType>; class operator LessThan(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): Boolean; class operator GreaterThan(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): Boolean; class operator LessThanOrEqual(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): Boolean; class operator GreaterThanOrEqual(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): Boolean; function IsNull: Boolean; function IsUnassigned: Boolean; function IsNullOrUnassigned: Boolean; function AsBoolean: Boolean; function AsInt16: SmallInt; function AsInt32: LongInt; function AsInt64: Int64; function AsDouble: Double; function AsDateTime: TDateTime; function AsString: TGMString; //function AsDisplayString: TGMString; function AsPointer: Pointer; function AsBooleanDflt(const ADefaultValue: Boolean = False): Boolean; function AsInt16Dflt(const ADefaultValue: SmallInt = 0): SmallInt; function AsInt32Dflt(const ADefaultValue: LongInt = 0): LongInt; function AsInt64Dflt(const ADefaultValue: Int64 = 0): Int64; function AsDoubleDflt(const ADefaultValue: Double = 0.0): Double; function AsDateTimeDflt(const ADefaultValue: TDateTime = 0): TDateTime; function AsPointerDflt(const ADefaultValue: Pointer = nil): Pointer; function AsStringDflt(const ADefaultValue: TGMString = ''): TGMString; function ValueTypeName: TGMSTring; case ValueType: EGMUnionValueType of //uvtString: (StringValue: Pointer); uvtBoolean: (BoolValue: Boolean); uvtInt16: (Int16Value: SmallInt); uvtInt32: (Int32Value: LongInt); uvtInt64: (Int64Value: Int64); uvtDouble: (DoubleValue: Double); uvtDateTime: (DateTimeValue: TDateTime); uvtPointer: (PointerValue: Pointer); end; RGMUnionValue = RGMGenericUnionValue<TGMString>; PGMUnionValue = ^RGMUnionValue; IGMGetUnionValue = interface(IUnknown) ['{9BB4E638-DB7C-4583-9EE3-410E4FB0F5CB}'] function GetUnionValue: RGMUnionValue; property Value: RGMUnionValue read GetUnionValue; end; IGMGetSetUnionValue = interface(IGMGetUnionValue) ['{3A3129DF-850E-40DA-AEF0-945A0BF2539C}'] procedure SetUnionValue(const AValue: RGMUnionValue); property Value: RGMUnionValue read GetUnionValue write SetUnionValue; end; TGMUnionValToStrFunc = function (const AValue: RGMUnionValue): TGMString; // // To be used with GMNamesAndValuesAsString and GMSeparatedValues // function GMUnionValueAsString(const AValue: RGMUnionValue): TGMString; function GMUnionValueAsQuotedString(const AValue: RGMUnionValue): TGMString; function GMUnionValueTypeName(const AValueType: EGMUnionValueType): TGMString; function GMUnionValueAsType(const AValue: RGMUnionValue; const ADestType: EGMUnionValueType): RGMUnionValue; function GMUnionValueAsTypeDflt(const AValue: RGMUnionValue; const ADestType: EGMUnionValueType): RGMUnionValue; procedure RaiseExceptionFmt(const AMsg: TGMString = ''; const AObj: TObject = nil; const ARoutineName: TGMString = ''); resourcestring srValueConvertToErrFmt = 'Cannot convert union value of type %s to type %s'; srValueConvertFromErrFmt = 'Cannot convert from %s to union value'; implementation uses SysUtils, TypInfo, GMIntf, GMCommon; type XGMUnionValueConvertError = class(EGMException); resourcestring srValueTypeOutofBounds = 'UnionValue ValueType %d out of bounds, must be in [%d .. %d]'; function GMUnionValueTypeName(const AValueType: EGMUnionValueType): TGMString; begin // AValueType enum value out of bounds may happen if not proper initialized if GMIsInRange(Ord(AValueType), Ord(Low(AValueType)), Ord(High(AValueType))) then Result := GetEnumName(TypeInfo(EGMUnionValueType), Ord(AValueType)) else Result := GMFormat(srValueTypeOutofBounds, [Ord(AValueType), Low(AValueType), High(AValueType)]); end; function GMUnionValueAsString(const AValue: RGMUnionValue): TGMString; begin Result := AValue.AsStringDflt; end; function GMUnionValueAsQuotedString(const AValue: RGMUnionValue): TGMString; begin Result := GMQuote(GMUnionValueAsString(AValue), '"', '"'); end; function GMUnionValueAsType(const AValue: RGMUnionValue; const ADestType: EGMUnionValueType): RGMUnionValue; begin Result := Default(RGMUnionValue); case ADestType of uvtUnassigned: Result := uvtUnassigned; uvtNull: Result := uvtNull; uvtString: Result := AValue.AsString; uvtBoolean: Result := AValue.AsBoolean; uvtInt16: Result := AValue.AsInt16; uvtInt32: Result := AValue.AsInt32; uvtInt64: Result := AValue.AsInt64; uvtDouble: Result := AValue.AsDouble; uvtDateTime: Result := AValue.AsDateTime; uvtPointer: Result := AValue.AsPointer; end; end; function GMUnionValueAsTypeDflt(const AValue: RGMUnionValue; const ADestType: EGMUnionValueType): RGMUnionValue; begin Result := Default(RGMUnionValue); case ADestType of uvtUnassigned: Result := uvtUnassigned; uvtNull: Result := uvtNull; uvtString: Result := AValue.AsStringDflt; uvtBoolean: Result := AValue.AsBooleanDflt; uvtInt16: Result := AValue.AsInt16Dflt; uvtInt32: Result := AValue.AsInt32Dflt; uvtInt64: Result := AValue.AsInt64Dflt; uvtDouble: Result := AValue.AsDoubleDflt; uvtDateTime: Result := AValue.AsDateTimeDflt; uvtPointer: Result := AValue.AsPointerDflt; end; end; procedure RaiseExceptionFmt(const AMsg: TGMString = cDfltExceptionMsg; const AObj: TObject = nil; const ARoutineName: TGMString = cDfltRoutineName); //const ASeverityLevel: TGMSeverityLevel = svError; //const AHelpCtx: LongInt = cDfltHelpCtx); begin raise XGMUnionValueConvertError.ObjError(AMsg, AObj, ARoutineName); end; { ---------------------------------------- } { ---- RGMGenericUnionValue<TStrType> ---- } { ---------------------------------------- } //const // // cBoolInt: array [Boolean] of Int64 = (0, 1); moved to GMCommon unit // cBoolStr: array [Boolean] of TGMString = ('False', 'True'); moved to GMCommon unit class operator RGMGenericUnionValue<TStrType>.Initialize(var AUnionValue: RGMGenericUnionValue<TStrType>); begin FillByte(AUnionValue, SizeOf(AUnionValue), 0); //FillByte(AUnionValue. SizeOf(AUnionValue), 0); //AUnionValue := Default(RGMGenericUnionValue<TStrType>); //AUnionValue.ValueType := uvtUnassigned; //AUnionValue.StringValue := ''; //AUnionValue.DoubleValue := 0; end; //class operator RGMGenericUnionValue<TStrType>.Finalize(var AUnionValue: RGMGenericUnionValue<TStrType>); //begin // case AUnionValue.ValueType of // uvtString: if AUnionValue.StringValue <> nil then begin TStrType(AUnionValue.StringValue) := ''; AUnionValue.StringValue := nil; end; // end; //end; //class operator RGMGenericUnionValue<TStrType>.Copy(constref ASrc: RGMGenericUnionValue<TStrType>; var ADst: RGMGenericUnionValue<TStrType>); //begin // Finalize(ADst); // Initialize(ADst); // // case ASrc.ValueType of // uvtString: begin // ADst.ValueType := ASrc.ValueType; // TStrType(ADst.StringValue) := TStrType(ASrc.StringValue); // end; // else Move(Asrc, ADst, SizeOf(ADst)); // end; //end; function RGMGenericUnionValue<TStrType>.IsNull: Boolean; begin Result := ValueType = uvtNull; end; function RGMGenericUnionValue<TStrType>.IsUnassigned: Boolean; begin Result := ValueType = uvtUnassigned; end; function RGMGenericUnionValue<TStrType>.IsNullOrUnassigned: Boolean; begin Result := ValueType in [uvtUnassigned, uvtNull]; end; function RGMGenericUnionValue<TStrType>.ValueTypeName: TGMSTring; begin Result := GMUnionValueTypeName(ValueType); end; {$IFDEF FPC}{$push}{$WARN 5059 off : Function result variable does not seem to be initialized}{$ENDIF} class operator RGMGenericUnionValue<TStrType>.Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): Boolean; begin case AUnionValue.ValueType of //uvtUnassigned, uvtNull, uvtBoolean: Result := AUnionValue.BoolValue; uvtString: Result := GMStrToBool(AUnionValue.StringValue); // TStrType() uvtInt16: Result := AUnionValue.Int16Value <> 0; uvtInt32: Result := AUnionValue.Int32Value <> 0; uvtInt64: Result := AUnionValue.Int64Value <> 0; uvtDouble: Result := AUnionValue.DoubleValue <> 0.0; uvtDateTime: Result := AUnionValue.DateTimeValue <> 0.0; uvtPointer: Result := AUnionValue.PointerValue <> nil; // else raise XGMUnionValueConvertError.ObjError(GMFormat(srValueConvertToErrFmt, [AUnionValue.ValueTypeName, 'Boolean']), nil, 'RGMGenericUnionValue<TStrType>.Implicit'); else RaiseExceptionFmt(GMFormat(srValueConvertToErrFmt, [AUnionValue.ValueTypeName, 'Boolean']), nil, 'RGMGenericUnionValue<TStrType>.Implicit'); end; end; {$macro on} {$define InsertUnionIntCase:=//jvkUnassigned, jvkNull uvtBoolean: Result := cBoolInt[AUnionValue.BoolValue]; uvtString: Result := GMStrToInt(AUnionValue.StringValue); // TStrType() uvtInt16: Result := AUnionValue.Int16Value; uvtInt32: Result := AUnionValue.Int32Value; uvtInt64: Result := AUnionValue.Int64Value; uvtDouble: Result := Round(AUnionValue.DoubleValue); uvtDateTime: Result := Round(AUnionValue.DateTimeValue); // else raise XGMUnionValueConvertError.ObjError(GMFormat(srValueConvertToErrFmt, [AUnionValue.ValueTypeName, cResultTypeName]), nil, 'RGMGenericUnionValue<TStrType>.Implicit'); else RaiseExceptionFmt(GMFormat(srValueConvertToErrFmt, [AUnionValue.ValueTypeName, cResultTypeName]), nil, 'RGMGenericUnionValue<TStrType>.Implicit'); end;} class operator RGMGenericUnionValue<TStrType>.Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): Byte; const cResultTypeName = 'Byte'; begin case AUnionValue.ValueType of InsertUnionIntCase end; class operator RGMGenericUnionValue<TStrType>.Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): SmallInt; const cResultTypeName = 'SmallInt'; begin case AUnionValue.ValueType of InsertUnionIntCase end; class operator RGMGenericUnionValue<TStrType>.Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): Word; const cResultTypeName = 'Word'; begin case AUnionValue.ValueType of InsertUnionIntCase end; class operator RGMGenericUnionValue<TStrType>.Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): LongInt; const cResultTypeName = 'LongInt'; begin case AUnionValue.ValueType of {$IFDEF CPU32} uvtPointer: Result := LongInt(AUnionValue.PointerValue); {$ENDIF} InsertUnionIntCase end; class operator RGMGenericUnionValue<TStrType>.Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): LongWord; const cResultTypeName = 'LongWord'; begin case AUnionValue.ValueType of {$IFDEF CPU32} uvtPointer: Result := LongWord(AUnionValue.PointerValue); {$ENDIF} InsertUnionIntCase end; class operator RGMGenericUnionValue<TStrType>.Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): Int64; const cResultTypeName = 'Int64'; begin case AUnionValue.ValueType of {$IFDEF CPU64} uvtPointer: Result := Int64(AUnionValue.PointerValue); {$ELSE} uvtPointer: Result := PtrUInt(AUnionValue.PointerValue); {$ENDIF} InsertUnionIntCase end; class operator RGMGenericUnionValue<TStrType>.Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): QWord; const cResultTypeName = 'QWord'; begin case AUnionValue.ValueType of {$IFDEF CPU64} uvtPointer: Result := QWord(AUnionValue.PointerValue); {$ELSE} uvtPointer: Result := PtrUInt(AUnionValue.PointerValue); {$ENDIF} InsertUnionIntCase end; class operator RGMGenericUnionValue<TStrType>.Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): Pointer; begin case AUnionValue.ValueType of //jvkUnassigned, jvkNull //uvtBoolean: Result := cBoolInt[AUnionValue.BoolValue]; uvtString: Result := Pointer(AUnionValue.StringValue); //uvtInt16: Result := AUnionValue.Int16Value; {$IFDEF CPU64} uvtInt32: Result := Pointer(Int64(AUnionValue.Int32Value)); uvtInt64: Result := Pointer(AUnionValue.Int64Value); {$ELSE} uvtInt32: Result := Pointer(AUnionValue.Int32Value); uvtInt64: Result := Pointer(PtrUInt(AUnionValue.Int64Value)); {$ENDIF} //uvtDouble: Result := AUnionValue.DoubleValue; //uvtDateTime: Result := AUnionValue.DateTimeValue; uvtPointer: Result := AUnionValue.PointerValue; // else raise XGMUnionValueConvertError.ObjError(GMFormat(srValueConvertToErrFmt, [AUnionValue.ValueTypeName, 'Pointer']), nil, 'RGMGenericUnionValue<TStrType>.Implicit'); else RaiseExceptionFmt(GMFormat(srValueConvertToErrFmt, [AUnionValue.ValueTypeName, 'Pointer']), nil, 'RGMGenericUnionValue<TStrType>.Implicit'); end; end; class operator RGMGenericUnionValue<TStrType>.Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): Single; begin case AUnionValue.ValueType of //jvkUnassigned, jvkNull uvtBoolean: Result := cBoolInt[AUnionValue.BoolValue]; uvtString: Result := GMStrToSingle(AUnionValue.StringValue); // TStrType() uvtInt16: Result := AUnionValue.Int16Value; uvtInt32: Result := AUnionValue.Int32Value; uvtInt64: Result := AUnionValue.Int64Value; uvtDouble: Result := AUnionValue.DoubleValue; uvtDateTime: Result := AUnionValue.DateTimeValue; uvtPointer: Result := PtrUInt(AUnionValue.PointerValue); // else raise XGMUnionValueConvertError.ObjError(GMFormat(srValueConvertToErrFmt, [AUnionValue.ValueTypeName, 'Single']), nil, 'RGMGenericUnionValue<TStrType>.Implicit'); else RaiseExceptionFmt(GMFormat(srValueConvertToErrFmt, [AUnionValue.ValueTypeName, 'Single']), nil, 'RGMGenericUnionValue<TStrType>.Implicit'); end; end; class operator RGMGenericUnionValue<TStrType>.Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): Double; begin case AUnionValue.ValueType of //jvkUnassigned, jvkNull uvtBoolean: Result := cBoolInt[AUnionValue.BoolValue]; uvtString: Result := GMStrToDouble(AUnionValue.StringValue); // TStrType() uvtInt16: Result := AUnionValue.Int16Value; uvtInt32: Result := AUnionValue.Int32Value; uvtInt64: Result := AUnionValue.Int64Value; uvtDouble: Result := AUnionValue.DoubleValue; uvtDateTime: Result := AUnionValue.DateTimeValue; uvtPointer: Result := PtrUInt(AUnionValue.PointerValue); // else raise XGMUnionValueConvertError.ObjError(GMFormat(srValueConvertToErrFmt, [AUnionValue.ValueTypeName, 'Double']), nil, 'RGMGenericUnionValue<TStrType>.Implicit'); else RaiseExceptionFmt(GMFormat(srValueConvertToErrFmt, [AUnionValue.ValueTypeName, 'Double']), nil, 'RGMGenericUnionValue<TStrType>.Implicit'); end; end; class operator RGMGenericUnionValue<TStrType>.Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): TDateTime; begin case AUnionValue.ValueType of //jvkUnassigned, jvkNull uvtString: Result := GMFixedDecodeDateTime(AUnionValue.StringValue); // GMStrToDouble(AUnionValue.StringValue); TStrType() uvtInt16: Result := AUnionValue.Int16Value; uvtInt32: Result := AUnionValue.Int32Value; uvtInt64: Result := AUnionValue.Int64Value; uvtDouble: Result := AUnionValue.DoubleValue; uvtDateTime: Result := AUnionValue.DateTimeValue; // else raise XGMUnionValueConvertError.ObjError(GMFormat(srValueConvertToErrFmt, [AUnionValue.ValueTypeName, 'TDateTime']), nil, 'RGMGenericUnionValue<TStrType>.Implicit'); else RaiseExceptionFmt(GMFormat(srValueConvertToErrFmt, [AUnionValue.ValueTypeName, 'TDateTime']), nil, 'RGMGenericUnionValue<TStrType>.Implicit'); end; end; {$IFDEF FPC}{$pop}{$ENDIF} {$macro on} {$define InsertStringResultCommonPart:= case AUnionValue.ValueType of //jvkUnassigned, jvkNull uvtBoolean: Result := cBoolStr[AUnionValue.BoolValue]; uvtString: Result := AUnionValue.StringValue; // TStrType() uvtInt16: Result := GMIntToStr(AUnionValue.Int16Value); uvtInt32: Result := GMIntToStr(AUnionValue.Int32Value); uvtInt64: Result := GMIntToStr(AUnionValue.Int64Value); uvtDouble: Result := GMDoubleToStr(AUnionValue.DoubleValue); uvtDateTime: Result := GMFixedEncodeDateTime(AUnionValue.DateTimeValue); // GMDateTimeToStr(AUnionValue.DateTimeValue); uvtPointer: Result := GMIntToHexStr(PtrUInt(AUnionValue.PointerValue)); // else raise XGMUnionValueConvertError.ObjError(GMFormat(srValueConvertToErrFmt, [AUnionValue.ValueTypeName, cResultTypeName]), nil, 'RGMGenericUnionValue<TStrType>.Implicit'); else RaiseExceptionFmt(GMFormat(srValueConvertToErrFmt, [AUnionValue.ValueTypeName, cResultTypeName]), nil, 'RGMGenericUnionValue<TStrType>.Implicit'); end;} class operator RGMGenericUnionValue<TStrType>.Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): UnicodeString; const cResultTypeName = 'UnicodeString'; begin InsertStringResultCommonPart end; class operator RGMGenericUnionValue<TStrType>.Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): WideString; const cResultTypeName = 'WideString'; begin InsertStringResultCommonPart end; class operator RGMGenericUnionValue<TStrType>.Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): AnsiString; const cResultTypeName = 'AnsiString'; begin InsertStringResultCommonPart end; class operator RGMGenericUnionValue<TStrType>.Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): Utf8String; const cResultTypeName = 'Utf8String'; begin InsertStringResultCommonPart end; class operator RGMGenericUnionValue<TStrType>.Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): Variant; begin case AUnionValue.ValueType of //uvtUnassigned: Result := Unassigned; uvtNull: Result := Variants.Null; uvtString: Result := AUnionValue.AsString; uvtBoolean: Result := AUnionValue.BoolValue; uvtInt16: Result := AUnionValue.Int16Value; uvtInt32: Result := AUnionValue.Int32Value; uvtInt64: Result := AUnionValue.Int64Value; uvtDouble: Result := AUnionValue.DoubleValue; uvtDateTime: Result := AUnionValue.DateTimeValue; uvtPointer: Result := AUnionValue.PointerValue; else Result := Variants.Unassigned; end; end; class operator RGMGenericUnionValue<TStrType>.Implicit(const AUnionValue: RGMGenericUnionValue<TStrType>): OleVariant; begin case AUnionValue.ValueType of //uvtUnassigned: Result := Unassigned; uvtNull: Result := Variants.Null; uvtString: Result := AUnionValue.AsString; uvtBoolean: Result := AUnionValue.BoolValue; uvtInt16: Result := AUnionValue.Int16Value; uvtInt32: Result := AUnionValue.Int32Value; uvtInt64: Result := AUnionValue.Int64Value; uvtDouble: Result := AUnionValue.DoubleValue; uvtDateTime: Result := AUnionValue.DateTimeValue; //uvtPointer: Result := AValue.PointerValue; else Result := Variants.Unassigned; end; end; class operator RGMGenericUnionValue<TStrType>.Implicit(AValue: EGMUnionValueType): RGMGenericUnionValue<TStrType>; begin //Result := Default(RGMGenericUnionValue<TStrType>); // <- Result may be a re-used instance, assignmet will finalize it before assign Finalize(Result); case AValue of uvtUnassigned, uvtNull: begin //TStrType(Result.StringValue) := ''; //Result.StringValue := ''; Result.ValueType := AValue; end; // else raise XGMUnionValueConvertError.ObjError(GMFormat(srValueConvertFromErrFmt, [GMUnionValueTypeName(AValue)]), nil, 'RGMGenericUnionValue<TStrType>.Implicit'); else RaiseExceptionFmt(GMFormat(srValueConvertFromErrFmt, [GMUnionValueTypeName(AValue)]), nil, 'RGMGenericUnionValue<TStrType>.Implicit'); end; end; class operator RGMGenericUnionValue<TStrType>.Implicit(AValue: Boolean): RGMGenericUnionValue<TStrType>; begin //Result := Default(RGMGenericUnionValue<TStrType>); // <- Result may be a re-used instance, assignmet will finalize it before assign Finalize(Result); Result.BoolValue := AValue; Result.ValueType := uvtBoolean; end; class operator RGMGenericUnionValue<TStrType>.Implicit(AValue: SmallInt): RGMGenericUnionValue<TStrType>; begin //Result := Default(RGMGenericUnionValue<TStrType>); // <- Result may be a re-used instance, assignmet will finalize it before assign Finalize(Result); Result.Int16Value := AValue; Result.ValueType := uvtInt16; end; class operator RGMGenericUnionValue<TStrType>.Implicit(AValue: LongInt): RGMGenericUnionValue<TStrType>; begin //Result := Default(RGMGenericUnionValue<TStrType>); // <- Result may be a re-used instance, assignmet will finalize it before assign Finalize(Result); Result.Int32Value := AValue; Result.ValueType := uvtInt32; end; class operator RGMGenericUnionValue<TStrType>.Implicit(AValue: Int64): RGMGenericUnionValue<TStrType>; begin //Result := Default(RGMGenericUnionValue<TStrType>); // <- Result may be a re-used instance, assignmet will finalize it before assign Finalize(Result); Result.Int64Value := AValue; Result.ValueType := uvtInt64; end; class operator RGMGenericUnionValue<TStrType>.Implicit(AValue: Double): RGMGenericUnionValue<TStrType>; begin //Result := Default(RGMGenericUnionValue<TStrType>); // <- Result may be a re-used instance, assignmet will finalize it before assign Finalize(Result); Result.DoubleValue := AValue; Result.ValueType := uvtDouble; end; class operator RGMGenericUnionValue<TStrType>.Implicit(AValue: TDateTime): RGMGenericUnionValue<TStrType>; begin //Result := Default(RGMGenericUnionValue<TStrType>); // <- Result may be a re-used instance, assignmet will finalize it before assign Finalize(Result); Result.DateTimeValue := AValue; Result.ValueType := uvtDateTime; end; class operator RGMGenericUnionValue<TStrType>.Implicit(AValue: Pointer): RGMGenericUnionValue<TStrType>; begin //Result := Default(RGMGenericUnionValue<TStrType>); // <- Result may be a re-used instance, assignmet will finalize it before assign Finalize(Result); Result.PointerValue := AValue; Result.ValueType := uvtPointer; end; {$define InsertAssignFromStringCommonPart:= //Result := Default(RGMGenericUnionValue<TStrType>); // <- Result may be a re-used instance, assignmet will finalize it before assign Finalize(Result); Result.StringValue := AValue; // TStrType() Result.ValueType := uvtString;} class operator RGMGenericUnionValue<TStrType>.Implicit(AValue: UnicodeString): RGMGenericUnionValue<TStrType>; begin InsertAssignFromStringCommonPart end; class operator RGMGenericUnionValue<TStrType>.Implicit(AValue: WideString): RGMGenericUnionValue<TStrType>; begin InsertAssignFromStringCommonPart end; class operator RGMGenericUnionValue<TStrType>.Implicit(AValue: AnsiString): RGMGenericUnionValue<TStrType>; begin InsertAssignFromStringCommonPart end; class operator RGMGenericUnionValue<TStrType>.Implicit(AValue: Utf8String): RGMGenericUnionValue<TStrType>; begin InsertAssignFromStringCommonPart end; {$define InsertAssignFromVariantCommonPart:= //Result := Default(RGMGenericUnionValue<TStrType>); // <- Result may be a re-used instance, assignmet will finalize it before assign //Finalize(Result); case VarType(AValue) of //varEmpty: Result := uvtUnassigned; <- handeled by else case varNull: Result := uvtNull; varBoolean: Result := Boolean(AValue); // AValue.vBoolean; varSmallint: Result := SmallInt(AValue); // AValue.vSmallint; varWord: Result := Word(AValue); // AValue.vWord; varInteger: Result := LongInt(AValue); // AValue.vInteger; varLongword: Result := Longword(AValue); // AValue.vLongword; varInt64: Result := Int64(AValue); // AValue.vInt64; varQWord: Result := QWord(AValue); // AValue.vQWord; varSingle: Result := Single(AValue); // AValue.vSingle; varDouble: Result := Double(AValue); // AValue.vDouble; varDate: Result := TDateTime(AValue); // AValue.vDate; varOleStr, varString: Result := GMVarToStr(AValue); varUnknown, varDispatch: Result := Pointer(IUnknown(AValue)); // AValue.vUnknown; varError: Result := AValue.vError; else Result := uvtUnassigned; end;} class operator RGMGenericUnionValue<TStrType>.Implicit(AValue: Variant): RGMGenericUnionValue<TStrType>; begin InsertAssignFromVariantCommonPart end; class operator RGMGenericUnionValue<TStrType>.Implicit(AValue: OleVariant): RGMGenericUnionValue<TStrType>; begin InsertAssignFromVariantCommonPart end; class operator RGMGenericUnionValue<TStrType>.Equal(const AUnionValue: RGMGenericUnionValue<TStrType>; AValue: Int64): Boolean; begin Result := AUnionValue.AsInt64 = AValue; end; class operator RGMGenericUnionValue<TStrType>.Equal(const AUnionValue: RGMGenericUnionValue<TStrType>; AValue: LongInt): Boolean; begin Result := AUnionValue.AsInt32 = AValue; end; class operator RGMGenericUnionValue<TStrType>.Equal(const AUnionValue: RGMGenericUnionValue<TStrType>; AValue: Pointer): Boolean; begin Result := AUnionValue.AsPointer = AValue; end; class operator RGMGenericUnionValue<TStrType>.Equal(const AUnionValue: RGMGenericUnionValue<TStrType>; AValue: Double): Boolean; begin Result := AUnionValue.AsDouble = AValue; end; //class operator RGMGenericUnionValue<TStrType>.Equal(const AUnionValue: RGMGenericUnionValue<TStrType>; AValue: TDateTime): Boolean; //begin // Result := AUnionValue.AsDateTime = AValue; //end; class operator RGMGenericUnionValue<TStrType>.Equal(const AUnionValue: RGMGenericUnionValue<TStrType>; AValue: UnicodeString): Boolean; begin Result := AUnionValue.AsString = AValue; end; class operator RGMGenericUnionValue<TStrType>.Equal(const AUnionValue: RGMGenericUnionValue<TStrType>; AValue: WideString): Boolean; begin Result := AUnionValue.AsString = AValue; end; class operator RGMGenericUnionValue<TStrType>.Equal(const AUnionValue: RGMGenericUnionValue<TStrType>; AValue: AnsiString): Boolean; begin Result := AUnionValue.AsString = AValue; end; class operator RGMGenericUnionValue<TStrType>.Equal(const AUnionValue: RGMGenericUnionValue<TStrType>; AValue: Utf8String): Boolean; begin Result := AUnionValue.AsString = AValue; end; //class operator RGMGenericUnionValue<TStrType>.Equal(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): Boolean; //begin // Result := AValue1.ValueType = AValue2.ValueType; // if Result then // case AValue1.ValueType of // //uvtUnassigned, uvtNull, <- Nothing! // uvtString: Result := GMSameText(AValue1.StringValue, AValue2.StringValue); // uvtBoolean: Result := AValue1.BoolValue = AValue2.BoolValue; // uvtInt16: Result := AValue1.Int16Value = AValue2.Int16Value; // uvtInt32: Result := AValue1.Int32Value = AValue2.Int32Value; // uvtInt64: Result := AValue1.Int64Value = AValue2.Int64Value; // uvtDouble: Result := AValue1.DoubleValue = AValue2.DoubleValue; // uvtDateTime: Result := AValue1.DateTimeValue = AValue2.DateTimeValue; // uvtPointer: Result := AValue1.PointerValue = AValue2.PointerValue; // end; //end; class operator RGMGenericUnionValue<TStrType>.Equal(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): Boolean; begin case AValue1.ValueType of uvtUnassigned: Result := AValue2.ValueType = uvtUnassigned; uvtNull: Result := AValue2.ValueType = uvtNull; uvtString: case AValue2.ValueType of uvtString: Result := GMSameText(AValue1.AsString, AValue2.AsString); uvtBoolean, uvtInt16, uvtInt32, uvtInt64, uvtDouble, uvtDateTime: Result := GMSameText(AValue1.AsString, AValue2.AsString); else Result := False; end; uvtBoolean: case AValue2.ValueType of uvtBoolean: Result := AValue1.BoolValue = AValue2.BoolValue; uvtInt16, uvtInt32, uvtInt64: Result := AValue1.BoolValue = (AValue2.AsInt64 <> 0); uvtdouble: Result := AValue1.BoolValue = (AVAlue2.DoubleValue <> 0); uvtString: Result := AValue1.BoolValue = GMStrToBool(AValue2.StringValue); // TStrType() else Result := False; end; uvtInt16, uvtInt32, uvtInt64: case AValue2.ValueType of //uvtString: Result := AValue1.AsInt64 = //uvtBoolean, //uvtString: Result uvtInt16, uvtInt32, uvtInt64: Result := AValue1.AsInt64 = AValue2.AsInt64; uvtDouble, uvtDateTime: Result := AValue1.AsInt64 = AValue2.DoubleValue; else Result := False; end; uvtDouble, uvtDateTime: case AValue2.ValueType of //uvtString, uvtBoolean, //uvtString, uvtInt16, uvtInt32, uvtInt64: Result := AValue1.DoubleValue = AValue2.AsInt64; uvtDouble, uvtDateTime: Result := AValue1.DoubleValue = AValue2.DoubleValue; else Result := False; end; uvtPointer: case AValue2.ValueType of {$IFDEF CPU32} uvtInt32: Result := PtrUInt(AValue1.PointerValue) = PtrUInt(AValue2.AsInt32); {$ENDIF} {$IFDEF CPU64} uvtInt64: Result := PtrUInt(AValue1.PointerValue) = PtrUInt(AValue2.AsInt64); {$ENDIF} uvtPointer: Result := AValue1.PointerValue = AValue2.PointerValue; else Result := False; end; else Result := False; end; end; class operator RGMGenericUnionValue<TStrType>.NotEqual(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): Boolean; begin Result := not (AValue1 = AValue2); end; class operator RGMGenericUnionValue<TStrType>.Add(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): RGMGenericUnionValue<TStrType>; begin case AValue1.ValueType of uvtInt16, uvtInt32, uvtInt64: case AValue2.ValueType of uvtInt16, uvtInt32, uvtInt64: Result := AValue1.AsInt64 + AValue2.AsInt64; else Result := AValue1.AsDouble + AValue2.AsDouble; end; else Result := AValue1.AsDouble + AValue2.AsDouble; end; end; class operator RGMGenericUnionValue<TStrType>.Multiply(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): RGMGenericUnionValue<TStrType>; begin case AValue1.ValueType of uvtInt16, uvtInt32, uvtInt64: case AValue2.ValueType of uvtInt16, uvtInt32, uvtInt64: Result := AValue1.AsInt64 * AValue2.AsInt64; else Result := AValue1.AsDouble * AValue2.AsDouble; end; else Result := AValue1.AsDouble * AValue2.AsDouble; end; end; class operator RGMGenericUnionValue<TStrType>.Divide(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): RGMGenericUnionValue<TStrType>; begin case AValue1.ValueType of uvtInt16, uvtInt32, uvtInt64: case AValue2.ValueType of uvtInt16, uvtInt32, uvtInt64: Result := AValue1.AsInt64 / AValue2.AsInt64; else Result := AValue1.AsDouble / AValue2.AsDouble; end; else Result := AValue1.AsDouble / AValue2.AsDouble; end; end; class operator RGMGenericUnionValue<TStrType>.IntDivide(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): RGMGenericUnionValue<TStrType>; begin Result := AValue1.AsInt64 div AValue2.AsInt64; end; class operator RGMGenericUnionValue<TStrType>.Modulus(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): RGMGenericUnionValue<TStrType>; begin Result := AValue1.AsInt64 mod AValue2.AsInt64; end; class operator RGMGenericUnionValue<TStrType>.LessThan(const AValue1, AValue2: RGMGenericUnionValue<TStrType> ): Boolean; begin case AValue1.ValueType of uvtUnassigned, uvtNull: Result := AValue2.ValueType > uvtNull; uvtString: case AValue2.ValueType of uvtString: Result := GMCompareNames(AValue1.StringValue, AValue2.StringValue) = crALessThanB; // TStrType() uvtBoolean, uvtInt16, uvtInt32, uvtInt64, uvtDouble, uvtDateTime: Result := GMCompareNames(AValue1.StringValue, AValue2.AsString) = crALessThanB; // TStrType() else Result := False; end; uvtBoolean: case AValue2.ValueType of uvtBoolean: Result := AValue1.BoolValue < AValue2.BoolValue; uvtInt16, uvtInt32, uvtInt64: Result := AValue1.BoolValue < (AValue2.AsInt64 <> 0); uvtdouble: Result := AValue1.BoolValue < (AVAlue2.DoubleValue <> 0); uvtString: Result := AValue1.BoolValue < GMStrToBool(AValue2.StringValue); // TStrType() else Result := False; end; uvtInt16, uvtInt32, uvtInt64: case AValue2.ValueType of //uvtString: Result := AValue1.AsInt64 = //uvtBoolean, //uvtString: Result uvtInt16, uvtInt32, uvtInt64: Result := AValue1.AsInt64 < AValue2.AsInt64; uvtDouble, uvtDateTime: Result := AValue1.AsInt64 < AValue2.DoubleValue; else Result := False; end; uvtDouble, uvtDateTime: case AValue2.ValueType of //uvtString, uvtBoolean, //uvtString, uvtInt16, uvtInt32, uvtInt64: Result := AValue1.DoubleValue < AValue2.AsInt64; uvtDouble, uvtDateTime: Result := AValue1.DoubleValue < AValue2.DoubleValue; else Result := False; end; uvtPointer: case AValue2.ValueType of {$IFDEF CPU32} uvtInt32: Result := PtrUInt(AValue1.PointerValue) < PtrUInt(AValue2.AsInt32); {$ENDIF} {$IFDEF CPU64} uvtInt64: Result := PtrUInt(AValue1.PointerValue) < PtrUInt(AValue2.AsInt64); {$ENDIF} uvtPointer: Result := AValue1.PointerValue < AValue2.PointerValue; else Result := False; end; else Result := False; end; end; class operator RGMGenericUnionValue<TStrType>.GreaterThan(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): Boolean; begin Result := not ((AValue1 < AValue2) or (AValue1 = AValue2)); end; class operator RGMGenericUnionValue<TStrType>.LessThanOrEqual(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): Boolean; begin Result := (AValue1 < AValue2) or (AValue1 = AValue2); end; class operator RGMGenericUnionValue<TStrType>.GreaterThanOrEqual(const AValue1, AValue2: RGMGenericUnionValue<TStrType>): Boolean; begin Result := not (AValue1 < AValue2); // (AValue1 > AValue2) or (AValue1 = AValue2); end; function RGMGenericUnionValue<TStrType>.AsBoolean: Boolean; begin Result := Self; end; function RGMGenericUnionValue<TStrType>.AsInt16: SmallInt; begin Result := Self; end; function RGMGenericUnionValue<TStrType>.AsInt32: LongInt; begin Result := Self; end; function RGMGenericUnionValue<TStrType>.AsInt64: Int64; begin Result := Self; end; function RGMGenericUnionValue<TStrType>.AsDouble: Double; begin Result := Self; end; function RGMGenericUnionValue<TStrType>.AsDateTime: TDateTime; begin Result := Self; end; function RGMGenericUnionValue<TStrType>.AsString: TGMString; begin Result := Self; end; function RGMGenericUnionValue<TStrType>.AsPointer: Pointer; begin Result := Self; end; //function RGMGenericUnionValue<TStrType>.AsDisplayString: TGMString; //begin // case ValueType of // uvtUnassigned, uvtNull: Result := ''; // else Result := Self; // end; //end; function RGMGenericUnionValue<TStrType>.AsBooleanDflt(const ADefaultValue: Boolean): Boolean; begin case ValueType of uvtUnassigned, uvtNull: Result := ADefaultValue; else Result := Self; end; end; function RGMGenericUnionValue<TStrType>.AsInt16Dflt(const ADefaultValue: SmallInt): SmallInt; begin case ValueType of uvtUnassigned, uvtNull: Result := ADefaultValue; else Result := Self; end; end; function RGMGenericUnionValue<TStrType>.AsInt32Dflt(const ADefaultValue: LongInt): LongInt; begin case ValueType of uvtUnassigned, uvtNull: Result := ADefaultValue; else Result := Self; end; end; function RGMGenericUnionValue<TStrType>.AsInt64Dflt(const ADefaultValue: Int64): Int64; begin case ValueType of uvtUnassigned, uvtNull: Result := ADefaultValue; else Result := Self; end; end; function RGMGenericUnionValue<TStrType>.AsDoubleDflt(const ADefaultValue: Double): Double; begin case ValueType of uvtUnassigned, uvtNull: Result := ADefaultValue; else Result := Self; end; end; function RGMGenericUnionValue<TStrType>.AsDateTimeDflt(const ADefaultValue: TDateTime): TDateTime; begin case ValueType of uvtUnassigned, uvtNull: Result := ADefaultValue; else Result := Self; end; end; function RGMGenericUnionValue<TStrType>.AsPointerDflt(const ADefaultValue: Pointer): Pointer; begin case ValueType of uvtUnassigned, uvtNull: Result := ADefaultValue; else Result := Self; end; end; function RGMGenericUnionValue<TStrType>.AsStringDflt(const ADefaultValue: TGMString): TGMString; begin case ValueType of uvtUnassigned, uvtNull: Result := ADefaultValue; else Result := Self; end; end; end.