gooey/
utils.rs

1//! Utility functions
2
3use std::collections::VecDeque;
4use crate::interface;
5use crate::Tree;
6use crate::tree::{self, NodeId};
7
8/// Utility functions to extend the Tree interface for common operations.
9pub 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  /// Splice the subtree of `other` rooted at `other_id` into `tree` under
16  /// `parent_id` with the given `order`
17  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  /// Write formatted tree representation showing only the names of the nodes
25  /// (as opposed to the default `Tree::write_formatted` method).
26  fn write_formatted_names <W : std::fmt::Write> (&self, w : &mut W)
27    -> std::fmt::Result;
28}
29
30/// Map a function over the nodes of a tree returning a new tree
31pub 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  /// Unwraps and gets the underlying `Element` reference.
61  ///
62  /// &#9888; Panics if `node_id` does not exist.
63  #[inline]
64  fn get_element (&self, node_id : &NodeId) -> &interface::Element {
65    self.get (node_id).unwrap().data()
66  }
67
68  /// Unwraps and gets the underlying `Element` reference.
69  ///
70  /// &#9888; Panics if `node_id` does not exist.
71  #[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  /// Unwraps and gets the parent of the referenced `NodeId`.
77  ///
78  /// &#9888; Panics if `child` does not exist or if `child` does not have a parent.
79  #[inline]
80  fn get_parent (&self, child : &NodeId) -> &interface::Element {
81    self.get_element (self.get_parent_id (child))
82  }
83
84  /// Unwraps and gets the parent of the referenced `NodeId`.
85  ///
86  /// &#9888; Panics if `child` does not exist or if `child` does not have a parent.
87  #[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  /// Unwraps and gets the parent ID of the referenced `NodeId`.
94  ///
95  /// &#9888; Panics if `child` does not exist or if `child` does not have a parent.
96  #[inline]
97  fn get_parent_id (&self, child : &NodeId) -> &NodeId {
98    self.get (child).unwrap().parent().unwrap()
99  }
100
101  /// Splice the subtree of `other` rooted at `other_id` into `tree` under `parent_id`
102  /// with the given `order`
103  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}