当前位置: 首页 > news >正文

Qt软件开发-摄像头检测使用软件V1.1

系列文章目录

Qt软件开发-摄像头检测使用软件V1.1


文章目录

  • 系列文章目录
  • 前言
  • 一、V1.1增加了哪些功能?
  • 二、代码构成
    • 1.总体结构
    • 2. 代码内容
  • 三、效果展示图
  • 总结


前言

之前,在Qt软件开发-摄像头检测使用软件:https://blog.csdn.net/xuming2044/article/details/145054460?spm=1011.2415.3001.5331,实现了一个简单的本地摄像头检测和使用软件,由于功能比较单一,只能实现两个功能:1、本地摄像头的检测、2、摄像头的使用。于是在此基础上,有了Qt软件开发-摄像头检测使用软件V1.1。


一、V1.1增加了哪些功能?

  1. 拍照功能:对当前画面进行拍照并保存到指定目录下;
  2. 录像功能:对当前正在显示的摄像头画面进行录屏,并以视频文件的形式保存到本地指定目录下;

二、代码构成

1.总体结构

在这里插入图片描述

2. 代码内容

main.cpp:

#include "mainwindow.h"#include <QApplication>int main(int argc, char *argv[])
{QApplication a(argc, argv);MainWindow w;w.show();return a.exec();
}

mainwindow.cpp:

#include "mainwindow.h"
#include "ui_mainwindow.h"MainWindow::MainWindow(QWidget *parent): QMainWindow(parent), ui(new Ui::MainWindow)
{ui->setupUi(this);cameras = QMediaDevices::videoInputs();//这里会返回所有可用的摄像头//qDebug() << "cameras.size:" <<cameras.size();for (const QCameraDevice &cameraDevice : cameras) {//将摄像头列表的描述的信息显示到lw_camerasui->lw_cameras->addItem(cameraDevice.description());}//cam = new QCamera(QMediaDevices::defaultVideoInput());//使用默认的集成摄像头//QCamera defCamera(QCameraDevice());//cap = new QMediaCaptureSession(this);//cap->setCamera(cam);cap.setVideoOutput(ui->video_widget);//显示画面到videoWidgetimage_cap.setFileFormat(QImageCapture::JPEG);//设置拍照保存的格式image_cap.setQuality(QImageCapture::VeryHighQuality);//设置拍照的质量cap.setImageCapture(&image_cap);//设置保存图像的对象media_format.setFileFormat(QMediaFormat::AVI);//设置视频的文件格式// media_format.setAudioCodec(QMediaFormat::AudioCodec::MP3);//设置音频编解码// media_format.setVideoCodec(QMediaFormat::VideoCodec::MPEG2);//设置视频编解码video_recorder.setMediaFormat(media_format);//初始化设置文件格式信息video_recorder.setQuality(QMediaRecorder::VeryHighQuality);//设置视频的质量为超高质量cap.setRecorder(&video_recorder);//设置保存录制视频的对象timer.setInterval(1000);//定时器每一秒触发超时mm = 0;ss = 0;//视频录制持续时间初始化为0image_path = "C:/Users/chang/Desktop/"; //保存照片的目录video_path = "C:/Users/chang/Desktop/"; //保存视频的目录//connect(&cap,SIGNAL(imageCaptureChanged()),this,SLOT(on_imageCaptureChanged()));//收到拍屏成功的信号之后保存图片到本地//connect(&cap,SIGNAL(screenCaptureChanged()),this,SLOT(on_screenCaptureChanged()));connect(&image_cap,SIGNAL(imageSaved(int,QString)),this,SLOT(on_imageSaved(int,QString)));//收到保存图片成功的信号之后显示到lb_imagecapconnect(&video_recorder,SIGNAL(errorOccured(QMediaRecorder::Error,QString)),this,SLOT(on_errorOccured(QMediaRecorder::Error,QString)));connect(&timer,SIGNAL(timeout()),this,SLOT(on_timeout()));//每秒触发超时,然后更新已录制视频的时间connect(&video_recorder,SIGNAL(recorderStateChanged(QMediaRecorder::RecorderState)),this,SLOT(on_recorderStateChanged(QMediaRecorder::RecorderState)));//根据recorder的实际状态进行相应操作//qDebug() << 0.1;
}MainWindow::~MainWindow()
{delete ui;
}void MainWindow::on_pb_display_clicked()//显示lw_cameras当前选择的摄像头画面到QVideoWidget
{for (const QCameraDevice &cameraDevice : cameras) {if(cameraDevice.description() == ui->lw_cameras->currentItem()->text()){cam.setCameraDevice(cameraDevice);cap.setCamera(&cam);}}cam.start();//启动摄像头//cap->camera()->activeChanged(true);//qDebug() << "cap->camera()->cameraDevice().position():" << cap->camera()->isAvailable();//qDebug() << "cap->camera()->cameraDevice().position():" << cap->camera()->cameraDevice().position();if(!cap.camera()->isActive()){// qDebug() << 2;cap.camera()->setActive(1);}
}void MainWindow::on_pb_stopDisplay_clicked()//停止显示摄像头画面到QVideoWidget
{cam.stop();//关闭摄像头
}void MainWindow::on_pb_quit_clicked()//退出软件
{close();
}void MainWindow::on_lw_cameras_itemClicked(QListWidgetItem *item)//单机选中当前项目
{ui->lw_cameras->setCurrentItem(item);
}void MainWindow::on_lw_cameras_itemDoubleClicked(QListWidgetItem *item)//双击直接开启选中摄像头并显示画面
{ui->lw_cameras->setCurrentItem(item);on_pb_display_clicked();
}void MainWindow::on_pb_imagecap_clicked()//拍照
{//cap.imageCapture();QString file_name = image_path + QDateTime::currentDateTime().toString("yyyyMMddhhmmsszzz");qDebug() << file_name;if(image_cap.isReadyForCapture()){image_cap.captureToFile(file_name);}else{ui->lb_imagecap->setText("Image Capture is not for capture");}
}void MainWindow::on_imageSaved(int id, const QString &fileName)
{if(id != -1){ui->lb_imagecap->setText(QString("文件保存成功,文件ID:%1,文件名:%2").arg(id).arg(fileName));}else{ui->lb_imagecap->setText(QString("文件保存失败,文件ID:%1,文件名:%2").arg(id).arg(fileName));}
}void MainWindow::on_pb_videocap_clicked()//开始、停止录像
{if(video_recorder.recorderState() == QMediaRecorder::StoppedState){QString file_dst = video_path + QDateTime::currentDateTime().toString("yyyyMMddhhmmsszzz");video_recorder.setOutputLocation(QUrl::fromLocalFile(file_dst));//QUrl file_dst(video_path + QDateTime::currentDateTime().toString("yyyyMMddhhmmsszzz"));//video_recorder.setOutputLocation(file_dst);qDebug() << QUrl::fromLocalFile(file_dst);video_recorder.record();}else if(video_recorder.recorderState() == QMediaRecorder::RecordingState){video_recorder.stop();}else if(video_recorder.recorderState() == QMediaRecorder::PausedState){//未实现状态}
}void MainWindow::on_errorOccured(QMediaRecorder::Error error, const QString &errorString)
{QMessageBox::critical(this,"错误",QString("errorValue:%1\nerrorString:%2").arg(error).arg(errorString));
}void MainWindow::on_recorderStateChanged(QMediaRecorder::RecorderState state)
{if(state == QMediaRecorder::RecordingState){QString time = QString("%1:%2").arg(mm,2,10,QChar('0')).arg(ss,2,10,QChar('0'));//将每个数字转换为格式:(值,宽度,进制,填充符号)ui->lb_videocap->setText(time);//显示开始录制的时间timer.start();//启动定时器ui->pb_videocap->setText("停止录像");}else if(state == QMediaRecorder::StoppedState){timer.stop();//关闭定时器mm = 0;ss = 0;//录制完后时间清零,为下次录制时间做准备ui->pb_videocap->setText("开始录像");}else if(state == QMediaRecorder::PausedState){//...}
}void MainWindow::on_timeout()//每秒被定时器信号触发,更新已录制的时间
{if(++ss == 60)//秒先加一{ss = 0;mm++;}QString time = QString("%1:%2").arg(mm,2,10,QChar('0')).arg(ss,2,10,QChar('0'));//将每个数字转换为格式:(值,宽度,进制,填充符号)ui->lb_videocap->setText(time);//显示开始录制的时间
}

三、效果展示图

  1. 拍照:
    在这里插入图片描述

  2. 录屏:
    在这里插入图片描述

总结

  1. V1.1版本是对最初版本的功能添加,V1.1和windows系统自带相机软件唯一的区别:windows相机似乎只能使用电脑自带的摄像头,而V1.1可以任意选择所有通过USB连接到电脑上的摄像头;
  2. 该版本录制的视频未实现收音的功能,只有纯画面展示,如有需求可自行研究;
  3. 下一个版本可以考虑将摄像头画面实时推流到远程主机上,并实现由远端用户来选择查看哪个摄像头的画面。
http://www.xdnf.cn/news/155881.html

相关文章:

  • python 与Redis操作整理
  • 血泪之arduino库文件找不到ArduinoJSON.h: No such file or directory错误原因
  • 学习记录:DAY18
  • AI日报 - 2025年04月26日
  • Yocto项目实战教程-第8章-树莓派启动定制镜像-8.4小节-使用Wic工具创建分区镜像
  • 毕业项目-基于java的入侵检测与防御系统
  • 字节 AI 原生 IDE Trae 发布 v1.3.0,新增 MCP 支持
  • 使用MyBatis注解方式的完整示例,涵盖CRUD、动态SQL、分页、事务管理等场景,并附详细注释和对比表格
  • Java爬虫入门:从网页抓取到数据提取(正则表达式篇)
  • 单例设计模式之懒汉式以及线程安全问题
  • 【计算机视觉】CV项目实战- 深度解析TorchVision_Maskrcnn:基于PyTorch的实例分割实战指南
  • 从“拼凑”到“构建”:大语言模型系统设计指南!
  • 【Vue】Vue3项目创建
  • 美团Java后端二面面经!
  • 【数论分块】数论分块算法模板及真题
  • # 家庭网络IPv6地址的一些知识
  • 思科路由器重分发(静态路由+OSPF动态路由+RIP动态路由)
  • 基于MTF的1D-2D-CNN-BiLSTM-Attention时序图像多模态融合的故障分类识别(Matlab完整源码和数据),适合研究学习,附模型研究报告
  • Leetcode刷题 由浅入深之哈希法——454. 四数相加Ⅱ
  • Logi Options+ 的 Flow:端口信息
  • 驱动开发(1)|鲁班猫rk356x内核编译,及helloworld驱动程序编译
  • 微信小程序核心技术栈
  • ORACLE数据库备份入门:第四部分:2-备份场景举例
  • 计算机视觉——对比YOLOv12、YOLOv11、和基于Darknet的YOLOv7的微调对比
  • MyBatis 官方子项目详细说明及表格总结
  • JavaScript基础知识合集笔记1——数据类型
  • TDengine 中的压缩设计
  • 毕业项目-Web入侵检测系统
  • 关于TCP三次握手和四次挥手的疑点
  • 游戏状态管理:用Pygame实现场景切换与暂停功能