使用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 相关文章推荐
设置下载不需要倒计时cookie(倒计时代码)
Nov 19 Javascript
Javascript的时间戳和php的时间戳转换注意事项
Apr 12 Javascript
location对象的属性和方法应用(解析URL)
Apr 12 Javascript
javascript拖拽上传类库DropzoneJS使用方法
Dec 05 Javascript
jquery插件冲突(jquery.noconflict)解决方法分享
Mar 20 Javascript
jQuery中end()方法用法实例
Jan 08 Javascript
JavaScript错误处理和堆栈追踪详解
Apr 18 Javascript
基于JavaScript实现无限加载瀑布流
Jul 21 Javascript
jsTree事件和交互以及插件plugins详解
Aug 29 Javascript
通过实例讲解JS如何防抖动
Jun 15 Javascript
electron踩坑之dialog中的callback解决
Oct 06 Javascript
JS数据类型分类及常用判断方法
Nov 19 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
通过缓存数据库结果提高PHP性能的原理介绍
2012/09/05 PHP
php中sprintf与printf函数用法区别解析
2014/02/17 PHP
Laravel 5框架学习之表单
2015/04/08 PHP
PHP Streams(流)详细介绍及使用
2015/05/12 PHP
var与Javascript变量隐式声明
2009/09/17 Javascript
百度留言本js 大家可以参考下
2009/10/13 Javascript
js原型链原理看图说明
2012/07/07 Javascript
一个jquery实现的不错的多行文字图片滚动效果
2014/09/28 Javascript
JavaScript原生对象之Number对象的属性和方法详解
2015/03/13 Javascript
jquery仿百度百科底部浮动导航特效
2015/08/08 Javascript
jQuery中的通配符选择器使用总结
2016/05/30 Javascript
微信小程序 ES6Promise.all批量上传文件实现代码
2017/04/14 Javascript
xmlplus组件设计系列之路由(ViewStack)(7)
2017/05/02 Javascript
VeeValidate在vue项目里表单校验应用案例
2018/05/09 Javascript
ng-alain表单使用方式详解
2018/07/10 Javascript
javascript闭包的使用之按钮切换功能
2018/08/30 Javascript
react-native滑动吸顶效果的实现过程
2019/06/03 Javascript
为什么Vue3.0使用Proxy实现数据监听(defineProperty表示不背这个锅)
2019/10/14 Javascript
你准备好迎接vue3.0了吗
2020/04/28 Javascript
vue项目中openlayers绘制行政区划
2020/12/24 Vue.js
详解Python中内置的NotImplemented类型的用法
2015/03/31 Python
Python使用Beautiful Soup包编写爬虫时的一些关键点
2016/01/20 Python
Python 中的range(),以及列表切片方法
2018/07/02 Python
Python类装饰器实现方法详解
2018/12/21 Python
在python带权重的列表中随机取值的方法
2019/01/23 Python
对Python定时任务的启动和停止方法详解
2019/02/19 Python
Python  Django 母版和继承解析
2019/08/09 Python
python 计算两个列表的相关系数的实现
2019/08/29 Python
Python任务自动化工具tox使用教程
2020/03/17 Python
浅谈keras中自定义二分类任务评价指标metrics的方法以及代码
2020/06/11 Python
python 还原梯度下降算法实现一维线性回归
2020/10/22 Python
最便宜促销价格订机票:Airpaz(总部设在印尼,支持中文)
2018/11/13 全球购物
2014年民主评议党员个人总结
2014/09/24 职场文书
银行会计主管岗位职责
2014/10/01 职场文书
先进班组材料范文
2014/12/25 职场文书
bose降噪耳机音能消除人声吗
2022/04/19 数码科技