Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Simplify graph viewer rendering logic #8227

Merged
merged 32 commits into from
Nov 29, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
32 commits
Select commit Hold shift + click to select a range
e62c933
Prototype of scene
abey79 Nov 26, 2024
7cc8a42
Restruture `nodes` visualizer
grtlr Nov 26, 2024
cec7395
Implement text and cirlce rendering
grtlr Nov 26, 2024
48957e6
Properly register pan and zoom
grtlr Nov 26, 2024
c189ef5
Fix panning and re-add debug info
grtlr Nov 27, 2024
145a1a4
Restructure into `canvas`
grtlr Nov 27, 2024
45a2dc7
Convert to `DrawableNode`
grtlr Nov 27, 2024
22b5489
[skip ci] Get layout to work again
grtlr Nov 27, 2024
2a1535c
[skip ci] Introduce new `LayoutRequest` to get rid of `Discriminator`
grtlr Nov 27, 2024
226804c
[skip ci] Redo graph viewer
grtlr Nov 27, 2024
f8429ae
Add entiy rects
grtlr Nov 28, 2024
f50d698
[skip ci] Fix interactions and highlights
grtlr Nov 28, 2024
19902a3
Code cleanup
grtlr Nov 28, 2024
96927da
Rename `NodeIndex` to `NodeId`
grtlr Nov 28, 2024
0cc4fec
Fix double-click to center
grtlr Nov 28, 2024
2de4d1c
Re-add tracing
grtlr Nov 28, 2024
1874a67
Remove un-needed `size_hash`
grtlr Nov 28, 2024
dedd199
Fix combined hashing
grtlr Nov 28, 2024
8ff48a2
[skip ci] Try to fix ray intersection
grtlr Nov 28, 2024
86bb623
[skip ci] Add doccomment
grtlr Nov 28, 2024
7fcbb1a
Add `Option` to `LayoutResult` getters
grtlr Nov 28, 2024
db3a546
[ski ci] doc comments
grtlr Nov 28, 2024
98c9fc8
Provide many comments and try to clarify
grtlr Nov 28, 2024
96f7a53
[skip ci] unused for now
grtlr Nov 28, 2024
f8fd175
[skip ci] rename world -> scene
grtlr Nov 28, 2024
9df6694
[skip ci] display works, interactions don't
grtlr Nov 28, 2024
834fb2f
[skip ci] Fix tooltips and remaining interactions
grtlr Nov 28, 2024
9a5f261
Clean up
grtlr Nov 28, 2024
8d8056a
Remaining cleanup tasks
grtlr Nov 28, 2024
7a4ad6d
Merge branch 'main' into grtlr/simplify-graph-viewer
grtlr Nov 29, 2024
5c00d50
Fx lints
grtlr Nov 29, 2024
95e12e3
Fix typos
grtlr Nov 29, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion Cargo.lock
Original file line number Diff line number Diff line change
Expand Up @@ -6231,11 +6231,11 @@ name = "re_space_view_graph"
version = "0.21.0-alpha.1+dev"
dependencies = [
"ahash",
"bytemuck",
"egui",
"fjadra",
"nohash-hasher",
"re_chunk",
"re_entity_db",
"re_format",
"re_log_types",
"re_query",
Expand Down
2 changes: 1 addition & 1 deletion crates/viewer/re_space_view_graph/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ all-features = true

[dependencies]
re_chunk.workspace = true
re_entity_db.workspace = true
re_format.workspace = true
re_log_types.workspace = true
re_query.workspace = true
Expand All @@ -32,7 +33,6 @@ re_viewer_context.workspace = true
re_viewport_blueprint.workspace = true

ahash.workspace = true
bytemuck.workspace = true
egui.workspace = true
fjadra.workspace = true
nohash-hasher.workspace = true
15 changes: 7 additions & 8 deletions crates/viewer/re_space_view_graph/src/graph/index.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3,25 +3,24 @@ use re_types::components;

use super::GraphNodeHash;

#[derive(Clone, Copy, PartialEq, Eq)]
pub struct NodeIndex {
#[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord)]
pub struct NodeId {
pub entity_hash: EntityPathHash,
pub node_hash: GraphNodeHash,
}

impl nohash_hasher::IsEnabled for NodeIndex {}
impl nohash_hasher::IsEnabled for NodeId {}

// We implement `Hash` manually, because `nohash_hasher` requires a single call to `state.write_*`.
// More info: https://crates.io/crates/nohash-hasher
impl std::hash::Hash for NodeIndex {
impl std::hash::Hash for NodeId {
fn hash<H: std::hash::Hasher>(&self, state: &mut H) {
// TODO(grtlr): Consider using `write_usize` here, to further decrease the risk of collision.
let combined = self.entity_hash.hash64() << 32 | self.node_hash.hash64();
let combined = self.entity_hash.hash64() ^ self.node_hash.hash64();
state.write_u64(combined);
}
}

impl NodeIndex {
impl NodeId {
pub fn from_entity_node(entity_path: &EntityPath, node: &components::GraphNode) -> Self {
Self {
entity_hash: entity_path.hash(),
Expand All @@ -30,7 +29,7 @@ impl NodeIndex {
}
}

impl std::fmt::Debug for NodeIndex {
impl std::fmt::Debug for NodeId {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "NodeIndex({:?}@{:?})", self.node_hash, self.entity_hash)
}
Expand Down
159 changes: 118 additions & 41 deletions crates/viewer/re_space_view_graph/src/graph/mod.rs
Original file line number Diff line number Diff line change
@@ -1,87 +1,164 @@
mod hash;

use egui::{Pos2, Vec2};
pub(crate) use hash::GraphNodeHash;
mod index;
pub(crate) use index::NodeIndex;
pub(crate) use index::NodeId;

use re_chunk::EntityPath;
use re_types::components::{self, GraphType};

use re_types::components::{GraphNode, GraphType};
use crate::{
ui::DrawableLabel,
visualizers::{EdgeData, NodeData, NodeInstance},
};

use crate::visualizers::{EdgeData, EdgeInstance, NodeData, NodeInstance};
/// Describes the different kind of nodes that we can have in a graph.
pub enum Node {
/// An explicit node is a node that was provided via [`re_types::archetypes::GraphNodes`].
///
/// It therefore has an instance, as well as all properties that can be added via that archetype.
Explicit {
instance: NodeInstance,
label: DrawableLabel,
},

pub struct NodeInstanceImplicit {
pub node: GraphNode,
pub index: NodeIndex,
/// An implicit node is a node that was provided via [`re_types::archetypes::GraphEdges`], but does not have a corresponding [`re_types::components::GraphNode`] in an [`re_types::archetypes::GraphNodes`] archetype.
///
/// Because it was never specified directly, it also does not have many of the properties that an [`Node::Explicit`] has.
Implicit {
id: NodeId,
graph_node: components::GraphNode,
label: DrawableLabel,
},
}

impl std::hash::Hash for NodeInstanceImplicit {
fn hash<H: std::hash::Hasher>(&self, state: &mut H) {
self.index.hash(state);
impl Node {
pub fn id(&self) -> NodeId {
match self {
Self::Explicit { instance, .. } => instance.id,
Self::Implicit { id, .. } => *id,
}
}

/// The original [`components::GraphNode`] id that was logged by the user.
pub fn graph_node(&self) -> &components::GraphNode {
match self {
Self::Explicit { instance, .. } => &instance.graph_node,
Self::Implicit { graph_node, .. } => graph_node,
}
}

pub fn label(&self) -> &DrawableLabel {
match self {
Self::Explicit { label, .. } | Self::Implicit { label, .. } => label,
}
}

pub fn size(&self) -> Vec2 {
self.label().size()
}

pub fn position(&self) -> Option<Pos2> {
match self {
Self::Explicit {
instance: NodeInstance { position, .. },
..
} => *position,
Self::Implicit { .. } => None,
}
}
}

#[derive(Hash)]
pub struct Graph<'a> {
explicit: &'a [NodeInstance],
implicit: Vec<NodeInstanceImplicit>,
edges: &'a [EdgeInstance],
pub struct Edge {
pub from: NodeId,
pub to: NodeId,
pub arrow: bool,
}

pub struct Graph {
entity: EntityPath,
nodes: Vec<Node>,
edges: Vec<Edge>,
kind: GraphType,
}

impl<'a> Graph<'a> {
pub fn new(node_data: Option<&'a NodeData>, edge_data: Option<&'a EdgeData>) -> Self {
impl Graph {
pub fn new<'a>(
ui: &egui::Ui,
entity: EntityPath,
node_data: Option<&'a NodeData>,
edge_data: Option<&'a EdgeData>,
) -> Self {
// We keep track of the nodes to find implicit nodes.
let mut seen = ahash::HashSet::default();

let explicit = if let Some(data) = node_data {
seen.extend(data.nodes.iter().map(|n| n.index));
data.nodes.as_slice()
let mut nodes: Vec<Node> = if let Some(data) = node_data {
seen.extend(data.nodes.iter().map(|n| n.id));
// TODO(grtlr): We should see if we can get rid of some of the cloning here.
data.nodes
.iter()
.map(|n| Node::Explicit {
instance: n.clone(),
label: DrawableLabel::from_label(ui, &n.label),
})
.collect()
} else {
&[][..]
Vec::new()
};

let (edges, implicit, kind) = if let Some(data) = edge_data {
let mut implicit = Vec::new();
let (edges, kind) = if let Some(data) = edge_data {
for edge in &data.edges {
if !seen.contains(&edge.source_index) {
implicit.push(NodeInstanceImplicit {
node: edge.source.clone(),
index: edge.source_index,
nodes.push(Node::Implicit {
id: edge.source_index,
graph_node: edge.source.clone(),
label: DrawableLabel::implicit_circle(),
});
seen.insert(edge.source_index);
}
if !seen.contains(&edge.target_index) {
implicit.push(NodeInstanceImplicit {
node: edge.target.clone(),
index: edge.target_index,
nodes.push(Node::Implicit {
id: edge.target_index,
graph_node: edge.target.clone(),
label: DrawableLabel::implicit_circle(),
});
seen.insert(edge.target_index);
}
}
(data.edges.as_slice(), implicit, Some(data.graph_type))

let es = data.edges.iter().map(|e| Edge {
from: e.source_index,
to: e.target_index,
arrow: data.graph_type == GraphType::Directed,
});

(es.collect(), data.graph_type)
} else {
(&[][..], Vec::new(), None)
(Vec::new(), GraphType::default())
};

Self {
explicit,
implicit,
entity,
nodes,
edges,
kind: kind.unwrap_or_default(),
kind,
}
}

pub fn nodes_explicit(&self) -> impl Iterator<Item = &NodeInstance> {
self.explicit.iter()
}

pub fn nodes_implicit(&self) -> impl Iterator<Item = &NodeInstanceImplicit> + '_ {
self.implicit.iter()
pub fn nodes(&self) -> &[Node] {
&self.nodes
}

pub fn edges(&self) -> impl Iterator<Item = &EdgeInstance> {
self.edges.iter()
pub fn edges(&self) -> &[Edge] {
&self.edges
}

pub fn kind(&self) -> GraphType {
self.kind
}

pub fn entity(&self) -> &EntityPath {
&self.entity
}
}
Loading