BlosSOM
Interactive dimensionality reduction on large datasets (EmbedSOM and FLOWER combined)
ui_renderer.cpp
Go to the documentation of this file.
1/* This file is part of BlosSOM.
2 *
3 * Copyright (C) 2021 Sona Molnarova
4 *
5 * BlosSOM is free software: you can redistribute it and/or modify it under
6 * the terms of the GNU General Public License as published by the Free
7 * Software Foundation, either version 3 of the License, or (at your option)
8 * any later version.
9 *
10 * BlosSOM is distributed in the hope that it will be useful, but WITHOUT ANY
11 * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
12 * FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
13 * details.
14 *
15 * You should have received a copy of the GNU General Public License along with
16 * BlosSOM. If not, see <https://www.gnu.org/licenses/>.
17 */
18
19#include "ui_renderer.h"
20
21#include "shaders.h"
22
24 : rect_indices({ 0, 1, 3, 1, 2, 3 })
25 , draw_rect(false)
26 , update_rect_pos(false)
27 , rect_pressed(false)
28 , is_brushing_active(false)
29 , draw_circle(false)
30{
31}
32
33bool
35{
36 glGenVertexArrays(1, &VAO_r);
37 glGenBuffers(1, &VBO_r);
38 glGenBuffers(1, &EBO_r);
39
41
42 glGenVertexArrays(1, &VAO_c);
43 glGenBuffers(1, &VBO_c);
44
46}
47
48void
50{
51 glEnable(GL_BLEND);
52
54
55 if (draw_rect) {
56 shader_r.use();
57 shader_r.set_mat4("model", glm::mat4(1.0f));
58 shader_r.set_mat4("view", view.get_view_matrix());
59 shader_r.set_mat4("proj", view.get_proj_matrix());
60
61 glBindVertexArray(VAO_r);
62 glDrawElements(GL_TRIANGLES, 6, GL_UNSIGNED_INT, 0);
63 }
64
65 if (draw_circle) {
66 shader_c.use();
67 shader_c.set_mat4("model", glm::mat4(1.0f));
68 shader_c.set_mat4("view", view.get_view_matrix());
69 shader_c.set_mat4("proj", view.get_proj_matrix());
70
71 glBindVertexArray(VAO_c);
72 glDrawArrays(GL_LINES, 0, num_all_vtxs_circle);
73 }
74
75 glDisable(GL_BLEND);
76}
77
78void
79UiRenderer::prepare_data(float current_zoom)
80{
82 prepare_circle(current_zoom);
83}
84
85bool
86UiRenderer::is_rect_pressed(glm::vec2 mouse_pos)
87{
88 float mouse_x = mouse_pos.x;
89 float mouse_y = mouse_pos.y;
90
91 float min_x = std::min(rect_vtxs[0].x, rect_vtxs[3].x);
92 float min_y = std::min(rect_vtxs[0].y, rect_vtxs[1].y);
93 float max_x = std::max(rect_vtxs[0].x, rect_vtxs[3].x);
94 float max_y = std::max(rect_vtxs[0].y, rect_vtxs[1].y);
95
96 if ((mouse_x >= min_x && mouse_x <= max_x) &&
97 (mouse_y >= min_y && mouse_y <= max_y))
98 rect_pressed = true;
99 else
100 rect_pressed = false;
101
102 max_diff_x = max_x - mouse_x;
103 min_diff_x = mouse_x - min_x;
104 max_diff_y = max_y - mouse_y;
105 min_diff_y = mouse_y - min_y;
106
107 return rect_pressed;
108}
109
110void
112{
113 draw_rect = update_rect_pos = true;
114
115 rect_vtxs[0] = mouse_pos;
116 rect_vtxs[1] = mouse_pos;
117 rect_vtxs[2] = mouse_pos;
118 rect_vtxs[3] = mouse_pos;
119}
120
121void
122UiRenderer::set_rect_end_point(glm::vec2 mouse_pos, const LandmarkModel &model)
123{
124 float delta_x = mouse_pos.x - rect_vtxs[3].x;
125 float delta_y = mouse_pos.y - rect_vtxs[3].y;
126 rect_vtxs[0] = { rect_vtxs[3].x + delta_x, rect_vtxs[3].y };
127 rect_vtxs[1] = { rect_vtxs[3].x + delta_x, rect_vtxs[3].y + delta_y };
128 rect_vtxs[2] = { rect_vtxs[3].x, rect_vtxs[3].y + delta_y };
129
130 selected_landmarks.clear();
131
132 const auto &vertices = model.lodim_vertices;
133 for (size_t i = 0; i < vertices.size(); ++i) {
134 if (is_within_rect(vertices[i]))
135 selected_landmarks.emplace_back(i);
136 }
137}
138
139void
140UiRenderer::move_selection(glm::vec2 mouse_pos, LandmarkModel &landmarks)
141{
142 float x = mouse_pos.x;
143 float y = mouse_pos.y;
144
145 auto new_upper_r = glm::vec2(x + max_diff_x, y + max_diff_y);
146 auto new_bottom_r = glm::vec2(x + max_diff_x, y - min_diff_y);
147 auto new_bottom_l = glm::vec2(x - min_diff_x, y - min_diff_y);
148 auto new_upper_l = glm::vec2(x - min_diff_x, y + max_diff_y);
149
150 float delta_x = new_upper_r.x - rect_vtxs[0].x;
151 float delta_y = new_upper_r.y - rect_vtxs[0].y;
152
153 for (size_t i = 0; i < selected_landmarks.size(); ++i) {
154 size_t ind = selected_landmarks[i];
155 auto old_pos = landmarks.lodim_vertices[ind];
156 glm::vec2 new_pos = old_pos + glm::vec2(delta_x, delta_y);
157 landmarks.move(ind, new_pos);
158 }
159
160 rect_vtxs[0] = new_upper_r;
161 rect_vtxs[1] = new_bottom_r;
162 rect_vtxs[2] = new_bottom_l;
163 rect_vtxs[3] = new_upper_l;
164}
165
166void
167UiRenderer::should_draw_circle(const View &view, glm::vec2 mouse_pos, float r)
168{
169 // Convert radius from screen to model space
170 // Compute mouse + radius point in model
171 auto right = mouse_pos + glm::vec2(r, 0);
172 auto model_right = view.model_mouse_coords(right);
173 auto model_mouse = view.model_mouse_coords(mouse_pos);
174 float model_radius = fabs((model_right - model_mouse).x);
175
176 draw_circle = true;
177 circle_pos = model_mouse;
178 circle_radius = model_radius;
179}
180
181void
183{
184 glBindVertexArray(VAO_r);
185
186 glBindBuffer(GL_ARRAY_BUFFER, VBO_r);
187 glBufferData(GL_ARRAY_BUFFER,
188 rect_vtxs.size() * sizeof(glm::vec2),
189 &rect_vtxs[0],
190 GL_DYNAMIC_DRAW);
191
192 glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, EBO_r);
193 glBufferData(GL_ELEMENT_ARRAY_BUFFER,
194 rect_indices.size() * sizeof(unsigned int),
195 &rect_indices[0],
196 GL_STATIC_DRAW);
197
198 glVertexAttribPointer(
199 0, 2, GL_FLOAT, GL_FALSE, sizeof(glm::vec2), (void *)0);
200 glEnableVertexAttribArray(0);
201}
202
203void
205{
206 std::vector<float> circle_vtxs;
207
208 int sides = 24;
209 num_all_vtxs_circle = sides * 2;
210
211 double two_pi = 2.0f * M_PI;
212
213 for (int i = 0; i < sides + 1; ++i) {
214 float x_coor = circle_pos.x + (circle_radius * cos(i * two_pi / sides));
215 float y_coor = circle_pos.y + (circle_radius * sin(i * two_pi / sides));
216
217 circle_vtxs.emplace_back(x_coor);
218 circle_vtxs.emplace_back(y_coor);
219
220 // Add each point twice --- end of line and start
221 // of next line.
222 if (i != 0 && i != sides) {
223 circle_vtxs.emplace_back(x_coor);
224 circle_vtxs.emplace_back(y_coor);
225 }
226 }
227
228 glBindVertexArray(VAO_c);
229
230 glBindBuffer(GL_ARRAY_BUFFER, VBO_c);
231 glBufferData(GL_ARRAY_BUFFER,
232 circle_vtxs.size() * sizeof(float),
233 &circle_vtxs[0],
234 GL_DYNAMIC_DRAW);
235 glVertexAttribPointer(
236 0, 2, GL_FLOAT, GL_FALSE, 2 * sizeof(float), (void *)0);
237 glEnableVertexAttribArray(0);
238}
239
240bool
241UiRenderer::is_within_rect(glm::vec2 point) const
242{
243 float point_x = point.x;
244 float point_y = point.y;
245
246 float min_x = std::min(rect_vtxs[0].x, rect_vtxs[3].x);
247 float min_y = std::min(rect_vtxs[0].y, rect_vtxs[1].y);
248 float max_x = std::max(rect_vtxs[0].x, rect_vtxs[3].x);
249 float max_y = std::max(rect_vtxs[0].y, rect_vtxs[1].y);
250
251 if ((point_x >= min_x && point_x <= max_x) &&
252 (point_y >= min_y && point_y <= max_y))
253 return true;
254 else
255 return false;
256}
257
258bool
259UiRenderer::is_within_circle(const glm::vec2 &vert,
260 const glm::vec2 &pos,
261 float radius)
262{
263 if ((pos.x + radius >= roundf(vert.x)) &&
264 (pos.x - radius <= roundf(vert.x)) &&
265 (pos.y + radius >= roundf(vert.y)) &&
266 (pos.y - radius <= roundf(vert.y))) {
267 return true;
268 }
269}
void set_mat4(const std::string &name, glm::mat4 value) const
Bind the matrix 4x4 variable to the shader.
Definition: shader.cpp:103
void use()
Activate built shader.
Definition: shader.cpp:79
void build(const std::string &vs, const std::string &fs)
Read and build the shader.
Definition: shader.cpp:27
A small utility class that manages the viewport coordinates, together with the virtual "camera" posit...
Definition: view.h:37
glm::mat4 get_proj_matrix() const
Compute projection matrix for orthographic projection.
Definition: view.h:108
float current_zoom
Definition: view.h:53
glm::mat4 get_view_matrix() const
Compute view matrix for drawing into the "view" space.
Definition: view.h:98
glm::vec2 model_mouse_coords(glm::vec2 mouse) const
Convert mouse coordinates ([0,0] in the upper left corner), to model coordinates ([0,...
Definition: view.h:139
const std::string ui_r_vs
Definition: shaders.h:124
const std::string ui_r_fs
Definition: shaders.h:134
const std::string ui_c_fs
Definition: shaders.h:151
const std::string ui_c_vs
Definition: shaders.h:141
Model of the high- and low-dimensional landmarks.
void move(size_t ind, const glm::vec2 &mouse_pos)
Sets two-dimensional position of the pressed landmark to mouse position.
std::vector< glm::vec2 > lodim_vertices
Array storing two-dimensional landmark coordinates.
glm::vec2 circle_pos
Definition: ui_renderer.h:85
float max_diff_x
Definition: ui_renderer.h:78
unsigned int VBO_r
Definition: ui_renderer.h:67
float min_diff_y
Definition: ui_renderer.h:81
void draw(const View &v)
Definition: ui_renderer.cpp:49
unsigned int VBO_c
Definition: ui_renderer.h:73
void set_rect_start_point(glm::vec2 mouse_pos)
bool is_within_rect(glm::vec2 point) const
const std::array< unsigned int, 6 > rect_indices
Definition: ui_renderer.h:76
bool update_rect_pos
Definition: ui_renderer.h:38
Shader shader_r
Definition: ui_renderer.h:65
bool is_rect_pressed(glm::vec2 mouse_pos)
Definition: ui_renderer.cpp:86
bool draw_rect
Definition: ui_renderer.h:37
int num_all_vtxs_circle
Definition: ui_renderer.h:70
bool draw_circle
Definition: ui_renderer.h:45
std::vector< size_t > selected_landmarks
Definition: ui_renderer.h:83
std::array< glm::vec2, 4 > rect_vtxs
Definition: ui_renderer.h:75
void prepare_data(float current_zoom)
Definition: ui_renderer.cpp:79
void should_draw_circle(const View &view, glm::vec2 mouse_pos, float r)
unsigned int EBO_r
Definition: ui_renderer.h:68
float min_diff_x
Definition: ui_renderer.h:79
unsigned int VAO_r
Definition: ui_renderer.h:66
float max_diff_y
Definition: ui_renderer.h:80
Shader shader_c
Definition: ui_renderer.h:71
bool is_within_circle(const glm::vec2 &vert, const glm::vec2 &pos, float radius)
void set_rect_end_point(glm::vec2 mouse_pos, const LandmarkModel &model)
void prepare_circle(float current_zoom)
bool init()
Definition: ui_renderer.cpp:34
float circle_radius
Definition: ui_renderer.h:86
unsigned int VAO_c
Definition: ui_renderer.h:72
void move_selection(glm::vec2 mouse_pos, LandmarkModel &landmarks)
void prepare_rectangle()
Prepare rectangle data used for multiselect.
bool rect_pressed
Definition: ui_renderer.h:40