zoukankan      html  css  js  c++  java
  • JavaScript实战(带收放动画效果的导航菜单)

    虽然有很多插件可用,但为了共同提高,我做了一系列JavaScript实战系列的实例,分享给大家,前辈们若有好的建议,请务必指出,免得误人子弟啊!

    ( 原创文章,转摘请注明:苏福:http://www.cnblogs.com/susufufu/p/5768402.html )

    今天是第一战:带收放动画效果的菜单,效果如下图:(样式有点丑(-^-))

    ( 由于在写本文时,用的编辑器不同,暂时添加不了演示效果,这里有:最终完整代码和演示 

    动画效果:鼠标hover改变所有目标的背景和字体颜色,鼠标移动到‘首页导航’,显示下面的分组菜单,分组菜单有子菜单,点击可缩放,带动画过度效果!而且,可以随便添加和删除导航菜单和子菜单,不影响效果!

    如何实现呢?

    第一步:用什么来实现菜单?HTML代码设计如下,遵循JS代码和HTML代码分离的原则!这里你看不到一句JS代码

    未应用样式之前是这个样子的:很古老吧!!!

    第二步:CSS样式。鼠标hover改变所有目标的背景和字体颜色,直接用CSS的transition和:hover,而其他的CSS样式布局就不全部列举了,大家自己动手吧,主要注意以下几点:

            #ul{
                ....
                z-index: 100;
            }
            #ul li{
                display: inline-block;
                position: relative;
                top: 0;
                left: -25px;
                width: 10%;
                min-width: 70px;
                height: 30px;
                text-align: center;
                line-height: 30px;
                border: 1px solid gray;
                border-radius:10px;
                background-color: aliceblue;
                cursor: pointer;
                -webkit-transition: all ease-in-out 0.3s;
                -moz-transition: all ease-in-out 0.3s;
                -ms-transition: all ease-in-out 0.3s;
                -o-transition: all ease-in-out 0.3s;
                transition: all ease-in-out 0.3s;
            }
            #ul li:hover{background-color: aquamarine;color: red;}
            ...
            .show-hide:hover{background-color: beige}
            .a-div{
                background-color: aquamarine;
                border-radius:10px;
                color: black;
                display: none;
                opacity: 0
            }
            .a{
                z-index: -1;
                display: block;
                ...
            }                      
    第三步:这一步是重点。如果给每个菜单选项和分组都添加事件监听,个人觉得好麻烦,且代码量肯定多不少,有没有什么办法就在一个元素上加监听就能实现呢?

    答案肯定是有的,利用事件的冒泡机制!在父元素ul标签上添加事件监听,而在监听函数里直接改变触发事件的元素样式就可以了,就这么简单!代码如下:
    var ul = document.getElementById('ul');
    ul.addEventListener('mouseover',listener1,false);
    ul.addEventListener('mouseout',listener2,false);
    ul.addEventListener('click',listener3,false);
    因为IE8及以下版本没有addEventListener,如果要兼容,还得加attachEvent对应的代码。

    第四部:主角登场!实现listener1、listener2、listener3监听函数。
    首先来最简单的listener1函数,代码如下:
    function listener1(event){
        //event = event||window.event; //兼容IE8及以前版本
        var target = event.target||event.srcElement; //兼容IE8及以前版本
        if(target.tagName.toLowerCase() === 'li'){
            var div1 = target.getElementsByTagName('div')[0];
            div1.style.display = 'block';
            var i = 0;
            var id;
            (function foo(){
                if(i>=1){clearTimeout(id);id=null;return;}
                i+=0.2;
                div1.style.opacity = i;
                id = setTimeout(function(){clearTimeout(id);foo()},30); 
         })();
      }
    }
    同样,一切为了IE8及更旧版本,
      1.因为它的event没有target属性,只有相对应得srcElement属性
      2.而这一句event = event||window.event;这里其实是可以省略的,只有当用属性来设置注册事件监听时,如ul.onmouseover = function(){},或<ul onmouseover='func'>,IE8及更旧版本只能通过window.event来取得当前的Event对象

    好了,现在获得了当前触发事件的target,事情就简单很多了,通过他就可以改变它自己和它的亲戚!
    下面是listener2函数,用在mouseout时触发,主要是操控target的子元素DIV,代码如下:
               function listener2(event){
                    //event = event||window.event;
                    var target = event.target||event.srcElement;
                    if(target.tagName.toLowerCase() === 'li'){
                        var div1 = target.getElementsByTagName('div')[0];
                        div1.onmouseover = function(){
                            div1.style.display = 'block';
                            div1.style.opacity = 1;
                        };
                        div1.onmouseout = function(){
                            div1.style.display = 'none';
                            div1.style.opacity = 0;
                        };
                        div1.style.display = 'none'; //这一组是为了实现当鼠标从上方出去时隐藏div1
                        div1.style.opacity = 0;
                    }
                }    
    好了,到这里,已经实现了大部分效果了,还有最后一步,那就是1号主角了:listener3函数,它主要负责鼠标点击时的缩放效果!
    实现原理:
      1.函数外面定义一个bool变量当做开关,鼠标点一下开,再点一下关;
      2.通过setTimeout来实现动画效果,动态的改变子菜单的height和opacity属性,还有display属性;
    完整代码如下:
     1 var bool = true;
     2             function listener3(event) {
     3                 var event = event || window.event;
     4                 var target = event.target || event.srcElement;
     5                 if (target.className === 'show-hide') {
     6                     var parent = target.parentElement;
     7                     var adiv = parent.getElementsByClassName('a-div')[0];
     8                     if (window.getComputedStyle(adiv,null).opacity>0.5){bool=false}else{bool=true}
     9                     var height = 90,
    10                             changeH,
    11                             opacity,
    12                             id;
    13                     if (bool) {
    14                         changeH = 0;
    15                         opacity = 0;
    16                         target.innerHTML = '财经 -';
    17                         (function show() {
    18                             if (changeH > height) {clearTimeout(id);return}
    19                             changeH += 5;
    20                             opacity += 0.06;
    21                             //console.log('opacity:'+adiv.style.opacity+',height :'+adiv.style.height);
    22                             adiv.style.height = changeH + 'px';
    23                             adiv.style.opacity = opacity;
    24                             adiv.style.display = 'block';
    25                             id = setTimeout(function () {
                         clearTimeout(id);
    26 show(); 27 }, 16.7); 28 })(); 29 30 bool = false; 31 } else { 32 changeH = height; 33 opacity = 1; 34 target.innerHTML = '财经 +'; 35 (function hidden() { 36 if (changeH < 0) {clearTimeout(id);adiv.style.display = 'none';return} 37 changeH -= 10; 38 opacity -= 0.11; 39 //console.log('opacity:'+adiv.style.opacity+',height :'+adiv.style.height); 40 adiv.style.height = changeH + 'px'; 41 adiv.style.opacity = opacity; 42 id = setTimeout(function () {
                         clearTimeout(id);
    43 hidden(); 44 }, 16.7); 45 })(); 46 bool = true; 47 } 48 } 49 }
    注意几点:
      1.记得清除setTimeout的ID,然后退出,否则死循环,如if (changeH < 0) {clearTimeout(id);adiv.style.display = 'none';return}
      2.setTimeout的延迟时间设置为16.7是因为符合屏幕的刷新率60FPS,看着舒服
      3.调试过程中,设置changeH和opacity的递增递减值时,记得打印出来,方便调试:console.log('opacity:'+adiv.style.opacity+',height :'+adiv.style.height);
      4.最后,整个菜单的实现中,最关键的是下面这一句,如果没有这一句,你无法完美实现所有功能,比如:你点开一组子菜单,然后移动到其它组点击的时候,情况将有很大不同;而window.getComputedStyle用这个的原因是,首次打开时,点任意组的第一下都没反应,因为直接通过event.target在点第一下时是取不到opacity值的。
    if (window.getComputedStyle(adiv,null).opacity>0.5){bool=false}else{bool=true};

    不过,IE9以下不支持getComputedStyle方法,IE的Element对象有currentStyle属性;

    如果你对CSS的处理不是很熟悉,看看我的总结:
    用原生JS读写CSS样式的方法总结
    如果你想多了解setTimeout()方法的应用,看看这个:你真的知道setTimeout是如何运行的吗
    对于事件的处理机制,可以看看这个:DOM中的事件处理概览与原理的全面剖析
    好了,到此结束,有不对和更好的地方,欢迎指教!

    
    
  • 相关阅读:
    html background 背景颜色美化 类似毛玻璃
    HTML
    export、exports、modules.exports 和 require 、import 的一些组合套路和坑
    C#实现监控网络流量
    PHP乱码问题,UTF-8(乱码)
    LitDB笔记
    LitDB文章
    NoSQL 35 个非主流数据库
    mysql中int转varchar
    CSS设置DIV背景色渐变显示
  • 原文地址:https://www.cnblogs.com/susufufu/p/5768402.html
Copyright © 2011-2022 走看看