轮廓图编程-自定义QChartView

news2024/11/24 12:30:44

目录

一、功能需求

二、实现效果

三、实现方法


一、功能需求

3D测量软件中,需要在轮廓上进行二次编程,需要显示轮廓线,然后可以调节矩形框的范围的获取参数,如华汉的HyperShape3D软件,对轮廓的编程界面如下。

二、实现效果

2个矩形框 + 1个轮廓

1个矩形框 + 1个轮廓

 

三、实现方法

找到了一篇比较符合需求的博客,基于QChartView实现;

Qt Charts使用(重写QChartView,实现一些自定义功能)_讳疾忌医丶的博客-CSDN博客_qtcharts使用

该文中重写了paintEvent(QPaintEvent *event),绘制2个矩形框。然而当我需要添加更多的矩形框时,发现需要添加很多重复性的代码,扩展性极差。为此我在此基础上做了改进,QChartView是基于qt的Graphics/View框架实现的,因此我可以将矩形框封装成一个图元QGraphicsItem,然后在场景中添加图元。

关键代码如下

自定义QChartView

#ifndef MYCHARTS_H
#define MYCHARTS_H

#include <QWidget>
#include <QtCharts/QLineSeries>
#include <QtCharts/QValueAxis>
#include <QChart>
#include <QChartView>
#include "processpropertyeditor_global.h"

QT_CHARTS_USE_NAMESPACE

class GraphicsRangeRectItem;
class PROCESSPROPERTYEDITORSHARED_EXPORT ContourChartView : public QChartView
{
    Q_OBJECT
public:
    enum ItemTheme
    {
        LightPink = 0, /* 浅粉红 */
        Violet,        /* 紫罗兰 */
        SkyBlue,       /* 天蓝色 */
        Cyan,          /* 青色 */
        SeaGreen,      /* 海洋绿 */
        Yellow,        /* 纯黄 */
        Gold,          /* 金 */
        LightGrey,     /* 浅灰色 */
    };

    explicit ContourChartView(QWidget *parent = nullptr);
    GraphicsRangeRectItem *addItem(ItemTheme theme, const QPointF &topleft);
    void lineSeriesAppend(const QList<QPointF> &points);

protected:
    void paintEvent(QPaintEvent *event) override;
    void mousePressEvent(QMouseEvent *event) override;
    void mouseMoveEvent(QMouseEvent *event) override;
    void mouseReleaseEvent(QMouseEvent *event) override;
    void contextMenuEvent(QContextMenuEvent *event) override;

public:
    QMap<ItemTheme, QColor > m_colors;
    QLineSeries *m_pSeries;
    QChart *m_pChart;
};



//
#include "ContourChartView.h"
#include "GraphicsRangeRectItem.h"

#include <QMenu>
#include <QDebug>
#include <QtMath>

ContourChartView::ContourChartView(QWidget *parent)
    : QChartView(parent)
{
    int a = 100;
    m_colors.insert(ItemTheme::LightPink, QColor(255, 182, 193, a));
    m_colors.insert(ItemTheme::Violet, QColor(238, 130, 238, a));
    m_colors.insert(ItemTheme::SkyBlue, QColor(135, 206, 235, a));
    m_colors.insert(ItemTheme::Cyan, QColor(0, 255, 255, a));
    m_colors.insert(ItemTheme::SeaGreen, QColor(46, 139, 87, a));
    m_colors.insert(ItemTheme::Yellow, QColor(255, 255, 0, a));
    m_colors.insert(ItemTheme::Gold, QColor(255, 215, 0, a));
    m_colors.insert(ItemTheme::LightGrey, QColor(211, 211, 211, a));

    m_pSeries = new QLineSeries();
    m_pSeries->setUseOpenGL(true);

    m_pChart = new QChart();
    m_pChart->setTheme(QChart::ChartThemeDark);
    m_pChart->legend()->hide();
    this->setChart(m_pChart);
    this->setRenderHints(QPainter::Antialiasing);
}

GraphicsRangeRectItem *ContourChartView::addItem(ContourChartView::ItemTheme theme,
        const QPointF &topleft)
{
    GraphicsRangeRectItem *rectItem = new GraphicsRangeRectItem;
    rectItem->setRect(QRectF(topleft,  QSizeF(60, 700)));
    rectItem->setPen(QPen(QColor(205, 104, 57, 100)));
    rectItem->setBrush(m_colors.value(theme));
    rectItem->setFlags(QGraphicsItem::ItemIsSelectable | QGraphicsItem::ItemIsMovable);
    rectItem->setZValue(100);
    this->scene()->addItem(rectItem);
    return rectItem;
}

void ContourChartView::lineSeriesAppend(const QList<QPointF> &points)
{
    QList<QAbstractSeries *>series =  m_pChart->series();
    if(series.contains(m_pSeries))
    {
        m_pChart->removeSeries(m_pSeries);
    }
    m_pSeries->clear();
    m_pSeries->append(points);
    m_pChart->addSeries(m_pSeries);
    m_pChart->createDefaultAxes();
}

void ContourChartView::paintEvent(QPaintEvent *event)
{
    QChartView::paintEvent(event);
}

void ContourChartView::mousePressEvent(QMouseEvent *event)
{
    QChartView::mousePressEvent(event);
}

void ContourChartView::mouseMoveEvent(QMouseEvent *event)
{
    QChartView::mouseMoveEvent(event);
}

void ContourChartView::mouseReleaseEvent(QMouseEvent *event)
{
    QChartView::mouseReleaseEvent(event);
}

void ContourChartView::contextMenuEvent(QContextMenuEvent *event)
{
    QMenu menu(this);
    menu.addAction(QStringLiteral("放大"), [ = ]()
    {
        chart()->zoom(1.2);
    });
    menu.addAction(QStringLiteral("缩小"), [ = ]()
    {
        chart()->zoom(0.8);
    });
    menu.addAction(QStringLiteral("还原"), [ = ]()
    {
        chart()->zoomReset();
    });
    menu.exec(event->globalPos());
}

 自定义矩形框图元

#ifndef GRAPHICSRANGERECTITEM_H
#define GRAPHICSRANGERECTITEM_H

#include <QGraphicsRectItem>

class GraphicsRangeRectItem: public QObject, public QGraphicsRectItem
{
    Q_OBJECT

public:
    enum E_HandleFlag : int
    {
        Default = 0x00,
        AtLeft = 0x01,
        AtRight = 0x02,
        AtCenter = 0x03
    };

    explicit GraphicsRangeRectItem(QGraphicsItem *parent = Q_NULLPTR);
    ~GraphicsRangeRectItem();

signals:
    void stateChanged();

protected:
    virtual void hoverMoveEvent(QGraphicsSceneHoverEvent *event) override;
    virtual void hoverLeaveEvent(QGraphicsSceneHoverEvent *event) override;
    virtual void mousePressEvent(QGraphicsSceneMouseEvent *event) override;
    virtual void mouseMoveEvent(QGraphicsSceneMouseEvent *event) override;
    virtual void mouseReleaseEvent(QGraphicsSceneMouseEvent *event) override;

private:
    E_HandleFlag handleAt(const QPointF &pos);

private:
    QMap<E_HandleFlag, Qt::CursorShape> handleCursors;
    E_HandleFlag handleSelected = Default;
    const int MIN_BOX_WIDTH = 20;
};

#endif // GRAPHICSRANGERECTITEM_H


//

#include "GraphicsRangeRectItem.h"
#include <QCursor>
#include <QGraphicsSceneHoverEvent>
#include <QDebug>
#include <QGraphicsScene>
#include <QGraphicsView>

GraphicsRangeRectItem::GraphicsRangeRectItem(QGraphicsItem *parent)
    : QGraphicsRectItem(parent)
{
    handleCursors[Default] = Qt::ArrowCursor;
    handleCursors[AtLeft] = Qt::SizeHorCursor;
    handleCursors[AtRight] = Qt::SizeHorCursor;
    this->setFlags(QGraphicsItem::ItemIsSelectable |
                   QGraphicsItem::ItemIsFocusable);
    setAcceptHoverEvents(true);
}

GraphicsRangeRectItem::~GraphicsRangeRectItem()
{

}

void GraphicsRangeRectItem::hoverMoveEvent(QGraphicsSceneHoverEvent *event)
{
    if(this->isSelected())
    {
        E_HandleFlag handle = this->handleAt(event->pos());
        this->setCursor(QCursor(handleCursors.value(handle)));
    }
    QGraphicsRectItem::hoverMoveEvent(event);
}

void GraphicsRangeRectItem::hoverLeaveEvent(QGraphicsSceneHoverEvent *event)
{
    this->setCursor(QCursor(Qt::ArrowCursor));
    QGraphicsRectItem::hoverLeaveEvent(event);
}

void GraphicsRangeRectItem::mousePressEvent(QGraphicsSceneMouseEvent *event)
{
    handleSelected = this->handleAt(event->pos());
    QGraphicsRectItem::mousePressEvent(event);
}

void GraphicsRangeRectItem::mouseMoveEvent(QGraphicsSceneMouseEvent *event)
{
    switch (handleSelected)
    {
        case E_HandleFlag::AtLeft:
            {
                QRectF oldRect = this->rect();
                /* 最小宽度限制 */
                if(oldRect.right() - event->pos().x() < MIN_BOX_WIDTH)
                {
                    return;
                }
                QRectF newRect = oldRect;
                newRect.setLeft(event->pos().x());
                this->setRect(newRect);
                update();
            }
            break;
        case E_HandleFlag::AtRight:
            {
                QRectF oldRect = this->rect();
                /* 最小宽度限制 */
                if(event->pos().x() - oldRect.left() < MIN_BOX_WIDTH)
                {
                    return;
                }
                QRectF newRect = oldRect;
                newRect.setRight(event->pos().x());
                this->setRect(newRect);
                update();
            }
            break;
        case E_HandleFlag::AtCenter:
            {
                QRectF oldRect = this->rect();
                /* 防止超出左右边界 */
                int leftLimit = 0;
                int rightLimit = this->scene()->views().at(0)->size().width();
                if(event->pos().x() <= leftLimit || event->pos().x() >= rightLimit)
                {
                    return;
                }
                QRectF newRect = oldRect;
                newRect.moveCenter(QPointF(event->pos().x(), oldRect.center().y()));
                this->setRect(newRect);
                update();
            }
            break;
        default:
            break;
    }
}

void GraphicsRangeRectItem::mouseReleaseEvent(QGraphicsSceneMouseEvent *event)
{
    QGraphicsItem::mouseReleaseEvent(event);
    handleSelected = Default;
    if(event->button() == Qt::LeftButton)
    {
        emit stateChanged();
    }
    this->update();
}

GraphicsRangeRectItem::E_HandleFlag GraphicsRangeRectItem::handleAt(const QPointF &pos)
{
    QRectF rect = this->rect();
    static const int check_radius = 3;
    if (std::abs(pos.x() - rect.right()) < check_radius)
    {
        return E_HandleFlag::AtRight;
    }
    else if (std::abs(pos.x() - rect.left()) < check_radius)
    {
        return E_HandleFlag::AtLeft;
    }
    else if(rect.contains(pos))
    {
        return E_HandleFlag::AtCenter;
    }
    return E_HandleFlag::Default;
}

使用

创建窗口

ContourChartView *graphicsView = new ContourChartView();

添加轮廓

graphicsView->lineSeriesAppend(const QList<QPointF> &points);

添加矩形框

 GraphicsRangeRectItem *rectItem1 = chartView()->addItem(ContourChartView::ItemTheme::LightPink, QPointF(100, 10));
    rectItem1->setToolTip(QStringLiteral("基准对象"));
 GraphicsRangeRectItem *rectItem2 = chartView()->addItem(ContourChartView::ItemTheme::Violet, QPointF(400, 10));
    rectItem2->setToolTip(QStringLiteral("测量对象"));
    
坐标映射,需要将场景中的坐标转换到chart图表上的坐标值
    connect(rectItem1, &GraphicsRangeRectItem::stateChanged, this, [ = ]()
    {
    double xl = chartView()->chart()->mapToValue(QPointF(rectItem1->rect().left(),  0)).x();
    double xr = chartView()->chart()->mapToValue(QPointF(rectItem1->rect().right(), 0)).x();
    });

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

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

相关文章

【Java基础知识复盘】String、StringBuffer、StringBuilder篇——持续更新中

本人知识复盘系列的博客并非全部原创&#xff0c;大部分摘自网络&#xff0c;只是为了记录在自己的博客方便查阅&#xff0c;往后也会陆续在本篇博客更新本人查阅到的新的知识点&#xff0c;望悉知&#xff01; String类 在 Java 中字符串属于对象&#xff0c;Java 提供了 Str…

《融合视觉显著性和局部熵的红外弱小目标检测》论文复现

1.复现论文概要 复现的论文为《融合视觉显著性和局部熵的红外弱小目标检测》&#xff08;赵鹏鹏&#xff0c;李庶中等&#xff0c;中国光学2022&#xff0c;http://www.chineseoptics.net.cn/cn/article/doi/10.37188/CO.2021-0170&#xff0c;以下简称论文&#xff09;。论文…

【库存控制】基于象鼻虫损害优化算法求解库存控制问题(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

交换机的结构特点及主要功能

什么是交换机&#xff1f; Switch 意为“交换机”&#xff0c;是一种用于转发电 ( 光 ) 信号的网络设备。它可以为访问交换机的任意两个网络节点提供专用的电信号路径。非常常见的 是以太网交换机。其他常见的有电话语音交换机、光纤交换机等。 它的工作原理是什么&#xff1f…

美国藤校Top30大学对IB成绩的要求

众所周知&#xff0c;IB体系是目前全球认可度比较高的国际课程。那么&#xff0c;用IBDP成绩申请美国TOP30大学需要什么样的成绩&#xff1f; 小智今天来带大家好好研究下美国TOP30大学对IB成绩的要求。普林斯顿大学 Princeton University 普林斯顿大学对于IB成绩没有具体要求&…

nginx架构解析:朴实中见真知

目录前言为什么高并发很重要Apache可以做到吗使用nginx会更有优势吗&#xff1f;nginx架构概览代码结构Workers模型nginx进程规则nginx缓存概览nginx配置nginx内部典型的HTTP请求处理循环课程总结前言 nginx&#xff08;发音“engine x”&#xff09;是俄国的软件工程师Igor S…

Godzilla(哥斯拉)安装与使用

Godzilla安装与使用1.Godzilla介绍2.Godzliia下载与安装2.1.Godzilla下载2.2.运行环境2.3.Godzilla安装2.3.1.执行文件2.3.2.安装完成3.Godzliia3.1.Godzliia使用3.1.1.生成木马3.1.2.存储木马3.1.3.放入木马3.1.4.访问木马3.1.5.连接木马3.1.6.测试连接3.1.7.连接进入3.2.Godz…

麒麟水乡,IU酒店进驻云南旅游咽喉之地曲靖

曲靖位于云南省东北部&#xff0c;是云南连接内地的重要陆路通道&#xff0c;素有“滇黔锁钥”、“入滇门户”、“云南咽喉”之称&#xff0c;是仅次于昆明的云南第二大城市。曾入选“中国十佳宜居城市”榜单10次的城市&#xff0c;拥有3000多年的文明史&#xff0c;早在三国魏…

Python解题 - CSDN周赛第17期 - 拯救公主

本期又出现了题目测试数据的问题&#xff0c;而且题目和算法关系也不太大&#xff0c;基本就属于用代码代替手工解答算术题的感觉。不禁让人怀疑官方题库是否已经没有高质量的题了&#xff0c;同时也怀疑长期满分却又不更新博客拿奖品的某位选手是不是托。。。 第一题&#xff…

VS系列多通道振弦传感器无线采发仪与参数配置工具连接

VS101~VS432 设备配备了专门的参数配置工具 SETP 来完成设备工作参数的查看和修改工作。 连接前的准备工作 &#xff08;1&#xff09;数据接口与计算机连接 使用标配的通讯线与计算机 RS232 接口连接。若需基于手机网络发送数据&#xff0c;请在开机前安装 SIM 卡。 若…

D. Lucky Chains(GCD+素数筛)

input: 4 5 15 13 37 8 9 10009 20000output: 0 1 -1 79题目大意&#xff1a; 如果一个数对(x,y)是幸运的&#xff0c;当且仅当gcd(x,y)1,一条链可以由以下规律的数对组成&#xff0c;(x,y),(x1,y1),(x2,y2)……(xk,yk)&#xff0c;如果说一条链是幸运的&#xff0c;当且仅当…

非零基础自学Golang 第16章 正则表达式 16.3 regexp包 16.4 小结 16.5 知识拓展

非零基础自学Golang 文章目录非零基础自学Golang第16章 正则表达式16.3 regexp包16.3.1 MatchString函数16.3.2 FindStringIndex函数16.3.3 ReplaceAllString函数16.4 小结16.5 知识拓展16.5.1 常用正则表达式参考第16章 正则表达式 16.3 regexp包 Go在处理正则表达式时主要使…

拆箱phper最适合入门的go框架beego

beego beego 是一个快速开发 Go 应用的 HTTP 框架&#xff0c;他可以用来快速开发 API、Web 及后端服务等各种应用&#xff0c;是一个 RESTful 的框架&#xff0c;主要设计灵感来源于 tornado、sinatra 和 flask 这三个框架&#xff0c;但是结合了 Go 本身的一些特性&#xff…

路由信息协议RIP(计算机网络)

目录 路由选择协议 路由信息协议 RIP中的距离 RIP协议的具体算法 RIP协议路由表的更新 例题 距离向量算法 RIP 协议的要点 路由选择协议 路由器之间要互相不断交换网络拓扑和状态信息&#xff0c;根据信息求出到所有目的网络的最佳路由 用于交换路由信息&#xff0c;进…

软件测试实战教程系列—接口测试用例和报告模板|收藏版

文章目录 接口测试为什么会如此重要呢&#xff1f; 接口测试的必要性 获取接口相关信息 接口测试的流程 接口文档 是接口测试的参照&#xff0c;至少包括&#xff1a; 接口测试用例设计 接口测试用例模板 &#xff08;可根据项目实际情况设计增减&#xff09; 接口测试…

前端基础_组合多个图形

组合多个图形 在前面的实例中&#xff0c;我们看到使用Canvas API可以将一个图形重叠绘制在另一个图形上面&#xff0c;但图形中能够被看到的部分完全取决于以哪种方式进行组合&#xff0c;这时需要使用到Canvas API的图形组合技术。在HTML5中&#xff0c;只要用图形上下文对象…

模数转换器(ADC)

目录 一、简介&#xff1a; 二、主要特征 三、校准&#xff08;CLB&#xff09; 四、ADC时钟 五、ADCON开关 六、规则组和注入组 七、转换模式 八、注入通道管理 九、可编程的采样时间 十、外部触发 十一、温度传感器和内部参考电压 一、简介&#xff1a; 12位ADC是…

CSS -- 使用纯CSS实现旋转木马相册的效果

如果对3D转换不熟悉可以参考&#xff1a;CSS – CSS3中3D转换相关属性讲解&#xff08;translate3d&#xff0c;rotate3d&#xff0c;perspective&#xff0c;transform-style&#xff09; 如果对动画不熟悉可以参考&#xff1a;CSS – CSS3基础动画讲解 旋转木马图片相册 <…

centos 7安装mysql

一.安装之前检测系统是否有自带的MySQL(若是没有安装直接忽略该步骤) rpm -qa | grep mysql #检查是否安装过MySQL rpm -qa | grep mariadb #检查是否存在 mariadb 数据库&#xff08;内置的MySQL数据库&#xff09;&#xff0c;有则强制删除 rpm -e --nodeps mariadb-libs-…

信而泰BGP Flow Spec防攻击测试解决方案

随着互联网行业的迅猛发展&#xff0c;越来越多的业务都从线下走到了线上。互联网在给大家生活带来便利的同时也面临着防护自身安全的各种挑战。 DoS/DDoS攻击是对网络安全的重大威胁&#xff0c;攻击者通过多个控制端控制成千上万的攻击设备对同一个目的地址、网段或服务器同…