BlosSOM
Interactive dimensionality reduction on large datasets (EmbedSOM and FLOWER combined)
ui_save.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 "ui_save.h"
21
22#include "imgui_stdlib.h"
23
24#include <glm/glm.hpp>
25
26#include <algorithm>
27#include <exception>
28#include <fstream>
29
31 : show_window(false)
32 , saver(ImGuiFileBrowserFlags_SelectDirectory |
33 ImGuiFileBrowserFlags_CreateNewDir)
34 , all(false)
35 , data_flags{ false, false, false, false, false }
36 , file_names{ "points_hd.tsv",
37 "landmarks_hd.tsv",
38 "points_2d.tsv",
39 "landmarks_2d.tsv",
40 "clusters.tsv" }
41{
42 saver.SetTitle("Select directory");
43}
44
45void
46UiSaver::render(State &state, ImGuiWindowFlags window_flags)
47{
48 if (!show_window)
49 return;
50
51 saver.Display();
52
53 if (saver.HasSelected()) {
54 try {
55 save_data(state, saver.GetSelected().string());
56 } catch (std::exception &e) {
57 saving_error = e.what();
58 }
59
60 saver.ClearSelected();
61 }
62
63 if (!saving_error.empty()) {
64 ImGui::Begin("Saving error", nullptr, window_flags);
65 ImGui::Text(saving_error.c_str());
66 if (ImGui::Button("OK"))
67 saving_error = "";
68 ImGui::End();
69 }
70
71 if (ImGui::Begin("Save##window", &show_window, window_flags)) {
72 auto save_line = [&](const char *text, int type) {
73 ImGui::Text(text);
74 std::string checkbox_name =
75 "##save_checkbox" + std::to_string(type);
76 ImGui::Checkbox(checkbox_name.data(), &data_flags[type]);
77 ImGui::SameLine();
78 std::string inputtext_name =
79 "file name##save" + std::to_string(type);
80 ImGui::InputText(inputtext_name.data(), &file_names[type]);
81 };
82
83 save_line("Save hidim points:", UiSaver::Types::POINTS_HD);
84
85 save_line("Save hidim landmarks:", UiSaver::Types::LAND_HD);
86
87 save_line("Save 2D points:", UiSaver::Types::POINTS_2D);
88
89 save_line("Save 2D landmarks:", UiSaver::Types::LAND_2D);
90
91 save_line("Save clusters:", UiSaver::Types::CLUSTERS);
92 ImGui::NewLine();
93
94 if (ImGui::Checkbox("Save all", &all))
95 if (all)
96 std::fill(data_flags.begin(), data_flags.end(), true);
97 else
98 std::fill(data_flags.begin(), data_flags.end(), false);
99
100 ImGui::NewLine();
101
102 if (ImGui::Button("Save##button"))
103 saver.Open();
104
105 ImGui::End();
106 }
107}
108
109void
110UiSaver::save_data(const State &state, const std::string &dir_name) const
111{
112 if (data_flags[UiSaver::Types::POINTS_HD])
113 write(UiSaver::Types::POINTS_HD, state, dir_name);
114
115 if (data_flags[UiSaver::Types::LAND_HD])
116 write(UiSaver::Types::LAND_HD, state, dir_name);
117
118 if (data_flags[UiSaver::Types::POINTS_2D])
119 write(UiSaver::Types::POINTS_2D, state, dir_name);
120
121 if (data_flags[UiSaver::Types::LAND_2D])
122 write(UiSaver::Types::LAND_2D, state, dir_name);
123
124 if (data_flags[UiSaver::Types::CLUSTERS])
125 write(UiSaver::Types::CLUSTERS, state, dir_name);
126}
127
128/**
129 * @brief Writes multi-dimensional data into the file by a given handler.
130 *
131 * @param dim Dimension of the data.
132 * @param data Multi-dimensional data to be written.
133 * @param handle Handle to the opened file for writing.
134 */
135static void
137 const std::vector<float> &data,
138 std::ofstream &handle)
139{
140 for (size_t i = 0; i < data.size(); i += dim) {
141 for (size_t j = 0; j < dim - 1; ++j) {
142 handle << data[i + j] << '\t';
143 }
144 handle << data[i + dim - 1] << '\n';
145 }
146};
147
148/**
149 * @brief Writes two-dimensional data into the file by a given handler.
150 *
151 * @param data Two-dimensional data to be written.
152 * @param handle Handle to the opened file for writing.
153 */
154static void
155write_data_2d(const std::vector<glm::vec2> &data, std::ofstream &handle)
156{
157 for (size_t i = 0; i < data.size(); ++i) {
158 handle << data[i].x << '\t' << data[i].y << '\n';
159 }
160};
161
162static void
163write_clusters(std::vector<std::pair<const glm::vec3 *, int>> landmarks,
164 const std::map<int, std::pair<glm::vec3, std::string>> &clusters,
165 std::ofstream &handle)
166{
167 for (size_t i = 0; i < landmarks.size(); ++i) {
168 auto id = landmarks[i].second;
169 auto color = clusters.at(id).first;
170 auto name = clusters.at(id).second;
171 handle << i << '\t' << color.r << '\t' << color.g << '\t' << color.b
172 << '\t' << name << '\n';
173 }
174}
175
176void
178 const State &state,
179 const std::string &dir_name) const
180{
181 std::string path = dir_name + "/" + file_names[type];
182 std::ofstream handle(path, std::ios::out);
183 if (!handle)
184 throw std::domain_error("Can not open file");
185
186 switch (type) {
187 case UiSaver::Types::POINTS_HD:
188 write_data_float(state.scaled.dim(), state.scaled.data, handle);
189 break;
190 case UiSaver::Types::LAND_HD:
192 state.landmarks.d, state.landmarks.hidim_vertices, handle);
193 break;
194 case UiSaver::Types::POINTS_2D:
195 write_data_2d(state.scatter.points, handle);
196 break;
197 case UiSaver::Types::LAND_2D:
199 break;
200 case UiSaver::Types::CLUSTERS:
202 state.colors.landmarks, state.colors.clustering.clusters, handle);
203 break;
204 }
205
206 handle.close();
207}
std::map< int, std::pair< glm::vec3, std::string > > clusters
Cluster colors and names for brushing, with id of cluster as a key.
Definition: cluster_data.h:49
std::vector< std::pair< const glm::vec3 *, int > > landmarks
Colors of the landmarks and id of the cluster.
Definition: color_data.h:63
ClusterData clustering
Definition: color_data.h:71
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.
size_t d
Dimension size.
std::vector< float > data
Scaled data in the same format as DataModel::data.
Definition: scaled_data.h:55
size_t dim() const
Returns dimension of the scaled data.
Definition: scaled_data.h:66
std::vector< glm::vec2 > points
Coordinates of the two-dimensional data points.
Definition: scatter_model.h:44
Storage of data of used algorithms and input events.
Definition: state.h:50
ScaledData scaled
Definition: state.h:54
ScatterModel scatter
Definition: state.h:64
LandmarkModel landmarks
Definition: state.h:55
ColorData colors
Definition: state.h:63
Types
Types of data to be exported.
Definition: ui_save.h:42
std::array< std::string, UiSaver::Types::COUNT > file_names
Names of the exported files for each export data type.
Definition: ui_save.h:68
void render(State &state, ImGuiWindowFlags window_flags)
Renders save file window, opens save file dialog window and calls save_data() if a directory was sele...
Definition: ui_save.cpp:46
UiSaver()
Initializes saver settings and initializes variables with default values.
Definition: ui_save.cpp:30
void save_data(const State &state, const std::string &dir_name) const
Calls write() for selected export data types.
Definition: ui_save.cpp:110
bool show_window
If the save file window should be rendered.
Definition: ui_save.h:55
std::array< bool, UiSaver::Types::COUNT > data_flags
Array of flags for each export data type indicating which data should be exported and which not.
Definition: ui_save.h:66
void write(UiSaver::Types type, const State &state, const std::string &dir_name) const
Writes given data into the file in the tsv format.
Definition: ui_save.cpp:177
std::string saving_error
Error message of the saving file that will be shown in the error window.
Definition: ui_save.h:60
ImGui::FileBrowser saver
ImGui file system dialog window handler.
Definition: ui_save.h:57
bool all
If all types of data should be exported.
Definition: ui_save.h:63
static void write_data_2d(const std::vector< glm::vec2 > &data, std::ofstream &handle)
Writes two-dimensional data into the file by a given handler.
Definition: ui_save.cpp:155
static void write_clusters(std::vector< std::pair< const glm::vec3 *, int > > landmarks, const std::map< int, std::pair< glm::vec3, std::string > > &clusters, std::ofstream &handle)
Definition: ui_save.cpp:163
static void write_data_float(size_t dim, const std::vector< float > &data, std::ofstream &handle)
Writes multi-dimensional data into the file by a given handler.
Definition: ui_save.cpp:136