Flutter Text_ 扶我起来

InkWell(
child: Image.asset(
‘assets/candies.png’,
width: 20,
height: 20,
),
onTap: () {
launch(‘https://github.com/fluttercandies/extended_text’);
},
)
],
),
),
),
);

image.png

无法指定文本溢出(省略号)的位置

无法指定文本溢出(省略号)的位置,比如下面两种常见场景。

  1. 省略号在中间

image.png

  1. 省略号在前面

image.png

Flutter 不支持,那么下面我们看看 原生 以及 web 端是否支持。(如有不对,望指正。)

平台开头中间结尾
androidandroid:ellipsize = "start"android:ellipsize = "middle"android:ellipsize = "end"
IosNSLineBreakByTruncatingHeadNSLineBreakByTruncatingMiddleNSLineBreakByTruncatingTail
webtext-overflow: ellipsis clip不支持text-overflow: clip ellipsis
flutter不支持不支持TextOverflow.ellipsis

注意的是 android 只有在一行的情况才有效,需要设置 android:singleline = "true"web 不支持中间;Ios 是其中支持最好的。

可以看到,原生 以及 web 端对该功能的支持比 Flutter 好太多了。那么我们 Flutter 就没办法了吗?当然不是,之前解决 文本溢出(省略号)没法自定义 ,其原理完全可以应用在这个功能上面,以下为最终效果图。

image.png

performLayout

performLayout 方法中,TextPainter 经过 layout 就可以知道当前文本是否有溢出。

final Size textSize = _textPainter.size;
final bool textDidExceedMaxLines = _textPainter.didExceedMaxLines;
size = constraints.constrain(textSize);

final bool didOverflowHeight = size.height < textSize.height || textDidExceedMaxLines;
final bool didOverflowWidth = size.width < textSize.width;
// TODO(abarth): We’re only measuring the sizes of the line boxes here. If
// the glyphs draw outside the line boxes, we might think that there isn’t
// visual overflow when there actually is visual overflow. This can become
// a problem if we start having horizontal overflow and introduce a clip
// that affects the actual (but undetected) vertical overflow.
final bool hasVisualOverflow = didOverflowWidth || didOverflowHeight;

计算不溢出的情况

这个还是要依靠 TextPainter,通过二分查找,找到 不溢出的溢出 临界点 X

开头中间结尾
[0,X][m,X]m 为中间文本的索引位置文本不需要改变,无需计算

Range 范围内的文本将不会显示。

处理文本

比如 abcdef, 我们找到的 Range[2,3] ,即最终显示 ab...ef。考虑支持选择复制,所以我们这里不能简单丢掉 cd。这里利用到 SpecialTextSpan 的一个功能。

你见到的并不是真实的

SpecialTextSpan(
‘abef’,
actualText: ‘abcdef’,
);

溢出内容绘制
开头中间结尾
画在文本开始位置画在文本中间位置画在文本结尾位置

实际上,我们要做的时候,在这些位置绘制溢出内容,并且遮住下面的文字。为了达到这个目的,

  1. 根据 Range 获得 TextSelection,利用 TextPainter.getBoxesForSelection 获取该段遮蔽文字的区域 overflowRect
  2. overflowRect 跟溢出内容的大小取并集。
  3. 移动 Range 继续取并集, 直到 overflowRect 跟溢出内容的大小完全包容。(确保溢出内容显示以及遮蔽文字不会露出来)

代码在_setOverflowRect 方法中。

遮蔽文字

在之前的版本中,由于 Paint()..blendMode = BlendMode.clear 无法对文字进行清除,只好利用 canvas.clipPath(path)overflowRect 部分的文字裁剪掉。现在官方已经修复掉这个问题,现在你可以这样做。(实际中小伙伴经常提起不知道怎么做摄像头的蒙层,其实你可以利用这个办法,将蒙层挖一个孔。)

if (_overflowRect != null) {
context.canvas.saveLayer(_offset & size, Paint());
}
// 绘制文字
_paint(context, offset);
if (_overflowRect != null) {
// 清除掉这块区域的文字
context.canvas.drawRect(
_overflowRect!.shift(_offset), Paint()…blendMode = BlendMode.clear);
context.canvas.restore();
}

更多的细节

其实上面只是大概讲了一下思路,其中是包含很多计算的,感兴趣的童鞋,可以看看这个文件.

缺点

  • 利用 TextPainter 的一些方法来实现最终计算,TextPainter 其实也是有很多限制的,比如 对 WidgetSpan 的计算错误,虽然已经利用一些 workaround 来规避了。但是官方引擎的每次改动都可能对结果有影响,反正是被坑过,懂的都懂。
  • 对面 middle 这种情况,文字每一行行高如果相差太大的话,极可能会导致计算失败,TextPainter 并没有提供一个 api,来直接告诉我们这些信息。

换行和溢出不友好

换行和溢出不友好, 比如在 Flutter 中显示下面一段话。

您的糖果服务已续期,服务订单号: 12345678910987654321110。

现实期望
image.pngimage.png
image.pngimage.png

这个问题也不是 Flutter 独有的,关系到对 word 排版的处理。目前的解决方案,是向文本中插入 zero-width space(\u{200B}).

Characters(‘abc’).join(‘\u{200B}’);

当然你可以直接将 ExtendedText.joinZeroWidthSpace 设置为 true

ExtendedText(
joinZeroWidthSpace: true,
)

或者你也可以利用 ExtendedTextLibrary 库中的扩展方法。

  1. 文本

String input=‘abc’.joinChar(‘\u{200B}’);

  1. InlineSpan

InlineSpan innerTextSpan;
innerTextSpan = joinChar(
innerTextSpan,
Accumulator(),
‘\u{200B}’,
);

但是你需要注意以下问题:

  1. word 不再是 word,你将无法通过双击选择 word

  2. 文本被修改了, 如果 ExtendedText.selectionEnabledtrue, 你需要重写 TextSelectionControlshandleCopy,将 zero-width space(\u{200B}) 去掉。

class MyTextSelectionControls extends TextSelectionControls {

@override
void handleCopy(TextSelectionDelegate delegate,
ClipboardStatusNotifier? clipboardStatus) {
final TextEditingValue value = delegate.textEditingValue;

String data = value.selection.textInside(value.text);
// remove zeroWidthSpace
data = data.replaceAll(‘\u{200B}’, ‘’);

Clipboard.setData(ClipboardData(
text: value.selection.textInside(value.text),

最后

小编这些年深知大多数初中级Android工程师,想要提升自己,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助

因此我收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、电子书籍、讲解视频,并且后续会持续更新

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人

都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

如果你需要这些资料, ⬅ 专栏获取
aE0eVC-1719183969405)]

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、电子书籍、讲解视频,并且后续会持续更新

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人

都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

如果你需要这些资料, ⬅ 专栏获取

  • 15
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Flutter Zoomable Image 是一个用于 Flutter 应用程序的库,它提供了一个可缩放和拖动的图像小部件。使用 Flutter Zoomable Image,您可以轻松地实现图像的缩放、拖动和捏放手势操作。这对于创建具有可交互性的图像查看器和画廊等应用程序非常有用。 要使用 Flutter Zoomable Image,您需要在项目的 `pubspec.yaml` 文件中添加依赖项,并运行 `flutter packages get` 命令来获取库。 以下是一个简单的示例代码,演示了如何在 Flutter 中使用 Zoomable Image: ```dart import 'package:flutter/material.dart'; import 'package:flutter_zoomable_image/flutter_zoomable_image.dart'; void main() { runApp(MyApp()); } class MyApp extends StatelessWidget { @override Widget build(BuildContext context) { return MaterialApp( title: 'Zoomable Image Demo', theme: ThemeData( primarySwatch: Colors.blue, ), home: MyHomePage(), ); } } class MyHomePage extends StatelessWidget { @override Widget build(BuildContext context) { return Scaffold( appBar: AppBar( title: Text('Zoomable Image Demo'), ), body: Center( child: ZoomableImage( AssetImage('path/to/your/image.jpg'), placeholder: Center(child: CircularProgressIndicator()), backgroundColor: Colors.black, ), ), ); } } ``` 在上面的示例中,我们创建了一个简单的 Flutter 应用程序,其中包含一个使用 ZoomableImage 小部件的页面。ZoomableImage 接受一个 AssetImage 对象作为图像源,并提供了一些可选参数,例如 placeholder(用于在图像加载期间显示的小部件)和 backgroundColor(用于设置图像背景色)。 您可以根据自己的需求定制 Zoomable Image 的样式和行为。要了解更多关于 Flutter Zoomable Image 的信息和用法,请参考官方文档或库的 GitHub 页面。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值