CloudCompare二次开发之如何设计界面ui与功能实现?

2023-11-05 12:20

本文主要是介绍CloudCompare二次开发之如何设计界面ui与功能实现?,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 0.引言
  • 1.创建界面ui相关文件
  • 2.添加界面ui相关文件到CloudCompare工程
  • 3.修改工程相关文件
  • 4.结果展示

0.引言

  CloudCompare源代码编译成功后,即可进行二次开发,可以通过修改源码实现二次开发,二次开发基础功能见(CloudCompare如何进行二次开发?),若想要实现更多自定义功能,可以自定义界面ui,并操作CloudCompare程序处理数据。本文讲解界面ui设计,修改相应的文件,使CloudCompare能够被扩展的ui界面进行克隆点云操作。

1.创建界面ui相关文件

  (1)在CloudCompare工程源码文件下新建Qt工程
  在这里插入图片描述

  本文在E:\CloudCompare-2.10.x\qCC\文件夹下创建初始Qt文件(如何在VS中使用Qt创建Qt工程详见:Visual Studio如何使用Qt开发桌面软件?)。
  在这里插入图片描述

  本文根据创建的Qt文件,主要使用的文件目录为:E:\CloudCompare-2.10.x\qCC\MyForm\MyForm\
  将该目录包含进CloudCompare搜索路径:CloudCompare工程→属性→配置属性→VC++目录→包含目录和库目录。
  在这里插入图片描述

2.添加界面ui相关文件到CloudCompare工程

  (1)用Visual Studio打开CloudCompare源码工程,添加MyForm项目的文件.ui、.h、.cpp
  ①工程新建一个筛选器存放MyForm相关文件
  在这里插入图片描述

  ②添加文件
  在这里插入图片描述

  (2)设计界面
  在这里插入图片描述

3.修改工程相关文件

  (1)修改MyForm相关文件
  ①修改MyForm.ui属性,并编译
  在这里插入图片描述

//命令行:填入以下内容
setlocal  
D:\Qt\Qt5.9.6\5.9.6\msvc2015\bin\uic.exe -o E:\CloudCompare-2.10.x\qCC\MyForm\MyForm\ui_MyForm.h E:\CloudCompare-2.10.x\qCC\MyForm\MyForm\MyForm.ui  
if %errorlevel% neq 0 goto :cmEnd  
:cmEnd  
endlocal & call :cmErrorLevel %errorlevel% & goto :cmDone  
:cmErrorLevel  
exit /b %1  
:cmDone  
if %errorlevel% neq 0 goto :VCEnd  //说明:填入以下内容  
Generating ui_MyForm.h  //输出:填入以下内容  
E:\CloudCompare-2.10.x\qCC\MyForm\MyForm\ui_MyForm.h

  在这里插入图片描述

  将生成的ui_MyForm.h加入工程。
  在这里插入图片描述

  ②修改MyForm.h属性,并编译
  在这里插入图片描述

//命令行:填入以下内容
setlocal  
D:\Qt\Qt5.9.6\5.9.6\msvc2015\bin\moc.exe E:\CloudCompare-2.10.x\qCC\MyForm\MyForm\MyForm.h -o E:\CloudCompare-2.10.x\qCC\MyForm\MyForm\moc_MyForm.cpp  
if %errorlevel% neq 0 goto :cmEnd  
:cmEnd  
endlocal & call :cmErrorLevel %errorlevel% & goto :cmDone  
:cmErrorLevel  
exit /b %1  
:cmDone  
if %errorlevel% neq 0 goto :VCEnd  //说明:填入以下内容  
Generating moc_MyForm.cpp  //输出:填入以下内容  
moc_MyForm.cpp  //附加依赖项:填入以下内容  
‪moc.exe;MyForm.h

  在这里插入图片描述

  将生成的moc_MyForm.cpp加入工程。
  在这里插入图片描述

  ③修改MyForm.h代码
  在这里插入图片描述

#pragma once
#include "ccOverlayDialog.h"  
#include "ui_MyForm.h"  class QMdiSubWindow;  
class ccGenericPointCloud;  
class ccPointCloud;  
class MainWindow;  class MyForm :public ccOverlayDialog  
{  Q_OBJECT  
public:  //MyFirstQt(QWidget *parent = Q_NULLPTR);  explicit MyForm(QWidget* parent, ccPointCloud* pc);  virtual ~MyForm() override;  bool start() override;  public slots:  void onClone();   //点云克隆  void GetSelectedCloud();  void onClear();  void closeForm();//关闭窗体  
private:  Ui::MyFormClass* m_ui = nullptr;  //功能界面  ccPointCloud* m_cloud = nullptr;  //DB-Tree中被选中的点云  MainWindow* m_app = nullptr;   //主窗体单例  
};

  ④修改MyForm.cpp代码

#pragma once
#include "MyForm.h"  
#include "ccGLWindow.h"  
#include <ccGLWidget.h>  
#include <QMdiSubWindow>  
#include "ccPointCloud.h"  
#include "mainwindow.h"  
#include "ccDBRoot.h"  
#include "ccConsole.h"  MyForm::MyForm(QWidget* parent, ccPointCloud* pc)  :ccOverlayDialog(parent)  , m_ui(new Ui::MyFormClass)  , m_cloud(pc)  {  //m_ui->setupUi(this);  m_ui->setupUi(this);  //获取主窗体单例  m_app = MainWindow::TheInstance();  //信号槽连接  connect(m_ui->pushButton, &amp;QAbstractButton::clicked, this, &amp;MyForm::onClone);  connect(m_ui->pushButton_2, &amp;QAbstractButton::clicked, this, &amp;MyForm::closeForm);  
}  
MyForm::~MyForm()  
{  if (m_ui) {  delete m_ui;  m_ui = nullptr;  }  if (m_cloud) {  delete m_cloud;  m_cloud = nullptr;  }  
}  bool MyForm::start()  
{  ccOverlayDialog::start();  m_processing = false;  return true;  
}  void MyForm::GetSelectedCloud()  
{  if (m_cloud)  {  //设置取消选中上一块点云,同时包围盒消失  m_cloud->setSelected(false);  }  //DB-Tree中所有被选中的实体存入container  ccHObject::Container container;  m_app->db()->getSelectedEntities(container);  if (container.size())  {  //拿到第一个被选中的实体  ccHObject* ent = container[0];  if (!ent->isKindOf(CC_TYPES::POINT_CLOUD))  {  ccConsole::Error(QStringLiteral("选择的对象不是点云类型!"));  return;  }  ccGenericPointCloud* cloud = ccHObjectCaster::ToGenericPointCloud(ent);  m_cloud = static_cast<ccPointCloud*>(cloud);  //设置选中当前点云,同时显示包围盒  m_cloud->setSelected(true);  }  //刷新  m_app->RefreshAllGLWindow();  
}  void MyForm::onClear()  
{  m_cloud->clear();  m_cloud = nullptr;  
}  void MyForm::onClone()  
{  if (!m_cloud)  {  return;  }  //法一(点到点的克隆)  ccPointCloud* pc = new ccPointCloud(m_cloud->getName() + QString("-Clone"));  //为克隆对象分配内存  pc->reserve(m_cloud->size());  size_t pointSize = m_cloud->size();  for (size_t i = 0; i < pointSize; ++i)  {  pc->addPoint(*m_cloud->getPoint(i));  }  //法二(调用ccPointCloud的接口克隆)  //ccPointCloud* pc = new ccPointCloud(m_cloud->getName() + QString("-Clone"));  //m_cloud->clone(pc);  //将克隆点云pc添加到与m_cloud相同的目录下,并选中。  m_cloud->getParent()->addChild(pc);  m_app->db()->selectEntity(pc);  m_app->addToDB(pc);  m_app->UpdateUI();  
}  void MyForm::closeForm()  
{  this->close();  
}

  (2)修改mainwindow源文件
  ①修改mainwindow.h代码
  在这里插入图片描述

  ②修改mainwindow.cpp代码
  在这里插入图片描述

void MainWindow::doclone()
{  //判断DB-Tree内是否选中对象  if (!haveOneSelection())  {  ccConsole::Error(QStringLiteral("请在 DB Tree 内选择点云实体对象!"));  return;  }  //获取当前DB-Tree中选择的实体对象  ccHObject* ent = m_selectedEntities[0];  //判断是否为点云对象(这里也可能是矢量栅格对象)  if (!ent->isKindOf(CC_TYPES::POINT_CLOUD))  {  ccConsole::Error(QStringLiteral("选择的对象不是点云类型!"));  return;  }  //将实体对象ent转换成ccGenericPointCloud对象  ccGenericPointCloud* gCloud = ccHObjectCaster::ToGenericPointCloud(ent);  //ccGenericPointCloud是抽象类,不能被实例化,所以向下转换成子类对象ccPointCloud,我们可以通过ccPointCloud来操作点云  ccPointCloud* pCloud = static_cast<ccPointCloud*>(gCloud);  //Qt MDI框架  QMdiSubWindow* qWin = m_mdiArea->activeSubWindow();  if (!qWin)  return;  if (!myForm)  {  //自定义对象m_colorDlg传入点云pCloud  myForm = new MyForm(qWin, pCloud);  //连接信号槽:使得后台可以实时获取用户在DB-Tree内所选中的点云  connect(m_ccRoot, &amp;ccDBRoot::selectionChanged, myForm, &amp;MyForm::GetSelectedCloud);  //当DB-Tree为空时,清空点云  connect(m_ccRoot, &amp;ccDBRoot::dbIsEmpty, myForm, &amp;MyForm::onClear);  //使得m_colorDlg能够停靠在点云显示窗口的右上角  registerOverlayDialog(myForm, Qt::TopRightCorner);  }  //显示m_colorDlg  myForm->start();  updateOverlayDialogsPlacement();  
}

(3)生成CloudCompare
  在这里插入图片描述

4.结果展示

  在这里插入图片描述

参考资料:
[1] cacrle. Visual Studio如何使用Qt开发桌面软件?; 2023-04-18 [accessed 2023-04-19].
[2] cacrle. CloudCompare如何进行二次开发?; 2023-04-19 [accessed 2023-04-19].
[3] 问也去. CloudCompare实现点选点云功能; 2021-09-23 [accessed 2023-04-17].
[4] 进击の小黑. CloudCompare简单二次开发教程 上(界面设计与ui文件编译); 2020-12-17 [accessed 2023-04-17].
[5] 进击の小黑. CloudCompare简单二次开发 下(功能实现); 2020-12-18 [accessed 2023-04-17].
[6] shaomq2187. VS2019已有项目中添加Qt; 2021-11-08 [accessed 2023-04-19].
[7] wb175208. VS2013 在配置中手动添加宏定义; 2018-04-08 [accessed 2023-04-19].

这篇关于CloudCompare二次开发之如何设计界面ui与功能实现?的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/349813

相关文章

分布式锁在Spring Boot应用中的实现过程

《分布式锁在SpringBoot应用中的实现过程》文章介绍在SpringBoot中通过自定义Lock注解、LockAspect切面和RedisLockUtils工具类实现分布式锁,确保多实例并发操作... 目录Lock注解LockASPect切面RedisLockUtils工具类总结在现代微服务架构中,分布

Java使用Thumbnailator库实现图片处理与压缩功能

《Java使用Thumbnailator库实现图片处理与压缩功能》Thumbnailator是高性能Java图像处理库,支持缩放、旋转、水印添加、裁剪及格式转换,提供易用API和性能优化,适合Web应... 目录1. 图片处理库Thumbnailator介绍2. 基本和指定大小图片缩放功能2.1 图片缩放的

Python使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我

深度解析Spring Security 中的 SecurityFilterChain核心功能

《深度解析SpringSecurity中的SecurityFilterChain核心功能》SecurityFilterChain通过组件化配置、类型安全路径匹配、多链协同三大特性,重构了Spri... 目录Spring Security 中的SecurityFilterChain深度解析一、Security

Redis客户端连接机制的实现方案

《Redis客户端连接机制的实现方案》本文主要介绍了Redis客户端连接机制的实现方案,包括事件驱动模型、非阻塞I/O处理、连接池应用及配置优化,具有一定的参考价值,感兴趣的可以了解一下... 目录1. Redis连接模型概述2. 连接建立过程详解2.1 连php接初始化流程2.2 关键配置参数3. 最大连

Python实现网格交易策略的过程

《Python实现网格交易策略的过程》本文讲解Python网格交易策略,利用ccxt获取加密货币数据及backtrader回测,通过设定网格节点,低买高卖获利,适合震荡行情,下面跟我一起看看我们的第一... 网格交易是一种经典的量化交易策略,其核心思想是在价格上下预设多个“网格”,当价格触发特定网格时执行买

python设置环境变量路径实现过程

《python设置环境变量路径实现过程》本文介绍设置Python路径的多种方法:临时设置(Windows用`set`,Linux/macOS用`export`)、永久设置(系统属性或shell配置文件... 目录设置python路径的方法临时设置环境变量(适用于当前会话)永久设置环境变量(Windows系统

Mysql中设计数据表的过程解析

《Mysql中设计数据表的过程解析》数据库约束通过NOTNULL、UNIQUE、DEFAULT、主键和外键等规则保障数据完整性,自动校验数据,减少人工错误,提升数据一致性和业务逻辑严谨性,本文介绍My... 目录1.引言2.NOT NULL——制定某列不可以存储NULL值2.UNIQUE——保证某一列的每一

Python对接支付宝支付之使用AliPay实现的详细操作指南

《Python对接支付宝支付之使用AliPay实现的详细操作指南》支付宝没有提供PythonSDK,但是强大的github就有提供python-alipay-sdk,封装里很多复杂操作,使用这个我们就... 目录一、引言二、准备工作2.1 支付宝开放平台入驻与应用创建2.2 密钥生成与配置2.3 安装ali

Spring Security 单点登录与自动登录机制的实现原理

《SpringSecurity单点登录与自动登录机制的实现原理》本文探讨SpringSecurity实现单点登录(SSO)与自动登录机制,涵盖JWT跨系统认证、RememberMe持久化Token... 目录一、核心概念解析1.1 单点登录(SSO)1.2 自动登录(Remember Me)二、代码分析三、