《微信小程序七日谈》- 第四天:页面路径最多五层?导航可以这么玩
《微信小程序七日谈》系列文章:
- 第一天:人生若只如初见;
- 第二天:你可能要抛弃原来的响应式开发思维;
- 第三天:玩转Page组件的生命周期;
- 第四天:页面路径最多五层?导航可以这么玩;
- 第五天:你可能要在登录功能上花费大力气;
- 第六天:小程序devtool隐藏的秘密;
- 第七天:不要捡了芝麻丢了西瓜
本系列的文章并非初学教程,而是笔者在具体开发过程中遇到的问题以及部分解决方案。
微信小程序提供导航相关的API:
wx.navigateTo();wx.redirectTo();wx.navigateBack()。
使用wx.navigateTo()或者<navigator>组件跳转的页面路径最多只有5层,这些页面路径是可以通过wx.navigateBack()API或者左上角返回按钮按顺序返回的。当页面路径大于5层时,使用wx.navigateTo()进行下一页吗跳转会抛出错误:
navigateTo:fail webview count limit exceed.
但是某些业务场景存在多页面互动的交互逻辑,远远不止5层页面栈。比如笔者近期参与开发的58到家小程序中存在如下的业务场景:
1、用户进入小程序,展示首页;
2、首页存在一个如下图的底部导航栏:

用户点击“我的”进入个人中心,此时页面栈为首页->个人中心,共2层。
3、个人中心页面存在“我的收入”入口,如下:

4、用户从个人中心进入我的收入页面,此时页面栈为首页->个人中心->我的收入,共3层;
5、我的收入页面提供“提现”页面的入口,如下:

6、用户进入提现页面,此时的页面栈为首页->个人中心->我的收入->提现,共4层。此时留给我们可支配的页面栈只剩下一层了。提现流程如下:

提现流程存在多页面直接的数据共享和交互,如果是常规的webapp,我们通常会考虑使用hash路由或者干脆做成独立的几个页面使用url传参进行数据通信。但是进入提现页面之后,我们最多只能再添加一个独立页面了。也就是说,银行列表页、绑定银行卡页和提交成功页三者只能再使用一个页面栈(并非一个页面)承载。如何用仅剩的最后一层页面栈实现上述复杂的提现流程呢?
逻辑行为梳理
第一步:细分交互行为
首先第一步是将提现行为细分,因为只能再添加一个独立页面,所以需要合并一些可在一个页面完成的行为。上文的流程图其实遗漏了一个行为:绑定银行卡页面点击银行卡需要显示银行列表页。也就是允许用户重新选择银行。所以其实整体的提现流程如下:

小程序标题栏左上角返回按钮的行为(图中标红的线条)是返回页面栈的上一页面,代码是无法干预的。
整个流程中必须支持“返回”按钮正常返回上一页面的行为有:
- 前置页面进入提现页面,正常返回前置页面;
- 提现页面进入的银行列表页面,正常返回提现页面。
要保证第二条“提现页面进入的银行列表页面,正常返回提现页面”,就必须将银行列表页独立为一个页面。至此,最后一层页面栈就定型了。那么剩下的绑定银行卡和提交成功页面怎么办呢?
第二步:合并逻辑页面
需要注意的是,银行列表页面与绑定银行卡页面之间有一个双向的交互行为,由于最后一个独立页面已经确定为银行列表页了,所以不得不从中牺牲一定的用户体验:绑定银行卡页面跳转到银行列表页后不能正常返回。有了这个前提,我们可以把银行列表和绑定银行卡两个逻辑页面合并为一个实体页面,通过子路由控制行为展示。
再次回顾上文的交互流程图还可以得到另外一个信息:提交成功页面的返回逻辑与提现页面完全相同。所以,两者同样可以合并为一个实体页面,由子路由控制行为展示。
第三步:梳理行为逻辑
以第二步的合并规则为准,实体页面的交互流程如下:

使用data.route实现子路由
微信小程序的Page是没有子路由概念的,我们在此讨论的子路由其实就是根据Page组件的某个data字段进行不同模板的分发渲染。
首先定义支持的子路由列表:
// 路由列表
const ROUTES = {
index: 'index',
banklist: 'banklist',
setcard: 'setcard',
done: 'done'
};
我们在代码上又进一步的融合,将第四层页面和第五层页面两个实体页面融合为同一个Page组件,通过子路由控制模板的渲染,之所以这样做有以下几点考虑:
- 两个实体页面之间有很多共用的数据;
- 58到家的小程序是复用了现存的部分接口,两个页面之间的数据是混合在一起的,融合也是为了避免接口的重构;
既然融合为一个Page组件,那么如何实现页面的跳转呢?其实很简单,使用wx.navigateTo()API如下:
wx.navigateTo({
url: './index?route=' + ROUTES.banklist
});
上述代码实现了跳转到同一Page组件的功能,并且跳转的页面会被加入到页面栈中。
然后在index.wxml中增加路由数据的逻辑判断分发:
<block wx:if="{{route=='index'}}">
<include src="_part/basic/index.wxml"/>
</block>
<block wx:if="{{route=='banklist'}}">
<include src="_part/banklist/index.wxml"/>
</block>
<block wx:if="{{route=='setcard'}}">
<include src="_part/setcard/index.wxml"/>
</block>
<block wx:if="{{route=='done'}}">
<include src="_part/done/index.wxml"/>
</block>
前置页面进入第四层页面时默认的是index子路由页面,有第五层页面的绑定银行卡提交后返回第四层页面时显示done子路由。这个逻辑中需要注意的是:
- 第四层页面跳转第五层页面是隐藏(Hide)而不是卸载(Unload);
- 第五层页面返回第四层页面后会触发第四层页面的
onShow钩子函数。
也就是说,我们可以再onShow钩子函数中进行路由的分发。但是如何获取路由字段呢?大家可能想到的第一个方案就是通过url传参,可惜这个方案是行不通的。首先,微信小程序官方文档中关于Page组件钩子函数的说明,只有onLoad函数可以获取由url query传递的数据,其余的任何钩子函数都不能获取;其次,第五层页面的提交行为返回第四层页面是由wx.navigateBack()API实现的,这个API的功能是返回页面栈中的上一层页面,并不支持指定的修改url,所以url传参这条路是走不通的。
那么使用cookie是否可行呢?虽然微信小程序不支持cookie,但cookie的理念可以提供给我们解决问题的思路:将数据先储存在本地,跳转页面后获取本地数据进行相应处理。
有了思路,自然而然地便想到类似cookie的本地storage。
使用storage进行页面间的数据通信
第一步:点击第五层页面的提交按钮后,首先在storage中储存第四层页面的route值:
wx.setStorage({
key: 'dj_deposits_route',
data: ROUTES.done
});
第二步:在第四层页面的onShow函数内获取storage中的route数据并赋值给data中的route字段,模板便会同步刷新:
let _route = wx.getStorageSync('dj_deposits_route');
this.setData({
route: _route
});
总结
微信小程序的页面路径限制为最多5层,多于5层的页面将不会跳转并且会抛出错误信息。而我们产品的某些业务场景不止存在5层的页面路径,在这种情况下,我们不得不牺牲一定的用户体验,以保证功能的完整。本文提到的方案是与业务场景息息相关的,只是一家之言,并非最佳实践。希望能够给大家一点参考。
《微信小程序七日谈》- 第四天:页面路径最多五层?导航可以这么玩的更多相关文章
- 微信小程序开发(四)页面跳转
承接上篇博客. 通过点击按钮跳转到新的页面. 先创建新页面home: 代码如下: // home.js Page({}) // 注册页面 // home.json {} // home.wxml &l ...
- 《微信小程序七日谈》- 第三天:玩转Page组件的生命周期
<微信小程序七日谈>系列文章: 第一天:人生若只如初见: 第二天:你可能要抛弃原来的响应式开发思维: 第三天:玩转Page组件的生命周期: 第四天:页面路径最多五层?导航可以这么玩 前两篇 ...
- 《微信小程序七日谈》- 第二天:你可能要抛弃原来的响应式开发思维
<微信小程序七日谈>系列文章: 第一天:人生若只如初见: 第二天:你可能要抛弃原来的响应式开发思维: 第三天:玩转Page组件的生命周期: 第四天:页面路径最多五层?导航可以这么玩 上篇文 ...
- 《微信小程序七日谈》- 第一天:人生若只如初见
<微信小程序七日谈>系列文章: 第一天:人生若只如初见: 第二天:你可能要抛弃原来的响应式开发思维: 第三天:玩转Page组件的生命周期: 第四天:页面路径最多五层?导航可以这么玩 微信小 ...
- 《微信小程序七日谈》- 第五天:你可能要在登录功能上花费大力气
<微信小程序七日谈>系列文章: 第一天:人生若只如初见: 第二天:你可能要抛弃原来的响应式开发思维: 第三天:玩转Page组件的生命周期: 第四天:页面路径最多五层?导航可以这么玩: 第五 ...
- 《微信小程序七日谈》- 第六天:小程序devtool隐藏的秘密
<微信小程序七日谈>系列文章: 第一天:人生若只如初见: 第二天:你可能要抛弃原来的响应式开发思维: 第三天:玩转Page组件的生命周期: 第四天:页面路径最多五层?导航可以这么玩: 第五 ...
- 《微信小程序七日谈》- 第七天:不要捡了芝麻丢了西瓜
<微信小程序七日谈>系列文章: 第一天:人生若只如初见: 第二天:你可能要抛弃原来的响应式开发思维: 第三天:玩转Page组件的生命周期: 第四天:页面路径最多五层?导航可以这么玩: 第五 ...
- 微信小程序把玩(四十)animation API
原文:微信小程序把玩(四十)animation API 动画水还是比较深的,这里只是简单介绍下小程序中动画的一些属性和注意事项,做动画前一定要整理好思路将动画一步步分解,再进行组合!这里只做引入. w ...
- 微信小程序把玩(四十一)canvas API
原文:微信小程序把玩(四十一)canvas API 绘图是每个移动应用必备的技术,基本上和Android,IOS,等移动开发都是相同的,创建个上下文,给你个画布再上画,官网给的小例子都比较全了自己去看 ...
随机推荐
- asp.net关于页面不回发的问题,寻求完美解决方案
原文地址:http://www.sufeinet.com/thread-4564-1-1.html 这个问题我相信有不少人见过,就是使用系统的分页功能时,或者是使用系统控件,都会有一个回发的功能, 这 ...
- Speech两种使用方法
COM组件使用speech: public class Speach { private static Speach _Instance = null ; private SpeechLib.SpVo ...
- Access to the path '20141211142713.gif' is denied.
给network service加上读写权限即可
- [转] Redis系统性介绍
Redis系统性介绍 http://blog.nosqlfan.com/html/3139.html?ref=rediszt 虽然Redis已经很火了,相信还是有很多同学对Redis只是有所听闻或者了 ...
- android国际化(多语言)
2013-03-18 23:45 13390人阅读 评论(0) 收藏 举报 1. 很大程度上,为什么 ...
- android开发常见编程错误总结
1.设置TextView的文本颜色 1 2 3 TextView tv; ... tv.setTextColor(R.color.white); 其实这样设置的颜色是 R.color.white的资源 ...
- GNU Radio 之 rtl-sdr
http://sdr.osmocom.org/trac/wiki/rtl-sdr 我使用的是去年买的一个电视棒(ezcap DVB-TFMDAB),50多元,它的频宽为52Mhz - 2.2GHz ! ...
- Mac OS 下安装wget
环境: Mac OS X 10.9.4 1 下载源码 地址: http://ftp.gnu.org/gnu/wget/wget-1.9.tar.gz 2 解压安装 tar zxvf wget-1.9 ...
- PL-SQL 存储函数和存储过程
PL-SQL 存储函数和存储过程 ORACLE 提供能够把PL/SQL 程序存储在数据库中,并能够在不论什么地方来执行它.这样就叫存储过程或函数. 过程和函数统称为PL/SQL子程序.他们是被命 ...
- SQL SERVER 服务启动后停止,某些服务由其它服务或程序使用时将自动停止
发生症状: 先是服务器挂掉,之后服务器可以登陆,但是实例却不能登陆进去(部分).出现的错误日志如下: :: R2 (SP2) - 10.50.4000.0 (X64) Jun :: Copyright ...