我一直在使用最新版本的WPFMediaKit。我要做的是编写一个示例应用程序,它将使用Samplegrabber捕获视频文件的视频帧,以便将它们作为单独的位图。
到目前为止,在构建和渲染图形时,我对以下代码运气不错。但是,当我使用此代码播放.wmv视频文件时,如果连接了samplegrabber,它将播放跳跃或波涛汹涌。如果我注释掉我添加samplegrabber过滤器的行,它可以正常工作。同样,它可以使用AVI / MPEG等正确使用samplegrabber。
protected virtual void OpenSource()
{
FrameCount = 0;
/* Make sure we clean up any remaining mess */
FreeResources();
if (m_sourceUri == null)
return;
string fileSource = m_sourceUri.OriginalString;
if (string.IsNullOrEmpty(fileSource))
return;
try
{
/* Creates the GraphBuilder COM object */
m_graph = new FilterGraphNoThread() as IGraphBuilder;
if (m_graph == null)
throw new Exception("Could not create a graph");
/* Add our prefered audio renderer */
InsertAudioRenderer(AudioRenderer);
var filterGraph = m_graph as IFilterGraph2;
if (filterGraph == null)
throw new Exception("Could not QueryInterface for the IFilterGraph2");
IBaseFilter renderer = CreateVideoMixingRenderer9(m_graph, 1);
IBaseFilter sourceFilter;
/* Have DirectShow find the correct source filter for the Uri */
var hr = filterGraph.AddSourceFilter(fileSource, fileSource, out sourceFilter);
DsError.ThrowExceptionForHR(hr);
/* We will want to enum all the pins on the source filter */
IEnumPins pinEnum;
hr = sourceFilter.EnumPins(out pinEnum);
DsError.ThrowExceptionForHR(hr);
IntPtr fetched = IntPtr.Zero;
IPin[] pins = { null };
/* Counter for how many pins successfully rendered */
int pinsRendered = 0;
m_sampleGrabber = (ISampleGrabber)new SampleGrabber();
SetupSampleGrabber(m_sampleGrabber);
hr = m_graph.AddFilter(m_sampleGrabber as IBaseFilter, "SampleGrabber");
DsError.ThrowExceptionForHR(hr);
/* Loop over each pin of the source filter */
while (pinEnum.Next(pins.Length, pins, fetched) == 0)
{
if (filterGraph.RenderEx(pins[0],
AMRenderExFlags.RenderToExistingRenderers,
IntPtr.Zero) >= 0)
pinsRendered++;
Marshal.ReleaseComObject(pins[0]);
}
Marshal.ReleaseComObject(pinEnum);
Marshal.ReleaseComObject(sourceFilter);
if (pinsRendered == 0)
throw new Exception("Could not render any streams from the source Uri");
/* Configure the graph in the base class */
SetupFilterGraph(m_graph);
HasVideo = true;
/* Sets the NaturalVideoWidth/Height */
//SetNativePixelSizes(renderer);
}
catch (Exception ex)
{
/* This exection will happen usually if the media does
* not exist or could not open due to not having the
* proper filters installed */
FreeResources();
/* Fire our failed event */
InvokeMediaFailed(new MediaFailedEventArgs(ex.Message, ex));
}
InvokeMediaOpened();
}
和
private void SetupSampleGrabber(ISampleGrabber sampleGrabber)
{
FrameCount = 0;
var mediaType = new AMMediaType
{
majorType = MediaType.Video,
subType = MediaSubType.RGB24,
formatType = FormatType.VideoInfo
};
int hr = sampleGrabber.SetMediaType(mediaType);
DsUtils.FreeAMMediaType(mediaType);
DsError.ThrowExceptionForHR(hr);
hr = sampleGrabber.SetCallback(this, 0);
DsError.ThrowExceptionForHR(hr);
}
我已经阅读了一些说明.wmv或.asf格式是异步的东西。我试图插入一个WMAsfReader来解码哪个有效,但是一旦它进入VMR9就会产生相同的行为。此外,当我注释掉IBaseFilter renderer = CreateVideoMixingRenderer9(m_graph, 1);
行并且filterGraph.Render(pins[0]);
时,我已经让它正常工作 - 唯一的缺点是它现在呈现在自己的Activemovie寡妇而不是我的控制中,但是samplegrabber功能正常,没有任何跳过。所以我认为这个bug存在于VMR9 / samplegrabbing中。
有任何帮助吗?我是新手。
答案 0 :(得分:1)
有些解码器会使用DXVA进行硬件加速。这是通过协商部分解码的格式,并将该部分解码的数据传递给渲染器以完成解码和渲染来实现的。如果在解码器和渲染器之间插入配置为RGB24的样本采集器,则将禁用硬件加速。
我确信,这是问题的症结所在。我担心细节仍然有点模糊,例如当你使用默认的VMR-7时它的工作原理,但是当你使用VMR-9时它会失败。我猜想解码器正在尝试使用dxva并在vmr-9情况下失败,但是它有一个合理的纯软件备份,在vmr-7中运行良好。
我不熟悉WPFMediaKit,但我认为最简单的解决方案是使用显式vmr-7创建替换显式vmr-9创建。也就是说,如果解码器仅使用vmr-7进行软件操作,那么请使用它并集中精力修复窗口重新定位问题。
答案 1 :(得分:0)
最终我发布的代码(这本身就是我从Jeremiah Morrill的WPFMediakit源代码中稍微修改过一些无耻的)实际上足以呈现.WMV文件并被样本抓取。
看起来波动与VS调试器或VS2008本身有关。在可视化编辑器中使用XAML一段时间后,然后运行应用程序,我将引入这种不稳定的行为。关闭VS2008似乎可以解决它。 :P
所以没有多少答案,但至少有一个(烦人的 - 重启VS2008)解决了它的问题。