Mercurial > repos > mingchen0919 > aurora_deseq2
comparison vakata-jstree-3.3.5/dist/jstree.js @ 0:55d2db17c67c draft
planemo upload commit 841d8b22bf9f1aaed6bfe8344b60617f45b275b2-dirty
author | mingchen0919 |
---|---|
date | Fri, 14 Dec 2018 00:21:26 -0500 |
parents | |
children |
comparison
equal
deleted
inserted
replaced
-1:000000000000 | 0:55d2db17c67c |
---|---|
1 /*globals jQuery, define, module, exports, require, window, document, postMessage */ | |
2 (function (factory) { | |
3 "use strict"; | |
4 if (typeof define === 'function' && define.amd) { | |
5 define(['jquery'], factory); | |
6 } | |
7 else if(typeof module !== 'undefined' && module.exports) { | |
8 module.exports = factory(require('jquery')); | |
9 } | |
10 else { | |
11 factory(jQuery); | |
12 } | |
13 }(function ($, undefined) { | |
14 "use strict"; | |
15 /*! | |
16 * jsTree 3.3.5 | |
17 * http://jstree.com/ | |
18 * | |
19 * Copyright (c) 2014 Ivan Bozhanov (http://vakata.com) | |
20 * | |
21 * Licensed same as jquery - under the terms of the MIT License | |
22 * http://www.opensource.org/licenses/mit-license.php | |
23 */ | |
24 /*! | |
25 * if using jslint please allow for the jQuery global and use following options: | |
26 * jslint: loopfunc: true, browser: true, ass: true, bitwise: true, continue: true, nomen: true, plusplus: true, regexp: true, unparam: true, todo: true, white: true | |
27 */ | |
28 /*jshint -W083 */ | |
29 | |
30 // prevent another load? maybe there is a better way? | |
31 if($.jstree) { | |
32 return; | |
33 } | |
34 | |
35 /** | |
36 * ### jsTree core functionality | |
37 */ | |
38 | |
39 // internal variables | |
40 var instance_counter = 0, | |
41 ccp_node = false, | |
42 ccp_mode = false, | |
43 ccp_inst = false, | |
44 themes_loaded = [], | |
45 src = $('script:last').attr('src'), | |
46 document = window.document; // local variable is always faster to access then a global | |
47 | |
48 /** | |
49 * holds all jstree related functions and variables, including the actual class and methods to create, access and manipulate instances. | |
50 * @name $.jstree | |
51 */ | |
52 $.jstree = { | |
53 /** | |
54 * specifies the jstree version in use | |
55 * @name $.jstree.version | |
56 */ | |
57 version : '3.3.5', | |
58 /** | |
59 * holds all the default options used when creating new instances | |
60 * @name $.jstree.defaults | |
61 */ | |
62 defaults : { | |
63 /** | |
64 * configure which plugins will be active on an instance. Should be an array of strings, where each element is a plugin name. The default is `[]` | |
65 * @name $.jstree.defaults.plugins | |
66 */ | |
67 plugins : [] | |
68 }, | |
69 /** | |
70 * stores all loaded jstree plugins (used internally) | |
71 * @name $.jstree.plugins | |
72 */ | |
73 plugins : {}, | |
74 path : src && src.indexOf('/') !== -1 ? src.replace(/\/[^\/]+$/,'') : '', | |
75 idregex : /[\\:&!^|()\[\]<>@*'+~#";.,=\- \/${}%?`]/g, | |
76 root : '#' | |
77 }; | |
78 | |
79 /** | |
80 * creates a jstree instance | |
81 * @name $.jstree.create(el [, options]) | |
82 * @param {DOMElement|jQuery|String} el the element to create the instance on, can be jQuery extended or a selector | |
83 * @param {Object} options options for this instance (extends `$.jstree.defaults`) | |
84 * @return {jsTree} the new instance | |
85 */ | |
86 $.jstree.create = function (el, options) { | |
87 var tmp = new $.jstree.core(++instance_counter), | |
88 opt = options; | |
89 options = $.extend(true, {}, $.jstree.defaults, options); | |
90 if(opt && opt.plugins) { | |
91 options.plugins = opt.plugins; | |
92 } | |
93 $.each(options.plugins, function (i, k) { | |
94 if(i !== 'core') { | |
95 tmp = tmp.plugin(k, options[k]); | |
96 } | |
97 }); | |
98 $(el).data('jstree', tmp); | |
99 tmp.init(el, options); | |
100 return tmp; | |
101 }; | |
102 /** | |
103 * remove all traces of jstree from the DOM and destroy all instances | |
104 * @name $.jstree.destroy() | |
105 */ | |
106 $.jstree.destroy = function () { | |
107 $('.jstree:jstree').jstree('destroy'); | |
108 $(document).off('.jstree'); | |
109 }; | |
110 /** | |
111 * the jstree class constructor, used only internally | |
112 * @private | |
113 * @name $.jstree.core(id) | |
114 * @param {Number} id this instance's index | |
115 */ | |
116 $.jstree.core = function (id) { | |
117 this._id = id; | |
118 this._cnt = 0; | |
119 this._wrk = null; | |
120 this._data = { | |
121 core : { | |
122 themes : { | |
123 name : false, | |
124 dots : false, | |
125 icons : false, | |
126 ellipsis : false | |
127 }, | |
128 selected : [], | |
129 last_error : {}, | |
130 working : false, | |
131 worker_queue : [], | |
132 focused : null | |
133 } | |
134 }; | |
135 }; | |
136 /** | |
137 * get a reference to an existing instance | |
138 * | |
139 * __Examples__ | |
140 * | |
141 * // provided a container with an ID of "tree", and a nested node with an ID of "branch" | |
142 * // all of there will return the same instance | |
143 * $.jstree.reference('tree'); | |
144 * $.jstree.reference('#tree'); | |
145 * $.jstree.reference($('#tree')); | |
146 * $.jstree.reference(document.getElementByID('tree')); | |
147 * $.jstree.reference('branch'); | |
148 * $.jstree.reference('#branch'); | |
149 * $.jstree.reference($('#branch')); | |
150 * $.jstree.reference(document.getElementByID('branch')); | |
151 * | |
152 * @name $.jstree.reference(needle) | |
153 * @param {DOMElement|jQuery|String} needle | |
154 * @return {jsTree|null} the instance or `null` if not found | |
155 */ | |
156 $.jstree.reference = function (needle) { | |
157 var tmp = null, | |
158 obj = null; | |
159 if(needle && needle.id && (!needle.tagName || !needle.nodeType)) { needle = needle.id; } | |
160 | |
161 if(!obj || !obj.length) { | |
162 try { obj = $(needle); } catch (ignore) { } | |
163 } | |
164 if(!obj || !obj.length) { | |
165 try { obj = $('#' + needle.replace($.jstree.idregex,'\\$&')); } catch (ignore) { } | |
166 } | |
167 if(obj && obj.length && (obj = obj.closest('.jstree')).length && (obj = obj.data('jstree'))) { | |
168 tmp = obj; | |
169 } | |
170 else { | |
171 $('.jstree').each(function () { | |
172 var inst = $(this).data('jstree'); | |
173 if(inst && inst._model.data[needle]) { | |
174 tmp = inst; | |
175 return false; | |
176 } | |
177 }); | |
178 } | |
179 return tmp; | |
180 }; | |
181 /** | |
182 * Create an instance, get an instance or invoke a command on a instance. | |
183 * | |
184 * If there is no instance associated with the current node a new one is created and `arg` is used to extend `$.jstree.defaults` for this new instance. There would be no return value (chaining is not broken). | |
185 * | |
186 * If there is an existing instance and `arg` is a string the command specified by `arg` is executed on the instance, with any additional arguments passed to the function. If the function returns a value it will be returned (chaining could break depending on function). | |
187 * | |
188 * If there is an existing instance and `arg` is not a string the instance itself is returned (similar to `$.jstree.reference`). | |
189 * | |
190 * In any other case - nothing is returned and chaining is not broken. | |
191 * | |
192 * __Examples__ | |
193 * | |
194 * $('#tree1').jstree(); // creates an instance | |
195 * $('#tree2').jstree({ plugins : [] }); // create an instance with some options | |
196 * $('#tree1').jstree('open_node', '#branch_1'); // call a method on an existing instance, passing additional arguments | |
197 * $('#tree2').jstree(); // get an existing instance (or create an instance) | |
198 * $('#tree2').jstree(true); // get an existing instance (will not create new instance) | |
199 * $('#branch_1').jstree().select_node('#branch_1'); // get an instance (using a nested element and call a method) | |
200 * | |
201 * @name $().jstree([arg]) | |
202 * @param {String|Object} arg | |
203 * @return {Mixed} | |
204 */ | |
205 $.fn.jstree = function (arg) { | |
206 // check for string argument | |
207 var is_method = (typeof arg === 'string'), | |
208 args = Array.prototype.slice.call(arguments, 1), | |
209 result = null; | |
210 if(arg === true && !this.length) { return false; } | |
211 this.each(function () { | |
212 // get the instance (if there is one) and method (if it exists) | |
213 var instance = $.jstree.reference(this), | |
214 method = is_method && instance ? instance[arg] : null; | |
215 // if calling a method, and method is available - execute on the instance | |
216 result = is_method && method ? | |
217 method.apply(instance, args) : | |
218 null; | |
219 // if there is no instance and no method is being called - create one | |
220 if(!instance && !is_method && (arg === undefined || $.isPlainObject(arg))) { | |
221 $.jstree.create(this, arg); | |
222 } | |
223 // if there is an instance and no method is called - return the instance | |
224 if( (instance && !is_method) || arg === true ) { | |
225 result = instance || false; | |
226 } | |
227 // if there was a method call which returned a result - break and return the value | |
228 if(result !== null && result !== undefined) { | |
229 return false; | |
230 } | |
231 }); | |
232 // if there was a method call with a valid return value - return that, otherwise continue the chain | |
233 return result !== null && result !== undefined ? | |
234 result : this; | |
235 }; | |
236 /** | |
237 * used to find elements containing an instance | |
238 * | |
239 * __Examples__ | |
240 * | |
241 * $('div:jstree').each(function () { | |
242 * $(this).jstree('destroy'); | |
243 * }); | |
244 * | |
245 * @name $(':jstree') | |
246 * @return {jQuery} | |
247 */ | |
248 $.expr.pseudos.jstree = $.expr.createPseudo(function(search) { | |
249 return function(a) { | |
250 return $(a).hasClass('jstree') && | |
251 $(a).data('jstree') !== undefined; | |
252 }; | |
253 }); | |
254 | |
255 /** | |
256 * stores all defaults for the core | |
257 * @name $.jstree.defaults.core | |
258 */ | |
259 $.jstree.defaults.core = { | |
260 /** | |
261 * data configuration | |
262 * | |
263 * If left as `false` the HTML inside the jstree container element is used to populate the tree (that should be an unordered list with list items). | |
264 * | |
265 * You can also pass in a HTML string or a JSON array here. | |
266 * | |
267 * It is possible to pass in a standard jQuery-like AJAX config and jstree will automatically determine if the response is JSON or HTML and use that to populate the tree. | |
268 * In addition to the standard jQuery ajax options here you can suppy functions for `data` and `url`, the functions will be run in the current instance's scope and a param will be passed indicating which node is being loaded, the return value of those functions will be used. | |
269 * | |
270 * The last option is to specify a function, that function will receive the node being loaded as argument and a second param which is a function which should be called with the result. | |
271 * | |
272 * __Examples__ | |
273 * | |
274 * // AJAX | |
275 * $('#tree').jstree({ | |
276 * 'core' : { | |
277 * 'data' : { | |
278 * 'url' : '/get/children/', | |
279 * 'data' : function (node) { | |
280 * return { 'id' : node.id }; | |
281 * } | |
282 * } | |
283 * }); | |
284 * | |
285 * // direct data | |
286 * $('#tree').jstree({ | |
287 * 'core' : { | |
288 * 'data' : [ | |
289 * 'Simple root node', | |
290 * { | |
291 * 'id' : 'node_2', | |
292 * 'text' : 'Root node with options', | |
293 * 'state' : { 'opened' : true, 'selected' : true }, | |
294 * 'children' : [ { 'text' : 'Child 1' }, 'Child 2'] | |
295 * } | |
296 * ] | |
297 * } | |
298 * }); | |
299 * | |
300 * // function | |
301 * $('#tree').jstree({ | |
302 * 'core' : { | |
303 * 'data' : function (obj, callback) { | |
304 * callback.call(this, ['Root 1', 'Root 2']); | |
305 * } | |
306 * }); | |
307 * | |
308 * @name $.jstree.defaults.core.data | |
309 */ | |
310 data : false, | |
311 /** | |
312 * configure the various strings used throughout the tree | |
313 * | |
314 * You can use an object where the key is the string you need to replace and the value is your replacement. | |
315 * Another option is to specify a function which will be called with an argument of the needed string and should return the replacement. | |
316 * If left as `false` no replacement is made. | |
317 * | |
318 * __Examples__ | |
319 * | |
320 * $('#tree').jstree({ | |
321 * 'core' : { | |
322 * 'strings' : { | |
323 * 'Loading ...' : 'Please wait ...' | |
324 * } | |
325 * } | |
326 * }); | |
327 * | |
328 * @name $.jstree.defaults.core.strings | |
329 */ | |
330 strings : false, | |
331 /** | |
332 * determines what happens when a user tries to modify the structure of the tree | |
333 * If left as `false` all operations like create, rename, delete, move or copy are prevented. | |
334 * You can set this to `true` to allow all interactions or use a function to have better control. | |
335 * | |
336 * __Examples__ | |
337 * | |
338 * $('#tree').jstree({ | |
339 * 'core' : { | |
340 * 'check_callback' : function (operation, node, node_parent, node_position, more) { | |
341 * // operation can be 'create_node', 'rename_node', 'delete_node', 'move_node', 'copy_node' or 'edit' | |
342 * // in case of 'rename_node' node_position is filled with the new node name | |
343 * return operation === 'rename_node' ? true : false; | |
344 * } | |
345 * } | |
346 * }); | |
347 * | |
348 * @name $.jstree.defaults.core.check_callback | |
349 */ | |
350 check_callback : false, | |
351 /** | |
352 * a callback called with a single object parameter in the instance's scope when something goes wrong (operation prevented, ajax failed, etc) | |
353 * @name $.jstree.defaults.core.error | |
354 */ | |
355 error : $.noop, | |
356 /** | |
357 * the open / close animation duration in milliseconds - set this to `false` to disable the animation (default is `200`) | |
358 * @name $.jstree.defaults.core.animation | |
359 */ | |
360 animation : 200, | |
361 /** | |
362 * a boolean indicating if multiple nodes can be selected | |
363 * @name $.jstree.defaults.core.multiple | |
364 */ | |
365 multiple : true, | |
366 /** | |
367 * theme configuration object | |
368 * @name $.jstree.defaults.core.themes | |
369 */ | |
370 themes : { | |
371 /** | |
372 * the name of the theme to use (if left as `false` the default theme is used) | |
373 * @name $.jstree.defaults.core.themes.name | |
374 */ | |
375 name : false, | |
376 /** | |
377 * the URL of the theme's CSS file, leave this as `false` if you have manually included the theme CSS (recommended). You can set this to `true` too which will try to autoload the theme. | |
378 * @name $.jstree.defaults.core.themes.url | |
379 */ | |
380 url : false, | |
381 /** | |
382 * the location of all jstree themes - only used if `url` is set to `true` | |
383 * @name $.jstree.defaults.core.themes.dir | |
384 */ | |
385 dir : false, | |
386 /** | |
387 * a boolean indicating if connecting dots are shown | |
388 * @name $.jstree.defaults.core.themes.dots | |
389 */ | |
390 dots : true, | |
391 /** | |
392 * a boolean indicating if node icons are shown | |
393 * @name $.jstree.defaults.core.themes.icons | |
394 */ | |
395 icons : true, | |
396 /** | |
397 * a boolean indicating if node ellipsis should be shown - this only works with a fixed with on the container | |
398 * @name $.jstree.defaults.core.themes.ellipsis | |
399 */ | |
400 ellipsis : false, | |
401 /** | |
402 * a boolean indicating if the tree background is striped | |
403 * @name $.jstree.defaults.core.themes.stripes | |
404 */ | |
405 stripes : false, | |
406 /** | |
407 * a string (or boolean `false`) specifying the theme variant to use (if the theme supports variants) | |
408 * @name $.jstree.defaults.core.themes.variant | |
409 */ | |
410 variant : false, | |
411 /** | |
412 * a boolean specifying if a reponsive version of the theme should kick in on smaller screens (if the theme supports it). Defaults to `false`. | |
413 * @name $.jstree.defaults.core.themes.responsive | |
414 */ | |
415 responsive : false | |
416 }, | |
417 /** | |
418 * if left as `true` all parents of all selected nodes will be opened once the tree loads (so that all selected nodes are visible to the user) | |
419 * @name $.jstree.defaults.core.expand_selected_onload | |
420 */ | |
421 expand_selected_onload : true, | |
422 /** | |
423 * if left as `true` web workers will be used to parse incoming JSON data where possible, so that the UI will not be blocked by large requests. Workers are however about 30% slower. Defaults to `true` | |
424 * @name $.jstree.defaults.core.worker | |
425 */ | |
426 worker : true, | |
427 /** | |
428 * Force node text to plain text (and escape HTML). Defaults to `false` | |
429 * @name $.jstree.defaults.core.force_text | |
430 */ | |
431 force_text : false, | |
432 /** | |
433 * Should the node should be toggled if the text is double clicked . Defaults to `true` | |
434 * @name $.jstree.defaults.core.dblclick_toggle | |
435 */ | |
436 dblclick_toggle : true, | |
437 /** | |
438 * Should the loaded nodes be part of the state. Defaults to `false` | |
439 * @name $.jstree.defaults.core.loaded_state | |
440 */ | |
441 loaded_state : false, | |
442 /** | |
443 * Should the last active node be focused when the tree container is blurred and the focused again. This helps working with screen readers. Defaults to `true` | |
444 * @name $.jstree.defaults.core.restore_focus | |
445 */ | |
446 restore_focus : true, | |
447 /** | |
448 * Default keyboard shortcuts (an object where each key is the button name or combo - like 'enter', 'ctrl-space', 'p', etc and the value is the function to execute in the instance's scope) | |
449 * @name $.jstree.defaults.core.keyboard | |
450 */ | |
451 keyboard : { | |
452 'ctrl-space': function (e) { | |
453 // aria defines space only with Ctrl | |
454 e.type = "click"; | |
455 $(e.currentTarget).trigger(e); | |
456 }, | |
457 'enter': function (e) { | |
458 // enter | |
459 e.type = "click"; | |
460 $(e.currentTarget).trigger(e); | |
461 }, | |
462 'left': function (e) { | |
463 // left | |
464 e.preventDefault(); | |
465 if(this.is_open(e.currentTarget)) { | |
466 this.close_node(e.currentTarget); | |
467 } | |
468 else { | |
469 var o = this.get_parent(e.currentTarget); | |
470 if(o && o.id !== $.jstree.root) { this.get_node(o, true).children('.jstree-anchor').focus(); } | |
471 } | |
472 }, | |
473 'up': function (e) { | |
474 // up | |
475 e.preventDefault(); | |
476 var o = this.get_prev_dom(e.currentTarget); | |
477 if(o && o.length) { o.children('.jstree-anchor').focus(); } | |
478 }, | |
479 'right': function (e) { | |
480 // right | |
481 e.preventDefault(); | |
482 if(this.is_closed(e.currentTarget)) { | |
483 this.open_node(e.currentTarget, function (o) { this.get_node(o, true).children('.jstree-anchor').focus(); }); | |
484 } | |
485 else if (this.is_open(e.currentTarget)) { | |
486 var o = this.get_node(e.currentTarget, true).children('.jstree-children')[0]; | |
487 if(o) { $(this._firstChild(o)).children('.jstree-anchor').focus(); } | |
488 } | |
489 }, | |
490 'down': function (e) { | |
491 // down | |
492 e.preventDefault(); | |
493 var o = this.get_next_dom(e.currentTarget); | |
494 if(o && o.length) { o.children('.jstree-anchor').focus(); } | |
495 }, | |
496 '*': function (e) { | |
497 // aria defines * on numpad as open_all - not very common | |
498 this.open_all(); | |
499 }, | |
500 'home': function (e) { | |
501 // home | |
502 e.preventDefault(); | |
503 var o = this._firstChild(this.get_container_ul()[0]); | |
504 if(o) { $(o).children('.jstree-anchor').filter(':visible').focus(); } | |
505 }, | |
506 'end': function (e) { | |
507 // end | |
508 e.preventDefault(); | |
509 this.element.find('.jstree-anchor').filter(':visible').last().focus(); | |
510 }, | |
511 'f2': function (e) { | |
512 // f2 - safe to include - if check_callback is false it will fail | |
513 e.preventDefault(); | |
514 this.edit(e.currentTarget); | |
515 } | |
516 } | |
517 }; | |
518 $.jstree.core.prototype = { | |
519 /** | |
520 * used to decorate an instance with a plugin. Used internally. | |
521 * @private | |
522 * @name plugin(deco [, opts]) | |
523 * @param {String} deco the plugin to decorate with | |
524 * @param {Object} opts options for the plugin | |
525 * @return {jsTree} | |
526 */ | |
527 plugin : function (deco, opts) { | |
528 var Child = $.jstree.plugins[deco]; | |
529 if(Child) { | |
530 this._data[deco] = {}; | |
531 Child.prototype = this; | |
532 return new Child(opts, this); | |
533 } | |
534 return this; | |
535 }, | |
536 /** | |
537 * initialize the instance. Used internally. | |
538 * @private | |
539 * @name init(el, optons) | |
540 * @param {DOMElement|jQuery|String} el the element we are transforming | |
541 * @param {Object} options options for this instance | |
542 * @trigger init.jstree, loading.jstree, loaded.jstree, ready.jstree, changed.jstree | |
543 */ | |
544 init : function (el, options) { | |
545 this._model = { | |
546 data : {}, | |
547 changed : [], | |
548 force_full_redraw : false, | |
549 redraw_timeout : false, | |
550 default_state : { | |
551 loaded : true, | |
552 opened : false, | |
553 selected : false, | |
554 disabled : false | |
555 } | |
556 }; | |
557 this._model.data[$.jstree.root] = { | |
558 id : $.jstree.root, | |
559 parent : null, | |
560 parents : [], | |
561 children : [], | |
562 children_d : [], | |
563 state : { loaded : false } | |
564 }; | |
565 | |
566 this.element = $(el).addClass('jstree jstree-' + this._id); | |
567 this.settings = options; | |
568 | |
569 this._data.core.ready = false; | |
570 this._data.core.loaded = false; | |
571 this._data.core.rtl = (this.element.css("direction") === "rtl"); | |
572 this.element[this._data.core.rtl ? 'addClass' : 'removeClass']("jstree-rtl"); | |
573 this.element.attr('role','tree'); | |
574 if(this.settings.core.multiple) { | |
575 this.element.attr('aria-multiselectable', true); | |
576 } | |
577 if(!this.element.attr('tabindex')) { | |
578 this.element.attr('tabindex','0'); | |
579 } | |
580 | |
581 this.bind(); | |
582 /** | |
583 * triggered after all events are bound | |
584 * @event | |
585 * @name init.jstree | |
586 */ | |
587 this.trigger("init"); | |
588 | |
589 this._data.core.original_container_html = this.element.find(" > ul > li").clone(true); | |
590 this._data.core.original_container_html | |
591 .find("li").addBack() | |
592 .contents().filter(function() { | |
593 return this.nodeType === 3 && (!this.nodeValue || /^\s+$/.test(this.nodeValue)); | |
594 }) | |
595 .remove(); | |
596 this.element.html("<"+"ul class='jstree-container-ul jstree-children' role='group'><"+"li id='j"+this._id+"_loading' class='jstree-initial-node jstree-loading jstree-leaf jstree-last' role='tree-item'><i class='jstree-icon jstree-ocl'></i><"+"a class='jstree-anchor' href='#'><i class='jstree-icon jstree-themeicon-hidden'></i>" + this.get_string("Loading ...") + "</a></li></ul>"); | |
597 this.element.attr('aria-activedescendant','j' + this._id + '_loading'); | |
598 this._data.core.li_height = this.get_container_ul().children("li").first().outerHeight() || 24; | |
599 this._data.core.node = this._create_prototype_node(); | |
600 /** | |
601 * triggered after the loading text is shown and before loading starts | |
602 * @event | |
603 * @name loading.jstree | |
604 */ | |
605 this.trigger("loading"); | |
606 this.load_node($.jstree.root); | |
607 }, | |
608 /** | |
609 * destroy an instance | |
610 * @name destroy() | |
611 * @param {Boolean} keep_html if not set to `true` the container will be emptied, otherwise the current DOM elements will be kept intact | |
612 */ | |
613 destroy : function (keep_html) { | |
614 /** | |
615 * triggered before the tree is destroyed | |
616 * @event | |
617 * @name destroy.jstree | |
618 */ | |
619 this.trigger("destroy"); | |
620 if(this._wrk) { | |
621 try { | |
622 window.URL.revokeObjectURL(this._wrk); | |
623 this._wrk = null; | |
624 } | |
625 catch (ignore) { } | |
626 } | |
627 if(!keep_html) { this.element.empty(); } | |
628 this.teardown(); | |
629 }, | |
630 /** | |
631 * Create a prototype node | |
632 * @name _create_prototype_node() | |
633 * @return {DOMElement} | |
634 */ | |
635 _create_prototype_node : function () { | |
636 var _node = document.createElement('LI'), _temp1, _temp2; | |
637 _node.setAttribute('role', 'treeitem'); | |
638 _temp1 = document.createElement('I'); | |
639 _temp1.className = 'jstree-icon jstree-ocl'; | |
640 _temp1.setAttribute('role', 'presentation'); | |
641 _node.appendChild(_temp1); | |
642 _temp1 = document.createElement('A'); | |
643 _temp1.className = 'jstree-anchor'; | |
644 _temp1.setAttribute('href','#'); | |
645 _temp1.setAttribute('tabindex','-1'); | |
646 _temp2 = document.createElement('I'); | |
647 _temp2.className = 'jstree-icon jstree-themeicon'; | |
648 _temp2.setAttribute('role', 'presentation'); | |
649 _temp1.appendChild(_temp2); | |
650 _node.appendChild(_temp1); | |
651 _temp1 = _temp2 = null; | |
652 | |
653 return _node; | |
654 }, | |
655 _kbevent_to_func : function (e) { | |
656 var keys = { | |
657 8: "Backspace", 9: "Tab", 13: "Return", 19: "Pause", 27: "Esc", | |
658 32: "Space", 33: "PageUp", 34: "PageDown", 35: "End", 36: "Home", | |
659 37: "Left", 38: "Up", 39: "Right", 40: "Down", 44: "Print", 45: "Insert", | |
660 46: "Delete", 96: "Numpad0", 97: "Numpad1", 98: "Numpad2", 99 : "Numpad3", | |
661 100: "Numpad4", 101: "Numpad5", 102: "Numpad6", 103: "Numpad7", | |
662 104: "Numpad8", 105: "Numpad9", '-13': "NumpadEnter", 112: "F1", | |
663 113: "F2", 114: "F3", 115: "F4", 116: "F5", 117: "F6", 118: "F7", | |
664 119: "F8", 120: "F9", 121: "F10", 122: "F11", 123: "F12", 144: "Numlock", | |
665 145: "Scrolllock", 16: 'Shift', 17: 'Ctrl', 18: 'Alt', | |
666 48: '0', 49: '1', 50: '2', 51: '3', 52: '4', 53: '5', | |
667 54: '6', 55: '7', 56: '8', 57: '9', 59: ';', 61: '=', 65: 'a', | |
668 66: 'b', 67: 'c', 68: 'd', 69: 'e', 70: 'f', 71: 'g', 72: 'h', | |
669 73: 'i', 74: 'j', 75: 'k', 76: 'l', 77: 'm', 78: 'n', 79: 'o', | |
670 80: 'p', 81: 'q', 82: 'r', 83: 's', 84: 't', 85: 'u', 86: 'v', | |
671 87: 'w', 88: 'x', 89: 'y', 90: 'z', 107: '+', 109: '-', 110: '.', | |
672 186: ';', 187: '=', 188: ',', 189: '-', 190: '.', 191: '/', 192: '`', | |
673 219: '[', 220: '\\',221: ']', 222: "'", 111: '/', 106: '*', 173: '-' | |
674 }; | |
675 var parts = []; | |
676 if (e.ctrlKey) { parts.push('ctrl'); } | |
677 if (e.altKey) { parts.push('alt'); } | |
678 if (e.shiftKey) { parts.push('shift'); } | |
679 parts.push(keys[e.which] || e.which); | |
680 parts = parts.sort().join('-').toLowerCase(); | |
681 | |
682 var kb = this.settings.core.keyboard, i, tmp; | |
683 for (i in kb) { | |
684 if (kb.hasOwnProperty(i)) { | |
685 tmp = i; | |
686 if (tmp !== '-' && tmp !== '+') { | |
687 tmp = tmp.replace('--', '-MINUS').replace('+-', '-MINUS').replace('++', '-PLUS').replace('-+', '-PLUS'); | |
688 tmp = tmp.split(/-|\+/).sort().join('-').replace('MINUS', '-').replace('PLUS', '+').toLowerCase(); | |
689 } | |
690 if (tmp === parts) { | |
691 return kb[i]; | |
692 } | |
693 } | |
694 } | |
695 return null; | |
696 }, | |
697 /** | |
698 * part of the destroying of an instance. Used internally. | |
699 * @private | |
700 * @name teardown() | |
701 */ | |
702 teardown : function () { | |
703 this.unbind(); | |
704 this.element | |
705 .removeClass('jstree') | |
706 .removeData('jstree') | |
707 .find("[class^='jstree']") | |
708 .addBack() | |
709 .attr("class", function () { return this.className.replace(/jstree[^ ]*|$/ig,''); }); | |
710 this.element = null; | |
711 }, | |
712 /** | |
713 * bind all events. Used internally. | |
714 * @private | |
715 * @name bind() | |
716 */ | |
717 bind : function () { | |
718 var word = '', | |
719 tout = null, | |
720 was_click = 0; | |
721 this.element | |
722 .on("dblclick.jstree", function (e) { | |
723 if(e.target.tagName && e.target.tagName.toLowerCase() === "input") { return true; } | |
724 if(document.selection && document.selection.empty) { | |
725 document.selection.empty(); | |
726 } | |
727 else { | |
728 if(window.getSelection) { | |
729 var sel = window.getSelection(); | |
730 try { | |
731 sel.removeAllRanges(); | |
732 sel.collapse(); | |
733 } catch (ignore) { } | |
734 } | |
735 } | |
736 }) | |
737 .on("mousedown.jstree", $.proxy(function (e) { | |
738 if(e.target === this.element[0]) { | |
739 e.preventDefault(); // prevent losing focus when clicking scroll arrows (FF, Chrome) | |
740 was_click = +(new Date()); // ie does not allow to prevent losing focus | |
741 } | |
742 }, this)) | |
743 .on("mousedown.jstree", ".jstree-ocl", function (e) { | |
744 e.preventDefault(); // prevent any node inside from losing focus when clicking the open/close icon | |
745 }) | |
746 .on("click.jstree", ".jstree-ocl", $.proxy(function (e) { | |
747 this.toggle_node(e.target); | |
748 }, this)) | |
749 .on("dblclick.jstree", ".jstree-anchor", $.proxy(function (e) { | |
750 if(e.target.tagName && e.target.tagName.toLowerCase() === "input") { return true; } | |
751 if(this.settings.core.dblclick_toggle) { | |
752 this.toggle_node(e.target); | |
753 } | |
754 }, this)) | |
755 .on("click.jstree", ".jstree-anchor", $.proxy(function (e) { | |
756 e.preventDefault(); | |
757 if(e.currentTarget !== document.activeElement) { $(e.currentTarget).focus(); } | |
758 this.activate_node(e.currentTarget, e); | |
759 }, this)) | |
760 .on('keydown.jstree', '.jstree-anchor', $.proxy(function (e) { | |
761 if(e.target.tagName && e.target.tagName.toLowerCase() === "input") { return true; } | |
762 if(this._data.core.rtl) { | |
763 if(e.which === 37) { e.which = 39; } | |
764 else if(e.which === 39) { e.which = 37; } | |
765 } | |
766 var f = this._kbevent_to_func(e); | |
767 if (f) { | |
768 var r = f.call(this, e); | |
769 if (r === false || r === true) { | |
770 return r; | |
771 } | |
772 } | |
773 }, this)) | |
774 .on("load_node.jstree", $.proxy(function (e, data) { | |
775 if(data.status) { | |
776 if(data.node.id === $.jstree.root && !this._data.core.loaded) { | |
777 this._data.core.loaded = true; | |
778 if(this._firstChild(this.get_container_ul()[0])) { | |
779 this.element.attr('aria-activedescendant',this._firstChild(this.get_container_ul()[0]).id); | |
780 } | |
781 /** | |
782 * triggered after the root node is loaded for the first time | |
783 * @event | |
784 * @name loaded.jstree | |
785 */ | |
786 this.trigger("loaded"); | |
787 } | |
788 if(!this._data.core.ready) { | |
789 setTimeout($.proxy(function() { | |
790 if(this.element && !this.get_container_ul().find('.jstree-loading').length) { | |
791 this._data.core.ready = true; | |
792 if(this._data.core.selected.length) { | |
793 if(this.settings.core.expand_selected_onload) { | |
794 var tmp = [], i, j; | |
795 for(i = 0, j = this._data.core.selected.length; i < j; i++) { | |
796 tmp = tmp.concat(this._model.data[this._data.core.selected[i]].parents); | |
797 } | |
798 tmp = $.vakata.array_unique(tmp); | |
799 for(i = 0, j = tmp.length; i < j; i++) { | |
800 this.open_node(tmp[i], false, 0); | |
801 } | |
802 } | |
803 this.trigger('changed', { 'action' : 'ready', 'selected' : this._data.core.selected }); | |
804 } | |
805 /** | |
806 * triggered after all nodes are finished loading | |
807 * @event | |
808 * @name ready.jstree | |
809 */ | |
810 this.trigger("ready"); | |
811 } | |
812 }, this), 0); | |
813 } | |
814 } | |
815 }, this)) | |
816 // quick searching when the tree is focused | |
817 .on('keypress.jstree', $.proxy(function (e) { | |
818 if(e.target.tagName && e.target.tagName.toLowerCase() === "input") { return true; } | |
819 if(tout) { clearTimeout(tout); } | |
820 tout = setTimeout(function () { | |
821 word = ''; | |
822 }, 500); | |
823 | |
824 var chr = String.fromCharCode(e.which).toLowerCase(), | |
825 col = this.element.find('.jstree-anchor').filter(':visible'), | |
826 ind = col.index(document.activeElement) || 0, | |
827 end = false; | |
828 word += chr; | |
829 | |
830 // match for whole word from current node down (including the current node) | |
831 if(word.length > 1) { | |
832 col.slice(ind).each($.proxy(function (i, v) { | |
833 if($(v).text().toLowerCase().indexOf(word) === 0) { | |
834 $(v).focus(); | |
835 end = true; | |
836 return false; | |
837 } | |
838 }, this)); | |
839 if(end) { return; } | |
840 | |
841 // match for whole word from the beginning of the tree | |
842 col.slice(0, ind).each($.proxy(function (i, v) { | |
843 if($(v).text().toLowerCase().indexOf(word) === 0) { | |
844 $(v).focus(); | |
845 end = true; | |
846 return false; | |
847 } | |
848 }, this)); | |
849 if(end) { return; } | |
850 } | |
851 // list nodes that start with that letter (only if word consists of a single char) | |
852 if(new RegExp('^' + chr.replace(/[-\/\\^$*+?.()|[\]{}]/g, '\\$&') + '+$').test(word)) { | |
853 // search for the next node starting with that letter | |
854 col.slice(ind + 1).each($.proxy(function (i, v) { | |
855 if($(v).text().toLowerCase().charAt(0) === chr) { | |
856 $(v).focus(); | |
857 end = true; | |
858 return false; | |
859 } | |
860 }, this)); | |
861 if(end) { return; } | |
862 | |
863 // search from the beginning | |
864 col.slice(0, ind + 1).each($.proxy(function (i, v) { | |
865 if($(v).text().toLowerCase().charAt(0) === chr) { | |
866 $(v).focus(); | |
867 end = true; | |
868 return false; | |
869 } | |
870 }, this)); | |
871 if(end) { return; } | |
872 } | |
873 }, this)) | |
874 // THEME RELATED | |
875 .on("init.jstree", $.proxy(function () { | |
876 var s = this.settings.core.themes; | |
877 this._data.core.themes.dots = s.dots; | |
878 this._data.core.themes.stripes = s.stripes; | |
879 this._data.core.themes.icons = s.icons; | |
880 this._data.core.themes.ellipsis = s.ellipsis; | |
881 this.set_theme(s.name || "default", s.url); | |
882 this.set_theme_variant(s.variant); | |
883 }, this)) | |
884 .on("loading.jstree", $.proxy(function () { | |
885 this[ this._data.core.themes.dots ? "show_dots" : "hide_dots" ](); | |
886 this[ this._data.core.themes.icons ? "show_icons" : "hide_icons" ](); | |
887 this[ this._data.core.themes.stripes ? "show_stripes" : "hide_stripes" ](); | |
888 this[ this._data.core.themes.ellipsis ? "show_ellipsis" : "hide_ellipsis" ](); | |
889 }, this)) | |
890 .on('blur.jstree', '.jstree-anchor', $.proxy(function (e) { | |
891 this._data.core.focused = null; | |
892 $(e.currentTarget).filter('.jstree-hovered').mouseleave(); | |
893 this.element.attr('tabindex', '0'); | |
894 }, this)) | |
895 .on('focus.jstree', '.jstree-anchor', $.proxy(function (e) { | |
896 var tmp = this.get_node(e.currentTarget); | |
897 if(tmp && tmp.id) { | |
898 this._data.core.focused = tmp.id; | |
899 } | |
900 this.element.find('.jstree-hovered').not(e.currentTarget).mouseleave(); | |
901 $(e.currentTarget).mouseenter(); | |
902 this.element.attr('tabindex', '-1'); | |
903 }, this)) | |
904 .on('focus.jstree', $.proxy(function () { | |
905 if(+(new Date()) - was_click > 500 && !this._data.core.focused && this.settings.core.restore_focus) { | |
906 was_click = 0; | |
907 var act = this.get_node(this.element.attr('aria-activedescendant'), true); | |
908 if(act) { | |
909 act.find('> .jstree-anchor').focus(); | |
910 } | |
911 } | |
912 }, this)) | |
913 .on('mouseenter.jstree', '.jstree-anchor', $.proxy(function (e) { | |
914 this.hover_node(e.currentTarget); | |
915 }, this)) | |
916 .on('mouseleave.jstree', '.jstree-anchor', $.proxy(function (e) { | |
917 this.dehover_node(e.currentTarget); | |
918 }, this)); | |
919 }, | |
920 /** | |
921 * part of the destroying of an instance. Used internally. | |
922 * @private | |
923 * @name unbind() | |
924 */ | |
925 unbind : function () { | |
926 this.element.off('.jstree'); | |
927 $(document).off('.jstree-' + this._id); | |
928 }, | |
929 /** | |
930 * trigger an event. Used internally. | |
931 * @private | |
932 * @name trigger(ev [, data]) | |
933 * @param {String} ev the name of the event to trigger | |
934 * @param {Object} data additional data to pass with the event | |
935 */ | |
936 trigger : function (ev, data) { | |
937 if(!data) { | |
938 data = {}; | |
939 } | |
940 data.instance = this; | |
941 this.element.triggerHandler(ev.replace('.jstree','') + '.jstree', data); | |
942 }, | |
943 /** | |
944 * returns the jQuery extended instance container | |
945 * @name get_container() | |
946 * @return {jQuery} | |
947 */ | |
948 get_container : function () { | |
949 return this.element; | |
950 }, | |
951 /** | |
952 * returns the jQuery extended main UL node inside the instance container. Used internally. | |
953 * @private | |
954 * @name get_container_ul() | |
955 * @return {jQuery} | |
956 */ | |
957 get_container_ul : function () { | |
958 return this.element.children(".jstree-children").first(); | |
959 }, | |
960 /** | |
961 * gets string replacements (localization). Used internally. | |
962 * @private | |
963 * @name get_string(key) | |
964 * @param {String} key | |
965 * @return {String} | |
966 */ | |
967 get_string : function (key) { | |
968 var a = this.settings.core.strings; | |
969 if($.isFunction(a)) { return a.call(this, key); } | |
970 if(a && a[key]) { return a[key]; } | |
971 return key; | |
972 }, | |
973 /** | |
974 * gets the first child of a DOM node. Used internally. | |
975 * @private | |
976 * @name _firstChild(dom) | |
977 * @param {DOMElement} dom | |
978 * @return {DOMElement} | |
979 */ | |
980 _firstChild : function (dom) { | |
981 dom = dom ? dom.firstChild : null; | |
982 while(dom !== null && dom.nodeType !== 1) { | |
983 dom = dom.nextSibling; | |
984 } | |
985 return dom; | |
986 }, | |
987 /** | |
988 * gets the next sibling of a DOM node. Used internally. | |
989 * @private | |
990 * @name _nextSibling(dom) | |
991 * @param {DOMElement} dom | |
992 * @return {DOMElement} | |
993 */ | |
994 _nextSibling : function (dom) { | |
995 dom = dom ? dom.nextSibling : null; | |
996 while(dom !== null && dom.nodeType !== 1) { | |
997 dom = dom.nextSibling; | |
998 } | |
999 return dom; | |
1000 }, | |
1001 /** | |
1002 * gets the previous sibling of a DOM node. Used internally. | |
1003 * @private | |
1004 * @name _previousSibling(dom) | |
1005 * @param {DOMElement} dom | |
1006 * @return {DOMElement} | |
1007 */ | |
1008 _previousSibling : function (dom) { | |
1009 dom = dom ? dom.previousSibling : null; | |
1010 while(dom !== null && dom.nodeType !== 1) { | |
1011 dom = dom.previousSibling; | |
1012 } | |
1013 return dom; | |
1014 }, | |
1015 /** | |
1016 * get the JSON representation of a node (or the actual jQuery extended DOM node) by using any input (child DOM element, ID string, selector, etc) | |
1017 * @name get_node(obj [, as_dom]) | |
1018 * @param {mixed} obj | |
1019 * @param {Boolean} as_dom | |
1020 * @return {Object|jQuery} | |
1021 */ | |
1022 get_node : function (obj, as_dom) { | |
1023 if(obj && obj.id) { | |
1024 obj = obj.id; | |
1025 } | |
1026 var dom; | |
1027 try { | |
1028 if(this._model.data[obj]) { | |
1029 obj = this._model.data[obj]; | |
1030 } | |
1031 else if(typeof obj === "string" && this._model.data[obj.replace(/^#/, '')]) { | |
1032 obj = this._model.data[obj.replace(/^#/, '')]; | |
1033 } | |
1034 else if(typeof obj === "string" && (dom = $('#' + obj.replace($.jstree.idregex,'\\$&'), this.element)).length && this._model.data[dom.closest('.jstree-node').attr('id')]) { | |
1035 obj = this._model.data[dom.closest('.jstree-node').attr('id')]; | |
1036 } | |
1037 else if((dom = $(obj, this.element)).length && this._model.data[dom.closest('.jstree-node').attr('id')]) { | |
1038 obj = this._model.data[dom.closest('.jstree-node').attr('id')]; | |
1039 } | |
1040 else if((dom = $(obj, this.element)).length && dom.hasClass('jstree')) { | |
1041 obj = this._model.data[$.jstree.root]; | |
1042 } | |
1043 else { | |
1044 return false; | |
1045 } | |
1046 | |
1047 if(as_dom) { | |
1048 obj = obj.id === $.jstree.root ? this.element : $('#' + obj.id.replace($.jstree.idregex,'\\$&'), this.element); | |
1049 } | |
1050 return obj; | |
1051 } catch (ex) { return false; } | |
1052 }, | |
1053 /** | |
1054 * get the path to a node, either consisting of node texts, or of node IDs, optionally glued together (otherwise an array) | |
1055 * @name get_path(obj [, glue, ids]) | |
1056 * @param {mixed} obj the node | |
1057 * @param {String} glue if you want the path as a string - pass the glue here (for example '/'), if a falsy value is supplied here, an array is returned | |
1058 * @param {Boolean} ids if set to true build the path using ID, otherwise node text is used | |
1059 * @return {mixed} | |
1060 */ | |
1061 get_path : function (obj, glue, ids) { | |
1062 obj = obj.parents ? obj : this.get_node(obj); | |
1063 if(!obj || obj.id === $.jstree.root || !obj.parents) { | |
1064 return false; | |
1065 } | |
1066 var i, j, p = []; | |
1067 p.push(ids ? obj.id : obj.text); | |
1068 for(i = 0, j = obj.parents.length; i < j; i++) { | |
1069 p.push(ids ? obj.parents[i] : this.get_text(obj.parents[i])); | |
1070 } | |
1071 p = p.reverse().slice(1); | |
1072 return glue ? p.join(glue) : p; | |
1073 }, | |
1074 /** | |
1075 * get the next visible node that is below the `obj` node. If `strict` is set to `true` only sibling nodes are returned. | |
1076 * @name get_next_dom(obj [, strict]) | |
1077 * @param {mixed} obj | |
1078 * @param {Boolean} strict | |
1079 * @return {jQuery} | |
1080 */ | |
1081 get_next_dom : function (obj, strict) { | |
1082 var tmp; | |
1083 obj = this.get_node(obj, true); | |
1084 if(obj[0] === this.element[0]) { | |
1085 tmp = this._firstChild(this.get_container_ul()[0]); | |
1086 while (tmp && tmp.offsetHeight === 0) { | |
1087 tmp = this._nextSibling(tmp); | |
1088 } | |
1089 return tmp ? $(tmp) : false; | |
1090 } | |
1091 if(!obj || !obj.length) { | |
1092 return false; | |
1093 } | |
1094 if(strict) { | |
1095 tmp = obj[0]; | |
1096 do { | |
1097 tmp = this._nextSibling(tmp); | |
1098 } while (tmp && tmp.offsetHeight === 0); | |
1099 return tmp ? $(tmp) : false; | |
1100 } | |
1101 if(obj.hasClass("jstree-open")) { | |
1102 tmp = this._firstChild(obj.children('.jstree-children')[0]); | |
1103 while (tmp && tmp.offsetHeight === 0) { | |
1104 tmp = this._nextSibling(tmp); | |
1105 } | |
1106 if(tmp !== null) { | |
1107 return $(tmp); | |
1108 } | |
1109 } | |
1110 tmp = obj[0]; | |
1111 do { | |
1112 tmp = this._nextSibling(tmp); | |
1113 } while (tmp && tmp.offsetHeight === 0); | |
1114 if(tmp !== null) { | |
1115 return $(tmp); | |
1116 } | |
1117 return obj.parentsUntil(".jstree",".jstree-node").nextAll(".jstree-node:visible").first(); | |
1118 }, | |
1119 /** | |
1120 * get the previous visible node that is above the `obj` node. If `strict` is set to `true` only sibling nodes are returned. | |
1121 * @name get_prev_dom(obj [, strict]) | |
1122 * @param {mixed} obj | |
1123 * @param {Boolean} strict | |
1124 * @return {jQuery} | |
1125 */ | |
1126 get_prev_dom : function (obj, strict) { | |
1127 var tmp; | |
1128 obj = this.get_node(obj, true); | |
1129 if(obj[0] === this.element[0]) { | |
1130 tmp = this.get_container_ul()[0].lastChild; | |
1131 while (tmp && tmp.offsetHeight === 0) { | |
1132 tmp = this._previousSibling(tmp); | |
1133 } | |
1134 return tmp ? $(tmp) : false; | |
1135 } | |
1136 if(!obj || !obj.length) { | |
1137 return false; | |
1138 } | |
1139 if(strict) { | |
1140 tmp = obj[0]; | |
1141 do { | |
1142 tmp = this._previousSibling(tmp); | |
1143 } while (tmp && tmp.offsetHeight === 0); | |
1144 return tmp ? $(tmp) : false; | |
1145 } | |
1146 tmp = obj[0]; | |
1147 do { | |
1148 tmp = this._previousSibling(tmp); | |
1149 } while (tmp && tmp.offsetHeight === 0); | |
1150 if(tmp !== null) { | |
1151 obj = $(tmp); | |
1152 while(obj.hasClass("jstree-open")) { | |
1153 obj = obj.children(".jstree-children").first().children(".jstree-node:visible:last"); | |
1154 } | |
1155 return obj; | |
1156 } | |
1157 tmp = obj[0].parentNode.parentNode; | |
1158 return tmp && tmp.className && tmp.className.indexOf('jstree-node') !== -1 ? $(tmp) : false; | |
1159 }, | |
1160 /** | |
1161 * get the parent ID of a node | |
1162 * @name get_parent(obj) | |
1163 * @param {mixed} obj | |
1164 * @return {String} | |
1165 */ | |
1166 get_parent : function (obj) { | |
1167 obj = this.get_node(obj); | |
1168 if(!obj || obj.id === $.jstree.root) { | |
1169 return false; | |
1170 } | |
1171 return obj.parent; | |
1172 }, | |
1173 /** | |
1174 * get a jQuery collection of all the children of a node (node must be rendered), returns false on error | |
1175 * @name get_children_dom(obj) | |
1176 * @param {mixed} obj | |
1177 * @return {jQuery} | |
1178 */ | |
1179 get_children_dom : function (obj) { | |
1180 obj = this.get_node(obj, true); | |
1181 if(obj[0] === this.element[0]) { | |
1182 return this.get_container_ul().children(".jstree-node"); | |
1183 } | |
1184 if(!obj || !obj.length) { | |
1185 return false; | |
1186 } | |
1187 return obj.children(".jstree-children").children(".jstree-node"); | |
1188 }, | |
1189 /** | |
1190 * checks if a node has children | |
1191 * @name is_parent(obj) | |
1192 * @param {mixed} obj | |
1193 * @return {Boolean} | |
1194 */ | |
1195 is_parent : function (obj) { | |
1196 obj = this.get_node(obj); | |
1197 return obj && (obj.state.loaded === false || obj.children.length > 0); | |
1198 }, | |
1199 /** | |
1200 * checks if a node is loaded (its children are available) | |
1201 * @name is_loaded(obj) | |
1202 * @param {mixed} obj | |
1203 * @return {Boolean} | |
1204 */ | |
1205 is_loaded : function (obj) { | |
1206 obj = this.get_node(obj); | |
1207 return obj && obj.state.loaded; | |
1208 }, | |
1209 /** | |
1210 * check if a node is currently loading (fetching children) | |
1211 * @name is_loading(obj) | |
1212 * @param {mixed} obj | |
1213 * @return {Boolean} | |
1214 */ | |
1215 is_loading : function (obj) { | |
1216 obj = this.get_node(obj); | |
1217 return obj && obj.state && obj.state.loading; | |
1218 }, | |
1219 /** | |
1220 * check if a node is opened | |
1221 * @name is_open(obj) | |
1222 * @param {mixed} obj | |
1223 * @return {Boolean} | |
1224 */ | |
1225 is_open : function (obj) { | |
1226 obj = this.get_node(obj); | |
1227 return obj && obj.state.opened; | |
1228 }, | |
1229 /** | |
1230 * check if a node is in a closed state | |
1231 * @name is_closed(obj) | |
1232 * @param {mixed} obj | |
1233 * @return {Boolean} | |
1234 */ | |
1235 is_closed : function (obj) { | |
1236 obj = this.get_node(obj); | |
1237 return obj && this.is_parent(obj) && !obj.state.opened; | |
1238 }, | |
1239 /** | |
1240 * check if a node has no children | |
1241 * @name is_leaf(obj) | |
1242 * @param {mixed} obj | |
1243 * @return {Boolean} | |
1244 */ | |
1245 is_leaf : function (obj) { | |
1246 return !this.is_parent(obj); | |
1247 }, | |
1248 /** | |
1249 * loads a node (fetches its children using the `core.data` setting). Multiple nodes can be passed to by using an array. | |
1250 * @name load_node(obj [, callback]) | |
1251 * @param {mixed} obj | |
1252 * @param {function} callback a function to be executed once loading is complete, the function is executed in the instance's scope and receives two arguments - the node and a boolean status | |
1253 * @return {Boolean} | |
1254 * @trigger load_node.jstree | |
1255 */ | |
1256 load_node : function (obj, callback) { | |
1257 var k, l, i, j, c; | |
1258 if($.isArray(obj)) { | |
1259 this._load_nodes(obj.slice(), callback); | |
1260 return true; | |
1261 } | |
1262 obj = this.get_node(obj); | |
1263 if(!obj) { | |
1264 if(callback) { callback.call(this, obj, false); } | |
1265 return false; | |
1266 } | |
1267 // if(obj.state.loading) { } // the node is already loading - just wait for it to load and invoke callback? but if called implicitly it should be loaded again? | |
1268 if(obj.state.loaded) { | |
1269 obj.state.loaded = false; | |
1270 for(i = 0, j = obj.parents.length; i < j; i++) { | |
1271 this._model.data[obj.parents[i]].children_d = $.vakata.array_filter(this._model.data[obj.parents[i]].children_d, function (v) { | |
1272 return $.inArray(v, obj.children_d) === -1; | |
1273 }); | |
1274 } | |
1275 for(k = 0, l = obj.children_d.length; k < l; k++) { | |
1276 if(this._model.data[obj.children_d[k]].state.selected) { | |
1277 c = true; | |
1278 } | |
1279 delete this._model.data[obj.children_d[k]]; | |
1280 } | |
1281 if (c) { | |
1282 this._data.core.selected = $.vakata.array_filter(this._data.core.selected, function (v) { | |
1283 return $.inArray(v, obj.children_d) === -1; | |
1284 }); | |
1285 } | |
1286 obj.children = []; | |
1287 obj.children_d = []; | |
1288 if(c) { | |
1289 this.trigger('changed', { 'action' : 'load_node', 'node' : obj, 'selected' : this._data.core.selected }); | |
1290 } | |
1291 } | |
1292 obj.state.failed = false; | |
1293 obj.state.loading = true; | |
1294 this.get_node(obj, true).addClass("jstree-loading").attr('aria-busy',true); | |
1295 this._load_node(obj, $.proxy(function (status) { | |
1296 obj = this._model.data[obj.id]; | |
1297 obj.state.loading = false; | |
1298 obj.state.loaded = status; | |
1299 obj.state.failed = !obj.state.loaded; | |
1300 var dom = this.get_node(obj, true), i = 0, j = 0, m = this._model.data, has_children = false; | |
1301 for(i = 0, j = obj.children.length; i < j; i++) { | |
1302 if(m[obj.children[i]] && !m[obj.children[i]].state.hidden) { | |
1303 has_children = true; | |
1304 break; | |
1305 } | |
1306 } | |
1307 if(obj.state.loaded && dom && dom.length) { | |
1308 dom.removeClass('jstree-closed jstree-open jstree-leaf'); | |
1309 if (!has_children) { | |
1310 dom.addClass('jstree-leaf'); | |
1311 } | |
1312 else { | |
1313 if (obj.id !== '#') { | |
1314 dom.addClass(obj.state.opened ? 'jstree-open' : 'jstree-closed'); | |
1315 } | |
1316 } | |
1317 } | |
1318 dom.removeClass("jstree-loading").attr('aria-busy',false); | |
1319 /** | |
1320 * triggered after a node is loaded | |
1321 * @event | |
1322 * @name load_node.jstree | |
1323 * @param {Object} node the node that was loading | |
1324 * @param {Boolean} status was the node loaded successfully | |
1325 */ | |
1326 this.trigger('load_node', { "node" : obj, "status" : status }); | |
1327 if(callback) { | |
1328 callback.call(this, obj, status); | |
1329 } | |
1330 }, this)); | |
1331 return true; | |
1332 }, | |
1333 /** | |
1334 * load an array of nodes (will also load unavailable nodes as soon as the appear in the structure). Used internally. | |
1335 * @private | |
1336 * @name _load_nodes(nodes [, callback]) | |
1337 * @param {array} nodes | |
1338 * @param {function} callback a function to be executed once loading is complete, the function is executed in the instance's scope and receives one argument - the array passed to _load_nodes | |
1339 */ | |
1340 _load_nodes : function (nodes, callback, is_callback, force_reload) { | |
1341 var r = true, | |
1342 c = function () { this._load_nodes(nodes, callback, true); }, | |
1343 m = this._model.data, i, j, tmp = []; | |
1344 for(i = 0, j = nodes.length; i < j; i++) { | |
1345 if(m[nodes[i]] && ( (!m[nodes[i]].state.loaded && !m[nodes[i]].state.failed) || (!is_callback && force_reload) )) { | |
1346 if(!this.is_loading(nodes[i])) { | |
1347 this.load_node(nodes[i], c); | |
1348 } | |
1349 r = false; | |
1350 } | |
1351 } | |
1352 if(r) { | |
1353 for(i = 0, j = nodes.length; i < j; i++) { | |
1354 if(m[nodes[i]] && m[nodes[i]].state.loaded) { | |
1355 tmp.push(nodes[i]); | |
1356 } | |
1357 } | |
1358 if(callback && !callback.done) { | |
1359 callback.call(this, tmp); | |
1360 callback.done = true; | |
1361 } | |
1362 } | |
1363 }, | |
1364 /** | |
1365 * loads all unloaded nodes | |
1366 * @name load_all([obj, callback]) | |
1367 * @param {mixed} obj the node to load recursively, omit to load all nodes in the tree | |
1368 * @param {function} callback a function to be executed once loading all the nodes is complete, | |
1369 * @trigger load_all.jstree | |
1370 */ | |
1371 load_all : function (obj, callback) { | |
1372 if(!obj) { obj = $.jstree.root; } | |
1373 obj = this.get_node(obj); | |
1374 if(!obj) { return false; } | |
1375 var to_load = [], | |
1376 m = this._model.data, | |
1377 c = m[obj.id].children_d, | |
1378 i, j; | |
1379 if(obj.state && !obj.state.loaded) { | |
1380 to_load.push(obj.id); | |
1381 } | |
1382 for(i = 0, j = c.length; i < j; i++) { | |
1383 if(m[c[i]] && m[c[i]].state && !m[c[i]].state.loaded) { | |
1384 to_load.push(c[i]); | |
1385 } | |
1386 } | |
1387 if(to_load.length) { | |
1388 this._load_nodes(to_load, function () { | |
1389 this.load_all(obj, callback); | |
1390 }); | |
1391 } | |
1392 else { | |
1393 /** | |
1394 * triggered after a load_all call completes | |
1395 * @event | |
1396 * @name load_all.jstree | |
1397 * @param {Object} node the recursively loaded node | |
1398 */ | |
1399 if(callback) { callback.call(this, obj); } | |
1400 this.trigger('load_all', { "node" : obj }); | |
1401 } | |
1402 }, | |
1403 /** | |
1404 * handles the actual loading of a node. Used only internally. | |
1405 * @private | |
1406 * @name _load_node(obj [, callback]) | |
1407 * @param {mixed} obj | |
1408 * @param {function} callback a function to be executed once loading is complete, the function is executed in the instance's scope and receives one argument - a boolean status | |
1409 * @return {Boolean} | |
1410 */ | |
1411 _load_node : function (obj, callback) { | |
1412 var s = this.settings.core.data, t; | |
1413 var notTextOrCommentNode = function notTextOrCommentNode () { | |
1414 return this.nodeType !== 3 && this.nodeType !== 8; | |
1415 }; | |
1416 // use original HTML | |
1417 if(!s) { | |
1418 if(obj.id === $.jstree.root) { | |
1419 return this._append_html_data(obj, this._data.core.original_container_html.clone(true), function (status) { | |
1420 callback.call(this, status); | |
1421 }); | |
1422 } | |
1423 else { | |
1424 return callback.call(this, false); | |
1425 } | |
1426 // return callback.call(this, obj.id === $.jstree.root ? this._append_html_data(obj, this._data.core.original_container_html.clone(true)) : false); | |
1427 } | |
1428 if($.isFunction(s)) { | |
1429 return s.call(this, obj, $.proxy(function (d) { | |
1430 if(d === false) { | |
1431 callback.call(this, false); | |
1432 } | |
1433 else { | |
1434 this[typeof d === 'string' ? '_append_html_data' : '_append_json_data'](obj, typeof d === 'string' ? $($.parseHTML(d)).filter(notTextOrCommentNode) : d, function (status) { | |
1435 callback.call(this, status); | |
1436 }); | |
1437 } | |
1438 // return d === false ? callback.call(this, false) : callback.call(this, this[typeof d === 'string' ? '_append_html_data' : '_append_json_data'](obj, typeof d === 'string' ? $(d) : d)); | |
1439 }, this)); | |
1440 } | |
1441 if(typeof s === 'object') { | |
1442 if(s.url) { | |
1443 s = $.extend(true, {}, s); | |
1444 if($.isFunction(s.url)) { | |
1445 s.url = s.url.call(this, obj); | |
1446 } | |
1447 if($.isFunction(s.data)) { | |
1448 s.data = s.data.call(this, obj); | |
1449 } | |
1450 return $.ajax(s) | |
1451 .done($.proxy(function (d,t,x) { | |
1452 var type = x.getResponseHeader('Content-Type'); | |
1453 if((type && type.indexOf('json') !== -1) || typeof d === "object") { | |
1454 return this._append_json_data(obj, d, function (status) { callback.call(this, status); }); | |
1455 //return callback.call(this, this._append_json_data(obj, d)); | |
1456 } | |
1457 if((type && type.indexOf('html') !== -1) || typeof d === "string") { | |
1458 return this._append_html_data(obj, $($.parseHTML(d)).filter(notTextOrCommentNode), function (status) { callback.call(this, status); }); | |
1459 // return callback.call(this, this._append_html_data(obj, $(d))); | |
1460 } | |
1461 this._data.core.last_error = { 'error' : 'ajax', 'plugin' : 'core', 'id' : 'core_04', 'reason' : 'Could not load node', 'data' : JSON.stringify({ 'id' : obj.id, 'xhr' : x }) }; | |
1462 this.settings.core.error.call(this, this._data.core.last_error); | |
1463 return callback.call(this, false); | |
1464 }, this)) | |
1465 .fail($.proxy(function (f) { | |
1466 this._data.core.last_error = { 'error' : 'ajax', 'plugin' : 'core', 'id' : 'core_04', 'reason' : 'Could not load node', 'data' : JSON.stringify({ 'id' : obj.id, 'xhr' : f }) }; | |
1467 callback.call(this, false); | |
1468 this.settings.core.error.call(this, this._data.core.last_error); | |
1469 }, this)); | |
1470 } | |
1471 if ($.isArray(s)) { | |
1472 t = $.extend(true, [], s); | |
1473 } else if ($.isPlainObject(s)) { | |
1474 t = $.extend(true, {}, s); | |
1475 } else { | |
1476 t = s; | |
1477 } | |
1478 if(obj.id === $.jstree.root) { | |
1479 return this._append_json_data(obj, t, function (status) { | |
1480 callback.call(this, status); | |
1481 }); | |
1482 } | |
1483 else { | |
1484 this._data.core.last_error = { 'error' : 'nodata', 'plugin' : 'core', 'id' : 'core_05', 'reason' : 'Could not load node', 'data' : JSON.stringify({ 'id' : obj.id }) }; | |
1485 this.settings.core.error.call(this, this._data.core.last_error); | |
1486 return callback.call(this, false); | |
1487 } | |
1488 //return callback.call(this, (obj.id === $.jstree.root ? this._append_json_data(obj, t) : false) ); | |
1489 } | |
1490 if(typeof s === 'string') { | |
1491 if(obj.id === $.jstree.root) { | |
1492 return this._append_html_data(obj, $($.parseHTML(s)).filter(notTextOrCommentNode), function (status) { | |
1493 callback.call(this, status); | |
1494 }); | |
1495 } | |
1496 else { | |
1497 this._data.core.last_error = { 'error' : 'nodata', 'plugin' : 'core', 'id' : 'core_06', 'reason' : 'Could not load node', 'data' : JSON.stringify({ 'id' : obj.id }) }; | |
1498 this.settings.core.error.call(this, this._data.core.last_error); | |
1499 return callback.call(this, false); | |
1500 } | |
1501 //return callback.call(this, (obj.id === $.jstree.root ? this._append_html_data(obj, $(s)) : false) ); | |
1502 } | |
1503 return callback.call(this, false); | |
1504 }, | |
1505 /** | |
1506 * adds a node to the list of nodes to redraw. Used only internally. | |
1507 * @private | |
1508 * @name _node_changed(obj [, callback]) | |
1509 * @param {mixed} obj | |
1510 */ | |
1511 _node_changed : function (obj) { | |
1512 obj = this.get_node(obj); | |
1513 if (obj && $.inArray(obj.id, this._model.changed) === -1) { | |
1514 this._model.changed.push(obj.id); | |
1515 } | |
1516 }, | |
1517 /** | |
1518 * appends HTML content to the tree. Used internally. | |
1519 * @private | |
1520 * @name _append_html_data(obj, data) | |
1521 * @param {mixed} obj the node to append to | |
1522 * @param {String} data the HTML string to parse and append | |
1523 * @trigger model.jstree, changed.jstree | |
1524 */ | |
1525 _append_html_data : function (dom, data, cb) { | |
1526 dom = this.get_node(dom); | |
1527 dom.children = []; | |
1528 dom.children_d = []; | |
1529 var dat = data.is('ul') ? data.children() : data, | |
1530 par = dom.id, | |
1531 chd = [], | |
1532 dpc = [], | |
1533 m = this._model.data, | |
1534 p = m[par], | |
1535 s = this._data.core.selected.length, | |
1536 tmp, i, j; | |
1537 dat.each($.proxy(function (i, v) { | |
1538 tmp = this._parse_model_from_html($(v), par, p.parents.concat()); | |
1539 if(tmp) { | |
1540 chd.push(tmp); | |
1541 dpc.push(tmp); | |
1542 if(m[tmp].children_d.length) { | |
1543 dpc = dpc.concat(m[tmp].children_d); | |
1544 } | |
1545 } | |
1546 }, this)); | |
1547 p.children = chd; | |
1548 p.children_d = dpc; | |
1549 for(i = 0, j = p.parents.length; i < j; i++) { | |
1550 m[p.parents[i]].children_d = m[p.parents[i]].children_d.concat(dpc); | |
1551 } | |
1552 /** | |
1553 * triggered when new data is inserted to the tree model | |
1554 * @event | |
1555 * @name model.jstree | |
1556 * @param {Array} nodes an array of node IDs | |
1557 * @param {String} parent the parent ID of the nodes | |
1558 */ | |
1559 this.trigger('model', { "nodes" : dpc, 'parent' : par }); | |
1560 if(par !== $.jstree.root) { | |
1561 this._node_changed(par); | |
1562 this.redraw(); | |
1563 } | |
1564 else { | |
1565 this.get_container_ul().children('.jstree-initial-node').remove(); | |
1566 this.redraw(true); | |
1567 } | |
1568 if(this._data.core.selected.length !== s) { | |
1569 this.trigger('changed', { 'action' : 'model', 'selected' : this._data.core.selected }); | |
1570 } | |
1571 cb.call(this, true); | |
1572 }, | |
1573 /** | |
1574 * appends JSON content to the tree. Used internally. | |
1575 * @private | |
1576 * @name _append_json_data(obj, data) | |
1577 * @param {mixed} obj the node to append to | |
1578 * @param {String} data the JSON object to parse and append | |
1579 * @param {Boolean} force_processing internal param - do not set | |
1580 * @trigger model.jstree, changed.jstree | |
1581 */ | |
1582 _append_json_data : function (dom, data, cb, force_processing) { | |
1583 if(this.element === null) { return; } | |
1584 dom = this.get_node(dom); | |
1585 dom.children = []; | |
1586 dom.children_d = []; | |
1587 // *%$@!!! | |
1588 if(data.d) { | |
1589 data = data.d; | |
1590 if(typeof data === "string") { | |
1591 data = JSON.parse(data); | |
1592 } | |
1593 } | |
1594 if(!$.isArray(data)) { data = [data]; } | |
1595 var w = null, | |
1596 args = { | |
1597 'df' : this._model.default_state, | |
1598 'dat' : data, | |
1599 'par' : dom.id, | |
1600 'm' : this._model.data, | |
1601 't_id' : this._id, | |
1602 't_cnt' : this._cnt, | |
1603 'sel' : this._data.core.selected | |
1604 }, | |
1605 func = function (data, undefined) { | |
1606 if(data.data) { data = data.data; } | |
1607 var dat = data.dat, | |
1608 par = data.par, | |
1609 chd = [], | |
1610 dpc = [], | |
1611 add = [], | |
1612 df = data.df, | |
1613 t_id = data.t_id, | |
1614 t_cnt = data.t_cnt, | |
1615 m = data.m, | |
1616 p = m[par], | |
1617 sel = data.sel, | |
1618 tmp, i, j, rslt, | |
1619 parse_flat = function (d, p, ps) { | |
1620 if(!ps) { ps = []; } | |
1621 else { ps = ps.concat(); } | |
1622 if(p) { ps.unshift(p); } | |
1623 var tid = d.id.toString(), | |
1624 i, j, c, e, | |
1625 tmp = { | |
1626 id : tid, | |
1627 text : d.text || '', | |
1628 icon : d.icon !== undefined ? d.icon : true, | |
1629 parent : p, | |
1630 parents : ps, | |
1631 children : d.children || [], | |
1632 children_d : d.children_d || [], | |
1633 data : d.data, | |
1634 state : { }, | |
1635 li_attr : { id : false }, | |
1636 a_attr : { href : '#' }, | |
1637 original : false | |
1638 }; | |
1639 for(i in df) { | |
1640 if(df.hasOwnProperty(i)) { | |
1641 tmp.state[i] = df[i]; | |
1642 } | |
1643 } | |
1644 if(d && d.data && d.data.jstree && d.data.jstree.icon) { | |
1645 tmp.icon = d.data.jstree.icon; | |
1646 } | |
1647 if(tmp.icon === undefined || tmp.icon === null || tmp.icon === "") { | |
1648 tmp.icon = true; | |
1649 } | |
1650 if(d && d.data) { | |
1651 tmp.data = d.data; | |
1652 if(d.data.jstree) { | |
1653 for(i in d.data.jstree) { | |
1654 if(d.data.jstree.hasOwnProperty(i)) { | |
1655 tmp.state[i] = d.data.jstree[i]; | |
1656 } | |
1657 } | |
1658 } | |
1659 } | |
1660 if(d && typeof d.state === 'object') { | |
1661 for (i in d.state) { | |
1662 if(d.state.hasOwnProperty(i)) { | |
1663 tmp.state[i] = d.state[i]; | |
1664 } | |
1665 } | |
1666 } | |
1667 if(d && typeof d.li_attr === 'object') { | |
1668 for (i in d.li_attr) { | |
1669 if(d.li_attr.hasOwnProperty(i)) { | |
1670 tmp.li_attr[i] = d.li_attr[i]; | |
1671 } | |
1672 } | |
1673 } | |
1674 if(!tmp.li_attr.id) { | |
1675 tmp.li_attr.id = tid; | |
1676 } | |
1677 if(d && typeof d.a_attr === 'object') { | |
1678 for (i in d.a_attr) { | |
1679 if(d.a_attr.hasOwnProperty(i)) { | |
1680 tmp.a_attr[i] = d.a_attr[i]; | |
1681 } | |
1682 } | |
1683 } | |
1684 if(d && d.children && d.children === true) { | |
1685 tmp.state.loaded = false; | |
1686 tmp.children = []; | |
1687 tmp.children_d = []; | |
1688 } | |
1689 m[tmp.id] = tmp; | |
1690 for(i = 0, j = tmp.children.length; i < j; i++) { | |
1691 c = parse_flat(m[tmp.children[i]], tmp.id, ps); | |
1692 e = m[c]; | |
1693 tmp.children_d.push(c); | |
1694 if(e.children_d.length) { | |
1695 tmp.children_d = tmp.children_d.concat(e.children_d); | |
1696 } | |
1697 } | |
1698 delete d.data; | |
1699 delete d.children; | |
1700 m[tmp.id].original = d; | |
1701 if(tmp.state.selected) { | |
1702 add.push(tmp.id); | |
1703 } | |
1704 return tmp.id; | |
1705 }, | |
1706 parse_nest = function (d, p, ps) { | |
1707 if(!ps) { ps = []; } | |
1708 else { ps = ps.concat(); } | |
1709 if(p) { ps.unshift(p); } | |
1710 var tid = false, i, j, c, e, tmp; | |
1711 do { | |
1712 tid = 'j' + t_id + '_' + (++t_cnt); | |
1713 } while(m[tid]); | |
1714 | |
1715 tmp = { | |
1716 id : false, | |
1717 text : typeof d === 'string' ? d : '', | |
1718 icon : typeof d === 'object' && d.icon !== undefined ? d.icon : true, | |
1719 parent : p, | |
1720 parents : ps, | |
1721 children : [], | |
1722 children_d : [], | |
1723 data : null, | |
1724 state : { }, | |
1725 li_attr : { id : false }, | |
1726 a_attr : { href : '#' }, | |
1727 original : false | |
1728 }; | |
1729 for(i in df) { | |
1730 if(df.hasOwnProperty(i)) { | |
1731 tmp.state[i] = df[i]; | |
1732 } | |
1733 } | |
1734 if(d && d.id) { tmp.id = d.id.toString(); } | |
1735 if(d && d.text) { tmp.text = d.text; } | |
1736 if(d && d.data && d.data.jstree && d.data.jstree.icon) { | |
1737 tmp.icon = d.data.jstree.icon; | |
1738 } | |
1739 if(tmp.icon === undefined || tmp.icon === null || tmp.icon === "") { | |
1740 tmp.icon = true; | |
1741 } | |
1742 if(d && d.data) { | |
1743 tmp.data = d.data; | |
1744 if(d.data.jstree) { | |
1745 for(i in d.data.jstree) { | |
1746 if(d.data.jstree.hasOwnProperty(i)) { | |
1747 tmp.state[i] = d.data.jstree[i]; | |
1748 } | |
1749 } | |
1750 } | |
1751 } | |
1752 if(d && typeof d.state === 'object') { | |
1753 for (i in d.state) { | |
1754 if(d.state.hasOwnProperty(i)) { | |
1755 tmp.state[i] = d.state[i]; | |
1756 } | |
1757 } | |
1758 } | |
1759 if(d && typeof d.li_attr === 'object') { | |
1760 for (i in d.li_attr) { | |
1761 if(d.li_attr.hasOwnProperty(i)) { | |
1762 tmp.li_attr[i] = d.li_attr[i]; | |
1763 } | |
1764 } | |
1765 } | |
1766 if(tmp.li_attr.id && !tmp.id) { | |
1767 tmp.id = tmp.li_attr.id.toString(); | |
1768 } | |
1769 if(!tmp.id) { | |
1770 tmp.id = tid; | |
1771 } | |
1772 if(!tmp.li_attr.id) { | |
1773 tmp.li_attr.id = tmp.id; | |
1774 } | |
1775 if(d && typeof d.a_attr === 'object') { | |
1776 for (i in d.a_attr) { | |
1777 if(d.a_attr.hasOwnProperty(i)) { | |
1778 tmp.a_attr[i] = d.a_attr[i]; | |
1779 } | |
1780 } | |
1781 } | |
1782 if(d && d.children && d.children.length) { | |
1783 for(i = 0, j = d.children.length; i < j; i++) { | |
1784 c = parse_nest(d.children[i], tmp.id, ps); | |
1785 e = m[c]; | |
1786 tmp.children.push(c); | |
1787 if(e.children_d.length) { | |
1788 tmp.children_d = tmp.children_d.concat(e.children_d); | |
1789 } | |
1790 } | |
1791 tmp.children_d = tmp.children_d.concat(tmp.children); | |
1792 } | |
1793 if(d && d.children && d.children === true) { | |
1794 tmp.state.loaded = false; | |
1795 tmp.children = []; | |
1796 tmp.children_d = []; | |
1797 } | |
1798 delete d.data; | |
1799 delete d.children; | |
1800 tmp.original = d; | |
1801 m[tmp.id] = tmp; | |
1802 if(tmp.state.selected) { | |
1803 add.push(tmp.id); | |
1804 } | |
1805 return tmp.id; | |
1806 }; | |
1807 | |
1808 if(dat.length && dat[0].id !== undefined && dat[0].parent !== undefined) { | |
1809 // Flat JSON support (for easy import from DB): | |
1810 // 1) convert to object (foreach) | |
1811 for(i = 0, j = dat.length; i < j; i++) { | |
1812 if(!dat[i].children) { | |
1813 dat[i].children = []; | |
1814 } | |
1815 if(!dat[i].state) { | |
1816 dat[i].state = {}; | |
1817 } | |
1818 m[dat[i].id.toString()] = dat[i]; | |
1819 } | |
1820 // 2) populate children (foreach) | |
1821 for(i = 0, j = dat.length; i < j; i++) { | |
1822 if (!m[dat[i].parent.toString()]) { | |
1823 this._data.core.last_error = { 'error' : 'parse', 'plugin' : 'core', 'id' : 'core_07', 'reason' : 'Node with invalid parent', 'data' : JSON.stringify({ 'id' : dat[i].id.toString(), 'parent' : dat[i].parent.toString() }) }; | |
1824 this.settings.core.error.call(this, this._data.core.last_error); | |
1825 continue; | |
1826 } | |
1827 | |
1828 m[dat[i].parent.toString()].children.push(dat[i].id.toString()); | |
1829 // populate parent.children_d | |
1830 p.children_d.push(dat[i].id.toString()); | |
1831 } | |
1832 // 3) normalize && populate parents and children_d with recursion | |
1833 for(i = 0, j = p.children.length; i < j; i++) { | |
1834 tmp = parse_flat(m[p.children[i]], par, p.parents.concat()); | |
1835 dpc.push(tmp); | |
1836 if(m[tmp].children_d.length) { | |
1837 dpc = dpc.concat(m[tmp].children_d); | |
1838 } | |
1839 } | |
1840 for(i = 0, j = p.parents.length; i < j; i++) { | |
1841 m[p.parents[i]].children_d = m[p.parents[i]].children_d.concat(dpc); | |
1842 } | |
1843 // ?) three_state selection - p.state.selected && t - (if three_state foreach(dat => ch) -> foreach(parents) if(parent.selected) child.selected = true; | |
1844 rslt = { | |
1845 'cnt' : t_cnt, | |
1846 'mod' : m, | |
1847 'sel' : sel, | |
1848 'par' : par, | |
1849 'dpc' : dpc, | |
1850 'add' : add | |
1851 }; | |
1852 } | |
1853 else { | |
1854 for(i = 0, j = dat.length; i < j; i++) { | |
1855 tmp = parse_nest(dat[i], par, p.parents.concat()); | |
1856 if(tmp) { | |
1857 chd.push(tmp); | |
1858 dpc.push(tmp); | |
1859 if(m[tmp].children_d.length) { | |
1860 dpc = dpc.concat(m[tmp].children_d); | |
1861 } | |
1862 } | |
1863 } | |
1864 p.children = chd; | |
1865 p.children_d = dpc; | |
1866 for(i = 0, j = p.parents.length; i < j; i++) { | |
1867 m[p.parents[i]].children_d = m[p.parents[i]].children_d.concat(dpc); | |
1868 } | |
1869 rslt = { | |
1870 'cnt' : t_cnt, | |
1871 'mod' : m, | |
1872 'sel' : sel, | |
1873 'par' : par, | |
1874 'dpc' : dpc, | |
1875 'add' : add | |
1876 }; | |
1877 } | |
1878 if(typeof window === 'undefined' || typeof window.document === 'undefined') { | |
1879 postMessage(rslt); | |
1880 } | |
1881 else { | |
1882 return rslt; | |
1883 } | |
1884 }, | |
1885 rslt = function (rslt, worker) { | |
1886 if(this.element === null) { return; } | |
1887 this._cnt = rslt.cnt; | |
1888 var i, m = this._model.data; | |
1889 for (i in m) { | |
1890 if (m.hasOwnProperty(i) && m[i].state && m[i].state.loading && rslt.mod[i]) { | |
1891 rslt.mod[i].state.loading = true; | |
1892 } | |
1893 } | |
1894 this._model.data = rslt.mod; // breaks the reference in load_node - careful | |
1895 | |
1896 if(worker) { | |
1897 var j, a = rslt.add, r = rslt.sel, s = this._data.core.selected.slice(); | |
1898 m = this._model.data; | |
1899 // if selection was changed while calculating in worker | |
1900 if(r.length !== s.length || $.vakata.array_unique(r.concat(s)).length !== r.length) { | |
1901 // deselect nodes that are no longer selected | |
1902 for(i = 0, j = r.length; i < j; i++) { | |
1903 if($.inArray(r[i], a) === -1 && $.inArray(r[i], s) === -1) { | |
1904 m[r[i]].state.selected = false; | |
1905 } | |
1906 } | |
1907 // select nodes that were selected in the mean time | |
1908 for(i = 0, j = s.length; i < j; i++) { | |
1909 if($.inArray(s[i], r) === -1) { | |
1910 m[s[i]].state.selected = true; | |
1911 } | |
1912 } | |
1913 } | |
1914 } | |
1915 if(rslt.add.length) { | |
1916 this._data.core.selected = this._data.core.selected.concat(rslt.add); | |
1917 } | |
1918 | |
1919 this.trigger('model', { "nodes" : rslt.dpc, 'parent' : rslt.par }); | |
1920 | |
1921 if(rslt.par !== $.jstree.root) { | |
1922 this._node_changed(rslt.par); | |
1923 this.redraw(); | |
1924 } | |
1925 else { | |
1926 // this.get_container_ul().children('.jstree-initial-node').remove(); | |
1927 this.redraw(true); | |
1928 } | |
1929 if(rslt.add.length) { | |
1930 this.trigger('changed', { 'action' : 'model', 'selected' : this._data.core.selected }); | |
1931 } | |
1932 cb.call(this, true); | |
1933 }; | |
1934 if(this.settings.core.worker && window.Blob && window.URL && window.Worker) { | |
1935 try { | |
1936 if(this._wrk === null) { | |
1937 this._wrk = window.URL.createObjectURL( | |
1938 new window.Blob( | |
1939 ['self.onmessage = ' + func.toString()], | |
1940 {type:"text/javascript"} | |
1941 ) | |
1942 ); | |
1943 } | |
1944 if(!this._data.core.working || force_processing) { | |
1945 this._data.core.working = true; | |
1946 w = new window.Worker(this._wrk); | |
1947 w.onmessage = $.proxy(function (e) { | |
1948 rslt.call(this, e.data, true); | |
1949 try { w.terminate(); w = null; } catch(ignore) { } | |
1950 if(this._data.core.worker_queue.length) { | |
1951 this._append_json_data.apply(this, this._data.core.worker_queue.shift()); | |
1952 } | |
1953 else { | |
1954 this._data.core.working = false; | |
1955 } | |
1956 }, this); | |
1957 if(!args.par) { | |
1958 if(this._data.core.worker_queue.length) { | |
1959 this._append_json_data.apply(this, this._data.core.worker_queue.shift()); | |
1960 } | |
1961 else { | |
1962 this._data.core.working = false; | |
1963 } | |
1964 } | |
1965 else { | |
1966 w.postMessage(args); | |
1967 } | |
1968 } | |
1969 else { | |
1970 this._data.core.worker_queue.push([dom, data, cb, true]); | |
1971 } | |
1972 } | |
1973 catch(e) { | |
1974 rslt.call(this, func(args), false); | |
1975 if(this._data.core.worker_queue.length) { | |
1976 this._append_json_data.apply(this, this._data.core.worker_queue.shift()); | |
1977 } | |
1978 else { | |
1979 this._data.core.working = false; | |
1980 } | |
1981 } | |
1982 } | |
1983 else { | |
1984 rslt.call(this, func(args), false); | |
1985 } | |
1986 }, | |
1987 /** | |
1988 * parses a node from a jQuery object and appends them to the in memory tree model. Used internally. | |
1989 * @private | |
1990 * @name _parse_model_from_html(d [, p, ps]) | |
1991 * @param {jQuery} d the jQuery object to parse | |
1992 * @param {String} p the parent ID | |
1993 * @param {Array} ps list of all parents | |
1994 * @return {String} the ID of the object added to the model | |
1995 */ | |
1996 _parse_model_from_html : function (d, p, ps) { | |
1997 if(!ps) { ps = []; } | |
1998 else { ps = [].concat(ps); } | |
1999 if(p) { ps.unshift(p); } | |
2000 var c, e, m = this._model.data, | |
2001 data = { | |
2002 id : false, | |
2003 text : false, | |
2004 icon : true, | |
2005 parent : p, | |
2006 parents : ps, | |
2007 children : [], | |
2008 children_d : [], | |
2009 data : null, | |
2010 state : { }, | |
2011 li_attr : { id : false }, | |
2012 a_attr : { href : '#' }, | |
2013 original : false | |
2014 }, i, tmp, tid; | |
2015 for(i in this._model.default_state) { | |
2016 if(this._model.default_state.hasOwnProperty(i)) { | |
2017 data.state[i] = this._model.default_state[i]; | |
2018 } | |
2019 } | |
2020 tmp = $.vakata.attributes(d, true); | |
2021 $.each(tmp, function (i, v) { | |
2022 v = $.trim(v); | |
2023 if(!v.length) { return true; } | |
2024 data.li_attr[i] = v; | |
2025 if(i === 'id') { | |
2026 data.id = v.toString(); | |
2027 } | |
2028 }); | |
2029 tmp = d.children('a').first(); | |
2030 if(tmp.length) { | |
2031 tmp = $.vakata.attributes(tmp, true); | |
2032 $.each(tmp, function (i, v) { | |
2033 v = $.trim(v); | |
2034 if(v.length) { | |
2035 data.a_attr[i] = v; | |
2036 } | |
2037 }); | |
2038 } | |
2039 tmp = d.children("a").first().length ? d.children("a").first().clone() : d.clone(); | |
2040 tmp.children("ins, i, ul").remove(); | |
2041 tmp = tmp.html(); | |
2042 tmp = $('<div />').html(tmp); | |
2043 data.text = this.settings.core.force_text ? tmp.text() : tmp.html(); | |
2044 tmp = d.data(); | |
2045 data.data = tmp ? $.extend(true, {}, tmp) : null; | |
2046 data.state.opened = d.hasClass('jstree-open'); | |
2047 data.state.selected = d.children('a').hasClass('jstree-clicked'); | |
2048 data.state.disabled = d.children('a').hasClass('jstree-disabled'); | |
2049 if(data.data && data.data.jstree) { | |
2050 for(i in data.data.jstree) { | |
2051 if(data.data.jstree.hasOwnProperty(i)) { | |
2052 data.state[i] = data.data.jstree[i]; | |
2053 } | |
2054 } | |
2055 } | |
2056 tmp = d.children("a").children(".jstree-themeicon"); | |
2057 if(tmp.length) { | |
2058 data.icon = tmp.hasClass('jstree-themeicon-hidden') ? false : tmp.attr('rel'); | |
2059 } | |
2060 if(data.state.icon !== undefined) { | |
2061 data.icon = data.state.icon; | |
2062 } | |
2063 if(data.icon === undefined || data.icon === null || data.icon === "") { | |
2064 data.icon = true; | |
2065 } | |
2066 tmp = d.children("ul").children("li"); | |
2067 do { | |
2068 tid = 'j' + this._id + '_' + (++this._cnt); | |
2069 } while(m[tid]); | |
2070 data.id = data.li_attr.id ? data.li_attr.id.toString() : tid; | |
2071 if(tmp.length) { | |
2072 tmp.each($.proxy(function (i, v) { | |
2073 c = this._parse_model_from_html($(v), data.id, ps); | |
2074 e = this._model.data[c]; | |
2075 data.children.push(c); | |
2076 if(e.children_d.length) { | |
2077 data.children_d = data.children_d.concat(e.children_d); | |
2078 } | |
2079 }, this)); | |
2080 data.children_d = data.children_d.concat(data.children); | |
2081 } | |
2082 else { | |
2083 if(d.hasClass('jstree-closed')) { | |
2084 data.state.loaded = false; | |
2085 } | |
2086 } | |
2087 if(data.li_attr['class']) { | |
2088 data.li_attr['class'] = data.li_attr['class'].replace('jstree-closed','').replace('jstree-open',''); | |
2089 } | |
2090 if(data.a_attr['class']) { | |
2091 data.a_attr['class'] = data.a_attr['class'].replace('jstree-clicked','').replace('jstree-disabled',''); | |
2092 } | |
2093 m[data.id] = data; | |
2094 if(data.state.selected) { | |
2095 this._data.core.selected.push(data.id); | |
2096 } | |
2097 return data.id; | |
2098 }, | |
2099 /** | |
2100 * parses a node from a JSON object (used when dealing with flat data, which has no nesting of children, but has id and parent properties) and appends it to the in memory tree model. Used internally. | |
2101 * @private | |
2102 * @name _parse_model_from_flat_json(d [, p, ps]) | |
2103 * @param {Object} d the JSON object to parse | |
2104 * @param {String} p the parent ID | |
2105 * @param {Array} ps list of all parents | |
2106 * @return {String} the ID of the object added to the model | |
2107 */ | |
2108 _parse_model_from_flat_json : function (d, p, ps) { | |
2109 if(!ps) { ps = []; } | |
2110 else { ps = ps.concat(); } | |
2111 if(p) { ps.unshift(p); } | |
2112 var tid = d.id.toString(), | |
2113 m = this._model.data, | |
2114 df = this._model.default_state, | |
2115 i, j, c, e, | |
2116 tmp = { | |
2117 id : tid, | |
2118 text : d.text || '', | |
2119 icon : d.icon !== undefined ? d.icon : true, | |
2120 parent : p, | |
2121 parents : ps, | |
2122 children : d.children || [], | |
2123 children_d : d.children_d || [], | |
2124 data : d.data, | |
2125 state : { }, | |
2126 li_attr : { id : false }, | |
2127 a_attr : { href : '#' }, | |
2128 original : false | |
2129 }; | |
2130 for(i in df) { | |
2131 if(df.hasOwnProperty(i)) { | |
2132 tmp.state[i] = df[i]; | |
2133 } | |
2134 } | |
2135 if(d && d.data && d.data.jstree && d.data.jstree.icon) { | |
2136 tmp.icon = d.data.jstree.icon; | |
2137 } | |
2138 if(tmp.icon === undefined || tmp.icon === null || tmp.icon === "") { | |
2139 tmp.icon = true; | |
2140 } | |
2141 if(d && d.data) { | |
2142 tmp.data = d.data; | |
2143 if(d.data.jstree) { | |
2144 for(i in d.data.jstree) { | |
2145 if(d.data.jstree.hasOwnProperty(i)) { | |
2146 tmp.state[i] = d.data.jstree[i]; | |
2147 } | |
2148 } | |
2149 } | |
2150 } | |
2151 if(d && typeof d.state === 'object') { | |
2152 for (i in d.state) { | |
2153 if(d.state.hasOwnProperty(i)) { | |
2154 tmp.state[i] = d.state[i]; | |
2155 } | |
2156 } | |
2157 } | |
2158 if(d && typeof d.li_attr === 'object') { | |
2159 for (i in d.li_attr) { | |
2160 if(d.li_attr.hasOwnProperty(i)) { | |
2161 tmp.li_attr[i] = d.li_attr[i]; | |
2162 } | |
2163 } | |
2164 } | |
2165 if(!tmp.li_attr.id) { | |
2166 tmp.li_attr.id = tid; | |
2167 } | |
2168 if(d && typeof d.a_attr === 'object') { | |
2169 for (i in d.a_attr) { | |
2170 if(d.a_attr.hasOwnProperty(i)) { | |
2171 tmp.a_attr[i] = d.a_attr[i]; | |
2172 } | |
2173 } | |
2174 } | |
2175 if(d && d.children && d.children === true) { | |
2176 tmp.state.loaded = false; | |
2177 tmp.children = []; | |
2178 tmp.children_d = []; | |
2179 } | |
2180 m[tmp.id] = tmp; | |
2181 for(i = 0, j = tmp.children.length; i < j; i++) { | |
2182 c = this._parse_model_from_flat_json(m[tmp.children[i]], tmp.id, ps); | |
2183 e = m[c]; | |
2184 tmp.children_d.push(c); | |
2185 if(e.children_d.length) { | |
2186 tmp.children_d = tmp.children_d.concat(e.children_d); | |
2187 } | |
2188 } | |
2189 delete d.data; | |
2190 delete d.children; | |
2191 m[tmp.id].original = d; | |
2192 if(tmp.state.selected) { | |
2193 this._data.core.selected.push(tmp.id); | |
2194 } | |
2195 return tmp.id; | |
2196 }, | |
2197 /** | |
2198 * parses a node from a JSON object and appends it to the in memory tree model. Used internally. | |
2199 * @private | |
2200 * @name _parse_model_from_json(d [, p, ps]) | |
2201 * @param {Object} d the JSON object to parse | |
2202 * @param {String} p the parent ID | |
2203 * @param {Array} ps list of all parents | |
2204 * @return {String} the ID of the object added to the model | |
2205 */ | |
2206 _parse_model_from_json : function (d, p, ps) { | |
2207 if(!ps) { ps = []; } | |
2208 else { ps = ps.concat(); } | |
2209 if(p) { ps.unshift(p); } | |
2210 var tid = false, i, j, c, e, m = this._model.data, df = this._model.default_state, tmp; | |
2211 do { | |
2212 tid = 'j' + this._id + '_' + (++this._cnt); | |
2213 } while(m[tid]); | |
2214 | |
2215 tmp = { | |
2216 id : false, | |
2217 text : typeof d === 'string' ? d : '', | |
2218 icon : typeof d === 'object' && d.icon !== undefined ? d.icon : true, | |
2219 parent : p, | |
2220 parents : ps, | |
2221 children : [], | |
2222 children_d : [], | |
2223 data : null, | |
2224 state : { }, | |
2225 li_attr : { id : false }, | |
2226 a_attr : { href : '#' }, | |
2227 original : false | |
2228 }; | |
2229 for(i in df) { | |
2230 if(df.hasOwnProperty(i)) { | |
2231 tmp.state[i] = df[i]; | |
2232 } | |
2233 } | |
2234 if(d && d.id) { tmp.id = d.id.toString(); } | |
2235 if(d && d.text) { tmp.text = d.text; } | |
2236 if(d && d.data && d.data.jstree && d.data.jstree.icon) { | |
2237 tmp.icon = d.data.jstree.icon; | |
2238 } | |
2239 if(tmp.icon === undefined || tmp.icon === null || tmp.icon === "") { | |
2240 tmp.icon = true; | |
2241 } | |
2242 if(d && d.data) { | |
2243 tmp.data = d.data; | |
2244 if(d.data.jstree) { | |
2245 for(i in d.data.jstree) { | |
2246 if(d.data.jstree.hasOwnProperty(i)) { | |
2247 tmp.state[i] = d.data.jstree[i]; | |
2248 } | |
2249 } | |
2250 } | |
2251 } | |
2252 if(d && typeof d.state === 'object') { | |
2253 for (i in d.state) { | |
2254 if(d.state.hasOwnProperty(i)) { | |
2255 tmp.state[i] = d.state[i]; | |
2256 } | |
2257 } | |
2258 } | |
2259 if(d && typeof d.li_attr === 'object') { | |
2260 for (i in d.li_attr) { | |
2261 if(d.li_attr.hasOwnProperty(i)) { | |
2262 tmp.li_attr[i] = d.li_attr[i]; | |
2263 } | |
2264 } | |
2265 } | |
2266 if(tmp.li_attr.id && !tmp.id) { | |
2267 tmp.id = tmp.li_attr.id.toString(); | |
2268 } | |
2269 if(!tmp.id) { | |
2270 tmp.id = tid; | |
2271 } | |
2272 if(!tmp.li_attr.id) { | |
2273 tmp.li_attr.id = tmp.id; | |
2274 } | |
2275 if(d && typeof d.a_attr === 'object') { | |
2276 for (i in d.a_attr) { | |
2277 if(d.a_attr.hasOwnProperty(i)) { | |
2278 tmp.a_attr[i] = d.a_attr[i]; | |
2279 } | |
2280 } | |
2281 } | |
2282 if(d && d.children && d.children.length) { | |
2283 for(i = 0, j = d.children.length; i < j; i++) { | |
2284 c = this._parse_model_from_json(d.children[i], tmp.id, ps); | |
2285 e = m[c]; | |
2286 tmp.children.push(c); | |
2287 if(e.children_d.length) { | |
2288 tmp.children_d = tmp.children_d.concat(e.children_d); | |
2289 } | |
2290 } | |
2291 tmp.children_d = tmp.children_d.concat(tmp.children); | |
2292 } | |
2293 if(d && d.children && d.children === true) { | |
2294 tmp.state.loaded = false; | |
2295 tmp.children = []; | |
2296 tmp.children_d = []; | |
2297 } | |
2298 delete d.data; | |
2299 delete d.children; | |
2300 tmp.original = d; | |
2301 m[tmp.id] = tmp; | |
2302 if(tmp.state.selected) { | |
2303 this._data.core.selected.push(tmp.id); | |
2304 } | |
2305 return tmp.id; | |
2306 }, | |
2307 /** | |
2308 * redraws all nodes that need to be redrawn. Used internally. | |
2309 * @private | |
2310 * @name _redraw() | |
2311 * @trigger redraw.jstree | |
2312 */ | |
2313 _redraw : function () { | |
2314 var nodes = this._model.force_full_redraw ? this._model.data[$.jstree.root].children.concat([]) : this._model.changed.concat([]), | |
2315 f = document.createElement('UL'), tmp, i, j, fe = this._data.core.focused; | |
2316 for(i = 0, j = nodes.length; i < j; i++) { | |
2317 tmp = this.redraw_node(nodes[i], true, this._model.force_full_redraw); | |
2318 if(tmp && this._model.force_full_redraw) { | |
2319 f.appendChild(tmp); | |
2320 } | |
2321 } | |
2322 if(this._model.force_full_redraw) { | |
2323 f.className = this.get_container_ul()[0].className; | |
2324 f.setAttribute('role','group'); | |
2325 this.element.empty().append(f); | |
2326 //this.get_container_ul()[0].appendChild(f); | |
2327 } | |
2328 if(fe !== null) { | |
2329 tmp = this.get_node(fe, true); | |
2330 if(tmp && tmp.length && tmp.children('.jstree-anchor')[0] !== document.activeElement) { | |
2331 tmp.children('.jstree-anchor').focus(); | |
2332 } | |
2333 else { | |
2334 this._data.core.focused = null; | |
2335 } | |
2336 } | |
2337 this._model.force_full_redraw = false; | |
2338 this._model.changed = []; | |
2339 /** | |
2340 * triggered after nodes are redrawn | |
2341 * @event | |
2342 * @name redraw.jstree | |
2343 * @param {array} nodes the redrawn nodes | |
2344 */ | |
2345 this.trigger('redraw', { "nodes" : nodes }); | |
2346 }, | |
2347 /** | |
2348 * redraws all nodes that need to be redrawn or optionally - the whole tree | |
2349 * @name redraw([full]) | |
2350 * @param {Boolean} full if set to `true` all nodes are redrawn. | |
2351 */ | |
2352 redraw : function (full) { | |
2353 if(full) { | |
2354 this._model.force_full_redraw = true; | |
2355 } | |
2356 //if(this._model.redraw_timeout) { | |
2357 // clearTimeout(this._model.redraw_timeout); | |
2358 //} | |
2359 //this._model.redraw_timeout = setTimeout($.proxy(this._redraw, this),0); | |
2360 this._redraw(); | |
2361 }, | |
2362 /** | |
2363 * redraws a single node's children. Used internally. | |
2364 * @private | |
2365 * @name draw_children(node) | |
2366 * @param {mixed} node the node whose children will be redrawn | |
2367 */ | |
2368 draw_children : function (node) { | |
2369 var obj = this.get_node(node), | |
2370 i = false, | |
2371 j = false, | |
2372 k = false, | |
2373 d = document; | |
2374 if(!obj) { return false; } | |
2375 if(obj.id === $.jstree.root) { return this.redraw(true); } | |
2376 node = this.get_node(node, true); | |
2377 if(!node || !node.length) { return false; } // TODO: quick toggle | |
2378 | |
2379 node.children('.jstree-children').remove(); | |
2380 node = node[0]; | |
2381 if(obj.children.length && obj.state.loaded) { | |
2382 k = d.createElement('UL'); | |
2383 k.setAttribute('role', 'group'); | |
2384 k.className = 'jstree-children'; | |
2385 for(i = 0, j = obj.children.length; i < j; i++) { | |
2386 k.appendChild(this.redraw_node(obj.children[i], true, true)); | |
2387 } | |
2388 node.appendChild(k); | |
2389 } | |
2390 }, | |
2391 /** | |
2392 * redraws a single node. Used internally. | |
2393 * @private | |
2394 * @name redraw_node(node, deep, is_callback, force_render) | |
2395 * @param {mixed} node the node to redraw | |
2396 * @param {Boolean} deep should child nodes be redrawn too | |
2397 * @param {Boolean} is_callback is this a recursion call | |
2398 * @param {Boolean} force_render should children of closed parents be drawn anyway | |
2399 */ | |
2400 redraw_node : function (node, deep, is_callback, force_render) { | |
2401 var obj = this.get_node(node), | |
2402 par = false, | |
2403 ind = false, | |
2404 old = false, | |
2405 i = false, | |
2406 j = false, | |
2407 k = false, | |
2408 c = '', | |
2409 d = document, | |
2410 m = this._model.data, | |
2411 f = false, | |
2412 s = false, | |
2413 tmp = null, | |
2414 t = 0, | |
2415 l = 0, | |
2416 has_children = false, | |
2417 last_sibling = false; | |
2418 if(!obj) { return false; } | |
2419 if(obj.id === $.jstree.root) { return this.redraw(true); } | |
2420 deep = deep || obj.children.length === 0; | |
2421 node = !document.querySelector ? document.getElementById(obj.id) : this.element[0].querySelector('#' + ("0123456789".indexOf(obj.id[0]) !== -1 ? '\\3' + obj.id[0] + ' ' + obj.id.substr(1).replace($.jstree.idregex,'\\$&') : obj.id.replace($.jstree.idregex,'\\$&')) ); //, this.element); | |
2422 if(!node) { | |
2423 deep = true; | |
2424 //node = d.createElement('LI'); | |
2425 if(!is_callback) { | |
2426 par = obj.parent !== $.jstree.root ? $('#' + obj.parent.replace($.jstree.idregex,'\\$&'), this.element)[0] : null; | |
2427 if(par !== null && (!par || !m[obj.parent].state.opened)) { | |
2428 return false; | |
2429 } | |
2430 ind = $.inArray(obj.id, par === null ? m[$.jstree.root].children : m[obj.parent].children); | |
2431 } | |
2432 } | |
2433 else { | |
2434 node = $(node); | |
2435 if(!is_callback) { | |
2436 par = node.parent().parent()[0]; | |
2437 if(par === this.element[0]) { | |
2438 par = null; | |
2439 } | |
2440 ind = node.index(); | |
2441 } | |
2442 // m[obj.id].data = node.data(); // use only node's data, no need to touch jquery storage | |
2443 if(!deep && obj.children.length && !node.children('.jstree-children').length) { | |
2444 deep = true; | |
2445 } | |
2446 if(!deep) { | |
2447 old = node.children('.jstree-children')[0]; | |
2448 } | |
2449 f = node.children('.jstree-anchor')[0] === document.activeElement; | |
2450 node.remove(); | |
2451 //node = d.createElement('LI'); | |
2452 //node = node[0]; | |
2453 } | |
2454 node = this._data.core.node.cloneNode(true); | |
2455 // node is DOM, deep is boolean | |
2456 | |
2457 c = 'jstree-node '; | |
2458 for(i in obj.li_attr) { | |
2459 if(obj.li_attr.hasOwnProperty(i)) { | |
2460 if(i === 'id') { continue; } | |
2461 if(i !== 'class') { | |
2462 node.setAttribute(i, obj.li_attr[i]); | |
2463 } | |
2464 else { | |
2465 c += obj.li_attr[i]; | |
2466 } | |
2467 } | |
2468 } | |
2469 if(!obj.a_attr.id) { | |
2470 obj.a_attr.id = obj.id + '_anchor'; | |
2471 } | |
2472 node.setAttribute('aria-selected', !!obj.state.selected); | |
2473 node.setAttribute('aria-level', obj.parents.length); | |
2474 node.setAttribute('aria-labelledby', obj.a_attr.id); | |
2475 if(obj.state.disabled) { | |
2476 node.setAttribute('aria-disabled', true); | |
2477 } | |
2478 | |
2479 for(i = 0, j = obj.children.length; i < j; i++) { | |
2480 if(!m[obj.children[i]].state.hidden) { | |
2481 has_children = true; | |
2482 break; | |
2483 } | |
2484 } | |
2485 if(obj.parent !== null && m[obj.parent] && !obj.state.hidden) { | |
2486 i = $.inArray(obj.id, m[obj.parent].children); | |
2487 last_sibling = obj.id; | |
2488 if(i !== -1) { | |
2489 i++; | |
2490 for(j = m[obj.parent].children.length; i < j; i++) { | |
2491 if(!m[m[obj.parent].children[i]].state.hidden) { | |
2492 last_sibling = m[obj.parent].children[i]; | |
2493 } | |
2494 if(last_sibling !== obj.id) { | |
2495 break; | |
2496 } | |
2497 } | |
2498 } | |
2499 } | |
2500 | |
2501 if(obj.state.hidden) { | |
2502 c += ' jstree-hidden'; | |
2503 } | |
2504 if (obj.state.loading) { | |
2505 c += ' jstree-loading'; | |
2506 } | |
2507 if(obj.state.loaded && !has_children) { | |
2508 c += ' jstree-leaf'; | |
2509 } | |
2510 else { | |
2511 c += obj.state.opened && obj.state.loaded ? ' jstree-open' : ' jstree-closed'; | |
2512 node.setAttribute('aria-expanded', (obj.state.opened && obj.state.loaded) ); | |
2513 } | |
2514 if(last_sibling === obj.id) { | |
2515 c += ' jstree-last'; | |
2516 } | |
2517 node.id = obj.id; | |
2518 node.className = c; | |
2519 c = ( obj.state.selected ? ' jstree-clicked' : '') + ( obj.state.disabled ? ' jstree-disabled' : ''); | |
2520 for(j in obj.a_attr) { | |
2521 if(obj.a_attr.hasOwnProperty(j)) { | |
2522 if(j === 'href' && obj.a_attr[j] === '#') { continue; } | |
2523 if(j !== 'class') { | |
2524 node.childNodes[1].setAttribute(j, obj.a_attr[j]); | |
2525 } | |
2526 else { | |
2527 c += ' ' + obj.a_attr[j]; | |
2528 } | |
2529 } | |
2530 } | |
2531 if(c.length) { | |
2532 node.childNodes[1].className = 'jstree-anchor ' + c; | |
2533 } | |
2534 if((obj.icon && obj.icon !== true) || obj.icon === false) { | |
2535 if(obj.icon === false) { | |
2536 node.childNodes[1].childNodes[0].className += ' jstree-themeicon-hidden'; | |
2537 } | |
2538 else if(obj.icon.indexOf('/') === -1 && obj.icon.indexOf('.') === -1) { | |
2539 node.childNodes[1].childNodes[0].className += ' ' + obj.icon + ' jstree-themeicon-custom'; | |
2540 } | |
2541 else { | |
2542 node.childNodes[1].childNodes[0].style.backgroundImage = 'url("'+obj.icon+'")'; | |
2543 node.childNodes[1].childNodes[0].style.backgroundPosition = 'center center'; | |
2544 node.childNodes[1].childNodes[0].style.backgroundSize = 'auto'; | |
2545 node.childNodes[1].childNodes[0].className += ' jstree-themeicon-custom'; | |
2546 } | |
2547 } | |
2548 | |
2549 if(this.settings.core.force_text) { | |
2550 node.childNodes[1].appendChild(d.createTextNode(obj.text)); | |
2551 } | |
2552 else { | |
2553 node.childNodes[1].innerHTML += obj.text; | |
2554 } | |
2555 | |
2556 | |
2557 if(deep && obj.children.length && (obj.state.opened || force_render) && obj.state.loaded) { | |
2558 k = d.createElement('UL'); | |
2559 k.setAttribute('role', 'group'); | |
2560 k.className = 'jstree-children'; | |
2561 for(i = 0, j = obj.children.length; i < j; i++) { | |
2562 k.appendChild(this.redraw_node(obj.children[i], deep, true)); | |
2563 } | |
2564 node.appendChild(k); | |
2565 } | |
2566 if(old) { | |
2567 node.appendChild(old); | |
2568 } | |
2569 if(!is_callback) { | |
2570 // append back using par / ind | |
2571 if(!par) { | |
2572 par = this.element[0]; | |
2573 } | |
2574 for(i = 0, j = par.childNodes.length; i < j; i++) { | |
2575 if(par.childNodes[i] && par.childNodes[i].className && par.childNodes[i].className.indexOf('jstree-children') !== -1) { | |
2576 tmp = par.childNodes[i]; | |
2577 break; | |
2578 } | |
2579 } | |
2580 if(!tmp) { | |
2581 tmp = d.createElement('UL'); | |
2582 tmp.setAttribute('role', 'group'); | |
2583 tmp.className = 'jstree-children'; | |
2584 par.appendChild(tmp); | |
2585 } | |
2586 par = tmp; | |
2587 | |
2588 if(ind < par.childNodes.length) { | |
2589 par.insertBefore(node, par.childNodes[ind]); | |
2590 } | |
2591 else { | |
2592 par.appendChild(node); | |
2593 } | |
2594 if(f) { | |
2595 t = this.element[0].scrollTop; | |
2596 l = this.element[0].scrollLeft; | |
2597 node.childNodes[1].focus(); | |
2598 this.element[0].scrollTop = t; | |
2599 this.element[0].scrollLeft = l; | |
2600 } | |
2601 } | |
2602 if(obj.state.opened && !obj.state.loaded) { | |
2603 obj.state.opened = false; | |
2604 setTimeout($.proxy(function () { | |
2605 this.open_node(obj.id, false, 0); | |
2606 }, this), 0); | |
2607 } | |
2608 return node; | |
2609 }, | |
2610 /** | |
2611 * opens a node, revaling its children. If the node is not loaded it will be loaded and opened once ready. | |
2612 * @name open_node(obj [, callback, animation]) | |
2613 * @param {mixed} obj the node to open | |
2614 * @param {Function} callback a function to execute once the node is opened | |
2615 * @param {Number} animation the animation duration in milliseconds when opening the node (overrides the `core.animation` setting). Use `false` for no animation. | |
2616 * @trigger open_node.jstree, after_open.jstree, before_open.jstree | |
2617 */ | |
2618 open_node : function (obj, callback, animation) { | |
2619 var t1, t2, d, t; | |
2620 if($.isArray(obj)) { | |
2621 obj = obj.slice(); | |
2622 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
2623 this.open_node(obj[t1], callback, animation); | |
2624 } | |
2625 return true; | |
2626 } | |
2627 obj = this.get_node(obj); | |
2628 if(!obj || obj.id === $.jstree.root) { | |
2629 return false; | |
2630 } | |
2631 animation = animation === undefined ? this.settings.core.animation : animation; | |
2632 if(!this.is_closed(obj)) { | |
2633 if(callback) { | |
2634 callback.call(this, obj, false); | |
2635 } | |
2636 return false; | |
2637 } | |
2638 if(!this.is_loaded(obj)) { | |
2639 if(this.is_loading(obj)) { | |
2640 return setTimeout($.proxy(function () { | |
2641 this.open_node(obj, callback, animation); | |
2642 }, this), 500); | |
2643 } | |
2644 this.load_node(obj, function (o, ok) { | |
2645 return ok ? this.open_node(o, callback, animation) : (callback ? callback.call(this, o, false) : false); | |
2646 }); | |
2647 } | |
2648 else { | |
2649 d = this.get_node(obj, true); | |
2650 t = this; | |
2651 if(d.length) { | |
2652 if(animation && d.children(".jstree-children").length) { | |
2653 d.children(".jstree-children").stop(true, true); | |
2654 } | |
2655 if(obj.children.length && !this._firstChild(d.children('.jstree-children')[0])) { | |
2656 this.draw_children(obj); | |
2657 //d = this.get_node(obj, true); | |
2658 } | |
2659 if(!animation) { | |
2660 this.trigger('before_open', { "node" : obj }); | |
2661 d[0].className = d[0].className.replace('jstree-closed', 'jstree-open'); | |
2662 d[0].setAttribute("aria-expanded", true); | |
2663 } | |
2664 else { | |
2665 this.trigger('before_open', { "node" : obj }); | |
2666 d | |
2667 .children(".jstree-children").css("display","none").end() | |
2668 .removeClass("jstree-closed").addClass("jstree-open").attr("aria-expanded", true) | |
2669 .children(".jstree-children").stop(true, true) | |
2670 .slideDown(animation, function () { | |
2671 this.style.display = ""; | |
2672 if (t.element) { | |
2673 t.trigger("after_open", { "node" : obj }); | |
2674 } | |
2675 }); | |
2676 } | |
2677 } | |
2678 obj.state.opened = true; | |
2679 if(callback) { | |
2680 callback.call(this, obj, true); | |
2681 } | |
2682 if(!d.length) { | |
2683 /** | |
2684 * triggered when a node is about to be opened (if the node is supposed to be in the DOM, it will be, but it won't be visible yet) | |
2685 * @event | |
2686 * @name before_open.jstree | |
2687 * @param {Object} node the opened node | |
2688 */ | |
2689 this.trigger('before_open', { "node" : obj }); | |
2690 } | |
2691 /** | |
2692 * triggered when a node is opened (if there is an animation it will not be completed yet) | |
2693 * @event | |
2694 * @name open_node.jstree | |
2695 * @param {Object} node the opened node | |
2696 */ | |
2697 this.trigger('open_node', { "node" : obj }); | |
2698 if(!animation || !d.length) { | |
2699 /** | |
2700 * triggered when a node is opened and the animation is complete | |
2701 * @event | |
2702 * @name after_open.jstree | |
2703 * @param {Object} node the opened node | |
2704 */ | |
2705 this.trigger("after_open", { "node" : obj }); | |
2706 } | |
2707 return true; | |
2708 } | |
2709 }, | |
2710 /** | |
2711 * opens every parent of a node (node should be loaded) | |
2712 * @name _open_to(obj) | |
2713 * @param {mixed} obj the node to reveal | |
2714 * @private | |
2715 */ | |
2716 _open_to : function (obj) { | |
2717 obj = this.get_node(obj); | |
2718 if(!obj || obj.id === $.jstree.root) { | |
2719 return false; | |
2720 } | |
2721 var i, j, p = obj.parents; | |
2722 for(i = 0, j = p.length; i < j; i+=1) { | |
2723 if(i !== $.jstree.root) { | |
2724 this.open_node(p[i], false, 0); | |
2725 } | |
2726 } | |
2727 return $('#' + obj.id.replace($.jstree.idregex,'\\$&'), this.element); | |
2728 }, | |
2729 /** | |
2730 * closes a node, hiding its children | |
2731 * @name close_node(obj [, animation]) | |
2732 * @param {mixed} obj the node to close | |
2733 * @param {Number} animation the animation duration in milliseconds when closing the node (overrides the `core.animation` setting). Use `false` for no animation. | |
2734 * @trigger close_node.jstree, after_close.jstree | |
2735 */ | |
2736 close_node : function (obj, animation) { | |
2737 var t1, t2, t, d; | |
2738 if($.isArray(obj)) { | |
2739 obj = obj.slice(); | |
2740 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
2741 this.close_node(obj[t1], animation); | |
2742 } | |
2743 return true; | |
2744 } | |
2745 obj = this.get_node(obj); | |
2746 if(!obj || obj.id === $.jstree.root) { | |
2747 return false; | |
2748 } | |
2749 if(this.is_closed(obj)) { | |
2750 return false; | |
2751 } | |
2752 animation = animation === undefined ? this.settings.core.animation : animation; | |
2753 t = this; | |
2754 d = this.get_node(obj, true); | |
2755 | |
2756 obj.state.opened = false; | |
2757 /** | |
2758 * triggered when a node is closed (if there is an animation it will not be complete yet) | |
2759 * @event | |
2760 * @name close_node.jstree | |
2761 * @param {Object} node the closed node | |
2762 */ | |
2763 this.trigger('close_node',{ "node" : obj }); | |
2764 if(!d.length) { | |
2765 /** | |
2766 * triggered when a node is closed and the animation is complete | |
2767 * @event | |
2768 * @name after_close.jstree | |
2769 * @param {Object} node the closed node | |
2770 */ | |
2771 this.trigger("after_close", { "node" : obj }); | |
2772 } | |
2773 else { | |
2774 if(!animation) { | |
2775 d[0].className = d[0].className.replace('jstree-open', 'jstree-closed'); | |
2776 d.attr("aria-expanded", false).children('.jstree-children').remove(); | |
2777 this.trigger("after_close", { "node" : obj }); | |
2778 } | |
2779 else { | |
2780 d | |
2781 .children(".jstree-children").attr("style","display:block !important").end() | |
2782 .removeClass("jstree-open").addClass("jstree-closed").attr("aria-expanded", false) | |
2783 .children(".jstree-children").stop(true, true).slideUp(animation, function () { | |
2784 this.style.display = ""; | |
2785 d.children('.jstree-children').remove(); | |
2786 if (t.element) { | |
2787 t.trigger("after_close", { "node" : obj }); | |
2788 } | |
2789 }); | |
2790 } | |
2791 } | |
2792 }, | |
2793 /** | |
2794 * toggles a node - closing it if it is open, opening it if it is closed | |
2795 * @name toggle_node(obj) | |
2796 * @param {mixed} obj the node to toggle | |
2797 */ | |
2798 toggle_node : function (obj) { | |
2799 var t1, t2; | |
2800 if($.isArray(obj)) { | |
2801 obj = obj.slice(); | |
2802 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
2803 this.toggle_node(obj[t1]); | |
2804 } | |
2805 return true; | |
2806 } | |
2807 if(this.is_closed(obj)) { | |
2808 return this.open_node(obj); | |
2809 } | |
2810 if(this.is_open(obj)) { | |
2811 return this.close_node(obj); | |
2812 } | |
2813 }, | |
2814 /** | |
2815 * opens all nodes within a node (or the tree), revaling their children. If the node is not loaded it will be loaded and opened once ready. | |
2816 * @name open_all([obj, animation, original_obj]) | |
2817 * @param {mixed} obj the node to open recursively, omit to open all nodes in the tree | |
2818 * @param {Number} animation the animation duration in milliseconds when opening the nodes, the default is no animation | |
2819 * @param {jQuery} reference to the node that started the process (internal use) | |
2820 * @trigger open_all.jstree | |
2821 */ | |
2822 open_all : function (obj, animation, original_obj) { | |
2823 if(!obj) { obj = $.jstree.root; } | |
2824 obj = this.get_node(obj); | |
2825 if(!obj) { return false; } | |
2826 var dom = obj.id === $.jstree.root ? this.get_container_ul() : this.get_node(obj, true), i, j, _this; | |
2827 if(!dom.length) { | |
2828 for(i = 0, j = obj.children_d.length; i < j; i++) { | |
2829 if(this.is_closed(this._model.data[obj.children_d[i]])) { | |
2830 this._model.data[obj.children_d[i]].state.opened = true; | |
2831 } | |
2832 } | |
2833 return this.trigger('open_all', { "node" : obj }); | |
2834 } | |
2835 original_obj = original_obj || dom; | |
2836 _this = this; | |
2837 dom = this.is_closed(obj) ? dom.find('.jstree-closed').addBack() : dom.find('.jstree-closed'); | |
2838 dom.each(function () { | |
2839 _this.open_node( | |
2840 this, | |
2841 function(node, status) { if(status && this.is_parent(node)) { this.open_all(node, animation, original_obj); } }, | |
2842 animation || 0 | |
2843 ); | |
2844 }); | |
2845 if(original_obj.find('.jstree-closed').length === 0) { | |
2846 /** | |
2847 * triggered when an `open_all` call completes | |
2848 * @event | |
2849 * @name open_all.jstree | |
2850 * @param {Object} node the opened node | |
2851 */ | |
2852 this.trigger('open_all', { "node" : this.get_node(original_obj) }); | |
2853 } | |
2854 }, | |
2855 /** | |
2856 * closes all nodes within a node (or the tree), revaling their children | |
2857 * @name close_all([obj, animation]) | |
2858 * @param {mixed} obj the node to close recursively, omit to close all nodes in the tree | |
2859 * @param {Number} animation the animation duration in milliseconds when closing the nodes, the default is no animation | |
2860 * @trigger close_all.jstree | |
2861 */ | |
2862 close_all : function (obj, animation) { | |
2863 if(!obj) { obj = $.jstree.root; } | |
2864 obj = this.get_node(obj); | |
2865 if(!obj) { return false; } | |
2866 var dom = obj.id === $.jstree.root ? this.get_container_ul() : this.get_node(obj, true), | |
2867 _this = this, i, j; | |
2868 if(dom.length) { | |
2869 dom = this.is_open(obj) ? dom.find('.jstree-open').addBack() : dom.find('.jstree-open'); | |
2870 $(dom.get().reverse()).each(function () { _this.close_node(this, animation || 0); }); | |
2871 } | |
2872 for(i = 0, j = obj.children_d.length; i < j; i++) { | |
2873 this._model.data[obj.children_d[i]].state.opened = false; | |
2874 } | |
2875 /** | |
2876 * triggered when an `close_all` call completes | |
2877 * @event | |
2878 * @name close_all.jstree | |
2879 * @param {Object} node the closed node | |
2880 */ | |
2881 this.trigger('close_all', { "node" : obj }); | |
2882 }, | |
2883 /** | |
2884 * checks if a node is disabled (not selectable) | |
2885 * @name is_disabled(obj) | |
2886 * @param {mixed} obj | |
2887 * @return {Boolean} | |
2888 */ | |
2889 is_disabled : function (obj) { | |
2890 obj = this.get_node(obj); | |
2891 return obj && obj.state && obj.state.disabled; | |
2892 }, | |
2893 /** | |
2894 * enables a node - so that it can be selected | |
2895 * @name enable_node(obj) | |
2896 * @param {mixed} obj the node to enable | |
2897 * @trigger enable_node.jstree | |
2898 */ | |
2899 enable_node : function (obj) { | |
2900 var t1, t2; | |
2901 if($.isArray(obj)) { | |
2902 obj = obj.slice(); | |
2903 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
2904 this.enable_node(obj[t1]); | |
2905 } | |
2906 return true; | |
2907 } | |
2908 obj = this.get_node(obj); | |
2909 if(!obj || obj.id === $.jstree.root) { | |
2910 return false; | |
2911 } | |
2912 obj.state.disabled = false; | |
2913 this.get_node(obj,true).children('.jstree-anchor').removeClass('jstree-disabled').attr('aria-disabled', false); | |
2914 /** | |
2915 * triggered when an node is enabled | |
2916 * @event | |
2917 * @name enable_node.jstree | |
2918 * @param {Object} node the enabled node | |
2919 */ | |
2920 this.trigger('enable_node', { 'node' : obj }); | |
2921 }, | |
2922 /** | |
2923 * disables a node - so that it can not be selected | |
2924 * @name disable_node(obj) | |
2925 * @param {mixed} obj the node to disable | |
2926 * @trigger disable_node.jstree | |
2927 */ | |
2928 disable_node : function (obj) { | |
2929 var t1, t2; | |
2930 if($.isArray(obj)) { | |
2931 obj = obj.slice(); | |
2932 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
2933 this.disable_node(obj[t1]); | |
2934 } | |
2935 return true; | |
2936 } | |
2937 obj = this.get_node(obj); | |
2938 if(!obj || obj.id === $.jstree.root) { | |
2939 return false; | |
2940 } | |
2941 obj.state.disabled = true; | |
2942 this.get_node(obj,true).children('.jstree-anchor').addClass('jstree-disabled').attr('aria-disabled', true); | |
2943 /** | |
2944 * triggered when an node is disabled | |
2945 * @event | |
2946 * @name disable_node.jstree | |
2947 * @param {Object} node the disabled node | |
2948 */ | |
2949 this.trigger('disable_node', { 'node' : obj }); | |
2950 }, | |
2951 /** | |
2952 * determines if a node is hidden | |
2953 * @name is_hidden(obj) | |
2954 * @param {mixed} obj the node | |
2955 */ | |
2956 is_hidden : function (obj) { | |
2957 obj = this.get_node(obj); | |
2958 return obj.state.hidden === true; | |
2959 }, | |
2960 /** | |
2961 * hides a node - it is still in the structure but will not be visible | |
2962 * @name hide_node(obj) | |
2963 * @param {mixed} obj the node to hide | |
2964 * @param {Boolean} skip_redraw internal parameter controlling if redraw is called | |
2965 * @trigger hide_node.jstree | |
2966 */ | |
2967 hide_node : function (obj, skip_redraw) { | |
2968 var t1, t2; | |
2969 if($.isArray(obj)) { | |
2970 obj = obj.slice(); | |
2971 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
2972 this.hide_node(obj[t1], true); | |
2973 } | |
2974 if (!skip_redraw) { | |
2975 this.redraw(); | |
2976 } | |
2977 return true; | |
2978 } | |
2979 obj = this.get_node(obj); | |
2980 if(!obj || obj.id === $.jstree.root) { | |
2981 return false; | |
2982 } | |
2983 if(!obj.state.hidden) { | |
2984 obj.state.hidden = true; | |
2985 this._node_changed(obj.parent); | |
2986 if(!skip_redraw) { | |
2987 this.redraw(); | |
2988 } | |
2989 /** | |
2990 * triggered when an node is hidden | |
2991 * @event | |
2992 * @name hide_node.jstree | |
2993 * @param {Object} node the hidden node | |
2994 */ | |
2995 this.trigger('hide_node', { 'node' : obj }); | |
2996 } | |
2997 }, | |
2998 /** | |
2999 * shows a node | |
3000 * @name show_node(obj) | |
3001 * @param {mixed} obj the node to show | |
3002 * @param {Boolean} skip_redraw internal parameter controlling if redraw is called | |
3003 * @trigger show_node.jstree | |
3004 */ | |
3005 show_node : function (obj, skip_redraw) { | |
3006 var t1, t2; | |
3007 if($.isArray(obj)) { | |
3008 obj = obj.slice(); | |
3009 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
3010 this.show_node(obj[t1], true); | |
3011 } | |
3012 if (!skip_redraw) { | |
3013 this.redraw(); | |
3014 } | |
3015 return true; | |
3016 } | |
3017 obj = this.get_node(obj); | |
3018 if(!obj || obj.id === $.jstree.root) { | |
3019 return false; | |
3020 } | |
3021 if(obj.state.hidden) { | |
3022 obj.state.hidden = false; | |
3023 this._node_changed(obj.parent); | |
3024 if(!skip_redraw) { | |
3025 this.redraw(); | |
3026 } | |
3027 /** | |
3028 * triggered when an node is shown | |
3029 * @event | |
3030 * @name show_node.jstree | |
3031 * @param {Object} node the shown node | |
3032 */ | |
3033 this.trigger('show_node', { 'node' : obj }); | |
3034 } | |
3035 }, | |
3036 /** | |
3037 * hides all nodes | |
3038 * @name hide_all() | |
3039 * @trigger hide_all.jstree | |
3040 */ | |
3041 hide_all : function (skip_redraw) { | |
3042 var i, m = this._model.data, ids = []; | |
3043 for(i in m) { | |
3044 if(m.hasOwnProperty(i) && i !== $.jstree.root && !m[i].state.hidden) { | |
3045 m[i].state.hidden = true; | |
3046 ids.push(i); | |
3047 } | |
3048 } | |
3049 this._model.force_full_redraw = true; | |
3050 if(!skip_redraw) { | |
3051 this.redraw(); | |
3052 } | |
3053 /** | |
3054 * triggered when all nodes are hidden | |
3055 * @event | |
3056 * @name hide_all.jstree | |
3057 * @param {Array} nodes the IDs of all hidden nodes | |
3058 */ | |
3059 this.trigger('hide_all', { 'nodes' : ids }); | |
3060 return ids; | |
3061 }, | |
3062 /** | |
3063 * shows all nodes | |
3064 * @name show_all() | |
3065 * @trigger show_all.jstree | |
3066 */ | |
3067 show_all : function (skip_redraw) { | |
3068 var i, m = this._model.data, ids = []; | |
3069 for(i in m) { | |
3070 if(m.hasOwnProperty(i) && i !== $.jstree.root && m[i].state.hidden) { | |
3071 m[i].state.hidden = false; | |
3072 ids.push(i); | |
3073 } | |
3074 } | |
3075 this._model.force_full_redraw = true; | |
3076 if(!skip_redraw) { | |
3077 this.redraw(); | |
3078 } | |
3079 /** | |
3080 * triggered when all nodes are shown | |
3081 * @event | |
3082 * @name show_all.jstree | |
3083 * @param {Array} nodes the IDs of all shown nodes | |
3084 */ | |
3085 this.trigger('show_all', { 'nodes' : ids }); | |
3086 return ids; | |
3087 }, | |
3088 /** | |
3089 * called when a node is selected by the user. Used internally. | |
3090 * @private | |
3091 * @name activate_node(obj, e) | |
3092 * @param {mixed} obj the node | |
3093 * @param {Object} e the related event | |
3094 * @trigger activate_node.jstree, changed.jstree | |
3095 */ | |
3096 activate_node : function (obj, e) { | |
3097 if(this.is_disabled(obj)) { | |
3098 return false; | |
3099 } | |
3100 if(!e || typeof e !== 'object') { | |
3101 e = {}; | |
3102 } | |
3103 | |
3104 // ensure last_clicked is still in the DOM, make it fresh (maybe it was moved?) and make sure it is still selected, if not - make last_clicked the last selected node | |
3105 this._data.core.last_clicked = this._data.core.last_clicked && this._data.core.last_clicked.id !== undefined ? this.get_node(this._data.core.last_clicked.id) : null; | |
3106 if(this._data.core.last_clicked && !this._data.core.last_clicked.state.selected) { this._data.core.last_clicked = null; } | |
3107 if(!this._data.core.last_clicked && this._data.core.selected.length) { this._data.core.last_clicked = this.get_node(this._data.core.selected[this._data.core.selected.length - 1]); } | |
3108 | |
3109 if(!this.settings.core.multiple || (!e.metaKey && !e.ctrlKey && !e.shiftKey) || (e.shiftKey && (!this._data.core.last_clicked || !this.get_parent(obj) || this.get_parent(obj) !== this._data.core.last_clicked.parent ) )) { | |
3110 if(!this.settings.core.multiple && (e.metaKey || e.ctrlKey || e.shiftKey) && this.is_selected(obj)) { | |
3111 this.deselect_node(obj, false, e); | |
3112 } | |
3113 else { | |
3114 this.deselect_all(true); | |
3115 this.select_node(obj, false, false, e); | |
3116 this._data.core.last_clicked = this.get_node(obj); | |
3117 } | |
3118 } | |
3119 else { | |
3120 if(e.shiftKey) { | |
3121 var o = this.get_node(obj).id, | |
3122 l = this._data.core.last_clicked.id, | |
3123 p = this.get_node(this._data.core.last_clicked.parent).children, | |
3124 c = false, | |
3125 i, j; | |
3126 for(i = 0, j = p.length; i < j; i += 1) { | |
3127 // separate IFs work whem o and l are the same | |
3128 if(p[i] === o) { | |
3129 c = !c; | |
3130 } | |
3131 if(p[i] === l) { | |
3132 c = !c; | |
3133 } | |
3134 if(!this.is_disabled(p[i]) && (c || p[i] === o || p[i] === l)) { | |
3135 if (!this.is_hidden(p[i])) { | |
3136 this.select_node(p[i], true, false, e); | |
3137 } | |
3138 } | |
3139 else { | |
3140 this.deselect_node(p[i], true, e); | |
3141 } | |
3142 } | |
3143 this.trigger('changed', { 'action' : 'select_node', 'node' : this.get_node(obj), 'selected' : this._data.core.selected, 'event' : e }); | |
3144 } | |
3145 else { | |
3146 if(!this.is_selected(obj)) { | |
3147 this.select_node(obj, false, false, e); | |
3148 } | |
3149 else { | |
3150 this.deselect_node(obj, false, e); | |
3151 } | |
3152 } | |
3153 } | |
3154 /** | |
3155 * triggered when an node is clicked or intercated with by the user | |
3156 * @event | |
3157 * @name activate_node.jstree | |
3158 * @param {Object} node | |
3159 * @param {Object} event the ooriginal event (if any) which triggered the call (may be an empty object) | |
3160 */ | |
3161 this.trigger('activate_node', { 'node' : this.get_node(obj), 'event' : e }); | |
3162 }, | |
3163 /** | |
3164 * applies the hover state on a node, called when a node is hovered by the user. Used internally. | |
3165 * @private | |
3166 * @name hover_node(obj) | |
3167 * @param {mixed} obj | |
3168 * @trigger hover_node.jstree | |
3169 */ | |
3170 hover_node : function (obj) { | |
3171 obj = this.get_node(obj, true); | |
3172 if(!obj || !obj.length || obj.children('.jstree-hovered').length) { | |
3173 return false; | |
3174 } | |
3175 var o = this.element.find('.jstree-hovered'), t = this.element; | |
3176 if(o && o.length) { this.dehover_node(o); } | |
3177 | |
3178 obj.children('.jstree-anchor').addClass('jstree-hovered'); | |
3179 /** | |
3180 * triggered when an node is hovered | |
3181 * @event | |
3182 * @name hover_node.jstree | |
3183 * @param {Object} node | |
3184 */ | |
3185 this.trigger('hover_node', { 'node' : this.get_node(obj) }); | |
3186 setTimeout(function () { t.attr('aria-activedescendant', obj[0].id); }, 0); | |
3187 }, | |
3188 /** | |
3189 * removes the hover state from a nodecalled when a node is no longer hovered by the user. Used internally. | |
3190 * @private | |
3191 * @name dehover_node(obj) | |
3192 * @param {mixed} obj | |
3193 * @trigger dehover_node.jstree | |
3194 */ | |
3195 dehover_node : function (obj) { | |
3196 obj = this.get_node(obj, true); | |
3197 if(!obj || !obj.length || !obj.children('.jstree-hovered').length) { | |
3198 return false; | |
3199 } | |
3200 obj.children('.jstree-anchor').removeClass('jstree-hovered'); | |
3201 /** | |
3202 * triggered when an node is no longer hovered | |
3203 * @event | |
3204 * @name dehover_node.jstree | |
3205 * @param {Object} node | |
3206 */ | |
3207 this.trigger('dehover_node', { 'node' : this.get_node(obj) }); | |
3208 }, | |
3209 /** | |
3210 * select a node | |
3211 * @name select_node(obj [, supress_event, prevent_open]) | |
3212 * @param {mixed} obj an array can be used to select multiple nodes | |
3213 * @param {Boolean} supress_event if set to `true` the `changed.jstree` event won't be triggered | |
3214 * @param {Boolean} prevent_open if set to `true` parents of the selected node won't be opened | |
3215 * @trigger select_node.jstree, changed.jstree | |
3216 */ | |
3217 select_node : function (obj, supress_event, prevent_open, e) { | |
3218 var dom, t1, t2, th; | |
3219 if($.isArray(obj)) { | |
3220 obj = obj.slice(); | |
3221 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
3222 this.select_node(obj[t1], supress_event, prevent_open, e); | |
3223 } | |
3224 return true; | |
3225 } | |
3226 obj = this.get_node(obj); | |
3227 if(!obj || obj.id === $.jstree.root) { | |
3228 return false; | |
3229 } | |
3230 dom = this.get_node(obj, true); | |
3231 if(!obj.state.selected) { | |
3232 obj.state.selected = true; | |
3233 this._data.core.selected.push(obj.id); | |
3234 if(!prevent_open) { | |
3235 dom = this._open_to(obj); | |
3236 } | |
3237 if(dom && dom.length) { | |
3238 dom.attr('aria-selected', true).children('.jstree-anchor').addClass('jstree-clicked'); | |
3239 } | |
3240 /** | |
3241 * triggered when an node is selected | |
3242 * @event | |
3243 * @name select_node.jstree | |
3244 * @param {Object} node | |
3245 * @param {Array} selected the current selection | |
3246 * @param {Object} event the event (if any) that triggered this select_node | |
3247 */ | |
3248 this.trigger('select_node', { 'node' : obj, 'selected' : this._data.core.selected, 'event' : e }); | |
3249 if(!supress_event) { | |
3250 /** | |
3251 * triggered when selection changes | |
3252 * @event | |
3253 * @name changed.jstree | |
3254 * @param {Object} node | |
3255 * @param {Object} action the action that caused the selection to change | |
3256 * @param {Array} selected the current selection | |
3257 * @param {Object} event the event (if any) that triggered this changed event | |
3258 */ | |
3259 this.trigger('changed', { 'action' : 'select_node', 'node' : obj, 'selected' : this._data.core.selected, 'event' : e }); | |
3260 } | |
3261 } | |
3262 }, | |
3263 /** | |
3264 * deselect a node | |
3265 * @name deselect_node(obj [, supress_event]) | |
3266 * @param {mixed} obj an array can be used to deselect multiple nodes | |
3267 * @param {Boolean} supress_event if set to `true` the `changed.jstree` event won't be triggered | |
3268 * @trigger deselect_node.jstree, changed.jstree | |
3269 */ | |
3270 deselect_node : function (obj, supress_event, e) { | |
3271 var t1, t2, dom; | |
3272 if($.isArray(obj)) { | |
3273 obj = obj.slice(); | |
3274 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
3275 this.deselect_node(obj[t1], supress_event, e); | |
3276 } | |
3277 return true; | |
3278 } | |
3279 obj = this.get_node(obj); | |
3280 if(!obj || obj.id === $.jstree.root) { | |
3281 return false; | |
3282 } | |
3283 dom = this.get_node(obj, true); | |
3284 if(obj.state.selected) { | |
3285 obj.state.selected = false; | |
3286 this._data.core.selected = $.vakata.array_remove_item(this._data.core.selected, obj.id); | |
3287 if(dom.length) { | |
3288 dom.attr('aria-selected', false).children('.jstree-anchor').removeClass('jstree-clicked'); | |
3289 } | |
3290 /** | |
3291 * triggered when an node is deselected | |
3292 * @event | |
3293 * @name deselect_node.jstree | |
3294 * @param {Object} node | |
3295 * @param {Array} selected the current selection | |
3296 * @param {Object} event the event (if any) that triggered this deselect_node | |
3297 */ | |
3298 this.trigger('deselect_node', { 'node' : obj, 'selected' : this._data.core.selected, 'event' : e }); | |
3299 if(!supress_event) { | |
3300 this.trigger('changed', { 'action' : 'deselect_node', 'node' : obj, 'selected' : this._data.core.selected, 'event' : e }); | |
3301 } | |
3302 } | |
3303 }, | |
3304 /** | |
3305 * select all nodes in the tree | |
3306 * @name select_all([supress_event]) | |
3307 * @param {Boolean} supress_event if set to `true` the `changed.jstree` event won't be triggered | |
3308 * @trigger select_all.jstree, changed.jstree | |
3309 */ | |
3310 select_all : function (supress_event) { | |
3311 var tmp = this._data.core.selected.concat([]), i, j; | |
3312 this._data.core.selected = this._model.data[$.jstree.root].children_d.concat(); | |
3313 for(i = 0, j = this._data.core.selected.length; i < j; i++) { | |
3314 if(this._model.data[this._data.core.selected[i]]) { | |
3315 this._model.data[this._data.core.selected[i]].state.selected = true; | |
3316 } | |
3317 } | |
3318 this.redraw(true); | |
3319 /** | |
3320 * triggered when all nodes are selected | |
3321 * @event | |
3322 * @name select_all.jstree | |
3323 * @param {Array} selected the current selection | |
3324 */ | |
3325 this.trigger('select_all', { 'selected' : this._data.core.selected }); | |
3326 if(!supress_event) { | |
3327 this.trigger('changed', { 'action' : 'select_all', 'selected' : this._data.core.selected, 'old_selection' : tmp }); | |
3328 } | |
3329 }, | |
3330 /** | |
3331 * deselect all selected nodes | |
3332 * @name deselect_all([supress_event]) | |
3333 * @param {Boolean} supress_event if set to `true` the `changed.jstree` event won't be triggered | |
3334 * @trigger deselect_all.jstree, changed.jstree | |
3335 */ | |
3336 deselect_all : function (supress_event) { | |
3337 var tmp = this._data.core.selected.concat([]), i, j; | |
3338 for(i = 0, j = this._data.core.selected.length; i < j; i++) { | |
3339 if(this._model.data[this._data.core.selected[i]]) { | |
3340 this._model.data[this._data.core.selected[i]].state.selected = false; | |
3341 } | |
3342 } | |
3343 this._data.core.selected = []; | |
3344 this.element.find('.jstree-clicked').removeClass('jstree-clicked').parent().attr('aria-selected', false); | |
3345 /** | |
3346 * triggered when all nodes are deselected | |
3347 * @event | |
3348 * @name deselect_all.jstree | |
3349 * @param {Object} node the previous selection | |
3350 * @param {Array} selected the current selection | |
3351 */ | |
3352 this.trigger('deselect_all', { 'selected' : this._data.core.selected, 'node' : tmp }); | |
3353 if(!supress_event) { | |
3354 this.trigger('changed', { 'action' : 'deselect_all', 'selected' : this._data.core.selected, 'old_selection' : tmp }); | |
3355 } | |
3356 }, | |
3357 /** | |
3358 * checks if a node is selected | |
3359 * @name is_selected(obj) | |
3360 * @param {mixed} obj | |
3361 * @return {Boolean} | |
3362 */ | |
3363 is_selected : function (obj) { | |
3364 obj = this.get_node(obj); | |
3365 if(!obj || obj.id === $.jstree.root) { | |
3366 return false; | |
3367 } | |
3368 return obj.state.selected; | |
3369 }, | |
3370 /** | |
3371 * get an array of all selected nodes | |
3372 * @name get_selected([full]) | |
3373 * @param {mixed} full if set to `true` the returned array will consist of the full node objects, otherwise - only IDs will be returned | |
3374 * @return {Array} | |
3375 */ | |
3376 get_selected : function (full) { | |
3377 return full ? $.map(this._data.core.selected, $.proxy(function (i) { return this.get_node(i); }, this)) : this._data.core.selected.slice(); | |
3378 }, | |
3379 /** | |
3380 * get an array of all top level selected nodes (ignoring children of selected nodes) | |
3381 * @name get_top_selected([full]) | |
3382 * @param {mixed} full if set to `true` the returned array will consist of the full node objects, otherwise - only IDs will be returned | |
3383 * @return {Array} | |
3384 */ | |
3385 get_top_selected : function (full) { | |
3386 var tmp = this.get_selected(true), | |
3387 obj = {}, i, j, k, l; | |
3388 for(i = 0, j = tmp.length; i < j; i++) { | |
3389 obj[tmp[i].id] = tmp[i]; | |
3390 } | |
3391 for(i = 0, j = tmp.length; i < j; i++) { | |
3392 for(k = 0, l = tmp[i].children_d.length; k < l; k++) { | |
3393 if(obj[tmp[i].children_d[k]]) { | |
3394 delete obj[tmp[i].children_d[k]]; | |
3395 } | |
3396 } | |
3397 } | |
3398 tmp = []; | |
3399 for(i in obj) { | |
3400 if(obj.hasOwnProperty(i)) { | |
3401 tmp.push(i); | |
3402 } | |
3403 } | |
3404 return full ? $.map(tmp, $.proxy(function (i) { return this.get_node(i); }, this)) : tmp; | |
3405 }, | |
3406 /** | |
3407 * get an array of all bottom level selected nodes (ignoring selected parents) | |
3408 * @name get_bottom_selected([full]) | |
3409 * @param {mixed} full if set to `true` the returned array will consist of the full node objects, otherwise - only IDs will be returned | |
3410 * @return {Array} | |
3411 */ | |
3412 get_bottom_selected : function (full) { | |
3413 var tmp = this.get_selected(true), | |
3414 obj = [], i, j; | |
3415 for(i = 0, j = tmp.length; i < j; i++) { | |
3416 if(!tmp[i].children.length) { | |
3417 obj.push(tmp[i].id); | |
3418 } | |
3419 } | |
3420 return full ? $.map(obj, $.proxy(function (i) { return this.get_node(i); }, this)) : obj; | |
3421 }, | |
3422 /** | |
3423 * gets the current state of the tree so that it can be restored later with `set_state(state)`. Used internally. | |
3424 * @name get_state() | |
3425 * @private | |
3426 * @return {Object} | |
3427 */ | |
3428 get_state : function () { | |
3429 var state = { | |
3430 'core' : { | |
3431 'open' : [], | |
3432 'loaded' : [], | |
3433 'scroll' : { | |
3434 'left' : this.element.scrollLeft(), | |
3435 'top' : this.element.scrollTop() | |
3436 }, | |
3437 /*! | |
3438 'themes' : { | |
3439 'name' : this.get_theme(), | |
3440 'icons' : this._data.core.themes.icons, | |
3441 'dots' : this._data.core.themes.dots | |
3442 }, | |
3443 */ | |
3444 'selected' : [] | |
3445 } | |
3446 }, i; | |
3447 for(i in this._model.data) { | |
3448 if(this._model.data.hasOwnProperty(i)) { | |
3449 if(i !== $.jstree.root) { | |
3450 if(this._model.data[i].state.loaded && this.settings.core.loaded_state) { | |
3451 state.core.loaded.push(i); | |
3452 } | |
3453 if(this._model.data[i].state.opened) { | |
3454 state.core.open.push(i); | |
3455 } | |
3456 if(this._model.data[i].state.selected) { | |
3457 state.core.selected.push(i); | |
3458 } | |
3459 } | |
3460 } | |
3461 } | |
3462 return state; | |
3463 }, | |
3464 /** | |
3465 * sets the state of the tree. Used internally. | |
3466 * @name set_state(state [, callback]) | |
3467 * @private | |
3468 * @param {Object} state the state to restore. Keep in mind this object is passed by reference and jstree will modify it. | |
3469 * @param {Function} callback an optional function to execute once the state is restored. | |
3470 * @trigger set_state.jstree | |
3471 */ | |
3472 set_state : function (state, callback) { | |
3473 if(state) { | |
3474 if(state.core && state.core.selected && state.core.initial_selection === undefined) { | |
3475 state.core.initial_selection = this._data.core.selected.concat([]).sort().join(','); | |
3476 } | |
3477 if(state.core) { | |
3478 var res, n, t, _this, i; | |
3479 if(state.core.loaded) { | |
3480 if(!this.settings.core.loaded_state || !$.isArray(state.core.loaded) || !state.core.loaded.length) { | |
3481 delete state.core.loaded; | |
3482 this.set_state(state, callback); | |
3483 } | |
3484 else { | |
3485 this._load_nodes(state.core.loaded, function (nodes) { | |
3486 delete state.core.loaded; | |
3487 this.set_state(state, callback); | |
3488 }); | |
3489 } | |
3490 return false; | |
3491 } | |
3492 if(state.core.open) { | |
3493 if(!$.isArray(state.core.open) || !state.core.open.length) { | |
3494 delete state.core.open; | |
3495 this.set_state(state, callback); | |
3496 } | |
3497 else { | |
3498 this._load_nodes(state.core.open, function (nodes) { | |
3499 this.open_node(nodes, false, 0); | |
3500 delete state.core.open; | |
3501 this.set_state(state, callback); | |
3502 }); | |
3503 } | |
3504 return false; | |
3505 } | |
3506 if(state.core.scroll) { | |
3507 if(state.core.scroll && state.core.scroll.left !== undefined) { | |
3508 this.element.scrollLeft(state.core.scroll.left); | |
3509 } | |
3510 if(state.core.scroll && state.core.scroll.top !== undefined) { | |
3511 this.element.scrollTop(state.core.scroll.top); | |
3512 } | |
3513 delete state.core.scroll; | |
3514 this.set_state(state, callback); | |
3515 return false; | |
3516 } | |
3517 if(state.core.selected) { | |
3518 _this = this; | |
3519 if (state.core.initial_selection === undefined || | |
3520 state.core.initial_selection === this._data.core.selected.concat([]).sort().join(',') | |
3521 ) { | |
3522 this.deselect_all(); | |
3523 $.each(state.core.selected, function (i, v) { | |
3524 _this.select_node(v, false, true); | |
3525 }); | |
3526 } | |
3527 delete state.core.initial_selection; | |
3528 delete state.core.selected; | |
3529 this.set_state(state, callback); | |
3530 return false; | |
3531 } | |
3532 for(i in state) { | |
3533 if(state.hasOwnProperty(i) && i !== "core" && $.inArray(i, this.settings.plugins) === -1) { | |
3534 delete state[i]; | |
3535 } | |
3536 } | |
3537 if($.isEmptyObject(state.core)) { | |
3538 delete state.core; | |
3539 this.set_state(state, callback); | |
3540 return false; | |
3541 } | |
3542 } | |
3543 if($.isEmptyObject(state)) { | |
3544 state = null; | |
3545 if(callback) { callback.call(this); } | |
3546 /** | |
3547 * triggered when a `set_state` call completes | |
3548 * @event | |
3549 * @name set_state.jstree | |
3550 */ | |
3551 this.trigger('set_state'); | |
3552 return false; | |
3553 } | |
3554 return true; | |
3555 } | |
3556 return false; | |
3557 }, | |
3558 /** | |
3559 * refreshes the tree - all nodes are reloaded with calls to `load_node`. | |
3560 * @name refresh() | |
3561 * @param {Boolean} skip_loading an option to skip showing the loading indicator | |
3562 * @param {Mixed} forget_state if set to `true` state will not be reapplied, if set to a function (receiving the current state as argument) the result of that function will be used as state | |
3563 * @trigger refresh.jstree | |
3564 */ | |
3565 refresh : function (skip_loading, forget_state) { | |
3566 this._data.core.state = forget_state === true ? {} : this.get_state(); | |
3567 if(forget_state && $.isFunction(forget_state)) { this._data.core.state = forget_state.call(this, this._data.core.state); } | |
3568 this._cnt = 0; | |
3569 this._model.data = {}; | |
3570 this._model.data[$.jstree.root] = { | |
3571 id : $.jstree.root, | |
3572 parent : null, | |
3573 parents : [], | |
3574 children : [], | |
3575 children_d : [], | |
3576 state : { loaded : false } | |
3577 }; | |
3578 this._data.core.selected = []; | |
3579 this._data.core.last_clicked = null; | |
3580 this._data.core.focused = null; | |
3581 | |
3582 var c = this.get_container_ul()[0].className; | |
3583 if(!skip_loading) { | |
3584 this.element.html("<"+"ul class='"+c+"' role='group'><"+"li class='jstree-initial-node jstree-loading jstree-leaf jstree-last' role='treeitem' id='j"+this._id+"_loading'><i class='jstree-icon jstree-ocl'></i><"+"a class='jstree-anchor' href='#'><i class='jstree-icon jstree-themeicon-hidden'></i>" + this.get_string("Loading ...") + "</a></li></ul>"); | |
3585 this.element.attr('aria-activedescendant','j'+this._id+'_loading'); | |
3586 } | |
3587 this.load_node($.jstree.root, function (o, s) { | |
3588 if(s) { | |
3589 this.get_container_ul()[0].className = c; | |
3590 if(this._firstChild(this.get_container_ul()[0])) { | |
3591 this.element.attr('aria-activedescendant',this._firstChild(this.get_container_ul()[0]).id); | |
3592 } | |
3593 this.set_state($.extend(true, {}, this._data.core.state), function () { | |
3594 /** | |
3595 * triggered when a `refresh` call completes | |
3596 * @event | |
3597 * @name refresh.jstree | |
3598 */ | |
3599 this.trigger('refresh'); | |
3600 }); | |
3601 } | |
3602 this._data.core.state = null; | |
3603 }); | |
3604 }, | |
3605 /** | |
3606 * refreshes a node in the tree (reload its children) all opened nodes inside that node are reloaded with calls to `load_node`. | |
3607 * @name refresh_node(obj) | |
3608 * @param {mixed} obj the node | |
3609 * @trigger refresh_node.jstree | |
3610 */ | |
3611 refresh_node : function (obj) { | |
3612 obj = this.get_node(obj); | |
3613 if(!obj || obj.id === $.jstree.root) { return false; } | |
3614 var opened = [], to_load = [], s = this._data.core.selected.concat([]); | |
3615 to_load.push(obj.id); | |
3616 if(obj.state.opened === true) { opened.push(obj.id); } | |
3617 this.get_node(obj, true).find('.jstree-open').each(function() { to_load.push(this.id); opened.push(this.id); }); | |
3618 this._load_nodes(to_load, $.proxy(function (nodes) { | |
3619 this.open_node(opened, false, 0); | |
3620 this.select_node(s); | |
3621 /** | |
3622 * triggered when a node is refreshed | |
3623 * @event | |
3624 * @name refresh_node.jstree | |
3625 * @param {Object} node - the refreshed node | |
3626 * @param {Array} nodes - an array of the IDs of the nodes that were reloaded | |
3627 */ | |
3628 this.trigger('refresh_node', { 'node' : obj, 'nodes' : nodes }); | |
3629 }, this), false, true); | |
3630 }, | |
3631 /** | |
3632 * set (change) the ID of a node | |
3633 * @name set_id(obj, id) | |
3634 * @param {mixed} obj the node | |
3635 * @param {String} id the new ID | |
3636 * @return {Boolean} | |
3637 * @trigger set_id.jstree | |
3638 */ | |
3639 set_id : function (obj, id) { | |
3640 obj = this.get_node(obj); | |
3641 if(!obj || obj.id === $.jstree.root) { return false; } | |
3642 var i, j, m = this._model.data, old = obj.id; | |
3643 id = id.toString(); | |
3644 // update parents (replace current ID with new one in children and children_d) | |
3645 m[obj.parent].children[$.inArray(obj.id, m[obj.parent].children)] = id; | |
3646 for(i = 0, j = obj.parents.length; i < j; i++) { | |
3647 m[obj.parents[i]].children_d[$.inArray(obj.id, m[obj.parents[i]].children_d)] = id; | |
3648 } | |
3649 // update children (replace current ID with new one in parent and parents) | |
3650 for(i = 0, j = obj.children.length; i < j; i++) { | |
3651 m[obj.children[i]].parent = id; | |
3652 } | |
3653 for(i = 0, j = obj.children_d.length; i < j; i++) { | |
3654 m[obj.children_d[i]].parents[$.inArray(obj.id, m[obj.children_d[i]].parents)] = id; | |
3655 } | |
3656 i = $.inArray(obj.id, this._data.core.selected); | |
3657 if(i !== -1) { this._data.core.selected[i] = id; } | |
3658 // update model and obj itself (obj.id, this._model.data[KEY]) | |
3659 i = this.get_node(obj.id, true); | |
3660 if(i) { | |
3661 i.attr('id', id); //.children('.jstree-anchor').attr('id', id + '_anchor').end().attr('aria-labelledby', id + '_anchor'); | |
3662 if(this.element.attr('aria-activedescendant') === obj.id) { | |
3663 this.element.attr('aria-activedescendant', id); | |
3664 } | |
3665 } | |
3666 delete m[obj.id]; | |
3667 obj.id = id; | |
3668 obj.li_attr.id = id; | |
3669 m[id] = obj; | |
3670 /** | |
3671 * triggered when a node id value is changed | |
3672 * @event | |
3673 * @name set_id.jstree | |
3674 * @param {Object} node | |
3675 * @param {String} old the old id | |
3676 */ | |
3677 this.trigger('set_id',{ "node" : obj, "new" : obj.id, "old" : old }); | |
3678 return true; | |
3679 }, | |
3680 /** | |
3681 * get the text value of a node | |
3682 * @name get_text(obj) | |
3683 * @param {mixed} obj the node | |
3684 * @return {String} | |
3685 */ | |
3686 get_text : function (obj) { | |
3687 obj = this.get_node(obj); | |
3688 return (!obj || obj.id === $.jstree.root) ? false : obj.text; | |
3689 }, | |
3690 /** | |
3691 * set the text value of a node. Used internally, please use `rename_node(obj, val)`. | |
3692 * @private | |
3693 * @name set_text(obj, val) | |
3694 * @param {mixed} obj the node, you can pass an array to set the text on multiple nodes | |
3695 * @param {String} val the new text value | |
3696 * @return {Boolean} | |
3697 * @trigger set_text.jstree | |
3698 */ | |
3699 set_text : function (obj, val) { | |
3700 var t1, t2; | |
3701 if($.isArray(obj)) { | |
3702 obj = obj.slice(); | |
3703 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
3704 this.set_text(obj[t1], val); | |
3705 } | |
3706 return true; | |
3707 } | |
3708 obj = this.get_node(obj); | |
3709 if(!obj || obj.id === $.jstree.root) { return false; } | |
3710 obj.text = val; | |
3711 if(this.get_node(obj, true).length) { | |
3712 this.redraw_node(obj.id); | |
3713 } | |
3714 /** | |
3715 * triggered when a node text value is changed | |
3716 * @event | |
3717 * @name set_text.jstree | |
3718 * @param {Object} obj | |
3719 * @param {String} text the new value | |
3720 */ | |
3721 this.trigger('set_text',{ "obj" : obj, "text" : val }); | |
3722 return true; | |
3723 }, | |
3724 /** | |
3725 * gets a JSON representation of a node (or the whole tree) | |
3726 * @name get_json([obj, options]) | |
3727 * @param {mixed} obj | |
3728 * @param {Object} options | |
3729 * @param {Boolean} options.no_state do not return state information | |
3730 * @param {Boolean} options.no_id do not return ID | |
3731 * @param {Boolean} options.no_children do not include children | |
3732 * @param {Boolean} options.no_data do not include node data | |
3733 * @param {Boolean} options.no_li_attr do not include LI attributes | |
3734 * @param {Boolean} options.no_a_attr do not include A attributes | |
3735 * @param {Boolean} options.flat return flat JSON instead of nested | |
3736 * @return {Object} | |
3737 */ | |
3738 get_json : function (obj, options, flat) { | |
3739 obj = this.get_node(obj || $.jstree.root); | |
3740 if(!obj) { return false; } | |
3741 if(options && options.flat && !flat) { flat = []; } | |
3742 var tmp = { | |
3743 'id' : obj.id, | |
3744 'text' : obj.text, | |
3745 'icon' : this.get_icon(obj), | |
3746 'li_attr' : $.extend(true, {}, obj.li_attr), | |
3747 'a_attr' : $.extend(true, {}, obj.a_attr), | |
3748 'state' : {}, | |
3749 'data' : options && options.no_data ? false : $.extend(true, $.isArray(obj.data)?[]:{}, obj.data) | |
3750 //( this.get_node(obj, true).length ? this.get_node(obj, true).data() : obj.data ), | |
3751 }, i, j; | |
3752 if(options && options.flat) { | |
3753 tmp.parent = obj.parent; | |
3754 } | |
3755 else { | |
3756 tmp.children = []; | |
3757 } | |
3758 if(!options || !options.no_state) { | |
3759 for(i in obj.state) { | |
3760 if(obj.state.hasOwnProperty(i)) { | |
3761 tmp.state[i] = obj.state[i]; | |
3762 } | |
3763 } | |
3764 } else { | |
3765 delete tmp.state; | |
3766 } | |
3767 if(options && options.no_li_attr) { | |
3768 delete tmp.li_attr; | |
3769 } | |
3770 if(options && options.no_a_attr) { | |
3771 delete tmp.a_attr; | |
3772 } | |
3773 if(options && options.no_id) { | |
3774 delete tmp.id; | |
3775 if(tmp.li_attr && tmp.li_attr.id) { | |
3776 delete tmp.li_attr.id; | |
3777 } | |
3778 if(tmp.a_attr && tmp.a_attr.id) { | |
3779 delete tmp.a_attr.id; | |
3780 } | |
3781 } | |
3782 if(options && options.flat && obj.id !== $.jstree.root) { | |
3783 flat.push(tmp); | |
3784 } | |
3785 if(!options || !options.no_children) { | |
3786 for(i = 0, j = obj.children.length; i < j; i++) { | |
3787 if(options && options.flat) { | |
3788 this.get_json(obj.children[i], options, flat); | |
3789 } | |
3790 else { | |
3791 tmp.children.push(this.get_json(obj.children[i], options)); | |
3792 } | |
3793 } | |
3794 } | |
3795 return options && options.flat ? flat : (obj.id === $.jstree.root ? tmp.children : tmp); | |
3796 }, | |
3797 /** | |
3798 * create a new node (do not confuse with load_node) | |
3799 * @name create_node([par, node, pos, callback, is_loaded]) | |
3800 * @param {mixed} par the parent node (to create a root node use either "#" (string) or `null`) | |
3801 * @param {mixed} node the data for the new node (a valid JSON object, or a simple string with the name) | |
3802 * @param {mixed} pos the index at which to insert the node, "first" and "last" are also supported, default is "last" | |
3803 * @param {Function} callback a function to be called once the node is created | |
3804 * @param {Boolean} is_loaded internal argument indicating if the parent node was succesfully loaded | |
3805 * @return {String} the ID of the newly create node | |
3806 * @trigger model.jstree, create_node.jstree | |
3807 */ | |
3808 create_node : function (par, node, pos, callback, is_loaded) { | |
3809 if(par === null) { par = $.jstree.root; } | |
3810 par = this.get_node(par); | |
3811 if(!par) { return false; } | |
3812 pos = pos === undefined ? "last" : pos; | |
3813 if(!pos.toString().match(/^(before|after)$/) && !is_loaded && !this.is_loaded(par)) { | |
3814 return this.load_node(par, function () { this.create_node(par, node, pos, callback, true); }); | |
3815 } | |
3816 if(!node) { node = { "text" : this.get_string('New node') }; } | |
3817 if(typeof node === "string") { | |
3818 node = { "text" : node }; | |
3819 } else { | |
3820 node = $.extend(true, {}, node); | |
3821 } | |
3822 if(node.text === undefined) { node.text = this.get_string('New node'); } | |
3823 var tmp, dpc, i, j; | |
3824 | |
3825 if(par.id === $.jstree.root) { | |
3826 if(pos === "before") { pos = "first"; } | |
3827 if(pos === "after") { pos = "last"; } | |
3828 } | |
3829 switch(pos) { | |
3830 case "before": | |
3831 tmp = this.get_node(par.parent); | |
3832 pos = $.inArray(par.id, tmp.children); | |
3833 par = tmp; | |
3834 break; | |
3835 case "after" : | |
3836 tmp = this.get_node(par.parent); | |
3837 pos = $.inArray(par.id, tmp.children) + 1; | |
3838 par = tmp; | |
3839 break; | |
3840 case "inside": | |
3841 case "first": | |
3842 pos = 0; | |
3843 break; | |
3844 case "last": | |
3845 pos = par.children.length; | |
3846 break; | |
3847 default: | |
3848 if(!pos) { pos = 0; } | |
3849 break; | |
3850 } | |
3851 if(pos > par.children.length) { pos = par.children.length; } | |
3852 if(!node.id) { node.id = true; } | |
3853 if(!this.check("create_node", node, par, pos)) { | |
3854 this.settings.core.error.call(this, this._data.core.last_error); | |
3855 return false; | |
3856 } | |
3857 if(node.id === true) { delete node.id; } | |
3858 node = this._parse_model_from_json(node, par.id, par.parents.concat()); | |
3859 if(!node) { return false; } | |
3860 tmp = this.get_node(node); | |
3861 dpc = []; | |
3862 dpc.push(node); | |
3863 dpc = dpc.concat(tmp.children_d); | |
3864 this.trigger('model', { "nodes" : dpc, "parent" : par.id }); | |
3865 | |
3866 par.children_d = par.children_d.concat(dpc); | |
3867 for(i = 0, j = par.parents.length; i < j; i++) { | |
3868 this._model.data[par.parents[i]].children_d = this._model.data[par.parents[i]].children_d.concat(dpc); | |
3869 } | |
3870 node = tmp; | |
3871 tmp = []; | |
3872 for(i = 0, j = par.children.length; i < j; i++) { | |
3873 tmp[i >= pos ? i+1 : i] = par.children[i]; | |
3874 } | |
3875 tmp[pos] = node.id; | |
3876 par.children = tmp; | |
3877 | |
3878 this.redraw_node(par, true); | |
3879 /** | |
3880 * triggered when a node is created | |
3881 * @event | |
3882 * @name create_node.jstree | |
3883 * @param {Object} node | |
3884 * @param {String} parent the parent's ID | |
3885 * @param {Number} position the position of the new node among the parent's children | |
3886 */ | |
3887 this.trigger('create_node', { "node" : this.get_node(node), "parent" : par.id, "position" : pos }); | |
3888 if(callback) { callback.call(this, this.get_node(node)); } | |
3889 return node.id; | |
3890 }, | |
3891 /** | |
3892 * set the text value of a node | |
3893 * @name rename_node(obj, val) | |
3894 * @param {mixed} obj the node, you can pass an array to rename multiple nodes to the same name | |
3895 * @param {String} val the new text value | |
3896 * @return {Boolean} | |
3897 * @trigger rename_node.jstree | |
3898 */ | |
3899 rename_node : function (obj, val) { | |
3900 var t1, t2, old; | |
3901 if($.isArray(obj)) { | |
3902 obj = obj.slice(); | |
3903 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
3904 this.rename_node(obj[t1], val); | |
3905 } | |
3906 return true; | |
3907 } | |
3908 obj = this.get_node(obj); | |
3909 if(!obj || obj.id === $.jstree.root) { return false; } | |
3910 old = obj.text; | |
3911 if(!this.check("rename_node", obj, this.get_parent(obj), val)) { | |
3912 this.settings.core.error.call(this, this._data.core.last_error); | |
3913 return false; | |
3914 } | |
3915 this.set_text(obj, val); // .apply(this, Array.prototype.slice.call(arguments)) | |
3916 /** | |
3917 * triggered when a node is renamed | |
3918 * @event | |
3919 * @name rename_node.jstree | |
3920 * @param {Object} node | |
3921 * @param {String} text the new value | |
3922 * @param {String} old the old value | |
3923 */ | |
3924 this.trigger('rename_node', { "node" : obj, "text" : val, "old" : old }); | |
3925 return true; | |
3926 }, | |
3927 /** | |
3928 * remove a node | |
3929 * @name delete_node(obj) | |
3930 * @param {mixed} obj the node, you can pass an array to delete multiple nodes | |
3931 * @return {Boolean} | |
3932 * @trigger delete_node.jstree, changed.jstree | |
3933 */ | |
3934 delete_node : function (obj) { | |
3935 var t1, t2, par, pos, tmp, i, j, k, l, c, top, lft; | |
3936 if($.isArray(obj)) { | |
3937 obj = obj.slice(); | |
3938 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
3939 this.delete_node(obj[t1]); | |
3940 } | |
3941 return true; | |
3942 } | |
3943 obj = this.get_node(obj); | |
3944 if(!obj || obj.id === $.jstree.root) { return false; } | |
3945 par = this.get_node(obj.parent); | |
3946 pos = $.inArray(obj.id, par.children); | |
3947 c = false; | |
3948 if(!this.check("delete_node", obj, par, pos)) { | |
3949 this.settings.core.error.call(this, this._data.core.last_error); | |
3950 return false; | |
3951 } | |
3952 if(pos !== -1) { | |
3953 par.children = $.vakata.array_remove(par.children, pos); | |
3954 } | |
3955 tmp = obj.children_d.concat([]); | |
3956 tmp.push(obj.id); | |
3957 for(i = 0, j = obj.parents.length; i < j; i++) { | |
3958 this._model.data[obj.parents[i]].children_d = $.vakata.array_filter(this._model.data[obj.parents[i]].children_d, function (v) { | |
3959 return $.inArray(v, tmp) === -1; | |
3960 }); | |
3961 } | |
3962 for(k = 0, l = tmp.length; k < l; k++) { | |
3963 if(this._model.data[tmp[k]].state.selected) { | |
3964 c = true; | |
3965 break; | |
3966 } | |
3967 } | |
3968 if (c) { | |
3969 this._data.core.selected = $.vakata.array_filter(this._data.core.selected, function (v) { | |
3970 return $.inArray(v, tmp) === -1; | |
3971 }); | |
3972 } | |
3973 /** | |
3974 * triggered when a node is deleted | |
3975 * @event | |
3976 * @name delete_node.jstree | |
3977 * @param {Object} node | |
3978 * @param {String} parent the parent's ID | |
3979 */ | |
3980 this.trigger('delete_node', { "node" : obj, "parent" : par.id }); | |
3981 if(c) { | |
3982 this.trigger('changed', { 'action' : 'delete_node', 'node' : obj, 'selected' : this._data.core.selected, 'parent' : par.id }); | |
3983 } | |
3984 for(k = 0, l = tmp.length; k < l; k++) { | |
3985 delete this._model.data[tmp[k]]; | |
3986 } | |
3987 if($.inArray(this._data.core.focused, tmp) !== -1) { | |
3988 this._data.core.focused = null; | |
3989 top = this.element[0].scrollTop; | |
3990 lft = this.element[0].scrollLeft; | |
3991 if(par.id === $.jstree.root) { | |
3992 if (this._model.data[$.jstree.root].children[0]) { | |
3993 this.get_node(this._model.data[$.jstree.root].children[0], true).children('.jstree-anchor').focus(); | |
3994 } | |
3995 } | |
3996 else { | |
3997 this.get_node(par, true).children('.jstree-anchor').focus(); | |
3998 } | |
3999 this.element[0].scrollTop = top; | |
4000 this.element[0].scrollLeft = lft; | |
4001 } | |
4002 this.redraw_node(par, true); | |
4003 return true; | |
4004 }, | |
4005 /** | |
4006 * check if an operation is premitted on the tree. Used internally. | |
4007 * @private | |
4008 * @name check(chk, obj, par, pos) | |
4009 * @param {String} chk the operation to check, can be "create_node", "rename_node", "delete_node", "copy_node" or "move_node" | |
4010 * @param {mixed} obj the node | |
4011 * @param {mixed} par the parent | |
4012 * @param {mixed} pos the position to insert at, or if "rename_node" - the new name | |
4013 * @param {mixed} more some various additional information, for example if a "move_node" operations is triggered by DND this will be the hovered node | |
4014 * @return {Boolean} | |
4015 */ | |
4016 check : function (chk, obj, par, pos, more) { | |
4017 obj = obj && obj.id ? obj : this.get_node(obj); | |
4018 par = par && par.id ? par : this.get_node(par); | |
4019 var tmp = chk.match(/^move_node|copy_node|create_node$/i) ? par : obj, | |
4020 chc = this.settings.core.check_callback; | |
4021 if(chk === "move_node" || chk === "copy_node") { | |
4022 if((!more || !more.is_multi) && (obj.id === par.id || (chk === "move_node" && $.inArray(obj.id, par.children) === pos) || $.inArray(par.id, obj.children_d) !== -1)) { | |
4023 this._data.core.last_error = { 'error' : 'check', 'plugin' : 'core', 'id' : 'core_01', 'reason' : 'Moving parent inside child', 'data' : JSON.stringify({ 'chk' : chk, 'pos' : pos, 'obj' : obj && obj.id ? obj.id : false, 'par' : par && par.id ? par.id : false }) }; | |
4024 return false; | |
4025 } | |
4026 } | |
4027 if(tmp && tmp.data) { tmp = tmp.data; } | |
4028 if(tmp && tmp.functions && (tmp.functions[chk] === false || tmp.functions[chk] === true)) { | |
4029 if(tmp.functions[chk] === false) { | |
4030 this._data.core.last_error = { 'error' : 'check', 'plugin' : 'core', 'id' : 'core_02', 'reason' : 'Node data prevents function: ' + chk, 'data' : JSON.stringify({ 'chk' : chk, 'pos' : pos, 'obj' : obj && obj.id ? obj.id : false, 'par' : par && par.id ? par.id : false }) }; | |
4031 } | |
4032 return tmp.functions[chk]; | |
4033 } | |
4034 if(chc === false || ($.isFunction(chc) && chc.call(this, chk, obj, par, pos, more) === false) || (chc && chc[chk] === false)) { | |
4035 this._data.core.last_error = { 'error' : 'check', 'plugin' : 'core', 'id' : 'core_03', 'reason' : 'User config for core.check_callback prevents function: ' + chk, 'data' : JSON.stringify({ 'chk' : chk, 'pos' : pos, 'obj' : obj && obj.id ? obj.id : false, 'par' : par && par.id ? par.id : false }) }; | |
4036 return false; | |
4037 } | |
4038 return true; | |
4039 }, | |
4040 /** | |
4041 * get the last error | |
4042 * @name last_error() | |
4043 * @return {Object} | |
4044 */ | |
4045 last_error : function () { | |
4046 return this._data.core.last_error; | |
4047 }, | |
4048 /** | |
4049 * move a node to a new parent | |
4050 * @name move_node(obj, par [, pos, callback, is_loaded]) | |
4051 * @param {mixed} obj the node to move, pass an array to move multiple nodes | |
4052 * @param {mixed} par the new parent | |
4053 * @param {mixed} pos the position to insert at (besides integer values, "first" and "last" are supported, as well as "before" and "after"), defaults to integer `0` | |
4054 * @param {function} callback a function to call once the move is completed, receives 3 arguments - the node, the new parent and the position | |
4055 * @param {Boolean} is_loaded internal parameter indicating if the parent node has been loaded | |
4056 * @param {Boolean} skip_redraw internal parameter indicating if the tree should be redrawn | |
4057 * @param {Boolean} instance internal parameter indicating if the node comes from another instance | |
4058 * @trigger move_node.jstree | |
4059 */ | |
4060 move_node : function (obj, par, pos, callback, is_loaded, skip_redraw, origin) { | |
4061 var t1, t2, old_par, old_pos, new_par, old_ins, is_multi, dpc, tmp, i, j, k, l, p; | |
4062 | |
4063 par = this.get_node(par); | |
4064 pos = pos === undefined ? 0 : pos; | |
4065 if(!par) { return false; } | |
4066 if(!pos.toString().match(/^(before|after)$/) && !is_loaded && !this.is_loaded(par)) { | |
4067 return this.load_node(par, function () { this.move_node(obj, par, pos, callback, true, false, origin); }); | |
4068 } | |
4069 | |
4070 if($.isArray(obj)) { | |
4071 if(obj.length === 1) { | |
4072 obj = obj[0]; | |
4073 } | |
4074 else { | |
4075 //obj = obj.slice(); | |
4076 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
4077 if((tmp = this.move_node(obj[t1], par, pos, callback, is_loaded, false, origin))) { | |
4078 par = tmp; | |
4079 pos = "after"; | |
4080 } | |
4081 } | |
4082 this.redraw(); | |
4083 return true; | |
4084 } | |
4085 } | |
4086 obj = obj && obj.id ? obj : this.get_node(obj); | |
4087 | |
4088 if(!obj || obj.id === $.jstree.root) { return false; } | |
4089 | |
4090 old_par = (obj.parent || $.jstree.root).toString(); | |
4091 new_par = (!pos.toString().match(/^(before|after)$/) || par.id === $.jstree.root) ? par : this.get_node(par.parent); | |
4092 old_ins = origin ? origin : (this._model.data[obj.id] ? this : $.jstree.reference(obj.id)); | |
4093 is_multi = !old_ins || !old_ins._id || (this._id !== old_ins._id); | |
4094 old_pos = old_ins && old_ins._id && old_par && old_ins._model.data[old_par] && old_ins._model.data[old_par].children ? $.inArray(obj.id, old_ins._model.data[old_par].children) : -1; | |
4095 if(old_ins && old_ins._id) { | |
4096 obj = old_ins._model.data[obj.id]; | |
4097 } | |
4098 | |
4099 if(is_multi) { | |
4100 if((tmp = this.copy_node(obj, par, pos, callback, is_loaded, false, origin))) { | |
4101 if(old_ins) { old_ins.delete_node(obj); } | |
4102 return tmp; | |
4103 } | |
4104 return false; | |
4105 } | |
4106 //var m = this._model.data; | |
4107 if(par.id === $.jstree.root) { | |
4108 if(pos === "before") { pos = "first"; } | |
4109 if(pos === "after") { pos = "last"; } | |
4110 } | |
4111 switch(pos) { | |
4112 case "before": | |
4113 pos = $.inArray(par.id, new_par.children); | |
4114 break; | |
4115 case "after" : | |
4116 pos = $.inArray(par.id, new_par.children) + 1; | |
4117 break; | |
4118 case "inside": | |
4119 case "first": | |
4120 pos = 0; | |
4121 break; | |
4122 case "last": | |
4123 pos = new_par.children.length; | |
4124 break; | |
4125 default: | |
4126 if(!pos) { pos = 0; } | |
4127 break; | |
4128 } | |
4129 if(pos > new_par.children.length) { pos = new_par.children.length; } | |
4130 if(!this.check("move_node", obj, new_par, pos, { 'core' : true, 'origin' : origin, 'is_multi' : (old_ins && old_ins._id && old_ins._id !== this._id), 'is_foreign' : (!old_ins || !old_ins._id) })) { | |
4131 this.settings.core.error.call(this, this._data.core.last_error); | |
4132 return false; | |
4133 } | |
4134 if(obj.parent === new_par.id) { | |
4135 dpc = new_par.children.concat(); | |
4136 tmp = $.inArray(obj.id, dpc); | |
4137 if(tmp !== -1) { | |
4138 dpc = $.vakata.array_remove(dpc, tmp); | |
4139 if(pos > tmp) { pos--; } | |
4140 } | |
4141 tmp = []; | |
4142 for(i = 0, j = dpc.length; i < j; i++) { | |
4143 tmp[i >= pos ? i+1 : i] = dpc[i]; | |
4144 } | |
4145 tmp[pos] = obj.id; | |
4146 new_par.children = tmp; | |
4147 this._node_changed(new_par.id); | |
4148 this.redraw(new_par.id === $.jstree.root); | |
4149 } | |
4150 else { | |
4151 // clean old parent and up | |
4152 tmp = obj.children_d.concat(); | |
4153 tmp.push(obj.id); | |
4154 for(i = 0, j = obj.parents.length; i < j; i++) { | |
4155 dpc = []; | |
4156 p = old_ins._model.data[obj.parents[i]].children_d; | |
4157 for(k = 0, l = p.length; k < l; k++) { | |
4158 if($.inArray(p[k], tmp) === -1) { | |
4159 dpc.push(p[k]); | |
4160 } | |
4161 } | |
4162 old_ins._model.data[obj.parents[i]].children_d = dpc; | |
4163 } | |
4164 old_ins._model.data[old_par].children = $.vakata.array_remove_item(old_ins._model.data[old_par].children, obj.id); | |
4165 | |
4166 // insert into new parent and up | |
4167 for(i = 0, j = new_par.parents.length; i < j; i++) { | |
4168 this._model.data[new_par.parents[i]].children_d = this._model.data[new_par.parents[i]].children_d.concat(tmp); | |
4169 } | |
4170 dpc = []; | |
4171 for(i = 0, j = new_par.children.length; i < j; i++) { | |
4172 dpc[i >= pos ? i+1 : i] = new_par.children[i]; | |
4173 } | |
4174 dpc[pos] = obj.id; | |
4175 new_par.children = dpc; | |
4176 new_par.children_d.push(obj.id); | |
4177 new_par.children_d = new_par.children_d.concat(obj.children_d); | |
4178 | |
4179 // update object | |
4180 obj.parent = new_par.id; | |
4181 tmp = new_par.parents.concat(); | |
4182 tmp.unshift(new_par.id); | |
4183 p = obj.parents.length; | |
4184 obj.parents = tmp; | |
4185 | |
4186 // update object children | |
4187 tmp = tmp.concat(); | |
4188 for(i = 0, j = obj.children_d.length; i < j; i++) { | |
4189 this._model.data[obj.children_d[i]].parents = this._model.data[obj.children_d[i]].parents.slice(0,p*-1); | |
4190 Array.prototype.push.apply(this._model.data[obj.children_d[i]].parents, tmp); | |
4191 } | |
4192 | |
4193 if(old_par === $.jstree.root || new_par.id === $.jstree.root) { | |
4194 this._model.force_full_redraw = true; | |
4195 } | |
4196 if(!this._model.force_full_redraw) { | |
4197 this._node_changed(old_par); | |
4198 this._node_changed(new_par.id); | |
4199 } | |
4200 if(!skip_redraw) { | |
4201 this.redraw(); | |
4202 } | |
4203 } | |
4204 if(callback) { callback.call(this, obj, new_par, pos); } | |
4205 /** | |
4206 * triggered when a node is moved | |
4207 * @event | |
4208 * @name move_node.jstree | |
4209 * @param {Object} node | |
4210 * @param {String} parent the parent's ID | |
4211 * @param {Number} position the position of the node among the parent's children | |
4212 * @param {String} old_parent the old parent of the node | |
4213 * @param {Number} old_position the old position of the node | |
4214 * @param {Boolean} is_multi do the node and new parent belong to different instances | |
4215 * @param {jsTree} old_instance the instance the node came from | |
4216 * @param {jsTree} new_instance the instance of the new parent | |
4217 */ | |
4218 this.trigger('move_node', { "node" : obj, "parent" : new_par.id, "position" : pos, "old_parent" : old_par, "old_position" : old_pos, 'is_multi' : (old_ins && old_ins._id && old_ins._id !== this._id), 'is_foreign' : (!old_ins || !old_ins._id), 'old_instance' : old_ins, 'new_instance' : this }); | |
4219 return obj.id; | |
4220 }, | |
4221 /** | |
4222 * copy a node to a new parent | |
4223 * @name copy_node(obj, par [, pos, callback, is_loaded]) | |
4224 * @param {mixed} obj the node to copy, pass an array to copy multiple nodes | |
4225 * @param {mixed} par the new parent | |
4226 * @param {mixed} pos the position to insert at (besides integer values, "first" and "last" are supported, as well as "before" and "after"), defaults to integer `0` | |
4227 * @param {function} callback a function to call once the move is completed, receives 3 arguments - the node, the new parent and the position | |
4228 * @param {Boolean} is_loaded internal parameter indicating if the parent node has been loaded | |
4229 * @param {Boolean} skip_redraw internal parameter indicating if the tree should be redrawn | |
4230 * @param {Boolean} instance internal parameter indicating if the node comes from another instance | |
4231 * @trigger model.jstree copy_node.jstree | |
4232 */ | |
4233 copy_node : function (obj, par, pos, callback, is_loaded, skip_redraw, origin) { | |
4234 var t1, t2, dpc, tmp, i, j, node, old_par, new_par, old_ins, is_multi; | |
4235 | |
4236 par = this.get_node(par); | |
4237 pos = pos === undefined ? 0 : pos; | |
4238 if(!par) { return false; } | |
4239 if(!pos.toString().match(/^(before|after)$/) && !is_loaded && !this.is_loaded(par)) { | |
4240 return this.load_node(par, function () { this.copy_node(obj, par, pos, callback, true, false, origin); }); | |
4241 } | |
4242 | |
4243 if($.isArray(obj)) { | |
4244 if(obj.length === 1) { | |
4245 obj = obj[0]; | |
4246 } | |
4247 else { | |
4248 //obj = obj.slice(); | |
4249 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
4250 if((tmp = this.copy_node(obj[t1], par, pos, callback, is_loaded, true, origin))) { | |
4251 par = tmp; | |
4252 pos = "after"; | |
4253 } | |
4254 } | |
4255 this.redraw(); | |
4256 return true; | |
4257 } | |
4258 } | |
4259 obj = obj && obj.id ? obj : this.get_node(obj); | |
4260 if(!obj || obj.id === $.jstree.root) { return false; } | |
4261 | |
4262 old_par = (obj.parent || $.jstree.root).toString(); | |
4263 new_par = (!pos.toString().match(/^(before|after)$/) || par.id === $.jstree.root) ? par : this.get_node(par.parent); | |
4264 old_ins = origin ? origin : (this._model.data[obj.id] ? this : $.jstree.reference(obj.id)); | |
4265 is_multi = !old_ins || !old_ins._id || (this._id !== old_ins._id); | |
4266 | |
4267 if(old_ins && old_ins._id) { | |
4268 obj = old_ins._model.data[obj.id]; | |
4269 } | |
4270 | |
4271 if(par.id === $.jstree.root) { | |
4272 if(pos === "before") { pos = "first"; } | |
4273 if(pos === "after") { pos = "last"; } | |
4274 } | |
4275 switch(pos) { | |
4276 case "before": | |
4277 pos = $.inArray(par.id, new_par.children); | |
4278 break; | |
4279 case "after" : | |
4280 pos = $.inArray(par.id, new_par.children) + 1; | |
4281 break; | |
4282 case "inside": | |
4283 case "first": | |
4284 pos = 0; | |
4285 break; | |
4286 case "last": | |
4287 pos = new_par.children.length; | |
4288 break; | |
4289 default: | |
4290 if(!pos) { pos = 0; } | |
4291 break; | |
4292 } | |
4293 if(pos > new_par.children.length) { pos = new_par.children.length; } | |
4294 if(!this.check("copy_node", obj, new_par, pos, { 'core' : true, 'origin' : origin, 'is_multi' : (old_ins && old_ins._id && old_ins._id !== this._id), 'is_foreign' : (!old_ins || !old_ins._id) })) { | |
4295 this.settings.core.error.call(this, this._data.core.last_error); | |
4296 return false; | |
4297 } | |
4298 node = old_ins ? old_ins.get_json(obj, { no_id : true, no_data : true, no_state : true }) : obj; | |
4299 if(!node) { return false; } | |
4300 if(node.id === true) { delete node.id; } | |
4301 node = this._parse_model_from_json(node, new_par.id, new_par.parents.concat()); | |
4302 if(!node) { return false; } | |
4303 tmp = this.get_node(node); | |
4304 if(obj && obj.state && obj.state.loaded === false) { tmp.state.loaded = false; } | |
4305 dpc = []; | |
4306 dpc.push(node); | |
4307 dpc = dpc.concat(tmp.children_d); | |
4308 this.trigger('model', { "nodes" : dpc, "parent" : new_par.id }); | |
4309 | |
4310 // insert into new parent and up | |
4311 for(i = 0, j = new_par.parents.length; i < j; i++) { | |
4312 this._model.data[new_par.parents[i]].children_d = this._model.data[new_par.parents[i]].children_d.concat(dpc); | |
4313 } | |
4314 dpc = []; | |
4315 for(i = 0, j = new_par.children.length; i < j; i++) { | |
4316 dpc[i >= pos ? i+1 : i] = new_par.children[i]; | |
4317 } | |
4318 dpc[pos] = tmp.id; | |
4319 new_par.children = dpc; | |
4320 new_par.children_d.push(tmp.id); | |
4321 new_par.children_d = new_par.children_d.concat(tmp.children_d); | |
4322 | |
4323 if(new_par.id === $.jstree.root) { | |
4324 this._model.force_full_redraw = true; | |
4325 } | |
4326 if(!this._model.force_full_redraw) { | |
4327 this._node_changed(new_par.id); | |
4328 } | |
4329 if(!skip_redraw) { | |
4330 this.redraw(new_par.id === $.jstree.root); | |
4331 } | |
4332 if(callback) { callback.call(this, tmp, new_par, pos); } | |
4333 /** | |
4334 * triggered when a node is copied | |
4335 * @event | |
4336 * @name copy_node.jstree | |
4337 * @param {Object} node the copied node | |
4338 * @param {Object} original the original node | |
4339 * @param {String} parent the parent's ID | |
4340 * @param {Number} position the position of the node among the parent's children | |
4341 * @param {String} old_parent the old parent of the node | |
4342 * @param {Number} old_position the position of the original node | |
4343 * @param {Boolean} is_multi do the node and new parent belong to different instances | |
4344 * @param {jsTree} old_instance the instance the node came from | |
4345 * @param {jsTree} new_instance the instance of the new parent | |
4346 */ | |
4347 this.trigger('copy_node', { "node" : tmp, "original" : obj, "parent" : new_par.id, "position" : pos, "old_parent" : old_par, "old_position" : old_ins && old_ins._id && old_par && old_ins._model.data[old_par] && old_ins._model.data[old_par].children ? $.inArray(obj.id, old_ins._model.data[old_par].children) : -1,'is_multi' : (old_ins && old_ins._id && old_ins._id !== this._id), 'is_foreign' : (!old_ins || !old_ins._id), 'old_instance' : old_ins, 'new_instance' : this }); | |
4348 return tmp.id; | |
4349 }, | |
4350 /** | |
4351 * cut a node (a later call to `paste(obj)` would move the node) | |
4352 * @name cut(obj) | |
4353 * @param {mixed} obj multiple objects can be passed using an array | |
4354 * @trigger cut.jstree | |
4355 */ | |
4356 cut : function (obj) { | |
4357 if(!obj) { obj = this._data.core.selected.concat(); } | |
4358 if(!$.isArray(obj)) { obj = [obj]; } | |
4359 if(!obj.length) { return false; } | |
4360 var tmp = [], o, t1, t2; | |
4361 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
4362 o = this.get_node(obj[t1]); | |
4363 if(o && o.id && o.id !== $.jstree.root) { tmp.push(o); } | |
4364 } | |
4365 if(!tmp.length) { return false; } | |
4366 ccp_node = tmp; | |
4367 ccp_inst = this; | |
4368 ccp_mode = 'move_node'; | |
4369 /** | |
4370 * triggered when nodes are added to the buffer for moving | |
4371 * @event | |
4372 * @name cut.jstree | |
4373 * @param {Array} node | |
4374 */ | |
4375 this.trigger('cut', { "node" : obj }); | |
4376 }, | |
4377 /** | |
4378 * copy a node (a later call to `paste(obj)` would copy the node) | |
4379 * @name copy(obj) | |
4380 * @param {mixed} obj multiple objects can be passed using an array | |
4381 * @trigger copy.jstree | |
4382 */ | |
4383 copy : function (obj) { | |
4384 if(!obj) { obj = this._data.core.selected.concat(); } | |
4385 if(!$.isArray(obj)) { obj = [obj]; } | |
4386 if(!obj.length) { return false; } | |
4387 var tmp = [], o, t1, t2; | |
4388 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
4389 o = this.get_node(obj[t1]); | |
4390 if(o && o.id && o.id !== $.jstree.root) { tmp.push(o); } | |
4391 } | |
4392 if(!tmp.length) { return false; } | |
4393 ccp_node = tmp; | |
4394 ccp_inst = this; | |
4395 ccp_mode = 'copy_node'; | |
4396 /** | |
4397 * triggered when nodes are added to the buffer for copying | |
4398 * @event | |
4399 * @name copy.jstree | |
4400 * @param {Array} node | |
4401 */ | |
4402 this.trigger('copy', { "node" : obj }); | |
4403 }, | |
4404 /** | |
4405 * get the current buffer (any nodes that are waiting for a paste operation) | |
4406 * @name get_buffer() | |
4407 * @return {Object} an object consisting of `mode` ("copy_node" or "move_node"), `node` (an array of objects) and `inst` (the instance) | |
4408 */ | |
4409 get_buffer : function () { | |
4410 return { 'mode' : ccp_mode, 'node' : ccp_node, 'inst' : ccp_inst }; | |
4411 }, | |
4412 /** | |
4413 * check if there is something in the buffer to paste | |
4414 * @name can_paste() | |
4415 * @return {Boolean} | |
4416 */ | |
4417 can_paste : function () { | |
4418 return ccp_mode !== false && ccp_node !== false; // && ccp_inst._model.data[ccp_node]; | |
4419 }, | |
4420 /** | |
4421 * copy or move the previously cut or copied nodes to a new parent | |
4422 * @name paste(obj [, pos]) | |
4423 * @param {mixed} obj the new parent | |
4424 * @param {mixed} pos the position to insert at (besides integer, "first" and "last" are supported), defaults to integer `0` | |
4425 * @trigger paste.jstree | |
4426 */ | |
4427 paste : function (obj, pos) { | |
4428 obj = this.get_node(obj); | |
4429 if(!obj || !ccp_mode || !ccp_mode.match(/^(copy_node|move_node)$/) || !ccp_node) { return false; } | |
4430 if(this[ccp_mode](ccp_node, obj, pos, false, false, false, ccp_inst)) { | |
4431 /** | |
4432 * triggered when paste is invoked | |
4433 * @event | |
4434 * @name paste.jstree | |
4435 * @param {String} parent the ID of the receiving node | |
4436 * @param {Array} node the nodes in the buffer | |
4437 * @param {String} mode the performed operation - "copy_node" or "move_node" | |
4438 */ | |
4439 this.trigger('paste', { "parent" : obj.id, "node" : ccp_node, "mode" : ccp_mode }); | |
4440 } | |
4441 ccp_node = false; | |
4442 ccp_mode = false; | |
4443 ccp_inst = false; | |
4444 }, | |
4445 /** | |
4446 * clear the buffer of previously copied or cut nodes | |
4447 * @name clear_buffer() | |
4448 * @trigger clear_buffer.jstree | |
4449 */ | |
4450 clear_buffer : function () { | |
4451 ccp_node = false; | |
4452 ccp_mode = false; | |
4453 ccp_inst = false; | |
4454 /** | |
4455 * triggered when the copy / cut buffer is cleared | |
4456 * @event | |
4457 * @name clear_buffer.jstree | |
4458 */ | |
4459 this.trigger('clear_buffer'); | |
4460 }, | |
4461 /** | |
4462 * put a node in edit mode (input field to rename the node) | |
4463 * @name edit(obj [, default_text, callback]) | |
4464 * @param {mixed} obj | |
4465 * @param {String} default_text the text to populate the input with (if omitted or set to a non-string value the node's text value is used) | |
4466 * @param {Function} callback a function to be called once the text box is blurred, it is called in the instance's scope and receives the node, a status parameter (true if the rename is successful, false otherwise) and a boolean indicating if the user cancelled the edit. You can access the node's title using .text | |
4467 */ | |
4468 edit : function (obj, default_text, callback) { | |
4469 var rtl, w, a, s, t, h1, h2, fn, tmp, cancel = false; | |
4470 obj = this.get_node(obj); | |
4471 if(!obj) { return false; } | |
4472 if(!this.check("edit", obj, this.get_parent(obj))) { | |
4473 this.settings.core.error.call(this, this._data.core.last_error); | |
4474 return false; | |
4475 } | |
4476 tmp = obj; | |
4477 default_text = typeof default_text === 'string' ? default_text : obj.text; | |
4478 this.set_text(obj, ""); | |
4479 obj = this._open_to(obj); | |
4480 tmp.text = default_text; | |
4481 | |
4482 rtl = this._data.core.rtl; | |
4483 w = this.element.width(); | |
4484 this._data.core.focused = tmp.id; | |
4485 a = obj.children('.jstree-anchor').focus(); | |
4486 s = $('<span>'); | |
4487 /*! | |
4488 oi = obj.children("i:visible"), | |
4489 ai = a.children("i:visible"), | |
4490 w1 = oi.width() * oi.length, | |
4491 w2 = ai.width() * ai.length, | |
4492 */ | |
4493 t = default_text; | |
4494 h1 = $("<"+"div />", { css : { "position" : "absolute", "top" : "-200px", "left" : (rtl ? "0px" : "-1000px"), "visibility" : "hidden" } }).appendTo("body"); | |
4495 h2 = $("<"+"input />", { | |
4496 "value" : t, | |
4497 "class" : "jstree-rename-input", | |
4498 // "size" : t.length, | |
4499 "css" : { | |
4500 "padding" : "0", | |
4501 "border" : "1px solid silver", | |
4502 "box-sizing" : "border-box", | |
4503 "display" : "inline-block", | |
4504 "height" : (this._data.core.li_height) + "px", | |
4505 "lineHeight" : (this._data.core.li_height) + "px", | |
4506 "width" : "150px" // will be set a bit further down | |
4507 }, | |
4508 "blur" : $.proxy(function (e) { | |
4509 e.stopImmediatePropagation(); | |
4510 e.preventDefault(); | |
4511 var i = s.children(".jstree-rename-input"), | |
4512 v = i.val(), | |
4513 f = this.settings.core.force_text, | |
4514 nv; | |
4515 if(v === "") { v = t; } | |
4516 h1.remove(); | |
4517 s.replaceWith(a); | |
4518 s.remove(); | |
4519 t = f ? t : $('<div></div>').append($.parseHTML(t)).html(); | |
4520 this.set_text(obj, t); | |
4521 nv = !!this.rename_node(obj, f ? $('<div></div>').text(v).text() : $('<div></div>').append($.parseHTML(v)).html()); | |
4522 if(!nv) { | |
4523 this.set_text(obj, t); // move this up? and fix #483 | |
4524 } | |
4525 this._data.core.focused = tmp.id; | |
4526 setTimeout($.proxy(function () { | |
4527 var node = this.get_node(tmp.id, true); | |
4528 if(node.length) { | |
4529 this._data.core.focused = tmp.id; | |
4530 node.children('.jstree-anchor').focus(); | |
4531 } | |
4532 }, this), 0); | |
4533 if(callback) { | |
4534 callback.call(this, tmp, nv, cancel); | |
4535 } | |
4536 h2 = null; | |
4537 }, this), | |
4538 "keydown" : function (e) { | |
4539 var key = e.which; | |
4540 if(key === 27) { | |
4541 cancel = true; | |
4542 this.value = t; | |
4543 } | |
4544 if(key === 27 || key === 13 || key === 37 || key === 38 || key === 39 || key === 40 || key === 32) { | |
4545 e.stopImmediatePropagation(); | |
4546 } | |
4547 if(key === 27 || key === 13) { | |
4548 e.preventDefault(); | |
4549 this.blur(); | |
4550 } | |
4551 }, | |
4552 "click" : function (e) { e.stopImmediatePropagation(); }, | |
4553 "mousedown" : function (e) { e.stopImmediatePropagation(); }, | |
4554 "keyup" : function (e) { | |
4555 h2.width(Math.min(h1.text("pW" + this.value).width(),w)); | |
4556 }, | |
4557 "keypress" : function(e) { | |
4558 if(e.which === 13) { return false; } | |
4559 } | |
4560 }); | |
4561 fn = { | |
4562 fontFamily : a.css('fontFamily') || '', | |
4563 fontSize : a.css('fontSize') || '', | |
4564 fontWeight : a.css('fontWeight') || '', | |
4565 fontStyle : a.css('fontStyle') || '', | |
4566 fontStretch : a.css('fontStretch') || '', | |
4567 fontVariant : a.css('fontVariant') || '', | |
4568 letterSpacing : a.css('letterSpacing') || '', | |
4569 wordSpacing : a.css('wordSpacing') || '' | |
4570 }; | |
4571 s.attr('class', a.attr('class')).append(a.contents().clone()).append(h2); | |
4572 a.replaceWith(s); | |
4573 h1.css(fn); | |
4574 h2.css(fn).width(Math.min(h1.text("pW" + h2[0].value).width(),w))[0].select(); | |
4575 $(document).one('mousedown.jstree touchstart.jstree dnd_start.vakata', function (e) { | |
4576 if (h2 && e.target !== h2) { | |
4577 $(h2).blur(); | |
4578 } | |
4579 }); | |
4580 }, | |
4581 | |
4582 | |
4583 /** | |
4584 * changes the theme | |
4585 * @name set_theme(theme_name [, theme_url]) | |
4586 * @param {String} theme_name the name of the new theme to apply | |
4587 * @param {mixed} theme_url the location of the CSS file for this theme. Omit or set to `false` if you manually included the file. Set to `true` to autoload from the `core.themes.dir` directory. | |
4588 * @trigger set_theme.jstree | |
4589 */ | |
4590 set_theme : function (theme_name, theme_url) { | |
4591 if(!theme_name) { return false; } | |
4592 if(theme_url === true) { | |
4593 var dir = this.settings.core.themes.dir; | |
4594 if(!dir) { dir = $.jstree.path + '/themes'; } | |
4595 theme_url = dir + '/' + theme_name + '/style.css'; | |
4596 } | |
4597 if(theme_url && $.inArray(theme_url, themes_loaded) === -1) { | |
4598 $('head').append('<'+'link rel="stylesheet" href="' + theme_url + '" type="text/css" />'); | |
4599 themes_loaded.push(theme_url); | |
4600 } | |
4601 if(this._data.core.themes.name) { | |
4602 this.element.removeClass('jstree-' + this._data.core.themes.name); | |
4603 } | |
4604 this._data.core.themes.name = theme_name; | |
4605 this.element.addClass('jstree-' + theme_name); | |
4606 this.element[this.settings.core.themes.responsive ? 'addClass' : 'removeClass' ]('jstree-' + theme_name + '-responsive'); | |
4607 /** | |
4608 * triggered when a theme is set | |
4609 * @event | |
4610 * @name set_theme.jstree | |
4611 * @param {String} theme the new theme | |
4612 */ | |
4613 this.trigger('set_theme', { 'theme' : theme_name }); | |
4614 }, | |
4615 /** | |
4616 * gets the name of the currently applied theme name | |
4617 * @name get_theme() | |
4618 * @return {String} | |
4619 */ | |
4620 get_theme : function () { return this._data.core.themes.name; }, | |
4621 /** | |
4622 * changes the theme variant (if the theme has variants) | |
4623 * @name set_theme_variant(variant_name) | |
4624 * @param {String|Boolean} variant_name the variant to apply (if `false` is used the current variant is removed) | |
4625 */ | |
4626 set_theme_variant : function (variant_name) { | |
4627 if(this._data.core.themes.variant) { | |
4628 this.element.removeClass('jstree-' + this._data.core.themes.name + '-' + this._data.core.themes.variant); | |
4629 } | |
4630 this._data.core.themes.variant = variant_name; | |
4631 if(variant_name) { | |
4632 this.element.addClass('jstree-' + this._data.core.themes.name + '-' + this._data.core.themes.variant); | |
4633 } | |
4634 }, | |
4635 /** | |
4636 * gets the name of the currently applied theme variant | |
4637 * @name get_theme() | |
4638 * @return {String} | |
4639 */ | |
4640 get_theme_variant : function () { return this._data.core.themes.variant; }, | |
4641 /** | |
4642 * shows a striped background on the container (if the theme supports it) | |
4643 * @name show_stripes() | |
4644 */ | |
4645 show_stripes : function () { | |
4646 this._data.core.themes.stripes = true; | |
4647 this.get_container_ul().addClass("jstree-striped"); | |
4648 /** | |
4649 * triggered when stripes are shown | |
4650 * @event | |
4651 * @name show_stripes.jstree | |
4652 */ | |
4653 this.trigger('show_stripes'); | |
4654 }, | |
4655 /** | |
4656 * hides the striped background on the container | |
4657 * @name hide_stripes() | |
4658 */ | |
4659 hide_stripes : function () { | |
4660 this._data.core.themes.stripes = false; | |
4661 this.get_container_ul().removeClass("jstree-striped"); | |
4662 /** | |
4663 * triggered when stripes are hidden | |
4664 * @event | |
4665 * @name hide_stripes.jstree | |
4666 */ | |
4667 this.trigger('hide_stripes'); | |
4668 }, | |
4669 /** | |
4670 * toggles the striped background on the container | |
4671 * @name toggle_stripes() | |
4672 */ | |
4673 toggle_stripes : function () { if(this._data.core.themes.stripes) { this.hide_stripes(); } else { this.show_stripes(); } }, | |
4674 /** | |
4675 * shows the connecting dots (if the theme supports it) | |
4676 * @name show_dots() | |
4677 */ | |
4678 show_dots : function () { | |
4679 this._data.core.themes.dots = true; | |
4680 this.get_container_ul().removeClass("jstree-no-dots"); | |
4681 /** | |
4682 * triggered when dots are shown | |
4683 * @event | |
4684 * @name show_dots.jstree | |
4685 */ | |
4686 this.trigger('show_dots'); | |
4687 }, | |
4688 /** | |
4689 * hides the connecting dots | |
4690 * @name hide_dots() | |
4691 */ | |
4692 hide_dots : function () { | |
4693 this._data.core.themes.dots = false; | |
4694 this.get_container_ul().addClass("jstree-no-dots"); | |
4695 /** | |
4696 * triggered when dots are hidden | |
4697 * @event | |
4698 * @name hide_dots.jstree | |
4699 */ | |
4700 this.trigger('hide_dots'); | |
4701 }, | |
4702 /** | |
4703 * toggles the connecting dots | |
4704 * @name toggle_dots() | |
4705 */ | |
4706 toggle_dots : function () { if(this._data.core.themes.dots) { this.hide_dots(); } else { this.show_dots(); } }, | |
4707 /** | |
4708 * show the node icons | |
4709 * @name show_icons() | |
4710 */ | |
4711 show_icons : function () { | |
4712 this._data.core.themes.icons = true; | |
4713 this.get_container_ul().removeClass("jstree-no-icons"); | |
4714 /** | |
4715 * triggered when icons are shown | |
4716 * @event | |
4717 * @name show_icons.jstree | |
4718 */ | |
4719 this.trigger('show_icons'); | |
4720 }, | |
4721 /** | |
4722 * hide the node icons | |
4723 * @name hide_icons() | |
4724 */ | |
4725 hide_icons : function () { | |
4726 this._data.core.themes.icons = false; | |
4727 this.get_container_ul().addClass("jstree-no-icons"); | |
4728 /** | |
4729 * triggered when icons are hidden | |
4730 * @event | |
4731 * @name hide_icons.jstree | |
4732 */ | |
4733 this.trigger('hide_icons'); | |
4734 }, | |
4735 /** | |
4736 * toggle the node icons | |
4737 * @name toggle_icons() | |
4738 */ | |
4739 toggle_icons : function () { if(this._data.core.themes.icons) { this.hide_icons(); } else { this.show_icons(); } }, | |
4740 /** | |
4741 * show the node ellipsis | |
4742 * @name show_icons() | |
4743 */ | |
4744 show_ellipsis : function () { | |
4745 this._data.core.themes.ellipsis = true; | |
4746 this.get_container_ul().addClass("jstree-ellipsis"); | |
4747 /** | |
4748 * triggered when ellisis is shown | |
4749 * @event | |
4750 * @name show_ellipsis.jstree | |
4751 */ | |
4752 this.trigger('show_ellipsis'); | |
4753 }, | |
4754 /** | |
4755 * hide the node ellipsis | |
4756 * @name hide_ellipsis() | |
4757 */ | |
4758 hide_ellipsis : function () { | |
4759 this._data.core.themes.ellipsis = false; | |
4760 this.get_container_ul().removeClass("jstree-ellipsis"); | |
4761 /** | |
4762 * triggered when ellisis is hidden | |
4763 * @event | |
4764 * @name hide_ellipsis.jstree | |
4765 */ | |
4766 this.trigger('hide_ellipsis'); | |
4767 }, | |
4768 /** | |
4769 * toggle the node ellipsis | |
4770 * @name toggle_icons() | |
4771 */ | |
4772 toggle_ellipsis : function () { if(this._data.core.themes.ellipsis) { this.hide_ellipsis(); } else { this.show_ellipsis(); } }, | |
4773 /** | |
4774 * set the node icon for a node | |
4775 * @name set_icon(obj, icon) | |
4776 * @param {mixed} obj | |
4777 * @param {String} icon the new icon - can be a path to an icon or a className, if using an image that is in the current directory use a `./` prefix, otherwise it will be detected as a class | |
4778 */ | |
4779 set_icon : function (obj, icon) { | |
4780 var t1, t2, dom, old; | |
4781 if($.isArray(obj)) { | |
4782 obj = obj.slice(); | |
4783 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
4784 this.set_icon(obj[t1], icon); | |
4785 } | |
4786 return true; | |
4787 } | |
4788 obj = this.get_node(obj); | |
4789 if(!obj || obj.id === $.jstree.root) { return false; } | |
4790 old = obj.icon; | |
4791 obj.icon = icon === true || icon === null || icon === undefined || icon === '' ? true : icon; | |
4792 dom = this.get_node(obj, true).children(".jstree-anchor").children(".jstree-themeicon"); | |
4793 if(icon === false) { | |
4794 dom.removeClass('jstree-themeicon-custom ' + old).css("background","").removeAttr("rel"); | |
4795 this.hide_icon(obj); | |
4796 } | |
4797 else if(icon === true || icon === null || icon === undefined || icon === '') { | |
4798 dom.removeClass('jstree-themeicon-custom ' + old).css("background","").removeAttr("rel"); | |
4799 if(old === false) { this.show_icon(obj); } | |
4800 } | |
4801 else if(icon.indexOf("/") === -1 && icon.indexOf(".") === -1) { | |
4802 dom.removeClass(old).css("background",""); | |
4803 dom.addClass(icon + ' jstree-themeicon-custom').attr("rel",icon); | |
4804 if(old === false) { this.show_icon(obj); } | |
4805 } | |
4806 else { | |
4807 dom.removeClass(old).css("background",""); | |
4808 dom.addClass('jstree-themeicon-custom').css("background", "url('" + icon + "') center center no-repeat").attr("rel",icon); | |
4809 if(old === false) { this.show_icon(obj); } | |
4810 } | |
4811 return true; | |
4812 }, | |
4813 /** | |
4814 * get the node icon for a node | |
4815 * @name get_icon(obj) | |
4816 * @param {mixed} obj | |
4817 * @return {String} | |
4818 */ | |
4819 get_icon : function (obj) { | |
4820 obj = this.get_node(obj); | |
4821 return (!obj || obj.id === $.jstree.root) ? false : obj.icon; | |
4822 }, | |
4823 /** | |
4824 * hide the icon on an individual node | |
4825 * @name hide_icon(obj) | |
4826 * @param {mixed} obj | |
4827 */ | |
4828 hide_icon : function (obj) { | |
4829 var t1, t2; | |
4830 if($.isArray(obj)) { | |
4831 obj = obj.slice(); | |
4832 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
4833 this.hide_icon(obj[t1]); | |
4834 } | |
4835 return true; | |
4836 } | |
4837 obj = this.get_node(obj); | |
4838 if(!obj || obj === $.jstree.root) { return false; } | |
4839 obj.icon = false; | |
4840 this.get_node(obj, true).children(".jstree-anchor").children(".jstree-themeicon").addClass('jstree-themeicon-hidden'); | |
4841 return true; | |
4842 }, | |
4843 /** | |
4844 * show the icon on an individual node | |
4845 * @name show_icon(obj) | |
4846 * @param {mixed} obj | |
4847 */ | |
4848 show_icon : function (obj) { | |
4849 var t1, t2, dom; | |
4850 if($.isArray(obj)) { | |
4851 obj = obj.slice(); | |
4852 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
4853 this.show_icon(obj[t1]); | |
4854 } | |
4855 return true; | |
4856 } | |
4857 obj = this.get_node(obj); | |
4858 if(!obj || obj === $.jstree.root) { return false; } | |
4859 dom = this.get_node(obj, true); | |
4860 obj.icon = dom.length ? dom.children(".jstree-anchor").children(".jstree-themeicon").attr('rel') : true; | |
4861 if(!obj.icon) { obj.icon = true; } | |
4862 dom.children(".jstree-anchor").children(".jstree-themeicon").removeClass('jstree-themeicon-hidden'); | |
4863 return true; | |
4864 } | |
4865 }; | |
4866 | |
4867 // helpers | |
4868 $.vakata = {}; | |
4869 // collect attributes | |
4870 $.vakata.attributes = function(node, with_values) { | |
4871 node = $(node)[0]; | |
4872 var attr = with_values ? {} : []; | |
4873 if(node && node.attributes) { | |
4874 $.each(node.attributes, function (i, v) { | |
4875 if($.inArray(v.name.toLowerCase(),['style','contenteditable','hasfocus','tabindex']) !== -1) { return; } | |
4876 if(v.value !== null && $.trim(v.value) !== '') { | |
4877 if(with_values) { attr[v.name] = v.value; } | |
4878 else { attr.push(v.name); } | |
4879 } | |
4880 }); | |
4881 } | |
4882 return attr; | |
4883 }; | |
4884 $.vakata.array_unique = function(array) { | |
4885 var a = [], i, j, l, o = {}; | |
4886 for(i = 0, l = array.length; i < l; i++) { | |
4887 if(o[array[i]] === undefined) { | |
4888 a.push(array[i]); | |
4889 o[array[i]] = true; | |
4890 } | |
4891 } | |
4892 return a; | |
4893 }; | |
4894 // remove item from array | |
4895 $.vakata.array_remove = function(array, from) { | |
4896 array.splice(from, 1); | |
4897 return array; | |
4898 //var rest = array.slice((to || from) + 1 || array.length); | |
4899 //array.length = from < 0 ? array.length + from : from; | |
4900 //array.push.apply(array, rest); | |
4901 //return array; | |
4902 }; | |
4903 // remove item from array | |
4904 $.vakata.array_remove_item = function(array, item) { | |
4905 var tmp = $.inArray(item, array); | |
4906 return tmp !== -1 ? $.vakata.array_remove(array, tmp) : array; | |
4907 }; | |
4908 $.vakata.array_filter = function(c,a,b,d,e) { | |
4909 if (c.filter) { | |
4910 return c.filter(a, b); | |
4911 } | |
4912 d=[]; | |
4913 for (e in c) { | |
4914 if (~~e+''===e+'' && e>=0 && a.call(b,c[e],+e,c)) { | |
4915 d.push(c[e]); | |
4916 } | |
4917 } | |
4918 return d; | |
4919 }; | |
4920 | |
4921 | |
4922 /** | |
4923 * ### Changed plugin | |
4924 * | |
4925 * This plugin adds more information to the `changed.jstree` event. The new data is contained in the `changed` event data property, and contains a lists of `selected` and `deselected` nodes. | |
4926 */ | |
4927 | |
4928 $.jstree.plugins.changed = function (options, parent) { | |
4929 var last = []; | |
4930 this.trigger = function (ev, data) { | |
4931 var i, j; | |
4932 if(!data) { | |
4933 data = {}; | |
4934 } | |
4935 if(ev.replace('.jstree','') === 'changed') { | |
4936 data.changed = { selected : [], deselected : [] }; | |
4937 var tmp = {}; | |
4938 for(i = 0, j = last.length; i < j; i++) { | |
4939 tmp[last[i]] = 1; | |
4940 } | |
4941 for(i = 0, j = data.selected.length; i < j; i++) { | |
4942 if(!tmp[data.selected[i]]) { | |
4943 data.changed.selected.push(data.selected[i]); | |
4944 } | |
4945 else { | |
4946 tmp[data.selected[i]] = 2; | |
4947 } | |
4948 } | |
4949 for(i = 0, j = last.length; i < j; i++) { | |
4950 if(tmp[last[i]] === 1) { | |
4951 data.changed.deselected.push(last[i]); | |
4952 } | |
4953 } | |
4954 last = data.selected.slice(); | |
4955 } | |
4956 /** | |
4957 * triggered when selection changes (the "changed" plugin enhances the original event with more data) | |
4958 * @event | |
4959 * @name changed.jstree | |
4960 * @param {Object} node | |
4961 * @param {Object} action the action that caused the selection to change | |
4962 * @param {Array} selected the current selection | |
4963 * @param {Object} changed an object containing two properties `selected` and `deselected` - both arrays of node IDs, which were selected or deselected since the last changed event | |
4964 * @param {Object} event the event (if any) that triggered this changed event | |
4965 * @plugin changed | |
4966 */ | |
4967 parent.trigger.call(this, ev, data); | |
4968 }; | |
4969 this.refresh = function (skip_loading, forget_state) { | |
4970 last = []; | |
4971 return parent.refresh.apply(this, arguments); | |
4972 }; | |
4973 }; | |
4974 | |
4975 /** | |
4976 * ### Checkbox plugin | |
4977 * | |
4978 * This plugin renders checkbox icons in front of each node, making multiple selection much easier. | |
4979 * It also supports tri-state behavior, meaning that if a node has a few of its children checked it will be rendered as undetermined, and state will be propagated up. | |
4980 */ | |
4981 | |
4982 var _i = document.createElement('I'); | |
4983 _i.className = 'jstree-icon jstree-checkbox'; | |
4984 _i.setAttribute('role', 'presentation'); | |
4985 /** | |
4986 * stores all defaults for the checkbox plugin | |
4987 * @name $.jstree.defaults.checkbox | |
4988 * @plugin checkbox | |
4989 */ | |
4990 $.jstree.defaults.checkbox = { | |
4991 /** | |
4992 * a boolean indicating if checkboxes should be visible (can be changed at a later time using `show_checkboxes()` and `hide_checkboxes`). Defaults to `true`. | |
4993 * @name $.jstree.defaults.checkbox.visible | |
4994 * @plugin checkbox | |
4995 */ | |
4996 visible : true, | |
4997 /** | |
4998 * a boolean indicating if checkboxes should cascade down and have an undetermined state. Defaults to `true`. | |
4999 * @name $.jstree.defaults.checkbox.three_state | |
5000 * @plugin checkbox | |
5001 */ | |
5002 three_state : true, | |
5003 /** | |
5004 * a boolean indicating if clicking anywhere on the node should act as clicking on the checkbox. Defaults to `true`. | |
5005 * @name $.jstree.defaults.checkbox.whole_node | |
5006 * @plugin checkbox | |
5007 */ | |
5008 whole_node : true, | |
5009 /** | |
5010 * a boolean indicating if the selected style of a node should be kept, or removed. Defaults to `true`. | |
5011 * @name $.jstree.defaults.checkbox.keep_selected_style | |
5012 * @plugin checkbox | |
5013 */ | |
5014 keep_selected_style : true, | |
5015 /** | |
5016 * This setting controls how cascading and undetermined nodes are applied. | |
5017 * If 'up' is in the string - cascading up is enabled, if 'down' is in the string - cascading down is enabled, if 'undetermined' is in the string - undetermined nodes will be used. | |
5018 * If `three_state` is set to `true` this setting is automatically set to 'up+down+undetermined'. Defaults to ''. | |
5019 * @name $.jstree.defaults.checkbox.cascade | |
5020 * @plugin checkbox | |
5021 */ | |
5022 cascade : '', | |
5023 /** | |
5024 * This setting controls if checkbox are bound to the general tree selection or to an internal array maintained by the checkbox plugin. Defaults to `true`, only set to `false` if you know exactly what you are doing. | |
5025 * @name $.jstree.defaults.checkbox.tie_selection | |
5026 * @plugin checkbox | |
5027 */ | |
5028 tie_selection : true, | |
5029 | |
5030 /** | |
5031 * This setting controls if cascading down affects disabled checkboxes | |
5032 * @name $.jstree.defaults.checkbox.cascade_to_disabled | |
5033 * @plugin checkbox | |
5034 */ | |
5035 cascade_to_disabled : true, | |
5036 | |
5037 /** | |
5038 * This setting controls if cascading down affects hidden checkboxes | |
5039 * @name $.jstree.defaults.checkbox.cascade_to_hidden | |
5040 * @plugin checkbox | |
5041 */ | |
5042 cascade_to_hidden : true | |
5043 }; | |
5044 $.jstree.plugins.checkbox = function (options, parent) { | |
5045 this.bind = function () { | |
5046 parent.bind.call(this); | |
5047 this._data.checkbox.uto = false; | |
5048 this._data.checkbox.selected = []; | |
5049 if(this.settings.checkbox.three_state) { | |
5050 this.settings.checkbox.cascade = 'up+down+undetermined'; | |
5051 } | |
5052 this.element | |
5053 .on("init.jstree", $.proxy(function () { | |
5054 this._data.checkbox.visible = this.settings.checkbox.visible; | |
5055 if(!this.settings.checkbox.keep_selected_style) { | |
5056 this.element.addClass('jstree-checkbox-no-clicked'); | |
5057 } | |
5058 if(this.settings.checkbox.tie_selection) { | |
5059 this.element.addClass('jstree-checkbox-selection'); | |
5060 } | |
5061 }, this)) | |
5062 .on("loading.jstree", $.proxy(function () { | |
5063 this[ this._data.checkbox.visible ? 'show_checkboxes' : 'hide_checkboxes' ](); | |
5064 }, this)); | |
5065 if(this.settings.checkbox.cascade.indexOf('undetermined') !== -1) { | |
5066 this.element | |
5067 .on('changed.jstree uncheck_node.jstree check_node.jstree uncheck_all.jstree check_all.jstree move_node.jstree copy_node.jstree redraw.jstree open_node.jstree', $.proxy(function () { | |
5068 // only if undetermined is in setting | |
5069 if(this._data.checkbox.uto) { clearTimeout(this._data.checkbox.uto); } | |
5070 this._data.checkbox.uto = setTimeout($.proxy(this._undetermined, this), 50); | |
5071 }, this)); | |
5072 } | |
5073 if(!this.settings.checkbox.tie_selection) { | |
5074 this.element | |
5075 .on('model.jstree', $.proxy(function (e, data) { | |
5076 var m = this._model.data, | |
5077 p = m[data.parent], | |
5078 dpc = data.nodes, | |
5079 i, j; | |
5080 for(i = 0, j = dpc.length; i < j; i++) { | |
5081 m[dpc[i]].state.checked = m[dpc[i]].state.checked || (m[dpc[i]].original && m[dpc[i]].original.state && m[dpc[i]].original.state.checked); | |
5082 if(m[dpc[i]].state.checked) { | |
5083 this._data.checkbox.selected.push(dpc[i]); | |
5084 } | |
5085 } | |
5086 }, this)); | |
5087 } | |
5088 if(this.settings.checkbox.cascade.indexOf('up') !== -1 || this.settings.checkbox.cascade.indexOf('down') !== -1) { | |
5089 this.element | |
5090 .on('model.jstree', $.proxy(function (e, data) { | |
5091 var m = this._model.data, | |
5092 p = m[data.parent], | |
5093 dpc = data.nodes, | |
5094 chd = [], | |
5095 c, i, j, k, l, tmp, s = this.settings.checkbox.cascade, t = this.settings.checkbox.tie_selection; | |
5096 | |
5097 if(s.indexOf('down') !== -1) { | |
5098 // apply down | |
5099 if(p.state[ t ? 'selected' : 'checked' ]) { | |
5100 for(i = 0, j = dpc.length; i < j; i++) { | |
5101 m[dpc[i]].state[ t ? 'selected' : 'checked' ] = true; | |
5102 } | |
5103 | |
5104 this._data[ t ? 'core' : 'checkbox' ].selected = this._data[ t ? 'core' : 'checkbox' ].selected.concat(dpc); | |
5105 } | |
5106 else { | |
5107 for(i = 0, j = dpc.length; i < j; i++) { | |
5108 if(m[dpc[i]].state[ t ? 'selected' : 'checked' ]) { | |
5109 for(k = 0, l = m[dpc[i]].children_d.length; k < l; k++) { | |
5110 m[m[dpc[i]].children_d[k]].state[ t ? 'selected' : 'checked' ] = true; | |
5111 } | |
5112 this._data[ t ? 'core' : 'checkbox' ].selected = this._data[ t ? 'core' : 'checkbox' ].selected.concat(m[dpc[i]].children_d); | |
5113 } | |
5114 } | |
5115 } | |
5116 } | |
5117 | |
5118 if(s.indexOf('up') !== -1) { | |
5119 // apply up | |
5120 for(i = 0, j = p.children_d.length; i < j; i++) { | |
5121 if(!m[p.children_d[i]].children.length) { | |
5122 chd.push(m[p.children_d[i]].parent); | |
5123 } | |
5124 } | |
5125 chd = $.vakata.array_unique(chd); | |
5126 for(k = 0, l = chd.length; k < l; k++) { | |
5127 p = m[chd[k]]; | |
5128 while(p && p.id !== $.jstree.root) { | |
5129 c = 0; | |
5130 for(i = 0, j = p.children.length; i < j; i++) { | |
5131 c += m[p.children[i]].state[ t ? 'selected' : 'checked' ]; | |
5132 } | |
5133 if(c === j) { | |
5134 p.state[ t ? 'selected' : 'checked' ] = true; | |
5135 this._data[ t ? 'core' : 'checkbox' ].selected.push(p.id); | |
5136 tmp = this.get_node(p, true); | |
5137 if(tmp && tmp.length) { | |
5138 tmp.attr('aria-selected', true).children('.jstree-anchor').addClass( t ? 'jstree-clicked' : 'jstree-checked'); | |
5139 } | |
5140 } | |
5141 else { | |
5142 break; | |
5143 } | |
5144 p = this.get_node(p.parent); | |
5145 } | |
5146 } | |
5147 } | |
5148 | |
5149 this._data[ t ? 'core' : 'checkbox' ].selected = $.vakata.array_unique(this._data[ t ? 'core' : 'checkbox' ].selected); | |
5150 }, this)) | |
5151 .on(this.settings.checkbox.tie_selection ? 'select_node.jstree' : 'check_node.jstree', $.proxy(function (e, data) { | |
5152 var self = this, | |
5153 obj = data.node, | |
5154 m = this._model.data, | |
5155 par = this.get_node(obj.parent), | |
5156 i, j, c, tmp, s = this.settings.checkbox.cascade, t = this.settings.checkbox.tie_selection, | |
5157 sel = {}, cur = this._data[ t ? 'core' : 'checkbox' ].selected; | |
5158 | |
5159 for (i = 0, j = cur.length; i < j; i++) { | |
5160 sel[cur[i]] = true; | |
5161 } | |
5162 | |
5163 // apply down | |
5164 if(s.indexOf('down') !== -1) { | |
5165 //this._data[ t ? 'core' : 'checkbox' ].selected = $.vakata.array_unique(this._data[ t ? 'core' : 'checkbox' ].selected.concat(obj.children_d)); | |
5166 var selectedIds = this._cascade_new_checked_state(obj.id, true); | |
5167 var temp = obj.children_d.concat(obj.id); | |
5168 for (i = 0, j = temp.length; i < j; i++) { | |
5169 if (selectedIds.indexOf(temp[i]) > -1) { | |
5170 sel[temp[i]] = true; | |
5171 } | |
5172 else { | |
5173 delete sel[temp[i]]; | |
5174 } | |
5175 } | |
5176 } | |
5177 | |
5178 // apply up | |
5179 if(s.indexOf('up') !== -1) { | |
5180 while(par && par.id !== $.jstree.root) { | |
5181 c = 0; | |
5182 for(i = 0, j = par.children.length; i < j; i++) { | |
5183 c += m[par.children[i]].state[ t ? 'selected' : 'checked' ]; | |
5184 } | |
5185 if(c === j) { | |
5186 par.state[ t ? 'selected' : 'checked' ] = true; | |
5187 sel[par.id] = true; | |
5188 //this._data[ t ? 'core' : 'checkbox' ].selected.push(par.id); | |
5189 tmp = this.get_node(par, true); | |
5190 if(tmp && tmp.length) { | |
5191 tmp.attr('aria-selected', true).children('.jstree-anchor').addClass(t ? 'jstree-clicked' : 'jstree-checked'); | |
5192 } | |
5193 } | |
5194 else { | |
5195 break; | |
5196 } | |
5197 par = this.get_node(par.parent); | |
5198 } | |
5199 } | |
5200 | |
5201 cur = []; | |
5202 for (i in sel) { | |
5203 if (sel.hasOwnProperty(i)) { | |
5204 cur.push(i); | |
5205 } | |
5206 } | |
5207 this._data[ t ? 'core' : 'checkbox' ].selected = cur; | |
5208 }, this)) | |
5209 .on(this.settings.checkbox.tie_selection ? 'deselect_all.jstree' : 'uncheck_all.jstree', $.proxy(function (e, data) { | |
5210 var obj = this.get_node($.jstree.root), | |
5211 m = this._model.data, | |
5212 i, j, tmp; | |
5213 for(i = 0, j = obj.children_d.length; i < j; i++) { | |
5214 tmp = m[obj.children_d[i]]; | |
5215 if(tmp && tmp.original && tmp.original.state && tmp.original.state.undetermined) { | |
5216 tmp.original.state.undetermined = false; | |
5217 } | |
5218 } | |
5219 }, this)) | |
5220 .on(this.settings.checkbox.tie_selection ? 'deselect_node.jstree' : 'uncheck_node.jstree', $.proxy(function (e, data) { | |
5221 var self = this, | |
5222 obj = data.node, | |
5223 dom = this.get_node(obj, true), | |
5224 i, j, tmp, s = this.settings.checkbox.cascade, t = this.settings.checkbox.tie_selection, | |
5225 cur = this._data[ t ? 'core' : 'checkbox' ].selected, sel = {}, | |
5226 stillSelectedIds = [], | |
5227 allIds = obj.children_d.concat(obj.id); | |
5228 | |
5229 // apply down | |
5230 if(s.indexOf('down') !== -1) { | |
5231 var selectedIds = this._cascade_new_checked_state(obj.id, false); | |
5232 | |
5233 cur = cur.filter(function(id) { | |
5234 return allIds.indexOf(id) === -1 || selectedIds.indexOf(id) > -1; | |
5235 }); | |
5236 } | |
5237 | |
5238 // only apply up if cascade up is enabled and if this node is not selected | |
5239 // (if all child nodes are disabled and cascade_to_disabled === false then this node will till be selected). | |
5240 if(s.indexOf('up') !== -1 && cur.indexOf(obj.id) === -1) { | |
5241 for(i = 0, j = obj.parents.length; i < j; i++) { | |
5242 tmp = this._model.data[obj.parents[i]]; | |
5243 tmp.state[ t ? 'selected' : 'checked' ] = false; | |
5244 if(tmp && tmp.original && tmp.original.state && tmp.original.state.undetermined) { | |
5245 tmp.original.state.undetermined = false; | |
5246 } | |
5247 tmp = this.get_node(obj.parents[i], true); | |
5248 if(tmp && tmp.length) { | |
5249 tmp.attr('aria-selected', false).children('.jstree-anchor').removeClass(t ? 'jstree-clicked' : 'jstree-checked'); | |
5250 } | |
5251 } | |
5252 | |
5253 cur = cur.filter(function(id) { | |
5254 return obj.parents.indexOf(id) === -1; | |
5255 }); | |
5256 } | |
5257 | |
5258 this._data[ t ? 'core' : 'checkbox' ].selected = cur; | |
5259 }, this)); | |
5260 } | |
5261 if(this.settings.checkbox.cascade.indexOf('up') !== -1) { | |
5262 this.element | |
5263 .on('delete_node.jstree', $.proxy(function (e, data) { | |
5264 // apply up (whole handler) | |
5265 var p = this.get_node(data.parent), | |
5266 m = this._model.data, | |
5267 i, j, c, tmp, t = this.settings.checkbox.tie_selection; | |
5268 while(p && p.id !== $.jstree.root && !p.state[ t ? 'selected' : 'checked' ]) { | |
5269 c = 0; | |
5270 for(i = 0, j = p.children.length; i < j; i++) { | |
5271 c += m[p.children[i]].state[ t ? 'selected' : 'checked' ]; | |
5272 } | |
5273 if(j > 0 && c === j) { | |
5274 p.state[ t ? 'selected' : 'checked' ] = true; | |
5275 this._data[ t ? 'core' : 'checkbox' ].selected.push(p.id); | |
5276 tmp = this.get_node(p, true); | |
5277 if(tmp && tmp.length) { | |
5278 tmp.attr('aria-selected', true).children('.jstree-anchor').addClass(t ? 'jstree-clicked' : 'jstree-checked'); | |
5279 } | |
5280 } | |
5281 else { | |
5282 break; | |
5283 } | |
5284 p = this.get_node(p.parent); | |
5285 } | |
5286 }, this)) | |
5287 .on('move_node.jstree', $.proxy(function (e, data) { | |
5288 // apply up (whole handler) | |
5289 var is_multi = data.is_multi, | |
5290 old_par = data.old_parent, | |
5291 new_par = this.get_node(data.parent), | |
5292 m = this._model.data, | |
5293 p, c, i, j, tmp, t = this.settings.checkbox.tie_selection; | |
5294 if(!is_multi) { | |
5295 p = this.get_node(old_par); | |
5296 while(p && p.id !== $.jstree.root && !p.state[ t ? 'selected' : 'checked' ]) { | |
5297 c = 0; | |
5298 for(i = 0, j = p.children.length; i < j; i++) { | |
5299 c += m[p.children[i]].state[ t ? 'selected' : 'checked' ]; | |
5300 } | |
5301 if(j > 0 && c === j) { | |
5302 p.state[ t ? 'selected' : 'checked' ] = true; | |
5303 this._data[ t ? 'core' : 'checkbox' ].selected.push(p.id); | |
5304 tmp = this.get_node(p, true); | |
5305 if(tmp && tmp.length) { | |
5306 tmp.attr('aria-selected', true).children('.jstree-anchor').addClass(t ? 'jstree-clicked' : 'jstree-checked'); | |
5307 } | |
5308 } | |
5309 else { | |
5310 break; | |
5311 } | |
5312 p = this.get_node(p.parent); | |
5313 } | |
5314 } | |
5315 p = new_par; | |
5316 while(p && p.id !== $.jstree.root) { | |
5317 c = 0; | |
5318 for(i = 0, j = p.children.length; i < j; i++) { | |
5319 c += m[p.children[i]].state[ t ? 'selected' : 'checked' ]; | |
5320 } | |
5321 if(c === j) { | |
5322 if(!p.state[ t ? 'selected' : 'checked' ]) { | |
5323 p.state[ t ? 'selected' : 'checked' ] = true; | |
5324 this._data[ t ? 'core' : 'checkbox' ].selected.push(p.id); | |
5325 tmp = this.get_node(p, true); | |
5326 if(tmp && tmp.length) { | |
5327 tmp.attr('aria-selected', true).children('.jstree-anchor').addClass(t ? 'jstree-clicked' : 'jstree-checked'); | |
5328 } | |
5329 } | |
5330 } | |
5331 else { | |
5332 if(p.state[ t ? 'selected' : 'checked' ]) { | |
5333 p.state[ t ? 'selected' : 'checked' ] = false; | |
5334 this._data[ t ? 'core' : 'checkbox' ].selected = $.vakata.array_remove_item(this._data[ t ? 'core' : 'checkbox' ].selected, p.id); | |
5335 tmp = this.get_node(p, true); | |
5336 if(tmp && tmp.length) { | |
5337 tmp.attr('aria-selected', false).children('.jstree-anchor').removeClass(t ? 'jstree-clicked' : 'jstree-checked'); | |
5338 } | |
5339 } | |
5340 else { | |
5341 break; | |
5342 } | |
5343 } | |
5344 p = this.get_node(p.parent); | |
5345 } | |
5346 }, this)); | |
5347 } | |
5348 }; | |
5349 /** | |
5350 * get an array of all nodes whose state is "undetermined" | |
5351 * @name get_undetermined([full]) | |
5352 * @param {boolean} full: if set to `true` the returned array will consist of the full node objects, otherwise - only IDs will be returned | |
5353 * @return {Array} | |
5354 * @plugin checkbox | |
5355 */ | |
5356 this.get_undetermined = function (full) { | |
5357 if (this.settings.checkbox.cascade.indexOf('undetermined') === -1) { | |
5358 return []; | |
5359 } | |
5360 var i, j, k, l, o = {}, m = this._model.data, t = this.settings.checkbox.tie_selection, s = this._data[ t ? 'core' : 'checkbox' ].selected, p = [], tt = this, r = []; | |
5361 for(i = 0, j = s.length; i < j; i++) { | |
5362 if(m[s[i]] && m[s[i]].parents) { | |
5363 for(k = 0, l = m[s[i]].parents.length; k < l; k++) { | |
5364 if(o[m[s[i]].parents[k]] !== undefined) { | |
5365 break; | |
5366 } | |
5367 if(m[s[i]].parents[k] !== $.jstree.root) { | |
5368 o[m[s[i]].parents[k]] = true; | |
5369 p.push(m[s[i]].parents[k]); | |
5370 } | |
5371 } | |
5372 } | |
5373 } | |
5374 // attempt for server side undetermined state | |
5375 this.element.find('.jstree-closed').not(':has(.jstree-children)') | |
5376 .each(function () { | |
5377 var tmp = tt.get_node(this), tmp2; | |
5378 | |
5379 if(!tmp) { return; } | |
5380 | |
5381 if(!tmp.state.loaded) { | |
5382 if(tmp.original && tmp.original.state && tmp.original.state.undetermined && tmp.original.state.undetermined === true) { | |
5383 if(o[tmp.id] === undefined && tmp.id !== $.jstree.root) { | |
5384 o[tmp.id] = true; | |
5385 p.push(tmp.id); | |
5386 } | |
5387 for(k = 0, l = tmp.parents.length; k < l; k++) { | |
5388 if(o[tmp.parents[k]] === undefined && tmp.parents[k] !== $.jstree.root) { | |
5389 o[tmp.parents[k]] = true; | |
5390 p.push(tmp.parents[k]); | |
5391 } | |
5392 } | |
5393 } | |
5394 } | |
5395 else { | |
5396 for(i = 0, j = tmp.children_d.length; i < j; i++) { | |
5397 tmp2 = m[tmp.children_d[i]]; | |
5398 if(!tmp2.state.loaded && tmp2.original && tmp2.original.state && tmp2.original.state.undetermined && tmp2.original.state.undetermined === true) { | |
5399 if(o[tmp2.id] === undefined && tmp2.id !== $.jstree.root) { | |
5400 o[tmp2.id] = true; | |
5401 p.push(tmp2.id); | |
5402 } | |
5403 for(k = 0, l = tmp2.parents.length; k < l; k++) { | |
5404 if(o[tmp2.parents[k]] === undefined && tmp2.parents[k] !== $.jstree.root) { | |
5405 o[tmp2.parents[k]] = true; | |
5406 p.push(tmp2.parents[k]); | |
5407 } | |
5408 } | |
5409 } | |
5410 } | |
5411 } | |
5412 }); | |
5413 for (i = 0, j = p.length; i < j; i++) { | |
5414 if(!m[p[i]].state[ t ? 'selected' : 'checked' ]) { | |
5415 r.push(full ? m[p[i]] : p[i]); | |
5416 } | |
5417 } | |
5418 return r; | |
5419 }; | |
5420 /** | |
5421 * set the undetermined state where and if necessary. Used internally. | |
5422 * @private | |
5423 * @name _undetermined() | |
5424 * @plugin checkbox | |
5425 */ | |
5426 this._undetermined = function () { | |
5427 if(this.element === null) { return; } | |
5428 var p = this.get_undetermined(false), i, j, s; | |
5429 | |
5430 this.element.find('.jstree-undetermined').removeClass('jstree-undetermined'); | |
5431 for (i = 0, j = p.length; i < j; i++) { | |
5432 s = this.get_node(p[i], true); | |
5433 if(s && s.length) { | |
5434 s.children('.jstree-anchor').children('.jstree-checkbox').addClass('jstree-undetermined'); | |
5435 } | |
5436 } | |
5437 }; | |
5438 this.redraw_node = function(obj, deep, is_callback, force_render) { | |
5439 obj = parent.redraw_node.apply(this, arguments); | |
5440 if(obj) { | |
5441 var i, j, tmp = null, icon = null; | |
5442 for(i = 0, j = obj.childNodes.length; i < j; i++) { | |
5443 if(obj.childNodes[i] && obj.childNodes[i].className && obj.childNodes[i].className.indexOf("jstree-anchor") !== -1) { | |
5444 tmp = obj.childNodes[i]; | |
5445 break; | |
5446 } | |
5447 } | |
5448 if(tmp) { | |
5449 if(!this.settings.checkbox.tie_selection && this._model.data[obj.id].state.checked) { tmp.className += ' jstree-checked'; } | |
5450 icon = _i.cloneNode(false); | |
5451 if(this._model.data[obj.id].state.checkbox_disabled) { icon.className += ' jstree-checkbox-disabled'; } | |
5452 tmp.insertBefore(icon, tmp.childNodes[0]); | |
5453 } | |
5454 } | |
5455 if(!is_callback && this.settings.checkbox.cascade.indexOf('undetermined') !== -1) { | |
5456 if(this._data.checkbox.uto) { clearTimeout(this._data.checkbox.uto); } | |
5457 this._data.checkbox.uto = setTimeout($.proxy(this._undetermined, this), 50); | |
5458 } | |
5459 return obj; | |
5460 }; | |
5461 /** | |
5462 * show the node checkbox icons | |
5463 * @name show_checkboxes() | |
5464 * @plugin checkbox | |
5465 */ | |
5466 this.show_checkboxes = function () { this._data.core.themes.checkboxes = true; this.get_container_ul().removeClass("jstree-no-checkboxes"); }; | |
5467 /** | |
5468 * hide the node checkbox icons | |
5469 * @name hide_checkboxes() | |
5470 * @plugin checkbox | |
5471 */ | |
5472 this.hide_checkboxes = function () { this._data.core.themes.checkboxes = false; this.get_container_ul().addClass("jstree-no-checkboxes"); }; | |
5473 /** | |
5474 * toggle the node icons | |
5475 * @name toggle_checkboxes() | |
5476 * @plugin checkbox | |
5477 */ | |
5478 this.toggle_checkboxes = function () { if(this._data.core.themes.checkboxes) { this.hide_checkboxes(); } else { this.show_checkboxes(); } }; | |
5479 /** | |
5480 * checks if a node is in an undetermined state | |
5481 * @name is_undetermined(obj) | |
5482 * @param {mixed} obj | |
5483 * @return {Boolean} | |
5484 */ | |
5485 this.is_undetermined = function (obj) { | |
5486 obj = this.get_node(obj); | |
5487 var s = this.settings.checkbox.cascade, i, j, t = this.settings.checkbox.tie_selection, d = this._data[ t ? 'core' : 'checkbox' ].selected, m = this._model.data; | |
5488 if(!obj || obj.state[ t ? 'selected' : 'checked' ] === true || s.indexOf('undetermined') === -1 || (s.indexOf('down') === -1 && s.indexOf('up') === -1)) { | |
5489 return false; | |
5490 } | |
5491 if(!obj.state.loaded && obj.original.state.undetermined === true) { | |
5492 return true; | |
5493 } | |
5494 for(i = 0, j = obj.children_d.length; i < j; i++) { | |
5495 if($.inArray(obj.children_d[i], d) !== -1 || (!m[obj.children_d[i]].state.loaded && m[obj.children_d[i]].original.state.undetermined)) { | |
5496 return true; | |
5497 } | |
5498 } | |
5499 return false; | |
5500 }; | |
5501 /** | |
5502 * disable a node's checkbox | |
5503 * @name disable_checkbox(obj) | |
5504 * @param {mixed} obj an array can be used too | |
5505 * @trigger disable_checkbox.jstree | |
5506 * @plugin checkbox | |
5507 */ | |
5508 this.disable_checkbox = function (obj) { | |
5509 var t1, t2, dom; | |
5510 if($.isArray(obj)) { | |
5511 obj = obj.slice(); | |
5512 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
5513 this.disable_checkbox(obj[t1]); | |
5514 } | |
5515 return true; | |
5516 } | |
5517 obj = this.get_node(obj); | |
5518 if(!obj || obj.id === $.jstree.root) { | |
5519 return false; | |
5520 } | |
5521 dom = this.get_node(obj, true); | |
5522 if(!obj.state.checkbox_disabled) { | |
5523 obj.state.checkbox_disabled = true; | |
5524 if(dom && dom.length) { | |
5525 dom.children('.jstree-anchor').children('.jstree-checkbox').addClass('jstree-checkbox-disabled'); | |
5526 } | |
5527 /** | |
5528 * triggered when an node's checkbox is disabled | |
5529 * @event | |
5530 * @name disable_checkbox.jstree | |
5531 * @param {Object} node | |
5532 * @plugin checkbox | |
5533 */ | |
5534 this.trigger('disable_checkbox', { 'node' : obj }); | |
5535 } | |
5536 }; | |
5537 /** | |
5538 * enable a node's checkbox | |
5539 * @name disable_checkbox(obj) | |
5540 * @param {mixed} obj an array can be used too | |
5541 * @trigger enable_checkbox.jstree | |
5542 * @plugin checkbox | |
5543 */ | |
5544 this.enable_checkbox = function (obj) { | |
5545 var t1, t2, dom; | |
5546 if($.isArray(obj)) { | |
5547 obj = obj.slice(); | |
5548 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
5549 this.enable_checkbox(obj[t1]); | |
5550 } | |
5551 return true; | |
5552 } | |
5553 obj = this.get_node(obj); | |
5554 if(!obj || obj.id === $.jstree.root) { | |
5555 return false; | |
5556 } | |
5557 dom = this.get_node(obj, true); | |
5558 if(obj.state.checkbox_disabled) { | |
5559 obj.state.checkbox_disabled = false; | |
5560 if(dom && dom.length) { | |
5561 dom.children('.jstree-anchor').children('.jstree-checkbox').removeClass('jstree-checkbox-disabled'); | |
5562 } | |
5563 /** | |
5564 * triggered when an node's checkbox is enabled | |
5565 * @event | |
5566 * @name enable_checkbox.jstree | |
5567 * @param {Object} node | |
5568 * @plugin checkbox | |
5569 */ | |
5570 this.trigger('enable_checkbox', { 'node' : obj }); | |
5571 } | |
5572 }; | |
5573 | |
5574 this.activate_node = function (obj, e) { | |
5575 if($(e.target).hasClass('jstree-checkbox-disabled')) { | |
5576 return false; | |
5577 } | |
5578 if(this.settings.checkbox.tie_selection && (this.settings.checkbox.whole_node || $(e.target).hasClass('jstree-checkbox'))) { | |
5579 e.ctrlKey = true; | |
5580 } | |
5581 if(this.settings.checkbox.tie_selection || (!this.settings.checkbox.whole_node && !$(e.target).hasClass('jstree-checkbox'))) { | |
5582 return parent.activate_node.call(this, obj, e); | |
5583 } | |
5584 if(this.is_disabled(obj)) { | |
5585 return false; | |
5586 } | |
5587 if(this.is_checked(obj)) { | |
5588 this.uncheck_node(obj, e); | |
5589 } | |
5590 else { | |
5591 this.check_node(obj, e); | |
5592 } | |
5593 this.trigger('activate_node', { 'node' : this.get_node(obj) }); | |
5594 }; | |
5595 | |
5596 /** | |
5597 * Cascades checked state to a node and all its descendants. This function does NOT affect hidden and disabled nodes (or their descendants). | |
5598 * However if these unaffected nodes are already selected their ids will be included in the returned array. | |
5599 * @private | |
5600 * @param {string} id the node ID | |
5601 * @param {bool} checkedState should the nodes be checked or not | |
5602 * @returns {Array} Array of all node id's (in this tree branch) that are checked. | |
5603 */ | |
5604 this._cascade_new_checked_state = function (id, checkedState) { | |
5605 var self = this; | |
5606 var t = this.settings.checkbox.tie_selection; | |
5607 var node = this._model.data[id]; | |
5608 var selectedNodeIds = []; | |
5609 var selectedChildrenIds = [], i, j, selectedChildIds; | |
5610 | |
5611 if ( | |
5612 (this.settings.checkbox.cascade_to_disabled || !node.state.disabled) && | |
5613 (this.settings.checkbox.cascade_to_hidden || !node.state.hidden) | |
5614 ) { | |
5615 //First try and check/uncheck the children | |
5616 if (node.children) { | |
5617 for (i = 0, j = node.children.length; i < j; i++) { | |
5618 var childId = node.children[i]; | |
5619 selectedChildIds = self._cascade_new_checked_state(childId, checkedState); | |
5620 selectedNodeIds = selectedNodeIds.concat(selectedChildIds); | |
5621 if (selectedChildIds.indexOf(childId) > -1) { | |
5622 selectedChildrenIds.push(childId); | |
5623 } | |
5624 } | |
5625 } | |
5626 | |
5627 var dom = self.get_node(node, true); | |
5628 | |
5629 //A node's state is undetermined if some but not all of it's children are checked/selected . | |
5630 var undetermined = selectedChildrenIds.length > 0 && selectedChildrenIds.length < node.children.length; | |
5631 | |
5632 if(node.original && node.original.state && node.original.state.undetermined) { | |
5633 node.original.state.undetermined = undetermined; | |
5634 } | |
5635 | |
5636 //If a node is undetermined then remove selected class | |
5637 if (undetermined) { | |
5638 node.state[ t ? 'selected' : 'checked' ] = false; | |
5639 dom.attr('aria-selected', false).children('.jstree-anchor').removeClass(t ? 'jstree-clicked' : 'jstree-checked'); | |
5640 } | |
5641 //Otherwise, if the checkedState === true (i.e. the node is being checked now) and all of the node's children are checked (if it has any children), | |
5642 //check the node and style it correctly. | |
5643 else if (checkedState && selectedChildrenIds.length === node.children.length) { | |
5644 node.state[ t ? 'selected' : 'checked' ] = checkedState; | |
5645 selectedNodeIds.push(node.id); | |
5646 | |
5647 dom.attr('aria-selected', true).children('.jstree-anchor').addClass(t ? 'jstree-clicked' : 'jstree-checked'); | |
5648 } | |
5649 else { | |
5650 node.state[ t ? 'selected' : 'checked' ] = false; | |
5651 dom.attr('aria-selected', false).children('.jstree-anchor').removeClass(t ? 'jstree-clicked' : 'jstree-checked'); | |
5652 } | |
5653 } | |
5654 else { | |
5655 selectedChildIds = this.get_checked_descendants(id); | |
5656 | |
5657 if (node.state[ t ? 'selected' : 'checked' ]) { | |
5658 selectedChildIds.push(node.id); | |
5659 } | |
5660 | |
5661 selectedNodeIds = selectedNodeIds.concat(selectedChildIds); | |
5662 } | |
5663 | |
5664 return selectedNodeIds; | |
5665 }; | |
5666 | |
5667 /** | |
5668 * Gets ids of nodes selected in branch (of tree) specified by id (does not include the node specified by id) | |
5669 * @name get_checked_descendants(obj) | |
5670 * @param {string} id the node ID | |
5671 * @return {Array} array of IDs | |
5672 * @plugin checkbox | |
5673 */ | |
5674 this.get_checked_descendants = function (id) { | |
5675 var self = this; | |
5676 var t = self.settings.checkbox.tie_selection; | |
5677 var node = self._model.data[id]; | |
5678 | |
5679 return node.children_d.filter(function(_id) { | |
5680 return self._model.data[_id].state[ t ? 'selected' : 'checked' ]; | |
5681 }); | |
5682 }; | |
5683 | |
5684 /** | |
5685 * check a node (only if tie_selection in checkbox settings is false, otherwise select_node will be called internally) | |
5686 * @name check_node(obj) | |
5687 * @param {mixed} obj an array can be used to check multiple nodes | |
5688 * @trigger check_node.jstree | |
5689 * @plugin checkbox | |
5690 */ | |
5691 this.check_node = function (obj, e) { | |
5692 if(this.settings.checkbox.tie_selection) { return this.select_node(obj, false, true, e); } | |
5693 var dom, t1, t2, th; | |
5694 if($.isArray(obj)) { | |
5695 obj = obj.slice(); | |
5696 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
5697 this.check_node(obj[t1], e); | |
5698 } | |
5699 return true; | |
5700 } | |
5701 obj = this.get_node(obj); | |
5702 if(!obj || obj.id === $.jstree.root) { | |
5703 return false; | |
5704 } | |
5705 dom = this.get_node(obj, true); | |
5706 if(!obj.state.checked) { | |
5707 obj.state.checked = true; | |
5708 this._data.checkbox.selected.push(obj.id); | |
5709 if(dom && dom.length) { | |
5710 dom.children('.jstree-anchor').addClass('jstree-checked'); | |
5711 } | |
5712 /** | |
5713 * triggered when an node is checked (only if tie_selection in checkbox settings is false) | |
5714 * @event | |
5715 * @name check_node.jstree | |
5716 * @param {Object} node | |
5717 * @param {Array} selected the current selection | |
5718 * @param {Object} event the event (if any) that triggered this check_node | |
5719 * @plugin checkbox | |
5720 */ | |
5721 this.trigger('check_node', { 'node' : obj, 'selected' : this._data.checkbox.selected, 'event' : e }); | |
5722 } | |
5723 }; | |
5724 /** | |
5725 * uncheck a node (only if tie_selection in checkbox settings is false, otherwise deselect_node will be called internally) | |
5726 * @name uncheck_node(obj) | |
5727 * @param {mixed} obj an array can be used to uncheck multiple nodes | |
5728 * @trigger uncheck_node.jstree | |
5729 * @plugin checkbox | |
5730 */ | |
5731 this.uncheck_node = function (obj, e) { | |
5732 if(this.settings.checkbox.tie_selection) { return this.deselect_node(obj, false, e); } | |
5733 var t1, t2, dom; | |
5734 if($.isArray(obj)) { | |
5735 obj = obj.slice(); | |
5736 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
5737 this.uncheck_node(obj[t1], e); | |
5738 } | |
5739 return true; | |
5740 } | |
5741 obj = this.get_node(obj); | |
5742 if(!obj || obj.id === $.jstree.root) { | |
5743 return false; | |
5744 } | |
5745 dom = this.get_node(obj, true); | |
5746 if(obj.state.checked) { | |
5747 obj.state.checked = false; | |
5748 this._data.checkbox.selected = $.vakata.array_remove_item(this._data.checkbox.selected, obj.id); | |
5749 if(dom.length) { | |
5750 dom.children('.jstree-anchor').removeClass('jstree-checked'); | |
5751 } | |
5752 /** | |
5753 * triggered when an node is unchecked (only if tie_selection in checkbox settings is false) | |
5754 * @event | |
5755 * @name uncheck_node.jstree | |
5756 * @param {Object} node | |
5757 * @param {Array} selected the current selection | |
5758 * @param {Object} event the event (if any) that triggered this uncheck_node | |
5759 * @plugin checkbox | |
5760 */ | |
5761 this.trigger('uncheck_node', { 'node' : obj, 'selected' : this._data.checkbox.selected, 'event' : e }); | |
5762 } | |
5763 }; | |
5764 | |
5765 /** | |
5766 * checks all nodes in the tree (only if tie_selection in checkbox settings is false, otherwise select_all will be called internally) | |
5767 * @name check_all() | |
5768 * @trigger check_all.jstree, changed.jstree | |
5769 * @plugin checkbox | |
5770 */ | |
5771 this.check_all = function () { | |
5772 if(this.settings.checkbox.tie_selection) { return this.select_all(); } | |
5773 var tmp = this._data.checkbox.selected.concat([]), i, j; | |
5774 this._data.checkbox.selected = this._model.data[$.jstree.root].children_d.concat(); | |
5775 for(i = 0, j = this._data.checkbox.selected.length; i < j; i++) { | |
5776 if(this._model.data[this._data.checkbox.selected[i]]) { | |
5777 this._model.data[this._data.checkbox.selected[i]].state.checked = true; | |
5778 } | |
5779 } | |
5780 this.redraw(true); | |
5781 /** | |
5782 * triggered when all nodes are checked (only if tie_selection in checkbox settings is false) | |
5783 * @event | |
5784 * @name check_all.jstree | |
5785 * @param {Array} selected the current selection | |
5786 * @plugin checkbox | |
5787 */ | |
5788 this.trigger('check_all', { 'selected' : this._data.checkbox.selected }); | |
5789 }; | |
5790 /** | |
5791 * uncheck all checked nodes (only if tie_selection in checkbox settings is false, otherwise deselect_all will be called internally) | |
5792 * @name uncheck_all() | |
5793 * @trigger uncheck_all.jstree | |
5794 * @plugin checkbox | |
5795 */ | |
5796 this.uncheck_all = function () { | |
5797 if(this.settings.checkbox.tie_selection) { return this.deselect_all(); } | |
5798 var tmp = this._data.checkbox.selected.concat([]), i, j; | |
5799 for(i = 0, j = this._data.checkbox.selected.length; i < j; i++) { | |
5800 if(this._model.data[this._data.checkbox.selected[i]]) { | |
5801 this._model.data[this._data.checkbox.selected[i]].state.checked = false; | |
5802 } | |
5803 } | |
5804 this._data.checkbox.selected = []; | |
5805 this.element.find('.jstree-checked').removeClass('jstree-checked'); | |
5806 /** | |
5807 * triggered when all nodes are unchecked (only if tie_selection in checkbox settings is false) | |
5808 * @event | |
5809 * @name uncheck_all.jstree | |
5810 * @param {Object} node the previous selection | |
5811 * @param {Array} selected the current selection | |
5812 * @plugin checkbox | |
5813 */ | |
5814 this.trigger('uncheck_all', { 'selected' : this._data.checkbox.selected, 'node' : tmp }); | |
5815 }; | |
5816 /** | |
5817 * checks if a node is checked (if tie_selection is on in the settings this function will return the same as is_selected) | |
5818 * @name is_checked(obj) | |
5819 * @param {mixed} obj | |
5820 * @return {Boolean} | |
5821 * @plugin checkbox | |
5822 */ | |
5823 this.is_checked = function (obj) { | |
5824 if(this.settings.checkbox.tie_selection) { return this.is_selected(obj); } | |
5825 obj = this.get_node(obj); | |
5826 if(!obj || obj.id === $.jstree.root) { return false; } | |
5827 return obj.state.checked; | |
5828 }; | |
5829 /** | |
5830 * get an array of all checked nodes (if tie_selection is on in the settings this function will return the same as get_selected) | |
5831 * @name get_checked([full]) | |
5832 * @param {mixed} full if set to `true` the returned array will consist of the full node objects, otherwise - only IDs will be returned | |
5833 * @return {Array} | |
5834 * @plugin checkbox | |
5835 */ | |
5836 this.get_checked = function (full) { | |
5837 if(this.settings.checkbox.tie_selection) { return this.get_selected(full); } | |
5838 return full ? $.map(this._data.checkbox.selected, $.proxy(function (i) { return this.get_node(i); }, this)) : this._data.checkbox.selected; | |
5839 }; | |
5840 /** | |
5841 * get an array of all top level checked nodes (ignoring children of checked nodes) (if tie_selection is on in the settings this function will return the same as get_top_selected) | |
5842 * @name get_top_checked([full]) | |
5843 * @param {mixed} full if set to `true` the returned array will consist of the full node objects, otherwise - only IDs will be returned | |
5844 * @return {Array} | |
5845 * @plugin checkbox | |
5846 */ | |
5847 this.get_top_checked = function (full) { | |
5848 if(this.settings.checkbox.tie_selection) { return this.get_top_selected(full); } | |
5849 var tmp = this.get_checked(true), | |
5850 obj = {}, i, j, k, l; | |
5851 for(i = 0, j = tmp.length; i < j; i++) { | |
5852 obj[tmp[i].id] = tmp[i]; | |
5853 } | |
5854 for(i = 0, j = tmp.length; i < j; i++) { | |
5855 for(k = 0, l = tmp[i].children_d.length; k < l; k++) { | |
5856 if(obj[tmp[i].children_d[k]]) { | |
5857 delete obj[tmp[i].children_d[k]]; | |
5858 } | |
5859 } | |
5860 } | |
5861 tmp = []; | |
5862 for(i in obj) { | |
5863 if(obj.hasOwnProperty(i)) { | |
5864 tmp.push(i); | |
5865 } | |
5866 } | |
5867 return full ? $.map(tmp, $.proxy(function (i) { return this.get_node(i); }, this)) : tmp; | |
5868 }; | |
5869 /** | |
5870 * get an array of all bottom level checked nodes (ignoring selected parents) (if tie_selection is on in the settings this function will return the same as get_bottom_selected) | |
5871 * @name get_bottom_checked([full]) | |
5872 * @param {mixed} full if set to `true` the returned array will consist of the full node objects, otherwise - only IDs will be returned | |
5873 * @return {Array} | |
5874 * @plugin checkbox | |
5875 */ | |
5876 this.get_bottom_checked = function (full) { | |
5877 if(this.settings.checkbox.tie_selection) { return this.get_bottom_selected(full); } | |
5878 var tmp = this.get_checked(true), | |
5879 obj = [], i, j; | |
5880 for(i = 0, j = tmp.length; i < j; i++) { | |
5881 if(!tmp[i].children.length) { | |
5882 obj.push(tmp[i].id); | |
5883 } | |
5884 } | |
5885 return full ? $.map(obj, $.proxy(function (i) { return this.get_node(i); }, this)) : obj; | |
5886 }; | |
5887 this.load_node = function (obj, callback) { | |
5888 var k, l, i, j, c, tmp; | |
5889 if(!$.isArray(obj) && !this.settings.checkbox.tie_selection) { | |
5890 tmp = this.get_node(obj); | |
5891 if(tmp && tmp.state.loaded) { | |
5892 for(k = 0, l = tmp.children_d.length; k < l; k++) { | |
5893 if(this._model.data[tmp.children_d[k]].state.checked) { | |
5894 c = true; | |
5895 this._data.checkbox.selected = $.vakata.array_remove_item(this._data.checkbox.selected, tmp.children_d[k]); | |
5896 } | |
5897 } | |
5898 } | |
5899 } | |
5900 return parent.load_node.apply(this, arguments); | |
5901 }; | |
5902 this.get_state = function () { | |
5903 var state = parent.get_state.apply(this, arguments); | |
5904 if(this.settings.checkbox.tie_selection) { return state; } | |
5905 state.checkbox = this._data.checkbox.selected.slice(); | |
5906 return state; | |
5907 }; | |
5908 this.set_state = function (state, callback) { | |
5909 var res = parent.set_state.apply(this, arguments); | |
5910 if(res && state.checkbox) { | |
5911 if(!this.settings.checkbox.tie_selection) { | |
5912 this.uncheck_all(); | |
5913 var _this = this; | |
5914 $.each(state.checkbox, function (i, v) { | |
5915 _this.check_node(v); | |
5916 }); | |
5917 } | |
5918 delete state.checkbox; | |
5919 this.set_state(state, callback); | |
5920 return false; | |
5921 } | |
5922 return res; | |
5923 }; | |
5924 this.refresh = function (skip_loading, forget_state) { | |
5925 if(!this.settings.checkbox.tie_selection) { | |
5926 this._data.checkbox.selected = []; | |
5927 } | |
5928 return parent.refresh.apply(this, arguments); | |
5929 }; | |
5930 }; | |
5931 | |
5932 // include the checkbox plugin by default | |
5933 // $.jstree.defaults.plugins.push("checkbox"); | |
5934 | |
5935 | |
5936 /** | |
5937 * ### Conditionalselect plugin | |
5938 * | |
5939 * This plugin allows defining a callback to allow or deny node selection by user input (activate node method). | |
5940 */ | |
5941 | |
5942 /** | |
5943 * a callback (function) which is invoked in the instance's scope and receives two arguments - the node and the event that triggered the `activate_node` call. Returning false prevents working with the node, returning true allows invoking activate_node. Defaults to returning `true`. | |
5944 * @name $.jstree.defaults.checkbox.visible | |
5945 * @plugin checkbox | |
5946 */ | |
5947 $.jstree.defaults.conditionalselect = function () { return true; }; | |
5948 $.jstree.plugins.conditionalselect = function (options, parent) { | |
5949 // own function | |
5950 this.activate_node = function (obj, e) { | |
5951 if(this.settings.conditionalselect.call(this, this.get_node(obj), e)) { | |
5952 return parent.activate_node.call(this, obj, e); | |
5953 } | |
5954 }; | |
5955 }; | |
5956 | |
5957 | |
5958 /** | |
5959 * ### Contextmenu plugin | |
5960 * | |
5961 * Shows a context menu when a node is right-clicked. | |
5962 */ | |
5963 | |
5964 /** | |
5965 * stores all defaults for the contextmenu plugin | |
5966 * @name $.jstree.defaults.contextmenu | |
5967 * @plugin contextmenu | |
5968 */ | |
5969 $.jstree.defaults.contextmenu = { | |
5970 /** | |
5971 * a boolean indicating if the node should be selected when the context menu is invoked on it. Defaults to `true`. | |
5972 * @name $.jstree.defaults.contextmenu.select_node | |
5973 * @plugin contextmenu | |
5974 */ | |
5975 select_node : true, | |
5976 /** | |
5977 * a boolean indicating if the menu should be shown aligned with the node. Defaults to `true`, otherwise the mouse coordinates are used. | |
5978 * @name $.jstree.defaults.contextmenu.show_at_node | |
5979 * @plugin contextmenu | |
5980 */ | |
5981 show_at_node : true, | |
5982 /** | |
5983 * an object of actions, or a function that accepts a node and a callback function and calls the callback function with an object of actions available for that node (you can also return the items too). | |
5984 * | |
5985 * Each action consists of a key (a unique name) and a value which is an object with the following properties (only label and action are required). Once a menu item is activated the `action` function will be invoked with an object containing the following keys: item - the contextmenu item definition as seen below, reference - the DOM node that was used (the tree node), element - the contextmenu DOM element, position - an object with x/y properties indicating the position of the menu. | |
5986 * | |
5987 * * `separator_before` - a boolean indicating if there should be a separator before this item | |
5988 * * `separator_after` - a boolean indicating if there should be a separator after this item | |
5989 * * `_disabled` - a boolean indicating if this action should be disabled | |
5990 * * `label` - a string - the name of the action (could be a function returning a string) | |
5991 * * `title` - a string - an optional tooltip for the item | |
5992 * * `action` - a function to be executed if this item is chosen, the function will receive | |
5993 * * `icon` - a string, can be a path to an icon or a className, if using an image that is in the current directory use a `./` prefix, otherwise it will be detected as a class | |
5994 * * `shortcut` - keyCode which will trigger the action if the menu is open (for example `113` for rename, which equals F2) | |
5995 * * `shortcut_label` - shortcut label (like for example `F2` for rename) | |
5996 * * `submenu` - an object with the same structure as $.jstree.defaults.contextmenu.items which can be used to create a submenu - each key will be rendered as a separate option in a submenu that will appear once the current item is hovered | |
5997 * | |
5998 * @name $.jstree.defaults.contextmenu.items | |
5999 * @plugin contextmenu | |
6000 */ | |
6001 items : function (o, cb) { // Could be an object directly | |
6002 return { | |
6003 "create" : { | |
6004 "separator_before" : false, | |
6005 "separator_after" : true, | |
6006 "_disabled" : false, //(this.check("create_node", data.reference, {}, "last")), | |
6007 "label" : "Create", | |
6008 "action" : function (data) { | |
6009 var inst = $.jstree.reference(data.reference), | |
6010 obj = inst.get_node(data.reference); | |
6011 inst.create_node(obj, {}, "last", function (new_node) { | |
6012 try { | |
6013 inst.edit(new_node); | |
6014 } catch (ex) { | |
6015 setTimeout(function () { inst.edit(new_node); },0); | |
6016 } | |
6017 }); | |
6018 } | |
6019 }, | |
6020 "rename" : { | |
6021 "separator_before" : false, | |
6022 "separator_after" : false, | |
6023 "_disabled" : false, //(this.check("rename_node", data.reference, this.get_parent(data.reference), "")), | |
6024 "label" : "Rename", | |
6025 /*! | |
6026 "shortcut" : 113, | |
6027 "shortcut_label" : 'F2', | |
6028 "icon" : "glyphicon glyphicon-leaf", | |
6029 */ | |
6030 "action" : function (data) { | |
6031 var inst = $.jstree.reference(data.reference), | |
6032 obj = inst.get_node(data.reference); | |
6033 inst.edit(obj); | |
6034 } | |
6035 }, | |
6036 "remove" : { | |
6037 "separator_before" : false, | |
6038 "icon" : false, | |
6039 "separator_after" : false, | |
6040 "_disabled" : false, //(this.check("delete_node", data.reference, this.get_parent(data.reference), "")), | |
6041 "label" : "Delete", | |
6042 "action" : function (data) { | |
6043 var inst = $.jstree.reference(data.reference), | |
6044 obj = inst.get_node(data.reference); | |
6045 if(inst.is_selected(obj)) { | |
6046 inst.delete_node(inst.get_selected()); | |
6047 } | |
6048 else { | |
6049 inst.delete_node(obj); | |
6050 } | |
6051 } | |
6052 }, | |
6053 "ccp" : { | |
6054 "separator_before" : true, | |
6055 "icon" : false, | |
6056 "separator_after" : false, | |
6057 "label" : "Edit", | |
6058 "action" : false, | |
6059 "submenu" : { | |
6060 "cut" : { | |
6061 "separator_before" : false, | |
6062 "separator_after" : false, | |
6063 "label" : "Cut", | |
6064 "action" : function (data) { | |
6065 var inst = $.jstree.reference(data.reference), | |
6066 obj = inst.get_node(data.reference); | |
6067 if(inst.is_selected(obj)) { | |
6068 inst.cut(inst.get_top_selected()); | |
6069 } | |
6070 else { | |
6071 inst.cut(obj); | |
6072 } | |
6073 } | |
6074 }, | |
6075 "copy" : { | |
6076 "separator_before" : false, | |
6077 "icon" : false, | |
6078 "separator_after" : false, | |
6079 "label" : "Copy", | |
6080 "action" : function (data) { | |
6081 var inst = $.jstree.reference(data.reference), | |
6082 obj = inst.get_node(data.reference); | |
6083 if(inst.is_selected(obj)) { | |
6084 inst.copy(inst.get_top_selected()); | |
6085 } | |
6086 else { | |
6087 inst.copy(obj); | |
6088 } | |
6089 } | |
6090 }, | |
6091 "paste" : { | |
6092 "separator_before" : false, | |
6093 "icon" : false, | |
6094 "_disabled" : function (data) { | |
6095 return !$.jstree.reference(data.reference).can_paste(); | |
6096 }, | |
6097 "separator_after" : false, | |
6098 "label" : "Paste", | |
6099 "action" : function (data) { | |
6100 var inst = $.jstree.reference(data.reference), | |
6101 obj = inst.get_node(data.reference); | |
6102 inst.paste(obj); | |
6103 } | |
6104 } | |
6105 } | |
6106 } | |
6107 }; | |
6108 } | |
6109 }; | |
6110 | |
6111 $.jstree.plugins.contextmenu = function (options, parent) { | |
6112 this.bind = function () { | |
6113 parent.bind.call(this); | |
6114 | |
6115 var last_ts = 0, cto = null, ex, ey; | |
6116 this.element | |
6117 .on("init.jstree loading.jstree ready.jstree", $.proxy(function () { | |
6118 this.get_container_ul().addClass('jstree-contextmenu'); | |
6119 }, this)) | |
6120 .on("contextmenu.jstree", ".jstree-anchor", $.proxy(function (e, data) { | |
6121 if (e.target.tagName.toLowerCase() === 'input') { | |
6122 return; | |
6123 } | |
6124 e.preventDefault(); | |
6125 last_ts = e.ctrlKey ? +new Date() : 0; | |
6126 if(data || cto) { | |
6127 last_ts = (+new Date()) + 10000; | |
6128 } | |
6129 if(cto) { | |
6130 clearTimeout(cto); | |
6131 } | |
6132 if(!this.is_loading(e.currentTarget)) { | |
6133 this.show_contextmenu(e.currentTarget, e.pageX, e.pageY, e); | |
6134 } | |
6135 }, this)) | |
6136 .on("click.jstree", ".jstree-anchor", $.proxy(function (e) { | |
6137 if(this._data.contextmenu.visible && (!last_ts || (+new Date()) - last_ts > 250)) { // work around safari & macOS ctrl+click | |
6138 $.vakata.context.hide(); | |
6139 } | |
6140 last_ts = 0; | |
6141 }, this)) | |
6142 .on("touchstart.jstree", ".jstree-anchor", function (e) { | |
6143 if(!e.originalEvent || !e.originalEvent.changedTouches || !e.originalEvent.changedTouches[0]) { | |
6144 return; | |
6145 } | |
6146 ex = e.originalEvent.changedTouches[0].clientX; | |
6147 ey = e.originalEvent.changedTouches[0].clientY; | |
6148 cto = setTimeout(function () { | |
6149 $(e.currentTarget).trigger('contextmenu', true); | |
6150 }, 750); | |
6151 }) | |
6152 .on('touchmove.vakata.jstree', function (e) { | |
6153 if(cto && e.originalEvent && e.originalEvent.changedTouches && e.originalEvent.changedTouches[0] && (Math.abs(ex - e.originalEvent.changedTouches[0].clientX) > 10 || Math.abs(ey - e.originalEvent.changedTouches[0].clientY) > 10)) { | |
6154 clearTimeout(cto); | |
6155 $.vakata.context.hide(); | |
6156 } | |
6157 }) | |
6158 .on('touchend.vakata.jstree', function (e) { | |
6159 if(cto) { | |
6160 clearTimeout(cto); | |
6161 } | |
6162 }); | |
6163 | |
6164 /*! | |
6165 if(!('oncontextmenu' in document.body) && ('ontouchstart' in document.body)) { | |
6166 var el = null, tm = null; | |
6167 this.element | |
6168 .on("touchstart", ".jstree-anchor", function (e) { | |
6169 el = e.currentTarget; | |
6170 tm = +new Date(); | |
6171 $(document).one("touchend", function (e) { | |
6172 e.target = document.elementFromPoint(e.originalEvent.targetTouches[0].pageX - window.pageXOffset, e.originalEvent.targetTouches[0].pageY - window.pageYOffset); | |
6173 e.currentTarget = e.target; | |
6174 tm = ((+(new Date())) - tm); | |
6175 if(e.target === el && tm > 600 && tm < 1000) { | |
6176 e.preventDefault(); | |
6177 $(el).trigger('contextmenu', e); | |
6178 } | |
6179 el = null; | |
6180 tm = null; | |
6181 }); | |
6182 }); | |
6183 } | |
6184 */ | |
6185 $(document).on("context_hide.vakata.jstree", $.proxy(function (e, data) { | |
6186 this._data.contextmenu.visible = false; | |
6187 $(data.reference).removeClass('jstree-context'); | |
6188 }, this)); | |
6189 }; | |
6190 this.teardown = function () { | |
6191 if(this._data.contextmenu.visible) { | |
6192 $.vakata.context.hide(); | |
6193 } | |
6194 parent.teardown.call(this); | |
6195 }; | |
6196 | |
6197 /** | |
6198 * prepare and show the context menu for a node | |
6199 * @name show_contextmenu(obj [, x, y]) | |
6200 * @param {mixed} obj the node | |
6201 * @param {Number} x the x-coordinate relative to the document to show the menu at | |
6202 * @param {Number} y the y-coordinate relative to the document to show the menu at | |
6203 * @param {Object} e the event if available that triggered the contextmenu | |
6204 * @plugin contextmenu | |
6205 * @trigger show_contextmenu.jstree | |
6206 */ | |
6207 this.show_contextmenu = function (obj, x, y, e) { | |
6208 obj = this.get_node(obj); | |
6209 if(!obj || obj.id === $.jstree.root) { return false; } | |
6210 var s = this.settings.contextmenu, | |
6211 d = this.get_node(obj, true), | |
6212 a = d.children(".jstree-anchor"), | |
6213 o = false, | |
6214 i = false; | |
6215 if(s.show_at_node || x === undefined || y === undefined) { | |
6216 o = a.offset(); | |
6217 x = o.left; | |
6218 y = o.top + this._data.core.li_height; | |
6219 } | |
6220 if(this.settings.contextmenu.select_node && !this.is_selected(obj)) { | |
6221 this.activate_node(obj, e); | |
6222 } | |
6223 | |
6224 i = s.items; | |
6225 if($.isFunction(i)) { | |
6226 i = i.call(this, obj, $.proxy(function (i) { | |
6227 this._show_contextmenu(obj, x, y, i); | |
6228 }, this)); | |
6229 } | |
6230 if($.isPlainObject(i)) { | |
6231 this._show_contextmenu(obj, x, y, i); | |
6232 } | |
6233 }; | |
6234 /** | |
6235 * show the prepared context menu for a node | |
6236 * @name _show_contextmenu(obj, x, y, i) | |
6237 * @param {mixed} obj the node | |
6238 * @param {Number} x the x-coordinate relative to the document to show the menu at | |
6239 * @param {Number} y the y-coordinate relative to the document to show the menu at | |
6240 * @param {Number} i the object of items to show | |
6241 * @plugin contextmenu | |
6242 * @trigger show_contextmenu.jstree | |
6243 * @private | |
6244 */ | |
6245 this._show_contextmenu = function (obj, x, y, i) { | |
6246 var d = this.get_node(obj, true), | |
6247 a = d.children(".jstree-anchor"); | |
6248 $(document).one("context_show.vakata.jstree", $.proxy(function (e, data) { | |
6249 var cls = 'jstree-contextmenu jstree-' + this.get_theme() + '-contextmenu'; | |
6250 $(data.element).addClass(cls); | |
6251 a.addClass('jstree-context'); | |
6252 }, this)); | |
6253 this._data.contextmenu.visible = true; | |
6254 $.vakata.context.show(a, { 'x' : x, 'y' : y }, i); | |
6255 /** | |
6256 * triggered when the contextmenu is shown for a node | |
6257 * @event | |
6258 * @name show_contextmenu.jstree | |
6259 * @param {Object} node the node | |
6260 * @param {Number} x the x-coordinate of the menu relative to the document | |
6261 * @param {Number} y the y-coordinate of the menu relative to the document | |
6262 * @plugin contextmenu | |
6263 */ | |
6264 this.trigger('show_contextmenu', { "node" : obj, "x" : x, "y" : y }); | |
6265 }; | |
6266 }; | |
6267 | |
6268 // contextmenu helper | |
6269 (function ($) { | |
6270 var right_to_left = false, | |
6271 vakata_context = { | |
6272 element : false, | |
6273 reference : false, | |
6274 position_x : 0, | |
6275 position_y : 0, | |
6276 items : [], | |
6277 html : "", | |
6278 is_visible : false | |
6279 }; | |
6280 | |
6281 $.vakata.context = { | |
6282 settings : { | |
6283 hide_onmouseleave : 0, | |
6284 icons : true | |
6285 }, | |
6286 _trigger : function (event_name) { | |
6287 $(document).triggerHandler("context_" + event_name + ".vakata", { | |
6288 "reference" : vakata_context.reference, | |
6289 "element" : vakata_context.element, | |
6290 "position" : { | |
6291 "x" : vakata_context.position_x, | |
6292 "y" : vakata_context.position_y | |
6293 } | |
6294 }); | |
6295 }, | |
6296 _execute : function (i) { | |
6297 i = vakata_context.items[i]; | |
6298 return i && (!i._disabled || ($.isFunction(i._disabled) && !i._disabled({ "item" : i, "reference" : vakata_context.reference, "element" : vakata_context.element }))) && i.action ? i.action.call(null, { | |
6299 "item" : i, | |
6300 "reference" : vakata_context.reference, | |
6301 "element" : vakata_context.element, | |
6302 "position" : { | |
6303 "x" : vakata_context.position_x, | |
6304 "y" : vakata_context.position_y | |
6305 } | |
6306 }) : false; | |
6307 }, | |
6308 _parse : function (o, is_callback) { | |
6309 if(!o) { return false; } | |
6310 if(!is_callback) { | |
6311 vakata_context.html = ""; | |
6312 vakata_context.items = []; | |
6313 } | |
6314 var str = "", | |
6315 sep = false, | |
6316 tmp; | |
6317 | |
6318 if(is_callback) { str += "<"+"ul>"; } | |
6319 $.each(o, function (i, val) { | |
6320 if(!val) { return true; } | |
6321 vakata_context.items.push(val); | |
6322 if(!sep && val.separator_before) { | |
6323 str += "<"+"li class='vakata-context-separator'><"+"a href='#' " + ($.vakata.context.settings.icons ? '' : 'style="margin-left:0px;"') + "> <"+"/a><"+"/li>"; | |
6324 } | |
6325 sep = false; | |
6326 str += "<"+"li class='" + (val._class || "") + (val._disabled === true || ($.isFunction(val._disabled) && val._disabled({ "item" : val, "reference" : vakata_context.reference, "element" : vakata_context.element })) ? " vakata-contextmenu-disabled " : "") + "' "+(val.shortcut?" data-shortcut='"+val.shortcut+"' ":'')+">"; | |
6327 str += "<"+"a href='#' rel='" + (vakata_context.items.length - 1) + "' " + (val.title ? "title='" + val.title + "'" : "") + ">"; | |
6328 if($.vakata.context.settings.icons) { | |
6329 str += "<"+"i "; | |
6330 if(val.icon) { | |
6331 if(val.icon.indexOf("/") !== -1 || val.icon.indexOf(".") !== -1) { str += " style='background:url(\"" + val.icon + "\") center center no-repeat' "; } | |
6332 else { str += " class='" + val.icon + "' "; } | |
6333 } | |
6334 str += "><"+"/i><"+"span class='vakata-contextmenu-sep'> <"+"/span>"; | |
6335 } | |
6336 str += ($.isFunction(val.label) ? val.label({ "item" : i, "reference" : vakata_context.reference, "element" : vakata_context.element }) : val.label) + (val.shortcut?' <span class="vakata-contextmenu-shortcut vakata-contextmenu-shortcut-'+val.shortcut+'">'+ (val.shortcut_label || '') +'</span>':'') + "<"+"/a>"; | |
6337 if(val.submenu) { | |
6338 tmp = $.vakata.context._parse(val.submenu, true); | |
6339 if(tmp) { str += tmp; } | |
6340 } | |
6341 str += "<"+"/li>"; | |
6342 if(val.separator_after) { | |
6343 str += "<"+"li class='vakata-context-separator'><"+"a href='#' " + ($.vakata.context.settings.icons ? '' : 'style="margin-left:0px;"') + "> <"+"/a><"+"/li>"; | |
6344 sep = true; | |
6345 } | |
6346 }); | |
6347 str = str.replace(/<li class\='vakata-context-separator'\><\/li\>$/,""); | |
6348 if(is_callback) { str += "</ul>"; } | |
6349 /** | |
6350 * triggered on the document when the contextmenu is parsed (HTML is built) | |
6351 * @event | |
6352 * @plugin contextmenu | |
6353 * @name context_parse.vakata | |
6354 * @param {jQuery} reference the element that was right clicked | |
6355 * @param {jQuery} element the DOM element of the menu itself | |
6356 * @param {Object} position the x & y coordinates of the menu | |
6357 */ | |
6358 if(!is_callback) { vakata_context.html = str; $.vakata.context._trigger("parse"); } | |
6359 return str.length > 10 ? str : false; | |
6360 }, | |
6361 _show_submenu : function (o) { | |
6362 o = $(o); | |
6363 if(!o.length || !o.children("ul").length) { return; } | |
6364 var e = o.children("ul"), | |
6365 xl = o.offset().left, | |
6366 x = xl + o.outerWidth(), | |
6367 y = o.offset().top, | |
6368 w = e.width(), | |
6369 h = e.height(), | |
6370 dw = $(window).width() + $(window).scrollLeft(), | |
6371 dh = $(window).height() + $(window).scrollTop(); | |
6372 // може да се спести е една проверка - дали няма някой от класовете вече нагоре | |
6373 if(right_to_left) { | |
6374 o[x - (w + 10 + o.outerWidth()) < 0 ? "addClass" : "removeClass"]("vakata-context-left"); | |
6375 } | |
6376 else { | |
6377 o[x + w > dw && xl > dw - x ? "addClass" : "removeClass"]("vakata-context-right"); | |
6378 } | |
6379 if(y + h + 10 > dh) { | |
6380 e.css("bottom","-1px"); | |
6381 } | |
6382 | |
6383 //if does not fit - stick it to the side | |
6384 if (o.hasClass('vakata-context-right')) { | |
6385 if (xl < w) { | |
6386 e.css("margin-right", xl - w); | |
6387 } | |
6388 } else { | |
6389 if (dw - x < w) { | |
6390 e.css("margin-left", dw - x - w); | |
6391 } | |
6392 } | |
6393 | |
6394 e.show(); | |
6395 }, | |
6396 show : function (reference, position, data) { | |
6397 var o, e, x, y, w, h, dw, dh, cond = true; | |
6398 if(vakata_context.element && vakata_context.element.length) { | |
6399 vakata_context.element.width(''); | |
6400 } | |
6401 switch(cond) { | |
6402 case (!position && !reference): | |
6403 return false; | |
6404 case (!!position && !!reference): | |
6405 vakata_context.reference = reference; | |
6406 vakata_context.position_x = position.x; | |
6407 vakata_context.position_y = position.y; | |
6408 break; | |
6409 case (!position && !!reference): | |
6410 vakata_context.reference = reference; | |
6411 o = reference.offset(); | |
6412 vakata_context.position_x = o.left + reference.outerHeight(); | |
6413 vakata_context.position_y = o.top; | |
6414 break; | |
6415 case (!!position && !reference): | |
6416 vakata_context.position_x = position.x; | |
6417 vakata_context.position_y = position.y; | |
6418 break; | |
6419 } | |
6420 if(!!reference && !data && $(reference).data('vakata_contextmenu')) { | |
6421 data = $(reference).data('vakata_contextmenu'); | |
6422 } | |
6423 if($.vakata.context._parse(data)) { | |
6424 vakata_context.element.html(vakata_context.html); | |
6425 } | |
6426 if(vakata_context.items.length) { | |
6427 vakata_context.element.appendTo("body"); | |
6428 e = vakata_context.element; | |
6429 x = vakata_context.position_x; | |
6430 y = vakata_context.position_y; | |
6431 w = e.width(); | |
6432 h = e.height(); | |
6433 dw = $(window).width() + $(window).scrollLeft(); | |
6434 dh = $(window).height() + $(window).scrollTop(); | |
6435 if(right_to_left) { | |
6436 x -= (e.outerWidth() - $(reference).outerWidth()); | |
6437 if(x < $(window).scrollLeft() + 20) { | |
6438 x = $(window).scrollLeft() + 20; | |
6439 } | |
6440 } | |
6441 if(x + w + 20 > dw) { | |
6442 x = dw - (w + 20); | |
6443 } | |
6444 if(y + h + 20 > dh) { | |
6445 y = dh - (h + 20); | |
6446 } | |
6447 | |
6448 vakata_context.element | |
6449 .css({ "left" : x, "top" : y }) | |
6450 .show() | |
6451 .find('a').first().focus().parent().addClass("vakata-context-hover"); | |
6452 vakata_context.is_visible = true; | |
6453 /** | |
6454 * triggered on the document when the contextmenu is shown | |
6455 * @event | |
6456 * @plugin contextmenu | |
6457 * @name context_show.vakata | |
6458 * @param {jQuery} reference the element that was right clicked | |
6459 * @param {jQuery} element the DOM element of the menu itself | |
6460 * @param {Object} position the x & y coordinates of the menu | |
6461 */ | |
6462 $.vakata.context._trigger("show"); | |
6463 } | |
6464 }, | |
6465 hide : function () { | |
6466 if(vakata_context.is_visible) { | |
6467 vakata_context.element.hide().find("ul").hide().end().find(':focus').blur().end().detach(); | |
6468 vakata_context.is_visible = false; | |
6469 /** | |
6470 * triggered on the document when the contextmenu is hidden | |
6471 * @event | |
6472 * @plugin contextmenu | |
6473 * @name context_hide.vakata | |
6474 * @param {jQuery} reference the element that was right clicked | |
6475 * @param {jQuery} element the DOM element of the menu itself | |
6476 * @param {Object} position the x & y coordinates of the menu | |
6477 */ | |
6478 $.vakata.context._trigger("hide"); | |
6479 } | |
6480 } | |
6481 }; | |
6482 $(function () { | |
6483 right_to_left = $("body").css("direction") === "rtl"; | |
6484 var to = false; | |
6485 | |
6486 vakata_context.element = $("<ul class='vakata-context'></ul>"); | |
6487 vakata_context.element | |
6488 .on("mouseenter", "li", function (e) { | |
6489 e.stopImmediatePropagation(); | |
6490 | |
6491 if($.contains(this, e.relatedTarget)) { | |
6492 // премахнато заради delegate mouseleave по-долу | |
6493 // $(this).find(".vakata-context-hover").removeClass("vakata-context-hover"); | |
6494 return; | |
6495 } | |
6496 | |
6497 if(to) { clearTimeout(to); } | |
6498 vakata_context.element.find(".vakata-context-hover").removeClass("vakata-context-hover").end(); | |
6499 | |
6500 $(this) | |
6501 .siblings().find("ul").hide().end().end() | |
6502 .parentsUntil(".vakata-context", "li").addBack().addClass("vakata-context-hover"); | |
6503 $.vakata.context._show_submenu(this); | |
6504 }) | |
6505 // тестово - дали не натоварва? | |
6506 .on("mouseleave", "li", function (e) { | |
6507 if($.contains(this, e.relatedTarget)) { return; } | |
6508 $(this).find(".vakata-context-hover").addBack().removeClass("vakata-context-hover"); | |
6509 }) | |
6510 .on("mouseleave", function (e) { | |
6511 $(this).find(".vakata-context-hover").removeClass("vakata-context-hover"); | |
6512 if($.vakata.context.settings.hide_onmouseleave) { | |
6513 to = setTimeout( | |
6514 (function (t) { | |
6515 return function () { $.vakata.context.hide(); }; | |
6516 }(this)), $.vakata.context.settings.hide_onmouseleave); | |
6517 } | |
6518 }) | |
6519 .on("click", "a", function (e) { | |
6520 e.preventDefault(); | |
6521 //}) | |
6522 //.on("mouseup", "a", function (e) { | |
6523 if(!$(this).blur().parent().hasClass("vakata-context-disabled") && $.vakata.context._execute($(this).attr("rel")) !== false) { | |
6524 $.vakata.context.hide(); | |
6525 } | |
6526 }) | |
6527 .on('keydown', 'a', function (e) { | |
6528 var o = null; | |
6529 switch(e.which) { | |
6530 case 13: | |
6531 case 32: | |
6532 e.type = "click"; | |
6533 e.preventDefault(); | |
6534 $(e.currentTarget).trigger(e); | |
6535 break; | |
6536 case 37: | |
6537 if(vakata_context.is_visible) { | |
6538 vakata_context.element.find(".vakata-context-hover").last().closest("li").first().find("ul").hide().find(".vakata-context-hover").removeClass("vakata-context-hover").end().end().children('a').focus(); | |
6539 e.stopImmediatePropagation(); | |
6540 e.preventDefault(); | |
6541 } | |
6542 break; | |
6543 case 38: | |
6544 if(vakata_context.is_visible) { | |
6545 o = vakata_context.element.find("ul:visible").addBack().last().children(".vakata-context-hover").removeClass("vakata-context-hover").prevAll("li:not(.vakata-context-separator)").first(); | |
6546 if(!o.length) { o = vakata_context.element.find("ul:visible").addBack().last().children("li:not(.vakata-context-separator)").last(); } | |
6547 o.addClass("vakata-context-hover").children('a').focus(); | |
6548 e.stopImmediatePropagation(); | |
6549 e.preventDefault(); | |
6550 } | |
6551 break; | |
6552 case 39: | |
6553 if(vakata_context.is_visible) { | |
6554 vakata_context.element.find(".vakata-context-hover").last().children("ul").show().children("li:not(.vakata-context-separator)").removeClass("vakata-context-hover").first().addClass("vakata-context-hover").children('a').focus(); | |
6555 e.stopImmediatePropagation(); | |
6556 e.preventDefault(); | |
6557 } | |
6558 break; | |
6559 case 40: | |
6560 if(vakata_context.is_visible) { | |
6561 o = vakata_context.element.find("ul:visible").addBack().last().children(".vakata-context-hover").removeClass("vakata-context-hover").nextAll("li:not(.vakata-context-separator)").first(); | |
6562 if(!o.length) { o = vakata_context.element.find("ul:visible").addBack().last().children("li:not(.vakata-context-separator)").first(); } | |
6563 o.addClass("vakata-context-hover").children('a').focus(); | |
6564 e.stopImmediatePropagation(); | |
6565 e.preventDefault(); | |
6566 } | |
6567 break; | |
6568 case 27: | |
6569 $.vakata.context.hide(); | |
6570 e.preventDefault(); | |
6571 break; | |
6572 default: | |
6573 //console.log(e.which); | |
6574 break; | |
6575 } | |
6576 }) | |
6577 .on('keydown', function (e) { | |
6578 e.preventDefault(); | |
6579 var a = vakata_context.element.find('.vakata-contextmenu-shortcut-' + e.which).parent(); | |
6580 if(a.parent().not('.vakata-context-disabled')) { | |
6581 a.click(); | |
6582 } | |
6583 }); | |
6584 | |
6585 $(document) | |
6586 .on("mousedown.vakata.jstree", function (e) { | |
6587 if(vakata_context.is_visible && vakata_context.element[0] !== e.target && !$.contains(vakata_context.element[0], e.target)) { | |
6588 $.vakata.context.hide(); | |
6589 } | |
6590 }) | |
6591 .on("context_show.vakata.jstree", function (e, data) { | |
6592 vakata_context.element.find("li:has(ul)").children("a").addClass("vakata-context-parent"); | |
6593 if(right_to_left) { | |
6594 vakata_context.element.addClass("vakata-context-rtl").css("direction", "rtl"); | |
6595 } | |
6596 // also apply a RTL class? | |
6597 vakata_context.element.find("ul").hide().end(); | |
6598 }); | |
6599 }); | |
6600 }($)); | |
6601 // $.jstree.defaults.plugins.push("contextmenu"); | |
6602 | |
6603 | |
6604 /** | |
6605 * ### Drag'n'drop plugin | |
6606 * | |
6607 * Enables dragging and dropping of nodes in the tree, resulting in a move or copy operations. | |
6608 */ | |
6609 | |
6610 /** | |
6611 * stores all defaults for the drag'n'drop plugin | |
6612 * @name $.jstree.defaults.dnd | |
6613 * @plugin dnd | |
6614 */ | |
6615 $.jstree.defaults.dnd = { | |
6616 /** | |
6617 * a boolean indicating if a copy should be possible while dragging (by pressint the meta key or Ctrl). Defaults to `true`. | |
6618 * @name $.jstree.defaults.dnd.copy | |
6619 * @plugin dnd | |
6620 */ | |
6621 copy : true, | |
6622 /** | |
6623 * a number indicating how long a node should remain hovered while dragging to be opened. Defaults to `500`. | |
6624 * @name $.jstree.defaults.dnd.open_timeout | |
6625 * @plugin dnd | |
6626 */ | |
6627 open_timeout : 500, | |
6628 /** | |
6629 * a function invoked each time a node is about to be dragged, invoked in the tree's scope and receives the nodes about to be dragged as an argument (array) and the event that started the drag - return `false` to prevent dragging | |
6630 * @name $.jstree.defaults.dnd.is_draggable | |
6631 * @plugin dnd | |
6632 */ | |
6633 is_draggable : true, | |
6634 /** | |
6635 * a boolean indicating if checks should constantly be made while the user is dragging the node (as opposed to checking only on drop), default is `true` | |
6636 * @name $.jstree.defaults.dnd.check_while_dragging | |
6637 * @plugin dnd | |
6638 */ | |
6639 check_while_dragging : true, | |
6640 /** | |
6641 * a boolean indicating if nodes from this tree should only be copied with dnd (as opposed to moved), default is `false` | |
6642 * @name $.jstree.defaults.dnd.always_copy | |
6643 * @plugin dnd | |
6644 */ | |
6645 always_copy : false, | |
6646 /** | |
6647 * when dropping a node "inside", this setting indicates the position the node should go to - it can be an integer or a string: "first" (same as 0) or "last", default is `0` | |
6648 * @name $.jstree.defaults.dnd.inside_pos | |
6649 * @plugin dnd | |
6650 */ | |
6651 inside_pos : 0, | |
6652 /** | |
6653 * when starting the drag on a node that is selected this setting controls if all selected nodes are dragged or only the single node, default is `true`, which means all selected nodes are dragged when the drag is started on a selected node | |
6654 * @name $.jstree.defaults.dnd.drag_selection | |
6655 * @plugin dnd | |
6656 */ | |
6657 drag_selection : true, | |
6658 /** | |
6659 * controls whether dnd works on touch devices. If left as boolean true dnd will work the same as in desktop browsers, which in some cases may impair scrolling. If set to boolean false dnd will not work on touch devices. There is a special third option - string "selected" which means only selected nodes can be dragged on touch devices. | |
6660 * @name $.jstree.defaults.dnd.touch | |
6661 * @plugin dnd | |
6662 */ | |
6663 touch : true, | |
6664 /** | |
6665 * controls whether items can be dropped anywhere on the node, not just on the anchor, by default only the node anchor is a valid drop target. Works best with the wholerow plugin. If enabled on mobile depending on the interface it might be hard for the user to cancel the drop, since the whole tree container will be a valid drop target. | |
6666 * @name $.jstree.defaults.dnd.large_drop_target | |
6667 * @plugin dnd | |
6668 */ | |
6669 large_drop_target : false, | |
6670 /** | |
6671 * controls whether a drag can be initiated from any part of the node and not just the text/icon part, works best with the wholerow plugin. Keep in mind it can cause problems with tree scrolling on mobile depending on the interface - in that case set the touch option to "selected". | |
6672 * @name $.jstree.defaults.dnd.large_drag_target | |
6673 * @plugin dnd | |
6674 */ | |
6675 large_drag_target : false, | |
6676 /** | |
6677 * controls whether use HTML5 dnd api instead of classical. That will allow better integration of dnd events with other HTML5 controls. | |
6678 * @reference http://caniuse.com/#feat=dragndrop | |
6679 * @name $.jstree.defaults.dnd.use_html5 | |
6680 * @plugin dnd | |
6681 */ | |
6682 use_html5: false | |
6683 }; | |
6684 var drg, elm; | |
6685 // TODO: now check works by checking for each node individually, how about max_children, unique, etc? | |
6686 $.jstree.plugins.dnd = function (options, parent) { | |
6687 this.init = function (el, options) { | |
6688 parent.init.call(this, el, options); | |
6689 this.settings.dnd.use_html5 = this.settings.dnd.use_html5 && ('draggable' in document.createElement('span')); | |
6690 }; | |
6691 this.bind = function () { | |
6692 parent.bind.call(this); | |
6693 | |
6694 this.element | |
6695 .on(this.settings.dnd.use_html5 ? 'dragstart.jstree' : 'mousedown.jstree touchstart.jstree', this.settings.dnd.large_drag_target ? '.jstree-node' : '.jstree-anchor', $.proxy(function (e) { | |
6696 if(this.settings.dnd.large_drag_target && $(e.target).closest('.jstree-node')[0] !== e.currentTarget) { | |
6697 return true; | |
6698 } | |
6699 if(e.type === "touchstart" && (!this.settings.dnd.touch || (this.settings.dnd.touch === 'selected' && !$(e.currentTarget).closest('.jstree-node').children('.jstree-anchor').hasClass('jstree-clicked')))) { | |
6700 return true; | |
6701 } | |
6702 var obj = this.get_node(e.target), | |
6703 mlt = this.is_selected(obj) && this.settings.dnd.drag_selection ? this.get_top_selected().length : 1, | |
6704 txt = (mlt > 1 ? mlt + ' ' + this.get_string('nodes') : this.get_text(e.currentTarget)); | |
6705 if(this.settings.core.force_text) { | |
6706 txt = $.vakata.html.escape(txt); | |
6707 } | |
6708 if(obj && obj.id && obj.id !== $.jstree.root && (e.which === 1 || e.type === "touchstart" || e.type === "dragstart") && | |
6709 (this.settings.dnd.is_draggable === true || ($.isFunction(this.settings.dnd.is_draggable) && this.settings.dnd.is_draggable.call(this, (mlt > 1 ? this.get_top_selected(true) : [obj]), e))) | |
6710 ) { | |
6711 drg = { 'jstree' : true, 'origin' : this, 'obj' : this.get_node(obj,true), 'nodes' : mlt > 1 ? this.get_top_selected() : [obj.id] }; | |
6712 elm = e.currentTarget; | |
6713 if (this.settings.dnd.use_html5) { | |
6714 $.vakata.dnd._trigger('start', e, { 'helper': $(), 'element': elm, 'data': drg }); | |
6715 } else { | |
6716 this.element.trigger('mousedown.jstree'); | |
6717 return $.vakata.dnd.start(e, drg, '<div id="jstree-dnd" class="jstree-' + this.get_theme() + ' jstree-' + this.get_theme() + '-' + this.get_theme_variant() + ' ' + ( this.settings.core.themes.responsive ? ' jstree-dnd-responsive' : '' ) + '"><i class="jstree-icon jstree-er"></i>' + txt + '<ins class="jstree-copy" style="display:none;">+</ins></div>'); | |
6718 } | |
6719 } | |
6720 }, this)); | |
6721 if (this.settings.dnd.use_html5) { | |
6722 this.element | |
6723 .on('dragover.jstree', function (e) { | |
6724 e.preventDefault(); | |
6725 $.vakata.dnd._trigger('move', e, { 'helper': $(), 'element': elm, 'data': drg }); | |
6726 return false; | |
6727 }) | |
6728 //.on('dragenter.jstree', this.settings.dnd.large_drop_target ? '.jstree-node' : '.jstree-anchor', $.proxy(function (e) { | |
6729 // e.preventDefault(); | |
6730 // $.vakata.dnd._trigger('move', e, { 'helper': $(), 'element': elm, 'data': drg }); | |
6731 // return false; | |
6732 // }, this)) | |
6733 .on('drop.jstree', $.proxy(function (e) { | |
6734 e.preventDefault(); | |
6735 $.vakata.dnd._trigger('stop', e, { 'helper': $(), 'element': elm, 'data': drg }); | |
6736 return false; | |
6737 }, this)); | |
6738 } | |
6739 }; | |
6740 this.redraw_node = function(obj, deep, callback, force_render) { | |
6741 obj = parent.redraw_node.apply(this, arguments); | |
6742 if (obj && this.settings.dnd.use_html5) { | |
6743 if (this.settings.dnd.large_drag_target) { | |
6744 obj.setAttribute('draggable', true); | |
6745 } else { | |
6746 var i, j, tmp = null; | |
6747 for(i = 0, j = obj.childNodes.length; i < j; i++) { | |
6748 if(obj.childNodes[i] && obj.childNodes[i].className && obj.childNodes[i].className.indexOf("jstree-anchor") !== -1) { | |
6749 tmp = obj.childNodes[i]; | |
6750 break; | |
6751 } | |
6752 } | |
6753 if(tmp) { | |
6754 tmp.setAttribute('draggable', true); | |
6755 } | |
6756 } | |
6757 } | |
6758 return obj; | |
6759 }; | |
6760 }; | |
6761 | |
6762 $(function() { | |
6763 // bind only once for all instances | |
6764 var lastmv = false, | |
6765 laster = false, | |
6766 lastev = false, | |
6767 opento = false, | |
6768 marker = $('<div id="jstree-marker"> </div>').hide(); //.appendTo('body'); | |
6769 | |
6770 $(document) | |
6771 .on('dnd_start.vakata.jstree', function (e, data) { | |
6772 lastmv = false; | |
6773 lastev = false; | |
6774 if(!data || !data.data || !data.data.jstree) { return; } | |
6775 marker.appendTo('body'); //.show(); | |
6776 }) | |
6777 .on('dnd_move.vakata.jstree', function (e, data) { | |
6778 var isDifferentNode = data.event.target !== lastev.target; | |
6779 if(opento) { | |
6780 if (!data.event || data.event.type !== 'dragover' || isDifferentNode) { | |
6781 clearTimeout(opento); | |
6782 } | |
6783 } | |
6784 if(!data || !data.data || !data.data.jstree) { return; } | |
6785 | |
6786 // if we are hovering the marker image do nothing (can happen on "inside" drags) | |
6787 if(data.event.target.id && data.event.target.id === 'jstree-marker') { | |
6788 return; | |
6789 } | |
6790 lastev = data.event; | |
6791 | |
6792 var ins = $.jstree.reference(data.event.target), | |
6793 ref = false, | |
6794 off = false, | |
6795 rel = false, | |
6796 tmp, l, t, h, p, i, o, ok, t1, t2, op, ps, pr, ip, tm, is_copy, pn; | |
6797 // if we are over an instance | |
6798 if(ins && ins._data && ins._data.dnd) { | |
6799 marker.attr('class', 'jstree-' + ins.get_theme() + ( ins.settings.core.themes.responsive ? ' jstree-dnd-responsive' : '' )); | |
6800 is_copy = data.data.origin && (data.data.origin.settings.dnd.always_copy || (data.data.origin.settings.dnd.copy && (data.event.metaKey || data.event.ctrlKey))); | |
6801 data.helper | |
6802 .children().attr('class', 'jstree-' + ins.get_theme() + ' jstree-' + ins.get_theme() + '-' + ins.get_theme_variant() + ' ' + ( ins.settings.core.themes.responsive ? ' jstree-dnd-responsive' : '' )) | |
6803 .find('.jstree-copy').first()[ is_copy ? 'show' : 'hide' ](); | |
6804 | |
6805 // if are hovering the container itself add a new root node | |
6806 //console.log(data.event); | |
6807 if( (data.event.target === ins.element[0] || data.event.target === ins.get_container_ul()[0]) && ins.get_container_ul().children().length === 0) { | |
6808 ok = true; | |
6809 for(t1 = 0, t2 = data.data.nodes.length; t1 < t2; t1++) { | |
6810 ok = ok && ins.check( (data.data.origin && (data.data.origin.settings.dnd.always_copy || (data.data.origin.settings.dnd.copy && (data.event.metaKey || data.event.ctrlKey)) ) ? "copy_node" : "move_node"), (data.data.origin && data.data.origin !== ins ? data.data.origin.get_node(data.data.nodes[t1]) : data.data.nodes[t1]), $.jstree.root, 'last', { 'dnd' : true, 'ref' : ins.get_node($.jstree.root), 'pos' : 'i', 'origin' : data.data.origin, 'is_multi' : (data.data.origin && data.data.origin !== ins), 'is_foreign' : (!data.data.origin) }); | |
6811 if(!ok) { break; } | |
6812 } | |
6813 if(ok) { | |
6814 lastmv = { 'ins' : ins, 'par' : $.jstree.root, 'pos' : 'last' }; | |
6815 marker.hide(); | |
6816 data.helper.find('.jstree-icon').first().removeClass('jstree-er').addClass('jstree-ok'); | |
6817 if (data.event.originalEvent && data.event.originalEvent.dataTransfer) { | |
6818 data.event.originalEvent.dataTransfer.dropEffect = is_copy ? 'copy' : 'move'; | |
6819 } | |
6820 return; | |
6821 } | |
6822 } | |
6823 else { | |
6824 // if we are hovering a tree node | |
6825 ref = ins.settings.dnd.large_drop_target ? $(data.event.target).closest('.jstree-node').children('.jstree-anchor') : $(data.event.target).closest('.jstree-anchor'); | |
6826 if(ref && ref.length && ref.parent().is('.jstree-closed, .jstree-open, .jstree-leaf')) { | |
6827 off = ref.offset(); | |
6828 rel = (data.event.pageY !== undefined ? data.event.pageY : data.event.originalEvent.pageY) - off.top; | |
6829 h = ref.outerHeight(); | |
6830 if(rel < h / 3) { | |
6831 o = ['b', 'i', 'a']; | |
6832 } | |
6833 else if(rel > h - h / 3) { | |
6834 o = ['a', 'i', 'b']; | |
6835 } | |
6836 else { | |
6837 o = rel > h / 2 ? ['i', 'a', 'b'] : ['i', 'b', 'a']; | |
6838 } | |
6839 $.each(o, function (j, v) { | |
6840 switch(v) { | |
6841 case 'b': | |
6842 l = off.left - 6; | |
6843 t = off.top; | |
6844 p = ins.get_parent(ref); | |
6845 i = ref.parent().index(); | |
6846 break; | |
6847 case 'i': | |
6848 ip = ins.settings.dnd.inside_pos; | |
6849 tm = ins.get_node(ref.parent()); | |
6850 l = off.left - 2; | |
6851 t = off.top + h / 2 + 1; | |
6852 p = tm.id; | |
6853 i = ip === 'first' ? 0 : (ip === 'last' ? tm.children.length : Math.min(ip, tm.children.length)); | |
6854 break; | |
6855 case 'a': | |
6856 l = off.left - 6; | |
6857 t = off.top + h; | |
6858 p = ins.get_parent(ref); | |
6859 i = ref.parent().index() + 1; | |
6860 break; | |
6861 } | |
6862 ok = true; | |
6863 for(t1 = 0, t2 = data.data.nodes.length; t1 < t2; t1++) { | |
6864 op = data.data.origin && (data.data.origin.settings.dnd.always_copy || (data.data.origin.settings.dnd.copy && (data.event.metaKey || data.event.ctrlKey))) ? "copy_node" : "move_node"; | |
6865 ps = i; | |
6866 if(op === "move_node" && v === 'a' && (data.data.origin && data.data.origin === ins) && p === ins.get_parent(data.data.nodes[t1])) { | |
6867 pr = ins.get_node(p); | |
6868 if(ps > $.inArray(data.data.nodes[t1], pr.children)) { | |
6869 ps -= 1; | |
6870 } | |
6871 } | |
6872 ok = ok && ( (ins && ins.settings && ins.settings.dnd && ins.settings.dnd.check_while_dragging === false) || ins.check(op, (data.data.origin && data.data.origin !== ins ? data.data.origin.get_node(data.data.nodes[t1]) : data.data.nodes[t1]), p, ps, { 'dnd' : true, 'ref' : ins.get_node(ref.parent()), 'pos' : v, 'origin' : data.data.origin, 'is_multi' : (data.data.origin && data.data.origin !== ins), 'is_foreign' : (!data.data.origin) }) ); | |
6873 if(!ok) { | |
6874 if(ins && ins.last_error) { laster = ins.last_error(); } | |
6875 break; | |
6876 } | |
6877 } | |
6878 if(v === 'i' && ref.parent().is('.jstree-closed') && ins.settings.dnd.open_timeout) { | |
6879 if (!data.event || data.event.type !== 'dragover' || isDifferentNode) { | |
6880 if (opento) { clearTimeout(opento); } | |
6881 opento = setTimeout((function (x, z) { return function () { x.open_node(z); }; }(ins, ref)), ins.settings.dnd.open_timeout); | |
6882 } | |
6883 } | |
6884 if(ok) { | |
6885 pn = ins.get_node(p, true); | |
6886 if (!pn.hasClass('.jstree-dnd-parent')) { | |
6887 $('.jstree-dnd-parent').removeClass('jstree-dnd-parent'); | |
6888 pn.addClass('jstree-dnd-parent'); | |
6889 } | |
6890 lastmv = { 'ins' : ins, 'par' : p, 'pos' : v === 'i' && ip === 'last' && i === 0 && !ins.is_loaded(tm) ? 'last' : i }; | |
6891 marker.css({ 'left' : l + 'px', 'top' : t + 'px' }).show(); | |
6892 data.helper.find('.jstree-icon').first().removeClass('jstree-er').addClass('jstree-ok'); | |
6893 if (data.event.originalEvent && data.event.originalEvent.dataTransfer) { | |
6894 data.event.originalEvent.dataTransfer.dropEffect = is_copy ? 'copy' : 'move'; | |
6895 } | |
6896 laster = {}; | |
6897 o = true; | |
6898 return false; | |
6899 } | |
6900 }); | |
6901 if(o === true) { return; } | |
6902 } | |
6903 } | |
6904 } | |
6905 $('.jstree-dnd-parent').removeClass('jstree-dnd-parent'); | |
6906 lastmv = false; | |
6907 data.helper.find('.jstree-icon').removeClass('jstree-ok').addClass('jstree-er'); | |
6908 if (data.event.originalEvent && data.event.originalEvent.dataTransfer) { | |
6909 data.event.originalEvent.dataTransfer.dropEffect = 'none'; | |
6910 } | |
6911 marker.hide(); | |
6912 }) | |
6913 .on('dnd_scroll.vakata.jstree', function (e, data) { | |
6914 if(!data || !data.data || !data.data.jstree) { return; } | |
6915 marker.hide(); | |
6916 lastmv = false; | |
6917 lastev = false; | |
6918 data.helper.find('.jstree-icon').first().removeClass('jstree-ok').addClass('jstree-er'); | |
6919 }) | |
6920 .on('dnd_stop.vakata.jstree', function (e, data) { | |
6921 $('.jstree-dnd-parent').removeClass('jstree-dnd-parent'); | |
6922 if(opento) { clearTimeout(opento); } | |
6923 if(!data || !data.data || !data.data.jstree) { return; } | |
6924 marker.hide().detach(); | |
6925 var i, j, nodes = []; | |
6926 if(lastmv) { | |
6927 for(i = 0, j = data.data.nodes.length; i < j; i++) { | |
6928 nodes[i] = data.data.origin ? data.data.origin.get_node(data.data.nodes[i]) : data.data.nodes[i]; | |
6929 } | |
6930 lastmv.ins[ data.data.origin && (data.data.origin.settings.dnd.always_copy || (data.data.origin.settings.dnd.copy && (data.event.metaKey || data.event.ctrlKey))) ? 'copy_node' : 'move_node' ](nodes, lastmv.par, lastmv.pos, false, false, false, data.data.origin); | |
6931 } | |
6932 else { | |
6933 i = $(data.event.target).closest('.jstree'); | |
6934 if(i.length && laster && laster.error && laster.error === 'check') { | |
6935 i = i.jstree(true); | |
6936 if(i) { | |
6937 i.settings.core.error.call(this, laster); | |
6938 } | |
6939 } | |
6940 } | |
6941 lastev = false; | |
6942 lastmv = false; | |
6943 }) | |
6944 .on('keyup.jstree keydown.jstree', function (e, data) { | |
6945 data = $.vakata.dnd._get(); | |
6946 if(data && data.data && data.data.jstree) { | |
6947 if (e.type === "keyup" && e.which === 27) { | |
6948 if (opento) { clearTimeout(opento); } | |
6949 lastmv = false; | |
6950 laster = false; | |
6951 lastev = false; | |
6952 opento = false; | |
6953 marker.hide().detach(); | |
6954 $.vakata.dnd._clean(); | |
6955 } else { | |
6956 data.helper.find('.jstree-copy').first()[ data.data.origin && (data.data.origin.settings.dnd.always_copy || (data.data.origin.settings.dnd.copy && (e.metaKey || e.ctrlKey))) ? 'show' : 'hide' ](); | |
6957 if(lastev) { | |
6958 lastev.metaKey = e.metaKey; | |
6959 lastev.ctrlKey = e.ctrlKey; | |
6960 $.vakata.dnd._trigger('move', lastev); | |
6961 } | |
6962 } | |
6963 } | |
6964 }); | |
6965 }); | |
6966 | |
6967 // helpers | |
6968 (function ($) { | |
6969 $.vakata.html = { | |
6970 div : $('<div />'), | |
6971 escape : function (str) { | |
6972 return $.vakata.html.div.text(str).html(); | |
6973 }, | |
6974 strip : function (str) { | |
6975 return $.vakata.html.div.empty().append($.parseHTML(str)).text(); | |
6976 } | |
6977 }; | |
6978 // private variable | |
6979 var vakata_dnd = { | |
6980 element : false, | |
6981 target : false, | |
6982 is_down : false, | |
6983 is_drag : false, | |
6984 helper : false, | |
6985 helper_w: 0, | |
6986 data : false, | |
6987 init_x : 0, | |
6988 init_y : 0, | |
6989 scroll_l: 0, | |
6990 scroll_t: 0, | |
6991 scroll_e: false, | |
6992 scroll_i: false, | |
6993 is_touch: false | |
6994 }; | |
6995 $.vakata.dnd = { | |
6996 settings : { | |
6997 scroll_speed : 10, | |
6998 scroll_proximity : 20, | |
6999 helper_left : 5, | |
7000 helper_top : 10, | |
7001 threshold : 5, | |
7002 threshold_touch : 10 | |
7003 }, | |
7004 _trigger : function (event_name, e, data) { | |
7005 if (data === undefined) { | |
7006 data = $.vakata.dnd._get(); | |
7007 } | |
7008 data.event = e; | |
7009 $(document).triggerHandler("dnd_" + event_name + ".vakata", data); | |
7010 }, | |
7011 _get : function () { | |
7012 return { | |
7013 "data" : vakata_dnd.data, | |
7014 "element" : vakata_dnd.element, | |
7015 "helper" : vakata_dnd.helper | |
7016 }; | |
7017 }, | |
7018 _clean : function () { | |
7019 if(vakata_dnd.helper) { vakata_dnd.helper.remove(); } | |
7020 if(vakata_dnd.scroll_i) { clearInterval(vakata_dnd.scroll_i); vakata_dnd.scroll_i = false; } | |
7021 vakata_dnd = { | |
7022 element : false, | |
7023 target : false, | |
7024 is_down : false, | |
7025 is_drag : false, | |
7026 helper : false, | |
7027 helper_w: 0, | |
7028 data : false, | |
7029 init_x : 0, | |
7030 init_y : 0, | |
7031 scroll_l: 0, | |
7032 scroll_t: 0, | |
7033 scroll_e: false, | |
7034 scroll_i: false, | |
7035 is_touch: false | |
7036 }; | |
7037 $(document).off("mousemove.vakata.jstree touchmove.vakata.jstree", $.vakata.dnd.drag); | |
7038 $(document).off("mouseup.vakata.jstree touchend.vakata.jstree", $.vakata.dnd.stop); | |
7039 }, | |
7040 _scroll : function (init_only) { | |
7041 if(!vakata_dnd.scroll_e || (!vakata_dnd.scroll_l && !vakata_dnd.scroll_t)) { | |
7042 if(vakata_dnd.scroll_i) { clearInterval(vakata_dnd.scroll_i); vakata_dnd.scroll_i = false; } | |
7043 return false; | |
7044 } | |
7045 if(!vakata_dnd.scroll_i) { | |
7046 vakata_dnd.scroll_i = setInterval($.vakata.dnd._scroll, 100); | |
7047 return false; | |
7048 } | |
7049 if(init_only === true) { return false; } | |
7050 | |
7051 var i = vakata_dnd.scroll_e.scrollTop(), | |
7052 j = vakata_dnd.scroll_e.scrollLeft(); | |
7053 vakata_dnd.scroll_e.scrollTop(i + vakata_dnd.scroll_t * $.vakata.dnd.settings.scroll_speed); | |
7054 vakata_dnd.scroll_e.scrollLeft(j + vakata_dnd.scroll_l * $.vakata.dnd.settings.scroll_speed); | |
7055 if(i !== vakata_dnd.scroll_e.scrollTop() || j !== vakata_dnd.scroll_e.scrollLeft()) { | |
7056 /** | |
7057 * triggered on the document when a drag causes an element to scroll | |
7058 * @event | |
7059 * @plugin dnd | |
7060 * @name dnd_scroll.vakata | |
7061 * @param {Mixed} data any data supplied with the call to $.vakata.dnd.start | |
7062 * @param {DOM} element the DOM element being dragged | |
7063 * @param {jQuery} helper the helper shown next to the mouse | |
7064 * @param {jQuery} event the element that is scrolling | |
7065 */ | |
7066 $.vakata.dnd._trigger("scroll", vakata_dnd.scroll_e); | |
7067 } | |
7068 }, | |
7069 start : function (e, data, html) { | |
7070 if(e.type === "touchstart" && e.originalEvent && e.originalEvent.changedTouches && e.originalEvent.changedTouches[0]) { | |
7071 e.pageX = e.originalEvent.changedTouches[0].pageX; | |
7072 e.pageY = e.originalEvent.changedTouches[0].pageY; | |
7073 e.target = document.elementFromPoint(e.originalEvent.changedTouches[0].pageX - window.pageXOffset, e.originalEvent.changedTouches[0].pageY - window.pageYOffset); | |
7074 } | |
7075 if(vakata_dnd.is_drag) { $.vakata.dnd.stop({}); } | |
7076 try { | |
7077 e.currentTarget.unselectable = "on"; | |
7078 e.currentTarget.onselectstart = function() { return false; }; | |
7079 if(e.currentTarget.style) { | |
7080 e.currentTarget.style.touchAction = "none"; | |
7081 e.currentTarget.style.msTouchAction = "none"; | |
7082 e.currentTarget.style.MozUserSelect = "none"; | |
7083 } | |
7084 } catch(ignore) { } | |
7085 vakata_dnd.init_x = e.pageX; | |
7086 vakata_dnd.init_y = e.pageY; | |
7087 vakata_dnd.data = data; | |
7088 vakata_dnd.is_down = true; | |
7089 vakata_dnd.element = e.currentTarget; | |
7090 vakata_dnd.target = e.target; | |
7091 vakata_dnd.is_touch = e.type === "touchstart"; | |
7092 if(html !== false) { | |
7093 vakata_dnd.helper = $("<div id='vakata-dnd'></div>").html(html).css({ | |
7094 "display" : "block", | |
7095 "margin" : "0", | |
7096 "padding" : "0", | |
7097 "position" : "absolute", | |
7098 "top" : "-2000px", | |
7099 "lineHeight" : "16px", | |
7100 "zIndex" : "10000" | |
7101 }); | |
7102 } | |
7103 $(document).on("mousemove.vakata.jstree touchmove.vakata.jstree", $.vakata.dnd.drag); | |
7104 $(document).on("mouseup.vakata.jstree touchend.vakata.jstree", $.vakata.dnd.stop); | |
7105 return false; | |
7106 }, | |
7107 drag : function (e) { | |
7108 if(e.type === "touchmove" && e.originalEvent && e.originalEvent.changedTouches && e.originalEvent.changedTouches[0]) { | |
7109 e.pageX = e.originalEvent.changedTouches[0].pageX; | |
7110 e.pageY = e.originalEvent.changedTouches[0].pageY; | |
7111 e.target = document.elementFromPoint(e.originalEvent.changedTouches[0].pageX - window.pageXOffset, e.originalEvent.changedTouches[0].pageY - window.pageYOffset); | |
7112 } | |
7113 if(!vakata_dnd.is_down) { return; } | |
7114 if(!vakata_dnd.is_drag) { | |
7115 if( | |
7116 Math.abs(e.pageX - vakata_dnd.init_x) > (vakata_dnd.is_touch ? $.vakata.dnd.settings.threshold_touch : $.vakata.dnd.settings.threshold) || | |
7117 Math.abs(e.pageY - vakata_dnd.init_y) > (vakata_dnd.is_touch ? $.vakata.dnd.settings.threshold_touch : $.vakata.dnd.settings.threshold) | |
7118 ) { | |
7119 if(vakata_dnd.helper) { | |
7120 vakata_dnd.helper.appendTo("body"); | |
7121 vakata_dnd.helper_w = vakata_dnd.helper.outerWidth(); | |
7122 } | |
7123 vakata_dnd.is_drag = true; | |
7124 $(vakata_dnd.target).one('click.vakata', false); | |
7125 /** | |
7126 * triggered on the document when a drag starts | |
7127 * @event | |
7128 * @plugin dnd | |
7129 * @name dnd_start.vakata | |
7130 * @param {Mixed} data any data supplied with the call to $.vakata.dnd.start | |
7131 * @param {DOM} element the DOM element being dragged | |
7132 * @param {jQuery} helper the helper shown next to the mouse | |
7133 * @param {Object} event the event that caused the start (probably mousemove) | |
7134 */ | |
7135 $.vakata.dnd._trigger("start", e); | |
7136 } | |
7137 else { return; } | |
7138 } | |
7139 | |
7140 var d = false, w = false, | |
7141 dh = false, wh = false, | |
7142 dw = false, ww = false, | |
7143 dt = false, dl = false, | |
7144 ht = false, hl = false; | |
7145 | |
7146 vakata_dnd.scroll_t = 0; | |
7147 vakata_dnd.scroll_l = 0; | |
7148 vakata_dnd.scroll_e = false; | |
7149 $($(e.target).parentsUntil("body").addBack().get().reverse()) | |
7150 .filter(function () { | |
7151 return (/^auto|scroll$/).test($(this).css("overflow")) && | |
7152 (this.scrollHeight > this.offsetHeight || this.scrollWidth > this.offsetWidth); | |
7153 }) | |
7154 .each(function () { | |
7155 var t = $(this), o = t.offset(); | |
7156 if(this.scrollHeight > this.offsetHeight) { | |
7157 if(o.top + t.height() - e.pageY < $.vakata.dnd.settings.scroll_proximity) { vakata_dnd.scroll_t = 1; } | |
7158 if(e.pageY - o.top < $.vakata.dnd.settings.scroll_proximity) { vakata_dnd.scroll_t = -1; } | |
7159 } | |
7160 if(this.scrollWidth > this.offsetWidth) { | |
7161 if(o.left + t.width() - e.pageX < $.vakata.dnd.settings.scroll_proximity) { vakata_dnd.scroll_l = 1; } | |
7162 if(e.pageX - o.left < $.vakata.dnd.settings.scroll_proximity) { vakata_dnd.scroll_l = -1; } | |
7163 } | |
7164 if(vakata_dnd.scroll_t || vakata_dnd.scroll_l) { | |
7165 vakata_dnd.scroll_e = $(this); | |
7166 return false; | |
7167 } | |
7168 }); | |
7169 | |
7170 if(!vakata_dnd.scroll_e) { | |
7171 d = $(document); w = $(window); | |
7172 dh = d.height(); wh = w.height(); | |
7173 dw = d.width(); ww = w.width(); | |
7174 dt = d.scrollTop(); dl = d.scrollLeft(); | |
7175 if(dh > wh && e.pageY - dt < $.vakata.dnd.settings.scroll_proximity) { vakata_dnd.scroll_t = -1; } | |
7176 if(dh > wh && wh - (e.pageY - dt) < $.vakata.dnd.settings.scroll_proximity) { vakata_dnd.scroll_t = 1; } | |
7177 if(dw > ww && e.pageX - dl < $.vakata.dnd.settings.scroll_proximity) { vakata_dnd.scroll_l = -1; } | |
7178 if(dw > ww && ww - (e.pageX - dl) < $.vakata.dnd.settings.scroll_proximity) { vakata_dnd.scroll_l = 1; } | |
7179 if(vakata_dnd.scroll_t || vakata_dnd.scroll_l) { | |
7180 vakata_dnd.scroll_e = d; | |
7181 } | |
7182 } | |
7183 if(vakata_dnd.scroll_e) { $.vakata.dnd._scroll(true); } | |
7184 | |
7185 if(vakata_dnd.helper) { | |
7186 ht = parseInt(e.pageY + $.vakata.dnd.settings.helper_top, 10); | |
7187 hl = parseInt(e.pageX + $.vakata.dnd.settings.helper_left, 10); | |
7188 if(dh && ht + 25 > dh) { ht = dh - 50; } | |
7189 if(dw && hl + vakata_dnd.helper_w > dw) { hl = dw - (vakata_dnd.helper_w + 2); } | |
7190 vakata_dnd.helper.css({ | |
7191 left : hl + "px", | |
7192 top : ht + "px" | |
7193 }); | |
7194 } | |
7195 /** | |
7196 * triggered on the document when a drag is in progress | |
7197 * @event | |
7198 * @plugin dnd | |
7199 * @name dnd_move.vakata | |
7200 * @param {Mixed} data any data supplied with the call to $.vakata.dnd.start | |
7201 * @param {DOM} element the DOM element being dragged | |
7202 * @param {jQuery} helper the helper shown next to the mouse | |
7203 * @param {Object} event the event that caused this to trigger (most likely mousemove) | |
7204 */ | |
7205 $.vakata.dnd._trigger("move", e); | |
7206 return false; | |
7207 }, | |
7208 stop : function (e) { | |
7209 if(e.type === "touchend" && e.originalEvent && e.originalEvent.changedTouches && e.originalEvent.changedTouches[0]) { | |
7210 e.pageX = e.originalEvent.changedTouches[0].pageX; | |
7211 e.pageY = e.originalEvent.changedTouches[0].pageY; | |
7212 e.target = document.elementFromPoint(e.originalEvent.changedTouches[0].pageX - window.pageXOffset, e.originalEvent.changedTouches[0].pageY - window.pageYOffset); | |
7213 } | |
7214 if(vakata_dnd.is_drag) { | |
7215 /** | |
7216 * triggered on the document when a drag stops (the dragged element is dropped) | |
7217 * @event | |
7218 * @plugin dnd | |
7219 * @name dnd_stop.vakata | |
7220 * @param {Mixed} data any data supplied with the call to $.vakata.dnd.start | |
7221 * @param {DOM} element the DOM element being dragged | |
7222 * @param {jQuery} helper the helper shown next to the mouse | |
7223 * @param {Object} event the event that caused the stop | |
7224 */ | |
7225 if (e.target !== vakata_dnd.target) { | |
7226 $(vakata_dnd.target).off('click.vakata'); | |
7227 } | |
7228 $.vakata.dnd._trigger("stop", e); | |
7229 } | |
7230 else { | |
7231 if(e.type === "touchend" && e.target === vakata_dnd.target) { | |
7232 var to = setTimeout(function () { $(e.target).click(); }, 100); | |
7233 $(e.target).one('click', function() { if(to) { clearTimeout(to); } }); | |
7234 } | |
7235 } | |
7236 $.vakata.dnd._clean(); | |
7237 return false; | |
7238 } | |
7239 }; | |
7240 }($)); | |
7241 | |
7242 // include the dnd plugin by default | |
7243 // $.jstree.defaults.plugins.push("dnd"); | |
7244 | |
7245 | |
7246 /** | |
7247 * ### Massload plugin | |
7248 * | |
7249 * Adds massload functionality to jsTree, so that multiple nodes can be loaded in a single request (only useful with lazy loading). | |
7250 */ | |
7251 | |
7252 /** | |
7253 * massload configuration | |
7254 * | |
7255 * It is possible to set this to a standard jQuery-like AJAX config. | |
7256 * In addition to the standard jQuery ajax options here you can supply functions for `data` and `url`, the functions will be run in the current instance's scope and a param will be passed indicating which node IDs need to be loaded, the return value of those functions will be used. | |
7257 * | |
7258 * You can also set this to a function, that function will receive the node IDs being loaded as argument and a second param which is a function (callback) which should be called with the result. | |
7259 * | |
7260 * Both the AJAX and the function approach rely on the same return value - an object where the keys are the node IDs, and the value is the children of that node as an array. | |
7261 * | |
7262 * { | |
7263 * "id1" : [{ "text" : "Child of ID1", "id" : "c1" }, { "text" : "Another child of ID1", "id" : "c2" }], | |
7264 * "id2" : [{ "text" : "Child of ID2", "id" : "c3" }] | |
7265 * } | |
7266 * | |
7267 * @name $.jstree.defaults.massload | |
7268 * @plugin massload | |
7269 */ | |
7270 $.jstree.defaults.massload = null; | |
7271 $.jstree.plugins.massload = function (options, parent) { | |
7272 this.init = function (el, options) { | |
7273 this._data.massload = {}; | |
7274 parent.init.call(this, el, options); | |
7275 }; | |
7276 this._load_nodes = function (nodes, callback, is_callback, force_reload) { | |
7277 var s = this.settings.massload, | |
7278 nodesString = JSON.stringify(nodes), | |
7279 toLoad = [], | |
7280 m = this._model.data, | |
7281 i, j, dom; | |
7282 if (!is_callback) { | |
7283 for(i = 0, j = nodes.length; i < j; i++) { | |
7284 if(!m[nodes[i]] || ( (!m[nodes[i]].state.loaded && !m[nodes[i]].state.failed) || force_reload) ) { | |
7285 toLoad.push(nodes[i]); | |
7286 dom = this.get_node(nodes[i], true); | |
7287 if (dom && dom.length) { | |
7288 dom.addClass("jstree-loading").attr('aria-busy',true); | |
7289 } | |
7290 } | |
7291 } | |
7292 this._data.massload = {}; | |
7293 if (toLoad.length) { | |
7294 if($.isFunction(s)) { | |
7295 return s.call(this, toLoad, $.proxy(function (data) { | |
7296 var i, j; | |
7297 if(data) { | |
7298 for(i in data) { | |
7299 if(data.hasOwnProperty(i)) { | |
7300 this._data.massload[i] = data[i]; | |
7301 } | |
7302 } | |
7303 } | |
7304 for(i = 0, j = nodes.length; i < j; i++) { | |
7305 dom = this.get_node(nodes[i], true); | |
7306 if (dom && dom.length) { | |
7307 dom.removeClass("jstree-loading").attr('aria-busy',false); | |
7308 } | |
7309 } | |
7310 parent._load_nodes.call(this, nodes, callback, is_callback, force_reload); | |
7311 }, this)); | |
7312 } | |
7313 if(typeof s === 'object' && s && s.url) { | |
7314 s = $.extend(true, {}, s); | |
7315 if($.isFunction(s.url)) { | |
7316 s.url = s.url.call(this, toLoad); | |
7317 } | |
7318 if($.isFunction(s.data)) { | |
7319 s.data = s.data.call(this, toLoad); | |
7320 } | |
7321 return $.ajax(s) | |
7322 .done($.proxy(function (data,t,x) { | |
7323 var i, j; | |
7324 if(data) { | |
7325 for(i in data) { | |
7326 if(data.hasOwnProperty(i)) { | |
7327 this._data.massload[i] = data[i]; | |
7328 } | |
7329 } | |
7330 } | |
7331 for(i = 0, j = nodes.length; i < j; i++) { | |
7332 dom = this.get_node(nodes[i], true); | |
7333 if (dom && dom.length) { | |
7334 dom.removeClass("jstree-loading").attr('aria-busy',false); | |
7335 } | |
7336 } | |
7337 parent._load_nodes.call(this, nodes, callback, is_callback, force_reload); | |
7338 }, this)) | |
7339 .fail($.proxy(function (f) { | |
7340 parent._load_nodes.call(this, nodes, callback, is_callback, force_reload); | |
7341 }, this)); | |
7342 } | |
7343 } | |
7344 } | |
7345 return parent._load_nodes.call(this, nodes, callback, is_callback, force_reload); | |
7346 }; | |
7347 this._load_node = function (obj, callback) { | |
7348 var data = this._data.massload[obj.id], | |
7349 rslt = null, dom; | |
7350 if(data) { | |
7351 rslt = this[typeof data === 'string' ? '_append_html_data' : '_append_json_data']( | |
7352 obj, | |
7353 typeof data === 'string' ? $($.parseHTML(data)).filter(function () { return this.nodeType !== 3; }) : data, | |
7354 function (status) { callback.call(this, status); } | |
7355 ); | |
7356 dom = this.get_node(obj.id, true); | |
7357 if (dom && dom.length) { | |
7358 dom.removeClass("jstree-loading").attr('aria-busy',false); | |
7359 } | |
7360 delete this._data.massload[obj.id]; | |
7361 return rslt; | |
7362 } | |
7363 return parent._load_node.call(this, obj, callback); | |
7364 }; | |
7365 }; | |
7366 | |
7367 /** | |
7368 * ### Search plugin | |
7369 * | |
7370 * Adds search functionality to jsTree. | |
7371 */ | |
7372 | |
7373 /** | |
7374 * stores all defaults for the search plugin | |
7375 * @name $.jstree.defaults.search | |
7376 * @plugin search | |
7377 */ | |
7378 $.jstree.defaults.search = { | |
7379 /** | |
7380 * a jQuery-like AJAX config, which jstree uses if a server should be queried for results. | |
7381 * | |
7382 * A `str` (which is the search string) parameter will be added with the request, an optional `inside` parameter will be added if the search is limited to a node id. The expected result is a JSON array with nodes that need to be opened so that matching nodes will be revealed. | |
7383 * Leave this setting as `false` to not query the server. You can also set this to a function, which will be invoked in the instance's scope and receive 3 parameters - the search string, the callback to call with the array of nodes to load, and the optional node ID to limit the search to | |
7384 * @name $.jstree.defaults.search.ajax | |
7385 * @plugin search | |
7386 */ | |
7387 ajax : false, | |
7388 /** | |
7389 * Indicates if the search should be fuzzy or not (should `chnd3` match `child node 3`). Default is `false`. | |
7390 * @name $.jstree.defaults.search.fuzzy | |
7391 * @plugin search | |
7392 */ | |
7393 fuzzy : false, | |
7394 /** | |
7395 * Indicates if the search should be case sensitive. Default is `false`. | |
7396 * @name $.jstree.defaults.search.case_sensitive | |
7397 * @plugin search | |
7398 */ | |
7399 case_sensitive : false, | |
7400 /** | |
7401 * Indicates if the tree should be filtered (by default) to show only matching nodes (keep in mind this can be a heavy on large trees in old browsers). | |
7402 * This setting can be changed at runtime when calling the search method. Default is `false`. | |
7403 * @name $.jstree.defaults.search.show_only_matches | |
7404 * @plugin search | |
7405 */ | |
7406 show_only_matches : false, | |
7407 /** | |
7408 * Indicates if the children of matched element are shown (when show_only_matches is true) | |
7409 * This setting can be changed at runtime when calling the search method. Default is `false`. | |
7410 * @name $.jstree.defaults.search.show_only_matches_children | |
7411 * @plugin search | |
7412 */ | |
7413 show_only_matches_children : false, | |
7414 /** | |
7415 * Indicates if all nodes opened to reveal the search result, should be closed when the search is cleared or a new search is performed. Default is `true`. | |
7416 * @name $.jstree.defaults.search.close_opened_onclear | |
7417 * @plugin search | |
7418 */ | |
7419 close_opened_onclear : true, | |
7420 /** | |
7421 * Indicates if only leaf nodes should be included in search results. Default is `false`. | |
7422 * @name $.jstree.defaults.search.search_leaves_only | |
7423 * @plugin search | |
7424 */ | |
7425 search_leaves_only : false, | |
7426 /** | |
7427 * If set to a function it wil be called in the instance's scope with two arguments - search string and node (where node will be every node in the structure, so use with caution). | |
7428 * If the function returns a truthy value the node will be considered a match (it might not be displayed if search_only_leaves is set to true and the node is not a leaf). Default is `false`. | |
7429 * @name $.jstree.defaults.search.search_callback | |
7430 * @plugin search | |
7431 */ | |
7432 search_callback : false | |
7433 }; | |
7434 | |
7435 $.jstree.plugins.search = function (options, parent) { | |
7436 this.bind = function () { | |
7437 parent.bind.call(this); | |
7438 | |
7439 this._data.search.str = ""; | |
7440 this._data.search.dom = $(); | |
7441 this._data.search.res = []; | |
7442 this._data.search.opn = []; | |
7443 this._data.search.som = false; | |
7444 this._data.search.smc = false; | |
7445 this._data.search.hdn = []; | |
7446 | |
7447 this.element | |
7448 .on("search.jstree", $.proxy(function (e, data) { | |
7449 if(this._data.search.som && data.res.length) { | |
7450 var m = this._model.data, i, j, p = [], k, l; | |
7451 for(i = 0, j = data.res.length; i < j; i++) { | |
7452 if(m[data.res[i]] && !m[data.res[i]].state.hidden) { | |
7453 p.push(data.res[i]); | |
7454 p = p.concat(m[data.res[i]].parents); | |
7455 if(this._data.search.smc) { | |
7456 for (k = 0, l = m[data.res[i]].children_d.length; k < l; k++) { | |
7457 if (m[m[data.res[i]].children_d[k]] && !m[m[data.res[i]].children_d[k]].state.hidden) { | |
7458 p.push(m[data.res[i]].children_d[k]); | |
7459 } | |
7460 } | |
7461 } | |
7462 } | |
7463 } | |
7464 p = $.vakata.array_remove_item($.vakata.array_unique(p), $.jstree.root); | |
7465 this._data.search.hdn = this.hide_all(true); | |
7466 this.show_node(p, true); | |
7467 this.redraw(true); | |
7468 } | |
7469 }, this)) | |
7470 .on("clear_search.jstree", $.proxy(function (e, data) { | |
7471 if(this._data.search.som && data.res.length) { | |
7472 this.show_node(this._data.search.hdn, true); | |
7473 this.redraw(true); | |
7474 } | |
7475 }, this)); | |
7476 }; | |
7477 /** | |
7478 * used to search the tree nodes for a given string | |
7479 * @name search(str [, skip_async]) | |
7480 * @param {String} str the search string | |
7481 * @param {Boolean} skip_async if set to true server will not be queried even if configured | |
7482 * @param {Boolean} show_only_matches if set to true only matching nodes will be shown (keep in mind this can be very slow on large trees or old browsers) | |
7483 * @param {mixed} inside an optional node to whose children to limit the search | |
7484 * @param {Boolean} append if set to true the results of this search are appended to the previous search | |
7485 * @plugin search | |
7486 * @trigger search.jstree | |
7487 */ | |
7488 this.search = function (str, skip_async, show_only_matches, inside, append, show_only_matches_children) { | |
7489 if(str === false || $.trim(str.toString()) === "") { | |
7490 return this.clear_search(); | |
7491 } | |
7492 inside = this.get_node(inside); | |
7493 inside = inside && inside.id ? inside.id : null; | |
7494 str = str.toString(); | |
7495 var s = this.settings.search, | |
7496 a = s.ajax ? s.ajax : false, | |
7497 m = this._model.data, | |
7498 f = null, | |
7499 r = [], | |
7500 p = [], i, j; | |
7501 if(this._data.search.res.length && !append) { | |
7502 this.clear_search(); | |
7503 } | |
7504 if(show_only_matches === undefined) { | |
7505 show_only_matches = s.show_only_matches; | |
7506 } | |
7507 if(show_only_matches_children === undefined) { | |
7508 show_only_matches_children = s.show_only_matches_children; | |
7509 } | |
7510 if(!skip_async && a !== false) { | |
7511 if($.isFunction(a)) { | |
7512 return a.call(this, str, $.proxy(function (d) { | |
7513 if(d && d.d) { d = d.d; } | |
7514 this._load_nodes(!$.isArray(d) ? [] : $.vakata.array_unique(d), function () { | |
7515 this.search(str, true, show_only_matches, inside, append, show_only_matches_children); | |
7516 }); | |
7517 }, this), inside); | |
7518 } | |
7519 else { | |
7520 a = $.extend({}, a); | |
7521 if(!a.data) { a.data = {}; } | |
7522 a.data.str = str; | |
7523 if(inside) { | |
7524 a.data.inside = inside; | |
7525 } | |
7526 if (this._data.search.lastRequest) { | |
7527 this._data.search.lastRequest.abort(); | |
7528 } | |
7529 this._data.search.lastRequest = $.ajax(a) | |
7530 .fail($.proxy(function () { | |
7531 this._data.core.last_error = { 'error' : 'ajax', 'plugin' : 'search', 'id' : 'search_01', 'reason' : 'Could not load search parents', 'data' : JSON.stringify(a) }; | |
7532 this.settings.core.error.call(this, this._data.core.last_error); | |
7533 }, this)) | |
7534 .done($.proxy(function (d) { | |
7535 if(d && d.d) { d = d.d; } | |
7536 this._load_nodes(!$.isArray(d) ? [] : $.vakata.array_unique(d), function () { | |
7537 this.search(str, true, show_only_matches, inside, append, show_only_matches_children); | |
7538 }); | |
7539 }, this)); | |
7540 return this._data.search.lastRequest; | |
7541 } | |
7542 } | |
7543 if(!append) { | |
7544 this._data.search.str = str; | |
7545 this._data.search.dom = $(); | |
7546 this._data.search.res = []; | |
7547 this._data.search.opn = []; | |
7548 this._data.search.som = show_only_matches; | |
7549 this._data.search.smc = show_only_matches_children; | |
7550 } | |
7551 | |
7552 f = new $.vakata.search(str, true, { caseSensitive : s.case_sensitive, fuzzy : s.fuzzy }); | |
7553 $.each(m[inside ? inside : $.jstree.root].children_d, function (ii, i) { | |
7554 var v = m[i]; | |
7555 if(v.text && !v.state.hidden && (!s.search_leaves_only || (v.state.loaded && v.children.length === 0)) && ( (s.search_callback && s.search_callback.call(this, str, v)) || (!s.search_callback && f.search(v.text).isMatch) ) ) { | |
7556 r.push(i); | |
7557 p = p.concat(v.parents); | |
7558 } | |
7559 }); | |
7560 if(r.length) { | |
7561 p = $.vakata.array_unique(p); | |
7562 for(i = 0, j = p.length; i < j; i++) { | |
7563 if(p[i] !== $.jstree.root && m[p[i]] && this.open_node(p[i], null, 0) === true) { | |
7564 this._data.search.opn.push(p[i]); | |
7565 } | |
7566 } | |
7567 if(!append) { | |
7568 this._data.search.dom = $(this.element[0].querySelectorAll('#' + $.map(r, function (v) { return "0123456789".indexOf(v[0]) !== -1 ? '\\3' + v[0] + ' ' + v.substr(1).replace($.jstree.idregex,'\\$&') : v.replace($.jstree.idregex,'\\$&'); }).join(', #'))); | |
7569 this._data.search.res = r; | |
7570 } | |
7571 else { | |
7572 this._data.search.dom = this._data.search.dom.add($(this.element[0].querySelectorAll('#' + $.map(r, function (v) { return "0123456789".indexOf(v[0]) !== -1 ? '\\3' + v[0] + ' ' + v.substr(1).replace($.jstree.idregex,'\\$&') : v.replace($.jstree.idregex,'\\$&'); }).join(', #')))); | |
7573 this._data.search.res = $.vakata.array_unique(this._data.search.res.concat(r)); | |
7574 } | |
7575 this._data.search.dom.children(".jstree-anchor").addClass('jstree-search'); | |
7576 } | |
7577 /** | |
7578 * triggered after search is complete | |
7579 * @event | |
7580 * @name search.jstree | |
7581 * @param {jQuery} nodes a jQuery collection of matching nodes | |
7582 * @param {String} str the search string | |
7583 * @param {Array} res a collection of objects represeing the matching nodes | |
7584 * @plugin search | |
7585 */ | |
7586 this.trigger('search', { nodes : this._data.search.dom, str : str, res : this._data.search.res, show_only_matches : show_only_matches }); | |
7587 }; | |
7588 /** | |
7589 * used to clear the last search (removes classes and shows all nodes if filtering is on) | |
7590 * @name clear_search() | |
7591 * @plugin search | |
7592 * @trigger clear_search.jstree | |
7593 */ | |
7594 this.clear_search = function () { | |
7595 if(this.settings.search.close_opened_onclear) { | |
7596 this.close_node(this._data.search.opn, 0); | |
7597 } | |
7598 /** | |
7599 * triggered after search is complete | |
7600 * @event | |
7601 * @name clear_search.jstree | |
7602 * @param {jQuery} nodes a jQuery collection of matching nodes (the result from the last search) | |
7603 * @param {String} str the search string (the last search string) | |
7604 * @param {Array} res a collection of objects represeing the matching nodes (the result from the last search) | |
7605 * @plugin search | |
7606 */ | |
7607 this.trigger('clear_search', { 'nodes' : this._data.search.dom, str : this._data.search.str, res : this._data.search.res }); | |
7608 if(this._data.search.res.length) { | |
7609 this._data.search.dom = $(this.element[0].querySelectorAll('#' + $.map(this._data.search.res, function (v) { | |
7610 return "0123456789".indexOf(v[0]) !== -1 ? '\\3' + v[0] + ' ' + v.substr(1).replace($.jstree.idregex,'\\$&') : v.replace($.jstree.idregex,'\\$&'); | |
7611 }).join(', #'))); | |
7612 this._data.search.dom.children(".jstree-anchor").removeClass("jstree-search"); | |
7613 } | |
7614 this._data.search.str = ""; | |
7615 this._data.search.res = []; | |
7616 this._data.search.opn = []; | |
7617 this._data.search.dom = $(); | |
7618 }; | |
7619 | |
7620 this.redraw_node = function(obj, deep, callback, force_render) { | |
7621 obj = parent.redraw_node.apply(this, arguments); | |
7622 if(obj) { | |
7623 if($.inArray(obj.id, this._data.search.res) !== -1) { | |
7624 var i, j, tmp = null; | |
7625 for(i = 0, j = obj.childNodes.length; i < j; i++) { | |
7626 if(obj.childNodes[i] && obj.childNodes[i].className && obj.childNodes[i].className.indexOf("jstree-anchor") !== -1) { | |
7627 tmp = obj.childNodes[i]; | |
7628 break; | |
7629 } | |
7630 } | |
7631 if(tmp) { | |
7632 tmp.className += ' jstree-search'; | |
7633 } | |
7634 } | |
7635 } | |
7636 return obj; | |
7637 }; | |
7638 }; | |
7639 | |
7640 // helpers | |
7641 (function ($) { | |
7642 // from http://kiro.me/projects/fuse.html | |
7643 $.vakata.search = function(pattern, txt, options) { | |
7644 options = options || {}; | |
7645 options = $.extend({}, $.vakata.search.defaults, options); | |
7646 if(options.fuzzy !== false) { | |
7647 options.fuzzy = true; | |
7648 } | |
7649 pattern = options.caseSensitive ? pattern : pattern.toLowerCase(); | |
7650 var MATCH_LOCATION = options.location, | |
7651 MATCH_DISTANCE = options.distance, | |
7652 MATCH_THRESHOLD = options.threshold, | |
7653 patternLen = pattern.length, | |
7654 matchmask, pattern_alphabet, match_bitapScore, search; | |
7655 if(patternLen > 32) { | |
7656 options.fuzzy = false; | |
7657 } | |
7658 if(options.fuzzy) { | |
7659 matchmask = 1 << (patternLen - 1); | |
7660 pattern_alphabet = (function () { | |
7661 var mask = {}, | |
7662 i = 0; | |
7663 for (i = 0; i < patternLen; i++) { | |
7664 mask[pattern.charAt(i)] = 0; | |
7665 } | |
7666 for (i = 0; i < patternLen; i++) { | |
7667 mask[pattern.charAt(i)] |= 1 << (patternLen - i - 1); | |
7668 } | |
7669 return mask; | |
7670 }()); | |
7671 match_bitapScore = function (e, x) { | |
7672 var accuracy = e / patternLen, | |
7673 proximity = Math.abs(MATCH_LOCATION - x); | |
7674 if(!MATCH_DISTANCE) { | |
7675 return proximity ? 1.0 : accuracy; | |
7676 } | |
7677 return accuracy + (proximity / MATCH_DISTANCE); | |
7678 }; | |
7679 } | |
7680 search = function (text) { | |
7681 text = options.caseSensitive ? text : text.toLowerCase(); | |
7682 if(pattern === text || text.indexOf(pattern) !== -1) { | |
7683 return { | |
7684 isMatch: true, | |
7685 score: 0 | |
7686 }; | |
7687 } | |
7688 if(!options.fuzzy) { | |
7689 return { | |
7690 isMatch: false, | |
7691 score: 1 | |
7692 }; | |
7693 } | |
7694 var i, j, | |
7695 textLen = text.length, | |
7696 scoreThreshold = MATCH_THRESHOLD, | |
7697 bestLoc = text.indexOf(pattern, MATCH_LOCATION), | |
7698 binMin, binMid, | |
7699 binMax = patternLen + textLen, | |
7700 lastRd, start, finish, rd, charMatch, | |
7701 score = 1, | |
7702 locations = []; | |
7703 if (bestLoc !== -1) { | |
7704 scoreThreshold = Math.min(match_bitapScore(0, bestLoc), scoreThreshold); | |
7705 bestLoc = text.lastIndexOf(pattern, MATCH_LOCATION + patternLen); | |
7706 if (bestLoc !== -1) { | |
7707 scoreThreshold = Math.min(match_bitapScore(0, bestLoc), scoreThreshold); | |
7708 } | |
7709 } | |
7710 bestLoc = -1; | |
7711 for (i = 0; i < patternLen; i++) { | |
7712 binMin = 0; | |
7713 binMid = binMax; | |
7714 while (binMin < binMid) { | |
7715 if (match_bitapScore(i, MATCH_LOCATION + binMid) <= scoreThreshold) { | |
7716 binMin = binMid; | |
7717 } else { | |
7718 binMax = binMid; | |
7719 } | |
7720 binMid = Math.floor((binMax - binMin) / 2 + binMin); | |
7721 } | |
7722 binMax = binMid; | |
7723 start = Math.max(1, MATCH_LOCATION - binMid + 1); | |
7724 finish = Math.min(MATCH_LOCATION + binMid, textLen) + patternLen; | |
7725 rd = new Array(finish + 2); | |
7726 rd[finish + 1] = (1 << i) - 1; | |
7727 for (j = finish; j >= start; j--) { | |
7728 charMatch = pattern_alphabet[text.charAt(j - 1)]; | |
7729 if (i === 0) { | |
7730 rd[j] = ((rd[j + 1] << 1) | 1) & charMatch; | |
7731 } else { | |
7732 rd[j] = ((rd[j + 1] << 1) | 1) & charMatch | (((lastRd[j + 1] | lastRd[j]) << 1) | 1) | lastRd[j + 1]; | |
7733 } | |
7734 if (rd[j] & matchmask) { | |
7735 score = match_bitapScore(i, j - 1); | |
7736 if (score <= scoreThreshold) { | |
7737 scoreThreshold = score; | |
7738 bestLoc = j - 1; | |
7739 locations.push(bestLoc); | |
7740 if (bestLoc > MATCH_LOCATION) { | |
7741 start = Math.max(1, 2 * MATCH_LOCATION - bestLoc); | |
7742 } else { | |
7743 break; | |
7744 } | |
7745 } | |
7746 } | |
7747 } | |
7748 if (match_bitapScore(i + 1, MATCH_LOCATION) > scoreThreshold) { | |
7749 break; | |
7750 } | |
7751 lastRd = rd; | |
7752 } | |
7753 return { | |
7754 isMatch: bestLoc >= 0, | |
7755 score: score | |
7756 }; | |
7757 }; | |
7758 return txt === true ? { 'search' : search } : search(txt); | |
7759 }; | |
7760 $.vakata.search.defaults = { | |
7761 location : 0, | |
7762 distance : 100, | |
7763 threshold : 0.6, | |
7764 fuzzy : false, | |
7765 caseSensitive : false | |
7766 }; | |
7767 }($)); | |
7768 | |
7769 // include the search plugin by default | |
7770 // $.jstree.defaults.plugins.push("search"); | |
7771 | |
7772 | |
7773 /** | |
7774 * ### Sort plugin | |
7775 * | |
7776 * Automatically sorts all siblings in the tree according to a sorting function. | |
7777 */ | |
7778 | |
7779 /** | |
7780 * the settings function used to sort the nodes. | |
7781 * It is executed in the tree's context, accepts two nodes as arguments and should return `1` or `-1`. | |
7782 * @name $.jstree.defaults.sort | |
7783 * @plugin sort | |
7784 */ | |
7785 $.jstree.defaults.sort = function (a, b) { | |
7786 //return this.get_type(a) === this.get_type(b) ? (this.get_text(a) > this.get_text(b) ? 1 : -1) : this.get_type(a) >= this.get_type(b); | |
7787 return this.get_text(a) > this.get_text(b) ? 1 : -1; | |
7788 }; | |
7789 $.jstree.plugins.sort = function (options, parent) { | |
7790 this.bind = function () { | |
7791 parent.bind.call(this); | |
7792 this.element | |
7793 .on("model.jstree", $.proxy(function (e, data) { | |
7794 this.sort(data.parent, true); | |
7795 }, this)) | |
7796 .on("rename_node.jstree create_node.jstree", $.proxy(function (e, data) { | |
7797 this.sort(data.parent || data.node.parent, false); | |
7798 this.redraw_node(data.parent || data.node.parent, true); | |
7799 }, this)) | |
7800 .on("move_node.jstree copy_node.jstree", $.proxy(function (e, data) { | |
7801 this.sort(data.parent, false); | |
7802 this.redraw_node(data.parent, true); | |
7803 }, this)); | |
7804 }; | |
7805 /** | |
7806 * used to sort a node's children | |
7807 * @private | |
7808 * @name sort(obj [, deep]) | |
7809 * @param {mixed} obj the node | |
7810 * @param {Boolean} deep if set to `true` nodes are sorted recursively. | |
7811 * @plugin sort | |
7812 * @trigger search.jstree | |
7813 */ | |
7814 this.sort = function (obj, deep) { | |
7815 var i, j; | |
7816 obj = this.get_node(obj); | |
7817 if(obj && obj.children && obj.children.length) { | |
7818 obj.children.sort($.proxy(this.settings.sort, this)); | |
7819 if(deep) { | |
7820 for(i = 0, j = obj.children_d.length; i < j; i++) { | |
7821 this.sort(obj.children_d[i], false); | |
7822 } | |
7823 } | |
7824 } | |
7825 }; | |
7826 }; | |
7827 | |
7828 // include the sort plugin by default | |
7829 // $.jstree.defaults.plugins.push("sort"); | |
7830 | |
7831 /** | |
7832 * ### State plugin | |
7833 * | |
7834 * Saves the state of the tree (selected nodes, opened nodes) on the user's computer using available options (localStorage, cookies, etc) | |
7835 */ | |
7836 | |
7837 var to = false; | |
7838 /** | |
7839 * stores all defaults for the state plugin | |
7840 * @name $.jstree.defaults.state | |
7841 * @plugin state | |
7842 */ | |
7843 $.jstree.defaults.state = { | |
7844 /** | |
7845 * A string for the key to use when saving the current tree (change if using multiple trees in your project). Defaults to `jstree`. | |
7846 * @name $.jstree.defaults.state.key | |
7847 * @plugin state | |
7848 */ | |
7849 key : 'jstree', | |
7850 /** | |
7851 * A space separated list of events that trigger a state save. Defaults to `changed.jstree open_node.jstree close_node.jstree`. | |
7852 * @name $.jstree.defaults.state.events | |
7853 * @plugin state | |
7854 */ | |
7855 events : 'changed.jstree open_node.jstree close_node.jstree check_node.jstree uncheck_node.jstree', | |
7856 /** | |
7857 * Time in milliseconds after which the state will expire. Defaults to 'false' meaning - no expire. | |
7858 * @name $.jstree.defaults.state.ttl | |
7859 * @plugin state | |
7860 */ | |
7861 ttl : false, | |
7862 /** | |
7863 * A function that will be executed prior to restoring state with one argument - the state object. Can be used to clear unwanted parts of the state. | |
7864 * @name $.jstree.defaults.state.filter | |
7865 * @plugin state | |
7866 */ | |
7867 filter : false, | |
7868 /** | |
7869 * Should loaded nodes be restored (setting this to true means that it is possible that the whole tree will be loaded for some users - use with caution). Defaults to `false` | |
7870 * @name $.jstree.defaults.state.preserve_loaded | |
7871 * @plugin state | |
7872 */ | |
7873 preserve_loaded : false | |
7874 }; | |
7875 $.jstree.plugins.state = function (options, parent) { | |
7876 this.bind = function () { | |
7877 parent.bind.call(this); | |
7878 var bind = $.proxy(function () { | |
7879 this.element.on(this.settings.state.events, $.proxy(function () { | |
7880 if(to) { clearTimeout(to); } | |
7881 to = setTimeout($.proxy(function () { this.save_state(); }, this), 100); | |
7882 }, this)); | |
7883 /** | |
7884 * triggered when the state plugin is finished restoring the state (and immediately after ready if there is no state to restore). | |
7885 * @event | |
7886 * @name state_ready.jstree | |
7887 * @plugin state | |
7888 */ | |
7889 this.trigger('state_ready'); | |
7890 }, this); | |
7891 this.element | |
7892 .on("ready.jstree", $.proxy(function (e, data) { | |
7893 this.element.one("restore_state.jstree", bind); | |
7894 if(!this.restore_state()) { bind(); } | |
7895 }, this)); | |
7896 }; | |
7897 /** | |
7898 * save the state | |
7899 * @name save_state() | |
7900 * @plugin state | |
7901 */ | |
7902 this.save_state = function () { | |
7903 var tm = this.get_state(); | |
7904 if (!this.settings.state.preserve_loaded) { | |
7905 delete tm.core.loaded; | |
7906 } | |
7907 var st = { 'state' : tm, 'ttl' : this.settings.state.ttl, 'sec' : +(new Date()) }; | |
7908 $.vakata.storage.set(this.settings.state.key, JSON.stringify(st)); | |
7909 }; | |
7910 /** | |
7911 * restore the state from the user's computer | |
7912 * @name restore_state() | |
7913 * @plugin state | |
7914 */ | |
7915 this.restore_state = function () { | |
7916 var k = $.vakata.storage.get(this.settings.state.key); | |
7917 if(!!k) { try { k = JSON.parse(k); } catch(ex) { return false; } } | |
7918 if(!!k && k.ttl && k.sec && +(new Date()) - k.sec > k.ttl) { return false; } | |
7919 if(!!k && k.state) { k = k.state; } | |
7920 if(!!k && $.isFunction(this.settings.state.filter)) { k = this.settings.state.filter.call(this, k); } | |
7921 if(!!k) { | |
7922 if (!this.settings.state.preserve_loaded) { | |
7923 delete k.core.loaded; | |
7924 } | |
7925 this.element.one("set_state.jstree", function (e, data) { data.instance.trigger('restore_state', { 'state' : $.extend(true, {}, k) }); }); | |
7926 this.set_state(k); | |
7927 return true; | |
7928 } | |
7929 return false; | |
7930 }; | |
7931 /** | |
7932 * clear the state on the user's computer | |
7933 * @name clear_state() | |
7934 * @plugin state | |
7935 */ | |
7936 this.clear_state = function () { | |
7937 return $.vakata.storage.del(this.settings.state.key); | |
7938 }; | |
7939 }; | |
7940 | |
7941 (function ($, undefined) { | |
7942 $.vakata.storage = { | |
7943 // simply specifying the functions in FF throws an error | |
7944 set : function (key, val) { return window.localStorage.setItem(key, val); }, | |
7945 get : function (key) { return window.localStorage.getItem(key); }, | |
7946 del : function (key) { return window.localStorage.removeItem(key); } | |
7947 }; | |
7948 }($)); | |
7949 | |
7950 // include the state plugin by default | |
7951 // $.jstree.defaults.plugins.push("state"); | |
7952 | |
7953 /** | |
7954 * ### Types plugin | |
7955 * | |
7956 * Makes it possible to add predefined types for groups of nodes, which make it possible to easily control nesting rules and icon for each group. | |
7957 */ | |
7958 | |
7959 /** | |
7960 * An object storing all types as key value pairs, where the key is the type name and the value is an object that could contain following keys (all optional). | |
7961 * | |
7962 * * `max_children` the maximum number of immediate children this node type can have. Do not specify or set to `-1` for unlimited. | |
7963 * * `max_depth` the maximum number of nesting this node type can have. A value of `1` would mean that the node can have children, but no grandchildren. Do not specify or set to `-1` for unlimited. | |
7964 * * `valid_children` an array of node type strings, that nodes of this type can have as children. Do not specify or set to `-1` for no limits. | |
7965 * * `icon` a string - can be a path to an icon or a className, if using an image that is in the current directory use a `./` prefix, otherwise it will be detected as a class. Omit to use the default icon from your theme. | |
7966 * * `li_attr` an object of values which will be used to add HTML attributes on the resulting LI DOM node (merged with the node's own data) | |
7967 * * `a_attr` an object of values which will be used to add HTML attributes on the resulting A DOM node (merged with the node's own data) | |
7968 * | |
7969 * There are two predefined types: | |
7970 * | |
7971 * * `#` represents the root of the tree, for example `max_children` would control the maximum number of root nodes. | |
7972 * * `default` represents the default node - any settings here will be applied to all nodes that do not have a type specified. | |
7973 * | |
7974 * @name $.jstree.defaults.types | |
7975 * @plugin types | |
7976 */ | |
7977 $.jstree.defaults.types = { | |
7978 'default' : {} | |
7979 }; | |
7980 $.jstree.defaults.types[$.jstree.root] = {}; | |
7981 | |
7982 $.jstree.plugins.types = function (options, parent) { | |
7983 this.init = function (el, options) { | |
7984 var i, j; | |
7985 if(options && options.types && options.types['default']) { | |
7986 for(i in options.types) { | |
7987 if(i !== "default" && i !== $.jstree.root && options.types.hasOwnProperty(i)) { | |
7988 for(j in options.types['default']) { | |
7989 if(options.types['default'].hasOwnProperty(j) && options.types[i][j] === undefined) { | |
7990 options.types[i][j] = options.types['default'][j]; | |
7991 } | |
7992 } | |
7993 } | |
7994 } | |
7995 } | |
7996 parent.init.call(this, el, options); | |
7997 this._model.data[$.jstree.root].type = $.jstree.root; | |
7998 }; | |
7999 this.refresh = function (skip_loading, forget_state) { | |
8000 parent.refresh.call(this, skip_loading, forget_state); | |
8001 this._model.data[$.jstree.root].type = $.jstree.root; | |
8002 }; | |
8003 this.bind = function () { | |
8004 this.element | |
8005 .on('model.jstree', $.proxy(function (e, data) { | |
8006 var m = this._model.data, | |
8007 dpc = data.nodes, | |
8008 t = this.settings.types, | |
8009 i, j, c = 'default', k; | |
8010 for(i = 0, j = dpc.length; i < j; i++) { | |
8011 c = 'default'; | |
8012 if(m[dpc[i]].original && m[dpc[i]].original.type && t[m[dpc[i]].original.type]) { | |
8013 c = m[dpc[i]].original.type; | |
8014 } | |
8015 if(m[dpc[i]].data && m[dpc[i]].data.jstree && m[dpc[i]].data.jstree.type && t[m[dpc[i]].data.jstree.type]) { | |
8016 c = m[dpc[i]].data.jstree.type; | |
8017 } | |
8018 m[dpc[i]].type = c; | |
8019 if(m[dpc[i]].icon === true && t[c].icon !== undefined) { | |
8020 m[dpc[i]].icon = t[c].icon; | |
8021 } | |
8022 if(t[c].li_attr !== undefined && typeof t[c].li_attr === 'object') { | |
8023 for (k in t[c].li_attr) { | |
8024 if (t[c].li_attr.hasOwnProperty(k)) { | |
8025 if (k === 'id') { | |
8026 continue; | |
8027 } | |
8028 else if (m[dpc[i]].li_attr[k] === undefined) { | |
8029 m[dpc[i]].li_attr[k] = t[c].li_attr[k]; | |
8030 } | |
8031 else if (k === 'class') { | |
8032 m[dpc[i]].li_attr['class'] = t[c].li_attr['class'] + ' ' + m[dpc[i]].li_attr['class']; | |
8033 } | |
8034 } | |
8035 } | |
8036 } | |
8037 if(t[c].a_attr !== undefined && typeof t[c].a_attr === 'object') { | |
8038 for (k in t[c].a_attr) { | |
8039 if (t[c].a_attr.hasOwnProperty(k)) { | |
8040 if (k === 'id') { | |
8041 continue; | |
8042 } | |
8043 else if (m[dpc[i]].a_attr[k] === undefined) { | |
8044 m[dpc[i]].a_attr[k] = t[c].a_attr[k]; | |
8045 } | |
8046 else if (k === 'href' && m[dpc[i]].a_attr[k] === '#') { | |
8047 m[dpc[i]].a_attr['href'] = t[c].a_attr['href']; | |
8048 } | |
8049 else if (k === 'class') { | |
8050 m[dpc[i]].a_attr['class'] = t[c].a_attr['class'] + ' ' + m[dpc[i]].a_attr['class']; | |
8051 } | |
8052 } | |
8053 } | |
8054 } | |
8055 } | |
8056 m[$.jstree.root].type = $.jstree.root; | |
8057 }, this)); | |
8058 parent.bind.call(this); | |
8059 }; | |
8060 this.get_json = function (obj, options, flat) { | |
8061 var i, j, | |
8062 m = this._model.data, | |
8063 opt = options ? $.extend(true, {}, options, {no_id:false}) : {}, | |
8064 tmp = parent.get_json.call(this, obj, opt, flat); | |
8065 if(tmp === false) { return false; } | |
8066 if($.isArray(tmp)) { | |
8067 for(i = 0, j = tmp.length; i < j; i++) { | |
8068 tmp[i].type = tmp[i].id && m[tmp[i].id] && m[tmp[i].id].type ? m[tmp[i].id].type : "default"; | |
8069 if(options && options.no_id) { | |
8070 delete tmp[i].id; | |
8071 if(tmp[i].li_attr && tmp[i].li_attr.id) { | |
8072 delete tmp[i].li_attr.id; | |
8073 } | |
8074 if(tmp[i].a_attr && tmp[i].a_attr.id) { | |
8075 delete tmp[i].a_attr.id; | |
8076 } | |
8077 } | |
8078 } | |
8079 } | |
8080 else { | |
8081 tmp.type = tmp.id && m[tmp.id] && m[tmp.id].type ? m[tmp.id].type : "default"; | |
8082 if(options && options.no_id) { | |
8083 tmp = this._delete_ids(tmp); | |
8084 } | |
8085 } | |
8086 return tmp; | |
8087 }; | |
8088 this._delete_ids = function (tmp) { | |
8089 if($.isArray(tmp)) { | |
8090 for(var i = 0, j = tmp.length; i < j; i++) { | |
8091 tmp[i] = this._delete_ids(tmp[i]); | |
8092 } | |
8093 return tmp; | |
8094 } | |
8095 delete tmp.id; | |
8096 if(tmp.li_attr && tmp.li_attr.id) { | |
8097 delete tmp.li_attr.id; | |
8098 } | |
8099 if(tmp.a_attr && tmp.a_attr.id) { | |
8100 delete tmp.a_attr.id; | |
8101 } | |
8102 if(tmp.children && $.isArray(tmp.children)) { | |
8103 tmp.children = this._delete_ids(tmp.children); | |
8104 } | |
8105 return tmp; | |
8106 }; | |
8107 this.check = function (chk, obj, par, pos, more) { | |
8108 if(parent.check.call(this, chk, obj, par, pos, more) === false) { return false; } | |
8109 obj = obj && obj.id ? obj : this.get_node(obj); | |
8110 par = par && par.id ? par : this.get_node(par); | |
8111 var m = obj && obj.id ? (more && more.origin ? more.origin : $.jstree.reference(obj.id)) : null, tmp, d, i, j; | |
8112 m = m && m._model && m._model.data ? m._model.data : null; | |
8113 switch(chk) { | |
8114 case "create_node": | |
8115 case "move_node": | |
8116 case "copy_node": | |
8117 if(chk !== 'move_node' || $.inArray(obj.id, par.children) === -1) { | |
8118 tmp = this.get_rules(par); | |
8119 if(tmp.max_children !== undefined && tmp.max_children !== -1 && tmp.max_children === par.children.length) { | |
8120 this._data.core.last_error = { 'error' : 'check', 'plugin' : 'types', 'id' : 'types_01', 'reason' : 'max_children prevents function: ' + chk, 'data' : JSON.stringify({ 'chk' : chk, 'pos' : pos, 'obj' : obj && obj.id ? obj.id : false, 'par' : par && par.id ? par.id : false }) }; | |
8121 return false; | |
8122 } | |
8123 if(tmp.valid_children !== undefined && tmp.valid_children !== -1 && $.inArray((obj.type || 'default'), tmp.valid_children) === -1) { | |
8124 this._data.core.last_error = { 'error' : 'check', 'plugin' : 'types', 'id' : 'types_02', 'reason' : 'valid_children prevents function: ' + chk, 'data' : JSON.stringify({ 'chk' : chk, 'pos' : pos, 'obj' : obj && obj.id ? obj.id : false, 'par' : par && par.id ? par.id : false }) }; | |
8125 return false; | |
8126 } | |
8127 if(m && obj.children_d && obj.parents) { | |
8128 d = 0; | |
8129 for(i = 0, j = obj.children_d.length; i < j; i++) { | |
8130 d = Math.max(d, m[obj.children_d[i]].parents.length); | |
8131 } | |
8132 d = d - obj.parents.length + 1; | |
8133 } | |
8134 if(d <= 0 || d === undefined) { d = 1; } | |
8135 do { | |
8136 if(tmp.max_depth !== undefined && tmp.max_depth !== -1 && tmp.max_depth < d) { | |
8137 this._data.core.last_error = { 'error' : 'check', 'plugin' : 'types', 'id' : 'types_03', 'reason' : 'max_depth prevents function: ' + chk, 'data' : JSON.stringify({ 'chk' : chk, 'pos' : pos, 'obj' : obj && obj.id ? obj.id : false, 'par' : par && par.id ? par.id : false }) }; | |
8138 return false; | |
8139 } | |
8140 par = this.get_node(par.parent); | |
8141 tmp = this.get_rules(par); | |
8142 d++; | |
8143 } while(par); | |
8144 } | |
8145 break; | |
8146 } | |
8147 return true; | |
8148 }; | |
8149 /** | |
8150 * used to retrieve the type settings object for a node | |
8151 * @name get_rules(obj) | |
8152 * @param {mixed} obj the node to find the rules for | |
8153 * @return {Object} | |
8154 * @plugin types | |
8155 */ | |
8156 this.get_rules = function (obj) { | |
8157 obj = this.get_node(obj); | |
8158 if(!obj) { return false; } | |
8159 var tmp = this.get_type(obj, true); | |
8160 if(tmp.max_depth === undefined) { tmp.max_depth = -1; } | |
8161 if(tmp.max_children === undefined) { tmp.max_children = -1; } | |
8162 if(tmp.valid_children === undefined) { tmp.valid_children = -1; } | |
8163 return tmp; | |
8164 }; | |
8165 /** | |
8166 * used to retrieve the type string or settings object for a node | |
8167 * @name get_type(obj [, rules]) | |
8168 * @param {mixed} obj the node to find the rules for | |
8169 * @param {Boolean} rules if set to `true` instead of a string the settings object will be returned | |
8170 * @return {String|Object} | |
8171 * @plugin types | |
8172 */ | |
8173 this.get_type = function (obj, rules) { | |
8174 obj = this.get_node(obj); | |
8175 return (!obj) ? false : ( rules ? $.extend({ 'type' : obj.type }, this.settings.types[obj.type]) : obj.type); | |
8176 }; | |
8177 /** | |
8178 * used to change a node's type | |
8179 * @name set_type(obj, type) | |
8180 * @param {mixed} obj the node to change | |
8181 * @param {String} type the new type | |
8182 * @plugin types | |
8183 */ | |
8184 this.set_type = function (obj, type) { | |
8185 var m = this._model.data, t, t1, t2, old_type, old_icon, k, d, a; | |
8186 if($.isArray(obj)) { | |
8187 obj = obj.slice(); | |
8188 for(t1 = 0, t2 = obj.length; t1 < t2; t1++) { | |
8189 this.set_type(obj[t1], type); | |
8190 } | |
8191 return true; | |
8192 } | |
8193 t = this.settings.types; | |
8194 obj = this.get_node(obj); | |
8195 if(!t[type] || !obj) { return false; } | |
8196 d = this.get_node(obj, true); | |
8197 if (d && d.length) { | |
8198 a = d.children('.jstree-anchor'); | |
8199 } | |
8200 old_type = obj.type; | |
8201 old_icon = this.get_icon(obj); | |
8202 obj.type = type; | |
8203 if(old_icon === true || !t[old_type] || (t[old_type].icon !== undefined && old_icon === t[old_type].icon)) { | |
8204 this.set_icon(obj, t[type].icon !== undefined ? t[type].icon : true); | |
8205 } | |
8206 | |
8207 // remove old type props | |
8208 if(t[old_type] && t[old_type].li_attr !== undefined && typeof t[old_type].li_attr === 'object') { | |
8209 for (k in t[old_type].li_attr) { | |
8210 if (t[old_type].li_attr.hasOwnProperty(k)) { | |
8211 if (k === 'id') { | |
8212 continue; | |
8213 } | |
8214 else if (k === 'class') { | |
8215 m[obj.id].li_attr['class'] = (m[obj.id].li_attr['class'] || '').replace(t[old_type].li_attr[k], ''); | |
8216 if (d) { d.removeClass(t[old_type].li_attr[k]); } | |
8217 } | |
8218 else if (m[obj.id].li_attr[k] === t[old_type].li_attr[k]) { | |
8219 m[obj.id].li_attr[k] = null; | |
8220 if (d) { d.removeAttr(k); } | |
8221 } | |
8222 } | |
8223 } | |
8224 } | |
8225 if(t[old_type] && t[old_type].a_attr !== undefined && typeof t[old_type].a_attr === 'object') { | |
8226 for (k in t[old_type].a_attr) { | |
8227 if (t[old_type].a_attr.hasOwnProperty(k)) { | |
8228 if (k === 'id') { | |
8229 continue; | |
8230 } | |
8231 else if (k === 'class') { | |
8232 m[obj.id].a_attr['class'] = (m[obj.id].a_attr['class'] || '').replace(t[old_type].a_attr[k], ''); | |
8233 if (a) { a.removeClass(t[old_type].a_attr[k]); } | |
8234 } | |
8235 else if (m[obj.id].a_attr[k] === t[old_type].a_attr[k]) { | |
8236 if (k === 'href') { | |
8237 m[obj.id].a_attr[k] = '#'; | |
8238 if (a) { a.attr('href', '#'); } | |
8239 } | |
8240 else { | |
8241 delete m[obj.id].a_attr[k]; | |
8242 if (a) { a.removeAttr(k); } | |
8243 } | |
8244 } | |
8245 } | |
8246 } | |
8247 } | |
8248 | |
8249 // add new props | |
8250 if(t[type].li_attr !== undefined && typeof t[type].li_attr === 'object') { | |
8251 for (k in t[type].li_attr) { | |
8252 if (t[type].li_attr.hasOwnProperty(k)) { | |
8253 if (k === 'id') { | |
8254 continue; | |
8255 } | |
8256 else if (m[obj.id].li_attr[k] === undefined) { | |
8257 m[obj.id].li_attr[k] = t[type].li_attr[k]; | |
8258 if (d) { | |
8259 if (k === 'class') { | |
8260 d.addClass(t[type].li_attr[k]); | |
8261 } | |
8262 else { | |
8263 d.attr(k, t[type].li_attr[k]); | |
8264 } | |
8265 } | |
8266 } | |
8267 else if (k === 'class') { | |
8268 m[obj.id].li_attr['class'] = t[type].li_attr[k] + ' ' + m[obj.id].li_attr['class']; | |
8269 if (d) { d.addClass(t[type].li_attr[k]); } | |
8270 } | |
8271 } | |
8272 } | |
8273 } | |
8274 if(t[type].a_attr !== undefined && typeof t[type].a_attr === 'object') { | |
8275 for (k in t[type].a_attr) { | |
8276 if (t[type].a_attr.hasOwnProperty(k)) { | |
8277 if (k === 'id') { | |
8278 continue; | |
8279 } | |
8280 else if (m[obj.id].a_attr[k] === undefined) { | |
8281 m[obj.id].a_attr[k] = t[type].a_attr[k]; | |
8282 if (a) { | |
8283 if (k === 'class') { | |
8284 a.addClass(t[type].a_attr[k]); | |
8285 } | |
8286 else { | |
8287 a.attr(k, t[type].a_attr[k]); | |
8288 } | |
8289 } | |
8290 } | |
8291 else if (k === 'href' && m[obj.id].a_attr[k] === '#') { | |
8292 m[obj.id].a_attr['href'] = t[type].a_attr['href']; | |
8293 if (a) { a.attr('href', t[type].a_attr['href']); } | |
8294 } | |
8295 else if (k === 'class') { | |
8296 m[obj.id].a_attr['class'] = t[type].a_attr['class'] + ' ' + m[obj.id].a_attr['class']; | |
8297 if (a) { a.addClass(t[type].a_attr[k]); } | |
8298 } | |
8299 } | |
8300 } | |
8301 } | |
8302 | |
8303 return true; | |
8304 }; | |
8305 }; | |
8306 // include the types plugin by default | |
8307 // $.jstree.defaults.plugins.push("types"); | |
8308 | |
8309 | |
8310 /** | |
8311 * ### Unique plugin | |
8312 * | |
8313 * Enforces that no nodes with the same name can coexist as siblings. | |
8314 */ | |
8315 | |
8316 /** | |
8317 * stores all defaults for the unique plugin | |
8318 * @name $.jstree.defaults.unique | |
8319 * @plugin unique | |
8320 */ | |
8321 $.jstree.defaults.unique = { | |
8322 /** | |
8323 * Indicates if the comparison should be case sensitive. Default is `false`. | |
8324 * @name $.jstree.defaults.unique.case_sensitive | |
8325 * @plugin unique | |
8326 */ | |
8327 case_sensitive : false, | |
8328 /** | |
8329 * Indicates if white space should be trimmed before the comparison. Default is `false`. | |
8330 * @name $.jstree.defaults.unique.trim_whitespace | |
8331 * @plugin unique | |
8332 */ | |
8333 trim_whitespace : false, | |
8334 /** | |
8335 * A callback executed in the instance's scope when a new node is created and the name is already taken, the two arguments are the conflicting name and the counter. The default will produce results like `New node (2)`. | |
8336 * @name $.jstree.defaults.unique.duplicate | |
8337 * @plugin unique | |
8338 */ | |
8339 duplicate : function (name, counter) { | |
8340 return name + ' (' + counter + ')'; | |
8341 } | |
8342 }; | |
8343 | |
8344 $.jstree.plugins.unique = function (options, parent) { | |
8345 this.check = function (chk, obj, par, pos, more) { | |
8346 if(parent.check.call(this, chk, obj, par, pos, more) === false) { return false; } | |
8347 obj = obj && obj.id ? obj : this.get_node(obj); | |
8348 par = par && par.id ? par : this.get_node(par); | |
8349 if(!par || !par.children) { return true; } | |
8350 var n = chk === "rename_node" ? pos : obj.text, | |
8351 c = [], | |
8352 s = this.settings.unique.case_sensitive, | |
8353 w = this.settings.unique.trim_whitespace, | |
8354 m = this._model.data, i, j, t; | |
8355 for(i = 0, j = par.children.length; i < j; i++) { | |
8356 t = m[par.children[i]].text; | |
8357 if (!s) { | |
8358 t = t.toLowerCase(); | |
8359 } | |
8360 if (w) { | |
8361 t = t.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, ''); | |
8362 } | |
8363 c.push(t); | |
8364 } | |
8365 if(!s) { n = n.toLowerCase(); } | |
8366 if (w) { n = n.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, ''); } | |
8367 switch(chk) { | |
8368 case "delete_node": | |
8369 return true; | |
8370 case "rename_node": | |
8371 t = obj.text || ''; | |
8372 if (!s) { | |
8373 t = t.toLowerCase(); | |
8374 } | |
8375 if (w) { | |
8376 t = t.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, ''); | |
8377 } | |
8378 i = ($.inArray(n, c) === -1 || (obj.text && t === n)); | |
8379 if(!i) { | |
8380 this._data.core.last_error = { 'error' : 'check', 'plugin' : 'unique', 'id' : 'unique_01', 'reason' : 'Child with name ' + n + ' already exists. Preventing: ' + chk, 'data' : JSON.stringify({ 'chk' : chk, 'pos' : pos, 'obj' : obj && obj.id ? obj.id : false, 'par' : par && par.id ? par.id : false }) }; | |
8381 } | |
8382 return i; | |
8383 case "create_node": | |
8384 i = ($.inArray(n, c) === -1); | |
8385 if(!i) { | |
8386 this._data.core.last_error = { 'error' : 'check', 'plugin' : 'unique', 'id' : 'unique_04', 'reason' : 'Child with name ' + n + ' already exists. Preventing: ' + chk, 'data' : JSON.stringify({ 'chk' : chk, 'pos' : pos, 'obj' : obj && obj.id ? obj.id : false, 'par' : par && par.id ? par.id : false }) }; | |
8387 } | |
8388 return i; | |
8389 case "copy_node": | |
8390 i = ($.inArray(n, c) === -1); | |
8391 if(!i) { | |
8392 this._data.core.last_error = { 'error' : 'check', 'plugin' : 'unique', 'id' : 'unique_02', 'reason' : 'Child with name ' + n + ' already exists. Preventing: ' + chk, 'data' : JSON.stringify({ 'chk' : chk, 'pos' : pos, 'obj' : obj && obj.id ? obj.id : false, 'par' : par && par.id ? par.id : false }) }; | |
8393 } | |
8394 return i; | |
8395 case "move_node": | |
8396 i = ( (obj.parent === par.id && (!more || !more.is_multi)) || $.inArray(n, c) === -1); | |
8397 if(!i) { | |
8398 this._data.core.last_error = { 'error' : 'check', 'plugin' : 'unique', 'id' : 'unique_03', 'reason' : 'Child with name ' + n + ' already exists. Preventing: ' + chk, 'data' : JSON.stringify({ 'chk' : chk, 'pos' : pos, 'obj' : obj && obj.id ? obj.id : false, 'par' : par && par.id ? par.id : false }) }; | |
8399 } | |
8400 return i; | |
8401 } | |
8402 return true; | |
8403 }; | |
8404 this.create_node = function (par, node, pos, callback, is_loaded) { | |
8405 if(!node || node.text === undefined) { | |
8406 if(par === null) { | |
8407 par = $.jstree.root; | |
8408 } | |
8409 par = this.get_node(par); | |
8410 if(!par) { | |
8411 return parent.create_node.call(this, par, node, pos, callback, is_loaded); | |
8412 } | |
8413 pos = pos === undefined ? "last" : pos; | |
8414 if(!pos.toString().match(/^(before|after)$/) && !is_loaded && !this.is_loaded(par)) { | |
8415 return parent.create_node.call(this, par, node, pos, callback, is_loaded); | |
8416 } | |
8417 if(!node) { node = {}; } | |
8418 var tmp, n, dpc, i, j, m = this._model.data, s = this.settings.unique.case_sensitive, w = this.settings.unique.trim_whitespace, cb = this.settings.unique.duplicate, t; | |
8419 n = tmp = this.get_string('New node'); | |
8420 dpc = []; | |
8421 for(i = 0, j = par.children.length; i < j; i++) { | |
8422 t = m[par.children[i]].text; | |
8423 if (!s) { | |
8424 t = t.toLowerCase(); | |
8425 } | |
8426 if (w) { | |
8427 t = t.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, ''); | |
8428 } | |
8429 dpc.push(t); | |
8430 } | |
8431 i = 1; | |
8432 t = n; | |
8433 if (!s) { | |
8434 t = t.toLowerCase(); | |
8435 } | |
8436 if (w) { | |
8437 t = t.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, ''); | |
8438 } | |
8439 while($.inArray(t, dpc) !== -1) { | |
8440 n = cb.call(this, tmp, (++i)).toString(); | |
8441 t = n; | |
8442 if (!s) { | |
8443 t = t.toLowerCase(); | |
8444 } | |
8445 if (w) { | |
8446 t = t.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, ''); | |
8447 } | |
8448 } | |
8449 node.text = n; | |
8450 } | |
8451 return parent.create_node.call(this, par, node, pos, callback, is_loaded); | |
8452 }; | |
8453 }; | |
8454 | |
8455 // include the unique plugin by default | |
8456 // $.jstree.defaults.plugins.push("unique"); | |
8457 | |
8458 | |
8459 /** | |
8460 * ### Wholerow plugin | |
8461 * | |
8462 * Makes each node appear block level. Making selection easier. May cause slow down for large trees in old browsers. | |
8463 */ | |
8464 | |
8465 var div = document.createElement('DIV'); | |
8466 div.setAttribute('unselectable','on'); | |
8467 div.setAttribute('role','presentation'); | |
8468 div.className = 'jstree-wholerow'; | |
8469 div.innerHTML = ' '; | |
8470 $.jstree.plugins.wholerow = function (options, parent) { | |
8471 this.bind = function () { | |
8472 parent.bind.call(this); | |
8473 | |
8474 this.element | |
8475 .on('ready.jstree set_state.jstree', $.proxy(function () { | |
8476 this.hide_dots(); | |
8477 }, this)) | |
8478 .on("init.jstree loading.jstree ready.jstree", $.proxy(function () { | |
8479 //div.style.height = this._data.core.li_height + 'px'; | |
8480 this.get_container_ul().addClass('jstree-wholerow-ul'); | |
8481 }, this)) | |
8482 .on("deselect_all.jstree", $.proxy(function (e, data) { | |
8483 this.element.find('.jstree-wholerow-clicked').removeClass('jstree-wholerow-clicked'); | |
8484 }, this)) | |
8485 .on("changed.jstree", $.proxy(function (e, data) { | |
8486 this.element.find('.jstree-wholerow-clicked').removeClass('jstree-wholerow-clicked'); | |
8487 var tmp = false, i, j; | |
8488 for(i = 0, j = data.selected.length; i < j; i++) { | |
8489 tmp = this.get_node(data.selected[i], true); | |
8490 if(tmp && tmp.length) { | |
8491 tmp.children('.jstree-wholerow').addClass('jstree-wholerow-clicked'); | |
8492 } | |
8493 } | |
8494 }, this)) | |
8495 .on("open_node.jstree", $.proxy(function (e, data) { | |
8496 this.get_node(data.node, true).find('.jstree-clicked').parent().children('.jstree-wholerow').addClass('jstree-wholerow-clicked'); | |
8497 }, this)) | |
8498 .on("hover_node.jstree dehover_node.jstree", $.proxy(function (e, data) { | |
8499 if(e.type === "hover_node" && this.is_disabled(data.node)) { return; } | |
8500 this.get_node(data.node, true).children('.jstree-wholerow')[e.type === "hover_node"?"addClass":"removeClass"]('jstree-wholerow-hovered'); | |
8501 }, this)) | |
8502 .on("contextmenu.jstree", ".jstree-wholerow", $.proxy(function (e) { | |
8503 if (this._data.contextmenu) { | |
8504 e.preventDefault(); | |
8505 var tmp = $.Event('contextmenu', { metaKey : e.metaKey, ctrlKey : e.ctrlKey, altKey : e.altKey, shiftKey : e.shiftKey, pageX : e.pageX, pageY : e.pageY }); | |
8506 $(e.currentTarget).closest(".jstree-node").children(".jstree-anchor").first().trigger(tmp); | |
8507 } | |
8508 }, this)) | |
8509 /*! | |
8510 .on("mousedown.jstree touchstart.jstree", ".jstree-wholerow", function (e) { | |
8511 if(e.target === e.currentTarget) { | |
8512 var a = $(e.currentTarget).closest(".jstree-node").children(".jstree-anchor"); | |
8513 e.target = a[0]; | |
8514 a.trigger(e); | |
8515 } | |
8516 }) | |
8517 */ | |
8518 .on("click.jstree", ".jstree-wholerow", function (e) { | |
8519 e.stopImmediatePropagation(); | |
8520 var tmp = $.Event('click', { metaKey : e.metaKey, ctrlKey : e.ctrlKey, altKey : e.altKey, shiftKey : e.shiftKey }); | |
8521 $(e.currentTarget).closest(".jstree-node").children(".jstree-anchor").first().trigger(tmp).focus(); | |
8522 }) | |
8523 .on("dblclick.jstree", ".jstree-wholerow", function (e) { | |
8524 e.stopImmediatePropagation(); | |
8525 var tmp = $.Event('dblclick', { metaKey : e.metaKey, ctrlKey : e.ctrlKey, altKey : e.altKey, shiftKey : e.shiftKey }); | |
8526 $(e.currentTarget).closest(".jstree-node").children(".jstree-anchor").first().trigger(tmp).focus(); | |
8527 }) | |
8528 .on("click.jstree", ".jstree-leaf > .jstree-ocl", $.proxy(function (e) { | |
8529 e.stopImmediatePropagation(); | |
8530 var tmp = $.Event('click', { metaKey : e.metaKey, ctrlKey : e.ctrlKey, altKey : e.altKey, shiftKey : e.shiftKey }); | |
8531 $(e.currentTarget).closest(".jstree-node").children(".jstree-anchor").first().trigger(tmp).focus(); | |
8532 }, this)) | |
8533 .on("mouseover.jstree", ".jstree-wholerow, .jstree-icon", $.proxy(function (e) { | |
8534 e.stopImmediatePropagation(); | |
8535 if(!this.is_disabled(e.currentTarget)) { | |
8536 this.hover_node(e.currentTarget); | |
8537 } | |
8538 return false; | |
8539 }, this)) | |
8540 .on("mouseleave.jstree", ".jstree-node", $.proxy(function (e) { | |
8541 this.dehover_node(e.currentTarget); | |
8542 }, this)); | |
8543 }; | |
8544 this.teardown = function () { | |
8545 if(this.settings.wholerow) { | |
8546 this.element.find(".jstree-wholerow").remove(); | |
8547 } | |
8548 parent.teardown.call(this); | |
8549 }; | |
8550 this.redraw_node = function(obj, deep, callback, force_render) { | |
8551 obj = parent.redraw_node.apply(this, arguments); | |
8552 if(obj) { | |
8553 var tmp = div.cloneNode(true); | |
8554 //tmp.style.height = this._data.core.li_height + 'px'; | |
8555 if($.inArray(obj.id, this._data.core.selected) !== -1) { tmp.className += ' jstree-wholerow-clicked'; } | |
8556 if(this._data.core.focused && this._data.core.focused === obj.id) { tmp.className += ' jstree-wholerow-hovered'; } | |
8557 obj.insertBefore(tmp, obj.childNodes[0]); | |
8558 } | |
8559 return obj; | |
8560 }; | |
8561 }; | |
8562 // include the wholerow plugin by default | |
8563 // $.jstree.defaults.plugins.push("wholerow"); | |
8564 if(document.registerElement && Object && Object.create) { | |
8565 var proto = Object.create(HTMLElement.prototype); | |
8566 proto.createdCallback = function () { | |
8567 var c = { core : {}, plugins : [] }, i; | |
8568 for(i in $.jstree.plugins) { | |
8569 if($.jstree.plugins.hasOwnProperty(i) && this.attributes[i]) { | |
8570 c.plugins.push(i); | |
8571 if(this.getAttribute(i) && JSON.parse(this.getAttribute(i))) { | |
8572 c[i] = JSON.parse(this.getAttribute(i)); | |
8573 } | |
8574 } | |
8575 } | |
8576 for(i in $.jstree.defaults.core) { | |
8577 if($.jstree.defaults.core.hasOwnProperty(i) && this.attributes[i]) { | |
8578 c.core[i] = JSON.parse(this.getAttribute(i)) || this.getAttribute(i); | |
8579 } | |
8580 } | |
8581 $(this).jstree(c); | |
8582 }; | |
8583 // proto.attributeChangedCallback = function (name, previous, value) { }; | |
8584 try { | |
8585 document.registerElement("vakata-jstree", { prototype: proto }); | |
8586 } catch(ignore) { } | |
8587 } | |
8588 | |
8589 })); |