Skip to content

Instantly share code, notes, and snippets.

@sancarn
Created June 26, 2020 23:08
Show Gist options
  • Save sancarn/bb9702558a6a94230aa9e04acffbb047 to your computer and use it in GitHub Desktop.
Save sancarn/bb9702558a6a94230aa9e04acffbb047 to your computer and use it in GitHub Desktop.

Put object can be used to register instances to the ROT

Option Explicit
 
Private Declare Function CreateFileMoniker Lib "ole32" (ByVal lpszPathName As Long, pResult As IUnknown) As Long
Private Declare Function GetRunningObjectTable Lib "ole32" (ByVal dwReserved As Long, pResult As IUnknown) As Long
Private Declare Function DispCallFunc Lib "oleaut32" (ByVal pvInstance As Long, ByVal oVft As Long, ByVal lCc As Long, ByVal vtReturn As VbVarType, ByVal cActuals As Long, prgVt As Any, prgpVarg As Any, pvargResult As Variant) As Long
 
Private Sub Form_Load()
    Dim lCookie     As Long
    Dim oCol        As Collection
    Dim lColCookie  As Long
    
    lCookie = PutObject(Me, "MyMoniker")
    Debug.Print TypeName(GetObject("MyMoniker"))    '--- returns Form1
    '--- warning: "MyMoniker" path is "stacked"
    Set oCol = New Collection
    lColCookie = PutObject(oCol, "MyMoniker")
    Debug.Print TypeName(GetObject("MyMoniker"))    '--- still Form1
    '--- "pops" Form1
    RevokeObject lCookie
    Debug.Print TypeName(GetObject("MyMoniker"))    '--- returns Collection now
    RevokeObject lColCookie
End Sub
 
Private Function PutObject(oObj As Object, sPathName As String) As Long
    Const ROTFLAGS_REGISTRATIONKEEPSALIVE As Long = 1
    Const IDX_REGISTER  As Long = 3
    Dim pROT            As IUnknown
    Dim pMoniker        As IUnknown
    
    Call GetRunningObjectTable(0, pROT)
    Call CreateFileMoniker(StrPtr(sPathName), pMoniker)
    DispCallByVtbl pROT, IDX_REGISTER, ROTFLAGS_REGISTRATIONKEEPSALIVE, ObjPtr(oObj), ObjPtr(pMoniker), VarPtr(PutObject)
End Function
 
Private Sub RevokeObject(ByVal lCookie As Long)
    Const IDX_REVOKE    As Long = 4
    Dim pROT            As IUnknown
    
    Call GetRunningObjectTable(0, pROT)
    DispCallByVtbl pROT, IDX_REVOKE, lCookie
End Sub
 
Private Function DispCallByVtbl(pUnk As IUnknown, ByVal lIndex As Long, ParamArray A() As Variant) As Variant
    Const CC_STDCALL    As Long = 4
    Dim lIdx            As Long
    Dim vParam()        As Variant
    Dim vType(0 To 63)  As Integer
    Dim vPtr(0 To 63)   As Long
    Dim hResult         As Long
    
    vParam = A
    For lIdx = 0 To UBound(vParam)
        vType(lIdx) = VarType(vParam(lIdx))
        vPtr(lIdx) = VarPtr(vParam(lIdx))
    Next
    hResult = DispCallFunc(ObjPtr(pUnk), lIndex * 4, CC_STDCALL, vbLong, lIdx, vType(0), vPtr(0), DispCallByVtbl)
    If hResult < 0 Then
        Err.Raise hResult
    End If
End Function
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment