缘由
近期由于个人项目需要,要做一个支持BBCode的富文本编辑器,可能不需要支持太多功能,但至少粗体、斜体、表情、链接是需要的。
Flutter自带的官方例子有一个示例,本系列文章会从这个示例看看如何在flutter里写一个简易的富文本编辑器。
结构
直接clone:
|
|
然后用IDE打开里面的simplistic_editor
文件夹。
首先看看依赖pubspec.yaml
:
|
|
挺干净的嘛,什么也没带。
看看目录结构:
|
|
还好还好,就这几个文件,而且大概能看出来是干嘛的。
代码量呢?
|
|
What’s up,怎么两千多行,这下慢慢看吧。没事,两千多行说明示例挺完整的,这把稳了。
入口
入口main.dart
,其中核心为_MyHomePageState
:
已删去部分不重要的代码,下文不再赘述。
|
|
用到的组件:
ReplacementTextEditingController
:用文本的实际样式替换原有的纯文本样式,也就是给文字加特效,如粗体斜体颜色。AppStateManager
: 持有AppState
的InheritedWidget
,供全局共享状态。BasicTextField
:展示内容输出的StatefulWidget
,将样式传给内部的BasicTextInputClient
,同时定义了各种手势操作的回调。这部分是内容最多的,内部的client占了一千余行。
下面先从简单和底层的部分讲起。
ReplacementTextEditingController
先上注释里的描述:
|
|
简而言之,这个controller的功能为:将一串纯文本按照文字位置和该位置上对应的样式,将纯文本渲染成带有样式外观的富文本。实际渲染时不是一个文字一个文字渲染,而是一段一段的,每一段内的样式相同,这样一段一段渲染。
而记载了“哪一段文字具有什么样的样式”这件事是由TextEditingInlineSpanReplacement
承担:
|
|
目前来看,controller做的功能只有:
- 保存当前现有的文本样式。
- 对用户输入(或者说外部的)变更进行响应,根据变更,更新自己持有的全文的文本样式。
- 根据持有的文本样式,返回一个大的富文本
TextSpan
。
TextEditingInlineSpanReplacement
代表一块文本区域和该文本区域的样式,实际持有的成员为:
TextRange
:就是一个(start, end)
的范围标记。InlineSpanGenerator
:实际是InlineSpan Function(String, TextRange)
这样的回调,函数体记载了文本该附加的样式。
除此以外,包含对刚才说的操作(添加,删除,替换)的回调onDelete
,onInsertion
,onReplacement
,onNonTextUpdate
。
操作类型
有关操作类型是如何定义的,可直接在在线demo上体验。
- 添加:键盘输入,添加了文本。
- 删除:键盘按退格键删除了文本。
- 替换:粘贴进来了文本,或者复制/剪切走了文本。哪怕粘贴的时候没有选中任何文字,看上去是在两个文字中间插入了剪切板里的文字,也看作是替换。
- 光标移动:这个刚才没讲,包含通过键盘和鼠标移动光标位置,以及全选。
到此,该讲对应的回调,来解释刚才在controller里syncReplacementRanges
留下的疑问。
onInsertion
从键盘插入文本时发生,不包含粘贴,复制,剪切这三种情况。
|
|
参数TextEditingDeltaInsertion
是flutter自带的类,代表一次插入操作,包含的信息有:
oldText
:在这次插入发生前的文本。textInserted
:这次插入的文本,通常是一个字符。如果是中文或其他输入法的话,候选词会先insertsion,然后用户选择的最终文本会以replacement的形式上屏。insertionOffset
:在oldText
的哪个位置发生的插入事件。这个位置指的是插入开始的位置。
比如:从abc变成a123bc,oldText
是abc ,textInserted
是abc ,insertionOffset
是整数1。
更新方式:
- 如果插入的位置刚好是当前范围结束的地方,而且
expand
为true,将插入的这部分文本也加到当前的style里。这是很典型的做法,像word里如果当前光标前面的文本有一些样式,插入的文本也会有这些样式。(不然写一个字更新一次样式也太蠢了) - 如果
end < offset
,也就是插到后面了,并且没挨着,将什么都不做。 - 如果
start >= offset && end > offset
,说明文本插入到当前块前面了,需要给start
和end
都加上插入文本的长度,也就是往后挪了挪。 - 如果
start < offset && end > offset
,文本是插到当前块内部,start
不变,end
加上插入文本的长度。
onDelete
当使用键盘的退格键删除文本时触发onDelete
回调。
|
|
参数TextEditingDeltaDeletion
构成:
oldText
: 删除前的文本。deletedRange
:oldText
中被删除的部分的起止标记。
更新方式:
- 当
offset.start <= start < offset.end < end
时,当前块左边部分被删除,当前区域会向左移并且缩短,变成\(offset.start, end - offset.length\)
。- 代码写的是
start = offset.end - offset.length
,实际直接start = offset.start
应该也一样。
- 代码写的是
- 当
start < offset.start < end < offset.end
时,从中间某个位置开始,后面的均被删除,当前区域会缩短成\(start, offset.start\)
。 - 当
start < offset.start && offset.end < end
时,中间有一段被删除,变成,\(start, end - offset.length\)
。 - 当
offset.start <= start && end <= offset.end
时,当前块被完全删除,返回null。 - 当
offset.start < start && end <= start
时,被删除部分在当前块的左边,只需要向左移,变成\(start - offset.length, end - offset.length\)
。 - 当
end <= offset.start && end < offset.end
时,被删除部分在当前块的右边,什么也不用做。
onReplacement
参数TextEditingDeltaReplacement
,包含:
oldText
。replacementText
:替换进去的文本。replacedRange
:被替换掉的位置范围。
这部分代码略长:
|
|
总结
以上是入口组件这部分,剩下的内容慢慢讲。