一、概述 用Qt进行开发界面时,既想要实现友好的用户交互又想界面漂亮,那么自定义界面就必不可少。其中有一个操作就是是我们每一个Qter开发者都要会的,而且是经常进行的。 Qt::FramelessWindowHint这个属性想必大家都使用过,有些同学可能对这个属性很了解,也用的是炉火纯青,今天我们也来说说这个属性。 关于这个无边框属性网上也有一些文章,有些谈论的是bug,当然了这是针对不同os而言,也有些是跟其他第三方库混合使用时的问题。可是问题归问题,想要实现自定义的优秀界面这个属性也是必不可少的。 今天我们就来实现一个无边框窗体最大化时,支持拖拽标题栏进行还原的功能。 无边框窗体支持缩放、移动这些不属于本篇文章的内容,本篇文章主要讲解怎么实现最大化时拖拽标题栏进行还原窗体,本篇文章的代码依赖于博主之前封装的一个拖拽代理类。二、效果展示 如效果图所示,做了一个简单的事例,双击标题栏窗体最大化,这个时候如果进行标题栏拖拽,当鼠标按下并移动一段距离时窗体恢复normal状态。 恢复normal状态下的窗体仍然支持放大和缩小,有接口可以设置。 三、demo制作 demo的制作过程还是比较简单的,分为如下几步1、设计窗体 通过desinger设计器我们拖拽了一个大致窗体内容,为了更好的展示效果,标题栏加上了icon和背景色 2、双击放大 鼠标双击标题栏放大这个功能实现起来方法也比较多,这里博主选择了代码量最少并且实现起来最简单的方式,直接把标题栏的事件循环安装到了主窗体上。ui。widgetinstallEventFilter(this); 接下来我们就需要重写主窗口的eventFilter函数即可boolDragWidget::eventFilter(QObjectwatched,QEventevent){if(watchedui。widget){if(eventtype()QEvent::MouseButtonDblClick){if(isMaximized()){showNormal();mhandler。setWidgetResizable(true);mhandler。setWidgetMovable(true);}else{showMaximized();mhandler。setWidgetResizable(false);mhandler。setWidgetMovable(false);}}}returnQWidget::eventFilter(watched,event);} 细心的同学就会发现代码里有一个mhandler变量,这个类就是博主之前自己封装的一个拖拽代理,通过接口可以设置被代理的窗体,并设置需要代理哪些行为。 本篇文章中所演示的事例代码,我们代理了主窗口上标题栏部分的移动事件和整个窗体的缩放事件,设置代码如下所示mhandler。activateOn(this);mhandler。useLocalMoveabled(true);mhandler。addLocalWidget(ui。widget);mhandler。setMaximumMove(true,true); 拖拽代理类内容比较多,本篇文章暂不讲解。四、拖拽 为了更好的理解本篇文章,这里需要把拖拽代理类的头文件放出来,这样更有利于大家理解。 接口都比较简单,代码中也有注释,大家自行阅读。classWidgetResizeHandler:publicQObject{public:explicitWidgetResizeHandler(QObjectparent0);WidgetResizeHandler();public:voidactivateOn(QWidgettopLevelWidget);添加topLevelWidget事件代理voidremoveFrom(QWidgettopLevelWidget);移除topLevelWidget事件代理Qt::CursorShapeCursorShape(QWidgetwidget);窗口移动default:truevoidsetWidgetMovable(boolmovable);boolisWidgetMovable();大小可变default:truevoidsetWidgetResizable(boolresizable);boolisWidgetResizable();橡胶式窗口移动default:falsevoiduseRubberBandOnMove(booluse);boolisUsingRubberBandOnMove();橡胶式修改大小default:falsevoiduseRubberBandOnResize(booluse);boolisUsingRubberBandOnResisze();voidsetBorderWidth(intnewBorderWidth);intborderWidth();局部可移动voiduseLocalMoveabled(booluse);voidaddLocalWidget(QWidget);最大化时支持拖拽参数2表示是否可放大缩小voidsetMaximumMove(boolmove,boolresizefalse);protected:virtualbooleventFilter(QObjectobj,QEventevent)QDECLOVERRIDE;private:WidgetResizeHandlerI}; 值得注意的是最后一个setMaximumMove接口,他就是我们今天的猪脚是否支持最大化时拖拽。当我们设置了这个接口后,窗体最大化时也就能进行拖拽,并还原到之前的normal状态。 文章第三小节讲解demo时,说过主窗体已经被代理拖拽类进行了事件代理,那么主窗体的所有事件首先都会传递给这个代理类,这里我们需要重点关注下鼠标按下时移动事件。voidWidgetData::handleMouseMoveEvent(QMouseEventevent){if(mLeftButtonPressed){if(dptrmWidgetResizablemPressedMousePos。onEdges){resizeWidget(eventglobalPos());}elseif(dptrmWidgetMovable){moveWidget(eventglobalPos());}elseif(dptrmMaxMovable){if(mWidgetisMaximized()TryMoveWidget(event)){dptrmWidgetMdptrmWidgetR}}}elseif(dptrmWidgetResizable){updateCursorShape(eventglobalPos());}} 这段代码包含有其他缩放窗体和正常移动的逻辑,最大化时支持移动的逻辑应该不难找木九十TryMoveWidget这个函数,该函数中我们进行了充分的逻辑判断,一旦触发了窗体移动,那么我们把mWidgetMovable变量置为true,下一次鼠标按下移动事件就会触发正常的拖拽逻辑。 仔细思考上边一段话,其中有2个关键信息触发窗体移动,并还原到之前的normal状态进行了第一步后,需要把mWidgetMovable变量置为true,之后走正常的窗体移动流程 窗体移动 尝试移动窗体,当鼠标当前位置距离鼠标按下时的距离大于20px时,进行窗体还原操作,并返回true,代表窗体已经被重置到normal态。boolWidgetData::TryMoveWidget(QMouseEventevent){QPointdistanceeventglobalPos()mDragPintlengthdistance。manhattanLength();if(length20){QRectrectmWidgetnormalGeometry();intdesXmDragPos。x()rect。width()mWidgetgeometry()。width();intdesYmDragPos。y();rect。moveTopLeft(eventglobalPos()QPoint(desX,desY));mWidgetshowNormal();mWidgetsetGeometry(rect);mDragPosQPoint(desX,desY);mIsMaxM}} 上述代码中的mIsMaxMove标识是为了在一次窗体还原操作后,释放鼠标时可以正常的设置缩放标识而设。 有了上述代码之后,窗体就能还原到最大化之前的大小,并且为之也移动到了鼠标相应的位置,关于这个新位置的计算这里需要说明下。 x坐标 x轴坐标使用了比例计算方式。窗体全屏时鼠标按下的位置在窗体上的位置在窗体还原后依然保持不变,这样计算比较简单而且不会出错,保证窗体还原后,鼠标会一直在标题栏内。 如果需要优化x轴坐标的计算方法,只需要重新计算上述代码中的desX值即可。 y坐标 y轴坐标这里没有做特殊处理。因为窗体还原时,标题栏的高度是没有发生变化的,因此这里不需要做特殊处理。