JavaScript练手小技巧:我破解了原神官网全屏滚动的秘密

news2025/1/6 19:02:43

这个标题有点夺人眼球,哈啊哈~骗点击率的。

“原神”官网当真的做的很漂亮,虽然我没玩过这个游戏,但是禁不住喜欢这个网站啊。

https://ys.mihoyo.com/

最近居家教学上网课。除了上课,实在不想做学校安排的其它任务,太烦了。果然在家没的工作动力啊~ 但是学习、coding、追番的动力还是有的。

今天来做一个模仿原神官网的全屏滚动效果页面。 以前这种页面用的是 fullpage 插件,网上的文章很多,就不再提它了。今天用原生的JS来写一个,结合了CSS3动画。

一、HTML结构

本demo页面制作设计了5个板块。读者大大可以根据自己的需要任意增加板块。

<!-- 内容板块 -->
<div class="wrap"  id="wrap">
    <div class="section"><span>01</span></div>
    <div class="section"><span>02</span></div>
    <div class="section"><span>03</span></div>
    <div class="section"><span>04</span></div>
    <div class="section"><span>05</span></div>
</div>
<!-- 内容板块 end-->
<!-- 控制块,方便用户切换板块 -->
<div class="ctrlBtns"  id="ctrlBtns">

</div>
<!-- 控制块 end -->

二、CSS样式

给 span 标签写了个小动画,这么做的目的是扩展后期更复杂的动画需要。

注意.section.show 是给 section 类设定了一个动画显示类 show让有这个类的时候,其下的span标签执行一个过渡动画。

以下CSS可以屏蔽掉浏览器的滚动条:

body{
    overflow: hidden;
}

为了让 wrap 能丝滑运动,我让它绝对定位了。

完整CSS代码。 

body{
    overflow: hidden;
}
.wrap{
    position: absolute;
    width: 100%;
}
.section{
    width: 100%;
    height: 100vh;
    font-size: 60px;
}
.section:nth-child(1){
    background: #ffdddd;
}
.section:nth-child(2){
    background: #ddffe4;
}
.section:nth-child(3){
    background: #d7dcfd;
}
.section:nth-child(4){
    background: #ffddf1;
}
.section:nth-child(5){
    background: #ddf8ff;
}

.ctrlBtns{
    position: fixed;
    right:20px;
    top:50%;
    width: 20px;
}
.ctrlBtns a{
    display: block;
    width: 20px;
    height: 20px;
    background: #fff;
    opacity: 0.8;
    margin-top: 10px;
    margin-bottom: 10px;
}
.ctrlBtns a.current{
    background: #f90;
    opacity: 1;
}

/* 做一个小动画作为示例。可以根据项目需要设计更复杂的动画 */
.section span{
    transition: all 0.5s;
    display: inline-block; /* 因为 inline 标签变形 transform 无效*/
}
.section.show  span{
   transform: translateX(300px);
}

 三、JavaScript代码

这次代码有几个关键点:

1. 防止浏览器刷新后,停留在当前位置

如果浏览器不是强制刷新(Ctrl + F5),而是普通刷新(点击刷新按钮,或者按下 F5 刷新),则页面重新载入完毕后会调到之前访问的位置。

这对全屏页面简直就是个灾难。因为,刷新后一些数据都初始化了。必须让页面回到初始的位置,初始的样子。借用以下代码可以实现 。在页面的任意位置执行下面几行 JS 代码就可以了:

// 防止浏览器刷新后,停留在当前位置。
if (history.scrollRestoration) {
    history.scrollRestoration = 'manual';
}

history.scrollRestoration 有两个属性值:

  • auto:默认值,表示滚动位置会被存储。
  • manual:单词的意思就是手动。表示,滚动的位置不会被存储。

2. 鼠标滚轮事件

可以参考我这篇文章:鼠标滚轮事件_stones4zd的博客-CSDN博客

3. 获取兄弟标签

 可以参考我这篇文章:https://blog.csdn.net/weixin_42703239/article/details/88764774

4. 获取窗口高度

let wh = window.innerHeight ;

 5. 过渡动画结束事件

之前我写过一篇文章,写了帧动画的结束事件,JavaScript 练手小技巧:animationend 事件及其应用小案例_stones4zd的博客-CSDN博客

其实过渡动画也有类似事件:transitionend

 6. 思路

  • 整个效果重点是切换板块

切换板块其实就是移动 wrap 标签。移动它,更改的就是它的 top 属性值。

定义一个全局的索引变量 index,top 值就是 index * 窗口高度

  • 整个效果是通过滚动鼠标滚轮驱动

要借用鼠标滚轮事件 wheel 。当滚动鼠标的时候,让 index 加1 或者 减 1 ,进而更改 wrap 的top属性。

重点:当正在滚动的时候,不能再次驱动滚动事件。防止一次性滚动过多。必须等这次滚动动画(本质执行transition过渡动画)执行完毕后,才能开始下一次滚动。

因此,使用了一个开关变量 isScrolling,默认值 true,默认允许滚动。当滚动开始的时候,就要让它为false。滚动结束,就要变为 true。

完整 JavaScript 代码,里面写好了注释:

let index = 0;
let wrap = document.getElementById("wrap");
let ctrlBtns = document.getElementById("ctrlBtns");
let sections = wrap.getElementsByClassName("section");
let secNum = sections.length ; // 页面板块个数
let isScrolling = true;    // 滚动开关。true,允许滚动;false,不允许滚动
/*
* 开关的目的是,当页面正在滚动的时候,让滚动事件不再执行代码。
* */
// 防止浏览器刷新后,停留在当前位置。
if (history.scrollRestoration) {
    history.scrollRestoration = 'manual';
}

/*
* 工具方法:找兄弟标签
* 参数:目标标签
* 返回值:目标标签的兄弟标签数组
* */
function findSiblings(tag){
    let parent = tag.parentNode;
    let children = parent.children;
    let siblings = [];
    for(let i=0; i<=children.length-1; i++){
        if( children[i] !== tag ){
            siblings.push( children[i] );
        }
    }
    return siblings;
}

/* 板块切换函数
* 参数:window.innerHeight 窗口高度
* index:当前显示的板块的索引值,全局变量
* */
function scrollWrap(wh,index){
    // 板块切换
    wrap.style.top = `-${wh*index}px`;
    //  控制块切换
    let btnIndex = ctrlBtns.children[index]; // index 对应的控制块
    let sibilingsA = findSiblings(btnIndex);   // 它的兄弟们
    // btnIndex 要添加 current,用以突出显示
    btnIndex.classList.add("current");
    // 兄弟们要去掉类 current
    sibilingsA.forEach(function (v,i) {
        v.classList.remove("current");
    })
}

/*
* 初始化函数:
* 初始索引变量 index 为 0
* 初始 wrap 的 top 值为 0
* 生成控制块
* 默认第一个 section 执行动画。
* */
function init(){
    index = 0;
    wrap.style.top  = 0;
    // 生成控制块
    for(let i=0; i <= secNum-1; i++){
        let a = document.createElement("a");
        a.href = "javascript:void(0)";
        ctrlBtns.appendChild(a);
    }
    let ctrlHeight = ctrlBtns.offsetHeight ;
    // 调整控制块垂直方向的位置。
    ctrlBtns.style.marginTop = - ctrlHeight/2 + "px";
    // 默认控制块第一个 a 为突出显示。
    ctrlBtns.children[0].classList.add("current");
    // 点击控制块,实现板块切换
    for(let i=0; i <= secNum-1; i++){
        ctrlBtns.children[i].addEventListener("click",function(){
            index = i ;
            // 切换板块
            scrollWrap(window.innerHeight,index);
        });
    }
    // 默认第一个标签执行动画
    sections[index].classList.add("show");
}
function doScroll(e){
    // 开关为 true:才执行滚动处理函数
    if(isScrolling ){
        // 进入开关,就关闭开关,防止多次滚动处理。
        wrap.style.transition ="all 0.5s";
        isScrolling = false;
        let evt = e || window.event;
        let wh = window.innerHeight ;
        evt.preventDefault();   // 阻止浏览器默认事件
        if( evt.deltaY>0 ){
            console.info("页面向上滚动");
            index++;
            if( index >= secNum ){  // 当在最后一页的时候,还要滚动鼠标
                index = secNum-1 ;
                isScrolling = true ;
                wrap.style.transition ="none"; // 去掉过渡效果,防止用户拖拉窗口,画面异常
            }else{
                scrollWrap(wh,index);
            }
        }else{
            console.info("页面向下滚动");
            index--;
            if(index<0){   // 当在第一页的时候,用户还要滚动鼠标
                index = 0; // 保持 index 为 0
                isScrolling = true ;  // 允许滚动
                wrap.style.transition ="none"; // 去掉过渡效果,防止用户拖拉窗口,画面异常
            }else{
                scrollWrap(wh,index);
            }

        }
        console.info( evt.type,evt.deltaY, index);
    }

}

window.addEventListener("wheel",doScroll,{ passive: false });
// 让过渡动画执行完毕的时候,就开启开关,允许滚动。
wrap.addEventListener("transitionend",function(){
    isScrolling = true;   // 允许滚动
    wrap.style.transition ="none"; // 去掉过渡效果,防止用户拖拉窗口,画面异常
    // 对应板块显示动画
    let secIndex = sections[index];  // 对应板块
    let secSiblings = findSiblings(secIndex); // 它的兄弟们
    secIndex.classList.add("show");
    // 其它版权要去掉动画效果
    for(let i = 0 ; i<=secSiblings.length-1; i++){
        secSiblings[i].classList.remove("show");
    }
});

window.addEventListener("resize",function(){
    wrap.style.top = `-${index*window.innerHeight}px`;
})

init();  // 初始化

哦对了,长安汽车的官网也是全屏滚动的页面。https://www.changan.com.cn/

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/21804.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

热量衡算习题课

第一部分 --- 传热计算综合例题 1.qm是质量流量&#xff0c;T1&#xff0c;2和 t1,2对应的是热流体和冷流体分别在管道进口和出口的温度 2.吊塔tm是传热温差 3.α1是热流体的对流传热系数&#xff0c;α2是冷流体的对流传热系数&#xff0c;K是整个对流传热过程的总的传热系…

【Linux】在Xilinx平台上实现UVC Gadget(2)- 解决dwc3驱动bug

【Linux】在Xilinx平台上实现UVC Gadget&#xff08;2&#xff09;- 解决dwc3驱动bug一、bug描述二、具体修改方法1. 找到内核源码位置并复制到其他目录2. Petalinux里面设置使用自定义内核源码1) 选第2个Linux Components Selection2) 选linux-kernel&#xff0c;回车&#xf…

【笔试题】【day22】

文章目录第一题&#xff08;循环队列的元素个数&#xff09;第二题&#xff08;二叉排序树插入规则&#xff09;第三题&#xff08;线性探测的平均查找长度&#xff09;第四题&#xff08;关键字比较次数与初始序列无关的&#xff09;第一题&#xff08;循环队列的元素个数&…

m短波宽带通信系统的信道建模matlab仿真

目录 1.算法概述 1.1 Watterson信道模型理论简介 1.2 Nakagami信道模型 2.仿真效果预览 3.MATLAB部分代码预览 4.完整MATLAB程序 1.算法概述 宽带短波信道的研究是设计宽带大容量短波通信的基础&#xff0c;但是传统的短波信道的研究仅限于窄带信号传输&#xff0c;这已经…

论文阅读-Whisper语音识别(OpenAI)

一、论文信息 论文名称&#xff1a;Robust Speech Recognition via Large-Scale Weak Supervision 代码地址&#xff1a;https://github.com/openai/whisper 官方博客&#xff1a;https://openai.com/blog/whisper 作者团队&#xff1a;OpenAI 二、介绍 Whisper是一个通用…

HAProxy实现负载均衡

目录 一、HAProxy介绍 二、环境准备 三、实验拓扑 四、HAProxy部署 五、配置HAProxy状态页面 一、HAProxy介绍 ha-proxy是一款高性能的负载均衡软件&#xff0c;主要是做7层负载均衡&#xff0c;也可以做4层负载均衡。因为其专注于负载均衡这一些事情&#xff0c;因此与ng…

8、如何使用FactoryBean向Spring容器中注册bean?

&#x1f4eb;作者简介&#xff1a;zhz小白 公众号&#xff1a;小白的Java进阶之路 专业技能&#xff1a; 1、Java基础&#xff0c;并精通多线程的开发&#xff0c;熟悉JVM原理 2、熟悉Java基础&#xff0c;并精通多线程的开发&#xff0c;熟悉JVM原理&#xff0c;具备⼀定的线…

openEuler快速入门-openEuler命令行基础操作

系列文章目录 第一章 openEuler快速入门(一)-openEuler操作系统介绍 文章目录系列文章目录前言一、shell是什么二、Linux命令行操作技巧三、基础命令3.1、Linux命令分类3.2、目录和文件3.2.1 相对路径和绝对路径3.2.2 处理目录的常用命令ls&#xff1a;cd&#xff1a;pwd&…

tkinter绘制组件(36)——树状图

tkinter绘制组件&#xff08;36&#xff09;——树状图引言布局函数结构内容数据格式整体框架绘制元素与重绘宽度标识元素展开与闭合完整函数代码效果测试代码最终效果github项目pip下载结语引言 TinUI的第38个元素控件&#xff0c;也是TinUI-4.0-添加的第一个组件&#xff0c…

Rsync下行同步+inotify实时同步介绍和部署

一、Rsync&#xff08;Remote Sync&#xff0c;远程同步&#xff09; 是一个开源的快速备份工具&#xff0c;可以在不同主机之间镜像同步整个目录树&#xff0c;支持增量备份&#xff0c;并保持链接和权限&#xff0c;且采用优化的同步算法&#xff0c;传输前执行压缩&#xf…

2022世界杯结果预测,简单AI模型最有效?附代码!

2022世界杯冠军是谁&#xff1f;本文将为你揭晓一个利用简单AI模型得到的靠谱预测。 许多人称足球为“不可预测的比赛”&#xff0c;因为一场足球比赛有不同的因素可以改变最终比分。 这是真的……在某种程度上。 北大出版社&#xff0c;人工智能原理与实践 人工智能和数据科…

面试官:什么是伪共享,如何避免?

本文已收录到 GitHub AndroidFamily&#xff0c;有 Android 进阶知识体系&#xff0c;欢迎 Star。技术和职场问题&#xff0c;请关注公众号 [彭旭锐] 加入 Android 交流群。 前言 大家好&#xff0c;我是小彭。 在前面的文章里&#xff0c;我们聊到了 CPU 的高速缓存机制。…

【electron】判断当前进程是否是开发环境的五种方案(获取一些资源的路径)

文章目录导读需求开发环境判断是否是开发环境方案一&#xff1a;location.protocol方案二&#xff1a;全局变量__static方案三&#xff1a;全局变量process.resourcesPath方案四&#xff1a;全局变量process.env.WEBPACK_DEV_SERVER方案五&#xff1a;app.isPackaged获取配置目…

OS-process

process 什么是进程 进程是被加载到内存中、正在运行的程序&#xff1b;多个进程可能对于同一个程序、一个正在运行的OS中会有多个进程 进程是程序的一次执行过程&#xff0c;是操作系统分配资源的基本单位 作业等同于进程 进程的布局&#xff1a; 每个进程都有一个不同的…

内网渗透中最常见的十种漏洞分析总结

【环境搭建资料、工具包、全套视频…等籽料】私信聆取 以下信息是根据2020年和2019年为全球各种中型组织和企业完成的60多个渗透测试报告汇总而来的&#xff0c;在跳转到列表之前&#xff0c;让我们简要介绍一下全面的测试方法。 一、测试方法 目的是使用白盒(灰盒)方法在现场…

spirngboot项目.mvn/wrapper/maven-wrapper.properties‘ does not exist.

rm -rf ~ 命令后&#xff0c;项目出现一些问题&#xff0c;如下 执行命令&#xff1a; ./mvnw clean package -am -pl bistoury-dist -P$PROFILR -Dmaven.test.skip -Denforcer.skiptrue 报异常&#xff1a; starting to build bistoury agent Exception in thread "ma…

[附源码]java毕业设计同德佳苑物业管理系统论文

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

SpringBoot数据层解决方案/ 内置持久化解决方案JdbcTemplate使用/内置数据库H2使用以及RedisNoSql使用

写在前面&#xff1a; 继续记录自己的SpringBoot学习之旅&#xff0c;这次是SpringBoot应用相关知识学习记录。若看不懂则建议先看前几篇博客&#xff0c;详细代码可在我的Gitee仓库SpringBoot克隆下载学习使用&#xff01; 3.4 数据层解决方案 3.4.1 使用项目 新建空项目&a…

redis的事务

Redis的事务&#xff1a;一次事务操作&#xff0c;改成功的成功&#xff0c;该失败的失败。 提交事务 如果遇见错误 成功的成功&#xff0c;失败的失败&#xff0c;不会回滚 先开启事务&#xff0c;执行一些列的命令&#xff0c;但是命令不会立即执行&#xff0c;会被放在一…

web分享会

文章目录前言平台1. 在线练习平台2. 自己搭建靶场3. CTF真题平台分享前言 个人的学习过程仅供参考。 个人情况 我是大二进的工作室&#xff0c;算是比较晚的了&#xff0c;工作室的师兄人也很好&#xff0c;我有问题也会去找他们。我是一位女生&#xff0c;非常欢迎师妹们加入&…