首页 > 解决方案 > 使用 c++ Builder 将 Directshow 渲染到面板上

问题描述

我正在尝试让 directshow 播放器显示在 C++ 构建器中主窗口以外的面板上。我采用了 Microsoft 示例播放器代码并将其制作成一个 c++ 应用程序,该应用程序创建了表单并将视频播放到窗口上。

我现在想添加到 vcl 应用程序并希望将视频显示到特定面板上。每当我尝试在 VCL 下运行图形时,它只会在我将句柄作为应用程序句柄而不是面板句柄传递时构建和运行。我知道图表会运行,因为我听到了音频,但我无法将视频渲染到面板上。如果我将 hwnd 作为任何 VCL 窗口的句柄传递,则过滤器图 (SetNotifyWindow) 的回调分配失败,如果我通过 Application->Handle,它将起作用。

    hr = m_pEvent->SetNotifyWindow((OAHWND)m_hwnd, WM_GRAPH_EVENT, NULL);
    if (FAILED(hr))
    {
        goto done;
    }

视频渲染功能可以找到最适合的渲染器,如下所示

HRESULT DShowPlayer::CreateVideoRenderer()
{
    HRESULT hr = E_FAIL;

    enum { Try_EVR, Try_VMR9, Try_VMR7 };

    for (DWORD i = Try_EVR; i <= Try_VMR7; i++)
    {
        switch (i)
        {
        case Try_EVR:
            m_pVideo = new (std::nothrow) CEVR();
            break;

        case Try_VMR9:
            m_pVideo = new (std::nothrow) CVMR9();
            break;

        case Try_VMR7:
            m_pVideo = new (std::nothrow) CVMR7();
            break;
        }

        if (m_pVideo == NULL)
        {
            hr = E_OUTOFMEMORY;
            break;
        }

        hr = m_pVideo->AddToGraph(m_pGraph, m_hwnd);
        if (SUCCEEDED(hr))
        {
            break;
        }

        delete m_pVideo;
        m_pVideo = NULL;
    }
    return hr;
}

这些中的每一个都在无窗口模式下初始化,例如 VMR9

HRESULT InitWindowlessVMR9( 
    IBaseFilter *pVMR,              // Pointer to the VMR
    HWND hwnd,                      // Clipping window
    IVMRWindowlessControl9** ppWC   // Receives a pointer to the VMR.
    ) 
{ 

    IVMRFilterConfig9 * pConfig = NULL; 
    IVMRWindowlessControl9 *pWC = NULL;

    // Set the rendering mode.  
    HRESULT hr = pVMR->QueryInterface(IID_PPV_ARGS(&pConfig)); 
    if (FAILED(hr))
    {
        goto done;
    }

    hr = pConfig->SetRenderingMode(VMR9Mode_Windowless); 
    if (FAILED(hr))
    {
        goto done;
    }

    // Query for the windowless control interface.
    hr = pVMR->QueryInterface(IID_PPV_ARGS(&pWC));
    if (FAILED(hr))
    {
        goto done;
    }

    // Set the clipping window.
    hr = pWC->SetVideoClippingWindow(hwnd);
    if (FAILED(hr))
    {
        goto done;
    }

    // Preserve aspect ratio by letter-boxing
    hr = pWC->SetAspectRatioMode(VMR9ARMode_LetterBox);
    if (FAILED(hr))
    {
        goto done;
    }

    // Return the IVMRWindowlessControl pointer to the caller.
    *ppWC = pWC;
    (*ppWC)->AddRef();

done:
    SafeRelease(&pConfig);
    SafeRelease(&pWC);
    return hr; 
} 

他们将渲染过滤器添加到图中,如下所示:

HRESULT CVMR9::AddToGraph(IGraphBuilder *pGraph, HWND hwnd)
{
    IBaseFilter *pVMR = NULL;

    HRESULT hr = AddFilterByCLSID(pGraph, CLSID_VideoMixingRenderer9, &pVMR, L"VMR-9");
    if (SUCCEEDED(hr))
    {
        // Set windowless mode on the VMR. This must be done before the VMR 
        // is connected.
        hr = InitWindowlessVMR9(pVMR, hwnd, &m_pWindowless);
    }
    SafeRelease(&pVMR);
    return hr;
}

如何让播放器显示在面板上?

标签: videoc++builderdirectshow

解决方案


请注意,IMediaEventEx::SetNotifyWindow您在代码段中引用的内容并未定义视频的显示位置。此调用旨在让过滤图管理器知道您希望在哪里接收通知消息。

您没有指定您使用的视频渲染器模式,所以我认为它是最简单的(尽管称为“旧版”) -窗口模式IVideoWindow::put_Owner以及IVideoWindow::SetWindowPosition如何定义视频在窗口模式下的显示位置,put_Owner特别是采用父窗口句柄。

2019-06-04 更新

因此,您正在尝试更改DShowPlayer 示例和教程,并且代码使用的是无窗口模式。

说明 Windows Vista 示例将使用 EVR 代码路径,所以我不确定您为什么更喜欢参考 VMR-9。尽管如此,它们都有“无窗口初始化”方法HWND作为参数。这是嵌入视频的窗口句柄。您应该使用可见的窗口句柄( EVR 的参考)并提供有效的剪切矩形(参考)。该示例已经这样做了,您可以更改它并以类似的方式使用不同的窗口,例如您的面板。


推荐阅读