题记
—— 执剑天涯,从你的点滴积累开始,所及之处,必精益求精。
【x1】点击查看提示

【x2】各种系列的教程

一个程序员的修炼日记


在实际业务开发中,时常会有这种一段Html格式的标签,看下图的情况 :


在 Flutter 中,有点发愁,因为 Flutter 提供的 Text 与 RichText 还解析不了这种格式的,但是你也不能使用 WebView 插件,如果使用了,你会在每一个Item中嵌入一个浏览器内核,再强的手机,也会卡,当然肯定不能这样做,因为这样就是错误的做法。

小编经过大量的尝试与思考,终于写出来了一个插件可以来解析了,现分享给大家。

1 基本使用实现

1.2 添加依赖

小编依旧,来个pub方式:【不用说 快捷入口在这】【当然也有github】 【夸张点还有 视频支持】

dependencies:flutter_html_rich_text: ^1.0.0

1.3 加载解析 HTML 片段标签

核心方法如下:

///htmlText 就是你的 HTML 片段了HtmlRichText(htmlText: txt,),

如下代码清单 1-3-1 就是上述图中的效果:

/// 代码清单 1-3-1
class TestHtmlPage extends StatefulWidget {@override_TestPageState createState() => _TestPageState();
}class _TestPageState extends State<TestHtmlPage> {String txt ="<p>长途轮 <h4>高速驱动</h4><span style='background-color:#ff3333'>""<span style='color:#ffffff;padding:10px'> 3条立减 购胎抽奖</span></span></p>""<p>长途高速驱动轮<span ><span style='color:#cc00ff;'> 3条立减 购胎抽奖</span></span></p>";@overrideWidget build(BuildContext context) {return Scaffold(///一个标题appBar: AppBar(title: Text('A页面'),),body: Center(///一个列表child: ListView.builder(itemBuilder: (BuildContext context, int postiont) {return buildItemWidget(postiont);},itemCount: 100,),),);}///ListView的条目Widget buildItemWidget(int postiont) {return Container(///内容边距padding: EdgeInsets.all(8),child: Column(///子Widget左对齐crossAxisAlignment: CrossAxisAlignment.start,///内容包裹mainAxisSize: MainAxisSize.min,children: [Text("测试标题 $postiont",style: TextStyle(fontWeight: FontWeight.w500),),///html富文本标签Container(margin: EdgeInsets.only(top: 8),child: HtmlRichText(htmlText: txt,),)],),);}
}

以下是解析思考 烧脑的实践


2 烧脑思考实践一

Flutter 应用程序被 Android iOS平台加载,在原生 Android 中,使用TextView就可轻松实现解析(如下代码清单2-1),当然在iOS中使用UILabel也可轻松实现(如下代码清单2-2)。

// Android 原生 TextView加载Html的核心方法
//代码清单2-1
// MxgsaTagHandler 定义的一个 TagHandler 用来处理点击事件
lTextView.setText(Html.fromHtml(myContent, null, new MxgsaTagHandler(context)));
lTextView.setClickable(true);
lTextView.setMovementMethod(LinkMovementMethod.getInstance());
// iOS 原生 UILabel加载Html的核心方法
//代码清单2-2
//返回的HTML文本 如 <font color = 'red'></font>
NSString *str = @"htmlText";
NSString *HTMLString = [NSString stringWithFormat:@"<html><body>%@</body></html>", str ];NSDictionary *options = @{NSDocumentTypeDocumentAttribute : NSHTMLTextDocumentType,NSCharacterEncodingDocumentAttribute : @(NSUTF8StringEncoding)};
NSData *data = [HTMLString dataUsingEncoding:NSUTF8StringEncoding];NSMutableAttributedString * attributedString = [[NSMutableAttributedString alloc] initWithData:data options:options documentAttributes:nil error:nil];NSMutableParagraphStyle *paragraphStyle = [[NSMutableParagraphStyle alloc] init];   // 调整行间距
paragraphStyle.lineSpacing = 8.0;
paragraphStyle.alignment = NSTextAlignmentJustified;
[attributedString addAttribute:NSParagraphStyleAttributeName value:paragraphStyle range:NSMakeRange(0, attributedString.length)];[attributedString addAttribute:NSFontAttributeName value:[UIFont systemFontOfSize:15] range:NSMakeRange(0, attributedString.length)];_uiLabel.backgroundColor = [UIColor cyanColor];
_uiLabel.numberOfLines = 0;
_uiLabel.attributedText = attributedString;
[_uiLabel sizeToFit];

然后对于 Flutter 来讲是可以顺利的加载原生 View的 【在这有讲述】,如下代码清单 2-3所示就是在Flutter中通过 AndroidView 与 UiKitView来实现。

  //Flutter中加载原生View核心方法//代码清单2-3buildAndroidView() {return AndroidView(//设置标识viewType: "com.studyon./text_html",//参数的编码方式creationParamsCodec: const StandardMessageCodec(),);}/// 通过 UiKitView 来加载 iOS原生ViewbuildUIKitView() {return UiKitView(//标识viewType: "com.studyon./text_html",//参数的编码方式creationParamsCodec: const StandardMessageCodec(),);}

于是小编开发了第一波操作,开发了这样的一个插件来调用原生 View 实现渲染富文本标签【源码在这里】,这个插件使用方式很简单,如下所示:

HTMLTextWidet(htmlText: "测试一下",)

这一步操作真是所谓的骚操作,其实小编在开发前就觉得不太合适,不过以小编的个性,非得尝试验证一下,现结果出来了,就是在加载时,由于应用在列表中,使用 HTMLTextWidet 会有短暂的黑屏效果,而且内存出吃不消,如下图所示:

为什么会黑屏,闲鱼技术团队有过论述在Flutter中嵌入Native组件的正确姿势 以及 文章 深入了解Flutter界面开发中有详细论述 。

所以结果是 :不可行。


3 烧脑思考实践二

用 Java 的思想来解析 String 的方式来处理 HTML 字符串,处理成小片段,然后使用Text结合 流式布局 Wrap 来组合,核心代码如下清单 3-1 所示为解析:

  /*解析标签*/List<TagColorModel> findBackGroundColor(String htmlStr) {List<TagColorModel> tagColorModelList = [];List<String> colorSpiltList = [];String driverAdvertisement = htmlStr;if (driverAdvertisement != null) {colorSpiltList = driverAdvertisement.split("background-color");for (var i = 0; i < colorSpiltList.length; i++) {TagColorModel itemColorModel = TagColorModel();String colorsStr = colorSpiltList[i];List<String> itemSpiltList = colorsStr.split(":#");for (var j = 0; j < itemSpiltList.length; ++j) {String item = itemSpiltList[j];String itemColor = "";String itemText = "";try {if (item.length >= 6) {itemColor = item.toString().substring(0, 6);if (itemColor.trim().toUpperCase() == "FFFFFF") {itemColorModel.backGroundColor = ColorUtils.getRandomColor();} else {itemColorModel.backGroundColor = new Color(int.parse(itemColor.trim(), radix: 16) + 0xFF000000);}int startIndex = item.indexOf("\">");int endIndex = item.indexOf("</");if (startIndex != -1 && endIndex >= startIndex) {LogUtil.e("startIndex  $startIndex  endIndex $endIndex ");itemText = item.substring(startIndex + 2, endIndex);LogUtil.e("itemColor  $itemColor  itemText $itemText ");itemColorModel.text = itemText;tagColorModelList.add(itemColorModel);}}} catch (e) {///解析异常的 不必处理}}}}LogUtil.e("${tagColorModelList.length} \n\n ");return tagColorModelList;}

然后 TagColorModel 的定义如下代码清单 3-2所示:

///代码清单 3-2
class TagColorModel {///背景Color backGroundColor;///文本颜色Color textColor;///文本String text;TagColorModel({this.text = "",this.backGroundColor = Colors.transparent,this.textColor = Colors.white});
}

然后就是使用 Wrap 来使用解析的内容,如下代码清单3-3所示:

///代码清单 3-3
///获取背景颜色List<TagColorModel> colorList = findBackGroundColor(htmlStr);List<Widget> tagList = [];for (var i = 0; i < colorList.length; ++i) {TagColorModel model = colorList[i];tagList.add(Container(margin: EdgeInsets.only(right: 2, left: 4, top: 4),padding: EdgeInsets.only(left: 6, right: 6),decoration: BoxDecoration(color: model.backGroundColor,borderRadius: BorderRadius.all(Radius.circular(2)),),child: Text("${model.text}",style: TextStyle(fontSize: 12, color: model.textColor),),));}///然后再使用 Wrap 包裹Wrap(alignment: WrapAlignment.spaceBetween,children: tagList,),

实践结果:可行,但是有兼容性差,效率低。

当然闲鱼团队在文章 如何低成本实现Flutter富文本,看这一篇就够了! 中也有详细论述。

4 烧脑思考实践三

当在Flutter中 Dart 从网站中提取数据时,html依赖库是一个不错的选择,html 是一个开源的 Dart 包,主要用于从 HTML 中提取数据,从中获取节点的属性、文本和 HTML以及各种节点的内容。Html pub仓库

dependencies:html: ^0.14.0+3

于是乎小编也开始尝试,首先是使用 Html 库解析 HTML文本块,将解析的 Document 通过递归方式遍历出来所有的 node 节点,如下代码清单4-1所示:

代码清单4-1
import 'package:html/parser.dart' as parser;
import 'package:html/dom.dart' as dom;List<Widget> parse(String originHtmlString) {// 空格替换 去除所有 br 标签用 \n 代替,originHtmlString = originHtmlString.replaceAll('<br/>', '\n');originHtmlString = originHtmlString.replaceAll('<br>', '\n');originHtmlString = originHtmlString.replaceAll('<br />', '\n');///html 依赖库解析dom.Document document = parser.parse(originHtmlString);///获取 DOM 中的 node 节点dom.Node cloneNode = document.body.clone(true);// 注意: 先序遍历找到所有关键节点(由于是引用传值,所以需要重新获取一遍 hashCode)List<dom.Node> keyNodeList = new List<dom.Node>();int nodeIndex = 0;///递归遍历parseNodesTree(cloneNode, callBack: (dom.Node childNode) {if (childNode is dom.Element &&truncateTagList.indexOf(childNode.localName) != -1) {print('TEST: truncate tag nodeIndex = ${nodeIndex++}');keyNodeList.add(childNode);// 注意: 对于占据整行的图片也作为关键节点处理} else if (childNode is dom.Element &&childNode.localName == 'img' &&checkImageNeedNewLine(childNode)) {print('TEST: one line image nodeIndex = ${nodeIndex++}');keyNodeList.add(childNode);}});}
///递归遍历
void parseNodesTree(dom.Node node,{NodeTreeCallBack callBack = printNodeName}) {///遍历 Node 节点for (var i = 0; i < node.nodes.length; ++i) {dom.Node item = node.nodes[i];callBack(item);parseNodesTree(item, callBack: callBack);}
}

然后就是将 得出的 node 节点 与 Flutter 组件映射,文本使用 TextSpan ,图片使用 Image ,然后将 样式使用 TextStyle 映射,然后最后将解析的结果组件使用 Wrap 来包裹,就达到了现在的插件 flutter_html_rich_text

综合实现思路就是 使用 HTML 库完善了【烧脑思考实践二】中的解析。

解析篇幅较长,大家有兴趣可以看下 github 源码。


目前小编在西瓜视频上免费刊登 Flutter 系列教程,每日更新,欢迎关注接收提醒点击查看提示 各种系列的教程

2020.09.12 开发笔记

  • iOS中加载Flutter中的图片
  • Flutter组件精讲【01】 MateriaApp使用概述
  • Flutter组件精讲【02】MaterialApp组件的基本使用
  • Flutter组件精讲【03】 MateriaApp组件路由routes配制
  • Flutter组件精讲【04】 MateriaApp配置默认启动页面
  • Flutter组件精讲【05】 MateriaApp之页面的跳转
  • Flutter组件精讲【06】MaterialApp 组件配置 404 页面-01
  • Flutter组件精讲【07】MaterialApp 组件配置 404 页面-02

Flutter中富文件标签的解决方案相关推荐

  1. IE6,IE7和8浏览器兼容HTML5标签的解决方案

    html5大行其道的时代已经到来.如果你还在等待浏览器兼容,说明你已经与web脱节几条街了.当然,这得益于移动客户端的蓬勃发展.如果还在纠 结于,是否应该掌握html5和css3技术时,请狠狠的抽自己 ...

  2. C#实现浮动和多标签窗体解决方案---使用Dockpanel

    首先声明:在此感谢Dockpanel 实现浮动和多标签窗体解决方案给我提供思路和灵感.本文的形成也多有借鉴,再次表示感谢!在此记录一下Dockpanel框架的搭建过程(主要步骤),以及需要注意的几点. ...

  3. Java多文件压缩下载解决方案

    Java多文件压缩下载解决方案 需求: 会员运营平台经过改版后页面增加了许多全部下载链接,上周上线比较仓促,全部下载是一个直接下载ZIP压缩文件的链接,每个ZIP压缩文件都是由公司运营人员将页面需要下 ...

  4. java大文件上传解决方案

    最近遇见一个需要上传百兆大文件的需求,调研了七牛和腾讯云的切片分段上传功能,因此在此整理前端大文件上传相关功能的实现. 在某些业务中,大文件上传是一个比较重要的交互场景,如上传入库比较大的Excel表 ...

  5. 大文件上传+解决方案

    最近遇见一个需要上传百兆大文件的需求,调研了七牛和腾讯云的切片分段上传功能,因此在此整理前端大文件上传相关功能的实现. 在某些业务中,大文件上传是一个比较重要的交互场景,如上传入库比较大的Excel表 ...

  6. CGB2005 JT-1(jt概述 SqlYog 物理模型图PD 表结构 pom文件标签说明 jt环境搭建 创建项目2种,创建各种文件 idea导入,打包,删除项目,启动原理)

    注意事项: 1.京淘项目概述和动吧项目缺点描述 2.把sql文件导入数据库:通过SqlYog可视化工具或者Dos命令窗口. 3.SqlYog说明,和制作物理模型图的工具pd用发. 4.表结构学习 5. ...

  7. Flutter 5 大本地数据库解决方案

    Flutter 5 大本地数据库解决方案 原文 https://levelup.gitconnected.com/top-5-local-database-solutions-for-flutter- ...

  8. linux rz 无法上传文件,linux rz上传文件及出错解决方案

    在把Windows上的文件传至Linux端时用到SecureCRT,一般小文件都没有问题,文件太大时则出现了上传后的文件只有几K大小,当然大于2个G的是不可能传的上去的了.对于几百M到1G多的大文件要 ...

  9. ios转换html标签,iOS html标签解析解决方案

    由于项目是基于b/s结构,因为在pc端上是没有客户端这概念只有浏览器,而对于web与移动端的关系,实际上还是类似于C/S结构,因此web端与web端的通信是通过发送HTML5标签来实现,常规的做法中p ...

最新文章

  1. 这几个模型不讲“模德”,我劝它们耗子尾汁
  2. 非常精美的全能视频转换器 注册版
  3. JavaScript、jQuery、HTML5、Node.js实例大全-读书笔记3
  4. Spring如何将事件分配给专门的监听器?
  5. 最新最详细最简洁Eclipse调试PHP配置详解(Xdebug,Zend Debugger)
  6. Redis——学习之路一(初识redis)
  7. 潭州课堂25班:Ph201805201 爬虫基础 第九课 图像处理- PIL (课堂笔记)
  8. 【快速入眠】高效睡眠 - 把失眠踩在脚下
  9. 通过SharpShell快速实现Windows Shell扩展
  10. STM32笔记记录2
  11. tokenizer.tokenize(), tokenizer.encode() , tokenizer.encode_plus() 方法介绍及其区别
  12. LoRaWAN入网方式以及加密进阶版
  13. 网络安全定义和安全威胁
  14. wps的linux文字显示模糊,WPS字体模糊不清晰怎么回事
  15. 计算机管理不小心删除了e盘,【J.C.X】计算机的D盘和E盘突然消失. 小编帮你找回来...
  16. 你唯一能控制的,是自己的脾气与努力!
  17. 在重庆如何利用NFC给一卡通充值
  18. 如何使用Git提交代码
  19. 50个html+css+js项目小练习(二:动画的倒计时效果)
  20. OA系统十八:请假申请四:【请假申请】这个内嵌界面中【提交请假表单数据】的Controller层;

热门文章

  1. AI大时代下,零基础进入人工智能领域该如何学习?
  2. CV新赛事:密集场景行人检测
  3. NeurIPS 2019丨深度双线性转换改进细粒度图像分类
  4. 阿里巴巴开源通用机器学习算法平台Alink
  5. MediaPipe: Google Research 开源的跨平台多媒体机器学习模型应用框架
  6. 精度高、模型小、速度快!梯形DenseNets结构实现语义分割新高度!
  7. 霍夫变换是怎么发明的?
  8. 几行Python代码,爱上Python编程
  9. 看程序员小哥如何机智应对电信诈骗
  10. 重磅开源!新型VOLO打破多项记录!