forked from servo/html5ever
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrc-consumer.rs
146 lines (119 loc) · 3.33 KB
/
rc-consumer.rs
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
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
#![feature(globs)]
use std::rc::{Rc, Weak};
use std::cell::RefCell;
use std::iter::Repeat;
use tree_builder::*;
mod tree_builder;
struct Node {
pub name: StrBuf,
pub parent: Option<WeakHandle>,
pub children: Vec<Handle>,
}
impl Node {
fn parent(&self) -> Handle {
self.parent.as_ref().expect("no parent!").upgrade()
}
}
#[unsafe_destructor]
impl Drop for Node {
fn drop(&mut self) {
println!("deleting {:s}", self.name);
}
}
#[deriving(Clone)]
struct Handle {
ptr: Rc<RefCell<Node>>,
}
impl Handle {
fn new(n: Node) -> Handle {
Handle {
ptr: Rc::new(RefCell::new(n)),
}
}
}
// Implement an object-identity Eq for use by position_elem().
impl Eq for Handle {
fn eq(&self, other: &Handle) -> bool {
// FIXME: This shouldn't really need to touch the borrow flags, right?
(&*self.ptr.borrow() as *Node) == (&*other.ptr.borrow() as *Node)
}
}
impl Deref<Rc<RefCell<Node>>> for Handle {
fn deref<'a>(&'a self) -> &'a Rc<RefCell<Node>> {
&self.ptr
}
}
// Don't implement DerefMut, because of mozilla/rust#12825
// We don't need it anyway, because RefCell gives interior
// mutability from an immutable ref.
struct WeakHandle {
ptr: Weak<RefCell<Node>>,
}
impl WeakHandle {
fn new(h: Handle) -> WeakHandle {
WeakHandle {
ptr: h.downgrade(),
}
}
fn upgrade(&self) -> Handle {
Handle {
ptr: self.ptr.upgrade().expect("dangling weak pointer!"),
}
}
}
struct Sink {
root: Option<Handle>,
}
impl TreeSink<Handle> for Sink {
fn create_element(&mut self, name: Atom) -> Handle {
println!("creating {:s}", name);
Handle::new(Node {
name: name,
children: vec!(),
parent: None,
})
}
fn create_html_element_set_as_root(&mut self) -> Handle {
let h = self.create_element(StrBuf::from_str("html"));
self.root = Some(h.clone());
h
}
fn detach_from_parent(&mut self, child_hdl: Handle) {
{
let child = child_hdl.borrow();
println!("detaching {:s}", child.name);
let parent = (*child).parent();
let mut parent = parent.borrow_mut();
let i = parent.children.as_slice().position_elem(&child_hdl).expect("not found!");
parent.children.remove(i).expect("not found!");
}
let mut child = child_hdl.borrow_mut();
(*child).parent = None;
}
fn append_element(&mut self, parent_hdl: Handle, child_hdl: Handle) {
let mut parent = parent_hdl.borrow_mut();
let mut child = child_hdl.borrow_mut();
println!("appending {:s} to {:s}", child.name, parent.name);
(*child).parent = Some(WeakHandle::new(parent_hdl.clone()));
parent.children.push(child_hdl.clone());
}
}
fn walk(node: Handle, depth: uint) {
let node = node.borrow();
let spaces: StrBuf = Repeat::new(' ').take(depth).collect();
println!("{:s}<{:s}>", spaces, node.name);
for c in node.children.iter() {
walk(c.clone(), depth+4);
}
println!("{:s}</{:s}>", spaces, node.name);
}
fn main() {
let mut s = Sink {
root: None,
};
{
let mut b = TreeBuilder::new(&mut s);
b.build();
}
walk(s.root.expect("no root!").clone(), 0);
}