数据源用数组混json结构,实现了基本的功能。效率一般,跟 dhtree 梅花雪树对比了下,都差不多。 (ps感觉比dhtree快点,跟梅花雪树差不多,个人测试)

这个实现树的原理是根据json,不断的生成ul li, 下面是一个简单的例子(只有涉及到生成树,也就是说只是展示,tree类代码只有64行) 没有用innerHTML生成,全是是创建节点来创建ul li,所以创建节点碎片添加,然后再一次性添加很重要啊,确实能提高速度。

<!DOCTYPE html>
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=gb-2312" />
<title>easytree</title>
<style type="text/css">
ul,li{
list-style: none outside none;
margin: 0;
overflow: hidden;
padding: 0;
}
#demo{
background-color:#F0FBEB;
}
#demo .root{
margin-left: -20px;
}
#demo ul{
padding-left: 20px;
}
#demo .folder{
background: url("../../images/201202/r_i.png") repeat-y scroll 0 21px transparent;
}
#demo img{
background: url("../../images/201202/r_icon.png") no-repeat scroll 0 0 transparent;
border: medium none;
height: 20px;
vertical-align: top;
width: 20px;
}
#demo .tvicon-open{
background-position: -160px -40px;
}
#demo .tvdash-f{
background-position: -240px -40px;
}
#demo .tvdash-f-open{
background-position: -200px -40px;
}
#demo .tvdash-fl{
background-position: -100px -40px;
}
#demo .tvdash-fl-open{
background-position: -80px -40px;
}
#demo .tvdash-t{
background-position: -180px -40px;
}
#demo .tvdash-tl{
background-position: -60px -40px;
}
</style>
</head>
<body>
<div id='sss'></div>
<div id='demo'></div>
<script>
(function(doc,undefined){
var window = this;
window.Sys = function (ua){
var b = {
ie: /msie/.test(ua) && !/opera/.test(ua),
opera: /opera/.test(ua),
safari: /webkit/.test(ua) && !/chrome/.test(ua),
firefox: /firefox/.test(ua),
chrome: /chrome/.test(ua)
},vMark = "";
for (var i in b) {
if (b[i]) { vMark = "safari" == i ? "version" : i; break; }
}
b.version = vMark && RegExp("(?:" + vMark + ")[\\/: ]([\\d.]+)").test(ua) ? RegExp.$1 : "0";
b.ie6 = b.ie && parseInt(b.version, 10) == 6;
b.ie7 = b.ie && parseInt(b.version, 10) == 7;
b.ie8 = b.ie && parseInt(b.version, 10) == 8;
return b;
}(window.navigator.userAgent.toLowerCase());
window.Sys.ie6&&doc.execCommand("BackgroundImageCache", false, true);
window.$$ = function(Id){
return doc.getElementById(Id);
};
window.$c = function(name,parent){
var elem = doc.createElement(name);
parent&&parent.appendChild(elem);
return elem;
};
window.addListener = function(element,e,fn){
!element.events&&(element.events = {});
element.events[e]&&(element.events[e][addListener.guid++]=fn)||(element.events[e] = {'0':fn});
element.addEventListener?element.addEventListener(e,fn,false):element.attachEvent("on" + e,fn);
};
window.addListener.guid = 1;
window.removeListener = function(element,e,fn){
var handlers = element.events[e],type;
if(fn){
for(type in handlers)
if(handlers[type]===fn){
element.removeEventListener?element.removeEventListener(e,fn,false):element.detachEvent("on" + e,fn);
delete handlers[type];
}
}else{
for(type in handlers){
element.removeEventListener?element.removeEventListener(e,handlers[type],false):element.detachEvent("on" + e,handlers[type]);
delete handlers[type];
}
}
};
window.setStyle = function(e,o){
if(typeof o=="string")
e.style.cssText=o;
else
for(var i in o)
e.style[i] = o[i];
};
var slice = Array.prototype.slice;
window.Bind = function(object, fun) {
var args = slice.call(arguments).slice(2);
return function() {
return fun.apply(object, args);
};
};
window.BindAsEventListener = function(object, fun,args) {
var args = slice.call(arguments).slice(2);
return function(event) {
return fun.apply(object, [event || window.event].concat(args));
}
};
//copy from jQ
window.extend = function(){
var target = arguments[0] || {}, i = 1, length = arguments.length, deep = true, options;
if ( typeof target === "boolean" ) {
deep = target;
target = arguments[1] || {};
i = 2;
}
if ( typeof target !== "object" && Object.prototype.toString.call(target)!="[object Function]")
target = {};
for(;i<length;i++){
if ( (options = arguments[ i ]) != null )
for(var name in options){
var src = target[ name ], copy = options[ name ];
if ( target === copy )
continue;
if ( deep && copy && typeof copy === "object" && !copy.nodeType ){
target[ name ] = arguments.callee( deep, src || ( copy.length != null ? [ ] : { } ), copy );
}
else if(copy !== undefined)
target[ name ] = copy;
}
}
return target;
};
//copy from jQ
window.each = function ( object, callback, args ) {
var name, i = 0, length = object.length;
if ( args ) {
args = Array.prototype.slice.call(arguments).slice(2);
if ( length === undefined ) {
for ( name in object )
if ( callback.apply( object[ name ],[name,object[ name ]].concat(args) ) === false )
break;
} else
for ( ; i < length; i++)
if ( callback.apply( object[ i ],[i,object[ i ]].concat(args)) === false ) //
break;
} else {
if ( length === undefined ) {
for ( name in object )
if ( callback.call( object[ name ], name, object[ name ] ) === false )
break;
} else
for ( var value = object[0];
i < length && callback.call( value, i, value ) !== false; value = object[++i] ){}
}
return object;
};
window.currentStyle = function(element){
return element.currentStyle || doc.defaultView.getComputedStyle(element, null);
};
window.objPos = function(elem){
var left = 0, top = 0, right = 0, bottom = 0,doc = elem ? elem.ownerDocument : doc;
if ( !elem.getBoundingClientRect || window.Sys.ie8 ) {
var n = elem;
while (n) { left += n.offsetLeft, top += n.offsetTop; n = n.offsetParent; };
right = left + elem.offsetWidth; bottom = top + elem.offsetHeight;
} else {
var rect = elem.getBoundingClientRect();
left = right = doc.documentElement.scrollLeft || doc.body.scrollLeft;
top = bottom = doc.documentElement.scrollLeft || doc.body.scrollLeft;
left += rect.left; right += rect.right;
top += rect.top; bottom += rect.bottom;
}
return { "left": left, "top": top, "right": right, "bottom": bottom };
};
window.hasClass = function(element, className){
return element.className.match(new RegExp('(\\s|^)'+className+'(\\s|$)'));
};
window.addClass = function(element, className){
!window.hasClass(element, className)&&(element.className += " "+className);
};
window.removeClass = function(element, className){
window.hasClass(element, className)&&(element.className = element.className.replace(new RegExp('(\\s|^)'+className+'(\\s|$)'),' '));
}
})(document);
//------------------------------------------------------------
(function(doc,undefined){
var win = this;
win.easyTree = function(){
this.init.apply(this,arguments);
};
win.easyTree.prototype = {
options : {
},
init : function(options){
this.deflaut = extend(this.options,options);
this.container = options.container;
this.data = options.data;
//根节点直接在这里生成
if(this.data.length>=1){
var ul = $c('ul',this.container),
self = this,
li;
each(this.data,function(i,o){
li = $c('li',ul);
li.className = 'root';
li.innerHTML = '<img src="http://images.cnblogs.com/cnblogs_com/wtcsy/294024/r_space.png" class="tvicon-open"><a href="#">'+o.name+'</a>';
o.child
&&self.bulidTree(o.child,li,'t')
});
}
},
bulidTree : function(data,parent,first){
if(data.length===0)
return;
var ul = $c('ul',parent),
self = this,
len = data.length,
imgClassName,
li;
first
? ul.style.paddingLeft = '0px'
: ul.style.display = 'none';
each(data,function(i,o){
var isLast = (len-1)===i;
li = $c('li',ul);
li.className = 'folder'+(isLast?'l':'');
imgClassName = o.child
? isLast?'tvdash-fl':'tvdash-f'
: isLast?'tvdash-tl':'tvdash-t';
li.innerHTML = '<img src="http://images.cnblogs.com/cnblogs_com/wtcsy/294024/r_space.png" class="+imgClassName+"><a href="#">'+o.name+'</a>';
img = li.getElementsByTagName('img')[0];
addListener(img,'click',BindAsEventListener(self,self.openOrClose,li,img));
o.child
&&self.bulidTree(o.child,li);
});
},
openOrClose : function(e,li,img){
var ul = li.getElementsByTagName('ul')[0];
if(ul){
ul.style.display = ul.style.display === 'none'?'':'none';
var name = img.className,
is = img.className.indexOf('open')>-1;
img.className = is?name.replace('-open',''):name+'-open';
}
}
}
})(document);
window.onload = function(){
var i = 0;
var data = [{id:i,name:'根节点',child:[]}];
while(i<100){
i++;
data[0].child.push({id:i,name:'节点'+i,child:[
{id:i*10+1,name:'节点'+i+'-1',child:[
{id:i*11+1,name:'节点'+i+'-1-2'},
{id:i*11+2,name:'节点'+i+'-1-3'},
{id:i*11+3,name:'节点'+i+'-1-4'}
]},
{id:i*10+2,name:'节点'+i+'-2',child:[
{id:i*11+4,name:'节点'+i+'-2-2'},
{id:i*11+5,name:'节点'+i+'-2-3'},
{id:i*11+6,name:'节点'+i+'-2-4'}
]},
{id:i*10+3,name:'节点'+i+'1-3'}
]});
}
var t = new Date().getTime();
new easyTree({
container : $$('demo'),
data : data
});
$$('sss').innerHTML ='一共1000个节点 共耗时:'+ (new Date().getTime()-t)+'毫秒'
}
</script>
</body>
</html>

上面只是一个简单的树,但是树应该是有input的,并且能够选中。记录选中能够开打任何节点,能够删除,不能够编辑、保存。说一下我在这里怎么完成上面的功能。选中应该是比较麻烦的,因为选中了一个节点,要让他所有的子节点都选中,他的父节点也有可能要选中(如果同级的都是选中状态的)。接着他的父的父也有可能要选中,所以设计到很多的查找,当生成一个li的时候,给li一个属性:

1 li.child.push({
2         li    : li,
3         input : isInput ?$q('input',li)[0]:undefined,
4         data  : o
5     });

记录下他下一级(是下一级 不是所有的子 )的所有的input 和li同时记录父的li 和 input:

1 li.parent = {
2         li    : parent,
3         input : isInput ?$q('input',parent)[0]:undefined
4     };

这样查找起来很方便,但是内存就要用的多些了。

<!DOCTYPE html>
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=gb-2312" />
<title>easytree</title>
<style type="text/css">
body{font:12px/1.5 '宋体',tahoma;}
ul,li{
list-style: none outside none;
margin: 0;
overflow: hidden;
padding: 0;
}
#demo span{
background: url("../../images/201202/icon.png") no-repeat scroll 0 0 transparent;
border: medium none;
height: 20px;
vertical-align: top;
width: 20px;
display :inline-block;
}
#demo .root{
margin-left: -20px;
}
#demo .root input{
vertical-align: middle;
}
#demo ul{
padding-left: 20px;
}
#demo .folder{
background: url("../../images/201202/i.gif") repeat-y scroll 0 21px transparent;
}
#demo img{
background: url("../../images/201202/icon.png") no-repeat scroll 0 0 transparent;
border: medium none;
height: 20px;
vertical-align: top;
width: 20px;
}
#demo .tvicon-open{
background-position: -160px -40px;
}
#demo .tvdash-f{
background-position: -240px -40px;
}
#demo .tvdash-f-open{
background-position: -200px -40px;
}
#demo .tvdash-fl{
background-position: -100px -40px;
}
#demo .tvdash-fl-open{
background-position: -80px -40px;
}
#demo .tvdash-t{
background-position: -180px -40px;
}
#demo .tvdash-tl{
background-position: -60px -40px;
}
#demo .tvicon-f{
background-position: -120px -40px;
}
#demo .tvicon-f-open{
background-position: -140px -40px;
}
#demo .tvicon-tl{
background-position: -220px -40px;
}
</style>
</head>
<body>
<div id='sss'></div>
<div>
输入id值<br>
<input value='101' type="text" id='v'>
<input value='展开节点'type="button"onClick="ch(document.getElementById('v').value)"><br><br>
返回的是节点id值 也可以是主键值
<input value='查看有多少节点被选中了'type="button"onClick="ss(this.value)">
<br>
</div>
<div id='demo'></div>
<script>
(function(doc,undefined){
var window = this;
window.Sys = function (ua){
var b = {
ie: /msie/.test(ua) && !/opera/.test(ua),
opera: /opera/.test(ua),
safari: /webkit/.test(ua) && !/chrome/.test(ua),
firefox: /firefox/.test(ua),
chrome: /chrome/.test(ua)
},vMark = "";
for (var i in b) {
if (b[i]) { vMark = "safari" == i ? "version" : i; break; }
}
b.version = vMark && RegExp("(?:" + vMark + ")[\\/: ]([\\d.]+)").test(ua) ? RegExp.$1 : "0";
b.ie6 = b.ie && parseInt(b.version, 10) == 6;
b.ie7 = b.ie && parseInt(b.version, 10) == 7;
b.ie8 = b.ie && parseInt(b.version, 10) == 8;
return b;
}(window.navigator.userAgent.toLowerCase());
window.Sys.ie6&&doc.execCommand("BackgroundImageCache", false, true);
window.$$ = function(Id){
return doc.getElementById(Id);
};
window.$q = function(name,parent){
return parent.getElementsByTagName(name);
};
window.$c = function(name,parent){
var elem = typeof name==='object'?name:doc.createElement(name);
parent&&parent.appendChild(elem);
return elem;
};
window.addListener = function(element,e,fn){
!element.events&&(element.events = {});
element.events[e]&&(element.events[e][addListener.guid++]=fn)||(element.events[e] = {'0':fn});
element.addEventListener?element.addEventListener(e,fn,false):element.attachEvent("on" + e,fn);
};
window.addListener.guid = 1;
window.removeListener = function(element,e,fn){
var handlers = element.events[e],type;
if(fn){
for(type in handlers)
if(handlers[type]===fn){
element.removeEventListener?element.removeEventListener(e,fn,false):element.detachEvent("on" + e,fn);
delete handlers[type];
}
}else{
for(type in handlers){
element.removeEventListener?element.removeEventListener(e,handlers[type],false):element.detachEvent("on" + e,handlers[type]);
delete handlers[type];
}
}
};
window.setStyle = function(e,o){
if(typeof o=="string")
e.style.cssText=o;
else
for(var i in o)
e.style[i] = o[i];
};
var slice = Array.prototype.slice;
window.Bind = function(object, fun) {
var args = slice.call(arguments).slice(2);
return function() {
return fun.apply(object, args);
};
};
window.BindAsEventListener = function(object, fun,args) {
var args = slice.call(arguments).slice(2);
return function(event) {
return fun.apply(object, [event || window.event].concat(args));
}
};
//copy from jQ
window.extend = function(){
var target = arguments[0] || {}, i = 1, length = arguments.length, deep = true, options;
if ( typeof target === "boolean" ) {
deep = target;
target = arguments[1] || {};
i = 2;
}
if ( typeof target !== "object" && Object.prototype.toString.call(target)!="[object Function]")
target = {};
for(;i<length;i++){
if ( (options = arguments[ i ]) != null )
for(var name in options){
var src = target[ name ], copy = options[ name ];
if ( target === copy )
continue;
if ( deep && copy && typeof copy === "object" && !copy.nodeType ){
target[ name ] = arguments.callee( deep, src || ( copy.length != null ? [ ] : { } ), copy );
}
else if(copy !== undefined)
target[ name ] = copy;
}
}
return target;
};
//copy from jQ
window.each = function ( object, callback, args ) {
var name, i = 0, length = object.length;
if ( args ) {
args = Array.prototype.slice.call(arguments).slice(2);
if ( length === undefined ) {
for ( name in object )
if ( callback.apply( object[ name ],[name,object[ name ]].concat(args) ) === false )
break;
} else
for ( ; i < length; i++)
if ( callback.apply( object[ i ],[i,object[ i ]].concat(args)) === false ) //
break;
} else {
if ( length === undefined ) {
for ( name in object )
if ( callback.call( object[ name ], name, object[ name ] ) === false )
break;
} else
for ( var value = object[0];
i < length && callback.call( value, i, value ) !== false; value = object[++i] ){}
}
return object;
};
window.currentStyle = function(element){
return element.currentStyle || doc.defaultView.getComputedStyle(element, null);
};
window.objPos = function(elem){
var left = 0, top = 0, right = 0, bottom = 0,doc = elem ? elem.ownerDocument : doc;
if ( !elem.getBoundingClientRect || window.Sys.ie8 ) {
var n = elem;
while (n) { left += n.offsetLeft, top += n.offsetTop; n = n.offsetParent; };
right = left + elem.offsetWidth; bottom = top + elem.offsetHeight;
} else {
var rect = elem.getBoundingClientRect();
left = right = doc.documentElement.scrollLeft || doc.body.scrollLeft;
top = bottom = doc.documentElement.scrollLeft || doc.body.scrollLeft;
left += rect.left; right += rect.right;
top += rect.top; bottom += rect.bottom;
}
return { "left": left, "top": top, "right": right, "bottom": bottom };
};
window.hasClass = function(element, className){
return element.className.match(new RegExp('(\\s|^)'+className+'(\\s|$)'));
};
window.addClass = function(element, className){
!window.hasClass(element, className)&&(element.className += " "+className);
};
window.removeClass = function(element, className){
window.hasClass(element, className)&&(element.className = element.className.replace(new RegExp('(\\s|^)'+className+'(\\s|$)'),' '));
}
})(document);
//------------------------------------------------------------------------------------
(function(doc,undefined){
var win = this,
primary;
win.easyTree = function(){
this.init.apply(this,arguments);
};
win.easyTree.prototype = {
options : {
primary : 'id',
isInput : false,
isIco : false
},
init : function(options){
this.deflaut = extend(this.options,options);
this.container = options.container;
//数据源 形式{id:data[i]}
this.data={};
//被勾选的项
this.selected = {};
//根节点直接在这里生成
if(options.data.length>=1){
primary = this.deflaut.primary;
var frag=document.createDocumentFragment()
var ul = $c('ul',frag),//this.container
self = this,
input = this.deflaut.isInput?'<input type="checkbox">':'',
li;
each(options.data,function(i,o){
li = $c('li',ul);
li.className = 'root';
li.innerHTML = [
'<span class="tvicon-open"></span>',
input,
'<a href="#">'+o.name+'</a>'
].join('');
li.setAttribute('primary',o[primary]);
self.data[o[primary]] = {elem:li,data:o};
o.child
&&self.bulidTree(o.child,li,'t');
});
//只绑定一个节点 展开和checkedbox判断都在里面
addListener(ul,'click',BindAsEventListener(self,self.operation));
this.container.appendChild(frag);
}
},
bulidTree : function(data,parent,first){
if(data.length===0)
return;
var ul = $c('ul',parent),
self = this,
len = data.length,
isInput = this.deflaut.isInput,
input = isInput?'<input type="checkbox">':'',
isIco = this.deflaut.isIco,
imgClassName,
img,
li;
first
? ul.style.paddingLeft = '0px'
: ul.style.display = 'none';
parent.child = [];
each(data,function(i,o){
var isLast = (len-1)===i,
ico = isIco
? o.child
? '<span class="tvicon-f"></span>'
: '<span class="tvicon-tl"></span>'
: '';
li = $c('li',ul);
li.className = 'folder'+(isLast?'l':'');
imgClassName = o.child
? isLast?'tvdash-fl':'tvdash-f'
: isLast?'tvdash-tl':'tvdash-t';
li.innerHTML = [
'<span class="+imgClassName+"></span>',
ico,
input,
'<a href="#">'+o.name+'</a>'
].join('');
//当前li保存父li的相关信息 只是一个对象
li.parent = {
li : parent,
input : isInput ?$q('input',parent)[0]:undefined
};
//当前li保存子li的相关信息 是一个数组
parent.child.push({
li : li,
input : isInput ?$q('input',li)[0]:undefined,
data : o
});
li.setAttribute('primary',o[primary]);
li.getElementsByTagName('span')[0].setAttribute('mark','mark');
self.data[o[primary]] = {elem:li,data:o};
o.child
&&self.bulidTree(o.child,li);
});
},
operation : function(e){
var elem = e.srcElement || e.target,
nodeName = elem.nodeName.toLocaleLowerCase();
/*显示隐藏子菜单*/
if((nodeName==='span'&&elem.getAttribute('mark'))||(nodeName==='a')){
var li = elem.parentNode,
ul = $q('ul',li)[0],
img = $q('span',li)[0],
ico = $q('span',li)[1];
if(ul){
ul.style.display = ul.style.display === 'none'?'':'none';
var name = img.className,
is = img.className.indexOf('open')>-1;
img.className = is?name.replace('-open',''):name+'-open';
if(ico){
ico.className = is?ico.className.replace('-open',''):ico.className+'-open';
}
}
};
//选中checkedbox的时候的相应的操作
if(nodeName==='input'){
var isCheck = elem.checked,
li = elem.parentNode,
ul = $q('ul',li)[0],
id = li.getAttribute('primary'),
//该元素的索引是否已经添加到了 this.selected中
isIn = id in this.selected;

//操作this.selected 从里面添加或者删除
isCheck
? !isIn
&&(this.selected[id]=id)
: isIn
&&delete this.selected[id];
this.findChild(li,isCheck);
this.findParent(li,isCheck);
}
},
findParent : function(child,b){
if(!child.parent)
return;
var parentLi = child.parent.li,
id = parentLi.getAttribute('primary')
isAll = true;
if(b){
//看他父下面的所有li是不是都是选中状态的
each(parentLi.child,function(i,o){
if(!o.input.checked){
isAll = false;
return false;
}
});
//如果都是选中状态
//继续查找他的父
if(isAll){
child.parent.input.checked = true;
!(id in this.selected)
&&(this.selected[id] = id);
this.findParent(parentLi,b);
}
}else{
if(child.parent.input.checked){
child.parent.input.checked = false;
id in this.selected
&& delete this.selected[id];
this.findParent(parentLi,b);
}
}
},
/*
查找子节点 将所有没有选中的字节点选中 并放到this.selected里面去
或者取消所有的子节点
*/
findChild : function(parent,b){
if(!parent.child)
return;
var self = this;
b
? each(parent.child,function(i,o){
if(!o.input.checked){
o.input.checked = true;
!(o.data[primary] in self.selected)
&&(self.selected[o.data[primary]] = o.data[primary]);
o.li.child
&&self.findChild(o.li,b);
}
})
: each(parent.child,function(i,o){
if(o.input.checked){
o.input.checked = false;
o.data[primary] in self.selected
&& delete self.selected[o.data[primary]];
o.li.child
&&self.findChild(o.li,b);
}
});
},
getAllSelect : function(){
var arr = [],
id;
for(id in this.selected)
arr.push(id);
return arr.join(',')
},
openNode : function(id){
if(id in this.data){
var li = this.data[id].elem,
parent = li.parentNode,
img,
ico;
while(parent.style.display==='none'){
parent.style.display = '';
li = li.parent.li;
img =$q('span',li)[0];
ico =$q('span',li)[1];
img.className=img.className+'-open';
ico
&&(ico.className=ico.className+'-open');
parent = li.parentNode;
}
}
}
}
})(document);
window.onload = function(){
var i = 0,j=100;
var data1 = [{id:i,name:'根节点',child:[]}];
while(i<100){
i++;
data1[0].child.push({id:i,name:'节点'+i,child:[
{id:++j,name:'节点'+i+'-1',child:[
{id:++j,name:'节点'+i+'-1-1'},
{id:++j,name:'节点'+i+'-1-2'},
{id:++j,name:'节点'+i+'-1-3'}
]},
{id:++j,name:'节点'+i+'-2',child:[
{id:++j,name:'节点'+i+'-2-1'},
{id:++j,name:'节点'+i+'-2-2'},
{id:++j,name:'节点'+i+'-2-3'}
]},
{id:++j,name:'节点'+i+'-3-1'}
]});
}

var t = new Date().getTime();
window.tt= new easyTree({
container : $$('demo'),
isInput : true,
isIco : true,
data : data1
});
//alert(new Date().getTime()-t1)
$$('sss').innerHTML ='一共1000个节点 共耗时:'+ (new Date().getTime()-t)+'毫秒'

}
function ch(v){
tt.openNode(v)
}
function ss(){
alert(tt.getAllSelect());
}
</script>
</body>
</html>

基本能达到要求了。

用JavaScript实现一个简单的树结构的更多相关文章

  1. javascript编写一个简单的编译器(理解抽象语法树AST)

    javascript编写一个简单的编译器(理解抽象语法树AST) 编译器 是一种接收一段代码,然后把它转成一些其他一种机制.我们现在来做一个在一张纸上画出一条线,那么我们画出一条线需要定义的条件如下: ...

  2. 使用JavaScript实现一个简单的编译器

    在前端开发中也会或多或少接触到一些与编译相关的内容,常见的有 将ES6.7代码编译成ES5的代码 将SCSS.LESS代码转换成浏览器支持的CSS代码 通过uglifyjs.uglifycss等工具压 ...

  3. 使用javascript完成一个简单工厂设计模式。

    在JS中创建对象会习惯的使用new关键字和类构造函数(也是可以用对象字面量). 工厂模式就是一种有助于消除两个类依赖性的模式. 工厂模式分为简单工厂模式和复杂工厂模式,这篇主要讲简单工厂模式. 简单工 ...

  4. 用JavaScript写一个简单的计算器

    本文使用js实现了一个简单的加.减.乘.除计算器. 以下是css部分代码: *{ padding:0; margin:0; color: #424242; } .outer{ width:300px; ...

  5. 用javascript编写一个简单的随机验证码程序

    简单模拟网页的随机数字验证码,效果图如下: html代码: <div id="content"> <div class="left"> ...

  6. JavaScript 实现一个简单的MVVM前端框架(ES6语法)

    前言 随着前端各大框架的崛起,为我们平时的开发带来了相当的便利,我们不能一直停留在应用层面,今天就自己动手实现一个乞丐版的MVVM小框架 完整代码github地址 效果 html代码 <div ...

  7. 用JavaScript写一个简单的倒计时,可以应用在发送短信验证码的“59秒后重新发送验证短信”

    倒计时——从10倒数到0,点击按钮会还原倒计时 <body> <!-- 将textvalue值设为10,从10倒数 --> <input type="text& ...

  8. 原生JavaScript实现一个简单的todo-list

    直接上代码: <!DOCTYPE html> <html lang="en"> <head> <meta charset="UT ...

  9. JavaScript实现一个简单的密码输入功能

    常见的密码输入框当输入字符后会被替换成‘*’,而且旁边会有个小眼睛可以查看原本的字符,虽然input标签有这个功能,但这只是自己正在看正则表达式的时候突然想到的,就当做个练习,自己手动实现下: < ...

随机推荐

  1. jmeter测试报告优化

    1.下载jmeter.results.shanhe.me.xsl 将该文件拷贝到jmeter\extras目录下 2.修改jmeter.results.shanhe.me.xsl 这里直接拷贝 jme ...

  2. 封装一个ExcelHelper,方便将Excel直接转成Datatable对象

    public class ExcelHelper { /// <summary> /// Excel转换成DataTable /// </summary> /// <pa ...

  3. 在spring+beranate中多数据源中使用 ThreadLocal ,总结的原理 --费元星

    设计模式 首先,ThreadLocal 不是用来解决共享对象的多线程访问问题的,一般情况下,通过ThreadLocal.set() 到线程中的对象是该线程自己使用的对象,其他线程是不需要访问的,也访问 ...

  4. 虚拟接VMnet1 和VMnet8的区别

    vmnet1是host-only,也就是说,选择用vmnet1的话就相当于VMware给你提供了一个虚拟交换机,仅将虚拟机和真实系统连上了,虚拟机可以与真实系统相互共享文件,但是虚拟机无法访问外部互联 ...

  5. SVN 使用时的小错误

    在使用SVN的时候总是出现一些小问题,今天又出现了一个,诶,分享一下吧!  Error:(个人文件夹名http://www.qdjhu.com/anli_xq/f_wancheng.php)  is ...

  6. 不同浏览器css引入外部字体的方式

    /** * 字体后缀和浏览器有关,如下所示 * .TTF或.OTF,适用于Firefox 3.5.Safari.Opera * .EOT,适用于Internet Explorer 4.0+ * .SV ...

  7. 适用于Linux的windows子系统

    Windows基于图形界面的易用性是有目共睹的,这也是很多普通用户往往难以舍弃的原因.但是Linux系统更强大的网络应用开发能力,却又是Windows系统所无法比拟的.一直以来,很多人都在试图采用各种 ...

  8. Windows下nginx作为静态资源服务器使用

    一.Nginx下载与安装 1.nginx官方下载地址:http://nginx.org/ 2.下载完后将压缩包解压即可 3.nginx配置文件为根目录下conf\nginx.conf 二.Nginx常 ...

  9. Vue学习(三):数据绑定语法

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  10. Linux-Shell脚本编程-学习-4-Shell编程-操作数字-加减乘除计算

    对于任何一种编程语言都很重要的特性就是操作数字的能力,遗憾的是,对于shell脚本来说,这个过程比较麻烦,在shell脚本中有两种途径来进行数学运算操作. 1.expr 最开始的时候,shell提供了 ...