热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

小程序自定义导航栏(适配所有手机)

背景在做小程序时,关于默认导航栏,我们遇到了以下的问题:Android、IOS手机对于页面title的展示不一致,安卓title的显示不居中页面的title只支持纯文本级别的样式控

背景

在做小程序时,关于默认导航栏,我们遇到了以下的问题:

  • Android、IOS手机对于页面title的展示不一致,安卓title的显示不居中
  • 页面的title只支持纯文本级别的样式控制,不能够做更丰富的title效果
  • 左上角的事件无法监听、定制
  • 路由导航单一,只能够返回上一页,深层级页面的返回不够友好

探索

小程序自定义导航栏已开放许久>>了解一下,相信不少小伙伴已使用过这个功能,同时不少小伙伴也会发现一些坑:

  • 机型多如牛毛:自定义导航栏高度在不同机型始终无法达到视觉上的统一
  • 调皮的胶囊按钮:导航栏元素(文字,图标等)怎么也对不齐那该死的胶囊按钮
  • 各种尺寸的全面屏,奇怪的刘海屏,简直要抓狂

一探究竟

为了搞明白原理,我先去翻了官方文档,>>飞机,点过去是不是很惊喜,很意外,通篇大文尽然只有最下方的一张图片与这个问题有关,并且啥也看不清,汗汗汗…

我特意找了一张图片来
《小程序自定义导航栏(适配所有手机)》
分析上图,我得到如下信息:

  • Android跟iOS有差异,表现在顶部到胶囊按钮之间的距离差了6pt
  • 胶囊按钮高度为32pt, iOS和Android一致

动手分析

我们写一个状态栏,通过wx.getSystemInfoSync().statusBarHeight设置高度

Android:
《小程序自定义导航栏(适配所有手机)》
iOS:
《小程序自定义导航栏(适配所有手机)》

可以看出,iOS胶囊按钮与状态栏之间距离为:4px, Android为8px,是不是所有手机都是这种情况呢?

答案是:苹果手机确实都是4px,安卓大部分都是7和8 也会有其他的情况(可以自己打印getSystemInfo验证)如何快速便捷算出这个高度,请接着往下看

如何计算

导航栏分为状态栏和标题栏,只要能算出每台手机的导航栏高度问题就迎刃而解

  • 导航栏高度 = 胶囊按钮高度 + 状态栏到胶囊按钮间距 * 2 + 状态栏高度

注:由于胶囊按钮是原生组件,为表现一致,其单位在各种手机中都为px,所以我们自定义导航栏的单位都必需是px(切记不能用rpx),才能完美适配。

解决问题

现在我们明白了原理,可以利用胶囊按钮的位置信息和statusBarHeight高度动态计算导航栏的高度,贴一个实现此功能最重要的方法

let systemInfo = wx.getSystemInfoSync();
let rect = Taro.getMenuButtonBoundingClientRect ? Taro.getMenuButtonBoundingClientRect() : null; //胶囊按钮位置信息
Taro.getMenuButtonBoundingClientRect();
let navBarHeight = (function() { //导航栏高度
let gap = rect.top - systemInfo.statusBarHeight; //动态计算每台手机状态栏到胶囊按钮间距
return 2 * gap + rect.height;
})();

gap信息就是不同的手机其状态栏到胶囊按钮间距,具体更多代码实现和使用demo请移步下方代码仓库,代码中还会有输入框文字跳动解决办法,安卓手机输入框文字飞出解决办法,左侧按钮边框太粗解决办法等等

吐槽

这么重要的问题,官方尽然没有提供解决方案…竟然提供了一张看不清的图片???

网上有很多ios设置44,android设置48,还有根据不同的手机型号设置不同高度,通过长时间的开发和尝试,本人发现以上方案并不完美,并且bug很多

代码库

  • Taro组件gitHub地址详细用法请参考README
  • 原生组件npm构建版本gitHub地址详细用法请参考README
  • 原生组件简易版gitHub地址详细用法请参考README
  • 由于本人精力有限,目前只计划发布维护好这2种组件,其他组件请自行修改代码,有问题请联系

备注

  • 上方2种组件在最下方30多款手机测试情况表现良好
  • iPhone手机打电话和开热点导致导航栏样式错乱,问题已经解决啦,请去demo里测试,这里特别感谢moments网友提出的问题
  • 本文章并无任何商业性质,如有侵权请联系本人修改或删除
  • 文章少量部分内容是本人查询搜集而来
  • 如有问题可以下方留言讨论,微信zhijunxh

比较

斗鱼:
《小程序自定义导航栏(适配所有手机)》

虎牙:
《小程序自定义导航栏(适配所有手机)》

微博:
《小程序自定义导航栏(适配所有手机)》

酷狗:
《小程序自定义导航栏(适配所有手机)》

知乎:
《小程序自定义导航栏(适配所有手机)》
《小程序自定义导航栏(适配所有手机)》

知乎是这里边做的最好的,但是我个人认为有几个可以优化的小问题

  • 打电话或者开启热点导致样式错落,这也是大部门小程序的问题
  • 导航栏下边距太小,看起来不舒服
  • 搜索框距离2侧按钮组距离不对等
  • 自定义返回和home按钮中的竖线颜色重了,并且感觉太粗

如果您看到了此篇文章,请赶快修改自己的代码,并运用在实践中吧

扫码体验我的小程序:
《小程序自定义导航栏(适配所有手机)》

创作不易,如果对你有帮助,请移步Taro组件
gitHub原生组件
gitHub给个星星 star✨✨ 谢谢

测试信息

手机型号胶囊位置信息statusBarHeight测试情况
iPhoneX80 32 281 369 48 8844通过
iPhone8 plus56 32 320 408 24 8820通过
iphone756 32 281 368 24 8720通过
iPhone6 plus56 32 320 408 24 8820通过
iPhone656 32 281 368 24 8720通过
HUAWEI SLA-AL0064 32 254 350 32 9624通过
HUAWEI VTR-AL0064 32 254 350 32 9624通过
HUAWEI EVA-AL0064 32 254 350 32 9624通过
HUAWEI EML-AL0068 32 254 350 36 9629通过
HUAWEI VOG-AL0065 32 254 350 33 9625通过
HUAWEI ATU-TL1064 32 254 350 32 9624通过
HUAWEI SMARTISAN OS10564 32 326 422 32 9624通过
XIAOMI MI659 28 265 352 31 8723通过
XIAOMI MI4LTE60 32 254 350 28 9620通过
XIAOMI MIX374 32 287 383 42 9635通过
REDMI NOTE364 32 254 350 32 9624通过
REDMI NOTE464 32 254 350 32 9624通过
REDMI NOTE355 28 255 351 27 9620通过
REDMI 5plus67 32 287 383 35 9628通过
MEIZU M571C65 32 254 350 33 9625通过
MEIZU M6 NOTE62 32 254 350 30 9622通过
MEIZU MX4 PRO62 32 278 374 30 9622通过
OPPO A3365 32 254 350 33 9626通过
OPPO R1158 32 254 350 26 9618通过
VIVO Y5564 32 254 350 32 9624通过
HONOR BLN-AL2064 32 254 350 32 9624通过
HONOR NEM-AL1059 28 265 352 31 8724通过
HONOR BND-AL1064 32 254 350 32 9624通过
HONOR duk-al2064 32 254 350 32 9624通过
SAMSUNG SM-G955064 32 305 401 32 9624通过
360 1801-A0164 32 254 350 32 9624通过

推荐阅读
  • 本文探讨了使用JavaScript在不同页面间传递参数的技术方法。具体而言,从a.html页面跳转至b.html时,如何携带参数并使b.html替代当前页面显示,而非新开窗口。文中详细介绍了实现这一功能的代码及注释,帮助开发者更好地理解和应用该技术。 ... [详细]
  • 本文详细介绍了一种利用 ESP8266 01S 模块构建 Web 服务器的成功实践方案。通过具体的代码示例和详细的步骤说明,帮助读者快速掌握该模块的使用方法。在疫情期间,作者重新审视并研究了这一未被充分利用的模块,最终成功实现了 Web 服务器的功能。本文不仅提供了完整的代码实现,还涵盖了调试过程中遇到的常见问题及其解决方法,为初学者提供了宝贵的参考。 ... [详细]
  • 本文介绍了如何利用 Delphi 中的 IdTCPServer 和 IdTCPClient 控件实现高效的文件传输。这些控件在默认情况下采用阻塞模式,并且服务器端已经集成了多线程处理,能够支持任意大小的文件传输,无需担心数据包大小的限制。与传统的 ClientSocket 相比,Indy 控件提供了更为简洁和可靠的解决方案,特别适用于开发高性能的网络文件传输应用程序。 ... [详细]
  • 小程序的授权和登陆
    小程序的授权和登陆 ... [详细]
  • Cookie学习小结
    Cookie学习小结 ... [详细]
  • 本文将介绍如何在混合开发(Hybrid)应用中实现Native与HTML5的交互,包括基本概念、学习目标以及具体的实现步骤。 ... [详细]
  • 本文详细解析了ASP.NET 2.0中的Callback机制,不仅介绍了基本的使用方法,还深入探讨了其背后的实现原理。通过对比Atlas框架,帮助读者更好地理解和应用这一机制。 ... [详细]
  • 本文详细介绍了如何在PHP中记录和管理行为日志,包括ThinkPHP框架中的日志记录方法、日志的用途、实现原理以及相关配置。 ... [详细]
  • 微信公众号推送模板40036问题
    返回码错误码描述说明40001invalidcredential不合法的调用凭证40002invalidgrant_type不合法的grant_type40003invalidop ... [详细]
  • 命令模式是一种行为设计模式,它将请求封装成一个独立的对象,从而允许你参数化不同的请求、队列请求或者记录请求日志。本文将详细介绍命令模式的基本概念、组件及其在实际场景中的应用。 ... [详细]
  • 在PHP中如何正确调用JavaScript变量及定义PHP变量的方法详解 ... [详细]
  • 【实例简介】本文详细介绍了如何在PHP中实现微信支付的退款功能,并提供了订单创建类的完整代码及调用示例。在配置过程中,需确保正确设置相关参数,特别是证书路径应根据项目实际情况进行调整。为了保证系统的安全性,存放证书的目录需要设置为可读权限。值得注意的是,普通支付操作无需证书,但在执行退款操作时必须提供证书。此外,本文还对常见的错误处理和调试技巧进行了说明,帮助开发者快速定位和解决问题。 ... [详细]
  • 如何使用 `org.eclipse.rdf4j.query.impl.MapBindingSet.getValue()` 方法及其代码示例详解 ... [详细]
  • 本文介绍了如何利用Shell脚本高效地部署MHA(MySQL High Availability)高可用集群。通过详细的脚本编写和配置示例,展示了自动化部署过程中的关键步骤和注意事项。该方法不仅简化了集群的部署流程,还提高了系统的稳定性和可用性。 ... [详细]
  • 本文介绍了如何利用Struts1框架构建一个简易的四则运算计算器。通过采用DispatchAction来处理不同类型的计算请求,并使用动态Form来优化开发流程,确保代码的简洁性和可维护性。同时,系统提供了用户友好的错误提示,以增强用户体验。 ... [详细]
author-avatar
松原电信曹玉威_203
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有