基于Web标准的UI组件 — 树状菜单(2)


Posted in Javascript onSeptember 18, 2006

从这篇开始,你需要拥有一些Javascript和DOM相关的知识才能顺利地学习下去。由于Javascript和DOM都不是三言两语可以说完的东西,如果你对它们还不熟悉,请先到这里学习一下再继续:Javascript在线教程(英文)、DOM在线教程(英文)。

getElementsByClassName()

为了从一大堆HTML代码中找出我们的树状菜单(也许有多个),我们先来实现一个通过className找DOM节点的方法:getElementsByClassName。这是对浏览器自有DOM方法的一个简单但实用的扩充。

此方法有两个参数:ele指出以哪个DOM节点为根节点寻找(也就是说只找ele的子节点),className指出符合条件的节点的class属性中必须包含怎样的className。它的返回值是一个数组,存放了所有符合条件的节点。

function getElementsByClassName(ele,className) {
 //获取所有子节点
 if(document.all){
  var children = ele.all;
 }else{
  var children = ele.getElementsByTagName('*');
 }
 //遍历子节点并检查className属性
 var elements = new Array();
 for (var i = 0; i < children.length; i++) {
  var child = children[i];
  var classNames = child.className.split(' ');
  for (var j = 0; j < classNames.length; j++) {
   if (classNames[j] == className) {
    elements[elements.length] = child;
    break;
   }
  }
 }
 return elements;
}

最前面的一个if-else语是为了兼容IE5(IE5不能运行document.getElementsByTagName('*'))。需要注意的是千万不要用浏览器检测的方法来写脚本,而应该直接使用将要用到的语句来测试是否可以执行,如果返回值为nullundefined,那再换一种方法。这样的脚本可以有更好的兼容性,也更健壮。

elements[elements.length] = child;,这句同样是为了兼容IE5才没有使用数组的push方法。如果你一定要使用push方法,那么可以在执行getElementsByClassName()之前先重载一遍push方法。代码如下:

Array.prototype.push = function(value){
 this[this.length] = value;
}

注:原本我希望getElementsByClassName也能像push方法一样写,比如HTMLElement.prototype.getElementsByClassName = ...。不过实际操作的时候发现在运行时HTMLElement这个对象并不是固定的,每种tag似乎都不一样,只能作罢。如果你有解决方案请告诉我,谢谢。

现在我们就可以方便地找出页面上所有的树状菜单了:

var trees	= getElementsByClassName(document,'TreeView');
for(var i=0;i<trees.length;i++){
 alert('我是第' + i + '个树状菜单');
 //在这里可以更细致地处理每一个树状菜单
}

最后把上面这几句加到window.onload事件中运行,以便文档一加载完就对树状菜单进行初始化。完整的代码请查看下面例子的源代码。

查看效果(例1)

区分树枝与树叶

在上一篇中我们说到树枝和树叶的区别就是这个节点有没有子节点,所以判断树枝和树叶的方法也可以从这个角度来考虑。一个比较直观的方法就是遍历整个树状菜单的DOM树(注意这里两个“树”的区别),看看每个节点是不是拥有子节点,如果有的话我们就给这个节点一个专门的class以示区分。我们这里用一种比较取巧的方法,就是判断各个节点的innerHTML中有没有出现字符串'<ul>'。如果有的话,那么很显然它拥有一个或多个子节点。

var trees	= getElementsByClassName(document,'TreeView');
for(var i=0;i<trees.length;i++){
 //先把所有的节点找出来
 var nodes = trees[i].getElementsByTagName('LI');
 //判断各个节点是否有子节点
 for(var j=0;j<nodes.length;j++){
  if(nodes[j].innerHTML.toLowerCase().indexOf('<ul>') > -1)
   nodes[j].className += 'Open';
 }
}

这里给每个树枝加了一个className:Open,因为我们现在还不能打开关闭树枝,所以只要是树枝那就是open的。当然后面我们会用到Close的:)。相应的修改一下CSS,给树枝一个带减号的图标,表示它是打开的:

.TreeView li.Open{
 background:transparent url(opened.gif) 12px 2px no-repeat;
}

查看效果(例2)

高亮选中项

接下来实现把当前选中的树枝(或树叶)高亮的功能。有两个时候需要高亮:菜单初始化的时候和点击某个菜单项的时候。

初始化的时候比较容易处理,直接给需要高亮的节点一个特殊的Class即可,比如“Selected”:

.TreeView li.Selected a:link,
.TreeView li.Selected a:visited,
.TreeView li.Selected a:hover,
.TreeView li.Selected a:active{
 background-color:#05F;
 color:#FFF;
 text-decoration:none;/*去除下划线*/
 cursor:default;/*让光标变为普通箭头,假装是不能点的^_^*/
 padding:0 2px;/*为了美观考虑,也可以不要这句*/
}

查看效果(例3)

这里有几点可能还需要补充说明一下:

  1. 选择器(Selector)的前面为什么要加上.TreeView,这不是冗余代码吗?
    在这个例子中确实是冗余代码,但在实际项目中,一个页面上可能会有各种不同的组件,比如还有一个菜单,被选中的菜单项也用.Selected来表示。这时就需要在选择符的前面先指出是什么组件的选中项以防冲突。当然还有其他的解决办法,比如这里的类不取名为Selectd,改为TreeSelected或者其他什么的,但是这样做人为的把命名方案复杂化了,我个人不推荐这样做。
  2. 选择器为什么分作四行来写?
    因为我们之前已经设置过a的样式,为了提高优先级重载旧的样式,所以需要指定a的四种伪状态(还有其他提高优先级的办法,关于优先级算法,在《网站重构》一书中有详细说明)。
  3. Selected为什么要用在li上,而不直接用在a上?
    这又是一个不太容易说明白的地方,因为很大程度上它是一种个人习惯,只是我个人觉得这样做更合适一些。事实上,写在li上或a上都是可以的,至少看上去(表现层的视角)不会有太大的区别,但是如果你从“表现层”中跳出来,站在“结构层”的视角来看,无论这个菜单的树结构还是DOM结构,一个节点都是由一个li来表达的,a只不过是这个节点内的更细节的部分。虽然我最终是希望给a指定一个特殊的样式(为什么不指定给li?你可以自己试一下),但从XHTML结构来说,这个class="Selected"还是写在li上更合适。(上帝保佑我说清楚了……)

下一篇讲什么?

这篇文章是我第一次加入Javascript内容,不是很清楚是说浅了还是说深了,请大家在右边留言告诉我你的想法。从下一篇开始,我们开始进入部署鼠标事件和响应鼠标事件方面的内容。也许从下下篇开始再加入一些Javascript面向对象编程的内容,待定待定……hehe^_^

Javascript 相关文章推荐
超级简单的图片防盗(HTML),好用
Apr 08 Javascript
location.href 在IE6中不跳转的解决方法与推荐使用代码
Jul 08 Javascript
原生js ActiveXObject获取execl里面的值
Nov 01 Javascript
jQuery避免$符和其他JS库冲突的方法对比
Feb 20 Javascript
javascript常用函数(2)
Nov 05 Javascript
学习javascript文件加载优化
Feb 19 Javascript
基于jquery实现弹幕效果
Sep 29 Javascript
vue2.0开发实践总结之疑难篇
Dec 07 Javascript
在javascript中,null>=0 为真,null==0却为假,null的值详解
Feb 22 Javascript
vue 2.0组件与v-model详解
Mar 27 Javascript
vue 简单自动补全的输入框的示例
Mar 12 Javascript
微信小程序HTTP请求从0到1封装
Sep 09 Javascript
JavaScript中的私有成员
Sep 18 #Javascript
javascript的事件描述
Sep 08 #Javascript
由浅到深了解JavaScript类
Sep 08 #Javascript
js常用函数 不错
Sep 08 #Javascript
Javascript 不能释放内存.
Sep 07 #Javascript
一些有关检查数据的JS代码
Sep 07 #Javascript
Mozilla中显示textarea中选择的文字
Sep 07 #Javascript
You might like
ajax php 实现写入数据库
2009/09/02 PHP
php下获取客户端ip地址的函数
2010/03/15 PHP
php统计文件大小,以GB、MB、KB、B输出
2011/05/29 PHP
php生成二维码
2015/08/10 PHP
验证坐标在某坐标区域内php代码
2016/10/08 PHP
完美解决在ThinkPHP控制器中命名空间的问题
2017/05/05 PHP
document.open() 与 document.write()的区别
2007/08/13 Javascript
jquery tools之tabs 选项卡/页签
2009/07/25 Javascript
一个简单的实现下拉框多选的插件可移植性比较好
2014/05/05 Javascript
JS实现网页游戏中滑块响应鼠标点击移动效果
2015/10/19 Javascript
Highcharts使用简例及异步动态读取数据
2015/12/30 Javascript
js中window.open的参数及注意注意事项
2016/07/06 Javascript
关于input全选反选恶心的异常情况
2016/07/24 Javascript
vue2.0安装style/css loader的方法
2018/03/14 Javascript
jquery登录的异步验证操作示例
2019/05/09 jQuery
小程序实现长按保存图片的方法
2019/12/31 Javascript
JavaScript实现简单日历效果
2020/09/11 Javascript
vue实现标签云效果的示例
2020/11/09 Javascript
使用python中的in ,not in来检查元素是不是在列表中的方法
2018/07/06 Python
详解Python中正则匹配TAB及空格的小技巧
2019/07/26 Python
基于Django框架的权限组件rbac实例讲解
2019/08/31 Python
Python实现线性插值和三次样条插值的示例代码
2019/11/13 Python
使用pandas库对csv文件进行筛选保存
2020/05/25 Python
python的help函数如何使用
2020/06/11 Python
沙特阿拉伯网上购物:Sayidaty Mall
2018/05/06 全球购物
国际商务专业学生个人的自我评价
2013/09/28 职场文书
同事打架检讨书
2014/02/04 职场文书
学习十八届三中全会精神实施方案
2014/02/17 职场文书
健康家庭事迹材料
2014/05/02 职场文书
2014年银行员工年终自我评价
2014/09/19 职场文书
2014年招生工作总结
2014/11/26 职场文书
校本培训个人总结
2015/02/28 职场文书
高三毕业感言
2015/07/30 职场文书
OpenCV实现普通阈值
2021/11/17 Java/Android
windows11怎么查看wifi密码? win11查看wifi密码的技巧
2021/11/21 数码科技
Java工作中实用的代码优化技巧分享
2022/04/21 Java/Android