12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354 |
- {
- This file is part of the Free Pascal run time library.
- Copyright (c) 1999-2000 by Florian Klaempfl
- member of the Free Pascal development team
- See the file COPYING.FPC, included in this distribution,
- for details about the copyright.
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- **********************************************************************}
- { This unit provides the same Functionality as the TypInfo Unit }
- { of Delphi }
- unit typinfo;
- interface
- {$MODE objfpc}
- {$MODESWITCH AdvancedRecords}
- {$inline on}
- {$macro on}
- {$h+}
- uses SysUtils;
- // temporary types:
- type
- {$MINENUMSIZE 1 this saves a lot of memory }
- {$ifdef FPC_RTTI_PACKSET1}
- { for Delphi compatibility }
- {$packset 1}
- {$endif}
- // if you change one of the following enumeration types
- // you have also to change the compiler in an appropriate way !
- TTypeKind = (tkUnknown,tkInteger,tkChar,tkEnumeration,tkFloat,
- tkSet,tkMethod,tkSString,tkLString,tkAString,
- tkWString,tkVariant,tkArray,tkRecord,tkInterface,
- tkClass,tkObject,tkWChar,tkBool,tkInt64,tkQWord,
- tkDynArray,tkInterfaceRaw,tkProcVar,tkUString,tkUChar,
- tkHelper,tkFile,tkClassRef,tkPointer);
- TOrdType = (otSByte,otUByte,otSWord,otUWord,otSLong,otULong);
- {$ifndef FPUNONE}
- TFloatType = (ftSingle,ftDouble,ftExtended,ftComp,ftCurr);
- {$endif}
- TMethodKind = (mkProcedure,mkFunction,mkConstructor,mkDestructor,
- mkClassProcedure,mkClassFunction,mkClassConstructor,
- mkClassDestructor,mkOperatorOverload);
- TParamFlag = (pfVar,pfConst,pfArray,pfAddress,pfReference,pfOut);
- TParamFlags = set of TParamFlag;
- TIntfFlag = (ifHasGuid,ifDispInterface,ifDispatch,ifHasStrGUID);
- TIntfFlags = set of TIntfFlag;
- TIntfFlagsBase = set of TIntfFlag;
- // don't rely on integer values of TCallConv since it includes all conventions
- // which both delphi and fpc support. In the future delphi can support more and
- // fpc own conventions will be shifted/reordered accordinly
- TCallConv = (ccReg, ccCdecl, ccPascal, ccStdCall, ccSafeCall,
- ccCppdecl, ccFar16, ccOldFPCCall, ccInternProc,
- ccSysCall, ccSoftFloat, ccMWPascal);
- {$MINENUMSIZE DEFAULT}
- const
- ptField = 0;
- ptStatic = 1;
- ptVirtual = 2;
- ptConst = 3;
- type
- TTypeKinds = set of TTypeKind;
- ShortStringBase = string[255];
- PVmtFieldEntry = ^TVmtFieldEntry;
- TVmtFieldEntry =
- {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}
- packed
- {$endif FPC_REQUIRES_PROPER_ALIGNMENT}
- record
- FieldOffset: PtrUInt;
- TypeIndex: Word;
- Name: ShortString;
- end;
- PVmtFieldTable = ^TVmtFieldTable;
- TVmtFieldTable =
- {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}
- packed
- {$endif FPC_REQUIRES_PROPER_ALIGNMENT}
- record
- Count: Word;
- ClassTab: Pointer;
- { should be array[Word] of TFieldInfo; but
- Elements have variant size! force at least proper alignment }
- Fields: array[0..0] of TVmtFieldEntry
- end;
- {$PACKRECORDS 1}
- TTypeInfo = record
- Kind : TTypeKind;
- Name : ShortString;
- // here the type data follows as TTypeData record
- end;
- PTypeInfo = ^TTypeInfo;
- PPTypeInfo = ^PTypeInfo;
- { Note: these are only for backwards compatibility. New type references should
- only use PPTypeInfo directly! }
- {$ifdef ver3_0}
- {$define TypeInfoPtr := PTypeInfo}
- {$else}
- {$define TypeInfoPtr := PPTypeInfo}
- {$endif}
- {$PACKRECORDS C}
- // members of TTypeData
- TArrayTypeData =
- {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}
- packed
- {$endif FPC_REQUIRES_PROPER_ALIGNMENT}
- record
- private
- function GetElType: PTypeInfo; inline;
- function GetDims(aIndex: Byte): PTypeInfo; inline;
- public
- property ElType: PTypeInfo read GetElType;
- property Dims[Index: Byte]: PTypeInfo read GetDims;
- public
- Size: SizeInt;
- ElCount: SizeInt;
- ElTypeRef: TypeInfoPtr;
- DimCount: Byte;
- DimsRef: array[0..255] of TypeInfoPtr;
- end;
- PManagedField = ^TManagedField;
- TManagedField =
- {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}
- packed
- {$endif FPC_REQUIRES_PROPER_ALIGNMENT}
- record
- private
- function GetTypeRef: PTypeInfo; inline;
- public
- property TypeRef: PTypeInfo read GetTypeRef;
- public
- TypeRefRef: TypeInfoPtr;
- FldOffset: SizeInt;
- end;
- PProcedureParam = ^TProcedureParam;
- TProcedureParam =
- {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}
- packed
- {$endif FPC_REQUIRES_PROPER_ALIGNMENT}
- record
- private
- function GetParamType: PTypeInfo; inline;
- public
- property ParamType: PTypeInfo read GetParamType;
- public
- Flags: Byte;
- ParamTypeRef: TypeInfoPtr;
- Name: ShortString;
- end;
- TProcedureSignature =
- {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}
- packed
- {$endif FPC_REQUIRES_PROPER_ALIGNMENT}
- record
- private
- function GetResultType: PTypeInfo; inline;
- public
- property ResultType: PTypeInfo read GetResultType;
- public
- Flags: Byte;
- CC: TCallConv;
- ResultTypeRef: TypeInfoPtr;
- ParamCount: Byte;
- {Params: array[0..ParamCount - 1] of TProcedureParam;}
- function GetParam(ParamIndex: Integer): PProcedureParam;
- end;
- PTypeData = ^TTypeData;
- TTypeData =
- {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}
- packed
- {$endif FPC_REQUIRES_PROPER_ALIGNMENT}
- record
- private
- function GetBaseType: PTypeInfo; inline;
- function GetCompType: PTypeInfo; inline;
- function GetParentInfo: PTypeInfo; inline;
- function GetHelperParent: PTypeInfo; inline;
- function GetExtendedInfo: PTypeInfo; inline;
- function GetIntfParent: PTypeInfo; inline;
- function GetRawIntfParent: PTypeInfo; inline;
- function GetIIDStr: ShortString; inline;
- function GetElType: PTypeInfo; inline;
- function GetElType2: PTypeInfo; inline;
- function GetInstanceType: PTypeInfo; inline;
- function GetRefType: PTypeInfo; inline;
- public
- { tkEnumeration }
- property BaseType: PTypeInfo read GetBaseType;
- { tkSet }
- property CompType: PTypeInfo read GetCompType;
- { tkClass }
- property ParentInfo: PTypeInfo read GetParentInfo;
- { tkHelper }
- property HelperParent: PTypeInfo read GetHelperParent;
- property ExtendedInfo: PTypeInfo read GetExtendedInfo;
- { tkInterface }
- property IntfParent: PTypeInfo read GetIntfParent;
- { tkInterfaceRaw }
- property RawIntfParent: PTypeInfo read GetRawIntfParent;
- property IIDStr: ShortString read GetIIDStr;
- { tkDynArray }
- property ElType2: PTypeInfo read GetElType2;
- property ElType: PTypeInfo read GetElType;
- { tkClassRef }
- property InstanceType: PTypeInfo read GetInstanceType;
- { tkPointer }
- property RefType: PTypeInfo read GetRefType;
- public
- case TTypeKind of
- tkUnKnown,tkLString,tkWString,tkVariant,tkUString:
- ();
- tkAString:
- (CodePage: Word);
- tkInteger,tkChar,tkEnumeration,tkBool,tkWChar,tkSet:
- (OrdType : TOrdType;
- case TTypeKind of
- tkInteger,tkChar,tkEnumeration,tkBool,tkWChar : (
- MinValue,MaxValue : Longint;
- case TTypeKind of
- tkEnumeration:
- (
- BaseTypeRef : TypeInfoPtr;
- NameList : ShortString;
- {EnumUnitName: ShortString;})
- );
- tkSet:
- (CompTypeRef : TypeInfoPtr)
- );
- {$ifndef FPUNONE}
- tkFloat:
- (FloatType : TFloatType);
- {$endif}
- tkSString:
- (MaxLength : Byte);
- tkClass:
- (ClassType : TClass;
- ParentInfoRef : TypeInfoPtr;
- PropCount : SmallInt;
- UnitName : ShortString
- // here the properties follow as array of TPropInfo
- );
- tkRecord:
- (
- {$ifndef VER3_0}
- RecInitTable: PPointer;
- {$endif VER3_0}
- RecSize: Integer;
- ManagedFldCount: Integer;
- {ManagedFields: array[1..ManagedFldCount] of TManagedField}
- );
- tkHelper:
- (HelperParentRef : TypeInfoPtr;
- ExtendedInfoRef : TypeInfoPtr;
- HelperProps : SmallInt;
- HelperUnit : ShortString
- // here the properties follow as array of TPropInfo
- );
- tkMethod:
- (MethodKind : TMethodKind;
- ParamCount : Byte;
- ParamList : array[0..1023] of Char
- {in reality ParamList is a array[1..ParamCount] of:
- record
- Flags : TParamFlags;
- ParamName : ShortString;
- TypeName : ShortString;
- end;
- followed by
- ResultType : ShortString // for mkFunction, mkClassFunction only
- ResultTypeRef : PPTypeInfo; // for mkFunction, mkClassFunction only
- CC : TCallConv;
- ParamTypeRefs : array[1..ParamCount] of PPTypeInfo;}
- );
- tkProcVar:
- (ProcSig: TProcedureSignature);
- tkInt64:
- (MinInt64Value, MaxInt64Value: Int64);
- tkQWord:
- (MinQWordValue, MaxQWordValue: QWord);
- tkInterface:
- (
- IntfParentRef: TypeInfoPtr;
- IntfFlags : TIntfFlagsBase;
- GUID: TGUID;
- IntfUnit: ShortString;
- { here the properties follow as Word Count & array of TPropInfo }
- );
- tkInterfaceRaw:
- (
- RawIntfParentRef: TypeInfoPtr;
- RawIntfFlags : TIntfFlagsBase;
- IID: TGUID;
- RawIntfUnit: ShortString;
- { IIDStr: ShortString; }
- { here the properties follow as Word Count & array of TPropInfo }
- );
- tkArray:
- (ArrayData: TArrayTypeData);
- tkDynArray:
- (
- elSize : PtrUInt;
- elType2Ref : TypeInfoPtr;
- varType : Longint;
- elTypeRef : TypeInfoPtr;
- DynUnitName: ShortStringBase
- );
- tkClassRef:
- (InstanceTypeRef: TypeInfoPtr);
- tkPointer:
- (RefTypeRef: TypeInfoPtr);
- end;
- TPropData =
- {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}
- packed
- {$endif FPC_REQUIRES_PROPER_ALIGNMENT}
- record
- PropCount : Word;
- PropList : record _alignmentdummy : ptrint; end;
- end;
- {$PACKRECORDS 1}
- PPropInfo = ^TPropInfo;
- TPropInfo = packed record
- private
- function GetPropType: PTypeInfo; inline;
- public
- PropTypeRef : TypeInfoPtr;
- GetProc : CodePointer;
- SetProc : CodePointer;
- StoredProc : CodePointer;
- Index : Integer;
- Default : Longint;
- NameIndex : SmallInt;
- // contains the type of the Get/Set/Storedproc, see also ptxxx
- // bit 0..1 GetProc
- // 2..3 SetProc
- // 4..5 StoredProc
- // 6 : true, constant index property
- PropProcs : Byte;
- Name : ShortString;
- property PropType: PTypeInfo read GetPropType;
- end;
- TProcInfoProc = Procedure(PropInfo : PPropInfo) of object;
- PPropList = ^TPropList;
- TPropList = array[0..{$ifdef cpu16}(32768 div sizeof(PPropInfo))-2{$else}65535{$endif}] of PPropInfo;
- const
- tkString = tkSString;
- tkProcedure = tkProcVar; // for compatibility with Delphi
- tkAny = [Low(TTypeKind)..High(TTypeKind)];
- tkMethods = [tkMethod];
- tkProperties = tkAny-tkMethods-[tkUnknown];
- // general property handling
- Function GetTypeData(TypeInfo : PTypeInfo) : PTypeData;
- Function GetPropInfo(TypeInfo: PTypeInfo;const PropName: string): PPropInfo;
- Function GetPropInfo(TypeInfo: PTypeInfo;const PropName: string; AKinds: TTypeKinds): PPropInfo;
- Function GetPropInfo(Instance: TObject; const PropName: string): PPropInfo;
- Function GetPropInfo(Instance: TObject; const PropName: string; AKinds: TTypeKinds): PPropInfo;
- Function GetPropInfo(AClass: TClass; const PropName: string): PPropInfo;
- Function GetPropInfo(AClass: TClass; const PropName: string; AKinds: TTypeKinds): PPropInfo;
- Function FindPropInfo(Instance: TObject; const PropName: string): PPropInfo;
- Function FindPropInfo(Instance: TObject; const PropName: string; AKinds: TTypeKinds): PPropInfo;
- Function FindPropInfo(AClass: TClass; const PropName: string): PPropInfo;
- Function FindPropInfo(AClass: TClass; const PropName: string; AKinds: TTypeKinds): PPropInfo;
- Procedure GetPropInfos(TypeInfo: PTypeInfo; PropList: PPropList);
- Function GetPropList(TypeInfo: PTypeInfo; TypeKinds: TTypeKinds; PropList: PPropList; Sorted: boolean = true): longint;
- Function GetPropList(TypeInfo: PTypeInfo; out PropList: PPropList): SizeInt;
- function GetPropList(AClass: TClass; out PropList: PPropList): Integer;
- function GetPropList(Instance: TObject; out PropList: PPropList): Integer;
- // Property information routines.
- Function IsStoredProp(Instance: TObject;PropInfo : PPropInfo) : Boolean;
- Function IsStoredProp(Instance: TObject; const PropName: string): Boolean;
- Function IsPublishedProp(Instance: TObject; const PropName: string): Boolean;
- Function IsPublishedProp(AClass: TClass; const PropName: string): Boolean;
- Function PropType(Instance: TObject; const PropName: string): TTypeKind;
- Function PropType(AClass: TClass; const PropName: string): TTypeKind;
- Function PropIsType(Instance: TObject; const PropName: string; TypeKind: TTypeKind): Boolean;
- Function PropIsType(AClass: TClass; const PropName: string; TypeKind: TTypeKind): Boolean;
- // subroutines to read/write properties
- Function GetOrdProp(Instance: TObject; PropInfo : PPropInfo) : Int64;
- Function GetOrdProp(Instance: TObject; const PropName: string): Int64;
- Procedure SetOrdProp(Instance: TObject; PropInfo : PPropInfo; Value : Int64);
- Procedure SetOrdProp(Instance: TObject; const PropName: string; Value: Int64);
- Function GetEnumProp(Instance: TObject; const PropName: string): string;
- Function GetEnumProp(Instance: TObject; const PropInfo: PPropInfo): string;
- Procedure SetEnumProp(Instance: TObject; const PropName: string;const Value: string);
- Procedure SetEnumProp(Instance: TObject; const PropInfo: PPropInfo;const Value: string);
- Function GetSetProp(Instance: TObject; const PropName: string): string;
- Function GetSetProp(Instance: TObject; const PropName: string; Brackets: Boolean): string;
- Function GetSetProp(Instance: TObject; const PropInfo: PPropInfo; Brackets: Boolean): string;
- Procedure SetSetProp(Instance: TObject; const PropName: string; const Value: string);
- Procedure SetSetProp(Instance: TObject; const PropInfo: PPropInfo; const Value: string);
- Function GetStrProp(Instance: TObject; PropInfo : PPropInfo) : Ansistring;
- Function GetStrProp(Instance: TObject; const PropName: string): string;
- Procedure SetStrProp(Instance: TObject; const PropName: string; const Value: AnsiString);
- Procedure SetStrProp(Instance: TObject; PropInfo : PPropInfo; const Value : Ansistring);
- Function GetWideStrProp(Instance: TObject; PropInfo: PPropInfo): WideString;
- Function GetWideStrProp(Instance: TObject; const PropName: string): WideString;
- Procedure SetWideStrProp(Instance: TObject; const PropName: string; const Value: WideString);
- Procedure SetWideStrProp(Instance: TObject; PropInfo: PPropInfo; const Value: WideString);
- Function GetUnicodeStrProp(Instance: TObject; PropInfo: PPropInfo): UnicodeString;
- Function GetUnicodeStrProp(Instance: TObject; const PropName: string): UnicodeString;
- Procedure SetUnicodeStrProp(Instance: TObject; const PropName: string; const Value: UnicodeString);
- Procedure SetUnicodeStrProp(Instance: TObject; PropInfo: PPropInfo; const Value: UnicodeString);
- {$ifndef FPUNONE}
- Function GetFloatProp(Instance: TObject; PropInfo : PPropInfo) : Extended;
- Function GetFloatProp(Instance: TObject; const PropName: string): Extended;
- Procedure SetFloatProp(Instance: TObject; const PropName: string; Value: Extended);
- Procedure SetFloatProp(Instance: TObject; PropInfo : PPropInfo; Value : Extended);
- {$endif}
- Function GetObjectProp(Instance: TObject; const PropName: string): TObject;
- Function GetObjectProp(Instance: TObject; const PropName: string; MinClass: TClass): TObject;
- Function GetObjectProp(Instance: TObject; PropInfo: PPropInfo): TObject;
- Function GetObjectProp(Instance: TObject; PropInfo: PPropInfo; MinClass: TClass): TObject;
- Procedure SetObjectProp(Instance: TObject; const PropName: string; Value: TObject);
- Procedure SetObjectProp(Instance: TObject; PropInfo: PPropInfo; Value: TObject);
- Function GetObjectPropClass(Instance: TObject; const PropName: string): TClass;
- Function GetObjectPropClass(AClass: TClass; const PropName: string): TClass;
- Function GetMethodProp(Instance: TObject; PropInfo: PPropInfo) : TMethod;
- Function GetMethodProp(Instance: TObject; const PropName: string): TMethod;
- Procedure SetMethodProp(Instance: TObject; PropInfo: PPropInfo; const Value : TMethod);
- Procedure SetMethodProp(Instance: TObject; const PropName: string; const Value: TMethod);
- Function GetInt64Prop(Instance: TObject; PropInfo: PPropInfo): Int64;
- Function GetInt64Prop(Instance: TObject; const PropName: string): Int64;
- Procedure SetInt64Prop(Instance: TObject; PropInfo: PPropInfo; const Value: Int64);
- Procedure SetInt64Prop(Instance: TObject; const PropName: string; const Value: Int64);
- Function GetPropValue(Instance: TObject; const PropName: string): Variant;
- Function GetPropValue(Instance: TObject; const PropName: string; PreferStrings: Boolean): Variant;
- Function GetPropValue(Instance: TObject; PropInfo: PPropInfo): Variant;
- Function GetPropValue(Instance: TObject; PropInfo: PPropInfo; PreferStrings: Boolean): Variant;
- Procedure SetPropValue(Instance: TObject; const PropName: string; const Value: Variant);
- Procedure SetPropValue(Instance: TObject; PropInfo: PPropInfo; const Value: Variant);
- Function GetVariantProp(Instance: TObject; PropInfo : PPropInfo): Variant;
- Function GetVariantProp(Instance: TObject; const PropName: string): Variant;
- Procedure SetVariantProp(Instance: TObject; const PropName: string; const Value: Variant);
- Procedure SetVariantProp(Instance: TObject; PropInfo : PPropInfo; const Value: Variant);
- function GetInterfaceProp(Instance: TObject; const PropName: string): IInterface;
- function GetInterfaceProp(Instance: TObject; PropInfo: PPropInfo): IInterface;
- procedure SetInterfaceProp(Instance: TObject; const PropName: string; const Value: IInterface);
- procedure SetInterfaceProp(Instance: TObject; PropInfo: PPropInfo; const Value: IInterface);
- function GetRawInterfaceProp(Instance: TObject; const PropName: string): Pointer;
- function GetRawInterfaceProp(Instance: TObject; PropInfo: PPropInfo): Pointer;
- procedure SetRawInterfaceProp(Instance: TObject; const PropName: string; const Value: Pointer);
- procedure SetRawInterfaceProp(Instance: TObject; PropInfo: PPropInfo; const Value: Pointer);
- function GetDynArrayProp(Instance: TObject; const PropName: string): Pointer;
- function GetDynArrayProp(Instance: TObject; PropInfo: PPropInfo): Pointer;
- procedure SetDynArrayProp(Instance: TObject; const PropName: string; const Value: Pointer);
- procedure SetDynArrayProp(Instance: TObject; PropInfo: PPropInfo; const Value: Pointer);
- // Auxiliary routines, which may be useful
- Function GetEnumName(TypeInfo : PTypeInfo;Value : Integer) : string;
- Function GetEnumValue(TypeInfo : PTypeInfo;const Name : string) : Integer;
- function GetEnumNameCount(enum1: PTypeInfo): SizeInt;
- function SetToString(TypeInfo: PTypeInfo; Value: Integer; Brackets: Boolean) : String;
- function SetToString(PropInfo: PPropInfo; Value: Integer; Brackets: Boolean) : String;
- function SetToString(PropInfo: PPropInfo; Value: Integer) : String;
- function StringToSet(PropInfo: PPropInfo; const Value: string): Integer;
- function StringToSet(TypeInfo: PTypeInfo; const Value: string): Integer;
- const
- BooleanIdents: array[Boolean] of String = ('False', 'True');
- DotSep: String = '.';
- Type
- EPropertyError = Class(Exception);
- TGetPropValue = Function (Instance: TObject; PropInfo: PPropInfo; PreferStrings: Boolean) : Variant;
- TSetPropValue = Procedure (Instance: TObject; PropInfo: PPropInfo; const Value: Variant);
- TGetVariantProp = Function (Instance: TObject; PropInfo : PPropInfo): Variant;
- TSetVariantProp = Procedure (Instance: TObject; PropInfo : PPropInfo; const Value: Variant);
- EPropertyConvertError = class(Exception); // Not used (yet), but defined for compatibility.
- Const
- OnGetPropValue : TGetPropValue = Nil;
- OnSetPropValue : TSetPropValue = Nil;
- OnGetVariantprop : TGetVariantProp = Nil;
- OnSetVariantprop : TSetVariantProp = Nil;
- { for inlining }
- function DerefTypeInfoPtr(Info: TypeInfoPtr): PTypeInfo; inline;
- Implementation
- uses rtlconsts;
- type
- PMethod = ^TMethod;
- { ---------------------------------------------------------------------
- Auxiliary methods
- ---------------------------------------------------------------------}
- function aligntoptr(p : pointer) : pointer;inline;
- begin
- {$ifdef FPC_REQUIRES_PROPER_ALIGNMENT}
- result:=align(p,sizeof(p));
- {$else FPC_REQUIRES_PROPER_ALIGNMENT}
- result:=p;
- {$endif FPC_REQUIRES_PROPER_ALIGNMENT}
- end;
- function DerefTypeInfoPtr(Info: TypeInfoPtr): PTypeInfo; inline;
- begin
- {$ifdef ver3_0}
- Result := Info;
- {$else}
- if not Assigned(Info) then
- Result := Nil
- else
- Result := Info^;
- {$endif}
- end;
- Function GetEnumName(TypeInfo : PTypeInfo;Value : Integer) : string;
- Var PS : PShortString;
- PT : PTypeData;
- begin
- PT:=GetTypeData(TypeInfo);
- if TypeInfo^.Kind=tkBool then
- begin
- case Value of
- 0,1:
- Result:=BooleanIdents[Boolean(Value)];
- else
- Result:='';
- end;
- end
- else
- begin
- PS:=@PT^.NameList;
- dec(Value,PT^.MinValue);
- While Value>0 Do
- begin
- PS:=PShortString(pointer(PS)+PByte(PS)^+1);
- Dec(Value);
- end;
- Result:=PS^;
- end;
- end;
- Function GetEnumValue(TypeInfo : PTypeInfo;const Name : string) : Integer;
- Var PS : PShortString;
- PT : PTypeData;
- Count : longint;
- sName: shortstring;
- begin
- If Length(Name)=0 then
- exit(-1);
- sName := Name;
- PT:=GetTypeData(TypeInfo);
- Count:=0;
- Result:=-1;
- if TypeInfo^.Kind=tkBool then
- begin
- If CompareText(BooleanIdents[false],Name)=0 then
- result:=0
- else if CompareText(BooleanIdents[true],Name)=0 then
- result:=1;
- end
- else
- begin
- PS:=@PT^.NameList;
- While (Result=-1) and (PByte(PS)^<>0) do
- begin
- If ShortCompareText(PS^, sName) = 0 then
- Result:=Count+PT^.MinValue;
- PS:=PShortString(pointer(PS)+PByte(PS)^+1);
- Inc(Count);
- end;
- end;
- end;
- function GetEnumNameCount(enum1: PTypeInfo): SizeInt;
- var
- PS: PShortString;
- PT: PTypeData;
- Count: SizeInt;
- begin
- PT:=GetTypeData(enum1);
- if enum1^.Kind=tkBool then
- Result:=2
- else
- begin
- Count:=0;
- Result:=0;
- PS:=@PT^.NameList;
- While (PByte(PS)^<>0) do
- begin
- PS:=PShortString(pointer(PS)+PByte(PS)^+1);
- Inc(Count);
- end;
- { the last string is the unit name }
- Result := Count - 1;
- end;
- end;
- Function SetToString(PropInfo: PPropInfo; Value: Integer; Brackets: Boolean) : String;
- begin
- Result:=SetToString(PropInfo^.PropType,Value,Brackets);
- end;
- Function SetToString(TypeInfo: PTypeInfo; Value: Integer; Brackets: Boolean) : String;
- type
- tsetarr = bitpacked array[0..SizeOf(Integer)*8-1] of 0..1;
- Var
- I : Integer;
- PTI : PTypeInfo;
- begin
- {$if defined(FPC_BIG_ENDIAN)}
- { On big endian systems, set element 0 is in the most significant bit,
- and the same goes for the elements of bitpacked arrays there. }
- case GetTypeData(TypeInfo)^.OrdType of
- otSByte,otUByte: Value:=Value shl (SizeOf(Integer)*8-8);
- otSWord,otUWord: Value:=Value shl (SizeOf(Integer)*8-16);
- end;
- {$endif}
- PTI:=GetTypeData(TypeInfo)^.CompType;
- Result:='';
- For I:=0 to SizeOf(Integer)*8-1 do
- begin
- if (tsetarr(Value)[i]<>0) then
- begin
- If Result='' then
- Result:=GetEnumName(PTI,i)
- else
- Result:=Result+','+GetEnumName(PTI,I);
- end;
- end;
- if Brackets then
- Result:='['+Result+']';
- end;
- Function SetToString(PropInfo: PPropInfo; Value: Integer) : String;
- begin
- Result:=SetToString(PropInfo,Value,False);
- end;
- Const
- SetDelim = ['[',']',',',' '];
- Function GetNextElement(Var S : String) : String;
- Var
- J : Integer;
- begin
- J:=1;
- Result:='';
- If Length(S)>0 then
- begin
- While (J<=Length(S)) and Not (S[j] in SetDelim) do
- Inc(j);
- Result:=Copy(S,1,j-1);
- Delete(S,1,j);
- end;
- end;
- Function StringToSet(PropInfo: PPropInfo; const Value: string): Integer;
- begin
- Result:=StringToSet(PropInfo^.PropType,Value);
- end;
- Function StringToSet(TypeInfo: PTypeInfo; const Value: string): Integer;
- Var
- S,T : String;
- I : Integer;
- PTI : PTypeInfo;
- begin
- Result:=0;
- PTI:=GetTypeData(TypeInfo)^.Comptype;
- S:=Value;
- I:=1;
- If Length(S)>0 then
- begin
- While (I<=Length(S)) and (S[i] in SetDelim) do
- Inc(I);
- Delete(S,1,i-1);
- end;
- While (S<>'') do
- begin
- T:=GetNextElement(S);
- if T<>'' then
- begin
- I:=GetEnumValue(PTI,T);
- if (I<0) then
- raise EPropertyError.CreateFmt(SErrUnknownEnumValue, [T]);
- Result:=Result or (1 shl i);
- end;
- end;
- end;
- Function GetTypeData(TypeInfo : PTypeInfo) : PTypeData;
- begin
- GetTypeData:=PTypeData(aligntoptr(PTypeData(pointer(TypeInfo)+2+PByte(pointer(TypeInfo)+1)^)));
- end;
- { ---------------------------------------------------------------------
- Basic Type information functions.
- ---------------------------------------------------------------------}
- Function GetPropInfo(TypeInfo : PTypeInfo;const PropName : string) : PPropInfo;
- var
- hp : PTypeData;
- i : longint;
- p : shortstring;
- pd : ^TPropData;
- begin
- P:=PropName; // avoid Ansi<->short conversion in a loop
- while Assigned(TypeInfo) do
- begin
- // skip the name
- hp:=GetTypeData(Typeinfo);
- // the class info rtti the property rtti follows immediatly
- pd:=aligntoptr(pointer(pointer(@hp^.UnitName)+Length(hp^.UnitName)+1));
- Result:=PPropInfo(@pd^.PropList);
- for i:=1 to pd^.PropCount do
- begin
- // found a property of that name ?
- if ShortCompareText(Result^.Name, P) = 0 then
- exit;
- // skip to next property
- Result:=PPropInfo(aligntoptr(pointer(@Result^.Name)+byte(Result^.Name[0])+1));
- end;
- // parent class
- Typeinfo:=hp^.ParentInfo;
- end;
- Result:=Nil;
- end;
- Function GetPropInfo(TypeInfo : PTypeInfo;const PropName : string; Akinds : TTypeKinds) : PPropInfo;
- begin
- Result:=GetPropInfo(TypeInfo,PropName);
- If (Akinds<>[]) then
- If (Result<>Nil) then
- If Not (Result^.PropType^.Kind in AKinds) then
- Result:=Nil;
- end;
- Function GetPropInfo(AClass: TClass; const PropName: string; AKinds: TTypeKinds) : PPropInfo;
- begin
- Result:=GetPropInfo(PTypeInfo(AClass.ClassInfo),PropName,AKinds);
- end;
- Function GetPropInfo(Instance: TObject; const PropName: string; AKinds: TTypeKinds) : PPropInfo;
- begin
- Result:=GetPropInfo(Instance.ClassType,PropName,AKinds);
- end;
- Function GetPropInfo(Instance: TObject; const PropName: string): PPropInfo;
- begin
- Result:=GetPropInfo(Instance,PropName,[]);
- end;
- Function GetPropInfo(AClass: TClass; const PropName: string): PPropInfo;
- begin
- Result:=GetPropInfo(AClass,PropName,[]);
- end;
- Function FindPropInfo(Instance: TObject; const PropName: string): PPropInfo;
- begin
- result:=GetPropInfo(Instance, PropName);
- if Result=nil then
- Raise EPropertyError.CreateFmt(SErrPropertyNotFound, [PropName]);
- end;
- Function FindPropInfo(Instance: TObject; const PropName: string; AKinds: TTypeKinds): PPropInfo;
- begin
- result:=GetPropInfo(Instance, PropName, AKinds);
- if Result=nil then
- Raise EPropertyError.CreateFmt(SErrPropertyNotFound, [PropName]);
- end;
- Function FindPropInfo(AClass: TClass; const PropName: string): PPropInfo;
- begin
- result:=GetPropInfo(AClass, PropName);
- if result=nil then
- Raise EPropertyError.CreateFmt(SErrPropertyNotFound, [PropName]);
- end;
- Function FindPropInfo(AClass: TClass; const PropName: string; AKinds: TTypeKinds): PPropInfo;
- begin
- result:=GetPropInfo(AClass, PropName, AKinds);
- if result=nil then
- Raise EPropertyError.CreateFmt(SErrPropertyNotFound, [PropName]);
- end;
- Function IsStoredProp(Instance : TObject;PropInfo : PPropInfo) : Boolean;
- type
- TBooleanIndexFunc=function(Index:integer):boolean of object;
- TBooleanFunc=function:boolean of object;
- var
- AMethod : TMethod;
- begin
- case (PropInfo^.PropProcs shr 4) and 3 of
- ptField:
- Result:=PBoolean(Pointer(Instance)+PtrUInt(PropInfo^.StoredProc))^;
- ptConst:
- Result:=LongBool(PropInfo^.StoredProc);
- ptStatic,
- ptVirtual:
- begin
- if (PropInfo^.PropProcs shr 4) and 3=ptstatic then
- AMethod.Code:=PropInfo^.StoredProc
- else
- AMethod.Code:=pcodepointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.StoredProc))^;
- AMethod.Data:=Instance;
- if ((PropInfo^.PropProcs shr 6) and 1)<>0 then
- Result:=TBooleanIndexFunc(AMethod)(PropInfo^.Index)
- else
- Result:=TBooleanFunc(AMethod)();
- end;
- end;
- end;
- Procedure GetPropInfos(TypeInfo : PTypeInfo;PropList : PPropList);
- {
- Store Pointers to property information in the list pointed
- to by proplist. PRopList must contain enough space to hold ALL
- properties.
- }
- Var
- TD : PTypeData;
- TP : PPropInfo;
- Count : Longint;
- begin
- // Get this objects TOTAL published properties count
- TD:=GetTypeData(TypeInfo);
- // Clear list
- FillChar(PropList^,TD^.PropCount*sizeof(Pointer),0);
- repeat
- TD:=GetTypeData(TypeInfo);
- // published properties count for this object
- TP:=aligntoptr(PPropInfo(aligntoptr((Pointer(@TD^.UnitName)+Length(TD^.UnitName)+1))));
- Count:=PWord(TP)^;
- // Now point TP to first propinfo record.
- Inc(Pointer(TP),SizeOF(Word));
- tp:=aligntoptr(tp);
- While Count>0 do
- begin
- // Don't overwrite properties with the same name
- if PropList^[TP^.NameIndex]=nil then
- PropList^[TP^.NameIndex]:=TP;
- // Point to TP next propinfo record.
- // Located at Name[Length(Name)+1] !
- TP:=aligntoptr(PPropInfo(pointer(@TP^.Name)+PByte(@TP^.Name)^+1));
- Dec(Count);
- end;
- TypeInfo:=TD^.Parentinfo;
- until TypeInfo=nil;
- end;
- Procedure InsertProp (PL : PProplist;PI : PPropInfo; Count : longint);
- Var
- I : Longint;
- begin
- I:=0;
- While (I<Count) and (PI^.Name>PL^[I]^.Name) do
- Inc(I);
- If I<Count then
- Move(PL^[I], PL^[I+1], (Count - I) * SizeOf(Pointer));
- PL^[I]:=PI;
- end;
- Procedure InsertPropnosort (PL : PProplist;PI : PPropInfo; Count : longint);
- begin
- PL^[Count]:=PI;
- end;
- Type TInsertProp = Procedure (PL : PProplist;PI : PPropInfo; Count : longint);
- //Const InsertProps : array[false..boolean] of TInsertProp = (InsertPropNoSort,InsertProp);
- Function GetPropList(TypeInfo : PTypeInfo;TypeKinds : TTypeKinds; PropList : PPropList;Sorted : boolean = true):longint;
- {
- Store Pointers to property information OF A CERTAIN KIND in the list pointed
- to by proplist. PRopList must contain enough space to hold ALL
- properties.
- }
- Var
- TempList : PPropList;
- PropInfo : PPropinfo;
- I,Count : longint;
- DoInsertProp : TInsertProp;
- begin
- if sorted then
- DoInsertProp:=@InsertProp
- else
- DoInsertProp:=@InsertPropnosort;
- Result:=0;
- Count:=GetTypeData(TypeInfo)^.Propcount;
- If Count>0 then
- begin
- GetMem(TempList,Count*SizeOf(Pointer));
- Try
- GetPropInfos(TypeInfo,TempList);
- For I:=0 to Count-1 do
- begin
- PropInfo:=TempList^[i];
- If PropInfo^.PropType^.Kind in TypeKinds then
- begin
- If (PropList<>Nil) then
- DoInsertProp(PropList,PropInfo,Result);
- Inc(Result);
- end;
- end;
- finally
- FreeMem(TempList,Count*SizeOf(Pointer));
- end;
- end;
- end;
- Function GetPropList(TypeInfo: PTypeInfo; out PropList: PPropList): SizeInt;
- begin
- result:=GetTypeData(TypeInfo)^.Propcount;
- if result>0 then
- begin
- getmem(PropList,result*sizeof(pointer));
- GetPropInfos(TypeInfo,PropList);
- end
- else
- PropList:=Nil;
- end;
- function GetPropList(AClass: TClass; out PropList: PPropList): Integer;
- begin
- Result := GetPropList(PTypeInfo(AClass.ClassInfo), PropList);
- end;
- function GetPropList(Instance: TObject; out PropList: PPropList): Integer;
- begin
- Result := GetPropList(Instance.ClassType, PropList);
- end;
- { ---------------------------------------------------------------------
- Property access functions
- ---------------------------------------------------------------------}
- { ---------------------------------------------------------------------
- Ordinal properties
- ---------------------------------------------------------------------}
- Function GetOrdProp(Instance : TObject;PropInfo : PPropInfo) : Int64;
- type
- TGetInt64ProcIndex=function(index:longint):Int64 of object;
- TGetInt64Proc=function():Int64 of object;
- TGetIntegerProcIndex=function(index:longint):longint of object;
- TGetIntegerProc=function:longint of object;
- TGetWordProcIndex=function(index:longint):word of object;
- TGetWordProc=function:word of object;
- TGetByteProcIndex=function(index:longint):Byte of object;
- TGetByteProc=function:Byte of object;
- var
- TypeInfo: PTypeInfo;
- AMethod : TMethod;
- DataSize: Integer;
- OrdType: TOrdType;
- Signed: Boolean;
- begin
- Result:=0;
- TypeInfo := PropInfo^.PropType;
- Signed := false;
- DataSize := 4;
- case TypeInfo^.Kind of
- {$ifdef cpu64}
- tkInterface,
- tkInterfaceRaw,
- tkDynArray,
- tkClass:
- DataSize:=8;
- {$endif cpu64}
- tkChar, tkBool:
- DataSize:=1;
- tkWChar:
- DataSize:=2;
- tkSet,
- tkEnumeration,
- tkInteger:
- begin
- OrdType:=GetTypeData(TypeInfo)^.OrdType;
- case OrdType of
- otSByte,otUByte: DataSize := 1;
- otSWord,otUWord: DataSize := 2;
- end;
- Signed := OrdType in [otSByte,otSWord,otSLong];
- end;
- tkInt64 :
- begin
- DataSize:=8;
- Signed:=true;
- end;
- tkQword :
- begin
- DataSize:=8;
- Signed:=false;
- end;
- end;
- case (PropInfo^.PropProcs) and 3 of
- ptField:
- if Signed then begin
- case DataSize of
- 1: Result:=PShortInt(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;
- 2: Result:=PSmallInt(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;
- 4: Result:=PLongint(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;
- 8: Result:=PInt64(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;
- end;
- end else begin
- case DataSize of
- 1: Result:=PByte(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;
- 2: Result:=PWord(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;
- 4: Result:=PLongint(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;
- 8: Result:=PInt64(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;
- end;
- end;
- ptStatic,
- ptVirtual:
- begin
- if (PropInfo^.PropProcs and 3)=ptStatic then
- AMethod.Code:=PropInfo^.GetProc
- else
- AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;
- AMethod.Data:=Instance;
- if ((PropInfo^.PropProcs shr 6) and 1)<>0 then begin
- case DataSize of
- 1: Result:=TGetByteProcIndex(AMethod)(PropInfo^.Index);
- 2: Result:=TGetWordProcIndex(AMethod)(PropInfo^.Index);
- 4: Result:=TGetIntegerProcIndex(AMethod)(PropInfo^.Index);
- 8: result:=TGetInt64ProcIndex(AMethod)(PropInfo^.Index)
- end;
- end else begin
- case DataSize of
- 1: Result:=TGetByteProc(AMethod)();
- 2: Result:=TGetWordProc(AMethod)();
- 4: Result:=TGetIntegerProc(AMethod)();
- 8: result:=TGetInt64Proc(AMethod)();
- end;
- end;
- if Signed then begin
- case DataSize of
- 1: Result:=ShortInt(Result);
- 2: Result:=SmallInt(Result);
- end;
- end;
- end;
- end;
- end;
- Procedure SetOrdProp(Instance : TObject;PropInfo : PPropInfo;Value : Int64);
- type
- TSetInt64ProcIndex=procedure(index:longint;i:Int64) of object;
- TSetInt64Proc=procedure(i:Int64) of object;
- TSetIntegerProcIndex=procedure(index,i:longint) of object;
- TSetIntegerProc=procedure(i:longint) of object;
- var
- DataSize: Integer;
- AMethod : TMethod;
- begin
- if PropInfo^.PropType^.Kind in [tkInt64,tkQword
- { why do we have to handle classes here, see also below? (FK) }
- {$ifdef cpu64}
- ,tkInterface
- ,tkInterfaceRaw
- ,tkDynArray
- ,tkClass
- {$endif cpu64}
- ] then
- DataSize := 8
- else
- DataSize := 4;
- if not(PropInfo^.PropType^.Kind in [tkInt64,tkQword,tkClass,tkInterface,tkInterfaceRaw,tkDynArray]) then
- begin
- { cut off unnecessary stuff }
- case GetTypeData(PropInfo^.PropType)^.OrdType of
- otSWord,otUWord:
- begin
- Value:=Value and $ffff;
- DataSize := 2;
- end;
- otSByte,otUByte:
- begin
- Value:=Value and $ff;
- DataSize := 1;
- end;
- end;
- end;
- case (PropInfo^.PropProcs shr 2) and 3 of
- ptField:
- case DataSize of
- 1: PByte(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Byte(Value);
- 2: PWord(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Word(Value);
- 4: PLongint(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Longint(Value);
- 8: PInt64(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Value;
- end;
- ptStatic,
- ptVirtual:
- begin
- if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then
- AMethod.Code:=PropInfo^.SetProc
- else
- AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;
- AMethod.Data:=Instance;
- if datasize=8 then
- begin
- if ((PropInfo^.PropProcs shr 6) and 1)<>0 then
- TSetInt64ProcIndex(AMethod)(PropInfo^.Index,Value)
- else
- TSetInt64Proc(AMethod)(Value);
- end
- else
- begin
- if ((PropInfo^.PropProcs shr 6) and 1)<>0 then
- TSetIntegerProcIndex(AMethod)(PropInfo^.Index,Value)
- else
- TSetIntegerProc(AMethod)(Value);
- end;
- end;
- end;
- end;
- Function GetOrdProp(Instance: TObject; const PropName: string): Int64;
- begin
- Result:=GetOrdProp(Instance,FindPropInfo(Instance,PropName));
- end;
- Procedure SetOrdProp(Instance: TObject; const PropName: string; Value: Int64);
- begin
- SetOrdProp(Instance,FindPropInfo(Instance,PropName),Value);
- end;
- Function GetEnumProp(Instance: TObject; Const PropInfo: PPropInfo): string;
- begin
- Result:=GetEnumName(PropInfo^.PropType, GetOrdProp(Instance, PropInfo));
- end;
- Function GetEnumProp(Instance: TObject; const PropName: string): string;
- begin
- Result:=GetEnumProp(Instance,FindPropInfo(Instance,PropName));
- end;
- Procedure SetEnumProp(Instance: TObject; const PropName: string; const Value: string);
- begin
- SetEnumProp(Instance,FindPropInfo(Instance,PropName),Value);
- end;
- Procedure SetEnumProp(Instance: TObject; Const PropInfo : PPropInfo; const Value: string);
- Var
- PV : Longint;
- begin
- If PropInfo<>Nil then
- begin
- PV:=GetEnumValue(PropInfo^.PropType, Value);
- if (PV<0) then
- raise EPropertyError.CreateFmt(SErrUnknownEnumValue, [Value]);
- SetOrdProp(Instance, PropInfo,PV);
- end;
- end;
- { ---------------------------------------------------------------------
- Int64 wrappers
- ---------------------------------------------------------------------}
- Function GetInt64Prop(Instance: TObject; PropInfo: PPropInfo): Int64;
- begin
- Result:=GetOrdProp(Instance,PropInfo);
- end;
- procedure SetInt64Prop(Instance: TObject; PropInfo: PPropInfo; const Value: Int64);
- begin
- SetOrdProp(Instance,PropInfo,Value);
- end;
- Function GetInt64Prop(Instance: TObject; const PropName: string): Int64;
- begin
- Result:=GetInt64Prop(Instance,FindPropInfo(Instance,PropName));
- end;
- Procedure SetInt64Prop(Instance: TObject; const PropName: string; const Value: Int64);
- begin
- SetInt64Prop(Instance,FindPropInfo(Instance,PropName),Value);
- end;
- { ---------------------------------------------------------------------
- Set properties
- ---------------------------------------------------------------------}
- Function GetSetProp(Instance: TObject; const PropName: string): string;
- begin
- Result:=GetSetProp(Instance,PropName,False);
- end;
- Function GetSetProp(Instance: TObject; const PropName: string; Brackets: Boolean): string;
- begin
- Result:=GetSetProp(Instance,FindPropInfo(Instance,PropName),Brackets);
- end;
- Function GetSetProp(Instance: TObject; const PropInfo: PPropInfo; Brackets: Boolean): string;
- begin
- Result:=SetToString(PropInfo,GetOrdProp(Instance,PropInfo),Brackets);
- end;
- Procedure SetSetProp(Instance: TObject; const PropName: string; const Value: string);
- begin
- SetSetProp(Instance,FindPropInfo(Instance,PropName),Value);
- end;
- Procedure SetSetProp(Instance: TObject; const PropInfo: PPropInfo; const Value: string);
- begin
- SetOrdProp(Instance,PropInfo,StringToSet(PropInfo,Value));
- end;
- { ---------------------------------------------------------------------
- Object properties
- ---------------------------------------------------------------------}
- Function GetObjectProp(Instance: TObject; const PropName: string): TObject;
- begin
- Result:=GetObjectProp(Instance,PropName,Nil);
- end;
- Function GetObjectProp(Instance: TObject; const PropName: string; MinClass: TClass): TObject;
- begin
- Result:=GetObjectProp(Instance,FindPropInfo(Instance,PropName),MinClass);
- end;
- Function GetObjectProp(Instance: TObject; PropInfo : PPropInfo): TObject;
- begin
- Result:=GetObjectProp(Instance,PropInfo,Nil);
- end;
- Function GetObjectProp(Instance: TObject; PropInfo : PPropInfo; MinClass: TClass): TObject;
- begin
- {$ifdef cpu64}
- Result:=TObject(GetInt64Prop(Instance,PropInfo));
- {$else cpu64}
- Result:=TObject(PtrInt(GetOrdProp(Instance,PropInfo)));
- {$endif cpu64}
- If (MinClass<>Nil) and (Result<>Nil) Then
- If Not Result.InheritsFrom(MinClass) then
- Result:=Nil;
- end;
- Procedure SetObjectProp(Instance: TObject; const PropName: string; Value: TObject);
- begin
- SetObjectProp(Instance,FindPropInfo(Instance,PropName),Value);
- end;
- Procedure SetObjectProp(Instance: TObject; PropInfo : PPropInfo; Value: TObject);
- begin
- {$ifdef cpu64}
- SetInt64Prop(Instance,PropInfo,Int64(Value));
- {$else cpu64}
- SetOrdProp(Instance,PropInfo,PtrInt(Value));
- {$endif cpu64}
- end;
- Function GetObjectPropClass(Instance: TObject; const PropName: string): TClass;
- begin
- Result:=GetTypeData(FindPropInfo(Instance,PropName,[tkClass])^.PropType)^.ClassType;
- end;
- Function GetObjectPropClass(AClass: TClass; const PropName: string): TClass;
- begin
- Result:=GetTypeData(FindPropInfo(AClass,PropName,[tkClass])^.PropType)^.ClassType;
- end;
- { ---------------------------------------------------------------------
- Interface wrapprers
- ---------------------------------------------------------------------}
- function GetInterfaceProp(Instance: TObject; const PropName: string): IInterface;
- begin
- Result:=GetInterfaceProp(Instance,FindPropInfo(Instance,PropName));
- end;
- function GetInterfaceProp(Instance: TObject; PropInfo: PPropInfo): IInterface;
- type
- TGetInterfaceProc=function:IInterface of object;
- TGetInterfaceProcIndex=function(index:longint):IInterface of object;
- var
- AMethod : TMethod;
- begin
- Result:=nil;
- case (PropInfo^.PropProcs) and 3 of
- ptField:
- Result:=IInterface(PPointer(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^);
- ptStatic,
- ptVirtual:
- begin
- if (PropInfo^.PropProcs and 3)=ptStatic then
- AMethod.Code:=PropInfo^.GetProc
- else
- AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;
- AMethod.Data:=Instance;
- if ((PropInfo^.PropProcs shr 6) and 1)<>0 then
- Result:=TGetInterfaceProcIndex(AMethod)(PropInfo^.Index)
- else
- Result:=TGetInterfaceProc(AMethod)();
- end;
- end;
- end;
- procedure SetInterfaceProp(Instance: TObject; const PropName: string; const Value: IInterface);
- begin
- SetInterfaceProp(Instance,FindPropInfo(Instance,PropName),Value);
- end;
- procedure SetInterfaceProp(Instance: TObject; PropInfo: PPropInfo; const Value: IInterface);
- type
- TSetIntfStrProcIndex=procedure(index:longint;const i:IInterface) of object;
- TSetIntfStrProc=procedure(i:IInterface) of object;
- var
- AMethod : TMethod;
- begin
- case Propinfo^.PropType^.Kind of
- tkInterface:
- begin
- case (PropInfo^.PropProcs shr 2) and 3 of
- ptField:
- PInterface(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Value;
- ptStatic,
- ptVirtual:
- begin
- if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then
- AMethod.Code:=PropInfo^.SetProc
- else
- AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;
- AMethod.Data:=Instance;
- if ((PropInfo^.PropProcs shr 6) and 1)<>0 then
- TSetIntfStrProcIndex(AMethod)(PropInfo^.Index,Value)
- else
- TSetIntfStrProc(AMethod)(Value);
- end;
- end;
- end;
- tkInterfaceRaw:
- Raise Exception.Create('Cannot set RAW interface from IUnknown interface');
- end;
- end;
- { ---------------------------------------------------------------------
- RAW (Corba) Interface wrapprers
- ---------------------------------------------------------------------}
- function GetRawInterfaceProp(Instance: TObject; const PropName: string): Pointer;
- begin
- Result:=GetRawInterfaceProp(Instance,FindPropInfo(Instance,PropName));
- end;
- function GetRawInterfaceProp(Instance: TObject; PropInfo: PPropInfo): Pointer;
- begin
- {$ifdef cpu64}
- Result:=Pointer(GetInt64Prop(Instance,PropInfo));
- {$else cpu64}
- Result:=Pointer(PtrInt(GetOrdProp(Instance,PropInfo)));
- {$endif cpu64}
- end;
- procedure SetRawInterfaceProp(Instance: TObject; const PropName: string; const Value: Pointer);
- begin
- SetRawInterfaceProp(Instance,FindPropInfo(Instance,PropName),Value);
- end;
- procedure SetRawInterfaceProp(Instance: TObject; PropInfo: PPropInfo; const Value: Pointer);
- type
- TSetPointerProcIndex=procedure(index:longint;const i:Pointer) of object;
- TSetPointerProc=procedure(i:Pointer) of object;
- var
- AMethod : TMethod;
- begin
- case Propinfo^.PropType^.Kind of
- tkInterfaceRaw:
- begin
- case (PropInfo^.PropProcs shr 2) and 3 of
- ptField:
- PPointer(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Value;
- ptStatic,
- ptVirtual:
- begin
- if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then
- AMethod.Code:=PropInfo^.SetProc
- else
- AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;
- AMethod.Data:=Instance;
- if ((PropInfo^.PropProcs shr 6) and 1)<>0 then
- TSetPointerProcIndex(AMethod)(PropInfo^.Index,Value)
- else
- TSetPointerProc(AMethod)(Value);
- end;
- end;
- end;
- tkInterface:
- Raise Exception.Create('Cannot set interface from RAW interface');
- end;
- end;
- { ---------------------------------------------------------------------
- Dynamic array properties
- ---------------------------------------------------------------------}
- function GetDynArrayProp(Instance: TObject; const PropName: string): Pointer;
- begin
- Result:=GetDynArrayProp(Instance,FindPropInfo(Instance,PropName));
- end;
- function GetDynArrayProp(Instance: TObject; PropInfo: PPropInfo): Pointer;
- type
- { we need a dynamic array as that type is usually passed differently from
- a plain pointer }
- TDynArray=array of Byte;
- TGetDynArrayProc=function:TDynArray of object;
- TGetDynArrayProcIndex=function(index:longint):TDynArray of object;
- var
- AMethod : TMethod;
- begin
- Result:=nil;
- if PropInfo^.PropType^.Kind<>tkDynArray then
- Exit;
- case (PropInfo^.PropProcs) and 3 of
- ptField:
- Result:=PPointer(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;
- ptStatic,
- ptVirtual:
- begin
- if (PropInfo^.PropProcs and 3)=ptStatic then
- AMethod.Code:=PropInfo^.GetProc
- else
- AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;
- AMethod.Data:=Instance;
- if ((PropInfo^.PropProcs shr 6) and 1)<>0 then
- Result:=Pointer(TGetDynArrayProcIndex(AMethod)(PropInfo^.Index))
- else
- Result:=Pointer(TGetDynArrayProc(AMethod)());
- end;
- end;
- end;
- procedure SetDynArrayProp(Instance: TObject; const PropName: string; const Value: Pointer);
- begin
- SetDynArrayProp(Instance,FindPropInfo(Instance,PropName),Value);
- end;
- procedure SetDynArrayProp(Instance: TObject; PropInfo: PPropInfo; const Value: Pointer);
- type
- { we need a dynamic array as that type is usually passed differently from
- a plain pointer }
- TDynArray=array of Byte;
- TSetDynArrayProcIndex=procedure(index:longint;const i:TDynArray) of object;
- TSetDynArrayProc=procedure(i:TDynArray) of object;
- var
- AMethod: TMethod;
- begin
- if PropInfo^.PropType^.Kind<>tkDynArray then
- Exit;
- case (PropInfo^.PropProcs shr 2) and 3 of
- ptField:
- CopyArray(PPointer(Pointer(Instance)+PtrUInt(PropInfo^.SetProc)), @Value, PropInfo^.PropType, 1);
- ptStatic,
- ptVirtual:
- begin
- if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then
- AMethod.Code:=PropInfo^.SetProc
- else
- AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;
- AMethod.Data:=Instance;
- if ((PropInfo^.PropProcs shr 6) and 1)<>0 then
- TSetDynArrayProcIndex(AMethod)(PropInfo^.Index,TDynArray(Value))
- else
- TSetDynArrayProc(AMethod)(TDynArray(Value));
- end;
- end;
- end;
- { ---------------------------------------------------------------------
- String properties
- ---------------------------------------------------------------------}
- Function GetStrProp(Instance: TObject; PropInfo: PPropInfo): AnsiString;
- type
- TGetShortStrProcIndex=function(index:longint):ShortString of object;
- TGetShortStrProc=function():ShortString of object;
- TGetAnsiStrProcIndex=function(index:longint):AnsiString of object;
- TGetAnsiStrProc=function():AnsiString of object;
- var
- AMethod : TMethod;
- begin
- Result:='';
- case Propinfo^.PropType^.Kind of
- tkWString:
- Result:=AnsiString(GetWideStrProp(Instance,PropInfo));
- tkUString:
- Result := AnsiString(GetUnicodeStrProp(Instance,PropInfo));
- tkSString:
- begin
- case (PropInfo^.PropProcs) and 3 of
- ptField:
- Result := PShortString(Pointer(Instance) + LongWord(PropInfo^.GetProc))^;
- ptStatic,
- ptVirtual:
- begin
- if (PropInfo^.PropProcs and 3)=ptStatic then
- AMethod.Code:=PropInfo^.GetProc
- else
- AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;
- AMethod.Data:=Instance;
- if ((PropInfo^.PropProcs shr 6) and 1)<>0 then
- Result:=TGetShortStrProcIndex(AMethod)(PropInfo^.Index)
- else
- Result:=TGetShortStrProc(AMethod)();
- end;
- end;
- end;
- tkAString:
- begin
- case (PropInfo^.PropProcs) and 3 of
- ptField:
- Result := PAnsiString(Pointer(Instance) + LongWord(PropInfo^.GetProc))^;
- ptStatic,
- ptVirtual:
- begin
- if (PropInfo^.PropProcs and 3)=ptStatic then
- AMethod.Code:=PropInfo^.GetProc
- else
- AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;
- AMethod.Data:=Instance;
- if ((PropInfo^.PropProcs shr 6) and 1)<>0 then
- Result:=TGetAnsiStrProcIndex(AMethod)(PropInfo^.Index)
- else
- Result:=TGetAnsiStrProc(AMethod)();
- end;
- end;
- end;
- end;
- end;
- Procedure SetStrProp(Instance : TObject;PropInfo : PPropInfo; const Value : AnsiString);
- type
- TSetShortStrProcIndex=procedure(index:longint;const s:ShortString) of object;
- TSetShortStrProc=procedure(const s:ShortString) of object;
- TSetAnsiStrProcIndex=procedure(index:longint;s:AnsiString) of object;
- TSetAnsiStrProc=procedure(s:AnsiString) of object;
- var
- AMethod : TMethod;
- begin
- case Propinfo^.PropType^.Kind of
- tkWString:
- SetWideStrProp(Instance,PropInfo,WideString(Value));
- tkUString:
- SetUnicodeStrProp(Instance,PropInfo,UnicodeString(Value));
- tkSString:
- begin
- case (PropInfo^.PropProcs shr 2) and 3 of
- ptField:
- PShortString(Pointer(Instance) + LongWord(PropInfo^.SetProc))^:=Value;
- ptStatic,
- ptVirtual:
- begin
- if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then
- AMethod.Code:=PropInfo^.SetProc
- else
- AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;
- AMethod.Data:=Instance;
- if ((PropInfo^.PropProcs shr 6) and 1)<>0 then
- TSetShortStrProcIndex(AMethod)(PropInfo^.Index,Value)
- else
- TSetShortStrProc(AMethod)(Value);
- end;
- end;
- end;
- tkAString:
- begin
- case (PropInfo^.PropProcs shr 2) and 3 of
- ptField:
- PAnsiString(Pointer(Instance) + LongWord(PropInfo^.SetProc))^:=Value;
- ptStatic,
- ptVirtual:
- begin
- if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then
- AMethod.Code:=PropInfo^.SetProc
- else
- AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;
- AMethod.Data:=Instance;
- if ((PropInfo^.PropProcs shr 6) and 1)<>0 then
- TSetAnsiStrProcIndex(AMethod)(PropInfo^.Index,Value)
- else
- TSetAnsiStrProc(AMethod)(Value);
- end;
- end;
- end;
- end;
- end;
- Function GetStrProp(Instance: TObject; const PropName: string): string;
- begin
- Result:=GetStrProp(Instance,FindPropInfo(Instance,PropName));
- end;
- Procedure SetStrProp(Instance: TObject; const PropName: string; const Value: AnsiString);
- begin
- SetStrProp(Instance,FindPropInfo(Instance,PropName),Value);
- end;
- Function GetWideStrProp(Instance: TObject; const PropName: string): WideString;
- begin
- Result:=GetWideStrProp(Instance, FindPropInfo(Instance, PropName));
- end;
- procedure SetWideStrProp(Instance: TObject; const PropName: string; const Value: WideString);
- begin
- SetWideStrProp(Instance,FindPropInfo(Instance,PropName),Value);
- end;
- Function GetWideStrProp(Instance: TObject; PropInfo: PPropInfo): WideString;
- type
- TGetWideStrProcIndex=function(index:longint):WideString of object;
- TGetWideStrProc=function():WideString of object;
- var
- AMethod : TMethod;
- begin
- Result:='';
- case Propinfo^.PropType^.Kind of
- tkSString,tkAString:
- Result:=WideString(GetStrProp(Instance,PropInfo));
- tkUString :
- Result := GetUnicodeStrProp(Instance,PropInfo);
- tkWString:
- begin
- case (PropInfo^.PropProcs) and 3 of
- ptField:
- Result := PWideString(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;
- ptStatic,
- ptVirtual:
- begin
- if (PropInfo^.PropProcs and 3)=ptStatic then
- AMethod.Code:=PropInfo^.GetProc
- else
- AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;
- AMethod.Data:=Instance;
- if ((PropInfo^.PropProcs shr 6) and 1)<>0 then
- Result:=TGetWideStrProcIndex(AMethod)(PropInfo^.Index)
- else
- Result:=TGetWideStrProc(AMethod)();
- end;
- end;
- end;
- end;
- end;
- Procedure SetWideStrProp(Instance: TObject; PropInfo: PPropInfo; const Value: WideString);
- type
- TSetWideStrProcIndex=procedure(index:longint;s:WideString) of object;
- TSetWideStrProc=procedure(s:WideString) of object;
- var
- AMethod : TMethod;
- begin
- case Propinfo^.PropType^.Kind of
- tkSString,tkAString:
- SetStrProp(Instance,PropInfo,AnsiString(Value));
- tkUString:
- SetUnicodeStrProp(Instance,PropInfo,Value);
- tkWString:
- begin
- case (PropInfo^.PropProcs shr 2) and 3 of
- ptField:
- PWideString(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Value;
- ptStatic,
- ptVirtual:
- begin
- if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then
- AMethod.Code:=PropInfo^.SetProc
- else
- AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;
- AMethod.Data:=Instance;
- if ((PropInfo^.PropProcs shr 6) and 1)<>0 then
- TSetWideStrProcIndex(AMethod)(PropInfo^.Index,Value)
- else
- TSetWideStrProc(AMethod)(Value);
- end;
- end;
- end;
- end;
- end;
- Function GetUnicodeStrProp(Instance: TObject; const PropName: string): UnicodeString;
- begin
- Result:=GetUnicodeStrProp(Instance, FindPropInfo(Instance, PropName));
- end;
- procedure SetUnicodeStrProp(Instance: TObject; const PropName: string; const Value: UnicodeString);
- begin
- SetUnicodeStrProp(Instance,FindPropInfo(Instance,PropName),Value);
- end;
- Function GetUnicodeStrProp(Instance: TObject; PropInfo: PPropInfo): UnicodeString;
- type
- TGetUnicodeStrProcIndex=function(index:longint):UnicodeString of object;
- TGetUnicodeStrProc=function():UnicodeString of object;
- var
- AMethod : TMethod;
- begin
- Result:='';
- case Propinfo^.PropType^.Kind of
- tkSString,tkAString:
- Result:=UnicodeString(GetStrProp(Instance,PropInfo));
- tkWString:
- Result:=GetWideStrProp(Instance,PropInfo);
- tkUString:
- begin
- case (PropInfo^.PropProcs) and 3 of
- ptField:
- Result := PUnicodeString(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;
- ptStatic,
- ptVirtual:
- begin
- if (PropInfo^.PropProcs and 3)=ptStatic then
- AMethod.Code:=PropInfo^.GetProc
- else
- AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;
- AMethod.Data:=Instance;
- if ((PropInfo^.PropProcs shr 6) and 1)<>0 then
- Result:=TGetUnicodeStrProcIndex(AMethod)(PropInfo^.Index)
- else
- Result:=TGetUnicodeStrProc(AMethod)();
- end;
- end;
- end;
- end;
- end;
- Procedure SetUnicodeStrProp(Instance: TObject; PropInfo: PPropInfo; const Value: UnicodeString);
- type
- TSetUnicodeStrProcIndex=procedure(index:longint;s:UnicodeString) of object;
- TSetUnicodeStrProc=procedure(s:UnicodeString) of object;
- var
- AMethod : TMethod;
- begin
- case Propinfo^.PropType^.Kind of
- tkSString,tkAString:
- SetStrProp(Instance,PropInfo,AnsiString(Value));
- tkWString:
- SetWideStrProp(Instance,PropInfo,Value);
- tkUString:
- begin
- case (PropInfo^.PropProcs shr 2) and 3 of
- ptField:
- PUnicodeString(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Value;
- ptStatic,
- ptVirtual:
- begin
- if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then
- AMethod.Code:=PropInfo^.SetProc
- else
- AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;
- AMethod.Data:=Instance;
- if ((PropInfo^.PropProcs shr 6) and 1)<>0 then
- TSetUnicodeStrProcIndex(AMethod)(PropInfo^.Index,Value)
- else
- TSetUnicodeStrProc(AMethod)(Value);
- end;
- end;
- end;
- end;
- end;
- {$ifndef FPUNONE}
- { ---------------------------------------------------------------------
- Float properties
- ---------------------------------------------------------------------}
- function GetFloatProp(Instance : TObject;PropInfo : PPropInfo) : Extended;
- type
- TGetExtendedProc = function:Extended of object;
- TGetExtendedProcIndex = function(Index: integer): Extended of object;
- TGetDoubleProc = function:Double of object;
- TGetDoubleProcIndex = function(Index: integer): Double of object;
- TGetSingleProc = function:Single of object;
- TGetSingleProcIndex = function(Index: integer):Single of object;
- TGetCurrencyProc = function : Currency of object;
- TGetCurrencyProcIndex = function(Index: integer) : Currency of object;
- var
- AMethod : TMethod;
- begin
- Result:=0.0;
- case PropInfo^.PropProcs and 3 of
- ptField:
- Case GetTypeData(PropInfo^.PropType)^.FloatType of
- ftSingle:
- Result:=PSingle(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;
- ftDouble:
- Result:=PDouble(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;
- ftExtended:
- Result:=PExtended(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;
- ftcomp:
- Result:=PComp(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;
- ftcurr:
- Result:=PCurrency(Pointer(Instance)+PtrUInt(PropInfo^.GetProc))^;
- end;
- ptStatic,
- ptVirtual:
- begin
- if (PropInfo^.PropProcs and 3)=ptStatic then
- AMethod.Code:=PropInfo^.GetProc
- else
- AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;
- AMethod.Data:=Instance;
- Case GetTypeData(PropInfo^.PropType)^.FloatType of
- ftSingle:
- if ((PropInfo^.PropProcs shr 6) and 1)=0 then
- Result:=TGetSingleProc(AMethod)()
- else
- Result:=TGetSingleProcIndex(AMethod)(PropInfo^.Index);
- ftDouble:
- if ((PropInfo^.PropProcs shr 6) and 1)=0 then
- Result:=TGetDoubleProc(AMethod)()
- else
- Result:=TGetDoubleProcIndex(AMethod)(PropInfo^.Index);
- ftExtended:
- if ((PropInfo^.PropProcs shr 6) and 1)=0 then
- Result:=TGetExtendedProc(AMethod)()
- else
- Result:=TGetExtendedProcIndex(AMethod)(PropInfo^.Index);
- ftCurr:
- if ((PropInfo^.PropProcs shr 6) and 1)=0 then
- Result:=TGetCurrencyProc(AMethod)()
- else
- Result:=TGetCurrencyProcIndex(AMethod)(PropInfo^.Index);
- end;
- end;
- end;
- end;
- Procedure SetFloatProp(Instance : TObject;PropInfo : PPropInfo; Value : Extended);
- type
- TSetExtendedProc = procedure(const AValue: Extended) of object;
- TSetExtendedProcIndex = procedure(Index: integer; AValue: Extended) of object;
- TSetDoubleProc = procedure(const AValue: Double) of object;
- TSetDoubleProcIndex = procedure(Index: integer; AValue: Double) of object;
- TSetSingleProc = procedure(const AValue: Single) of object;
- TSetSingleProcIndex = procedure(Index: integer; AValue: Single) of object;
- TSetCurrencyProc = procedure(const AValue: Currency) of object;
- TSetCurrencyProcIndex = procedure(Index: integer; AValue: Currency) of object;
- Var
- AMethod : TMethod;
- begin
- case (PropInfo^.PropProcs shr 2) and 3 of
- ptfield:
- Case GetTypeData(PropInfo^.PropType)^.FloatType of
- ftSingle:
- PSingle(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Value;
- ftDouble:
- PDouble(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Value;
- ftExtended:
- PExtended(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Value;
- {$ifdef FPC_COMP_IS_INT64}
- ftComp:
- PComp(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=trunc(Value);
- {$else FPC_COMP_IS_INT64}
- ftComp:
- PComp(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Comp(Value);
- {$endif FPC_COMP_IS_INT64}
- ftCurr:
- PCurrency(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^:=Value;
- end;
- ptStatic,
- ptVirtual:
- begin
- if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then
- AMethod.Code:=PropInfo^.SetProc
- else
- AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;
- AMethod.Data:=Instance;
- Case GetTypeData(PropInfo^.PropType)^.FloatType of
- ftSingle:
- if ((PropInfo^.PropProcs shr 6) and 1)=0 then
- TSetSingleProc(AMethod)(Value)
- else
- TSetSingleProcIndex(AMethod)(PropInfo^.Index,Value);
- ftDouble:
- if ((PropInfo^.PropProcs shr 6) and 1)=0 then
- TSetDoubleProc(AMethod)(Value)
- else
- TSetDoubleProcIndex(AMethod)(PropInfo^.Index,Value);
- ftExtended:
- if ((PropInfo^.PropProcs shr 6) and 1)=0 then
- TSetExtendedProc(AMethod)(Value)
- else
- TSetExtendedProcIndex(AMethod)(PropInfo^.Index,Value);
- ftCurr:
- if ((PropInfo^.PropProcs shr 6) and 1)=0 then
- TSetCurrencyProc(AMethod)(Value)
- else
- TSetCurrencyProcIndex(AMethod)(PropInfo^.Index,Value);
- end;
- end;
- end;
- end;
- function GetFloatProp(Instance: TObject; const PropName: string): Extended;
- begin
- Result:=GetFloatProp(Instance,FindPropInfo(Instance,PropName))
- end;
- Procedure SetFloatProp(Instance: TObject; const PropName: string; Value: Extended);
- begin
- SetFloatProp(Instance,FindPropInfo(Instance,PropName),Value);
- end;
- {$endif}
- { ---------------------------------------------------------------------
- Method properties
- ---------------------------------------------------------------------}
- Function GetMethodProp(Instance : TObject;PropInfo : PPropInfo) : TMethod;
- type
- TGetMethodProcIndex=function(Index: Longint): TMethod of object;
- TGetMethodProc=function(): TMethod of object;
- var
- value: PMethod;
- AMethod : TMethod;
- begin
- Result.Code:=nil;
- Result.Data:=nil;
- case (PropInfo^.PropProcs) and 3 of
- ptField:
- begin
- Value:=PMethod(Pointer(Instance)+PtrUInt(PropInfo^.GetProc));
- if Value<>nil then
- Result:=Value^;
- end;
- ptStatic,
- ptVirtual:
- begin
- if (PropInfo^.PropProcs and 3)=ptStatic then
- AMethod.Code:=PropInfo^.GetProc
- else
- AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.GetProc))^;
- AMethod.Data:=Instance;
- if ((PropInfo^.PropProcs shr 6) and 1)<>0 then
- Result:=TGetMethodProcIndex(AMethod)(PropInfo^.Index)
- else
- Result:=TGetMethodProc(AMethod)();
- end;
- end;
- end;
- Procedure SetMethodProp(Instance : TObject;PropInfo : PPropInfo; const Value : TMethod);
- type
- TSetMethodProcIndex=procedure(index:longint;p:TMethod) of object;
- TSetMethodProc=procedure(p:TMethod) of object;
- var
- AMethod : TMethod;
- begin
- case (PropInfo^.PropProcs shr 2) and 3 of
- ptField:
- PMethod(Pointer(Instance)+PtrUInt(PropInfo^.SetProc))^ := Value;
- ptStatic,
- ptVirtual:
- begin
- if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then
- AMethod.Code:=PropInfo^.SetProc
- else
- AMethod.Code:=PCodePointer(Pointer(Instance.ClassType)+PtrUInt(PropInfo^.SetProc))^;
- AMethod.Data:=Instance;
- if ((PropInfo^.PropProcs shr 6) and 1)<>0 then
- TSetMethodProcIndex(AMethod)(PropInfo^.Index,Value)
- else
- TSetMethodProc(AMethod)(Value);
- end;
- end;
- end;
- Function GetMethodProp(Instance: TObject; const PropName: string): TMethod;
- begin
- Result:=GetMethodProp(Instance,FindPropInfo(Instance,PropName));
- end;
- Procedure SetMethodProp(Instance: TObject; const PropName: string; const Value: TMethod);
- begin
- SetMethodProp(Instance,FindPropInfo(Instance,PropName),Value);
- end;
- { ---------------------------------------------------------------------
- Variant properties
- ---------------------------------------------------------------------}
- Procedure CheckVariantEvent(P : CodePointer);
- begin
- If (P=Nil) then
- Raise Exception.Create(SErrNoVariantSupport);
- end;
- Function GetVariantProp(Instance : TObject;PropInfo : PPropInfo): Variant;
- begin
- CheckVariantEvent(CodePointer(OnGetVariantProp));
- Result:=OnGetVariantProp(Instance,PropInfo);
- end;
- Procedure SetVariantProp(Instance : TObject;PropInfo : PPropInfo; const Value: Variant);
- begin
- CheckVariantEvent(CodePointer(OnSetVariantProp));
- OnSetVariantProp(Instance,PropInfo,Value);
- end;
- Function GetVariantProp(Instance: TObject; const PropName: string): Variant;
- begin
- Result:=GetVariantProp(Instance,FindPropInfo(Instance,PropName));
- end;
- Procedure SetVariantProp(Instance: TObject; const PropName: string; const Value: Variant);
- begin
- SetVariantprop(instance,FindpropInfo(Instance,PropName),Value);
- end;
- { ---------------------------------------------------------------------
- All properties through variant.
- ---------------------------------------------------------------------}
- Function GetPropValue(Instance: TObject; const PropName: string): Variant;
- begin
- Result := GetPropValue(Instance,FindPropInfo(Instance, PropName));
- end;
- Function GetPropValue(Instance: TObject; const PropName: string; PreferStrings: Boolean): Variant;
- begin
- Result := GetPropValue(Instance,FindPropInfo(Instance, PropName),PreferStrings);
- end;
- Function GetPropValue(Instance: TObject; PropInfo: PPropInfo): Variant;
- begin
- Result := GetPropValue(Instance, PropInfo, True);
- end;
- Function GetPropValue(Instance: TObject; PropInfo: PPropInfo; PreferStrings: Boolean): Variant;
- begin
- CheckVariantEvent(CodePointer(OnGetPropValue));
- Result:=OnGetPropValue(Instance,PropInfo,PreferStrings);
- end;
- Procedure SetPropValue(Instance: TObject; const PropName: string; const Value: Variant);
- begin
- SetPropValue(Instance, FindPropInfo(Instance, PropName), Value);
- end;
- Procedure SetPropValue(Instance: TObject; PropInfo: PPropInfo; const Value: Variant);
- begin
- CheckVariantEvent(CodePointer(OnSetPropValue));
- OnSetPropValue(Instance,PropInfo,Value);
- end;
- { ---------------------------------------------------------------------
- Easy access methods that appeared in Delphi 5
- ---------------------------------------------------------------------}
- Function IsPublishedProp(Instance: TObject; const PropName: string): Boolean;
- begin
- Result:=GetPropInfo(Instance,PropName)<>Nil;
- end;
- Function IsPublishedProp(AClass: TClass; const PropName: string): Boolean;
- begin
- Result:=GetPropInfo(AClass,PropName)<>Nil;
- end;
- Function PropIsType(Instance: TObject; const PropName: string; TypeKind: TTypeKind): Boolean;
- begin
- Result:=PropType(Instance,PropName)=TypeKind
- end;
- Function PropIsType(AClass: TClass; const PropName: string; TypeKind: TTypeKind): Boolean;
- begin
- Result:=PropType(AClass,PropName)=TypeKind
- end;
- Function PropType(Instance: TObject; const PropName: string): TTypeKind;
- begin
- Result:=FindPropInfo(Instance,PropName)^.PropType^.Kind;
- end;
- Function PropType(AClass: TClass; const PropName: string): TTypeKind;
- begin
- Result:=FindPropInfo(AClass,PropName)^.PropType^.Kind;
- end;
- Function IsStoredProp(Instance: TObject; const PropName: string): Boolean;
- begin
- Result:=IsStoredProp(instance,FindPropInfo(Instance,PropName));
- end;
- { TProcedureParam }
- function TProcedureParam.GetParamType: PTypeInfo;
- begin
- Result := DerefTypeInfoPtr(ParamTypeRef);
- end;
- { TManagedField }
- function TManagedField.GetTypeRef: PTypeInfo;
- begin
- Result := DerefTypeInfoPtr(TypeRefRef);
- end;
- { TArrayTypeData }
- function TArrayTypeData.GetElType: PTypeInfo;
- begin
- Result := DerefTypeInfoPtr(ElTypeRef);
- end;
- function TArrayTypeData.GetDims(aIndex: Byte): PTypeInfo;
- begin
- Result := DerefTypeInfoPtr(DimsRef[aIndex]);
- end;
- { TProcedureSignature }
- function TProcedureSignature.GetResultType: PTypeInfo;
- begin
- Result := DerefTypeInfoPtr(ResultTypeRef);
- end;
- function TProcedureSignature.GetParam(ParamIndex: Integer): PProcedureParam;
- begin
- if (ParamIndex<0)or(ParamIndex>=ParamCount) then
- Exit(nil);
- Result := PProcedureParam(PByte(@Flags) + SizeOf(Self));
- while ParamIndex > 0 do
- begin
- Result := PProcedureParam(aligntoptr((PByte(@Result^.Name) + (Length(Result^.Name) + 1) * SizeOf(AnsiChar))));
- dec(ParamIndex);
- end;
- end;
- { TTypeData }
- function TTypeData.GetBaseType: PTypeInfo;
- begin
- Result := DerefTypeInfoPtr(BaseTypeRef);
- end;
- function TTypeData.GetCompType: PTypeInfo;
- begin
- Result := DerefTypeInfoPtr(CompTypeRef);
- end;
- function TTypeData.GetParentInfo: PTypeInfo;
- begin
- Result := DerefTypeInfoPtr(ParentInfoRef);
- end;
- function TTypeData.GetHelperParent: PTypeInfo;
- begin
- Result := DerefTypeInfoPtr(HelperParentRef);
- end;
- function TTypeData.GetExtendedInfo: PTypeInfo;
- begin
- Result := DerefTypeInfoPtr(ExtendedInfoRef);
- end;
- function TTypeData.GetIntfParent: PTypeInfo;
- begin
- Result := DerefTypeInfoPtr(IntfParentRef);
- end;
- function TTypeData.GetRawIntfParent: PTypeInfo;
- begin
- Result := DerefTypeInfoPtr(RawIntfParentRef);
- end;
- function TTypeData.GetIIDStr: ShortString;
- begin
- Result := PShortString(Pointer(@RawIntfUnit) + Length(RawIntfUnit) + 1)^;
- end;
- function TTypeData.GetElType: PTypeInfo;
- begin
- Result := DerefTypeInfoPtr(elTypeRef);
- end;
- function TTypeData.GetElType2: PTypeInfo;
- begin
- Result := DerefTypeInfoPtr(elType2Ref);
- end;
- function TTypeData.GetInstanceType: PTypeInfo;
- begin
- Result := DerefTypeInfoPtr(InstanceTypeRef);
- end;
- function TTypeData.GetRefType: PTypeInfo;
- begin
- Result := DerefTypeInfoPtr(RefTypeRef);
- end;
- { TPropInfo }
- function TPropInfo.GetPropType: PTypeInfo;
- begin
- Result := DerefTypeInfoPtr(PropTypeRef);
- end;
- end.
|