在移动端页面中,由于屏幕空间有限,导航条扮演着非常重要的角色,提供了快速导航到不同页面或功能的方式。用户也通常会在导航条中寻找他们感兴趣的内容,因此导航条的曝光率较高。在这样的背景下,提供一个动态灵活的导航条,为产品赋能,变得尤其重要。
拿 iOS 原生导航条为例,导航条作为页面进出栈的根视图连接器,以及生命周期的管理器。尤其是在作为 webView Controller 的父容器的时候,面对 webview 中 h5 页面灵活的的路由属性,以及一些难料的异常情况,原生很难也不便于频繁操作根试图容器,因此也产生了一些性能差、体验差、开发成本高、测试场景难覆盖等问题。安卓也有类似情况。
•ssr
预渲染时,无法对原生导航条进行预加载。对于百亿,便宜包邮等使用 ssr 预渲染的频道,因为原生导航栏无法进行预加载,导致上屏较慢等问题。
•原生导航条生命周期耦合。原生导航条作为 webviewController 的根容器,一旦操作时机不当,很可能影响到线上页面,而且最大的问题在于这种场景测试很难覆盖。比如:window.href.url
使用这种方式更新当前页面时,由于不同频道操作同一根导航条,会引发不可预知的问题;
•场景有限。站外场景无法使用原生导航条,一些业务方往往需要单独处理站内外,造成开发资源浪费。
•webview 初始化时会预置一个默认的导航条,然后根据前端配置,再去设置导航条的不同样式,无法避免的存在一个过渡期,体验较差。
•window.location.reload()
刷新当前页面的时候,即便是在 js 中隐藏了导航条,webview
为了兼容一个线上问题,执行 reload 时此时会先展示原生导航条,直到执行了 js 的隐藏逻辑,才会被隐藏,体验较差。
•无法扩展交互动效。得益于移动端页面中,导航条得天独厚的位置,产品往往希望有更生动的交互性,来提高曝光、粘性、活动触达率等。比如导航栏上挂载搜索框、以及吸顶、延伸动画、沉浸式、炫酷的营销 icon 等等。遗憾的是原生系统导航条不能全部支持,其实无论从视图层级上来说,还是从导航条职责上来说,apple
并不希望过多操作导航栏上的元素。也就造成了高曝光位置的资源浪费。
•因为要依赖原生 JS 桥,就一定会存在版本限制问题。造成需求迭代慢,甚至随着时间的推移,版本卡口原因无迹可寻,代码调整战战兢兢,版本审核慢、周期长等问题。
基于原生导航条现状,百亿补贴频道沉淀出了通用 H5 导航条组件@pango/navigation-bar
,具有以下优势:
•支持 ssr 预渲染,上屏较快。
•人力节省百分之 90% 以上,以 plus 95 折为例,对接只需 0.5/人日。
•无场景限制。可用于站内外,ssr 以及 csr 场景,无需站内外多次开发。
•可配置。 @pango/navigation-bar
使用 config 的形式配置 item,这么做的好处是一旦业务需求改动,只需调整配置,无需调整组件逻辑,极大降低开发和测试成本。另外如果你使用主站的 webview 并且配置了 config,那么只需要简单的改动 config,代码迁移成本低。
•导航条在频道内和其他普通楼层无异,生命周期隔离清晰,不会影响别的页面,测试成本低。
•单向数据流设计,外部数据变化,组件 UI 及时响应,不存在原生的操作窗口问题,开发体验佳;
•生命周期和其他楼层保持同步,规避了原生容器和 H5 页面天然的生命周期无法同步的问题,也就不存在两者之间的过渡问题,体验佳。
•采用左、中、右、状态栏、导航栏分层设计的模式,支持传入 React.ReactElement,比原生定制性更强,可灵活定制目前站内绝大部分导航条样式以及交互动画,合理高效利用导航条资源;
•参考原生导航栏异形屏适配方案,参考原生绝对布局思路,完美适配折叠屏、异形屏。
•iOS9 - 最新 、Android5 - 最新均兼容性良好,未发现线上兼容异常。
纯手工打造,未使用第三方库,不会对宿主造成依赖冲突,随时改动随时发布不存在版本控制,最大程度的降低和隔断对原生容器的版本依赖。
原生导航条作为根试图容器,容器内子视图异常不会影响根试图的展示,所以不用特殊处理 html 下载失败,js 执行异常,服务挂掉等异常情况。但是 H5 导航条遇到这些异常情况,也要保证用户可以点击返回按钮返回上一页。
目前方案已和通天塔以及 hybrid 团队打通,方案如下:
异常场景 1:业务 js 执行异常。
• @pango/navigation-bar 组件使用 a 标签渲染返回按钮,保证 js 执行异常时依然展示返回按钮,并且能正常响应返回事件。
•业务展示兜底错误页时,会使用导航条兜底数据渲染导航条确保可返回上一级。
异常场景 2:webview 加载 html 失败。
为了消除上面提到的过渡问题,业务链接中新增了qurey
参数hideNavi=1
,原生 webview 会通过该字段在 webview 出现之前隐藏导航条。但是因此也引发了一个风险:html 加载失败时,会造成无头的问题。因此需要 webview 配合改造,一旦监测到 html 加载失败,原生 webview 要展示原生导航条。
异常场景 3:通天塔服务异常。
同样是场景 2 中的问题,需要通天塔配合改造通天塔服务异常的场景:依据链接中hideNavi
字段添加返回按钮或者通知 webview 展示默认导航条。
观察多个竞品以及兄弟频道,发现在上述的异常场景 2、3 下,均未做特别处理,展示无头错误页。如果此时原生禁用了右滑返回手势,页面将无法返回上一级,这无异是一个非常严重的缺陷(事实上有些竞品页面以及我们某些频道确实无法返回上一级)。
目前使用该组件的项目:百亿补贴、月黑风高、PLUS95 折。
参考原生navigationBar
的设计思路,把整个导航栏分为左、右、中
三个区域,左、右区域根据内容自适应宽度,剩余空间为中间区域。左右区域接受 items 数组,可根据 item 接口协议设置左右的 items,协议可自定义图片、尺寸、事件、间距、下拉菜单、是否动画响应等,已默认包含了关注、返回、更多、频道 logo 等常用元素,当然如有需要也可以自定义一个React.ReactElement
。中间区域只接受React.ReactElement
,你可以自由定制元素,传入navigation-bar
即可,一张图片一段文字,或者是一个搜索框……不管是伸缩或者是上滑吸顶,都可自定义。
该组件使用 react + ts 开发,大小只有 4.1K。
文件结构:
npm i @pango/navigation-bar --registry=http://registry.m.jd.com
你可以自由配置 items 除了"follow", "more","back","logo"
,这些已知的元素外还可以设置type:"common",
是一个通用类型的 item;
scrollCallBack
会回调上滑比例,可根据该比例做交互动画;
import {
BACK_ICON,
FEEDBACK_ICON,
FEEDBACK_URL,
INavigationParams,
MORE_ICON,
RULE_ICON,
SHARE_ICON,
} from "@pango/navigation-bar";
setH5NavigationButton = (headerData) => {
const extend = headerData?.navigationBar?.extend;
const followInfo = headerData?.navigationBar?.followInfo;
const follow = {
type: "follow",
collectionId: String(followInfo?.themeId),
gapWidth: 12,
width: 55,
height: 22,
};
const moreItem = {
type: "more",
menuBackgroundColor: "white",
img: MORE_ICON,
title: "更多",
menuList: [],
};
moreItem.menuList.push({
icon: RULE_ICON,
title: "规则页",
menuEventData: extend?.guideUrl,
});
moreItem.menuList.push({
icon: SHARE_ICON,
title: "分享",
type: "share",
menuEventData: extend?.share,
});
const backItem = {
type: "back",
img: BACK_ICON,
canClick: !margicWindow,
title: "返回",
};
const backLogo = {
type: "logo",
img: DEFAULT_LOGO,
isAnimation: true,
gapWidth: 5,
width: 176,
height: 34
};
const navBarParams: INavigationParams = {
leftItems: [],
rightItems: [],
backgroundColor: "#FD4D00",
navHeight: this.status.navHeight,
};
navBarParams.leftItems.push(backItem, backLogo);
navBarParams.rightItems.push(moreItem, follow);
navBarParams.titleImgItem = TitleSearch({});
navBarParams.scrollCallBack = (scale) => {
this.setStatus({
navigationBarParams: Object.assign(this.status.navigationBarParams, {
titleImgItem: TitleSearch({ isCollapse: scale === 1 })
})
});
}
return navBarParams;
};
特别注意 titleImgItem,这个属性是导航条中间区域的展示内容,TitleSearch
是百亿补贴的搜索框,你可以参考该元素自定义中间区域。
import { INavigationParams, NavigationBar } from "@pango/navigation-bar";
import "@pango/navigation-bar/lib/navigation-bar.scss";
css
.nav-bar {
width: 750px;
z-index: 1;
top: 0px;
}
<NavigationBar
className="nav-bar"
params={data.navParams}
barHeight={200} //自定义导航栏高度
event={do somethings}
/>
Q:若原生导航条隐藏,此时异常怎么办?
异常分为以下 3 类:
异常场景 1:业务 js 执行异常。
• @pango/navigation-bar 组件使用 a 标签渲染返回按钮,保证 js 执行异常时依然展示该标签,并且能正常相应出栈事件。
•业务展示兜底错误页时,会使用导航条兜底数据渲染导航条。
异常场景 2:webview 加载 html 失败。
为了消除上面提到的过渡问题,业务链接中新增了qurey
参数hideNavi=1
,原生 webview 会通过该字段在 webview 出现之前隐藏导航条。但是因此也引发了一个风险:html 加载失败时,会造成无头的问题。因此需要 webview 配合改造,一旦监测到 html 加载失败,原生 webview 要展示原生导航条。
异常场景 3:通天塔服务异常。
同样是场景 2 中的问题,需要通天塔配合改造通天塔服务异常的场景:依据链接中hideNavi
字段添加返回按钮或者通知 webview 展示默认导航条。
若发现其他异常,麻烦提醒我。
Q:折叠屏怎么适配?
折叠屏适配一直是前端适配的噩梦,噩梦的根本原因在于:宽度于高度的比例非常值,前端布局是往往会把 px 转换成 vw,因此造成了异形屏适配难的问题。
•参考原生系统导航栏的绝对布局方案:@pango/navigation-bar
把导航条拆分为状态栏和导航栏上下两部分, 导航条宽度屏幕自适应,导航条高度跟随设备变化,并采用大写的PX
单位来固定元素尺寸。根据协议 item 宽高、间距仍可自定义,但是大写的PX
保证了 item 不会随着屏幕宽度而异常变化。
navigation-bar {
width: 750px; // 会转换成vw
height: 44PX; // 不会转换成vw
display: flex;
position: absolute;
.left-items-bg {
margin-left: 16PX; // 不会转换成vw
height: 22PX;
margin-top: 11PX;
width: fit-content;
display: flex;
align-items: center;
justify-content: center;
}
}
Q:原生导航条优化?
现状中的几个异常场景,仍需要 webview 配合一起整改,所以目前整改方案为:
业务链接中新增qurey
参数hideNavi=1
,此时 webview 通过该字段在 webview 出现之前隐藏导航条。由 webview 负责整改,跟版 12.1.4。
经安全部门审核之后,会向外开源。
•导航条在移动端页面中的重要性无需多言,我们最终的目的是面向全集团,和通天塔以及 hybrid 团队,一起打造一根规范通用的 H5 导航栏,如果你在使用过程中发现一些我们没有考虑到的异常场景或者设计规范,请与我联系,我们共同完善。
•目前该组件下拉刷新还是要依赖原生的下拉刷新事件,后期会定制 H5 自己的下拉刷新。
•一个规范的 UI 组件应该是一个有严格 UI 设计规范的,比如间距,字体大小、图片规范等。但是一期的设计中我们为了灵活,通过协议把 UI 把控留给了用户,也希望后面的迭代开发中融入更多规范的设计语言。
作者:京东零售 张松超
来源:京东云开发者社区 转载请注明来源