get object if you want

This commit is contained in:
Adam D. Ruppe 2023-10-03 15:26:28 -04:00
parent b5da2be1e6
commit 2d97891c11
1 changed files with 49 additions and 0 deletions

49
com.d
View File

@ -209,6 +209,21 @@ struct ComResult {
return ComIntermediary(newComObject, dispid, memberName);
}
T getD(T)() {
import std.conv;
switch(result.vt) {
case 3: // int
static if(is(T : const long))
return result.intVal;
throw new Exception("cannot convert variant of type int to requested " ~ T.stringof);
case 8: // string
static if(is(T : const string))
return makeUtf8StringFromWindowsString(result.bstrVal); // FIXME free?
throw new Exception("cannot convert variant of type string to requested " ~ T.stringof);
default: throw new Exception("can't handle this type " ~ to!string(result.vt));
}
}
}
struct ComIntermediary {
@ -222,6 +237,11 @@ struct ComIntermediary {
import std.stdio; writeln("Object ", cast(void*) a, " method ", c, " = ", name);
}
T getD(T)() {
auto res = _fetchProperty();
return res.getD!T;
}
ComResult _fetchProperty() {
DISPPARAMS disp_params;
@ -619,6 +639,35 @@ auto createComObject(T = Dynamic)(GUID classId) {
return ComClient!(Dify!T, typeof(obj))(obj);
}
auto getComObject(T = Dynamic)(wstring c, bool tryCreateIfGetFails = true) {
initializeClassicCom();
auto guid = guidForClassName(c);
auto get() {
auto iid = IID_IDispatch;
IUnknown obj;
ComCheck(GetActiveObject(&guid, null, &obj), "Get Object"); // code 0x800401e3 is operation unavailable if it isn't there i think
if(obj is null)
throw new Exception("null");
IDispatch disp;
ComCheck(obj.QueryInterface(&iid, cast(void**) &disp), "QueryInterface");
auto client = ComClient!(Dify!T, typeof(disp))(disp);
disp.AddRef();
return client;
}
if(tryCreateIfGetFails)
try
return get();
catch(Exception e)
return createComObject(guid);
else
return get();
}
// FIXME: add one to get by ProgID rather than always guid
// FIXME: add a dynamic com object that uses IDispatch