Qt4利用MVC开发曲线数据编辑器

news2024/11/28 4:48:08

目录

1 需求

2 开发流程

1 搭建框架

2 构造函数

3 打开工程

4 实现应用程序参数加载

5 QCustomPlot和TableView的联动

6 数据的可视化修改

7 列表点击事件事先键盘控制

8 表格实现复制,粘贴,删除等一系列功能

9 曲线实现自适应范围和统一范围


1 需求

之前编过1个曲线编辑器,但有几个问题,1是加载太慢,2是没法保存工程。

现在将需求重新整理一下,再开发个曲线编辑器。此外也总结了三点技术问题,分别为:

(1)曲线空间QCustomPlot和表格控件TableView的联动,目的是实现曲线编辑;

(2)数据分类显示,目的是数据按不同分类来绘图,避免叠合在一起看不清。

(3)表格实现复制,粘贴,删除等一系列功能。

(4)列表控件实现键盘控制,解放鼠标,加速曲线切换。

(5)不同曲线实现自适应范围和统一范围。用于对比。

2 开发流程

1 搭建框架

新建main window工程,并使其支持中文

#include <QtGui/QApplication>
#include "mainwindow.h"
#include <QTextCodec>

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);

    // 文本编码规定
    QTextCodec::setCodecForTr(QTextCodec::codecForName("UTF8"));
    QTextCodec::setCodecForLocale(QTextCodec::codecForName("UTF8"));
    QTextCodec::setCodecForCStrings(QTextCodec::codecForName("UTF8"));

    MainWindow w;
    w.show();

    return a.exec();
}

2 构造函数

在构造函数中定好模型,视图以及控件初始化。

MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    winName = "磁测剖面编辑器 V1.0";
    setWindowTitle(winName);

    // 默认编辑开关为否
    customEditOn = false;

    // 默认曲线范围为自适应
    curveXfit = true;
    curveYfit = true;

    // 表格模型视图
    model = new QStandardItemModel(0,22);
    model->setHorizontalHeaderLabels(QStringList()<<"Line"<<"Point"<<"PLon"<<"PLat"<<"DiuCorr"
                                     <<"Read"<<"Sq"<<"Lon"<<"Lat"<<"Elevation"
                                     <<"Date"<<"Time"<<"Instr"<<"GeoMag"<<"ΔT"
                                     <<"Note"<<"PX"<<"PY"<<"X"<<"Y"<<"Pdistance"<<"Anomaly");

    connect(model,SIGNAL(dataChanged(QModelIndex,QModelIndex)),this,SLOT(slotDataChanged(QModelIndex,QModelIndex)));

    proxyModel = new QSortFilterProxyModel(this);
    proxyModel->setSourceModel(model);
    proxyModel->setFilterKeyColumn(0);
    ui->tableView->setModel(proxyModel);
    ui->tableView->resizeColumnsToContents();
    ui->tableView->resizeRowsToContents();


    // 列表模型视图
    lines = new QStringListModel;
    ui->listView->setModel(lines);
    ui->listView->setEditTriggers(false);

    // plot点击事件
    connect(ui->curveView, SIGNAL(plottableClick(QCPAbstractPlottable*,int,QMouseEvent*)),
            this, SLOT(graphClicked(QCPAbstractPlottable*,int)));

    // 进度条
    ui->progressBar->setRange(0,100);
    ui->progressBar->setValue(0);
    ui->progressBar->hide();

}

3 打开工程

这里要重点改一下,先来看之前的逻辑

  • 指定文件名
  • 读取文件到局部变量data
  • 计算公里网
  • 多级排序
  • 计算测线名称及点数
  • 计算距离和异常列
  • 写入模型
  • 挂载测线列表并触发点击
  • 控件显示。

这里要重点改新增加的这几列,有些列需要预留。

更改原始文件,将投影变换的功能移除,现在的逻辑变为:

  • 读取数据
  • 点名分离为线和点
  • 计算测线
  • 计算点数(这个实际MVC中不用,只是用于后期用户统计)
  • 数据写入模型
  • 挂载测线列表
  • 触发点击事件。

此逻辑较老版的更为简介,且加载速度更快。

4 实现应用程序参数加载

为在主程序中调用曲线编辑器,可采用QProcess来调用,这时候需要改造构造函数,并重写打开action。具体代码如下:

// 打开line文件
void MainWindow::on_actionOpen_triggered()
{
    // 1指定文件
    QString tmpName = QFileDialog::getOpenFileName(this,"Open","","*.txt");
    if(tmpName.isEmpty())
        return;
    fileName = tmpName;
    open(QStringList()<<fileName);

}

所有的第3步提到的业务逻辑全部打包至open函数中,这样就可以实现在构造函数中调用,代码如下:

 if(!inNameList.isEmpty())
        open(inNameList);

这里要注意的是,传入参数是个list,需要取第0个值作为文件名。

5 QCustomPlot和TableView的联动

由于QCustomPlot仅仅为绘图库,不是MVC结构,因此只能实现2个单向的联动,以此来模拟MVC机制。

主要实现2个流程:

(1)当绘图数据点击时,实现表格的选择;

(2)当表格数据修改时,实现绘图的更新。

具体第1个方法的代码是:



void MainWindow::graphClicked(QCPAbstractPlottable *plottable, int dataIndex)
{
    // plot中只有QCPGraphs,因此可以立即调用interface1D()
    // 建议先检查interface1D()是否返回非零
    double dataValue = plottable->interface1D()->dataMainValue(dataIndex);
    QString message = QString("Clicked on graph '%1' at data point #%2 with value %3.").arg(plottable->name()).arg(dataIndex).arg(dataValue);
    ui->statusBar->showMessage(message);
    // 少数据的话就不准了
    ui->tableView->setCurrentIndex(proxyModel->index(dataIndex,19,QModelIndex()));
}

第2个方法的实现代码为:

/* 数据改动回调 */
void MainWindow::slotDataChanged(QModelIndex ind1, QModelIndex ind2)
{
    Q_UNUSED(ind1)
    Q_UNUSED(ind2)
    if(customEditOn == true)
        plot();
}

代码解析:1个是绘图库控件单击事件,并不修改表格数据;1个是表格数据的修改事件槽函数,调用重新绘图方法。因此两个并不是咬合的关系,但却恰恰实现了需求的功能。

6 数据的可视化修改

下面来展示下第5点所属的修改过程。

文件加载了数十条曲线,上图显示的第13条。这条曲线明显被一些废点所影响了,如果单纯看表格是很难找到的,毕竟有上千个数据。而曲线则可以轻易的找到废点位置。

只需要点击曲线,就能找到废点在表格中的位置。

选中之后,用户可以选择删除或者修改。

可见删除之后,绘图库控件即时进行了刷新。曲线恢复了正常形态。其余的废点也可以按照这种放方法进行处理。

7 列表点击事件事先键盘控制

在切换曲线时,需要鼠标逐个选择列表项,这相当的麻烦,因此需要实现键盘事件,以此来加速曲线的切换操作。

具体的代码为:


// 方向键上下加回车可调用点击事件
void MainWindow::on_listView_activated(const QModelIndex &index)
{
    on_listView_clicked(index);
}

代码较为简单,仅仅是调用了点击事件。

// 测线列表 单击事件
void MainWindow::on_listView_clicked(const QModelIndex &index)
{
    // 模型过滤后绘图
    int row = index.row();
    QString ln = lines->stringList().at(row);
    // 关闭编辑标记后再修改代理模型 避免在修改时频繁调用plot
    customEditOn = false;
    proxyModel->setFilterRegExp(QRegExp(ln, Qt::CaseInsensitive, QRegExp::FixedString));
    plot();
    customEditOn = true;
}

具体逻辑为:先获取选中行的序号,再找到字符串,之后设置proxymodel的正则化过滤器,刷新绘图后,打开修改开关。

通过上述2个函数配合,就可以实现回车与点击事件的同步操作。简化了曲线切换的麻烦。

8 表格实现复制,粘贴,删除等一系列功能

表格数据需要实现复制,粘贴,删除等一系列功能,这就涉及到tableview的子类化问题,上一篇博文我们用mainwindow来实现复制粘贴,本节则采用对qtableview子类化的方式,重写event函数来达到此目的。

下面是代码:


/*  实现多选的复制粘贴 */
void TableView::keyPressEvent(QKeyEvent *keyEvent)
{
    if(keyEvent->matches(QKeySequence::Copy))//复制
    {
        QModelIndexList indexList = selectionModel()->selectedIndexes();
        if(indexList.isEmpty())
            return;
        int startRow = indexList.first().row();
        int endRow = indexList.last().row();
        int startCol = indexList.first().column();
        int endCol = indexList.last().column();
        QStringList clipboardTextList;
        for(int i = startRow;i <= endRow;i++)
        {
            QStringList rowText;
            for(int j = startCol;j <= endCol;j++)
            {
                rowText.append(model()->data(model()->index(i,j)).toString());

            }
            clipboardTextList.append(rowText.join("\t"));
        }
        QString clipboardText = clipboardTextList.join("\n" );
        QApplication::clipboard()->setText(clipboardText);
    }
    else if (keyEvent->matches(QKeySequence::Paste))
    {
        QString clipboardText = QApplication::clipboard()->text();
        if(clipboardText.isEmpty())
            return;
        QStringList rowTextList = clipboardText.split('\n');
        if(rowTextList.last().isEmpty())//从word或者excel复制的内容后面可能会带'\n',导致split出来后面有个空字符串。
            rowTextList.removeLast();
        QModelIndexList indexList = selectionModel()->selectedIndexes();
        if(indexList.isEmpty())
            return;
        QModelIndex startIndex = indexList.first();
        for(int i = 0;i < rowTextList.size();i++)
        {
            QStringList itemTextList = rowTextList.at(i).split('\t');
            for(int j = 0;j < itemTextList.size();j++)
            {
                QModelIndex curIndex = model()->index(i + startIndex.row(),j + startIndex.column());
                if(curIndex.isValid())
                {
                    model()->setData(curIndex,itemTextList.at(j));
                }
            }
        }
    }
    else if (keyEvent->matches(QKeySequence::Delete))
    {
        // 获取选中行
        QItemSelectionModel *selections = selectionModel();
        QModelIndexList selected = selections->selectedIndexes();
        // 循环选中的各个index并写为空
        foreach(QModelIndex index,selected)
        {
            model()->setData(index,"");
        }
    }
    else if (keyEvent->matches(QKeySequence::SelectAll))
    {
        QModelIndex topLeft;
        QModelIndex bottomRight;
        topLeft = model()->index(0,0);
        bottomRight = model()->index(model()->rowCount()-1,model()->columnCount()-1);
        QItemSelection selection(topLeft,bottomRight);
        selectionModel()->select(selection,QItemSelectionModel::Select);
    }
    else if (keyEvent->matches(QKeySequence::MoveToNextLine))
    {
        if(currentIndex().row()>-1)
        {
            if(currentIndex().row()<model()->rowCount()-1)
                setCurrentIndex(model()->index(currentIndex().row()+1,currentIndex().column()));
        }
    }
    else if (keyEvent->matches(QKeySequence::MoveToPreviousLine))
    {
        if(currentIndex().row()>0)
        {
            setCurrentIndex(model()->index(currentIndex().row()-1,currentIndex().column()));
        }
    }
    else if (keyEvent->matches(QKeySequence::MoveToNextChar))
    {
        if(currentIndex().column()>-1)
        {
            if(currentIndex().column()<model()->columnCount()-1)
                setCurrentIndex(model()->index(currentIndex().row(),currentIndex().column()+1));
        }
    }
    else if (keyEvent->matches(QKeySequence::MoveToPreviousChar))
    {
        if(currentIndex().column()>0)
        {
            setCurrentIndex(model()->index(currentIndex().row(),currentIndex().column()-1));
        }
    }
    else if (keyEvent->matches(QKeySequence::MoveToNextPage))
    {
        int row = currentIndex().row();
        if(row>-1)
        {
            int col = currentIndex().column();
            int step = 20;
            int count = model()->rowCount();
            if(row+step<count-1)
                setCurrentIndex(model()->index(row+step,col));
            else
                setCurrentIndex(model()->index(count-1,col));
        }
    }
    else if (keyEvent->matches(QKeySequence::MoveToPreviousPage))
    {
        int row = currentIndex().row();
        if(row>-1)
        {
            int col = currentIndex().column();
            int step = 20;
            if(row-step>0)
                setCurrentIndex(model()->index(row-step,col));
            else
                setCurrentIndex(model()->index(0,col));
        }
    }
}

代码外层是比较简单的判断语句,分别实现了ctrl+c,ctrl+v,ctrl+a,delete等功能。这样就可以对表格实现较多的单选,多选,指定区域的复制,粘贴,删除等操作。

9 曲线实现自适应范围和统一范围

主要用lineEdit控件和绘图库控件配合完成。

代码如下:


// 横坐标范围模式切换
void MainWindow::on_actionXlim_toggled(bool arg1)
{
    if(arg1==true)
        curveXfit = false;
    else
        curveXfit = true;
}

// 纵坐标范围模式切换
void MainWindow::on_actionYlim_toggled(bool arg1)
{
    if(arg1==true)
        curveYfit = false;
    else
        curveYfit = true;
}

通过切换自适应开关来实现绘图范围的控制。在绘图plot中实现:

   // 设置绘图范围
    QString xmin = ui->xmin->text();
    QString xmax = ui->xmax->text();
    if(curveXfit==true || xmin=="" || xmax=="")
        ui->curveView->xAxis->rescale();
    else
        ui->curveView->xAxis->setRange(xmin.toDouble(),xmax.toDouble());

    QString ymin = ui->ymin->text();
    QString ymax = ui->ymax->text();
    if(curveYfit==true || ymin=="" || ymax=="")
        ui->curveView->yAxis->rescale();
    else
        ui->curveView->yAxis->setRange(ymin.toDouble(),ymax.toDouble());

用一个判断来实现绘图范围控制的切换,以此来实现曲线的对比和显示。

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

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

相关文章

【Web】攻防世界Web_php_wrong_nginx_config

这题考察了绕过登录、目录浏览、后门利用 进来先是一个登录框&#xff0c;随便怎么输前端都直接弹窗 禁用js后再输入后登录 查看源码&#xff0c;好家伙&#xff0c;不管输什么都进不去 直接扫目录 访问/robots.txt 访问/hint.php 访问/Hack.php 抓包看一下 cookie里isLogin0…

【JAVA学习笔记】71 - JDBC入门

项目代码 https://github.com/yinhai1114/Java_Learning_Code/tree/main/IDEA_Chapter25/src/com/yinhai/dao_ 一、JDBC概述 1.基本介绍 1. JDBC为访问不同的数据库提供了统一的接口&#xff0c;为使用者屏蔽了细节问题。 2. Java程序员使用JDBC,可以连接任何提供了JDBC驱动…

菜单的hover不同动画背景

CSS常用示例100专栏目录 本专栏记录的是经常使用的CSS示例与技巧&#xff0c;主要包含CSS布局&#xff0c;CSS特效&#xff0c;CSS花边信息三部分内容。其中CSS布局主要是列出一些常用的CSS布局信息点&#xff0c;CSS特效主要是一些动画示例&#xff0c;CSS花边是描述了一些CSS…

虚拟化逻辑架构: LBR 网桥基础管理

目录 一、理论 1.Linux Bridge 二、实验 1.LBR 网桥管理 三、问题 1.Linux虚拟交换机如何增删 一、理论 1.Linux Bridge Linux Bridge&#xff08;网桥&#xff09;是用纯软件实现的虚拟交换机&#xff0c;有着和物理交换机相同的功能&#xff0c;例如二层交换&#…

IT问题解答类型网站源码

问答网是一款为IT工程师提供的问答平台&#xff0c;旨在帮助用户在线获取专业知识和相关问题的答案。在问答网&#xff0c;用户可以轻松找到其他人的问答问题&#xff0c;并在这里寻求解答。如果您有任何想要解决的问题&#xff0c;都可以在此发布问题并得到其他同行的解答。 …

YARN工作流程详解

图1 图2 图1 -作业提交阶段&#xff1a; 1、client 提交job,向 ResourceManager【RM】 申请job_id; 2、RM 返回 job_id 及资源提交路径 给 client 3、client 把job所需的资源提交 到 3中指定的路径中 4、client 上传完成资源后&#xff0c;向RM 发送执行作业请求&#xff0c;RM…

React中通过children prop或者React.memo来优化子组件渲染【react性能优化】

文章目录 前言未优化之前的代码问题解决方案一&#xff0c;通过children prop解决方案二&#xff0c;通过React.memo后言 前言 hello world欢迎来到前端的新世界 &#x1f61c;当前文章系列专栏&#xff1a;react.js &#x1f431;‍&#x1f453;博主在前端领域还有很多知识和…

1.9 字符数组

1.9 字符数组 一、字符数组概述二、练习 一、字符数组概述 所谓字符数组&#xff0c;就是char类型的数组&#xff0c;比如 char a[]&#xff0c;是C语言中最常用的数组类型&#xff0c;先看一个程序 #include <stdio.h> #define MAXLINE 1000 //最大行长度限制 int get…

Telesquare TLR-2005Ksh 路由器 RCE漏洞复现

0x01 产品简介 Telesquare Tlr-2005Ksh是韩国Telesquare公司的一款 Sk 电讯 Lte 路由器。 0x02 漏洞概述 Telesquare TLR-2005Ksh存在安全漏洞&#xff0c;未经授权的攻击者可通过setSyncTimeHost执行任意命令获取服务器权限。 0x03 复现环境 FOFA&#xff1a;app"TELE…

Windows开启SQL Server服及1433端口

需求&#xff1a;Windows开启SQL Server服务及1433端口 目前端口没有启动 解决&#xff1a; 打开SQL Server配置管理器&#xff08;winR&#xff09; 各个sqlserver版本在textbox中输入对应的命令如下&#xff1a; SQLServerManager15.msc&#xff08;对于 SQL Server 2019 &am…

序列化基础

1、简介 对象序列化的目标是将对象保存到磁盘中&#xff0c;或允许在网络中直接传输对象。它允许把内存中的 Java 对象转换成平台无关的二进制流&#xff08;序列化&#xff0c;也称编码&#xff09;&#xff0c;并持久地保存在磁盘上或通过网络把这种二进制流传输到另一个网络…

2023年第十六届山东省职业院校技能大赛中职组“网络安全”赛项竞赛正式试题

第十六届山东省职业院校技能大赛中职组 “网络安全”赛项竞赛试题 目录 一、竞赛时间 二、竞赛阶段 三、竞赛任务书内容 &#xff08;一&#xff09;拓扑图 &#xff08;二&#xff09;A模块基础设施设置/安全加固&#xff08;200分&#xff09; &#xff08;三&#xf…

【学习记录】从0开始的Linux学习之旅——驱动模块编译与加载

一、概述 Linux操作系统通常是基于Linux内核&#xff0c;并结合GNU项目中的工具和应用程序而成。Linux操作系统支持多用户、多任务和多线程&#xff0c;具有强大的网络功能和良好的兼容性。本文主要讲述如何编译及加载linux驱动模块。 二、概念及原理 应用程序通过系统调用与内…

【C/C++】如何不使用 sizeof 求数据类型占用的字节数

实现代码&#xff1a; #include <stdio.h>#define GET_TYPE_SIZE(TYPE) ((char *)(&TYPE 1) - (char *) & TYPE)int main(void) {char a a;short b 0;int c 0;long d 0;long long e 0;float f 0.0;double g 0.0;long double h 0.0;char* i NULL;print…

STK Components 二次开发-创建地面站

1.地面站只需要知道地面站的经纬高。 // Define the location of the facility using cartographic coordinates.var location new Cartographic(Trig.DegreesToRadians(-75.596766667), Trig.DegreesToRadians(40.0388333333), 0.0); 2.创建地面站 创建方式和卫星一样生成对…

MUI框架从新手入门【webapp开发教程】

文章目录 MUI -最接近原生APP体验的高性能前端框架APP开发3.25 开发记录miu框架介绍头部/搜索框&#xff1a;身体>轮播图轮播图设置数据自动跳转&#xff1a;九宫格图片九宫格图文列表底部选项卡按钮选择器手机模拟器 心得与总结&#xff1a;MUI框架在移动应用开发中的应用M…

Linux shell编程学习笔记30:打造彩色的选项菜单

1 需求分析 在 Linux shell编程学习笔记21&#xff1a;用select in循环语句打造菜单https://blog.csdn.net/Purpleendurer/article/details/134212033?spm1001.2014.3001.5501 中&#xff0c;我们利用select in循环语句打造的菜单中&#xff0c;菜单项都是用系统设置的颜色配…

大屏可视化编辑器

前言&#xff1a; 乐吾乐Le5le大屏可视化设计器&#xff0c;零代码实现物联网、工业智能制造等领域的可视化大屏、触摸屏端UI以及工控可视化的解决方案。同时也是一个Web组态工具&#xff0c;支持2D、3D等多种形式&#xff0c;用于构建具有实时数据展示、监控预警、丰富交互的组…

一、Spring_IOCDI(1)

&#x1f33b;&#x1f33b; 目录 一、前提介绍1.1 为什么要学?1.2 学什么?1.3 怎么学? 二、Spring相关概念2.1 初始Spring2.1.1 Spring家族2.1.2 了解 Spring 发展史 2.2 Spring系统架构2.2.1 系统架构图2.2.2 课程学习路线 2.3 Spring核心概念2.3.1 目前项目中的问题2.3.2…

【古诗生成AI实战】之一——实战项目总览

[1] 总览 【古诗生成AI实战】系列共五篇文章&#xff1a; 【古诗生成AI实战】之一——实战项目总览   【古诗生成AI实战】之二——项目架构设计   【古诗生成AI实战】之三——任务加载器与预处理器   【古诗生成AI实战】之四——模型包装器与模型的训练   【古诗生成AI…