那些年原生js实现的楼层跳转

  最近做一个需求~~楼层跳转(京东、淘宝侧边导航),由于现在项目都用框架,所以 jquery是不能再用了,只好自己原生写一个,其实实现起来很简单,无非就是获取到每个楼层距离文档顶部的距离,然后通过控制滚动条来实现,麻烦的是需求在变,由于突然引出了一个锚点导航遇到了fixed,所以只好重新思考,满足需求。

  简单的楼层跳转样例可以先看一下之前写的demo(https://lewiscutey.github.io/yintai),看起来效果很好吧,下面详解其中的坑:

1.锚点导航遇到fixed

  现在布局一般都要box-sizing:border-box/content-box,所以不要再用padding-top:50px;margin-top:-50px;这种写法了,最好直接写一个子元素来定位,这样既不占空间,又可以精确定位,下面是简易的demo:

.floor {
    position: relative;
    #anchor{
        position: absolute;
        top: -56px;
        width: 100%;
    }
}

2.子元素永远比父元素层级z-index高

  大家总是习以为常的拿z-index来决定层级,当有时候就算z-index:9999!important;还是不起作用的时候,也许就是父子元素定位出了问题,这样唯一的解决办法就是给他们在加一个父元素,原来的父子元素现在变成了兄弟方可有效,呜呜,不幸中招。

3.慎用getBoundingClientRect()

  大家可能对这个方法有点陌生,其实就是用来获取dom节点对于bom的相对位置,今天要说的这个坑是当页面中有锚点链接时,直接路由进来的页面会显示到锚点的内容区域,从而通过getBoundingClientRect()获取到的值就会乱套,因此如果用作楼层跳转里楼层高度获取值时,就会失效了,所以慎用这个获取相对位置的方法,我们一般用offsetTop来获取dom节点对于bom的相对位置,这就要求把该dom直接作为body的子元素,经检测方便可靠,下面是一个简单的demo:

let flag = true;
let floorHeights = [];
let slideBar = document.querySelector("#slide-bar");
let floors = document.querySelectorAll(".floor");
let floorLinks = document.querySelectorAll("#slide-bar .floor-link");
for (let i = 0; i < floors.length; i++){
    floorHeights.push(floors[i].offsetTop - 56);
}
window.onscroll = function() {
    let scrollTop = document.body.scrollTop||document.documentElement.scrollTop;
    if(scrollTop > 750){
        if(flag){
            flag = false;
            slideBar.style.bottom = '50%';
        }
    }else{
        if(!flag){
            flag = true;
            slideBar.style.bottom = '-200%';
        }
    }
    for (let i = 0; i < floorHeights.length; i++) {
        if(floorHeights[i] < scrollTop + 200){
            for (let j = 0; j < floorHeights.length; j++) {
                floorLinks[j].style.opacity = 0;
            }
            floorLinks[i].style.opacity = 1;
        }
    }     
}
window.onscroll();
for (let i = 0; i < floorLinks.length; i++) {
    floorLinks[i].index = i;
    floorLinks[i].onmouseover = function(){
        for (let j = 0; j < floorLinks.length; j++) {
            floorLinks[j].style.opacity = 0;
        }
        this.style.opacity = 1;
    }

    floorLinks[i].onclick = function() {
        for (let j = 0; j < floorLinks.length-1; j++) {
            floorLinks[j].style.opacity = 0;
        }
        document.body.scrollTop = floorHeights[this.index];
        document.documentElement.scrollTop = floorHeights[this.index];
    }

    floorLinks[floorLinks.length-1].onclick = function() {
        document.body.scrollTop = 0;
        document.documentElement.scrollTop = 0;
    }
}

   哈哈,命名即思维,我相信通过代码大家可以弄的一清二楚,下面附上经典老图(一图胜千言)!

  以后要把这些琐碎的知识点记录下来,就如阮一峰老师所说的:炫耀从来不是我的动机,好奇才是,一边学习,一边记录!

posted @ 2017-11-03 22:54  新哥·lewis  阅读(3184)  评论(0编辑  收藏  举报