QT+opencv将图像显示到ui界面上(含资源)

前言

界面显示摄像头大概有两个方面,一个是定时器的使用,一个是数据类型的转换。界面的更新方式需要通过定时器访问,需要不断的释放摄像头资源,清理内存。数据类型就是QT ui界面常见问题,Mat转为QImage类型。
**

ui界面

首先就是新建一个ui界面,拉拉扯扯的做成自己喜欢的样子,我就简简单单拉扯成,,,这样:
ui界面

.Pro文件

原有的pro文件保持不变,QT的dll文件调用如下:

QT       += core gui
TARGET = XXX
TEMPLATE = app
CONFIG += c++11

greaterThan(QT_MAJOR_VERSION, 4): QT += widgets
DEFINES += QT_DEPRECATED_WARNINGS

opencv环境配置如下,我使用的是opencv3.4.0,用的MSVC2017-64bit编译器:

INCLUDEPATH +=D:\opencv\build\include
INCLUDEPATH +=D:\opencv\build\include\opencv
INCLUDEPATH +=D:\opencv\build\include\opencv2

win32:CONFIG(release, debug|release): LIBS += -L$$PWD/../../../opencv/build/x64/vc14/lib/ -lopencv_world340
else:win32:CONFIG(debug, debug|release): LIBS += -L$$PWD/../../../opencv/build/x64/vc14/lib/ -lopencv_world340d

INCLUDEPATH += $$PWD/../../../opencv/build/x64/vc14
DEPENDPATH += $$PWD/../../../opencv/build/x64/vc14

mainwindow.cpp

  1. 制作定时器
MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    timer = new QTimer(this);
    image = new QImage();
    connect(timer,SIGNAL(timeout()),this,SLOT(readFarme()));
}
  1. 数据类型转换
//Mat转成QImage
QImage MainWindow::MatImageToQt(const Mat &src)
{
    //CV_8UC1 8位无符号的单通道---灰度图片
    if(src.type() == CV_8UC1)
    {
        //使用给定的大小和格式构造图像
        //QImage(int width, int height, Format format)
        QImage qImage(src.cols,src.rows,QImage::Format_Indexed8);
        //扩展颜色表的颜色数目
        qImage.setColorCount(256);

        //在给定的索引设置颜色
        for(int i = 0; i < 256; i ++)
        {
            //得到一个黑白图
            qImage.setColor(i,qRgb(i,i,i));
        }
        //复制输入图像,data数据段的首地址
        uchar *pSrc = src.data;
        //
        for(int row = 0; row < src.rows; row ++)
        {
            //遍历像素指针
            uchar *pDest = qImage.scanLine(row);
            //从源src所指的内存地址的起始位置开始拷贝n个
            //字节到目标dest所指的内存地址的起始位置中
            memcmp(pDest,pSrc,src.cols);
            //图像层像素地址
            pSrc += src.step;
        }
        return qImage;
    }
    //为3通道的彩色图片
    else if(src.type() == CV_8UC3)
    {
        //得到图像的的首地址
        const uchar *pSrc = (const uchar*)src.data;
        //以src构造图片
        QImage qImage(pSrc,src.cols,src.rows,src.step,QImage::Format_RGB888);
        //在不改变实际图像数据的条件下,交换红蓝通道
        return qImage.rgbSwapped();
    }
    //四通道图片,带Alpha通道的RGB彩色图像
    else if(src.type() == CV_8UC4)
    {
        const uchar *pSrc = (const uchar*)src.data;
        QImage qImage(pSrc, src.cols, src.rows, src.step, QImage::Format_ARGB32);
        //返回图像的子区域作为一个新图像
        return qImage.copy();
    }
    else
    {
        return QImage();
    }
}

  1. 开启、关闭摄像头(定时器的使用、按钮的使用):
void MainWindow::on_Open_clicked()
{
    cap.open(0);
    timer->start(33);
}

void MainWindow::on_Stop_clicked()
{
    // 停止读取数据。
    timer->stop();
    cap.release();
    ui->cameraView->clear();
}

按钮转到槽的操作这里不多赘述,本教程只针对对QT界面略有了解的小白

  1. 取流,显示图像:
void MainWindow::readFarme()
{
    cap.read(src_image);

    QImage imag = MatImageToQt(src_image);
    ui->cameraView->setPixmap(QPixmap::fromImage(imag));
}

此时,每一帧图像都会不断覆盖src_image,显示在cameraView标签上

mainwindow.h

这里就简简单单定义一个类:

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    explicit MainWindow(QWidget *parent = nullptr);
    //Mat转QLabel
    QImage MatImageToQt(const Mat &src);
    ~MainWindow();

private slots:
    void readFarme();
    void on_Open_clicked();
    void on_Stop_clicked();

private:
    Ui::MainWindow *ui;
    VideoCapture cap;
    Mat src_image;
    QTimer *timer;
    QImage *image;
};

结语

这里使用的是cap.open()打开摄像头,所以界面只能打开免驱或者板载的摄像头,例如笔记本摄像头等,至于工业相机,只能根据工业相机自己的取流例程改了。
分装好的QT工程资源供大家免费下载:https://download.csdn.net/download/qq_46046959/18972862
GitHub

评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值