Flutter 126: 图解自定义两侧对齐 ACETabBar 标签导航栏

简介: 0 基础学习 Flutter,第一百二十六步:简单定义一个可设置左右两侧对齐方式及两侧固定位 Icon 的 ACETabBar 导航栏!

    小菜在实践学习过程中,需要把 TabBar 标签栏默认居左,而 TabBar 默认是居中状态;小菜借此机会学习一下 TabBar 源码,稍微调整一下对齐方式;

ACETabBar

ACETabBarAlignType 对齐方式

    小菜添加了一个 alignType 用于设置 ACETabBar 对齐方式;同时设置 isScrollable = true;当 isScrollable = false 时与 TabBar 默认占满屏幕均分效果一致;

enum ACETabBarAlignType { left, center, right }

源码分析

    小菜分析 TabBar 源码,在 _TabBarStateTabBar 绘制过程中,多个子 Tab 通过 SingleChildScrollView 存放,最简单的方式,在 SingleChildScrollView 外添加可以设置对齐方式的 Container 即可;

if (widget.isScrollable) {
  _scrollController ??= _TabBarScrollController(this);
  tabBar = Container(
      alignment: _alignType(widget.alignType ?? ACETabBarAlignType.center),
      child: _scrollView(tabBar));
}

_alignType(alignType) {
  Alignment _type;
  switch (alignType) {
    case ACETabBarAlignType.left:
      _type = Alignment.centerLeft;
      break;
    case ACETabBarAlignType.center:
      _type = Alignment.center;
      break;
    case ACETabBarAlignType.right:
      _type = Alignment.centerRight;
      break;
  }
  return _type;
}

_scrollView(tabBar) {
  return SingleChildScrollView(
      dragStartBehavior: widget.dragStartBehavior,
      scrollDirection: Axis.horizontal,
      controller: _scrollController,
      physics: widget.physics,
      child: tabBar);
}

案例尝试

    小菜尝试了在 isScrollable 是否可滑动两种状态下,ACETabBar 的对齐方式;

_tabBar01() => ACETabBar(
    isScrollable: true, 
    controller: _tabController, tabs: <Widget>[
      Tab(text: '今日', icon: Icon(Icons.account_circle)),
      Tab(text: '今日爆款土货生鲜'),
      Tab(text: '分类')
    ]);
    
_tabBar02() => ACETabBar(
    isScrollable: true,
    alignType: ACETabBarAlignType.left,
    controller: _tabController,
    tabs: <Widget>[
      Tab(text: '今日爆款'),
      Tab(text: '今日爆款土货生鲜'),
      Tab(text: '分类')
    ]);

_tabBar03() => ACETabBar(
    isScrollable: true,
    alignType: ACETabBarAlignType.right,
    controller: _tabController,
    tabs: <Widget>[
      Tab(text: '今日爆款'),
      Tab(text: '今日爆款土货生鲜'),
      Tab(text: '分类')
    ]);
    
_tabBar04() => ACETabBar(
    isScrollable: false,
    alignType: ACETabBarAlignType.right,
    controller: _tabController,
    tabs: <Widget>[
      Tab(text: '今日爆款'),
      Tab(text: '今日爆款土货生鲜'),
      Tab(text: '分类')
    ]);

startIcon & endIcon 固定位图标

    类似很多新闻类或商城类 app,在 TabBar 所在的左右两侧通常会有固定的图标或文字等小 Widget;而小菜也在设置完对齐方式后增加了 startIcon & endIcon 两个图标位;

源码分析

    小菜在设置对齐方式的时了解到 _TabBarState 用于绘制展示是否可滑动的 TabBar,小菜增加两个 startIcon & endIcon 两个属性,在最终 return tabBar 时进行判断是否展示添加到导航栏中;而是否添加点击事件可以通过添加 Widget 时进行处理;

Widget tabBar = CustomPaint(
    painter: _indicatorPainter,
    child: _TabStyle(
        animation: kAlwaysDismissedAnimation,
        selected: false,
        labelColor: widget.labelColor,
        unselectedLabelColor: widget.unselectedLabelColor,
        labelStyle: widget.labelStyle,
        unselectedLabelStyle: widget.unselectedLabelStyle,
        child: _TabLabelBar(onPerformLayout: _saveTabOffsets, children: wrappedTabs)));

if (widget.isScrollable) {
  _scrollController ??= _TabBarScrollController(this);
  tabBar = Container(
      alignment: _alignType(widget.alignType ?? ACETabBarAlignType.center),
      child: _scrollView(tabBar));
}
tabBar = Row(children: [
  widget.startIcon ?? Container(), Flexible(child: tabBar), widget.endIcon ?? Container()
]);
return tabBar;

案例尝试

    小菜尝试在 isScrollable 是否可滑动两种状态下,在导航栏中添加左右两个固定位图标;

_tabBar05(type, isLeft, isRight, {isScrollable}) => ACETabBar(
    isScrollable: isScrollable ?? true,
    alignType: ACETabBarAlignType.left,
    startIcon: isLeft
        ? Padding(
            padding: EdgeInsets.symmetric(horizontal: 10.0),
            child: FlutterLogo()) : null,
    endIcon: isRight
        ? GestureDetector(
            onTap: () => print('----endIcon.click----'),
            child: Padding(
                padding: EdgeInsets.symmetric(horizontal: 10.0),
                child: Icon(Icons.add, color: Colors.white)))
        : null,
    controller: type == 0 ? _tabController : _tabController2,
    tabs: type == 0 ? _tabData02 : _tabData04);
    
_tabBarWid07() => Container(
    height: 200.0,
    child: Scaffold(
        appBar: AppBar(
            title: Text('true & LeftIcon & RightIcon'),
            bottom: _tabBar05(1, true, true)),
        body: _tabBarView(1)));

_tabBarWid08() => Container(
    height: 200.0,
    child: Scaffold(
        appBar: AppBar(
            title: Text('false & LeftIcon & RightIcon'),
            bottom: _tabBar05(1, true, true, isScrollable: false)),
        body: _tabBarView(1)));

小扩展

    小菜在了解 TabBar 源码过程中,简单学习了 Tab Item 以及对应 indicator 标签指标的绘制及对应的滑动过程;其中在 TabBar 绘制过程中会用到 PreferredSizeWidget 小组件;

    PreferredSizeWidget 小组件作为一个抽象接口类,主要用于返回该小部件在不受其他限制的情况下设定的较理想的大小;若没有进行约束高度,则会使用 PreferredSizeWidget 指定的高度;而 TabBar 就是实现了 preferredSize 方法,用于设置高度,小菜尝试调整 preferredSize 即可调整 TabBar 默认高度;

@override
Size get preferredSize {
  for (final Widget item in tabs) {
    if (item is Tab) {
      final Tab tab = item;
      if ((tab.text != null || tab.child != null) && tab.icon != null)
        return Size.fromHeight(_kTextAndIconTabHeight + indicatorWeight);
    }
  }
  return Size.fromHeight(_kTabHeight + indicatorWeight - 50.0);
}


    ACETabBar 案例源码


来源: 阿策小和尚
目录
相关文章
|
10月前
|
Dart 前端开发
【05】flutter完成注册页面完善样式bug-增加自定义可复用组件widgets-严格规划文件和目录结构-规范入口文件-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草央千澈
【05】flutter完成注册页面完善样式bug-增加自定义可复用组件widgets-严格规划文件和目录结构-规范入口文件-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草央千澈
368 75
【05】flutter完成注册页面完善样式bug-增加自定义可复用组件widgets-严格规划文件和目录结构-规范入口文件-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草央千澈
|
9月前
|
前端开发 Java Shell
【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
621 20
【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
|
10月前
|
Dart 前端开发 容器
【07】flutter完成主页-完成底部菜单栏并且做自定义组件-完整短视频仿抖音上下滑动页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草央千澈
【07】flutter完成主页-完成底部菜单栏并且做自定义组件-完整短视频仿抖音上下滑动页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草央千澈
345 18
【07】flutter完成主页-完成底部菜单栏并且做自定义组件-完整短视频仿抖音上下滑动页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草央千澈
|
UED 开发者 容器
Flutter&鸿蒙next 的 Sliver 实现自定义滚动效果
Flutter 提供了强大的滚动组件,如 ListView 和 GridView,但当需要更复杂的滚动效果时,Sliver 组件是一个强大的工具。本文介绍了如何使用 Sliver 实现自定义滚动效果,包括 SliverAppBar、SliverList 等常用组件的使用方法,以及通过 CustomScrollView 组合多个 Sliver 组件实现复杂布局的示例。通过具体代码示例,展示了如何实现带有可伸缩 AppBar 和可滚动列表的页面。
442 1
|
UED
Flutter&鸿蒙next 中的 Drawer 导航栏
在 Flutter 中,Drawer 是一个常用的侧边栏导航组件,通过点击菜单按钮或滑动屏幕显示。它用于展示导航项、用户信息和应用设置等。本文通过一个简单的示例代码,介绍了如何使用 Drawer 实现多页面导航,包括 Drawer 的基本结构、ListView 和 ListTile 的使用,以及页面内容的切换。希望对理解和使用 Flutter 的 Drawer 组件有所帮助。
332 1
Flutter 自定义组件继承与调用的高级使用方式
本文深入探讨了 Flutter 中自定义组件的高级使用方式,包括创建基本自定义组件、继承现有组件、使用 Mixins 和组合模式等。通过这些方法,您可以构建灵活、可重用且易于维护的 UI 组件,从而提升开发效率和代码质量。
384 1
|
前端开发 开发者
深入探索 Flutter 鸿蒙版的画笔使用与高级自定义动画
本文深入探讨了 Flutter 中的绘图功能,重点介绍了 CustomPainter 和 Canvas 的使用方法。通过示例代码,详细讲解了如何绘制自定义图形、设置 Paint 对象的属性以及实现高级自定义动画。内容涵盖基本绘图、动画基础、渐变动画和路径动画,帮助读者掌握 Flutter 绘图与动画的核心技巧。
270 1
|
Dart UED 开发者
Flutter&鸿蒙next中的按钮封装:自定义样式与交互
在Flutter应用开发中,按钮是用户界面的重要组成部分。Flutter提供了多种内置按钮组件,但有时这些样式无法满足特定设计需求。因此,封装一个自定义按钮组件变得尤为重要。自定义按钮组件可以确保应用中所有按钮的一致性、可维护性和可扩展性,同时提供更高的灵活性,支持自定义颜色、形状和点击事件。本文介绍了如何创建一个名为CustomButton的自定义按钮组件,并详细说明了其样式、形状、颜色和点击事件的处理方法。
267 1
|
Dart 搜索推荐 API
Flutter & 鸿蒙next版本:自定义对话框与表单验证的动态反馈与错误处理
在现代移动应用开发中,用户体验至关重要。本文探讨了如何在 Flutter 与鸿蒙操作系统(HarmonyOS)中创建自定义对话框,并结合表单验证实现动态反馈与错误处理,提升用户体验。通过自定义对话框和表单验证,开发者可以提供更加丰富和友好的交互体验,同时利用鸿蒙next版本拓展应用的受众范围。
268 1
|
前端开发
Flutter快速实现自定义折线图,支持数据改变过渡动画
Flutter快速实现自定义折线图,支持数据改变过渡动画
351 4
Flutter快速实现自定义折线图,支持数据改变过渡动画