12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601 |
- {
- 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}
- {$inline on}
- {$h+}
- uses SysUtils;
- // temporary types:
- type
- {$MINENUMSIZE 1 this saves a lot of memory }
- // 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);
- TOrdType = (otSByte,otUByte,otSWord,otUWord,otSLong,otULong);
- TFloatType = (ftSingle,ftDouble,ftExtended,ftComp,ftCurr);
- TMethodKind = (mkProcedure,mkFunction,mkConstructor,mkDestructor,
- mkClassProcedure, mkClassFunction);
- TParamFlag = (pfVar,pfConst,pfArray,pfAddress,pfReference,pfOut);
- TParamFlags = set of TParamFlag;
- TIntfFlag = (ifHasGuid,ifDispInterface,ifDispatch,ifHasStrGUID);
- TIntfFlags = set of TIntfFlag;
- TIntfFlagsBase = set of TIntfFlag;
- {$MINENUMSIZE DEFAULT}
- const
- ptField = 0;
- ptStatic = 1;
- ptVirtual = 2;
- ptConst = 3;
- tkString = tkSString;
- type
- TTypeKinds = set of TTypeKind;
- {$PACKRECORDS 1}
- TTypeInfo = record
- Kind : TTypeKind;
- Name : ShortString;
- // here the type data follows as TTypeData record
- end;
- PTypeInfo = ^TTypeInfo;
- PPTypeInfo = ^PTypeInfo;
- {$PACKRECORDS C}
- PTypeData = ^TTypeData;
- TTypeData =
- {$ifndef FPC_REQUIRES_PROPER_ALIGNMENT}
- packed
- {$endif FPC_REQUIRES_PROPER_ALIGNMENT}
- record
- case TTypeKind of
- tkUnKnown,tkLString,tkWString,tkAString,tkVariant:
- ();
- tkInteger,tkChar,tkEnumeration,tkWChar,tkSet:
- (OrdType : TOrdType;
- case TTypeKind of
- tkInteger,tkChar,tkEnumeration,tkBool,tkWChar : (
- MinValue,MaxValue : Longint;
- case TTypeKind of
- tkEnumeration:
- (
- BaseType : PTypeInfo;
- NameList : ShortString)
- );
- tkSet:
- (CompType : PTypeInfo)
- );
- tkFloat:
- (FloatType : TFloatType);
- tkSString:
- (MaxLength : Byte);
- tkClass:
- (ClassType : TClass;
- ParentInfo : PTypeInfo;
- PropCount : SmallInt;
- UnitName : 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}
- );
- tkInt64:
- (MinInt64Value, MaxInt64Value: Int64);
- tkQWord:
- (MinQWordValue, MaxQWordValue: QWord);
- tkInterface:
- (
- IntfParent: PTypeInfo;
- IntfFlags : TIntfFlagsBase;
- GUID: TGUID;
- IntfUnit: ShortString;
- );
- tkInterfaceRaw:
- (
- RawIntfParent: PTypeInfo;
- RawIntfFlags : TIntfFlagsBase;
- IID: TGUID;
- RawIntfUnit: ShortString;
- IIDStr: ShortString;
- );
- end;
- // unsed, just for completeness
- 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
- PropType : PTypeInfo;
- GetProc : Pointer;
- SetProc : Pointer;
- StoredProc : Pointer;
- 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;
- end;
- TProcInfoProc = Procedure(PropInfo : PPropInfo) of object;
- PPropList = ^TPropList;
- TPropList = array[0..65535] of PPropInfo;
- const
- 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; AKinds: TTypeKinds) : PPropInfo;
- Function GetPropInfo(Instance: TObject; const PropName: string): PPropInfo;
- Function GetPropInfo(AClass: TClass; const PropName: string; AKinds: TTypeKinds) : PPropInfo;
- Function GetPropInfo(AClass: TClass; const PropName: string): PPropInfo;
- Function FindPropInfo(Instance: TObject; const PropName: string): PPropInfo;
- Function FindPropInfo(AClass:TClass;const PropName: string): 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(AObject: 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 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);
- 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 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;
- Procedure SetPropValue(Instance: TObject; const PropName: string; 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);
- // 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; const PropName: string; PreferStrings: Boolean) : Variant;
- TSetPropValue = Procedure (Instance: TObject; const PropName: string; const Value: Variant);
- TGetVariantProp = Function (Instance: TObject; PropInfo : PPropInfo): Variant;
- TSetVariantProp = Procedure (Instance: TObject; PropInfo : PPropInfo; const Value: Variant);
- Const
- OnGetPropValue : TGetPropValue = Nil;
- OnSetPropValue : TSetPropValue = Nil;
- OnGetVariantprop : TGetVariantProp = Nil;
- OnSetVariantprop : TSetVariantProp = Nil;
- 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 GetEnumName(TypeInfo : PTypeInfo;Value : Integer) : string;
- Var PS : PShortString;
- PT : PTypeData;
- begin
- PT:=GetTypeData(TypeInfo);
- // ^.BaseType);
- // If PT^.MinValue<0 then Value:=Ord(Value<>0); {map to 0/1}
- PS:=@PT^.NameList;
- While Value>0 Do
- begin
- PS:=PShortString(pointer(PS)+PByte(PS)^+1);
- Dec(Value);
- end;
- Result:=PS^;
- end;
- Function GetEnumValue(TypeInfo : PTypeInfo;const Name : string) : Integer;
- Var PS : PShortString;
- PT : PTypeData;
- Count : longint;
- begin
- If Length(Name)=0 then
- exit(-1);
- PT:=GetTypeData(TypeInfo);
- Count:=0;
- Result:=-1;
- PS:=@PT^.NameList;
- While (Result=-1) and (PByte(PS)^<>0) do
- begin
- If CompareText(PS^, Name) = 0 then
- Result:=Count;
- PS:=PShortString(pointer(PS)+PByte(PS)^+1);
- Inc(Count);
- end;
- end;
- function GetEnumNameCount(enum1: PTypeInfo): SizeInt;
- var
- PS: PShortString;
- PT: PTypeData;
- Count: SizeInt;
- begin
- PT:=GetTypeData(enum1);
- Count:=0;
- Result:=0;
- PS:=@PT^.NameList;
- While (PByte(PS)^<>0) do
- begin
- PS:=PShortString(pointer(PS)+PByte(PS)^+1);
- Inc(Count);
- end;
- Result := Count;
- 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;
- Var
- I : Integer;
- PTI : PTypeInfo;
- begin
- PTI:=GetTypeData(TypeInfo)^.CompType;
- Result:='';
- For I:=0 to SizeOf(Integer)*8-1 do
- begin
- if ((Value and 1)<>0) then
- begin
- If Result='' then
- Result:=GetEnumName(PTI,i)
- else
- Result:=Result+','+GetEnumName(PTI,I);
- end;
- Value:=Value shr 1;
- 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 : string;
- pd : ^TPropData;
- begin
- P:=UpCase(PropName);
- 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 Upcase(Result^.Name)=P 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(AClass:TClass;const PropName: string): PPropInfo;
- begin
- result:=GetPropInfo(AClass,PropName);
- 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)+Longint(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:=ppointer(Pointer(Instance.ClassType)+Longint(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;
- end;
- function GetPropList(AObject: TObject; out PropList: PPropList): Integer;
- begin
- Result := GetPropList(PTypeInfo(AObject.ClassInfo), 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)+Ptrint(PropInfo^.GetProc))^;
- 2: Result:=PSmallInt(Pointer(Instance)+Ptrint(PropInfo^.GetProc))^;
- 4: Result:=PLongint(Pointer(Instance)+Ptrint(PropInfo^.GetProc))^;
- 8: Result:=PInt64(Pointer(Instance)+Ptrint(PropInfo^.GetProc))^;
- end;
- end else begin
- case DataSize of
- 1: Result:=PByte(Pointer(Instance)+Ptrint(PropInfo^.GetProc))^;
- 2: Result:=PWord(Pointer(Instance)+Ptrint(PropInfo^.GetProc))^;
- 4: Result:=PLongint(Pointer(Instance)+Ptrint(PropInfo^.GetProc))^;
- 8: Result:=PInt64(Pointer(Instance)+Ptrint(PropInfo^.GetProc))^;
- end;
- end;
- ptstatic,
- ptvirtual :
- begin
- if (PropInfo^.PropProcs and 3)=ptStatic then
- AMethod.Code:=PropInfo^.GetProc
- else
- AMethod.Code:=PPointer(Pointer(Instance.ClassType)+Ptrint(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]) 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)+Ptrint(PropInfo^.SetProc))^:=Byte(Value);
- 2: PWord(Pointer(Instance)+Ptrint(PropInfo^.SetProc))^:=Word(Value);
- 4: PLongint(Pointer(Instance)+Ptrint(PropInfo^.SetProc))^:=Longint(Value);
- 8: PInt64(Pointer(Instance)+Ptrint(PropInfo^.SetProc))^:=Value;
- end;
- ptstatic,
- ptvirtual :
- begin
- if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then
- AMethod.Code:=PropInfo^.SetProc
- else
- AMethod.Code:=PPointer(Pointer(Instance.ClassType)+Ptrint(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,Integer(Value));
- {$endif cpu64}
- end;
- Function GetObjectPropClass(Instance: TObject; const PropName: string): TClass;
- begin
- Result:=GetTypeData(FindPropInfo(Instance,PropName)^.PropType)^.ClassType;
- 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:=GetWideStrProp(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:=PPointer(Pointer(Instance.ClassType)+Ptrint(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:=PPointer(Pointer(Instance.ClassType)+Ptrint(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,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 and 3)=ptStatic then
- AMethod.Code:=PropInfo^.SetProc
- else
- AMethod.Code:=PPointer(Pointer(Instance.ClassType)+Ptrint(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:=PPointer(Pointer(Instance.ClassType)+Ptrint(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:=GetStrProp(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:=PPointer(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,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:=PPointer(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;
- { ---------------------------------------------------------------------
- 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)+Ptrint(PropInfo^.GetProc))^;
- ftDouble:
- Result:=PDouble(Pointer(Instance)+Ptrint(PropInfo^.GetProc))^;
- ftExtended:
- Result:=PExtended(Pointer(Instance)+Ptrint(PropInfo^.GetProc))^;
- ftcomp:
- Result:=PComp(Pointer(Instance)+Ptrint(PropInfo^.GetProc))^;
- ftcurr:
- Result:=PCurrency(Pointer(Instance)+Ptrint(PropInfo^.GetProc))^;
- end;
- ptStatic,
- ptVirtual:
- begin
- if (PropInfo^.PropProcs and 3)=ptStatic then
- AMethod.Code:=PropInfo^.GetProc
- else
- AMethod.Code:=PPointer(Pointer(Instance.ClassType)+Ptrint(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)+Ptrint(PropInfo^.SetProc))^:=Value;
- ftDouble:
- PDouble(Pointer(Instance)+Ptrint(PropInfo^.SetProc))^:=Value;
- ftExtended:
- PExtended(Pointer(Instance)+Ptrint(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))^:=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:=PPointer(Pointer(Instance.ClassType)+Ptrint(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;
- { ---------------------------------------------------------------------
- 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)+Ptrint(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:=PPointer(Pointer(Instance.ClassType)+Ptrint(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)+Ptrint(PropInfo^.SetProc))^ := Value;
- ptstatic,
- ptvirtual :
- begin
- if ((PropInfo^.PropProcs shr 2) and 3)=ptStatic then
- AMethod.Code:=PropInfo^.SetProc
- else
- AMethod.Code:=PPointer(Pointer(Instance.ClassType)+Ptrint(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 : Pointer);
- begin
- If (P=Nil) then
- Raise Exception.Create(SErrNoVariantSupport);
- end;
- Function GetVariantProp(Instance : TObject;PropInfo : PPropInfo): Variant;
- begin
- CheckVariantEvent(Pointer(OnGetVariantProp));
- Result:=OnGetVariantProp(Instance,PropInfo);
- end;
- Procedure SetVariantProp(Instance : TObject;PropInfo : PPropInfo; const Value: Variant);
- begin
- CheckVariantEvent(Pointer(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,PropName,True);
- end;
- Function GetPropValue(Instance: TObject; const PropName: string; PreferStrings: Boolean): Variant;
- begin
- CheckVariantEvent(Pointer(OnGetPropValue));
- Result:=OnGetPropValue(Instance,PropName,PreferStrings)
- end;
- Procedure SetPropValue(Instance: TObject; const PropName: string; const Value: Variant);
- begin
- CheckVariantEvent(Pointer(OnSetPropValue));
- OnSetPropValue(Instance,PropName,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:=FindPropInfo(Instance,PropName)^.PropType^.Kind=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;
- end.
|