1

嗨,我正在尝试使用ISampleGrabberCB::SampleCB来从实时预览中获取图像,然后再将它们显示在我的表单中。

我希望能够将每个新帧转换为要处理的位图(例如扫描+添加水印类型的图像)。

目前我正在尝试通过以下方式进行操作:

int ISampleGrabberCB.SampleCB(double SampleTime, IMediaSample sample)
    {
        int hr;
        IntPtr buffer;
        AMMediaType mediaType;
        VideoInfoHeader videoInfo;
        int frameWidth;
        int frameHeight;
        int stride;
        int bufferLength;


        hr = sample.GetPointer(out buffer);
        DsError.ThrowExceptionForHR(hr);

        hr = sample.GetMediaType(out mediaType);
        DsError.ThrowExceptionForHR(hr);

        bufferLength = sample.GetSize();

        try
        {
            videoInfo = new VideoInfoHeader();
            Marshal.PtrToStructure(mediaType.formatPtr, videoInfo);

            frameWidth = videoInfo.BmiHeader.Width;
            frameHeight = videoInfo.BmiHeader.Height;
            stride = frameWidth * (videoInfo.BmiHeader.BitCount / 8);

            CopyMemory(imageBuffer, buffer, bufferLength);                

            Bitmap bitmapOfFrame = new Bitmap(frameWidth, frameHeight, stride, PixelFormat.Format24bppRgb, buffer);
            bitmapOfFrame.Save("C:\\Users\\...\\...\\...\\....jpg");


        }
        catch (Exception ex)
        {
            Console.WriteLine(ex.ToString());
        }

        return 0;
    }

理论上这应该得到媒体类型,然后用于获取图像的宽度、高度和步幅,然后用于创建位图。然后从 IMediaSample 的指针中获取缓冲区。

但是,这似乎不起作用(我假设这是因为位图永远不会保存)。那么我将如何将每个新帧转换为位图呢?

设置引脚的附加功能:

public void setupGraphForSampleGrabber(DsDevice webcamDevice, Control displayBox)
    {
        int hr;
        ISampleGrabber sampleGrabber = null;
        IPin capturePin = null;
        IPin samplePin = null;
        IPin renderPin = null;
        IBaseFilter captureFilter;
        filtergraph = new FilterGraph() as IFilterGraph2;

        try
        {


            //Add the webcam
            hr = filtergraph.AddSourceFilterForMoniker(webcamDevice.Mon, null, webcamDevice.Name, out captureFilter);
            DsError.ThrowExceptionForHR(hr);

            //Get the still pin
            stillPin = DsFindPin.ByCategory(captureFilter, PinCategory.Still, 0);

            if (stillPin == null)
            {
                stillPin = DsFindPin.ByCategory(captureFilter, PinCategory.Preview, 0);
            }

            if (stillPin == null)
            {
                IPin outputPin = null;
                IPin inputPin = null;

                //As there is still no still pin set this to null
                videoControl = null;

                // Add a splitter
                IBaseFilter smartTee = (IBaseFilter)new SmartTee();

                try
                {
                    hr = filtergraph.AddFilter(smartTee, "SmartTee");
                    DsError.ThrowExceptionForHR(hr);

                    //Obtain the capture pin from the webcam and the input pin from the spliter and assign them to the outputPin and inputPin respectivly
                    outputPin = DsFindPin.ByCategory(captureFilter, PinCategory.Capture, 0);
                    inputPin = DsFindPin.ByDirection(smartTee, PinDirection.Input, 0);

                    //Then connect both of them to the graph
                    hr = filtergraph.Connect(outputPin, inputPin);
                    DsError.ThrowExceptionForHR(hr);

                    //Set the capture and still pins so we can use them with the rest of the program
                    stillPin = DsFindPin.ByName(smartTee, "Preview");
                    capturePin = DsFindPin.ByName(smartTee, "Capture");

                    setParameters(outputPin);

                }
                    //Release all the com objects to avoid problems as the program is to be used for extended periods

                finally
                {
                    if (outputPin != null)
                    {
                        Marshal.ReleaseComObject(outputPin);
                    }
                    if (outputPin != inputPin)
                    {
                        Marshal.ReleaseComObject(inputPin);
                    }
                    if (outputPin != smartTee)
                    {
                        Marshal.ReleaseComObject(smartTee);
                    }
                }
            }
            else
            {
                videoControl = captureFilter as IAMVideoControl;
                capturePin = DsFindPin.ByCategory(captureFilter, PinCategory.Capture, 0);
                setParameters(stillPin);
            }

            //Get interface
            sampleGrabber = new SampleGrabber() as ISampleGrabber;

            //Configure the samplegrabber
            IBaseFilter baseFilter = sampleGrabber as IBaseFilter;
            configureSampleGrabber(sampleGrabber);
            samplePin = DsFindPin.ByDirection(baseFilter, PinDirection.Input, 0);

            //Video Renderer
            IBaseFilter render = new VideoRendererDefault() as IBaseFilter;
            hr = filtergraph.AddFilter(render, "Renderer");
            DsError.ThrowExceptionForHR(hr);

            renderPin = DsFindPin.ByDirection(render, PinDirection.Input, 0);

            //Add samplegrabber to graph
            hr = filtergraph.AddFilter(baseFilter, "SampleGrabber");
            DsError.ThrowExceptionForHR(hr);

            if (videoControl == null)
            {
                //Connect still pin to samplegrabber
                hr = filtergraph.Connect(stillPin, samplePin);
                DsError.ThrowExceptionForHR(hr);

                //Connect capture pin to render
                hr = filtergraph.Connect(capturePin, renderPin);
                DsError.ThrowExceptionForHR(hr);
            }
            else
            {
                //Connect capture pin to render
                hr = filtergraph.Connect(capturePin, renderPin);
                DsError.ThrowExceptionForHR(hr);

                //Connect still pin to samplegrabber
                hr = filtergraph.Connect(stillPin, samplePin);
                DsError.ThrowExceptionForHR(hr);
            }

            //Get video properties
            saveVideoInfo(sampleGrabber);
            ConfigureVideoLocation(displayBox);

            //Run Graph
            IMediaControl mediaControl = filtergraph as IMediaControl;
            hr = mediaControl.Run();
            DsError.ThrowExceptionForHR(hr);


        }
        finally
        {
            if (sampleGrabber != null)
            {
                Marshal.ReleaseComObject(sampleGrabber);
                sampleGrabber = null;
            }
            if (capturePin != null)
            {
                Marshal.ReleaseComObject(capturePin);
                capturePin = null;
            }
            if (renderPin != null)
            {
                Marshal.ReleaseComObject(renderPin);
                renderPin = null;
            }
            if (samplePin != null)
            {
                Marshal.ReleaseComObject(samplePin);
                samplePin = null;
            }
        }
    }

此外,我像这样配置我的 samplegrabber:

    public void configureSampleGrabber(ISampleGrabber sampleGrabber)
    {
        int hr;
        AMMediaType mediaType = new AMMediaType();

        //Set the values for media type and format
        mediaType.majorType = MediaType.Video;
        mediaType.subType = MediaSubType.RGB24;
        mediaType.formatType = FormatType.VideoInfo;
        hr = sampleGrabber.SetMediaType(mediaType);
        DsError.ThrowExceptionForHR(hr);

        DsUtils.FreeAMMediaType(mediaType);
        mediaType = null;

        //Configure
        hr = sampleGrabber.SetCallback(this, 0);
        DsError.ThrowExceptionForHR(hr);

    }
4

2 回答 2

1

Sample Grabber 不会为每个抓取的样本提供正确的 MediaType,所以不要费心从这里的样本中请求它。取而代之的是获取抓取器的输入引脚或摄像头的输出引脚并调用 IPin.ConnectionMediaType() 以了解此流的正确媒体类型。

于 2013-10-31T07:10:11.487 回答
1

我认为您的图表设置不正确,并且样本从未通过样本采集器。乍一看,我认为样本采集器永远不会被渲染。将样本采集器输出引脚连接到空渲染器,一切正常。

于 2013-11-01T18:57:48.533 回答