1use std::collections::VecDeque;
4use crate::interface;
5use crate::Tree;
6use crate::tree::{self, NodeId};
7
8pub trait TreeHelper {
10 fn get_element (&self, node_id : &NodeId) -> &interface::Element;
11 fn get_element_mut (&mut self, node_id : &NodeId) -> &mut interface::Element;
12 fn get_parent (&self, child : &NodeId) -> &interface::Element;
13 fn get_parent_mut (&mut self, child : &NodeId) -> &mut interface::Element;
14 fn get_parent_id (&self, child : &NodeId) -> &NodeId;
15 fn splice_subtree (
18 &mut self,
19 parent_id : &NodeId,
20 other : &Tree <interface::Element>,
21 other_id : &NodeId,
22 order : interface::CreateOrder
23 ) -> NodeId;
24 fn write_formatted_names <W : std::fmt::Write> (&self, w : &mut W)
27 -> std::fmt::Result;
28}
29
30pub fn map_tree <T, U, F> (tree : &Tree <T>, f : F) -> Tree <U> where
32 F : Fn (&T) -> U
33{
34 use tree::{InsertBehavior, Node};
35 let mut out = Tree::new();
36 if let Some (root_id) = tree.root_node_id() {
37 let root = tree.get (root_id).unwrap();
38 let new_root_id = out
39 .insert (Node::new (f (root.data())), InsertBehavior::AsRoot).unwrap();
40 let mut parent_child_ids = root.children().iter()
41 .map (|child_id| (new_root_id.clone(), child_id.clone()))
42 .collect::<VecDeque <_>>();
43 while let Some ((new_parent_id, old_child_id)) =
44 parent_child_ids.pop_front()
45 {
46 let child = tree.get (&old_child_id).unwrap();
47 let new_child_id = out.insert (
48 Node::new (f (child.data())),
49 InsertBehavior::UnderNode (&new_parent_id)
50 ).unwrap();
51 for child_id in child.children().iter() {
52 parent_child_ids.push_back ((new_child_id.clone(), child_id.clone()));
53 }
54 }
55 }
56 out
57}
58
59impl TreeHelper for Tree <interface::Element> {
60 #[inline]
64 fn get_element (&self, node_id : &NodeId) -> &interface::Element {
65 self.get (node_id).unwrap().data()
66 }
67
68 #[inline]
72 fn get_element_mut (&mut self, node_id : &NodeId) -> &mut interface::Element {
73 self.get_mut (node_id).unwrap().data_mut()
74 }
75
76 #[inline]
80 fn get_parent (&self, child : &NodeId) -> &interface::Element {
81 self.get_element (self.get_parent_id (child))
82 }
83
84 #[inline]
88 fn get_parent_mut (&mut self, child : &NodeId) -> &mut interface::Element {
89 let parent_id = self.get_parent_id (child).clone();
90 self.get_element_mut (&parent_id)
91 }
92
93 #[inline]
97 fn get_parent_id (&self, child : &NodeId) -> &NodeId {
98 self.get (child).unwrap().parent().unwrap()
99 }
100
101 fn splice_subtree (
104 &mut self,
105 parent_id : &NodeId,
106 other : &Tree <interface::Element>,
107 other_id : &NodeId,
108 order : interface::CreateOrder
109 ) -> NodeId {
110 use tree::Node;
111 use interface::CreateOrder;
112 let subtree_root = Node::new (other.get_element (other_id).clone());
113 let subtree_id = self.insert (
114 subtree_root, tree::InsertBehavior::UnderNode (parent_id)
115 ).unwrap();
116 match order {
117 CreateOrder::Append => {}
118 CreateOrder::Prepend => {
119 let _ = self.make_first_sibling (&subtree_id).unwrap();
120 }
121 CreateOrder::NthSibling (n) =>
122 self.make_nth_sibling (&subtree_id, n as usize).unwrap()
123 }
124 for child_id in other.children_ids (other_id).unwrap() {
125 self.splice_subtree (&subtree_id, other, child_id, CreateOrder::Append);
126 }
127 subtree_id
128 }
129
130 fn write_formatted_names<W: std::fmt::Write>(&self, w: &mut W)
131 -> std::fmt::Result
132 {
133 if let Some(node_id) = self.root_node_id() {
134 let childn = 0;
135 let level = 0;
136 let last = vec![];
137 let mut stack = vec![(node_id, childn, level, last)];
138 while let Some((node_id, childn, level, last)) = stack.pop() {
139 debug_assert_eq!(
140 last.len(), level,
141 "each previous level should indicate whether it has reached the last node"
142 );
143 let node = self.get(node_id)
144 .expect("getting node of existing node ref id");
145 if childn == 0 {
146 for i in 1..level {
147 if last[i - 1] {
148 write!(w, " ")?;
149 } else {
150 write!(w, "│ ")?;
151 }
152 }
153 if level > 0 {
154 if last[level - 1] {
155 write!(w, "└── ")?;
156 } else {
157 write!(w, "├── ")?;
158 }
159 }
160 let data = node.data();
161 let state = match data.controller.state {
162 interface::controller::State::Focused => "*",
163 interface::controller::State::Enabled => "",
164 interface::controller::State::Disabled => "†"
165 };
166 writeln!(w, "{}{}", data.name, state)?;
167 }
168 let mut children = node.children().iter().skip(childn);
169 if let Some(child_id) = children.next() {
170 let mut next_last = last.clone();
171 if children.next().is_some() {
172 stack.push((node_id, childn + 1, level, last));
173 next_last.push(false);
174 } else {
175 next_last.push(true);
176 }
177 stack.push((child_id, 0, level + 1, next_last));
178 }
179 }
180 }
181 Ok(())
182 }
183}
184
185#[cfg(test)]
186mod tests {
187 use crate::tree::{InsertBehavior, Node};
188 use super::*;
189 #[test]
190 fn map_tree() {
191 use super::map_tree;
192 let mut tree = Tree::new();
193 let root_id = tree.insert (Node::new (0u32), InsertBehavior::AsRoot).unwrap();
194 let child_id = tree.insert (Node::new (1), InsertBehavior::UnderNode (&root_id))
195 .unwrap();
196 tree.insert (Node::new (2), InsertBehavior::UnderNode (&root_id)).unwrap();
197 tree.insert (Node::new (3), InsertBehavior::UnderNode (&child_id)).unwrap();
198 tree.insert (Node::new (4), InsertBehavior::UnderNode (&child_id)).unwrap();
199 let mut s = String::new();
200 tree.write_formatted (&mut s).unwrap();
201 println!("old tree:\n{s}");
202 let new_tree : Tree <String> = map_tree (&tree, |i| (*i).to_string());
203 let mut s = String::new();
204 new_tree.write_formatted (&mut s).unwrap();
205 println!("new tree:\n{s}");
206 }
207}