zoukankan      html  css  js  c++  java
  • Qt仿win7自动顶部最大化左侧右侧半屏效果

    Win7系统不得不说是非常好用的,也是目前为止占用份额最大的操作系统,其中win7有个效果,将窗体拖动到顶部时会自动最大化,拖动到左侧右侧时会自动半屏显示,再次拖动窗体到其他位置,会重新恢复之前的大小,这个效果还是比较人性化的,大大方便了很多用户的操作习惯。

    在Qt中,如果是无边框窗体,(有边框窗体和操作系统窗体效果一致)并没有相关的API接口来实现这个效果,必须自己写代码来模拟这个效果,原理很简单,绑定事件过滤器,自动计算当前无边框窗体的位置和鼠标按下去的坐标,当到达顶部或者左侧右侧时,自动设置该窗体的geometry即可。

    为了复用代码,我这里绑定的全局事件过滤器,这样只需要无边框窗体界面设置两行即可,无需重复编码。

    无边框窗体代码:

    this->setProperty("canMove", true);
    this->setWindowFlags(Qt::FramelessWindowHint | Qt::WindowSystemMenuHint | Qt::WindowMinMaxButtonsHint);
    

    核心代码:

    #include "appinit.h"
    #include "qapplication.h"
    #include "qdesktopwidget.h"
    #include "qevent.h"
    #include "qwidget.h"
    #include "qdebug.h"
    
    AppInit *AppInit::self = 0;
    AppInit::AppInit(QObject *parent) : QObject(parent)
    {
    }
    
    bool AppInit::eventFilter(QObject *obj, QEvent *evt)
    {
        QWidget *w = (QWidget *)obj;
        if (!w->property("canMove").toBool()) {
            return QObject::eventFilter(obj, evt);
        }
    
        //存储桌面宽高以及全屏/左侧半屏/右侧半屏 区域
        static int deskWidth = qApp->desktop()->availableGeometry().width();
        static int deskHeight = qApp->desktop()->availableGeometry().height();
        static QRect fullRect = qApp->desktop()->availableGeometry();
        static QRect leftRect = QRect(0, 0, deskWidth / 2, deskHeight);
        static QRect rightRect = QRect(deskWidth / 2, 0, deskWidth / 2, deskHeight);
        bool autoRect = w->property("autoRect").toBool();
    
        static QPoint mousePoint;
        static bool mousePressed = false;
    
        QMouseEvent *event = static_cast<QMouseEvent *>(evt);
        if (event->type() == QEvent::MouseButtonPress) {
            if (event->button() == Qt::LeftButton) {
                mousePressed = true;
                mousePoint = event->globalPos() - w->pos();
                return true;
            }
        } else if (event->type() == QEvent::MouseButtonRelease) {
            mousePressed = false;
    
            //计算全局坐标
            int x = event->globalPos().x();
            int y = event->globalPos().y();
            int offset = 10;
    
            //如果Y坐标在桌面顶部,则自动最大化
            //如果X坐标在桌面左侧,则自动左侧半屏幕
            //如果X坐标在桌面右侧,则自动右侧半屏幕
            //自动变化后记住当前窗体是自动产生的位置,以便下次恢复时自动应用变化前的位置
            if (!autoRect) {
                //存储最后一次的位置,自动矫正负数的坐标
                int oldX = w->geometry().x();
                oldX = oldX < 0 ? 0 : oldX;
                int oldY = w->geometry().y();
                oldY = oldY < 0 ? 0 : oldY;
                QRect oldRect = QRect(oldX, oldY, w->geometry().width(), w->geometry().height());
    
                if (y < offset) {
                    w->setProperty("autoRect", true);
                    w->setProperty("oldRect", oldRect);
                    w->setGeometry(fullRect);
                } else if (x < offset) {
                    w->setProperty("autoRect", true);
                    w->setProperty("oldRect", oldRect);
                    w->setGeometry(leftRect);
                } else if (x > (deskWidth - offset)) {
                    w->setProperty("autoRect", true);
                    w->setProperty("oldRect", oldRect);
                    w->setGeometry(rightRect);
                }
            }
    
            return true;
        } else if (event->type() == QEvent::MouseMove) {
            if (mousePressed && (event->buttons() && Qt::LeftButton)) {
                if (!autoRect) {
                    w->move(event->globalPos() - mousePoint);
                } else {
                    QRect oldRect = w->property("oldRect").toRect();
                    w->setProperty("autoRect", false);
                    w->setGeometry(oldRect);
                }
    
                return true;
            }
        }
    
        return QObject::eventFilter(obj, evt);
    }
    
    void AppInit::start()
    {
        qApp->installEventFilter(this);
    }
    

    完整源码下载:http://git.oschina.net/feiyangqingyun/win7full

  • 相关阅读:
    Vue中v-on的指令以及一些其他指令
    Vue模板语法中数据绑定
    Vue介绍以及模板语法-插值
    JQ中的Ajax的封装
    跨域问题的解决?
    Ajax的快速入门
    如何使用npm的部分用法以及npm被墙的解决方法
    使用Node.js简单创建一个服务器
    机智云连接ESP8266--远程控制点亮RGB灯
    1、索引的概念和分类
  • 原文地址:https://www.cnblogs.com/feiyangqingyun/p/6718249.html
Copyright © 2011-2022 走看看