Differences
This shows you the differences between two versions of the page.
Both sides previous revision Previous revision Next revision | Previous revision | ||
guides:machine_code:com [2025-03-26 15:42] – Add IDispatch example 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 159: | Line 164: | ||
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. | 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> | <runner ahk2> | ||
Line 235: | Line 266: | ||
// --- MyObj Method Definitions ------------------------------------------------ | // --- MyObj Method Definitions ------------------------------------------------ | ||
- | // Returns the specified | + | // 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 246: | Line 277: | ||
} | } | ||
- | // Here are some other interfaces that may be requested by AutoHotkey | + | // Here are some of the 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 314: | Line 345: | ||
} | } | ||
- | // Dispatches | + | // Dispatches |
// https:// | // https:// | ||
HRESULT __stdcall MyObj_Invoke( | HRESULT __stdcall MyObj_Invoke( | ||
Line 435: | Line 466: | ||
; Retrieve an IDispatch COM Object | ; Retrieve an IDispatch COM Object | ||
MyObj := ComObjFromPtr(lib.NewMyObj()) | MyObj := ComObjFromPtr(lib.NewMyObj()) | ||
+ | |||
MyObj.FirstAddend := 1 | MyObj.FirstAddend := 1 | ||
MyObj.SecondAddend := 2 | MyObj.SecondAddend := 2 | ||
MsgBox MyObj.FirstAddend " + " MyObj.SecondAddend " = " MyObj.GetSum() | 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) | ||
+ | </ | ||
+ |