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

Flutter 底部导航与顶部选项卡实战:状态保持与鸿蒙适配

综述由AI生成Flutter 底部导航与顶部选项卡是构建 App 骨架的核心。本文对比了 BottomNavigationBar 与 Material 3 NavigationBar 的差异,演示了 TabBar 配合 DefaultTabController 的滑动切换方案。重点解决了页面切换后状态丢失问题,介绍了 IndexedStack 与 AutomaticKeepAliveClientMixin 两种保持状态的方法。同时针对 OpenHarmony 系统手势条进行了安全区域适配,确保底部导航不被遮挡。

灰度发布发布于 2026/3/21更新于 2026/6/518 浏览
Flutter 底部导航与顶部选项卡实战:状态保持与鸿蒙适配

Flutter 底部导航与顶部选项卡实战:状态保持与鸿蒙适配

一个复杂的 App 通常包含多个功能模块。打开微信、淘宝或抖音,你会发现它们都有一个共同的架构:底部有 4-5 个图标切换主页面,顶部还有'关注/推荐'这样的分类切换。这就是移动端最经典的 '底 Tab + 顶 Tab' 双导航架构。

一、底部导航:App 的根基

Flutter 提供了两种主流的底部导航组件,选择哪种取决于你的设计风格和版本需求。

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, // 超过 3 个 item 必须设置 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 使用。

2.1 DefaultTabController (推荐)

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

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

  @override
  Widget build(BuildContext context) {
    return DefaultTabController(
      length: 3, // Tab 数量
      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。它会一次性加载所有页面,只是不可见部分不渲染 UI。

// 修改 Scaffold body
body: IndexedStack(
  index: _currentIndex,
  children: _pages, // 所有页面都会被保留在 Widget 树中,只是不可见
),

注意:如果页面较多或内存敏感,需谨慎使用此方案。

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); // 3. 必须调用 super.build
    return ListView.builder(
      itemCount: 100,
      itemBuilder: (c, i) => ListTile(title: Text('Item $i')),
    );
  }
}

四、OpenHarmony 鸿蒙适配专题

现在的鸿蒙手机(如 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,视觉更现代;老项目用 BottomNavigationBar 也没问题。
  2. 顶部 Tab:使用 DefaultTabController 配合 TabBarView 最省事。
  3. 状态保持:不想每次切换都重加载?请记住 IndexedStack (简单粗暴) 或 AutomaticKeepAliveClientMixin (精细控制)。
  4. 鸿蒙适配:时刻留意底部的安全区域,不要让按钮贴底太近,避免被手势条遮挡。

目录

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

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

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

更多推荐文章

查看全部
  • 2026 届学位论文 AIGC 检测率要求及应对策略
  • C++ 并发核心:内存序、可见性与指令重排
  • 无人机视角山区泥石流与滑坡图像识别数据集
  • Python 环境变量配置指南
  • 基于 Python 的 Telegram 信息检索机器人搭建指南
  • GitHub 日榜精选:AI 代理与开发工具趋势 (2026-02-24)
  • 医疗 NLP 实战:从电子病历分析到智能问答模型落地
  • 交换瓶子问题 Java 最小交换次数解法
  • Ubuntu 部署 OpenClaw 并接入飞书机器人
  • GitHub Copilot Agent Skills:打造跨项目 AI 专属工具箱
  • HarmonyOS DevEco Studio 安装与应用工程创建指南
  • SheetJS:全场景适用的 JavaScript 电子表格处理工具
  • SpringBoot 整合 LangChain4j 与 Tavily 实现联网搜索及 API Key 获取
  • 机器人实践:Foxglove 开发环境搭建指南及常见问题
  • Web Worker:前端多线程开发的隐形引擎
  • C++ 四十年演进:引用、内联与空指针的三大基石
  • 产品经理的多维度划分与进阶路径
  • 攻防世界 MISC 进阶题:图片隐写与 UUencode 解密实战
  • PostgreSQL 动态分区裁剪技术:查询性能优化实战
  • 前端大文件分片上传与断点续传实现方案

相关免费在线工具

  • 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

  • JSON美化和格式化

    将JSON字符串修饰为友好的可读格式。 在线工具,JSON美化和格式化在线工具,online