diff options
author | mensonge <mensonge@b3834d28-1941-0410-a4f8-b48e95affb8f> | 2008-11-13 09:49:11 +0000 |
---|---|---|
committer | mensonge <mensonge@b3834d28-1941-0410-a4f8-b48e95affb8f> | 2008-11-13 09:49:11 +0000 |
commit | e44a7e37b6c7b5961adaffc62b9042b8d442938e (patch) | |
tree | 95b67c356e93163467db2451f2b8cce84ed5d582 /includes/js/dojo/dnd/Container.js | |
parent | a62b9742ee5e28bcec6872d88f50f25b820914f6 (diff) | |
download | semanticscuttle-e44a7e37b6c7b5961adaffc62b9042b8d442938e.tar.gz semanticscuttle-e44a7e37b6c7b5961adaffc62b9042b8d442938e.tar.bz2 |
New feature: basic Ajax suggestion for tags and implementation of Dojo toolkit
git-svn-id: https://semanticscuttle.svn.sourceforge.net/svnroot/semanticscuttle/trunk@151 b3834d28-1941-0410-a4f8-b48e95affb8f
Diffstat (limited to 'includes/js/dojo/dnd/Container.js')
-rw-r--r-- | includes/js/dojo/dnd/Container.js | 311 |
1 files changed, 311 insertions, 0 deletions
diff --git a/includes/js/dojo/dnd/Container.js b/includes/js/dojo/dnd/Container.js new file mode 100644 index 0000000..92b3211 --- /dev/null +++ b/includes/js/dojo/dnd/Container.js @@ -0,0 +1,311 @@ +if(!dojo._hasResource["dojo.dnd.Container"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code. +dojo._hasResource["dojo.dnd.Container"] = true; +dojo.provide("dojo.dnd.Container"); + +dojo.require("dojo.dnd.common"); +dojo.require("dojo.parser"); + +/* + Container states: + "" - normal state + "Over" - mouse over a container + Container item states: + "" - normal state + "Over" - mouse over a container item +*/ + +dojo.declare("dojo.dnd.Container", null, { + // summary: a Container object, which knows when mouse hovers over it, + // and know over which element it hovers + + // object attributes (for markup) + skipForm: false, + + constructor: function(node, params){ + // summary: a constructor of the Container + // node: Node: node or node's id to build the container on + // params: Object: a dict of parameters, recognized parameters are: + // creator: Function: a creator function, which takes a data item, and returns an object like that: + // {node: newNode, data: usedData, type: arrayOfStrings} + // skipForm: Boolean: don't start the drag operation, if clicked on form elements + // _skipStartup: Boolean: skip startup(), which collects children, for deferred initialization + // (this is used in the markup mode) + this.node = dojo.byId(node); + if(!params){ params = {}; } + this.creator = params.creator || null; + this.skipForm = params.skipForm; + this.defaultCreator = dojo.dnd._defaultCreator(this.node); + + // class-specific variables + this.map = {}; + this.current = null; + + // states + this.containerState = ""; + dojo.addClass(this.node, "dojoDndContainer"); + + // mark up children + if(!(params && params._skipStartup)){ + this.startup(); + } + + // set up events + this.events = [ + dojo.connect(this.node, "onmouseover", this, "onMouseOver"), + dojo.connect(this.node, "onmouseout", this, "onMouseOut"), + // cancel text selection and text dragging + dojo.connect(this.node, "ondragstart", this, "onSelectStart"), + dojo.connect(this.node, "onselectstart", this, "onSelectStart") + ]; + }, + + // object attributes (for markup) + creator: function(){}, // creator function, dummy at the moment + + // abstract access to the map + getItem: function(/*String*/ key){ + // summary: returns a data item by its key (id) + return this.map[key]; // Object + }, + setItem: function(/*String*/ key, /*Object*/ data){ + // summary: associates a data item with its key (id) + this.map[key] = data; + }, + delItem: function(/*String*/ key){ + // summary: removes a data item from the map by its key (id) + delete this.map[key]; + }, + forInItems: function(/*Function*/ f, /*Object?*/ o){ + // summary: iterates over a data map skipping members, which + // are present in the empty object (IE and/or 3rd-party libraries). + o = o || dojo.global; + var m = this.map, e = dojo.dnd._empty; + for(var i in this.map){ + if(i in e){ continue; } + f.call(o, m[i], i, m); + } + }, + clearItems: function(){ + // summary: removes all data items from the map + this.map = {}; + }, + + // methods + getAllNodes: function(){ + // summary: returns a list (an array) of all valid child nodes + return dojo.query("> .dojoDndItem", this.parent); // NodeList + }, + insertNodes: function(data, before, anchor){ + // summary: inserts an array of new nodes before/after an anchor node + // data: Array: a list of data items, which should be processed by the creator function + // before: Boolean: insert before the anchor, if true, and after the anchor otherwise + // anchor: Node: the anchor node to be used as a point of insertion + if(!this.parent.firstChild){ + anchor = null; + }else if(before){ + if(!anchor){ + anchor = this.parent.firstChild; + } + }else{ + if(anchor){ + anchor = anchor.nextSibling; + } + } + if(anchor){ + for(var i = 0; i < data.length; ++i){ + var t = this._normalizedCreator(data[i]); + this.setItem(t.node.id, {data: t.data, type: t.type}); + this.parent.insertBefore(t.node, anchor); + } + }else{ + for(var i = 0; i < data.length; ++i){ + var t = this._normalizedCreator(data[i]); + this.setItem(t.node.id, {data: t.data, type: t.type}); + this.parent.appendChild(t.node); + } + } + return this; // self + }, + destroy: function(){ + // summary: prepares the object to be garbage-collected + dojo.forEach(this.events, dojo.disconnect); + this.clearItems(); + this.node = this.parent = this.current; + }, + + // markup methods + markupFactory: function(params, node){ + params._skipStartup = true; + return new dojo.dnd.Container(node, params); + }, + startup: function(){ + // summary: collects valid child items and populate the map + + // set up the real parent node + this.parent = this.node; + if(this.parent.tagName.toLowerCase() == "table"){ + var c = this.parent.getElementsByTagName("tbody"); + if(c && c.length){ this.parent = c[0]; } + } + + // process specially marked children + this.getAllNodes().forEach(function(node){ + if(!node.id){ node.id = dojo.dnd.getUniqueId(); } + var type = node.getAttribute("dndType"), + data = node.getAttribute("dndData"); + this.setItem(node.id, { + data: data ? data : node.innerHTML, + type: type ? type.split(/\s*,\s*/) : ["text"] + }); + }, this); + }, + + // mouse events + onMouseOver: function(e){ + // summary: event processor for onmouseover + // e: Event: mouse event + var n = e.relatedTarget; + while(n){ + if(n == this.node){ break; } + try{ + n = n.parentNode; + }catch(x){ + n = null; + } + } + if(!n){ + this._changeState("Container", "Over"); + this.onOverEvent(); + } + n = this._getChildByEvent(e); + if(this.current == n){ return; } + if(this.current){ this._removeItemClass(this.current, "Over"); } + if(n){ this._addItemClass(n, "Over"); } + this.current = n; + }, + onMouseOut: function(e){ + // summary: event processor for onmouseout + // e: Event: mouse event + for(var n = e.relatedTarget; n;){ + if(n == this.node){ return; } + try{ + n = n.parentNode; + }catch(x){ + n = null; + } + } + if(this.current){ + this._removeItemClass(this.current, "Over"); + this.current = null; + } + this._changeState("Container", ""); + this.onOutEvent(); + }, + onSelectStart: function(e){ + // summary: event processor for onselectevent and ondragevent + // e: Event: mouse event + if(!this.skipForm || !dojo.dnd.isFormElement(e)){ + dojo.stopEvent(e); + } + }, + + // utilities + onOverEvent: function(){ + // summary: this function is called once, when mouse is over our container + }, + onOutEvent: function(){ + // summary: this function is called once, when mouse is out of our container + }, + _changeState: function(type, newState){ + // summary: changes a named state to new state value + // type: String: a name of the state to change + // newState: String: new state + var prefix = "dojoDnd" + type; + var state = type.toLowerCase() + "State"; + //dojo.replaceClass(this.node, prefix + newState, prefix + this[state]); + dojo.removeClass(this.node, prefix + this[state]); + dojo.addClass(this.node, prefix + newState); + this[state] = newState; + }, + _addItemClass: function(node, type){ + // summary: adds a class with prefix "dojoDndItem" + // node: Node: a node + // type: String: a variable suffix for a class name + dojo.addClass(node, "dojoDndItem" + type); + }, + _removeItemClass: function(node, type){ + // summary: removes a class with prefix "dojoDndItem" + // node: Node: a node + // type: String: a variable suffix for a class name + dojo.removeClass(node, "dojoDndItem" + type); + }, + _getChildByEvent: function(e){ + // summary: gets a child, which is under the mouse at the moment, or null + // e: Event: a mouse event + var node = e.target; + if(node){ + for(var parent = node.parentNode; parent; node = parent, parent = node.parentNode){ + if(parent == this.parent && dojo.hasClass(node, "dojoDndItem")){ return node; } + } + } + return null; + }, + _normalizedCreator: function(item, hint){ + // summary: adds all necessary data to the output of the user-supplied creator function + var t = (this.creator ? this.creator : this.defaultCreator)(item, hint); + if(!dojo.isArray(t.type)){ t.type = ["text"]; } + if(!t.node.id){ t.node.id = dojo.dnd.getUniqueId(); } + dojo.addClass(t.node, "dojoDndItem"); + return t; + } +}); + +dojo.dnd._createNode = function(tag){ + // summary: returns a function, which creates an element of given tag + // (SPAN by default) and sets its innerHTML to given text + // tag: String: a tag name or empty for SPAN + if(!tag){ return dojo.dnd._createSpan; } + return function(text){ // Function + var n = dojo.doc.createElement(tag); + n.innerHTML = text; + return n; + }; +}; + +dojo.dnd._createTrTd = function(text){ + // summary: creates a TR/TD structure with given text as an innerHTML of TD + // text: String: a text for TD + var tr = dojo.doc.createElement("tr"); + var td = dojo.doc.createElement("td"); + td.innerHTML = text; + tr.appendChild(td); + return tr; // Node +}; + +dojo.dnd._createSpan = function(text){ + // summary: creates a SPAN element with given text as its innerHTML + // text: String: a text for SPAN + var n = dojo.doc.createElement("span"); + n.innerHTML = text; + return n; // Node +}; + +// dojo.dnd._defaultCreatorNodes: Object: a dicitionary, which maps container tag names to child tag names +dojo.dnd._defaultCreatorNodes = {ul: "li", ol: "li", div: "div", p: "div"}; + +dojo.dnd._defaultCreator = function(node){ + // summary: takes a container node, and returns an appropriate creator function + // node: Node: a container node + var tag = node.tagName.toLowerCase(); + var c = tag == "table" ? dojo.dnd._createTrTd : dojo.dnd._createNode(dojo.dnd._defaultCreatorNodes[tag]); + return function(item, hint){ // Function + var isObj = dojo.isObject(item) && item; + var data = (isObj && item.data) ? item.data : item; + var type = (isObj && item.type) ? item.type : ["text"]; + var t = String(data), n = (hint == "avatar" ? dojo.dnd._createSpan : c)(t); + n.id = dojo.dnd.getUniqueId(); + return {node: n, data: data, type: type}; + }; +}; + +} |