Compare commits

..

No commits in common. "d20124af807e5aee49e54b8a163b582b59a4c38c" and "233adba0aeb5a3155d28aff78316065761b67cfd" have entirely different histories.

3 changed files with 63 additions and 119 deletions

View File

@ -1,5 +1,5 @@
cmake_minimum_required(VERSION 3.25) cmake_minimum_required(VERSION 3.25)
project(graphs VERSION 0.0.23) project(graphs VERSION 0.0.20)
option(ENABLE_ADDRSAN "Enable the address sanitizer" OFF) option(ENABLE_ADDRSAN "Enable the address sanitizer" OFF)
option(ENABLE_UBSAN "Enable the ub sanitizer" OFF) option(ENABLE_UBSAN "Enable the ub sanitizer" OFF)

View File

@ -1,7 +1,7 @@
#!/bin/bash #!/bin/bash
mkdir cmake-build-emrelwithdebinfo mkdir cmake-build-emrelwithdebinfo
cd cmake-build-emrelwithdebinfo cd cmake-build-emrelwithdebinfo
#emcmake cmake -DCMAKE_BUILD_TYPE=RelWithDebInfo ../ emcmake cmake -DCMAKE_BUILD_TYPE=RelWithDebInfo ../
emcmake cmake -DCMAKE_BUILD_TYPE=Release ../ #emcmake cmake -DCMAKE_BUILD_TYPE=Release ../
emmake make -j 16 emmake make -j 16
cp graphs.js graphs.data graphs.wasm /var/www/html/playground/ cp graphs.js graphs.data graphs.wasm /var/www/html/playground/

View File

@ -21,7 +21,6 @@
#include "blt/gfx/renderer/camera.h" #include "blt/gfx/renderer/camera.h"
#include <blt/gfx/framebuffer.h> #include <blt/gfx/framebuffer.h>
#include <imgui.h> #include <imgui.h>
#include <memory>
#include <random> #include <random>
#include <blt/std/ranges.h> #include <blt/std/ranges.h>
#include <blt/std/assert.h> #include <blt/std/assert.h>
@ -32,6 +31,7 @@ blt::gfx::matrix_state_manager global_matrices;
blt::gfx::resource_manager resources; blt::gfx::resource_manager resources;
blt::gfx::batch_renderer_2d renderer_2d(resources); blt::gfx::batch_renderer_2d renderer_2d(resources);
blt::gfx::first_person_camera_2d camera; blt::gfx::first_person_camera_2d camera;
blt::gfx::fbo_t render_texture;
blt::u64 lastTime; blt::u64 lastTime;
double ft = 0; double ft = 0;
double fps = 0; double fps = 0;
@ -106,16 +106,11 @@ struct edge_hash
struct equation_variables struct equation_variables
{ {
float repulsive_constant = 24.0; float repulsive_constant = 12.0;
float spring_constant = 12.0; float spring_constant = 24.0;
float ideal_spring_length = 175.0; float ideal_spring_length = 175.0;
float initial_temperature = 69.5; float initial_temperature = 69.5;
float cooling_rate = 0.999; float cooling_rate = 0.999;
float min_cooling = 0;
equation_variables() = default;
//equation_variables(const equation_variables&) = delete;
//equation_variables& operator=(const equation_variables&) = delete;
}; };
class force_equation class force_equation
@ -123,14 +118,14 @@ class force_equation
public: public:
using node_pair = const std::pair<blt::size_t, node>&; using node_pair = const std::pair<blt::size_t, node>&;
protected: protected:
const equation_variables& variables; const equation_variables variables;
struct equation_data struct equation_data
{ {
blt::vec2 unit, unit_inv; blt::vec2 unit;
float mag, mag_sq; float mag, mag_sq;
equation_data(blt::vec2 unit, blt::vec2 unit_inv, float mag, float mag_sq): unit(unit), unit_inv(unit_inv), mag(mag), mag_sq(mag_sq) equation_data(blt::vec2 unit, float mag, float mag_sq): unit(unit), mag(mag), mag_sq(mag_sq)
{} {}
}; };
@ -142,13 +137,10 @@ class force_equation
inline static equation_data calc_data(node_pair v1, node_pair v2) inline static equation_data calc_data(node_pair v1, node_pair v2)
{ {
auto dir = dir_v(v1, v2); auto dir = dir_v(v1, v2);
auto dir2 = dir_v(v2, v1);
auto mag = dir.magnitude(); auto mag = dir.magnitude();
auto mag2 = dir2.magnitude();
auto unit = mag == 0 ? blt::vec2() : dir / mag; auto unit = mag == 0 ? blt::vec2() : dir / mag;
auto unit_inv = mag2 == 0 ? blt::vec2() : dir2 / mag2;
auto mag_sq = mag * mag; auto mag_sq = mag * mag;
return {unit, unit_inv, mag, mag_sq}; return {unit, mag, mag_sq};
} }
public: public:
@ -164,7 +156,7 @@ class force_equation
[[nodiscard]] virtual float cooling_factor(int t) const [[nodiscard]] virtual float cooling_factor(int t) const
{ {
return std::max(static_cast<float>(variables.initial_temperature * std::pow(variables.cooling_rate, t)), variables.min_cooling); return static_cast<float>(variables.initial_temperature * std::pow(variables.cooling_rate, t));
} }
virtual ~force_equation() = default; virtual ~force_equation() = default;
@ -182,16 +174,15 @@ class Eades_equation : public force_equation
auto ideal = std::log(data.mag / variables.ideal_spring_length); auto ideal = std::log(data.mag / variables.ideal_spring_length);
return (variables.spring_constant * ideal * data.unit) - rep(v1, v2); return variables.spring_constant * ideal * data.unit;
} }
[[nodiscard]] blt::vec2 rep(node_pair v1, node_pair v2) const final [[nodiscard]] blt::vec2 rep(node_pair v1, node_pair v2) const final
{ {
auto data = calc_data(v1, v2); auto data = calc_data(v1, v2);
// scaling factor included because of the scales this algorithm is working on (large viewport) auto scale = variables.repulsive_constant / data.mag_sq;
auto scale = (variables.repulsive_constant * 10000) / data.mag_sq; return scale * -data.unit;
return scale * data.unit_inv;
} }
[[nodiscard]] std::string name() const final [[nodiscard]] std::string name() const final
@ -212,7 +203,7 @@ class Fruchterman_Reingold_equation : public force_equation
float scale = data.mag_sq / variables.ideal_spring_length; float scale = data.mag_sq / variables.ideal_spring_length;
return (scale * data.unit); return scale * data.unit;
} }
[[nodiscard]] blt::vec2 rep(node_pair v1, node_pair v2) const final [[nodiscard]] blt::vec2 rep(node_pair v1, node_pair v2) const final
@ -221,7 +212,7 @@ class Fruchterman_Reingold_equation : public force_equation
float scale = (variables.ideal_spring_length * variables.ideal_spring_length) / data.mag; float scale = (variables.ideal_spring_length * variables.ideal_spring_length) / data.mag;
return scale * data.unit_inv; return scale * -data.unit;
} }
[[nodiscard]] float cooling_factor(int t) const override [[nodiscard]] float cooling_factor(int t) const override
@ -251,22 +242,19 @@ struct bounding_box
class graph class graph
{ {
private: private:
equation_variables variables;
std::vector<node> nodes; std::vector<node> nodes;
blt::hashset_t<edge, edge_hash> edges; blt::hashset_t<edge, edge_hash> edges;
blt::hashmap_t<blt::u64, blt::hashset_t<blt::u64>> connected_nodes; blt::hashmap_t<blt::u64, blt::hashset_t<blt::u64>> connected_nodes;
bool sim = false; bool sim = false;
bool run_infinitely = true;
float sim_speed = 1; float sim_speed = 1;
float threshold = 0.01; float threshold = 0.01;
float max_force_last = 1; float max_force_last = 1;
int current_iterations = 0; int current_iterations = 0;
int max_iterations = 5000; int max_iterations = 5000;
equation_variables variables;
std::unique_ptr<force_equation> equation; std::unique_ptr<force_equation> equation;
static constexpr float POINT_SIZE = 35; 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) void create_random_graph(bounding_box bb, blt::size_t min_nodes, blt::size_t max_nodes, blt::f64 connectivity)
{ {
// don't allow points too close to the edges of the window. // don't allow points too close to the edges of the window.
@ -346,7 +334,7 @@ class graph
public: public:
graph() = default; graph() = default;
void make_new(const bounding_box& bb, blt::size_t min_nodes, blt::size_t max_nodes, blt::f64 connectivity) graph(const bounding_box& bb, blt::size_t min_nodes, blt::size_t max_nodes, blt::f64 connectivity)
{ {
create_random_graph(bb, min_nodes, max_nodes, connectivity); create_random_graph(bb, min_nodes, max_nodes, connectivity);
use_Eades(); use_Eades();
@ -377,7 +365,7 @@ class graph
void render(double frame_time) void render(double frame_time)
{ {
if (sim && (current_iterations < max_iterations || run_infinitely) && max_force_last > threshold) if (sim && current_iterations < max_iterations && max_force_last > threshold)
{ {
for (int _ = 0; _ < sub_ticks; _++) for (int _ = 0; _ < sub_ticks; _++)
{ {
@ -390,8 +378,7 @@ class graph
{ {
if (v1.first == v2.first) if (v1.first == v2.first)
continue; continue;
if (connected(v1.first, v2.first)) attractive += equation->attr(v1, v2);
attractive += equation->attr(v1, v2);
repulsive += equation->rep(v1, v2); repulsive += equation->rep(v1, v2);
} }
v1.second.getVelocityRef() = attractive + repulsive; v1.second.getVelocityRef() = attractive + repulsive;
@ -424,54 +411,6 @@ class graph
} }
} }
void reset_mouse_drag()
{
current_node = -1;
}
void process_mouse_drag(blt::i32, blt::i32 height)
{
auto mx = static_cast<float>(blt::gfx::getMouseX());
auto my = static_cast<float>(height - blt::gfx::getMouseY());
auto mv = blt::vec2(mx, my);
const auto& ovm = global_matrices.computedOVM();
auto adj_mv = ovm * blt::vec4(mv.x(), mv.y(), 0, 1);
auto adj_size = ovm * blt::vec4(POINT_SIZE, POINT_SIZE, POINT_SIZE, POINT_SIZE);
float new_size = std::max(std::abs(adj_size.x()), std::abs(adj_size.y()));
//BLT_TRACE_STREAM << "adj_mv: ";
//BLT_TRACE_STREAM << adj_mv << "\n";
//BLT_TRACE_STREAM << "adj_size: ";
//BLT_TRACE_STREAM << adj_size << "\n";
if (current_node < 0)
{
for (const auto& n : blt::enumerate(nodes))
{
auto pos = n.second.getPosition();
auto dist = pos - mv;
auto mag = dist.magnitude();
if (mag < POINT_SIZE)
{
current_node = static_cast<blt::i32>(n.first);
break;
}
}
} else
{
auto pos = nodes[current_node].getPosition();
auto adj_pos = ovm * blt::vec4(pos.x(), pos.y(), 0, 1);
//BLT_TRACE_STREAM << "adj_pos: ";
//BLT_TRACE_STREAM << adj_pos << "\n";
nodes[current_node].getPositionRef() = mv;
}
}
void use_Eades() void use_Eades()
{ {
equation = std::make_unique<Eades_equation>(variables); equation = std::make_unique<Eades_equation>(variables);
@ -497,21 +436,6 @@ class graph
return equation->name(); return equation->name();
} }
auto getCoolingFactor()
{
return equation->cooling_factor(current_iterations);
}
void reset_iterations()
{
current_iterations = 0;
}
bool& getIterControl()
{
return run_infinitely;
}
float& getSimSpeed() float& getSimSpeed()
{ {
return sim_speed; return sim_speed;
@ -522,9 +446,29 @@ class graph
return threshold; return threshold;
} }
auto& getVariables() float& getSpringConstant()
{ {
return variables; return variables.spring_constant;
}
float& getInitialTemperature()
{
return variables.initial_temperature;
}
float& getCoolingRate()
{
return variables.cooling_rate;
}
float& getIdealSpringLength()
{
return variables.ideal_spring_length;
}
float& getRepulsionConstant()
{
return variables.repulsive_constant;
} }
int& getMaxIterations() int& getMaxIterations()
@ -560,8 +504,13 @@ void init(const blt::gfx::window_data& data)
renderer_2d.create(); renderer_2d.create();
bounding_box bb(0, 0, data.width, data.height); bounding_box bb(0, 0, data.width, data.height);
main_graph.make_new(bb, 5, 25, 0.2); main_graph = graph(bb, 5, 25, 0.2);
lastTime = blt::system::nanoTime(); lastTime = blt::system::nanoTime();
//render_texture = fbo_t::make_multisample_render_texture(1440, 720, 4);
//render_texture = fbo_t::make_multisample_render_target(1440, 720, 8);
//render_texture = fbo_t::make_render_target(1440, 720);
//render_texture = fbo_t::make_render_texture(1440, 720);
} }
float x = 50, y = 50; float x = 50, y = 50;
@ -589,7 +538,7 @@ void update(const blt::gfx::window_data& data)
static int min_nodes = 5; static int min_nodes = 5;
static int max_nodes = 25; static int max_nodes = 25;
static bounding_box bb{0, 0, data.width, data.height}; static bounding_box bb {0, 0, data.width, data.height};
static float connectivity = 0.12; static float connectivity = 0.12;
@ -599,8 +548,7 @@ void update(const blt::gfx::window_data& data)
im::SetNextItemOpen(true, ImGuiCond_Once); im::SetNextItemOpen(true, ImGuiCond_Once);
if (im::CollapsingHeader("Help")) if (im::CollapsingHeader("Help"))
{ {
im::Text("You can use W/A/S/D to move the camera around");
im::Text("Q/E can be used to zoom in/out the camera");
} }
if (im::CollapsingHeader("Graph Generation Settings")) if (im::CollapsingHeader("Graph Generation Settings"))
{ {
@ -613,7 +561,9 @@ void update(const blt::gfx::window_data& data)
result |= im::InputInt("Min Y", &bb.min_y, 5, 100); result |= im::InputInt("Min Y", &bb.min_y, 5, 100);
result |= im::InputInt("Max Y", &bb.max_y, 5, 100); result |= im::InputInt("Max Y", &bb.max_y, 5, 100);
if (result) if (result)
{
bb.is_screen = false; bb.is_screen = false;
}
} }
if (bb.is_screen) if (bb.is_screen)
{ {
@ -635,29 +585,25 @@ void update(const blt::gfx::window_data& data)
if (im::CollapsingHeader("Simulation Settings")) if (im::CollapsingHeader("Simulation Settings"))
{ {
im::InputInt("Max Iterations", &main_graph.getMaxIterations()); im::InputInt("Max Iterations", &main_graph.getMaxIterations());
im::Checkbox("Run Infinitely", &main_graph.getIterControl());
im::InputInt("Sub-ticks Per Frame", &sub_ticks); im::InputInt("Sub-ticks Per Frame", &sub_ticks);
im::InputFloat("Threshold", &main_graph.getThreshold(), 0.01, 1); im::InputFloat("Threshold", &main_graph.getThreshold(), 0.01, 1);
im::InputFloat("Repulsive Constant", &main_graph.getVariables().repulsive_constant, 0.25, 10); im::InputFloat("Repulsive Constant", &main_graph.getRepulsionConstant(), 0.25, 10);
im::InputFloat("Spring Constant", &main_graph.getVariables().spring_constant, 0.25, 10); im::InputFloat("Spring Constant", &main_graph.getSpringConstant(), 0.25, 10);
im::InputFloat("Ideal Spring Length", &main_graph.getVariables().ideal_spring_length, 2.5, 10); im::InputFloat("Ideal Spring Length", &main_graph.getIdealSpringLength(), 2.5, 10);
im::SliderFloat("Initial Temperature", &main_graph.getVariables().initial_temperature, 1, 100); im::SliderFloat("Initial Temperature", &main_graph.getInitialTemperature(), 1, 100);
im::SliderFloat("Cooling Rate", &main_graph.getVariables().cooling_rate, 0, 0.999999, "%.6f"); im::SliderFloat("Cooling Rate", &main_graph.getCoolingRate(), 0, 0.999999, "%.6f");
im::InputFloat("Min Cooling", &main_graph.getVariables().min_cooling, 0.5, 1);
im::Text("Current Cooling Factor: %f", main_graph.getCoolingFactor());
im::SliderFloat("Simulation Speed", &main_graph.getSimSpeed(), 0, 4); im::SliderFloat("Simulation Speed", &main_graph.getSimSpeed(), 0, 4);
if (im::Button("Start")) if (im::Button("Start"))
main_graph.start_sim(); main_graph.start_sim();
im::SameLine(); im::SameLine();
if (im::Button("Stop")) if (im::Button("Stop"))
main_graph.stop_sim(); main_graph.stop_sim();
if (im::Button("Reset Iterations"))
main_graph.reset_iterations();
} }
im::SetNextItemOpen(true, ImGuiCond_Once); im::SetNextItemOpen(true, ImGuiCond_Once);
if (im::CollapsingHeader("System Controls")) if (im::CollapsingHeader("System Controls"))
{ {
im::Text("Select a system:"); //im::Text("Current System: %s", main_graph.getSimulatorName().c_str());
//im::
auto current_sim = main_graph.getSimulatorName(); auto current_sim = main_graph.getSimulatorName();
const char* items[] = {"Eades", "Fruchterman & Reingold"}; const char* items[] = {"Eades", "Fruchterman & Reingold"};
static int item_current = 0; static int item_current = 0;
@ -684,12 +630,10 @@ void update(const blt::gfx::window_data& data)
im::End(); im::End();
} }
auto& io = ImGui::GetIO(); //renderer_2d.drawLine(blt::vec4{1, 0, 1, 1}, 0.0f, blt::vec2{x, y}, blt::vec2{500, 500}, 5.0f);
//renderer_2d.drawLine(blt::vec4{1, 0, 0, 1}, 0.0f, blt::vec2{0, 150}, blt::vec2{240, 0}, 12.0f);
if (!io.WantCaptureMouse && blt::gfx::isMousePressed(0)) //renderer_2d.drawPoint(blt::vec4{0, 1, 0, 1}, 1.0f, blt::vec2{500, 500}, 50.0f);
main_graph.process_mouse_drag(data.width, data.height); //renderer_2d.drawPoint("parkercat", 1.0f, blt::vec2{800, 500}, 256.0f);
else
main_graph.reset_mouse_drag();
main_graph.render(ft); main_graph.render(ft);