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

Flutter简介

Flutter是谷歌的移动UI框架,可以快速在iOS和Android上构建高质量的原生用户界面。 Flutter可以与现有的代码一起工作。在全世界,Flutter正在被越来越多的开发者和组织使用,并且Flutter是完全免费、开源的。
它也是构建未来的Google Fuchsia 应用的主要方式。

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

实现

先上一张实现效果图

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高度

总结

以上所述是小编给大家介绍的Flutter 侧滑栏及城市选择UI的实现方法,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对我们网站的支持!
如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!

(0)

相关推荐

  • Android5.0多种侧滑栏效果实例代码

    1.普通侧滑 效果图: 思路:通过自定义View继承HorizontalScrollView,然后重写onMeasure(),onLayout(),onTouchEvent() 方法并设置menu(通过动画使menu开始时处于隐藏状态)布局和content布局.(注意:使用ViewHelper类需要导入nineoldandroids-2.4.0.jar包) menu(left_menu)布局代码: <?xml version="1.0" encoding="utf-8&

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

    Flutter简介 Flutter是谷歌的移动UI框架,可以快速在iOS和Android上构建高质量的原生用户界面. Flutter可以与现有的代码一起工作.在全世界,Flutter正在被越来越多的开发者和组织使用,并且Flutter是完全免费.开源的. 它也是构建未来的Google Fuchsia 应用的主要方式. 目前移动市场上很多业务都需要开发Android/IOS两个端,开发成本比较高. Flutter 在跨端上凭借着性能优势关注量,使用度也持续上升.今天给大家分享在去年就写的一个Flu

  • flutter 动手撸一个城市选择citypicker功能

    城市选择器在项目开发中一般都会用到,基于flutter版本的也有一个city_pickers但是已经很久没有人维护了,项目中之前也用的是这个,最近升级到flutter1.17.x后,发现有一定的概率闪退,无奈之下,只能自动动手撸一个了 demo下载地址:https://github.com/qqcc1388/city_picker CityPickerView能够实现以下功能 显示省市区地址,市或者区可以为空白数据 省市区数据支持自定义,但是格式要按照city.json中个格式来,如果需要外部传

  • js实现仿阿里巴巴城市选择框效果实例

    本文实例讲述了js实现仿阿里巴巴城市选择框效果.分享给大家供大家参考.具体分析如下: 这并不是一个城市选择插件,在这里介绍只是为了mark一下二级联动的方法,此效果适用于有二级子菜单的效果,如导航栏.城市选择.类别选择等等. 样式效果是基于阿里的样式,懒得做其他调整,在area.css中仅仅是为了修改浏览器兼容性略做了一点调整. 城市数据是通过js构造,当然也可以通过后端取得数据,不过感觉没必要. <!doctype html> <html> <head> <me

  • Mint UI实现A-Z字母排序的城市选择列表

    本文实例为大家分享了Mint Ul实现A-Z字母排序的城市选择列表的具体代码,供大家参考,具体内容如下 效果图如下: 项目文件存放路径图: pinying.js 和 city.json文件下载传送门 所有代码如下: <template> <mt-index-list> <mt-index-section v-for="letter in citySortArr" :key="letter" :index="letter&quo

  • Flutter实现自定义下拉选择框的示例详解

    在一些列表页面中,我们经常会有上方筛选项的的需求,点击出现一个下拉菜单,多选.单选.列表选等,而在Flutter中,并没有现成的这样的组件,找第三方的扩展有时候又会受到一定限制,所以最好我们可以自己做一个,这样即使扩展我们也会得心应手. 先看效果图: 关键点:弹出.收回动画.状态改变.选项联动 思路: 我们可以看到一个完整的下拉框有头部和具体的下拉选项两部分组成,头部又和下拉组进行了联动, 把头部当做1个数组,下方选项作为1个数组,两个数组数量一致之间形成一个完整的下拉选择框可以更好的控制联动效

  • Android自定义view实现侧滑栏详解

    目录 前言 需求 效果图 编写代码 主要问题 前言 上一篇文章学了下自定义View的onDraw函数及自定义属性,做出来的滚动选择控件还算不错,就是逻辑复杂了一些.这篇文章打算利用自定义view的知识,直接手撕一个安卓侧滑栏,涉及到自定义LayoutParams.带padding和margin的measure和layout.利用requestLayout实现动画效果等,有一定难度,但能重新学到很多知识! 需求 这里类似旧版QQ(我特别喜欢之前的侧滑栏),有两层页面,滑动不是最左侧才触发的,而是从

  • jQuery实现火车票买票城市选择切换功能

    效果图如下所示: <!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8"> <title></title> <style> select { width: 200px; background-color: teal; height: 200px; font-size: 20px; } .btn-box { width

  • JavaScript实现城市选择控件的效果

    在淘宝旅行上看到的城市选择效果,感觉还不错,就自己的理解重新实现一遍,先看效果(有人说IE9下面有BUG,LZ用的是落后的XP,居然装不上IE9,去公司在搞搞好了),然后再细说实现原理,支持鼠标上下键选择城市,支持直接输入城市名称,拼音首字母,全拼,支持IE6遮盖SELECT,压缩后12K. 实现的步骤: 一.先用一定的格式罗列出控件所需要的城市以及拼音等,我这里是按照如下格式罗列成一个数组, 如果需要增加城市,直接增加在数组里面即可: 城市我是一个一个手打的... ['北京|beijing|b

  • vue基于mint-ui实现城市选择三级联动

    项目是基于vue2 的移动端项目,供大家参考,具体内容如下 1.实际效果 地址三级联动 mint-ui picker.png 2.首先你需要去下载一个包含中国省份,城市,区县的数据 如下: (这个地址里面包含二级联动数据,三级联动数据,四级联动数据等,找到自己需要的) (一个更好的中国地区数据,推荐用这个) 3.具体代码 主要是用到了mint-ui的picker组件,关于mint-ui的使用就自行看官网 Ⅰ .html组件 <div> <mt-picker :slots="my

随机推荐