本篇代码示例:github

前提:不关注移动端浏览器的前进事件

涵盖功能:
 1,管理路由的历史记录

 2,切页动画的实现

 3,处理流程类页面的回退事件

描述:

   流程类页面的回退事件的解释:
 以注册页面举例, 为了拉长整个流程,把输入项分开。以下是整个流程(懒得画图,大家将就看)

 1.首页 - 点击首页注册按钮可进入注册输入页

 2.注册输入页 - 输入姓名

 3.注册输入页 - 输入电话

 4.提交注册页 - 点击提交按钮,获取注册状态(成功或者失败),如果成功,则跳转到指定页面(假定是产品页)

 5.产品页

 以上是一个简单的注册的流程,现在写一下业务场景和需求

 1,注册未完成时

   注册未完成时,页面可能处于 1,2,3,4 这几个页面,此时,点击返回(无论是页面本身自带的返回,还是浏览器或者设备-[如安卓]的返回键触发的返回) 均可回到上一页

 2.注册完成时

  指注册成功并已经进入 5 页面 ,此时,点击返回(浏览器或者设备-[如安卓]的返回键触发的返回),回到 4 页面之后,要求直接跳回 1 页面

准备工作

 1.切页动画的实现

  切页动画使用vue自带的 过渡效果 transition 实现, 这个是附带的,这里不讲,大家自己看代码,代码如下

<template>
<div class="wapper">
<transition :name="transition">
<router-view class="child-view"></router-view>
</transition>
</div>
</template> <script> export default {
name: 'tableDemo',
data () {
return {
transition: 'slide-left'
}
},
methods: {
doSomething (e) {
this.$router.delLastRouter() // 页面回退时,路由做对应的处理
}
},
created () {
window.addEventListener('popstate', this.doSomething, false) // 监听回退事件
},
beforeRouteUpdate (to, from, next) {
window.document.title = to.meta.title
let isBack = this.$router.isBack
if (isBack) {
this.transition = 'slide-right'
next()
} else {
// 进入页面之前判断是否是进入流程页
if (from.meta.isStartPage) {
this.$router.setProcessStatus(to.meta.group, false)
}
this.transition = 'slide-left'
next()
}
}
}
</script> <style lang="scss">
.wapper { position: relative; width: 100%; height: 100%;}
.child-view { position: absolute; width:100%; height: 100%;}
.slide-left-leave { -webkit-transform: translate(0, 0); transform: translate(0, 0);}
.slide-left-leave-active { transition: all .4s cubic-bezier(.55,0,.1,1);}
.slide-left-leave-to { -webkit-transform: translate(100%, 0); transform: translate(100%, 0);}
.slide-left-enter { -webkit-transform: translate(-100%, 0); transform: translate(-100%, 0);}
.slide-left-enter-active { transition: all .4s cubic-bezier(.55,0,.1,1);}
.slide-left-enter-to { -webkit-transform: translate(0, 0); transform: translate(0, 0);} .slide-right-leave { -webkit-transform: translate(0, 0); transform: translate(0, 0);}
.slide-right-leave-active { transition: all .4s cubic-bezier(.55,0,.1,1);}
.slide-right-leave-to { -webkit-transform: translate(-100%, 0); transform: translate(-100%, 0);}
.slide-right-enter { -webkit-transform: translate(100%, 0); transform: translate(100%, 0);}
.slide-right-enter-active { transition: all .4s cubic-bezier(.55,0,.1,1);}
.slide-right-enter-to { -webkit-transform: translate(0, 0); transform: translate(0, 0);} </style>

 2.路由的管理

  1. 为vue-router设置一些状态

Router.prototype.routerArray = []; // 存储过往的路由信息
Router.prototype.processArray = []; // 存储流程页的状态
Router.prototype.isBack = false; // 是否是返回,动画使用

  2. 使用 beforeRouteUpdate 来检查路由的变化 这里要注意的事情是 beforeRouteUpdate 只针对子路由生效  ,所以这段代码放在父路由指向的页面中

  在这个页面,我们通过 this.$router.isBack 来区分页面是前进还是后退,并调整对应的动画效果。

beforeRouteUpdate (to, from, next) {
window.document.title = to.meta.title
let isBack = this.$router.isBack
if (isBack) {
this.transition = 'slide-right'
next()
} else {
// 进入页面之前判断是否是进入流程页
if (from.meta.isStartPage) {
this.$router.setProcessStatus(to.meta.group, false)
}
this.transition = 'slide-left'
next()
}
}

  3. 重写路由的push方法  在push方法中 如果push指向的页面已经存在于历史记录中,则当做返回处理, 前进则为 历史记录添加一条数据

// 修改 Router本身的push, 跳转之前做一些判断
let lPush = Router.prototype.push;
Router.prototype.push = function (location, onComplete, onAbort) {
let lGoCount = this.havRouterHistory(location)
if (lGoCount < 0) {
this.isBack = true
this.go(lGoCount);
} else {
this.isBack = false
this.routerArray.push(this.history.current)
lPush.call(this, location, onComplete, onAbort)
}
};

  4.重写路由的go方法

// 重写 Router 本身的 go, 回退的同时要清空记录中的数据
let lGo = Router.prototype.go
Router.prototype.go = function (n) {
if (n > 0) {
lGo.call(this, n);
return;
}
let lLen = 0 - n;
lLen = this.routerArray.length > lLen ? lLen : this.routerArray.length;
let lStart = this.routerArray.length - lLen;
this.routerArray.splice(lStart, lLen);
lGo.call(this, n);
};

  5.判断 push 时 指向的页面是否存在于历史记录中

// 判断历史记录中是否存在将要跳转的路由,如果有,执行回退操作
Router.prototype.havRouterHistory = function (location) {
// debugger
let lPath = location.path;
let lName = location.name;
let lLen = this.routerArray.length;
for (let i = 0; i < lLen; i++) {
if (this.routerArray[i].path === lPath || this.routerArray[i].name === lName ) {
return i - lLen; // 存在则返回应该回退的步数,注意回退的步数是负值
}
}
return 1 // 如果存在则必然小于0,所以这里写返回1代表不存在没有问题
};

  6.浏览器或者设备物理键触发回退

  在父路由页面设置全局的事件监听,在触发事件的时候处理路由

methods: {
doSomething (e) {
this.$router.delLastRouter() // 页面回退时,路由做对应的处理
}
},
created () {
window.addEventListener('popstate', this.doSomething, false) // 监听回退事件
},
// 浏览器回退时,清除最后一条路由数据  这串代码在 路由配置文件 中
Router.prototype.delLastRouter = function () {
this.routerArray.pop()
}

  7. 路由本身的回退事件   无需对路由的back事件做处理,因为vue-router本身的back 调用的  go(-1) go函数我们已经重写过了

  8.根据流程相关页面,对路由做相对应的配置

  isStartPage: true // 表示这是一个流程入口页,相当于上面的 页面1

  group  // group分组,代表了一组页面同属于一个流程中

/**
* 返回效果的Demo
*/
export default [
{
path: '/backDemoIndex',
name: 'backDemoIndex',
component: r => require.ensure([], () => r(require('../../Page/BackDemo/index')), 'BackDemo/index'),
meta: { title: '入口选择页面' }
},
{
path: '/backAPage',
name: 'backAPage',
component: r => require.ensure([], () => r(require('../../Page/BackDemo/aPage')), 'BackDemo/aPage'),
meta: { title: '操作起始页面aaaaa', isStartPage: true }
},
{
path: '/backAPageA',
name: 'backAPageA',
component: r => require.ensure([], () => r(require('../../Page/BackDemo/aPagea')), 'BackDemo/aPagea'),
meta: { title: '操作起始页面bbb', isStartPage: true }
},
{
path: '/backAPageB',
name: 'backAPageB',
component: r => require.ensure([], () => r(require('../../Page/BackDemo/aPageb')), 'BackDemo/aPageb'),
meta: { title: '操作起始页面CCC', isStartPage: true }
},
{
path: '/backBPage',
name: 'backBPage',
component: r => require.ensure([], () => r(require('../../Page/BackDemo/bPage')), 'BackDemo/bPage'),
meta: { title: '第一个输入页面', group: 'backDemo' }
},
{
path: '/backCPage',
name: 'backCPage',
component: r => require.ensure([], () => r(require('../../Page/BackDemo/cPage')), 'BackDemo/cPage'),
meta: { title: '第二个输入页面', group: 'backDemo' }
},
{
path: '/backDPage',
name: 'backDPage',
component: r => require.ensure([], () => r(require('../../Page/BackDemo/dPage')), 'BackDemo/dPage'),
meta: { title: '结果页面', group: 'backDemo' }
},
{
path: '/other',
name: 'other',
component: r => require.ensure([], () => r(require('../../Page/BackDemo/other')), 'BackDemo/other'),
meta: { title: '结果页面' }
}
]

  9.在流程起始页进入流程页是,修改流程的状态

  this.$router.setProcessStatus( ‘backDemo’, false)  //  groupName 流程分组名称   status  流程状态 true为已经完成  false 为 未完成

// 修改一个流程的状态
Router.prototype.setProcessStatus = function (groupName, status) {
this.processArray[groupName] = status
};

  10.在每个流程页 检查流程的完成情况

  this.$router.onPageLoad()

// 找到最近的入口页面
Router.prototype.findInsterPage = function () {
let lLen = this.routerArray.length;
for (let i = lLen - 1; i >= 0; i--) {
if (this.routerArray[i].meta.isStartPage) {
return i - lLen
}
}
return 1; // 如果存在则必然小于0,所以这里写返回1代表不存在没有问题
} // 如果进入一个流程页,并且该页面的状态已经是完成,则应该回退到最近的流程入口页面,或者 跳转到指定的页面
Router.prototype.onPageLoad = function (query) {
// debugger
if (this.history.current.meta.group) {
let lNowRouter = this.history.current.meta.group
let lStatus = this.processArray[lNowRouter]
if (lStatus) {
let lRes = this.findInsterPage()
if (lRes < 0) {
this.go(lRes)
}
}
}
};

  11. 在流程结果处理页面,更改流程状态为完成  this.$router.setProcessStatus( ‘backDemo’, true)

  

基于vue-router的移动端网页的路由管理的更多相关文章

  1. 基于Vue.js PC桌面端弹出框组件|vue自定义弹层组件|vue模态框

    vue.js构建的轻量级PC网页端交互式弹层组件VLayer. 前段时间有分享过一个vue移动端弹窗组件,今天给大家分享一个最近开发的vue pc端弹出层组件. VLayer 一款集Alert.Dia ...

  2. 基于VUE.JS的移动端框架Mint UI

    Mint UI GitHub:github.com/ElemeFE/mint 项目主页:mint-ui.github.io/# Demo:elemefe.github.io/mint- 文档:mint ...

  3. 在微信框架模块中,基于Vue&Element前端的事件和内容的管理

    在微信后台管理中,我们需要定义好菜单对应的事件管理,因为微信通过菜单触发相关的事件,因此菜单事件的响应关系,我们如果处理好,就能构建出我们的微信应用入口了.通过入口,我们可以响应用户菜单的事件,如响应 ...

  4. 基于 Vue.js 的移动端组件库mint-ui实现无限滚动加载更多

    通过多次爬坑,发现了这些监听滚动来加载更多的组件的共同点, 因为这些加载更多的方法是绑定在需要加载更多的内容的元素上的, 所以是进入页面则直接触发一次,当监听到滚动事件之后,继续加载更多, 所以对于无 ...

  5. Vue 爬坑之路(十一)—— 基于 Nuxt.js 实现服务端渲染(SSR)

    直接使用 Vue 构建前端单页面应用,页面源码时只有简单的几行 html,这并不利于网站的 SEO,这时候就需要服务端渲染 2016 年 10 月 25 日,zeit.co 背后的团队对外发布了一个 ...

  6. 优秀的基于VUE移动端UI框架合集

    1. vonic 一个基于 vue.js 和 ionic 样式的 UI 框架,用于快速构建移动端单页应用,很简约,是我喜欢的风格 star 2.3k 中文文档 在线预览 2.vux 基于WeUI和Vu ...

  7. 前端基于vue,后台采用springboot+shiro的方式搭建的一个移动端商品展示平台

    基于vue实现的移动端商品展示页,可以web-view的方式嵌入到小程序中,布局简约.大气,减少初学者或开发者不必要的工作量.后台维护采用的springboot+shiro的方式,为广大爱好者提供展示 ...

  8. [Vue 牛刀小试]:第十二章 - 使用 Vue Router 实现 Vue 中的前端路由控制

    一.前言 前端路由是什么?如果你之前从事的是后端的工作,或者虽然有接触前端,但是并没有使用到单页面应用的话,这个概念对你来说还是会很陌生的.那么,为什么会在单页面应用中存在这么一个概念,以及,前端路由 ...

  9. 新建一个基于vue.js+Mint UI的项目

    上篇文章里面讲到如何新建一个基于vue,js的项目(详细文章请戳用Vue创建一个新的项目). 该项目如果需要组件等都需要自己去写,今天就学习一下如何新建一个基于vue.js+Mint UI的项目,直接 ...

随机推荐

  1. 「JLOI2014」松鼠的新家

    「JLOI2014」松鼠的新家 传送门 两种做法: 树上差分 \(O(n)\) 树链剖分 \(O(nlogn)\) 树剖比较好写而且无脑,树上差分复杂度优秀一些但是会有点难调. 这里给出树剖写法: 唯 ...

  2. liux 防火墙以及开关

    一.service方式 查看防火墙状态: [root@centos6 ~]# service iptables status iptables:未运行防火墙. 开启防火墙: [root@centos6 ...

  3. 设计模式课程 设计模式精讲 3-8 迪米法特原则讲解及Coding

    1 课程讲解 1.1 定义 1.2 特质 1.3 重点 2 代码演练 2.1 反例 2.2 正例 1 课程讲解 1.1 定义 定义:一个对象应该对其他对象保持最少的了解.又叫最少知道原则. 1.2 特 ...

  4. 栈结构Stack

    package seday12; import java.util.Deque; import java.util.LinkedList; /** * @author xingsir * 栈结构. 栈 ...

  5. K8S-OVS使用Openvswitch为提供SDN功能支持单租户模式和多租户模式

    k8s-ovs ============================== 最近在寻求一些工作机会,如果有kubernetes相关研发招聘的朋友,欢迎随时联系我.我的个人简历可以通过百度网盘:htt ...

  6. day11 作业

    # 1.编写装饰器,为多个函数加上认证的功能(用户的账号密码来源于文件), # 要求登录成功一次,后续的函数都无需再输入用户名和密码 # FLAG = False # def login(func): ...

  7. Week of Code:GG

    题意是给出一个数n,一个长度为n-1的字符串.求的是1到n符合要求的序列的数量,该序列需要满足当该位置为G时,这个位置的数大于后面位置的数.当该位置为L时,这个位置的数要小于后面位置的数.最后数量模m ...

  8. Linux命令:vi | vim命令

    vim - vi 增强版.文本编辑器 格式:vim [options] [file ..] 说明:如果file存在,文件被打开并显示内容,如果文件不存在,当编辑后第一次存盘时创建它 [options] ...

  9. MongoDB基础篇2:数据库/用户/数据集合的增删改

    一.数据库操作 创建并进入数据库: 命令:use DATABASE_NAME 示例:use tms   查看所有数据库: 命令:show dbs   注意: (1)新创建的数据库在show dbs命令 ...

  10. CSP2019 括号树

    Description: 给定括号树,每个节点都是 ( 或 ) ,定义节点的权值为根到该节点的简单路径所构成的括号序列中不同合法子串的个数(子串需要连续,子串所在的位置不同即为不同.)与节点编号的乘积 ...