这将是一篇很长的文章,因为我想向您展示我尝试完成这项工作的所有步骤:)
我有 C++ COM dll,其中包含一个 VideoPlayer 类,该类使用 Media Foundation API 来显示视频。
VideoPlayer 类是使用 IDL 文件定义的:
[
object,
uuid(74FDBBB1-BFFB-4F7E-ACA3-ADB0C7232790),
dual,
nonextensible,
pointer_default(unique)
]
interface IVideoPlayer : IDispatch {
[id(1)] HRESULT Initialize([in] HWND* video_hwnd, [in] HWND* event_hwnd);
[id(2)] HRESULT OpenUrl([in] BSTR url_path);
[id(3)] HRESULT Play();
[id(4)] HRESULT HandleEvent([in] INT pEventPtr);
[id(5)] HRESULT Repaint(void);
[id(6)] HRESULT Resize([in] LONG width, [in] LONG height);
};
此类在内部使用自定义演示器(基于WPFMediaKit项目),它在 IDirect3DSurface9 对象中输出视频帧。
自定义 Presenter 需要一个 IEVRPresenterCallback 类型的回调,定义如下:
MIDL_INTERFACE("B92D8991-6C42-4e51-B942-E61CB8696FCB")
IEVRPresenterCallback : public IUnknown
{
public:
virtual HRESULT STDMETHODCALLTYPE PresentSurfaceCB(IDirect3DSurface9 *pSurface) = 0;
};
如您所见,它不是在 IDL 文件中定义的,而是在头文件中声明的。
我需要向 VideoPlayer 类添加一个新函数,该函数允许调用 C# 代码传递从 IEVRPresenterCallback 继承的类的实例,该实例将设置为自定义演示者。
我尝试将此行添加到 VideoPlayer 的 IDL 文件中:
[id(7)] HRESULT RegisterCallback2([in] IEVRPresenterCallback * p_PresenterCallback);
但我收到一个错误:
错误 MIDL2025:语法错误:期望“IEVRPresenterCallback”附近的类型规范
我想这很正常,因为我没有在 IDL 中导入任何内容。这是正常的,因为 IEVRPresenterCallback 是在头文件中定义的。
我尝试导入头文件,但 IEVRPresenterCallback 定义的 MIDL_INTERFACE 宏生成错误:
错误 MIDL2025:语法错误:需要接口名称或 DispatchInterfaceName 或 CoclassName 或 ModuleName 或 LibraryName 或 ContractName 或“MIDL_INTERFACE”附近的类型规范
然后我尝试转发声明接口,但出现此错误:
错误 MIDL2011:未解析的类型声明:IEVRPresenterCallback [过程“RegisterCallback2”(接口“IVideoPlayer”)的参数“p_PresenterCallback”]
我最后一次尝试是更改 RegisterCallback 的定义,使用指向 IUnknown 而不是 IEVRPresenterCallback 的指针,并且在函数的声明中,我将指针转换为正确的接口。
这使得 C++ dll 可以正确编译。
在 C# 应用程序中,我将回调设置如下:
[ComVisible(true), ComImport, SuppressUnmanagedCodeSecurity, Guid("B92D8991-6C42-4e51-B942-E61CB8696FCB"), InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
internal interface IEVRPresenterCallback
{
[PreserveSig]
int PresentSurfaceCB(IntPtr pSurface);
}
internal class EVRPresenterCallback : IEVRPresenterCallback
{
public int PresentSurfaceCB(IntPtr pSurface)
{
return 0;
}
}
public partial class MainWindow : Window
{
private EmideeMediaFoundationLib.IVideoPlayer videoPlayer = new EmideeMediaFoundationLib.VideoPlayer();
private EVRPresenterCallback callback = new EVRPresenterCallback();
public MainWindow()
{
InitializeComponent();
}
private void Button_Click(object sender, RoutedEventArgs e)
{
videoHost.VideoPlayer.RegisterCallback(callback);
videoHost.VideoPlayer.OpenUrl(@"C:\Users\Public\Videos\Sample Videos\wildlife.wmv");
}
}
我遇到的问题是,尽管自定义演示者调用了回调,但我再也没有回到 C# PresentSurfaceCB 函数中。
我现在完全卡住了,我不知道问题出在哪里,也不知道如何解决它:(
有任何想法吗?
提前致谢