看到一款树形结构,比较喜欢它的样式,就参照它的外观自己做了一个,练习一下CSS。

做出来的效果如下:

li { position: relative; padding: 5px 0; margin:0; }
#tree-div.tree>ul { padding: 0; margin: 0 }
.tree>ul ul>li::after { content: " "; position: absolute; top: 20px; left: -45px; width: 45px; border: none; border-top: 1px solid #ddd }
.tree ul>li:not(:last-child)::before { content: " "; position: absolute; top: 0; left: -45px; height: 100%; border: none; border-left: 1px solid #ddd }
.tree ul>li:last-child::before { content: " "; position: absolute; top: 0; left: -45px; height: 20px; border: none; border-left: 1px solid #ddd }
.tree span:hover,.tree span:hover+ul span { color: #fff; background-color: orange }
.tree span:hover,.tree span:hover+ul span,.tree span:hover+ul li::before,.tree span:hover+ul li::after { border-color: orange }
.tree .fa::before { margin-right: 5px }
.tree .fa-minus-circle,.tree .fa-plus-circle { cursor: pointer }
-->

  • 拉莫小学

    • 一年级

      • 一班
      • 二班
    • 二年级
    • 三年级
      • 一班
      • 二班
      • 三班

树的dom结构:

<div class="tree">
<ul>
<li>
<span><i class="fa fa-minus-circle"></i>拉莫小学</span>
<ul>
<li>
<span><i class="fa fa-minus-circle"></i>一年级</span>
<ul>
<li><span>一班</span></li><li><span>二班</span></li>
</ul>
</li>
<li>
<span>二年级</span>
</li>
<li>
<span><i class="fa fa-minus-circle"></i>三年级</span>
<ul>
<li><span>一班</span></li>
<li><span>二班</span></li>
<li><span>三班</span></li>
</ul>
</li>
</ul>
</li>
</ul>
</div>

CSS代码:

/** tree.css zyj 2018.4.21 */
ul,li{list-style-type:none;}
.tree{display:block;position:relative;padding:5px 15px;}
.tree span{display:inline-block;box-sizing:border-box;height:30px;line-height:28px;min-width:60px;text-align:center;color:#888;border:1px solid #ddd;border-radius:5px;padding:0 8px;}
.tree ul{position:relative;padding-left:60px;margin:;}
.tree ul>li{position:relative;padding:5px 0;}
.tree>ul{padding:;margin:;}
/** 水平方向连线 */
.tree>ul ul>li:after{content:' ';position:absolute;top:20px;left:-45px;width:45px;border:none;border-top:1px solid #ddd;}
/** 垂直方向连线 */
.tree ul>li:not(:last-child):before{content:' ';position:absolute;top:;left:-45px;height:100%;border:none;border-left:1px solid #ddd;}
.tree ul>li:last-child:before{content:' ';position:absolute;top:;left:-45px;height:20px;border:none;border-left:1px solid #ddd;}
/** 控制鼠标移上去的颜色 */
.tree span:hover, .tree span:hover+ul span{color:#fff;background-color:orange;}
.tree span:hover, .tree span:hover+ul span, .tree span:hover+ul li:before, .tree span:hover+ul li:after{border-color:orange;}
/** 折叠图标 */
.tree .fa:before{margin-right:5px;}
.tree .fa-minus-circle, .tree .fa-plus-circle{cursor:pointer;}

里面引的fontawesome图标没法加载进来,导致折叠按钮显示不出,下面是原始树状图的截图:

数据是我用JS加载的,写了个加载数据的tree.js文件,源码如下:

/** tree.js zyj 2018.4.22 */
(function(name){
var tree, outer, defaultDateFormat; outer = {
setData : setData,
}; defaultDateFormat = {
unfold : true,
name : 'name',
childName : 'children'
}; function getDataFormat(dataFormat){
var index;
if(!dataFormat){
return defaultDateFormat;
}
for(index in defaultDateFormat){
dataFormat[index] = typeof dataFormat[index] == 'undefined'? defaultDateFormat[index] : dataFormat[index];
}
return dataFormat
} function initTreeJs(name){
var tree;
if(checkTreeNameUsed(name)){return;}
window[name] = outer;
initFoldIcon($('.tree'));
} function checkTreeNameUsed(name){
if(window[name]){
console.error("The window object name [" + name + "] has been used, tree.js can't be loaded! You can try another name." );
return true;
}
return false;
} function initFoldIcon(target){
target.off('click', 'span>i.fa').on('click', 'span>i.fa', function(e){
var ele = $(e.target);
if(ele.hasClass('fa-minus-circle')){
ele.removeClass('fa-minus-circle').addClass('fa-plus-circle').parent().next('ul').hide(200);
}else if(ele.hasClass('fa-plus-circle')){
ele.removeClass('fa-plus-circle').addClass('fa-minus-circle').parent().next('ul').show(200);
}
})
} function getJqueryObjectBySelector(selector){
var ele = $(selector);
if(typeof selector != 'string'){
console.error("The first parameter jquery selector [" + selector + "] must be a string!" );
return;
}
if(!ele.hasClass('tree')){
ele = ele.find('.tree');
}
if(ele.length != 1){
console.error("The selector [" + selector + "] expect only one element!" );
return;
}
return ele;
} function setData(selector, data, dataFormat){
var ele = getJqueryObjectBySelector(selector);
if(!ele){return;}
if(!data){return;}
if(!data.length){
data = [data];
}
dataFormat = getDataFormat(dataFormat);
dataFormat.topElement = true;
ele.empty().append(getTreeList(data, dataFormat));
initFoldIcon(ele);
} function getTreeList(data, dataFormat){
var i, single, name, children, childDataFormat,
array = [];
childDataFormat = dataFormat.child || dataFormat;
if(dataFormat.unfold){
array.push('<ul>');
}else if(dataFormat.topElement){
dataFormat.topElement = false;
array.push('<ul>');
}else{
array.push('<ul style="display:none;">');
}
for(i=0; i<data.length; i++){
single = data[i];
if(typeof dataFormat.name == 'function'){
name = dataFormat.name(single);
}else if(typeof dataFormat.name == 'string'){
name = single[dataFormat.name];
}else{
name = single['name'];
}
if(typeof dataFormat.childName == 'string'){
children = single[dataFormat.childName];
}else{
children = single['children'];
}
array.push('<li>');
array.push('<span>');
if(children && children.length > 0){
if(dataFormat.unfold){
array.push('<i class="fa fa-minus-circle"></i>');
}else{
array.push('<i class="fa fa-plus-circle"></i>');
}
array.push(name);
array.push('</span>');
array.push(getTreeList(children, childDataFormat));
}else{
array.push(name);
array.push('</span>');
}
array.push('</li>');
}
array.push('</ul>');
return array.join('');
} initTreeJs(name);
}('tree'))

偷懒没写注释,tree.js中目前只写了一个对外的接口 tree.setData(selector, data, dataFormat) 。参数selector是jQuery选择器,data是数据,dataFormat是数据格式。

比如加载上图的数据:

var dataTest = {
name:'拉莫小学',
children:[
{
name:'一年级',
children:[
{name:'一班'},
{name:'二班'}
]
},
{
name:'二年级'
},
{
name:'三年级',
children:[
{name:'一班'},
{name:'二班'},
{name:'三班'}
]
}
]
}; tree.setData('.tree', dataTest);

由于后台加载的数据不一定是按照{name:'*', children:[{name:'*'},...]}这种结构,所以留了dataFormat参数,自己去定义数据格式。

简单举个例子,假如后台数据格式是

var data =
{
id : '1',
title : '百度',
url : 'http://www.baidu.com',
subWeb :
[
{
id : '2',
title : '百度新闻',
url : 'http://news.baidu.com'
},
{
id : '3',
title : '百度知道',
url : 'http://zhidao.baidu.com'
},
{
id : '4',
title : '百度图片',
url : 'http://image.baidu.com'
},
]
}

那么dataFormat可以定义为

var dataFormat =
{
name : function(data){
return '<a href="' + data.url + '">' + data.title + '</a>';
},
childName : 'subWeb'
}

至于效果,读者自己去试咯。

CSS实现树形结构 + js加载数据的更多相关文章

  1. css 动态导入css文件 @import 动态js加载 都是静态的

    @import "http://apps.bdimg.com/libs/bootstrap/3.3.4/css/bootstrap.css" /*-防止各大cdn公共库加载地址失效 ...

  2. iScroll.js 向上滑动异步加载数据回弹问题

    iScroll是一款用于移动设备web开发的一款插件.像缩放.下拉刷新.滑动切换等移动应用上常见的一些效果都可以轻松实现. 现在最新版本是5.X,官网这里:http://iscrolljs.com/ ...

  3. js中对arry数组的各种操作小结 瀑布流AJAX无刷新加载数据列表--当页面滚动到Id时再继续加载数据 web前端url传递值 js加密解密 HTML中让表单input等文本框为只读不可编辑的方法 js监听用户的键盘敲击事件,兼容各大主流浏览器 HTML特殊字符

    js中对arry数组的各种操作小结   最近工作比较轻松,于是就花时间从头到尾的对js进行了详细的学习和复习,在看书的过程中,发现自己平时在做项目的过程中有很多地方想得不过全面,写的不够合理,所以说啊 ...

  4. 关于使用Iscroll.js异步加载数据后不能滑动到最底端的问题解决方案

    关于使用Iscroll.js异步加载数据后不能滑动到最底端,拉到最下边又弹回去的问题困扰了我老半天,相信很多朋友都遇到了.我刚好不小心解决了,和大家分享一下.由于各种忙,下边就直接上代码吧. (前提是 ...

  5. 定时器详解和应用、js加载阻塞、css加载阻塞

    1.setTimeout().setInterval()详解和应用 1.1 详解: setTimeout.setInterval执行时机 1.2 存在问题: setInterval重复定时器可能存在的 ...

  6. FusionCharts简单教程(二)-----使用js加载图像和setDataXML()加载数据

          前面一篇对FusionCharts进行了一个简单的介绍,而且建立了我们第一个图形,但是那个是在HTML中使用<OBJECT>和<EMBED>标记来加载图形的,但是这 ...

  7. [JS前端开发] js/jquery控制页面动态加载数据 滑动滚动条自动加载事件

    页面滚动动态加载数据,页面下拉自动加载内容 相信很多人都见过瀑布流图片布局,那些图片是动态加载出来的,效果很好,对服务器的压力相对来说也小了很多 有手机的相信都见过这样的效果:进入qq空间,向下拉动空 ...

  8. 使用js加载图像和setDataXML()加载数据

    使用js加载图像和setDataXML()加载数据 前面一篇对FusionCharts进行了一个简单的介绍,而且建立了我们第一个图形,但是那个是在HTML中使用<OBJECT>和<E ...

  9. 新手教程:不写JS,在MIP页中实现异步加载数据

    从需求谈起:在 MIP 页中异步加载数据 MIP(移动网页加速器) 的 加速原理 除了靠谱的 MIP-Cache CDN 加速外,最值得一提的就是组件系统.所有 JS 交互都需要使用 MIP 组件实现 ...

随机推荐

  1. 浅谈EditText控件的inputType类型

    android:inputType="none"--默认 android:inputType="text"--输入文本字符 android:inputType= ...

  2. NoHttp封装--08 用一个实体类接收所有接口数据

    1.用户信息获取--bean实体类形式返回数据 ①服务器端: 代码: protected void onHandler(HttpServletRequest request, HttpServletR ...

  3. git 入门教程之本地和远程仓库的本质

    本地仓库和远程仓库在本质上没有太大区别,只不过一个是本地电脑,一个是远程电脑. 远程仓库不一定非得是 github 那种专门的"中央服务器",甚至局域网的另外一台电脑也可以充当&q ...

  4. Django 2.0 URL新版配置介绍

    实例 先看一个例子: from django.urls import path from . import views urlpatterns = [ path('articles/2003/', v ...

  5. MSSQL sql server order by 1,2 的具体含义

    转自:http://www.maomao365.com/?p=5416 摘要: order by 1,2 的含义是对表的第一列  按照从小到大的顺序进行排列 然后再对第二列按照从小到大的顺序进行排列 ...

  6. scrapy中pipeline的一点综合知识

    初次学习scrapy ,觉得spider代码才是最重要的,越往后学,发现pipeline中的代码也很有趣, 今天顺便把pipeline中三种储存方法写下来,算是对自己学习的一点鼓励吧,也可以为后来者的 ...

  7. 登录Windows界面前执行自定义脚本

    通常情况下,进入Windows界面之前都有一个登录过程,如何在登录前让系统执行脚本呢?下面介绍一种方法. 1.打开组策略,在Run(运行)中输入GREDIT.MSC,点击确认. 2.依次点击Compu ...

  8. Linux 小知识翻译 - 「邮件服务器」

    这次聊聊「邮件服务器」. 邮件服务器上通常会运行2个服务端软件,「SMTP服务器」和「POP服务器或者IMAP服务器」. 这2个东西,也许使用邮件客户端的人立马就明白了.因为设置邮件客户端的时候,需要 ...

  9. SecureCRT8.1+SecureCRT_keygen完成注册

    原文:https://www.cnblogs.com/qiyawei/p/7822957.html 1.下载完secureCRT8.1之后,进行安装. 点击secureCRT.exe的时候会出现如下图 ...

  10. (转)Spring Boot(十七):使用 Spring Boot 上传文件

    http://www.ityouknow.com/springboot/2018/01/12/spring-boot-upload-file.html 上传文件是互联网中常常应用的场景之一,最典型的情 ...