0
点赞
收藏
分享

微信扫一扫

动态旋转的3D爱心

点亮自己的那盏灯 2022-01-16 阅读 52
3dc++qt

之前为了在对象面前露一手(啊不是 闲得无聊),为了证明这两年多的图像没有白学,总不能做个控制台黑窗口print循环打印爱心字符串吧,于是乎做了一个可以动态旋转的3D爱心,效果如下图所示。虽然效果也很简陋,还被鄙视了(惨兮兮),但毕竟也是花了两个小时写出来的demo嘛。
在这里插入图片描述

基于Win10+QT平台开发,用了一些常用的2D/3D图像库比如OpenCV、PCL和VTK,其实主体也就百来行代码,思路就是调用库绘制已知方程的爱心点云然后在对点云进行变换的同时刷新可视化窗口。3D爱心的方程是
在这里插入图片描述

具体程序如下:
mainwindow.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>
#include <QVTKWidget.h>
#include <QMouseEvent>
#include <Windows.h>

#include <iostream>
#include <opencv2/opencv.hpp>
#include <pcl/io/pcd_io.h>
#include <pcl/io/ply_io.h>
#include <pcl/point_types.h>
#include <pcl/visualization/pcl_visualizer.h>
#include <vtkAutoInit.h>
#include <vtkRenderWindow.h>


VTK_MODULE_INIT(vtkRenderingOpenGL2)
VTK_MODULE_INIT(vtkInteractionStyle)


namespace Ui {
class MainWindow;
}

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    explicit MainWindow(QWidget *parent = 0);
    ~MainWindow();

private:
    Ui::MainWindow *ui;

    pcl::PointCloud<pcl::PointXYZRGB>::Ptr cloud;

    pcl::PointCloud<pcl::PointXYZRGB>::Ptr cloud2;

    boost::shared_ptr<pcl::visualization::PCLVisualizer> viewer;

    bool flag = true;

public slots:
    void stopLoop();

    void mousePressEvent(QMouseEvent *event);
};

#endif // MAINWINDOW_H

mainwindow.cpp

#include "mainwindow.h"
#include "ui_mainwindow.h"

MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)
{
    ui->setupUi(this);

    this->setMouseTracking(true);
    this->setFocusPolicy(Qt::StrongFocus);
    setAttribute(Qt::WA_QuitOnClose);

    cloud.reset(new pcl::PointCloud<pcl::PointXYZRGB>);
    for (float x = -1.2; x <= 1.2; x+= 0.01)
    {
        for (float y = -0.8; y <= 0.8; y+= 0.01)
        {
            for (float z = -1; z <= 1.2; z+= 0.01)
            {
                if (pow(x*x + 9 / 4 * y*y + z*z - 1, 3) - x*x*z*z*z - 9 / 80 * y*y*z*z*z <= 0)
                {
                    pcl::PointXYZRGB p;
                    p.x = x; p.y = y; p.z = z; p.r = 255; p.g = 0; p.b =0;
                    cloud->push_back(p);
                }
            }
        }
    }

    viewer.reset(new pcl::visualization::PCLVisualizer("viewer", false));
    viewer->addPointCloud(cloud, "cloud");
    viewer->setCameraPosition(0, 12, 0, 0, -1, 0, 0, 0, 1);

    ui->qvtkWidget->SetRenderWindow(viewer->getRenderWindow());
    viewer->setupInteractor(ui->qvtkWidget->GetInteractor(), ui->qvtkWidget->GetRenderWindow());
    ui->qvtkWidget->update();
}

MainWindow::~MainWindow()
{
    delete ui;
}

void MainWindow::stopLoop()
{
    flag = false;
}

void MainWindow::mousePressEvent(QMouseEvent *event)
{
    if(event->button())
    {
        srand((unsigned int)time(0));

        while(flag)
        {
            int r = rand() % 256, g = rand() % 256, b = rand() % 256;
            for(int i = 0; i < cloud->points.size(); ++i)
            {
                float x = cloud->points[i].x, y = cloud->points[i].y;
                cloud->points[i].x = cos(0.1) * x - sin(0.1) * y;
                cloud->points[i].y = sin(0.1) * x + cos(0.1) * y;
                cloud->points[i].r = r;
                cloud->points[i].g = g;
                cloud->points[i].b = b;
            }
            viewer->updatePointCloud(cloud);
            ui->qvtkWidget->update();
            cv::waitKey(1);
        }
    }
}

main.cpp

#include "mainwindow.h"
#include <QApplication>

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    MainWindow *w = new MainWindow;
    w->show();
    a.connect(&a, SIGNAL(lastWindowClosed()), w, SLOT(stopLoop()));
    return a.exec();
}

项目工程文件在此:
链接: https://pan.baidu.com/s/1UbO5ktO7zjtA1r9aXZCJAg?pwd=zbfp 提取码: zbfp 复制这段内容后打开百度网盘手机App,操作更方便哦
–来自百度网盘超级会员v2的分享

举报

相关推荐

0 条评论