Dandelion 1.1.1
A light-weight 3D builder for educational usage
载入中...
搜索中...
未找到
scene.h
浏览该文件的文档.
1#ifndef DANDELION_SCENE_SCENE_H
2#define DANDELION_SCENE_SCENE_H
3
4#include <memory>
5#include <string>
6#include <vector>
7#include <list>
8#include <optional>
9#include <chrono>
10
11#include <spdlog/spdlog.h>
12
13#include "group.h"
14#include "camera.h"
15#include "light.h"
16#include "../platform/gl.hpp"
20
21/*!
22 * \file scene/scene.h
23 * \ingroup rendering
24 */
25
26/*!
27 * \ingroup rendering
28 * \ingroup simulation
29 * \~chinese
30 * \brief 表示一个包含相机、光源、物体的完整场景。
31 *
32 * `Scene` 是一个对象语义的类,它既包含完成图形设计和渲染所需的全部数据,
33 * 也具有渲染自身预览效果的功能。当前的设计是:控制器 (Controller) 在 OpenGL
34 * 上下文被创建后即创建场景实例,该实例在程序的整个生命周期中都是唯一的。因此,
35 * 复制或移动 `Scene` 实例都是错误的行为,因为这有可能导致场景数据不一致。
36 * 未来加入修改历史记录时,此设计思路有可能被改变。
37 *
38 * 场景可以是空的,如果它非空,则可以包含若干个物体组 (Group);
39 * 每个组也可以为空或包含若干物体 (Object),物体是场景中的最小单元,不能再分了。
40 * 各种数据遵循“尽可能存储到更小单元中”的原则,因此 `Scene` 类中只存储相机、
41 * 光源这些与任何物体都无关的数据,而材质、mesh 等数据则存储到物体中。
42 * 所有物体都由 `Scene` 对象创建并持有,当 `Scene` 对象析构时,所有的组、
43 * 物体都会自行析构。
44 */
45class Scene
46{
47public:
48 Scene();
49 ///@{
50 /*! \~chinese 禁止复制场景。 */
51 Scene(Scene& other) = delete;
52 Scene(const Scene& other) = delete;
53 ///@}
54 /*! \~chinese 禁止移动场景。 */
55 Scene(Scene&& other) = delete;
56 ~Scene() = default;
57 /*!
58 * \~chinese
59 * \brief 从指定路径加载模型文件到这个场景中。
60 *
61 * 这个函数只会根据文件名创建一个物体组,然后调用物体组的 `load` 方法加载文件。
62 */
63 bool load(const std::string& file_path);
64 /*! \~chinese 备份物体当前状态并开始模拟,此后每一帧都会调用所有物体的 `update` 方法。 */
65 void start_simulation();
66 /*! \~chinese 停止模拟,此后不再调用物体的 `update` 方法。 */
67 void stop_simulation();
68 /*! \~chinese 恢复物体在动画开始前的状态。 */
69 void reset_simulation();
70 /*! \~chinese 查询当前是否正在进行物理模拟。 */
72 /*! \~chinese
73 * \brief 绘制整个场景。
74 *
75 * `render` 方法是场景对外的绘制接口,不会直接绘制任何内容,只负责调用每个 Object 的 `render`
76 * 方法、`render_camera` 和 `render_lights` 方法。
77 */
78 void render(const Shader& shader, WorkingMode mode);
79
80 /*! \~chinese 场景中所有的物体组。 */
81 std::vector<std::unique_ptr<Group>> groups;
82 /*!
83 * \~chinese
84 * \brief 当前被选中的物体。
85 *
86 * 在布局模式和物理模拟模式下,它决定当前被高亮的物体;在建模模式下,仅有这个物体被绘制。
87 */
89 /*! \~chinese 用于 **离线渲染** 的相机,和用于预览场景的观察相机(主相机)无关。 */
91 /*! \~chinese 用于 **离线渲染** 的光源,和预览时照亮物体的光源无关。 */
92 std::list<Light> lights;
93 /*! \~chinese 用于建模模式的半边网格。 */
94 std::unique_ptr<HalfedgeMesh> halfedge_mesh;
95
96private:
97 /*! \~chinese
98 * \brief 绘制空间坐标轴和 \f$y=1\f$ 平面上表示地面的网格线。
99 *
100 * 这个函数在 \f$x\f$ 和 \f$z\f$ 方向各绘制 1000 条网格线,并分别用红、绿、蓝三色绘制
101 * \f$x,y,z\f$ 三轴正半轴。
102 */
103 static void render_ground(const Shader& shader);
104 /*! \~chinese
105 * \brief 在渲染模式 (Rendering mode) 下绘制代表相机视锥的线框。
106 *
107 * 根据场景对象的相机参数(位置、目标点、远近平面、视角),绘制四棱锥形的相机视锥。
108 * 由于近平面通常离相机视点很近,这个函数不会绘制近平面。离线渲染时,
109 * 会裁剪掉视锥范围外的所有内容。
110 */
111 void render_camera(const Shader& shader);
112 /*! \~chinese
113 * \brief 在渲染模式下绘制光源。
114 *
115 * 这个函数将场景中每个点光源绘制成六个亮点。
116 */
117 void render_lights(const Shader& shader);
118 /*!
119 * \~chinese
120 * \brief 计算场景中所有物体下一帧要渲染的运动状态。
121 *
122 * 这个函数按照固定的时间步长模拟物体运动。
123 *
124 * 每一帧的渲染过程可以概括为更新数据(几何、运动等等)和渲染图像两步。
125 * 约定当前屏幕上显示的是上一帧的图像,正在计算的是当前帧的数据。
126 * 更新运动状态时首先用当前时间减去 `last_update` 得到上一帧和之前帧剩余时长之和;
127 * 再循环模拟物体运动。每循环一次走过一个长度为 `time_step` 的时间步、
128 * 剩余时长减去 `time_step` ;当剩余时长不足一个 `time_step` 时停止模拟,
129 * 并将这一帧模拟走过的总时长累加到 `last_update` 上。
130 */
131 void simulation_update();
132 /*! \~chinese 状态变量,表示当前是否正在进行物理模拟。 */
134 /*! \~chinese 上一次将模拟状态同步到渲染的时间点。 */
135 std::chrono::time_point<std::chrono::steady_clock> last_update;
136 /*! \~chinese 碰撞检测时记录所有物体,其他情况下无效。 */
137 std::vector<Object*> all_objects;
138 /*! \~chinese 用于在物理模拟模式下显示速度向量。 */
140 /*! \~chinese 日志记录器。 */
141 std::shared_ptr<spdlog::logger> logger;
142};
143
144#endif // DANDELION_SCENE_SCENE_H
表示物体的类。
定义 object.h:40
bool during_animation
定义 scene.h:133
void reset_simulation()
定义 scene.cpp:224
bool check_during_simulation()
定义 scene.cpp:237
Scene(Scene &other)=delete
void render_lights(const Shader &shader)
在渲染模式下绘制光源。
定义 scene.cpp:151
void render(const Shader &shader, WorkingMode mode)
绘制整个场景。
定义 scene.cpp:242
std::vector< std::unique_ptr< Group > > groups
定义 scene.h:81
void render_camera(const Shader &shader)
在渲染模式 (Rendering mode) 下绘制代表相机视锥的线框。
定义 scene.cpp:104
static void render_ground(const Shader &shader)
绘制空间坐标轴和 平面上表示地面的网格线。
定义 scene.cpp:44
std::shared_ptr< spdlog::logger > logger
定义 scene.h:141
void stop_simulation()
定义 scene.cpp:219
GL::LineSet arrows
定义 scene.h:139
std::list< Light > lights
定义 scene.h:92
void start_simulation()
定义 scene.cpp:202
bool load(const std::string &file_path)
从指定路径加载模型文件到这个场景中。
定义 scene.cpp:186
std::unique_ptr< HalfedgeMesh > halfedge_mesh
定义 scene.h:94
std::chrono::time_point< std::chrono::steady_clock > last_update
定义 scene.h:135
std::vector< Object * > all_objects
定义 scene.h:137
Object * selected_object
当前被选中的物体。
定义 scene.h:88
Scene(Scene &&other)=delete
void simulation_update()
计算场景中所有物体下一帧要渲染的运动状态。
定义 scene.cpp:297
Camera camera
定义 scene.h:90
对 GLSL Shader 的简单封装。
定义 shader.hpp:24
半边网格所需各种类型的公共头文件。
这个文件定义了一些和渲染(离线渲染或场景预览)相关的常量、枚举等。
WorkingMode
定义 rendering.hpp:46
表示观察点的相机,既可以用于预览视角,也可以用于渲染视角。
定义 camera.h:22
在预览场景时绘制若干线条。
定义 gl.hpp:270