BlosSOM
Interactive dimensionality reduction on large datasets (EmbedSOM and FLOWER combined)
graph_renderer.cpp
Go to the documentation of this file.
1/* This file is part of BlosSOM.
2 *
3 * Copyright (C) 2021 Mirek Kratochvil
4 * Sona Molnarova
5 *
6 * BlosSOM is free software: you can redistribute it and/or modify it under
7 * the terms of the GNU General Public License as published by the Free
8 * Software Foundation, either version 3 of the License, or (at your option)
9 * any later version.
10 *
11 * BlosSOM is distributed in the hope that it will be useful, but WITHOUT ANY
12 * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
13 * FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
14 * details.
15 *
16 * You should have received a copy of the GNU General Public License along with
17 * BlosSOM. If not, see <https://www.gnu.org/licenses/>.
18 */
19
20#include "graph_renderer.h"
21
22#include "glm/gtc/matrix_transform.hpp"
23
24#include <cmath>
25#include <iostream>
26
27#include "shaders.h"
28
30 : vert_pressed(false)
31 , vert_ind(0)
32{
33}
34
35void
37{
38 glGenVertexArrays(1, &VAO_v);
39 glGenBuffers(1, &VBO_v_pos);
40 glGenBuffers(1, &VBO_v_col);
41
43
44 glGenVertexArrays(1, &VAO_v_outline);
45 glGenBuffers(1, &VBO_v_pos_outline);
46
48
49 glGenVertexArrays(1, &VAO_e);
50 glGenBuffers(1, &VBO_e);
51
53}
54
55void
57 const LandmarkModel &model,
58 const ColorData &colors)
59{
60 glEnable(GL_BLEND);
61
62 prepare_data(view.current_zoom, model, colors);
63
64 shader_e.use();
65 shader_e.set_mat4("model", glm::mat4(1.0f));
66 shader_e.set_mat4("view", view.get_view_matrix());
67 shader_e.set_mat4("proj", view.get_proj_matrix());
68
69 glBindVertexArray(VAO_e);
70 glDrawArrays(GL_LINES, 0, 2 * model.edges.size());
71
72 shader_v.use();
73 shader_v.set_mat4("model", glm::mat4(1.0f));
74 shader_v.set_mat4("view", view.get_view_matrix());
75 shader_v.set_mat4("proj", view.get_proj_matrix());
76
77 glBindVertexArray(VAO_v);
78
79 for (size_t i = 0; i < model.lodim_vertices.size(); ++i) {
80 glDrawArrays(GL_TRIANGLE_FAN, i * num_all_vtxs, num_all_vtxs);
81 }
82
84 shader_v_outline.set_mat4("model", glm::mat4(1.0f));
87
88 glBindVertexArray(VAO_v_outline);
89 for (size_t i = 0; i < model.lodim_vertices.size(); ++i) {
90 glDrawArrays(
92 }
93
94 glDisable(GL_BLEND);
95}
96
97bool
98GraphRenderer::is_vert_pressed(const View &view, glm::vec2 mouse)
99{
100 float radius = vertex_size;
101
102 for (size_t i = 0; i < vertices.size(); ++i) {
103 glm::vec2 vert = view.screen_coords(vertices[i]);
104
105 if ((mouse.x >= roundf(vert.x) - radius) &&
106 (mouse.x <= roundf(vert.x) + radius) &&
107 (mouse.y >= roundf(vert.y) - radius) &&
108 (mouse.y <= roundf(vert.y) + radius)) {
109 vert_ind = i;
110 return true;
111 }
112 }
113
114 return false;
115}
116void
118 const LandmarkModel &model,
119 const ColorData &colors)
120{
121 prepare_vertices(current_zoom, model, colors);
122 prepare_edges(model);
123}
124
125void
127 float middle_y,
128 float zoom,
129 std::vector<float> &all_vtxs,
130 std::vector<float> &vtxs_outlines,
131 std::vector<glm::vec3> &all_colors,
132 const glm::vec3 &color)
133{
134 int sides = 12;
135 float radius = 0.05f;
136 num_all_vtxs = sides + 2;
137 num_all_vtxs_outlines = sides * 2;
138
139 double two_pi = 2.0f * M_PI;
140
141 all_vtxs.emplace_back(middle_x);
142 all_vtxs.emplace_back(middle_y);
143 all_colors.emplace_back(color);
144
145 for (int i = 1; i < num_all_vtxs; i++) {
146 float x_coor =
147 middle_x + (radius * cos(i * two_pi / sides)) * zoom * 130;
148 float y_coor =
149 middle_y + (radius * sin(i * two_pi / sides)) * zoom * 130;
150 all_vtxs.emplace_back(x_coor);
151 all_vtxs.emplace_back(y_coor);
152
153 all_colors.emplace_back(color);
154
155 vtxs_outlines.emplace_back(x_coor);
156 vtxs_outlines.emplace_back(y_coor);
157 // Add each point twice --- end of line and start
158 // of next line.
159 if (i != 1 && i != num_all_vtxs - 1) {
160 vtxs_outlines.emplace_back(x_coor);
161 vtxs_outlines.emplace_back(y_coor);
162 }
163 }
164}
165
166void
168 const LandmarkModel &model,
169 const ColorData &colors)
170{
171 if (vertices.size() != model.lodim_vertices.size()) {
172 vertices.clear();
173 vertices.resize(model.lodim_vertices.size());
174 }
175
176 std::vector<float> all_vtxs;
177 std::vector<float> vtx_outlines;
178 std::vector<glm::vec3> all_colors;
179
180 for (size_t i = 0; i < vertices.size(); ++i) {
181 vertices[i] = model.lodim_vertices[i];
182 add_circle(vertices[i].x,
183 vertices[i].y,
184 current_zoom,
185 all_vtxs,
186 vtx_outlines,
187 all_colors,
188 (colors.coloring == colors.BRUSHING
189 ? *colors.landmarks[i].first
190 : colors.default_landmark_color));
191 }
192
193 glBindVertexArray(VAO_v);
194
195 glBindBuffer(GL_ARRAY_BUFFER, VBO_v_pos);
196 glBufferData(GL_ARRAY_BUFFER,
197 all_vtxs.size() * sizeof(float),
198 &all_vtxs[0],
199 GL_DYNAMIC_DRAW);
200 glVertexAttribPointer(
201 0, 2, GL_FLOAT, GL_FALSE, 2 * sizeof(float), (void *)0);
202 glEnableVertexAttribArray(0);
203
204 glBindBuffer(GL_ARRAY_BUFFER, VBO_v_col);
205 glBufferData(GL_ARRAY_BUFFER,
206 all_colors.size() * sizeof(glm::vec3),
207 &all_colors[0],
208 GL_DYNAMIC_DRAW);
209 glVertexAttribPointer(
210 1, 3, GL_FLOAT, GL_FALSE, sizeof(glm::vec3), (void *)0);
211 glEnableVertexAttribArray(1);
212
213 glBindVertexArray(VAO_v_outline);
214
215 glBindBuffer(GL_ARRAY_BUFFER, VBO_v_pos_outline);
216 glBufferData(GL_ARRAY_BUFFER,
217 vtx_outlines.size() * sizeof(float),
218 &vtx_outlines[0],
219 GL_DYNAMIC_DRAW);
220 glVertexAttribPointer(
221 0, 2, GL_FLOAT, GL_FALSE, 2 * sizeof(float), (void *)0);
222 glEnableVertexAttribArray(0);
223}
224
225void
227{
228 std::vector<glm::vec2> edge_lines(2 * model.edges.size());
229 for (size_t i = 0; i < model.edges.size(); ++i) {
230 edge_lines[2 * i + 0] = vertices[model.edges[i].first];
231 edge_lines[2 * i + 1] = vertices[model.edges[i].second];
232 }
233
234 glBindVertexArray(VAO_e);
235
236 glBindBuffer(GL_ARRAY_BUFFER, VBO_e);
237 glBufferData(GL_ARRAY_BUFFER,
238 edge_lines.size() * sizeof(glm::vec2),
239 &edge_lines[0],
240 GL_DYNAMIC_DRAW);
241 glVertexAttribPointer(
242 0, 2, GL_FLOAT, GL_FALSE, sizeof(glm::vec2), (void *)0);
243 glEnableVertexAttribArray(0);
244}
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 screen_coords(glm::vec2 point) const
Converts point to screen coordinates([0,0] in the middle of the screen).
Definition: view.h:157
const std::string graph_v_outline_vs
Definition: shaders.h:88
const std::string graph_e_fs
Definition: shaders.h:117
const std::string graph_v_outline_fs
Definition: shaders.h:98
const std::string graph_v_fs
Definition: shaders.h:80
const std::string graph_v_vs
Definition: shaders.h:67
const std::string graph_e_vs
Definition: shaders.h:107
Storage of the color data.
Definition: color_data.h:40
std::vector< std::pair< const glm::vec3 *, int > > landmarks
Colors of the landmarks and id of the cluster.
Definition: color_data.h:63
int coloring
Type of the coloring method.
Definition: color_data.h:65
const glm::vec3 default_landmark_color
Definition: color_data.h:52
std::vector< glm::vec2 > vertices
Cached screen coordinates of the vertices.
unsigned int VAO_v_outline
void prepare_vertices(float current_zoom, const LandmarkModel &model, const ColorData &colors)
Prepare graph vertices that are rendered as circles.
bool is_vert_pressed(const View &view, glm::vec2 mouse)
Checks if some vertex was pressed.
unsigned int VAO_v
static constexpr float vertex_size
Radius of the vertex used for comparing, if the landmark was pressed.
unsigned int VBO_v_col
Shader shader_v_outline
unsigned int VAO_e
void draw(const View &v, const LandmarkModel &m, const ColorData &colors)
Draw event of the 2D landmark graph.
int num_all_vtxs
Number of all vertices for rendering circles(graph vertices).
unsigned int VBO_v_pos_outline
void add_circle(float middle_x, float middle_y, float zoom, std::vector< float > &all_vtxs, std::vector< float > &vtxs_outlines, std::vector< glm::vec3 > &all_colors, const glm::vec3 &color)
Add vertices for TRIANGLE_FAN that creates circle at given position.
unsigned int VBO_v_pos
int num_all_vtxs_outlines
size_t vert_ind
Index of the pressed vertex.
void prepare_edges(const LandmarkModel &model)
Prepare graph edges that are rendered as lines.
void prepare_data(float current_zoom, const LandmarkModel &model, const ColorData &colors)
Prepare data to render vertices and edges.
unsigned int VBO_e
Model of the high- and low-dimensional landmarks.
std::vector< glm::vec2 > lodim_vertices
Array storing two-dimensional landmark coordinates.
std::vector< std::pair< size_t, size_t > > edges
Array of vertex ID pairs.