原生JS

原生态JS是指遵循ECMAScript标准的javascript,在不使用框架的情况下考察我们对于对JS语法的了解和API的掌握。虽然目前大都是根据框架来进行前端的编程,但对于原生JS的掌握还是很重要的吧,本代码实现的效果如下:

根据传入的图片数量在页面的底部呈现轮播图的图片数量和当前为第几张图片,点击相应的圆圈则可以跳转至其他图片;左右两个按钮同样可以实现图片的切换,如果当前为第一张或者最后一张图片,点击左移或者右移按钮即可完成循环操作,从第一张图片跳转至最后一张图片,且切换图片的效果还在。具体的实现如下:

思路阐述

HTML部分

既然说是原生JS实现,那么html和CSS我们能少用就少用。整个html内,仅写入一个div,其CSS格式表示其铺满整个屏幕,引入额外写的index.js后,在body下新建一个Slider对象,并插入我们要轮播的图片。整体代码仅此而已,其详细代码如下:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>last experiment</title>
    <script src="index.js"></script>
</head>
<body>
    <div style="
                position: fixed;
                top:0px;
                bottom: 0px;
                left: 0px;
                right: 0px;
                background-color: #333;">
    </div>
</body>
<script>
    new Slider(
        document.querySelector('div'),
        [
            'images/1.png',
            'images/3.jpg',
            'images/25.png'
        ]
    )
</script>
</html>

JS部分

我的思路是将所有图片加载在一个div中,我们称该div之为innerct,该div采取fixed布局,让图片在一整行内显示,每个图片与该div的大小一致。根据该div平移的距离即可实现页面呈现的图片的切换,同时带平移效果。

思路实现的目标就是下图第一行的内容,红框表示当前展示的图片,如现在页面展示的就是编号为 Ⅱ 的图片。但是假如当前图片为第一张,同时我又按下了左移按钮,根据轮播图的特性,我应该跳转至图片三,同时页面呈现出图片向右滑动,即左移的效果,呈现出编号为 Ⅲ 的图片。

为了实现这个效果,那么innerct内的图片排列应该是下图第二行的这种情况。同时,当页面呈现的图片为第二行的第一个 编号为 Ⅲ 时,将此时的第二行的第一个 Ⅲ 瞬移至 第二行的第四个图片 Ⅲ 。同理,当页面呈现第二行的第五个图片 Ⅰ 时,再将图片瞬移至第二行的第二个 Ⅰ 图片即可不露声色的完成轮播图的效果实现。那么此时我们需要解决的就是如何实现“瞬移“操作。

图片的平移效果我们采取了CSS样式中的过渡效果,那我们将平移的过程设置在在1s内完成,就可以展示出平移的过程,而页面不会呈现出图片的瞬间切换。根据这个平移的实现思路,我们就可以设置当页面呈现的图片为第二行的第一个Ⅲ 和 第五个 Ⅰ时,将平移的过程设置在0s完成,将图片瞬移至第二行的第四个 Ⅲ 和 第二行的第二个 Ⅰ。其他位置时,平移过程的过渡效果再次恢复为1s内完成即可。

代码实现

根据以上的思路,我们就可以进行JS的编程了。代码分为两部分一个是Slider类,来实现轮播图对象;一个是工具函数来实现异步睡眠操作。由于前端的知识点比较杂糅,具体的讲解,我在代码中进行注释来一一说明,下面我们开始 >灬<。

class Slider {
    constructor(ct, imgs) { //该类传入的参数为一个dom元素和要轮播图片的list

        this.currentPage = 0; //当前页面展示的图片下标
        this.pageCount = imgs.length; //轮播图的数量 aka,图片list的 大小
        // let width = window.innerWidth;//全屏
        let width = ct.offsetWidth; //此处实现的是让图片全屏显示,使用上面的语句也可以实现,但由于本次html中只有一个div,用该div的外轮廓宽也是可以的。
        this.width = width;
        let height = ct.offsetHeight;

        this.innerCt = document.createElement('div');
        this.innerCt.style.cssText = `width: ${width * (imgs.length + 2)}px;//设置图片所在div的宽度,其宽度为 图片的数据 + 首部的最后一张图片 和 尾部的第一张图片,aka,图片list的长度 + 2。
        height:100%;padding:0;margin:0;transition:transform 1s ease;left:0`;//设置transition 过渡效果,平移 持续时间为1s 过渡方式为ease
        // 'width:' + width * imgs.length + 'px;' + 'height:100%;padding:0;' + 'margin:0;' + 'transition:transform 1s ease';//ease先快后慢;放图像个数的窗口
        ct.appendChild(this.innerCt);//将innerct放入到dom元素中去

        let circleCt = document.createElement('div');//此处为页面底部的圆圈
        circleCt.style.cssText = `position:fixed;opacity:0.6;bottom:0px;width:100%;padding:10px 0`;//让圆圈布局在底部
        circleCt.setAttribute('align', 'center');//效果 剧中
        ct.appendChild(circleCt);
        this.circles = [];

        let ct3 = document.createElement('div');//新建一个div,将最后一个图片放入其中,再将该div放入innerct中
        ct3.style.cssText = `width:${width}px;height:${height};float:left`;
        ct3.setAttribute('align', 'center');//左右居中
        let img3 = new Image();
        img3.src = imgs[this.pageCount - 1];
        img3.style.cssText = `max-width:${width}px;max-height:${height}px`;
        img3.onload = () => {
            img3.style.marginTop = (height - img3.height) / 2 + 'px';//垂直居中
        }
        ct3.appendChild(img3);
        this.innerCt.appendChild(ct3);

        imgs.forEach((item, index) => {//将图片按照 list的顺序放入其中,并在其中添加圆圈的点击事件
            let ct1 = document.createElement('div');
            ct1.style.cssText = `width:${width}px;height:${height};float:left`;
            // 'width:' + width + 'px;' + 'height:' + height + 'px;' + 'float:left;';
            ct1.setAttribute('align', 'center');//左右居中
            let img = new Image();
            img.src = item;
            img.style.cssText = `max-width:${width}px;max-height:${height}px`;
            // 'max-width:' + width + 'px;' + 'max-height:' + height + 'px;';
            img.onload = () => {
                img.style.marginTop = (height - img.height) / 2 + 'px';//垂直居中
            }
            ct1.appendChild(img);
            this.innerCt.appendChild(ct1);
            let c = document.createElement('div');
            c.style.cssText = `width:20px;height:20px;
            border-radius:10px;background-coclor:white;
            display:inline-block;margin-right:10px;`;
            this.circles.push(c);
            circleCt.appendChild(c);
            c.addEventListener('click', () => {
                this.sliderTo(index);//点击小圆点有滑动的效果
            });
        });
        let ct2 = document.createElement('div');//新建一个div,将第一个图片放入其中,再将该div放入innerct中
        ct2.style.cssText = `width:${width}px;height:${height};float:left`;
        ct2.setAttribute('align', 'center');//左右居中
        let img2 = new Image();
        img2.src = imgs[0];
        img2.style.cssText = `max-width:${width}px;max-height:${height}px`;
        img2.onload = () => {
            img2.style.marginTop = (height - img2.height) / 2 + 'px';//垂直居中
        }
        ct2.appendChild(img2);
        this.innerCt.appendChild(ct2);

        let css = `position:absolute;//es6的新特性,反引号,可以查一下,使用极其方便,好像是加载速度不如''。
        top:50%;
        margin-top: -20px;
        height:40px;opacity:0.6;
        margin:0px auto;
        width:40px;
        padding:0 20px
        line-height:40px;
        background-color:#bbb;
        font-size:28px;
        cursor:pointer;`;
        let btnleft = document.createElement('div')
        btnleft.innerHTML = '<';
        btnleft.style.cssText = css;
        btnleft.style.left = '0px';

        let btnright = document.createElement('div')
        btnright.innerHTML = '>';
        btnright.style.cssText = css + `text-align: right;`;
        btnright.style.right = '0px';

        ct.appendChild(btnleft);
        ct.appendChild(btnright);//设置左右两个切换按钮,其下面所添加的点击事件就是之前思路中所强调的部分。

        btnleft.addEventListener('click', () => {
            if (this.currentPage === 0) {//当在第一个图片时点击了左移按钮。
                this.circles[this.currentPage].style.backgroundColor = 'white';当前的底部圆圈颜色变白。
                this.circles[this.pageCount - 1].style.backgroundColor = 'red';最后一个图片的底部圆圈的变为选中的颜色 红色
                let left = 0 * this.width;此时innerct平移的距离为 0 根据此前innerct里面插入div 的顺序,那么此时呈现的就是最后一个图片。
                this.innerCt.style.transform = 'translate(' + left + 'px,0px)';
                sleep(1000).then(() => {睡眠1秒完成,此处必须使用es6中的箭头函数,不如会引起this指向的改变
                    this.innerCt.style.transitionDuration = "0s";调用js中的 transition的api 将过渡时间设置为0
                    this.sliderTo(this.pageCount - 1);常规的移动图片至最后一张
                });
                sleep(1100).then(() => {同样等待上面的操作完成后,再将过渡时间设置为1s,此处只需要比上面完成的时间少一点即可
                    this.innerCt.style.transitionDuration = "1s";
                })
                return;
            }
            this.sliderTo(this.currentPage - 1);//当不涉及特色情况时,进行左移即可。
        });
        btnright.addEventListener('click', () => {//同理
            console.log('pagecount', this.currentPage);
            if (this.currentPage === this.pageCount - 1) {
                this.circles[this.currentPage].style.backgroundColor = 'white';
                this.circles[0].style.backgroundColor = 'red';
                let left = -(this.pageCount + 1) * this.width;
                this.innerCt.style.transform = 'translate(' + left + 'px,0px)';
                this.currentPage = 0;
                sleep(1000).then(() => {
                    this.innerCt.style.transitionDuration = "0s";
                    this.sliderTo(0);
                });
                sleep(1100).then(() => {
                    this.innerCt.style.transitionDuration = "1s";
                })
                return;
            }
            this.sliderTo(this.currentPage + 1);


        })
        for (let i = 0; i < this.circles.length; ++i) {
            this.circles[i].style.backgroundColor = 'white';
        }//底部的圆圈初始化

        this.circles[0].style.backgroundColor = 'white';
        this.circles[0].style.backgroundColor = 'red';
        let left = -1 * this.width;
        this.innerCt.style.transform = 'translate(' + left + 'px,0px)';
        this.currentPage = 0;//设置显示list中的第一个图片

    }
    sliderTo(num) {
        this.circles[this.currentPage].style.backgroundColor = 'white';
        this.circles[num].style.backgroundColor = 'red';//底部圆圈的切换
        let left = -(num + 1) * this.width;//innerct 平移距离的改变
        this.innerCt.style.transform = 'translate(' + left + 'px,0px)';
        this.currentPage = num;//当前照片的改变
        console.log('pagecurrent::::', this.currentPage);
    }
}

function sleep(time) {//es6增加的异步操作,此处需要异步来完成睡眠
    return new Promise((resolve) => setTimeout(resolve, time));
}

上面代码讲的十分详细了,我就不过多赘述了。

碎碎念

又到了每次必有的碎碎念环节了,其实一开始不是很想写这个博客的,昨晚偶然看见了Love ,Death & Robot 第三季的宣传动画,我不知去向的小熊猫就又出来了。5.20日,这部科幻动画天花板第三季回归,让我看看网飞会再度给我们带来什么样的惊喜吧!!!

下面写点关于自己的东西吧,从川北野生小熊猫到不知去向小熊猫。这其中经历了不少的心路历程,哎,自己的迷茫还是只有自己知道啊,疯癫过,伤心过,愤怒过。但是!!既然选择了就旁若无人地继续走吧,去寻觅地探索吧,让自己的title能再变一下,从川北野生,到不知去向,在到什么呢,让我们共同努力,去找到答案吧。

杰诺德姆, See u in the future.