动态旋转变色的3D爱心

之前为了在女朋友面前露一手(啊不是 闲得无聊),为了证明两年多的图像没有白学,总不能做个控制台黑窗口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;

    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/1Ll8-qv98upb6ugXLi9G5pQ?pwd=d5o2
提取码:d5o2

  • 1
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

给算法爸爸上香

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值