{ +-------------------------------------------------------------+ }
{ |                                                             | }
{ |   GM-Software                                               | }
{ |   ===========                                               | }
{ |                                                             | }
{ |   Project: All Projects                                     | }
{ |                                                             | }
{ |   Description: GM - Interfaces                              | }
{ |                                                             | }
{ |                                                             | }
{ |   Copyright (C) - 1996 - Gerrit Moeller.                    | }
{ |                                                             | }
{ |   Source code distributed under MIT license.                | }
{ |                                                             | }
{ |   See: https://www.gm-software.de                           | }
{ |                                                             | }
{ +-------------------------------------------------------------+ }

{$INCLUDE GMCompilerSettings.inc}

unit GMIntf;

interface

uses {$IFDEF JEDIAPI}jwaWinType, jwaWinError, jwaWinNT, jwaWinUser,{$ELSE}Windows,{$ENDIF} 
     GMStrDef ,GMActiveX, GMUnionValue, SysUtils, TypInfo;

const

  cDfltRoutineName = '';
  cStrFieldDataTypeName = 'TGMDBColumnDataType';
  CGMUnknownState = -1;
  CGMUnknownPosition = -1;
  cGMUnknownCount = -1;
  CInvalidEnumPos = High(LongInt);
  cCurrentStrmPos = -1;

  cDfltCoCeateContext = CLSCTX_SERVER;

  cDfltReadString = '';
  cDfltReadInteger = 0;
  cDfltReadBoolean = False;
  cDfltReadFloat = 0;
  cDfltReadDateTime = 0;

  cDfltCopyBufferSize = $10000; // <- 64 KB
  cDfltUiResponseMS = 400; // 330; // <- Milliseconds
  cFormatBufSize = 4096;

  cDfltCoInitFlags = COINIT_APARTMENTTHREADED; // COINIT_MULTITHREADED

  scSubNodesDirSeparator = '.';

  //cStrUnknown = '?';

  cCustomHrError: LongInt = LongInt($A0000000);

  cNewLine = #13#10;
  c2NewLine = cNewLine + cNewLine;


type

  {$IFNDEF JEDIAPI}
  {$IFNDEF FPC}
  {$IFDEF CPU64}
  PtrInt = Int64;
  PtrUInt = QWord;
  {$ELSE}
  PtrInt = LongInt;
  PtrUInt = LongWord;
  {$ENDIF}
  PPtrInt = ^PtrInt;
  PPtrUInt = ^PtrUInt;
  {$ENDIF}
  {$ENDIF}

  TGMCompareResult = (crALessThanB, crAEqualToB, crAGreaterThanB);
  TGMSeverityLevel = (svNone, svConfirmation, svInformation, svWarning, svError);
  //TGMSeverityLevel = svInformation .. svError;
  //TSeverityLevels = set of TGMSeverityLevelBase;

  TGMCharKind = (ckUnknown, ckAnsi, ckUtf8, ckUtf16LE, ckUtf16BE);
  EGMLineEndKind = (lekUnknown, lekCR, lekCRLF, lekLF, leLFCR);
  //TGMCharKind = (ckAnsi, ckUtf8, ckUtf16LE, jccISO_8859_1);


  TGMCheckRefCountProc = procedure(const ARefCount: LongInt; const AObj: TObject);

  PGMPtrIntArray = ^TGMPtrIntArray;
  TGMPtrIntArray = array of PtrInt;

  procedure GMAddIntegersToArray(var ADest: TGMPtrIntArray; const AValues: array of PtrInt);



  //
  // Optionally exported function of plugins to write their resource string PO files
  //

  type

    TDllWriteResStrPOFileProc = procedure (FileExtension: TApiString = ''; OutputFolder: TApiString = ''); stdcall;


  const

    cStrDllWriteResStrPOFileProcName = 'DllWriteResStrPOFile';


  { ------------------------------ }
  { ---- Type Safe Interfaces ---- }
  { ------------------------------ }

  type

  //
  // This generic prevents you from using a different (wrong) IID than the variable was declared for
  // TIntf is meant to be an interface type derived from IUnknown
  //
  RGMTypedIntf<TIntf> = record
   //private
   // class operator Implicit(const ATypedIntf: RGMTypedIntf<TIntf>): IUnknown;
   // class operator Implicit(AValue: Pointer): RGMTypedIntf<TIntf>;

   public
    Intf: TIntf;

    class operator Implicit (const ATypedIntf: RGMTypedIntf<TIntf>): TIntf;
    class operator := (AValue: IUnknown): RGMTypedIntf<TIntf>;
    class operator := (AObj: TObject): RGMTypedIntf<TIntf>;

    // ACallingName is only relevant/used when ACheckResult = True
    function QueryFrom(AIntf: IUnknown; ACheckResult: Boolean = False; const ACallingName: TGMString = cDfltRoutineName): Boolean;
    function GetFrom(AObj: TObject; ACheckResult: Boolean = False; const ACallingName: TGMString = cDfltRoutineName): Boolean;
    //function GetWeakFrom(AObj: TObject; ACheckResult: Boolean = False): Boolean;
    function Call(ACaller: TObject = nil): TIntf;
  end;


  { ----------------------- }
  { ---- String Arrays ---- }
  { ----------------------- }

  type

  PGMStringArray = ^TGMStringArray;
  TGMStringArray = array of TGMString;

  function GMStringArray(const AStrings: array of TGMString): TGMStringArray;
  procedure GMAddStrToArray(const AValue: TGMString; var AStringArray: TGMStringArray; const AAddEmptyStrings: Boolean = False);
  function GMStrArrayAsText(const AStrings: TGMStringArray; const ASeparator: TGMString = cNewLine): TGMString;
  function GMIndexOfStrInArray(const AValue: TGMString; const AStringArray: TGMStringArray): PtrInt;
  function GMFindStrInArray(const AValue: TGMString; const AStringArray: TGMStringArray; var AIdx: PtrInt): Boolean;
  procedure GMDeleteStrInArray(var AStringArray: TGMStringArray; const AIdx: PtrInt);


  { -------------------------------- }
  { ---- Thread synchronization ---- }
  { -------------------------------- }

  type

  IGMCriticalSection = interface(IUnknown)
    ['{278BDF06-1387-4181-A83D-8DDF4E18CE03}']
    procedure EnterCriticalSection;
    procedure LeaveCriticalSection;
    //function TryEnterCriticalSection: Boolean; 
  end;


  procedure GMEnterCriticalSection(const ACriticalSection: IUnknown);
  procedure GMLeaveCriticalSection(const ACriticalSection: IUnknown);


  { ----------------- }
  { ---- Objects ---- }
  { ----------------- }

  type
  //
  // Getting the object implementing an interface
  //
  IGMObjInfo = interface(IUnknown)
    ['{F8FF8365-3C8F-4730-B628-82280DCC75FB}']
    function GetClassName: TGMString;
    function GetClassType: TClass;
    function GetInstance: TObject;
    function GetTypeInfo: PTypeInfo; // <- will return nil unless class has been declared with $M+ compiler directive!
    function GetRefCount: LongInt;
    property ClassName: TGMString read GetClassName;
    property ClassType: TClass read GetClassType;
    property Instance: TObject read GetInstance;
    property TypeInfo: PTypeInfo read GetTypeInfo;
    property RefCount: LongInt read GetRefCount;
  end;


  TGMHashCode = PtrInt;

  IGMHashCode = interface(IUnknown)
    ['{9C61B58B-41DF-4695-9716-AC4A343DC2DB}']
    function HashCode: TGMHashCode;
  end;


  { -------------------- }
  { ---- Exceptions ---- }
  { -------------------- }

  IGMGetHRCode = interface(IUnknown)
    ['{B3526DBB-F29F-474e-94A0-C8B3218DABC0}']
    function GetHRCode: HResult; stdcall;
    property HRCode: HResult read GetHRCode;
  end;


  function GMGetObjHRCode(const AObj: TObject; const ADefaultHrCode: HResult = S_OK): HResult;
  function GMGetIntfHRCode(const AIntf: IUnknown; const ADefaultHrCode: HResult = S_OK): HResult;
  function GMIsOneOfIntegers(const AValue: PtrInt; const AIntValues: array of PtrInt): Boolean;


  type

  IGMSetExceptionInformation = interface(IUnknown)
    ['{07870B30-5CFA-4F90-8A22-EEB7EE796543}']
    procedure SetMessage(AMessage: PGMChar); stdcall;
    procedure SetSeverityLevel(ASeverityLevel: TGMSeverityLevel); stdcall;
  end;


  IGMExceptionInformation = interface(IUnknown)
    ['{E9D30915-FAF3-43b3-A3FA-B3AE9E24EA02}']
    //
    // The TGMExceptionHandlerObj will ask raised Exceptions for
    // this Interface. If they have this interface the information
    // will be used to display more comprehensive error information.
    //
    function GetGMMessage: PGMChar; stdcall;
    function GetExceptionClassName: PGMChar; stdcall;
    function GetExceptAddress: Pointer; stdcall;
    function GetRaisorName: PGMChar; stdcall;
    function GetRaisorClassName: PGMChar; stdcall;
    function GetRoutineName: PGMChar; stdcall;
    function GetSeverityLevel: TGMSeverityLevel; stdcall;
    function GetHelpCtx: LongInt; stdcall;

    property GMMessage: PGMChar read GetGMMessage;
    property ExceptionClassName: PGMChar read GetExceptionClassName;
    property ExceptAddress: Pointer read GetExceptAddress;
    property RaisorName: PGMChar read GetRaisorName;
    property RaisorClassName: PGMChar read GetRaisorClassName;
    property RoutineName: PGMChar read GetRoutineName;
    property SeverityLevel: TGMSeverityLevel read GetSeverityLevel;
    property HelpContext: LongInt read GetHelpCtx;
  end;


  TGMDfltVerticalAlignment  = (vaDefault, vaTop, vaCenter, vaBottom);
  TGMVerticalAlignment = vaTop .. vaBottom;

  TGMDfltHorizontalAlignment = (haDefault, haLeft, haCenter, haRight);
  TGMHorizontalAlignment = haLeft .. haRight;


  TGMColumnDescRec = record
   Title: TGMString;
   Width: LongInt;
   Alignment: TGMDfltHorizontalAlignment;
  end;


  IGMClear = interface(IUnknown)
    ['{EC95F51C-8238-40AB-A1A2-406C761E1456}']
    procedure Clear(const ANotify: Boolean = True);
  end;


  { ----------------------------- }
  { ---- Activatable Objects ---- }
  { ----------------------------- }

  IGMGetActive = interface(IUnknown)
    ['{BEB78B21-2B77-11d5-AB38-000021DCAD19}']
    //
    // Anything that can be activated, like:
    //  - Recordsets
    //  - Players
    //  - Connections
    //
    function GetActive: Boolean; stdcall;
    property Active: Boolean read GetActive;
  end;


  IGMGetSetActive = interface(IGMGetActive)
    ['{731F9581-1642-11d5-A5E4-00E0987755DD}']
    procedure SetActive(const Value: Boolean); stdcall;
    property Active: Boolean read GetActive write SetActive;
  end;


  IGMActiveChangeNotifications = interface(IUnknown)
    ['{4D3692B3-522B-4bf2-BA82-EAB98198B755}']
    procedure BeforeActiveChange(const NewActive: Boolean); stdcall;
    procedure AfterActiveChange(const NewActive: Boolean); stdcall;
  end;


  IGMVerifyActivation = interface(IUnknown)
    ['{290C3D48-1C5A-11d5-AB38-000021DCAD19}']
    //
    // Verify the State on an activatable object. Raise if it is not.
    //
    procedure CheckIsActive(const ACallingName: TGMString = cDfltRoutineName); stdcall;
    procedure CheckIsInactive(const NeedInActiveName: TGMString = cDfltRoutineName); stdcall;
  end;


  function GMObjIsActive(const AObj: TObject; const ADefaultValue: Boolean = False): Boolean;
  function GMIntfIsActive(const AIntf: IUnknown; const ADefaultValue: Boolean = False): Boolean;

  function GMSetObjActive(const AObj: TObject; const Active: Boolean; const ACallingName: TGMString = cDfltRoutineName): Boolean;

  function GMSetIntfActive(const AIntf: IUnknown; const Active: Boolean; const ACallingName: TGMString = cDfltRoutineName): Boolean;

  procedure GMCheckObjIsInActive(const AObj: TObject; const NeedInActiveName: TGMString);
  procedure GMCheckIntfIsInActive(const AIntf: IUnknown; const NeedInActiveName: TGMString);

  procedure GMCheckObjIsActive(const AObj: TObject; const ACallingName: TGMString = cDfltRoutineName);
  procedure GMCheckIntfIsActive(const AIntf: IUnknown; const ACallingName: TGMString = cDfltRoutineName);


  type

  { ------------------------------------- }
  { ---- Connections between Objects ---- }
  { ------------------------------------- }

  //
  // There are many situations when Objects need to be connected to each other.
  // Instead of designing my own set of interfaces for connecting objects I decided
  // to use th? interfaces IConnectionPointContainer and IConnectionPoint.
  //
  // There are implementations for Connection points and containers in unit gmdbbase.
  // I did my own implementation because I couldn't agree with the borland one.
  //

  IGMCreateConnectionPoint = interface(IUnknown)
    ['{FD024422-1DE8-11d5-AB38-000021DCAD19}']
    //
    // A Implementation of a IConnectionPointContainer may implement this interface
    // too to let others create new ConnectionPoints from outside.
    //
    procedure CreateConnectionPoint(const AIID: TGUID); stdcall;
  end;


  IGMDisconnectFromConnectionPoint = interface(IUnknown)
    ['{0D697004-1D84-11d5-AB38-000021DCAD19}']
    //
    // A Object that is connected to a connection point may implement this interface.
    // The connection point will call DisconnectFromConnectionPoint on all connected
    // Objects if it wants the Objects to disconnect from itself (because of closing down or destruction).
    //
    procedure DisconnectFromConnectionPoint(const ConnectionPointContainer: IUnknown; const AIID: TGUID; const Cookie: LongInt); stdcall;
  end;



  { ----------------- }
  { ---- Handles ---- }
  { ----------------- }

  //
  // Note: Even if the datatype of a handle can be technically marshalled to
  //       other processes or computers normally a handle will be invalid in
  //       another process (computer).
  //

  IGMGetHandle = interface(IUnknown)
    ['{5BB45961-15A9-11d5-A5E4-00E0987755DD}']
    //
    // Anything that has a Handle:
    // ===========================
    //  - Windows
    //  - Files
    //  - DB Objects
    //  - GDI Objects etc.
    //
    function GetHandle: THandle; stdcall;
    property Handle: THandle read GetHandle;
  end;


  IGMGetSetHandle = interface(IGMGetHandle)
    ['{A6D2A402-1F1E-11d5-AB38-000021DCAD19}']
    //
    // This one allows to set the Handle too.
    //
    procedure SetHandle(const Value: THandle); stdcall;
    property Handle: THandle read GetHandle write SetHandle;
  end;


  IGMGetHandleType = interface(IUnknown)
    ['{39ED6E5A-DD09-4956-B5C3-58F203EA1A08}']
    function GetHandleType: Longword; stdcall;
    property HandleType: Longword read GetHandleType;
  end;


  IGMHandleAllocated = interface(IUnknown)
    ['{81BED96A-D5A7-418a-8808-31CD51E22117}']
    // Windows handles are often created when accessed.
    // So Handle <> 0 doesnt work since it creates the handle if neccessary.
    // In such situations use IGMHandleAllocated.
    function GetHandleAllocated: Boolean; stdcall;
    property HandleAllocated: Boolean read GetHandleAllocated;
  end;


  {IGMEnableDisablePaint = interface(IUnknown)
    ['655E0FB2-6AC8-4b5d-864B-EFCA768090F9']
    function EnablePaint: LongInt; stdcall;
    function DisablePaint: LongInt; stdcall;
    function GetPaintDisabledCount: LongInt; stdcall;
    property PaintDisabledCount: LongInt read GetPaintDisabledCount;
  end;}


  { ---------------------- }
  { ---- Names / Text ---- }
  { ---------------------- }

  type

  IGMGetName = interface(IUnknown)
    ['{D7242466-1DD0-494e-97CF-7944331A9116}']
    function GetName: TGMString; stdcall;
    property Name: TGMString read GetName;
  end;

  IGMGetSetName = Interface(IGMGetName)
    ['{423674CA-F5D1-4c34-ADF6-E592ACD9D4AA}']
    procedure SetName(const Value: TGMString); stdcall;
    property Name: TGMString read GetName write SetName;
  end;


  IGMGetFileName = interface(IUnknown)
    ['{D3ECCB42-A563-4cc4-B375-79931031ECBA}']
    function GetFileName: TGMString; stdcall;
    property FileName: TGMString read GetFileName;
  end;

  IGMGetSetFileName = Interface(IGMGetFileName)
    ['{ECFB879F-86F6-41a3-A685-0C899A2B5BCA}']
    procedure SetFileName(const Value: TGMString); stdcall;
    property FileName: TGMString read GetFileName write SetFileName;
  end;


  IGMGetHint = interface(IUnknown)
    ['{08E916C0-5208-4513-BB23-71747E5140C3}']
    function GetHint: TGMString; stdcall;
    property Hint: TGMString read GetHint;
  end;

  IGMGetSetHint = interface(IGMGetHint)
    ['{225B8C2D-4FD3-4d6e-B331-6AA331F761BC}']
    procedure SetHint(const Value: TGMString); stdcall;
    property Hint: TGMString read GetHint write SetHint;
  end;


  IGMGetText = interface(IUnknown)
    ['{BBDCFBC0-B9A3-4208-AFC1-3EE2903C21C2}']
    function GetText: TGMString; stdcall;
    property Text: TGMString read GetText;
  end;

  IGMGetSetText = interface(IGMGetText)
    ['{547DA16A-5C69-45a2-9FB0-25D93F749168}']
    procedure SetText(const Value: TGMString); stdcall;
    property Text: TGMString read GetText write SetText;
  end;


  //
  // A record implementing IGMGetName for the following purposes:
  //  - Can reside on the stack, avoiding memory manager overhead and improving concurrency
  //  - Reducing memory consumption: 12 Byte instead 36 Byte (TGMNameObj) per instance
  //
  PGMNameRec = ^RGMNameRec;
  RGMNameRec = record
   VTable: Pointer;   // <- automatically assigned by Initialize
   RefCount: Integer; // <- automatically assigned by Initialize
   Name: TGMString;

   class operator Initialize(var ANameRec: RGMNameRec);
   class operator Implicit(const ANameValRec: RGMNameRec): IUnknown;

   //class function QueryIntferface(AInstance: PGMNameRec; const AIID: TGUID; out AObj): HResult; stdcall; static;
   //class function Addref(AInstance: PGMNameRec): Integer; stdcall; static;
   //class function Release(AInstance: PGMNameRec): Integer; stdcall; static;

   function QueryIntferface(const AIID: TGUID; out AObj): HResult; stdcall;
   function Addref: Integer; stdcall;
   function Release: Integer; stdcall;
   function GetName: TGMString; stdcall;
  end;


  { ------------------------------------ }
  { ---- Objects that have an owner ---- }
  { ------------------------------------ }

  type                  

  IGMReleaseReferences = interface(IUnknown)
    ['{A213DBA5-1F9B-11d5-AB38-000021DCAD19}']
    procedure ReleaseReferences; stdcall;
  end;

  procedure GMReleaseMembers(const AIntf: IUnknown);


  type

  IGMGetParentObj = interface(IUnknown)
    ['{B8487F83-E75D-4e5a-BB2D-BA857CC846FE}']
    function GetParentObj: TObject;
    property ParentObj: TObject read GetParentObj;
  end;

  IGMGetSetParentObj = interface(IGMGetParentObj)
    ['{C7306C62-6C64-4d05-85C1-2593F8960951}']
    procedure SetParentObj(const Value: TObject; const Relayout: Boolean = True);
    //property ParentObj: TObject read GetParentObj write SetParentObj;
  end;


  { ------------------------- }
  { ---- File Attributes ---- }
  { ------------------------- }

  TFileAttribute = (faArchive, faCompressed, faDirectory, faEncrypted, faHidden, faNormal, faOffline, faReadOnly, faReparsePoint, faSparse, faSystem, faTemporary);
  TFileAttributes = set of TFileAttribute;

  IGMFileProperties = interface(IUnknown)
    ['{1F883BC2-19F8-4f67-9633-B0CA563F7DC0}']
    function GetFileName: TGMString; stdcall;
    function GetDisplayName: TGMString; stdcall;
    function GetAttributes: TFileAttributes; stdcall;
    function GetCreationTime: TDateTime; stdcall;
    function GetLastAccessTime: TDateTime; stdcall;
    function GetLastWriteTime: TDateTime; stdcall;
    function GetSizeInBytes: Int64; stdcall;

    property FileName: TGMString read GetFileName;
    property DisplayName: TGMString read GetDisplayName;
    property Attributes: TFileAttributes read GetAttributes;
    property CreationTime: TDateTime read GetCreationTime;
    property LastAccessTime: TDateTime read GetLastAccessTime;
    property LastWriteTime: TDateTime read GetLastWriteTime;
    property SizeInBytes: Int64 read GetSizeInBytes;
  end;


  function GMDWordToFileAttributes(const AValue: DWORD): TFileAttributes;
  function GMFileAttributesToDWORD(const AValue: TFileAttributes): DWORD;


  { ------------------------- }
  { ---- Enumarte things ---- }
  { ------------------------- }

  type

  IGMTellEnumString = interface(IUnknown)
    ['{C74E385E-C615-4398-BECE-8B7D537DA87E}']
    procedure TellEnumString(const ItemKind: LongInt; const Value: TGMString; const Parameter: Pointer = nil); stdcall;
  end;


  IGMTellEnumIntf = interface(IUnknown)
    ['{54EDEAAA-C059-41d1-9F7A-2611D05A8D73}']
    procedure TellEnumIntf(const Sender: IUnknown; const ItemKind: LongInt; const Value: IUnknown; const Parameter: Pointer = nil); stdcall;
  end;


  IGMEnumerateItems = interface(IUnknown)
    ['{525D8C22-275A-11d5-AB38-000021DCAD19}']
    procedure EnumerateItems(const ItemKind: LongInt; const TellEnumSink: IUnknown; const Parameter: Pointer = nil); stdcall;
  end;


  { ------------------------- }
  { ---- Locate / Lookup ---- }
  { ------------------------- }

  type

  IGMLookupValues = interface(IUnknown)
    ['{40B69A61-2819-11d5-AB38-000021DCAD19}']
    function LookupValues(const SQLCriteria: TGMString; const Values: IUnknown): Boolean; stdcall;
  end;


  TMatchKind = (mkExactMatch, mkNearestMatch);

  IGMPositionOfValues = interface(IUnknown)
    ['{52E31022-3907-4fc8-82CF-7FD9B96285F9}']
    function PositionOfValues(const Values: IUnknown; var FindPos: LongInt): Boolean; stdcall;
  end;

  IGMLocateValues = interface(IUnknown)
    ['{D19ECF62-28E7-11d5-AB38-000021DCAD19}']
    function LocateValues(const Values: IUnknown): Boolean; stdcall;
  end;


  { ------------------------------------- }
  { ---- Getting Interfaces indirect ---- }
  { ------------------------------------- }

  IGMGetIntfByName = interface(IUnknown)
    ['{4694A883-24F6-11d5-AB38-000021DCAD19}']
    function GetIntfByName(const Name: TGMString; const AIID: TGUID; out AIntf): HResult; stdcall;
  end;

  function GMFieldDisplayText(const FieldName: TGMString; const PIFieldByName: IGMGetIntfByName): TGMString;


  type

  IGMGetIntfByPosition = interface(IUnknown)
    ['{4694A884-24F6-11d5-AB38-000021DCAD19}']
    function GetIntfByPosition(const Position: PtrInt; const AIID: TGUID; out AIntf): HResult; stdcall;
  end;


  IGMGetPropertyIntf = interface(IUnknown)
    ['{746A0021-33EC-11d5-AB38-000021DCAD19}']
    function GetPropertyIntf(const PropertyName: TGMString; const AIID: TGUID; out AIntf): HResult; stdcall;
  end;


  IGMGetSubItems = interface(IUnknown)
    ['{967AED57-89B8-4547-8A8F-36A3D8C7CD32}']
    function GetSubItems(const ParentFieldName: TGMString; const ParentFieldValue: RGMUnionValue; const AIID: TGUID; out AIntf): HResult;
  end;


  { ---------------------- }
  { ---- Data Sources ---- }
  { ---------------------- }

  type

  IGMGetInterfaceSource = interface(IUnknown)
    ['{DFB04E9F-BC8D-474c-9134-B242699810EC}']
    function GetInterfaceSource: IUnknown; stdcall;
    property InterfaceSource: IUnknown read GetInterfaceSource;
  end;


  IGMGetSetInterfaceSource = interface(IGMGetInterfaceSource)
    ['{06B67B17-78B5-4a54-8959-3EDE25F0CDEA}']
    procedure SetInterfaceSource(const Value: IUnknown); stdcall;
    property InterfaceSource: IUnknown read GetInterfaceSource write SetInterfaceSource;
  end;


  function GMGetInterfaceSource(const AContainer: IUnknown): IUnknown;
  procedure GMSetInterfaceSource(const AContainer, AIntfSource: IUnknown);


  { ---------------- }
  { ---- Values ---- }
  { ---------------- }

  //
  // The following Value related interfaces may be implemnted by:
  // ============================================================
  //  - Fields
  //  - Parameters
  //  - Cached values
  //  - Anything else that maintains a Value
  //

  type

  IGMGetOleValue = interface(IUnknown)
    ['{5DC49C42-1BBE-11d5-AB38-000021DCAD19}']
    //
    // Anything that has a Value that can be read
    //
    function GetOleValue: OleVariant; stdcall;
    property Value: OleVariant read GetOleValue;
  end;


  IGMGetSetOleValue = interface(IGMGetOleValue)
    ['{5DC49C44-1BBE-11d5-AB38-000021DCAD19}']
    //
    // Anything that has a Read/Write Value
    //
    procedure SetOleValue(const AValue: OleVariant); stdcall;
    property Value: OleVariant read GetOleValue write SetOleValue;
  end;


  function GMGetItemValue(const AContainer: IUnknown; const AItemName: TGMString): RGMUnionValue; overload;
  function GMGetItemValue(const AContainer: IUnknown; const AItemPosition: LongInt): RGMUnionValue; overload;

  function GMCheckGetItemValue(const AContainer: IUnknown; const ItemName: TGMString; const ACallingName: TGMString = cDfltRoutineName): RGMUnionValue; overload;
  function GMCheckGetItemValue(const AContainer: IUnknown; const ItemPosition: LongInt; const ACallingName: TGMString = cDfltRoutineName): RGMUnionValue; overload;

  procedure GMSetItemValue(const AContainer: IUnknown; const AItemName: TGMString; const AValue: RGMUnionValue);
  procedure GMCheckSetItemValue(const AContainer: IUnknown; const ItemName: TGMString; const AValue: RGMUnionValue; const ACallingName: TGMString = cDfltRoutineName);

  function GMGetFieldValueImpl(const AContainer: IUnknown; const AIndex: RGMUnionValue): RGMUnionValue;
  procedure GMSetFieldValueImpl(const AContainer: IUnknown; const AIndex: RGMUnionValue; const AValue: RGMUnionValue);


  type

  IGMGetStringValue = interface(IUnknown)
    ['{601C32E7-A887-44D3-9604-06D128CF0670}']
    function GetStringValue: TGMString;
    property StringValue: TGMString read GetStringValue;
  end;


  IGMGetSetStringValue = interface(IGMGetStringValue)
    ['{D96903C3-4063-401D-AB17-1FC69E87F23A}']
    procedure SetStringValue(const ANewValue: TGMString);
    property StringValue: TGMString read GetStringValue write SetStringValue;
  end;


  IGMGetGUID = interface(IUnknown)
    ['{54A2C24B-EAF3-4de7-9359-27A88214E02D}']
    function GetGUID: TGUID; stdcall;
    property GUID: TGUID read GetGUID;
  end;


  IGMGetSetGUID = interface(IGMGetGUID)
    ['{7A0B2960-72B6-4cee-BAC9-69C58CFCAF3E}']
    procedure SetGUID(const Value: TGUID); stdcall;
    property GUID: TGUID read GetGUID write SetGUID;
  end;


  IGMGetModified = interface(IUnknown)
    ['{CC345281-29B8-11d5-AB38-000021DCAD19}']
    function GetModified: Boolean; stdcall;
    property Modified: Boolean read GetModified;
  end;


  IGMGetSetModified = interface(IGMGetModified)
    ['{CC345282-29B8-11d5-AB38-000021DCAD19}']
    procedure SetModified(const Value: Boolean); stdcall;
    property Modified: Boolean read GetModified write SetModified;
  end;


  IGMGetIsNull = interface(IUnknown)
    ['{DC42E57B-E518-4E90-9748-D78A6B569625}']
    function GetIsNull: Boolean;
    property IsNull: Boolean read GetIsNull;
  end;


  IGMValidateValues = interface(IUnknown)
    ['{BB458E41-2C19-11d5-AB38-000021DCAD19}']
    procedure ValidateValues;
  end;


  IGMGetReadOnly = interface(IUnknown)
    ['{88A90C63-47A5-49ae-89CC-E123ACB6D629}']
    function GetReadonly: Boolean; stdcall;
    property ReadOnly: Boolean read GetReadOnly;
  end;

  IGMGetSetReadOnly = interface(IGMGetReadOnly)
    ['{B551AD30-8A05-4b59-9D1C-75FD8F9A478E}']
    procedure SetReadOnly(const Value: Boolean); stdcall;
    property ReadOnly: Boolean read GetReadOnly write SetReadOnly;
  end;


  IGMGetEnabled = interface(IUnknown)
    ['{B460F84A-929B-47d9-8E19-77B63276DB23}']
    function GetEnabled: Boolean; stdcall;
    property Enabled: Boolean read GetEnabled;
  end;

  IGMGetSetEnabled = interface(IGMGetEnabled)
    ['{FB390167-27D1-41a2-8A94-0322E0B64EDD}']
    procedure SetEnabled(const Value: Boolean); stdcall;
    property Enabled: Boolean read GetEnabled write SetEnabled;
  end;

  procedure GMEnableObj(const AObj: TObject; const AEnabled: Boolean);
  procedure GMEnableIntf(const AIntf: IUnknown; const AEnabled: Boolean);

  function GMGetObjEnabled(const AObj: TObject; const ADefaultValue: Boolean = False): Boolean;
  function GMGetIntfEnabled(const AIntf: IUnknown; const ADefaultValue: Boolean = False): Boolean;


  type

  EGMValueBufferAccessMode = (baRead, baWrite);
  EGMValueBufferInstance = (vbiValue, vbiOldValue);

  const

  cStrBufAccessTypeName = 'EGMValueBufferAccessMode';
  cStrValBufInstTypeName = 'EGMValueBufferInstance';


  type

  //IGMAccessValueBuffer = interface(IUnknown)
  //  ['{478594C1-2CE4-11d5-AB38-000021DCAD19}']
  //  function AccessValueBuffer(const AccessMode: LongInt; const AIID: TGUID; out AIntf; const ValueBufferInstance: LongInt = Ord(vbiValue)): HResult; stdcall;
  //end;


  IGMGetValueBufferIntf = interface(IUnknown)
    ['{48E70686-D324-4ef7-B1D7-E646EA0F612A}']
    function GetValueBufferIntf(const ValueBufferInstance: LongInt; const AIID: TGUID; out AIntf): HResult; stdcall;
  end;


  { ----------------------------------- }
  { ---- Objects that have a State ---- }
  { ----------------------------------- }

  IGMGetState = interface(IUnknown)
    ['{427EF9A1-220B-11d5-AB38-000021DCAD19}']
    function GetState: LongInt; stdcall;
    property State: LongInt read GetState;
  end;


  IGMGetSetState = interface(IGMGetState)
    ['{427EF9A2-220B-11d5-AB38-000021DCAD19}']
    procedure SetState(const Value: LongInt); stdcall;
    property State: LongInt read GetState write SetState;
  end;


  IGMStateChangeNotifications = interface(IUnknown)
    ['{C3A9CA45-1ACD-11d5-AB38-000021DCAD19}']
    //
    // Implemented by:
    // ===============
    //  - Fields
    //  - Data aware controls
    //
    // The implementing Object wants to be informed when a Recordset changes it's state.
    //
    // To connect to a recordset the IconnectionPointContainer/IConnectionPoint
    // interfaces should be used.
    //
    procedure BeforeStateChange(const OldState, NewState: LongInt); stdcall;
    procedure AfterStateChange(const OldState, NewState: LongInt); stdcall;
  end;


  IGMSaveRestoreState = interface
    ['{D5DC937A-959B-47d9-96BF-2FB011C6071D}']
    function CaptureState: IUnknown; stdcall; // <- Give the current state represented as IUnknown
    procedure RestoreState(const State: IUnknown); stdcall; // <- Restore from a previously saved state
  end;


  { ------------------ }
  { ---- Progress ---- }
  { ------------------ }

  type

  TGMCalcProgressKind = (cpkAbsolute, cpkRelative);

  TGMOnProgressProc = procedure (const AProgress: Int64; var ACancel: BOOL) of object; stdcall;
//TGMOnTellTotalProgressProc = procedure (const ATotalProgress: Int64) of object; stdcall;

  IGMOnProgress = interface(IUnknown)
    ['{3128A854-FB14-41f1-AFF1-77E67C7F4EA8}']
    procedure OnProgress(const AProgress: Int64; var ACancel: BOOL; const ACalcProgressKind: TGMCalcProgressKind = cpkAbsolute); stdcall;
  end;


  IGMSetCounter = interface(IUnknown)
    ['{63C0E4E8-FDF8-4B75-9133-8C09CC9584A3}']
    procedure SetCounter(const ACounterId: LongInt; const AValue: Int64; const ACalcCountKind: TGMCalcProgressKind = cpkAbsolute); stdcall;
  end;


  IGMSetProgressMax = interface(IGMOnProgress)
    ['{837A4D36-0F99-4ef2-82CF-0F375A522E5C}']
    procedure SetProgressMax(const AProgressMax: Int64); stdcall;
  end;


  IGMSetProgressDescription = interface(IUnknown)
    ['{289F562F-EFC1-4652-AD1E-BDC57A07F245}']
    procedure SetProgressDescription(const AProgressDescription: TGMString; const ATextColor: COLORREF = 0); stdcall;
  end;


//IGMMultiPhaseProgress = interface(IGMSetProgressMax)
//  ['{5B90D670-915B-4900-BAEB-2B0C1633D6F3}']
//  procedure SetPhaseCount(const Value: LongInt);
//  procedure NextProgressPhase(const APhaseDescription: TGMString);
//  //procedure SetupPhaseProgressRange(const AMin, AMax: LongInt; const ACurrentProgress: LongInt = cInvalidProgress);
//end;


  procedure GMSetProgressAndCheckCanceled(const AProgresssable: IUnknown; const AProgress: Int64; var ACancel: BOOL; const ACalcProgressKind: TGMCalcProgressKind = cpkAbsolute);
  procedure GMSetProgressMax(const AProgresssable: IUnknown; const AProgressMax: Int64);
  procedure GMSetProgressDescription(const AProgresssable: IUnknown; const AProgressDescription: TGMString; const ATextColor: COLORREF = 0);

  type

  IGMAppendText = interface(IUnknown)
    ['{3E9D2E3A-7B1A-4757-A7C6-89DE6FF08DD8}']
    procedure AppendText(const Text: TGMString; const Color: COLORREF = 0; const Bold: Boolean = False; const AFontSize: LongInt = 0);
  end;


  IGMAppendTextFromDLL = interface(IUnknown)
    ['{C3502DE5-296C-4274-9283-B8625EA5C639}']
    procedure AppendTextFromDLL(const Text: PGMChar; const Color: COLORREF = 0; const Bold: Boolean = False; const AFontSize: LongInt = 0); stdcall;
  end;



  IGMGetOperationCanceled = interface(IUnknown)
    ['{2DC47225-B370-4C0A-BA1A-4A6FAE9044B4}']
    function GetOperationCanceled: BOOL; stdcall;
    property OperationCanceled: BOOL read GetOperationCanceled;
  end;



  { ------------------- }
  { ---- Questions ---- }
  { ------------------- }

  TGMBoolAskResult = (barUnknown, barFalse, barTrue);

  TGMAskBoolValueId = (// bvFirstByteRead, bvDisplayValue, bvDisplayText, bvDataFetchNeeded,
                       bvAlwaysNotify, bvIsMultiLine, bvPositionalInsert,
                       //bvIsSigned, bvIsAutoIncrementing,
                       bvConfirmDeletions, bvHasFocus, bvMatchCase, bvDoColorChange,
                       // bvCanSetPosition,
                       bvWantsActivationClick, bvCustomStoreValues);

  IGMAskBoolean = interface(IUnknown)
    ['{7AAFA581-DC29-4c9a-B182-0E3F2BB412BB}']
    //function AskBoolean(const ValueId: LongInt): TGMBoolAskResult; stdcall;
    function AskBoolean(const ValueId: LongInt): LongInt; stdcall;
  end;

  function GMBooleanAskResult(const AValue: Boolean): LongInt;
  function GMAskBoolean(const AObj: TObject; const AValueId: LongInt; const ADefaultResult: Boolean = False): Boolean; overload;
  function GMAskBoolean(const AIntf: IUnknown; const AValueId: LongInt; const ADefaultResult: Boolean = False): Boolean; overload;
  function GMAskUnkBoolean(const AIntf: IUnknown; const AValueId: LongInt): TGMBoolAskResult;
  function GMCheckAskBoolean(const AObj: TObject; const AValueId: LongInt; const ACallingName: TGMString = cDfltRoutineName): Boolean; overload;
  function GMCheckAskBoolean(const AIntf: IUnknown; const AValueId: LongInt; const ACallingName: TGMString = cDfltRoutineName): Boolean; overload;


  const cInvalidIntValue = Low(LongInt);

  type

  TGMAskIntValueId = (ivDataLength, ivMatchKind, ivId, ivImageIndex); // ivFieldCount,, ivCursorType, ivDataSize, ivDisplayWidth, ivStoredDataLength , ivMaxEditLength,

  IGMAskInteger = interface(IUnknown)
    ['{501B3BB4-BF5F-4b7a-8094-76FF8FF5C847}']
    function AskInteger(const ValueId: LongInt): LongInt; stdcall;
  end;

  function GMAskInteger(const AObj: TObject; const AValueId: LongInt; const ADefaultValue: LongInt = cInvalidIntValue): LongInt; overload;
  function GMAskInteger(const AIntf: IUnknown; const AValueId: LongInt; const ADefaultValue: LongInt = cInvalidIntValue): LongInt; overload;
  function GMCheckAskInteger(const AObj: TObject; const AValueId: LongInt; const ACallingName: TGMString = cDfltRoutineName): LongInt; overload;
  function GMCheckAskInteger(const AIntf: IUnknown; const AValueId: LongInt; const ACallingName: TGMString = cDfltRoutineName): LongInt; overload;


  type

  IGMMapIntegerOnInteger = interface(IUnknown)
    ['{1BC6F7EB-C230-48ba-B383-B332D555DA6F}']
    function MapIntegerOnInteger(const MapValue: PtrInt): PtrInt; stdcall;
  end;


  //IGMIntegerMapChange = interface(IUnknown)
  //  ['{1EEF030C-E9BA-478f-809F-81E0EEE100B6}']
  //  procedure IntegerMapChanged(const Value: LongInt); stdcall;
  //end;


  { -------------------- }
  { ---- Operations ---- }
  { -------------------- }

  TGMOperation = (roEdit, roInsert, roDelete, roCancelChanges, roApplyChanges, roRefreshCurrent, roReExecuteStatement, roScheduleReExecution, roLeaveModifyingState, roSetSimplestConfiguration,
                  // opClear, // opFetchData, opInvalidate, opCompressData, opUncompressData,
                  goShow, goHide, goSetFocus, goKillFocus, goRebuildContextMenu, goInitialize, goEnable, goDisable,
                  noNotifyDataChange);


  IGMCanExecuteOperation = interface(IUnknown)
    ['{9BA70E7C-CB2C-4404-8203-2BD68ABED85B}']
    function CanExecuteOperation(const Operation: LongInt; const Parameter: IUnknown = nil): Boolean; stdcall;
  end;


  IGMExecuteOperation = interface(IUnknown)
    ['{DCC23FA6-D77E-44c9-95BA-DFAA264451FD}']
    function ExecuteOperation(const Operation: LongInt; const Parameter: IUnknown = nil): Boolean; stdcall;
  end;


  function GMCanExecOperation(const AIntf: IUnknown; const Operation: LongInt; const Parameter: IUnknown = nil): Boolean;

  function GMExecuteOperation(const AObj: TObject; const Operation: LongInt; const Parameter: IUnknown = nil): Boolean; overload;
  function GMExecuteOperation(const AIntf: IUnknown; const Operation: LongInt; const Parameter: IUnknown = nil): Boolean; overload;

  procedure GMCheckExecOperation(const AObj: TObject; const Operation: LongInt; const OperationName: TGMString; const ACallingName: TGMString = cDfltRoutineName; const Parameter: IUnknown = nil); overload;
  procedure GMCheckExecOperation(const AIntf: IUnknown; const Operation: LongInt; const OperationName: TGMString; const ACallingName: TGMString = cDfltRoutineName; const Parameter: IUnknown = nil); overload;


  type

  IGMOperationNotifications = interface(IUnknown)
    ['{B9FCB8C1-10D6-41fe-83A3-6F85140419FC}']
    procedure BeforeOperation(const Operation: LongInt; const Parameter: IUnknown = nil); stdcall;
    procedure AfterOperation(const Operation: LongInt; const Parameter: IUnknown = nil); stdcall;
  end;


  TGMRefreshGranularity = (rgNone, rgRefreshPosition, rgRefreshCurrent, rgRefeshComplete);

  IGMEnableNotifications = interface(IUnknown)
    ['{D59116D7-62F4-4408-A482-DB2E2B8A5ACD}']
    function GetNotifyDisableCount: LongInt; stdcall;
    function EnableNotifications(const NotificationOnReEnable: LongInt = Ord(rgNone)): LongInt; stdcall;
    function DisableNotifications(const NotificationOnFirstDisable: LongInt = Ord(rgNone)): LongInt; stdcall;
    property NotifyDisableCount: LongInt read GetNotifyDisableCount;
  end;


  { ----------------- }
  { ---- Cursors ---- }
  { ----------------- }

  //
  // Abstract cursor interfaces, not specialized to anything.
  //
  // Implemented easily and meaningful by:
  // =====================================
  //  - TStrem or other stream like classes
  //  - TDataset or any other Recordset
  //  - List boxes, Grids or Comboboxes
  //  - File access classes
  //


  //
  // Cusorposition Changes will be notified by the
  // IGMPositionChangeNotifications interface.
  //

  IGMUnidirectionalCursor = interface(IUnknown)
    ['{C3A9CA46-1ACD-11d5-AB38-000021DCAD19}']
    function GetBOF: Boolean; stdcall;
    function GetEOF: Boolean; stdcall;
    procedure MoveToNext; stdcall;
    property BOF: Boolean read GetBOF;
    property EOF: Boolean read GetEOF;
  end;


  IGMBidirectionalCursor = interface(IGMUnidirectionalCursor)
    ['{C3A9CA47-1ACD-11d5-AB38-000021DCAD19}']
    procedure MoveToPrevious; stdcall;
  end;


  IGMCursorFirstLast = interface(IGMBidirectionalCursor)
    ['{C3A9CA48-1ACD-11d5-AB38-000021DCAD19}']
    procedure MoveToFirst; stdcall;
    procedure MoveToLast; stdcall;
  end;


  TGMCursorMove = (cmFirst, cmPrior, cmNext, cmLast);

  function GMIsValidCursorMove(const AIntf: IUnknown; const Move: TGMCursorMove): Boolean;
  procedure GMMoveCursor(const AIntf: IUnknown; const Move: TGMCursorMove);
  procedure GMSafeCursorMove(const AIntf: IUnknown; const Move: TGMCursorMove);


  { ------------------ }
  { ---- Position ---- }
  { ------------------ }

  //
  // All things that have a Position, like:
  // ======================================
  //  - Streams
  //  - Files
  //  - Recordsets
  //  - Listboxes, comboboxes, Grids
  //  - Scrollbars
  //  - Progress Bars
  //

  type

  IGMGetPosition = interface(IUnknown)
    ['{C3A9CA4A-1ACD-11d5-AB38-000021DCAD19}']
    function GetPosition: PtrInt; stdcall;
    property Position: PtrInt read GetPosition;
  end;


  IGMGetSetPosition = interface(IGMGetPosition)
    ['{5DC49C45-1BBE-11d5-AB38-000021DCAD19}']
    procedure SetPosition(const Value: PtrInt); stdcall;
    property Position: PtrInt read GetPosition write SetPosition;
  end;


  IGMPositionChangeNotifications = interface(IUnknown)
    ['{290C3D43-1C5A-11d5-AB38-000021DCAD19}']
    //
    // If others want to be informed when a Position changes
    // they can implement this interface.
    //
    // To connect to the Position holder the IconnectionPointContainer/IConnectionPoint
    // interfaces should be used.
    //
    procedure BeforePositionChange; stdcall;
    procedure AfterPositionChange; stdcall;
  end;


  IGMScrollBar = interface(IGMGetSetPosition)
    ['{3835397C-7C0F-422A-BFD9-763CA068E96C}']
    function GetMinPosition: LongInt; stdcall;
    function GetMaxPosition: LongInt; stdcall;
    function GetPageSize: LongInt; stdcall;
    procedure SetMinPosition(const AValue: LongInt); stdcall;
    procedure SetMaxPosition(const AValue: LongInt); stdcall;
    procedure SetPageSize(const AValue: LongInt); stdcall;
    property MinPosition: LongInt read GetMinPosition write SetMinPosition;
    property MaxPosition: LongInt read GetMaxPosition write SetMaxPosition;
    property PageSize: LongInt read GetPageSize write SetPageSize;
  end;

  function GMGetIntfPosition(const AIntf: IUnknown; const ADefaultPos: PtrInt = CGMUnknownPosition): PtrInt;
  procedure GMSetIntfPosition(const AIntf: IUnknown; const APosition: PtrInt);
  procedure GMMovePosition(const AIntf: IUnknown; const ADelta: PtrInt);


  { ---------------- }
  { ---- colors ---- }
  { ---------------- }

  type

  IGMBkgndColor = interface(IUnknown)
    ['{26A00902-95B8-4854-BCC4-01E5C2E80135}']
    function BkgndColor: COLORREF; stdcall;
    //property BkgndColor: COLORREF read GetBkgndColor;
  end;


  IGMGetSetBkgndColor = interface(IGMBkgndColor)
    ['{2667B51E-1EB9-4c76-99AF-51232F4E13C6}']
    procedure SetBkgndColor(const Value: COLORREF; const Repaint: Boolean = True); stdcall;
    //property BkgndColor: COLORREF read IGMBkgndColor; //  write SetBkgndColor;
  end;


  { --------------- }
  { ---- Count ---- }
  { --------------- }

  IGMGetCount = interface(IUnknown)
    ['{93880081-2684-11d5-AB38-000021DCAD19}']
    function GetCount: PtrInt; stdcall;
    property Count: PtrInt read GetCount;
  end;


  IGMGetSetCount = interface(IGMGetCount)
    ['{93880082-2684-11d5-AB38-000021DCAD19}']
    procedure SetCount(const Value: PtrInt); stdcall;
    property Count: PtrInt read GetCount write SetCount;
  end;


  function GMGetIntfCount(const AIntf: IUnknown; const ADefaultValue: PtrInt = 0): PtrInt;
  function GMObjIsEmpty(const AObj: TObject; const ADefaultResult: Boolean = True): Boolean;
  function GMIntfIsEmpty(const AIntf: IUnknown; const ADefaultResult: Boolean = True): Boolean;


  { ---------------- }
  { ---- Offset ---- }
  { ---------------- }

  type

  IGMGetOffset = interface(IUnknown)
    ['{AC94E5A1-38AB-11d5-AB38-000021DCAD19}']
    function GetOffset: PtrInt; stdcall;
    property Offset: PtrInt read GetOffset;
  end;


  IGMGetSetOffset = interface(IGMGetOffset)
    ['{AC94E5A2-38AB-11d5-AB38-000021DCAD19}']
    procedure SetOffset(PtrInt: PtrInt); stdcall;
    property Offset: PtrInt read GetOffset write SetOffset;
  end;


  IGMShiftOffset = interface(IUnknown)
    ['{9A57A8D4-2DFE-4aa0-BE1E-6365C3FD2B68}']
    procedure SetOffsetAndShiftData(const NewOffset: LongInt); stdcall;
  end;


  { ------------------- }
  { ---- Placement ---- }
  { ------------------- }

  IGMGetLeft = interface(IUnknown)
    ['{F2589550-F6A9-46d4-AC88-D694B080C0DE}']
    function GetLeft: LongInt; stdcall;
    property Left: LongInt read GetLeft;
  end;


  {IGMGetSetLeft = interface(IGMGetLeft)
    ['567DD2AA-B5B3-4bc3-B68E-2649698A0C05']
    procedure SetLeft(const Value: LongInt); stdcall;
    property Left: LongInt read GetLeft write SetLeft;
  end;}


  {IGMGetTop = interface(IUnknown)
    ['9ABE0B27-B9BD-4c19-878E-30FBB67F9D75']
    function GetTop: LongInt; stdcall;
    property Top: LongInt read GetTop;
  end;}


  {IGMGetSetTop = interface(IGMGetTop)
    ['BBAB652D-AD89-4468-8DDB-C8AE8F20926A']
    procedure SetTop(const Value: LongInt); stdcall;
    property Top: LongInt read GetTop write SetTop;
  end;}


  {IGMGetWidth = interface(IUnknown)
    ['7416B560-6800-4bfb-9F72-FF21F3B91EEA']
    function GetWidth: LongInt; stdcall;
    property Width: LongInt read GetWidth;
  end;}


  {IGMGetSetWidth = interface(IGMGetWidth)
    ['CE61FDA8-FD92-47e6-BCDF-A5599FFEA6AB']
    procedure SetWidth(const Value: LongInt); stdcall;
    property Width: LongInt read GetWidth write SetWidth;
  end;}


  {IGMGetHeight = interface(IUnknown)
    ['E703AB11-7475-45c8-A2DC-ECFB83906E21']
    function GetHeight: LongInt; stdcall;
    property Height: LongInt read GetHeight;
  end;}


  {IGMGetSetHeight = interface(IGMGetHeight)
    ['7E5751E5-C243-4054-8180-A440BDA79DBE']
    procedure SetHeight(const Value: LongInt); stdcall;
    property Height: LongInt read GetHeight write SetHeight;
  end;}                                             


  {IGMSetBounds = interface(IUnknown)
    ['FDEECACF-47A4-4b6d-B2FA-F062FCD3F9D3']
    procedure SetBounds(ALeft, ATop, AWidth, AHeight: LongInt); stdcall;
  end;}


  { ------------------------------- }
  { ---- Assign#ments / Clones ---- }
  { ------------------------------- }

  IGMAssignFromObj = interface(IUnknown)
    ['{347F1FD6-9775-4612-A9D4-828415E0E274}']
    procedure AssignFromObj(const ASource: TObject); stdcall;
  end;


  IGMAssignToObj = interface(IUnknown)
    ['{76BE483F-13F2-408c-A6DC-75F2F10EA32A}']
    procedure AssignToObj(const ADest: TObject); stdcall;
  end;


  IGMAssignFromIntf = interface(IUnknown)
    ['{B4047CE2-4821-4b08-BC3F-0EFA7D2A0481}']
    procedure AssignFromIntf(const ASource: IUnknown); stdcall;
  end;


  IGMAssignToIntf = interface(IUnknown)
    ['{5DC7F8E2-00BD-487f-A8F6-1FCF8706CF7D}']
    procedure AssignToIntf(const ADest: IUnknown); stdcall;
  end;


  IGMCreateCopyQI = interface(IUnknown)
    ['{82C8DD54-9DEC-455f-9C94-CF0D6CC7E0A7}']
    //
    // Anything that allows to create a Copy of itself.
    // Specify the Interface you want use to communicate to
    // the newly created Object in the AIID Parameter.
    //
    function CreateCopyQI(const AIID: TGUID; out AIntf): HResult; stdcall;
  end;


  { -------------------- }
  { ---- Attributes ---- }
  { -------------------- }

  //
  // Bitflag Field interpreted as a Set of Boolean Attributes.
  //

  IGMGetAttributes = interface(IUnknown)
    ['{5771C661-3BF8-11d5-AB38-000021DCAD19}']
    function GetAttributes: Longword; stdcall;
    property Attributes: Longword read GetAttributes;
  end;


  IGMGetSetAttributes = interface(IGMGetAttributes)
    ['{E94E2166-597A-4937-A700-FBE37EEDE291}']
    procedure SetAttributes(const Value: Longword); stdcall;
    property Attributes: Longword read GetAttributes write SetAttributes;
  end;


  { ---------------------- }
  { ---- User Account ---- }
  { ---------------------- }

  IGMUsernameAndPassword = interface(IUnknown)
    ['{CAE0D477-6191-4502-A4B6-7FA36BCC250B}']
    function GetUsername: TGMString;
    procedure SetUsername(const ABypass: TGMString);

    function GetPassword: TGMString;
    procedure SetPassword(const ABypass: TGMString);

    property Username: TGMString read GetUsername write SetUsername;
    property Password: TGMString read GetPassword write SetPassword;
  end;


  IGMUserAccount = interface(IUnknown)
    ['{42129443-AFC0-4103-91E1-0ED04AB8584B}']
    function GetUsername: PGMChar; stdcall;
    function GetPassword: PGMChar; stdcall;
//  function GetDomain: PGMChar; stdcall;
    function GetSaveUserData: Boolean; stdcall;
    procedure SetUsername(AuserName: PGMChar); stdcall;
    procedure SetPassword(APassword: PGMChar); stdcall;
//  procedure SetDomain(ADomain: PGMChar); stdcall;
    procedure SetSaveUserData(Value: Boolean); stdcall;
    property Username: PGMChar read GetUsername write SetUsername;
    property Password: PGMChar read GetPassword write SetPassword;
//  property Domain: PGMChar read GetDomain write SetDomain;
    property SaveUserData: Boolean read GetSaveUserData write SetSaveUserData;
  end;


  { ----------------- }
  { ---- Strings ---- }
  { ----------------- }

  {IGMStrings = interface(IUnknown)
    ['FA99E2AC-F393-417a-AA19-5C9501B200B5']
    //
    // Cannot be marshalled
    //
    procedure BeginUpdate;
    procedure EndUpdate;
    
    function Get(Index: LongInt): TGMString;
    function GetSorted: Boolean;
    procedure Put(Index: LongInt; const S: TGMString);
    procedure SetSorted(Value: Boolean);

    procedure Clear;
    function GetCount: LongInt;
    function IsEmpty: Boolean;

    function Add(const Value: TGMString): LongInt;
    procedure Delete(Index: LongInt);

    function IndexOf(const Value: TGMString): LongInt;
    function IndexOfNearest(const Value: TGMString): LongInt;
    function Find(const Value: TGMString; var Index: LongInt): Boolean;
    
    property Count: LongInt read GetCount;
    property Strings[Index: LongInt]: TGMString read Get write Put; default;
    property Sorted: Boolean read GetSorted write SetSorted;
  end;}


  { ------------------------ }
  { ---- Storing Values ---- }
  { ------------------------ }

  PGMCryptCtrlData = ^RGMCryptCtrlData;
  RGMCryptCtrlData = record
    KeyData: AnsiString;
    HashAlgoID: Cardinal;
    CryptAlgoID: Cardinal;
  end;


  TGMReadValStrFunc = function(const AValueName: TGMString; const ADefaultValue: TGMString = ''): TGMString of object; stdcall;
  TGMWriteValStrProc = procedure(const AValueName, AValue: TGMString) of object; stdcall;

  IGMStringStorage = interface(IUnknown)
    ['{6C1E6792-ED8D-4c16-A49E-12CB62F61E7E}']
    function ReadString(const ValueName: TGMString; const ADefaultValue: TGMString = ''): TGMString; stdcall;
    procedure WriteString(const ValueName, Value: TGMString); stdcall;
  end;


  IGMValueStorage = interface(IGMStringStorage)
    ['{5B79A555-D8BA-4062-B84E-A08AD279194B}']
    function ReadInteger(const ValueName: TGMString; const ADefaultValue: LongInt = cDfltReadInteger): LongInt; stdcall;
    function ReadInt64(const ValueName: TGMString; const ADefaultValue: Int64 = cDfltReadInteger): Int64; stdcall;
    function ReadBoolean(const ValueName: TGMString; const ADefaultValue: Boolean = cDfltReadBoolean): Boolean; stdcall;
    function ReadDateTime(const ValueName: TGMString; const ADefaultValue: Double = cDfltReadDateTime): Double; stdcall;
    function ReadDouble(const ValueName: TGMString; const ADefaultValue: Double = cDfltReadFloat): Double; stdcall;
    function ReadVariant(const ValueName: TGMString; const ADefaultValue: OleVariant): OleVariant; stdcall;
    function ReadUnionValue(const ValueName: TGMString; const ADefaultValue: RGMUnionValue): RGMUnionValue; stdcall;
    procedure WriteInteger(const ValueName: TGMString; const Value: LongInt); stdcall;
    procedure WriteInt64(const ValueName: TGMString; const Value: Int64); stdcall;
    procedure WriteBoolean(const ValueName: TGMString; const Value: Boolean); stdcall;
    procedure WriteDateTime(const ValueName: TGMString; const Value: Double); stdcall;
    procedure WriteDouble(const ValueName: TGMString; const Value: Double); stdcall;
    procedure WriteVariant(const ValueName: TGMString; const Value: OleVariant); stdcall;
    procedure WriteUnionValue(const ValueName: TGMString; const Value: RGMUnionValue); stdcall;
  end;

  TGMLoadStoreValuesProc = procedure (const AStorage: IGMValueStorage; const ACryptCtrlData: PGMCryptCtrlData = nil) of object;


  //IGMBinaryStorage = interface(IUnknown)
  //  ['EF4C4B29-EFEA-4e2d-B483-7EA3DAA608FA}']
  //  function ReadBinary(const ValueName: TGMString; out Data; const DataSize: LongInt; const ZeroInit: Boolean = True): LongWord; stdcall;
  //  procedure WriteBinary(const ValueName: TGMString; const Data; const DataSize: LongInt); stdcall;
  //end;

  IGMContainsValue = interface(IUnknown)
    ['{EFF05048-5460-4A8E-B6D1-2FCCDB184CC2}']
    function ContainsValue(const ValueName: TGMString): Boolean; stdcall;
  end;

  
  IGMValueStorageDirectory = interface(IGMContainsValue)
    ['{E8696A1E-B597-46cd-ABA3-0376EC82609D}']
    function OpenDir(const DirPath: TGMString; const CreateIfNotExist: Boolean = False): Boolean; stdcall;
//  function DirExists(const DirPath: TGMString): Boolean; stdcall;
    procedure ReadSubDirNames(var SubDirNames: TGMStringArray); stdcall;
    procedure ReadValueNames(var ValueNames: TGMStringArray); stdcall;
    function DeleteValue(const ValueName: TGMString): Boolean; stdcall;
    function DeleteDir(const DirPath: TGMString): Boolean; stdcall;
    function CurrentPath: TGMString; stdcall;
    procedure Commit; stdcall;

    function GetBasePath: TGMString; stdcall;
    function GetRootKey: HKEY; stdcall;
    procedure SetBasePath(const Value: TGMString); stdcall;
    procedure SetRootKey(const Value: HKEY); stdcall;

    property BasePath: TGMString read GetBasePath write SetBasePath;
    property RootKey: HKEY read GetRootKey write SetRootKey;
  end;


  IGMLoadStoreData = interface(IUnknown)
    ['{D8D48DE1-AE80-4132-AE40-ECA66F9256C6}']
    procedure LoadData(const ASource: IGMValueStorage; const ACryptCtrlData: PGMCryptCtrlData = nil); stdcall;
    procedure StoreData(const ADest: IGMValueStorage; const ACryptCtrlData: PGMCryptCtrlData = nil); stdcall;
  end;



  { ---------------------------- }
  { ----  Content Container ---- }
  { ---------------------------- }

  IGWriteToStream = interface(IUnknown)
    ['{AEFAA5A9-50C8-43F3-95ED-E0645E74791E}']
    procedure WriteToStream(const ADestStream: ISequentialStream; const ACharCoding: TGMCharKind = ckUtf8; const AIndent: TGMString = '');
  end;


  { ---------------- }
  { ----  Trees ---- }
  { ---------------- }

  IGMTreeable = Interface(IUnknown)
    ['{72EF5208-0379-4C0B-BBDF-4AB64B9050F3}']
    function Parent: IGMTreeable; // <- Level up
    function FirstChild: IGMTreeable; // <- Level down
    function NextSibling: IGMTreeable; // <- Next neighbour on same Level
    function PrevSibling: IGMTreeable; // <- Previous neighbour on same Level
  end;


  IGMCreateTreeNodeWithDataObj = interface(IUnknown)
    ['{6ED27D2A-0349-490E-A42D-F238DCF77D87}']
    function CreateTreeNodeWithDataObj(const ASource: IGMValueStorage; const ParentNode: IGMTreeable; const Parameters: IUnknown = nil): IGMTreeable; // stdcall;
  end;


  IGMCreateNewTreeNode = interface(IUnknown)
    ['{E1AC135B-AB0D-4228-B945-C4B111EE884E}']
    function CreateNewTreeNode(const ParentNode: IGMTreeable; const Title: TGMString; const ImgIdx, SelectedImgIdx: Integer; const DataObj: TObject = nil; const Parameters: IUnknown = nil): IGMTreeable; stdcall;
  end;


  IGMGetTreeNodeFromRaw = interface(IUnknown)
    ['{A1531AA5-68B5-4FD9-A2DB-85F0CD37880B}'] 
    function GetTreeNodeFromRaw(const ANode: Pointer): IGMTreeable; stdcall;
  end;


  IGMGetDataObject = interface(IUnknown)
    ['{66E08396-AD93-4996-B4A0-E506085CAD49}']
    function GetDataObject: TObject; stdcall;
  end;


  IGMSetDataObject = interface(IUnknown)
    ['{F0520DB3-D54A-4E9A-B57A-D2BD2F91F796}']
    procedure SetDataObject(const Value: TObject); stdcall;
  end;

  //
  // Return value True continues iteratiing further nodes, False stops iteration at current node
  //
  TGMNodeVisitFunc = function(const Node: IGMTreeable; const Parameter: Pointer = nil): Boolean;
  TGMNodeVisitMethod = function(const Node: IGMTreeable; const Parameter: Pointer = nil): Boolean of object;

  PGMNodeVisitData = ^RGMNodeVisitData;
  RGMNodeVisitData = record
    DataClass: TClass;
    NodeTitle: TGMString;
    Node: IGMTreeable;
    Index, SearchIdx: LongInt;
  end;


  function GMTreeableNodeLevel(ANode: IGMTreeable): Integer;

  function GMInitNodeVisitData(const ADataClass: TClass; const ANodeTitle: TGMString = ''; const ANode: IGMTreeable = nil; const ASearchIndex: LongInt = 0): RGMNodeVisitData;

  function GMGetDataObject(const AOwner: IUnknown): TObject;
  function GMSetDataObject(const AOwner: IUnknown; const ADataObj: TObject): TObject;

  function GMIntfHasDataClass(const AIntf: IUnknown; const ADataClass: TClass): Boolean;
  function GMGetIntfDataClass(const AIntf: IUnknown): TClass;

  function GMVisitNodesRootFirst(const AStartNode: IGMTreeable;
                                 const AVisitFunc: TGMNodeVisitFunc;
                                 const ARecurse: Boolean = True;
                                 const AParameter: Pointer = nil): Boolean; overload;

  function GMVisitNodesRootFirst(const AStartNode: IGMTreeable;
                                 const AVisitFunc: TGMNodeVisitMethod;
                                 const ARecurse: Boolean = True;
                                 const AParameter: Pointer = nil): Boolean; overload;


  function GMVisitNodesDepthFirst(const AStartNode: IGMTreeable;
                                  const AVisitFunc: TGMNodeVisitFunc;
                                  const ARecurse: Boolean = True;
                                  const AParameter: Pointer = nil): Boolean; overload;

  function GMVisitNodesDepthFirst(const AStartNode: IGMTreeable;
                                  const AVisitFunc: TGMNodeVisitMethod;
                                  const ARecurse: Boolean = True;
                                  const AParameter: Pointer = nil): Boolean; overload;


  function GMIsNodeMatch(ANode: IGMTreeable; const ADataClass: TClass; const ANodeTitle: TGMString = ''): Boolean;
  function GMFindParentNode(const AStartNode: IGMTreeable; const ADataClass: TClass; const ANodeTitle: TGMString = ''): IGMTreeable;

  function GMFindNode(const AStartNode: IGMTreeable; const ADecideFunc: TGMNodeVisitFunc; const ARecurse: Boolean; const AParameter: Pointer = nil): IGMTreeable;  overload;
  function GMFindNode(const AStartNode: IGMTreeable; const ADataClass: TClass; const ANodeTitle: TGMString = ''; const ARecurse: Boolean = True): IGMTreeable; overload;

  function GMFindRootNode(const ANode: IUnknown): IGMTreeable;

  function GMNodePath(const ANode: IUnknown; const ADelimStr: TGMString = '\'): TGMString;


  { ---------------------------------------------------------------- }
  { ----  Classes supporting reference counted lifetime control ---- }
  { ---------------------------------------------------------------- }

  type

  TGMRefCountedObj = class(TObject, IUnknown, IGMObjInfo, IGMCreateCopyQI)
   protected
    FRefLifeTime: Boolean;
    FRefCount: LongInt;

   public
    constructor Create(const ARefLifeTime: Boolean = False); virtual;

    procedure AfterConstruction; override;
    procedure BeforeDestruction; override;
    destructor Destroy; override;
    procedure OnFinalRelease; virtual;

    // IGMObjInfo
    function GetClassName: TGMString;
    function GetClassType: TClass;
    function GetInstance: TObject;
    function GetTypeInfo: PTypeInfo;
    function GetRefCount: LongInt;

    // IUnknown
    function QueryInterface({$IFDEF FPC_HAS_CONSTREF}constref{$ELSE}const{$ENDIF} AIID: TGUID; out AIntf): HResult; virtual; {$IFNDEF WINDOWS}cdecl{$ELSE}stdcall{$ENDIF};
    function _AddRef: LongInt; virtual; {$IFNDEF WINDOWS}cdecl{$ELSE}stdcall{$ENDIF};
    function _Release: LongInt; virtual; {$IFNDEF WINDOWS}cdecl{$ELSE}stdcall{$ENDIF};

    // IGMCreateCopyQI
    function CreateCopyQI(const AIID: TGUID; out AIntf): HResult; virtual; stdcall;

    property RefCount: LongInt read FRefCount;
    property RefLifeTime: Boolean read FRefLifeTime write FRefLifeTime;
  end;

  TGMRefCountedObjClass = class of TGMRefCountedObj;


  TGMAggregatableObj = class(TGMRefCountedObj, IUnknown)
   //
   // Intended to be aggregated to/by another class via "implements" compiler featrue.
   // Must not be created with RefLifetime when used as an aggregate (Interface delegation member).
   //
   // If used as interface delegation member the owner must refernece this class by a normal
   // object member and not an interface member. Because reference counts are routed back to the
   // owner by this class a cyclic reference by interfaces would keep the owner forever. For the same reason
   // this class must not reference other delegation classes of the owner by interfaces.
   //
   // Can be created standalone (AOwner = nil) too, with or without RefLifeTime.
   //
   protected
    FOwner: Pointer; // <- cannot be a Interface reference, would cause circular reference

   public
    constructor Create(const AOwner: IUnknown; const ARefLifeTime: Boolean = False); virtual; reintroduce; overload;

    function QueryInterface({$IFDEF FPC_HAS_CONSTREF}constref{$ELSE}const{$ENDIF} AIID: TGUID; out AIntf): HResult; override;
    function _AddRef: LongInt; override;
    function _Release: LongInt; override;

    function GetOwner: IUnknown;
    function GetOwnerObj: TObject;
    property Owner: IUnknown read GetOwner;
    property OwnerObj: TObject read GetOwnerObj;
  end;



  {$IFDEF DELPHIVCL}
  TGMRefLifePersistent = class(TPersistent, IUnknown, IGMObjInfo)
   protected
    FConstructed: Boolean;
    FRefCount: LongInt;
    FRefLifeTime: Boolean;

   public
    constructor Create(const ARefLifeTime: Boolean = False);
    procedure AfterConstruction; override;
    destructor Destroy; override;
    procedure OnFinalRelease; virtual;

    // IGMObjInfo
    function GetClassName: TGMString;
    function GetClassType: TClass;
    function GetInstance: TObject;
    function GetTypeInfo: PTypeInfo;

    // IUnknown
    function QueryInterface({$IFDEF FPC_HAS_CONSTREF}constref{$ELSE}const{$ENDIF} AIID: TGUID; out AIntf): HResult; virtual; {$IFNDEF WINDOWS}cdecl{$ELSE}stdcall{$ENDIF};
    function _AddRef: LongInt; virtual; {$IFNDEF WINDOWS}cdecl{$ELSE}stdcall{$ENDIF};
    function _Release: LongInt; virtual; {$IFNDEF WINDOWS}cdecl{$ELSE}stdcall{$ENDIF};

    property RefCount: LongInt read FRefCount;
    property RefLifeTime: Boolean read FRefLifeTime write FRefLifeTime;
  end;
  {$ENDIF}


  {$IFDEF DELPHIVCL}
  TGMRefLifeComponent = class;

  TGMRefLifeComponentClass = class of TGMRefLifeComponent;

  TGMRefLifeComponent = class(TComponent, IUnknown,
                                          IGMCreateCopyQI,
                                          IGMAssignFromObj,
                                          IGMAssignToObj,
                                          IGMGetName,
                                          IGMObjInfo)
   protected
    FConstructed: Boolean;
    FRefCount: LongInt;
    FRefLifeTime: Boolean;

    function CopyCreateClass: TGMRefLifeComponentClass;

   public
    constructor CreateIntf; // virtual; <- not needed, calls virtual create constructor
    procedure AfterConstruction; override;
    destructor Destroy; override;
    procedure OnFinalRelease; virtual;

    // IGMObjInfo
    function GetClassName: TGMString;
    function GetClassType: TClass;
    function GetInstance: TObject;
    function GetTypeInfo: PTypeInfo;

    // IGMGetName
    function GetName: TGMString; virtual; stdcall;

    // IGMAssignByObj
    procedure AssignFromObj(const ASource: TObject); virtual; stdcall;
    procedure AssignToObj(const ADest: TObject); virtual; stdcall;

    // IGMCreateCopyQI
    function CreateCopyQI(const AIID: TGUID; out AIntf): HResult; virtual; stdcall;

    // IUnknown
    function QueryInterface({$IFDEF FPC_HAS_CONSTREF}constref{$ELSE}const{$ENDIF} AIID: TGUID; out AIntf): HResult; virtual; {$IFNDEF WINDOWS}cdecl{$ELSE}stdcall{$ENDIF};
    function _AddRef: LongInt; virtual; {$IFNDEF WINDOWS}cdecl{$ELSE}stdcall{$ENDIF};
    function _Release: LongInt; virtual; {$IFNDEF WINDOWS}cdecl{$ELSE}stdcall{$ENDIF};

    property RefCount: LongInt read FRefCount;
    property RefLifeTime: Boolean read FRefLifeTime write FRefLifeTime;
  end;
  {$ENDIF}


  TCPCNotifyParams = array of RGMUnionValue; // OleVariant
  TCPCNotifyProc = procedure (const NotifySink: IUnknown; const Params: TCPCNotifyParams);
  TCPCNotifyMethod = procedure (const NotifySink: IUnknown; const Params: TCPCNotifyParams) of object;


  { ---------------------------------- }
  { ---- Synchronization Classes  ---- }
  { ---------------------------------- }

  TGMCriticalSection = class(TGMRefCountedObj, IGMCriticalSection)
   protected
    FCriticalSection: TRTLCriticalSection;
   public
    constructor Create(const ARefLifeTime: Boolean = True); override;
    destructor Destroy; override;
    procedure EnterCriticalSection;
    procedure LeaveCriticalSection;
    function TryEnterCriticalSection: Boolean; 
  end;


  TGMCriticalSectionLock = class(TGMRefCountedObj)
   protected
    FCriticalSection: IGMCriticalSection;
   public
    constructor Create(const ACriticalSection: IUnknown; const ARefLifeTime: Boolean = True); reintroduce;
    destructor Destroy; override;
  end deprecated 'Use RGMCriticalSectionLock instead';


  RGMCriticalSectionLock = record
   private
    LockCount: Int64;
    CriticalSection: IGMCriticalSection;

    class operator Initialize(var ACriticalSectionLock: RGMCriticalSectionLock);
    class operator Finalize(var ACriticalSectionLock: RGMCriticalSectionLock);

   public
    procedure Lock(const ACriticalSection: IGMCriticalSection); overload;
    procedure Lock(const ACriticalSection: IUnknown); overload;
    procedure Lock(const ACriticalSection: TObject); overload;
    procedure LockAgain;
    procedure Unlock;
    procedure UnlockAll;
    function GetLockCount: Int64;
  end;


  { ------------------------ }
  { ---- Smart Classes  ---- }
  { ------------------------ }

  {ToDo: Change TGMCOMInitializer to managed record}
  TGMCOMInitializer = class(TGMRefCountedObj)
   protected
    FInitialized: Boolean;
   public
    constructor Create(const ACoInitFlags: DWORD = cDfltCoInitFlags; const AHrCheck: Boolean = True; const ARefLifeTime: Boolean = True); reintroduce;
    destructor Destroy; override;
  end;


  {ToDo: Change TGMNotificationDisabler to managed record}
  TGMNotificationDisabler = class(TGMRefCountedObj)
   protected
    FNotifier: IGMEnableNotifications;
    FNotificationOnReEnable: LongInt;
   public
    constructor Create(const ANotifier: IUnknown; const ANotificationOnReEnable: LongInt = Ord(rgNone); const ANotificationOnFirstDisable: LongInt = Ord(rgNone); const ARefLifeTime: Boolean = True); reintroduce;
    destructor Destroy; override;
  end;


  {ToDo: Change TGMPositionKeeper to managed record}
  TGMPositionKeeper = class(TGMRefCountedObj)
   protected
    FPosition: LongInt;
    FObjWithPosition: IGMGetSetPosition;
   public
    constructor Create(const AObjWithPosition: IUnknown; const ANewposition: LongInt = -1; const ARefLifeTime: Boolean = True); reintroduce;
    destructor Destroy; override;
  end;


  {ToDo: Change TGMStateKeeper to managed record}
  TGMStateKeeper = class(TGMRefCountedObj)
   protected
    FObjWithState: IGMSaveRestoreState;
    FState: IUnknown;
   public
    constructor Create(const AObjWithState: IUnknown; const ARefLifeTime: Boolean = True); reintroduce;
    destructor Destroy; override;
  end;


  {ToDo: Change TGMQuietStateKeeper to managed record}
  TGMQuietStateKeeper = class(TGMRefCountedObj)
   protected
    FStateKeeper: IUnknown; // <- Released First
    FNotificationDisabler: IUnknown; // <- Released Second
   public
    constructor Create(const AObjWithState: IUnknown; const ANotificationOnReEnable: LongInt = Ord(rgNone); const ANotificationOnFirstDisable: LongInt = Ord(rgNone); const ARefLifeTime: Boolean = True); reintroduce;
  end;


  {ToDo: Change TGMActiveKeeper to managed record}
  TGMActiveKeeper = class(TGMRefCountedObj)
   protected
    FActivatableObj: IGMGetSetActive;
    FWasActive: Boolean;
   public
    constructor Create(const AActivatableObj: IUnknown; const ANewActive: Boolean = True; const ARefLifeTime: Boolean = True); reintroduce;
    destructor Destroy; override;
  end;


  {ToDo: Change TGMVsdDirPathKeeper to managed record}
  TGMVsdDirPathKeeper = class(TGMRefCountedObj)
   protected
    FValStorageDir: IGMValueStorageDirectory;
    FOldDirPath: TGMString;
   public
    constructor Create(const AValStorageDir: IUnknown; const ANewDirPath: TGMString = ''; const ACreateIfNotExists: Boolean = False; const ARefLifeTime: Boolean = True); reintroduce;
    destructor Destroy; override;
  end;


  {ToDo: Change TGMIStreamPosKeeper to managed record}
  TGMIStreamPosKeeper = class(TGMRefCountedObj)
   protected
    FStream: IStream;
    FOldPos: Int64;
   public
    constructor Create(const AStream: IUnknown; AStartPos: Int64 = cCurrentStrmPos; const ARefLifeTime: Boolean = True); reintroduce;
    destructor Destroy; override;
  end;


  {TGMPaintDisabler = class(TGMRefCountedObj)
   protected                                                                                
    FPaintObj: IGMEnableDisablePaint;
   public
    constructor Create(const APaintObj: IUnknown; const ARefLifeTime: Boolean = True);
    destructor Destroy; override;
  end;}


{ ---- Key Value Directory Routines ---- }

function GMVsdOpenDir(const AStorage: IUnknown; const ADirPath: TGMString; const ACreateIfNotExist: Boolean = False): Boolean;
function GMVsdDirExists(const AStorage: IUnknown; const ADirPath: TGMString): Boolean;
function GMVsdOpenAbsDir(const AStorage: IUnknown; const ADirPath: TGMString; const ACreateIfNotExist: Boolean = False): Boolean;
function GMVsdDeleteDir(const AStorage: IUnknown; const ADirPath: TGMString): Boolean;
function GMVsdDeleteAbsDir(const Storage: IUnknown; const ADirPath: TGMString): Boolean;
function GMVsdContainsValue(const AStorage: IUnknown; const AValueName: TGMString): Boolean;
function GMVsdDeleteValue(const AStorage: IUnknown; const AValueName: TGMString): Boolean;
//function GMVsdValueNameExists(const AStorage: IUnknown; const ValueName: TGMString): Boolean;
//procedure GMVsdDeletePrefixedValues(const AStorage: IUnknown; const PrefixStr: TGMString);
//procedure GMVsdDeleteValues(const AStorage: IUnknown);
procedure GMVsdReadSubDirNames(const AStorage: IUnknown; var ASubDirNames: TGMStringArray);
procedure GMVsdReadValueNames(const AStorage: IUnknown; var AValueNames: TGMStringArray);
procedure GMVsdCreatePath(const AStorage: IUnknown; const ADirPath: TGMString);
function GMVsdCurrentPath(const AStorage: IUnknown): TGMString;
procedure GMVsdCommit(const AStorage: IUnknown);

procedure GMVsdLoadFromDir(const ASource: IGMValueStorage; const ALoadPath: TGMString; const ALoadProc: TGMLoadStoreValuesProc; const ACryptCtrlData: PGMCryptCtrlData = nil); overload;
procedure GMVsdLoadFromDir(const ASource: IGMValueStorage; const ALoadPath: TGMString; const ALoadIntf: IUnknown; const ACryptCtrlData: PGMCryptCtrlData = nil); overload;
procedure GMVsdStoreToDir(const ADest: IGMValueStorage; const AStorePath: TGMString; const AStoreProc: TGMLoadStoreValuesProc; const ACryptCtrlData: PGMCryptCtrlData = nil); overload;
procedure GMVsdStoreToDir(const ADest: IGMValueStorage; const AStorePath: TGMString; const AStoreIntf: IUnknown; const ACryptCtrlData: PGMCryptCtrlData = nil); overload;

procedure GMVsdLoadTree(const ASource: IGMValueStorage;
                        const AParentTreeNode: IGMTreeable;
                        const ATreeNodeCreator: IGMCreateTreeNodeWithDataObj;
                        const AParameter: IUnknown = nil;
                        const ACryptCtrlData: PGMCryptCtrlData = nil;
                        const ASubDirSeparator: TGMString = scSubNodesDirSeparator);

procedure GMStoreTree(const ADest: IGMValueStorage; ANode: IGMTreeable; var ANodeIdx: LongInt; const AStoreSiblings: Boolean; const ACryptCtrlData: PGMCryptCtrlData = nil);

procedure GMVsdCopyDirValues(const ASource, ADest: IGMStringStorage);
procedure GMVsdCopyStorageContents(const ASource, ADest: IUnknown; AStartDirPath: TGMString = '');

procedure GMStoreString(const ADest: IGMStringStorage; const AValueName, AValue: TGMString; const ADefaultValue: TGMString = cDfltReadString);
procedure GMStoreInteger(const ADest: IGMValueStorage; const AValueName: TGMString; const AValue: LongInt; const ADefaultValue: LongInt = cDfltReadInteger);
procedure GMStoreInt64(const ADest: IGMValueStorage; const AValueName: TGMString; const AValue: Int64; const ADefaultValue: Int64 = cDfltReadInteger);
procedure GMStoreBoolean(const ADest: IGMValueStorage; const AValueName: TGMString; const AValue: Boolean; const ADefaultValue: Boolean = cDfltReadBoolean);
procedure GMStoreDateTime(const ADest: IGMValueStorage; const AValueName: TGMString; const AValue: TDateTime; const ADefaultValue: TDateTime = cDfltReadDateTime);
procedure GMStoreDouble(const ADest: IGMValueStorage; const AValueName: TGMString; const AValue: Double; const ADefaultValue: Double = cDfltReadFloat);
procedure GMStoreVariant(const ADest: IGMValueStorage; const AValueName: TGMString; const AValue, ADefaultValue: OleVariant);


{ ---- ILockByte Routines ---- }

function GMLockByteSize(const LockBytes: ILockBytes): Int64;
procedure GMCopyLockBytes(const ASource, ADest: ILockBytes; const AMaxBytesToCopy: LongInt = 0; const AVerfy: Boolean = True);
procedure GMLockByteSafeReadAt(const ASource: ILockBytes; const AOffset: Int64; const APData: Pointer; const ACount: LongInt; const ACallingName: TGMString = cDfltRoutineName);
procedure GMLockByteSafeWriteAt(const ADest: ILockBytes; const AOffset: Int64; const APData: Pointer; const ACount: LongInt; const ACallingName: TGMString = cDfltRoutineName);


{ ---- Istream Routines ---- }

function GMIStreamSize(const AStream: IUnknown): Int64;
function GMIStreamPos(const AStream: IUnknown): Int64;
function GMSetIStreamAbsPos(const AStream: IUnknown; const ANewPos: Int64; const ACallingName: TGMString = cDfltRoutineName): Int64;
function GMIStreamRead(const ASource: ISequentialStream; const Data: Pointer; const DataSizeInBytes: LongWord): Cardinal;
procedure GMSafeIStreamRead(const ASource: ISequentialStream; const AData: Pointer; const ADataSizeInBytes: LongWord; const ACallingName: TGMString = cDfltRoutineName);
procedure GMSafeIStreamWrite(const ADest: ISequentialStream; const AData: Pointer; const ADataSizeInBytes: LongWord; const ACallingName: TGMString = cDfltRoutineName);
function GMIStreamReadResult(const pcbOut: Pointer; const AllDone: Boolean): HResult;
function GMIStreamWriteResult(const pcbOut: Pointer; const AllDone: Boolean): HResult;
function GMIStreamContentAsString(const ASourceStrm: ISequentialStream; StartPos: Int64 = -1; const ACallingName: TGMString = ''): AnsiString;

procedure GMCopyIStreamTime(const ASourceStrm, ADestStrm: ISequentialStream;
                            const AOnProgressProc: TGMOnProgressProc = nil;
                            const ACallBackTimeInMS: LongWord = cDfltUiResponseMS;
                            const ACallingName: TGMString = cDfltRoutineName);

procedure GMCopyIStreamBufSize(const ASourceStrm, ADestStrm: ISequentialStream;
                               const ACopyBufferSize: LongWord = cDfltCopyBufferSize;
                               const AOnProgressProc: TGMOnProgressProc = nil;
                               const ACallingName: TGMString = cDfltRoutineName);

procedure GMCopyIStream(const ASourceStrm, ADestStrm: ISequentialStream;
                        const ACopyBufferSize: LongInt = cDfltCopyBufferSize;
                        const AOnProgressProc: TGMOnProgressProc = nil;
                        const ACallingName: TGMString = cDfltRoutineName);


{ ---- Objects / Interfaces / Classes ---- }

//function GMIsOneOfClasses(const AObj: TObject; const Classes: array of TClass): Boolean;
function GMIntfClassName(const AIntf: IUnknown): TGMString;
function GMObjFromIntf(const AIntf: IUnknown): TObject;
function GMCompareIUnknown(const AIntf1, AIntf2: IUnknown): TGMCompareResult;
function GMObjAsIntf(const AObj: TObject): IUnknown; inline;
function GMGetIntfName(const AIntf: IUnknown; const ADefaultName: TGMString = ''): TGMString;
function GMGetIntfDisplayName(const AIntf: IUnknown): TGMString; // ; const ADefaultValue: TGMString = cStrUnknown
function GMGetIntfFileName(const AIntf: IUnknown; const ADefaultFileName: TGMString = ''): TGMString;
function GMIntfIIDName(const AIID: TGuid): TGMString; overload;
function GMIntfIIDName(const IIDStr: TGMString): TGMString; overload;
function GMGetObjName(const AObj: TObject; const ADefaultName: TGMString = ''): TGMString;
function GMGetObjDisplayName(const AObj: TObject): TGMString;
function GMObjClassName(const AObj: TObject): TGMString;
function GMClassName(const AClass: TClass): TGMString;
function GMGetPropertyIntf(const AObj: TObject; const PropertyName: TGMString; const AIID: TGUID; out AIntf): HResult;
function GMGetPropIntfFromIntf(const AOwner: IUnknown; const APropertyName: TGMString; const AIID: TGUID; out AIntf): HResult;
//procedure GMCheckGetPropIntfFromIntf(const Owner: IUnknown; const PropertyName: TGMString; const AIID: TGUID; out AIntf; const ACallingName: TGMString = '');
function GMGetInterface(const AObj: TObject; const AIID: TGUID; out AIntf): Boolean; inline;
procedure GMCheckGetInterface(const AObj: TObject; const AIID: TGUID; out AIntf; const ACallingName: TGMString = cDfltRoutineName);
function GMGetWeakInterface(const AObj: TObject; const AIID: TGUID; out AIntf): Boolean; inline;
function GMQueryInterface(const ASource: IUnknown; const AIID: TGUID; out AIntf): Boolean; inline;
procedure GMCheckQueryInterface(const ASource: IUnknown; const AIID: TGUID; out AIntf; const ACallingName: TGMString = cDfltRoutineName);
function GMAllInterfacesSupported(const AIntf: IUnknown; const InterfaceIDs: array of TGUID): Boolean;
procedure GMCheckAllInterfacesSupported(const AIntf: IUnknown; const InterfaceIDs: array of TGUID; const ACallingName: TGMString = cDfltRoutineName);
procedure GMCheckGetIntfByPosition(const ACollection: IGMGetIntfByPosition; const Position: LongInt; const AIID: TGUID; out AIntf; const ElementName, ACallingName: TGMString);

function GMGetIntfHandle(const AIntf: IUnknown; const ADefaultValue: THandle = 0): THandle;
function GMIsHandleAllocated(const AIntf: IUnknown): Boolean;
function GMCheckGetIntfHandle(const AIntf: IUnknown; const ACallingName: TGMString): THandle;
function GMGetAllocatedIntfHandle(const AIntf: IUnknown; var Handle: THandle): Boolean;
function GMGetAllocatedObjHandle(const AObj: TObject; var AHandle: THandle): Boolean;

function GMCreateCopyQI(const ASource: IUnknown; const AIID: TGUID; out AIntf): HResult;
function GMObjCreateCopyQI(const ASource: TObject; const AIID: TGUID; out AIntf): HResult;
function GMFindParentObj(const AObj: TObject; const AParentClass: TClass; out AParent): Boolean;
function GMIsParentObj(AParent, AObj: TObject; const AllowIdentity: Boolean = False; const AStopAtClass: TClass = nil): Boolean;
procedure GMCheckFindParentObj(const AObj: TObject; const ParentClass: TClass; out Parent);

function GMGetObjText(const AObj: TObject; const ADefaultValue: TGMString = ''): TGMString;
procedure GMSetObjText(const AObj: TObject; const AValue: TGMString);
function GMGetIntfText(const AIntf: IUnknown; const ADefaultValue: TGMString = ''): TGMString;
procedure GMSetIntfText(const AIntf: IUnknown; const AValue: TGMString);

function GMGetIntfStrValue(const AIntf: IUnknown; const ADefaultValue: TGMString = ''): TGMSTring;


{ ---- Connection Point Helpers ---- }

function GMDoNotifySink(const ANotificationsEnabled: Boolean; const ANotifySink: IUnknown; const AIID: TGUID; out AIntf): Boolean;
procedure GMInterfaceConnect(const ASink: TObject; const AContainer: IUnknown; const AIID: TGUID; var ACookie: LongInt; const ACallingName: TGMString = cDfltRoutineName); overload;
procedure GMInterfaceConnect(const ASink, AContainer: IUnknown; const AIID: TGUID; var ACookie: LongInt; const AObj: TObject = nil; const ACallingName: TGMString = cDfltRoutineName); overload;
procedure GMQuietInterfaceConnect(const AObj: TObject; const AContainer: IUnknown; const AIID: TGUID; var ACookie: LongInt);
procedure GMInterfaceDisconnect(const AContainer: IUnknown; const AIID: TGUID; var ACookie: LongInt);
procedure GMCpcCallNotifySinks(const ACpc: IConnectionPointContainer; const AConnectionPointIID: TGUID; const ANotifyProc: TCPCNotifyProc; const ANotificationsEnabled: Boolean; const AParams: TCPCNotifyParams);
procedure GMCpcCallNotifySinksObj(const ACpc: IConnectionPointContainer; const AConnectionPointIID: TGUID; const ANotifyProc: TCPCNotifyMethod; const ANotificationsEnabled: Boolean; const AParams: TCPCNotifyParams);
procedure GMRequestCPCDisconnect(const ACpc: IConnectionPointContainer);


{ ---- CPC Notifiers ---- }

procedure GMCallSinkClose(const NotifySink: IUnknown; const Params: TCPCNotifyParams);
procedure GMCallSinkBeforeActiveChange(const NotifySink: IUnknown; const Params: TCPCNotifyParams);
procedure GMCallSinkAfterActiveChange(const NotifySink: IUnknown; const Params: TCPCNotifyParams);
procedure GMCallSinkBeforePositionChange(const NotifySink: IUnknown; const Params: TCPCNotifyParams);
procedure GMCallSinkAfterPositionChange(const NotifySink: IUnknown; const Params: TCPCNotifyParams);
procedure GMCallSinkBeforeOperation(const NotifySink: IUnknown; const Params: TCPCNotifyParams);
procedure GMCallSinkAfterOperation(const NotifySink: IUnknown; const Params: TCPCNotifyParams);
procedure GMCallSinkValidateValue(const NotifySink: IUnknown; const Params: TCPCNotifyParams);


{ ---- COM Helpers ---- }

function GMHrSucceeded(const AErrorCode: HResult): Boolean;
function GMSysErrorMsg(const AErrorCode: LongInt; const AParams: array of PGMChar): TGMString;
function GMCreateGuid: TGuid;
function GMIsGUID(const AGuidStr: TGMString): Boolean;
procedure GMCopyGuid(const ASource, ADest: IUnknown; const ACallingName: TGMString = cDfltRoutineName);
function GMGuidToString(const Guid: TGUID): TGMString;
function GMStringToGuid(const AGuidStr: TGMString; const ACaller: TObject = nil; const ACallingName: TGMString = cDfltRoutineName): TGUID;
function GMMakeGuidStr(const GuidStr: TGMString): TGMString;
function GMMakeGuid(GuidStr: TGMString; const AObj: TObject = nil; const ACallingName: TGMString = cDfltRoutineName): TGUID;
function GMCompareGuids(const GuidA, GuidB: TGUID): TGMCompareResult;
function GMEqualGuids(const GuidA, GuidB: TGUID): Boolean;
function GMCreateComObject(const ClassID: TGUID; const CreateContext: DWORD = cDfltCoCeateContext): IUnknown;
function GMCoClassIsRegistered(const ClassId: TGUID): Boolean;


{ ---- Storage Routines ---- }

function GMIsStorageGuid(const Storage: IStorage; const Guid: TGUID; const Caller: TObject = nil): Boolean;
function GMIsStorageFileGuid(const FileName: TGMString; const Guid: TGUID; const Caller: TObject = nil): Boolean;
procedure GMCheckIsStorageGuid(const Storage: IStorage; const Guid: TGUID; const Caller: TObject = nil);
procedure GMCheckIsStorageFileGuid(const FileName: TGMString; const Guid: TGUID; const Caller: TObject = nil);


{ ---- Error Message Format Helpers ---- }

function GMSeverityName(const Aseverity: TGMSeverityLevel): TGMString;
function BuildCallingName(const ACallingName, AThisName: TGMString): TGMString; inline;
function MsgInconsistentClassSize(const ClassName: TGMString): TGMString;
function MsgUnsupportedFieldDataType(const OrdinalValue: LongInt): TGMString;
function MsgUnknownValue(const ValueName: TGMString; const Value: LongInt): TGMString;
function MsgUnsupportedValue(const ValueName: TGMString; const Value: LongInt): TGMString;
function MsgUnknownFieldDataType(const OrdinalValue: LongInt): TGMString;
function MsgUnknownPropVal(const PropertyName: TGMString; const Value: LongInt): TGMString;
function MsgIntfNotSupported(const Name: TGMString; const AIID: TGuid): TGMString;
function MsgNoItemIntfPrefix(const Name: TGMString; const AIID: TGuid): TGMString;
function MsgNoOwnerClass(const ClassName: TGMString): TGMString;
function MsgOutOfRange(const ValueName: TGMString; const Value: LongInt; const MinVal: LongInt; const MaxVal: LongInt): TGMString;
function MsgUnsupportedOperation(const OrdinalValue: LongInt): TGMString;
function MsgInvalidStateTransition(const OldState, NewState: LongInt): TGMString;
function MsgUnknownQestionOrdinal(const Question: LongInt): TGMString;
function MsgMemoryTooSmall(const BufferName: TGMString; const BufferSize, RequiredSize: LongInt): TGMString;
function MsgPointerIsNil(const PointerName: TGMString): TGMString;
//function MsgModuleNotInstalled(const ModuleName, ErrorMsg, DownloadURL: TGMString): TGMString;


procedure GMCheckPointerAssigned(const Ptr: Pointer; const PointerName: TGMString; const AObj: TObject = nil; const ACallingName: TGMString = cDfltRoutineName);
procedure GMCheckIntRange(const ValueName: TGMString; const Value, MinValue, MaxValue: PtrInt; const AObj: TObject = nil; const ACallingName: TGMString = cDfltRoutineName);
procedure GMCheckMemorySize(const BufferName: TGMString; const BufferSize, RequiredSize: PtrInt; const AObj: TObject = nil; const ACallingName: TGMString = cDfltRoutineName);
procedure GMTraceQueryInterface(const AObj: TObject; const AIID: TGuid; const AResult: HResult);
{$IFDEF DEBUG}
procedure GMCheckRefCountIsZero(const ARefCount: LongInt; const AObj: TObject);
{$ENDIF}

function GMCoTaskStrDupW(const AValue: WideString): PWideChar;


{ ---- Lists as Strings ---- }

function GMStringJoin(const ALeftStr, ASeparator, ARightStr: TGMString): TGMString;
function GMSeparatedStrings(const AStrings: array of TGMString; const ASeparator: TGMString = ', '): TGMString;
function GMSeparatedNames(const ACollection: IUnknown; const ASeparator: TGMString = ', '): TGMString;
function GMSeparatedValues(const ACollection: IUnknown; const AVarToStrFunc: TGMUnionValToStrFunc; const ASeparator: TGMString = ', '): TGMString;
function GMNamesAndValuesAsString(const ACollection: IUnknown; const AVarToStrFunc: TGMUnionValToStrFunc; const ASeparator: TGMString = ', '; const AOperator: TGMString = ' = '; const AOmitNullValues: Boolean = False): TGMString;

function GMColumnDesc(const ATitle: TGMString; const AWidth: LongInt; const AAlignment: TGMDfltHorizontalAlignment = haLeft): TGMColumnDescRec;


{ ---- Additional error message modules ---- }

procedure GMRegisterErrorMsgModule(const AModuleName: TGMString);
procedure GMExtendExceptionMsg(const AException: TObject; const AAddition: TGMString; const ASeparator: TGMString = ': '; const APrefix: Boolean = True);

function GMFileAttributeName(const AFileAttribute: TFileAttribute): TGMString;
function GMMatchKindName(const AMatchKind: TMatchKind): TGMString;


resourcestring

  RStrInterface = '<Interface>';
  RStrNil = '<nil>';
  RStrCPC = 'Connection point container';

  RStrUnknownSysErrorMsgFmt = 'Unknown system error code: %d';

  RStrField = 'Field';
//RStrStreamRead = 'Stream read';
//RStrStreamWrite = 'Stream write';
  RStrProperty = 'Property';
  RStrMethod = 'Method';
  //RStrItemByNameNoIntf = '%s %s doesn''t exist or';
  RStrRefCountFmt = '%s [%p] "%s" Reference Count at Destruction = %d';
  RStrAskShowAgain = 'Shall this Message be shown again?';
  RStrNoCPForIntf = '%s has no Connection Point for the Interface: %s';
  RStrSysErrorCodeFmt = 'System Error, Code Hex: 0x%X,  Code Dez: %d';
  RStrInvalidParamCountFmt = 'Unsupported number of Parameter: %d';
  RStrMissingPropVal = 'The "%s" property must be set before the component can be activated';
  RStrUnsupportedIdxType = 'Unsupported Datatype "%s" used for indexing';
  RStrNoIntfFmt = '%s [%p] "%s" Interface not Supported: %s %s';
  RStrNoItemIntf = 'Unable to get interface %s for item "%s"';
  RStrNeedRTTI = 'Class "%s" must be declared with the $M+ compiler directive to access properties.';
  RStrFieldNotFound = 'Field "%s" not found in "%s"';
  RStrNoValueForItemFmt = 'Cannot get a value for item "%s" at position %d';
  RStrCreateDirFailed = 'Failed to create persistent directory "%s"';
  srCannotCallNilIntf = 'RGMTypedIntf<%s>.Call: Cannot call any method on NIL interface pointer';

  srCollectionelement = 'Collection element';
  RStrValue = 'Value';

  RStrExactMatch = 'Exact Match';
  RStrClosestMatch = 'Closest Match';

  RStrArchive = 'Archiv';
  RStrCompressed = 'Compressed';
  RStrDirectory = 'Directory';
  RStrEncrypted = 'Encrypted';
  RStrHidden = 'Hidden';
  RStrNormal = 'Normal';
  RStrOffline = 'Offline';
  RStrReadOnly = 'Readonly';
  RStrReparsePoint = 'Reparse point';
  RStrSparse = 'Sparse';
  RStrSystem = 'System';
  RStrTemporary = 'Temporary';

  RStrError = 'Error';
  RStrInformation = 'Information';
  RStrWarning = 'Warning';
  RStrConfirm = 'Confirm';

  RStrNullAllowed = 'NULL allowed';
  RStrNotNull = 'Not NULL';

  srAscending = 'Ascending';
  srDescending = 'Descending';

  RStrUnique = 'Unique';
  RStrDuplicatesAlloed = 'Duplicates allowed';


  RStrInvalidCompareResult = 'Invalid Compare Result';


const

  cEnumElementCount: array [Boolean] of LongInt = (0, 1);
  cHRSeverity: array [Boolean] of TGMSeverityLevel = (svInformation, svError);

  cQIResult: array [Boolean] of HResult = (E_NOINTERFACE, S_OK);

  GM_E_STREAMWRITE = STG_E_CANTSAVE;
  GM_E_STREAMREAD = E_FAIL; // E_UNEXPECTED;

  {$IFDEF FPC}
  {$IFNDEF JEDIAPI}
  {$EXTERNALSYM IDI_APPLICATION}
  IDI_APPLICATION = PGMChar(ULONG_PTR(WORD(32512)));
  {$EXTERNALSYM IDI_HAND}
  IDI_HAND = LPTSTR(ULONG_PTR(WORD(32513)));
  {$EXTERNALSYM IDI_QUESTION}
  IDI_QUESTION = LPTSTR(ULONG_PTR(WORD(32514)));
  {$EXTERNALSYM IDI_EXCLAMATION}
  IDI_EXCLAMATION = LPTSTR(ULONG_PTR(WORD(32515)));
  {$EXTERNALSYM IDI_ASTERISK}
  IDI_ASTERISK = LPTSTR(ULONG_PTR(WORD(32516)));
  {$EXTERNALSYM IDI_WINLOGO}
  IDI_WINLOGO = LPTSTR(ULONG_PTR(WORD(32517)));
  {$EXTERNALSYM IDI_WARNING}
  IDI_WARNING = IDI_EXCLAMATION;
  {$EXTERNALSYM IDI_ERROR}
  IDI_ERROR = IDI_HAND;
  {$EXTERNALSYM IDI_INFORMATION}
  IDI_INFORMATION = IDI_ASTERISK;
  {$ENDIF}
  {$ENDIF}


var

  vShowRefCountWarnings: Boolean = {$IFDEF DEBUG}True;{$ELSE}False;{$ENDIF}
  {$IFDEF DEBUG}
  vfGMCheckRefCountOnDestroy: TGMCheckRefCountProc = GMCheckRefCountIsZero;
  {$ELSE}
  vfGMCheckRefCountOnDestroy: TGMCheckRefCountProc = nil;
  {$ENDIF}

//vGMFileAttributeNames: array [TFileAttribute] of TGMString =
// (RStrArchive, RStrCompressed, RStrDirectory, RStrEncrypted, RStrHidden, RStrNormal, RStrOffline,
//  RStrReadOnly, RStrReparsePoint, RStrSparse, RStrSystem, RStrTemporary);

  vGMSevrityIcons: array [TGMSeverityLevel] of Pointer = (nil, IDI_QUESTION, IDI_INFORMATION, IDI_WARNING, IDI_ERROR);
//vMatchKindNames: array [TMatchKind] of TGMString = (RStrExactMatch, RStrClosestMatch);

  vGMComInitFlags: LongInt = cDfltCoInitFlags;


implementation

uses GMCommon, GMStrBuilder {$IFDEF JEDIAPI}, jwaWinReg, jwaWinBase{$ENDIF}
     {$IFDEF DELPHI6}, Variants{$ENDIF}
     ;

var

  vGMErrorMsgModules: TGMStringArray;


resourcestring

  RStrUnsupportedValue = 'Unsupported value (%d) for "%s"';
  RStrUnknownValue = 'Unknown Value (%d) for "%s"';
  RStrMsgOutOfRangeFmt = '%s out of range: %d. The Value must be in Interval [%d, %d]';
  RStrMsgUnsupportedOperation = 'Operation not supported, Ord(Operation): %d';
  RStrIntfNotSupported = '%s doesn''t support the Interface: %s';
  RStrUnknownPropVal = 'Unknown Value of Property "%s", value: %d';
  RStrNoOwnerClass = 'No owner Component with class "%s" found';
  RStrInvalidstateTransition = 'Invalid State transition, Ord(OldState): %d, Ord(NewState): %d';
  RStrCheckActive = 'The %s can only be used when the Object is Active';
  RStrCheckInactive = 'The %s cannot be changed while the Object is active';
  RStrUnknownQestionOrdinal = 'Unknown Ordinal Value for Boolean Question: %d';
  RStrOperationExecFailed = 'Failed to execute operation %s';
  RStrUnsupoortedBoolQuestion = 'Unsupported Question ID: %d';
  RStrUnsupportedValueId = 'Unsupported Value ID: %d';
  RStrInconsistentclassSize = 'Inconsistent class Size: %s';
  RStrInvalidStorageGuid = 'Invalid Storage GUID, requested GUID %s storage GUID %s';
  RStrNoParentFmt = 'The object "%s" has no parent of class "%s"';
  RStrMemoryTooSmall = '%s size too small (%d Bytes), must be at least %d Bytes';
  RStrPtrIsNil = '%s is <nil>';
  RStrTheObject = 'The Object';
  RStrTheList = 'The List';


  //RStrModuleNotInstalled = 'The ''%s'' has not been installed on this System.';
  //RStrModuleDownload = 'It can be downloaded from: %s.';
//RStrModuleErrorMsg = 'The System Error Message was:'#13 +
//                     '---------------------------------------------------'#13 +
//                     '%s';


const

  cNameRecVTable: array[0..3] of Pointer =
    (@RGMNameRec.QueryIntferface,
     @RGMNameRec.Addref,
     @RGMNameRec.Release,
     @RGMNameRec.GetName);


{ -------------------------------------- }
{ ---- Error message Format Helpers ---- }
{ -------------------------------------- }

function GMSeverityName(const ASeverity: TGMSeverityLevel): TGMString;
begin
  case ASeverity of
   svInformation: Result := RStrInformation;
   svConfirmation: Result := RStrConfirm;
   svWarning: Result := RStrWarning;
   svError: Result := RStrError;
   else Result := '';
  end;
end;

function GMFileAttributeName(const AFileAttribute: TFileAttribute): TGMString;
begin
  case AFileAttribute of
   faArchive: Result := RStrArchive;
   faCompressed: Result := RStrCompressed;
   faDirectory: Result := RStrDirectory;
   faEncrypted: Result := RStrEncrypted;
   faHidden: Result := RStrHidden;
   faNormal: Result := RStrNormal;
   faOffline: Result := RStrOffline;
   faReadOnly: Result := RStrReadOnly;
   faReparsePoint: Result := RStrReparsePoint;
   faSparse: Result := RStrSparse;
   faSystem: Result := RStrSystem;
   faTemporary: Result := RStrTemporary;
   else Result := '';
  end;
end;

function GMMatchKindName(const AMatchKind: TMatchKind): TGMString;
begin
  case AMatchKind of
   mkExactMatch: Result := RStrExactMatch;
   mkNearestMatch: Result := RStrClosestMatch;
   else Result := '';
  end;
end;

function BuildCallingName(const ACallingName, AThisName: TGMString): TGMString;
begin
  if ACallingName <> cDfltRoutineName then Result := ACallingName else Result := AThisName;
end;

function MsgInconsistentClassSize(const ClassName: TGMString): TGMString;
begin
  Result := GMFormat(RStrInconsistentclassSize, [ClassName]);
end;

function MsgUnsupportedFieldDataType(const OrdinalValue: LongInt): TGMString;
begin
  Result := GMFormat(RStrUnsupportedValue, [OrdinalValue, cStrFieldDataTypeName]);
end;

function MsgUnknownValue(const ValueName: TGMString; const Value: LongInt): TGMString;
begin
  Result := GMFormat(RStrUnknownValue, [Value, ValueName]);
end;

function MsgUnsupportedValue(const ValueName: TGMString; const Value: LongInt): TGMString;
begin
  Result := GMFormat(RStrUnsupportedValue, [Value, ValueName]);
end;

function MsgUnknownFieldDataType(const OrdinalValue: LongInt): TGMString;
begin
  Result := MsgUnknownValue(cStrFieldDataTypeName, OrdinalValue);
end;

function MsgUnknownPropVal(const PropertyName: TGMString; const Value: LongInt): TGMString;
begin
  Result := GMFormat(RStrUnknownPropVal, [PropertyName, Value]);
end;

function MsgIntfNotSupported(const Name: TGMString; const AIID: TGuid): TGMString;
begin
  Result := GMFormat(RStrIntfNotSupported, [Name, GMGuidToString(AIID)]);
end;

function MsgNoItemIntfPrefix(const Name: TGMString; const AIID: TGuid): TGMString;
begin
  Result := GMFormat(RStrNoItemIntf + ': ', [GMGuidToString(AIID), Name]);
end;

{function MsgNoIntfOrNotExist(const Kind, Name: TGMString; const AIID: TGuid): TGMString;
begin
  Result := MsgIntfNotSupported(GMFormat(RStrItemByNameNoIntf, [Kind, Name]), AIID);
end;}

function MsgNoOwnerClass(const ClassName: TGMString): TGMString;
begin
  Result := GMFormat(RStrNoOwnerClass, [ClassName]);
end;

function MsgOutOfRange(const ValueName: TGMString; const Value: LongInt; const MinVal: LongInt; const MaxVal: LongInt): TGMString;
begin
  Result := GMFormat(RStrMsgOutOfRangeFmt, [ValueName, Value, MinVal, MaxVal]);
end;

function MsgUnsupportedOperation(const OrdinalValue: LongInt): TGMString;
begin
  Result := GMFormat(RStrMsgUnsupportedOperation, [OrdinalValue]);
end;

function MsgInvalidStateTransition(const OldState, NewState: LongInt): TGMString;
begin
  Result := GMFormat(RStrInvalidstateTransition, [OldState, NewState]);
end;

function MsgUnknownQestionOrdinal(const Question: LongInt): TGMString;
begin
  Result := GMFormat(RStrUnknownQestionOrdinal, [Question]);
end;

function MsgMemoryTooSmall(const BufferName: TGMString; const BufferSize, RequiredSize: LongInt): TGMString;
begin
  Result := GMFormat(RStrMemoryTooSmall, [BufferName, BufferSize, RequiredSize]);
end;

function MsgPointerIsNil(const PointerName: TGMString): TGMString;
begin
  Result := GMFormat(RStrPtrIsNil, [PointerName]);
end;

//function MsgModuleNotInstalled(const ModuleName, ErrorMsg, DownloadURL: TGMString): TGMString;
////const C2Line = #13#13;
//begin
//Result := GMFormat(RStrModuleNotInstalled + c2NewLine, [ModuleName]);
//if DownloadURL <> '' then Result := Result + GMFormat(RStrModuleDownload + c2NewLine, [DownloadURL]);
//if ErrorMsg <> '' then Result := Result + GMFormat(RStrModuleErrorMsg, [ErrorMsg]);
//end;


{ ------------------------------- }
{ ---- String array routines ---- }
{ ------------------------------- }

function GMStringArray(const AStrings: array of TGMString): TGMStringArray;
var i: LongInt;
begin
  SetLength(Result, Length(AStrings));
  for i:=Low(AStrings) to High(AStrings) do Result[i] := AStrings[i];
end;

procedure GMAddStrToArray(const AValue: TGMString; var AStringArray: TGMStringArray; const AAddEmptyStrings: Boolean);
begin
  if AAddEmptyStrings or (Length(AValue) > 0) then
   begin
    SetLength(AStringArray, Length(AStringArray)+1);
    AStringArray[High(AStringArray)] := AValue;
   end;
end;

function GMStrArrayAsText(const AStrings: TGMStringArray; const ASeparator: TGMString = cNewLine): TGMString;
var i: LongInt; resStr: RGMStringBuilder;
begin
  //Result := '';
  for i:=Low(AStrings) to High(AStrings) do resStr.Join(ASeparator, AStrings[i]); // Result := GMStringJoin(Result, ASeparator, AStrings[i]);
  Result := resStr;
end;

function GMIndexOfStrInArray(const AValue: TGMString; const AStringArray: TGMStringArray): PtrInt;
begin
  for Result:=Low(AStringArray) to High(AStringArray) do if GMSameText(AStringArray[Result], AValue) then Exit;
  Result := cInvalidItemIdx;
end;

function GMFindStrInArray(const AValue: TGMString; const AStringArray: TGMStringArray; var AIdx: PtrInt): Boolean;
begin
  AIdx := GMIndexOfStrInArray(AValue, AStringArray);
  Result := AIdx <> cInvalidItemIdx;
end;

procedure GMDeleteStrInArray(var AStringArray: TGMStringArray; const AIdx: PtrInt);
var i: LongInt;
begin
  if not GMIsInRange(AIdx, Low(AStringArray), High(AStringArray)) then Exit;
  for i:=AIdx to High(AStringArray)-1 do AStringArray[i] := AStringArray[i+1];
  SetLength(AStringArray, Length(AStringArray)-1);
end;


{ ------------------------- }
{ ---- Buffer Checking ---- }
{ ------------------------- }

procedure GMCheckPointerAssigned(const Ptr: Pointer; const PointerName: TGMString; const AObj: TObject; const ACallingName: TGMString);
begin
  if Ptr = nil then raise EGMException.ObjError(MsgPointerIsNil(PointerName), AObj, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
end;

procedure GMCheckIntRange(const ValueName: TGMString; const Value, MinValue, MaxValue: PtrInt; const AObj: TObject; const ACallingName: TGMString);
begin
  if not GMIsInRange(Value, MinValue, MaxValue) then
     raise EGMException.ObjError(MsgOutOfRange(ValueName, Value, MinValue, MaxValue), AObj, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
end;

procedure GMCheckMemorySize(const BufferName: TGMString; const BufferSize, RequiredSize: PtrInt; const AObj: TObject; const ACallingName: TGMString);
begin
  if BufferSize < RequiredSize then
   raise EGMException.ObjError(MsgMemoryTooSmall(BufferName, BufferSize, RequiredSize), AObj, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
end;

function GMCoTaskStrDupW(const AValue: WideString): PWideChar;
begin
  Result := CoTaskMemAlloc((Length(AValue) + 1) * SizeOf(WideChar));
  if Result = nil then raise EGMHrException.ObjError(E_OUTOFMEMORY, [], nil, {$I %CurrentRoutine%});
  Move(PWideChar(AValue)^, Result^, (Length(AValue) + 1) * SizeOf(WideChar));
end;

{$IFDEF DEBUG}
procedure GMCheckRefCountIsZero(const ARefCount: LongInt; const AObj: TObject);
var Msg: TGMString;
begin
  if ARefCount <> 0 then
   begin
    Msg := GMFormat(RStrRefCountFmt, [GMObjClassName(AObj), Pointer(AObj), GMGetObjName(AObj), ARefCount]);
    GMTrace(Msg, tpWarning);  
    if vShowRefCountWarnings then vShowRefCountWarnings := vfGMMessageBox(Msg + c2NewLine + RStrAskShowAgain, svWarning, MB_YESNO) = IdYes;
   end;
end;
{$ENDIF}

procedure GMTraceQueryInterface(const AObj: TObject; const AIID: TGuid; const AResult: HResult);
begin
  if AResult = S_OK then Exit;
  GMTrace(GMFormat(RStrNoIntfFmt, [GMObjClassName(AObj), Pointer(AObj), GMGetObjName(AObj), GMIntfIIDName(AIID), GMGuidToString(AIID)]), tpInterface);
end;


{ ------------------------------------ }
{ ---- File Attribute conversions ---- }
{ ------------------------------------ }

function GMDWordToFileAttributes(const AValue: DWORD): TFileAttributes;
begin
  Result := [];
  if AValue and FILE_ATTRIBUTE_ARCHIVE <> 0 then Include(Result, faArchive);
  if AValue and FILE_ATTRIBUTE_COMPRESSED <> 0 then Include(Result, faCompressed);
  if AValue and FILE_ATTRIBUTE_DIRECTORY <> 0 then Include(Result, faDirectory);
  if AValue and FILE_ATTRIBUTE_ENCRYPTED <> 0 then Include(Result, faEncrypted);
  if AValue and FILE_ATTRIBUTE_HIDDEN <> 0 then Include(Result, faHidden);
  if AValue and FILE_ATTRIBUTE_NORMAL <> 0 then Include(Result, faNormal);
  if AValue and FILE_ATTRIBUTE_OFFLINE <> 0 then Include(Result, faOffline);
  if AValue and FILE_ATTRIBUTE_READONLY <> 0 then Include(Result, faReadOnly);
  if AValue and FILE_ATTRIBUTE_REPARSE_POINT <> 0 then Include(Result, faReparsePoint);
  if AValue and FILE_ATTRIBUTE_SPARSE_FILE <> 0 then Include(Result, faSparse);
  if AValue and FILE_ATTRIBUTE_SYSTEM <> 0 then Include(Result, faSystem);
  if AValue and FILE_ATTRIBUTE_TEMPORARY <> 0 then Include(Result, faTemporary);
end;

function GMFileAttributesToDWORD(const AValue: TFileAttributes): DWORD;
begin
  Result := 0;
  if faArchive in AValue then Result := Result or FILE_ATTRIBUTE_ARCHIVE;
  if faCompressed in AValue then Result := Result or FILE_ATTRIBUTE_COMPRESSED;
  if faDirectory in AValue then Result := Result or FILE_ATTRIBUTE_DIRECTORY;
  if faEncrypted in AValue then Result := Result or FILE_ATTRIBUTE_ENCRYPTED;
  if faHidden in AValue then Result := Result or FILE_ATTRIBUTE_HIDDEN;
  if faNormal in AValue then Result := Result or FILE_ATTRIBUTE_NORMAL;
  if faOffline in AValue then Result := Result or FILE_ATTRIBUTE_OFFLINE;
  if faReadOnly in AValue then Result := Result or FILE_ATTRIBUTE_READONLY;
  if faReparsePoint in AValue then Result := Result or FILE_ATTRIBUTE_REPARSE_POINT;
  if faSparse in AValue then Result := Result or FILE_ATTRIBUTE_SPARSE_FILE;
  if faSystem in AValue then Result := Result or FILE_ATTRIBUTE_SYSTEM;
  if faTemporary in AValue then Result := Result or FILE_ATTRIBUTE_TEMPORARY;
end;


{ ------------------------- }
{ ---- String Routines ---- }
{ ------------------------- }

function GMStringJoin(const ALeftStr, ASeparator, ARightStr: TGMString): TGMString;
begin
  if Length(ARightStr) <= 0 then Result := ALeftStr
  else
  if Length(ALeftStr) <= 0 then Result := ARightStr
  else
  if Length(ASeparator) <= 0 then Result := ALeftStr + ARightStr
  else
  Result := ALeftStr + ASeparator + ARightStr;
end;

function GMSeparatedStrings(const AStrings: array of TGMString; const ASeparator: TGMString): TGMString;
var i: LongInt; resStr: RGMStringBuilder;
begin
  //Result := '';
  for i:=Low(AStrings) to high(AStrings) do resStr.Join(ASeparator, AStrings[i]);
   //Result := GMStringJoin(Result, ASeparator, AStrings[i]);
  Result := resStr;
end;

function GMSeparatedNames(const ACollection: IUnknown; const ASeparator: TGMString): TGMString;
var i: LongInt; PICount: IGMGetCount; PIIntfByPosition: IGMGetIntfByPosition; PIName: IGMGetName; resStr: RGMStringBuilder;
begin                                                                             
  //Result := '';
  GMCheckQueryInterface(ACollection, IGMGetCount, PICount, {$I %CurrentRoutine%});
  GMCheckQueryInterface(ACollection, IGMGetIntfByPosition, PIIntfByPosition, {$I %CurrentRoutine%});

  for i:=0 to PICount.Count-1 do
   begin
    GMCheckGetIntfByPosition(PIIntfByPosition, i, IGMGetName, PIName, srCollectionelement, {$I %CurrentRoutine%});
    //Result := GMStringJoin(Result, ASeparator, PIName.Name);
    resStr.Join(ASeparator, PIName.Name);
   end;
  Result := resStr;
end;

function GMSeparatedValues(const ACollection: IUnknown; const AVarToStrFunc: TGMUnionValToStrFunc; const ASeparator: TGMString): TGMString;
var i: LongInt; PICount: IGMGetCount; PIIntfByPosition: IGMGetIntfByPosition; PIValue: IGMGetUnionValue; resStr: RGMStringBuilder;
begin
  //Result := '';
  Assert(Assigned(AVarToStrFunc), 'Assigned(AVarToStrFunc)');
  GMCheckQueryInterface(ACollection, IGMGetCount, PICount, {$I %CurrentRoutine%});
  GMCheckQueryInterface(ACollection, IGMGetIntfByPosition, PIIntfByPosition, {$I %CurrentRoutine%});

  for i:=0 to PICount.Count-1 do
   begin
    GMCheckGetIntfByPosition(PIIntfByPosition, i, IGMGetUnionValue, PIValue, srCollectionelement, {$I %CurrentRoutine%});
    //Result := GMStringJoin(Result, ASeparator, AVarToStrFunc(PIValue.Value));
    resStr.Join(ASeparator, AVarToStrFunc(PIValue.Value));
   end;
  Result := resStr;
end;

function GMNamesAndValuesAsString(const ACollection: IUnknown; const AVarToStrFunc: TGMUnionValToStrFunc; const ASeparator, AOperator: TGMString; const AOmitNullValues: Boolean): TGMString;
var i: LongInt; count: IGMGetCount; intfByPos: IGMGetIntfByPosition; getName: IGMGetName; getVal: IGMGetUnionValue; getText: IGMGetText;
    valStr: TGMString; resStr: RGMStringBuilder;
begin
  //Result := '';
  Assert(Assigned(AVarToStrFunc), 'Assigned(AVarToStrFunc)');
  GMCheckQueryInterface(ACollection, IGMGetCount, count, {$I %CurrentRoutine%});
  GMCheckQueryInterface(ACollection, IGMGetIntfByPosition, intfByPos, {$I %CurrentRoutine%});

  for i:=0 to count.Count-1 do
   begin
    GMCheckGetIntfByPosition(intfByPos, i, IGMGetName, getName, srCollectionelement, {$I %CurrentRoutine%});
//  GMCheckGetIntfByPosition(intfByPos, i, IGMGetUnionValue, getVal, srCollectionelement, {$I %CurrentRoutine%});

//  valStr := '';
    if intfByPos.GetIntfByPosition(i, IGMGetUnionValue, getVal) = S_OK then valStr := AVarToStrFunc(getVal.Value)
    else
    if intfByPos.GetIntfByPosition(i, IGMGetText, getText) = S_OK then valStr := getText.Text
    else raise EGMException.IntfError(GMFormat(RStrNoValueForItemFmt, [getName.Name, i]), getName, {$I %CurrentRoutine%});

    if not AOmitNullValues or not getVal.Value.IsNullOrUnassigned then
       //Result := GMStringJoin(Result, ASeparator, getName.Name + AOperator + valStr); // <- dont StrJoin the latter part! GMStringJoin(getName.Name, AOperator, valStr)
       resStr.Join(ASeparator, getName.Name + AOperator + valStr); // <- dont StrJoin the latter part! GMStringJoin(getName.Name, AOperator, valStr)
   end;
  Result := resStr;
end;

function GMColumnDesc(const ATitle: TGMString; const AWidth: LongInt; const AAlignment: TGMDfltHorizontalAlignment = haLeft): TGMColumnDescRec;
begin
  Result.Title := ATitle;
  Result.Width := AWidth;
  Result.Alignment := AAlignment;
end;


{ ------------------------------ }
{ ---- Objects / Interfaces ---- }
{ ------------------------------ }

{function GMIsOneOfClasses(const AObj: TObject; const Classes: array of TClass): Boolean;
var i: LongInt;
begin
  // if Classes is empty Result is True!
  Result := Length(Classes) = 0;
  if not Result then for i:=Low(Classes) to High(Classes) do if Obj is Classes[i] then begin Result := True; Break; end;
end;}

function GMIntfClassName(const AIntf: IUnknown): TGMString;
var obiInfo: IGMObjInfo;
begin
 if not GMQueryInterface(AIntf, IGMObjInfo, obiInfo) then
  Result := RStrInterface
 else
  Result := obiInfo.ClassName;
end;

function GMObjClassName(const AObj: TObject): TGMString;
begin
  if AObj = nil then Result := RStrNil else Result := AObj.ClassName;
end;

function GMClassName(const AClass: TClass): TGMString;
begin
  if AClass = nil then Result := RStrNil else Result := AClass.ClassName;
end;

function GMGetIntfName(const AIntf: IUnknown; const ADefaultName: TGMString): TGMString;
var getName: IGMGetName;
begin
  if GMQueryInterface(AIntf, IGMGetName, getName) then Result := getName.Name else Result := ADefaultName;
end;

function GMGetIntfDisplayName(const AIntf: IUnknown): TGMString; // ; const ADefaultValue: TGMString = cStrUnknown
var getName: IGMGetName;
begin
  if AIntf = nil then Result := RStrNil else
    if GMQueryInterface(AIntf, IGMGetName, getName) then Result := getName.Name else Result := RStrUnknown;
end;

function GMGetIntfFileName(const AIntf: IUnknown; const ADefaultFileName: TGMString): TGMString;
var fileName: IGMGetFileName;
begin
  if AIntf = nil then Result := ADefaultFileName else
   if GMQueryInterface(AIntf, IGMGetFileName, fileName) then Result := fileName.FileName else Result := ADefaultFileName;
end;

function GMIntfIIDName(const IIDStr: TGMString): TGMString;
begin
  Result := GMIntfIIDName(GMStringToGuid(IIDStr));
end;

function GMIntfIIDName(const AIID: TGuid): TGMString;
var RegKey: IGMRegKey;
begin
  RegKey := TGMRegKey.Create;
  if RegKey.Obj.OpenKey(HKEY_CLASSES_ROOT, '\Interface\' + GMGuidToString(AIID)) then
   Result := RegKey.Obj.ReadString('') else Result := RStrUnknown;
end;

function GMGetObjName(const AObj: TObject; const ADefaultName: TGMString = ''): TGMString;
var getName: IGMGetName;
begin
  if GMGetInterface(AObj, IGMGetName, getName) then Result := getName.Name else Result := ADefaultName;
end;

function GMGetObjDisplayName(const AObj: TObject): TGMString;
begin
  if AObj = nil then Result := RStrNil
  else
  {$IFDEF DELPHIVCL}
  if AObj is TComponent then Result := (>Obj as TComponent).Name
  else
  {$ENDIF}
  Result := GMGetObjName(AObj, RStrUnknown);
end;

procedure GMEnableObj(const AObj: TObject; const AEnabled: Boolean);
begin
  GMEnableIntf(GMObjAsIntf(AObj), AEnabled);
end;

function GMGetIntfEnabled(const AIntf: IUnknown; const ADefaultValue: Boolean = False): Boolean;
var getEnabled: IGMGetEnabled;
begin
  if GMQueryInterface(AIntf, IGMGetEnabled, getEnabled) then Result := getEnabled.Enabled else Result := ADefaultValue;
end;

function GMGetObjEnabled(const AObj: TObject; const ADefaultValue: Boolean = False): Boolean;
var getEnabled: IGMGetEnabled;
begin
  if GMGetInterface(AObj, IGMGetEnabled, getEnabled) then Result := getEnabled.Enabled else Result := ADefaultValue;
end;

procedure GMEnableIntf(const AIntf: IUnknown; const AEnabled: Boolean);
var enable: IGMGetSetEnabled;
begin
  if GMQueryInterface(AIntf, IGMGetSetEnabled, enable) then enable.SetEnabled(AEnabled);
end;

function GMTreeableNodeLevel(ANode: IGMTreeable): Integer;
begin
  Result := 0;
  while ANode <> nil do begin ANode := ANode.Parent; Inc(Result); end;
end;

function GMInitNodeVisitData(const ADataClass: TClass; const ANodeTitle: TGMString; const ANode: IGMTreeable; const ASearchIndex: LongInt): RGMNodeVisitData;
begin
  Result.DataClass := ADataClass;
  Result.NodeTitle := ANodeTitle;
  Result.Node := ANode;
  Result.Index := 0;
  Result.SearchIdx := ASearchIndex;
end;

{$push}
{$macro on}
{$define DoNodeVisitRootFirstInnerFunc:=function DoNodeVisit(ANode: IGMTreeable): Boolean;
  begin
    Result := True;
    while (ANode <> nil) and Result do
     begin
      Result := AVisitFunc(ANode, AParameter);
      if ARecurse and Result then Result := DoNodeVisit(ANode.FirstChild);
      ANode := ANode.NextSibling;
     end;
  end;}

function GMVisitNodesRootFirst(const AStartNode: IGMTreeable; const AVisitFunc: TGMNodeVisitFunc; const ARecurse: Boolean; const AParameter: Pointer): Boolean;
  {.$I DoNodeVisit.inc}
  DoNodeVisitRootFirstInnerFunc
begin
  if Assigned(AVisitFunc) then Result := DoNodeVisit(AStartNode) else Result := True;
end;

function GMVisitNodesRootFirst(const AStartNode: IGMTreeable; const AVisitFunc: TGMNodeVisitMethod; const ARecurse: Boolean; const AParameter: Pointer): Boolean;
  {.$I DoNodeVisit.inc}
  DoNodeVisitRootFirstInnerFunc
begin
  if Assigned(AVisitFunc) then Result := DoNodeVisit(AStartNode) else Result := True;
end;

{$define DoNodeVisitDepthFirstInnerFunc:=  function DoNodeVisitDepthFirst(ANode: IGMTreeable): Boolean;
  begin
    Result := True;
    while (ANode <> nil) and Result do
     begin
      if ARecurse and Result then Result := DoNodeVisitDepthFirst(ANode.FirstChild);
      if Result then Result := AVisitFunc(ANode, AParameter);
      ANode := ANode.NextSibling;
     end;
  end;}

function GMVisitNodesDepthfirst(const AStartNode: IGMTreeable; const AVisitFunc: TGMNodeVisitFunc; const ARecurse: Boolean; const AParameter: Pointer): Boolean;
  {.$I DoNodeVisitDepthfirst.inc}
  DoNodeVisitDepthFirstInnerFunc
begin
  if Assigned(AVisitFunc) then Result := DoNodeVisitDepthFirst(AStartNode) else Result := True;
end;

function GMVisitNodesDepthfirst(const AStartNode: IGMTreeable; const AVisitFunc: TGMNodeVisitMethod; const ARecurse: Boolean; const AParameter: Pointer): Boolean;
  {.$I DoNodeVisitDepthfirst.inc}
  DoNodeVisitDepthFirstInnerFunc
begin
  if Assigned(AVisitFunc) then Result := DoNodeVisitDepthFirst(AStartNode) else Result := True;
end;

{$pop}

function GMFindRootNode(const ANode: IUnknown): IGMTreeable;
begin
  GMQueryInterface(ANode, IGMTreeable, Result);
  while (Result <> nil) and (Result.Parent <> nil) do Result := Result.Parent;
  while (Result <> nil) and (Result.PrevSibling <> nil) do Result := Result.PrevSibling;
end;

function GMNodePath(const ANode: IUnknown; const ADelimStr: TGMString): TGMString;
var node: IGMTreeable;
begin
  Result := '';
  if GMQueryInterface(ANode, IGMTreeable, node) then
   while node <> nil do
    begin
     Result := GMStringJoin(GMGetIntfText(node), ADelimStr, Result);
     node := node.Parent;
    end;
end;

function GMIntfHasDataClass(const AIntf: IUnknown; const ADataClass: TClass): Boolean;
begin
  Result := GMGetDataObject(AIntf) is ADataClass;
end;

function GMGetIntfDataClass(const AIntf: IUnknown): TClass;
var dataObj: TObject;
begin
  dataObj := GMGetDataObject(AIntf);
  if dataObj <> nil then Result := dataObj.ClassType else Result := nil;
end;

function GMIsNodeMatch(ANode: IGMTreeable; const ADataClass: TClass; const ANodeTitle: TGMString): Boolean;
begin
  Result := ((ADataClass = nil) or (GMGetDataObject(ANode) is ADataClass)) and
            ((ANodeTitle = '') or GMSameText(ANodeTitle, GMGetIntfText(ANode)));
end;

function GMFindParentNode(const AStartNode: IGMTreeable; const ADataClass: TClass; const ANodeTitle: TGMString): IGMTreeable;
begin
  Result := AStartNode;
  while (Result <> nil) and not GMIsNodeMatch(Result, ADataClass, ANodeTitle) do Result := Result.Parent;
end;

function GMFindNode(const AStartNode: IGMTreeable; const ADecideFunc: TGMNodeVisitFunc; const ARecurse: Boolean; const AParameter: Pointer): IGMTreeable;
//
// Use the same return value semantics of ADecideFunc here as in GMVisitNodesXxxxx (True => continue iteration, False => stop iteration).
// Otherwise a TGMNodeVisitFunc could not be used for both GMFindNode and GMVisitNodesXxxxx.
//
  function DoFind(ANode: IGMTreeable): IGMTreeable;
  begin
    Result := nil;
    while (ANode <> nil) and (Result = nil) do
     begin
      if not ADecideFunc(ANode, AParameter) then Result := ANode else if ARecurse then Result := DoFind(ANode.FirstChild);
      ANode := ANode.NextSibling;
     end;
  end;
begin
  if Assigned(ADecideFunc) then Result := DoFind(AStartNode) else Result := nil;
end;

function GMNodeMatchVisitFunc(const ANode: IGMTreeable; const AParameter: Pointer): Boolean;
begin
  Result := (AParameter = nil) or not GMIsNodeMatch(ANode, PGMNodeVisitData(AParameter).DataClass, PGMNodeVisitData(AParameter).NodeTitle);
end;

function GMFindNode(const AStartNode: IGMTreeable; const ADataClass: TClass; const ANodeTitle: TGMString; const ARecurse: Boolean): IGMTreeable;
var visitData: RGMNodeVisitData;
begin
  visitData := GMInitNodeVisitData(ADataClass, ANodeTitle);
  Result := GMFindNode(AStartNode, GMNodeMatchVisitFunc, ARecurse, @visitData);
end;

function GMGetDataObject(const AOwner: IUnknown): TObject;
var getDataObj: IGMGetDataObject;
begin
  if GMQueryInterface(AOwner, IGMGetDataObject, getDataObj) then Result := getDataObj.GetDataObject else Result := nil;
end;

function GMSetDataObject(const AOwner: IUnknown; const ADataObj: TObject): TObject;
var PIDataObj: IGMSetDataObject;
begin
  if GMQueryInterface(AOwner, IGMSetDataObject, PIDataObj) then PIDataObj.SetDataObject(ADataObj);
  Result := ADataObj;
end;

function GMGetIntfText(const AIntf: IUnknown; const ADefaultValue: TGMString): TGMString;
var getText: IGMGetText;
begin
  if GMQueryInterface(AIntf, IGMGetText, getText) then Result := getText.Text else Result := ADefaultValue;
end;

procedure GMSetIntfText(const AIntf: IUnknown; const AValue: TGMString);
var setText: IGMGetSetText;
begin
  if GMQueryInterface(AIntf, IGMGetSetText, setText) then setText.Text := AValue;
end;

function GMObjFromIntf(const AIntf: IUnknown): TObject;
var PIObiInfo: IGMObjInfo;
begin
 if not GMQueryInterface(AIntf, IGMObjInfo, PIObiInfo) then Result := nil else Result := PIObiInfo.Instance;
end;

function GMGetIntfStrValue(const AIntf: IUnknown; const ADefaultValue: TGMString = ''): TGMSTring;
var strVal: IGMGetStringValue;
begin
 if GMQueryInterface(AIntf, IGMGetStringValue, strVal) then Result := strVal.StringValue else Result := ADefaultValue;
end;

function GMCompareIUnknown(const AIntf1, AIntf2: IUnknown): TGMCompareResult;
var unk1, unk2: IUnknown;
begin
  Result := crALessThanB;
  if GMQueryInterface(AIntf1, IUnknown, unk1) and GMQueryInterface(AIntf2, IUnknown, unk2) then
   begin
    if PtrUInt(unk1) = PtrUInt(unk2) then Result := crAEqualToB
      else
      if PtrUInt(unk1) > PtrUInt(unk2) then Result := crAGreaterThanB;
   end;
end;

function GMObjAsIntf(const AObj: TObject): IUnknown;
begin
  if (AObj = nil) or not AObj.GetInterface(IUnknown, Result) then Result := nil;
end;

procedure GMCheckGetInterface(const AObj: TObject; const AIID: TGUID; out AIntf; const ACallingName: TGMString);
begin
  GMCheckPointerAssigned(AObj, RStrTheObject, AObj, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
  if not AObj.GetInterface(AIID, AIntf) then raise EGMException.ObjError(MsgIntfNotSupported(RStrTheObject, AIID), AObj, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
end;

procedure GMCheckQueryInterface(const ASource: IUnknown; const AIID: TGUID; out AIntf; const ACallingName: TGMString);
var CallerName: TGMString; Hr: HResult;
  function LocalBuildCallingName: TGMString;
  begin
    if CallerName = '' then CallerName := GMStringJoin(BuildCallingName(ACallingName, {$I %CurrentRoutine%}), ' - ',
                                                       GMFormat('QueryInterface<%s>("%s")', [GMIntfClassName(ASource), GMGuidToString(AIID)]));
    Result := CallerName;
  end;
begin
  //GMCheckPointerAssigned(Pointer(Obj), RStrTheObject, nil, CallerName);
  if ASource = nil then raise EGMException.ObjError(MsgPointerIsNil(RStrTheObject), nil, LocalBuildCallingName);

  //GMHrCheckIntf(Obj.QueryInterface(AIID, AIntf), Obj, CallerName);
  Hr := ASource.QueryInterface(AIID, AIntf);
  if not GMHrSucceeded(Hr) then GMHrCheckIntf(ASource.QueryInterface(AIID, AIntf), ASource, LocalBuildCallingName);

  //if Obj.QueryInterface(AIID, AIntf) <> S_OK then raise EGMException.IntfError(MsgIntfNotSupported(RStrTheObject, AIID), Obj, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
end;

function GMQueryInterface(const ASource: IUnknown; const AIID: TGUID; out AIntf): Boolean;
begin
  Result := (ASource <> nil) and (ASource.QueryInterface(AIID, AIntf) = S_OK);
end;

function GMGetInterface(const AObj: TObject; const AIID: TGUID; out AIntf): Boolean;
begin
  Result := (AObj <> nil) and AObj.GetInterface(AIID, AIntf);
end;

function GMGetWeakInterface(const AObj: TObject; const AIID: TGUID; out AIntf): Boolean;
begin
  Result := (AObj <> nil) and AObj.GetInterfaceWeak(AIID, AIntf);
end;

function GMGetIntfHRCode(const AIntf: IUnknown; const ADefaultHrCode: HResult): HResult;
var getHrCode: IGMGetHRCode;
begin
  if GMQueryInterface(AIntf, IGMGetHRCode, getHrCode) then Result := getHrCode.HRCode else Result := ADefaultHrCode
end;

function GMGetObjHRCode(const AObj: TObject; const ADefaultHrCode: HResult): HResult;
begin
  Result := GMGetIntfHRCode(GMObjAsIntf(AObj), ADefaultHrCode);
end;

procedure GMEnterCriticalSection(const ACriticalSection: IUnknown);
var cs: IGMCriticalSection;
begin
  if GMQueryInterface(ACriticalSection, IGMCriticalSection, cs) then cs.EnterCriticalSection;
//if ACriticalSection <> nil then ACriticalSection.EnterCriticalSection;
end;

procedure GMLeaveCriticalSection(const ACriticalSection: IUnknown);
var cs: IGMCriticalSection;
begin
  if GMQueryInterface(ACriticalSection, IGMCriticalSection, cs) then cs.LeaveCriticalSection;
//if ACriticalSection <> nil then ACriticalSection.LeaveCriticalSection;
end;

function GMIsOneOfIntegers(const AValue: PtrInt; const AIntValues: array of PtrInt): Boolean;
var val: PtrInt; // i: PtrInt;
begin
  //for i:=Low(AIntValues) to High(AIntValues) do if AValue = AIntValues[i] then begin Result := True; Exit; end;
  for val in AIntValues do if val = AValue then Exit(True);
  Result := False;
end;

procedure GMAddIntegersToArray(var ADest: TGMPtrIntArray; const AValues: array of PtrInt);
var i: LongInt;
begin
  if Length(AValues) = 0 then Exit;
  SetLength(ADest, Length(ADest) + Length(AValues));
  for i:=Low(AValues) to High(AValues) do ADest[High(ADest) - Length(AValues) - Low(AValues) + i + 1] := AValues[i];
end;

procedure GMCheckAllInterfacesSupported(const AIntf: IUnknown; const InterfaceIDs: array of TGUID; const ACallingName: TGMString);
var i: LongInt; PIUnknown: IUnknown;
begin
  for i:=Low(InterfaceIDs) to High(InterfaceIDs) do GMCheckQueryInterface(AIntf, InterfaceIDs[i], PIUnknown, ACallingName);
end;

function GMAllInterfacesSupported(const AIntf: IUnknown; const InterfaceIDs: array of TGUID): Boolean;
var i: LongInt; PIUnknown: IUnknown;
begin
  if AIntf = nil then Result := False else
   begin
    Result := True;
    for i:=Low(InterfaceIDs) to High(InterfaceIDs) do
     if GMQueryInterface(AIntf, InterfaceIDs[i], PIUnknown) then
      begin Result := False; break; end;
   end;
end;

procedure GMCheckGetIntfByPosition(const ACollection: IGMGetIntfByPosition; const Position: LongInt; const AIID: TGUID; out AIntf; const ElementName, ACallingName: TGMString);
//var Hr: HResult;
begin
  GMCheckPointerAssigned(Pointer(ACollection), RStrTheList, nil, ACallingName);
  //Hr := ACollection.GetIntfByPosition(Position, AIID, AIntf); // <- Save MsgNoItemIntfPrefix overhead on success
  GMHrCheckIntf(ACollection.GetIntfByPosition(Position, AIID, AIntf), ACollection, ACallingName, MsgNoItemIntfPrefix(GMIntToStr(Position), AIID));
  //then raise EGMException.IntfError(MsgIntfNotSupported(GMFormat('%s %d', [ElementName, Position]), AIID), nil, ACallingName);
end;

function GMGetPropertyIntf(const AObj: TObject; const PropertyName: TGMString; const AIID: TGUID; out AIntf): HResult;
var Prop: LongInt;
begin
  if (PropertyName = '') or (AObj = nil) then Result := E_INVALIDARG else
   begin
    if AObj.ClassInfo = nil then raise EGMException.ObjError(GMFormat(RStrNeedRTTI, [AObj.ClassName]), AObj, 'GMGetPropertyIntf');
    if not GMGetOrdinalProperty(AObj, PropertyName, Prop) or (TObject(Prop) = nil) then
     Result := DISP_E_UNKNOWNNAME
    else
     Result := cQIResult[TObject(Prop).GetInterface(AIID, AIntf)];
   end;
   //GMGetOrdinalProperty(Obj, PropertyName, Prop) and
     //(TObject(Prop) <> nil) then Result := TObject(Prop).GetInterface(AIID, AIntf) else Result := False;
end;

function GMGetPropIntfFromIntf(const AOwner: IUnknown; const APropertyName: TGMString; const AIID: TGUID; out AIntf): HResult;
var getPropIntf: IGMGetPropertyIntf;
begin
  if AOwner = nil then begin Result := E_POINTER; Exit; end;
  Result := AOwner.QueryInterface(IGMGetPropertyIntf, getPropIntf);
  if Result = S_OK then Result := getPropIntf.GetPropertyIntf(APropertyName, AIID, AIntf);
end;

//procedure GMCheckGetPropIntfFromIntf(const Owner: IUnknown; const PropertyName: TGMString; const AIID: TGUID; out AIntf; const ACallingName: TGMString = '');
//var PIGetPropIntf: IGMGetPropertyIntf;
//begin
//  GMCheckQueryInterface(Owner, IGMGetPropertyIntf, PIGetPropIntf, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
//  if not PIGetPropIntf.GetPropertyIntf(PropertyName, AIID, AIntf) then
//   raise EGMException.IntfError(MsgNoIntfOrNotExist(RStrProperty, PropertyName, AIID), Owner, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
//end;

procedure GMReleaseMembers(const AIntf: IUnknown);
var ReleaseReferences: IGMReleaseReferences;
begin
  if GMQueryInterface(AIntf, IGMReleaseReferences, ReleaseReferences) then ReleaseReferences.ReleaseReferences;
end;

function GMGetIntfCount(const AIntf: IUnknown; const ADefaultValue: PtrInt): PtrInt;
var PICount: IGMGetCount;
begin
  if GMQueryInterface(AIntf, IGMGetCount, PICount) then Result := PICount.Count else Result := ADefaultValue;
end;

function GMGetIntfHandle(const AIntf: IUnknown; const ADefaultValue: THandle): THandle;
var PIHandle: IGMGetHandle;
begin
  if GMQueryInterface(AIntf, IGMGetHandle, PIHandle) then Result := PIHandle.Handle else Result := ADefaultValue;
end;

function GMCheckGetIntfHandle(const AIntf: IUnknown; const ACallingName: TGMString): THandle;
var PIHandle: IGMGetHandle;
begin
  GMCheckQueryInterface(AIntf, IGMGetHandle, PIHandle, ACallingName);
  Result := PIHandle.Handle;
end;

function GMGetAllocatedIntfHandle(const AIntf: IUnknown; var Handle: THandle): Boolean;
var PIHAlloc: IGMHandleAllocated; PIHandle: IGMGetHandle;
begin
  if (AIntf <> nil) and ((AIntf.QueryInterface(IGMHandleAllocated, PIHAlloc) <> S_OK) or PIHAlloc.HandleAllocated) and
     (AIntf.QueryInterface(IGMGetHandle, PIHandle) = S_OK) then
   begin Handle := PIHandle.Handle; Result := Handle <> 0; end else Result := False;
end;

function GMGetAllocatedObjHandle(const AObj: TObject; var AHandle: THandle): Boolean;
begin
  Result := GMGetAllocatedIntfHandle(GMObjAsIntf(AObj), AHandle);
end;

function GMIsHandleAllocated(const AIntf: IUnknown): Boolean;
var PIHandle: IGMGetHandle; PIHAlloc: IGMHandleAllocated;
begin
  //
  // If we can get a IGMHandleAllocated avoid accessing PIHandle.Handle because this may create a handle
  // but we only want to test for a valid handle here and dont want to create one!
  //
  Result :=  (AIntf <> nil) and ((AIntf.QueryInterface(IGMHandleAllocated, PIHAlloc) <> S_OK) or PIHAlloc.HandleAllocated) and
             (AIntf.QueryInterface(IGMGetHandle, PIHandle) = S_OK) and (PIHandle.Handle <> 0);
end;

function GMCreateCopyQI(const ASource: IUnknown; const AIID: TGUID; out AIntf): HResult;
var PICreateCopy: IGMCreateCopyQI;
begin
  if ASource = nil then Result := E_INVALIDARG else
   begin
    Result := ASource.QueryInterface(IGMCreateCopyQI, PICreateCopy);
    if Result <> S_OK then Exit;
    Result := PICreateCopy.CreateCopyQI(AIID, AIntf);
   end;
end;

function GMObjCreateCopyQI(const ASource: TObject; const AIID: TGUID; out AIntf): HResult;
var PICreateCopy: IGMCreateCopyQI;
begin
  if ASource = nil then Result := E_INVALIDARG else
   begin
    if not ASource.GetInterface(IGMCreateCopyQI, PICreateCopy) then begin Result := E_NOINTERFACE; Exit; end;
    Result := PICreateCopy.CreateCopyQI(AIID, AIntf);
   end;
end;

function GMFindParentObj(const AObj: TObject; const AParentClass: TClass; out AParent): Boolean;
var PIParent: IGMGetParentObj; Prnt: TObject;
begin
  Result := False; TObject(AParent) := nil;
  if AObj = nil then Exit;
  if AObj is AParentClass then begin TObject(AParent) := AObj; Result := True; Exit; end;
  if not AObj.GetInterface(IGMGetParentObj, PIParent) then Exit;
  Prnt := PIParent.ParentObj;
  while (Prnt <> nil) and not (Prnt is AParentClass) do
   begin
    if not Prnt.GetInterface(IGMGetParentObj, PIParent) then Exit;
    Prnt := PIParent.ParentObj;
   end;
  Result := Prnt <> nil;
  if Result then TObject(AParent) := Prnt;
end;

function GMIsParentObj(AParent, AObj: TObject; const AllowIdentity: Boolean; const AStopAtClass: TClass): Boolean;
var PIParent: IGMGetParentObj;
begin
  if not AllowIdentity and (AObj = AParent) then begin Result := False; Exit; end;
  repeat
   Result := (AObj <> nil) and (AObj = AParent);
   if Result or (AObj = nil) or ((AStopAtClass <> nil) and (AObj is AStopAtClass)) then Break;
   if not AObj.GetInterface(IGMGetParentObj, PIParent) then Break;
   AObj := PIParent.ParentObj;
  until Result;
end;

procedure GMCheckFindParentObj(const AObj: TObject; const ParentClass: TClass; out Parent);
begin
  GMCheckPointerAssigned(AObj, RStrTheObject, AObj, {$I %CurrentRoutine%});
  if not GMFindParentObj(AObj, ParentClass, Parent) then
   raise EGMException.ObjError(GMFormat(RStrNoParentFmt, [GMGetObjName(AObj), GMClassName(ParentClass)]), AObj, {$I %CurrentRoutine%});
end;

function GMGetObjText(const AObj: TObject; const ADefaultValue: TGMString): TGMString;
var PIText: IGMGetText;
begin
  if (AObj <> nil) and AObj.GetInterface(IGMGetText, PIText) then Result := PIText.Text else Result := ADefaultValue;
end;

procedure GMSetObjText(const AObj: TObject; const AValue: TGMString);
var PIText: IGMGetSetText;
begin
  if (AObj <> nil) and AObj.GetInterface(IGMGetSetText, PIText) then PIText.Text := AValue;
end;

//function GMFindParentObj(const Area: TObject; const ObjClass: TClass; out AObj): Boolean;
//var PIParent: IGMGetParentObj; PILayout: IGMUiArea; Parent: TObject;
//begin
//  GMCheckGetInterface(Area, IGMGetParentObj, PIParent, {$I %CurrentRoutine%});
//  Parent := PIParent.ParentObj;
//  while (Parent <> nil) and not (Parent is ObjClass) do
//   begin
//    GMCheckGetInterface(Parent, IGMGetParentObj, PIParent, {$I %CurrentRoutine%});
//    Parent := PIParent.ParentObj;
//   end;
//  Result := Parent <> nil;
//  if Result then TObject(AObj) := Parent;
//end;

//function GMObjHandleAllocated(const AObj: TObject): Boolean;
//var PIHandle: IGMGetHandle;
//begin
//  Result := (AObj <> nil) and AObj.GetInterface(IGMGetHandle, PIHandle) and (PIHandle.HAndle <> 0);
//end;

//function GMIntfHandleAllocated(const AIntf: IUnknown): Boolean;
//var PIHandle: IGMGetHandle;
//begin
//  Result := (AIntf <> nil) and (AIntf.QueryInterface(IGMGetHandle, PIHandle) = S_OK) and (PIHandle.HAndle <> 0);
//end;


{ --------------------- }
{ ---- COM Helpers ---- }
{ --------------------- }

function GMHrSucceeded(const AErrorCode: HResult): Boolean;
begin
  Result := AErrorCode and $80000000 = 0;
end;

//function GMSysErrorMsg(const AErrorCode: LongInt): TGMString;
//begin
//  Result := SysErrorMessage(AErrorCode);
//  if Result = '' then Result := GMFormat(RStrSysErrorCodeFmt, [AErrorCode, AErrorCode]);
//end;

procedure GMRegisterErrorMsgModule(const AModuleName: TGMString);
begin
  if not GMIsOneOfStrings(AModuleName, vGMErrorMsgModules) then GMAddStrToArray(AModuleName, vGMErrorMsgModules);
end;

procedure GMExtendExceptionMsg(const AException: TObject; const AAddition: TGMString; const ASeparator: TGMString = ': '; const APrefix: Boolean = True);
var setText: IGMGetSetText;
  function BuildMsg(const ACurrentMsg: TGMString): TGMString;
  begin
    if APrefix then
      Result := GMStringJoin(AAddition, ASeparator, ACurrentMsg)
    else
      Result := GMStringJoin(ACurrentMsg, ASeparator, AAddition);
  end;
begin
  if GMGetInterface(AException, IGMGetSetText, setText) then setText.Text := BuildMsg(setText.Text)
  else
  if GMIsClassByName(AException, Exception) then
     Exception(AException).Message := BuildMsg(Exception(AException).Message);
end;

function GMSysErrorMsg(const AErrorCode: LongInt; const AParams: array of PGMChar): TGMString;
var apiCode: DWORD; pParams: Pointer; i: Integer; unresolved: TGMString;
  function BuildSysErrMsg(AFlags: DWORD): TGMString;
  var buffer: PGMChar; len: DWORD;
  begin
    len := FormatMessage(AFlags or FORMAT_MESSAGE_ALLOCATE_BUFFER, nil, DWORD(AErrorCode), 0, PGMChar(@buffer), 0, pParams);
    if len = 0 then begin apiCode := GetLastError; Result := ''; end else
     begin
      apiCode := ERROR_SUCCESS;

      while (len > 0) do
       case buffer[len - 1] of
        #0..#32, '.': Dec(len);
        else Break;
       end;

      SetString(Result, buffer, len);
      if LocalFree(HLOCAL(buffer)) <> 0 then GMTrace('GMSysErrorMsg - LocalFree failed!', tpWarning);
     end;
  end;

  function DoInserts(const AMsg: TGMString): TGMString;
  var bitMaskInserted, chPos, i, argNo: Integer; strToInsert: TGMString; pCh, pChFound: PGMChar; // unResolved,
  begin
    Result := AMsg;  unResolved := '';  chPos := 1;  bitMaskInserted := 0;
    repeat
     pCh := @Result[chPos]; // <- Result may have been moved due to modifications!
     pChFound := GMStrLScan(pCh, '%', Length(Result) - chPos + 1);
     if pChFound <> nil then
      begin
       Inc(chPos, pChFound - pCh);
       Inc(pChFound);
       argNo := Ord(pChFound^) - 49;
       if GMIsInRange(argNo, Low(AParams), High(AParams)) then
        begin
         strToInsert := AParams[argNo];
         System.Delete(Result, chPos, 2);
         System.Insert(strToInsert, Result, chPos);
         Inc(chPos, Length(strToInsert));
         bitMaskInserted := bitMaskInserted or (1 shl argNo);
        end;
      end;
    until (pChFound = nil) or (pChFound^ = #0);

    for i:=Low(AParams) to High(AParams) do
      if bitMaskInserted and (1 shl i) = 0 then unResolved := GMStringJoin(unResolved, ', ', AParams[i]);
  end;

begin
  if Length(AParams) = 0 then pParams := nil else pParams := @AParams[Low(AParams)];

  //Result := BuildSysErrMsg(FORMAT_MESSAGE_FROM_SYSTEM or FORMAT_MESSAGE_ARGUMENT_ARRAY);

  //if (Length(Result) <= 0) and (apiCode = ERROR_INVALID_PARAMETER) then
  Result := BuildSysErrMsg(FORMAT_MESSAGE_FROM_SYSTEM or FORMAT_MESSAGE_ARGUMENT_ARRAY or FORMAT_MESSAGE_IGNORE_INSERTS);

  if (Length(Result) <= 0) then // and (Length(vGMErrorMsgModules) > 0)
   for i:=Low(vGMErrorMsgModules) to High(vGMErrorMsgModules) do
    begin
     Result := GMModuleErrorMessage(vGMErrorMsgModules[i], LongWord(AErrorCode and $0000FFFF));
     if Length(Result) > 0 then Break;
    end;

  unresolved := '';
  Result := DoInserts(Result);
  Result := GMStringJoin(Result, ': ', unresolved);

  if Length(Result) <= 0 then Result := GMFormat(RStrUnknownSysErrorMsgFmt, [AErrorCode]);
end;

function GMGuidToString(const Guid: TGUID): TGMString;
var PWStr: PWideChar;
begin
  GMHrCheckObj(StringFromCLSID(Guid, PWStr), nil, {$I %CurrentRoutine%});
  Result := PWStr;
  CoTaskMemFree(PWStr);
end;

procedure GMCopyGuid(const ASource, ADest: IUnknown; const ACallingName: TGMString);
var PISrcGuid: IGMGetGuid; PIDstGuid: IGMGetSetGuid;
begin
  GMCheckQueryInterface(ASource, IGMGetGuid, PISrcGuid, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
  GMCheckQueryInterface(ADest, IGMGetSetGuid, PIDstGuid, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
  PIDstGuid.GUID := PISrcGuid.GUID;
end;

function GMMakeGuidStr(const GuidStr: TGMString): TGMString;
begin
  Result := GMStrip(GuidStr, cGuidStripChars);
  if Result <> '' then Result := '{' + Result + '}';
end;

function GMCreateGuid: TGuid;
begin
  GMHrCheckObj(CoCreateGuid(Result), nil, 'GMCreateGuid');
end;

function GMIsGUID(const AGuidStr: TGMString): Boolean;
var GUID: TGUID; guidWStr: UnicodeString;
begin
  guidWStr := GMMakeGuidStr(AGuidStr);
  Result := GMHrSucceeded(CLSIDFromString(PWideChar(guidWStr), GUID));
end;

function GMStringToGuid(const AGuidStr: TGMString; const ACaller: TObject; const ACallingName: TGMString): TGUID;
var guidWStr: UnicodeString;
begin
  guidWStr := GMMakeGuidStr(AGuidStr);
  GMHrCheckObj(CLSIDFromString(PWideChar(guidWStr), Result), ACaller, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
end;

function GMMakeGuid(GuidStr: TGMString; const AObj: TObject; const ACallingName: TGMString): TGUID;
begin
  GuidStr := GMMakeGuidStr(GuidStr);
  if GuidStr <> '' then
   Result := GMStringToGuid(GuidStr, AObj, ACallingName)
  else
   Result := GUID_NULL;
end;

function GMCreateComObject(const ClassID: TGUID; const CreateContext: DWORD): IUnknown;
begin
  GMHrCheckObj(CoCreateInstance(ClassID, nil, CreateContext, IUnknown, Result), nil, {$I %CurrentRoutine%});
end;

function GMCompareGuids(const GuidA, GuidB: TGUID): TGMCompareResult;
var Cmp: LongInt;
begin
  Cmp := CompareText(GMGuidToString(GuidA), GMGuidToString(GuidB));
  if Cmp < 0 then Result := crALessThanB else if Cmp = 0 then Result := crAEqualToB else Result := crAGreaterThanB
end;

function GMEqualGuids(const GuidA, GuidB: TGUID): Boolean;
begin
  Result := CompareMem(@GuidA, @GuidB, SizeOf(TGUID)); // GMCompareGuids(GuidA, GuidB) = crAEqualToB;
end;

function GMCoClassIsRegistered(const ClassId: TGUID): Boolean;
var ProgId: PWideChar;
begin
  ProgId := nil;
  Result := ProgIDFromCLSID(ClassId, ProgId) = S_OK;
  if ProgId <> nil then CoTaskMemFree(ProgId);
end;


{ -------------------------- }
{ ---- Storage Routines ---- }
{ -------------------------- }

function GMIsStorageGuid(const Storage: IStorage; const Guid: TGUID; const Caller: TObject): Boolean;
var Stat: TStatStg;
begin
  if Storage = nil then Result := False else
   begin
    GMHrCheckObj(Storage.Stat(Stat, STATFLAG_NONAME), Caller, {$I %CurrentRoutine%});
    //if Stat.pwcsName <> nil then CoTaskMemFree(Stat.pwcsName);
    Result :=  IsEqualGuid(Stat.clsid, Guid);
   end;
end;

function GMIsStorageFileGuid(const FileName: TGMString; const Guid: TGUID; const Caller: TObject): Boolean;
var Storage: IStorage;
begin
  GMCheckFileExists(FileName, Caller, {$I %CurrentRoutine%});
  //if not FileExists(FileName) then raise EGMException.ObjError(GMFormat(RStrFileNotExists, [FileName]), Caller, {$I %CurrentRoutine%});
  GMHrCheckObj(StgOpenStorage(PWideChar(UnicodeString(FileName)), nil, STGM_READWRITE or STGM_SHARE_EXCLUSIVE, nil, 0, Storage), Caller, {$I %CurrentRoutine%});
  Result := GMIsStorageGuid(Storage, Guid);
end;

procedure GMCheckIsStorageGuid(const Storage: IStorage; const Guid: TGUID; const Caller: TObject);
var Stat: TStatStg;
begin
  if Storage <> nil then
   begin
    GMHrCheckObj(Storage.Stat(Stat, STATFLAG_NONAME), Caller, {$I %CurrentRoutine%});
    //if Stat.pwcsName <> nil then CoTaskMemFree(Stat.pwcsName);
    if not IsEqualGuid(Stat.clsid, Guid) then raise EGMException.ObjError(GMFormat(RStrInvalidStorageGuid, [GMGuidToString(Guid), GMGuidToString(Stat.clsid)]), Caller, {$I %CurrentRoutine%});
   end;
end;

procedure GMCheckIsStorageFileGuid(const FileName: TGMString; const Guid: TGUID; const Caller: TObject = nil);
var Storage: IStorage;
begin
  GMCheckFileExists(FileName, Caller, {$I %CurrentRoutine%});
  //if not FileExists(FileName) then raise EGMException.ObjError(GMFormat(RStrFileNotExists, [FileName]), Caller, {$I %CurrentRoutine%});
  GMHrCheckObj(StgOpenStorage(PWideChar(UnicodeString(FileName)), nil, STGM_READWRITE or STGM_SHARE_EXCLUSIVE, nil, 0, Storage), Caller, {$I %CurrentRoutine%});
  GMCheckIsStorageGuid(Storage, Guid, Caller);
end;


{ ---------------------------------- }
{ ---- Connection Point Helpers ---- }
{ ---------------------------------- }

function GMDoNotifySink(const ANotificationsEnabled: Boolean; const ANotifySink: IUnknown; const AIID: TGUID; out AIntf): Boolean;
begin
  if not GMQueryInterface(ANotifySink, AIID, AIntf) then Result := False else
   Result := ANotificationsEnabled or GMAskBoolean(ANotifySink, Ord(bvAlwaysNotify), False);
end;

procedure GMInterfaceConnect(const ASink: TObject; const AContainer: IUnknown; const AIID: TGUID; var ACookie: LongInt; const ACallingName: TGMString);
var PISink: IUnknown;
begin
  if (ASink <> nil) and (AContainer <> nil) then
   begin
    ACookie := cInvalidCPCookie;
    GMCheckGetInterface(ASink, IUnknown, PISink, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
    GMInterfaceConnect(PISink, AContainer, AIID, ACookie, ASink, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
   end;
end;

procedure GMInterfaceConnect(const ASink, AContainer: IUnknown; const AIID: TGUID; var ACookie: LongInt; const AObj: TObject; const ACallingName: TGMString); overload;
var CPC: IConnectionPointContainer; CP: IConnectionPoint;
begin
  if (ASink <> nil) and (AContainer <> nil) then
   begin
    ACookie := cInvalidCPCookie;
    if not GMQueryInterface(AContainer, IConnectionPointContainer, CPC) then raise EGMException.ObjError(MsgIntfNotSupported(RStrCPC, IConnectionPointContainer), AObj, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
    if CPC.FindConnectionPoint(AIID, CP) <> S_OK then raise EGMException.ObjError(GMFormat(RStrNoCPForIntf, [RStrCPC, GMGuidToString(AIID)]), AObj, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
    GMHrCheckObj(CP.Advise(ASink, ACookie), AObj, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
   end;
end;

procedure GMQuietInterfaceConnect(const AObj: TObject; const AContainer: IUnknown; const AIID: TGUID; var ACookie: LongInt);
var sink: IUnknown; cpc: IConnectionPointContainer; cp: IConnectionPoint;
begin
  if (AObj <> nil) and (AContainer <> nil) then
   begin
    ACookie := cInvalidCPCookie;
    if AObj.GetInterface(IUnknown, sink) and GMQueryInterface(AContainer, IConnectionPointContainer, cpc) and
       (cpc.FindConnectionPoint(AIID, cp) = S_OK) then cp.Advise(sink, ACookie);
   end;
end;

procedure GMInterfaceDisconnect(const AContainer: IUnknown; const AIID: TGUID; var ACookie: LongInt);
var CPC: IConnectionPointContainer; CP: IConnectionPoint;
begin
  if (AContainer <> nil) and (ACookie <> cInvalidCPCookie) then
   if GMHrSucceeded(AContainer.QueryInterface(IConnectionPointContainer, CPC)) then
    if GMHrSucceeded(CPC.FindConnectionPoint(AIID, CP)) then
     if GMHrSucceeded(CP.Unadvise(ACookie)) then ACookie := cInvalidCPCookie;
end;

{$push}
{$macro on}
{$define InsertNotifyConnectionsLocalProc:=
  procedure NotifyConnections;
  var enumConnections: IEnumConnections; ConnectData: tagConnectData; PISinkUnk: IUnknown;
  begin
   if (connectionPoint <> nil) and (connectionPoint.EnumConnections(enumConnections) = S_OK) then
    while enumConnections.Next(1, ConnectData, nil) = S_OK do
     if GMDoNotifySink(ANotificationsEnabled, ConnectData.pUnk, IUnknown, PISinkUnk) then ANotifyProc(ConnectData.pUnk, AParams);
     //if ConnectData.pUnk <> nil then ANotifyProc(ConnectData.pUnk, Params); {(ConnectData.pUnk.QueryInterface(AConnectionPointIID, NotifySink) = S_OK)}
  end;}

procedure GMCpcCallNotifySinks(const ACpc: IConnectionPointContainer; const AConnectionPointIID: TGUID; const ANotifyProc: TCPCNotifyProc; const ANotificationsEnabled: Boolean; const AParams: TCPCNotifyParams);
var EnumCP: IEnumConnectionPoints; connectionPoint: IConnectionPoint;
  InsertNotifyConnectionsLocalProc
begin
  if (ACpc <> nil) and Assigned(ANotifyProc) then
   begin
    if IsEqualGuid(AConnectionPointIID, GUID_NULL) then
     begin
      if ACpc.EnumConnectionPoints(EnumCP) = S_OK then
         while EnumCP.Next(1, connectionPoint, nil) = S_OK do NotifyConnections; //(connectionPoint);
     end
    else
     if ACpc.FindConnectionPoint(AConnectionPointIID, connectionPoint) = S_OK then NotifyConnections; // (connectionPoint);
   end;
end;

procedure GMCpcCallNotifySinksObj(const ACpc: IConnectionPointContainer; const AConnectionPointIID: TGUID; const ANotifyProc: TCPCNotifyMethod; const ANotificationsEnabled: Boolean; const AParams: TCPCNotifyParams);
var EnumCP: IEnumConnectionPoints; connectionPoint: IConnectionPoint;
  InsertNotifyConnectionsLocalProc
begin
  if (ACpc <> nil) and Assigned(ANotifyProc) then
   begin
    if IsEqualGuid(AConnectionPointIID, GUID_NULL) then
     begin
      if ACpc.EnumConnectionPoints(EnumCP) = S_OK then
         while EnumCP.Next(1, connectionPoint, nil) = S_OK do NotifyConnections; // (connectionPoint);
     end
    else
     if ACpc.FindConnectionPoint(AConnectionPointIID, connectionPoint) = S_OK then NotifyConnections; // (connectionPoint);
   end;
end;
{$pop}

procedure GMRequestCPCDisconnect(const ACpc: IConnectionPointContainer);
var EnumCP: IEnumConnectionPoints; ConnectionPoint: IConnectionPoint; CpIID: TGUID;
    EnumConnections: IEnumConnections; ConnectData: tagConnectData; ConnectedObj: IGMDisconnectFromConnectionPoint;
begin
  if (ACpc <> nil) and (ACpc.EnumConnectionPoints(EnumCP) = S_OK) then
   while EnumCP.Next(1, ConnectionPoint, nil) = S_OK do
    if (ConnectionPoint.GetConnectionInterface(CpIID) = S_OK) and (ConnectionPoint.EnumConnections(EnumConnections) = S_OK) then
     while EnumConnections.Next(1, ConnectData, nil) = S_OK do
      if GMQueryInterface(ConnectData.pUnk, IGMDisconnectFromConnectionPoint, ConnectedObj) then
       try ConnectedObj.DisconnectFromConnectionPoint(ACpc, CpIID, ConnectData.dwCookie); except end;
end;


{ ----------------------------------------- }
{ ---- Connection Point Notify Helpers ---- }
{ ----------------------------------------- }

procedure GMCallSinkClose(const NotifySink: IUnknown; const Params: TCPCNotifyParams);
var activate: IGMGetSetActive;
begin
  if GMQueryInterface(NotifySink, IGMGetSetActive, activate) then activate.Active := False; // <- Don't eat exceptions!
end;

procedure GMCallSinkBeforeActiveChange(const NotifySink: IUnknown; const Params: TCPCNotifyParams);
var Sink: IGMActiveChangeNotifications;
begin
  if (Length(Params) > 0) and GMQueryInterface(NotifySink, IGMActiveChangeNotifications, Sink) then Sink.BeforeActiveChange(Params[Low(Params)]);
end;

procedure GMCallSinkAfterActiveChange(const NotifySink: IUnknown; const Params: TCPCNotifyParams);
var Sink: IGMActiveChangeNotifications;
begin
  if (Length(Params) > 0) and GMQueryInterface(NotifySink, IGMActiveChangeNotifications, Sink) then {try} Sink.AfterActiveChange(Params[Low(Params)]); //except end;
end;

procedure GMCallSinkBeforePositionChange(const NotifySink: IUnknown; const Params: TCPCNotifyParams);
var Sink: IGMPositionChangeNotifications;
begin
  // No try Except around the Call Here!
  if GMQueryInterface(NotifySink, IGMPositionChangeNotifications, Sink) then Sink.BeforePositionChange;
end;

procedure GMCallSinkAfterPositionChange(const NotifySink: IUnknown; const Params: TCPCNotifyParams);
var Sink: IGMPositionChangeNotifications;
begin
  if GMQueryInterface(NotifySink, IGMPositionChangeNotifications, Sink) then try Sink.AfterPositionChange; except end;
end;

procedure GMCallSinkBeforeOperation(const NotifySink: IUnknown; const Params: TCPCNotifyParams);
var Sink: IGMOperationNotifications;
begin
  if GMQueryInterface(NotifySink, IGMOperationNotifications, Sink) then
   case Length(Params) of
    1: Sink.BeforeOperation(Params[Low(Params)]);
    2: Sink.BeforeOperation(Params[Low(Params)], Params[Low(Params) + 1]);
    else raise EGMException.ObjError(GMFormat(RStrInvalidParamCountFmt, [Length(Params)]), nil, {$I %CurrentRoutine%});
   end;
end;

procedure GMCallSinkAfterOperation(const NotifySink: IUnknown; const Params: TCPCNotifyParams);
var Sink: IGMOperationNotifications;
begin
  if GMQueryInterface(NotifySink, IGMOperationNotifications, Sink) then
   case Length(Params) of
    1: try Sink.AfterOperation(Params[Low(Params)]); except end;
    2: try Sink.AfterOperation(Params[Low(Params)], Params[Low(Params) + 1]); except end;
    else raise EGMException.ObjError(GMFormat(RStrInvalidParamCountFmt, [Length(Params)]), nil, {$I %CurrentRoutine%});
   end;
end;

procedure GMCallSinkValidateValue(const NotifySink: IUnknown; const Params: TCPCNotifyParams);
var Sink: IGMValidateValues;
begin
  // No try Except around the Call Here!
  if GMQueryInterface(NotifySink, IGMValidateValues, Sink) then Sink.ValidateValues;
end;


{ ------------------------------------------- }
{ ---- Value AStorage Directory Routines ---- }
{ ------------------------------------------- }

procedure GMVsdCreatePath(const AStorage: IUnknown; const ADirPath: TGMString);
var valStgDir: RGMTypedIntf<IGMValueStorageDirectory>; chPos: PtrInt; Dir: TGMString; //AbsPath: Boolean;
begin
  if not valStgDir.QueryFrom(AStorage) then Exit;
  chPos := 1;
  if not GMIsRelativePath(ADirPath) then GMVsdOpenDir(AStorage, '\', True);
  repeat
   Dir := GMNextWord(chPos, ADirPath, cDirSep);
   if Length(Dir) > 0 then GMVsdOpenDir(AStorage, Dir, True);
  until Dir = '';
end;

function GMVsdOpenDir(const AStorage: IUnknown; const ADirPath: TGMString; const ACreateIfNotExist: Boolean): Boolean;
var valStgDir: RGMTypedIntf<IGMValueStorageDirectory>;
begin
  if not valStgDir.QueryFrom(AStorage) then Result := False else
   begin
    Result := valStgDir.Intf.OpenDir(ADirPath, ACreateIfNotExist);
    if not Result and ACreateIfNotExist then raise EGMException.IntfError(GMFormat(RStrCreateDirFailed, [ADirPath]), AStorage, {$I %CurrentRoutine%});
   end;
end;

function GMVsdDirExists(const AStorage: IUnknown; const ADirPath: TGMString): Boolean;
var valStgDir: RGMTypedIntf<IGMValueStorageDirectory>; oldDirPath: TGMString; // dirPathKeeper: IUnknown;
begin
  if not valStgDir.QueryFrom(AStorage) then Result := False else
   begin
//  dirPathKeeper := TGMVsdDirPathKeeper.Create(AStorage);
    oldDirPath := valStgDir.Intf.CurrentPath;
    try
     Result := valStgDir.Intf.OpenDir(ADirPath, False);
    finally
     valStgDir.Intf.OpenDir(oldDirPath, False);
    end;
   end;
end;

function GMVsdOpenAbsDir(const AStorage: IUnknown; const ADirPath: TGMString; const ACreateIfNotExist: Boolean): Boolean;
begin
  Result := GMVsdOpenDir(AStorage, GMAbsPath(ADirPath), ACreateIfNotExist);
end;

function GMVsdDeleteAbsDir(const Storage: IUnknown; const ADirPath: TGMString): Boolean;
begin
  Result := GMVsdDeleteDir(Storage, GMAbsPath(ADirPath));
end;

function GMVsdDeleteDir(const AStorage: IUnknown; const ADirPath: TGMString): Boolean;
var valStgDir: RGMTypedIntf<IGMValueStorageDirectory>;
begin
  if valStgDir.QueryFrom(AStorage) then Result := valStgDir.Intf.DeleteDir(ADirPath) else Result := False;
end;

function GMVsdDeleteValue(const AStorage: IUnknown; const AValueName: TGMString): Boolean;
var valStgDir: RGMTypedIntf<IGMValueStorageDirectory>;
begin
  if valStgDir.QueryFrom(AStorage) then Result := valStgDir.Intf.DeleteValue(AValueName) else Result := False;
end;

function GMVsdContainsValue(const AStorage: IUnknown; const AValueName: TGMString): Boolean;
var containsVal: RGMTypedIntf<IGMContainsValue>;
begin
  if containsVal.QueryFrom(AStorage) then Result := containsVal.Intf.ContainsValue(AValueName) else Result := False;
end;

procedure GMVsdReadSubDirNames(const AStorage: IUnknown; var ASubDirNames: TGMStringArray);
var valStgDir: RGMTypedIntf<IGMValueStorageDirectory>;
begin
  if valStgDir.QueryFrom(AStorage) then valStgDir.Intf.ReadSubDirNames(ASubDirNames);
end;

procedure GMVsdReadValueNames(const AStorage: IUnknown; var AValueNames: TGMStringArray);
var valStgDir: RGMTypedIntf<IGMValueStorageDirectory>;
begin
  if valStgDir.QueryFrom(AStorage) then valStgDir.Intf.ReadValueNames(AValueNames);
end;

function GMVsdCurrentPath(const AStorage: IUnknown): TGMString;
var valStgDir: RGMTypedIntf<IGMValueStorageDirectory>;
begin
  if valStgDir.QueryFrom(AStorage) then Result := valStgDir.Intf.CurrentPath else Result := '';
end;

procedure GMVsdCommit(const AStorage: IUnknown);
var valStgDir: RGMTypedIntf<IGMValueStorageDirectory>;
begin
  if valStgDir.QueryFrom(AStorage) then valStgDir.Intf.Commit;
end;

{function GMVsdValueNameExists(const Storage: IUnknown; const ValueName: TGMString): Boolean;
var ValueNames: IGMStrings; valStgDir: IGMValueStorageDirectory;
begin
  if (Storage = nil) or (Storage.QueryInterface(IGMValueStorageDirectory, valStgDir) <> S_OK) then Result := False else
   begin
    ValueNames := TGMStringList.Create(False, True);
    ValueNames.Sorted := True;
    valStgDir.ReadValueNames(ValueNames);
    Result := ValueNames.IndexOf(ValueName) <> CInvalidItemIdx;
   end;
end;}

{procedure GMVsdDeletePrefixedValues(const Storage: IUnknown; const PrefixStr: TGMString);
var i: LongInt; ValueNames: IGMStrings; valStgDir: IGMValueStorageDirectory;
begin
  if (Storage <> nil) and (Storage.QueryInterface(IGMValueStorageDirectory, valStgDir) = S_OK) then
   begin
    ValueNames := TGMStringList.Create(False, True);
    valStgDir.ReadValueNames(ValueNames);
    for i:=0 to ValueNames.Count-1 do if GMIsPrefixStr(PrefixStr, ValueNames[i], True) then valStgDir.DeleteValue(ValueNames[i]);
   end;
end;

procedure GMVsdDeleteValues(const Storage: IUnknown);
begin
  GMVsdDeletePrefixedValues(Storage, '');
end; }

procedure GMStoreString(const ADest: IGMStringStorage; const AValueName, AValue: TGMString; const ADefaultValue: TGMString = cDfltReadString);
begin
  if ADest = nil then Exit;
  if not GMSameText(AValue, ADefaultValue) then ADest.WriteString(AValueName, AValue) else GMVsdDeleteValue(ADest, AValueName);
end;

procedure GMStoreInteger(const ADest: IGMValueStorage; const AValueName: TGMString; const AValue: LongInt; const ADefaultValue: LongInt = cDfltReadInteger);
begin
  if ADest = nil then Exit;
  if AValue <> ADefaultValue then ADest.WriteInteger(AValueName, AValue) else GMVsdDeleteValue(ADest, AValueName);
end;

procedure GMStoreInt64(const ADest: IGMValueStorage; const AValueName: TGMString; const AValue: Int64; const ADefaultValue: Int64 = cDfltReadInteger);
begin
  if ADest = nil then Exit;
  if AValue <> ADefaultValue then ADest.WriteInt64(AValueName, AValue) else GMVsdDeleteValue(ADest, AValueName);
end;

procedure GMStoreBoolean(const ADest: IGMValueStorage; const AValueName: TGMString; const AValue: Boolean; const ADefaultValue: Boolean = cDfltReadBoolean);
begin
  if ADest = nil then Exit;
  if AValue <> ADefaultValue then ADest.WriteBoolean(AValueName, AValue) else GMVsdDeleteValue(ADest, AValueName);
end;

procedure GMStoreDateTime(const ADest: IGMValueStorage; const AValueName: TGMString; const AValue: TDateTime; const ADefaultValue: TDateTime = cDfltReadDateTime);
begin
  if ADest = nil then Exit;
  if AValue <> ADefaultValue then ADest.WriteDateTime(AValueName, AValue) else GMVsdDeleteValue(ADest, AValueName);
end;

procedure GMStoreDouble(const ADest: IGMValueStorage; const AValueName: TGMString; const AValue: Double; const ADefaultValue: Double = cDfltReadFloat);
begin
  if ADest = nil then Exit;
  if AValue <> ADefaultValue then ADest.WriteDouble(AValueName, AValue) else GMVsdDeleteValue(ADest, AValueName);
end;

procedure GMStoreVariant(const ADest: IGMValueStorage; const AValueName: TGMString; const AValue, ADefaultValue: OleVariant);
begin
  if ADest = nil then Exit;
  if (VarType(AValue) <> VarType(ADefaultValue)) or (AValue <> ADefaultValue) then ADest.WriteVariant(AValueName, AValue)
     else GMVsdDeleteValue(ADest, AValueName);
end;

procedure GMVsdLoadFromDir(const ASource: IGMValueStorage; const ALoadPath: TGMString; const ALoadProc: TGMLoadStoreValuesProc; const ACryptCtrlData: PGMCryptCtrlData);
var threadSync: RGMCriticalSectionLock; pathKeeper: IUnknown;
begin
  if (ASource <> nil) and (ALoadPath <> '') and Assigned(ALoadProc) then
   begin
    threadSync.Lock(ASource);
    pathKeeper := TGMVsdDirPathKeeper.Create(ASource);
    if GMVsdOpenDir(ASource, ALoadPath, False) then ALoadProc(ASource, ACryptCtrlData);
   end;
end;

procedure GMVsdStoreToDir(const ADest: IGMValueStorage; const AStorePath: TGMString; const AStoreProc: TGMLoadStoreValuesProc; const ACryptCtrlData: PGMCryptCtrlData);
var threadSync: RGMCriticalSectionLock; pathKeeper: IUnknown;
begin
  if (ADest <> nil) and (AStorePath <> '') and Assigned(AStoreProc) then
   begin
    threadSync.Lock(ADest);
    pathKeeper := TGMVsdDirPathKeeper.Create(ADest, AStorePath, True);
    AStoreProc(ADest, ACryptCtrlData);
   end;
end;

procedure GMVsdLoadFromDir(const ASource: IGMValueStorage; const ALoadPath: TGMString; const ALoadIntf: IUnknown; const ACryptCtrlData: PGMCryptCtrlData);
var threadSync: RGMCriticalSectionLock; pathKeeper: IUnknown; PILoadStore: IGMLoadStoreData;
begin
  if (ASource <> nil) and (Length(ALoadPath) > 0) and GMQueryInterface(ALoadIntf, IGMLoadStoreData, PILoadStore) then
   begin
    threadSync.Lock(ASource);
    pathKeeper := TGMVsdDirPathKeeper.Create(ASource);
    if GMVsdOpenDir(ASource, ALoadPath, False) then PILoadStore.LoadData(ASource, ACryptCtrlData);
   end;
end;

procedure GMVsdStoreToDir(const ADest: IGMValueStorage; const AStorePath: TGMString; const AStoreIntf: IUnknown; const ACryptCtrlData: PGMCryptCtrlData);
var threadSync: RGMCriticalSectionLock; pathKeeper: IUnknown; PILoadStore: IGMLoadStoreData;
begin
  if (ADest <> nil) and (Length(AStorePath) > 0) and GMQueryInterface(AStoreIntf, IGMLoadStoreData, PILoadStore) then
   begin
    threadSync.Lock(ADest);
    pathKeeper := TGMVsdDirPathKeeper.Create(ADest, AStorePath, True);
    PILoadStore.StoreData(ADest, ACryptCtrlData);
   end;
end;

procedure GMVsdLoadTree(const ASource: IGMValueStorage;
                        const AParentTreeNode: IGMTreeable;
                        const ATreeNodeCreator: IGMCreateTreeNodeWithDataObj;
                        const AParameter: IUnknown;
                        const ACryptCtrlData: PGMCryptCtrlData;
                        const ASubDirSeparator: TGMString);
var i: Integer; subDirNames: TGMStringArray; newNode: IGMTreeable; dirKeeper: IUnknown; loadStore: IGMLoadStoreData;
begin
  if (ASubDirSeparator = '') or not GMVsdOpenDir(ASource, ASubDirSeparator) then Exit;

  GMVsdReadSubDirNames(ASource, subDirNames);
  for i:=Low(subDirNames) to High(subDirNames) do
   begin
    dirKeeper := nil; // <- restore previous dir
    dirKeeper := TGMVsdDirPathKeeper.Create(ASource);
    if not GMVsdOpenDir(ASource, subDirNames[i]) then Continue;
    newNode := ATreeNodeCreator.CreateTreeNodeWithDataObj(ASource, AParentTreeNode, AParameter);
    if GMQueryInterface(newNode, IGMLoadStoreData, loadStore) then loadStore.LoadData(ASource, ACryptCtrlData);
    //loadStore := nil; // <- release early, saves some memory
    GMVsdLoadTree(ASource, newNode, ATreeNodeCreator, AParameter, ACryptCtrlData, ASubDirSeparator);
   end;
end;

procedure GMStoreTree(const ADest: IGMValueStorage; ANode: IGMTreeable; var ANodeIdx: LongInt; const AStoreSiblings: Boolean; const ACryptCtrlData: PGMCryptCtrlData);
var dirKeeper: IUnknown; loadStore: IGMLoadStoreData;
begin
  while ANode <> nil do
   begin
    dirKeeper := nil; // <- restore previous dir
    // in TGMIniFileStorage name order must match integer order -> pad node idx with zeros
    dirKeeper := TGMVsdDirPathKeeper.Create(ADest, GMFormat('%s\%.5d', [scSubNodesDirSeparator, ANodeIdx]), True);
    if GMQueryInterface(ANode, IGMLoadStoreData, loadStore) then loadStore.StoreData(ADest, ACryptCtrlData);
    loadStore := nil; // <- release early, saves some memory
    Inc(ANodeIdx);
    GMStoreTree(ADest, ANode.FirstChild, ANodeIdx, True, ACryptCtrlData);
    if AStoreSiblings then ANode := ANode.NextSibling else ANode := nil;
   end;
end;

procedure GMVsdCopyDirValues(const ASource, ADest: IGMStringStorage);
const cDfltString = '';
var threadSync1, threadSync2: RGMCriticalSectionLock; i: LongInt; ValueNames: TGMStringArray;
begin
  if (ASource = nil) or (ADest = nil) then Exit;
  threadSync1.Lock(ASource);
  threadSync2.Lock(ADest);
  GMVsdReadValueNames(ASource, ValueNames);
  for i:=Low(ValueNames) to High(ValueNames) do
     GMStoreString(ADest, ValueNames[i], ASource.ReadString(ValueNames[i], cDfltString), cDfltString);
end;

procedure GMVsdCopyStorageContents(const ASource, ADest: IUnknown; AStartDirPath: TGMString);
var threadSync1, threadSync2: RGMCriticalSectionLock; mousePtrWait: IUnknown; srcValues, dstValues: IGMStringStorage;
  procedure CopyDirContents(const Dir: TGMString);
  var i: LongInt; SubDirNames: TGMStringArray; SrcDirKeeper, DstDirKeeper: IUnknown;
  begin
    SrcDirKeeper := TGMVsdDirPathKeeper.Create(ASource);
    if not GMVsdOpenDir(ASource, Dir, False) then Exit;
    DstDirKeeper := TGMVsdDirPathKeeper.Create(ADest, '\' + GMVsdCurrentPath(ASource), True);
    GMVsdCopyDirValues(srcValues, dstValues);
    GMVsdReadSubDirNames(ASource, SubDirNames);
    for i:=Low(SubDirNames) to High(SubDirNames) do CopyDirContents(SubDirNames[i]);
  end;
begin
  if (ASource = nil) or (ADest = nil) then Exit;
  threadSync1.Lock(ASource);
  threadSync2.Lock(ADest);

  GMCheckQueryInterface(ASource, IGMStringStorage, srcValues, {$I %CurrentRoutine%});
  GMCheckQueryInterface(ADest, IGMStringStorage, dstValues, {$I %CurrentRoutine%});

  if AStartDirPath = '' then AStartDirPath := '\' + GMVsdCurrentPath(ASource);
  mousePtrWait := TGMTempCursor.Create(vGMWaitCursor);
  CopyDirContents(AStartDirPath);
end;


{ ---------------------------- }
{ ---- ILockByte Routines ---- }
{ ---------------------------- }

function GMLockByteSize(const LockBytes: ILockBytes): Int64;
var Stat: TStatStg;
begin
  if LockBytes = nil then Result := 0 else
   begin
    GMHrCheckIntf(LockBytes.Stat(Stat, STATFLAG_NONAME), LockBytes, {$I %CurrentRoutine%});
    //if Stat.pwcsName <> nil then CoTaskMemFree(Stat.pwcsName);
    Result := Stat.cbSize;
   end;
end;

procedure GMCopyLockBytes(const ASource, ADest: ILockBytes; const AMaxBytesToCopy: LongInt = 0; const AVerfy: Boolean = True);
const CCopyBufSize = $20000;
var pBuffer: Pointer; bufSize, readCount, WriteCount, n, sourceSize, sourcePos, destPos: LongInt;
begin
  if (ASource <> nil) and (ADest <> nil) then
   begin
    sourceSize := GMLockByteSize(ASource);
    if AMaxBytesToCopy > 0 then sourceSize := Min(sourceSize, AMaxBytesToCopy);
    GMHrCheckIntf(ADest.SetSize(sourceSize), ADest, {$I %CurrentRoutine%});
    if sourceSize > 0 then
     begin
      sourcePos := 0;
      destPos := 0;
      bufSize := Min(CCopyBufSize, sourceSize);
      GetMem(pBuffer, bufSize);
      try
       while sourcePos < sourceSize do
        begin
         n := Max(0, Min(bufSize, sourceSize - sourcePos));
         GMHrCheckIntf(ASource.ReadAt(sourcePos, pBuffer, n, @readCount), ASource, {$I %CurrentRoutine%});
         if AVerfy and (readCount <> n) then raise EGMException.IntfError(GMFormat(RStrReadErrorFmt, [bufSize, readCount]), ASource, {$I %CurrentRoutine%});
         Inc(sourcePos, readCount);
         GMHrCheckIntf(ADest.WriteAt(destPos, pBuffer, readCount, @WriteCount), ADest, {$I %CurrentRoutine%});
         if AVerfy and (WriteCount <> readCount) then raise EGMException.IntfError(GMFormat(RStrWriteErrorFmt, [readCount, WriteCount]), ADest, {$I %CurrentRoutine%});
         Inc(destPos, WriteCount);
        end;
      finally
       FreeMem(pBuffer);
      end;
     end;
   end;
end;

procedure GMLockByteSafeReadAt(const ASource: ILockBytes; const AOffset: Int64; const APData: Pointer; const ACount: LongInt; const ACallingName: TGMString);
var n: LongInt;
begin
  if (ASource = nil) or (ACount <= 0) then Exit;
  GMHrCheckIntf(ASource.ReadAt(AOffset, APData, ACount, @n), ASource, ACallingName);
  if n <> ACount then raise EGMException.IntfError(GMFormat(RStrReadErrorFmt, [ACount, n]), ASource, ACallingName);
end;

procedure GMLockByteSafeWriteAt(const ADest: ILockBytes; const AOffset: Int64; const APData: Pointer; const ACount: LongInt; const ACallingName: TGMString);
var n: LongInt;
begin
  if (ADest = nil) or (ACount <= 0) then Exit;
  GMHrCheckIntf(ADest.WriteAt(AOffset, APData, ACount, @n), ADest, ACallingName);
  if n <> ACount then raise EGMException.IntfError(GMFormat(RStrWriteErrorFmt, [ACount, n]), ADest, ACallingName);
end;


{ --------------------------- }
{ ---- Progress Routines ---- }
{ --------------------------- }

procedure GMSetProgressAndCheckCanceled(const AProgresssable: IUnknown; const AProgress: Int64; var ACancel: BOOL; const ACalcProgressKind: TGMCalcProgressKind);
var OnProgress: IGMOnProgress;
begin
  if GMQueryInterface(AProgresssable, IGMOnProgress, OnProgress) then
   begin
    OnProgress.OnProgress(AProgress, ACancel, ACalcProgressKind);
    if ACancel then raise EGMAbort.Create(RStrOperationCanceled);
   end;
end;

procedure GMSetProgressMax(const AProgresssable: IUnknown; const AProgressMax: Int64);
var ProgressMax: IGMSetProgressMax;
begin
  if GMQueryInterface(AProgresssable, IGMSetProgressMax, ProgressMax) then ProgressMax.SetProgressMax(AProgressMax);
end;

procedure GMSetProgressDescription(const AProgresssable: IUnknown; const AProgressDescription: TGMString; const ATextColor: COLORREF);
var ProgressDescription: IGMSetProgressDescription;
begin
  if GMQueryInterface(AProgresssable, IGMSetProgressDescription, ProgressDescription) then
     ProgressDescription.SetProgressDescription(AProgressDescription, ATextColor);
end;


{ -------------------------- }
{ ---- Istream Routines ---- }
{ -------------------------- }

function GMIStreamSize(const AStream: IUnknown): Int64;
var Stat: TStatStg; strm: IStream;
begin
  if not GMQueryInterface(AStream, IStream, strm) then Result := 0 else
   begin
    FillByte(Stat, SizeOf(Stat), 0);
    GMHrCheckIntf(strm.Stat(Stat, STATFLAG_NONAME), strm, 'GMIStreamSize');
    if Stat.pwcsName <> nil then CoTaskMemFree(Stat.pwcsName);
    Result := Stat.cbSize;
   end;
end;

function GMIStreamPos(const AStream: IUnknown): Int64;
var strm: IStream;
begin
  if not GMQueryInterface(AStream, IStream, strm) then Result := 0 else
     GMHrCheckIntf(strm.Seek(0, STREAM_SEEK_CUR, @Result), strm, 'GMIStreamPos');
end;

function GMSetIStreamAbsPos(const AStream: IUnknown; const ANewPos: Int64; const ACallingName: TGMString): Int64;
var strm: IStream;
begin
  if not GMQueryInterface(AStream, IStream, strm) then Result := 0 else
     GMHrCheckIntf(strm.Seek(ANewPos, STREAM_SEEK_SET, @Result), strm, ACallingName);
end;

function GMIStreamRead(const ASource: ISequentialStream; const Data: Pointer; const DataSizeInBytes: LongWord): Cardinal;
begin
  if (ASource = nil) or (DataSizeInBytes = 0) then Result := 0 else ASource.Read(Data, DataSizeInBytes, Pointer(@Result));
end;

procedure GMSafeIStreamRead(const ASource: ISequentialStream; const AData: Pointer; const ADataSizeInBytes: LongWord; const ACallingName: TGMString);
var N: LongWord; //RtnName: TGMString;
begin
  if (ASource <> nil) and (ADataSizeInBytes > 0) then
   begin
    //if ACallingName <> cDfltRoutineName then RtnName := ACallingName else RtnName := {$I %CurrentRoutine%};
    N := 0;
    GMHrCheckIntf(ASource.Read(AData, ADataSizeInBytes, Pointer(@N)), ASource, ACallingName); // , RStrStreamRead + ': '
    if N <> ADataSizeInBytes then raise EGMException.IntfError(GMStringJoin(ACallingName, ': ', GMFormat(RStrReadErrorFmt, [ADataSizeInBytes, N])), ASource, ACallingName);
   end;
end;

procedure GMSafeIStreamWrite(const ADest: ISequentialStream; const AData: Pointer; const ADataSizeInBytes: LongWord; const ACallingName: TGMString);
var dwWritten: LongWord; //RtnName: TGMString;
begin
  if (ADest <> nil) and (ADataSizeInBytes > 0) then
   begin
    //if ACallingName <> cDfltRoutineName then RtnName := ACallingName else RtnName := {$I %CurrentRoutine%};
    dwWritten := 0;
    GMHrCheckIntf(ADest.Write(AData, ADataSizeInBytes, Pointer(@dwWritten)), ADest, ACallingName); // , RStrStreamWrite + ':'
    if dwWritten <> ADataSizeInBytes then raise EGMException.IntfError(GMFormat(RStrWriteErrorFmt, [ADataSizeInBytes, dwWritten]), ADest, ACallingName);
   end;
end;

function GMIStreamReadResult(const pcbOut: Pointer; const AllDone: Boolean): HResult;
const CResult: array [Boolean, Boolean] of HResult = ((S_FALSE, S_OK), (GM_E_STREAMREAD, S_OK));
begin
  Result := CResult[pcbOut = nil, AllDone];
end;

function GMIStreamWriteResult(const pcbOut: Pointer; const AllDone: Boolean): HResult;
const CResult: array [Boolean, Boolean] of HResult = ((S_FALSE, S_OK), (GM_E_STREAMWRITE, S_OK));
begin
  Result := CResult[pcbOut = nil, AllDone];
end;

procedure GMCopyIStreamTime(const ASourceStrm, ADestStrm: ISequentialStream; const AOnProgressProc: TGMOnProgressProc; const ACallBackTimeInMS: LongWord; const ACallingName: TGMString);
const cBufAlignSize = $1000;{4KB} cBufSizeMin = 1024; cBufSizeMax = $40000;{256KB} cBufSizeStart = 4096; cOneMilliSecond = 1 / (24 * 60 * 60 * 1000);
var n, bufferSize, newBufSize: LongWord; total: Int64; t1, t2, cbTime: TDateTime; buffer: RawByteString; canceled: BOOL;// PBuffer: Pointer; t1, t2: LongWord;
begin
  //
  // Adjust buffer size during copy so that OnProgress will be called every ACallBackTimeInMS
  //
  // Because Tickcount will wrap every 72 days better use a TDateTime values instead of TickCount here!
  //
  canceled := False; //PBuffer := nil;
  if (ASourceStrm = nil) or (ADestStrm = nil) or (ACallBackTimeInMS <= 0) then Exit;
   //try
    cbTime := ACallBackTimeInMS * cOneMilliSecond;
    bufferSize := 0;
    newBufSize := cBufSizeStart;
    total := 0;
    repeat
     if GMAlignedValue(bufferSize, cBufAlignSize) <> GMAlignedValue(newBufSize, cBufAlignSize) then SetLength(buffer, GMAlignedValue(newBufSize, cBufAlignSize)); //ReAllocMem(PBuffer, GMAlignedValue(newBufSize, cBufAlignSize));
     {$IFDEF DEBUG}
     //GMTrace('Stream Copy bufferSize: '+GMIntToStr(newBufSize)); // <- for testing
     {$ENDIF}
     bufferSize := newBufSize;
     t1 := now; // t1 := GetTickCount;
     n := 0;
     GMHrCheckIntf(ASourceStrm.Read(PAnsiChar(buffer), bufferSize, Pointer(@n)), ASourceStrm, ACallingName); // , RStrStreamRead + ': '
     GMSafeIStreamWrite(ADestStrm, PAnsiChar(buffer), n, ACallingName);
     //Sleep(170);  <- Test
     t2 := Now; // t2 := GetTickCount;
     //if Progress <> nil then begin Inc(total, n); Progress.OnProgress(total, Result); end;
     if Assigned(AOnProgressProc) then begin Inc(total, n); AOnProgressProc(total, canceled); end;
     //if t1 = t2 then newBufSize := cBufSizeMax else newBufSize := GMBoundedInt(Round(n * ACallBackTimeInMS / (t2 - t1)), 1, cBufSizeMax);
     if t1 = t2 then newBufSize := cBufSizeMax else newBufSize := GMBoundedInt(Round(n * (cbTime / (t2 - t1))), cBufSizeMin, cBufSizeMax);
//   Sleep(600);
    until (n < bufferSize) or canceled;
    if canceled then raise EGMAbort.Create(RStrOperationCanceled);
//  Result := not Result;
   //finally
    //FreeMem(PBuffer);
   //end;
end;

procedure GMCopyIStreamBufSize(const ASourceStrm, ADestStrm: ISequentialStream; const ACopyBufferSize: LongWord; const AOnProgressProc: TGMOnProgressProc; const ACallingName: TGMString);
var n: LongWord; total: Int64; buffer: RawByteString; canceled: BOOL; //PIBuffer: IGMMemoryBuffer;
begin
  canceled := False; total := 0;
  if (ASourceStrm = nil) or (ADestStrm = nil) or (ACopyBufferSize = 0) then Exit;
  //PIBuffer := TGMMemoryBuffer.Create(nil, ACopyBufferSize);
  SetLength(buffer, ACopyBufferSize);
  repeat
   n := 0;                 // PIBuffer.Memory
   GMHrCheckIntf(ASourceStrm.Read(PAnsiChar(buffer), ACopyBufferSize, Pointer(@n)), ASourceStrm, ACallingName); // , RStrStreamRead + ': '
   if n > 0 then GMSafeIStreamWrite(ADestStrm, PAnsiChar(buffer), n, ACallingName);
   //if Progress <> nil then begin Inc(total, n); Progress.OnProgress(total, Result); end;
   if Assigned(AOnProgressProc) then begin Inc(total, n); AOnProgressProc(total, canceled); end;
  until (n < ACopyBufferSize) or canceled;
  if canceled then raise EGMAbort.Create(RStrOperationCanceled);
//Result := not Result;
end;

procedure GMCopyIStream(const ASourceStrm, ADestStrm: ISequentialStream; const ACopyBufferSize: LongInt; const AOnProgressProc: TGMOnProgressProc; const ACallingName: TGMString);
begin
  if ACopyBufferSize = 0 then Exit; // begin Result := False; Exit; end;
  if ACopyBufferSize < 0 then
   //Result :=
   GMCopyIStreamTime(ASourceStrm, ADestStrm, AOnProgressProc, -ACopyBufferSize, ACallingName)
  else
   //Result :=
   GMCopyIStreamBufSize(ASourceStrm, ADestStrm, ACopyBufferSize, AOnProgressProc, ACallingName);
end;

function GMIStreamContentAsString(const ASourceStrm: ISequentialStream; StartPos: Int64; const ACallingName: TGMString): AnsiString;
var PosKeeper: IUnknown; seekStrm, strStream: IStream;
begin
  Result := '';
  if ASourceStrm = nil then Exit;

  if GMQueryInterface(ASourceStrm, IStream, seekStrm) then
   begin
    if StartPos < 0 then GMHrCheckObj(seekStrm.Seek(0, STREAM_SEEK_CUR, @StartPos), nil, ACallingName);
    PosKeeper := TGMIStreamPosKeeper.Create(seekStrm, StartPos);
   end;

  strStream := TGMByteStringIStream.Create('');
  GMCopyIStream(ASourceStrm, strStream);
  Result := GMGetIntfText(strStream);
//SetLength(Result, GMIStreamSize(ASourceStrm) - StartPos);
//if Length(Result) > 0 then GMSafeIStreamRead(ASourceStrm, PGMChar(Result), Length(Result), ACallingName);
end;


{ ----------------------------- }
{ ---- Activatable Objects ---- }
{ ----------------------------- }

function GMObjIsActive(const AObj: TObject; const ADefaultValue: Boolean = False): Boolean;
var getActive: RGMTypedIntf<IGMGetActive>;
begin
  if getActive.GetFrom(AObj) then Result := getActive.Intf.Active else Result := ADefaultValue;
  //if (AObj <> nil) and AObj.GetInterface(IGMGetActive, getActive) then Result := getActive.Active else Result := ADefaultValue;
  //Result := (AObj <> nil) and AObj.GetInterface(IGMGetActive, getActive) and getActive.Active;
end;

function GMIntfIsActive(const AIntf: IUnknown; const ADefaultValue: Boolean = False): Boolean;
var getActive: RGMTypedIntf<IGMGetActive>;
begin
  if getActive.QueryFrom(AIntf) then Result := getActive.Intf.Active else Result := ADefaultValue;
  //if GMQueryInterface(AIntf, IGMGetActive, getActive) then Result := getActive.Active else Result := ADefaultValue;
  //Result := (AIntf <> nil) and (AIntf.QueryInterface(IGMGetActive, getActive) = S_OK) and getActive.Active;
end;

procedure GMCheckObjIsActive(const AObj: TObject; const ACallingName: TGMString);
var PIActive: IGMGetActive; //RtnName: TGMString;
begin
  //if ACallingName then RtnName := {$I %CurrentRoutine%} else RtnName := ACallingName;
  GMCheckGetInterface(AObj, IGMGetActive, PIActive, {$I %CurrentRoutine%});
  if not PIActive.Active then raise EGMException.ObjError(GMFormat(RStrCheckActive, [ACallingName]), AObj, {$I %CurrentRoutine%});
end;

procedure GMCheckIntfIsActive(const AIntf: IUnknown; const ACallingName: TGMString);
var PIActive: IGMGetActive; //RtnName: TGMString;
begin
  //if ACallingName then RtnName := {$I %CurrentRoutine%} else RtnName := ACallingName;
  GMCheckQueryInterface(AIntf, IGMGetActive, PIActive, {$I %CurrentRoutine%});
  if not PIActive.Active then raise EGMException.IntfError(GMFormat(RStrCheckActive, [ACallingName]), AIntf, {$I %CurrentRoutine%});
end;

procedure GMCheckObjIsInActive(const AObj: TObject; const NeedInActiveName: TGMString);
var PIActive: IGMGetActive;
begin
  GMCheckGetInterface(AObj, IGMGetActive, PIActive, {$I %CurrentRoutine%});
  if PIActive.Active then raise EGMException.ObjError(GMFormat(RStrCheckInactive, [NeedInActiveName]), AObj, {$I %CurrentRoutine%});
end;

procedure GMCheckIntfIsInActive(const AIntf: IUnknown; const NeedInActiveName: TGMString);
var PIActive: IGMGetActive;
begin
  GMCheckQueryInterface(AIntf, IGMGetActive, PIActive, {$I %CurrentRoutine%});
  if PIActive.Active then raise EGMException.IntfError(GMFormat(RStrCheckInactive, [NeedInActiveName]), AIntf, {$I %CurrentRoutine%});
end;

function GMSetObjActive(const AObj: TObject; const Active: Boolean; const ACallingName: TGMString): Boolean;
//var PIActive: IGMGetSetActive;
begin
  //if (AObj <> nil) and AObj.GetInterface(IGMGetSetActive, PIActive) then PIActive.Active := Active;
  Result := GMSetIntfActive(GMObjAsIntf(AObj), Active, ACallingName);
end;

function GMSetIntfActive(const AIntf: IUnknown; const Active: Boolean; const ACallingName: TGMString): Boolean;
var PIActive: IGMGetSetActive;
begin
  if AIntf = nil then Result := False else
   begin
    GMCheckQueryInterface(AIntf, IGMGetSetActive, PIActive, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
    Result := PIActive.Active;
    PIActive.Active := Active;
   end; 
end;


{ -------------------------- }
{ ---- Interface Source ---- }
{ -------------------------- }

function GMGetInterfaceSource(const AContainer: IUnknown): IUnknown;
var getIntfSrc: IGMGetInterfaceSource;
begin
  if GMQueryInterface(AContainer, IGMGetInterfaceSource, getIntfSrc) then Result := getIntfSrc.InterfaceSource else Result := nil;
end;

procedure GMSetInterfaceSource(const AContainer, AIntfSource: IUnknown);
var setIntfSrc: IGMGetSetInterfaceSource;
begin
  if GMQueryInterface(AContainer, IGMGetSetInterfaceSource, setIntfSrc) then setIntfSrc.InterfaceSource := AIntfSource;
end;


{ ---------------------- }
{ ---- Field Values ---- }
{ ---------------------- }

function GMGetItemValue(const AContainer: IUnknown; const AItemName: TGMString): RGMUnionValue;
var fieldByName: IGMGetIntfByName; getVal: IGMGetUnionValue;
begin
  if (AItemName <> '') and GMQueryInterface(AContainer, IGMGetIntfByName, fieldByName) and
     (fieldByName.GetIntfByName(AItemName, IGMGetUnionValue, getVal) = S_OK) then Result := getVal.Value else Result := uvtNull;
end;

function GMGetItemValue(const AContainer: IUnknown; const AItemPosition: LongInt): RGMUnionValue;
var fieldByPos: IGMGetIntfByPosition; getVal: IGMGetUnionValue;
begin
  if GMQueryInterface(AContainer, IGMGetIntfByPosition, fieldByPos) and
     (fieldByPos.GetIntfByPosition(AItemPosition, IGMGetUnionValue, getVal) = S_OK) then Result := getVal.Value else Result := uvtNull;
end;


function GMCheckGetItemValue(const AContainer: IUnknown; const ItemName: TGMString; const ACallingName: TGMString = cDfltRoutineName): RGMUnionValue; overload;
var fieldByName: IGMGetIntfByName; getVal: IGMGetUnionValue;
begin
  GMCheckQueryInterface(AContainer, IGMGetIntfByName, fieldByName, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
  GMHrCheckIntf(fieldByName.GetIntfByName(ItemName, IGMGetUnionValue, getVal), AContainer, BuildCallingName(ACallingName, {$I %CurrentRoutine%}), MsgNoItemIntfPrefix(ItemName, IGMGetUnionValue));
  Result := getVal.Value;
end;

function GMCheckGetItemValue(const AContainer: IUnknown; const ItemPosition: LongInt; const ACallingName: TGMString = cDfltRoutineName): RGMUnionValue; overload;
var fieldByPos: IGMGetIntfByPosition; getVal: IGMGetUnionValue;
begin
  GMCheckQueryInterface(AContainer, IGMGetIntfByPosition, fieldByPos, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
  GMHrCheckIntf(fieldByPos.GetIntfByPosition(ItemPosition, IGMGetUnionValue, getVal), AContainer, BuildCallingName(ACallingName, {$I %CurrentRoutine%}), MsgNoItemIntfPrefix(GMIntToStr(ItemPosition), IGMGetUnionValue));
  Result := getVal.Value;
end;


procedure GMSetItemValue(const AContainer: IUnknown; const AItemName: TGMString; const AValue: RGMUnionValue);
var PIFieldByName: IGMGetIntfByName; getVal: IGMGetSetUnionValue;
begin
   if (AItemName <> '') and GMQueryInterface(AContainer, IGMGetIntfByName, PIFieldByName) and
      (PIFieldByName.GetIntfByName(AItemName, IGMGetSetUnionValue, getVal) = S_OK) then getVal.Value := AValue;
end;

procedure GMCheckSetItemValue(const AContainer: IUnknown; const ItemName: TGMString; const AValue: RGMUnionValue; const ACallingName: TGMString = cDfltRoutineName);
var PIFieldByName: IGMGetIntfByName; setVal: IGMGetSetUnionValue;
begin
  GMCheckQueryInterface(AContainer, IGMGetIntfByName, PIFieldByName, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
  GMHrCheckIntf(PIFieldByName.GetIntfByName(ItemName, IGMGetSetUnionValue, setVal), AContainer, BuildCallingName(ACallingName, {$I %CurrentRoutine%}), MsgNoItemIntfPrefix(ItemName, IGMGetSetUnionValue));
  setVal.Value := AValue;
end;


function GMGetFieldValueImpl(const AContainer: IUnknown; const AIndex: RGMUnionValue): RGMUnionValue;
var intfByPos: IGMGetIntfByPosition; intfByName: IGMGetIntfByName; val: IGMGetUnionValue; fieldPos: LongInt; fieldName: TGMString; hr: HResult;
begin
  GMCheckIntfIsActive(AContainer, 'FieldValue ' + RStrProperty);

  case AIndex.ValueType of
   uvtInt16, uvtInt32, uvtInt64, uvtDouble:
    begin
     GMCheckQueryInterface(AContainer, IGMGetIntfByPosition, intfByPos, {$I %CurrentRoutine%});
     fieldPos := AIndex;
     hr := intfByPos.GetIntfByPosition(fieldPos, IGMGetUnionValue, val);
     if not GMHrSucceeded(hr) then GMHrCheckIntf(hr, AContainer, {$I %CurrentRoutine%}, MsgNoItemIntfPrefix(GMIntToStr(fieldPos), IGMGetUnionValue));
    end;

   uvtString:
    begin
     GMCheckQueryInterface(AContainer, IGMGetIntfByName, intfByName, {$I %CurrentRoutine%});
     fieldName := AIndex;
     hr := intfByName.GetIntfByName(fieldName, IGMGetUnionValue, val);
     if not GMHrSucceeded(hr) then
      begin
       if hr = GMHResultFromWin32(ERROR_FILE_NOT_FOUND) then
         raise EGMException.IntfError(GMFormat(RStrFieldNotFound, [fieldName, GMGetIntfName(AContainer)]), AContainer, {$I %CurrentRoutine%})
       else
         GMHrCheckIntf(hr, AContainer, {$I %CurrentRoutine%}, MsgNoItemIntfPrefix(GMStringJoin(GMGetIntfName(AContainer), '.', fieldName), IGMGetUnionValue));
      end;
    end;

   else raise EGMException.IntfError(GMFormat(RStrUnsupportedIdxType, [AIndex.ValueTypeName]), AContainer, {$I %CurrentRoutine%});
  end;

  if val <> nil then Result := val.Value else Result := uvtNull;
end;

procedure GMSetFieldValueImpl(const AContainer: IUnknown; const AIndex: RGMUnionValue; const AValue: RGMUnionValue);
var intfByPos: IGMGetIntfByPosition; intfByName: IGMGetIntfByName; val: IGMGetSetUnionValue; fieldPos: LongInt; fieldName: TGMString; hr: HResult;
begin
  GMCheckIntfIsActive(AContainer, 'FieldValue ' + RStrProperty);

  case AIndex.ValueType of
   uvtInt16, uvtInt32, uvtInt64, uvtDouble:
    begin
     GMCheckQueryInterface(AContainer, IGMGetIntfByPosition, intfByPos, {$I %CurrentRoutine%});
     fieldPos := AIndex;
     hr := intfByPos.GetIntfByPosition(fieldPos, IGMGetSetUnionValue, val);
     if not GMHrSucceeded(hr) then GMHrCheckIntf(hr, AContainer, {$I %CurrentRoutine%}, MsgNoItemIntfPrefix(GMIntToStr(fieldPos), IGMGetSetUnionValue));
    end;

   uvtString:
    begin
     GMCheckQueryInterface(AContainer, IGMGetIntfByName, intfByName, {$I %CurrentRoutine%});
     fieldName := AIndex;
     hr := intfByName.GetIntfByName(fieldName, IGMGetSetUnionValue, val);
     if not GMHrSucceeded(hr) then
      begin
       if hr = GMHResultFromWin32(ERROR_FILE_NOT_FOUND) then
         raise EGMException.IntfError(GMFormat(RStrFieldNotFound, [fieldName, GMGetIntfName(AContainer)]), AContainer, {$I %CurrentRoutine%})
       else
         GMHrCheckIntf(hr, AContainer, {$I %CurrentRoutine%}, MsgNoItemIntfPrefix(GMStringJoin(GMGetIntfName(AContainer), '.', fieldName), IGMGetSetUnionValue));
      end;
    end;

   else raise EGMException.IntfError(GMFormat(RStrUnsupportedIdxType, [AIndex.ValueTypeName]), AContainer, {$I %CurrentRoutine%});
  end;

  if val <> nil then val.Value := AValue;
end;


{ --------------------- }
{ ---- DisplayText ---- }
{ --------------------- }

function GMFieldDisplayText(const FieldName: TGMString; const PIFieldByName: IGMGetIntfByName): TGMString;
var PIText: IGMGetText;
begin
  if (FieldName <> '') and (PIFieldByName <> nil) and (PIFieldByName.GetIntfByName(FieldName, IGMGetText, PIText) = S_OK) then
   Result := PIText.Text
  else
   Result := '';
end;


{ ------------------------------ }
{ ---- Executing Operations ---- }
{ ------------------------------ }

function GMCanExecOperation(const AIntf: IUnknown; const Operation: LongInt; const Parameter: IUnknown = nil): Boolean;
var canExecOp: RGMTypedIntf<IGMCanExecuteOperation>;
begin
  if AIntf = nil then Result := False
  else
   //if GMQueryInterface(AIntf, IGMCanExecuteOperation, canExecOp) then
   if canExecOp.QueryFrom(AIntf) then Result := canExecOp.Intf.CanExecuteOperation(Ord(Operation))
  else
   Result := True;
end;

procedure GMCheckExecOperation(const AObj: TObject; const Operation: LongInt; const OperationName: TGMString; const ACallingName: TGMString; const Parameter: IUnknown);
begin
  GMCheckExecOperation(GMObjAsIntf(AObj), Operation, OperationName, ACallingName, Parameter);
end;

procedure GMCheckExecOperation(const AIntf: IUnknown; const Operation: LongInt; const OperationName: TGMString; const ACallingName: TGMString; const Parameter: IUnknown);
var PIExecOp: IGMExecuteOperation; OpName: TGMString;
begin
  if not GMCanExecOperation(AIntf, Operation, Parameter) then Exit;
  GMCheckQueryInterface(AIntf, IGMExecuteOperation, PIExecOp, ACallingName);
  if not PIExecOp.ExecuteOperation(Operation, Parameter) then
   begin
    if OperationName = '' then OpName := GMFormat('(%d)', [Operation]) else OpName := GMFormat('(%d) "%s"', [Operation, OperationName]);
    raise EGMException.IntfError(GMFormat(RStrOperationExecFailed, [OpName]), AIntf, ACallingName);
   end;
end;

function GMExecuteOperation(const AObj: TObject; const Operation: LongInt; const Parameter: IUnknown = nil): Boolean;
begin
  Result := GMExecuteOperation(GMObjAsIntf(AObj), Operation, Parameter)
end;

function GMExecuteOperation(const AIntf: IUnknown; const Operation: LongInt; const Parameter: IUnknown = nil): Boolean;
var PIExecOp: IGMExecuteOperation; 
begin
  Result := GMCanExecOperation(AIntf, Operation, Parameter);
  if Result and GMQueryInterface(AIntf, IGMExecuteOperation, PIExecOp) then
     Result := PIExecOp.ExecuteOperation(Ord(Operation), Parameter);
end;


{ ---------------------------------- }
{ ---- Asking Boolean Questions ---- }
{ ---------------------------------- }

function GMBooleanAskResult(const AValue: Boolean): LongInt;
const cResultValues: array [Boolean] of LongInt = (Ord(barFalse), Ord(barTrue));
begin
  Result := cResultValues[AValue];
end;

function GMAskBoolean(const AObj: TObject; const AValueId: LongInt; const ADefaultResult: Boolean): Boolean;
begin
  Result := GMAskBoolean(GMObjAsIntf(AObj), AValueId, ADefaultResult)
end;

function GMAskBoolean(const AIntf: IUnknown; const AValueId: LongInt; const ADefaultResult: Boolean): Boolean;
var PIAskBoolean: IGMAskBoolean;
begin
  if not GMQueryInterface(AIntf, IGMAskBoolean, PIAskBoolean) then Result := ADefaultResult else
   case PIAskBoolean.AskBoolean(AValueId) of
    Ord(barTrue): Result := True;
    Ord(barFalse): Result := False;
    else Result := ADefaultResult;
   end;
end;

function GMAskUnkBoolean(const AIntf: IUnknown; const AValueId: LongInt): TGMBoolAskResult;
var PIAskBoolean: IGMAskBoolean;
begin
  if not GMQueryInterface(AIntf, IGMAskBoolean, PIAskBoolean) then Result := barUnknown else
   Result := TGMBoolAskResult(GMBoundedInt(PIAskBoolean.AskBoolean(AValueId), Ord(Low(TGMBoolAskResult)), Ord(High(TGMBoolAskResult))));
end;

function GMCheckAskBoolean(const AObj: TObject; const AValueId: LongInt; const ACallingName: TGMString): Boolean;
begin
  Result := GMCheckAskBoolean(GMObjAsIntf(AObj), AValueId, ACallingName);
end;

function GMCheckAskBoolean(const AIntf: IUnknown; const AValueId: LongInt; const ACallingName: TGMString): Boolean;
var PIAskBoolean: IGMAskBoolean; Answer: LongInt;
begin
  GMCheckQueryInterface(AIntf, IGMAskBoolean, PIAskBoolean, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
  Answer := PIAskBoolean.AskBoolean(AValueId);
  case Answer of
   Ord(barTrue): Result := True;
   Ord(barFalse): Result := False;
   else raise EGMException.IntfError(GMFormat(RStrUnsupoortedBoolQuestion, [AValueId]), AIntf, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
  end;
end;


{ ----------------------------- }
{ ---- Asking for Integers ---- }
{ ----------------------------- }

function GMAskInteger(const AObj: TObject; const AValueId: LongInt; const ADefaultValue: LongInt): LongInt; overload;
//var PIUnk: IUnknown;
begin
//  if (AObj <> nil) and (AObj.GetInterface(IUnknown, PIUnk)) then
//   Result := GMAskInteger(PIUnk, AValueId, ADefaultValue)
//  else
//   Result := ADefaultValue;
  Result := GMAskInteger(GMObjAsIntf(AObj), AValueId, ADefaultValue);
end;

function GMAskInteger(const AIntf: IUnknown; const AValueId: LongInt; const ADefaultValue: LongInt): LongInt; overload;
var PIAskInteger: IGMAskInteger;
begin
  if not GMQueryInterface(AIntf, IGMAskInteger, PIAskInteger) then Result := ADefaultValue else
   begin
    Result := PIAskInteger.AskInteger(AValueId);
    if Result = cInvalidIntValue then Result := ADefaultValue;
   end;
end;

function GMCheckAskInteger(const AObj: TObject; const AValueId: LongInt; const ACallingName: TGMString): LongInt;
begin
  Result := GMCheckAskInteger(GMObjAsIntf(AObj), AValueId, ACallingName);
end;

function GMCheckAskInteger(const AIntf: IUnknown; const AValueId: LongInt; const ACallingName: TGMString): LongInt;
var PIAskInteger: IGMAskInteger;
begin
  GMcheckQueryInterface(AIntf, IGMAskInteger, PIAskInteger, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
  Result := PIAskInteger.AskInteger(AValueId);
  if Result = CInvalidIntValue then raise EGMException.IntfError(GMFormat(RStrUnsupportedValueId, [AValueId]), AIntf, BuildCallingName(ACallingName, {$I %CurrentRoutine%}));
end;


{ ------------------------- }
{ ---- Position Helper ---- }
{ ------------------------- }

function GMGetIntfPosition(const AIntf: IUnknown; const ADefaultPos: PtrInt): PtrInt;
var getPos: RGMTypedIntf<IGMGetPosition>;
begin
  if getPos.QueryFrom(AIntf) then Result := getPos.Intf.Position else Result := ADefaultPos;
end;

procedure GMSetIntfPosition(const AIntf: IUnknown; const APosition: PtrInt);
var setPos: RGMTypedIntf<IGMGetSetPosition>;
begin
  if setPos.QueryFrom(AIntf) then setPos.Intf.Position := APosition;
end;

procedure GMMovePosition(const AIntf: IUnknown; const ADelta: PtrInt);
var setPos: RGMTypedIntf<IGMGetSetPosition>;
begin
  if setPos.QueryFrom(AIntf) then setPos.Intf.Position := setPos.Intf.Position + ADelta;
end;


{ ----------------------- }
{ ---- Empty Objects ---- }
{ ----------------------- }

function GMObjIsEmpty(const AObj: TObject; const ADefaultResult: Boolean): Boolean;
begin
  Result := GMIntfIsEmpty(GMObjAsIntf(AObj), ADefaultResult);
end;

function GMIntfIsEmpty(const AIntf: IUnknown; const ADefaultResult: Boolean): Boolean;
var PICount: IGMGetCount;
begin
  if GMQueryInterface(AIntf, IGMGetCount, PICount) then Result := PICount.Count <= 0 else Result := ADefaultResult;
end;


{ ---------------------------- }
{ ---- Cursor Move Helper ---- }
{ ---------------------------- }

function GMIsValidCursorMove(const AIntf: IUnknown; const Move: TGMCursorMove): Boolean;
var uniCur: RGMTypedIntf<IGMUnidirectionalCursor>;
    bidiCursor: RGMTypedIntf<IGMBidirectionalCursor>;
    firstLastCursor: RGMTypedIntf<IGMCursorFirstLast>;
begin
  if not GMIntfIsActive(AIntf) then Result := False else
   Case Move of
    cmNext: Result := uniCur.QueryFrom(AIntf) and not uniCur.Intf.EOF and not GMIntfIsEmpty(uniCur.Intf);
    cmPrior: Result := bidiCursor.QueryFrom(AIntf) and not bidiCursor.Intf.BOF and not GMIntfIsEmpty(bidiCursor.Intf);
    cmFirst: Result := firstLastCursor.QueryFrom(AIntf) and not firstLastCursor.Intf.BOF and not GMIntfIsEmpty(firstLastCursor.Intf);
    cmLast: Result := firstLastCursor.QueryFrom(AIntf) and not firstLastCursor.Intf.EOF and not GMIntfIsEmpty(firstLastCursor.Intf);
    else Result := False;
   end;
end;

procedure GMMoveCursor(const AIntf: IUnknown; const Move: TGMCursorMove);
var uniCur: RGMTypedIntf<IGMUnidirectionalCursor>;
    bidiCursor: RGMTypedIntf<IGMBidirectionalCursor>;
    firstLastCursor: RGMTypedIntf<IGMCursorFirstLast>;
begin
  if AIntf <> nil then
   case Move of
    cmNext:  if uniCur.QueryFrom(AIntf) and not uniCur.Intf.EOF then uniCur.Intf.MoveToNext;
    cmPrior: if bidiCursor.QueryFrom(AIntf) and not bidiCursor.Intf.BOF then bidiCursor.Intf.MoveToPrevious;
    cmFirst: if firstLastCursor.QueryFrom(AIntf) and not firstLastCursor.Intf.BOF then firstLastCursor.Intf.MoveToFirst;
    cmLast:  if firstLastCursor.QueryFrom(AIntf) and not firstLastCursor.Intf.EOF then firstLastCursor.Intf.MoveToLast;
   end;
end;

procedure GMSafeCursorMove(const AIntf: IUnknown; const Move: TGMCursorMove);
begin
  if GMIsValidCursorMove(AIntf, Move) then GMMoveCursor(AIntf, Move);
end;


{ --------------------- }
{ ---- OGMMemTable ---- }
{ --------------------- }

class operator RGMNameRec.Initialize(var ANameRec: RGMNameRec);
begin
  //ANameRec.Name := '';
  ANameRec.RefCount := 0;
  ANameRec.VTable := @cNameRecVTable;
end;

class operator RGMNameRec.Implicit(const ANameValRec: RGMNameRec): IUnknown;
begin
  Result := IUnknown(@ANameValRec);
end;

//class function RGMNameRec.QueryIntferface(AInstance: PGMNameRec; const AIID: TGUID; out AObj): HResult; stdcall;
//begin
//  if (AIID = IUnknown) or (AIID = IGMGetName) then begin Pointer(AObj) := AInstance; Result := S_OK; end
//  else Result := E_NOINTERFACE;
//end;

//class function RGMNameRec.Addref(AInstance: PGMNameRec): Integer; stdcall;
//begin
//  Result := InterlockedIncrement(AInstance.RefCount);
//end;

//class function RGMNameRec.Release(AInstance: PGMNameRec): Integer; stdcall;
//begin
//  Result := InterlockedDecrement(AInstance.RefCount);
//  //if result = 0 then Dispose(AInstance);
//end;

function RGMNameRec.QueryIntferface(const AIID: TGUID; out AObj): HResult; stdcall;
begin
  if (AIID = IUnknown) or (AIID = IGMGetName) then begin Pointer(AObj) := @Self; Result := S_OK; end
  else Result := E_NOINTERFACE;
end;

function RGMNameRec.Addref: Integer; stdcall;
begin
  Result := InterlockedIncrement(RefCount);
end;

function RGMNameRec.Release: Integer; stdcall;
begin
  Result := InterlockedDecrement(RefCount);
  // if RefLifeTime and (Result = 0) then Dispose(@Self);
end;

function RGMNameRec.GetName: TGMString; stdcall;
begin
  Result := Name;
end;


{ ---------------------- }
{ ---- RGMTypedIntf ---- }
{ ---------------------- }

//class operator RGMTypedIntf<TIntf>.Implicit(const ATypedIntf: RGMTypedIntf<TIntf>): Pointer;
//begin
//  Result := Pointer(ATypedIntf.Intf);
//end;

//class operator RGMTypedIntf<TIntf>.Implicit(const ATypedIntf: RGMTypedIntf<TIntf>): TIntf;
//begin
//  Result := ATypedIntf.Intf;
//end;

//class operator RGMTypedIntf<TIntf>.Implicit(AValue: Pointer): RGMTypedIntf<TIntf>;
//begin
//
//end;

//class operator RGMTypedIntf<TIntf>.Implicit(AValue: IUnknown): RGMTypedIntf<TIntf>;
//begin
//  if AValue = nil then Result.Intf := nil else
//   GMCheckQueryInterface(AValue, TIntf, Result.Intf, 'RGMTypedIntf<TIntf>.Implicit');
//end;

class operator RGMTypedIntf<TIntf>.:=(const ATypedIntf: RGMTypedIntf<TIntf>): TIntf;
begin
  Result := ATypedIntf.Intf;
end;

class operator RGMTypedIntf<TIntf>.:=(AValue: IUnknown): RGMTypedIntf<TIntf>;
begin
  if AValue = nil then Result.Intf := nil else
    GMCheckQueryInterface(AValue, TIntf, Result.Intf, 'RGMTypedIntf<'+GMGuidToString(TIntf)+'>.Assign');
end;

class operator RGMTypedIntf<TIntf>.:=(AObj: TObject): RGMTypedIntf<TIntf>;
begin
  if AObj = nil then Result.Intf := nil else
   GMCheckGetInterface(AObj, TIntf, Result.Intf, 'RGMTypedIntf<'+GMGuidToString(TIntf)+'>.Assign');
end;

function RGMTypedIntf<TIntf>.QueryFrom(AIntf: IUnknown; ACheckResult: Boolean; const ACallingName: TGMString): Boolean;
begin
  if ACheckResult then
   begin GMCheckQueryInterface(AIntf, TIntf, Intf, BuildCallingName(ACallingName, {$I %CurrentRoutine%})); Result := True; end
  else
    Result := GMQueryInterface(AIntf, TIntf, Intf);
end;

function RGMTypedIntf<TIntf>.GetFrom(AObj: TObject; ACheckResult: Boolean; const ACallingName: TGMString): Boolean;
begin
  if ACheckResult then
   begin GMCheckGetInterface(AObj, TIntf, Intf, BuildCallingName(ACallingName, {$I %CurrentRoutine%})); Result := True; end
  else
    Result := GMGetInterface(AObj, TIntf, Intf);
end;

//function RGMTypedIntf<TIntf>.GetWeakFrom(AObj: TObject; ACheckResult: Boolean): Boolean;
//begin
//  if not ACheckResult then Result := GMGetWeakInterface(AObj, TIntf, Intf) else
//   begin
//    Result := GMFormat(RStrIntfNotSupported, ['Name', GMGuidToString(TIntf)]);
//    //GMCheckPointerAssigned(AObj, RStrTheObject, AObj, {$I %CurrentRoutine%});
//    //if not AObj.GetInterfaceWeak(TIntf, Intf) then raise EGMException.ObjError(MsgIntfNotSupported(RStrTheObject, TIntf), AObj, {$I %CurrentRoutine%});
//    Result := True;
//   end;
//end;

function RGMTypedIntf<TIntf>.Call(ACaller: TObject): TIntf;
begin
  //Assert(Intf <> nil, 'RGMTypedIntf<TIntf>.Intf <> nil');
  if Intf = nil then EGMException.ObjError(GMFormat(srCannotCallNilIntf, [GMGuidToString(TIntf)]), ACaller, 'RGMTypedIntf<'+GMGuidToString(TIntf)+'>.Call');
  Result := Intf;
end;



{ -------------------------------- }
{ ---- RGMCriticalSectionLock ---- }
{ -------------------------------- }

class operator RGMCriticalSectionLock.Initialize(var ACriticalSectionLock: RGMCriticalSectionLock);
begin
  ACriticalSectionLock := Default(RGMCriticalSectionLock);
end;

class operator RGMCriticalSectionLock.Finalize(var ACriticalSectionLock: RGMCriticalSectionLock);
begin
  ACriticalSectionLock.UnlockAll;
end;

procedure RGMCriticalSectionLock.Lock(const ACriticalSection: IGMCriticalSection);
begin
  CriticalSection := ACriticalSection;
  LockAgain;
end;

procedure RGMCriticalSectionLock.Lock(const ACriticalSection: IUnknown);
var cs: IGMCriticalSection;
begin
  if GMQueryInterface(ACriticalSection, IGMCriticalSection, cs) then Lock(cs);
end;

procedure RGMCriticalSectionLock.Lock(const ACriticalSection: TObject);
var cs: IGMCriticalSection;
begin
  if GMGetInterface(ACriticalSection, IGMCriticalSection, cs) then Lock(cs);
end;

procedure RGMCriticalSectionLock.LockAgain;
begin
  if CriticalSection <> nil then begin CriticalSection.EnterCriticalSection; Inc(LockCount); end;
end;

procedure RGMCriticalSectionLock.Unlock;
begin
  if (CriticalSection <> nil) and (LockCount > 0) then begin CriticalSection.LeaveCriticalSection; Dec(LockCount); end;
end;

procedure RGMCriticalSectionLock.UnlockAll;
begin
  while LockCount > 0 do Unlock;
end;

function RGMCriticalSectionLock.GetLockCount: Int64;
begin
  Result := LockCount;
end;


{ ---------------------------- }
{ ---- TGMCriticalSection ---- }
{ ---------------------------- }

constructor TGMCriticalSection.Create(const ARefLifeTime: Boolean);
begin
  inherited Create(ARefLifeTime);
  {$IFNDEF JEDIAPI}windows{$ELSE}jwaWinBase{$ENDIF}.InitializeCriticalSection(FCriticalSection);
end;

destructor TGMCriticalSection.Destroy;
begin
  {$IFNDEF JEDIAPI}windows{$ELSE}jwaWinBase{$ENDIF}.DeleteCriticalSection(FCriticalSection);
  inherited Destroy;
end;

procedure TGMCriticalSection.EnterCriticalSection;
begin
  {$IFNDEF JEDIAPI}windows{$ELSE}jwaWinBase{$ENDIF}.EnterCriticalSection(FCriticalSection);
end;

procedure TGMCriticalSection.LeaveCriticalSection;
begin
  {$IFNDEF JEDIAPI}windows{$ELSE}jwaWinBase{$ENDIF}.LeaveCriticalSection(FCriticalSection);
end;

function TGMCriticalSection.TryEnterCriticalSection: Boolean;
begin
  Result := {$IFNDEF JEDIAPI}windows{$ELSE}jwaWinBase{$ENDIF}.TryEnterCriticalSection(FCriticalSection);
end;


{ -------------------------------- }
{ ---- TGMCriticalSectionLock ---- }
{ -------------------------------- }

constructor TGMCriticalSectionLock.Create(const ACriticalSection: IUnknown; const ARefLifeTime: Boolean);
begin
  inherited Create(ARefLifeTime);
  if ACriticalSection = nil then Exit; // <- allow nil!
  // But force ACriticalSection to support IGMCriticalSection interface if not nil!
  GMCheckQueryInterface(ACriticalSection, IGMCriticalSection, FCriticalSection, {$I %CurrentRoutine%});
  if FCriticalSection <> nil then FCriticalSection.EnterCriticalSection;
end;

destructor TGMCriticalSectionLock.Destroy;
begin
  if FCriticalSection <> nil then FCriticalSection.LeaveCriticalSection;
  inherited Destroy;
end;


{ --------------------------- }
{ ---- TGMCOMInitializer ---- }
{ --------------------------- }

constructor TGMCOMInitializer.Create(const ACoInitFlags: DWORD; const AHrCheck, ARefLifeTime: Boolean);
var hr: HResult;
begin
  inherited Create(ARefLifeTime);
  hr := CoInitializeEx(nil, ACoInitFlags);
  if AHrCheck then GMHrCheckObj(hr, Self, 'CoInitializeEx');
  FInitialized := GMHrSucceeded(hr);
end;

destructor TGMCOMInitializer.Destroy;
begin
  if FInitialized then CoUninitialize;
  inherited Destroy;
end;


{ --------------------------------- }
{ ---- TGMNotificationDisabler ---- }
{ --------------------------------- }

constructor TGMNotificationDisabler.Create(const ANotifier: IUnknown; const ANotificationOnReEnable: LongInt; const ANotificationOnFirstDisable: LongInt; const ARefLifeTime: Boolean);
begin
  inherited Create(ARefLifeTime);
  FNotificationOnReEnable := ANotificationOnReEnable;
  // follow "if possible" semantics here
  if not GMQueryInterface(ANotifier, IGMEnableNotifications, FNotifier) then Exit;
  //GMCheckQueryInterface(ANotifier, IGMEnableNotifications, FNotifier, {$I %CurrentRoutine%});
  FNotifier.DisableNotifications(ANotificationOnFirstDisable);
end;

destructor TGMNotificationDisabler.Destroy;
begin
  if FNotifier <> nil then try FNotifier.EnableNotifications(FNotificationOnReEnable); except end; // <- never raise exceptions in destructors
  inherited Destroy;
end;


{ --------------------------- }
{ ---- TGMPositionKeeper ---- }
{ --------------------------- }

constructor TGMPositionKeeper.Create(const AObjWithPosition: IUnknown; const ANewposition: LongInt; const ARefLifeTime: Boolean);
begin
  inherited Create(ARefLifeTime);
  // follow "if possible" semantics here
  if not GMQueryInterface(AObjWithPosition, IGMGetSetPosition, FObjWithPosition) then Exit;
  //GMCheckQueryInterface(AObjWithPosition, IGMGetSetPosition, FObjWithPosition, {$I %CurrentRoutine%});
  FPosition := FObjWithPosition.Position;
  if ANewposition <> -1 then FObjWithPosition.Position := ANewposition;
end;

destructor TGMPositionKeeper.Destroy;
//var PICount: IGMGetCount;
begin
  if FObjWithPosition <> nil then try FObjWithPosition.Position := FPosition; except end; // <- never raise exceptions in destructors
   //if FObjWithPosition.QueryInterface(IGMGetCount, PICount) = S_OK then FObjWithPosition.Position := Min(FPosition, PICount.Count) else
  inherited Destroy;
end;


{ ------------------------ }
{ ---- TGMStateKeeper ---- }
{ ------------------------ }

constructor TGMStateKeeper.Create(const AObjWithState: IUnknown; const ARefLifeTime: Boolean);
begin
  inherited Create(ARefLifeTime);
  // follow "if possible" semantics here
  if not GMQueryInterface(AObjWithState, IGMSaveRestoreState, FObjWithState) then Exit;
  //GMCheckQueryInterface(AObjWithState, IGMSaveRestoreState, FObjWithState, {$I %CurrentRoutine%});
  FState := FObjWithState.CaptureState;
end;

destructor TGMStateKeeper.Destroy;
begin
  // never raise exceptions in destructors
  if FObjWithState <> nil then try FObjWithState.RestoreState(FState); except end; // <- never raise exceptions in destructors
  inherited Destroy;
end;


{ ----------------------------- }
{ ---- TGMQuietStateKeeper ---- }
{ ----------------------------- }

constructor TGMQuietStateKeeper.Create(const AObjWithState: IUnknown; const ANotificationOnReEnable: LongInt; const ANotificationOnFirstDisable: LongInt; const ARefLifeTime: Boolean);
begin
  inherited Create(ARefLifeTime);
  FNotificationDisabler := TGMNotificationDisabler.Create(AObjWithState, ANotificationOnReEnable, ANotificationOnFirstDisable);
  FStateKeeper := TGMStateKeeper.Create(AObjWithState);
end;


{ ------------------------- }
{ ---- TGMActiveKeeper ---- }
{ ------------------------- }

constructor TGMActiveKeeper.Create(const AActivatableObj: IUnknown; const ANewActive: Boolean; const ARefLifeTime: Boolean);
begin
  inherited Create(ARefLifeTime);
  GMQueryInterface(AActivatableObj, IGMGetSetActive, FActivatableObj);
  FWasActive := GMSetIntfActive(FActivatableObj, ANewActive, {$I %CurrentRoutine%});
end;

destructor TGMActiveKeeper.Destroy;
begin
  try GMSetIntfActive(FActivatableObj, FWasActive, {$I %CurrentRoutine%}); except end; // <- never raise exceptions in destructors
  inherited Destroy;
end;


{ ----------------------------- }
{ ---- TGMVsdDirPathKeeper ---- }
{ ----------------------------- }

constructor TGMVsdDirPathKeeper.Create(const AValStorageDir: IUnknown; const ANewDirPath: TGMString; const ACreateIfNotExists: Boolean; const ARefLifeTime: Boolean);
begin
  inherited Create(ARefLifeTime);
  if AValStorageDir = nil then Exit;
  // force AValStorageDir to support IGMValueStorageDirectory interface
  GMCheckQueryInterface(AValStorageDir, IGMValueStorageDirectory, FValStorageDir, {$I %CurrentRoutine%});
  FOldDirPath := FValStorageDir.CurrentPath;
  if Length(ANewDirPath) > 0 then GMVsdOpenDir(FValStorageDir, ANewDirPath, ACreateIfNotExists);
end;

destructor TGMVsdDirPathKeeper.Destroy;
begin
  if FValStorageDir <> nil then GMVsdOpenAbsDir(FValStorageDir, FOldDirPath, False);
  inherited Destroy;
end;


{ ----------------------------- }
{ ---- TGMIStreamPosKeeper ---- }
{ ----------------------------- }

constructor TGMIStreamPosKeeper.Create(const AStream: IUnknown; AStartPos: Int64; const ARefLifeTime: Boolean);
var strm: IStream;
begin
  inherited Create(ARefLifeTime);
//if AStream = nil then Exit;
  if not GMQueryInterface(AStream, IStream, strm) then Exit;
  FStream := strm;
  if AStartPos >= 0 then
   GMHrCheckObj(FStream.Seek(AStartPos, STREAM_SEEK_SET, @FOldPos), Self, {$I %CurrentRoutine%})
  else
   GMHrCheckObj(FStream.Seek(0, STREAM_SEEK_CUR, @FOldPos), Self, {$I %CurrentRoutine%});
end;

destructor TGMIStreamPosKeeper.Destroy;
begin
  if FStream <> nil then FStream.Seek(FOldPos, STREAM_SEEK_SET, @FOldPos); // <- no exceptions in destructors!
  inherited Destroy;
end;


{ -------------------------- }
{ ---- TGMPaintDisabler ---- }
{ -------------------------- }

{constructor TGMPaintDisabler.Create(const APaintObj: IUnknown; const ARefLifeTime: Boolean);
begin
  inherited Create(ARefLifeTime);
  if APaintObj <> nil then APaintObj.QueryInterface(IGMEnableDisablePaint, FPaintObj);
  if FPaintObj <> nil then FPaintObj.DisablePaint;
end;

destructor TGMPaintDisabler.Destroy;
begin
  if FPaintObj <> nil then FPaintObj.EnablePaint;
  inherited Destroy;
end;}


{ -------------------------- }
{ ---- TGMRefCountedObj ---- }
{ -------------------------- }

constructor TGMRefCountedObj.Create(const ARefLifeTime: Boolean);
begin
  inherited Create;
  FRefLifeTime := ARefLifeTime;
  FRefCount := 1; // <- artificial RefCount during construction, avoiding immediate destruction
                  //    when local interface variables to this instance are used by derived constructors.
end;

procedure TGMRefCountedObj.AfterConstruction;
begin
  Dec(FRefCount);
  inherited AfterConstruction;
end;

procedure TGMRefCountedObj.BeforeDestruction;
begin
  inherited BeforeDestruction;
  InterlockedIncrement(FRefCount); // <- put an artificial RefCount during destruction to avoid reentering destructor if temporary references are used during destruction
end;

destructor TGMRefCountedObj.Destroy;
begin
  inherited Destroy;
  if Assigned(vfGMCheckRefCountOnDestroy) then vfGMCheckRefCountOnDestroy(RefCount-1, Self);
end;

procedure TGMRefCountedObj.OnFinalRelease;
begin
  Free;
end;

function TGMRefCountedObj.CreateCopyQI(const AIID: TGUID; out AIntf): HResult; stdcall;
var unkCopy: IUnknown; assignFrom: IGMAssignFromObj;
begin
  unkCopy := TGMRefCountedObjClass(ClassType).Create(True);
  Result := unkCopy.QueryInterface(AIID, AIntf);
  if (Result = S_OK) and (unkCopy.QueryInterface(IGMAssignFromObj, assignFrom) = S_OK) then assignFrom.AssignFromObj(Self);
end;


{ ---- IGMGetObjInfo ---- }

function TGMRefCountedObj.GetClassName: TGMString;
//var RetVal: TGMString;
begin
  //RetVal := ClassName;
  //Result := PGMChar(RetVal);
  Result := ClassName;
end;

function TGMRefCountedObj.GetClassType: TClass;
begin
  Result := ClassType;
end;

function TGMRefCountedObj.GetInstance: TObject;
begin
  Result := Self;
end;

function TGMRefCountedObj.GetTypeInfo: PTypeInfo;
begin
  Result := ClassInfo;
end;

function TGMRefCountedObj.GetRefCount: LongInt;
begin
  Result := FRefCount;
end;


{ ---- IUnknown ---- }

function TGMRefCountedObj.QueryInterface(constref AIID: TGUID; out AIntf): HResult; stdcall;
begin
  if GetInterface(AIID, AIntf) then Result := S_OK else Result := E_NOINTERFACE;
  //GMTraceQueryInterface(Self, AIID, Result);
end;

function TGMRefCountedObj._AddRef: LongInt; stdcall;
begin
  Result := InterlockedIncrement(FRefCount);
end;

function TGMRefCountedObj._Release: LongInt; stdcall;
begin
  Result := InterlockedDecrement(FRefCount);
  if (Result = 0) and RefLifeTime then OnFinalRelease;
end;


{ ---------------------------- }
{ ---- TGMAggregatableObj ---- }
{ ---------------------------- }

constructor TGMAggregatableObj.Create(const AOwner: IUnknown; const ARefLifeTime: Boolean);
begin
  Create(ARefLifeTime);
  FOwner := Pointer(AOwner);
  Assert((AOwner = nil) or not ARefLifeTime, '(AOwner = nil) or not ARefLifeTime');
end;

function TGMAggregatableObj.QueryInterface(constref AIID: TGUID; out AIntf): HResult; stdcall;
begin
  if FOwner <> nil then Result := IUnknown(FOwner).QueryInterface(AIID, AIntf) else Result := E_NOINTERFACE;
  if Result <> S_OK then Result := inherited QueryInterface(AIID, AIntf);
end;

function TGMAggregatableObj._AddRef: LongInt; stdcall;
begin
  if FOwner <> nil then Result := IUnknown(FOwner)._AddRef else Result := inherited _AddRef;
end;

function TGMAggregatableObj._Release: LongInt; stdcall;
begin
  if FOwner <> nil then Result := IUnknown(FOwner)._Release else Result := inherited _Release;
end;

function TGMAggregatableObj.GetOwner: IUnknown;
begin
  Result := IUnknown(FOwner);
end;

function TGMAggregatableObj.GetOwnerObj: TObject;
begin
  Result := GMObjFromIntf(IUnknown(FOwner));
end;


{ ------------------------------ }
{ ---- TGMRefLifePersistent ---- }
{ ------------------------------ }

{$IFDEF DELPHIVCL}
constructor TGMRefLifePersistent.Create(const ARefLifeTime: Boolean);
begin
  inherited Create;
  FRefLifeTime := ARefLifeTime;
end;

procedure TGMRefLifePersistent.AfterConstruction;
begin
  inherited AfterConstruction;
  FConstructed := True;
end;

destructor TGMRefLifePersistent.Destroy;
begin
  inherited Destroy;
  if Assigned(vfGMCheckRefCountOnDestroy) then vfGMCheckRefCountOnDestroy(RefCount, Self);
end;

procedure TGMRefLifePersistent.OnFinalRelease;
begin
  Free;
end;

{ ---- IGMGetObjInfo ---- }

function TGMRefLifePersistent.GetClassName: TGMString;
begin
  Result := ClassName;
end;

function TGMRefLifePersistent.GetClassType: TClass;
begin
  Result := ClassType;
end;

function TGMRefLifePersistent.GetInstance: TObject;
begin
  Result := Self;
end;

function TGMRefLifePersistent.GetTypeInfo: PTypeInfo;
begin
  Result := ClassInfo;
end;

{ ---- IUnknown ---- }

function TGMRefLifePersistent.QueryInterface(const AIID: TGUID; out AIntf): HResult;
begin
  if GetInterface(AIID, AIntf) then Result := S_OK else Result := E_NOINTERFACE;
  //GMTraceQueryInterface(Self, AIID, Result);
end;

function TGMRefLifePersistent._AddRef: LongInt;
begin
  Result := InterlockedIncrement(FRefCount);
end;

function TGMRefLifePersistent._Release: LongInt;
begin
  Result := InterlockedDecrement(FRefCount);
  if (Result = 0) and RefLifeTime and FConstructed then OnFinalRelease;
end;
{$ENDIF}


{ ----------------------------- }
{ ---- TGMRefLifeComponent ---- }
{ ----------------------------- }

{$IFDEF DELPHIVCL}
constructor TGMRefLifeComponent.CreateIntf;
begin
  RefLifeTime := True;
  Create(nil); // <- Important: call virtual constructor here, to let derived classes initialize their members
end;

procedure TGMRefLifeComponent.AfterConstruction;
begin
  inherited AfterConstruction;
  FConstructed := True;
end;

destructor TGMRefLifeComponent.Destroy;
begin
  inherited Destroy;
  {if RefLifeTime then} if Assigned(vfGMCheckRefCountOnDestroy) then vfGMCheckRefCountOnDestroy(RefCount, Self);
end;

function TGMRefLifeComponent.CopyCreateClass: TGMRefLifeComponentClass;
begin
  Result := TGMRefLifeComponentClass(ClassType);
end;

procedure TGMRefLifeComponent.OnFinalRelease;
begin
  Free;
end;

{ ---- IGMGetObjInfo ---- }

function TGMRefLifeComponent.GetClassName: TGMString;
begin
  Result := ClassName;
end;

function TGMRefLifeComponent.GetClassType: TClass;
begin
  Result := ClassType;
end;

function TGMRefLifeComponent.GetInstance: TObject;
begin
  Result := Self;
end;

function TGMRefLifeComponent.GetTypeInfo: PTypeInfo;
begin
  Result := ClassInfo;
end;

{ ---- IGMGetName ---- }

function TGMRefLifeComponent.GetName: TGMString;
begin
  Result := Name;
end;

{ ---- IGMAssignByObj ---- }

procedure TGMRefLifeComponent.AssignFromObj(const ASource: TObject);
begin
  if ASource is TPersistent then Assign(TPersistent(ASource));
end;

procedure TGMRefLifeComponent.AssignToObj(const ADest: TObject);
begin
  if ADest is TPersistent then TPersistent(ADest).Assign(Self);
end;

{ ---- IGMCreateCopyQI ---- }

function TGMRefLifeComponent.CreateCopyQI(const AIID: TGUID; out AIntf): HResult;
var PIUnknown: IUnknown; PIAssign: IGMAssignFromObj;
begin
  PIUnknown := CopyCreateClass.CreateIntf;
  Result := PIUnknown.QueryInterface(AIID, AIntf);
  if (Result = S_OK) and (PIUnknown.QueryInterface(IGMAssignFromObj, PIAssign) = S_OK) then PIAssign.AssignFromObj(Self);
end;

{ ---- IUnknown ---- }

function TGMRefLifeComponent.QueryInterface(const AIID: TGUID; out AIntf): HResult;
begin
  if GetInterface(AIID, AIntf) then Result := S_OK else Result := E_NOINTERFACE;
  //GMTraceQueryInterface(Self, AIID, Result);
end;

function TGMRefLifeComponent._AddRef: LongInt;
begin
  Result := InterlockedIncrement(FRefCount);
end;

function TGMRefLifeComponent._Release: LongInt;
begin
  Result := InterlockedDecrement(FRefCount);
  if (Result = 0) and RefLifeTime and FConstructed then OnFinalRelease;
end;
{$ENDIF}


end.