mirror of https://github.com/aziis98/asd-2024.git
some experiments
parent
49caa60d21
commit
425fcd8544
File diff suppressed because it is too large
Load Diff
File diff suppressed because one or more lines are too long
@ -0,0 +1,15 @@
|
|||||||
|
[package]
|
||||||
|
name = "configurable"
|
||||||
|
version = "0.1.0"
|
||||||
|
license = "MIT"
|
||||||
|
edition = "2021"
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
egui = "0.27"
|
||||||
|
eframe = "0.27"
|
||||||
|
petgraph = "0.6"
|
||||||
|
egui_graphs = { version = "0.20.0", features = ["events"] }
|
||||||
|
serde_json = "1.0"
|
||||||
|
fdg-sim = "0.9"
|
||||||
|
rand = "0.8"
|
||||||
|
crossbeam = "0.8"
|
@ -0,0 +1,10 @@
|
|||||||
|
# Configurable
|
||||||
|
Configurable example where you can toggle settings of the `GraphView` widget and see the result immediately.
|
||||||
|
It also contains controls to play with the graph.
|
||||||
|
|
||||||
|
This example also demonstrates the usage of force-directed layout implemented on client side.
|
||||||
|
|
||||||
|
## run
|
||||||
|
```bash
|
||||||
|
cargo run --release -p configurable
|
||||||
|
```
|
@ -0,0 +1,584 @@
|
|||||||
|
use std::time::Instant;
|
||||||
|
|
||||||
|
use crossbeam::channel::{unbounded, Receiver, Sender};
|
||||||
|
use eframe::{run_native, App, CreationContext};
|
||||||
|
use egui::{CollapsingHeader, Context, Pos2, ScrollArea, Slider, Ui};
|
||||||
|
use egui_graphs::events::Event;
|
||||||
|
use egui_graphs::{to_graph, DefaultEdgeShape, DefaultNodeShape, Graph, GraphView};
|
||||||
|
use fdg_sim::glam::Vec3;
|
||||||
|
use fdg_sim::{ForceGraph, ForceGraphHelper, Simulation, SimulationParameters};
|
||||||
|
use petgraph::stable_graph::{DefaultIx, EdgeIndex, NodeIndex, StableGraph};
|
||||||
|
use petgraph::Directed;
|
||||||
|
use rand::Rng;
|
||||||
|
use settings::{SettingsGraph, SettingsInteraction, SettingsNavigation, SettingsStyle};
|
||||||
|
|
||||||
|
mod settings;
|
||||||
|
|
||||||
|
const SIMULATION_DT: f32 = 0.035;
|
||||||
|
const EVENTS_LIMIT: usize = 100;
|
||||||
|
|
||||||
|
pub struct ConfigurableApp {
|
||||||
|
g: Graph<(), (), Directed, DefaultIx>,
|
||||||
|
sim: Simulation<(), f32>,
|
||||||
|
|
||||||
|
settings_graph: SettingsGraph,
|
||||||
|
settings_interaction: SettingsInteraction,
|
||||||
|
settings_navigation: SettingsNavigation,
|
||||||
|
settings_style: SettingsStyle,
|
||||||
|
|
||||||
|
last_events: Vec<String>,
|
||||||
|
|
||||||
|
simulation_stopped: bool,
|
||||||
|
|
||||||
|
fps: f64,
|
||||||
|
last_update_time: Instant,
|
||||||
|
frames_last_time_span: usize,
|
||||||
|
|
||||||
|
event_publisher: Sender<Event>,
|
||||||
|
event_consumer: Receiver<Event>,
|
||||||
|
|
||||||
|
pan: Option<[f32; 2]>,
|
||||||
|
zoom: Option<f32>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ConfigurableApp {
|
||||||
|
fn new(_: &CreationContext<'_>) -> Self {
|
||||||
|
let settings_graph = SettingsGraph::default();
|
||||||
|
let (g, sim) = generate(&settings_graph);
|
||||||
|
let (event_publisher, event_consumer) = unbounded();
|
||||||
|
Self {
|
||||||
|
g,
|
||||||
|
sim,
|
||||||
|
|
||||||
|
event_consumer,
|
||||||
|
event_publisher,
|
||||||
|
|
||||||
|
settings_graph,
|
||||||
|
|
||||||
|
settings_interaction: SettingsInteraction::default(),
|
||||||
|
settings_navigation: SettingsNavigation::default(),
|
||||||
|
settings_style: SettingsStyle::default(),
|
||||||
|
|
||||||
|
last_events: Vec::default(),
|
||||||
|
|
||||||
|
simulation_stopped: false,
|
||||||
|
|
||||||
|
fps: 0.,
|
||||||
|
last_update_time: Instant::now(),
|
||||||
|
frames_last_time_span: 0,
|
||||||
|
|
||||||
|
pan: Option::default(),
|
||||||
|
zoom: Option::default(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn update_simulation(&mut self) {
|
||||||
|
if self.simulation_stopped {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// the following manipulations is a hack to avoid having looped edges in the simulation
|
||||||
|
// because they cause the simulation to blow up;
|
||||||
|
// this is the issue of the fdg_sim engine we use for the simulation
|
||||||
|
// https://github.com/grantshandy/fdg/issues/10
|
||||||
|
// * remove loop edges
|
||||||
|
// * update simulation
|
||||||
|
// * restore loop edges
|
||||||
|
|
||||||
|
// remove looped edges
|
||||||
|
let looped_nodes = {
|
||||||
|
let graph = self.sim.get_graph_mut();
|
||||||
|
let mut looped_nodes = vec![];
|
||||||
|
let mut looped_edges = vec![];
|
||||||
|
graph.edge_indices().for_each(|idx| {
|
||||||
|
let edge = graph.edge_endpoints(idx).unwrap();
|
||||||
|
let looped = edge.0 == edge.1;
|
||||||
|
if looped {
|
||||||
|
looped_nodes.push((edge.0, ()));
|
||||||
|
looped_edges.push(idx);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
for idx in looped_edges {
|
||||||
|
graph.remove_edge(idx);
|
||||||
|
}
|
||||||
|
|
||||||
|
self.sim.update(SIMULATION_DT);
|
||||||
|
|
||||||
|
looped_nodes
|
||||||
|
};
|
||||||
|
|
||||||
|
// restore looped edges
|
||||||
|
let graph = self.sim.get_graph_mut();
|
||||||
|
for (idx, _) in looped_nodes.iter() {
|
||||||
|
graph.add_edge(*idx, *idx, 1.);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Syncs the graph with the simulation.
|
||||||
|
///
|
||||||
|
/// Changes location of nodes in `g` according to the locations in `sim`. If node from `g` is dragged its location is prioritized
|
||||||
|
/// over the location of the corresponding node from `sim` and this location is set to the node from the `sim`.
|
||||||
|
///
|
||||||
|
/// If node or edge is selected it is added to the corresponding selected field in `self`.
|
||||||
|
fn sync_graph_with_simulation(&mut self) {
|
||||||
|
let g_indices = self.g.g.node_indices().collect::<Vec<_>>();
|
||||||
|
for g_n_idx in &g_indices {
|
||||||
|
let g_n = self.g.g.node_weight_mut(*g_n_idx).unwrap();
|
||||||
|
let sim_n = self.sim.get_graph_mut().node_weight_mut(*g_n_idx).unwrap();
|
||||||
|
|
||||||
|
let loc = sim_n.location;
|
||||||
|
g_n.set_location(Pos2::new(loc.x, loc.y));
|
||||||
|
}
|
||||||
|
|
||||||
|
// reset the weights of the edges
|
||||||
|
self.sim.get_graph_mut().edge_weights_mut().for_each(|w| {
|
||||||
|
*w = 1.;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
fn update_fps(&mut self) {
|
||||||
|
self.frames_last_time_span += 1;
|
||||||
|
let now = Instant::now();
|
||||||
|
let elapsed = now.duration_since(self.last_update_time);
|
||||||
|
if elapsed.as_secs() >= 1 {
|
||||||
|
self.last_update_time = now;
|
||||||
|
self.fps = self.frames_last_time_span as f64 / elapsed.as_secs_f64();
|
||||||
|
self.frames_last_time_span = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn reset_graph(&mut self, ui: &mut Ui) {
|
||||||
|
let settings_graph = SettingsGraph::default();
|
||||||
|
let (g, sim) = generate(&settings_graph);
|
||||||
|
|
||||||
|
self.g = g;
|
||||||
|
self.sim = sim;
|
||||||
|
self.settings_graph = settings_graph;
|
||||||
|
self.last_events = Vec::default();
|
||||||
|
|
||||||
|
GraphView::<(), (), Directed, DefaultIx>::reset_metadata(ui);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn handle_events(&mut self) {
|
||||||
|
self.event_consumer.try_iter().for_each(|e| {
|
||||||
|
if self.last_events.len() > EVENTS_LIMIT {
|
||||||
|
self.last_events.remove(0);
|
||||||
|
}
|
||||||
|
self.last_events.push(serde_json::to_string(&e).unwrap());
|
||||||
|
|
||||||
|
match e {
|
||||||
|
Event::Pan(payload) => match self.pan {
|
||||||
|
Some(pan) => {
|
||||||
|
self.pan = Some([pan[0] + payload.diff[0], pan[1] + payload.diff[1]]);
|
||||||
|
}
|
||||||
|
None => {
|
||||||
|
self.pan = Some(payload.diff);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Event::Zoom(z) => {
|
||||||
|
match self.zoom {
|
||||||
|
Some(zoom) => {
|
||||||
|
self.zoom = Some(zoom + z.diff);
|
||||||
|
}
|
||||||
|
None => {
|
||||||
|
self.zoom = Some(z.diff);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
Event::NodeMove(payload) => {
|
||||||
|
let node_id = NodeIndex::new(payload.id);
|
||||||
|
let diff = Vec3::new(payload.diff[0], payload.diff[1], 0.);
|
||||||
|
|
||||||
|
let node = self.sim.get_graph_mut().node_weight_mut(node_id).unwrap();
|
||||||
|
node.location += diff;
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
fn random_node_idx(&self) -> Option<NodeIndex> {
|
||||||
|
let nodes_cnt = self.g.node_count();
|
||||||
|
if nodes_cnt == 0 {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
|
||||||
|
let random_n_idx = rand::thread_rng().gen_range(0..nodes_cnt);
|
||||||
|
self.g.g.node_indices().nth(random_n_idx)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn random_edge_idx(&self) -> Option<EdgeIndex> {
|
||||||
|
let edges_cnt = self.g.edge_count();
|
||||||
|
if edges_cnt == 0 {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
|
||||||
|
let random_e_idx = rand::thread_rng().gen_range(0..edges_cnt);
|
||||||
|
self.g.g.edge_indices().nth(random_e_idx)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn remove_random_node(&mut self) {
|
||||||
|
let idx = self.random_node_idx().unwrap();
|
||||||
|
self.remove_node(idx);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn add_random_node(&mut self) {
|
||||||
|
let random_n_idx = self.random_node_idx();
|
||||||
|
if random_n_idx.is_none() {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let random_n = self.g.node(random_n_idx.unwrap()).unwrap();
|
||||||
|
|
||||||
|
// location of new node is in surrounging of random existing node
|
||||||
|
let mut rng = rand::thread_rng();
|
||||||
|
let location = Pos2::new(
|
||||||
|
random_n.location().x + 10. + rng.gen_range(0. ..50.),
|
||||||
|
random_n.location().y + 10. + rng.gen_range(0. ..50.),
|
||||||
|
);
|
||||||
|
|
||||||
|
let idx = self.g.add_node_with_location((), location);
|
||||||
|
|
||||||
|
let mut sim_node = fdg_sim::Node::new(idx.index().to_string().as_str(), ());
|
||||||
|
sim_node.location = Vec3::new(location.x, location.y, 0.);
|
||||||
|
self.sim.get_graph_mut().add_node(sim_node);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn remove_node(&mut self, idx: NodeIndex) {
|
||||||
|
self.g.remove_node(idx);
|
||||||
|
|
||||||
|
self.sim.get_graph_mut().remove_node(idx).unwrap();
|
||||||
|
|
||||||
|
// update edges count
|
||||||
|
self.settings_graph.count_edge = self.g.edge_count();
|
||||||
|
}
|
||||||
|
|
||||||
|
fn add_random_edge(&mut self) {
|
||||||
|
let random_start = self.random_node_idx().unwrap();
|
||||||
|
let random_end = self.random_node_idx().unwrap();
|
||||||
|
|
||||||
|
self.add_edge(random_start, random_end);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn add_edge(&mut self, start: NodeIndex, end: NodeIndex) {
|
||||||
|
self.g.add_edge(start, end, ());
|
||||||
|
|
||||||
|
self.sim.get_graph_mut().add_edge(start, end, 1.);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn remove_random_edge(&mut self) {
|
||||||
|
let random_e_idx = self.random_edge_idx();
|
||||||
|
if random_e_idx.is_none() {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let endpoints = self.g.edge_endpoints(random_e_idx.unwrap()).unwrap();
|
||||||
|
|
||||||
|
self.remove_edge(endpoints.0, endpoints.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn remove_edge(&mut self, start: NodeIndex, end: NodeIndex) {
|
||||||
|
let (g_idx, _) = self.g.edges_connecting(start, end).next().unwrap();
|
||||||
|
self.g.remove_edge(g_idx);
|
||||||
|
|
||||||
|
let sim_idx = self.sim.get_graph_mut().find_edge(start, end).unwrap();
|
||||||
|
self.sim.get_graph_mut().remove_edge(sim_idx).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
fn draw_section_app(&mut self, ui: &mut Ui) {
|
||||||
|
CollapsingHeader::new("App Config")
|
||||||
|
.default_open(true)
|
||||||
|
.show(ui, |ui| {
|
||||||
|
ui.add_space(10.);
|
||||||
|
|
||||||
|
ui.label("Simulation");
|
||||||
|
ui.separator();
|
||||||
|
|
||||||
|
ui.horizontal(|ui| {
|
||||||
|
if ui
|
||||||
|
.button(match self.simulation_stopped {
|
||||||
|
true => "start",
|
||||||
|
false => "stop",
|
||||||
|
})
|
||||||
|
.clicked()
|
||||||
|
{
|
||||||
|
self.simulation_stopped = !self.simulation_stopped;
|
||||||
|
};
|
||||||
|
if ui.button("reset").clicked() {
|
||||||
|
self.reset_graph(ui);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
ui.add_space(10.);
|
||||||
|
|
||||||
|
self.draw_counts_sliders(ui);
|
||||||
|
|
||||||
|
ui.add_space(10.);
|
||||||
|
|
||||||
|
ui.separator();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
fn draw_section_widget(&mut self, ui: &mut Ui) {
|
||||||
|
CollapsingHeader::new("Widget")
|
||||||
|
.default_open(true)
|
||||||
|
.show(ui, |ui| {
|
||||||
|
CollapsingHeader::new("Navigation").default_open(true).show(ui, |ui|{
|
||||||
|
if ui
|
||||||
|
.checkbox(&mut self.settings_navigation.fit_to_screen_enabled, "fit_to_screen")
|
||||||
|
.changed()
|
||||||
|
&& self.settings_navigation.fit_to_screen_enabled
|
||||||
|
{
|
||||||
|
self.settings_navigation.zoom_and_pan_enabled = false
|
||||||
|
};
|
||||||
|
ui.label("Enable fit to screen to fit the graph to the screen on every frame.");
|
||||||
|
|
||||||
|
ui.add_space(5.);
|
||||||
|
|
||||||
|
ui.add_enabled_ui(!self.settings_navigation.fit_to_screen_enabled, |ui| {
|
||||||
|
ui.vertical(|ui| {
|
||||||
|
ui.checkbox(&mut self.settings_navigation.zoom_and_pan_enabled, "zoom_and_pan");
|
||||||
|
ui.label("Zoom with ctrl + mouse wheel, pan with middle mouse drag.");
|
||||||
|
}).response.on_disabled_hover_text("disable fit_to_screen to enable zoom_and_pan");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
CollapsingHeader::new("Style").show(ui, |ui| {
|
||||||
|
ui.checkbox(&mut self.settings_style.labels_always, "labels_always");
|
||||||
|
ui.label("Wheter to show labels always or when interacted only.");
|
||||||
|
});
|
||||||
|
|
||||||
|
CollapsingHeader::new("Interaction").show(ui, |ui| {
|
||||||
|
if ui.checkbox(&mut self.settings_interaction.dragging_enabled, "dragging_enabled").clicked() && self.settings_interaction.dragging_enabled {
|
||||||
|
self.settings_interaction.node_clicking_enabled = true;
|
||||||
|
};
|
||||||
|
ui.label("To drag use LMB click + drag on a node.");
|
||||||
|
|
||||||
|
ui.add_space(5.);
|
||||||
|
|
||||||
|
ui.add_enabled_ui(!(self.settings_interaction.dragging_enabled || self.settings_interaction.node_selection_enabled || self.settings_interaction.node_selection_multi_enabled), |ui| {
|
||||||
|
ui.vertical(|ui| {
|
||||||
|
ui.checkbox(&mut self.settings_interaction.node_clicking_enabled, "node_clicking_enabled");
|
||||||
|
ui.label("Check click events in last events");
|
||||||
|
}).response.on_disabled_hover_text("node click is enabled when any of the interaction is also enabled");
|
||||||
|
});
|
||||||
|
|
||||||
|
ui.add_space(5.);
|
||||||
|
|
||||||
|
ui.add_enabled_ui(!self.settings_interaction.node_selection_multi_enabled, |ui| {
|
||||||
|
ui.vertical(|ui| {
|
||||||
|
if ui.checkbox(&mut self.settings_interaction.node_selection_enabled, "node_selection_enabled").clicked() && self.settings_interaction.node_selection_enabled {
|
||||||
|
self.settings_interaction.node_clicking_enabled = true;
|
||||||
|
};
|
||||||
|
ui.label("Enable select to select nodes with LMB click. If node is selected clicking on it again will deselect it.");
|
||||||
|
}).response.on_disabled_hover_text("node_selection_multi_enabled enables select");
|
||||||
|
});
|
||||||
|
|
||||||
|
if ui.checkbox(&mut self.settings_interaction.node_selection_multi_enabled, "node_selection_multi_enabled").changed() && self.settings_interaction.node_selection_multi_enabled {
|
||||||
|
self.settings_interaction.node_clicking_enabled = true;
|
||||||
|
self.settings_interaction.node_selection_enabled = true;
|
||||||
|
}
|
||||||
|
ui.label("Enable multiselect to select multiple nodes.");
|
||||||
|
|
||||||
|
ui.add_space(5.);
|
||||||
|
|
||||||
|
ui.add_enabled_ui(!(self.settings_interaction.edge_selection_enabled || self.settings_interaction.edge_selection_multi_enabled), |ui| {
|
||||||
|
ui.vertical(|ui| {
|
||||||
|
ui.checkbox(&mut self.settings_interaction.edge_clicking_enabled, "edge_clicking_enabled");
|
||||||
|
ui.label("Check click events in last events");
|
||||||
|
}).response.on_disabled_hover_text("edge click is enabled when any of the interaction is also enabled");
|
||||||
|
});
|
||||||
|
|
||||||
|
ui.add_space(5.);
|
||||||
|
|
||||||
|
ui.add_enabled_ui(!self.settings_interaction.edge_selection_multi_enabled, |ui| {
|
||||||
|
ui.vertical(|ui| {
|
||||||
|
if ui.checkbox(&mut self.settings_interaction.edge_selection_enabled, "edge_selection_enabled").clicked() && self.settings_interaction.edge_selection_enabled {
|
||||||
|
self.settings_interaction.edge_clicking_enabled = true;
|
||||||
|
};
|
||||||
|
ui.label("Enable select to select edges with LMB click. If edge is selected clicking on it again will deselect it.");
|
||||||
|
}).response.on_disabled_hover_text("edge_selection_multi_enabled enables select");
|
||||||
|
});
|
||||||
|
|
||||||
|
if ui.checkbox(&mut self.settings_interaction.edge_selection_multi_enabled, "edge_selection_multi_enabled").changed() && self.settings_interaction.edge_selection_multi_enabled {
|
||||||
|
self.settings_interaction.edge_clicking_enabled = true;
|
||||||
|
self.settings_interaction.edge_selection_enabled = true;
|
||||||
|
}
|
||||||
|
ui.label("Enable multiselect to select multiple edges.");
|
||||||
|
});
|
||||||
|
|
||||||
|
CollapsingHeader::new("Selected").default_open(true).show(ui, |ui| {
|
||||||
|
ScrollArea::vertical().auto_shrink([false, true]).max_height(200.).show(ui, |ui| {
|
||||||
|
self.g.selected_nodes().iter().for_each(|node| {
|
||||||
|
ui.label(format!("{node:?}"));
|
||||||
|
});
|
||||||
|
self.g.selected_edges().iter().for_each(|edge| {
|
||||||
|
ui.label(format!("{edge:?}"));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
CollapsingHeader::new("Last Events").default_open(true).show(ui, |ui| {
|
||||||
|
ScrollArea::vertical().auto_shrink([false, true]).max_height(200.).show(ui, |ui| {
|
||||||
|
self.last_events.iter().rev().for_each(|event| {
|
||||||
|
ui.label(event);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
fn draw_section_debug(&mut self, ui: &mut Ui) {
|
||||||
|
CollapsingHeader::new("Debug")
|
||||||
|
.default_open(true)
|
||||||
|
.show(ui, |ui| {
|
||||||
|
if let Some(zoom) = self.zoom {
|
||||||
|
ui.label(format!("zoom: {:.5}", zoom));
|
||||||
|
};
|
||||||
|
if let Some(pan) = self.pan {
|
||||||
|
ui.label(format!("pan: [{:.5}, {:.5}]", pan[0], pan[1]));
|
||||||
|
};
|
||||||
|
|
||||||
|
ui.label(format!("FPS: {:.1}", self.fps));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
fn draw_counts_sliders(&mut self, ui: &mut Ui) {
|
||||||
|
ui.horizontal(|ui| {
|
||||||
|
let before = self.settings_graph.count_node as i32;
|
||||||
|
|
||||||
|
ui.add(Slider::new(&mut self.settings_graph.count_node, 1..=2500).text("nodes"));
|
||||||
|
|
||||||
|
let delta = self.settings_graph.count_node as i32 - before;
|
||||||
|
(0..delta.abs()).for_each(|_| {
|
||||||
|
if delta > 0 {
|
||||||
|
self.add_random_node();
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
self.remove_random_node();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
ui.horizontal(|ui| {
|
||||||
|
let before = self.settings_graph.count_edge as i32;
|
||||||
|
|
||||||
|
ui.add(Slider::new(&mut self.settings_graph.count_edge, 0..=5000).text("edges"));
|
||||||
|
|
||||||
|
let delta = self.settings_graph.count_edge as i32 - before;
|
||||||
|
(0..delta.abs()).for_each(|_| {
|
||||||
|
if delta > 0 {
|
||||||
|
self.add_random_edge();
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
self.remove_random_edge();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl App for ConfigurableApp {
|
||||||
|
fn update(&mut self, ctx: &Context, _: &mut eframe::Frame) {
|
||||||
|
egui::SidePanel::right("right_panel")
|
||||||
|
.min_width(250.)
|
||||||
|
.show(ctx, |ui| {
|
||||||
|
ScrollArea::vertical().show(ui, |ui| {
|
||||||
|
self.draw_section_app(ui);
|
||||||
|
ui.add_space(10.);
|
||||||
|
self.draw_section_debug(ui);
|
||||||
|
ui.add_space(10.);
|
||||||
|
self.draw_section_widget(ui);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
egui::CentralPanel::default().show(ctx, |ui| {
|
||||||
|
let settings_interaction = &egui_graphs::SettingsInteraction::new()
|
||||||
|
.with_node_selection_enabled(self.settings_interaction.node_selection_enabled)
|
||||||
|
.with_node_selection_multi_enabled(
|
||||||
|
self.settings_interaction.node_selection_multi_enabled,
|
||||||
|
)
|
||||||
|
.with_dragging_enabled(self.settings_interaction.dragging_enabled)
|
||||||
|
.with_node_clicking_enabled(self.settings_interaction.node_clicking_enabled)
|
||||||
|
.with_edge_clicking_enabled(self.settings_interaction.edge_clicking_enabled)
|
||||||
|
.with_edge_selection_enabled(self.settings_interaction.edge_selection_enabled)
|
||||||
|
.with_edge_selection_multi_enabled(
|
||||||
|
self.settings_interaction.edge_selection_multi_enabled,
|
||||||
|
);
|
||||||
|
let settings_navigation = &egui_graphs::SettingsNavigation::new()
|
||||||
|
.with_zoom_and_pan_enabled(self.settings_navigation.zoom_and_pan_enabled)
|
||||||
|
.with_fit_to_screen_enabled(self.settings_navigation.fit_to_screen_enabled)
|
||||||
|
.with_zoom_speed(self.settings_navigation.zoom_speed);
|
||||||
|
let settings_style = &egui_graphs::SettingsStyle::new()
|
||||||
|
.with_labels_always(self.settings_style.labels_always);
|
||||||
|
ui.add(
|
||||||
|
&mut GraphView::<_, _, _, _, DefaultNodeShape, DefaultEdgeShape>::new(&mut self.g)
|
||||||
|
.with_interactions(settings_interaction)
|
||||||
|
.with_navigations(settings_navigation)
|
||||||
|
.with_styles(settings_style)
|
||||||
|
.with_events(&self.event_publisher),
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
self.handle_events();
|
||||||
|
self.sync_graph_with_simulation();
|
||||||
|
|
||||||
|
self.update_simulation();
|
||||||
|
self.update_fps();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn generate(settings: &SettingsGraph) -> (Graph<(), (), Directed, DefaultIx>, Simulation<(), f32>) {
|
||||||
|
let g = generate_random_graph(settings.count_node, settings.count_edge);
|
||||||
|
let sim = construct_simulation(&g);
|
||||||
|
|
||||||
|
(g, sim)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn construct_simulation(g: &Graph<(), (), Directed, DefaultIx>) -> Simulation<(), f32> {
|
||||||
|
// create force graph
|
||||||
|
let mut force_graph = ForceGraph::with_capacity(g.g.node_count(), g.g.edge_count());
|
||||||
|
g.g.node_indices().for_each(|idx| {
|
||||||
|
let idx = idx.index();
|
||||||
|
force_graph.add_force_node(format!("{idx}").as_str(), ());
|
||||||
|
});
|
||||||
|
g.g.edge_indices().for_each(|idx| {
|
||||||
|
let (source, target) = g.g.edge_endpoints(idx).unwrap();
|
||||||
|
force_graph.add_edge(source, target, 1.);
|
||||||
|
});
|
||||||
|
|
||||||
|
// initialize simulation
|
||||||
|
let mut params = SimulationParameters::default();
|
||||||
|
let force = fdg_sim::force::fruchterman_reingold_weighted(100., 0.5);
|
||||||
|
params.set_force(force);
|
||||||
|
|
||||||
|
Simulation::from_graph(force_graph, params)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn generate_random_graph(node_count: usize, edge_count: usize) -> Graph<(), ()> {
|
||||||
|
let mut rng = rand::thread_rng();
|
||||||
|
let mut graph = StableGraph::new();
|
||||||
|
|
||||||
|
// add nodes
|
||||||
|
for _ in 0..node_count {
|
||||||
|
graph.add_node(());
|
||||||
|
}
|
||||||
|
|
||||||
|
// add random edges
|
||||||
|
for _ in 0..edge_count {
|
||||||
|
let source = rng.gen_range(0..node_count);
|
||||||
|
let target = rng.gen_range(0..node_count);
|
||||||
|
|
||||||
|
graph.add_edge(NodeIndex::new(source), NodeIndex::new(target), ());
|
||||||
|
}
|
||||||
|
|
||||||
|
to_graph(&graph)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
let native_options = eframe::NativeOptions::default();
|
||||||
|
run_native(
|
||||||
|
"egui_graphs_configurable_demo",
|
||||||
|
native_options,
|
||||||
|
Box::new(|cc| Box::new(ConfigurableApp::new(cc))),
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
}
|
@ -0,0 +1,47 @@
|
|||||||
|
pub struct SettingsGraph {
|
||||||
|
pub count_node: usize,
|
||||||
|
pub count_edge: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for SettingsGraph {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self {
|
||||||
|
count_node: 300,
|
||||||
|
count_edge: 500,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Default)]
|
||||||
|
pub struct SettingsInteraction {
|
||||||
|
pub dragging_enabled: bool,
|
||||||
|
pub node_clicking_enabled: bool,
|
||||||
|
pub node_selection_enabled: bool,
|
||||||
|
pub node_selection_multi_enabled: bool,
|
||||||
|
pub edge_clicking_enabled: bool,
|
||||||
|
pub edge_selection_enabled: bool,
|
||||||
|
pub edge_selection_multi_enabled: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct SettingsNavigation {
|
||||||
|
pub fit_to_screen_enabled: bool,
|
||||||
|
pub zoom_and_pan_enabled: bool,
|
||||||
|
pub screen_padding: f32,
|
||||||
|
pub zoom_speed: f32,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for SettingsNavigation {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self {
|
||||||
|
screen_padding: 0.3,
|
||||||
|
zoom_speed: 0.1,
|
||||||
|
fit_to_screen_enabled: true,
|
||||||
|
zoom_and_pan_enabled: false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Default)]
|
||||||
|
pub struct SettingsStyle {
|
||||||
|
pub labels_always: bool,
|
||||||
|
}
|
@ -0,0 +1,11 @@
|
|||||||
|
[package]
|
||||||
|
name = "egui-graph-viz"
|
||||||
|
version = "0.1.0"
|
||||||
|
license = "MIT"
|
||||||
|
edition = "2021"
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
egui = "0.27"
|
||||||
|
eframe = "0.27"
|
||||||
|
petgraph = "0.6"
|
||||||
|
egui_graphs = "0.20.0"
|
@ -0,0 +1,9 @@
|
|||||||
|
# Interactive
|
||||||
|
Modification of the [basic example](https://github.com/blitzarx1/egui_graph/tree/master/examples/basic) which shows how easy it is to enable interactivity.
|
||||||
|
|
||||||
|
Try to drag around and select some nodes.
|
||||||
|
|
||||||
|
## run
|
||||||
|
```bash
|
||||||
|
cargo run --release -p interactive
|
||||||
|
```
|
@ -0,0 +1,64 @@
|
|||||||
|
use eframe::{run_native, App, CreationContext};
|
||||||
|
use egui::Context;
|
||||||
|
use egui_graphs::{
|
||||||
|
DefaultEdgeShape, DefaultNodeShape, Graph, GraphView, SettingsInteraction, SettingsStyle,
|
||||||
|
};
|
||||||
|
use petgraph::stable_graph::StableGraph;
|
||||||
|
|
||||||
|
pub struct InteractiveApp {
|
||||||
|
g: Graph<(), ()>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl InteractiveApp {
|
||||||
|
fn new(_: &CreationContext<'_>) -> Self {
|
||||||
|
let g = generate_graph();
|
||||||
|
Self { g }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl App for InteractiveApp {
|
||||||
|
fn update(&mut self, ctx: &Context, _: &mut eframe::Frame) {
|
||||||
|
egui::CentralPanel::default().show(ctx, |ui| {
|
||||||
|
let interaction_settings = &SettingsInteraction::new()
|
||||||
|
.with_dragging_enabled(true)
|
||||||
|
.with_node_clicking_enabled(true)
|
||||||
|
.with_node_selection_enabled(true)
|
||||||
|
.with_node_selection_multi_enabled(true)
|
||||||
|
.with_edge_clicking_enabled(true)
|
||||||
|
.with_edge_selection_enabled(true)
|
||||||
|
.with_edge_selection_multi_enabled(true);
|
||||||
|
let style_settings = &SettingsStyle::new().with_labels_always(true);
|
||||||
|
ui.add(
|
||||||
|
&mut GraphView::<_, _, _, _, DefaultNodeShape, DefaultEdgeShape>::new(&mut self.g)
|
||||||
|
.with_styles(style_settings)
|
||||||
|
.with_interactions(interaction_settings),
|
||||||
|
);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn generate_graph() -> Graph<(), ()> {
|
||||||
|
let mut g = StableGraph::new();
|
||||||
|
|
||||||
|
let a = g.add_node(());
|
||||||
|
let b = g.add_node(());
|
||||||
|
let c = g.add_node(());
|
||||||
|
|
||||||
|
g.add_edge(a, a, ());
|
||||||
|
g.add_edge(a, b, ());
|
||||||
|
g.add_edge(a, b, ());
|
||||||
|
g.add_edge(b, c, ());
|
||||||
|
g.add_edge(c, a, ());
|
||||||
|
|
||||||
|
Graph::from(&g)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
let native_options = eframe::NativeOptions::default();
|
||||||
|
run_native(
|
||||||
|
"egui_graphs_interactive_demo",
|
||||||
|
native_options,
|
||||||
|
Box::new(|cc| Box::new(InteractiveApp::new(cc))),
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
|
}
|
@ -0,0 +1,26 @@
|
|||||||
|
struct AdjacencyGraph {
|
||||||
|
nodes: Vec<&str>,
|
||||||
|
adjacencies: HashMap<&str, Vec<&str>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AdjacencyGraph {
|
||||||
|
fn new() -> Self {
|
||||||
|
AdjacencyGraph {
|
||||||
|
nodes: Vec::new(),
|
||||||
|
adjacencies: HashMap::new(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn add_node(&mut self, node: &str) {
|
||||||
|
self.nodes.push(node);
|
||||||
|
self.adjacencies.insert(node, Vec::new());
|
||||||
|
}
|
||||||
|
|
||||||
|
fn add_edge(&mut self, from: &str, to: &str) {
|
||||||
|
self.adjacencies.get_mut(from).unwrap().push(to);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn neighbors(&self, node: &str) -> Option<&Vec<&str>> {
|
||||||
|
self.adjacencies.get(node)
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue