查看完整版本: [-- 仿win7窗体自动顶部最大化左侧右侧半屏效果 --]

QTCN开发网 -> Qt 作品展 -> 仿win7窗体自动顶部最大化左侧右侧半屏效果 [打印本页] 登录 -> 注册 -> 回复主题 -> 发表主题

liudianwu 2017-04-16 10:27

仿win7窗体自动顶部最大化左侧右侧半屏效果

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

核心代码:
  1. #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);
    }
完整源码:[attachment=16828]

仗剑天涯 2017-04-16 11:06
    

qing11 2017-04-16 18:29
  

pangwei 2017-04-17 08:37

圣域天子 2017-04-17 08:44
首先感谢楼主。
不过这不是我要的效果,我需要的是系统级的最大化和左右分屏,也就是鼠标没有放手前,有虚框显示最大化的效果。

再次感谢~~~

boylebao 2017-04-17 09:17
说白了,就是最大化预览,不过刘大师实现的效果鼠标再次拖动时会变位,拖不准。

圣域天子 2017-04-17 12:00
boylebao:说白了,就是最大化预览,不过刘大师实现的效果鼠标再次拖动时会变位,拖不准。 (2017-04-17 09:17) 

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

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

是不过不是系统的~~~

kimtaikee 2017-04-17 15:31
看来典武兄实在是看不下去了....

kimtaikee 2017-04-17 16:14
圣域是需要的这个效果吧?仅供示意,没有完善。

liuchangyin 2017-04-17 16:47

563255107 2017-04-18 00:31
kimtaikee:圣域是需要的这个效果吧?仅供示意,没有完善。[图片] (2017-04-17 16:14) 

又见大神现身,支持了

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

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

圣域天子 2017-04-18 10:16
stlcours:虚框用遮罩,刘大师的另一个帖子里有,稍加变化就可以。你太懒了啊。 (2017-04-18 02:42)

哪个?找来我看看?

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

liudianwu 2017-04-18 11:54
kimtaikee:看来典武兄实在是看不下去了.... (2017-04-17 15:31) 

卧槽,勇哥居然称呼我为典武兄,来来来,喝杯啤酒,交个朋友!

stlcours 2017-04-18 15:24
圣域天子:哪个?找来我看看?
不过听上去,就不象是系统级的。 (2017-04-18 10:16) 

这个:
http://www.qtcn.org/bbs/read-htm-tid-62587.html

圣域天子 2017-04-18 16:12
stlcours:这个:
http://www.qtcn.org/bbs/read-htm-tid-62587.html (2017-04-18 15:24) 

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


stlcours 2017-04-18 18:07
圣域天子:你好象不理解我提出的需求是什么。
 (2017-04-18 16:12) 

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

return 2017-04-18 20:20
      

619304288 2017-04-24 09:01
#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方法

圣域天子 2017-04-24 14:11
kimtaikee:圣域是需要的这个效果吧?仅供示意,没有完善。[图片] (2017-04-17 16:14)

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

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


前面两位都是高手。

而我是吹毛求疵的完美主义者~~~


查看完整版本: [-- 仿win7窗体自动顶部最大化左侧右侧半屏效果 --] [-- top --]



Powered by phpwind v8.7 Code ©2003-2011 phpwind
Gzip disabled