使用Flutter开发一个github用户端
Gitme 是Flutter中文网https://flutterchina.club/ 开发的一款github用户端,本文和大家分享一下我们用flutter从开始设计Gitme到动手开发,再到最后上线的整个过程中的少量思考、经验、以及趟过的坑。在阅读本文前,您可以先去我们的官网安装一下Gitme ,而后再比照本文中提到的点,才会有一个清晰的认识。
首先我们先来看几张gime软件截图:
开屏页
首页
issue页
user.jpg
目标
我们的目标是使用flutter做一个高性能的,同时支持Android和iOS的github用户端。但是,Github资源、功能比较多,并非所有功能我们都要在APP支持,在支持计划中的功能也必需划出优先级,首个版本应具有少量核心功能,少量优先级不高的功能随着日后版本迭代一点一点来完善。经过整理、探讨,我们列出了1.0中要支持的功能清单:
支持github账号登录、注销。
登录后使用户可以查看自己项目、动态等信息;支持编辑个人信息。
搜索;1.0支持搜索项目、使用户、issue;支持github搜索语法。
项目:支持对项目进行star/unstar、watch/unwatch,可以查看项目issue列表、升级动态、分支源码等信息。
使用户:支持查看使用户介绍;支持follow/unfollow使用户;假如使用户公开了邮箱,支持给使用户发送email。
Issue: 支持浏览、打开、关闭、编辑、评论issue;支持给issue增加label。
Label: 支持浏览、创立、删除label; 支持通过label挑选issue。
书签:关注内容可以加入书签收藏,以便下次可以快速打开。
国际化:支持中文简体与美国英语。
个性化:提供多套APP主题;提供深、浅两种代码主题。
技术点分析
确定目标后,就要对功能可能使用到的技术做一个分析整理,确定出哪些可以在flutter中完成,哪些需要插件。
UI
因为我们用的是flutter, 那么UI自然是在flutter来实现,主要熟习一下Flutter常使用widget.
数据与内容
github中绝大多数内容是源代码文件及markdown文本,还有少量就是图片等其它元数据。
对于源代码文件,需要渲染为等宽字体,并且排版时不能强制换行。
对于markdown文本(主要是issue、评论、文档),这是大多数使用户主要浏览的内容。为此必需有一个markdown解析器,这假如是在web端,没什么好担心的,成熟的轮子很多,但在flutter社区,情况却不容乐观,在pub仓库找到了一个flutter_markdown的包,经过测试发现坑很多,主要体现在markdown语法支持不足、样式自己设置困难、不支持tabel、不能自动识别url等,离可使用相差甚远。
对于github中的图片,主要是一般的图片(项目中的图片文件和网站的使用户头像等)和github的私有emoji。这里主要关注一下github emoji,它们有些特别,由于这些emoji在文本中只是少量标记,所以在渲染之前必需对文档进行解析,提取出emoji标记,而后转化为对应的图片,最后再进行渲染。而emoji会出现在很多地方,比方markdown中,所以这在解析markdown时也是应该考虑的点。
网络请求
Github API是开放的,v3是restful风格的,v4是graphQL风格。我们最终选择了v3版本,由于graphQL尽管灵活,可以做到按需取数据,绿色无白费,但在我们进行选型时,有两个因素让我们不得不放弃:
需要用户端开发者自己去汇总所需数据而后写出请求体;这非常耗时,刚开始时,我们根据github的API文档,在汇总时效率极低,一个小时才能完整的请求出两个业务接口。
返回数据嵌套层次太深;这让我们在将json数据转化成dart类(相似于java bean)时非常为难,假如把返回数据当成json数据,在开发时便不能取得ide的提醒会降低开发效率;在编译时会牺牲掉静态类型检查会添加潜在出错可能性(比方字段名输错了)。
确定选使用v3版本的api后我们需要一个合适的http库,我们希望http库具有:
良好的restful接口
请求响应阻拦器;这很重要,这意味着我们可以在底层统一对请求/响应进行预解决。
灵活的请求配置;比方可以统一配置请求基地址、公共header等,还有就是github 很多API在请求时都会涉及私有的
content-type
, 这意味着不同的请求可能需要不同的请求配置。支持超时; 因为重所周知的起因,在国内访问github时,有时可能需要较长的响应时间(有时甚至无法访问),所以支持超时是非常重要的。
最好支持请求取消;主要还是由于众所周知的起因,导致有时页面加载过慢,当使用户没有耐心继续等待下去返回时,能够将之前请求取消,避免在后端占使用资源、白费流量。
当然,一个优秀的Http库可能还包括cookie管理、文件下载/上传等功能,但是这两个功能在我们的需求场景中暂未使用到,所以就根据这5个指标去挑选。当时经过一圈查找,发现dart社区竟无一个同时满足这五点的(甚至同时满足前四点的也没有),这也是flutter社区刚起步生态还不好的尴尬,多希望有一个dart版的okhttp! 在这种时候,我一般都会找一个满足需求最高的开源项目,fork下来,而后定制。但是看了少量库的源码,发现实在是和需求相差较大,设计思路也相差太远,发现该轮子的成本已经大于从头造轮子的成本,没办法,历史上很多时候,就需要有少量人能够敢为人先,挺身而出,而后留下惊才绝艳的一笔…. 于是也便有了dio:
Dio is a powerful Http client for Dart, which supports Interceptors, FormData, Request Cancellation, File Downloading, Timeout etc.
值得一提的是,dio是flutter中文网开源项目之一,它主要借鉴了okhttp、axios、request、fly 四个开源库,所以无论是android开发者、还是前台、node开发者,相信都能很快上手dio。 目前dio在pub上得分是96分,github dart语言下项目排名22(正在快速上升中),在此,强烈向你推荐dio。
插件
Flutter的优势是在开发UI上,但因为Flutter用自绘引擎,并不能无缝集成原生控件(Android 原生控件及iOS UIToolkit), 而原生控件有一个比较大的优势就是可以集成系统能力,比方可以调使用相机(如surfaceView)、支持浏览网页(如webview),但在flutter中,因为绘制引擎skia只支持二维图形绘制,并不能直接结合原生功能,所以当我们要到这类原生相关的控件时,我们只能通过flutter插件来调使用原生控件来实现,在gitme中,主要涉及的是如何打开github文档、issue、评论里的url链接内容。
Webview
现在我们需要一个webview控件,能在应使用内显示h5网页,而要实现这个,我们只能通过flutter插件!
很多人问过我flutter中有没有相似于webview这样的widget,答案是现在没有,将来极大可能也不会有,起因很简单,假如在flutter中加一个webkit和v8你觉得flutter应使用的安装包有多大?
好了,现在看看有没有现成的轮子,pub中搜到的flutter_webview倒不少,但大多数都不能直接来使用,起因有两个:
我们需要对webview所在路由(android中的activity, iOS中的controller)的导航头进行少量自己设置,比方当页内跳转过多时给导航栏右侧加一个直接关闭当前路由的button以避免要连续屡次点击返回才能退出。
我们需要webview支持一套javascript bridge协议,已备日后方便集成h5功能。
但目前没有同时满足这两点的插件,所以,我们的webview插件还是得自己来写,最终我们通过:
Android: Webview + DSBridge-Andriod
iOS:WKWebview + DSBridge-IOS
实现了自己的webview插件。
其它插件
我们还使用到了fluttertoast 和 shared_preferences 插件,前者使用于少量需要提醒的场景来弹toast, 后者主要使用于应使用配置持久化。
设计模式及架构
遵循合适的设计模式,会让我们的代码逻辑清晰且易维护,一般来说不同端上都会有一套成熟的设计模式,如iOS上的mvc、android上的mvp、前台的mvvm等,那么我们的flutter代码中应该遵循怎么的设计模式?要答复这个问题,我们得先看一下flutter官方给出的编程范式(Flutter框架编程范式)以及google团队创造flutter时的灵感起源React-Native。
React-Native和flutter
RN最大的特点就是状态驱动的响应式编程,简而言之就是应使用程序维护一套状态(state),并提供一个UI模板,而模板可以绑定状态,而后当状态发生改变时框架根据状态的变化重新构建UI界面。可见,而整个过程中使用户不会直接操作UI控件树,构建过程(包括底层优化逻辑,几React中的diff算法)由框架完成。
在Flutter中,和RN非常类似,使用户可以创立有状态(stateful)和无状态(stateless) 的widget。 而后在build
方法中公告UI模板,当状态改变时,通过setState
方法通知flutter, flutter会在下一个frame中调使用使用户提供的build
方法来重建UI, 而底层的优化,如比照状态升级前后widget树的变化,只渲染变化部分的最小集,这些工作由flutter框架来完成,正如RN中的diff算法也是由框架完成一样。
所以很显著, Flutter是一个响应式框架,不记得mvxx这一套吧,假如你非要在flutter中套使用mvxx这一套设计模式,很可能就会变成过度设计。
Dart语言范式
Dart语言最主要的特点就是结合了编译性语言与脚本语言之所长,特点很多,在实际动手之前,我比较关注它最受诟病的一点:在flutter中,对于复杂一点的UI,嵌套层次太深!
这一点的确无法反驳,过多的嵌套的确让代码看起来很难维护,尤其是web前台开发者,早就受够javaScript “回调地狱 ”(callback hell)之苦,没想到现在到了flutter还是逃不掉。但其实,问题并没有那么糟糕,flutter中的嵌套和javascript中的回调嵌套是不同的,javascript中的回调嵌套一般是异步任务的回调,需要在回调中解决之前回调的逻辑, 而flutter中的嵌套一般来说并不是回调,而是UI widget的公告结构,它不需要再回调中再解决逻辑,所以,flutter中也就是嵌套层次深少量,但不会发生解决逻辑混乱。目前比较好的建议就是对于复杂的ui,最好将各个部分拆分成单独函数。
架构
其实flutter本身就是响应式的框架,我们只要遵循响应式编程的规范就行,但在程序逻辑结构上,我们也要多考虑一下。因为gitme主要是通过网络从github获取数据,而后再渲染UI. 我们可以在逻辑上对业务代码简单分成两层:底层数据IO+上层UI渲染,
数据层
关于数据请求的配置、逻辑等不要在UI层去控制,而由数据层自己完成。这也就是为什么我们队http库的要求中肯定要包含“支持请求/响应阻拦器”,由于只有支持阻拦器,我们才能将io逻辑更好分离。
UI渲染层
UI层我们主要用的事是material组件库,但我们并没有直接用 Scaffold
、 AppBar
这些基本每个页面都要使用的组件,而是在其上包装了一层,目的是程序风格发生变化时,我们只要要在包装组件中统一修改就可所有页面生效,而避免全局去替换(也许你会说可以设置主题,但是主题的精细粒度是不够的,有些需要自己设置的点主题并不支持)。除此之外,我们也封装了少量通使用的自己设置组件,如支持上拉加载、下拉刷新的无限列表。
编码
在想清楚上述问题后,我们对我们APP整体也就有了一个轮廓。接下来就是去逐一处理这些技术点就可。
UI布局
布局主要涉及Flutter中widget的用,这一步可以结合google官方 Gallery 中的示例先摸索,等自己动手写上几个页面后,布局就会轻松很多,flutter组件非常多,但常使用的也很固定。flutter sdk中的注释很详细,示例都在注释里(Flutter文档就是通过注释生成的), 在IDE中可以非常方便的跳转查看源码。总之,理解Flutter widget的第一资料就是源码。
Markdown支持
dart官方有一个markdown包,它可以将markdown文本解析成html。但是我们需要的是将markdown文本直接转化成flutter widget树,所以这个包是不能直接使用的,但是,假如我们要自己实现一个markdown到flutter的解析器,也并非易事。于是,我们想到了markdown包,看是否把它将markdown语法转化为html这一步替换为从markdown到flutter的widget,顺着这个思路,我们实现了最终的markdown解析器,并且工作良好。但是有一个问题就是:markdown包只支持纯粹markdown语法解析,假如在markdown文本中嵌入html代码,html代码是不支持的,所以现在我们的markdown解析器只支持markdown语法,对内嵌html代码不支持。这个我们希望markdown包作者能在后续版本中支持内嵌html语法,或者者等我这边腾出手再去给它提pr。
Emoji支持
Emoji支持是在markdown解析过程中完成的,将对应的emoji标记符先转换成markdown语法,而后再解析markdown。
Mock与缓存
因为gitme中用的网络库是dio, 而dio的开发与迭代基本与gitme是同时的,我们也花了不少的时间在dio库的迭代上。
Mock
在开发测试时,我们测试数据放在了一个git项目中,让后push到github,App访问git数据时就从github上的测试项目拉取,但是有一个问题就是每次打开页面时都要等待几秒,直到数据获取完成,这极大的影响了我们的开发效率。为理解决这个问题,我们在dio请求阻拦器中做了一层mock: 假如请求的是测试项目的数据,我们直接将本地工程对应的数据返回。这样一来有两个好处:
需要增加、改动测试数据时无需push到github远程仓库,本地该了就立即生效。
节省了网络请求时间。
缓存
因为github在墙外,国内访问有时可能会在速度和稳固性上存在少量问题,为了提高使用户体验,我们需要一个正当的缓存策略。一般来说,http协议有一套完整的策略,需要服务器与用户端配合(通过header来传递缓存策略信息),但是我们调使用的是github的接口,所以服务器对于我们来说是不可控的,所以我们不能用http协议本身的缓存策略,这的确比较遗憾,但是现在我们又有了一种新的思路,这还是多亏dio支持阻拦器,这让我们也可以在请求前/后来定制我们的缓存策略,值得一提的是,1.0中还没有加入缓存功能,这在我们后续版本迭代时会被支持。
链接阻拦
假如在markdown中点击url链接时,会进行统一的预解决,比方:检查假如是github链接的话,将其转换为App内路由,这样即可以在APP内打开,避免跳到网页中去,假如是邮箱地址,则调使用系统邮箱APP打开。
全局事件总线
gitme中有些场景需要全局状态共享,这和react中的redux或者vue中的vux很类似,不过gitme中需要共享的状态并不多,所以我们采使用了事件总线的方式来同步状态。
插件
正如上文所说,我们需要实现一个支持一种javascript bridge协议的webview插件,这个需要会原生开发,本身难度不大,就是gitme中实现了状态栏自动变色功能,会根据背景颜色自动调整前景文字、图标颜色,这使得我们的webview插件样式比较智能,并且非常容易自己设置主题。同时也实现了几个API,以供javascript调使用。
我们实现的另一个插件是版本升级插件,在其中我们也集成了mta统计sdk.
修轮子
在gitme中引入了少量第三方包,而其中近乎一半的第三方包无法直接用,对于这些包,我们的做法是fork其源码,而后修复、定制,而后在gitme中依赖我们fork的repo(flutter支持直接依赖git项目)。在开发gitme的过程中,我们深深的体会到了生态的重要性。
总结
在1.0开发完成后,首先根据之前设定的目标,check一下完成度, 而后在谈谈开发过程中躺过的坑。
目标完成度
1.0的目标基本都已完成,但仍有几个已知问题:
不支持markdown中嵌套的html代码。
代码染色能力不足。
对于第一个问题,上文已经谈过了,待日后优化。而代码染色问题比较辣手,这主要是由于编程语言种类繁多,而靠谱的染色方式都是需要通过将代码转化为笼统语法树(AST,Abstract Syntax Tree),而后再进行关键字、方法名、类名等提取,而后应使用不同样式渲染。假如是在web端,直接引入highlight.js,但dart中目前并没有这样的库,为此我们自己实现了一个简单的分析器,我们主要测试了Dart、Javascript、Java、php四种语言的成功率,gitme 1.0.0 结果如下:
语言 | 成功率 |
---|---|
Dart | > 95% |
Javascript | > 90% |
Java | > 90% |
php | 50% |
其它语言在1.0.0中染色成功率可能会非常低,因为良好的代码染色对gitme的使用户体验非常重要,因而,我们的下个版本主要的任务就是优化代码染色,根据目前1.0.1的开发进度,我们的分析器已经足够强大,就目前的测试结果,已经支持绝大多数编程语言,并且染色成功率都在90%以上,当然,在1.0.1上线前,我们还要进行更加全面的测试,最终的结果,敬请期待!
趟过的坑
严格来说,从一开始到现在遇到的问题是挺多的,但其中大部分是因为刚接触flutter,不太熟习,并不能说是坑,如各种widget的用等。下面列出几个在gitme开发过程中让我们花费了较多时间的问题:
不要将build函数中传入的
context
保存为全局变量(可能是为了后续用方便),build中传入的context会变,并且widget树不同部分构建时的context都不同,假如用保存的全局context,将会出现不可预期的错误。比方无法通过context正常获取local及主题信息(偶现);不要将需要缓存的数据保存在widget中。
因为Flutter响应式机制,每次状态变化都会重新build widget树,一般来说应该将需要缓存的数据保存在state中,因为widget和state生命周期不同,大多数情况下重新build时,state是复使用的,但是发现在TabView中切换tab时,每次tab都会完全重建(包括state), 这时缓存的数据就不能放在state中,有种做法是可以将数据保存在widget中,应为widget都是你在build方法中手动创立的,只需在创立时缓存一下widget(而不是每次build都重新
new
一个widget),这样只需widget不重建,即可以保证保存在widget中的数据不销毁,但我告诉你,千万不要这么做,由于你缓存widget的组件本身也是可能被重建的,这样就会导致你缓存的widget还是会被重建(原来保存的数据就销毁了); 假如你非要这么做,那么久必需保证从你缓存widget的组件开始到widget树根之间的所有widget都得被缓存,否则,一旦flutter调使用根widget的build方法,那么整个widget树都会被重新构建,之前缓存的数据也就自然不复存在了。正确的做法是放在全局状态管理器(如redux)或者全局变量中。ListView
结合RefreshIndicator
实现下拉刷新时, 列表项假如不满一屏,下拉刷新无效,此时需要将ListView
的primary
属性设置true
,但设置后就不能给ListView
设置controller,这是由于primary
属性设置为true
的ListView
会从他父辈widget中的PrimaryScrollController
获取它的controller(每个Scaffold
都会默认设置一个PrimaryScrollController
) 所以此时再设置controller时,flutter会报错,处理办法是自己手动设置一个PrimaryScrollController
。当自己设置导航栏(
AppBar
)的返回按钮时,iOS下右滑关闭手势会失效。这和iOS原生导航栏自己设置返回按钮会导致右滑手势失效是一样的。Android和iOS系统支持的字体不一样,不要以为flutter会自己用一套标准字体,flutter在绘制时也会用系统字体,所以在Text widget指定字体时肯定要看看能否两个平台都支持,gitme中在设置代码的等宽字体时发现了这个问题。
在替换图片、资源后或者构建release包之前要先执行
flutter clean
清理缓存,否则有些时候,新的改动不会生效。
其它相关问题
除上面所述,关于Flutter, 还有少量问题可能是大家比较关心的。例如:
包大小; gitme 1.0.0 release版,Android: 11.7M, iOS AppStore上架后38M,可见android包比ios包小很多,当然,ios中各种尺寸的icon和launchImage的确会比android多占使用些空间,但是这3倍的差距的确也大了少量。 笔者尚未研究flutter framework ios部分代码,至于优化空间,我想若能更好,谷歌是不会不采取行动的。
热升级; flutter release版默认是AOT,所以要实现热升级,那就只能依赖dart作为脚本语言的特性,采使用JIT模式,而flutter的debug模式默认就是JIT模式,而JIT模式和AOT模式性能差距是非常大的,假如要做热更,问题瓶颈应该在性能。但是随着苹果AppStore审核策略的收紧,用热更都会面临被拒风险,所以建议需要动态化的功能还是通过h5或者rn/weex这样的框架,当然h5的风险要比rn/weex更低。
性能; Flutter AOT模式下比JIT性能好很多,假如你开发时在debug模式感觉性能不佳,可以切换到AOT模式(打Release包)试试,整体来说,flutter的性能还是符合预期的,假如Release模式下性能仍然不佳,那么你就要考虑重构你的代码(或者者换种实现方式)。
反馈和建议
我们之所以做gitme,最初是想做一个flutter范例,使用户可以直接下载,能直观感受flutter。同时也是想做一款能够给开发者带来真正价值的APP。 我们(Flutter中文网)会继续迭代gitme,假如大家有什么好的建议或者发现了bug,欢迎反馈,请在gitme issue中反馈。
下个版本计划
下个版本我们主要会在代码染色和缓存方面来优化使用户体验。对于前者,上文已经仔细说过,不在赘述;对于后者,主要是由于github在墙外,在国内较慢,有时还会不稳固,所以我们考虑在APP中做少量适当的缓存策略。当然假如您有其它好的功能建议,欢迎反馈。
最后
我们欢迎您用Gitme ,假如您觉得好,欢迎把它推荐给您的朋友、同事(菜单>分享), 也欢迎您的建议。最后再次贴出gitme官网https://flutterchina.club/app/gm.html 。
我们有一个APP体验群,您可以扫描下面二维码加入,如二维码已过期,可以增加管理员微信Demons-du(增加时请备注”gitme使用户”), 他会将你拉进群。
gitme体验群
1. 本站所有资源来源于用户上传和网络,如有侵权请邮件联系站长!
2. 分享目的仅供大家学习和交流,您必须在下载后24小时内删除!
3. 不得使用于非法商业用途,不得违反国家法律。否则后果自负!
4. 本站提供的源码、模板、插件等等其他资源,都不包含技术服务请大家谅解!
5. 如有链接无法下载、失效或广告,请联系管理员处理!
6. 本站资源售价只是摆设,本站源码仅提供给会员学习使用!
7. 如遇到加密压缩包,请使用360解压,如遇到无法解压的请联系管理员
开心源码网 » 使用Flutter开发一个github用户端