时隔一年多,重新对视频监控系统的onvif内核重写,一方面为了兼容Qt6,一方面按功能分类提高效率。整体逻辑思路是一样的,主要的改动是由于Qt6不再支持QtXmlPatterns模块(其实这个模块在Qt5的后面的版本也逐渐提示为废弃模块),onvif协议通信中的数据都是带有命名空间的xml数据,用QtXmlPatterns模块去解析是最合适的,现在全部改成了用最原始最基础的QtXml模块去解析,毕竟QtXml模块肯定是一直在的,这是相当基础的模块,无论以后Qt7还是Qt100肯定都会有。
之前云台控制这块,已经实现了相对移动和绝对移动,后面在测试了几十种厂家的摄像机以后,发现有部分摄像机厂家对onvif协议支持不完全,打通了部分协议,比如云台控制这块就只支持连续移动,其实真的场景也绝大部分都是需要连续移动的,看起来相当丝滑,之前在没有实现连续移动命令的时候,云台控制默认用的是相对移动,为了实现连续移动的效果,搞了个定时器来不断执行命令模拟连续移动,效果不是很理想,看起来怪怪的,有时候停顿,近期才发现原来有连续移动的指令,这就舒服了,丝滑到爆,按下对应的方位触发连续移动,松开自动停止。
onvif主要的功能
onvif的处理流程
#include "frmipcptz.h"
#include "ui_frmipcptz.h"
#include "quihelper.h"
#include "devicehelper.h"
#include "deviceonvif.h"
frmIpcPtz::frmIpcPtz(QWidget *parent) : QWidget(parent), ui(new Ui::frmIpcPtz)
{
ui->setupUi(this);
this->initForm();
this->initIcon();
}
frmIpcPtz::~frmIpcPtz()
{
delete ui;
}
bool frmIpcPtz::eventFilter(QObject *watched, QEvent *event)
{
//鼠标按下指定松开自动停止
if (event->type() == QEvent::MouseButtonPress) {
int type = 255;
if (watched == ui->btnPtzZoomIn) {
type = 9;
} else if (watched == ui->btnPtzZoomOut) {
type = 10;
} else if (watched == ui->btnPtzFocusIn) {
type = 11;
} else if (watched == ui->btnPtzFocusOut) {
type = 12;
} else if (watched == ui->btnPtzApertureIn) {
type = 13;
} else if (watched == ui->btnPtzApertureOut) {
type = 14;
}
QMetaObject::invokeMethod(this, "ptzControl", Qt::QueuedConnection, Q_ARG(int, type));
} else if (event->type() == QEvent::MouseButtonRelease) {
QMetaObject::invokeMethod(this, "ptzControl", Qt::QueuedConnection, Q_ARG(int, 255));
}
return QWidget::eventFilter(watched, event);
}
void frmIpcPtz::initForm()
{
//安装事件过滤器识别松开自动停止
ui->btnPtzZoomIn->installEventFilter(this);
ui->btnPtzZoomOut->installEventFilter(this);
//绑定云台仪表盘按下和松开事件
ui->gaugeCloud->setAutoRepeat(false);
connect(ui->gaugeCloud, SIGNAL(mousePressed(int, QString)), this, SLOT(mousePressed(int, QString)));
connect(ui->gaugeCloud, SIGNAL(mouseReleased(int, QString)), this, SLOT(mouseReleased(int, QString)));
//关联样式改变信号自动重新设置图标
connect(AppEvent::Instance(), SIGNAL(changeStyle()), this, SLOT(initIcon()));
}
void frmIpcPtz::initIcon()
{
//设置图形字体
int fontSize = 15;
IconHelper::setIcon(ui->btnPtzZoomIn, 0xf067, fontSize);
IconHelper::setIcon(ui->btnPtzZoomOut, 0xf068, fontSize);
IconHelper::setIcon(ui->btnPtzFocusIn, 0xf067, fontSize);
IconHelper::setIcon(ui->btnPtzFocusOut, 0xf068, fontSize);
IconHelper::setIcon(ui->btnPtzApertureIn, 0xf067, fontSize);
IconHelper::setIcon(ui->btnPtzApertureOut, 0xf068, fontSize);
}
void frmIpcPtz::mousePressed(int position, const QString &strPosition)
{
DeviceHelper::addMsg(QString("按下云台 %1").arg(strPosition));
QMetaObject::invokeMethod(this, "ptzControl", Qt::QueuedConnection, Q_ARG(int, position));
}
void frmIpcPtz::mouseReleased(int position, const QString &strPosition)
{
DeviceHelper::addMsg(QString("松开云台 %1").arg(strPosition));
QMetaObject::invokeMethod(this, "ptzControl", Qt::QueuedConnection, Q_ARG(int, 255));
}
void frmIpcPtz::ptzControl(int position)
{
//计算速度,转为小数
double step = (double)ui->sliderPtzStep->value() / 10;
//根据按下的不同部位发送云台控制命令
//1. x、y、z 范围都在0-1之间。
//2. x为负数,表示左转,x为正数,表示右转。
//3. y为负数,表示下转,y为正数,表示上转。
//4. z为正数,表示拉近,z为负数,表示拉远。
//5. 通过x和y的组合,来实现云台的控制。
//6. 通过z的组合,来实现焦距控制。
//0-8依次表示底部/左下角/左侧/左上角/顶部/右上角/右侧/右下角/中间
if (position == 0) {
DeviceOnvif::ptzControl(3, AppData::CurrentUrl, 0.0, -step, 0.0);
} else if (position == 1) {
DeviceOnvif::ptzControl(3, AppData::CurrentUrl, -step, -step, 0.0);
} else if (position == 2) {
DeviceOnvif::ptzControl(3, AppData::CurrentUrl, -step, 0.0, 0.0);
} else if (position == 3) {
DeviceOnvif::ptzControl(3, AppData::CurrentUrl, -step, step, 0.0);
} else if (position == 4) {
DeviceOnvif::ptzControl(3, AppData::CurrentUrl, 0.0, step, 0.0);
} else if (position == 5) {
DeviceOnvif::ptzControl(3, AppData::CurrentUrl, step, step, 0.0);
} else if (position == 6) {
DeviceOnvif::ptzControl(3, AppData::CurrentUrl, step, 0.0, 0.0);
} else if (position == 7) {
DeviceOnvif::ptzControl(3, AppData::CurrentUrl, step, -step, 0.0);
} else if (position == 8) {
//先停止再复位
DeviceOnvif::ptzControl(0, AppData::CurrentUrl, 0.0, 0.0, 0.0);
//DeviceOnvif::ptzControl(1, AppData::CurrentUrl, 0.0, 0.0, 0.0);
} else if (position == 9) {
//变倍+
DeviceOnvif::ptzControl(3, AppData::CurrentUrl, 0.0, 0.0, step);
} else if (position == 10) {
//变倍-
DeviceOnvif::ptzControl(3, AppData::CurrentUrl, 0.0, 0.0, -step);
} else if (position == 11) {
//变焦+
} else if (position == 12) {
//变焦-
} else if (position == 13) {
//光圈+
} else if (position == 14) {
//光圈-
} else {
//停止
DeviceOnvif::ptzControl(0, AppData::CurrentUrl, 0.0, 0.0, 0.0);
}
}
原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。
如有侵权,请联系 cloudcommunity@tencent.com 删除。
原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。
如有侵权,请联系 cloudcommunity@tencent.com 删除。
扫码关注腾讯云开发者
领取腾讯云代金券
Copyright © 2013 - 2025 Tencent Cloud. All Rights Reserved. 腾讯云 版权所有
深圳市腾讯计算机系统有限公司 ICP备案/许可证号:粤B2-20090059 深公网安备号 44030502008569
腾讯云计算(北京)有限责任公司 京ICP证150476号 | 京ICP备11018762号 | 京公网安备号11010802020287
Copyright © 2013 - 2025 Tencent Cloud.
All Rights Reserved. 腾讯云 版权所有