


UI components based on web standards - tree menu (2)_javascript skills
从这篇开始,你需要拥有一些Javascript和DOM相关的知识才能顺利地学习下去。由于Javascript和DOM都不是三言两语可以说完的东西,如果你对它们还不熟悉,请先到这里学习一下再继续:Javascript在线教程(英文)、DOM在线教程(英文)。
getElementsByClassName()
为了从一大堆HTML代码中找出我们的树状菜单(也许有多个),我们先来实现一个通过className
找DOM节点的方法:getElementsByClassName
。这是对浏览器自有DOM方法的一个简单但实用的扩充。
此方法有两个参数:ele
指出以哪个DOM节点为根节点寻找(也就是说只找ele
的子节点),className
指出符合条件的节点的class属性中必须包含怎样的className
。它的返回值是一个数组,存放了所有符合条件的节点。
<code>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; }</CODE>
最前面的一个if-else语是为了兼容IE5(IE5不能运行document.getElementsByTagName('*')
)。需要注意的是千万不要用浏览器检测的方法来写脚本,而应该直接使用将要用到的语句来测试是否可以执行,如果返回值为null
或undefined
,那再换一种方法。这样的脚本可以有更好的兼容性,也更健壮。
elements[elements.length] = child;
,这句同样是为了兼容IE5才没有使用数组的push
方法。如果你一定要使用push
方法,那么可以在执行getElementsByClassName()
之前先重载一遍push
方法。代码如下:
<CODE>Array.prototype.push = function(value){ this[this.length] = value; }</CODE>
注:原本我希望getElementsByClassName
也能像push
方法一样写,比如HTMLElement.prototype.getElementsByClassName = ...
。不过实际操作的时候发现在运行时HTMLElement
这个对象并不是固定的,每种tag似乎都不一样,只能作罢。如果你有解决方案请告诉我,谢谢。
现在我们就可以方便地找出页面上所有的树状菜单了:
<CODE>var trees = getElementsByClassName(document,'TreeView'); for(var i=0;i<trees.length;i++){ alert('我是第' + i + '个树状菜单'); //在这里可以更细致地处理每一个树状菜单 }</CODE>
最后把上面这几句加到window.onload
事件中运行,以便文档一加载完就对树状菜单进行初始化。完整的代码请查看下面例子的源代码。
区分树枝与树叶
在上一篇中我们说到树枝和树叶的区别就是这个节点有没有子节点,所以判断树枝和树叶的方法也可以从这个角度来考虑。一个比较直观的方法就是遍历整个树状菜单的DOM树(注意这里两个“树”的区别),看看每个节点是不是拥有子节点,如果有的话我们就给这个节点一个专门的class以示区分。我们这里用一种比较取巧的方法,就是判断各个节点的innerHTML中有没有出现字符串'<ul>'
。如果有的话,那么很显然它拥有一个或多个子节点。
<code>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'; } } </code>
这里给每个树枝加了一个className:Open
,因为我们现在还不能打开关闭树枝,所以只要是树枝那就是open的。当然后面我们会用到Close的:)。相应的修改一下CSS,给树枝一个带减号的图标,表示它是打开的:
<code>.TreeView li.Open{ background:transparent url(opened.gif) 12px 2px no-repeat; }</code>
高亮选中项
接下来实现把当前选中的树枝(或树叶)高亮的功能。有两个时候需要高亮:菜单初始化的时候和点击某个菜单项的时候。
初始化的时候比较容易处理,直接给需要高亮的节点一个特殊的Class即可,比如“Selected
”:
<code>.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;/*为了美观考虑,也可以不要这句*/ }</code>
这里有几点可能还需要补充说明一下:
-
Why do we need to add
- in front of the selector? Isn’t this redundant code?
.TreeView
is indeed redundant code in this example, but in actual projects, there may be various components on a page, such as a menu, and the selected menu item is also represented by
. At this time, you need to indicate what component is selected in front of the selector to prevent conflicts. Of course there are other solutions, such as not naming the class here Selectd, but changing it to TreeSelected or something else, but this artificially complicates the naming scheme, and I personally do not recommend this..Selected
Why is the selector written in four lines? - Because we have set the style of
before, in order to overload the old style to improve the priority, we need to specify the four pseudo states ofa
(there are other ways to improve the priority, about the priority algorithm , described in detail in the book "a
Website Reconstruction"). Why should - be used on
Selected
instead of directly onli
?a
This is another place that is not easy to explain, because to a large extent it is a personal habit, but I personally feel that this is more appropriate. In fact, it’s okay to write on
orli
, at least it won’t look much different (from the perspective of the presentation layer), but if you jump out of the “presentation layer” and stand on From the perspective of the "structural layer", regardless of the tree structure or DOM structure of this menu, a node is expressed by aa
, andli
is just a more detailed part of this node. Although I ultimately hope to assign a special style toa
(why not assign it toa
? You can try it yourself), in terms of XHTML structure, it is more appropriate to write thisli
onclass="Selected"
. (God forbid I made it clear...)li
This article is the first time I have added Javascript content. It is not clear whether it is shallow or deep. Please leave a message on the right to tell me your thoughts. Starting from the next article, we start to get into the content of deploying mouse events and responding to mouse events. Maybe I will add some Javascript object-oriented programming content from the next article, to be determined... hehe^_^

Hot AI Tools

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Undress AI Tool
Undress images for free

Clothoff.io
AI clothes remover

AI Hentai Generator
Generate AI Hentai for free.

Hot Article

Hot Tools

Notepad++7.3.1
Easy-to-use and free code editor

SublimeText3 Chinese version
Chinese version, very easy to use

Zend Studio 13.0.1
Powerful PHP integrated development environment

Dreamweaver CS6
Visual web development tools

SublimeText3 Mac version
God-level code editing software (SublimeText3)

Hot Topics

Detailed explanation of JavaScript string replacement method and FAQ This article will explore two ways to replace string characters in JavaScript: internal JavaScript code and internal HTML for web pages. Replace string inside JavaScript code The most direct way is to use the replace() method: str = str.replace("find","replace"); This method replaces only the first match. To replace all matches, use a regular expression and add the global flag g: str = str.replace(/fi

Leverage jQuery for Effortless Web Page Layouts: 8 Essential Plugins jQuery simplifies web page layout significantly. This article highlights eight powerful jQuery plugins that streamline the process, particularly useful for manual website creation

So here you are, ready to learn all about this thing called AJAX. But, what exactly is it? The term AJAX refers to a loose grouping of technologies that are used to create dynamic, interactive web content. The term AJAX, originally coined by Jesse J

This post compiles helpful cheat sheets, reference guides, quick recipes, and code snippets for Android, Blackberry, and iPhone app development. No developer should be without them! Touch Gesture Reference Guide (PDF) A valuable resource for desig

jQuery is a great JavaScript framework. However, as with any library, sometimes it’s necessary to get under the hood to discover what’s going on. Perhaps it’s because you’re tracing a bug or are just curious about how jQuery achieves a particular UI

10 fun jQuery game plugins to make your website more attractive and enhance user stickiness! While Flash is still the best software for developing casual web games, jQuery can also create surprising effects, and while not comparable to pure action Flash games, in some cases you can also have unexpected fun in your browser. jQuery tic toe game The "Hello world" of game programming now has a jQuery version. Source code jQuery Crazy Word Composition Game This is a fill-in-the-blank game, and it can produce some weird results due to not knowing the context of the word. Source code jQuery mine sweeping game

Article discusses creating, publishing, and maintaining JavaScript libraries, focusing on planning, development, testing, documentation, and promotion strategies.

This tutorial demonstrates how to create a captivating parallax background effect using jQuery. We'll build a header banner with layered images that create a stunning visual depth. The updated plugin works with jQuery 1.6.4 and later. Download the
