setTimeout应用

自己写的很啰嗦,所以为转载文章


result.gif

html

<div id="nav" class="container">
        <ul>
            <li>
                <a href="javascript:;">站长之家</a>
            </li>
            <li>
                <a href="javascript:;">行业资讯</a>
                <div class="subnav">
                    <em class="arrow"></em>
                    <p>
                        <span>
                            <a href="javascript:;">业界动态</a>|
                            <a href="javascript:;">收购融资</a>|
                            <a href="javascript:;">门户动态</a>|
                            <a href="javascript:;">搜索引擎</a>|
                            <a href="javascript:;">网络游戏</a>|
                            <a href="javascript:;">电子商务</a>|
                            <a href="javascript:;">广告传媒</a>|
                            <a href="javascript:;">厂商开发</a>
                        </span>
                    </p>
                </div>
            </li>
            <li><a href="javascript:;">站长在线</a>
                <div class="subnav">
                    <em class="arrow"></em>
                    <p>
                        <span>
                            <a href="javascript:;">站长报道</a>|
                            <a href="javascript:;">好站推荐</a>|
                            <a href="javascript:;">站长聚会</a>|
                            <a href="javascript:;">站长访谈</a>|
                            <a href="javascript:;">站长茶馆</a>|
                            <a href="javascript:;">站长排行</a>
                        </span>
                    </p>
                </div>
            </li>

            <li><a href="javascript:;">网站运营</a>
                <div class="subnav">
                    <em class="arrow"></em>
                    <p>
                        <span>
                            <a href="javascript:;">建站经验</a>|
                            <a href="javascript:;">策划盈利</a>|
                            <a href="javascript:;">搜索优化</a>|
                            <a href="javascript:;">网站推广</a>|
                            <a href="javascript:;">免费资源</a>
                        </span>
                    </p>
                </div>
            </li>

            <li><a href="javascript:;">设计在线</a>
                <div class="subnav">
                    <em class="arrow"></em>
                    <p>
                        <span>
                            <a href="javascript:;">酷站推荐</a>|
                            <a href="javascript:;">网页设计</a>|
                            <a href="javascript:;">WEB标准</a>|
                            <a href="javascript:;">视频处理</a>|
                            <a href="javascript:;">设计活动</a>
                        </span>
                    </p>
                </div>
            </li>

            <li><a href="javascript:;">网络编程</a>
                <div class="subnav">
                    <em class="arrow"></em>
                    <p>
                        <span>
                            <a href="javascript:;">Asp编程</a>|
                            <a href="javascript:;">Php编程</a>|
                            <a href="javascript:;">.Net编程</a>|
                            <a href="javascript:;">Xml编程</a>|
                            <a href="javascript:;">Access</a>|
                            <a href="javascript:;">Mssql</a>|
                            <a href="javascript:;">Mysql</a>
                        </span>
                    </p>
                </div>
            </li>

            <li><a href="javascript:;">行业资讯</a>
                <div class="subnav">
                    <em class="arrow"></em>
                    <p>
                        <span>
                            <a href="javascript:;">联盟动态</a>|
                            <a href="javascript:;">联盟介绍</a>|
                            <a href="javascript:;">联盟点评</a>|
                            <a href="javascript:;">网赚技巧</a>
                        </span>
                    </p>
                </div>
            </li>

            <li><a href="javascript:;">服务器</a>
                <div class="subnav">
                    <em class="arrow"></em>
                    <p>
                        <span>
                            <a href="javascript:;">Web服务器</a>|
                            <a href="javascript:;">Ftp服务器</a>|
                            <a href="javascript:;">Mail服务器</a>|
                            <a href="javascript:;">Dns服务器</a>|
                            <a href="javascript:;">Win服务器</a>|
                            <a href="javascript:;">Linux服务器</a>|
                            <a href="javascript:;">安全防护</a>|
                            <a href="javascript:;">虚拟主机</a>
                        </span>
                    </p>
                </div>
            </li>

        </ul>
    </div>

导航一般都用ul括起来

<ul>
    <li><a></a></li>
    <li><a></a></li>
</ul>

<a>标签中的href不为空,设置为"javascript:;",因为这样点击页面不会刷新
每个元素的背景图片都是同一张图片的不同位置,所以subnav的背景是副标题的最左边,p的背景是副标题 的最右,span的背景是所有链接的背景

css

基本css

body,div,ul,li,p{
            margin:0;
            padding:0;
        }
        body{
            font:12px/1.5 Arial;
        }
        ul{
            list-style-type:none;
        }

添加背景图片

#nav,#nav ul,#nav ul li,#nav ul li a:hover,#nav .subnav,#nav .subnav p,#nav .subnav p span,#nav .subnav .arrow{
            background:url(http://fgm.cc/learn/lesson4/img/nav_bg.png) no-repeat;
        }

元素样式

/*最外层设置宽度,位置*/
        #nav{
            position:relative;/*对二级标题的位置设置有用*/
            width:910px;
            background-position:0 -36px;/*这是设置最左边的背景,属性值是相对于原来的位置偏移多少*/
            margin:10px auto;
        }
        /*ul设置高度还有整体的导航栏的左右宽度,多余的部分都隐藏起来*/
        #nav ul{
            height:36px;
            line-height:36px;
            margin-left:10px;
            padding-right:10px;
            overflow:hidden;
            background-position:right -72px;/*设置最右边的背景*/
        }
        /*ul的li需要全部靠左浮动,浮动后元素就可以设置宽高*/
        #nav ul li{
            float:left;
            width:110px;
            margin-left:-2px;
            background-position:0 -108px;
        }
        /*主要是设置字大小,颜色,居中显示,去掉<a>原来的修饰*/
        #nav ul li a{
            font-size:14px;
            color:#fff;
            width:102px;
            display:block;
            text-align:center;
            text-decoration:none;
            margin:0 2px 0 4px;
        }
        /*设置hover后的变化*/
        #nav ul li a:hover{
            font-weight:700;
            background-position:-3px -144px;
        }
        /*下面是设置二级标题,类似于一级标题*/
        #nav .subnav{
            display:none;
            position:absolute;
            top:41px;
            width:auto!important;
            min-width:110px;
            height:27px;
            line-height:27px;
            white-space:nowrap;/*规定段落中的文本不进行换行*/
            background-position:0 -180px;
        }
        #nav .subnav p{
            margin-left:10px;
            padding-right:10px;
            background-position:right -234px;
        }
        #nav .subnav p span{
            display:block;
            color:#235e99;
            background-repeat:repeat-x;/*沿着x轴的方向重复*/
            background-position:0 -207px;
        }
        #nav .subnav p a{
            font-size:12px;
            display:inline;/*a标签会继承父元素的display*/
            color:#235e99;
            text-decoration:none;
            margin:0 5px;
            padding:0 2px;
        }
        #nav .subnav p a:hover{
            font-weight:400;
            background-image:none;
            border-bottom:2px solid;
        }
        #nav .subnav .arrow{
            position:absolute;
            top:-4px;
            display:block;
            width:11px;
            height:5px;
            background-position:0 -261px;
        }

背景图片


nav_bg.png

background-positon背景图片显示的起始位置,负值代表图片向左向上移动,正值代表图片向右向下移动
ol ul 本身的padding-left有40px
em.arrow为箭头
subnav的背景图为左边圆角, p为右边圆角, span的背景图为内容背景图,沿x轴方向重复

JS部分

var get = {
        byId: function (id) {
            return document.getElementById(id)
        },
        byClass: function (sClass, oParent) {
            // 返回父元素oParent内类名为sClass的元素
            var aClass = [];
            var reClass = new RegExp("(^|)"+sClass+"(|$)");
            var aElem = this.byTagName('*', oParent);
            for (var i=0; i<aElem.length; i++) {
                reClass.test(aElem[i].className) && aClass.push(aElem[i]);
            }
            // return 放在for循环中,就不能循环,执行到return直接结束,return后的console.log()也不能执行
            return aClass;
        },
        byTagName: function (elem, obj) {
            // 返回 obj 下的 名为 elem 的 tag
            return (obj || document).getElementsByTagName(elem);
        }
    };

    window.onload = function () {
        var oNav = get.byId('nav');
        var aLi = get.byTagName('li', oNav);
        var aSubNav = get.byClass('subnav', oNav);
        var oSubNav = oEm = timer = null;
        var i = 0;

        for (i=1; i<aLi.length; i++) {
            aLi[i].onmouseover = function () {
                // 隐藏所有子菜单
                for (i=0; i<aSubNav.length; i++) aSubNav[i].style.display = 'none';
                // 获取该项下的子菜单
                oSubNav = get.byClass('subnav', this)[0];
                console.log(oSubNav);
                // 获取该项下的指示箭头
                oEm = get.byTagName('em', this)[0];
                // 显示该项下的子菜单
                oSubNav.style.display = 'block';
                // 判断显示区域
                console.log(oNav.offsetWidth - this.offsetLeft > oSubNav.offsetWidth)
                oNav.offsetWidth - this.offsetLeft > oSubNav.offsetWidth ?
                    // 如果在显示范围居左显示, 超出显示范围居右显示
                    oSubNav.style.left = this.offsetLeft + 'px' : oSubNav.style.right = 0;
                // 计算指标箭头显示位置
                oEm.style.left = this.offsetLeft - oSubNav.offsetLeft + 50 + 'px';
                clearTimeout(timer);
                // 阻止冒泡事件
                oSubNav.onmouseover = function (event) {
                    (event || window.event).cancelBubble = true;
                    clearTimeout(timer)
                }
            };

            aLi[i].onmouseout = function () {
                timer = setTimeout(function () {
                    oSubNav.style.display = 'none'
                }, 300)
            }
        }

    }

在主导航向subnav移动鼠标时,为防止鼠标一从主导航离开,subnav立刻小时,所以通过setTimeout延迟了300毫秒,在鼠标移动到subnav时,清除timer.

这样做的弊端就是在主导航之间滑动时,鼠标离开其中一块,移动到另一块区域时,timer仍在继续,所以要在每一个onmouseover中添加cleartimeout(timer)
(event || window.event).cancelBubble = true;阻止冒泡事件发生,防止在触发子元素的onmouseover事件时,会触发父元素的onmouseover事件,使其执行了回调函数,所以此处阻止冒泡,就不能触发父元素的onmouseover事件了

HTMLElement.offsetWidth 是一个只读属性,返回一个元素的布局宽度。offsetWidth是测量包含元素的边框(border)、水平线上的内边距(padding)、竖直方向滚动条(scrollbar)(如果存在的话)、以及CSS设置的宽度(width)的值。

HTMLElement.offsetLeft是一个只读属性,返回当前元素左上角相对于 HTMLElement.offsetParent 节点的左边界偏移的像素值。
HTMLElement.offsetParent指向的是离domElement最近的拥有position:relative/absolute属性的父级元素。若不存在,则指向 <body>元素。

父元素

子元素/二级导航

判断显示区域就是判断二级导航显示位置,根据主导航的总宽度减父元素左边界的位置, 所得长度如果大于二级导航的宽度,则设置二级导航的left为父元素的左边界的位置,否则,设置二级导航的right为0, 二级导航为绝对定位
箭头的位置 父元素的左边界 减 二级导航的 左边界 加 50px ,这样就能使箭头处于父元素的中间位置

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,293评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,604评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,958评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,729评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,719评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,630评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,000评论 3 397
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,665评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,909评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,646评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,726评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,400评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,986评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,959评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,996评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,481评论 2 342

推荐阅读更多精彩内容

  •   JavaScript 与 HTML 之间的交互是通过事件实现的。   事件,就是文档或浏览器窗口中发生的一些特...
    霜天晓阅读 3,470评论 1 11
  • HTML 5 HTML5概述 因特网上的信息是以网页的形式展示给用户的,因此网页是网络信息传递的载体。网页文件是用...
    阿啊阿吖丁阅读 3,828评论 0 0
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,426评论 25 707
  • CSS 是什么 css(Cascading Style Sheets),层叠样式表,选择器{属性:值;属性:值}h...
    崔敏嫣阅读 1,469评论 0 5
  • 介绍 Gensim是一个用于从文档中自动提取语义主题的Python库,足够智能,堪比无痛人流。Gensim可以处理...
    迅速傅里叶变换阅读 30,360评论 0 5