我正在尝试在我的C#应用程序中使用可自定义的文件打开对话框,我发现Common Item Dialog提供了用于自定义的简单API。然而,它需要一些COM Interop魔法,因为我是COM的完整noob,我认为这将是很好的学习练习。现在我遇到了以下问题:
我创建了以下 *。bat 文件来生成互操作程序集:
set OUT_DIR=.\com_interop
if not exist %OUT_DIR% mkdir "%OUT_DIR%
call "C:\Program Files (x86)\Microsoft Visual Studio 14.0\VC\bin\vcvars32.bat"
set SDK_DIR="C:\Program Files\Microsoft SDKs\Windows\v7.1"
%SDK_DIR%\Bin\midl.exe /server none /client stub /out "%OUT_DIR%" /tlb ShObjIdl.tlb /x64 %SDK_DIR%\Include\ShObjIdl.idl
%SDK_DIR%\Bin\TlbImp.exe "%OUT_DIR%\ShObjIdl.tlb" /out:ComInterop.ShObj.dll /namespace:ComInterop.ShObj /machine:X64
move /Y ComInterop.ShObj.dll "%OUT_DIR%\ComInterop.ShObj.dll"
所有与x64相关的开关都来自我(失败)尝试解决我的问题,我不确定它们是否有必要。
现在,问题是: midl.exe 会生成将句柄转换为_RemotableHandle
类型的符号。它生成Show(IntPtr)
:
RemoteShow(_RemotableHandle)
方法
[Guid("D57C7288-D4AD-4768-BE02-9D969532D960")]
[InterfaceType(1)]
public interface IFileOpenDialog : IFileDialog
{
void AddPlace(IShellItem psi, FDAP FDAP);
void Advise(IFileDialogEvents pfde, out uint pdwCookie);
void ClearClientData();
void Close(int hr);
void GetCurrentSelection(out IShellItem ppsi);
void GetFileName(out string pszName);
void GetFileTypeIndex(out uint piFileType);
void GetFolder(out IShellItem ppsi);
void GetOptions(out uint pfos);
void GetResult(out IShellItem ppsi);
void GetResults(out IShellItemArray ppenum);
void GetSelectedItems(out IShellItemArray ppsai);
void RemoteShow([ComAliasName("ComInterop.ShObj.wireHWND")] ref _RemotableHandle hwndOwner); // <===== argument, why u no HWND?
void SetClientGuid(ref Guid guid);
void SetDefaultExtension(string pszDefaultExtension);
void SetDefaultFolder(IShellItem psi);
void SetFileName(string pszName);
void SetFileNameLabel(string pszLabel);
void SetFileTypeIndex(uint iFileType);
void SetFileTypes(uint cFileTypes, ref _COMDLG_FILTERSPEC rgFilterSpec);
void SetFilter(IShellItemFilter pFilter);
void SetFolder(IShellItem psi);
void SetOkButtonLabel(string pszText);
void SetOptions(uint fos);
void SetTitle(string pszTitle);
void Unadvise(uint dwCookie);
}
是否可以生成使用常规句柄的方法?如果没有,我该如何从我的申请中拨打RemoteShow
? Bonus问题:为什么会这样发生(COM-wise)?
在其他一些主题中,我发现了使用以下代码将IntPtr
转换为_RemotableHandle
的建议:
_RemotableHandle HWNDtoRemotableHandle(IntPtr handle)
{
return (_RemotableHandle)Marshal.PtrToStructure(handle, typeof(_RemotableHandle));
}
但它因AccessViolation
错误而失败,我相信这样的转换不是很干净。