flutter创建省市区三级联动

话不多说先上效果图。
IMG_1179.PNG

这是一个从底部弹起的BottomSheet,选择完成确定后会回调所选择的内容。

下面直接上封装好的底部弹出的省市区视图类
import 'package:flutter/material.dart';
import 'package:my_flutter/areajsonlist.dart';
class ShaiXuanSelectWidget extends StatefulWidget {
  final List PinpaiListData;
  ShaiXuanSelectWidget(
      {
         @required this.PinpaiListData,
    });
  @override
  ShaiXuanSelectWidgetState createState() => ShaiXuanSelectWidgetState();
}

class ShaiXuanSelectWidgetState extends State<ShaiXuanSelectWidget> {
  //品牌字典组
  List _pinpaiGroup = [];
  // 当前选中的品牌
  int _nowClickPinpai;

  // 分类选择组
  List _typeGroup = [];
  int _nowClickFenlei;
  // 系列选择组
  List _xilieGroup = [];
  int _nowClickXilie;
  // 当前选择系列的数据
  // List _nowChooseXilieGroup = [];
String _nowSelectedXilieSte = "";
  bool _isFirst = true;

  @override
  void initState() {
    super.initState();
    if (_isFirst) {
      initData();
      _isFirst = false;
    }
  }
  @override
  Widget build(BuildContext context) {

   return StatefulBuilder(
      builder: (BuildContext context, StateSetter modalSetState) {
        return Container(
          child: Column(
            children: <Widget>[
              Expanded(
                child: buildDeviceContainer(modalSetState),
              ),
              Expanded(
                flex: 0,
                child: buildBottomButton(modalSetState),
              ),
            ],
          ),
        );
      },
    );
  }

  // 初始化数据
  void initData() {
    _pinpaiGroup = [];
    // whilePinpai(widget.PinpaiListData);
    whileBig(jsonlistclass.jsonlist);
    setState(() {});
  }
  void whileBig(data) {
    for (int i = 0; i < data.length; i++) {
      if (null != data[i]) {
        _pinpaiGroup
            .add({'name': data[i]["name"], 'city': data[i]["city"]});
      }
    }
  }
  // 格式化品牌组
  void whilePinpai(data) {
    for (int i = 0; i < data.length; i++) {
      if (null != data[i]) {
        _pinpaiGroup
            .add({'name': data[i].name, 'value': data[i].cid});
      }
    }
  }

  // 选择品牌
  buildDeviceContainer(StateSetter modalSetState) {
    return Row(
      children: <Widget>[
    Expanded(
    child: Column(
        children: <Widget>[
          Container(
            height: 20,
            child: Text('品牌',style: TextStyle(
              fontSize: 16,
            ),),
          ),
          Expanded(
            child: selectPinpaiItem(modalSetState),
          ),
        ])),

        Expanded(
          child: Column(
            children: <Widget>[
              Container(
                height: 20,
                child: Text('分类', style: TextStyle(
                  fontSize: 16,
                ),),
              ),
              Expanded(
                child: selectTypeItem(modalSetState),
              ),
            ],
          ),
        ),
        Expanded(
          child: Column(
            children: <Widget>[
              Container(
                      height: 20,
                      child: Text('系列',style: TextStyle(
                        fontSize: 16,
                      ),),
                    ),
              Expanded(
                child: selectXILieItem(modalSetState),
              ),
            ],
          ),
        ),
      ],
    );
  }

  // 底部操作按钮
  buildBottomButton(StateSetter modalSetState) {
    return Row(
        mainAxisAlignment: MainAxisAlignment.spaceEvenly,
        children: <Widget>[
          RaisedButton(
            child: Text('取消'),
            onPressed: () {
              modalSetState(() {
                _nowClickPinpai = null;
                _nowClickFenlei=null;
                _nowClickFenlei=null;
                _typeGroup = [];
                _xilieGroup = [];
                // _nowChooseXilieGroup = [];
                _nowSelectedXilieSte = "";
              });
            },
          ),
          RaisedButton(
            padding: EdgeInsets.all(0),
            color: Color.fromRGBO(
                253, 117, 80, 1),
            child: Text('确定',  textAlign: TextAlign.center,
              style: TextStyle(
                color: Colors.white,
                fontSize: 14,
              ),),
            onPressed: () {
              Navigator.pop(context, _nowSelectedXilieSte);
            },
          ),
        ],

    );
  }

  // 品牌列表
  selectPinpaiItem(StateSetter modalSetState) {
    return ListView.separated(
      itemBuilder: (BuildContext context, int index) {
        return Container(
          height: 40,
          child: GestureDetector(
            child: ListTile(
              title: Text(
                  _pinpaiGroup.length > 0 ? _pinpaiGroup[index]['name'] ?? ' ' : ' ',
                  maxLines: 1,
                  style: _nowClickPinpai == index
                      ? TextStyle(color:Color.fromRGBO(
                      253, 117, 80, 1), fontSize: 13,)
                      : TextStyle(fontSize: 13,color:Colors.black45)),
            ),
            onTap: () {
              if (_nowClickPinpai != index) {
              setState(() {
                _nowClickPinpai = index;
                whileType(_pinpaiGroup[index]);
              });
              }
            },
          ),
        );
      },
      separatorBuilder: (BuildContext context, int index) {
        return Divider(
          height: 0.0,
        );
      },
      itemCount: _pinpaiGroup.length,
    );

  }

  // 分类列表
  selectTypeItem(StateSetter modalSetState) {
    return ListView.separated(
      itemBuilder: (BuildContext context, int index) {
        return Container(
          height: 40,
          child: GestureDetector(
            child: ListTile(
              title: Text(
                  _typeGroup.length > 0 ? _typeGroup[index]['name'] ?? ' ' : ' ',
                  style: _nowClickFenlei==index
                      ? TextStyle(color: Color.fromRGBO(
                      253, 117, 80, 1), fontSize: 13,)
                      : TextStyle(fontSize: 13,color:Colors.black45)),
            ),
            onTap: () {
              if (_nowClickFenlei != index) {
                setState(() {
                  _nowClickFenlei = index;
                  whileXilie(_typeGroup[index]);
                });
              }
            },
          ),
        );
      },
      separatorBuilder: (BuildContext context, int index) {
        return Divider(
          height: 0.0,
        );
      },
      itemCount: _typeGroup.length,
    );
  }

  // 系列列表
  selectXILieItem(StateSetter modalSetState) {
    return ListView.separated(
      itemBuilder: (BuildContext context, int index) {
        return Container(
          height: 40,
          child: GestureDetector(
            child: ListTile(
              title: Text(
                _xilieGroup.length > 0 ? _xilieGroup[index]['name'] ?? ' ' : ' ',
                style: _nowClickXilie==index
                    ? TextStyle(color: Color.fromRGBO(
                    253, 117, 80, 1), fontSize: 13,)
                    : TextStyle(fontSize: 13,color:Colors.black45)),
              ),
            onTap: () {
              if (_nowClickXilie != index) {
                setState(() {
                  _nowClickXilie  = index;
                  _nowSelectedXilieSte =_xilieGroup[index]['name'];
                });
              }
            },
          ),
        );
      },
      separatorBuilder: (BuildContext context, int index) {
        return Divider(
          height: 0.0,
        );
      },
      itemCount: _xilieGroup.length,
    );
  }

// 获取分类名称
  whileType(dataMap) {
    _typeGroup=[];
    for (int i = 0; i < dataMap['city'].length; i++) {
      if (null != dataMap['city'][i]) {
          _typeGroup.add(
              {'name': dataMap['city'][i]['name'],'area': dataMap['city'][i]['area'], 'check': false});
      }
    }
  }

 // 获取系列
  whileXilie(areaMap) {
    _xilieGroup=[];
    for (int i = 0; i < areaMap['area'].length; i++) {
      if (null != areaMap['area'][i]) {
        _xilieGroup.add(
            {'name': areaMap['area'][i], 'check': false});
      }
    }
  }
}
用法:在需要的地方,自由调用即可
  MakeShaiXuan(BuildContext context) {
    showModalBottomSheet(
      context: context,
        builder: (context) => Container(
          child:ShaiXuanSelectWidget(PinpaiListData: Global.GlobalPinPaiItemsArrY),
          height: 600,
        ),
    ).then((val) {
      print(val);
      Soutchstr = val;
       _pageNo = 1;
       List _recodemodelitems = [];
        this._recodemodelitems.clear();
       RequestRecodeData(true);
    });
  }
结语:最后附上省市区的json地址:https://blog.csdn.net/youshi520000/article/details/70808580/
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 162,710评论 4 376
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 68,839评论 2 308
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 112,295评论 0 255
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,776评论 0 223
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 53,198评论 3 297
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 41,074评论 1 226
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 32,200评论 2 322
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,986评论 0 214
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,733评论 1 250
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,877评论 2 254
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,348评论 1 265
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,675评论 3 265
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,393评论 3 246
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,209评论 0 9
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,996评论 0 201
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 36,212评论 2 287
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 36,003评论 2 280

推荐阅读更多精彩内容