jQuery插件——多级联动菜单
jQuery插件——多级联动菜单
引言
开发中,有好多地方用到联动菜单,以前每次遇到联动菜单的时候都去重新写,代码重用率很低,前几天又遇到联动菜单的问题,总结了下,发现可以开发一个联动菜单的功能,以后想用的时候就方便多了。项目中每个页面都有引用jQuery,,开发个jQuery联动菜单插件,说动手就动手,下面跟大家分享分享。
我用的jQuery插件方式
- (function($){
- $.fn.casmenu=function(argvs){
- //your code
- }
- })(jQuery);
其中jQuery传入的是jquery对象,需要在扩展之前引用。在扩展中同样使用jQuery的短格式$。
$.fn是指jquery的命名空间,加在fn上的方法及属性,会对jquery实例每一个有效,看看下面的jquery源码101行左右:
- jQuery.fn = jQuery.prototype = {
- ......
- }
比如说后面要开发的$.fn.casmenu(),定义之后,在后面的jquery对象都可以使用该方法。
这里还有一种扩展的方法:
- $.extend({
- funName: function(){
- //your code
- },
- });
这种扩展方法和上面的是有区别的,要是拿类做类比的话,$.extend这种方法相当于类中的静态方法,上面的一种方式相当于非静态方法,必须有对象才可以使用。简单的理解,是下面这样的:
- //$.fn.casemenu 方式扩展的方法,必须在有jquery对象的时候才可以使用
- $("#mydiv").casmenu();
- //$.extend({}) 方式扩展的方法,可以直接使用
- $.add(2,3);
设计思路
首先是层级菜单的数据保存方式,看看下面的数据:
- var levels={
- //内容中有引号,必须使用单引号,外引号必须用双引号
- //name => value
- 1:{
- 退出应用: "code1003",
- 登录界面:"code1004",
- 跳转至个人资料界面:"code1005",
- },
- 2:{
- 退出应用:{
- 应用1:"gameid1",
- 应用2:"gameid2",
- 应用3:"gameid3",
- 应用4:"gameid4",
- 应用5:"gameid5",
- },
- 跳转至个人资料界面:{
- 主界面:"main interface",
- }
- },
- 3:{
- 应用1:{
- 中级场:"12",
- 高级场:"13",
- 职业场:"14",
- 比赛场:"15",
- }
- }
- }
对象levels中的直接键值1、2、3代表菜单的层级,没有就不用些,每一项name=>value代表select中option的名称和value。
层级有规律,某一层级中的某一项要是有下一级菜单,在下一层及有该项的名称,就像levels[1]['退出应用']在有下级菜单,就有levels[2]['退出应用'],要是继续有下级菜单,就像levels[2]['退出应用']['应用1'],会在下一层及中继续有levels[3]['应用1']。这样一来,就实现了无限级联动菜单,不同的联动菜单只需要修改菜单配置文件就可以了。
但是这么做又有一个遗憾,就是如果level2[2]中的子项有两个名称相同的,都有下级菜单,而且下级菜单内容还不一样,就会有问题,因此在设置的时候,有下级菜单的项要取不同的名称,这里要注意下。就目前这种来说,简单,好理解,也够用了。
代码实现
在代码中也用到了$.extend,用来扩展默认配置。
还有一个点要注意,在联动的时候会将实事的菜单值放入一个属性为hidden的input中,用默认逗号分割每个层级之间的值,可以很轻松的获取到联动菜单所有项的值
- if(typeof(AI.opts.saveinput) != "undefined" && (AI.opts.saveinput = AI.opts.saveinput.toString()) != ''){
- $("<input type='hidden' value='' name='"+AI.opts.saveinput+"' id='"+AI.opts.saveinput+"' />").appendTo($('body'));
- }
- (function($){
- //配置
- var AI={
- opts:{
- saveinput:"jumpcode", //是否将结果保存至input
- levels:{},
- ulObj:{},//保存生成好的ul列表
- length:0, //层级菜单的层级
- divide:",",//默认各个层级菜单之间的分隔符
- }
- };
- $.fn.casmenu=function(opts){
- AI.opts = $.extend(AI.opts, opts);
- if((AI.opts.length = Object.keys(AI.opts.levels).length) <= 0){
- throw "levels arr must not be empty";
- return;
- }
- var _levels = AI.opts.levels;
- if(_levels[1] == undefined){
- throw "menu level 1 must not be empty";
- return;
- }
- var _levels_1 = _levels[1];
- if(typeof(AI.opts.saveinput) != "undefined" && (AI.opts.saveinput = AI.opts.saveinput.toString()) != ''){
- $("<input type='hidden' value='' name='"+AI.opts.saveinput+"' id='"+AI.opts.saveinput+"' />").appendTo($('body'));
- }
- AI.opts.ulObj['level_1'] = '<select class="casmenu" level="1">';
- AI.opts.ulObj['level_1'] += '<option value="null">请选择</option>';
- $("#"+AI.opts.saveinput).val('null');
- for(var i in _levels_1){
- AI.opts.ulObj['level_1'] += '<option name="'+i+'" value="'+_levels_1[i]+'">'+i+'</option>';
- }
- AI.opts.ulObj['level_1'] += '</select>';
- $(AI.opts.ulObj['level_1']).appendTo(this);
- $("body").on("change", ".casmenu", function(){
- var level = $(this).attr("level");
- if(level > AI.opts.length) return;
- level++;
- //移除当前触发菜单之后的菜单
- for(var num=level;num<=AI.opts.length;num++){
- $(".casmenu[level="+num+"]").remove();
- }
- //设置input的值,级联菜单的值
- var _val = '';
- for(var val=1;val<=AI.opts.length;val++){
- var __val = $("select[level="+val+"]");
- if(__val.length <= 0)
- continue;
- _val += __val.val()+AI.opts.divide;
- }
- $("#"+AI.opts.saveinput).val(_val.substr(0, _val.length-1));
- //levels对象中不存在下一级别目录
- if(typeof(AI.opts.levels[level]) == "undefined") return;
- //获取下一级别目录的键值,值不存在的话返回
- var name = $(this).find("option:selected").attr("name");
- if(typeof(AI.opts.levels[level][name]) == "undefined") return;
- if(typeof(AI.opts.ulObj['level_'+level]) == "undefined" || typeof(AI.opts.ulObj['level_'+level][name]) == "undefined"){
- if(typeof(AI.opts.ulObj['level_'+level]) == "undefined")
- AI.opts.ulObj['level_'+level] = {};
- AI.opts.ulObj['level_'+level][name] = '<select class="casmenu" level="'+level+'">';
- AI.opts.ulObj['level_'+level][name] += '<option value="null">请选择</option>';
- var levelinfo = AI.opts.levels[level][name];
- for(var i in levelinfo){
- AI.opts.ulObj['level_'+level][name] += '<option name="'+i+'" value="'+levelinfo[i]+'" >'+i+'</option>';
- }
- AI.opts.ulObj['level_'+level][name] += '</select>';
- }
- $(AI.opts.ulObj['level_'+level][name]).appendTo($(this).parent());
- var _val = '';
- for(var val=1;val<=AI.opts.length;val++){
- var __val = $("select[level="+val+"]");
- if(__val.length <= 0)
- continue;
- _val += __val.val()+AI.opts.divide;
- }
- $("#"+AI.opts.saveinput).val(_val.substr(0, _val.length-1));
- });
- }
- })(jQuery);
效果
下面看看效果
例子:http://aizuyan.github.io/casmenu.html
我把完整的代码放在github上:https://github.com/aizuyan/jquery.plugin
本文版权归作者iforever(luluyrt@163.com)所有,未经作者本人同意禁止任何形式的转载,转载文章之后必须在文章页面明显位置给出作者和原文连接,否则保留追究法律责任的权利。
jQuery插件——多级联动菜单的更多相关文章
- 开发一个jQuery插件——多级联动菜单
引言 开发中,有好多地方用到联动菜单,以前每次遇到联动菜单的时候都去重新写,代码重用率很低,前几天又遇到联动菜单的问题,总结了下,发现可以开发一个联动菜单的功能,以后想用的时候就方便多了.项目中每个页 ...
- [ PHP+jQuery ] ajax 多级联动菜单的应用:电商网站的用户地址选择功能 ( 二 ) - 仿亚马逊下拉面板
/** jQuery version: 1.8.3 Author: 小dee Date: 2014.11.8 */ 接上一篇博客. 实现带缓存的仿亚马逊下拉面板 效果图: 图1 初始 图2 点击省份 ...
- jQuery cxSelect 多级联动下拉菜单
随着电商热门,这种多层次的互动更充分地体现在下拉菜单,最明显的是多级联动地址下拉选择,因此,这里是一个简单的分享 jQuery cxSelect 多级联动下拉菜单 cxSelect 它是基于 jQue ...
- 基于jQuery垂直多级导航菜单代码
基于jQuery垂直多级导航菜单代码是一款黑色风格的jQuery竖直导航菜单特效下载.效果图如下: 在线预览 源码下载 实现的代码. html代码: <ul class="ce&q ...
- jQuery实现三级联动菜单(鼠标悬停联动)
效果图: 代码: <!DOCTYPE html> <html> <head> <meta charset="utf-8"/> < ...
- 网站开发常用jQuery插件总结(十)菜单插件superfish
网站对于菜单的依赖其实并不是很大,我们完全可以不使用菜单来设计网站,显示网站内容.但是如果网站的分类太多,“也许”使用菜单作为网站导航可以使 用户 更方便的寻找内容.superfish插件就是用于实现 ...
- jquery插件-省市联动
由于项目需要需要实现一个省市联动,由于业务有一些特殊的需求,使用现有的插件略有不便,就自己实现了一个. 首先需要保存地区数据的JS数据文件,我这里命名为areaData.js,内容如下 ...
- jQuery省市区三级联动菜单
<style> select{ padding:5px 0; } .outer{ width:500px; margin:20px auto; } </style> <d ...
- vue地址插件多级联动自适应 + github地址
https://github.com/cqzyl/vue-manyAddress
随机推荐
- FireFox VS Chrome 之 调试篇
一个完美的调试工具,FireBug! 精确跟踪每一步.仅仅要按下图所看到的,选择"脚本",然后在下方选择脚本所在的文件就可以对该文本的运行进行断点跟踪. 而且仅当一个线程运行结束后 ...
- 【十三】注入框架RoboGuice采用:(Logging via Ln)
上一篇我们简单的介绍了一下RoboGuice的使用([十二]注入框架RoboGuice使用:(Your First Injected ContentProvider)),今天我们来看下Log日志使用. ...
- War文件部署(转)
其实,开始要求将源码压缩成War文件时,一头雾水! 公司项目要求做CAS SSO单点登录 也就是这玩意.... 其实war文件就是Java中web应用程序的打包.借用一个老兄的话,“当你一个web应用 ...
- Codeforces 459E Pashmak and Graph(dp+贪婪)
题目链接:Codeforces 459E Pashmak and Graph 题目大意:给定一张有向图,每条边有它的权值,要求选定一条路线,保证所经过的边权值严格递增,输出最长路径. 解题思路:将边依 ...
- plsql dev中Dynamic Performance Tables not accessible分析解决(转)
使用plsql dev的朋友多遇到过类此如下面的提示: Dynamic Performance Tables not accessible, Automatic Statistics Disabled ...
- hud 1312 Red and Black
题目: 链接:pid=1312">点击打开链接 题意: DFS搜索 算法: dfs 思路: 简单题 代码: #include<iostream> #include<c ...
- HDU 3065 病毒在继续 (AC自己主动机)
中国标题不解释 Sample Input 3 AA BB CC ooxxCC%dAAAoen....END Sample Output AA: 2 CC: 1 输出病毒出现的次数! #includ ...
- Visual Studio 原生开发的10个调试技巧(二)
原文:Visual Studio 原生开发的10个调试技巧(二) 我以前关于 Visual Studio 调试技巧的文章引起了大家很大的兴趣,以至于我决定分享更多调试的知识.以下的列表中你可以看到写原 ...
- Windows Phone 启动器
http://msdn.microsoft.com/zh-CN/library/gg278408(v=vs.92)#BKMK_Launchers using Microsoft.Phone.Contr ...
- InstallShield自定义安装界面
原文:InstallShield自定义安装界面 版权声明: 可以任意转载,转载时请务必以超链接形式标明文章原始出处和作者信息. 前言: 对于一些InstallShield用户或企业,对于安装包界面除了 ...