Qt基于QScrollArea实现界面嵌套移动

在实际的应用场景中,经常会出现软件界面战场图大于实际窗体大小,利用QScrollArea可以为widget窗体添加滚动条,可以实现小窗体利用滚动条显示大界面需求。实现如下:

QT创建一个qWidget界面

在ui界面中利用QT自带的widget控件布局一个如下图所示的层叠关系,widget_2界面大小需大于widget大小

界面布局好后,将widget_2提升为类,提升之前需为工程新添加一个设计界面类,添加完之后,将widget_2提升为类类名和前面新添加的设计界面类名一致

源码实现如下

patchwindow.h

#ifndef PATCHWINDOW_H
#define PATCHWINDOW_H

#include <QDebug>
#include <QPainter>
#include <QWidget>
#include <QMouseEvent>
#include <QStyleOption>
#include <QPaintEvent>

enum CursorRegion{
    NONE,
    TOPLEFT,
    TOPRIGHT,
    BOTTOMRIGHT,
    BOTTOMLEFT
};

namespace Ui {
class Patchwindow;
}

class Patchwindow : public QWidget
{
    Q_OBJECT

public:
    explicit Patchwindow(QWidget *parent = 0);
    ~Patchwindow();
    CursorRegion getCursorRegion(QPoint);

public:
    int borderWidth;
    int handleSize;

    bool mousePressed;
    QPoint previousPos;

private:
    Ui::Patchwindow *ui;

protected:
    void mousePressEvent(QMouseEvent*);
    void mouseReleaseEvent(QMouseEvent*);
    void mouseMoveEvent(QMouseEvent*);

signals:
    void send_widget_rx_ry(int rx,int ry);
};

#endif // PATCHWINDOW_H

patchwindow.cpp

#include "patchwindow.h"
#include "ui_patchwindow.h"

Patchwindow::Patchwindow(QWidget *parent) :
    QWidget(parent),
    ui(new Ui::Patchwindow)
{
    ui->setupUi(this);

    this->setMouseTracking(true);

    setFocusPolicy(Qt::StrongFocus);

    mousePressed = false;
    borderWidth = 1;
    handleSize = 8;

}

Patchwindow::~Patchwindow()
{
    delete ui;
}

//设置鼠标形状
CursorRegion Patchwindow::getCursorRegion(QPoint pos)
{
    if (pos.x() > 0 && pos.x() < (handleSize + borderWidth) &&
        pos.y() > 0 && pos.y() < (handleSize + borderWidth)  ){
        if (this->hasFocus())
            this->setCursor(QCursor(Qt::SizeFDiagCursor));
        return CursorRegion::TOPLEFT;
    }

    if (pos.x() > (this->width() - handleSize - borderWidth) && pos.x() < this->width() &&
        pos.y() > 0 && pos.y() < (handleSize + borderWidth)  ){
        if (this->hasFocus())
            this->setCursor(QCursor(Qt::SizeBDiagCursor));
        return CursorRegion::TOPRIGHT;
    }

    if (pos.x() > (this->width() - handleSize - borderWidth) && pos.x() < this->width() &&
        pos.y() > (this->height() - handleSize - borderWidth) && pos.y() < this->height()  ){
        if (this->hasFocus())
            this->setCursor(QCursor(Qt::SizeFDiagCursor));
        return CursorRegion::BOTTOMRIGHT;
    }

    if (pos.x() > 0 && pos.x() < (handleSize + borderWidth) &&
        pos.y() > (this->height() - handleSize - borderWidth) && pos.y() < this->height()  ){
        if (this->hasFocus())
            this->setCursor(QCursor(Qt::SizeBDiagCursor));
        return CursorRegion::BOTTOMLEFT;
    }

    this->setCursor(Qt::ArrowCursor);
    return CursorRegion::NONE;
}

void Patchwindow::mousePressEvent(QMouseEvent *event)
{
    mousePressed = true;
    previousPos = this->mapToParent(event->pos());
    //qDebug()<<"previousPos = "<<previousPos;
}

void Patchwindow::mouseReleaseEvent(QMouseEvent*)
{
    mousePressed = false;
}

void Patchwindow::mouseMoveEvent(QMouseEvent *event)
{
    if (mousePressed){
        QPoint _curPos = this->mapToParent(event->pos());
        QPoint _offPos = _curPos - previousPos;
        previousPos = _curPos;
        //qDebug()<<"_offPos = "<<_offPos;
        //qDebug()<<"_curPos = "<<_curPos;
        emit send_widget_rx_ry(_offPos.rx(),_offPos.ry());
    }
}

mainwindow.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>
#include <QHBoxLayout>
#include <QDebug>
#include <QScrollArea>

namespace Ui {
class MainWindow;
}

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    explicit MainWindow(QWidget *parent = 0);
    ~MainWindow();

    QScrollArea *m_pScroll;

private:
    Ui::MainWindow *ui;

private slots:
    void remove_widget(int r_x,int r_y);

};

#endif // MAINWINDOW_H

mainwindow.cpp

#include "mainwindow.h"
#include "ui_mainwindow.h"
#include <QPalette>

#include <QScrollBar>

MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    //this->resize(600,600);

    //给父窗体填充颜色
    QPalette palette = ui->widget_2->palette();
    palette.setBrush(QPalette::Window,QBrush(QColor(61,61,61)));
    ui->widget_2->setAutoFillBackground(true);
    ui->widget_2->setPalette(palette);

    ui->widget_2->setAttribute(Qt::WA_StyledBackground);
    ui->widget_2->setStyleSheet("QWidget{background: black}");

    ui->widget_3->setAttribute(Qt::WA_TransparentForMouseEvents, true);//设置该层鼠标事件透明,可以设置为显示层

    m_pScroll = new QScrollArea(ui->widget);
    m_pScroll->setWidget(ui->widget_2);//给widget_2设置滚动条
    //ui->widget_2->setMinimumSize(1500,1000);//这里注意,要比主窗体的尺寸要大,不然太小的话会留下一片空白

    QHBoxLayout *pLayout = new QHBoxLayout;
    pLayout->addWidget(m_pScroll);
    pLayout->setMargin(0);
    pLayout->setSpacing(0);
    ui->widget->setLayout(pLayout);

    connect(ui->widget_2,&Patchwindow::send_widget_rx_ry,this,&MainWindow::remove_widget);

}

MainWindow::~MainWindow()
{
    delete ui;
}

void MainWindow::remove_widget(int r_x,int r_y)
{
    r_y = m_pScroll->verticalScrollBar()->value()-r_y;
    r_x = m_pScroll->horizontalScrollBar()->value()-r_x;

    if((0 < r_y) | (r_y == 0))
    {
        if(r_y > m_pScroll->verticalScrollBar()->maximum())
        {
            r_y = m_pScroll->verticalScrollBar()->maximum();
        }
    }
    else
    {
        r_y = 0;
    }

    if((0 < r_x) | (r_x == 0))
    {
        if(r_x > m_pScroll->horizontalScrollBar()->maximum())
        {
            r_x = m_pScroll->horizontalScrollBar()->maximum();
        }
    }
    else
    {
        r_x = 0;
    }

    m_pScroll->verticalScrollBar()->setValue(r_y);
    m_pScroll->horizontalScrollBar()->setValue(r_x);

}

最终实现效果如下,可以通过滚轮滚动界面,也可以通过鼠标拖拽来实现界面拖拽效果:

到此这篇关于Qt基于QScrollArea实现界面嵌套移动的文章就介绍到这了,更多相关Qt QScrollArea界面嵌套移动内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Qt图形图像开发之QT滚动区控件(滚动条)QScrollArea的详细方法用法图解与实例

    QT滚动区控件(滚动条控件)QScrollArea简介 滚动区域控件QScrollArea用于显示一个画面中的子部件的内容.如果部件超过画面的大小,视图可以提供滚动条,这样就可以看到部件的整个区域. QScrollArea属于控件容器类,可以直接在ui中拖出来. 对于QScrollArea,最难搞懂的就是:如何控制它,才能让它在我们想要出现滚动条的时候出现滚动条. 我们拖入一个QScrollArea,再向他里面拖入4个button,观察信息如下: 可以发现,4个button并不是直接位于QScr

  • 在PYQT5中QscrollArea(滚动条)的使用方法

    如下所示: import sys from PyQt5.QtWidgets import * class MainWindow(QMainWindow): def __init__(self,): super(QMainWindow,self).__init__() self.number = 0 w = QWidget() self.setCentralWidget(w) self.topFiller = QWidget() self.topFiller.setMinimumSize(250,

  • PyQt5实现让QScrollArea支持鼠标拖动的操作方法

    如下所示: #!/usr/bin/evn python3 # -*- coding: utf-8 -*- import sys from PyQt5.QtWidgets import (QApplication, QWidget, QScrollArea, QLabel) from PyQt5.QtCore import QEvent class TestWindow(QWidget): def __init__(self): super(TestWindow, self).__init__()

  • pyqt5 QScrollArea设置在自定义侧(任何位置)

    本例设置为垂直左侧scroll 主要思想是利用一个长度为0的mid_frame,高度为待设置qwidget的高度,用mid_frame的moveEvent事件驱动qwidget的move 我项目的效果图: 代码及注释 from PyQt5.Qt import * from sys import argv # 主窗口 class Main(QMainWindow): def __init__(self): super().__init__(None) self.setGeometry(500, 5

  • Qt基于QScrollArea实现界面嵌套移动

    在实际的应用场景中,经常会出现软件界面战场图大于实际窗体大小,利用QScrollArea可以为widget窗体添加滚动条,可以实现小窗体利用滚动条显示大界面需求.实现如下: QT创建一个qWidget界面 在ui界面中利用QT自带的widget控件布局一个如下图所示的层叠关系,widget_2界面大小需大于widget大小 界面布局好后,将widget_2提升为类,提升之前需为工程新添加一个设计界面类,添加完之后,将widget_2提升为类类名和前面新添加的设计界面类名一致 源码实现如下 pat

  • Qt基于定时器实现动图展示效果

    本文实例为大家分享了Qt基于定时器实现动图展示的具体代码,供大家参考,具体内容如下 总体概述 (1)总体介绍 动图展示主要是将已有的动图逐帧图片连续输出,达到视觉上的动态图效果,本次介绍两例,分别为单一动图和分组动图. 主要原理是设置一个定时器,然后随设置的秒数将资源中的逐帧图片输出,让图片连续变化. (2)素材获得途径 关于素材来源,可以到网站下载现成的逐帧图片素材包,也可以找到自己喜欢的动态图,通过软件(如:爱奇艺万能播放器)将动图逐帧保存得到素材. (注意:不论通过哪种方式获得素材,都需要

  • 基于ElementUI中Table嵌套实现多选的示例代码

    前言: 写这个是因为帮朋友修改项目中的bug 我也是第一次写这个功能,有不对的希望大家指正,如果看完有帮助点个赞! 代码中关键是js中Tree的路径查找这个核心,有不懂的自行百度 多了不说了,有需要的可以私信找我要代码,来看下我怎么实现的 思路: 从头开始看这个需求,我们需要知道用到哪写东西 1.表格Table 2.多选&全选 3.嵌套数据(下拉操作) 正好我们可以找下ElementUI官方文档 找到了我们需要用到的API 在嵌套数据的时候需要使用tree-props 选中数据的时候使用togg

  • Qt 使用QDialog实现界面遮罩的示例(蒙版)

    写应用程序的过程中,弹窗是个避免不了的功能,显示中,假设弹窗背景色和主窗口背景色相差不多,甚至是一样的时候,就会存在一个比较严重的人机交互和UI显示的问题,找到弹窗的边界是比较麻烦的一件事.但是如果我们能在弹窗显示的时候被主窗口其他的部位增加一个背景色,这个背景色与弹窗的背景色形成比较大的色差,那么是不是就能够很容易的找到弹窗的边界.因此,我们实现一个自定义组件,可随便设置需要遮罩的主窗口,并且能够让其跟随主窗口的移动而移动. 先来看下效果: 根据需求功能,我们需要提供设置主窗口的接口,同样的,

  • QT基于TCP实现文件传输系统

    本文实例为大家分享了QT基于TCP实现文件传输系统的具体代码,供大家参考,具体内容如下 一.设计目标 1.发送端选择要传输的文件,进度条显示传输的进度,手动输入要传输的ip地址和端口号 2.接收到数据后显示接收进度,接收完毕后进度条自动消失 二.效果展示 三.实现过程 Qt文件传输发送端 1. 连接服务器 2. 当连接成功后会发送connected信号 3. 在槽函数中发送文件头信息 4. 当数据发送成功后会发送bytesWritten 5. 在槽函数中发送文件内容 UI布局 filesend.

  • QT基于TCP实现网络聊天室程序

    本文实例为大家分享了QT学习:基于TCP的网络聊天室程序,供大家参考,具体内容如下 TCP与UDP的差别如图: 一.TCP工作原理 如下图所示,TCP能够为应用程序提供可靠的通信连接,使一台计算机发出的字节流无差错 地送达网络上的其他计算机.因此,对可靠性要求高的数据通信系统往往使用TCP传输数据,但在正式收发数据前,通信双方必须首先建立连接. 二.TCP编程模型 下面介绍基于TCP的经典编程模型,TCP客户端与服务器间的交互时序如下图所示: 三.TCP服务器端编程实例 TCP服务器端的具体实现

  • Python开发之QT解决无边框界面拖动卡屏问题(附带源码)

    1.简介 看到很多才学QT的人都会问为啥无边框拖动为啥会花屏? 那是因为你每次拖动的过程中都一直在调用move()函数让QT重新绘制界面,如果资源过大,就会导致当前图形还未绘制完,便又重新改变坐标了,从而导致花屏. 2.如何解决 我们参考其它软件,比如QQ,浏览器等,可以看到我们如果在拖动它们的时候,会出现一个虚线框. 如下图所示,可以看到在白色背景下,拖出的虚线框是黑色的 而在黑色背景时,拖出的虚线框是白色的 显然这个虚线框会根据当前桌面的像素点而去取反(也就是255-currentRGB).

  • QT基于TCP网络聊天室

    本文实例为大家分享了QT实现网络聊天室的具体代码,供大家参考,具体内容如下 1.客户端 1.1UI设计 分两个部分,第一部分是消息区里面包含QPlainTextEdit和QListWidget,要显示接收的消息和在线的成员.第二部分QLineEdit发生字符. 1.2 子模块 1.2.1 登录界面 登录界面主要就是要有验证码,防止恶意程序的攻击.通过paintEvent画出一个白色矩形,在白色矩形里面显示四个不同颜色的字母以及随机出现的噪点. 代码: QLoginDialog.h #ifndef

  • Qt基于TCP实现客户端与服务端的连接

    Qt TCP的客户端与服务端的连接,供大家参考,具体内容如下 可以实现局域网内简单的信息传递(文件传输,稍后更新) 界面是用ui设计师做的简单设计 客户端 (1).ClientWidget.h 头文件 #ifndef CLIENTWIDGET_H #define CLIENTWIDGET_H #include <QWidget> #include "ui_ClientWidget.h" #include <QTcpSocket> #include <QHo

  • 基于MVC4+EasyUI的Web开发框架形成之旅之界面控件的使用

    在前面介绍了两篇关于我的基于MVC4+EasyUI技术的Web开发框架的随笔,本篇继续介绍其中界面部分的一些使用知识,包括控件的赋值.取值.清空,以及相关的使用. 我们知道,一般Web界面包括的界面控件有:单行文本框.多行文本框.密码文本框.下拉列表Combobox.日期输入控件.数值输入控件.单项选择.复选框.表格控件DataGrid.树形控件.布局控件.弹出式对话框.提示信息.列表控件等,这些界面控件的操作都有哪些不同,下面我们来逐一进行介绍. 1.单行文本框 使用easyui的控件,单行文

随机推荐