Отображение потока веб-камеры в PyQt4 с помощью OpenCV Camera Capture

Я использую этот скрипт Python для отображения моей веб-камеры:

from opencv.cv import *  
from opencv.highgui import *  

import sys

cvNamedWindow("w1", CV_WINDOW_AUTOSIZE)
camera_index = 0
capture = cvCreateCameraCapture(camera_index)

def repeat():
    global capture #declare as globals since we are assigning to them now
    global camera_index
    frame = cvQueryFrame(capture)
    cvShowImage("w1", frame)
    c = cvWaitKey(10)

    if c == "q":
        sys.exit(0)

if __name__ == "__main__":
    while True:
        repeat()

Он работает довольно хорошо, но я хотел бы установить этот дисплей в своем приложении Qt. Как я могу использовать IplImage Изображение OpenCV в Qt VideoWidget?

3 ответа

Решение

Я использовал приведенный ниже код, чтобы скрыть объект Iplimage в QImage. Мне потребовалось некоторое время, чтобы получить правильные форматы. Iplimage - это 3-канальный формат с порядком каналов BGR, в то время как QImage использует порядок каналов RGB.

camcapture = cv.CaptureFromCAM(0)       
cv.SetCaptureProperty(camcapture,cv.CV_CAP_PROP_FRAME_WIDTH, 1280)
cv.SetCaptureProperty(camcapture,cv.CV_CAP_PROP_FRAME_HEIGHT, 720);

frame = cv.QueryFrame(camcapture)
image = QImage(frame.tostring(), frame.width, frame.height, QImage.Format_RGB888).rgbSwapped()
pixmap = QPixmap.fromImage(image)

Похоже, нам нужно связать этот код C++ в Python:

QImage& cvxCopyIplImage(const IplImage *pIplImage, QImage &qImage)
{
        if(!CV_IS_IMAGE(pIplImage)) return qImage;

        int w = pIplImage->width;
        int h = pIplImage->height;


        if(qImage.width() != w || qImage.height() != h)
        {
                qImage = QImage(w, h, QImage::Format_RGB32);
        }

        int x, y;
        for(x = 0; x < pIplImage->width; ++x)
        {
                for(y = 0; y < pIplImage->height; ++y)
                {
                        CvScalar color = cvGet2D(pIplImage, y, x);

                        if(pIplImage->nChannels == 1)
                        {
                                int v = color.val[0];

                                qImage.setPixel(x, y, qRgb(v,v,v));
                        }
                        else
                        {
                                int r = color.val[2];
                                int g = color.val[1];
                                int b = color.val[0];

                                qImage.setPixel(x, y, qRgb(r,g,b));
                        }
                }
        }

        if(pIplImage->origin != IPL_ORIGIN_TL)
        {
                qImage = qImage.mirrored(false, true);
        }

        return qImage;
}

Этот работал для меня. Я работаю на Linux с Python 2.6.5:

import base64
import Image
import time
import urllib2
import cv

# Basic HTTP Authentication...
url = 'http://192.168.0.11:82/Videostream.cgi'
ww = 'Username:Password'
encodedstring = base64.encodestring(ww)[:-1]
auth = "Basic %s" % encodedstring
req = urllib2.Request(url,None, {"Authorization": auth })
handle = urllib2.urlopen(req)

def read_stream():
    buf = ''
    b = handle.readlines(45)
    for a in b:
        if a.startswith('Content-Length'):
            readlen = str(a).split()[1]
    b1 = handle.read(int(readlen)+4)
    return b1

def test():
    pass

def write_stream():
    imgc = read_stream()
    cv_img = cv.CreateImageHeader((640,480), cv.IPL_DEPTH_8U, 3)
    buf = Image.fromstring('RGB',(640,480),imgc[2:], 'jpeg', 'RGB', None)
    cv.SetData(cv_img, buf.tostring(), (640*3))
    return cv_img


fps = 10.0

if __name__ == "__main__":
    while True:
        frame = write_stream()
        cv.ShowImage('Camera', frame)
        k = cv.WaitKey(10)
        time.sleep(int(1.0/fps))
        if k == 0x10001b: # ESC
            cv.DestroyWindow("Camera")
            print 'ESC pressed. Exiting ...'
            break
Другие вопросы по тегам