Flutter 绘制原理

Flutter中和绘制相关的对象有三个,分别是Canvas、Layer 和 Scene:

  • Canvas:封装了Flutter Skia各种绘制指令,比如画线、画圆、画矩形等指令。
  • Layer:分为容器类和绘制类两种;暂时可以理解为是绘制产物的载体,比如调用 Canvas 的绘制 API 后,相应的绘制产物被保存在 PictureLayer.picture 对象中。
  • Scene:屏幕上将要要显示的元素。在上屏前,我们需要将Layer中保存的绘制产物关联到 Scene 上。

Flutter 绘制流程:

  1. 构建一个 Canvas,用于绘制;同时还需要创建一个绘制指令记录器,因为绘制指令最终是要传递给 Skia 的,而 Canvas 可能会连续发起多条绘制指令,指令记录器用于收集 Canvas 在一段时间内所有的绘制指令,因此Canvas 构造函数第一个参数必须传递一个 PictureRecorder 实例。
  2. Canvas 绘制完成后,通过 PictureRecorder 获取绘制产物,然后将其保存在 Layer 中。
  3. 构建 Scene 对象,将 layer 的绘制产物和 Scene 关联起来。
  4. 上屏;调用window.render API 将Scene上的绘制产物发送给GPU。

Picture

  1. Picture 实际上是一系列的图形绘制操作指令,这一点可以参考 Picture 类源码的注释。
  2. Picture 要显示在屏幕上,必然会经过光栅化,随后Flutter会将光栅化后的位图信息缓存起来,也就是说同一个 Picture 对象,其绘制指令只会执行一次,执行完成后绘制的位图就会被缓存起来。

综合以上两点,我们可以看到 PictureLayer 的“绘制产物”一开始是一些列“绘图指令”,当第一次绘制完成后,位图信息就会被缓存,绘制指令也就不会再被执行了,所以这时“绘制产物”就是位图了。为了便于理解,后续我们可以认为指的就是绘制好的位图。

Canvas绘制的位图转图片

既然 Picture 中保存的是绘制产物,那么它也应该能提供一个方法能将绘制产物导出,实际上,Picture有一个toImage方法,可以根据指定的大小导出Image。

//将图片导出为Uint8List
final Image image = await pictureLayer.picture.toImage();
final ByteData? byteData = await image.toByteData(format: ImageByteFormat.png);
final Uint8List pngBytes = byteData!.buffer.asUint8List();
print(pngBytes);

Layer

作用

  1. 可以在不同的frame之间复用绘制产物(如果没有发生变化)。
  2. 划分绘制边界,缩小重绘范围。

类型

  1. OffsetLayer:根 Layer,它继承自ContainerLayer,而ContainerLayer继承自 Layer 类,我们将直接继承自ContainerLayer 类的 Layer 称为容器类Layer,容器类 Layer 可以添加任意多个子Layer。
    1. 将组件树的绘制结构组成一棵树。

      因为 Flutter 中的 Widget 是树状结构,那么相应的 RenderObject 对应的绘制结构也应该是树状结构,Flutter 会根据一些“特定的规则”(后面解释)为组件树生成一棵 Layer 树,而容器类Layer就可以组成树状结构(父 Layer 可以包含任意多个子 Layer,子Layer又可以包含任意多个子Layer)。

    2. 可以对多个 layer 整体应用一些变换效果。

      容器类 Layer 可以对其子 Layer 整体做一些变换效果,比如剪裁效果(ClipRectLayer、ClipRRectLayer、ClipPathLayer)、过滤效果(ColorFilterLayer、ImageFilterLayer)、矩阵变换(TransformLayer)、透明变换(OpacityLayer)等。

  2. PictureLayer:保存绘制产物的 Layer,它直接继承自 Layer 类。我们将可以直接承载(或关联)绘制结果的 Layer 称为绘制类 Layer

相关推荐

  1. Flutter 绘制原理

    2024-03-28 11:06:04       19 阅读
  2. flutter绘制半圆和绘制其他图形

    2024-03-28 11:06:04       32 阅读
  3. Flutter核心原理

    2024-03-28 11:06:04       6 阅读
  4. Android UI底层绘制原理

    2024-03-28 11:06:04       10 阅读
  5. flutter嵌入原生view

    2024-03-28 11:06:04       12 阅读
  6. 绘制虚线圆角矩形的Flutter小部件

    2024-03-28 11:06:04       18 阅读
  7. 绘制虚线圆角矩形的Flutter小部件

    2024-03-28 11:06:04       20 阅读

最近更新

  1. TCP协议是安全的吗?

    2024-03-28 11:06:04       16 阅读
  2. 阿里云服务器执行yum,一直下载docker-ce-stable失败

    2024-03-28 11:06:04       16 阅读
  3. 【Python教程】压缩PDF文件大小

    2024-03-28 11:06:04       15 阅读
  4. 通过文章id递归查询所有评论(xml)

    2024-03-28 11:06:04       18 阅读

热门阅读

  1. 数据结构奇妙旅程之深入解析希尔排序

    2024-03-28 11:06:04       17 阅读
  2. Unity DOTS系列之IJobChunk来迭代处理数据

    2024-03-28 11:06:04       22 阅读
  3. 3月26日,每日信息差

    2024-03-28 11:06:04       15 阅读
  4. C++|char*大小端转换

    2024-03-28 11:06:04       18 阅读
  5. 【力扣】210 课程表(c++)解析

    2024-03-28 11:06:04       20 阅读