Dandelion 1.1.2
A light-weight 3D builder for educational usage
载入中...
搜索中...
未找到
controller.h
浏览该文件的文档.
1#ifndef DANDELION_UI_CONTROLLER_H
2#define DANDELION_UI_CONTROLLER_H
3
4/*!
5 * \file ui/controller.h
6 */
7
8#include <memory>
9
10#include <spdlog/spdlog.h>
11#include <nlohmann/json.hpp>
12
13#include "menubar.h"
14#include "toolbar.h"
15#include "selection_helper.h"
16#include "../platform/gl.hpp"
18#include "../scene/scene.h"
19
20/*!
21 * \ingroup ui
22 * \~chinese
23 * \brief 控制器管理所有的界面组件,并处理和预览视角操作(例如旋转、缩放或者平移)相关的输入。
24 *
25 * 控制器的生命周期就是整个程序 GUI 的生命周期,因此这是一个单例的类。其唯一一个实例可以通过
26 * `controller()` 这个静态方法访问,访问操作是线程安全的(但访问后的修改不是)。
27 * 控制器创建并持有所有与图形界面相关的资源:
28 *
29 * - 界面元素(菜单栏和工具栏)
30 * - 场景(包含所有物体、坐标轴等)
31 *
32 * 界面元素本身不存储场景,也不保存指向场景的引用等。控制器在每一帧将场景的引用传递给
33 * `Toolbar::render` 或 `Menubar::render` 函数,从而允许这些界面元素访问场景。
34 * 与 `Menubar` 或 `Toolbar` 保存引用相比,这种设计当前并没有额外有点,它只是参考 Scotty3D
35 * 思路的结果,将来有可能为引入全局撤销 (Undo) 和重做 (Redo) 操作提供方便。
36 */
38{
39public:
40
41 /*!
42 * \~chinese
43 * \brief 获取 `Controller` 类的唯一实例。
44 *
45 * 唯一的实例被定义成函数内的静态变量,在 C++ 11 及之后的标准中,这种定义方式是线程安全的。
46 */
47 static Controller& controller();
48 ///@{
49 /*! \~chinese 禁止复制构造。*/
50 Controller(Controller& other) = delete;
51 Controller& operator=(Controller& other) = delete;
52 ///@}
54 /*!
55 * \~chinese
56 * \brief 将鼠标拖动转换为旋转视角或平移视角操作。
57 *
58 * 当按下鼠标中键(或 Alt+鼠标左键)拖动时旋转视角,按下 Ctrl+鼠标左键拖动时平移视角。
59 * 同时按下 Alt 和 Ctrl 的效果与只按下 Alt 相同(只旋转不平移)。
60 *
61 * \param initial 为真表示开始拖动时调用,为假表示拖动过程中调用。
62 */
63 void on_mouse_dragged(bool initial);
64 /*!
65 * \~chinese
66 * \brief 根据鼠标点击选取场景中的物体。
67 *
68 * 这个函数将鼠标点击的屏幕坐标 (screen space) 转换为观察坐标 (view space),
69 * 然后将观察坐标转换回世界坐标 (world space),从而构造一条从观察相机(主相机)
70 * 发出的射线,按这条射线与场景中物体的相交结果拾取(选中)场景中的可选中对象。
71 * 工作于布局模式和物理模拟模式下时,它调用 `pick_object` 函数,工作于建模模式下时,
72 * 它调用 `pick_element` 函数。
73 */
74 void on_picking();
75 /*!
76 * \~chinese
77 * \brief 将拨动鼠标滚轮的操作转换为视角拉进或远离。
78 *
79 * 缩放视角的操作按指数规律执行:鼠标滚轮每滚动一个单位,观察相机到目标点的距离就乘上一个固定值
80 * `wheel_scroll_factor`,即
81 *
82 * \f[
83 * \mathbf{d}_\text{new}=\text{wheel_scroll_factor}^\text{wheel_delta}\cdot\mathbf{d}
84 * \f]
85 */
86 void on_wheel_scrolled();
87 /*!
88 * \~chinese
89 * \brief 在缩放窗口时调整轨迹球半径、预览视角(相机)Y 方向的 FoV,并更新自身记录的窗口尺寸。
90 *
91 * 虽然名字相同,但这个函数并非 GLFW 的窗口缩放回调函数。在缩放时它会被真正的回调函数
92 * `Platform::on_framebuffer_resized` 调用。
93 * \param width 缩放后的窗口宽度
94 * \param height 缩放后的窗口高度
95 */
96 void on_framebuffer_resized(float width, float height);
97 /*!
98 * \~chinese
99 * \brief 处理关于场景操作的输入。
100 *
101 * 这个函数判别鼠标操作是否对应于某种场景操作(调整视角、拾取物体等),并调用相应的处理函数
102 * (例如 `on_mouse_clicked` 或 `on_picking`)。与场景操作相对的是控件操作(例如点击按钮),
103 * 控件操作由相应的 ImGui 控件直接处理。
104 */
105 void process_input();
106 /*!
107 * \~chinese
108 * 渲染场景和各个 UI 组件。控制器本身不直接渲染任何内容,而是调用相应类的 `render()` 方法。
109 * \param shader 当前渲染使用的 shader,用于设置 shader 中的全局变量。
110 */
111 void render(const Shader& shader);
112 /*!
113 * \~chinese
114 * 在加载场景前重置 UI 的状态,避免加载场景时控制器仍有不当的数据引用。
115 */
117 /*!
118 * \~english All variables used for layout are float because ImVec2 expects float.
119 * \~chinese 由于 ImGui 使用浮点数表示尺寸,所有和布局相关的变量都是浮点型。
120 */
121 ///@{
124 ///@}
125
126private:
127
128 /*!
129 * \~chinese
130 * \brief 选择一个元素。
131 *
132 * 这个函数修改控制器记录的选择状态。它包括两步动作:
133 * 1. 调用 `unselect` 清除原先的选择状态及其副作用
134 * 2. 修改被选中元素 `selected_element`,调用 `select_[type]` 函数执行具体操作
135 *
136 * \param element 要选择的新元素
137 */
138 void select(SelectableType element);
139 /*!
140 * \~chinese
141 * \brief 取消选择。
142 *
143 * 这个函数清楚当前的选择状态,将`selected_element` 重置为 `std::monostate`
144 * 并清空 `highlighted_element` 或 `highlighted_halfedge` 这些高亮元素。
145 * 如果之前的被选中元素设置过 `Scene` 等对象的属性,它也会一并将其重置。
146 */
147 void unselect();
148 /*!
149 * \~chinese
150 * \brief 渲染当前选中的元素。
151 *
152 * 渲染选中元素时会禁用深度检测 `GL_DEPTH_TEST` ,直接在原先的绘制结果上叠加。
153 * 因此,即使将视角旋转到背面也会看到高亮出来的被选中元素。
154 */
155 void render_selected_element(const Shader& shader);
156 /*!
157 * \~chinese
158 * \brief 渲染帮助调试的元素。
159 *
160 * 根据 `debug_options` 中的各项设置,渲染帮助调试的结构,如 BVH
161 * 的所有包围盒等。
162 */
163 void render_debug_helpers(const Shader& shader);
164 /*!
165 * \~chinese
166 * \brief 拾取物体。
167 *
168 * 在布局或物理模拟模式下,根据射线求交的结果选择物体。
169 * \param ray 根据点击位置生成的射线(世界坐标系下)
170 */
171 void pick_object(Ray& ray);
172 /*!
173 * \~chinese
174 * \brief 拾取半边、顶点、边或面片。
175 *
176 * 在建模模式下,根据射线求交的结果选择半边网格上的基本元素。
177 * \param ray 根据点击位置生成的射线(世界坐标系下)
178 */
179 void pick_element(Ray& ray);
180 /*! \~chinese 选择物体并更新 `Scene::selected_object` 。 */
181 void select_object(Object* object);
182 /*! \~chinese 选择半边并更新 `highlighted_halfedge` 。 */
183 void select_halfedge(const Halfedge* halfedge);
184 /*! \~chinese 选择顶点并更新 `highlighted_element` 。 */
185 void select_vertex(Vertex* vertex);
186 /*! \~chinese 选择边并更新 `highlighted_element` 。 */
187 void select_edge(Edge* edge);
188 /*! \~chinese 选择面片并更新 `highlighted_element` 。 */
189 void select_face(Face* face);
190 /*! \~chinese 选择光源并更新 `highlighted_element` 。 */
191 void select_light(Light* light);
192 /*!
193 * \~chinese
194 * \brief 将鼠标拖动转换为轨迹球 (Trackball) 操作,用于旋转预览视角。
195 *
196 * 轨迹球曲面的推导参考 [Object Mouse
197 * Trackball](https://www.khronos.org/opengl/wiki/Object_Mouse_Trackball)。
198 * \param initial 为真表示开始拖动时调用,为假表示拖动过程中调用。
199 */
200 void on_rotating(bool initial);
201 /*!
202 * \~chinese
203 * \brief 将鼠标拖动转换为视角平移。
204 *
205 * 将鼠标拖动时屏幕坐标的变化量换算为观察空间 (view space) 中的坐标变化,
206 * 光标向右移动对应相机向左移动,光标向上移动对应相机向下移动。
207 * 相应的变化量会累加到主相机的 `position` 和 `target` 属性上,因此平移过程中视线方向不会改变。
208 *
209 * 视点到观察点的距离增大(减小)时,相同屏幕坐标变化量对应的观察坐标变化量也增大(减小),
210 * 平移量正比于视角远近,比例系数为 `mouse_translation_factor` 。
211 * \param initial 为真表示开始拖动时调用,为假表示拖动过程中调用。
212 */
213 void on_translating(bool initial);
214 /*! \~chinese 缩放视角的控制系数,详见 `on_wheel_scrolled` 方法。 */
215 static constexpr float wheel_scroll_factor = 0.8f;
216 /*! \~chinese 平移视角时屏幕坐标到观察坐标换算系数的固定部分。 */
217 static constexpr float mouse_translation_factor = 0.001f;
218 /*! \~chinese 构造函数是私有的。 */
219 Controller();
220 /*!
221 * \~chinese
222 * \brief 全局工作模式。
223 *
224 * 控制器维护着当前的工作模式,决定各种功能可用或禁用。其他的 GUI 组件持有对该变量的引用。
225 */
227 /*! \~chinese 一些帮助调试的选项,详见 `UI::DebugOptions` 类型说明。 */
229 /*! \~chinese 菜单栏。 */
230 std::unique_ptr<UI::Menubar> menubar;
231 /*! \~chinese 工具栏。 */
232 std::unique_ptr<UI::Toolbar> toolbar;
233 /*! \~chinese 包含所有三维数据的场景实例。 */
234 std::unique_ptr<Scene> scene;
235 /*!
236 * \~chinese
237 * \brief 当前被选中的元素。
238 *
239 * 根据当前所处的模式,物体、各类几何基本元素、光源都可能被选中,详见 `SelectableType`
240 * 的类型说明。当 `selected_element` 持有 `std::monostate` 类型时,
241 * 当前的选择状态为空(没有任何元素被选中)。
242 */
244 /*! \~chinese 日志记录器。 */
245 std::shared_ptr<spdlog::logger> logger;
246 /*! \~chinese 当前的轨迹球半径,决定轨迹球控制曲面上球面和双曲面部分的相切位置。 */
248 /*! \~chinese 被选中元素类型为顶点、边、面片或光源时使用的绘制对象。 */
250 /*! \~chinese 被选中元素类型为半边时使用的绘制对象。 */
252 /*! \~chinese 显示拾取射线用的绘制对象,对应 `UI::DebugOptions::show_picking_ray` 。 */
254};
255
256#endif // DANDELION_UI_CONTROLLER_H
void select_object(Object *object)
定义 controller.cpp:476
void select_light(Light *light)
定义 controller.cpp:527
void select_halfedge(const Halfedge *halfedge)
定义 controller.cpp:482
void on_rotating(bool initial)
将鼠标拖动转换为轨迹球 (Trackball) 操作,用于旋转预览视角。
定义 controller.cpp:540
void render_debug_helpers(const Shader &shader)
渲染帮助调试的元素。
定义 controller.cpp:368
std::unique_ptr< UI::Toolbar > toolbar
定义 controller.h:232
void on_framebuffer_resized(float width, float height)
在缩放窗口时调整轨迹球半径、预览视角(相机)Y 方向的 FoV,并更新自身记录的窗口尺寸。
定义 controller.cpp:146
void on_wheel_scrolled()
将拨动鼠标滚轮的操作转换为视角拉进或远离。
定义 controller.cpp:136
void select_vertex(Vertex *vertex)
定义 controller.cpp:490
void on_translating(bool initial)
将鼠标拖动转换为视角平移。
定义 controller.cpp:597
WorkingMode mode
全局工作模式。
定义 controller.h:226
void render_selected_element(const Shader &shader)
渲染当前选中的元素。
定义 controller.cpp:306
void return_to_safe_state()
定义 controller.cpp:619
std::shared_ptr< spdlog::logger > logger
定义 controller.h:245
void select_face(Face *face)
定义 controller.cpp:510
void pick_element(Ray &ray)
拾取半边、顶点、边或面片。
定义 controller.cpp:411
static constexpr float mouse_translation_factor
定义 controller.h:217
Controller & operator=(Controller &other)=delete
static Controller & controller()
获取 Controller 类的唯一实例。
定义 controller.cpp:34
void unselect()
取消选择。
定义 controller.cpp:269
SelectableType selected_element
当前被选中的元素。
定义 controller.h:243
static constexpr float wheel_scroll_factor
定义 controller.h:215
void render(const Shader &shader)
定义 controller.cpp:237
Controller()
定义 controller.cpp:40
GL::LineSet picking_ray
定义 controller.h:253
UI::DebugOptions debug_options
定义 controller.h:228
float window_height
定义 controller.h:122
void on_mouse_dragged(bool initial)
将鼠标拖动转换为旋转视角或平移视角操作。
定义 controller.cpp:95
GL::Mesh highlighted_element
定义 controller.h:249
void on_picking()
根据鼠标点击选取场景中的物体。
定义 controller.cpp:109
std::unique_ptr< Scene > scene
定义 controller.h:234
float window_width
定义 controller.h:122
float toolbar_width
定义 controller.h:123
Controller(Controller &other)=delete
GL::LineSet highlighted_halfedge
定义 controller.h:251
void select(SelectableType element)
选择一个元素。
定义 controller.cpp:253
float trackball_radius
定义 controller.h:247
void pick_object(Ray &ray)
拾取物体。
定义 controller.cpp:384
void select_edge(Edge *edge)
定义 controller.cpp:498
std::unique_ptr< UI::Menubar > menubar
定义 controller.h:230
void process_input()
处理关于场景操作的输入。
定义 controller.cpp:154
表示物体的类。
定义 object.h:42
对 GLSL Shader 的简单封装。
定义 shader.hpp:24
std::variant< std::monostate, Object *, const Halfedge *, Vertex *, Edge *, Face *, Light * > SelectableType
场景中可被选择的元素类型。
定义 selection_helper.h:38
WorkingMode
定义 rendering.hpp:46
包含场景的类。
半边网格中的边。
定义 halfedge.h:118
半边网格中的面片。
定义 halfedge.h:151
在预览场景时绘制若干线条。
定义 gl.hpp:334
用于场景预览渲染的 Mesh 类。
定义 gl.hpp:268
半边网格中最关键的几何元素。
定义 halfedge.h:46
表示一个点光源的类。
定义 light.h:19
定义 ray.h:24
辅助调试的 GUI 选项。
定义 menubar.h:21
半边网格中的顶点。
定义 halfedge.h:86