`

扩展dtree支持checkbox

阅读更多
最近仔细看了一下dtree的代码,它不支持checkbox很不爽,就自己动手加上了。修改后的代码与原来的dtree兼容,可以直接替换原工程中的dtree。详细示例请看附件。
/*--------------------------------------------------|
| dTree 2.05 | www.destroydrop.com/javascript/tree/ |
|---------------------------------------------------|
| Copyright (c) 2002-2003 Geir Landr?               |
|                                                   |
| This script can be used freely as long as all     |
| copyright messages are intact.                    |
|                                                   |
| Updated: 17.04.2003                               |
|--------------------------------------------------*/
//modified by wallimn. 20070831
//to support checkbox object
//updated 2009-02-13
//welcome to visit my blog: http://blog.csdn.net/wallimn
//please keep these message.
// Node object
function Node(id, pid, name, url, title, target, icon, iconOpen, open) {
	this.id = id;
	this.pid = pid;
	this.name = name;
	this.url = url;
	this.title = title;
	this.target = target;
	this.icon = icon;
	this.iconOpen = iconOpen;
	this._io = open || false;
	this._is = false;
	this._ls = false;
	this._hc = false;
	this._ai = 0;
	this._p;
};
// Tree object
function dTree(objName,iconPath) {
	this.config = {
		target					: null,
		folderLinks			: true,
		useSelection		: true,
		useCookies			: true,
		useLines				: true,
		useIcons				: true,
		useStatusText		: false,
		closeSameLevel	: false,
		inOrder					: false,
		check:false		//added by wallimn 2009-02-05, to control whether checkbox is showed
	};
	//modified by wallimn 2009-02-05, in order to make client can modify path of icons;
	if(!iconPath)iconPath="img/";//set the default value, be compatible with dtree
	this.icon = {
			root		: iconPath+'base.gif',
			folder		: iconPath+'folder.gif',
			folderOpen	: iconPath+'folderopen.gif',
			node		: iconPath+'page.gif',
			empty		: iconPath+'empty.gif',
			line		: iconPath+'line.gif',
			join		: iconPath+'join.gif',
			joinBottom	: iconPath+'joinbottom.gif',
			plus		: iconPath+'plus.gif',
			plusBottom	: iconPath+'plusbottom.gif',
			minus		: iconPath+'minus.gif',
			minusBottom	: iconPath+'minusbottom.gif',
			nlPlus		: iconPath+'nolines_plus.gif',
			nlMinus		: iconPath+'nolines_minus.gif'
		};
	//added by wallimn, to cache checkbox object and improve speed of set checked status
	this.cbCollection = new Object();
	this.obj = objName;
	this.aNodes = [];
	this.aIndent = [];
	this.root = new Node(-1);
	this.selectedNode = null;
	this.selectedFound = false;
	this.completed = false;
};

// Adds a new node to the node array
dTree.prototype.add = function(id, pid, name, url, title, target, icon, iconOpen, open) {
	this.aNodes[this.aNodes.length] = new Node(id, pid, name, url, title, target, icon, iconOpen, open);
};

// Open/close all nodes
dTree.prototype.openAll = function() {
	this.oAll(true);
};
dTree.prototype.closeAll = function() {
	this.oAll(false);
};

// Outputs the tree to the page
dTree.prototype.toString = function() {
	var str = '<div class="dtree">\n';
	if (document.getElementById) {
		if (this.config.useCookies) this.selectedNode = this.getSelected();
		str += this.addNode(this.root);
	} else str += 'Browser not supported.';
	str += '</div>';
	if (!this.selectedFound) this.selectedNode = null;
	this.completed = true;
	return str;
};

// Creates the tree structure
dTree.prototype.addNode = function(pNode) {
	var str = '';
	var n=0;
	if (this.config.inOrder) n = pNode._ai;
	for (n; n<this.aNodes.length; n++) {
		if (this.aNodes[n].pid == pNode.id) {
			var cn = this.aNodes[n];
			cn._p = pNode;
			cn._ai = n;
			this.setCS(cn);
			if (!cn.target && this.config.target) cn.target = this.config.target;
			if (cn._hc && !cn._io && this.config.useCookies) cn._io = this.isOpen(cn.id);
			if (!this.config.folderLinks && cn._hc) cn.url = null;
			if (this.config.useSelection && cn.id == this.selectedNode && !this.selectedFound) {
					cn._is = true;
					this.selectedNode = n;
					this.selectedFound = true;
			}
			str += this.node(cn, n);
			if (cn._ls) break;
		}
	}
	return str;
};
//get checkbox object by id(input by client)
//added by wallimn,
dTree.prototype.co=function(id){
	if (this.cbCollection[id])return this.cbCollection[id];
	for(var n=0; n<this.aNodes.length; n++){
		if(this.aNodes[n].id==id){
			this.cbCollection[id]=document.getElementById("c"+this.obj+n);
			break;
		}
	}
	return this.cbCollection[id];
};
//get the ids of the checkbox which check status is true;
//added by wallimn, updated: 2009-02-13
dTree.prototype.getCheckedNodes=function(){
	var res = new Array();
	var cko;//checkobject
	for(var n=0; n<this.aNodes.length; n++){
		//alert("c"+this.obj+n+(document.getElementById("c"+this.obj+n).checked));
		//cache the object to improve speed when you have very,very many nodes and call this function many times in one page
		//i.e. with ajax technology
		//document.getElementById("c"+this.obj+n).checked
		cko = this.co(this.aNodes[n].id);
		if(cko.checked==true){
			res[res.length]=this.aNodes[n].id;
		}
	}
	return res;
}
//added by wallimn
dTree.prototype.cc=function(nodeId){
	var cs = document.getElementById("c"+this.obj+nodeId).checked;
	var n,node = this.aNodes[nodeId];
	var len =this.aNodes.length;
	for (n=0; n<len; n++) {
		//alert(this.aNodes[n].pid+"--"+this.aNodes[n].id);
		if (this.aNodes[n].pid == node.id) {
			//if (this.aNodes[n]._io) this.nodeStatus(false, n, this.aNodes[n]._ls);
			//this.aNodes[n]._io = false;
			document.getElementById("c"+this.obj+n).checked=cs;
			this.cc(n);		
		}
	}
	if(cs==false)return;
	var pid=node.pid;
	var bSearch;
	do{
		bSearch=false;
		for(n=0;n<len;n++){
			if(this.aNodes[n].id==pid){
				document.getElementById("c"+this.obj+n).checked=true;
				pid=this.aNodes[n].pid;
				bSearch= true;				
				break;
			}
		}
	}while(bSearch==true)
}

// Creates the node icon, url and text
dTree.prototype.node = function(node, nodeId) {
	var str = '<div class="dTreeNode">' + this.indent(node, nodeId);
	if (this.config.useIcons) {
		if (!node.icon) node.icon = (this.root.id == node.pid) ? this.icon.root : ((node._hc) ? this.icon.folder : this.icon.node);
		if (!node.iconOpen) node.iconOpen = (node._hc) ? this.icon.folderOpen : this.icon.node;
		if (this.root.id == node.pid) {
			node.icon = this.icon.root;
			node.iconOpen = this.icon.root;
		}
		str += '<img id="i' + this.obj + nodeId + '" src="' + ((node._io)?node.iconOpen : node.icon) + '" alt="" />';
	}
	if(this.config.check==true){
		str+= '<input type="checkbox" class="cx" id="c'+  this.obj + nodeId + '" onclick="javascript:'+this.obj+'.cc('+nodeId+')"/>';
		//alert(str);
	}
	if (node.url) {
		str += '<a id="s' + this.obj + nodeId + '" class="' + ((this.config.useSelection) ? ((node._is ? 'nodeSel' : 'node')) : 'node') + '" href="' + node.url + '"';
		if (node.title) str += ' title="' + node.title + '"';
		if (node.target) str += ' target="' + node.target + '"';
		if (this.config.useStatusText) str += ' onmouseover="window.status=\'' + node.name + '\';return true;" onmouseout="window.status=\'\';return true;" ';
		if (this.config.useSelection && ((node._hc && this.config.folderLinks) || !node._hc))	str += ' onclick="javascript: ' + this.obj + '.s(' + nodeId + ');"';
		str += '>';
	}
	else if ((!this.config.folderLinks || !node.url) && node._hc && node.pid != this.root.id)
		str += '<a href="javascript: ' + this.obj + '.o(' + nodeId + ');" class="node">';
	str += node.name;
	if (node.url || ((!this.config.folderLinks || !node.url) && node._hc)) str += '</a>';
	str += '</div>';
	if (node._hc) {
		str += '<div id="d' + this.obj + nodeId + '" class="clip" style="display:' + ((this.root.id == node.pid || node._io) ? 'block' : 'none') + ';">';
		str += this.addNode(node);
		str += '</div>';
	}
	this.aIndent.pop();
	//alert(str);
	return str;
};

// Adds the empty and line icons
dTree.prototype.indent = function(node, nodeId) {
	var str = '';
	if (this.root.id != node.pid) {
		for (var n=0; n<this.aIndent.length; n++)
			str += '<img src="' + ( (this.aIndent[n] == 1 && this.config.useLines) ? this.icon.line : this.icon.empty ) + '" alt="" />';
		(node._ls) ? this.aIndent.push(0) : this.aIndent.push(1);
		if (node._hc) {
			str += '<a href="javascript: ' + this.obj + '.o(' + nodeId + ');"><img id="j' + this.obj + nodeId + '" src="';
			if (!this.config.useLines) str += (node._io) ? this.icon.nlMinus : this.icon.nlPlus;
			else str += ( (node._io) ? ((node._ls && this.config.useLines) ? this.icon.minusBottom : this.icon.minus) : ((node._ls && this.config.useLines) ? this.icon.plusBottom : this.icon.plus ) );
			str += '" alt="" /></a>';
		} else str += '<img src="' + ( (this.config.useLines) ? ((node._ls) ? this.icon.joinBottom : this.icon.join ) : this.icon.empty) + '" alt="" />';
	}
	return str;
};

// Checks if a node has any children and if it is the last sibling
dTree.prototype.setCS = function(node) {
	var lastId;
	for (var n=0; n<this.aNodes.length; n++) {
		if (this.aNodes[n].pid == node.id) node._hc = true;
		if (this.aNodes[n].pid == node.pid) lastId = this.aNodes[n].id;
	}
	if (lastId==node.id) node._ls = true;
};

// Returns the selected node
dTree.prototype.getSelected = function() {
	var sn = this.getCookie('cs' + this.obj);
	return (sn) ? sn : null;
};

// Highlights the selected node
dTree.prototype.s = function(id) {
	if (!this.config.useSelection) return;
	var cn = this.aNodes[id];
	if (cn._hc && !this.config.folderLinks) return;
	if (this.selectedNode != id) {
		if (this.selectedNode || this.selectedNode==0) {
			eOld = document.getElementById("s" + this.obj + this.selectedNode);
			eOld.className = "node";
		}
		eNew = document.getElementById("s" + this.obj + id);
		eNew.className = "nodeSel";
		this.selectedNode = id;
		if (this.config.useCookies) this.setCookie('cs' + this.obj, cn.id);
	}
};

// Toggle Open or close
dTree.prototype.o = function(id) {
	var cn = this.aNodes[id];
	this.nodeStatus(!cn._io, id, cn._ls);
	cn._io = !cn._io;
	if (this.config.closeSameLevel) this.closeLevel(cn);
	if (this.config.useCookies) this.updateCookie();
};

// Open or close all nodes
dTree.prototype.oAll = function(status) {
	for (var n=0; n<this.aNodes.length; n++) {
		if (this.aNodes[n]._hc && this.aNodes[n].pid != this.root.id) {
			this.nodeStatus(status, n, this.aNodes[n]._ls)
			this.aNodes[n]._io = status;
		}
	}
	if (this.config.useCookies) this.updateCookie();
};

// Opens the tree to a specific node
dTree.prototype.openTo = function(nId, bSelect, bFirst) {
	if (!bFirst) {
		for (var n=0; n<this.aNodes.length; n++) {
			if (this.aNodes[n].id == nId) {
				nId=n;
				break;
			}
		}
	}
	var cn=this.aNodes[nId];
	if (cn.pid==this.root.id || !cn._p) return;
	cn._io = true;
	cn._is = bSelect;
	if (this.completed && cn._hc) this.nodeStatus(true, cn._ai, cn._ls);
	if (this.completed && bSelect) this.s(cn._ai);
	else if (bSelect) this._sn=cn._ai;
	this.openTo(cn._p._ai, false, true);
};

// Closes all nodes on the same level as certain node
dTree.prototype.closeLevel = function(node) {
	for (var n=0; n<this.aNodes.length; n++) {
		if (this.aNodes[n].pid == node.pid && this.aNodes[n].id != node.id && this.aNodes[n]._hc) {
			this.nodeStatus(false, n, this.aNodes[n]._ls);
			this.aNodes[n]._io = false;
			this.closeAllChildren(this.aNodes[n]);
		}
	}
}

// Closes all children of a node
dTree.prototype.closeAllChildren = function(node) {
	for (var n=0; n<this.aNodes.length; n++) {
		if (this.aNodes[n].pid == node.id && this.aNodes[n]._hc) {
			if (this.aNodes[n]._io) this.nodeStatus(false, n, this.aNodes[n]._ls);
			this.aNodes[n]._io = false;
			this.closeAllChildren(this.aNodes[n]);		
		}
	}
}

// Change the status of a node(open or closed)
dTree.prototype.nodeStatus = function(status, id, bottom) {
	eDiv	= document.getElementById('d' + this.obj + id);
	eJoin	= document.getElementById('j' + this.obj + id);
	if (this.config.useIcons) {
		eIcon	= document.getElementById('i' + this.obj + id);
		eIcon.src = (status) ? this.aNodes[id].iconOpen : this.aNodes[id].icon;
	}
	eJoin.src = (this.config.useLines)?
	((status)?((bottom)?this.icon.minusBottom:this.icon.minus):((bottom)?this.icon.plusBottom:this.icon.plus)):
	((status)?this.icon.nlMinus:this.icon.nlPlus);
	eDiv.style.display = (status) ? 'block': 'none';
};
// [Cookie] Clears a cookie
dTree.prototype.clearCookie = function() {
	var now = new Date();
	var yesterday = new Date(now.getTime() - 1000 * 60 * 60 * 24);
	this.setCookie('co'+this.obj, 'cookieValue', yesterday);
	this.setCookie('cs'+this.obj, 'cookieValue', yesterday);
};

// [Cookie] Sets value in a cookie
dTree.prototype.setCookie = function(cookieName, cookieValue, expires, path, domain, secure) 

{
	document.cookie =
		escape(cookieName) + '=' + escape(cookieValue)
		+ (expires ? '; expires=' + expires.toGMTString() : '')
		+ (path ? '; path=' + path : '')
		+ (domain ? '; domain=' + domain : '')
		+ (secure ? '; secure' : '');
};

// [Cookie] Gets a value from a cookie
dTree.prototype.getCookie = function(cookieName) {
	var cookieValue = '';
	var posName = document.cookie.indexOf(escape(cookieName) + '=');
	if (posName != -1) {
		var posValue = posName + (escape(cookieName) + '=').length;
		var endPos = document.cookie.indexOf(';', posValue);
		if (endPos != -1) cookieValue = unescape(document.cookie.substring(posValue, endPos));
		else cookieValue = unescape(document.cookie.substring(posValue));
	}
	return (cookieValue);
};

// [Cookie] Returns ids of open nodes as a string
dTree.prototype.updateCookie = function() {
	var str = '';
	for (var n=0; n<this.aNodes.length; n++) {
		if (this.aNodes[n]._io && this.aNodes[n].pid != this.root.id) {
			if (str) str += '.';
			str += this.aNodes[n].id;
		}
	}
	this.setCookie('co' + this.obj, str);
};

// [Cookie] Checks if a node id is in a cookie
dTree.prototype.isOpen = function(id) {
	var aOpen = this.getCookie('co' + this.obj).split('.');
	for (var n=0; n<aOpen.length; n++)
		if (aOpen[n] == id) return true;
	return false;
};

// If Push and pop is not implemented by the browser
if (!Array.prototype.push) {
	Array.prototype.push = function array_push() {
		for(var i=0;i<arguments.length;i++)
			this[this.length]=arguments[i];
		return this.length;
	}
};
if (!Array.prototype.pop) {
	Array.prototype.pop = function array_pop() {
		lastElement = this[this.length-1];
		this.length = Math.max(this.length-1,0);
		return lastElement;
	}
};

/***********本人原创,欢迎转载,转载请保留本人信息*************/
作者:wallimn 电邮:wallimn@sohu.com 时间:2009-02-13
博客:http://blog.csdn.net/wallimn http://wallimn.iteye.com
网络硬盘:http://wallimn.ys168.com
/***********文章发表请与本人联系,作者保留所有权利*************/
19
2
分享到:
评论
22 楼 wallimn 2014-02-18  
xuexishoucangbaodian 写道
wallimn 写道
xuexishoucangbaodian 写道
在不在?想请教个问题,如何根据ID(例如:0,1001,1003)使checkbox回显啊?这个里面有现成的方法么?


可以遍历一下,一个一个的设置上。函数co可以返回id对应的checkbox,然后checked=true就行了。也就是mytreeco('1').checked=true。

再请教一个问题,回显以后我想要默认打开当前的节点菜单,是什么方法?在此先谢谢你


看看296行的代码,是不是满足你的要求。
21 楼 xuexishoucangbaodian 2014-02-18  
wallimn 写道
xuexishoucangbaodian 写道
在不在?想请教个问题,如何根据ID(例如:0,1001,1003)使checkbox回显啊?这个里面有现成的方法么?


可以遍历一下,一个一个的设置上。函数co可以返回id对应的checkbox,然后checked=true就行了。也就是mytreeco('1').checked=true。

再请教一个问题,回显以后我想要默认打开当前的节点菜单,是什么方法?在此先谢谢你
20 楼 wallimn 2014-02-17  
xuexishoucangbaodian 写道
在不在?想请教个问题,如何根据ID(例如:0,1001,1003)使checkbox回显啊?这个里面有现成的方法么?


可以遍历一下,一个一个的设置上。函数co可以返回id对应的checkbox,然后checked=true就行了。也就是mytreeco('1').checked=true。
19 楼 xuexishoucangbaodian 2014-02-17  
在不在?想请教个问题,如何根据ID(例如:0,1001,1003)使checkbox回显啊?这个里面有现成的方法么?
18 楼 wallimn 2012-07-31  
拼接成a=1&a=2的形式传递。或者用表单
17 楼 march_on 2012-07-29  
请问我怎么才能把我选择的dtree中的节点信息传递给另外一个jsp界面呢。
我在js文件里面写了一个retrieve函数可以获得我选择的dtree节点信息
function retrieve()
{
        var retrieveResult = new Array();
var selectedNodes = dtree.getCheckedNodes();
for( var i = 0; i < selectedNodes.length; ++i)
{
retrieveResult.push( dtree.aNodes[selectedNodes[i]].name );
}

}

但是下面我怎么把这个retrieveResult传递给另外的jsp界面呢
谢谢lz了
16 楼 luoting2762 2011-01-25  
请问如何确定选定checkbox对应的tree的nodeid?
15 楼 wallimn 2010-09-06  
修改级联选择,查看这个函数:dTree.prototype.cc=function(nodeId),可以通过对这个函数的修改实现自己的想法。不是很难,现在比较忙,顾不上弄这个。
14 楼 stards_cai 2010-09-04  
发现有些小问题,不知楼主是否能改善一下checkBox级联选中的算法:当所有同级节点未被同时选中的时候,父节点不应该被选中。(因为这样容易造成误解)
1.从未选到选取:点击节点的时候,判断该节点的所有同级节点是否都为checked==true,如果是,则设置父节点为选中;同时,如果它有子节点,则选中所有子节点。
2.从选中到取消:设置父节点为false; 同时,如果它有子节点,则设置所有子节点为false。
13 楼 wallimn 2010-08-26  
修改级联选择,查看这个函数:dTree.prototype.cc=function(nodeId),可以通过给dTree加一个变量的方式来进行控制(查看dTree函数的定义)。
12 楼 Quen 2010-08-26  
想问问,里面的级联选择(就是选定父节点时所有子节点也会被选定)是怎么设置的,如果我不想级联起来,那该怎么去修改呀!谢谢!
11 楼 hitman小血 2010-05-13  
能告诉我你的QQ号吗  那样说话方便
10 楼 wallimn 2010-05-13  
那应该不会,我也因了呀,节点不止30个呀,你的问题可能出在其它方面。
9 楼 hitman小血 2010-05-13  
我正在用您的修改后的dtree 但是发现个问题 当树形节点过多时 大概30个以上时 就获得不到被选中的id了  能告诉我怎么办么?
8 楼 wallimn 2010-01-20  
getCheckedNodes
7 楼 songaiyun 2010-01-20  
是不是 getCheckedNodes ? 能给具体说下么?着急找这个呢  感谢
6 楼 wallimn 2010-01-19  
有个函数,名字我忘记了,好像是getSelectedNodes
5 楼 songaiyun 2010-01-19  
很好 感谢! 然后提问:

请问如何确定选定checkbox对应的tree的nodeid?
4 楼 xiaosky 2009-11-26  
这个确实有用的,我也正需要。
关键是性能怎么样。
回复做这么多的测试题,真是麻烦啊。
3 楼 wallimn 2009-09-20  
我的网络硬盘上有例子,地址:wallimn.ys168.com

相关推荐

    dtree和dtree_checkbox(dtree复选框)

    该资料包括dtree和dtree_checkbox(dtree复选框),众所周知,dtree是个不错的JS控件,好资料当然要和大家分享^_^

    dtree+checkbox

    dtree +checkBox全选 。。。。dtree +checkBox全选 。。。。dtree +checkBox全选 。。。。dtree +checkBox全选 。。。。dtree +checkBox全选 。。。。dtree +checkBox全选 。。。。

    三个常用树控件dtree最新dtree带checkbox还有mztreeview1.0跟2.0

    三个常用树控件dtree最新dtree带checkbox还有mztreeview1.0跟2.0里面还有本人写的部分demo如果是常用开发请使用mztree1.0或dtree 如果有特殊需求请使用带checkbox的dtree或mztree2.0

    Dtree(checkbox)

    本人根据dtree添加了checkbox的功能,能默认选中,方便好用

    dtree带checkbox

    dtree带checkbox,java项目

    dtree-checkbox-muchlin修改版

    源文件:dtree-ck.source.js 15kb 压缩版:dtree-ck.compress.js 11kb 改动: 1、修改可单独设置节点有无checkbox。 2、调整节点open参数至target参数后面。 3、添加css文件自动加载方法,使用时只需在head区引用...

    带checkbox的dtree

    在dtree原有基础上添加了checkbox功能

    dtree_checkbox

    NULL 博文链接:https://liu-shui8.iteye.com/blog/591865

    DTree & CheckBox

    DTree CheckBox JS树 无限级树 灵活

    dtree checkBox

    dtree 带checkBox 选中父节点,子节点全选。子节点取消选中,父节点取消,并且此节点保持原有状态

    dtree 框架 还有checkbox 选择框

    dtree 框架 还有checkbox 选择框 dtree 框架 还有checkbox 选择框

    dtree_checkBox

    dtree+jquery+oracle+tomcat 从oracle数据库中取出dtree的节点信息,在jsp页面使用jquery动态生成带checkBox的dtree 测试时请访问main.jsp WebRoot/db目录为数据库建表脚本;

    dtree 加入checkbox

    dtree.js动态树增加checkbox功能属性

    dtree组件 带checkbox

    纯javascript实现,有checkbox的dtree,下载就可以试用,有完整实例

    带 checkbox 的 dtree 权限树

    有dtree做的 带checkbox的权限树

    dtree_role.rar_dtree checkbox ro

    修改了传统带checkbox的dtree中选中复选框时的操作,大大提升了全选,选父节点时子节点全部选中 等的速度

    dtree支持动态添加,删除节点

    dtree支持动态添加,删除节点dtree支持动态添加,删除节点dtree支持动态添加,删除节点dtree支持动态添加,删除节点

    dtree树形菜单(有带checkbox实现)

    本工程使用dTree组件实现树形菜单的展示 有静态树形菜单和动态(从数据库获取数据)树形菜单展示两种 ...菜单既有普通的树结构展示,也有方便做权限等功能的带选择框的展示(含带默认选中checkbox的处理)

    dtree新改版-添加checkbox复选框

    dtree修改版-添加checkbox: 由于原版dtree作者没有加上复选框功能,我用来做用户模块权限功能的时候很不好用,网上也找到一些基于dtree添加了checkbox的修改版,但是用起来很不顺手,最后不得不根据自己的需要一再...

Global site tag (gtag.js) - Google Analytics