Flutter Dynamic Widget - Build your widget on the fly

本人在某电商公司搬砖,一般来说电商APP的首页都需要非常动态灵活的展现方式,例如:某天产品突发奇想为后天的xx大促活动设计了一个酷炫的UI展现方式,跟技术说如果按照我的这种展示交互来做的话,这个转化率肯定能提升100%。问我们什么时候能搞定,我们一评估发现这个需要发版才能搞定。后天的xx大促肯定是来不及了。产品会觉得我们技术太差,我们也在产品面前抬不起头。

电商app

对于这种问题,APP有React Native、Weex这种解决方案,但是我们调研RN后发现,RN有很多问题,而且性能方面也不是太好。好像现在越来越多的公司也已经抛弃RN了。所以我们就决定自己定义了一套UI组件协议,在Android、iOS,JS设置小程序端分别实现。这种方案前期工作量确实很大,但是当你的UI组件协议越来越丰富时,带来的好处也是极大的。APP中的很多页面我们都能做到随时更新。甚至是能做到随时新增一些页面。避免APP一旦修改都需要发版的问题。极大的提升了产品迭代速度。带来极大的收益。

Flutter

其实早在17年的时候,我们就已经有关注Flutter了,那时候还是叫Sky。我们在团队内分享后觉得还是处于非常早期的阶段,后面也没有再持续关注了,没想到Flutter在18年12月份放出了1.0正式版。Flutter天然的跨平台,支持Android,iOS。Web和桌面也在支持的计划中。可以说几乎跨了所有的平台。而且是基于同一份代码,注意这里跟RN还是有区别的,RN是Learn once, write everywhere.

在学习Flutter的过程中发现写Flutter的Widget跟我们公司定义的那套UI协议是非常相似的。我们定义的UI协议是基于json语法,而Flutter的Widget代码跟json几乎一致。


sample

这里可以把Widget的类型对应到json string的type属性。其他的属性基本保持一致,这里color的属性在json中使用#aarrggbb就行。看到这里心中就涌现一个想法,如果我用Flutter来实现一个DynamicWidget,这个Widget可以通过json来创建的话,这样我只要用Flutter实现一遍就能天然的跨多端。而且Flutter的Widget就是天然的UI协议呀,我只要把Flutter官方的Widget转成json然后实现一遍就行了。这样用户只要会了解了Flutter的Widget基本上就能写出对应的json了。用户不需要再理解一套特殊的UI协议。我们公司的UI协议,为了兼容Android、iOS各端走的是一套特殊的UI协议,需要使用者去学习。

Flutter Dynamic Widget

说干就干,先定义一个WidgetParser接口。

/// extends this class to make a Flutter widget parser.
abstract class WidgetParser{

  /// parse the json map into a flutter widget.
  Widget parse(Map<String, dynamic> map);

  /// check the matched widget type. for example:
  /// {"type" : "Text", "data" : "Denny"}
  /// if you want to make a flutter Text widget, you should implement this
  /// method as "Text" == widgetName, for more details, please see
  /// @TextWidgetParser
  bool forWidget(String widgetName);
}

parse方法解析json得到Flutter的Widget,forWidget方法表示这个Parser是对应json中对应的type属性。我们来实现一个Container widget的Parser, 如下:

class ContainerWidgetParser extends WidgetParser{
  @override
  bool forWidget(String widgetName) {
    return "Container" == widgetName;
  }

  @override
  Widget parse(Map<String, dynamic> map) {
    Alignment alignment = parseAlignment(map['alignment']);
    Color color = parseHexColor(map['color']);
    BoxConstraints constraints = parseBoxConstraints(map['constraints']);
    //TODO: decoration, foregroundDecoration and transform properties to be implemented.
    EdgeInsetsGeometry margin = parseEdgeInsetsGeometry(map['margin']);
    EdgeInsetsGeometry padding = parseEdgeInsetsGeometry(map['padding']);
    Map<String, dynamic> childMap =  map['child'];
    Widget child;
    for(var parser in DynamicWidgetBuilder.parsers){
      if (parser.forWidget(childMap['type'])){
        child = parser.parse(childMap);
        break;
      }
    }
    return Container(
      alignment: alignment,
      padding: padding,
      color: color,
      margin: margin,
      width: map['width'],
      height: map['height'],
      constraints: constraints,
      child: child,
    );
  }

}

你需要做的就是解析json的各个属性,然后最后拼装一个Container对象。很简单把。基于以上的想法,我把实现放在github上,大家可以在这里查看源码和各个实现的Widget的demo。同时也发布到dart pub上了。
github:https://github.com/dengyin2000/dynamic_widget
dart pub:https://pub.dartlang.org/packages/dynamic_widget

目前只是实现了几个Flutter Widget,计划是把Flutter官方的widget都实现一遍。大家如果有问题的话,欢迎随时联系我。

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

推荐阅读更多精彩内容

  • 特别说明 当前博客平台账号已废弃,如果有使用细节问题请前往我新博客平台进行讨论交流。 个人博客平台 HuRuWo的...
    善笃有余劫阅读 4,935评论 0 30
  • 国庆后面两天在家学习整理了一波flutter,基本把能撸过能看到的代码都过了一遍,此文篇幅较长,建议保存(star...
    Nealyang阅读 4,337评论 1 17
  • 目录 一、Flutter 为何使用Dart开发语言二、Flutter的UI系统1.特点2.架构简介2.1 Flut...
    十拿九稳啦阅读 3,626评论 3 28
  • 原文在此,此处只为学习 Widget与ElementWidget主要接口Stateless WidgetState...
    lltree阅读 4,499评论 0 1
  • 14/21 李雯 深圳【每日一结构】结构思考力21天思维改善训练营 序:近期二手房买卖纠纷案件呈现井喷状态。 G:...
    闪亮Living阅读 159评论 0 0