| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336 | define("dijit/tree/_dndSelector", [	"dojo/_base/array", // array.filter array.forEach array.map	"dojo/_base/connect", // connect.isCopyKey	"dojo/_base/declare", // declare	"dojo/_base/lang", // lang.hitch	"dojo/mouse", // mouse.isLeft	"dojo/on",	"dojo/touch",	"dojo/_base/window", // win.global	"./_dndContainer"], function(array, connect, declare, lang, mouse, on, touch, win, _dndContainer){	// module:	//		dijit/tree/_dndSelector	// summary:	//		This is a base class for `dijit.tree.dndSource` , and isn't meant to be used directly.	//		It's based on `dojo.dnd.Selector`.	return declare("dijit.tree._dndSelector", _dndContainer, {		// summary:		//		This is a base class for `dijit.tree.dndSource` , and isn't meant to be used directly.		//		It's based on `dojo.dnd.Selector`.		// tags:		//		protected		/*=====		// selection: Hash<String, DomNode>		//		(id, DomNode) map for every TreeNode that's currently selected.		//		The DOMNode is the TreeNode.rowNode.		selection: {},		=====*/		constructor: function(){			// summary:			//		Initialization			// tags:			//		private			this.selection={};			this.anchor = null;			this.tree.domNode.setAttribute("aria-multiselect", !this.singular);			this.events.push(				on(this.tree.domNode, touch.press, lang.hitch(this,"onMouseDown")),				on(this.tree.domNode, touch.release, lang.hitch(this,"onMouseUp")),				on(this.tree.domNode, touch.move, lang.hitch(this,"onMouseMove"))			);		},		//	singular: Boolean		//		Allows selection of only one element, if true.		//		Tree hasn't been tested in singular=true mode, unclear if it works.		singular: false,		// methods		getSelectedTreeNodes: function(){			// summary:			//		Returns a list of selected node(s).			//		Used by dndSource on the start of a drag.			// tags:			//		protected			var nodes=[], sel = this.selection;			for(var i in sel){				nodes.push(sel[i]);			}			return nodes;		},		selectNone: function(){			// summary:			//		Unselects all items			// tags:			//		private			this.setSelection([]);			return this;	// self		},		destroy: function(){			// summary:			//		Prepares the object to be garbage-collected			this.inherited(arguments);			this.selection = this.anchor = null;		},		addTreeNode: function(/*dijit._TreeNode*/node, /*Boolean?*/isAnchor){			// summary:			//		add node to current selection			// node: Node			//		node to add			// isAnchor: Boolean			//		Whether the node should become anchor.			this.setSelection(this.getSelectedTreeNodes().concat( [node] ));			if(isAnchor){ this.anchor = node; }			return node;		},		removeTreeNode: function(/*dijit._TreeNode*/node){			// summary:			//		remove node from current selection			// node: Node			//		node to remove			this.setSelection(this._setDifference(this.getSelectedTreeNodes(), [node]));			return node;		},		isTreeNodeSelected: function(/*dijit._TreeNode*/node){			// summary:			//		return true if node is currently selected			// node: Node			//		the node to check whether it's in the current selection			return node.id && !!this.selection[node.id];		},		setSelection: function(/*dijit._treeNode[]*/ newSelection){			// summary:			//		set the list of selected nodes to be exactly newSelection. All changes to the			//		selection should be passed through this function, which ensures that derived			//		attributes are kept up to date. Anchor will be deleted if it has been removed			//		from the selection, but no new anchor will be added by this function.			// newSelection: Node[]			//		list of tree nodes to make selected			var oldSelection = this.getSelectedTreeNodes();			array.forEach(this._setDifference(oldSelection, newSelection), lang.hitch(this, function(node){				node.setSelected(false);				if(this.anchor == node){					delete this.anchor;				}				delete this.selection[node.id];			}));			array.forEach(this._setDifference(newSelection, oldSelection), lang.hitch(this, function(node){				node.setSelected(true);				this.selection[node.id] = node;			}));			this._updateSelectionProperties();		},		_setDifference: function(xs,ys){			// summary:			//		Returns a copy of xs which lacks any objects			//		occurring in ys. Checks for membership by			//		modifying and then reading the object, so it will			//		not properly handle sets of numbers or strings.			array.forEach(ys, function(y){ y.__exclude__ = true; });			var ret = array.filter(xs, function(x){ return !x.__exclude__; });			// clean up after ourselves.			array.forEach(ys, function(y){ delete y['__exclude__'] });			return ret;		},		_updateSelectionProperties: function(){			// summary:			//		Update the following tree properties from the current selection:			//		path[s], selectedItem[s], selectedNode[s]			var selected = this.getSelectedTreeNodes();			var paths = [], nodes = [];			array.forEach(selected, function(node){				nodes.push(node);				paths.push(node.getTreePath());			});			var items = array.map(nodes,function(node){ return node.item; });			this.tree._set("paths", paths);			this.tree._set("path", paths[0] || []);			this.tree._set("selectedNodes", nodes);			this.tree._set("selectedNode", nodes[0] || null);			this.tree._set("selectedItems", items);			this.tree._set("selectedItem", items[0] || null);		},		// mouse events		onMouseDown: function(e){			// summary:			//		Event processor for onmousedown/ontouchstart			// e: Event			//		onmousedown/ontouchstart event			// tags:			//		protected			// ignore click on expando node			if(!this.current || this.tree.isExpandoNode(e.target, this.current)){ return; }			if(e.type == "mousedown" && mouse.isLeft(e)){				// Prevent text selection while dragging on desktop, see #16328.   But don't call preventDefault()				// for mobile because it will break things completely, see #15838.  Also, don't preventDefault() on				// MSPointerDown or pointerdown events, because that stops the mousedown event from being generated,				// see #17709.				// TODO: remove this completely in 2.0.  It shouldn't be needed since dojo/dnd/Manager already				// calls preventDefault() for the "selectstart" event.  It can also be achieved via CSS:				// http://stackoverflow.com/questions/826782/css-rule-to-disable-text-selection-highlighting				e.preventDefault();			}else if(e.type != "touchstart"){				// Ignore right click				return;			}			var treeNode = this.current,			  copy = connect.isCopyKey(e), id = treeNode.id;			// if shift key is not pressed, and the node is already in the selection,			// delay deselection until onmouseup so in the case of DND, deselection			// will be canceled by onmousemove.			if(!this.singular && !e.shiftKey && this.selection[id]){				this._doDeselect = true;				return;			}else{				this._doDeselect = false;			}			this.userSelect(treeNode, copy, e.shiftKey);		},		onMouseUp: function(e){			// summary:			//		Event processor for onmouseup/ontouchend			// e: Event			//		onmouseup/ontouchend event			// tags:			//		protected			// _doDeselect is the flag to indicate that the user wants to either ctrl+click on			// a already selected item (to deselect the item), or click on a not-yet selected item			// (which should remove all current selection, and add the clicked item). This can not			// be done in onMouseDown, because the user may start a drag after mousedown. By moving			// the deselection logic here, the user can drags an already selected item.			if(!this._doDeselect){ return; }			this._doDeselect = false;			this.userSelect(this.current, connect.isCopyKey(e), e.shiftKey);		},		onMouseMove: function(/*===== e =====*/){			// summary:			//		event processor for onmousemove/ontouchmove			// e: Event			//		onmousemove/ontouchmove event			this._doDeselect = false;		},		_compareNodes: function(n1, n2){			if(n1 === n2){				return 0;			}			if('sourceIndex' in document.documentElement){ //IE				//TODO: does not yet work if n1 and/or n2 is a text node				return n1.sourceIndex - n2.sourceIndex;			}else if('compareDocumentPosition' in document.documentElement){ //FF, Opera				return n1.compareDocumentPosition(n2) & 2 ? 1: -1;			}else if(document.createRange){ //Webkit				var r1 = doc.createRange();				r1.setStartBefore(n1);				var r2 = doc.createRange();				r2.setStartBefore(n2);				return r1.compareBoundaryPoints(r1.END_TO_END, r2);			}else{				throw Error("dijit.tree._compareNodes don't know how to compare two different nodes in this browser");			}		},		userSelect: function(node, multi, range){			// summary:			//		Add or remove the given node from selection, responding			//		to a user action such as a click or keypress.			// multi: Boolean			//		Indicates whether this is meant to be a multi-select action (e.g. ctrl-click)			// range: Boolean			//		Indicates whether this is meant to be a ranged action (e.g. shift-click)			// tags:			//		protected			if(this.singular){				if(this.anchor == node && multi){					this.selectNone();				}else{					this.setSelection([node]);					this.anchor = node;				}			}else{				if(range && this.anchor){					var cr = this._compareNodes(this.anchor.rowNode, node.rowNode),					begin, end, anchor = this.anchor;					if(cr < 0){ //current is after anchor						begin = anchor;						end = node;					}else{ //current is before anchor						begin = node;						end = anchor;					}					var nodes = [];					//add everything betweeen begin and end inclusively					while(begin != end){						nodes.push(begin);						begin = this.tree._getNextNode(begin);					}					nodes.push(end);					this.setSelection(nodes);				}else{					if( this.selection[ node.id ] && multi ){						this.removeTreeNode( node );					}else if(multi){						this.addTreeNode(node, true);					}else{						this.setSelection([node]);						this.anchor = node;					}				}			}		},		getItem: function(/*String*/ key){			// summary:			//		Returns the dojo.dnd.Item (representing a dragged node) by it's key (id).			//		Called by dojo.dnd.Source.checkAcceptance().			// tags:			//		protected			var widget = this.selection[key];			return {				data: widget,				type: ["treeNode"]			}; // dojo.dnd.Item		},		forInSelectedItems: function(/*Function*/ f, /*Object?*/ o){			// summary:			//		Iterates over selected items;			//		see `dojo.dnd.Container.forInItems()` for details			o = o || win.global;			for(var id in this.selection){				// console.log("selected item id: " + id);				f.call(o, this.getItem(id), id, this);			}		}	});});
 |