OLD | NEW |
1 /* | 1 /* |
2 * Copyright (C) 2007 Apple Inc. All rights reserved. | 2 * Copyright (C) 2007 Apple Inc. All rights reserved. |
3 * | 3 * |
4 * Redistribution and use in source and binary forms, with or without | 4 * Redistribution and use in source and binary forms, with or without |
5 * modification, are permitted provided that the following conditions | 5 * modification, are permitted provided that the following conditions |
6 * are met: | 6 * are met: |
7 * | 7 * |
8 * 1. Redistributions of source code must retain the above copyright | 8 * 1. Redistributions of source code must retain the above copyright |
9 * notice, this list of conditions and the following disclaimer. | 9 * notice, this list of conditions and the following disclaimer. |
10 * 2. Redistributions in binary form must reproduce the above copyright | 10 * 2. Redistributions in binary form must reproduce the above copyright |
11 * notice, this list of conditions and the following disclaimer in the | 11 * notice, this list of conditions and the following disclaimer in the |
12 * documentation and/or other materials provided with the distribution. | 12 * documentation and/or other materials provided with the distribution. |
13 * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of | 13 * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of |
14 * its contributors may be used to endorse or promote products derived | 14 * its contributors may be used to endorse or promote products derived |
15 * from this software without specific prior written permission. | 15 * from this software without specific prior written permission. |
16 * | 16 * |
17 * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY | 17 * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY |
18 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | 18 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED |
19 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | 19 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE |
20 * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY | 20 * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY |
21 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | 21 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES |
22 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | 22 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; |
23 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | 23 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND |
24 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 24 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF | 25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF |
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
27 */ | 27 */ |
28 | |
29 /** | 28 /** |
30 * @constructor | 29 * @unrestricted |
31 * @extends {WebInspector.Object} | |
32 * @param {boolean=} nonFocusable | |
33 */ | 30 */ |
34 function TreeOutline(nonFocusable) | 31 var TreeOutline = class extends WebInspector.Object { |
35 { | 32 /** |
| 33 * @param {boolean=} nonFocusable |
| 34 */ |
| 35 constructor(nonFocusable) { |
| 36 super(); |
36 this._createRootElement(); | 37 this._createRootElement(); |
37 | 38 |
38 this.selectedTreeElement = null; | 39 this.selectedTreeElement = null; |
39 this.expandTreeElementsWhenArrowing = false; | 40 this.expandTreeElementsWhenArrowing = false; |
40 /** @type {?function(!TreeElement, !TreeElement):number} */ | 41 /** @type {?function(!TreeElement, !TreeElement):number} */ |
41 this._comparator = null; | 42 this._comparator = null; |
42 | 43 |
43 this.contentElement = this._rootElement._childrenListNode; | 44 this.contentElement = this._rootElement._childrenListNode; |
44 this.contentElement.addEventListener("keydown", this._treeKeyDown.bind(this)
, true); | 45 this.contentElement.addEventListener('keydown', this._treeKeyDown.bind(this)
, true); |
45 | 46 |
46 this.setFocusable(!nonFocusable); | 47 this.setFocusable(!nonFocusable); |
47 | 48 |
48 this.element = this.contentElement; | 49 this.element = this.contentElement; |
49 } | 50 } |
| 51 |
| 52 _createRootElement() { |
| 53 this._rootElement = new TreeElement(); |
| 54 this._rootElement.treeOutline = this; |
| 55 this._rootElement.root = true; |
| 56 this._rootElement.selectable = false; |
| 57 this._rootElement.expanded = true; |
| 58 this._rootElement._childrenListNode.classList.remove('children'); |
| 59 } |
| 60 |
| 61 /** |
| 62 * @return {!TreeElement} |
| 63 */ |
| 64 rootElement() { |
| 65 return this._rootElement; |
| 66 } |
| 67 |
| 68 /** |
| 69 * @return {?TreeElement} |
| 70 */ |
| 71 firstChild() { |
| 72 return this._rootElement.firstChild(); |
| 73 } |
| 74 |
| 75 /** |
| 76 * @param {!TreeElement} child |
| 77 */ |
| 78 appendChild(child) { |
| 79 this._rootElement.appendChild(child); |
| 80 } |
| 81 |
| 82 /** |
| 83 * @param {!TreeElement} child |
| 84 * @param {number} index |
| 85 */ |
| 86 insertChild(child, index) { |
| 87 this._rootElement.insertChild(child, index); |
| 88 } |
| 89 |
| 90 /** |
| 91 * @param {!TreeElement} child |
| 92 */ |
| 93 removeChild(child) { |
| 94 this._rootElement.removeChild(child); |
| 95 } |
| 96 |
| 97 removeChildren() { |
| 98 this._rootElement.removeChildren(); |
| 99 } |
| 100 |
| 101 /** |
| 102 * @param {number} x |
| 103 * @param {number} y |
| 104 * @return {?TreeElement} |
| 105 */ |
| 106 treeElementFromPoint(x, y) { |
| 107 var node = this.contentElement.ownerDocument.deepElementFromPoint(x, y); |
| 108 if (!node) |
| 109 return null; |
| 110 |
| 111 var listNode = node.enclosingNodeOrSelfWithNodeNameInArray(['ol', 'li']); |
| 112 if (listNode) |
| 113 return listNode.parentTreeElement || listNode.treeElement; |
| 114 return null; |
| 115 } |
| 116 |
| 117 /** |
| 118 * @param {?Event} event |
| 119 * @return {?TreeElement} |
| 120 */ |
| 121 treeElementFromEvent(event) { |
| 122 return event ? this.treeElementFromPoint(event.pageX, event.pageY) : null; |
| 123 } |
| 124 |
| 125 /** |
| 126 * @param {?function(!TreeElement, !TreeElement):number} comparator |
| 127 */ |
| 128 setComparator(comparator) { |
| 129 this._comparator = comparator; |
| 130 } |
| 131 |
| 132 /** |
| 133 * @param {boolean} focusable |
| 134 */ |
| 135 setFocusable(focusable) { |
| 136 if (focusable) |
| 137 this.contentElement.setAttribute('tabIndex', 0); |
| 138 else |
| 139 this.contentElement.removeAttribute('tabIndex'); |
| 140 } |
| 141 |
| 142 focus() { |
| 143 this.contentElement.focus(); |
| 144 } |
| 145 |
| 146 /** |
| 147 * @param {!TreeElement} element |
| 148 */ |
| 149 _bindTreeElement(element) { |
| 150 if (element.treeOutline) |
| 151 console.error('Binding element for the second time: ' + new Error().stack)
; |
| 152 element.treeOutline = this; |
| 153 element.onbind(); |
| 154 } |
| 155 |
| 156 /** |
| 157 * @param {!TreeElement} element |
| 158 */ |
| 159 _unbindTreeElement(element) { |
| 160 if (!element.treeOutline) |
| 161 console.error('Unbinding element that was not bound: ' + new Error().stack
); |
| 162 |
| 163 element.deselect(); |
| 164 element.onunbind(); |
| 165 element.treeOutline = null; |
| 166 } |
| 167 |
| 168 /** |
| 169 * @return {boolean} |
| 170 */ |
| 171 selectPrevious() { |
| 172 var nextSelectedElement = this.selectedTreeElement.traversePreviousTreeEleme
nt(true); |
| 173 while (nextSelectedElement && !nextSelectedElement.selectable) |
| 174 nextSelectedElement = nextSelectedElement.traversePreviousTreeElement(!thi
s.expandTreeElementsWhenArrowing); |
| 175 if (nextSelectedElement) { |
| 176 nextSelectedElement.reveal(); |
| 177 nextSelectedElement.select(false, true); |
| 178 return true; |
| 179 } |
| 180 return false; |
| 181 } |
| 182 |
| 183 /** |
| 184 * @return {boolean} |
| 185 */ |
| 186 selectNext() { |
| 187 var nextSelectedElement = this.selectedTreeElement.traverseNextTreeElement(t
rue); |
| 188 while (nextSelectedElement && !nextSelectedElement.selectable) |
| 189 nextSelectedElement = nextSelectedElement.traverseNextTreeElement(!this.ex
pandTreeElementsWhenArrowing); |
| 190 if (nextSelectedElement) { |
| 191 nextSelectedElement.reveal(); |
| 192 nextSelectedElement.select(false, true); |
| 193 return true; |
| 194 } |
| 195 return false; |
| 196 } |
| 197 |
| 198 /** |
| 199 * @param {!Event} event |
| 200 */ |
| 201 _treeKeyDown(event) { |
| 202 if (event.target !== this.contentElement) |
| 203 return; |
| 204 |
| 205 if (!this.selectedTreeElement || event.shiftKey || event.metaKey || event.ct
rlKey) |
| 206 return; |
| 207 |
| 208 var handled = false; |
| 209 var nextSelectedElement; |
| 210 if (event.key === 'ArrowUp' && !event.altKey) { |
| 211 handled = this.selectPrevious(); |
| 212 } else if (event.key === 'ArrowDown' && !event.altKey) { |
| 213 handled = this.selectNext(); |
| 214 } else if (event.key === 'ArrowLeft') { |
| 215 if (this.selectedTreeElement.expanded) { |
| 216 if (event.altKey) |
| 217 this.selectedTreeElement.collapseRecursively(); |
| 218 else |
| 219 this.selectedTreeElement.collapse(); |
| 220 handled = true; |
| 221 } else if (this.selectedTreeElement.parent && !this.selectedTreeElement.pa
rent.root) { |
| 222 handled = true; |
| 223 if (this.selectedTreeElement.parent.selectable) { |
| 224 nextSelectedElement = this.selectedTreeElement.parent; |
| 225 while (nextSelectedElement && !nextSelectedElement.selectable) |
| 226 nextSelectedElement = nextSelectedElement.parent; |
| 227 handled = nextSelectedElement ? true : false; |
| 228 } else if (this.selectedTreeElement.parent) |
| 229 this.selectedTreeElement.parent.collapse(); |
| 230 } |
| 231 } else if (event.key === 'ArrowRight') { |
| 232 if (!this.selectedTreeElement.revealed()) { |
| 233 this.selectedTreeElement.reveal(); |
| 234 handled = true; |
| 235 } else if (this.selectedTreeElement._expandable) { |
| 236 handled = true; |
| 237 if (this.selectedTreeElement.expanded) { |
| 238 nextSelectedElement = this.selectedTreeElement.firstChild(); |
| 239 while (nextSelectedElement && !nextSelectedElement.selectable) |
| 240 nextSelectedElement = nextSelectedElement.nextSibling; |
| 241 handled = nextSelectedElement ? true : false; |
| 242 } else { |
| 243 if (event.altKey) |
| 244 this.selectedTreeElement.expandRecursively(); |
| 245 else |
| 246 this.selectedTreeElement.expand(); |
| 247 } |
| 248 } |
| 249 } else if (event.keyCode === 8 /* Backspace */ || event.keyCode === 46 /* De
lete */) |
| 250 handled = this.selectedTreeElement.ondelete(); |
| 251 else if (isEnterKey(event)) |
| 252 handled = this.selectedTreeElement.onenter(); |
| 253 else if (event.keyCode === WebInspector.KeyboardShortcut.Keys.Space.code) |
| 254 handled = this.selectedTreeElement.onspace(); |
| 255 |
| 256 if (nextSelectedElement) { |
| 257 nextSelectedElement.reveal(); |
| 258 nextSelectedElement.select(false, true); |
| 259 } |
| 260 |
| 261 if (handled) |
| 262 event.consume(true); |
| 263 } |
| 264 |
| 265 /** |
| 266 * @param {!TreeElement} treeElement |
| 267 * @param {boolean} center |
| 268 */ |
| 269 _deferredScrollIntoView(treeElement, center) { |
| 270 if (!this._treeElementToScrollIntoView) |
| 271 this.element.window().requestAnimationFrame(deferredScrollIntoView.bind(th
is)); |
| 272 this._treeElementToScrollIntoView = treeElement; |
| 273 this._centerUponScrollIntoView = center; |
| 274 /** |
| 275 * @this {TreeOutline} |
| 276 */ |
| 277 function deferredScrollIntoView() { |
| 278 this._treeElementToScrollIntoView.listItemElement.scrollIntoViewIfNeeded(t
his._centerUponScrollIntoView); |
| 279 delete this._treeElementToScrollIntoView; |
| 280 delete this._centerUponScrollIntoView; |
| 281 } |
| 282 } |
| 283 }; |
50 | 284 |
51 /** @enum {symbol} */ | 285 /** @enum {symbol} */ |
52 TreeOutline.Events = { | 286 TreeOutline.Events = { |
53 ElementAttached: Symbol("ElementAttached"), | 287 ElementAttached: Symbol('ElementAttached'), |
54 ElementExpanded: Symbol("ElementExpanded"), | 288 ElementExpanded: Symbol('ElementExpanded'), |
55 ElementCollapsed: Symbol("ElementCollapsed"), | 289 ElementCollapsed: Symbol('ElementCollapsed'), |
56 ElementSelected: Symbol("ElementSelected") | 290 ElementSelected: Symbol('ElementSelected') |
57 }; | 291 }; |
58 | 292 |
59 TreeOutline.prototype = { | |
60 _createRootElement: function() | |
61 { | |
62 this._rootElement = new TreeElement(); | |
63 this._rootElement.treeOutline = this; | |
64 this._rootElement.root = true; | |
65 this._rootElement.selectable = false; | |
66 this._rootElement.expanded = true; | |
67 this._rootElement._childrenListNode.classList.remove("children"); | |
68 }, | |
69 | |
70 /** | |
71 * @return {!TreeElement} | |
72 */ | |
73 rootElement: function() | |
74 { | |
75 return this._rootElement; | |
76 }, | |
77 | |
78 /** | |
79 * @return {?TreeElement} | |
80 */ | |
81 firstChild: function() | |
82 { | |
83 return this._rootElement.firstChild(); | |
84 }, | |
85 | |
86 /** | |
87 * @param {!TreeElement} child | |
88 */ | |
89 appendChild: function(child) | |
90 { | |
91 this._rootElement.appendChild(child); | |
92 }, | |
93 | |
94 /** | |
95 * @param {!TreeElement} child | |
96 * @param {number} index | |
97 */ | |
98 insertChild: function(child, index) | |
99 { | |
100 this._rootElement.insertChild(child, index); | |
101 }, | |
102 | |
103 /** | |
104 * @param {!TreeElement} child | |
105 */ | |
106 removeChild: function(child) | |
107 { | |
108 this._rootElement.removeChild(child); | |
109 }, | |
110 | |
111 removeChildren: function() | |
112 { | |
113 this._rootElement.removeChildren(); | |
114 }, | |
115 | |
116 /** | |
117 * @param {number} x | |
118 * @param {number} y | |
119 * @return {?TreeElement} | |
120 */ | |
121 treeElementFromPoint: function(x, y) | |
122 { | |
123 var node = this.contentElement.ownerDocument.deepElementFromPoint(x, y); | |
124 if (!node) | |
125 return null; | |
126 | |
127 var listNode = node.enclosingNodeOrSelfWithNodeNameInArray(["ol", "li"])
; | |
128 if (listNode) | |
129 return listNode.parentTreeElement || listNode.treeElement; | |
130 return null; | |
131 }, | |
132 | |
133 /** | |
134 * @param {?Event} event | |
135 * @return {?TreeElement} | |
136 */ | |
137 treeElementFromEvent: function(event) | |
138 { | |
139 return event ? this.treeElementFromPoint(event.pageX, event.pageY) : nul
l; | |
140 }, | |
141 | |
142 /** | |
143 * @param {?function(!TreeElement, !TreeElement):number} comparator | |
144 */ | |
145 setComparator: function(comparator) | |
146 { | |
147 this._comparator = comparator; | |
148 }, | |
149 | |
150 /** | |
151 * @param {boolean} focusable | |
152 */ | |
153 setFocusable: function(focusable) | |
154 { | |
155 if (focusable) | |
156 this.contentElement.setAttribute("tabIndex", 0); | |
157 else | |
158 this.contentElement.removeAttribute("tabIndex"); | |
159 }, | |
160 | |
161 focus: function() | |
162 { | |
163 this.contentElement.focus(); | |
164 }, | |
165 | |
166 /** | |
167 * @param {!TreeElement} element | |
168 */ | |
169 _bindTreeElement: function(element) | |
170 { | |
171 if (element.treeOutline) | |
172 console.error("Binding element for the second time: " + new Error().
stack); | |
173 element.treeOutline = this; | |
174 element.onbind(); | |
175 }, | |
176 | |
177 /** | |
178 * @param {!TreeElement} element | |
179 */ | |
180 _unbindTreeElement: function(element) | |
181 { | |
182 if (!element.treeOutline) | |
183 console.error("Unbinding element that was not bound: " + new Error()
.stack); | |
184 | |
185 element.deselect(); | |
186 element.onunbind(); | |
187 element.treeOutline = null; | |
188 }, | |
189 | |
190 /** | |
191 * @return {boolean} | |
192 */ | |
193 selectPrevious: function() | |
194 { | |
195 var nextSelectedElement = this.selectedTreeElement.traversePreviousTreeE
lement(true); | |
196 while (nextSelectedElement && !nextSelectedElement.selectable) | |
197 nextSelectedElement = nextSelectedElement.traversePreviousTreeElemen
t(!this.expandTreeElementsWhenArrowing); | |
198 if (nextSelectedElement) { | |
199 nextSelectedElement.reveal(); | |
200 nextSelectedElement.select(false, true); | |
201 return true; | |
202 } | |
203 return false; | |
204 }, | |
205 | |
206 /** | |
207 * @return {boolean} | |
208 */ | |
209 selectNext: function() | |
210 { | |
211 var nextSelectedElement = this.selectedTreeElement.traverseNextTreeEleme
nt(true); | |
212 while (nextSelectedElement && !nextSelectedElement.selectable) | |
213 nextSelectedElement = nextSelectedElement.traverseNextTreeElement(!t
his.expandTreeElementsWhenArrowing); | |
214 if (nextSelectedElement) { | |
215 nextSelectedElement.reveal(); | |
216 nextSelectedElement.select(false, true); | |
217 return true; | |
218 } | |
219 return false; | |
220 }, | |
221 | |
222 /** | |
223 * @param {!Event} event | |
224 */ | |
225 _treeKeyDown: function(event) | |
226 { | |
227 if (event.target !== this.contentElement) | |
228 return; | |
229 | |
230 if (!this.selectedTreeElement || event.shiftKey || event.metaKey || even
t.ctrlKey) | |
231 return; | |
232 | |
233 var handled = false; | |
234 var nextSelectedElement; | |
235 if (event.key === "ArrowUp" && !event.altKey) { | |
236 handled = this.selectPrevious(); | |
237 } else if (event.key === "ArrowDown" && !event.altKey) { | |
238 handled = this.selectNext(); | |
239 } else if (event.key === "ArrowLeft") { | |
240 if (this.selectedTreeElement.expanded) { | |
241 if (event.altKey) | |
242 this.selectedTreeElement.collapseRecursively(); | |
243 else | |
244 this.selectedTreeElement.collapse(); | |
245 handled = true; | |
246 } else if (this.selectedTreeElement.parent && !this.selectedTreeElem
ent.parent.root) { | |
247 handled = true; | |
248 if (this.selectedTreeElement.parent.selectable) { | |
249 nextSelectedElement = this.selectedTreeElement.parent; | |
250 while (nextSelectedElement && !nextSelectedElement.selectabl
e) | |
251 nextSelectedElement = nextSelectedElement.parent; | |
252 handled = nextSelectedElement ? true : false; | |
253 } else if (this.selectedTreeElement.parent) | |
254 this.selectedTreeElement.parent.collapse(); | |
255 } | |
256 } else if (event.key === "ArrowRight") { | |
257 if (!this.selectedTreeElement.revealed()) { | |
258 this.selectedTreeElement.reveal(); | |
259 handled = true; | |
260 } else if (this.selectedTreeElement._expandable) { | |
261 handled = true; | |
262 if (this.selectedTreeElement.expanded) { | |
263 nextSelectedElement = this.selectedTreeElement.firstChild(); | |
264 while (nextSelectedElement && !nextSelectedElement.selectabl
e) | |
265 nextSelectedElement = nextSelectedElement.nextSibling; | |
266 handled = nextSelectedElement ? true : false; | |
267 } else { | |
268 if (event.altKey) | |
269 this.selectedTreeElement.expandRecursively(); | |
270 else | |
271 this.selectedTreeElement.expand(); | |
272 } | |
273 } | |
274 } else if (event.keyCode === 8 /* Backspace */ || event.keyCode === 46 /
* Delete */) | |
275 handled = this.selectedTreeElement.ondelete(); | |
276 else if (isEnterKey(event)) | |
277 handled = this.selectedTreeElement.onenter(); | |
278 else if (event.keyCode === WebInspector.KeyboardShortcut.Keys.Space.code
) | |
279 handled = this.selectedTreeElement.onspace(); | |
280 | |
281 if (nextSelectedElement) { | |
282 nextSelectedElement.reveal(); | |
283 nextSelectedElement.select(false, true); | |
284 } | |
285 | |
286 if (handled) | |
287 event.consume(true); | |
288 }, | |
289 | |
290 /** | |
291 * @param {!TreeElement} treeElement | |
292 * @param {boolean} center | |
293 */ | |
294 _deferredScrollIntoView: function(treeElement, center) | |
295 { | |
296 if (!this._treeElementToScrollIntoView) | |
297 this.element.window().requestAnimationFrame(deferredScrollIntoView.b
ind(this)); | |
298 this._treeElementToScrollIntoView = treeElement; | |
299 this._centerUponScrollIntoView = center; | |
300 /** | |
301 * @this {TreeOutline} | |
302 */ | |
303 function deferredScrollIntoView() | |
304 { | |
305 this._treeElementToScrollIntoView.listItemElement.scrollIntoViewIfNe
eded(this._centerUponScrollIntoView); | |
306 delete this._treeElementToScrollIntoView; | |
307 delete this._centerUponScrollIntoView; | |
308 } | |
309 }, | |
310 | |
311 __proto__: WebInspector.Object.prototype | |
312 }; | |
313 | |
314 /** | 293 /** |
315 * @constructor | 294 * @unrestricted |
316 * @extends {TreeOutline} | |
317 */ | 295 */ |
318 function TreeOutlineInShadow() | 296 var TreeOutlineInShadow = class extends TreeOutline { |
319 { | 297 constructor() { |
320 TreeOutline.call(this); | 298 super(); |
321 this.contentElement.classList.add("tree-outline"); | 299 this.contentElement.classList.add('tree-outline'); |
322 | 300 |
323 // Redefine element to the external one. | 301 // Redefine element to the external one. |
324 this.element = createElement("div"); | 302 this.element = createElement('div'); |
325 this._shadowRoot = WebInspector.createShadowRootWithCoreStyles(this.element,
"ui/treeoutline.css"); | 303 this._shadowRoot = WebInspector.createShadowRootWithCoreStyles(this.element,
'ui/treeoutline.css'); |
326 this._disclosureElement = this._shadowRoot.createChild("div", "tree-outline-
disclosure"); | 304 this._disclosureElement = this._shadowRoot.createChild('div', 'tree-outline-
disclosure'); |
327 this._disclosureElement.appendChild(this.contentElement); | 305 this._disclosureElement.appendChild(this.contentElement); |
328 this._renderSelection = true; | 306 this._renderSelection = true; |
329 } | 307 } |
330 | 308 |
331 TreeOutlineInShadow.prototype = { | 309 /** |
332 /** | 310 * @param {string} cssFile |
333 * @param {string} cssFile | 311 */ |
334 */ | 312 registerRequiredCSS(cssFile) { |
335 registerRequiredCSS: function(cssFile) | 313 WebInspector.appendStyle(this._shadowRoot, cssFile); |
336 { | 314 } |
337 WebInspector.appendStyle(this._shadowRoot, cssFile); | 315 |
338 }, | 316 hideOverflow() { |
339 | 317 this._disclosureElement.classList.add('tree-outline-disclosure-hide-overflow
'); |
340 hideOverflow: function() | 318 } |
341 { | 319 |
342 this._disclosureElement.classList.add("tree-outline-disclosure-hide-over
flow"); | 320 makeDense() { |
343 }, | 321 this.contentElement.classList.add('tree-outline-dense'); |
344 | 322 } |
345 makeDense: function() | |
346 { | |
347 this.contentElement.classList.add("tree-outline-dense"); | |
348 }, | |
349 | |
350 __proto__: TreeOutline.prototype | |
351 }; | 323 }; |
352 | 324 |
353 /** | 325 /** |
354 * @constructor | 326 * @unrestricted |
355 * @param {(string|!Node)=} title | |
356 * @param {boolean=} expandable | |
357 */ | 327 */ |
358 function TreeElement(title, expandable) | 328 var TreeElement = class { |
359 { | 329 /** |
| 330 * @param {(string|!Node)=} title |
| 331 * @param {boolean=} expandable |
| 332 */ |
| 333 constructor(title, expandable) { |
360 /** @type {?TreeOutline} */ | 334 /** @type {?TreeOutline} */ |
361 this.treeOutline = null; | 335 this.treeOutline = null; |
362 this.parent = null; | 336 this.parent = null; |
363 this.previousSibling = null; | 337 this.previousSibling = null; |
364 this.nextSibling = null; | 338 this.nextSibling = null; |
365 | 339 |
366 this._listItemNode = createElement("li"); | 340 this._listItemNode = createElement('li'); |
367 this._listItemNode.treeElement = this; | 341 this._listItemNode.treeElement = this; |
368 if (title) | 342 if (title) |
369 this.title = title; | 343 this.title = title; |
370 this._listItemNode.addEventListener("mousedown", this._handleMouseDown.bind(
this), false); | 344 this._listItemNode.addEventListener('mousedown', this._handleMouseDown.bind(
this), false); |
371 this._listItemNode.addEventListener("click", this._treeElementToggled.bind(t
his), false); | 345 this._listItemNode.addEventListener('click', this._treeElementToggled.bind(t
his), false); |
372 this._listItemNode.addEventListener("dblclick", this._handleDoubleClick.bind
(this), false); | 346 this._listItemNode.addEventListener('dblclick', this._handleDoubleClick.bind
(this), false); |
373 | 347 |
374 this._childrenListNode = createElement("ol"); | 348 this._childrenListNode = createElement('ol'); |
375 this._childrenListNode.parentTreeElement = this; | 349 this._childrenListNode.parentTreeElement = this; |
376 this._childrenListNode.classList.add("children"); | 350 this._childrenListNode.classList.add('children'); |
377 | 351 |
378 this._hidden = false; | 352 this._hidden = false; |
379 this._selectable = true; | 353 this._selectable = true; |
380 this.expanded = false; | 354 this.expanded = false; |
381 this.selected = false; | 355 this.selected = false; |
382 this.setExpandable(expandable || false); | 356 this.setExpandable(expandable || false); |
383 this._collapsible = true; | 357 this._collapsible = true; |
384 } | 358 } |
| 359 |
| 360 /** |
| 361 * @param {?TreeElement} ancestor |
| 362 * @return {boolean} |
| 363 */ |
| 364 hasAncestor(ancestor) { |
| 365 if (!ancestor) |
| 366 return false; |
| 367 |
| 368 var currentNode = this.parent; |
| 369 while (currentNode) { |
| 370 if (ancestor === currentNode) |
| 371 return true; |
| 372 currentNode = currentNode.parent; |
| 373 } |
| 374 |
| 375 return false; |
| 376 } |
| 377 |
| 378 /** |
| 379 * @param {?TreeElement} ancestor |
| 380 * @return {boolean} |
| 381 */ |
| 382 hasAncestorOrSelf(ancestor) { |
| 383 return this === ancestor || this.hasAncestor(ancestor); |
| 384 } |
| 385 |
| 386 /** |
| 387 * @return {!Array.<!TreeElement>} |
| 388 */ |
| 389 children() { |
| 390 return this._children || []; |
| 391 } |
| 392 |
| 393 /** |
| 394 * @return {number} |
| 395 */ |
| 396 childCount() { |
| 397 return this._children ? this._children.length : 0; |
| 398 } |
| 399 |
| 400 /** |
| 401 * @return {?TreeElement} |
| 402 */ |
| 403 firstChild() { |
| 404 return this._children ? this._children[0] : null; |
| 405 } |
| 406 |
| 407 /** |
| 408 * @return {?TreeElement} |
| 409 */ |
| 410 lastChild() { |
| 411 return this._children ? this._children[this._children.length - 1] : null; |
| 412 } |
| 413 |
| 414 /** |
| 415 * @param {number} index |
| 416 * @return {?TreeElement} |
| 417 */ |
| 418 childAt(index) { |
| 419 return this._children ? this._children[index] : null; |
| 420 } |
| 421 |
| 422 /** |
| 423 * @param {!TreeElement} child |
| 424 * @return {number} |
| 425 */ |
| 426 indexOfChild(child) { |
| 427 return this._children ? this._children.indexOf(child) : -1; |
| 428 } |
| 429 |
| 430 /** |
| 431 * @param {!TreeElement} child |
| 432 */ |
| 433 appendChild(child) { |
| 434 if (!this._children) |
| 435 this._children = []; |
| 436 |
| 437 var insertionIndex; |
| 438 if (this.treeOutline && this.treeOutline._comparator) |
| 439 insertionIndex = this._children.lowerBound(child, this.treeOutline._compar
ator); |
| 440 else |
| 441 insertionIndex = this._children.length; |
| 442 this.insertChild(child, insertionIndex); |
| 443 } |
| 444 |
| 445 /** |
| 446 * @param {!TreeElement} child |
| 447 * @param {number} index |
| 448 */ |
| 449 insertChild(child, index) { |
| 450 if (!this._children) |
| 451 this._children = []; |
| 452 |
| 453 if (!child) |
| 454 throw 'child can\'t be undefined or null'; |
| 455 |
| 456 console.assert( |
| 457 !child.parent, 'Attempting to insert a child that is already in the tree
, reparenting is not supported.'); |
| 458 |
| 459 var previousChild = (index > 0 ? this._children[index - 1] : null); |
| 460 if (previousChild) { |
| 461 previousChild.nextSibling = child; |
| 462 child.previousSibling = previousChild; |
| 463 } else { |
| 464 child.previousSibling = null; |
| 465 } |
| 466 |
| 467 var nextChild = this._children[index]; |
| 468 if (nextChild) { |
| 469 nextChild.previousSibling = child; |
| 470 child.nextSibling = nextChild; |
| 471 } else { |
| 472 child.nextSibling = null; |
| 473 } |
| 474 |
| 475 this._children.splice(index, 0, child); |
| 476 |
| 477 this.setExpandable(true); |
| 478 child.parent = this; |
| 479 |
| 480 if (this.treeOutline) |
| 481 this.treeOutline._bindTreeElement(child); |
| 482 for (var current = child.firstChild(); this.treeOutline && current; |
| 483 current = current.traverseNextTreeElement(false, child, true)) |
| 484 this.treeOutline._bindTreeElement(current); |
| 485 child.onattach(); |
| 486 child._ensureSelection(); |
| 487 if (this.treeOutline) |
| 488 this.treeOutline.dispatchEventToListeners(TreeOutline.Events.ElementAttach
ed, child); |
| 489 var nextSibling = child.nextSibling ? child.nextSibling._listItemNode : null
; |
| 490 this._childrenListNode.insertBefore(child._listItemNode, nextSibling); |
| 491 this._childrenListNode.insertBefore(child._childrenListNode, nextSibling); |
| 492 if (child.selected) |
| 493 child.select(); |
| 494 if (child.expanded) |
| 495 child.expand(); |
| 496 } |
| 497 |
| 498 /** |
| 499 * @param {number} childIndex |
| 500 */ |
| 501 removeChildAtIndex(childIndex) { |
| 502 if (childIndex < 0 || childIndex >= this._children.length) |
| 503 throw 'childIndex out of range'; |
| 504 |
| 505 var child = this._children[childIndex]; |
| 506 this._children.splice(childIndex, 1); |
| 507 |
| 508 var parent = child.parent; |
| 509 if (this.treeOutline && this.treeOutline.selectedTreeElement && |
| 510 this.treeOutline.selectedTreeElement.hasAncestorOrSelf(child)) { |
| 511 if (child.nextSibling) |
| 512 child.nextSibling.select(true); |
| 513 else if (child.previousSibling) |
| 514 child.previousSibling.select(true); |
| 515 else if (parent) |
| 516 parent.select(true); |
| 517 } |
| 518 |
| 519 if (child.previousSibling) |
| 520 child.previousSibling.nextSibling = child.nextSibling; |
| 521 if (child.nextSibling) |
| 522 child.nextSibling.previousSibling = child.previousSibling; |
| 523 child.parent = null; |
| 524 |
| 525 if (this.treeOutline) |
| 526 this.treeOutline._unbindTreeElement(child); |
| 527 for (var current = child.firstChild(); this.treeOutline && current; |
| 528 current = current.traverseNextTreeElement(false, child, true)) |
| 529 this.treeOutline._unbindTreeElement(current); |
| 530 |
| 531 child._detach(); |
| 532 } |
| 533 |
| 534 /** |
| 535 * @param {!TreeElement} child |
| 536 */ |
| 537 removeChild(child) { |
| 538 if (!child) |
| 539 throw 'child can\'t be undefined or null'; |
| 540 if (child.parent !== this) |
| 541 return; |
| 542 |
| 543 var childIndex = this._children.indexOf(child); |
| 544 if (childIndex === -1) |
| 545 throw 'child not found in this node\'s children'; |
| 546 |
| 547 this.removeChildAtIndex(childIndex); |
| 548 } |
| 549 |
| 550 removeChildren() { |
| 551 if (!this.root && this.treeOutline && this.treeOutline.selectedTreeElement &
& |
| 552 this.treeOutline.selectedTreeElement.hasAncestorOrSelf(this)) |
| 553 this.select(true); |
| 554 |
| 555 for (var i = 0; this._children && i < this._children.length; ++i) { |
| 556 var child = this._children[i]; |
| 557 child.previousSibling = null; |
| 558 child.nextSibling = null; |
| 559 child.parent = null; |
| 560 |
| 561 if (this.treeOutline) |
| 562 this.treeOutline._unbindTreeElement(child); |
| 563 for (var current = child.firstChild(); this.treeOutline && current; |
| 564 current = current.traverseNextTreeElement(false, child, true)) |
| 565 this.treeOutline._unbindTreeElement(current); |
| 566 child._detach(); |
| 567 } |
| 568 this._children = []; |
| 569 } |
| 570 |
| 571 get selectable() { |
| 572 if (this._hidden) |
| 573 return false; |
| 574 return this._selectable; |
| 575 } |
| 576 |
| 577 set selectable(x) { |
| 578 this._selectable = x; |
| 579 } |
| 580 |
| 581 get listItemElement() { |
| 582 return this._listItemNode; |
| 583 } |
| 584 |
| 585 get childrenListElement() { |
| 586 return this._childrenListNode; |
| 587 } |
| 588 |
| 589 /** |
| 590 * @return {string|!Node} |
| 591 */ |
| 592 get title() { |
| 593 return this._title; |
| 594 } |
| 595 |
| 596 /** |
| 597 * @param {string|!Node} x |
| 598 */ |
| 599 set title(x) { |
| 600 if (this._title === x) |
| 601 return; |
| 602 this._title = x; |
| 603 |
| 604 if (typeof x === 'string') { |
| 605 this._titleElement = createElementWithClass('span', 'tree-element-title'); |
| 606 this._titleElement.textContent = x; |
| 607 this.tooltip = x; |
| 608 } else { |
| 609 this._titleElement = x; |
| 610 this.tooltip = ''; |
| 611 } |
| 612 |
| 613 this._listItemNode.removeChildren(); |
| 614 if (this._iconElement) |
| 615 this._listItemNode.appendChild(this._iconElement); |
| 616 |
| 617 this._listItemNode.appendChild(this._titleElement); |
| 618 this._ensureSelection(); |
| 619 } |
| 620 |
| 621 /** |
| 622 * @return {string} |
| 623 */ |
| 624 titleAsText() { |
| 625 if (!this._title) |
| 626 return ''; |
| 627 if (typeof this._title === 'string') |
| 628 return this._title; |
| 629 return this._title.textContent; |
| 630 } |
| 631 |
| 632 /** |
| 633 * @param {!WebInspector.InplaceEditor.Config} editingConfig |
| 634 */ |
| 635 startEditingTitle(editingConfig) { |
| 636 WebInspector.InplaceEditor.startEditing(this._titleElement, editingConfig); |
| 637 this.treeOutline._shadowRoot.getSelection().setBaseAndExtent(this._titleElem
ent, 0, this._titleElement, 1); |
| 638 } |
| 639 |
| 640 createIcon() { |
| 641 if (!this._iconElement) { |
| 642 this._iconElement = createElementWithClass('div', 'icon'); |
| 643 this._listItemNode.insertBefore(this._iconElement, this._listItemNode.firs
tChild); |
| 644 this._ensureSelection(); |
| 645 } |
| 646 } |
| 647 |
| 648 /** |
| 649 * @return {string} |
| 650 */ |
| 651 get tooltip() { |
| 652 return this._tooltip || ''; |
| 653 } |
| 654 |
| 655 /** |
| 656 * @param {string} x |
| 657 */ |
| 658 set tooltip(x) { |
| 659 if (this._tooltip === x) |
| 660 return; |
| 661 this._tooltip = x; |
| 662 this._listItemNode.title = x; |
| 663 } |
| 664 |
| 665 /** |
| 666 * @return {boolean} |
| 667 */ |
| 668 isExpandable() { |
| 669 return this._expandable; |
| 670 } |
| 671 |
| 672 /** |
| 673 * @param {boolean} expandable |
| 674 */ |
| 675 setExpandable(expandable) { |
| 676 if (this._expandable === expandable) |
| 677 return; |
| 678 |
| 679 this._expandable = expandable; |
| 680 |
| 681 this._listItemNode.classList.toggle('parent', expandable); |
| 682 if (!expandable) |
| 683 this.collapse(); |
| 684 } |
| 685 |
| 686 /** |
| 687 * @param {boolean} collapsible |
| 688 */ |
| 689 setCollapsible(collapsible) { |
| 690 if (this._collapsible === collapsible) |
| 691 return; |
| 692 |
| 693 this._collapsible = collapsible; |
| 694 |
| 695 this._listItemNode.classList.toggle('always-parent', !collapsible); |
| 696 if (!collapsible) |
| 697 this.expand(); |
| 698 } |
| 699 |
| 700 get hidden() { |
| 701 return this._hidden; |
| 702 } |
| 703 |
| 704 set hidden(x) { |
| 705 if (this._hidden === x) |
| 706 return; |
| 707 |
| 708 this._hidden = x; |
| 709 |
| 710 this._listItemNode.classList.toggle('hidden', x); |
| 711 this._childrenListNode.classList.toggle('hidden', x); |
| 712 } |
| 713 |
| 714 invalidateChildren() { |
| 715 if (this._children) { |
| 716 this.removeChildren(); |
| 717 this._children = null; |
| 718 } |
| 719 } |
| 720 |
| 721 _ensureSelection() { |
| 722 if (!this.treeOutline || !this.treeOutline._renderSelection) |
| 723 return; |
| 724 if (!this._selectionElement) |
| 725 this._selectionElement = createElementWithClass('div', 'selection fill'); |
| 726 this._listItemNode.insertBefore(this._selectionElement, this.listItemElement
.firstChild); |
| 727 } |
| 728 |
| 729 /** |
| 730 * @param {!Event} event |
| 731 */ |
| 732 _treeElementToggled(event) { |
| 733 var element = event.currentTarget; |
| 734 if (element.treeElement !== this || element.hasSelection()) |
| 735 return; |
| 736 |
| 737 var toggleOnClick = this.toggleOnClick && !this.selectable; |
| 738 var isInTriangle = this.isEventWithinDisclosureTriangle(event); |
| 739 if (!toggleOnClick && !isInTriangle) |
| 740 return; |
| 741 |
| 742 if (event.target && event.target.enclosingNodeOrSelfWithNodeName('a')) |
| 743 return; |
| 744 |
| 745 if (this.expanded) { |
| 746 if (event.altKey) |
| 747 this.collapseRecursively(); |
| 748 else |
| 749 this.collapse(); |
| 750 } else { |
| 751 if (event.altKey) |
| 752 this.expandRecursively(); |
| 753 else |
| 754 this.expand(); |
| 755 } |
| 756 event.consume(); |
| 757 } |
| 758 |
| 759 /** |
| 760 * @param {!Event} event |
| 761 */ |
| 762 _handleMouseDown(event) { |
| 763 var element = event.currentTarget; |
| 764 if (!element) |
| 765 return; |
| 766 if (!this.selectable) |
| 767 return; |
| 768 if (element.treeElement !== this) |
| 769 return; |
| 770 |
| 771 if (this.isEventWithinDisclosureTriangle(event)) |
| 772 return; |
| 773 |
| 774 this.selectOnMouseDown(event); |
| 775 } |
| 776 |
| 777 /** |
| 778 * @param {!Event} event |
| 779 */ |
| 780 _handleDoubleClick(event) { |
| 781 var element = event.currentTarget; |
| 782 if (!element || element.treeElement !== this) |
| 783 return; |
| 784 |
| 785 var handled = this.ondblclick(event); |
| 786 if (handled) |
| 787 return; |
| 788 if (this._expandable && !this.expanded) |
| 789 this.expand(); |
| 790 } |
| 791 |
| 792 _detach() { |
| 793 this._listItemNode.remove(); |
| 794 this._childrenListNode.remove(); |
| 795 } |
| 796 |
| 797 collapse() { |
| 798 if (!this.expanded || !this._collapsible) |
| 799 return; |
| 800 this._listItemNode.classList.remove('expanded'); |
| 801 this._childrenListNode.classList.remove('expanded'); |
| 802 this.expanded = false; |
| 803 this.oncollapse(); |
| 804 if (this.treeOutline) |
| 805 this.treeOutline.dispatchEventToListeners(TreeOutline.Events.ElementCollap
sed, this); |
| 806 } |
| 807 |
| 808 collapseRecursively() { |
| 809 var item = this; |
| 810 while (item) { |
| 811 if (item.expanded) |
| 812 item.collapse(); |
| 813 item = item.traverseNextTreeElement(false, this, true); |
| 814 } |
| 815 } |
| 816 |
| 817 expand() { |
| 818 if (!this._expandable || (this.expanded && this._children)) |
| 819 return; |
| 820 |
| 821 // Set this before onpopulate. Since onpopulate can add elements, this makes |
| 822 // sure the expanded flag is true before calling those functions. This preve
nts the possibility |
| 823 // of an infinite loop if onpopulate were to call expand. |
| 824 |
| 825 this.expanded = true; |
| 826 |
| 827 this._populateIfNeeded(); |
| 828 this._listItemNode.classList.add('expanded'); |
| 829 this._childrenListNode.classList.add('expanded'); |
| 830 |
| 831 if (this.treeOutline) { |
| 832 this.onexpand(); |
| 833 this.treeOutline.dispatchEventToListeners(TreeOutline.Events.ElementExpand
ed, this); |
| 834 } |
| 835 } |
| 836 |
| 837 /** |
| 838 * @param {number=} maxDepth |
| 839 */ |
| 840 expandRecursively(maxDepth) { |
| 841 var item = this; |
| 842 var info = {}; |
| 843 var depth = 0; |
| 844 |
| 845 // The Inspector uses TreeOutlines to represents object properties, so recur
sive expansion |
| 846 // in some case can be infinite, since JavaScript objects can hold circular
references. |
| 847 // So default to a recursion cap of 3 levels, since that gives fairly good r
esults. |
| 848 if (isNaN(maxDepth)) |
| 849 maxDepth = 3; |
| 850 |
| 851 while (item) { |
| 852 if (depth < maxDepth) |
| 853 item.expand(); |
| 854 item = item.traverseNextTreeElement(false, this, (depth >= maxDepth), info
); |
| 855 depth += info.depthChange; |
| 856 } |
| 857 } |
| 858 |
| 859 /** |
| 860 * @param {boolean=} center |
| 861 */ |
| 862 reveal(center) { |
| 863 var currentAncestor = this.parent; |
| 864 while (currentAncestor && !currentAncestor.root) { |
| 865 if (!currentAncestor.expanded) |
| 866 currentAncestor.expand(); |
| 867 currentAncestor = currentAncestor.parent; |
| 868 } |
| 869 |
| 870 this.treeOutline._deferredScrollIntoView(this, !!center); |
| 871 } |
| 872 |
| 873 /** |
| 874 * @return {boolean} |
| 875 */ |
| 876 revealed() { |
| 877 var currentAncestor = this.parent; |
| 878 while (currentAncestor && !currentAncestor.root) { |
| 879 if (!currentAncestor.expanded) |
| 880 return false; |
| 881 currentAncestor = currentAncestor.parent; |
| 882 } |
| 883 |
| 884 return true; |
| 885 } |
| 886 |
| 887 selectOnMouseDown(event) { |
| 888 if (this.select(false, true)) |
| 889 event.consume(true); |
| 890 } |
| 891 |
| 892 /** |
| 893 * @param {boolean=} omitFocus |
| 894 * @param {boolean=} selectedByUser |
| 895 * @return {boolean} |
| 896 */ |
| 897 select(omitFocus, selectedByUser) { |
| 898 if (!this.treeOutline || !this.selectable || this.selected) |
| 899 return false; |
| 900 |
| 901 if (this.treeOutline.selectedTreeElement) |
| 902 this.treeOutline.selectedTreeElement.deselect(); |
| 903 this.treeOutline.selectedTreeElement = null; |
| 904 |
| 905 if (this.treeOutline._rootElement === this) |
| 906 return false; |
| 907 |
| 908 this.selected = true; |
| 909 |
| 910 if (!omitFocus) |
| 911 this.treeOutline.focus(); |
| 912 |
| 913 // Focusing on another node may detach "this" from tree. |
| 914 if (!this.treeOutline) |
| 915 return false; |
| 916 this.treeOutline.selectedTreeElement = this; |
| 917 this._listItemNode.classList.add('selected'); |
| 918 this.treeOutline.dispatchEventToListeners(TreeOutline.Events.ElementSelected
, this); |
| 919 return this.onselect(selectedByUser); |
| 920 } |
| 921 |
| 922 /** |
| 923 * @param {boolean=} omitFocus |
| 924 */ |
| 925 revealAndSelect(omitFocus) { |
| 926 this.reveal(true); |
| 927 this.select(omitFocus); |
| 928 } |
| 929 |
| 930 /** |
| 931 * @param {boolean=} supressOnDeselect |
| 932 */ |
| 933 deselect(supressOnDeselect) { |
| 934 if (!this.treeOutline || this.treeOutline.selectedTreeElement !== this || !t
his.selected) |
| 935 return; |
| 936 |
| 937 this.selected = false; |
| 938 this.treeOutline.selectedTreeElement = null; |
| 939 this._listItemNode.classList.remove('selected'); |
| 940 } |
| 941 |
| 942 _populateIfNeeded() { |
| 943 if (this.treeOutline && this._expandable && !this._children) { |
| 944 this._children = []; |
| 945 this.onpopulate(); |
| 946 } |
| 947 } |
| 948 |
| 949 onpopulate() { |
| 950 // Overridden by subclasses. |
| 951 } |
| 952 |
| 953 /** |
| 954 * @return {boolean} |
| 955 */ |
| 956 onenter() { |
| 957 return false; |
| 958 } |
| 959 |
| 960 /** |
| 961 * @return {boolean} |
| 962 */ |
| 963 ondelete() { |
| 964 return false; |
| 965 } |
| 966 |
| 967 /** |
| 968 * @return {boolean} |
| 969 */ |
| 970 onspace() { |
| 971 return false; |
| 972 } |
| 973 |
| 974 onbind() { |
| 975 } |
| 976 |
| 977 onunbind() { |
| 978 } |
| 979 |
| 980 onattach() { |
| 981 } |
| 982 |
| 983 onexpand() { |
| 984 } |
| 985 |
| 986 oncollapse() { |
| 987 } |
| 988 |
| 989 /** |
| 990 * @param {!Event} e |
| 991 * @return {boolean} |
| 992 */ |
| 993 ondblclick(e) { |
| 994 return false; |
| 995 } |
| 996 |
| 997 /** |
| 998 * @param {boolean=} selectedByUser |
| 999 * @return {boolean} |
| 1000 */ |
| 1001 onselect(selectedByUser) { |
| 1002 return false; |
| 1003 } |
| 1004 |
| 1005 /** |
| 1006 * @param {boolean} skipUnrevealed |
| 1007 * @param {?TreeElement=} stayWithin |
| 1008 * @param {boolean=} dontPopulate |
| 1009 * @param {!Object=} info |
| 1010 * @return {?TreeElement} |
| 1011 */ |
| 1012 traverseNextTreeElement(skipUnrevealed, stayWithin, dontPopulate, info) { |
| 1013 if (!dontPopulate) |
| 1014 this._populateIfNeeded(); |
| 1015 |
| 1016 if (info) |
| 1017 info.depthChange = 0; |
| 1018 |
| 1019 var element = skipUnrevealed ? (this.revealed() ? this.firstChild() : null)
: this.firstChild(); |
| 1020 if (element && (!skipUnrevealed || (skipUnrevealed && this.expanded))) { |
| 1021 if (info) |
| 1022 info.depthChange = 1; |
| 1023 return element; |
| 1024 } |
| 1025 |
| 1026 if (this === stayWithin) |
| 1027 return null; |
| 1028 |
| 1029 element = skipUnrevealed ? (this.revealed() ? this.nextSibling : null) : thi
s.nextSibling; |
| 1030 if (element) |
| 1031 return element; |
| 1032 |
| 1033 element = this; |
| 1034 while (element && !element.root && |
| 1035 !(skipUnrevealed ? (element.revealed() ? element.nextSibling : null)
: element.nextSibling) && |
| 1036 element.parent !== stayWithin) { |
| 1037 if (info) |
| 1038 info.depthChange -= 1; |
| 1039 element = element.parent; |
| 1040 } |
| 1041 |
| 1042 if (!element || element.root) |
| 1043 return null; |
| 1044 |
| 1045 return (skipUnrevealed ? (element.revealed() ? element.nextSibling : null) :
element.nextSibling); |
| 1046 } |
| 1047 |
| 1048 /** |
| 1049 * @param {boolean} skipUnrevealed |
| 1050 * @param {boolean=} dontPopulate |
| 1051 * @return {?TreeElement} |
| 1052 */ |
| 1053 traversePreviousTreeElement(skipUnrevealed, dontPopulate) { |
| 1054 var element = skipUnrevealed ? (this.revealed() ? this.previousSibling : nul
l) : this.previousSibling; |
| 1055 if (!dontPopulate && element) |
| 1056 element._populateIfNeeded(); |
| 1057 |
| 1058 while (element && (skipUnrevealed ? (element.revealed() && element.expanded
? element.lastChild() : null) : |
| 1059 element.lastChild())) { |
| 1060 if (!dontPopulate) |
| 1061 element._populateIfNeeded(); |
| 1062 element = |
| 1063 (skipUnrevealed ? (element.revealed() && element.expanded ? element.la
stChild() : null) : |
| 1064 element.lastChild()); |
| 1065 } |
| 1066 |
| 1067 if (element) |
| 1068 return element; |
| 1069 |
| 1070 if (!this.parent || this.parent.root) |
| 1071 return null; |
| 1072 |
| 1073 return this.parent; |
| 1074 } |
| 1075 |
| 1076 /** |
| 1077 * @return {boolean} |
| 1078 */ |
| 1079 isEventWithinDisclosureTriangle(event) { |
| 1080 // FIXME: We should not use getComputedStyle(). For that we need to get rid
of using ::before for disclosure triangle. (http://webk.it/74446) |
| 1081 var paddingLeftValue = window.getComputedStyle(this._listItemNode).paddingLe
ft; |
| 1082 console.assert(paddingLeftValue.endsWith('px')); |
| 1083 var computedLeftPadding = parseFloat(paddingLeftValue); |
| 1084 var left = this._listItemNode.totalOffsetLeft() + computedLeftPadding; |
| 1085 return event.pageX >= left && event.pageX <= left + TreeElement._ArrowToggle
Width && this._expandable; |
| 1086 } |
| 1087 }; |
385 | 1088 |
386 /** @const */ | 1089 /** @const */ |
387 TreeElement._ArrowToggleWidth = 10; | 1090 TreeElement._ArrowToggleWidth = 10; |
388 | |
389 TreeElement.prototype = { | |
390 /** | |
391 * @param {?TreeElement} ancestor | |
392 * @return {boolean} | |
393 */ | |
394 hasAncestor: function(ancestor) | |
395 { | |
396 if (!ancestor) | |
397 return false; | |
398 | |
399 var currentNode = this.parent; | |
400 while (currentNode) { | |
401 if (ancestor === currentNode) | |
402 return true; | |
403 currentNode = currentNode.parent; | |
404 } | |
405 | |
406 return false; | |
407 }, | |
408 | |
409 /** | |
410 * @param {?TreeElement} ancestor | |
411 * @return {boolean} | |
412 */ | |
413 hasAncestorOrSelf: function(ancestor) | |
414 { | |
415 return this === ancestor || this.hasAncestor(ancestor); | |
416 }, | |
417 | |
418 /** | |
419 * @return {!Array.<!TreeElement>} | |
420 */ | |
421 children: function() | |
422 { | |
423 return this._children || []; | |
424 }, | |
425 | |
426 /** | |
427 * @return {number} | |
428 */ | |
429 childCount: function() | |
430 { | |
431 return this._children ? this._children.length : 0; | |
432 }, | |
433 | |
434 /** | |
435 * @return {?TreeElement} | |
436 */ | |
437 firstChild: function() | |
438 { | |
439 return this._children ? this._children[0] : null; | |
440 }, | |
441 | |
442 /** | |
443 * @return {?TreeElement} | |
444 */ | |
445 lastChild: function() | |
446 { | |
447 return this._children ? this._children[this._children.length - 1] : null
; | |
448 }, | |
449 | |
450 /** | |
451 * @param {number} index | |
452 * @return {?TreeElement} | |
453 */ | |
454 childAt: function(index) | |
455 { | |
456 return this._children ? this._children[index] : null; | |
457 }, | |
458 | |
459 /** | |
460 * @param {!TreeElement} child | |
461 * @return {number} | |
462 */ | |
463 indexOfChild: function(child) | |
464 { | |
465 return this._children ? this._children.indexOf(child) : -1; | |
466 }, | |
467 | |
468 /** | |
469 * @param {!TreeElement} child | |
470 */ | |
471 appendChild: function(child) | |
472 { | |
473 if (!this._children) | |
474 this._children = []; | |
475 | |
476 var insertionIndex; | |
477 if (this.treeOutline && this.treeOutline._comparator) | |
478 insertionIndex = this._children.lowerBound(child, this.treeOutline._
comparator); | |
479 else | |
480 insertionIndex = this._children.length; | |
481 this.insertChild(child, insertionIndex); | |
482 }, | |
483 | |
484 /** | |
485 * @param {!TreeElement} child | |
486 * @param {number} index | |
487 */ | |
488 insertChild: function(child, index) | |
489 { | |
490 if (!this._children) | |
491 this._children = []; | |
492 | |
493 if (!child) | |
494 throw "child can't be undefined or null"; | |
495 | |
496 console.assert(!child.parent, "Attempting to insert a child that is alre
ady in the tree, reparenting is not supported."); | |
497 | |
498 var previousChild = (index > 0 ? this._children[index - 1] : null); | |
499 if (previousChild) { | |
500 previousChild.nextSibling = child; | |
501 child.previousSibling = previousChild; | |
502 } else { | |
503 child.previousSibling = null; | |
504 } | |
505 | |
506 var nextChild = this._children[index]; | |
507 if (nextChild) { | |
508 nextChild.previousSibling = child; | |
509 child.nextSibling = nextChild; | |
510 } else { | |
511 child.nextSibling = null; | |
512 } | |
513 | |
514 this._children.splice(index, 0, child); | |
515 | |
516 this.setExpandable(true); | |
517 child.parent = this; | |
518 | |
519 if (this.treeOutline) | |
520 this.treeOutline._bindTreeElement(child); | |
521 for (var current = child.firstChild(); this.treeOutline && current; curr
ent = current.traverseNextTreeElement(false, child, true)) | |
522 this.treeOutline._bindTreeElement(current); | |
523 child.onattach(); | |
524 child._ensureSelection(); | |
525 if (this.treeOutline) | |
526 this.treeOutline.dispatchEventToListeners(TreeOutline.Events.Element
Attached, child); | |
527 var nextSibling = child.nextSibling ? child.nextSibling._listItemNode :
null; | |
528 this._childrenListNode.insertBefore(child._listItemNode, nextSibling); | |
529 this._childrenListNode.insertBefore(child._childrenListNode, nextSibling
); | |
530 if (child.selected) | |
531 child.select(); | |
532 if (child.expanded) | |
533 child.expand(); | |
534 }, | |
535 | |
536 /** | |
537 * @param {number} childIndex | |
538 */ | |
539 removeChildAtIndex: function(childIndex) | |
540 { | |
541 if (childIndex < 0 || childIndex >= this._children.length) | |
542 throw "childIndex out of range"; | |
543 | |
544 var child = this._children[childIndex]; | |
545 this._children.splice(childIndex, 1); | |
546 | |
547 var parent = child.parent; | |
548 if (this.treeOutline && this.treeOutline.selectedTreeElement && this.tre
eOutline.selectedTreeElement.hasAncestorOrSelf(child)) { | |
549 if (child.nextSibling) | |
550 child.nextSibling.select(true); | |
551 else if (child.previousSibling) | |
552 child.previousSibling.select(true); | |
553 else if (parent) | |
554 parent.select(true); | |
555 } | |
556 | |
557 if (child.previousSibling) | |
558 child.previousSibling.nextSibling = child.nextSibling; | |
559 if (child.nextSibling) | |
560 child.nextSibling.previousSibling = child.previousSibling; | |
561 child.parent = null; | |
562 | |
563 if (this.treeOutline) | |
564 this.treeOutline._unbindTreeElement(child); | |
565 for (var current = child.firstChild(); this.treeOutline && current; curr
ent = current.traverseNextTreeElement(false, child, true)) | |
566 this.treeOutline._unbindTreeElement(current); | |
567 | |
568 child._detach(); | |
569 }, | |
570 | |
571 /** | |
572 * @param {!TreeElement} child | |
573 */ | |
574 removeChild: function(child) | |
575 { | |
576 if (!child) | |
577 throw "child can't be undefined or null"; | |
578 if (child.parent !== this) | |
579 return; | |
580 | |
581 var childIndex = this._children.indexOf(child); | |
582 if (childIndex === -1) | |
583 throw "child not found in this node's children"; | |
584 | |
585 this.removeChildAtIndex(childIndex); | |
586 }, | |
587 | |
588 removeChildren: function() | |
589 { | |
590 if (!this.root && this.treeOutline && this.treeOutline.selectedTreeEleme
nt && this.treeOutline.selectedTreeElement.hasAncestorOrSelf(this)) | |
591 this.select(true); | |
592 | |
593 for (var i = 0; this._children && i < this._children.length; ++i) { | |
594 var child = this._children[i]; | |
595 child.previousSibling = null; | |
596 child.nextSibling = null; | |
597 child.parent = null; | |
598 | |
599 if (this.treeOutline) | |
600 this.treeOutline._unbindTreeElement(child); | |
601 for (var current = child.firstChild(); this.treeOutline && current;
current = current.traverseNextTreeElement(false, child, true)) | |
602 this.treeOutline._unbindTreeElement(current); | |
603 child._detach(); | |
604 } | |
605 this._children = []; | |
606 }, | |
607 | |
608 get selectable() | |
609 { | |
610 if (this._hidden) | |
611 return false; | |
612 return this._selectable; | |
613 }, | |
614 | |
615 set selectable(x) | |
616 { | |
617 this._selectable = x; | |
618 }, | |
619 | |
620 get listItemElement() | |
621 { | |
622 return this._listItemNode; | |
623 }, | |
624 | |
625 get childrenListElement() | |
626 { | |
627 return this._childrenListNode; | |
628 }, | |
629 | |
630 get title() | |
631 { | |
632 return this._title; | |
633 }, | |
634 | |
635 /** | |
636 * @param {string|!Node} x | |
637 */ | |
638 set title(x) | |
639 { | |
640 if (this._title === x) | |
641 return; | |
642 this._title = x; | |
643 | |
644 if (typeof x === "string") { | |
645 this._titleElement = createElementWithClass("span", "tree-element-ti
tle"); | |
646 this._titleElement.textContent = x; | |
647 this.tooltip = x; | |
648 } else { | |
649 this._titleElement = x; | |
650 this.tooltip = ""; | |
651 } | |
652 | |
653 this._listItemNode.removeChildren(); | |
654 if (this._iconElement) | |
655 this._listItemNode.appendChild(this._iconElement); | |
656 | |
657 this._listItemNode.appendChild(this._titleElement); | |
658 this._ensureSelection(); | |
659 }, | |
660 | |
661 /** | |
662 * @return {string} | |
663 */ | |
664 titleAsText: function() | |
665 { | |
666 if (!this._title) | |
667 return ""; | |
668 if (typeof this._title === "string") | |
669 return this._title; | |
670 return this._title.textContent; | |
671 }, | |
672 | |
673 /** | |
674 * @param {!WebInspector.InplaceEditor.Config} editingConfig | |
675 */ | |
676 startEditingTitle: function(editingConfig) | |
677 { | |
678 WebInspector.InplaceEditor.startEditing(this._titleElement, editingConfi
g); | |
679 this.treeOutline._shadowRoot.getSelection().setBaseAndExtent(this._title
Element, 0, this._titleElement, 1); | |
680 }, | |
681 | |
682 createIcon() | |
683 { | |
684 if (!this._iconElement) { | |
685 this._iconElement = createElementWithClass("div", "icon"); | |
686 this._listItemNode.insertBefore(this._iconElement, this._listItemNod
e.firstChild); | |
687 this._ensureSelection(); | |
688 } | |
689 }, | |
690 | |
691 get tooltip() | |
692 { | |
693 return this._tooltip || ""; | |
694 }, | |
695 | |
696 /** | |
697 * @param {string} x | |
698 */ | |
699 set tooltip(x) | |
700 { | |
701 if (this._tooltip === x) | |
702 return; | |
703 this._tooltip = x; | |
704 this._listItemNode.title = x; | |
705 }, | |
706 | |
707 /** | |
708 * @return {boolean} | |
709 */ | |
710 isExpandable: function() | |
711 { | |
712 return this._expandable; | |
713 }, | |
714 | |
715 /** | |
716 * @param {boolean} expandable | |
717 */ | |
718 setExpandable: function(expandable) | |
719 { | |
720 if (this._expandable === expandable) | |
721 return; | |
722 | |
723 this._expandable = expandable; | |
724 | |
725 this._listItemNode.classList.toggle("parent", expandable); | |
726 if (!expandable) | |
727 this.collapse(); | |
728 }, | |
729 | |
730 /** | |
731 * @param {boolean} collapsible | |
732 */ | |
733 setCollapsible: function(collapsible) | |
734 { | |
735 if (this._collapsible === collapsible) | |
736 return; | |
737 | |
738 this._collapsible = collapsible; | |
739 | |
740 this._listItemNode.classList.toggle("always-parent", !collapsible); | |
741 if (!collapsible) | |
742 this.expand(); | |
743 }, | |
744 | |
745 get hidden() | |
746 { | |
747 return this._hidden; | |
748 }, | |
749 | |
750 set hidden(x) | |
751 { | |
752 if (this._hidden === x) | |
753 return; | |
754 | |
755 this._hidden = x; | |
756 | |
757 this._listItemNode.classList.toggle("hidden", x); | |
758 this._childrenListNode.classList.toggle("hidden", x); | |
759 }, | |
760 | |
761 invalidateChildren: function() | |
762 { | |
763 if (this._children) { | |
764 this.removeChildren(); | |
765 this._children = null; | |
766 } | |
767 }, | |
768 | |
769 _ensureSelection: function() | |
770 { | |
771 if (!this.treeOutline || !this.treeOutline._renderSelection) | |
772 return; | |
773 if (!this._selectionElement) | |
774 this._selectionElement = createElementWithClass("div", "selection fi
ll"); | |
775 this._listItemNode.insertBefore(this._selectionElement, this.listItemEle
ment.firstChild); | |
776 }, | |
777 | |
778 /** | |
779 * @param {!Event} event | |
780 */ | |
781 _treeElementToggled: function(event) | |
782 { | |
783 var element = event.currentTarget; | |
784 if (element.treeElement !== this || element.hasSelection()) | |
785 return; | |
786 | |
787 var toggleOnClick = this.toggleOnClick && !this.selectable; | |
788 var isInTriangle = this.isEventWithinDisclosureTriangle(event); | |
789 if (!toggleOnClick && !isInTriangle) | |
790 return; | |
791 | |
792 if (event.target && event.target.enclosingNodeOrSelfWithNodeName("a")) | |
793 return; | |
794 | |
795 if (this.expanded) { | |
796 if (event.altKey) | |
797 this.collapseRecursively(); | |
798 else | |
799 this.collapse(); | |
800 } else { | |
801 if (event.altKey) | |
802 this.expandRecursively(); | |
803 else | |
804 this.expand(); | |
805 } | |
806 event.consume(); | |
807 }, | |
808 | |
809 /** | |
810 * @param {!Event} event | |
811 */ | |
812 _handleMouseDown: function(event) | |
813 { | |
814 var element = event.currentTarget; | |
815 if (!element) | |
816 return; | |
817 if (!this.selectable) | |
818 return; | |
819 if (element.treeElement !== this) | |
820 return; | |
821 | |
822 if (this.isEventWithinDisclosureTriangle(event)) | |
823 return; | |
824 | |
825 this.selectOnMouseDown(event); | |
826 }, | |
827 | |
828 /** | |
829 * @param {!Event} event | |
830 */ | |
831 _handleDoubleClick: function(event) | |
832 { | |
833 var element = event.currentTarget; | |
834 if (!element || element.treeElement !== this) | |
835 return; | |
836 | |
837 var handled = this.ondblclick(event); | |
838 if (handled) | |
839 return; | |
840 if (this._expandable && !this.expanded) | |
841 this.expand(); | |
842 }, | |
843 | |
844 _detach: function() | |
845 { | |
846 this._listItemNode.remove(); | |
847 this._childrenListNode.remove(); | |
848 }, | |
849 | |
850 collapse: function() | |
851 { | |
852 if (!this.expanded || !this._collapsible) | |
853 return; | |
854 this._listItemNode.classList.remove("expanded"); | |
855 this._childrenListNode.classList.remove("expanded"); | |
856 this.expanded = false; | |
857 this.oncollapse(); | |
858 if (this.treeOutline) | |
859 this.treeOutline.dispatchEventToListeners(TreeOutline.Events.Element
Collapsed, this); | |
860 }, | |
861 | |
862 collapseRecursively: function() | |
863 { | |
864 var item = this; | |
865 while (item) { | |
866 if (item.expanded) | |
867 item.collapse(); | |
868 item = item.traverseNextTreeElement(false, this, true); | |
869 } | |
870 }, | |
871 | |
872 expand: function() | |
873 { | |
874 if (!this._expandable || (this.expanded && this._children)) | |
875 return; | |
876 | |
877 // Set this before onpopulate. Since onpopulate can add elements, this m
akes | |
878 // sure the expanded flag is true before calling those functions. This p
revents the possibility | |
879 // of an infinite loop if onpopulate were to call expand. | |
880 | |
881 this.expanded = true; | |
882 | |
883 this._populateIfNeeded(); | |
884 this._listItemNode.classList.add("expanded"); | |
885 this._childrenListNode.classList.add("expanded"); | |
886 | |
887 if (this.treeOutline) { | |
888 this.onexpand(); | |
889 this.treeOutline.dispatchEventToListeners(TreeOutline.Events.Element
Expanded, this); | |
890 } | |
891 }, | |
892 | |
893 /** | |
894 * @param {number=} maxDepth | |
895 */ | |
896 expandRecursively: function(maxDepth) | |
897 { | |
898 var item = this; | |
899 var info = {}; | |
900 var depth = 0; | |
901 | |
902 // The Inspector uses TreeOutlines to represents object properties, so r
ecursive expansion | |
903 // in some case can be infinite, since JavaScript objects can hold circu
lar references. | |
904 // So default to a recursion cap of 3 levels, since that gives fairly go
od results. | |
905 if (isNaN(maxDepth)) | |
906 maxDepth = 3; | |
907 | |
908 while (item) { | |
909 if (depth < maxDepth) | |
910 item.expand(); | |
911 item = item.traverseNextTreeElement(false, this, (depth >= maxDepth)
, info); | |
912 depth += info.depthChange; | |
913 } | |
914 }, | |
915 | |
916 /** | |
917 * @param {boolean=} center | |
918 */ | |
919 reveal: function(center) | |
920 { | |
921 var currentAncestor = this.parent; | |
922 while (currentAncestor && !currentAncestor.root) { | |
923 if (!currentAncestor.expanded) | |
924 currentAncestor.expand(); | |
925 currentAncestor = currentAncestor.parent; | |
926 } | |
927 | |
928 this.treeOutline._deferredScrollIntoView(this, !!center); | |
929 }, | |
930 | |
931 /** | |
932 * @return {boolean} | |
933 */ | |
934 revealed: function() | |
935 { | |
936 var currentAncestor = this.parent; | |
937 while (currentAncestor && !currentAncestor.root) { | |
938 if (!currentAncestor.expanded) | |
939 return false; | |
940 currentAncestor = currentAncestor.parent; | |
941 } | |
942 | |
943 return true; | |
944 }, | |
945 | |
946 selectOnMouseDown: function(event) | |
947 { | |
948 if (this.select(false, true)) | |
949 event.consume(true); | |
950 }, | |
951 | |
952 /** | |
953 * @param {boolean=} omitFocus | |
954 * @param {boolean=} selectedByUser | |
955 * @return {boolean} | |
956 */ | |
957 select: function(omitFocus, selectedByUser) | |
958 { | |
959 if (!this.treeOutline || !this.selectable || this.selected) | |
960 return false; | |
961 | |
962 if (this.treeOutline.selectedTreeElement) | |
963 this.treeOutline.selectedTreeElement.deselect(); | |
964 this.treeOutline.selectedTreeElement = null; | |
965 | |
966 if (this.treeOutline._rootElement === this) | |
967 return false; | |
968 | |
969 this.selected = true; | |
970 | |
971 if (!omitFocus) | |
972 this.treeOutline.focus(); | |
973 | |
974 // Focusing on another node may detach "this" from tree. | |
975 if (!this.treeOutline) | |
976 return false; | |
977 this.treeOutline.selectedTreeElement = this; | |
978 this._listItemNode.classList.add("selected"); | |
979 this.treeOutline.dispatchEventToListeners(TreeOutline.Events.ElementSele
cted, this); | |
980 return this.onselect(selectedByUser); | |
981 }, | |
982 | |
983 /** | |
984 * @param {boolean=} omitFocus | |
985 */ | |
986 revealAndSelect: function(omitFocus) | |
987 { | |
988 this.reveal(true); | |
989 this.select(omitFocus); | |
990 }, | |
991 | |
992 /** | |
993 * @param {boolean=} supressOnDeselect | |
994 */ | |
995 deselect: function(supressOnDeselect) | |
996 { | |
997 if (!this.treeOutline || this.treeOutline.selectedTreeElement !== this |
| !this.selected) | |
998 return; | |
999 | |
1000 this.selected = false; | |
1001 this.treeOutline.selectedTreeElement = null; | |
1002 this._listItemNode.classList.remove("selected"); | |
1003 }, | |
1004 | |
1005 _populateIfNeeded: function() | |
1006 { | |
1007 if (this.treeOutline && this._expandable && !this._children) { | |
1008 this._children = []; | |
1009 this.onpopulate(); | |
1010 } | |
1011 }, | |
1012 | |
1013 onpopulate: function() | |
1014 { | |
1015 // Overridden by subclasses. | |
1016 }, | |
1017 | |
1018 /** | |
1019 * @return {boolean} | |
1020 */ | |
1021 onenter: function() | |
1022 { | |
1023 return false; | |
1024 }, | |
1025 | |
1026 /** | |
1027 * @return {boolean} | |
1028 */ | |
1029 ondelete: function() | |
1030 { | |
1031 return false; | |
1032 }, | |
1033 | |
1034 /** | |
1035 * @return {boolean} | |
1036 */ | |
1037 onspace: function() | |
1038 { | |
1039 return false; | |
1040 }, | |
1041 | |
1042 onbind: function() | |
1043 { | |
1044 }, | |
1045 | |
1046 onunbind: function() | |
1047 { | |
1048 }, | |
1049 | |
1050 onattach: function() | |
1051 { | |
1052 }, | |
1053 | |
1054 onexpand: function() | |
1055 { | |
1056 }, | |
1057 | |
1058 oncollapse: function() | |
1059 { | |
1060 }, | |
1061 | |
1062 /** | |
1063 * @param {!Event} e | |
1064 * @return {boolean} | |
1065 */ | |
1066 ondblclick: function(e) | |
1067 { | |
1068 return false; | |
1069 }, | |
1070 | |
1071 /** | |
1072 * @param {boolean=} selectedByUser | |
1073 * @return {boolean} | |
1074 */ | |
1075 onselect: function(selectedByUser) | |
1076 { | |
1077 return false; | |
1078 }, | |
1079 | |
1080 /** | |
1081 * @param {boolean} skipUnrevealed | |
1082 * @param {?TreeElement=} stayWithin | |
1083 * @param {boolean=} dontPopulate | |
1084 * @param {!Object=} info | |
1085 * @return {?TreeElement} | |
1086 */ | |
1087 traverseNextTreeElement: function(skipUnrevealed, stayWithin, dontPopulate,
info) | |
1088 { | |
1089 if (!dontPopulate) | |
1090 this._populateIfNeeded(); | |
1091 | |
1092 if (info) | |
1093 info.depthChange = 0; | |
1094 | |
1095 var element = skipUnrevealed ? (this.revealed() ? this.firstChild() : nu
ll) : this.firstChild(); | |
1096 if (element && (!skipUnrevealed || (skipUnrevealed && this.expanded))) { | |
1097 if (info) | |
1098 info.depthChange = 1; | |
1099 return element; | |
1100 } | |
1101 | |
1102 if (this === stayWithin) | |
1103 return null; | |
1104 | |
1105 element = skipUnrevealed ? (this.revealed() ? this.nextSibling : null) :
this.nextSibling; | |
1106 if (element) | |
1107 return element; | |
1108 | |
1109 element = this; | |
1110 while (element && !element.root && !(skipUnrevealed ? (element.revealed(
) ? element.nextSibling : null) : element.nextSibling) && element.parent !== sta
yWithin) { | |
1111 if (info) | |
1112 info.depthChange -= 1; | |
1113 element = element.parent; | |
1114 } | |
1115 | |
1116 if (!element || element.root) | |
1117 return null; | |
1118 | |
1119 return (skipUnrevealed ? (element.revealed() ? element.nextSibling : nul
l) : element.nextSibling); | |
1120 }, | |
1121 | |
1122 /** | |
1123 * @param {boolean} skipUnrevealed | |
1124 * @param {boolean=} dontPopulate | |
1125 * @return {?TreeElement} | |
1126 */ | |
1127 traversePreviousTreeElement: function(skipUnrevealed, dontPopulate) | |
1128 { | |
1129 var element = skipUnrevealed ? (this.revealed() ? this.previousSibling :
null) : this.previousSibling; | |
1130 if (!dontPopulate && element) | |
1131 element._populateIfNeeded(); | |
1132 | |
1133 while (element && (skipUnrevealed ? (element.revealed() && element.expan
ded ? element.lastChild() : null) : element.lastChild())) { | |
1134 if (!dontPopulate) | |
1135 element._populateIfNeeded(); | |
1136 element = (skipUnrevealed ? (element.revealed() && element.expanded
? element.lastChild() : null) : element.lastChild()); | |
1137 } | |
1138 | |
1139 if (element) | |
1140 return element; | |
1141 | |
1142 if (!this.parent || this.parent.root) | |
1143 return null; | |
1144 | |
1145 return this.parent; | |
1146 }, | |
1147 | |
1148 /** | |
1149 * @return {boolean} | |
1150 */ | |
1151 isEventWithinDisclosureTriangle: function(event) | |
1152 { | |
1153 // FIXME: We should not use getComputedStyle(). For that we need to get
rid of using ::before for disclosure triangle. (http://webk.it/74446) | |
1154 var paddingLeftValue = window.getComputedStyle(this._listItemNode).paddi
ngLeft; | |
1155 console.assert(paddingLeftValue.endsWith("px")); | |
1156 var computedLeftPadding = parseFloat(paddingLeftValue); | |
1157 var left = this._listItemNode.totalOffsetLeft() + computedLeftPadding; | |
1158 return event.pageX >= left && event.pageX <= left + TreeElement._ArrowTo
ggleWidth && this._expandable; | |
1159 } | |
1160 }; | |
OLD | NEW |