QGraphicsView、QGraphicsScene和QGraphicsItem图形视图框架(二)疑难杂症

news2025/1/15 13:02:59

疑难杂症1

1.问题:

设置场景的背景图片时,采用setBackgroundBrush()方法和重写drawBackground()函数得到的结果很不一样,而且通过setSceneRect设置场景原点位置之后得到的结果也有很大区别。

如下图

第一个和第三个中重写了QGraphicsScene的drawBackground()函数,区别在于第一个的场景原点在左上角,第三个的原点在中心。

第二个和第四个直接用的QGraphicsScene通过setBackgroundBrush()函数设置背景图片,区别在于第二个的场景原点在左上角,第四个的原点在中心。

class GraphicsScene : public QGraphicsScene
{
    Q_OBJECT
public:
    explicit GraphicsScene(QObject *parent = nullptr)
        :QGraphicsScene(parent)
    {

    }

protected:
    void drawBackground(QPainter *painter, const QRectF &rect)
    {
        Q_UNUSED(rect)
        painter->drawPixmap(sceneRect().toRect(),QPixmap("/home/li/1.png"));
    }
};

便于查看原点位置,在原点绘制了十字架 

    // 绘制十字路径
    {
        // 创建 QPainterPath 对象
        QPainterPath path;
        path.moveTo(-50, 0);
        path.lineTo(50, 0);
        path.moveTo(0, -50);
        path.lineTo(0, 50);

        // 创建 QGraphicsPathItem 对象,并设置路径
        QGraphicsPathItem *pathItem = new QGraphicsPathItem(path);
        pathItem->setPen(QPen(Qt::black, 1));
        pathItem->setPos(0,0);
        scene1->addItem(pathItem);

        QGraphicsRectItem *rect = new QGraphicsRectItem(-1,-1,2,2);
        rect->setBrush(Qt::red);
        rect->setPen(QPen(Qt::red));
        rect->setPos(0,0);
        scene1->addItem(rect);
    }

四个不同的显示代码如下:

//1.
GraphicsScene *scene1 = new GraphicsScene;
scene1->setSceneRect(0,0,400,400);
ui->graphicsView->setScene(scene1);

//2.
QGraphicsScene *scene2 = new QGraphicsScene;
scene2->setSceneRect(0,0,400,400);
scene2->setBackgroundBrush(QPixmap("/home/li/1.png"));
ui->graphicsView_2->setScene(scene2);

//3.
GraphicsScene *scene3 = new GraphicsScene;
scene3->setSceneRect(-200,-200,400,400);
ui->graphicsView_3->setScene(scene3);

//4.
QGraphicsScene *scene4 = new QGraphicsScene;
scene4->setSceneRect(-200,-200,400,400);
scene4->setBackgroundBrush(QPixmap("/home/li/1.png"));
ui->graphicsView_4->setScene(scene4);

这是为什么呢???为什么显示的结果不一样?为什么直接通过setBackgroundBrush设置的背景,在超出场景边界的地方以场景的背景填充的?

2.解答

找了一下drawBackground()和setBackgroundBrush()的源码实现,如下:

void QGraphicsScene::setBackgroundBrush(const QBrush &brush)
{
    Q_D(QGraphicsScene);
    d->backgroundBrush = brush;
    for (QGraphicsView *view : std::as_const(d->views)) {
        view->resetCachedContent();
        view->viewport()->update();
    }
    update();
}
void QGraphicsScene::drawBackground(QPainter *painter, const QRectF &rect)
{
    Q_D(QGraphicsScene);

    if (d->backgroundBrush.style() != Qt::NoBrush) {
        if (d->painterStateProtection)
            painter->save();
        painter->setBrushOrigin(0, 0);
        painter->fillRect(rect, backgroundBrush());
        if (d->painterStateProtection)
            painter->restore();
    }
}

通过源码可以通过setBackgroundBrush设置背景本质上还是drawBackground函数绘制的背景。

[virtual protected] void QGraphicsScene::drawBackground(QPainter *painter, const QRectF &rect)
Draws the background of the scene using painter, before any items and the foreground are drawn. Reimplement this function to provide a custom background for the scene.
All painting is done in scene coordinates. The rect parameter is the exposed rectangle.
If all you want is to define a color, texture, or gradient for the background, you can call setBackgroundBrush() instead. 

drawBackground函数的官方解释中,参数rect是暴露的矩形。

图片中的第一个和第三个只显示了sceneRect()矩形内的背景是因为在重写的drawBackground函数中,painter->drawPixmap(sceneRect().toRect(),QPixmap("/home/li/1.png"));只在设置的场景矩形框内绘制背景。源码中painter->fillRect(rect, backgroundBrush());在给定的暴露的场景内绘制。

void QPainter::setBrushOrigin(const QPointF &position)

Sets the brush origin to position.

The brush origin specifies the (0, 0) coordinate of the painter's brush.

setBrushOrigin函数的官方解释,将画笔原点设置为给定位置。

图片中的第二个和第四个的差异是因为在源码中写到painter->setBrushOrigin(0, 0);将画笔原点设置为(0,0)点,所以绘制背景图片时是从(0,0)点作为起点的。

3.综合

如果想要以不规则的图片作为背景图片,需要重写QGraphicsScene,重写drawBackground()函数。

class GraphicsScene : public QGraphicsScene
{
    Q_OBJECT
public:
    explicit GraphicsScene(QObject *parent = nullptr)
        :QGraphicsScene(parent)
    {

    }

protected:
    void drawBackground(QPainter *painter, const QRectF &rect)
    {
        painter->drawPixmap(rect.toRect(),QPixmap("/home/li/1.png"));
    }
};

疑难杂症2

1.问题

默认情况下,当在 QGraphicsView 中设置一个 QGraphicsScene 实例时,视图将自动调整滚动区域以适应场景的大小,确保场景的中心在视图中居中显示。这一机制确保了无论场景大小如何变化,视图始终能保持场景中心点的可视性。

但是!!!下面出现了特殊情况。

由于工作中用到在QTabWidget控件中放两个完全一样的QGraphicsView并使用布局,然后在场景中心放十字架,发现总有一个view的十字架不在中心。

放到同一个tab里面用布局都不在中间

在tab中不用布局,直接设置view的大小就是正常的。

还尝试了除QTabWidget之外可以放多个窗口的控件,是没问题。

2.解答

通过上面的对比,可以发现是QTabWidget里面的布局对QGraphicsView中的滚动条位置产生了影响,使场景中心和视图的视口中心不能对应起来。(但是是为什么会有这个影响就不太清楚了)

 消除这个的影响只能主动设置一下QGraphicsView的滚动条位置,设置滚动条的位置在中心。

 ui->graphicsView->horizontalScrollBar()->setValue(-ui->graphicsView->viewport()->rect().width()/2);
 ui->graphicsView->verticalScrollBar()->setValue(-ui->graphicsView->viewport()->rect().height()/2);

注意:在设置时,需要在窗口完全显示之后设置,因为要获取QGraphicsView的宽度和高度,在未完全显示之前,拿到的宽高都是默认的原始宽高;只要显示出来之后,才可以拿到准确的宽高。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2110689.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

论文解读:《LAMM: Label Alignment for Multi-Modal Prompt Learning》

系列文章目录 文章目录 系列文章目录LAMM: Label Alignment for Multi-Modal Prompt Learning学习1、论文细节理解1、研究背景2、论文贡献3、方法框架4、研究思路5、实验6、限制 LAMM: Label Alignment for Multi-Modal Prompt Learning学习 1、论文细节理解 VL模型和下游任务…

Python输出多位数

作者制作不易,关注、点赞、收藏一下吧! 1.第一种:正常直接用循环 以三位数为例: for i in range(100, 1000):print(i) 运行结果( 展示一部分 ): 图1-1 2.第二种:特定位数 以三位数为例: for i in range(1, 5): # 括号内指定那个位的范围for j in r…

MySQL基础(5)- 运算符

目录 一、算数运算符 1.加法运算符 2.乘除运算符 3.取模运算 二、比较运算符 1. <> <> ! < < > > 2.IS NULL \ IN NOT NULL \ ISNULL 3.LEAST() \ GREATEST() 4.BETWWEEN 条件下界1 AND 条件上界2 5.in (set)\ not in (set) 6.LIKE :模糊查…

96.游戏的启动与多开-窗口多开检测与破解

免责声明&#xff1a;内容仅供学习参考&#xff0c;请合法利用知识&#xff0c;禁止进行违法犯罪活动&#xff01; 内容参考于&#xff1a;易道云信息技术研究院 上一个内容&#xff1a;95.游戏的启动与多开-信号量多开检测与破解 以 95.游戏的启动与多开-信号量多开检测与破…

流程图符号速查:快速掌握流程图绘制要点

流程图是一种以图形化方式表示算法或过程的步骤和逻辑关系的图表&#xff0c;它通过使用一系列标准的符号和连接线来清晰地展示流程的顺序和决策点。这种表示方法不仅使得复杂的过程更加易于理解&#xff0c;而且也便于跟踪和优化。以下是对流程图的进一步扩写&#xff0c;包括…

智谱AI八月全面升级

智谱系列升级&#xff1a; GLM-4-Flash&#xff1a;免费供应&#xff0c;提供快速响应的对话体验。GLM-4-Long&#xff1a;超长上下文模型&#xff0c;支持高达1百万tokens的上下文&#xff0c;成本为1元/百万tokens。GLM-4-Plus&#xff1a;旗舰版模型&#xff0c;性能升级&am…

Linux平台屏幕|摄像头采集并实现RTMP推送两种技术方案探究

技术背景 随着国产化操作系统的推进&#xff0c;市场对国产化操作系统下的生态构建&#xff0c;需求越来越迫切&#xff0c;特别是音视频这块&#xff0c;今天我们讨论的是如何在linux平台实现屏幕|摄像头采集&#xff0c;并推送至RTMP服务。 我们知道&#xff0c;Linux平台&…

js模块化 --- commonjs规范 原理详解

什么是commonjs规范 commonjs是一种模块化规范&#xff08;nodejs的默认模块化规范&#xff0c;新版的nodejs已经支持es6的模块化&#xff0c;但它默认任然使用的是commonjs&#xff09;&#xff0c;通俗的说它将代码分割成了一个一个的模块&#xff0c;让不同的模块拥有自己独…

使用“天聚数行”藏头诗生成API:轻松创作个性化诗词

在日常生活中&#xff0c;有时我们需要一些创意的方式来表达情感或增加趣味性。“天聚数行”提供的藏头诗生成API就是一个很好的工具&#xff0c;可以帮助我们轻松创作个性化的诗词。以下是关于如何使用这一API的详细介绍。 API概述 “天聚数行”的藏头诗生成API是上海觉克信息…

别给智能猫砂盆花冤枉钱了!这三款好用智能猫砂盆哪个更好用?

节假日回老家&#xff0c;不方便带猫咪怎么办&#xff1f;而且猫咪这么能拉&#xff0c;猫砂盆里拉满了又怎么办&#xff1f;猫砂盆一满&#xff0c;就会变脏&#xff0c;变脏了小猫就会抗拒上厕所&#xff0c;从而在家里找其他干净的地方排泄&#xff0c;那我们要怎么保证不在…

AIOT边缘计算机助力智慧储能,开启能源管理新时代

智慧储能能源管理正成为实现可持续发展和高效能源利用的关键。而 AIOT&#xff08;人工智能物联网&#xff09;边缘计算机的出现&#xff0c;为智慧储能能源管理带来了全新的机遇和突破。 一、AIOT 边缘计算机的特点 强大的计算能力 AIOT 边缘计算机具备高性能的处理器和充足…

widows安装配置mamba_ssm环境

由于现在mamba大火&#xff0c;所以将mamba_ssm在windows中的环境配置进行介绍&#xff0c;如果你想在windows中进行开发&#xff0c;那么这是你最好的选择&#xff01;&#xff01; 安装步骤 1.anaconda下载 官网下载&#xff1a;https://www.anaconda.com/download 下载好…

Java笔试面试题AI答之JDBC(4)

文章目录 19. 解释JDBC的ResultSet是什么 &#xff1f;20. JDBC编程有哪些不足之处&#xff0c;MyBatis是如何解决这些问题的&#xff1f;JDBC编程的不足之处MyBatis如何解决这些问题 21. 简述JDBC 能否处理 Blob 和 Clob &#xff1f;1. JDBC对Blob和Clob的支持2. 处理Blob和C…

【LeetCode】09.回文数

题目要求 解题思路 主要是提防越界问题 代码实现 class Solution { public:bool isPalindrome(int x) {//处理边界if(x<0) return false;long tempx,ret0;while(temp){retret*10temp%10;temp/10;}return xret;} };

Java并发编程实战 01 | 进程和线程

最早的计算机就像一个新手服务员&#xff0c;只有在接收到每一条指令时才会开始执行。当用户输入指令时&#xff0c;计算机会执行这条指令&#xff0c;然后等待下一条指令。如果用户在思考或者犹豫时&#xff0c;计算机就会乖乖地等待&#xff0c;效率实在是有点低&#xff0c;…

Swagger UI 无法发送 Cookie

文章目录 背景分析解决参考背景 项目中使用 Cookie 传递用户唯一标识,并在 Swagger 中添加 Cookie 作为全局请求参数。 尽管后端配置了 Cookie 请求参数,但在 Swagger UI 中使用 Try it out 发起请求时,发现请求中并没有 Cookie 传递过去。😭 首先,能看到 Curl 上已经有…

高级编程语言翻译例题

编译器的流程 源程序—词法分析—语法分析—语义分析—中间代码生成—代码优化—目标代码生成—目标程序 选项A&#xff1a;先进性词法分析&#xff0c;接着进行语法分析&#xff0c;最后进行语义分析 选项B&#xff1a;语法分析阶段只能发现程序上的语法错误&#xff0c;其…

软考高项通过率最高?!证书价值大吗?什么时候能报考?

近期有省份公布了2024年上半年软考合格人员名单&#xff0c;不少人说软考高项通过率最高&#xff0c;导致一些人十分想报名软考高项&#xff0c;从而拿证书。 那么&#xff0c;软考高项证书价值大吗&#xff1f;什么时候能报考&#xff1f; 1、通过率分析 在浙江公布的2024年…

YOLOv9改进策略【注意力机制篇】| GAM全局注意力机制: 保留信息以增强通道与空间的相互作用

一、本文介绍 本文记录的是基于GAM注意力模块的YOLOv9目标检测改进方法研究。GAM注意力模块通过3D排列和重新设计的子模块&#xff0c;能够在通道和空间方面保留信息&#xff0c;避免了先前方法中由于信息减少和维度分离而导致的全局空间-通道交互丢失的问题。本文利用GAM改进…

UML的图及其他图补充

一、UML图 1.类图 ‌类图‌是统一建模语言&#xff08;UML&#xff09;中的一种静态结构图&#xff0c;主要用于描述软件系统的静态结构。它显示了模型中的类、类的内部结构以及它们与其他类的关系。类图是面向对象建模的主要组成部分&#xff0c;用于对系统的词汇进行建模、对…