Qt 隐藏标题栏可拖拽,自由缩放

Qt在隐藏标题栏的情况下,实现拖拽很简单,可以看这里https://blog.csdn.net/z609932088/article/details/80865742 

或者这里:https://blog.csdn.net/z609932088/article/details/50898022

 

今天主要核心是在几年前,尝试过Qt在隐藏标题栏情况下实现可自由缩放的效果,

原来的坑在这里:https://blog.csdn.net/z609932088/article/details/53929904

今天这坑终于天上了,看下演示效果 参考链接忘记了,浏览记录里面没有找到,如有侵权,联系我

大致就是这样的,剩下的看下代码

 

//头文件

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>
#include <QPushButton>
#include <QMouseEvent>
#include <QApplication>
#include <QDebug>


enum windowEdge{
    TOPLEFT = 11,
    TOP = 12,
    TOPRIGHT = 13,
    LEFT = 21,
    CENTER = 22,
    RIGHT = 23,
    BUTTOMLEFT = 31,
    BUTTOM = 32,
    BUTTOMRIGHT = 33
};

class MainWindow : public QMainWindow
{
    Q_OBJECT
public:
    explicit MainWindow(QWidget *parent = nullptr);

signals:

protected:
    /**
     * @brief mousePressEvent
     * @param event
     * 鼠标按下事件
     */
    void mousePressEvent(QMouseEvent *event);
    /**
     * @brief mouseMoveEvent
     * @param event
     * 鼠标移动事件
     */
    void mouseMoveEvent(QMouseEvent *event);
    /**
     * @brief mouseReleaseEvent
     * @param event
     * 鼠标松开事件
     */
    void mouseReleaseEvent(QMouseEvent *event);
    /**
     * @brief mouseDoubleClickEvent
     * @param event
     * 鼠标双击事件
     */
    void mouseDoubleClickEvent(QMouseEvent *event);
    /**
     * @brief setCursorShape
     * @param mPos
     * 设置鼠标形状
     */
    void setCursorShape(int mPos);
    /**
     * @brief calCursorCol
     * @param pt
     * @return
     * 计算鼠标X的位置
     */
    int calCursorCol(QPoint pt);
    /**
     * @brief calCursorPos
     * @param pt
     * @param colPos
     * @return
     * 计算鼠标的位置
     */
    int calCursorPos(QPoint pt,int colPos);

private:

    QPoint m_mousePoint;            //用于存储鼠标位置
    bool m_moveFlag = false;        //窗口移动标志位
    bool m_resizeFlag = false;      //窗口大小重置标志
    const int m_titleHight = 30;    //用于标记标题栏高度
    const int m_frameShape = 2;     //用于鼠标区域判断
    int     m_iCalCursorPos;
    QRect   m_rtPreGeometry;
    QPoint  m_ptViewMousePos;

    QPushButton *m_pushbuttonClose = nullptr;

};

#endif // MAINWINDOW_H
//源文件

#include "mainwindow.h"

MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent)
{

    this->resize(800,600);
    this->setWindowFlags(Qt::FramelessWindowHint);
    this->setMouseTracking(true);
    m_pushbuttonClose = new QPushButton(this);
    m_pushbuttonClose->setGeometry(100,100,100,80);
    m_pushbuttonClose->setText("关闭");
    //    m_pushbuttonClose->setStyleSheet();
    connect(m_pushbuttonClose,&QPushButton::clicked,this,[=](){this->close();});


}

void MainWindow::mousePressEvent(QMouseEvent *event)
{
    if(event->y() < m_titleHight && event->y()> m_frameShape)
    {
        m_mousePoint = event->globalPos();
        m_moveFlag = true;
    }
    else
    {
        m_iCalCursorPos = calCursorPos(event->pos(),calCursorCol(event->pos()));
        if (event->button() == Qt::LeftButton)
        {
            if(m_iCalCursorPos != CENTER)
            {
                m_resizeFlag = true;
            }
        }
        m_rtPreGeometry = geometry();
        m_ptViewMousePos = event->globalPos();
    }
}

void MainWindow::mouseMoveEvent(QMouseEvent *event)
{
    if((event->y() < m_titleHight) && event->y()> m_frameShape && m_moveFlag)
    {
        int dx = event->globalX() - m_mousePoint.x();
        int dy = event->globalY() - m_mousePoint.y();
        m_mousePoint = event->globalPos();
        this->move(this->x()+dx,this->y()+dy);
    }
    else
    {
        if(Qt::WindowMaximized != windowState())
        {
            setCursorShape(calCursorPos(event->pos(),calCursorCol(event->pos())));
        }
        QPoint ptCurrentPos = QCursor::pos(); //获取当前的点,这个点是全局的
        QPoint ptMoveSize = ptCurrentPos - m_ptViewMousePos; //计算出移动的位置,当前点 - 鼠标左键按下的点
        QRect rtTempGeometry = m_rtPreGeometry;
        if(m_resizeFlag)
        {
            switch(m_iCalCursorPos)
            {
            case windowEdge::TOPLEFT:
                rtTempGeometry.setTopLeft(m_rtPreGeometry.topLeft()+ptMoveSize);
                break;
            case windowEdge::TOP:
                rtTempGeometry.setTop(m_rtPreGeometry.top()+ptMoveSize.y());
                break;
            case windowEdge::TOPRIGHT:
                rtTempGeometry.setTopRight(m_rtPreGeometry.topRight()+ptMoveSize);
                break;
            case windowEdge::LEFT:
                rtTempGeometry.setLeft(m_rtPreGeometry.left()+ptMoveSize.x());
                break;
            case windowEdge::RIGHT:
                rtTempGeometry.setRight(m_rtPreGeometry.right()+ptMoveSize.x());
                break;
            case windowEdge::BUTTOMLEFT:
                rtTempGeometry.setBottomLeft(m_rtPreGeometry.bottomLeft()+ptMoveSize);
                break;
            case windowEdge::BUTTOM:
                rtTempGeometry.setBottom(m_rtPreGeometry.bottom()+ptMoveSize.y());
                break;
            case windowEdge::BUTTOMRIGHT:
                rtTempGeometry.setBottomRight(m_rtPreGeometry.bottomRight()+ptMoveSize);
                break;
            default:
                break;
            }
            this->setGeometry(rtTempGeometry);
        }
    }
}

void MainWindow::mouseReleaseEvent(QMouseEvent *event)
{
    if(event->y() < m_titleHight && event->y()> m_frameShape && m_moveFlag)
    {
        int dx = event->globalX() - m_mousePoint.x();
        int dy = event->globalY() - m_mousePoint.y();
        m_mousePoint = event->globalPos();
        this->move(this->x()+dx,this->y()+dy);
        m_moveFlag = !m_moveFlag;
    }
    else
    {
        m_resizeFlag = !m_resizeFlag;
        QApplication::restoreOverrideCursor();
    }
}

void MainWindow::mouseDoubleClickEvent(QMouseEvent *event)
{
    if(event->button() == Qt::LeftButton)       //鼠标双击最大化/正常
    {
        if(event->y() < m_titleHight)
        {
            if(windowState() != Qt::WindowMaximized)
            {
                this->showMaximized();
            }
            else
            {
                this->showNormal();
            }
        }
    }
}

void MainWindow::setCursorShape(int mPos)
{
    Qt::CursorShape mCursor;
    switch (mPos)
    {
    case windowEdge::TOPLEFT:
    case windowEdge::BUTTOMRIGHT:
        mCursor = Qt::SizeFDiagCursor;
        break;
    case windowEdge::TOPRIGHT:
    case windowEdge::BUTTOMLEFT:
        mCursor = Qt::SizeBDiagCursor;
        break;
    case windowEdge::TOP:
    case windowEdge::BUTTOM:
        mCursor = Qt::SizeVerCursor;
        break;
    case windowEdge::LEFT:
    case windowEdge::RIGHT:
        mCursor = Qt::SizeHorCursor;
        break;
    default:
        mCursor = Qt::ArrowCursor;
        break;
    }
    this->setCursor(mCursor);
}

int MainWindow::calCursorCol(QPoint pt)
{
    return (pt.x() < m_frameShape ? 1 : ((pt.x() > this->width() - m_frameShape) ? 3 : 2));
}

int MainWindow::calCursorPos(QPoint pt, int colPos)
{
    return ((pt.y() < m_frameShape ? 10 : ((pt.y() > this->height() - m_frameShape) ? 30 : 20)) + colPos);
}

  • 1
    点赞
  • 27
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
Qt中,可以通过自定义窗口标题栏的方式来实现缩放功能。 首先,需要创建一个继承自QWidget的自定义窗口类(例如CustomWindow),并重写其鼠标按下、移动和释放事件的处理函数: ```c++ class CustomWindow : public QWidget { // ... protected: void mousePressEvent(QMouseEvent* event) override { if (event->button() == Qt::LeftButton) { // 记录鼠标按下时的窗口位置和大小 m_startPos = event->globalPos(); m_startGeometry = geometry(); } } void mouseMoveEvent(QMouseEvent* event) override { if (event->buttons() & Qt::LeftButton) { // 计算鼠标移动的偏移量 QPoint offset = event->globalPos() - m_startPos; // 根据偏移量来设置新的窗口位置和大小 QRect newGeometry = m_startGeometry.translated(offset); setGeometry(newGeometry); } } void mouseReleaseEvent(QMouseEvent* event) override { if (event->button() == Qt::LeftButton) { // 释放鼠标时重置起始位置和窗口大小 m_startPos = QPoint(); m_startGeometry = QRect(); } } private: QPoint m_startPos; QRect m_startGeometry; }; ``` 接下来,在主窗口中使用自定义窗口类替代原始的QWidget窗口: ```c++ int main(int argc, char *argv[]) { QApplication a(argc, argv); CustomWindow w; w.show(); return a.exec(); } ``` 这样就实现了自定义标题栏缩放功能了。当鼠标按下标题栏左键并移动时,窗口会跟随鼠标的移动而改变位置和大小。释放鼠标按钮后,窗口将保持在新的位置和大小上。 注意:为了实现这个功能,可能还需要对标题栏的样式进行进一步的自定义。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

DreamLife.

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值