Directshow & .Net - Растровое изображение показывает полосу справа на левой стороне изображения? - PullRequest
2 голосов
/ 30 августа 2009

Образец изображения: alt text

Я использую DirectShow.net, чтобы записать видео с веб-камеры в свою программу. Для этого я добавляю исходную камеру на график и VideoMixingRenderer9.

Эта часть работает плавно, но часть, где я извлекаю кадр с помощью GetCurrentImage (из lpDib), имеет то, что я могу описать только как странную проблему.

Я использую Marshal.PtrToSTructure для создания BitmapInfoHeader из lpDib, а затем вычисляю формат ширины / высоты / шага / & пикселя.

Проблема возникает, когда я смотрю на изображение, хранящееся в растровом изображении - у него есть линия шириной 10 пикселей вниз по левой стороне, полученная от того, что на самом деле является правым!

Стоит отметить, что данные, которые я получаю от вызова GetCurrentImage, фактически перевернуты - обратите внимание на вызов Cap.RotateFlip.

IntPtr lpDib;
windowlessCtrl.GetCurrentImage(out lpDib);

BitmapInfoHeader head;
head = (BitmapInfoHeader)Marshal.PtrToStructure(lpDib, typeof(BitmapInfoHeader));
int width = head.Width;
int height = head.Height;
int stride = width * (head.BitCount / 8);
PixelFormat pixelFormat = PixelFormat.Format24bppRgb;

switch (head.BitCount)
{
    case 24: pixelFormat = PixelFormat.Format24bppRgb; break;
    case 32: pixelFormat = PixelFormat.Format32bppRgb; break;
    case 48: pixelFormat = PixelFormat.Format48bppRgb; break;
    default: throw new Exception("Unknown BitCount");
}

Cap = new Bitmap(width, height, stride, pixelFormat, lpDib);
Cap.RotateFlip(RotateFlipType.RotateNoneFlipY);
//if we examine Cap here (Cap.Save, for example) I'm seeing the odd stripe.

Я полностью потерян здесь. Похоже на какую-то проблему смещения, и я попытался немного поработать, но безрезультатно (просто создает странный диагональный вид).

Ответы [ 3 ]

2 голосов
/ 04 июня 2013

Этот код сделан с использованием примеров DirectShowLib и работает:

public Bitmap GetCurrentImage()
        {
            Bitmap bmp = null;
            if (windowlessCtrl != null)
            {
                IntPtr currentImage = IntPtr.Zero;

                try
                {
                    int hr = windowlessCtrl.GetCurrentImage(out currentImage);
                    DsError.ThrowExceptionForHR(hr);

                    if (currentImage != IntPtr.Zero)
                    {
                        BitmapInfoHeader structure = new BitmapInfoHeader();
                        Marshal.PtrToStructure(currentImage, structure);

                        PixelFormat pixelFormat = PixelFormat.Format24bppRgb;
                        switch (structure.BitCount)
                        {
                            case 24:
                                pixelFormat = PixelFormat.Format24bppRgb;
                                break;
                            case 32:
                                pixelFormat = PixelFormat.Format32bppRgb;
                                break;
                            case 48:
                                pixelFormat = PixelFormat.Format48bppRgb;
                                break;
                            default:
                                throw new Exception("BitCount desconhecido");
                        }

                        // este trecho: new IntPtr(currentImage.ToInt64() + 40), é o que resolve o problema da faixa (strip) da direita na esquerda.
                        bmp = new Bitmap(structure.Width, structure.Height, (structure.BitCount / 8) * structure.Width, pixelFormat, new IntPtr(currentImage.ToInt64() + 40));
                        bmp.RotateFlip(RotateFlipType.RotateNoneFlipY);
                    }
                }
                catch (Exception anyException)
                {
                    MessageBox.Show("Falha gravando imagem da Webcam: " + anyException.ToString());
                }
                finally
                {
                    Marshal.FreeCoTaskMem(currentImage);
                }
            }
            return bmp;
        }
1 голос
/ 31 августа 2009

Рендерер видео расширяет растровое изображение в соответствии со своими потребностями выравнивания памяти, но он настроит тип носителя в соответствии с.VIDEOINFOHEADER (или структура VIDEOINFOHEADER2) в типе носителя будет иметь прямоугольник rcTarget, который определяет действительную область в пределах большего растрового изображения.Вы можете запросить текущий тип мультимедиа на входном выводе и получить эту информацию.

Вы обнаружите, что визуализатору нужен только этот расширенный шаг для некоторых форматов, поэтому, возможно, ваш самый простой подход состоит в том, чтобы вызвать другой захватформат.Альтернативой является использование фильтра захвата образца вместо VMR.

G

0 голосов
/ 02 июня 2016

Для тех, кто хочет избежать использования SampleGrabber. Проблема с «полосой» может быть исправлена ​​путем добавления смещения заголовка растрового изображения в IntPtr. Однако для этого требуется небезопасный код

    IntPtr pBuffer = IntPtr.Zero;
    int xBufferSize = 0;
    int xWidth, xHeight;

    basicVideo.get_VideoWidth(out xWidth);
    basicVideo.get_VideoHeight(out xHeight);

    int hr = basicVideo.GetCurrentImage(ref xBufferSize, IntPtr.Zero);
    pBuffer = Marshal.AllocCoTaskMem(xBufferSize);

    // Get the pixel buffer for the thumbnail
    hr = basicVideo.GetCurrentImage(ref xBufferSize, pBuffer);

    // Offset for BitmapHeader info
    var bitmapHeader = (BitmapInfoHeader)Marshal.PtrToStructure(pBuffer, typeof(BitmapInfoHeader));
    var pBitmapData = (byte*)pBuffer.ToPointer();
    pBitmapData += bitmapHeader.Size;

    // This will be the pointer to the bitmap pixels
    var bitmapData = new IntPtr(pBitmapData);

    //Change for your format type!
    System.Drawing.Imaging.PixelFormat xFormat = (System.Drawing.Imaging.PixelFormat.Format32bppRgb);

    int bitsPerPixel = ((int)xFormat & 0xff00) >> 8;
    int bytesPerPixel = (bitsPerPixel + 7) / 8;
    int stride = 4 * ((xWidth * bytesPerPixel + 3) / 4);

    Bitmap image = new Bitmap(xWidth, xHeight, stride, xFormat, bitmapData);
    image.RotateFlip(RotateFlipType.RotateNoneFlipY);
    return image;

Расчет для Stride можно найти здесь .

...