跳到主要内容
极客日志极客日志面向AI+效率的开发者社区
首页博客GitHub 精选镜像工具UI配色美学隐私政策关于联系
搜索内容 / 工具 / 仓库 / 镜像...⌘K搜索
注册
博客列表
Python

Flutter for OpenHarmony 实战:BottomNavigationBar 与 TabBar 多页切换

!底部导航组件对比 Flutter for OpenHarmony 实战:BottomNavigationBar 与 TabBar 多页切换 > **摘要**:一个复杂的 App 通常包含多个功能模块。将深入讲解 Flutter 中最核心的两种多页切换模式:底部导航 (BottomNavigationBar) 和顶部选项卡 (TabBar)。我们将探讨 Material 3 风格的新组件 Na…

指针猎手发布于 2026/4/6更新于 2026/5/2468K 浏览
Flutter for OpenHarmony 实战:BottomNavigationBar 与 TabBar 多页切换

底部导航组件对比

Flutter for OpenHarmony 实战:BottomNavigationBar 与 TabBar 多页切换

摘要:一个复杂的 App 通常包含多个功能模块。本文将深入讲解 Flutter 中最核心的两种多页切换模式:底部导航 (BottomNavigationBar) 和顶部选项卡 (TabBar)。我们将探讨 Material 3 风格的新组件 NavigationBar,解决页面切换时的状态丢失问题,并适配鸿蒙系统的底部手势条。

前言

打开你手机里的微信、淘宝或抖音,你会发现它们都有一个共同的架构:底部有 4-5 个图标,点击切换不同的主页面;顶部可能还有'关注/推荐/热榜'这样的分类切换。

这就是移动端最经典的 '底 Tab + 顶 Tab' 双导航架构。

本文你将学到:

  • BottomNavigationBar (经典) 与 NavigationBar (Material 3) 的区别
  • TabBar + TabBarView 实现滑动切换
  • 核心难点:如何让页面在切换后不重置?(AutomaticKeepAliveClientMixin)
  • 鸿蒙适配:底部导航栏如何避开系统手势条 (Home Indicator)

一、底部导航:App 的根基

Flutter 提供了两种主流的底部导航组件。

BottomNavigationBar 与 NavigationBar 对比

1.1 经典款:BottomNavigationBar

这是最传统、兼容性最好的组件。

class MainPage extends StatefulWidget {
  const MainPage({super.key});

  @override
  State<MainPage> createState() => _MainPageState();
}

class _MainPageState extends State<MainPage> {
  int _currentIndex = 0;

  final List<Widget> _pages = [
    const HomePage(),
    const CategoryPage(),
    const ProfilePage(),
  ];

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: _pages[_currentIndex],
      bottomNavigationBar: BottomNavigationBar(
        currentIndex: _currentIndex,
        onTap: (index) => setState(() => _currentIndex = index),
        type: BottomNavigationBarType.fixed,
        selectedItemColor: Colors.blue,
        unselectedItemColor: Colors.grey,
        items: const [
          BottomNavigationBarItem(icon: Icon(Icons.home), label: '首页'),
          BottomNavigationBarItem(icon: Icon(Icons.category), label: '发现'),
          BottomNavigationBarItem(icon: Icon(Icons.person), label: '我的'),
        ],
      ),
    );
  }
}
1.2 新潮款:NavigationBar (M3)

Material 3 引入了更高、更圆润的 NavigationBar。它自带点击涟漪和胶囊状的指示器,视觉效果更好。

NavigationBar(
  selectedIndex: _currentIndex,
  onDestinationSelected: (index) => setState(() => _currentIndex = index),
  destinations: const [
    NavigationDestination(
      icon: Icon(Icons.home_outlined),
      selectedIcon: Icon(Icons.home),
      label: '首页',
    ),
    NavigationDestination(
      icon: Icon(Icons.explore_outlined),
      selectedIcon: Icon(Icons.explore),
      label: '发现',
    ),
  ],
)

二、顶部选项卡:TabBar

TabBar 通常用于在同一个主栏目下,切换不同的子分类(如新闻 App 的频道)。它必须配合 TabController 使用。

TabBar 结构示意

2.1 DefaultTabController (推荐)

最简单的方法是在父级包裹一个 DefaultTabController,这样我们就不用手动管理 Controller 了。

class NewsPage extends StatelessWidget {
  const NewsPage({super.key});

  @override
  Widget build(BuildContext context) {
    return DefaultTabController(
      length: 3,
      child: Scaffold(
        appBar: AppBar(
          title: const Text('资讯'),
          bottom: const TabBar(
            tabs: [
              Tab(text: '推荐'),
              Tab(text: '科技'),
              Tab(text: '体育'),
            ],
          ),
        ),
        body: const TabBarView(
          children: [
            Center(child: Text('推荐内容列表')),
            Center(child: Text('科技内容列表')),
            Center(child: Text('体育内容列表')),
          ],
        ),
      ),
    );
  }
}

三、核心难点:页面状态保持

页面状态保持示意

默认情况下,当你从'首页'切换到'我的',再切回'首页'时,首页会被重建(列表滚动位置丢失,输入框清空)。这是因为 Scaffold.body 直接替换了 Widget。

3.1 解决方案:IndexedStack

如果你希望所有页面在初始化后一直存在,可以使用 IndexedStack。它会一次性加载所有页面(注意内存消耗)。

// 修改 Scaffold body
body: IndexedStack(
  index: _currentIndex,
  children: _pages,
),
3.2 解决方案:AutomaticKeepAliveClientMixin

如果你使用的是 PageView 或 TabBarView,更推荐让子页面自己决定是否保持状态。

class HomePage extends StatefulWidget {
  const HomePage({super.key});

  @override
  State<HomePage> createState() => _HomePageState();
}

// 1. 混入 AutomaticKeepAliveClientMixin
class _HomePageState extends State<HomePage> with AutomaticKeepAliveClientMixin {
  // 2. 重写 wantKeepAlive 返回 true
  @override
  bool get wantKeepAlive => true;

  @override
  Widget build(BuildContext context) {
    super.build(context);
    return ListView.builder(
      itemCount: 100,
      itemBuilder: (c, i) => ListTile(title: Text('Item $i')),
    );
  }
}

四、OpenHarmony 鸿蒙适配专题

底部导航栏与手势条

4.1 底部导航栏与手势条

现在的鸿蒙手机(如 Mate 60)默认开启全面屏手势,底部有一条'黑条'或'白条' (Home Indicator)。

如果你的 BottomNavigationBar 高度写死,或者没有适配 SafeArea,底部的图标可能会被这个手势条遮挡。

Flutter 的 Scaffold + BottomNavigationBar 默认已经处理了 SafeArea。但如果你使用了自定义的底部栏(比如 Stack 里的 Positioned),务必包裹 SafeArea 并设置 bottom: true。

Align(
  alignment: Alignment.bottomCenter,
  child: SafeArea(
    child: Container(
      height: 60,
      color: Colors.white,
      child: Row(...),
    ),
  ),
)

五、总结

搭建一个 App 的骨架,核心就是'一底一顶'。

核心要点
  1. 底部导航:推荐使用 M3 风格的 NavigationBar,视觉更现代。
  2. 顶部 Tab:使用 DefaultTabController 配合 TabBarView 最省事。
  3. 状态保持:不想每次切换都重加载?请记住 IndexedStack (简单粗暴) 或 AutomaticKeepAliveClientMixin (精细控制)。
  4. 鸿蒙适配:时刻留意底部的安全区域,不要让按钮贴底太近。
下一篇预告

基本的页面结构都有了,但是我们现在的布局还是太'循规蹈矩'了(一行一列)。如果我要做一个 Pinterest 那样的瀑布流,或者像相册一样的网格呢?

《Flutter for OpenHarmony 实战之基础组件:第十二篇 GridView 网格布局详解》 我们将突破线性布局的限制,探索二维空间的布局艺术。


目录

  1. Flutter for OpenHarmony 实战:BottomNavigationBar 与 TabBar 多页切换
  2. 前言
  3. 一、底部导航:App 的根基
  4. 1.1 经典款:BottomNavigationBar
  5. 1.2 新潮款:NavigationBar (M3)
  6. 二、顶部选项卡:TabBar
  7. 2.1 DefaultTabController (推荐)
  8. 三、核心难点:页面状态保持
  9. 3.1 解决方案:IndexedStack
  10. 3.2 解决方案:AutomaticKeepAliveClientMixin
  11. 四、OpenHarmony 鸿蒙适配专题
  12. 4.1 底部导航栏与手势条
  13. 五、总结
  14. 核心要点
  15. 下一篇预告
  • 💰 8折买阿里云服务器限时8折了解详情
  • Magick API 一键接入全球大模型注册送1000万token查看
  • 🤖 一键搭建Deepseek满血版了解详情
  • 一键打造专属AI 智能体了解详情
极客日志微信公众号二维码

微信扫一扫,关注极客日志

微信公众号「极客日志V2」,在微信中扫描左侧二维码关注。展示文案:极客日志V2 zeeklog

更多推荐文章

查看全部
  • FPGA 中 RS485 收发器应用与毛刺处理方案
  • SpringBoot 整合 LangChain4j 与 Tavily 实现联网搜索
  • Flutter 基础组件:BottomNavigationBar 与 TabBar 多页切换及鸿蒙适配
  • ChatMed 开源中文医疗大模型介绍与对比分析
  • 家居家电行业大模型落地应用与趋势分析
  • Python 核心应用领域与开发环境配置详解
  • 基于 C++11 手写前端 Promise 实现
  • Windows 系统下安装与编译 llama.cpp
  • 大语言模型(LLM)应用开发流程与实战项目
  • 树结构与二叉树核心概念及转换实战
  • Qwen3.5 多模态模型部署 OOM 排查:enforce-eager 参数配置详解
  • Rust 与 WebAssembly 实战:在浏览器与 Node.js 运行高性能代码
  • LightRAG 详解:基于图结构的检索增强生成系统实践
  • Qwen-Image-2512 V2 模型 ComfyUI 及 WebUI 整合部署指南
  • 基于浏览器扩展实现微信网页版访问方案
  • 大规模无人机检测数据集:11998张图像,支持YOLOv8/COCO/TensorFlow训练
  • 三数之和算法详解:排序与双指针
  • SpringBoot 整合 Langchain4j 对接主流大模型实战详解
  • 8 条实用 Python 代码案例与初学者核心知识点
  • DeepSeek-R1 大模型基于 MS-Swift 框架部署、推理与微调指南

相关免费在线工具

  • curl 转代码

    解析常见 curl 参数并生成 fetch、axios、PHP curl 或 Python requests 示例代码。 在线工具,curl 转代码在线工具,online

  • Base64 字符串编码/解码

    将字符串编码和解码为其 Base64 格式表示形式即可。 在线工具,Base64 字符串编码/解码在线工具,online

  • Base64 文件转换器

    将字符串、文件或图像转换为其 Base64 表示形式。 在线工具,Base64 文件转换器在线工具,online

  • Markdown转HTML

    将 Markdown(GFM)转为 HTML 片段,浏览器内 marked 解析;与 HTML转Markdown 互为补充。 在线工具,Markdown转HTML在线工具,online

  • HTML转Markdown

    将 HTML 片段转为 GitHub Flavored Markdown,支持标题、列表、链接、代码块与表格等;浏览器内处理,可链接预填。 在线工具,HTML转Markdown在线工具,online

  • JSON 压缩

    通过删除不必要的空白来缩小和压缩JSON。 在线工具,JSON 压缩在线工具,online