经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » JS/JS库/框架 » JavaScript » 查看文章
如何使用Javascript开发sliding-nav带滑动条效果的导航插件?
来源:cnblogs  作者:dosboy  时间:2021/3/29 9:09:21  对本文有异议

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

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

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

一、前言

效果如下图:

 

二、使用方法

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

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

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

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

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

 

三、开发过程

1. 模型示例

 

 

 

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

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

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

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

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

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

  1. function init() {
  2.  
  3. var navs = document.getElementsByClassName('sliding-nav');
  4.  
  5. for (var i = 0; i < navs.length; i++) {
  6.  
  7.  
  8. //创建一个DIV与当前导航竖向对齐
  9. var indi = document.createElement("div");
  10. indi.id = "slna-indicator"
  11.  
  12. indi.style.borderRadius = navs[i].getAttribute("sn-radius") || "0px"
  13. indi.style.height = navs[i].getAttribute("sn-height") || "3px"
  14. indi.style.backgroundColor = navs[i].getAttribute("sn-color") || "#F00"
  15.  
  16. indi.style.position = "absolute"
  17. indi.style.transition = "0.5s"
  18.  
  19. //查找当前子菜单项,如果有类名active或者是selected就视为当前项,如果没有使用第1项
  20. var selected = navs[i].getElementsByClassName('active')
  21. if (selected.length == 0) {
  22. selected = navs[i].getElementsByClassName('selected')
  23. }
  24. if (selected.length == 0) {
  25. selected = navs[i].children
  26. }
  27.  
  28. if (selected.length == 0) {
  29. throw Error('Sorry, Navigation bar has no item at all!');
  30. }
  31.  
  32. selected = selected[0];
  33.  
  34. indi.style.width = selected.offsetWidth + "px";
  35. indi.style.top = selected.offsetHeight + "px";
  36. indi.style.left = selected.offsetLeft + "px";
  37. navs[i].parentElement.insertBefore(indi, navs[i]);
  38.  
  39. //未完成,下面插入代码以绑定事件
  40.  
  41.  
  42.  
  43.  
  44. }
  45.  
  46. }

 

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

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

2、事件与动画

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

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

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

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

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

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

 

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

 

四、所有原代码

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

 

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

 

原文链接:http://www.cnblogs.com/dosboy/p/14590251.html

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号