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

相关文章

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略 1. 特权模式限制2. 宿主机资源隔离3. 用户和组管理4. 权限提升控制5. SELinux配置 💖The Begin💖点点关注,收藏不迷路💖 Kubernetes的PodSecurityPolicy(PSP)是一个关键的安全特性,它在Pod创建之前实施安全策略,确保P