201 lines
5.5 KiB
C
201 lines
5.5 KiB
C
|
#pragma once
|
||
|
/*
|
||
|
* Copyright (C) 2024 Brett Terpstra
|
||
|
*
|
||
|
* This program is free software: you can redistribute it and/or modify
|
||
|
* it under the terms of the GNU General Public License as published by
|
||
|
* the Free Software Foundation, either version 3 of the License, or
|
||
|
* (at your option) any later version.
|
||
|
*
|
||
|
* This program is distributed in the hope that it will be useful,
|
||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
* GNU General Public License for more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU General Public License
|
||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||
|
*/
|
||
|
|
||
|
#ifndef GRAPHS_GRAPH_H
|
||
|
#define GRAPHS_GRAPH_H
|
||
|
|
||
|
#include <graph_base.h>
|
||
|
#include <force_algorithms.h>
|
||
|
#include <blt/gfx/window.h>
|
||
|
|
||
|
namespace im = ImGui;
|
||
|
|
||
|
struct bounding_box
|
||
|
{
|
||
|
int min_x = 0;
|
||
|
int min_y = 0;
|
||
|
int max_x = 0;
|
||
|
int max_y = 0;
|
||
|
|
||
|
bounding_box(const int min_x, const int min_y, const int max_x, const int max_y): min_x(min_x), min_y(min_y), max_x(max_x), max_y(max_y)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
bool is_screen = true;
|
||
|
};
|
||
|
|
||
|
class graph_t
|
||
|
{
|
||
|
private:
|
||
|
std::vector<node> nodes;
|
||
|
blt::hashset_t<edge, edge_hash> edges;
|
||
|
blt::hashmap_t<blt::u64, blt::hashset_t<blt::u64>> connected_nodes;
|
||
|
bool sim = false;
|
||
|
bool run_infinitely = true;
|
||
|
float sim_speed = 1;
|
||
|
float threshold = 0;
|
||
|
float max_force_last = 1;
|
||
|
int current_iterations = 0;
|
||
|
int max_iterations = 5000;
|
||
|
std::unique_ptr<force_equation> equation;
|
||
|
static constexpr float POINT_SIZE = 35;
|
||
|
|
||
|
blt::i32 current_node = -1;
|
||
|
|
||
|
void create_random_graph(bounding_box bb, blt::size_t min_nodes, blt::size_t max_nodes, blt::f64 connectivity,
|
||
|
blt::f64 scaling_connectivity, blt::f64 distance_factor);
|
||
|
|
||
|
public:
|
||
|
graph_t() = default;
|
||
|
|
||
|
void make_new(const bounding_box& bb, const blt::size_t min_nodes, const blt::size_t max_nodes, const blt::f64 connectivity)
|
||
|
{
|
||
|
create_random_graph(bb, min_nodes, max_nodes, connectivity, 0, 25);
|
||
|
use_Eades();
|
||
|
}
|
||
|
|
||
|
void reset(const bounding_box& bb, const blt::size_t min_nodes, const blt::size_t max_nodes, const blt::f64 connectivity,
|
||
|
const blt::f64 scaling_connectivity, const blt::f64 distance_factor)
|
||
|
{
|
||
|
sim = false;
|
||
|
current_iterations = 0;
|
||
|
max_force_last = 1.0;
|
||
|
nodes.clear();
|
||
|
edges.clear();
|
||
|
connected_nodes.clear();
|
||
|
create_random_graph(bb, min_nodes, max_nodes, connectivity, scaling_connectivity, distance_factor);
|
||
|
}
|
||
|
|
||
|
void connect(const blt::u64 n1, const blt::u64 n2)
|
||
|
{
|
||
|
edges.insert(edge{n1, n2});
|
||
|
connected_nodes[n1].insert(n2);
|
||
|
connected_nodes[n2].insert(n1);
|
||
|
}
|
||
|
|
||
|
[[nodiscard]] bool connected(blt::u64 e1, blt::u64 e2) const
|
||
|
{
|
||
|
return edges.contains({e1, e2});
|
||
|
}
|
||
|
|
||
|
void render(double frame_time);
|
||
|
|
||
|
void reset_mouse_drag()
|
||
|
{
|
||
|
current_node = -1;
|
||
|
}
|
||
|
|
||
|
void process_mouse_drag(blt::i32 width, blt::i32 height);
|
||
|
|
||
|
void use_Eades()
|
||
|
{
|
||
|
equation = std::make_unique<Eades_equation>();
|
||
|
}
|
||
|
|
||
|
void use_Fruchterman_Reingold()
|
||
|
{
|
||
|
equation = std::make_unique<Fruchterman_Reingold_equation>();
|
||
|
}
|
||
|
|
||
|
void start_sim()
|
||
|
{
|
||
|
sim = true;
|
||
|
}
|
||
|
|
||
|
void stop_sim()
|
||
|
{
|
||
|
sim = false;
|
||
|
}
|
||
|
|
||
|
[[nodiscard]] std::string getSimulatorName() const
|
||
|
{
|
||
|
return equation->name();
|
||
|
}
|
||
|
|
||
|
[[nodiscard]] auto* getSimulator() const
|
||
|
{
|
||
|
return equation.get();
|
||
|
}
|
||
|
|
||
|
[[nodiscard]] auto getCoolingFactor() const
|
||
|
{
|
||
|
return equation->cooling_factor(current_iterations);
|
||
|
}
|
||
|
|
||
|
void reset_iterations()
|
||
|
{
|
||
|
current_iterations = 0;
|
||
|
}
|
||
|
|
||
|
[[nodiscard]] bool& getIterControl()
|
||
|
{
|
||
|
return run_infinitely;
|
||
|
}
|
||
|
|
||
|
[[nodiscard]] float& getSimSpeed()
|
||
|
{
|
||
|
return sim_speed;
|
||
|
}
|
||
|
|
||
|
[[nodiscard]] float& getThreshold()
|
||
|
{
|
||
|
return threshold;
|
||
|
}
|
||
|
|
||
|
[[nodiscard]] int& getMaxIterations()
|
||
|
{
|
||
|
return max_iterations;
|
||
|
}
|
||
|
|
||
|
[[nodiscard]] int numberOfNodes() const
|
||
|
{
|
||
|
return static_cast<int>(nodes.size());
|
||
|
}
|
||
|
};
|
||
|
|
||
|
class engine_t
|
||
|
{
|
||
|
private:
|
||
|
graph_t graph;
|
||
|
|
||
|
void draw_gui(const blt::gfx::window_data& data, double ft);
|
||
|
|
||
|
public:
|
||
|
void init(const blt::gfx::window_data& data)
|
||
|
{
|
||
|
graph.make_new({0, 0, data.width, data.height}, 5, 25, 0.2);
|
||
|
}
|
||
|
|
||
|
void render(const blt::gfx::window_data& data, const double ft)
|
||
|
{
|
||
|
draw_gui(data, ft);
|
||
|
|
||
|
auto& io = ImGui::GetIO();
|
||
|
|
||
|
if (!io.WantCaptureMouse && blt::gfx::isMousePressed(0))
|
||
|
graph.process_mouse_drag(data.width, data.height);
|
||
|
else
|
||
|
graph.reset_mouse_drag();
|
||
|
|
||
|
|
||
|
graph.render(ft);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
#endif //GRAPHS_GRAPH_H
|