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

Qt触摸屏双指缩放和单指移动界面(支持嵌入式设备)

本文介绍的QGraphicsView的双指缩放,QWidget更简单,可以参考当前内容。
方法一:(QTouchEvent事件实现)
使用场景:适用于paintevent绘制下的界面。
优点:不需要代码设置中心锚点(锚点:视图变化期间通过此点定位场景)。
缺点:界面上所有其它操作无法响应,需单独做处理才能做相应。(无法响应原因是使用了”return true“打断了”触摸点击“转化为”鼠标点击“。但是不使用”retuen true“又会造成”触摸点击“默认转化成了”鼠标点击“,触摸操作无法过度到TouchUpdate中来捕获到多点触摸了。
使用步骤:
1、首先需要打开触摸屏功能。
this->setAttribute(Qt::WA_AcceptTouchEvents);
2、在event事件管理器中接收触摸屏的三个事件,TouchBegin、TouchUpdate和TouchEnd。
3、判断单点触摸还是多点触摸。
4、如果多点触摸,通过比较前后两次两点间触摸位置来判断是放大还是缩小。
5、多点触摸时,会存在抖动情况,需要做防抖处理。
6、如果是单点触摸,通过比较手指放上去的位置和手指拖动时的位置来设置界面滚动条的位置。

代码:
属性设置

    this->setAttribute(Qt::WA_AcceptTouchEvents);

缩放和移动逻辑

bool MGraaphicsView::event(QEvent *e)
{static int index = 0;switch (e->type()) {case QEvent::TouchBegin:case QEvent::TouchUpdate:case QEvent::TouchEnd:{qDebug() <<"CProjectionPicture::event"<<e->type();QTouchEvent *touchEvent = static_cast<QTouchEvent *>(e);QList<QTouchEvent::TouchPoint> touchPoints = touchEvent->touchPoints();if (touchPoints.count() == 2) {//缩放const QTouchEvent::TouchPoint &touchPoint0 = touchPoints.first();const QTouchEvent::TouchPoint &touchPoint1 = touchPoints.last();qreal currentScaleFactor = QLineF(touchPoint0.pos(), touchPoint1.pos()).length()/ QLineF(touchPoint0.startPos(), touchPoint1.startPos()).length();if (currentScaleFactor  > _lastScaleFactor)index++;else if (currentScaleFactor  < _lastScaleFactor)index--;if (index == 5)//超过5次放大,则认为有效.防抖操作{index = 0;zoomOnce(true);}else if (index == -5){index = 0;zoomOnce(false);}qDebug()<<index<<currentScaleFactor<<_lastScaleFactor;_lastScaleFactor = currentScaleFactor;update();}else if (touchPoints.count() == 1){//移动 const QTouchEvent::TouchPoint &touchPoint = touchPoints.first();if (e->type() == QEvent::TouchBegin || e->type() == QEvent::TouchEnd)_startScenePos = mapToScene(touchPoint.pos().toPoint());if (e->type() == QEvent::TouchUpdate){QPointF endScenePos = mapToScene(touchPoint.pos().toPoint());QPointF delta = endScenePos - _startScenePos;int oposx = this->horizontalScrollBar()->value();int oposy = this->verticalScrollBar()->value();int nposx = oposx - delta.x();int nposy = oposy - delta.y();this->horizontalScrollBar()->setValue(nposx);this->verticalScrollBar()->setValue(nposy);}qDebug()<<"====="<<_startScenePos<<touchPoint.pos();}if (touchEvent->touchPointStates() & Qt::TouchPointReleased){qDebug()<<"Qt::TouchPointReleased";}return true;//一定不要调QGraphicsView::event(e);否则手指触摸会经常失效}default:break;}return QGraphicsView::event(e);
}

//缩放

void MGraaphicsView::zoomOnce(bool increase)
{if (increase)setZoom(1);elsesetZoom(-1);
}void MGraaphicsView::setZoom(int val)
{if (val > 0){m_zoom++;auto scaleValue = qPow(2, m_zoom);setTransform(QTransform::fromScale(scaleValue, scaleValue));}else{m_zoom--;auto scaleValue = qPow(2, m_zoom);setTransform(QTransform::fromScale(scaleValue, scaleValue));}
}

方法二:(QGesture事件实现)
使用场景:适用所有场景。
优点:界面上界面上所有操作都不受影响。
缺点:需要代码定位视图锚点。(QGraphicsView需要定位视图锚点,如果是QWidget则不需要定位锚点这一步,缩放大小和位置代码设置即可。)
使用步骤:
1、首先需要打开触摸屏功能和注册缩放手势。
grabGesture(Qt::PinchGesture);//”捏“手势
this->setAttribute(Qt::WA_AcceptTouchEvents);

所有手势介绍:
enum GestureType
{TapGesture        = 1, //轻拍手势。(1个手指单击)TapAndHoldGesture = 2,  //轻触并保持(长按)手势。(1个手指单击并长按)PanGesture        = 3,  //平移手势。(1个手指拖动)PinchGesture      = 4, //捏合缩放及缩放(2个手指捏合或转动)SwipeGesture      = 5, //滑动手势。(3个手指平移)CustomGesture     = 0x0100,  //可用于测试手势是否为用户定义的手势ID的标志。
};

自定义手势
2、在event事件管理器中接收QEvent::Gesture事件,并转化为QGestureEvent事件。
3、获取”捏“手势,并转化为QPinchGesture事件。
4、获取”捏“手势变化状态。
5、判断捏手势变化状态,根据QPinchGesture::ScaleFactorChanged状态变化计算缩放比例。
6、在GestureUpdated的变化下,根据缩放比例来处理缩小和扩大。
代码:

bool InteractiveMap::event(QEvent *e)
{static int index = 0;
#if 1if (e->type() == QEvent::Gesture){QGestureEvent* gEvent = static_cast<QGestureEvent *>(e);if (QGesture* pinch = gEvent->gesture(Qt::PinchGesture)){QPinchGesture* pEvent = static_cast<QPinchGesture *>(pinch);QPinchGesture::ChangeFlags changeFlags = pEvent->changeFlags();//旋转角度的变化记录if (changeFlags & QPinchGesture::RotationAngleChanged){
//                qreal rotationDelta = pEvent->rotationAngle() - pEvent->lastRotationAngle();
//                qDebug() << "pinchTriggered(): rotate by" << rotationDelta;}static qreal s_factor = 1.;//缩放比例的变化记录if (changeFlags & QPinchGesture::ScaleFactorChanged){s_factor *= pEvent->totalScaleFactor();}int id = (int)pEvent->state();switch (id){case Qt::GestureStarted:case Qt::GestureUpdated:{//视图在变换期间应如何定位场景。//QGraphicsView使用此属性来决定在变换矩阵发生变化以及视图的坐标系发生变换时如何在视口中定位场景。//默认行为AnchorViewCenter可确保视图中心的场景点在变换期间保持不变(例如,旋转时,场景将显示为围绕视图中心旋转)。setTransformationAnchor(QGraphicsView::AnchorViewCenter);  //视图中心的场景点用作锚点if (s_factor > 1)index++;elseindex--;if (index == 5) //超过5次放大,则认为有效.防抖操作{index = 0;zoomOnce(true);}else if (index == -5){index = 0;zoomOnce(false);}setTransformationAnchor(QGraphicsView::AnchorUnderMouse);  //鼠标下方的点用作锚点。}break;case Qt::GestureFinished:default:{s_factor = 1.;}break;}}}return QGraphicsView::event(e);
}
void MGraaphicsView::zoomOnce(bool increase)
{if (increase)setZoom(1);elsesetZoom(-1);
}void MGraaphicsView::setZoom(int val)
{if (val > 0){m_zoom++;auto scaleValue = qPow(2, m_zoom);setTransform(QTransform::fromScale(scaleValue, scaleValue));}else{m_zoom--;auto scaleValue = qPow(2, m_zoom);setTransform(QTransform::fromScale(scaleValue, scaleValue));}
}
http://www.lryc.cn/news/209694.html

相关文章:

  • 【Linux】虚拟机安装Linux、客户端工具,MobaXterm的使用,Linux常用命令
  • springboot-scanBasePackages包扫描
  • 【C语言数据结构——————排序(1万字)】
  • PyTorch基础(18)-- torch.stack()方法
  • 从lc560“和为 K 的子数组“带你认识“前缀和+哈希表“的解题思路
  • c:变参函数:汇编解析;va_list;marco 宏:__VA_ARGS__
  • eclipse安装教程(2021版)
  • 计算机网络重点概念整理-第二章 物理层【期末复习|考研复习】
  • 【计算机网络】从输入URL到页面都显示经历了什么??
  • [C++]——带你学习类和对象
  • Docker多平台、跨平台编译打包
  • LLM系列 | 22 : Code Llama实战(下篇):本地部署、量化及GPT-4对比
  • Nginx的进程结构实例演示
  • 【Nginx36】Nginx学习:SSI静态文件服务器端包含模块
  • StripedFly恶意软件框架感染了100万台Windows和Linux主机
  • 蓝桥杯每日一题2023.10.25
  • 【C++】详解map和set基本接口及使用
  • 如何学习 Linux 内核内存管理
  • 【计算机网络】(谢希仁第八版)第一章课后习题答案
  • Operator开发之operator-sdk入门
  • RabbitMQ生产者的可靠性
  • 集群节点批量执行 shell 命令
  • fl studio21.2水果软件怎么设置中文?
  • .NET CORE 3.1 集成JWT鉴权和授权2
  • nbcio-boot如何进行gitee第三方登录
  • 【C语言】字符函数、字符串函数与内存函数
  • 生成树协议:监控 STP 端口和交换机
  • 【黑产攻防道03】利用JS参数更新检测黑产的协议破解
  • 什么是web3.0?
  • 二、W5100S/W5500+RP2040树莓派Pico<DHCP>