Flutter 137: 图解自定义 ACEFoldTextView 折叠文本
2021/9/28 8:10:44
本文主要是介绍Flutter 137: 图解自定义 ACEFoldTextView 折叠文本,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!
小菜在学习 Flutter 过程中,有特别需求是对于文本过长的内容需要展示固定行数,而在文本右下角有提示用户点击展开和收起;小菜尝试自定义一个可折叠收缩的 ACEFoldTextView;
ACEFoldTextView
小菜首先简单梳理了一下设计流程,如下图所示;
- 当文本内容所占据行数小于等于限制的最大行数时,默认展示整个文本内容,不会有【展开/收起】;
- 当文本内容所占据行数大于限制的最大行数时,默认展示最大行数内容,并在右下角显示【展开】提示;
- 点击【展开】区域时,当文本内容最后一行内容与【展开】区域占据内容宽度之和小于最大宽度时,默认展示【收起】;
- 点击【展开】区域时,当文本内容最后一行内容与【展开】区域占据内容宽度之和大于等于最大宽度时,【收起】区域换行展示;
1. 透明渐变【展开/收起】
小菜整体通过 Stack 层级嵌套方式在右下角显示可点击的【展开/收起】文本区,为了提高显示效果,并防止完全遮挡内容文本,小菜尝试了两种方式来实现颜色透明度渐变;
1.1 ShaderMask 着色器
小菜之前有重点介绍过 ShaderMask 着色器,可以对子 Widget 进行颜色处理,包括遮罩层特效展示;小菜设置了一个 LinearGradient 线性渐变,但 ShaderMask 是对整个子 Widget 遮罩层生效,可能会影响 Text 文本显示效果,需要 Stack 层级使用;
_transparentWid02() => ShaderMask( shaderCallback: (bounds) => LinearGradient( colors: [_bgColor.withOpacity(0.0), _bgColor], ).createShader(bounds), child: Container( alignment: Alignment.centerRight, color: Colors.white, width: _kMoreWidth, child: Text((_temLines > _maxLines) ? '展开' : '收起', style: TextStyle(color: Theme.of(context).accentColor, fontSize: widget.textStyle?.fontSize ?? 14.0))));
1.2 Container BoxDecoration
第二种就是常用的 Container 配合设置 BoxDecoration 设置线性渐变色;该方式使用更为便捷;
_transparentWid01() => Container( alignment: Alignment.centerRight, decoration: BoxDecoration( gradient: LinearGradient( colors: [_bgColor.withOpacity(0.0), _bgColor], end: FractionalOffset(0.5, 0.5))), width: _kMoreWidth, child: Text((_temLines > _maxLines) ? '展开' : '收起', style: TextStyle(color: Theme.of(context).accentColor, fontSize: widget.textStyle?.fontSize ?? 14.0)));
2. Text 文本内容折叠
小菜想实现文本折叠,首先需要预先得知 Text 文本在范围内占据的行数,一般都需要通过 TextPainter 等方式获取;小菜尝试了两种方式进行判断;
2.1 TextPainter.didExceedMaxLines
小菜之前也有简单了解过 TextPainter 与 TextSpan 的应用,主要用于文本的绘制,当设置 maxLines 之后,可以通过 didExceedMaxLines 判断文本内容是否已经超行;小菜之后会对 TextPainter 再深入研究一下;
_checkOverMaxLines01(maxLines, maxWidth) { final textSpan = TextSpan(text: _textStr, style: widget.textStyle); final textPainter = TextPainter(text: textSpan, textDirection: TextDirection.ltr, maxLines: maxLines); textPainter.layout(maxWidth: widget.maxWidth ?? MediaQuery.of(context).size.width); return textPainter.didExceedMaxLines; }
2.2 LineMetrics
didExceedMaxLines 可以直接获取文本内容是否超行,但无法获取每行文本信息等;于是小菜尝试了 computeLineMetrics() 方式获取 LineMetrics 基线度量;可以获取每行内容所占据的宽高等;
当然 LineMetrics 也无法获取每行文本内容,以及在两种文本对齐方式共用时有注意事项,小菜之后会进一步研究;
Tips: 在使用 computeLineMetrics() 获取 LineMetrics 信息时,需要注意 TextPainter 必须设置好 textDirection 文本对齐方式,以及在 layout 布局之后才可以获取;
_checkOverMaxLines02(maxWidth) { final textSpan = TextSpan(text: _textStr, style: widget.textStyle); final textPainter = TextPainter(text: textSpan, textDirection: TextDirection.ltr); textPainter.layout(maxWidth: widget.maxWidth ?? MediaQuery.of(context).size.width); _lines = textPainter.computeLineMetrics(); return _lines; }
3. ACEFoldTextView
有了前面两步的基础,小菜将其结合起来,生成自定义 ACEFoldTextView;通过 LinearBuilder 约束子 Text 延迟加载;通过 LineMetrics 获取最后一行文本长度,与默认【展开】所在 Widget 计算总和,之后判断是否占据超过限制最大宽度;当超过最大宽度时,小菜将文本添加一个 \n 强制换行;
return LayoutBuilder(builder: (context, size) { _isOverFlow = _checkOverMaxLines01(_maxLines, widget.maxWidth); _temLines = _checkOverMaxLines02(widget.maxWidth)?.length; return (_temLines <= _maxLines) ? _itemText() : Stack(children: <Widget>[_itemText(), _moreText()]); }); _moreText() => Positioned( bottom: 0, right: 0, child: GestureDetector( child: _transparentWid02(), onTap: () => setState(() { if (_temLines > _maxLines) { if (_lines.last.width + _kMoreWidth >= widget.maxWidth) { _maxLines = _temLines + 1; _textStr = '${widget.text}\n'; } else { _maxLines = _temLines; } } else if (_temLines == _maxLines) { _maxLines = widget.maxLines; } })));
ACEFoldTextView 案例源码
小菜对 ACEFoldTextView 的绘制到此为止,其中涉及到 TextPainter 内容较浅显,小菜之后会进一步学习研究;如有错误,请多多指导!
来源: 阿策小和尚
这篇关于Flutter 137: 图解自定义 ACEFoldTextView 折叠文本的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!
- 2025-01-12深入理解 ECMAScript 2024 新特性:Map.groupBy() 分组操作
- 2025-01-11国产医疗级心电ECG采集处理模块
- 2025-01-10Rakuten 乐天积分系统从 Cassandra 到 TiDB 的选型与实战
- 2025-01-09CMS内容管理系统是什么?如何选择适合你的平台?
- 2025-01-08CCPM如何缩短项目周期并降低风险?
- 2025-01-08Omnivore 替代品 Readeck 安装与使用教程
- 2025-01-07Cursor 收费太贵?3分钟教你接入超低价 DeepSeek-V3,代码质量逼近 Claude 3.5
- 2025-01-06PingCAP 连续两年入选 Gartner 云数据库管理系统魔力象限“荣誉提及”
- 2025-01-05Easysearch 可搜索快照功能,看这篇就够了
- 2025-01-04BOT+EPC模式在基础设施项目中的应用与优势