如何解决我是否使用GMFBridge.DLL正确预览/捕获流?
| 我正在尝试使用GMFBuilder,以便可以预览网络摄像头中的流并定期保存它,而无需重新启动整个图形。但是我不确定这是否正确,我尝试遵循示例,但是代码已更新,并且情况已更改。 我尝试创建:WEbcam -> Smart Tee (preview) -> AVI Decompressor -> Video Renderer
Smart Tee (Capture) -> BridgeSinkFilter
并且:
BridgeSourceFilter -> ffdshow video encoder -> haali mastroska muxer
(只是因为它易于使用)
关于使代码正常运行的输入将不胜感激。
private void button2_Click(object sender,EventArgs e)
{
IGraphBuilder firstGraph = (IGraphBuilder)new FilterGraph();
IGraphBuilder secondGraph = (IGraphBuilder)new FilterGraph();
IBaseFilter BridgeSinkFilter;
IBaseFilter BridgeSourceFilter;
IBaseFilter Source;
IBaseFilter Mux;
IBaseFilter FileWriter;
IGMFBridgeController bridge = (IGMFBridgeController)new GMFBridgeController();
bridge.AddStream(1,eFormatType.eMuxInputs,1);
BridgeSinkFilter = (IBaseFilter)bridge.InsertSinkFilter(firstGraph);
Source = FindFilter(FilterCategory.VideoInputDevice,\"SG330\");
firstGraph.AddFilter(Source,\"source\");
IBaseFilter SmartTee = FindFilter(FilterCategory.LegacyAmFilterCategory,\"Smart Tee\");
firstGraph.AddFilter(SmartTee,\"Smart Tee\");
IPin pinin,pinout;
pinout = FindPinByDirection( Source,PinDirection.Output);
pinin = FindPinByDirection( SmartTee,PinDirection.Input);
firstGraph.Connect(pinout,pinin);
pinout = FindPinByDirection(SmartTee,PinDirection.Output);
pinin = FindPinByDirection(BridgeSinkFilter,pinin);
IBaseFilter Decomp = FindFilter(FilterCategory.LegacyAmFilterCategory,\"AVI Decompressor\");
firstGraph.AddFilter(Decomp,\"Avi Decompressor\");
pinout = FindPinByDirection(SmartTee,PinDirection.Output);
pinin = FindPinByDirection(Decomp,pinin);
IBaseFilter Renderer = FindFilter(FilterCategory.LegacyAmFilterCategory,\"Video Renderer\");
firstGraph.AddFilter(Renderer,\"Video Renderer\");
pinout = FindPinByDirection(Decomp,PinDirection.Output);
pinin = FindPinByDirection(Renderer,pinin);
DsROTEntry g = new DsROTEntry(firstGraph);
BridgeSourceFilter = (IBaseFilter)bridge.InsertSourceFilter(BridgeSinkFilter,secondGraph);
DsROTEntry h = new DsROTEntry(secondGraph);
IBaseFilter Muxe = FindFilter(FilterCategory.VideoCompressorCategory,\"ffdshow video encoder\");
secondGraph.AddFilter(Muxe,\"Mux\");
pinout = FindPinByDirection(BridgeSourceFilter,PinDirection.Output);
pinin = FindPinByDirection(Muxe,PinDirection.Input);
secondGraph.Connect(pinout,pinin);
IBaseFilter MKV = FindFilter(FilterCategory.LegacyAmFilterCategory,\"Haali Matroska Muxer\");
IFileSinkFilter fs = (IFileSinkFilter)MKV;
fs.SetFileName(\"c:\\\\cool.mkv\",null);
secondGraph.AddFilter(MKV,\"mux\");
pinout = FindPinByDirection(Muxe,PinDirection.Output);
pinin = FindPinByDirection(MKV,PinDirection.Input);
secondGraph.Connect(pinout,pinin);
bridge.BridgeGraphs(BridgeSinkFilter,BridgeSourceFilter);
IMediaControl mediacontrolforpartone = (IMediaControl)firstGraph;
mediacontrolforparttwo = (IMediaControl)secondGraph;
mediacontrolforpartone.Run();
mediacontrolforparttwo.Run();
}
解决方法
要以我目前的知识正确使用GMFBridge:
从以下位置获取GMFBridge DLL:http://www.gdcl.co.uk/gmfbridge/
从以下位置获取DirectShowLib DLL:https://sourceforge.net/projects/directshownet/files/DirectShowNET/
将它们都包含在您的项目中。
创建2个图。 1用于预览,第二用于捕获。
IGraphBuilder firstgraph = (IGraphBuilder) new FilterGraph();
IGraphBuilder secondgraph = (IGraphBuilder) new FilterGraph();
从GMFBridge dll创建一个网桥,它将连接两个图形
IGMFBridgeController Bridge = (IGMFBridgeController) new GMFBridgeController();
在这里,您可以设置桥接器以允许多路复用输入
Bridge.AddStream(1,eFormatType.eMuxInputs,1);
从这里,您可以添加源视频过滤器,它不需要连接到网桥,添加Smart Tee并将源连接到Smart Tee。
然后创建一个过滤器以容纳第一个将完成工作的桥式过滤器
BridgeSinkFilter = (IBaseFilter)Bridge.InsertSinkFilter(firstgraph);
此过滤器将连续接受来自Smart Tee捕获引脚的视频。如果第二图桥过滤器已连接并正在运行,它将把视频从BridgeSInkFilter传递到第二图。否则,它只会将其丢弃,但始终会运行。
将BridgeSinkFilter连接到Smart Tee的捕获引脚。
我发现连接引脚的最佳方法是通过https://splicer.svn.codeplex.com/svn/src/Splicer/Utilities/FilterGraphTools.cs使用FindPinByDirection,然后调用
firstgraph.connect(pinoutput,pininput)
从此处开始预览视频,应添加FilterCategory.LegacyAmFilterCategory的AVI Decompressor过滤器,并将其连接到Smart Tee的Preview引脚。然后添加了Video Renderer并连接到AVI Decompressor。
那应该照顾第一个图。
第二个图需要从桥开始。它会从第一个图bridgesinkfilter创建一个桥,并将其拉到第二个图,我们可以在其中做任何我们想做的事情。为此,我们需要桥的另一侧。
IBaseFilter BridgeSourceFilter = (IBaseFilter)Bridge.InsertSourceFilter(BridgeSinkFilter,secondgraph);
这将源设置为来自第一个图的接收器过滤器,但以BridgeSourceFilter为幌子将其放置在我们的第二个图上。
现在连接编码器,ffdshow视频编码器等。
将其连接到BridgeSourceFilter。
添加一个muxer,AVI Mux和文件编写器。连接他们。多数民众赞成在第二张图。
为了完成图表,我们需要创建2个可以启动和停止图表的媒体控制器。
IMediaControl MediaControl_FirstGraph = (IMediaControl)firstgraph;
IMediaControl MediaControl_SecondGraph = (IMediaControl)secondgraph;
现在我们可以打电话
MediaControl_FirstGraph.Run() to start previewing the video.
然后要捕获该视频,我们需要在第一张图和第二张图之间连接桥,然后运行第二张图。
Bridge.BridgeGraphs(BridgeSinkFilter,BridgeSourceFilter);
MediaControl_SecondGraph.Run()
在任何时候,您都可以通过断开网桥连接来停止捕获,然后停止第二个图形。
Bridge.BridgeGraphs(null,null);
MediaControl_SecondGraph.Stop();
我认为这涵盖了香港专业教育学院对GMF桥控制的发现:)
希望这符合Geraint Davies的标准
如果第二个图形的任何地方甚至有丝毫错误,则当您运行第二个图形时,它将停止第一个图形。这很好地表明出了点问题。如果给它一个无效的名称,如使Avi mux-> file writer尝试保存在非真实位置,则会停止图形
此代码将创建一个新窗口,并将视频粘贴在该新窗口中。要将视频流式传输到窗体的面板框中,您只需要添加4行代码即可。
IVideoWindow var = firstgraph as IVideoWindow();
var.put_Owner(panel1.handle);
var.put_windowstyle( windowstyle.child | windowstyle.clipchildren );
var.SetWindowPosition( panel1.clientrectangle.left,panel1.clientrectangle.top,panel1.clientrectangle.width,panel1.clientrectangle.height);
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。