1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
use cgmath::{Matrix4, Vector3, One};
use id_tree::*;
use ModelHandle;
use TextHandle;
use RenderGroup;

pub enum SceneNodeEntry {
    Model(ModelHandle),
    Text(TextHandle),
    Empty,
}

/// Scene entries have an optional model and a transform.
pub struct SceneNode {
    pub entry: SceneNodeEntry,
    pub group: RenderGroup,
    pub transform: Matrix4<f32>,
}

impl SceneNode {
    pub fn new(entry: SceneNodeEntry) -> Self {
        SceneNode {
            entry: entry,
            group: RenderGroup::Simple,
            transform: Matrix4::one(),
        }
    }

    pub fn empty() -> Self {
        SceneNode {
            entry: SceneNodeEntry::Empty,
            group: RenderGroup::None,
            transform: Matrix4::one(),
        }
    }

    pub fn translate(&mut self, translation: Vector3<f32>) {
        self.transform = self.transform * Matrix4::from_translation(translation);
    }
}

/// A directed acyclic graph for models with transformations at each node.
pub struct Scene {
    root_id: NodeId,
    pub graph: Tree<SceneNode>,
}

impl Scene {
    pub fn new() -> Self {
        use id_tree::InsertBehavior::*;

        let mut graph: Tree<SceneNode> = TreeBuilder::new().build();

        let root_node = SceneNode::empty();

        let root_id = graph.insert(Node::new(root_node), AsRoot).unwrap();

        Scene { root_id, graph }
    }

    pub fn traverse(&self) -> PreOrderTraversal<SceneNode> {
        self.graph.traverse_pre_order(&self.root_id).unwrap()
    }

    pub fn insert(&mut self, node: SceneNode, parent_id: &NodeId) -> NodeId {
        use id_tree::InsertBehavior::*;

        let child_id = self.graph
            .insert(Node::new(node), UnderNode(parent_id))
            .unwrap();

        child_id
    }

    pub fn get_root_id(&self) -> NodeId {
        self.root_id.clone()
    }

    pub fn get_graph(&mut self) -> &mut Tree<SceneNode> {
        &mut self.graph
    }
}