Flutter 侧滑栏及城市选择UI的实现

Flutter 侧滑栏及城市选择UI的实现

前言

  目前移动市场上很多业务都需要开发Android/IOS两个端,开发成本比较高. Flutter 在跨端上凭借着性能优势关注量,使用度也持续上升.今天给大家分享在去年就写的一个Flutter版本的侧滑栏.

实现

先上一张实现效果图

readMe_city.gif

SliderBar 实现

  侧边是一个支持手势滑动的SliderBar,一个自定义的StatefulWidget.可以观察到,当手势在侧边滑动时,中央显示选中的标签.

布局

  一个横向布局,里面放了一个元素。左边标签的容器尽量占满整个屏幕,右边固定宽度的一个列表(里面放需要展示的Label),代码如下:

new Row(
      mainAxisSize: MainAxisSize.min,
      crossAxisAlignment: CrossAxisAlignment.start,
      children: <Widget>[
        new Expanded(
            child: new Center(
                child: new Text(selectLabel,
                    style:
                        new TextStyle(color: Colors.orange, fontSize: 40.0)))),
        slide
      ],
    );

手势数据处理

   Flutter 提供 手势处理类 GestureDetector,当手势开始滑动是更新中央Label显示,停止或者取消时,取消Label显示并把对应的数据填充到Label上.

new GestureDetector(
      behavior: HitTestBehavior.translucent,
      child: slideWidget,
      onPanStart: (event) {
        updateLabel(context, event.globalPosition);
      },
      onPanDown: (event) {
        updateLabel(context, event.globalPosition);
      },
      onVerticalDragUpdate: (event) {
        updateLabel(context, event.globalPosition);
      },
      onPanCancel: () {
        setState(() {
          selectLabel = '';
        });
      },
      onVerticalDragEnd: (event) {
        setState(() {
          selectLabel = '';
        });
      },
    );

遇到的问题以及解决方法:

  • GestureDetector 监听的手势很多,当注册 onVerticalDragUpdate 后,onPanUpdate 不在回调,解决方法:将onPanUpdate逻辑全部移入onVerticalDragUpdate,
  • onPanUp 未监听到手势抬起,解决方法:换用onPanCancel,onVerticalDragEnd方法监听

updateLabel,获取具体选中Label的index 公式为 index = dy / widgetHeight * labelList.length,其中dy 为 以控件起始点y的位置偏移量,widgetHeight为高度,
labelList.length为Label的长度,刷新数据逻辑如下:

void updateLabel(BuildContext context, Offset globalPosition) {
    var object = globalKey?.currentContext?.findRenderObject();
    var translation = object?.getTransformTo(null)?.getTranslation();

    int index = ((globalPosition.dy - translation.y - topMargin) /
            (globalKey.currentContext.size.height - topMargin) *
            widget.showList.length)
        .toInt();
    if (index < widget.showList.length && index >= 0) {
      setState(() {
        selectLabel = widget.showList[index];
        if (widget.onChangeSelect != null) {
          widget.onChangeSelect(selectLabel);
        }
      });
    }
  }

其中,获取控件距离屏幕的距离方法为:

  var object = globalKey?.currentContext?.findRenderObject();
  var translation = object?.getTransformTo(null)?.getTranslation();

城市选择主界面实现

主布局

   采用了Flutter 的Stack布局(非常类似Android FrameLayout),下层是城市选择页面数据,上层盖了一层SliderBar

 new Scaffold(
        appBar: getAppBar(),
        body: new Stack(children: <Widget>[
          getShowContentView(),
          new SlideBar(
              cityListUtils.labelList, onChangeSelect)
        ]));

UI的下层 使用 ListView.builder 根据item类型返回不同类型的Widget

Widget rightCity = new Container(
       color: AppColor.white,
       padding: EdgeInsets.only(right: 20.0),
       child: new ListView.builder(
           controller: scrollController,
           itemCount: cityListUtils.cityList.length,
           itemBuilder: (listContext, position) {
             var city = cityListUtils.cityList[position];
             if (city is CityModel) {
               return new GestureDetector(
                   behavior: HitTestBehavior.translucent,
                   child: new Container(
                       decoration: new BoxDecoration(
                           border: new Border.all(
                               color: AppColor.bg1, width: 0.5)),
                       height: 48.0,
                       padding: EdgeInsets.only(left: 15.0),
                       alignment: Alignment.centerLeft,
                       child: new Text(city.name)),
                   onTap:selectCity(city));
             } else if (city is CityLabel) {
               return new Container(
                 width: MediaQuery.of(context).size.width,
                 height: 20.0,
                 padding: EdgeInsets.only(left: 15.0),
                 child: new Text(city.keyLabel),
                 color: AppColor.bg1,
               );
             }
           }));

城市列表数据处理

   城市列表的数据格式如下

{"A":[{"name":"澳门","id":"***","fullWord":"aomen","first":"am","isShow":"true"}]}

数据解析使用到dart:convert包,调用json.decode(jsonStr)解析的数据为map,在将Map转为具体的实体,实体解析工具推荐使用开源工具自动生成模型文件 FlutterJsonBeanFactory
得到城市实体的解析Model如下:

import 'dart:convert' show json;

class CityModel {
  String first;
  String fullWord;
  String id;
  String isShow;
  String name;
  bool isSelected = false;

  CityModel.fromParams(
      {this.first, this.fullWord, this.id, this.isShow, this.name});

  factory CityModel(jsonStr) => jsonStr is String
      ? CityModel.fromJson(json.decode(jsonStr))
      : CityModel.fromJson(jsonStr);

  CityModel.fromJson(jsonRes) {
    first = jsonRes['first'];
    fullWord = jsonRes['fullWord'];
    id = jsonRes['id'];
    isShow = jsonRes['isShow'];
    name = jsonRes['name'];
  }

  @override
  String toString() {
    return '{"first": ${first != null?'${json.encode(first)}':'null'},"fullWord": ${fullWord != null?'${json.encode(fullWord)}':'null'},"id": ${id != null?'${json.encode(id)}':'null'},"isShow": ${isShow != null?'${json.encode(isShow)}':'null'},"name": ${name != null?'${json.encode(name)}':'null'}}';
  }
}

将首字母,城市数据存入CityList里,并将首字母列表传入到SliderBar中,记录字母索引所在的位置

class CityListUtils {
  List cityList = [];
  List<String> labelList = [];
  Map<String, IndexPosition> mapKey = {};

  void parse(var map) {
    if (map is String) {
      map = json.decode(map);
    }
    Map mapList = map['destination'];
    int index = 0, labelPosition = 0;
    mapList.keys.forEach((key) {
      cityList.add(new CityLabel(key));
      labelList.add(key);
      mapKey[key] = new IndexPosition(labelPosition, index);
      labelPosition++;
      index++;
      for (var value in mapList[key]) {
        index++;
        cityList.add(new CityModel(value));
      }
      ;
    });
  }
}

联动处理

当滑动SliderBar时,应将城市列表滑到对应的位置,ListView 提供 ScrollController 去为ListView 添加监听及 Auto scroll ListView,
里面对应的有两个方法可以滑动,一个是带有动画 animateTo,一个不带有动画的滑动 jumpTo,此处使用不带有的方法,传递参数为
滑动的偏移量,实现如下

  OnChangeSelect onChangeSelect = (keyLabel) {
        IndexPosition index = cityListUtils.mapKey[keyLabel];
        scrollController.jumpTo(index.total * 48.0 - index.label * 28.0);
      };

其中 OnChangeSelect定义为

typedef OnChangeSelect(String keyLabel);

使用接口回调的方式将选中的key回传,并使用CityListUtils里存储的mapKey找到对应的首字母索引,计算出ListView应该滑动的偏移量

遇到的问题

计算的偏移量不准,导致滑动不能准确定位到首字母索引上。
原因:item 使用 Container布局 高度未限制,手动获取到的高度不准确
解决方法:使用固定的item高度

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

推荐阅读更多精彩内容