Flutter 自定义特殊形状的tabbar

先看效果!
自定义形状TabBar
实现

成员变量
List<Widget> pages = List<Widget>();用来存放tabs分别对应的页面。
tabbar作为内容的容器 CustomerNotification用来容器内部Widget来通知tabbar做出改变。

NotificationListener的’ child‘是页面显示内容,NotificationListener用来监听CustomerNotification。

看设计,TabBar是悬浮的设计,不能用官方库提供的TabBar。TabBar和后面的内容是堆叠的,所以用Stack widget。
Stack: 取代线性布局,Stack允许子 widget 堆叠, 你可以使用 Positioned 来定位他们相对于Stack的上下左右四条边的位置。Stacks是基于Web开发中的绝对定位(absolute positioning )布局模型设计的。
Positioned,用Positioned绝对定位悬浮的TabBar。 bottom: 15, left: 20, right: 20(top这里不需要设置)分别设计和容器的边距。

Positioned的child就是你自己需要定制的tabbar形状。

全部代码如下:

/*
 * File: /Users/ottpay/Git/fyh_transfer/lib/Account/accountHomePage.dart
 * Created Date: 2019-09-27 4:13:15
 * Author: 木穆  pengxuan.li@ott.ca
 * Copyright (c) 2019 OTT Pay HK
 */



import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart';
import 'package:fyh_transfer/Send/sendHomePage.dart';
import 'package:fyh_transfer/Order/orderHomePage.dart';
import 'package:fyh_transfer/Account/accountHomePage.dart';

class FYHTabbar extends StatefulWidget {
  @override
  FYHTabbarState createState() => FYHTabbarState();
}

class CustomerNotification extends Notification {
  CustomerNotification(this.enable);
  final bool enable;
}

class FYHTabbarState extends State<FYHTabbar>
     {
  List<Widget> pages = List<Widget>();

  bool sendeEnable = false;
  SendHomePage _sendHomePage = SendHomePage();

  int _activeIndex = 1; //激活项
  double _height = 48.0; //导航栏高度
  double _floatRadius = 30; //悬浮图标半径

  List _navs = [
    Icons.search,
    Icons.ondemand_video,
    Icons.music_video,
  ]; //导航项

  @override
  void initState() {
    super.initState();
    pages..add(OrderHomePage())..add(_sendHomePage)..add(AccountHomePage());
  }

  @override
  Widget build(BuildContext context) {
    double width = MediaQuery.of(context).size.width - 40;
    _floatRadius = _activeIndex == 1 ? 40 : 30;
    return NotificationListener<CustomerNotification>(
        onNotification: (notification) {
          setState(() {
            sendeEnable = notification.enable;
          });
        },
        child: Container(
          child: Stack(children: [
            pages[_activeIndex],//tabbar要存放的页面
            Positioned(//用Positioned绝对定位悬浮的TabBar
              bottom: 15,
              left: 20,
              right: 20,
              child: Container(
                padding: EdgeInsets.fromLTRB(0, 10, 0, 0),
                width: width,
                height: 58,
                child: Stack(
                  overflow: Overflow.visible,
                  children: <Widget>[
                    //所有图标
                    DecoratedBox(
                      decoration: BoxDecoration(
                        color: Color(0xFFf2f4f5), // 底色
                        shape: BoxShape.rectangle, // 默认值也是矩形
                        borderRadius:
                            new BorderRadius.circular(_height / 2), // 圆角度
                        boxShadow: [
                          BoxShadow(
                              color: Color(0xffc1c1c1),
                              offset: Offset(2.0, 2.0),
                              blurRadius: 50.0,
                              spreadRadius: 2.0)
                        ],
                      ),
                      child: SizedBox(
                        height: _height,
                        child: Row(
                          mainAxisAlignment: MainAxisAlignment.spaceAround,
                          crossAxisAlignment: CrossAxisAlignment.center,
                          children: _navs
                              .asMap()
                              .map((i, v) => MapEntry(
                                  i,
                                  GestureDetector(
                                    child: Container(
                                      color: Color(0xFFf2f4f5),
                                      margin: EdgeInsets.all(0),
                                      child: Icon(v,
                                          color: _activeIndex == i
                                              ? Colors.pink
                                              : Colors.grey),
                                      width: width / 3 - 30,
                                      height: _height - 10,
                                    ),
                                    onTap: () {
                                      _switchNav(i);
                                    },
                                  )))
                              .values
                              .toList(),
                        ),
                      ),
                    ),
                    //浮动图标
                    Positioned(
                      bottom: 4,
                      left: (width) / 2 - _floatRadius,
                      child: DecoratedBox(
                        decoration:
                            ShapeDecoration(shape: CircleBorder(), shadows: [
                          BoxShadow(
                              blurRadius: 2,
                              offset: Offset(0, 2),
                              spreadRadius: 0,
                              color: _activeIndex == 1
                                  ? Colors.pink
                                  : Colors.grey),
                        ]),
                        child: GestureDetector(
                          child: CircleAvatar(
                              radius: _floatRadius, //浮动图标和圆弧之间设置8pixel间隙
                              backgroundColor: Color(0xfff2f4f5),
                              child: Icon(_navs[1],
                                  color: sendeEnable && _activeIndex == 1
                                      ? Colors.pink
                                      : Colors.black38)),
                          onTap: () {
                            _clickMiddleBtn();
                          },
                        ),
                      ),
                    ),
                  ],
                ),
              ),
            )
          ]),
        ));
    ;
  }

  //中间按钮
  _clickMiddleBtn() {
    if (_activeIndex == 1) {
      if (sendeEnable) {
        print('send');
      }
    } else {
      _switchNav(1);
    }
  }

  //切换导航
  _switchNav(int newIndex) {
    print(newIndex);
    double oldPosition = _activeIndex.toDouble();
    double newPosition = newIndex.toDouble();
    if (oldPosition != newPosition) {
      setState(() {
        _activeIndex = newIndex;
      });
    }
  }

  @override
  void dispose() {
    super.dispose();
  }
}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,029评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,238评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,576评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,214评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,324评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,392评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,416评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,196评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,631评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,919评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,090评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,767评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,410评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,090评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,328评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,952评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,979评论 2 351

推荐阅读更多精彩内容