Differences
This shows you the differences between two versions of the page.
Next revision | Previous revision | ||
guides:machine_code:com [2025-03-26 13:48] – created geek | guides:machine_code:com [2025-03-28 13:28] (current) – [Using COM with MCode] Link MCode guide geek | ||
---|---|---|---|
Line 1: | Line 1: | ||
====== Using COM with MCode ====== | ====== Using COM with MCode ====== | ||
+ | |||
+ | Please read the basic [[guides: | ||
+ | |||
+ | If you have not, consider referencing the [[guides: | ||
When building integrations for AutoHotkey using MCode, it is often convenient to be able to work with COM objects. Either consuming COM objects in your MCode, or exporting COM objects from your MCode so they can be consumed by AHK. | When building integrations for AutoHotkey using MCode, it is often convenient to be able to work with COM objects. Either consuming COM objects in your MCode, or exporting COM objects from your MCode so they can be consumed by AHK. | ||
Line 59: | Line 63: | ||
// --- MyObj Method Definitions ------------------------------------------------ | // --- MyObj Method Definitions ------------------------------------------------ | ||
- | // Returns the IDispatch | + | // Returns the requested |
// https:// | // https:// | ||
HRESULT __stdcall MyObj_QueryInterface(MyObj *this, REFIID riid, void** ppvObject) { | HRESULT __stdcall MyObj_QueryInterface(MyObj *this, REFIID riid, void** ppvObject) { | ||
Line 70: | Line 74: | ||
} | } | ||
- | // Here are some other interfaces that may be requested by AutoHotkey | + | // Here are some other interfaces that are sometimes |
// IID_IDispatch {00020400-0000-0000-C000-000000000046} | // IID_IDispatch {00020400-0000-0000-C000-000000000046} | ||
// | // | ||
// IID_IProvideClassInfo {B196B283-BAB4-101A-B69C-00AA00341D07} | // IID_IProvideClassInfo {B196B283-BAB4-101A-B69C-00AA00341D07} | ||
// | // | ||
- | // IID_IObjectComCompatible { 0x619f7e25, 0x6d89, 0x4eb4, 0xb2, 0xfb, 0x18, 0xe7, 0xc7, 0x3c, 0xe, 0xa6 } | + | // IID_IObjectComCompatible {619F7E25-6D89-4EB4-B2Fb-18E7C73C0EA6} |
- | // | + | // |
return E_NOINTERFACE; | return E_NOINTERFACE; | ||
} | } | ||
Line 150: | Line 154: | ||
; Retrieve an IMyObj COM Object | ; Retrieve an IMyObj COM Object | ||
MyObj := lib.NewMyObj() | MyObj := lib.NewMyObj() | ||
+ | |||
ComCall(3, MyObj, " | ComCall(3, MyObj, " | ||
ComCall(4, MyObj, " | ComCall(4, MyObj, " | ||
Line 155: | Line 160: | ||
MsgBox "1 + 2 = " result | MsgBox "1 + 2 = " result | ||
</ | </ | ||
+ | |||
+ | ===== Exporting IDispatch from MCode ===== | ||
+ | |||
+ | By implementing the IDispatch interface, your custom COM object can be wrapped by AHK. Once wrapped, users can interact with the object using native object syntax. | ||
+ | |||
+ | IDispatch extends IUnknown, so it has the three IUnknown methods. After those, it includes these four methods: | ||
+ | * '' | ||
+ | * '' | ||
+ | * '' | ||
+ | * '' | ||
+ | |||
+ | The methods '' | ||
+ | |||
+ | The '' | ||
+ | |||
+ | If your object allows for new properties to be created dynamically, | ||
+ | |||
+ | The '' | ||
+ | * '' | ||
+ | * '' | ||
+ | * '' | ||
+ | * '' | ||
+ | |||
+ | '' | ||
+ | |||
+ | * '' | ||
+ | * '' | ||
+ | * '' | ||
+ | |||
+ | By implementing '' | ||
+ | |||
+ | <runner ahk2> | ||
+ | #Requires AutoHotkey v2.0 | ||
+ | #include <MCL> | ||
+ | |||
+ | lib := MCL.FromC(" | ||
+ | ( | ||
+ | #include < | ||
+ | #include < | ||
+ | #include < | ||
+ | #define DISP_E_TYPEMISMATCH 0x80020005 | ||
+ | |||
+ | MCL_IMPORT(int, | ||
+ | MCL_IMPORT(HRESULT, | ||
+ | |||
+ | |||
+ | // --- Interface Identifiers --------------------------------------------------- | ||
+ | |||
+ | // IID_IUnknown {00000000-0000-0000-C000-000000000046} | ||
+ | GUID IID_IUnknown = { 0x00000000, 0x0000, 0x0000, { 0xC0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x46 } }; | ||
+ | |||
+ | // IID_IDispatch {00020400-0000-0000-C000-000000000046} | ||
+ | GUID IID_IDispatch = { 0x00020400, 0x0000, 0x0000, { 0xC0, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x46 } }; | ||
+ | |||
+ | |||
+ | // --- MyObj Structural Definition --------------------------------------------- | ||
+ | |||
+ | // Forward declare the MyObj type so it can be referenced in MyObjVtbl | ||
+ | typedef struct MyObj MyObj; | ||
+ | |||
+ | // Define the virtual method table for objects of type MyObj | ||
+ | typedef struct MyObjVtbl { | ||
+ | // Implements IUnknown | ||
+ | __stdcall HRESULT(*QueryInterface) (MyObj *this, REFIID riid, void **ppvObject); | ||
+ | __stdcall ULONG(*AddRef) (MyObj *this); | ||
+ | __stdcall ULONG(*Release) (MyObj *this); | ||
+ | |||
+ | // Implements IDispatch | ||
+ | __stdcall HRESULT(*GetTypeInfoCount) (MyObj *this, UINT *pctinfo); | ||
+ | __stdcall HRESULT(*GetTypeInfo) (MyObj *this, UINT iTInfo, LCID lcid, void **ppTInfo); | ||
+ | __stdcall HRESULT(*GetIDsOfNames) ( | ||
+ | MyObj *This, | ||
+ | REFIID riid, | ||
+ | LPOLESTR *rgszNames, | ||
+ | UINT cNames, | ||
+ | LCID lcid, | ||
+ | DISPID *rgDispId); | ||
+ | __stdcall HRESULT(*Invoke) ( | ||
+ | MyObj *This, | ||
+ | DISPID dispIdMember, | ||
+ | REFIID riid, | ||
+ | LCID lcid, | ||
+ | WORD wFlags, | ||
+ | DISPPARAMS *pDispParams, | ||
+ | VARIANT *pVarResult, | ||
+ | EXCEPINFO *pExcepInfo, | ||
+ | UINT *puArgErr); | ||
+ | } MyObjVtbl; | ||
+ | |||
+ | // Define virtual method DISPIDs | ||
+ | #define MY_OBJ_DISPID_FIRST_ADDEND 1 | ||
+ | #define MY_OBJ_DISPID_SECOND_ADDEND 2 | ||
+ | #define MY_OBJ_DISPID_GET_SUM 3 | ||
+ | |||
+ | // Define the basic structure of MyObj instances so it can be referenced in | ||
+ | // method implementations | ||
+ | struct MyObj { | ||
+ | const MyObjVtbl* lpVtbl; | ||
+ | int refcount; | ||
+ | LONG firstAddend; | ||
+ | LONG secondAddend; | ||
+ | }; | ||
+ | |||
+ | |||
+ | // --- MyObj Method Definitions ------------------------------------------------ | ||
+ | |||
+ | // Returns the requested interface of the object when requested | ||
+ | // https:// | ||
+ | HRESULT __stdcall MyObj_QueryInterface(MyObj *this, REFIID riid, void** ppvObject) { | ||
+ | // If the interface requested is implemented by MyObj, return it | ||
+ | if (memcmp(riid, | ||
+ | memcmp(riid, | ||
+ | this-> | ||
+ | *ppvObject = this; | ||
+ | return S_OK; | ||
+ | } | ||
+ | |||
+ | // Here are some of the interfaces that are sometimes requested by AutoHotkey | ||
+ | // IID_IDispatch {00020400-0000-0000-C000-000000000046} | ||
+ | // | ||
+ | // IID_IProvideClassInfo {B196B283-BAB4-101A-B69C-00AA00341D07} | ||
+ | // | ||
+ | // IID_IObjectComCompatible {619F7E25-6D89-4EB4-B2Fb-18E7C73C0EA6} | ||
+ | // | ||
+ | return E_NOINTERFACE; | ||
+ | } | ||
+ | |||
+ | // Increments the reference counter | ||
+ | // https:// | ||
+ | ULONG __stdcall MyObj_AddRef(MyObj *this) { | ||
+ | ++this-> | ||
+ | return this-> | ||
+ | } | ||
+ | |||
+ | // Decrements the reference counter and frees the object when it hits 0 | ||
+ | // https:// | ||
+ | ULONG __stdcall MyObj_Release(MyObj* this) { | ||
+ | --this-> | ||
+ | if (this-> | ||
+ | return this-> | ||
+ | } | ||
+ | |||
+ | // Make sure to perform any code needed to free individual fields before | ||
+ | // freeing the entire object | ||
+ | free(this); | ||
+ | return 0; | ||
+ | } | ||
+ | |||
+ | // Allows debug-style inspection of the COM object | ||
+ | // https:// | ||
+ | HRESULT __stdcall MyObj_GetTypeInfoCount(MyObj *this, UINT* pctinfo) { | ||
+ | return E_NOTIMPL; | ||
+ | } | ||
+ | |||
+ | // Allows debug-style inspection of the COM object | ||
+ | // https:// | ||
+ | HRESULT __stdcall MyObj_GetTypeInfo(MyObj *this, UINT iTInfo, LCID lcid, void **ppTInfo) { | ||
+ | return E_NOTIMPL; | ||
+ | } | ||
+ | |||
+ | // Translates script-given method and property names to unique DISPID values. | ||
+ | // https:// | ||
+ | HRESULT __stdcall MyObj_GetIDsOfNames( | ||
+ | MyObj *this, | ||
+ | /* [in] */ REFIID riid, | ||
+ | /* [in] */ LPOLESTR *rgszNames, | ||
+ | /* [in] */ UINT cNames, | ||
+ | /* [in] */ LCID lcid, | ||
+ | /* [out] */ DISPID *rgDispId) { | ||
+ | HRESULT retval = S_OK; | ||
+ | for (int i = 0; i < cNames; i++) { | ||
+ | if (0 == _wcsicmp(rgszNames[i], | ||
+ | rgDispId[i] = MY_OBJ_DISPID_FIRST_ADDEND; | ||
+ | } else if (0 == _wcsicmp(rgszNames[i], | ||
+ | rgDispId[i] = MY_OBJ_DISPID_SECOND_ADDEND; | ||
+ | } else if (0 == _wcsicmp(rgszNames[i], | ||
+ | rgDispId[i] = MY_OBJ_DISPID_GET_SUM; | ||
+ | } else { | ||
+ | retval = DISP_E_UNKNOWNNAME; | ||
+ | rgDispId[i] = DISPID_UNKNOWN; | ||
+ | } | ||
+ | } | ||
+ | return retval; | ||
+ | } | ||
+ | |||
+ | // Dispatches MyObj' | ||
+ | // https:// | ||
+ | HRESULT __stdcall MyObj_Invoke( | ||
+ | MyObj *this, | ||
+ | /* [in] */ DISPID dispIdMember, | ||
+ | /* [in] */ REFIID riid, | ||
+ | /* [in] */ LCID lcid, | ||
+ | /* [in] */ WORD wFlags, | ||
+ | /* [out][in] */ DISPPARAMS *pDispParams, | ||
+ | /* [out][opt] */ VARIANT *pVarResult, | ||
+ | /* [out][opt] */ EXCEPINFO *pExcepInfo, | ||
+ | /* [out][opt] */ UINT *puArgErr) { | ||
+ | |||
+ | // Set First Addend | ||
+ | if (dispIdMember == MY_OBJ_DISPID_FIRST_ADDEND && (wFlags & DISPATCH_PROPERTYPUT)) { | ||
+ | // Require a single parameter | ||
+ | if (pDispParams-> | ||
+ | return DISP_E_BADPARAMCOUNT; | ||
+ | } | ||
+ | |||
+ | // Coerce to 4-byte integer (LONG) | ||
+ | if (S_OK != VariantChangeType( | ||
+ | & | ||
+ | & | ||
+ | 0, | ||
+ | VT_I4)) { | ||
+ | return DISP_E_TYPEMISMATCH; | ||
+ | } | ||
+ | |||
+ | this-> | ||
+ | pVarResult-> | ||
+ | pVarResult-> | ||
+ | |||
+ | return S_OK; | ||
+ | } | ||
+ | |||
+ | // Get First Addend | ||
+ | if (dispIdMember == MY_OBJ_DISPID_FIRST_ADDEND && (wFlags & DISPATCH_PROPERTYGET)) { | ||
+ | pVarResult-> | ||
+ | pVarResult-> | ||
+ | return S_OK; | ||
+ | } | ||
+ | |||
+ | // Set Second Addend | ||
+ | if (dispIdMember == MY_OBJ_DISPID_SECOND_ADDEND && (wFlags & DISPATCH_PROPERTYPUT)) { | ||
+ | // Require a single parameter | ||
+ | if (pDispParams-> | ||
+ | return DISP_E_BADPARAMCOUNT; | ||
+ | } | ||
+ | |||
+ | // Coerce to 4-byte integer (LONG) | ||
+ | if (S_OK != VariantChangeType( | ||
+ | & | ||
+ | & | ||
+ | 0, | ||
+ | VT_I4)) { | ||
+ | return DISP_E_TYPEMISMATCH; | ||
+ | } | ||
+ | |||
+ | this-> | ||
+ | pVarResult-> | ||
+ | pVarResult-> | ||
+ | |||
+ | return S_OK; | ||
+ | } | ||
+ | |||
+ | // Get Second Addend | ||
+ | if (dispIdMember == MY_OBJ_DISPID_SECOND_ADDEND && (wFlags & DISPATCH_PROPERTYGET)) { | ||
+ | pVarResult-> | ||
+ | pVarResult-> | ||
+ | return S_OK; | ||
+ | } | ||
+ | |||
+ | // Call GetSum | ||
+ | if (dispIdMember == MY_OBJ_DISPID_GET_SUM && (wFlags & DISPATCH_METHOD)) { | ||
+ | // Require no parameters | ||
+ | if (pDispParams-> | ||
+ | return DISP_E_BADPARAMCOUNT; | ||
+ | } | ||
+ | |||
+ | pVarResult-> | ||
+ | pVarResult-> | ||
+ | |||
+ | return S_OK; | ||
+ | } | ||
+ | |||
+ | return DISP_E_MEMBERNOTFOUND; | ||
+ | } | ||
+ | |||
+ | |||
+ | // --- Object Factory ---------------------------------------------------------- | ||
+ | |||
+ | // Create a constant instance of the virtual method table to be shared | ||
+ | // among the instance objects. Similar to AHK's prototype object. | ||
+ | static const MyObjVtbl MY_OBJ_VTBL = { | ||
+ | .QueryInterface = MyObj_QueryInterface, | ||
+ | .AddRef = MyObj_AddRef, | ||
+ | .Release = MyObj_Release, | ||
+ | .GetTypeInfoCount = MyObj_GetTypeInfoCount, | ||
+ | .GetTypeInfo = MyObj_GetTypeInfo, | ||
+ | .GetIDsOfNames = MyObj_GetIDsOfNames, | ||
+ | .Invoke = MyObj_Invoke | ||
+ | }; | ||
+ | |||
+ | // Define a factory for MyObj instances that can be called by AutoHotkey | ||
+ | MCL_EXPORT(NewMyObj, | ||
+ | MyObj* NewMyObj() { | ||
+ | MyObj* obj = malloc(sizeof(MyObj)); | ||
+ | |||
+ | // Initialize all fields | ||
+ | obj-> | ||
+ | obj-> | ||
+ | obj-> | ||
+ | obj-> | ||
+ | |||
+ | return obj; | ||
+ | } | ||
+ | )") | ||
+ | |||
+ | ; Retrieve an IDispatch COM Object | ||
+ | MyObj := ComObjFromPtr(lib.NewMyObj()) | ||
+ | |||
+ | MyObj.FirstAddend := 1 | ||
+ | MyObj.SecondAddend := 2 | ||
+ | MsgBox MyObj.FirstAddend " + " MyObj.SecondAddend " = " MyObj.GetSum() | ||
+ | </ | ||
+ | |||
+ | ===== Consuming AHK Objects from MCode ===== | ||
+ | |||
+ | AutoHotkey objects are, at their basic, IDispatch objects. Just like we can create IDispatch objects using basic C code, we can interact with IDispatch objects using basic C code. | ||
+ | |||
+ | <runner ahk2> | ||
+ | #Requires AutoHotkey v2.0 | ||
+ | #include <MCL> | ||
+ | |||
+ | lib := MCL.FromC(" | ||
+ | ( | ||
+ | #include < | ||
+ | #include < | ||
+ | #include < | ||
+ | |||
+ | MCL_IMPORT(HRESULT, | ||
+ | |||
+ | static IID IID_NULL = { 0x00000000, 0x0000, 0x0000, { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 } }; | ||
+ | |||
+ | MCL_EXPORT(SumValues, | ||
+ | int SumValues(IDispatch* pObj) { | ||
+ | // Get the ID of the " | ||
+ | DISPID dispidLength = DISPID_UNKNOWN; | ||
+ | LPOLESTR strLength = L" | ||
+ | pObj-> | ||
+ | if (dispidLength == DISPID_UNKNOWN) { | ||
+ | return 0; | ||
+ | } | ||
+ | |||
+ | // Invoke the " | ||
+ | DISPPARAMS dpLength = { .cArgs = 0, .cNamedArgs = 0 }; | ||
+ | VARIANT vLength = { .vt = VT_EMPTY }; | ||
+ | if (S_OK != pObj-> | ||
+ | pObj, dispidLength, | ||
+ | & | ||
+ | vLength.vt != VT_I4) { | ||
+ | return 0; | ||
+ | } | ||
+ | |||
+ | // Loop from i = 0 to Length | ||
+ | int out = 0; | ||
+ | for (int i = 0; i < vLength.lVal; | ||
+ | // Invoke the value property, with argument `i+1`. This is equivalent | ||
+ | // to `object[i+1]` | ||
+ | VARIANT vargGet = { .vt = VT_I4, .lVal = i + 1 }; | ||
+ | DISPPARAMS dpGet = { .cArgs = 1, .cNamedArgs = 0, .rgvarg = & | ||
+ | VARIANT vGet = { .vt = VT_EMPTY }; | ||
+ | if (S_OK != pObj-> | ||
+ | pObj, DISPID_VALUE, | ||
+ | &dpGet, &vGet, NULL, NULL)) { | ||
+ | return 0; | ||
+ | } | ||
+ | |||
+ | // Coerce to 4-byte integer (LONG) | ||
+ | if (S_OK != VariantChangeType(& | ||
+ | return 0; | ||
+ | } | ||
+ | |||
+ | out += vGet.lVal; | ||
+ | } | ||
+ | |||
+ | return out; | ||
+ | } | ||
+ | )") | ||
+ | |||
+ | input := [1, 2, 3] | ||
+ | MsgBox lib.SumValues(ObjPtr(input)) | ||
+ | </ | ||
+ | |||
+ | ==== Implementing QuickSort ==== | ||
+ | |||
+ | By invoking '' | ||
+ | |||
+ | <runner ahk2> | ||
+ | #Requires AutoHotkey v2.0 | ||
+ | #include <MCL> | ||
+ | |||
+ | lib := MCL.FromC(" | ||
+ | ( | ||
+ | #include < | ||
+ | #include < | ||
+ | #include < | ||
+ | |||
+ | #define E_FAIL 0x80004005 | ||
+ | |||
+ | MCL_IMPORT(HRESULT, | ||
+ | MCL_IMPORT(HRESULT, | ||
+ | |||
+ | HRESULT GetItem(IDispatch* array, ULONG index, VARIANT* pvarResult) { | ||
+ | VARIANT vargGet = { .vt = VT_I4, .lVal = index }; | ||
+ | DISPPARAMS dpGet = { .cArgs = 1, .cNamedArgs = 0, .rgvarg = & | ||
+ | return array-> | ||
+ | array, DISPID_VALUE, | ||
+ | &dpGet, pvarResult, NULL, NULL); | ||
+ | } | ||
+ | |||
+ | HRESULT PutItem(IDispatch* array, ULONG index, VARIANT* pvarItem) { | ||
+ | VARIANT vargIndex = { .vt = VT_I4, .lVal = index }; | ||
+ | VARIANT rgvargArgs[] = { *pvarItem, vargIndex }; | ||
+ | DISPPARAMS dpPut = { .cArgs = 2, .cNamedArgs = 0, .rgvarg = rgvargArgs }; | ||
+ | return array-> | ||
+ | array, DISPID_VALUE, | ||
+ | &dpPut, NULL, NULL, NULL); | ||
+ | } | ||
+ | |||
+ | HRESULT partition(IDispatch *array, ULONG lo, ULONG hi, ULONG* newPartition) { | ||
+ | HRESULT result = E_FAIL; | ||
+ | |||
+ | VARIANT varHi = { .vt = VT_EMPTY }; | ||
+ | VARIANT varI = { .vt = VT_EMPTY }; | ||
+ | VARIANT varJ = { .vt = VT_EMPTY }; | ||
+ | |||
+ | // Use highest index value as pivot | ||
+ | if (S_OK != GetItem(array, | ||
+ | |||
+ | ULONG i = lo; | ||
+ | for (ULONG j = lo; j < hi; j++) { | ||
+ | if (S_OK != GetItem(array, | ||
+ | // lcid = LOCALE_INVARIANT, | ||
+ | if ( | ||
+ | (varJ.vt != varHi.vt) ? (varJ.vt <= varHi.vt) : | ||
+ | VarCmp(& | ||
+ | if (S_OK != GetItem(array, | ||
+ | if (S_OK != PutItem(array, | ||
+ | if (S_OK != PutItem(array, | ||
+ | i = i + 1; | ||
+ | VariantClear(& | ||
+ | } | ||
+ | VariantClear(& | ||
+ | } | ||
+ | |||
+ | // Center the pivot value | ||
+ | if (S_OK != GetItem(array, | ||
+ | if (S_OK != PutItem(array, | ||
+ | if (S_OK != PutItem(array, | ||
+ | |||
+ | *newPartition = i; | ||
+ | result = S_OK; | ||
+ | |||
+ | fail: | ||
+ | VariantClear(& | ||
+ | VariantClear(& | ||
+ | VariantClear(& | ||
+ | return result; | ||
+ | } | ||
+ | |||
+ | MCL_EXPORT(quicksort, | ||
+ | HRESULT quicksort(IDispatch *array, ULONG lo, ULONG hi) { | ||
+ | if (lo >= hi) return S_OK; | ||
+ | ULONG p = 0; | ||
+ | if (S_OK != partition(array, | ||
+ | if (S_OK != quicksort(array, | ||
+ | if (S_OK != quicksort(array, | ||
+ | return S_OK; | ||
+ | } | ||
+ | |||
+ | )") | ||
+ | |||
+ | x := [" | ||
+ | |||
+ | MsgBox " | ||
+ | |||
+ | lib.quicksort(ObjPtr(x), | ||
+ | |||
+ | MsgBox "After Sorting: " JSON.Dump(x) | ||
+ | </ | ||
+ |