如何使用Javascript开发sliding-nav带滑动条效果的导航插件?

  • A+
所属分类:Web前端
摘要

本文介绍如何使用纯Javascript来开发一款简单的JS插件,本插件可以实现鼠标悬停在导航上时,下方的滑动条自动从当前菜单滑动到所选菜单当中去。

本文介绍如何使用纯Javascript来开发一款简单的JS插件,本插件可以实现鼠标悬停在导航上时,下方的滑动条自动从当前菜单滑动到所选菜单当中去。

本项目的源代码寄宿于GitHub,记得点小星星哦:

https://github.com/dosboy0716/sliding-nav

一、前言

效果如下图:

如何使用Javascript开发sliding-nav带滑动条效果的导航插件?

 

二、使用方法

本插件只需要如下的三步,就可以在您的项目中使用:

1、在</body>标记结束前,引用sliding-nav.js文件

2、在需要滑动条的菜单容器上加类名 sliding-nav,当前项使用类名:active

3、使用属性来定定外观:sn-color="颜色" sn-radius="圆度" sn-height="高度"

<script src="/path/to/sliding-nav.js"></script> <ul class="nav sliding-nav" sn-color="#F00" sn-radius="0px" sn-height="3px">     <li class="active">菜单项1</li>     <li>菜单项2</li>     <li>菜单项3</li> <ul> 

 

三、开发过程

1. 模型示例

 

如何使用Javascript开发sliding-nav带滑动条效果的导航插件?

 

 

导航菜单一般使用上图的层次型结构,外层容器使用<ul> 标记,菜单项使用<li>标记,假设如果要显示黄色小横条,如何定位很重要。

经过分析,虽然在视觉上小横条位于UL之内,为了不破坏原来导航的样式,小黄条必须使用absolute的绝对定位,并且初始位置与ul标记相同。

因此,我们把小横条插入<ul>标记的前面,如上面的小灰点,它就是小横条的初始位置即(left=0,top=0)的位置。

那么我们如何让小条看起来在菜单项的正下方呢?

  • 把小条的top属性赋值为菜单项的高度(即offsetHeight属性),
  • 把小条的left属性赋值为菜单项的左边距(即offsetLeft属性)

实现上面的功能可以使用如下的代码:

function init() {      var navs = document.getElementsByClassName('sliding-nav');      for (var i = 0; i < navs.length; i++) {           //创建一个DIV与当前导航竖向对齐         var indi = document.createElement("div");         indi.id = "slna-indicator"          indi.style.borderRadius = navs[i].getAttribute("sn-radius") || "0px"         indi.style.height = navs[i].getAttribute("sn-height") || "3px"         indi.style.backgroundColor = navs[i].getAttribute("sn-color") || "#F00"          indi.style.position = "absolute"         indi.style.transition = "0.5s"          //查找当前子菜单项,如果有类名active或者是selected就视为当前项,如果没有使用第1项         var selected = navs[i].getElementsByClassName('active')         if (selected.length == 0) {             selected = navs[i].getElementsByClassName('selected')         }         if (selected.length == 0) {             selected = navs[i].children         }          if (selected.length == 0) {             throw Error('Sorry, Navigation bar has no item at all!');         }          selected = selected[0];          indi.style.width = selected.offsetWidth + "px";         indi.style.top = selected.offsetHeight + "px";         indi.style.left = selected.offsetLeft + "px";         navs[i].parentElement.insertBefore(indi, navs[i]);          //未完成,下面插入代码以绑定事件         }  } 

 

如上的代码构建了初始化函数init(),此函数:

查找所有含有类名sliding-nav的标记,并且按照上面的方法,在前面插入div标记充当“指示条”,并且查找“活动”的菜单项,找到后通过这个菜单项的各个属性给“指示条”定位。

2、事件与动画

我们把"指示条"div 标记transition属性设置成了0.5s,那么只要在事件里直接设置该div的如下:

  • left属性就可以实现"指示条"的移动
  • width属性就可以设置"指示条"的宽度

所以可以在如上的代码末尾,插入如下的代码实现事件与动画:

for (var j = 0; j < navs[i].children.length; j++) {              hover(navs[i].children[j], function(e, elem) {                  indi.style.width = elem.offsetWidth + "px";                 indi.style.left = elem.offsetLeft + "px";              });              //移出导航就恢复默认             hover(navs[i], null, function(e, elem) {                 indi.style.width = selected.offsetWidth + "px";                 indi.style.left = selected.offsetLeft + "px";             });          }

其中代码,用到了自定义函数hover,该函数类似于实现hover事件,JS原生只有mouseover和mouseout事件。

hover(绑定DOM元素,移入事件函数,移出事件函数) 

 

函数作用是给DOM元素绑定鼠标移入和鼠标移出事件,具体实现的过程,可以看作者原代码。

 

四、所有原代码

本文实现的所有原代码如下,希望读者提出更加优化的建议,我们一起打造更加唯美的前端体验。

 

  1 /*   2    3 Usage   4 1. Include file sliding-nav.js before tag</body> in a HTML file.   5    6 <script src="/path/to/sliding-nav.js"></script>   7    8 2. Use class name sliding-nav in a navigation bar element,and use .active for a selected menu item.   9 3. Use following attributes to change its color,radius and height:sn-color, sn-radius,sn-height. (if no these attributes, default settings following will be used)  10   11 <ul class="nav sliding-nav" sn-color="#F00" sn-radius="0px" sn-height="3px">  12     <li class="active">menu-item 1</li>  13     <li>menu-item 2</li>  14     <li>menu-item 3</li>  15 <ul>       16   17 使用方法  18 1、在</body>标记结束前,引用sliding-nav.js文件  19 2、在需要滑动条的菜单容器上加类名 sliding-nav,当前项使用类名:active  20 3、使用属性来定定外观:sn-color="颜色" sn-radius="圆度" sn-height="高度"  21   22   23 <script src="/path/to/sliding-nav.js"></script>  24 <ul class="nav sliding-nav" sn-color="#F00" sn-radius="0px" sn-height="3px">  25     <li class="active">菜单项1</li>  26     <li>菜单项2</li>  27     <li>菜单项3</li>  28 <ul>  29       30 Sliding Navigation Bar By yan,ZHANG   31 Mailto: 26959368@qq.com  32 2020.02.06  33 */  34   35   36 window.onload = function() {  37     init();  38 };  39   40 function bind(elem, ev, callback) {  41     if (document.all) {  42         elem.attachEvent("on" + ev, callback);  43     } else {  44         elem.addEventListener(ev, callback, false);  45     }  46 }  47   48 function unbind(elem, ev, callback) {  49     if (typeof(callback) == "function") {  50         if (document.all) {  51             elem.detachEvent("on" + ev, callback);  52         } else {  53             elem.removeEventListener(ev, callback, false);  54         }  55     } else {  56         if (document.all) {  57             elem.detachEvent("on" + ev);  58         } else {  59             elem.removeEventListener(ev, false);  60         }  61     }  62 }  63   64 function hover(elem, overCallback, outCallback) { //实现hover事件  65     var isHover = false; //判断是否悬浮在上方  66     var preOvTime = new Date().getTime(); //上次悬浮时间  67     function over(e) {  68         var curOvTime = new Date().getTime();  69         isHover = true; //处于over状态  70         if (curOvTime - preOvTime > 10) { //时间间隔超过10毫秒,认为鼠标完成了mouseout事件  71             overCallback && overCallback(e, elem);  72         }  73         preOvTime = curOvTime;  74     }  75   76     function out(e) {  77         var curOvTime = new Date().getTime();  78         preOvTime = curOvTime;  79         isHover = false;  80         setTimeout(function() {  81             if (!isHover) {  82                 outCallback && outCallback(e, elem);  83             }  84         }, 10);  85     }  86     bind(elem, "mouseover", over);  87     bind(elem, "mouseout", out);  88 };  89   90   91   92   93 function init() {  94   95     var navs = document.getElementsByClassName('sliding-nav');  96   97     for (var i = 0; i < navs.length; i++) {  98   99  100         //创建一个DIV与当前导航竖向对齐 101         var indi = document.createElement("div"); 102         indi.id = "slna-indicator" 103  104         indi.style.borderRadius = navs[i].getAttribute("sn-radius") || "0px" 105         indi.style.height = navs[i].getAttribute("sn-height") || "3px" 106         indi.style.backgroundColor = navs[i].getAttribute("sn-color") || "#F00" 107  108         indi.style.position = "absolute" 109         indi.style.transition = "0.5s" 110  111         //查找当前子菜单项,如果有类名active或者是selected就视为当前项,如果没有使用第1项 112         var selected = navs[i].getElementsByClassName('active') 113         if (selected.length == 0) { 114             selected = navs[i].getElementsByClassName('selected') 115         } 116         if (selected.length == 0) { 117             selected = navs[i].children 118         } 119  120         if (selected.length == 0) { 121             throw Error('Sorry, Navigation bar has no item at all!'); 122         } 123  124         selected = selected[0]; 125  126         indi.style.width = selected.offsetWidth + "px"; 127         indi.style.top = selected.offsetHeight + "px"; 128         indi.style.left = selected.offsetLeft + "px"; 129         navs[i].parentElement.insertBefore(indi, navs[i]); 130  131         for (var j = 0; j < navs[i].children.length; j++) { 132  133             hover(navs[i].children[j], function(e, elem) { 134  135                 indi.style.width = elem.offsetWidth + "px"; 136                 indi.style.left = elem.offsetLeft + "px"; 137  138             }); 139  140             //移出导航就恢复默认 141             hover(navs[i], null, function(e, elem) { 142                 indi.style.width = selected.offsetWidth + "px"; 143                 indi.style.left = selected.offsetLeft + "px"; 144             }); 145  146         } 147  148  149  150  151     } 152  153 }