使用vue制作FullPage页面滚动效果


Posted in Javascript onAugust 21, 2017

前言

已经有好久没有更新博客了,大三下了要准备找实习了,才发现自己很多东西都不会,所以赶紧找了个现在流行的MVVM框架学习一下。我学习的是Vue,所以拿Vue写了一个FullPage的模板,可以供自己和其他人使用。

项目的地址是:https://github.com/hzxszsk/vue-fullpage

项目讲解

为了加深理解,我把我制作这个FullPage页面的思路和流程记录下来,也可以给其他和我一样的初学者一个参考。

分解思路

首先,我根据Vue的组件开发思想,把这个FullPage页面分为两个主要的组件模块:页面组件(Page)和页面控制组件(PageController)。除了这两个组件模块之外,还有一个整合的App.vue文件。

其中,Page组件主要负责每个FullPage页面的样式和位置控制,而PageController组件则用来发起页面的切换请求。

因为Vue是单向数据流形式的,因此将所有需要的数据都定义在App.vue文件中,由App组件向下传递数据。

数据需求分析

要制作一个FullPage页面,每个Page页面需要的是页面自身的内容(页面的内容和样式配置),而控制器所需要的数据有当前正在显示的页面以及总的页面数,用这两个属性计算出其他需要的属性。

其中,总页面数即是Page总数,有多少个Page,就有多少个Page的配置,因此可以通过计算Page配置信息对象的数量来获得总页面数。

我在App.vue的组件data中定义了两个属性:

  • currentPage表示的是当前页面的序号(从1开始计算);
  • options是一个数组,其中的每一个对象表示的是对应序号的页面的配置信息,可以通过修改里面的对象属性从而改变对应Page的样式。

Page组件只需要知道当前页面是第几个页面和自身的配置,因此只含有两个属性:

  • currentPage
  • option 表示自身的配置

PageController需要对页面进行切换控制,因此需要两个属性:

  • currentPage
  • pageNum 表示总的页面数量

编写页面样式

App组件的结构为:

<div id="app" class="app">
 <!-- page为单独的页面组件,page内可以编写任意的页面内容 -->
 <page :currentPage="currentPage"></page>
 <!-- page-controller为控制器组件 -->
 <page-controller :pageNum="pageNum" :currentPage="currentPage" @changePage="changePage" :arrowsType="arrowsType"></page-controller>
</div>

App组件主要样式:

/* 页面宽高为100%,overflow为hidden隐藏溢出部分 */
html,body {
 margin: 0;
 padding: 0;
 height: 100%;
 width: 100%;
 overflow: hidden;
}
.app {
 height: 100%;
 width: 100%;
}

Page组件结构为:

<section class="page" v-if="options"
:style="{background:options.background,color:options.color||'#fff'}" 
:class="{'page-before': options.index < currentPage,'page-after': options.index > currentPage}">
  <div :class="{'page-center': options.isCenter}">
    <slot></slot>
  </div>
</section>
<section class="page" v-else>页面正在渲染中。。。</section>

其中slot内为在App.vue的page标签内编写的html内容,并且根据Page组件内的option属性,为Page页面添加不同的样式(包括背景颜色、字体颜色、居中等,而且可以自由扩展)

Page组件的主要样式为:

.page {
  position: absolute;
  width: 100%;
  height: 100%;
  transition: all 0.5s ease 0s;
}
.page-before {
  transform: translate3d(0,-100%,0);
}
.page-after {
  transform: translate3d(0,100%,0);
}

通过比较当前页面的index与currentPage的大小,可以判断Page组件处于之前的页面,当前的页面和之后的页面三个状态中的哪一种。

将所有页面都用absolute定位在同一个位置上,对于处于之前的页面的Page,添加page-before类标签,使其在Y轴上向上偏移自身高度距离,同理,对之后的页面做Y轴上向下的自身高度偏移。

在page标签上添加transition样式,使其在切换时可以产生动画效果。

PageController组件的结构为:

<nav class="controller">
 <button v-if="arrowsType != 'no'" class="prev-btn" :class="{moving:arrowsType === 'animate'}" @click="changePage(prevIndex)"></button>
 <ul>
   <li @click="changePage(index)" v-for="index in pageNum" :key="'controller-'+index" class="controller-item"></li>
 </ul>
 <button v-if="arrowsType != 'no'" class="next-btn" :class="{moving:arrowsType === 'animate'}" @click="changePage(nextIndex)"></button>
</nav>

PageController组件中分为两块,控制上下滚动的上下箭头按钮和控制所有页面的ul列表。

因为PageController中的样式较多,因此不在文章中详细说明,只做简单介绍。

button和ul标签用fixed定位,使其处于整个页面的上方、下方和右侧。

button使用rotate和border属性,制作出箭头的样式,并添加对应的动画效果(可以通过传递props选择关闭)。

编写页面逻辑

切换页面逻辑

因为所有的页面切换都是由PageController发起,而控制currentPage的组件并不是PageController,所以需要有一个父子组件事件,由子组件PageController发起,传递一个参数表示要切换到第几个页面,因此在PageController中定义一个method:

changePage (index) {
 this.$emit('changePage', index);
}

父组件接受该事件并调用自己定义的changePage方法,修改自身的currentPage属性

changePage (index) {
 // 改变page
 this.currentPage = index;
}

箭头按钮上下滚动

为了实现上下滚动,需要知道当前页面的前一个页面和后一个页面分别是第几个页面,因此可以使用计算属性,计算出前一个和后一个页面的index值:

// PageController.vue
nextIndex () {
 if (this.currentPage === this.pageNum) {
 return 1;
 } else {
 return this.currentPage + 1;
 }
},
prevIndex () {
 if (this.currentPage === 1) {
 return this.pageNum;
 } else {
 return this.currentPage - 1;
 }
}

在点击箭头时,将对应的nextIndex或prevIndex值当做参数传给changePage方法。

滚轮滚动和移动端滚动

滚轮滚动和移动端滚动主要依靠window的监听事件,根据传入的event属性,计算出页面是应该向上还是向下滚动,将需要滚动的方向作为参数传递给处理函数handler。

因为代码略长,因此不全部显示在文章中,只显示处理函数相关逻辑

let _this = this;
let timer = null;
function scrollHandler (direction) {
 // 防止重复触发滚动事件
 if (timer != null) {
 return;
 }
 if (direction === 'down') {
 _this.changePage(_this.nextIndex);
 } else {
 _this.changePage(_this.prevIndex);
 }
 timer = setTimeout(function() {
 clearTimeout(timer);
 timer = null;
 }, 500);
}

需要注意的一点是,移动端做滚动判断时,要求touches和changedTouches之间需要一定的间隔,不然容易误触发滚动事件。

OPTIONS属性的分发

为了使使用者更加方便地编写页面内容而不在意具体的页面序号,我采用了自动对page内的option赋值的方法。其实现原理是在App.vue文件中,使用钩子函数mounted,对page中的option属性进行设置。

mounted () {
 this.$children.forEach((child, index) => {
 // 动态设置各个page内的options
 if (child.option === null) {
  let childOption = this.options[index];
  this.$set(childOption,'index',index+1);
  child.option = childOption;
 }
 });
}

高级属性:新的钩子函数

为了满足部分使用者的需求,我在设置了两个钩子函数:beforeLeave和afterEnter。

这两个钩子函数可以设置在对应页面的options属性对象中,并且含有一个默认的参数,为对应页面的page组件实例对象。

其实现方式为在原先的changePage函数(App.vue)内添加新的逻辑:

changePage (index) {
 // beforeLeave
 let beforeIndex = this.currentPage - 1;
 let leaveFunction = this.options[beforeIndex].beforeLeave;
 typeof leaveFunction === 'function' && leaveFunction.call(this,this.$children[beforeIndex]);
 // 改变page
 this.currentPage = index;
 // afterEnter
 let nextIndex = index-1;
 let enterFunction = this.options[nextIndex].afterEnter;
 this.$nextTick(function () {
 typeof enterFunction === 'function' && enterFunction.call(this,this.$children[nextIndex]);
 })
}

总结

这篇文章记录了我开发一个FullPage页面的总体流程,将主要的逻辑重新顺理了一遍,还有一些小的细节没有写在文章中,有兴趣的可以去具体的项目页面看源码

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
发现的以前不知道的函数
Sep 19 Javascript
JS在IE和FireFox之间常用函数的区别小结
Mar 12 Javascript
修改jQuery Validation里默认的验证方法
Feb 14 Javascript
JS 实现图片直接下载示例代码
Jul 22 Javascript
输入自动提示搜索提示功能的使用说明:sugggestion.txt
Sep 02 Javascript
将JavaScript的jQuery库中表单转化为JSON对象的方法
Nov 17 Javascript
JavaScript获取对象在页面中位置坐标的方法
Feb 03 Javascript
javascript history对象详解
Feb 09 Javascript
JavaScript实现简单的星星评分效果
May 18 Javascript
Vue之Watcher源码解析(1)
Jul 19 Javascript
服务端预渲染之Nuxt(使用篇)
Apr 08 Javascript
JavaScript浅层克隆与深度克隆示例详解
Sep 01 Javascript
详解Layer弹出层样式
Aug 21 #Javascript
JS数组操作之增删改查的简单实现
Aug 21 #Javascript
JS实现评价的星星功能
Aug 20 #Javascript
详解A标签中href=&quot;&quot;的几种用法
Aug 20 #Javascript
Cropper.js 实现裁剪图片并上传(PC端)
Aug 20 #Javascript
Bootstrap 模态框(Modal)带参数传值实例
Aug 20 #Javascript
mui 打开新窗口的方式总结及注意事项
Aug 20 #Javascript
You might like
咖啡知识 咖啡养豆要养多久 排气又是什么
2021/03/06 新手入门
ThinkPHP的cookie和session冲突造成Cookie不能使用的解决方法
2014/07/01 PHP
PHP命令行脚本接收传入参数的三种方式
2014/08/20 PHP
php实现webservice实例
2014/11/06 PHP
PHP模板引擎Smarty内建函数section,sectionelse用法详解
2016/04/11 PHP
php自定义扩展名获取函数示例
2016/12/12 PHP
支持汉转拼和拼音分词的PHP中文工具类ChineseUtil
2018/02/23 PHP
Yii2.0 RESTful API 基础配置教程详解
2018/12/26 PHP
在一个form用一个SUBMIT(或button)分别提交到两个处理表单页面的代码
2007/02/15 Javascript
Prototype Object对象 学习
2009/07/12 Javascript
jQuery帮助之筛选查找 children([expr])
2011/01/31 Javascript
js前台分页显示后端JAVA数据响应
2013/03/18 Javascript
JS模拟自动点击的简单实例
2013/08/08 Javascript
Jquery 分页插件之Jquery Pagination
2015/08/25 Javascript
去除字符串左右两边的空格(实现代码)
2016/05/12 Javascript
Vue制作Todo List网页
2017/04/26 Javascript
详解angular如何调用HTML字符串的方法
2018/06/30 Javascript
javascript和php使用ajax通信传递JSON的实例
2018/08/21 Javascript
vuex2中使用mapGetters/mapActions报错的解决方法
2018/10/20 Javascript
vue使用Sass时报错问题的解决方法
2020/10/14 Javascript
[02:05]2014DOTA2西雅图邀请赛 老队长全明星大猜想谁不服就按进显示器
2014/07/08 DOTA
Python正则表达式教程之一:基础篇
2017/03/02 Python
详解Pytorch 使用Pytorch拟合多项式(多项式回归)
2018/05/24 Python
Linux下Python安装完成后使用pip命令的详细教程
2018/11/22 Python
python爬虫 基于requests模块发起ajax的get请求实现解析
2019/08/20 Python
Python爬虫图片懒加载技术 selenium和PhantomJS解析
2019/09/18 Python
python的scipy.stats模块中正态分布常用函数总结
2021/02/19 Python
美国领先的奢侈美容零售商:Bluemercury
2017/07/26 全球购物
捷克原创男装和女装购物网站:Bolf.cz
2018/04/28 全球购物
岗位竞聘书范文
2014/03/31 职场文书
银行主办会计岗位职责
2014/08/13 职场文书
政府班子四风问题整改措施
2014/10/04 职场文书
幼儿园国庆节活动总结
2015/03/23 职场文书
导游词之上海豫园
2019/10/24 职场文书
Python基础教程,Python入门教程(超详细)
2021/06/24 Python
CDPR谈《巫师》新作用虚幻5原因 称不会为Epic独占
2022/04/06 其他游戏