花了半天时间实现一个多级联动下拉框,目的是对某一植物进行“门纲目科属”的归类。使用的技术是javascript+xml,之所以不用数据库,一来这方面的数据虽然量大但都是固定不变的,二来不希望加重服务器的负担,第三是因为这种多级从属关系的数据不太适合放在数据库里。
这是大概的思路:
1、读取xml文件
2、当一个下拉框选中某选项时,根据该选项,当前节点指向下一层,进入下一层下拉框的设置
3、取消当前下拉框的禁用,禁用下一层的下拉框
4、清空当前下拉框的选项
5、根据当前节点读取xml的数据,设置下拉框选项
6、返回步骤2
代码:
JavaScript
var xmlDoc;
var browserType;
var currentNode;//当前所在节点
setBrowserType();
loadXml("classify.xml");
//读取xml文件数据并设置门、纲、目、科、属的下拉框
//设置“门”的下拉框
function setPhylum(){
currentNode=xmlDoc.documentElement;
var phylums=xmlDoc.documentElement.childNodes;
var phylumName;
if(browserType=="IE"){
for(var i=0;i<phylums.length;i++){
//从门到属,都有name属性标签,并且所有下拉框选项索引都是从1开始
phylumName=phylums[i].selectNodes("name")[0].text;
document.forms[0].phylum.options[i+1]=new Option(phylumName,phylumName);
}
}
else{//FF
//FireFox没有selectNodes()方法,且其childNodes的对应索引是1,3,5,7...
for(var i=1;i<phylums.length;i=i+2){
phylumName=phylums[i].childNodes[1].textContent;
document.forms[0].phylum.options[(i+1)/2]=new Option(phylumName,phylumName);
}
document.forms[0].clazz.disabled="disabled";
document.forms[0].order.disabled="disabled";
document.forms[0].family.disabled="disabled";
document.forms[0].genus.disabled="disabled";
}
}
//设置“纲”的下拉框
function setClazz(selectedIndex){
//取消下拉框的禁用
//后面的下拉框禁用,这是因应各下拉框的无序选择可能产生的错误
//比如选了“科”又回头重新选“目”,或更改同一个下拉框选项)
document.forms[0].clazz.disabled=null;
document.forms[0].order.disabled="disabled";
document.forms[0].family.disabled="disabled";
document.forms[0].genus.disabled="disabled";
clearOption(document.forms[0].clazz);
var clazzes;
var clazzName;
//将选中的门节点作为当前节点,注意这里需要将索引回减1
//因为门的父节点没有name属性标签,而下拉框的索引又是从1开始
//currentNode的赋值应使用绝对定位,也是因应各下拉框的无序选择
//currentNode=currentNode.childNodes(selectedIndex-1);
if(browserType=="IE"){
currentNode=xmlDoc.documentElement.childNodes(selectedIndex-1);
clazzes=currentNode.childNodes;
//因为门节点的第一个子节点为name属性标签,故循环时索引从1开始
//相应的下拉框的索引就与纲节点的索引同步(不需要options[i+1]),目、科、属也是一样
for(var i=1;i<clazzes.length;i++){
clazzName=clazzes[i].selectNodes("name")[0].text;
document.forms[0].clazz.options[i]=new Option(clazzName,clazzName);
}
}
else{//FF
currentNode=xmlDoc.documentElement.childNodes[selectedIndex*2-1];
clazzes=currentNode.childNodes;
for(var i=1;i<clazzes.length-2;i=i+2){
clazzName=clazzes[i+2].childNodes[1].textContent;
document.forms[0].clazz.options[(i+1)/2]=new Option(clazzName,clazzName);
}
}
}
//设置“目”的下拉框
function setOrder(selectedIndex){
//取消下拉框的禁用
//后面的下拉框禁用,这是因应各下拉框的无序选择可能产生的错误(比如选了“科”又回头重新选“目”)
document.forms[0].order.disabled=null;
document.forms[0].family.disabled="disabled";
document.forms[0].genus.disabled="disabled";
clearOption(document.forms[0].order);
var orderName;
//currentNode的赋值应使用绝对定位
var phylumSI=document.forms[0].phylum.selectedIndex; //phylum selected index
if(browserType=="IE"){
currentNode=xmlDoc.documentElement
.childNodes[phylumSI-1]
.childNodes[selectedIndex];
var orders=currentNode.childNodes;
for(var i=1;i<orders.length;i++){
orderName=orders[i].selectNodes("name")[0].text;
document.forms[0].order.options[i]=new Option(orderName,orderName);
}
}else{
currentNode=xmlDoc.documentElement
.childNodes[phylumSI*2-1]
.childNodes[selectedIndex*2+1];
var orders=currentNode.childNodes;
for(var i=1;i<orders.length-2;i=i+2){
orderName=orders[i+2].childNodes[1].textContent;
document.forms[0].order.options[(i+1)/2]=new Option(orderName,orderName);
}
}
}
//设置“科”的下拉框
function setFamily(selectedIndex){
document.forms[0].family.disabled=null;//取消下拉框的禁用
document.forms[0].genus.disabled="disabled";//后面的下拉框禁用
//currentNode的赋值应使用绝对定位
var phylumSI=document.forms[0].phylum.selectedIndex;//phylum selected index
var clazzSI=document.forms[0].clazz.selectedIndex; //clazz selected index
clearOption(document.forms[0].family);
var families;
var familyName;
if(browserType=="IE"){
currentNode=xmlDoc.documentElement
.childNodes[phylumSI-1]
.childNodes[clazzSI]
.childNodes[selectedIndex];
families=currentNode.childNodes;
for(var i=1;i<families.length;i++){
familyName=families[i].selectNodes("name")[0].text;
document.forms[0].family.options[i]=new Option(familyName,familyName);
}
}
else{
currentNode=xmlDoc.documentElement
.childNodes[phylumSI*2-1]
.childNodes[clazzSI*2+1]
.childNodes[selectedIndex*2+1];
families=currentNode.childNodes;
for(var i=1;i<families.length-2;i=i+2){
familyName=families[i+2].childNodes[1].textContent;
document.forms[0].family.options[(i+1)/2]=new Option(familyName,familyName);
}
}
}
//设置“属”的下拉框
function setGenus(selectedIndex){
document.forms[0].genus.disabled=null;//取消下拉框的禁用
//currentNode的赋值应使用绝对定位
var phylumSI=document.forms[0].phylum.selectedIndex;//phylum selected index
var clazzSI=document.forms[0].clazz.selectedIndex; //clazz selected index
var orderSI=document.forms[0].order.selectedIndex; //order selected index
clearOption(document.forms[0].genus);
var genues;
var genusName;
if(browserType=="IE"){
currentNode=xmlDoc.documentElement
.childNodes(phylumSI-1)
.childNodes(clazzSI)
.childNodes(orderSI)
.childNodes(selectedIndex);
genuses=currentNode.childNodes;
for(var i=1;i<genuses.length;i++){
//属为叶节点
var genusName=genuses[i].text;
document.forms[0].genus.options[i]=new Option(genusName,genusName);
}
}
else{
currentNode=xmlDoc.documentElement
.childNodes[phylumSI*2-1]
.childNodes[clazzSI*2+1]
.childNodes[orderSI*2+1]
.childNodes[selectedIndex*2+1];
genuses=currentNode.childNodes;
for(var i=1;i<genuses.length-2;i=i+2){
//属为叶节点
var genusName=genuses[i+2].textContent;
document.forms[0].genus.options[(i+1)/2]=new Option(genusName,genusName);
}
}
}
//清空下拉框选项
function clearOption(selectElement){
for(var i=1;i<selectElement.options.length;i++){
selectElement.options[i]=null;
}
}
//判断浏览器类型
function setBrowserType(){
if (window.ActiveXObject){//IE
browserType="IE";
}else{
browserType="FireFox";
}
}
//载入xml
function loadXml(xmlName){
if (browserType=="IE"){//IE
xmlDoc = new ActiveXObject("Microsoft.XMLDOM");
xmlDoc.async = false;
xmlDoc.load(xmlName);
} else{
// xmlDoc=document.implementation.createDocument("", "", null);
// xmlDoc.async = false;
// xmlDoc.load("classify.xml");
browserType="FireFox";
var xmlHttp = new XMLHttpRequest();
xmlHttp.open( "GET", "classify.xml", false ) ;
xmlHttp.send(null) ;
xmlDoc=xmlHttp.responseXML;
//FireFox没有selectNodes()方法,且xml中,其childNodes的对应索引是1,3,5,7...
// alert(xmlDoc.getElementsByTagName("phylum")[1]
// .childNodes[3].childNodes[3].childNodes[1].textContent);
}
}
最后是xml文件的内容
<?xml version="1.0" encoding="UTF-8"?>
<plant>
<phylum>
<name>被子植物门</name>
<clazz>
<name>双子叶植物纲</name>
<order>
<name>菊目</name>
<family>
<name>菊科</name>
<genus>菊属</genus>
</family>
<family>
<name>桔梗科</name>
<genus>同钟花属</genus>
<genus>刺萼参属</genus>
</family>
</order>
<order>
<name>胡椒目</name>
<family>
<name>胡椒科</name>
<genus>胡椒属</genus>
<genus>草胡椒属</genus>
<genus>齐头绒属</genus>
</family>
</order>
</clazz>
</phylum>
<phylum>
<name>蕨类植物门</name>
<clazz>
<name>石松纲</name>
<order>
<name>石松目</name>
<family>
<name>石松科</name>
<genus>石松属</genus>
</family>
</order>
</clazz>
</phylum>
</plant>
这是部分效果图:
可以实现上下级下拉框的联动,支持无序选择,若向上重新选择,下下层下拉框将自动被禁用,下层下拉框选项也会相应改变。
有一点不足是,因为数据量实在太大,这样5个下拉框仍然可能出现某下拉框有几百甚至几千个选项,此时就失去了下拉框的意义,因此正在考虑是否应该做成输入框的形式,或者像搜索引擎那样带有输入提示功能,研究中,欢迎拍砖。
PS:重新修改了一下,可以支持FireFox了,这可真是麻烦的工程:FireFox的JavaScript的Element对象中没有selectNodes()方法,只有调用childNodes()或者getElementsByTagName();并且在FireFox中,xml中节点对应childNodes()的索引是1,3,5,7...,也就是说,如果你想读取xml某个节点下的第i个子节点,正常我们就会写someNode.childNodes[i-1],但在FireFox就必须写作someNode.childNodes[i*2-1]。
另外在使用数组时,IE允许把小括号当成中括号使用(即someArray[i]和someArray(i)均合法),FireFox则不行,所以最好统一写someArray[i]。
- 大小: 9.8 KB
- 大小: 9.3 KB
- 大小: 7.6 KB
- 大小: 6.7 KB
- 大小: 9.8 KB
分享到:
相关推荐
三级联动下拉框, 也可以扩展为N级联动, 自动判别 label, value, 速度快,项目中肯定用得到
在JavaScript开发中,实现多级联动下拉框的功能可能需要编写大量的DOM操作代码,而使用jQuery这样的库可以大大简化这个过程。jQuery提供了强大的DOM操作API,如`$(selector).val()`用于获取或设置元素的值,`$...
在IT行业中,网页开发经常会遇到需要实现省市区县等多级联动下拉框的需求,这在用户填写地址、选择服务区域等场景十分常见。本文将详细介绍如何使用HTML、JavaScript(通常配合jQuery)以及可能涉及的CSS来实现...
ajax 实现 (全国,省,是,区....) N级联 Demo js: <script type="text/javascript"> $(function () { $.fn.CascadeSelect({ url: '/CascadeSelect/CascadeSelect.ashx', //返回Json数据的一般处理文件 idKey: ...
总的来说,使用jQuery实现多级联动下拉框涉及到了HTML结构设计、CSS样式定制和JavaScript交互编程。开发者需要理解DOM操作、事件处理和数据通信的基本原理,同时也需具备一定的前端UI设计能力,才能创造出既实用又...
### JS三级联动下拉框知识点解析 #### 一、知识点概览 在Web开发中,三级联动下拉框是一种常见的用户交互方式,主要用于提供多级数据选择的功能,比如省份、城市、区县的选择等场景。该功能通过JavaScript实现,...
### JS下拉框多级联动实现——出生日期与世界各国选择 #### 一、概述 本文档将详细介绍如何使用JavaScript实现下拉框的多级联动功能,具体应用在两个场景:一是用户选择出生日期(年、月、日);二是用户选择国籍...
JavaScript 自定义多级联动下拉菜单
总的来说,理解和实现js多级联动多选checkbox插件涉及了JavaScript的多个核心概念,包括DOM操作、事件处理、数据结构、UI更新、性能优化以及模块化开发等。通过学习和实践,开发者可以创建出功能强大且易于使用的...
用XML存储地区数据,页面上使用: var Array_AreaName = ["","",""]; /// 地区值设置初始值 并在body 的onload 事件里添加:Set_AreaSelected(); ;"> 所在省份 所在地市 ...具体使用参考js文件
总结,"简单的vue三级联动下拉框代码"是一个实际项目中常见的需求,通过Vue.js的数据绑定、组件化以及计算属性等特性,可以方便地实现这种功能,提高开发效率。理解并掌握这些知识点对于提升Vue.js开发能力大有裨益...
在本项目中,"基于jQuery Selector.js插件实现的联动下拉框表单美化效果源码.zip" 提供了一种实现联动下拉框的方法,这种效果通常用于多级筛选或者关联数据的展示,例如在地理位置选择、产品分类等场景。联动下拉框...
- Vue.js/React.js/Angular.js:现代Web开发中,这些前端框架提供了更高效的数据绑定和组件化功能,可以方便地实现联动下拉框。 4. **数据结构**: - 数据通常以JSON格式存储,每个下拉框的选项是嵌套的,比如省-...
在JavaScript、HTML和JS的世界里,自定义多级联动下拉菜单是一种常见的交互设计,它通常用于提升用户体验,尤其是在处理复杂的数据层级结构时。本文将深入探讨如何利用这些技术实现自定义的select联动效果。 首先,...
Web多级联动下拉框(从数据库中获取数据,MVC模式,使用的是tomcat)-附件资源
在网页设计和开发中,"三级联动下拉框"是一种常见的交互元素,它通常用于实现多级关联选择,比如省份-城市-区县的选择。这种功能可以极大地提高用户体验,尤其是在处理大量分类信息时。下面我们将深入探讨这个话题,...
本文将详细介绍如何使用原生JavaScript实现一个可自定义级数的多级联动下拉列表,以及如何根据提供的"demo"文件进行应用。 首先,我们需要了解多级联动的基本原理。在联动下拉列表中,当用户在一个下拉列表中选择一...
本主题将深入探讨如何使用jQuery动态创建联动下拉框,这是一个常见的需求,特别是在构建多级选择或依赖性选择菜单时。 联动下拉框通常指的是两个或多个下拉框之间存在关联关系,当用户在一个下拉框中选择一个选项时...
在Struts框架中实现下拉框的多级联动是一项常见的需求,比如在地址选择时,可以选择国家、省份、城市等,每一级的选择都会影响下一级的选项。 在Struts框架中,实现下拉框的联动通常涉及到以下几个关键知识点: 1....
综上所述,Ajax无限级联动下拉框的实现涉及前端JavaScript编程、服务器端数据处理以及良好的用户体验设计。通过合理的架构和优化,我们可以创建出高效、流畅的多级筛选功能,提升用户在浏览和操作网站时的满意度。