BlosSOM
Interactive dimensionality reduction on large datasets (EmbedSOM and FLOWER combined)
landmark_model.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 "landmark_model.h"
21
22#include <limits>
23#include <random>
24
26 : d(0)
27{
28 init_grid(0);
29}
30
31void
33{
34 if (!d) {
35 hidim_vertices.clear();
36 lodim_vertices.clear();
37 edges.clear();
38 return;
39 }
40
41 lodim_vertices.resize(n * n);
42 hidim_vertices.resize(n * n * d);
43 edges.clear();
44
45 std::default_random_engine gen;
46 std::uniform_real_distribution<float> dist(-1, 1);
47
48 for (size_t i = 0; i < n * n; ++i) {
49 auto x = i % n;
50 auto y = i / n;
51 lodim_vertices[i] = glm::vec2(x, y);
52
53 for (size_t di = 0; di < d; ++di)
54 hidim_vertices[i * d + di] = (di & 1) ? x : y;
55 }
56
57 touch();
58}
59
60void
62{
63 if (dim == d)
64 return;
65 d = dim;
66 init_grid(2);
67}
68
69void
70LandmarkModel::move(size_t ind, const glm::vec2 &mouse_pos)
71{
72 lodim_vertices[ind] = mouse_pos;
73 touch();
74}
75
76void
78{
79 // Add new line to hidim
80 size_t line_idx = d * ind;
81 for (size_t i = 0; i < d; ++i) {
82 hidim_vertices.emplace_back(hidim_vertices[line_idx + i]);
83 }
84
85 // Add new vertex to lodim
86 lodim_vertices.emplace_back(
87 glm::vec2(lodim_vertices[ind].x + 0.3, lodim_vertices[ind].y));
88
89 touch();
90}
91
92void
93LandmarkModel::add(const glm::vec2 &mouse_pos)
94{
95 size_t vert_ind = closest_landmark(mouse_pos);
96
97 // Add new vertex to lodim
98 lodim_vertices.emplace_back(mouse_pos);
99
100 // Add new line to hidim
101 size_t line_idx = d * vert_ind;
102 for (size_t i = 0; i < d; ++i) {
103 hidim_vertices.emplace_back(hidim_vertices[line_idx + i]);
104 }
105
106 touch();
107}
108
109void
111{
112 lodim_vertices.erase(lodim_vertices.begin() + ind);
113 size_t line_idx = d * ind;
114 hidim_vertices.erase(hidim_vertices.begin() + line_idx,
115 hidim_vertices.begin() + line_idx + d);
116
117 // Remove edges.
118 std::vector<size_t> edge_idxs;
119 size_t edge_ind = 0;
120 for (auto i = edges.begin(); i != edges.end();) {
121 if (i->first == ind || i->second == ind) {
122 i = edges.erase(i);
123 edge_lengths.erase(edge_lengths.begin() + edge_ind);
124 continue;
125 }
126 ++edge_ind;
127 ++i;
128 }
129
130 // Update indices of vertices that are after the removing vertex
131 // so the edges have proper vertex indices
132 for (auto i = edges.begin(); i != edges.end(); ++i) {
133 if (i->first >= ind) {
134 --i->first;
135 }
136 if (i->second >= ind) {
137 --i->second;
138 }
139 }
140
141 touch();
142}
143
144static float
145distance(const glm::vec2 &x, const glm::vec2 &y)
146{
147 auto a = powf(x.x - y.x, 2);
148 auto b = powf(x.y - y.y, 2);
149 return sqrtf(a + b);
150}
151
152size_t
153LandmarkModel::closest_landmark(const glm::vec2 &mouse_pos) const
154{
155 auto min_dist = std::numeric_limits<float>::max();
156 size_t vert_ind = 0;
157 for (size_t i = 0; i < lodim_vertices.size(); ++i) {
158 auto dist = distance(lodim_vertices[i], mouse_pos);
159 if (dist < min_dist) {
160 min_dist = dist;
161 vert_ind = i;
162 }
163 }
164
165 return vert_ind;
166}
static float distance(const glm::vec2 &x, const glm::vec2 &y)
void touch()
Make the cache dirty.
Definition: dirty.h:43
void duplicate(size_t ind)
Creates new landmark with the same two- and high-dimensional coordinates as the given landmark.
void move(size_t ind, const glm::vec2 &mouse_pos)
Sets two-dimensional position of the pressed landmark to mouse position.
void add(const glm::vec2 &mouse_pos)
Creates new landmark with the two- and high-dimensional coordinates as the closeset landmark.
std::vector< glm::vec2 > lodim_vertices
Array storing two-dimensional landmark coordinates.
std::vector< float > hidim_vertices
One-dimensional array storing d-dimensional landmark coordinates in row-major order.
std::vector< float > edge_lengths
Lengths of all edges.
size_t d
Dimension size.
size_t closest_landmark(const glm::vec2 &mouse_pos) const
Counts closest landmark to the given position.
LandmarkModel()
Creates empty landmarks with dimension 0.
void update_dim(size_t dim)
Updates current dimension and calls init_grid().
void remove(size_t ind)
Removes landmark and corresponding edges.
void init_grid(size_t side)
Creates squared landmarks layout, without edges.
std::vector< std::pair< size_t, size_t > > edges
Array of vertex ID pairs.