• 7808阅读
  • 19回复

仿win7窗体自动顶部最大化左侧右侧半屏效果 [复制链接]

上一主题 下一主题
离线liudianwu
 

只看楼主 倒序阅读 楼主  发表于: 2017-04-16
版主 圣域天子,最近一年多一直在找这个效果,今天一觉醒来,又看到版主在CSDN的Qt板块寻找此效果,特意抽空随手写了下代码,基本实现该效果,纯qt代码实现,没有使用windows API,所以在其他平台上,运行效果一样。
原理:绑定事件过滤器,自动计算当前无边框窗体的位置和鼠标按下去的坐标,当到达顶部或者左侧右侧时,自动设置该窗体的geometry即可。
为了复用代码,我这里绑定的全局事件过滤器,这样只需要无边框窗体界面设置两行即可,无需重复编码。
无边框窗体代码:
  1. this->setProperty("canMove", true);
  2. this->setWindowFlags(Qt::FramelessWindowHint | Qt::WindowSystemMenuHint | Qt::WindowMinMaxButtonsHint);

核心代码:
  1. #include "appinit.h"
  2. #include "qapplication.h"
  3. #include "qdesktopwidget.h"
  4. #include "qevent.h"
  5. #include "qwidget.h"
  6. #include "qdebug.h"
  7. AppInit *AppInit::self = 0;
  8. AppInit::AppInit(QObject *parent) : QObject(parent)
  9. {
  10. }
  11. bool AppInit::eventFilter(QObject *obj, QEvent *evt)
  12. {
  13.     QWidget *w = (QWidget *)obj;
  14.     if (!w->property("canMove").toBool()) {
  15.         return QObject::eventFilter(obj, evt);
  16.     }
  17.     //存储桌面宽高以及全屏/左侧半屏/右侧半屏 区域
  18.     static int deskWidth = qApp->desktop()->availableGeometry().width();
  19.     static int deskHeight = qApp->desktop()->availableGeometry().height();
  20.     static QRect fullRect = qApp->desktop()->availableGeometry();
  21.     static QRect leftRect = QRect(0, 0, deskWidth / 2, deskHeight);
  22.     static QRect rightRect = QRect(deskWidth / 2, 0, deskWidth / 2, deskHeight);
  23.     bool autoRect = w->property("autoRect").toBool();
  24.     static QPoint mousePoint;
  25.     static bool mousePressed = false;
  26.     QMouseEvent *event = static_cast<QMouseEvent *>(evt);
  27.     if (event->type() == QEvent::MouseButtonPress) {
  28.         if (event->button() == Qt::LeftButton) {
  29.             mousePressed = true;
  30.             mousePoint = event->globalPos() - w->pos();
  31.             return true;
  32.         }
  33.     } else if (event->type() == QEvent::MouseButtonRelease) {
  34.         mousePressed = false;
  35.         //计算全局坐标
  36.         int x = event->globalPos().x();
  37.         int y = event->globalPos().y();
  38.         int offset = 10;
  39.         //如果Y坐标在桌面顶部,则自动最大化
  40.         //如果X坐标在桌面左侧,则自动左侧半屏幕
  41.         //如果X坐标在桌面右侧,则自动右侧半屏幕
  42.         //自动变化后记住当前窗体是自动产生的位置,以便下次恢复时自动应用变化前的位置
  43.         if (!autoRect) {
  44.             //存储最后一次的位置,自动矫正负数的坐标
  45.             int oldX = w->geometry().x();
  46.             oldX = oldX < 0 ? 0 : oldX;
  47.             int oldY = w->geometry().y();
  48.             oldY = oldY < 0 ? 0 : oldY;
  49.             QRect oldRect = QRect(oldX, oldY, w->geometry().width(), w->geometry().height());
  50.             if (y < offset) {
  51.                 w->setProperty("autoRect", true);
  52.                 w->setProperty("oldRect", oldRect);
  53.                 w->setGeometry(fullRect);
  54.             } else if (x < offset) {
  55.                 w->setProperty("autoRect", true);
  56.                 w->setProperty("oldRect", oldRect);
  57.                 w->setGeometry(leftRect);
  58.             } else if (x > (deskWidth - offset)) {
  59.                 w->setProperty("autoRect", true);
  60.                 w->setProperty("oldRect", oldRect);
  61.                 w->setGeometry(rightRect);
  62.             }
  63.         }
  64.         return true;
  65.     } else if (event->type() == QEvent::MouseMove) {
  66.         if (mousePressed && (event->buttons() && Qt::LeftButton)) {
  67.             if (!autoRect) {
  68.                 w->move(event->globalPos() - mousePoint);
  69.             } else {
  70.                 QRect oldRect = w->property("oldRect").toRect();
  71.                 w->setProperty("autoRect", false);
  72.                 w->setGeometry(oldRect);
  73.             }
  74.             return true;
  75.         }
  76.     }
  77.     return QObject::eventFilter(obj, evt);
  78. }
  79. void AppInit::start()
  80. {
  81.     qApp->installEventFilter(this);
  82. }
完整源码: untitled.zip (4 K) 下载次数:210
欢迎关注微信公众号:Qt实战/Qt入门和进阶(各种开源作品、经验整理、项目实战技巧,专注Qt/C++软件开发,视频监控、物联网、工业控制、嵌入式软件、国产化系统应用软件开发) QQ:517216493  WX:feiyangqingyun  QQ群:751439350
离线仗剑天涯

只看该作者 1楼 发表于: 2017-04-16
    
离线qing11

只看该作者 2楼 发表于: 2017-04-16
  
哈喽,大家好
离线pangwei

只看该作者 3楼 发表于: 2017-04-17
离线圣域天子

只看该作者 4楼 发表于: 2017-04-17
首先感谢楼主。
不过这不是我要的效果,我需要的是系统级的最大化和左右分屏,也就是鼠标没有放手前,有虚框显示最大化的效果。

再次感谢~~~
离线boylebao

只看该作者 5楼 发表于: 2017-04-17
说白了,就是最大化预览,不过刘大师实现的效果鼠标再次拖动时会变位,拖不准。
本帖提到的人: @liudianwu
为Qt打造具有强大生产力的软件。
离线圣域天子

只看该作者 6楼 发表于: 2017-04-17
回 boylebao 的帖子
boylebao:说白了,就是最大化预览,不过刘大师实现的效果鼠标再次拖动时会变位,拖不准。 (2017-04-17 09:17) 

这只是一个例子,细节地方自己是可以优化的。

位置问题我也早看到了,没进行缩放的比例计算而已,加几行代码就OK了。

是不过不是系统的~~~
离线kimtaikee

只看该作者 7楼 发表于: 2017-04-17
看来典武兄实在是看不下去了....

离线kimtaikee

只看该作者 8楼 发表于: 2017-04-17
圣域是需要的这个效果吧?仅供示意,没有完善。
附件: untitled.zip (8 K) 下载次数:49

离线liuchangyin

只看该作者 9楼 发表于: 2017-04-17
离线563255107

只看该作者 10楼 发表于: 2017-04-18
回 kimtaikee 的帖子
kimtaikee:圣域是需要的这个效果吧?仅供示意,没有完善。[图片] (2017-04-17 16:14) 

又见大神现身,支持了
离线stlcours

只看该作者 11楼 发表于: 2017-04-18
回 圣域天子 的帖子
圣域天子:首先感谢楼主。
不过这不是我要的效果,我需要的是系统级的最大化和左右分屏,也就是鼠标没有放手前,有虚框显示最大化的效果。
再次感谢~~~ (2017-04-17 08:44) 

虚框用遮罩,刘大师的另一个帖子里有,稍加变化就可以。你太懒了啊。
离线圣域天子

只看该作者 12楼 发表于: 2017-04-18
回 stlcours 的帖子
stlcours:虚框用遮罩,刘大师的另一个帖子里有,稍加变化就可以。你太懒了啊。 (2017-04-18 02:42)

哪个?找来我看看?

不过听上去,就不象是系统级的。
离线liudianwu

只看该作者 13楼 发表于: 2017-04-18
回 kimtaikee 的帖子
kimtaikee:看来典武兄实在是看不下去了.... (2017-04-17 15:31) 

卧槽,勇哥居然称呼我为典武兄,来来来,喝杯啤酒,交个朋友!
欢迎关注微信公众号:Qt实战/Qt入门和进阶(各种开源作品、经验整理、项目实战技巧,专注Qt/C++软件开发,视频监控、物联网、工业控制、嵌入式软件、国产化系统应用软件开发) QQ:517216493  WX:feiyangqingyun  QQ群:751439350
离线stlcours

只看该作者 14楼 发表于: 2017-04-18
回 圣域天子 的帖子
圣域天子:哪个?找来我看看?
不过听上去,就不象是系统级的。 (2017-04-18 10:16) 

这个:
http://www.qtcn.org/bbs/read-htm-tid-62587.html
离线圣域天子

只看该作者 15楼 发表于: 2017-04-18
回 stlcours 的帖子
stlcours:这个:
http://www.qtcn.org/bbs/read-htm-tid-62587.html (2017-04-18 15:24) 

你好象不理解我提出的需求是什么。

离线stlcours

只看该作者 16楼 发表于: 2017-04-18
回 圣域天子 的帖子
圣域天子:你好象不理解我提出的需求是什么。
 (2017-04-18 16:12) 

我觉得我明白。而且用WINAPI实现过。我的意思是,有了遮罩,你可以随心所欲的实现各种效果,因为背景你都可以作假了,还有什么不能做的。
离线return

只看该作者 17楼 发表于: 2017-04-18
      
离线619304288

只看该作者 18楼 发表于: 2017-04-24
#include "HMainWindow.h"

#include <QApplication>
#ifdef Q_OS_WIN
#include <qt_windows.h>
#include <dwmapi.h>

#ifndef GET_X_LPARAM
#define GET_X_LPARAM(lParam)    ((int)(short)LOWORD(lParam))
#endif
#ifndef GET_Y_LPARAM
#define GET_Y_LPARAM(lParam)    ((int)(short)HIWORD(lParam))
#endif

#endif
HMainWindow::HMainWindow(QWidget *parent) : QMainWindow(parent)
{
    setObjectName("HFramer");
    setWindowTitle("HFramer");
    setWidgetBorderless(this);
}


void HMainWindow::setWidgetBorderless(const QWidget *widget)
{
    setWindowFlags( Qt::WindowMinimizeButtonHint | Qt::FramelessWindowHint);
#ifdef Q_OS_WIN
    HWND hwnd = reinterpret_cast<HWND>(widget->winId());
    DWORD style = GetWindowLong(hwnd, GWL_STYLE);
    SetWindowLongPtr(hwnd, GWL_STYLE, style | WS_MAXIMIZEBOX | WS_THICKFRAME | WS_CAPTION);
#endif
}

bool HMainWindow::nativeEvent(const QByteArray &eventType, void *message, long *result)
{
#ifdef Q_OS_WIN
    if (eventType != "windows_generic_MSG")
        return false;

    MSG* msg = static_cast<MSG*>(message);
    QWidget* widget = QWidget::find(reinterpret_cast<WId>(msg->hwnd));
    if (!widget)
        return false;

    switch (msg->message) {

    case WM_NCCALCSIZE: {
        *result = 0;
        return true;
    }

    case WM_NCHITTEST: {
        const LONG borderWidth = 9;
        RECT winrect;
        GetWindowRect(msg->hwnd, &winrect);
        long x = GET_X_LPARAM(msg->lParam);
        long y = GET_Y_LPARAM(msg->lParam);

        // bottom right
        if (x < winrect.right && x >= winrect.right - borderWidth &&
                y < winrect.bottom && y >= winrect.bottom - borderWidth)
        {
            *result = HTBOTTOMRIGHT;
            return true;
        }

        return false;
    }

    case WM_GETMINMAXINFO: {
        if (::IsZoomed(msg->hwnd)) {

            RECT frame = { 0, 0, 0, 0 };
            AdjustWindowRectEx(&frame, WS_OVERLAPPEDWINDOW, FALSE, 0);
            frame.left = abs(frame.left);
            frame.top = abs(frame.bottom);
            widget->setContentsMargins(frame.left, frame.top, frame.right, frame.bottom);
        }
        else {
            widget->setContentsMargins(0, 0, 0, 0);
        }

        *result = ::DefWindowProc(msg->hwnd, msg->message, msg->wParam, msg->lParam);
        return true;
    }
        break;

    default:
        break;
    }

#endif

    return QMainWindow::nativeEvent(eventType, message, result);
}

这样可以实现系统的虚框,重新写nativeEvent方法
离线圣域天子

只看该作者 19楼 发表于: 2017-04-24
回 kimtaikee 的帖子
kimtaikee:圣域是需要的这个效果吧?仅供示意,没有完善。[图片] (2017-04-17 16:14)

也不是系统级的 ... ...

其实在 tdesktop 中有成功实现的,不过我看了一天多都没找到它是怎么实现的。


前面两位都是高手。

而我是吹毛求疵的完美主义者~~~
快速回复
限100 字节
 
上一个 下一个