【剪裁 widget】Flutter ClipPath

简介: 【剪裁 widget】Flutter ClipPath

image.png

Flutter ClipPath 用 path 去剪裁 child,path 以外的部分不显示,还能高效的实现动画。

ClipPath 介绍

和布局 widget 不同,剪裁 widget 功能实现是在绘制阶段。所以剪裁 widget 的 size 是不会变的,无论怎样剪裁。

剪裁是在绘制阶段,具体实现是在 paint 方法中调用 PaintingContext 类的 pushClipPath 方法进行剪裁。

void paint(PaintingContext context, Offset offset) { 
...
 layer = context.pushClipPath(
    needsCompositing,
    offset,
    Offset.zero & size,
    _clip!,
    super.paint,
    clipBehavior: clipBehavior,
    oldLayer: layer as ClipPathLayer?,
  );
 ...
}
复制代码

path 在文档中的解释大家自己去看下。如果你用过 potoshop 或 sketch 一类的绘图软件,可能会理解的比较深刻一些。

默认情况下,ClipPath 的剪裁路径是正好包含整个 child,只有溢出 child 的部分才会被裁剪。代码如下:

@override
  Path get _defaultClip => Path()..addRect(Offset.zero & size);
复制代码

对比一下之前 ClipRect 的 默认 clipper 代码

@override
  Rect get _defaultClip => Offset.zero & size;
复制代码

虽然返回的类型不同,但效果一样,ClipPath 的矩形是用 Path 来表达的,ClipRect 的矩形是用 Rect 来表达的。虽然表达方式不同,但最后的效果是一样的。

自定义裁剪

我们可以指定 clipper 参数进行自定义裁剪。

举个例子,我们想裁剪出花朵的部分。图片大小为 100 x 100。

image.png

class MyClipper extends CustomClipper<Path> {
  @override
  Path getClip(Size size) {
    return Path()..addRect(const Rect.fromLTWH(20, 10, 60, 60));
  }
  @override
  bool shouldReclip(covariant CustomClipper<Path> oldClipper) {
    return false;
  }
}
复制代码
Center(
      child: ClipPath(
        clipper: MyClipper(),
        child: Image.network(
          'https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/552bbefbf3ba42b7910ca5721cd61a6c~tplv-k3u1fbpfcp-watermark.image?',
          width: 100,
          height: 100,
          fit: BoxFit.fill,
        ),
      ))
复制代码

ClipRect 的 MyClipper 几乎完全一样,就是把类型换成了 Path。也许你会问,既然都一样,那都用 ClipRect  不就得了?原因是 ClipPath 更加强大,能裁剪各种形状。比如圆形,菱形,三角形,还能裁剪出动物的形状。比如下面的小鸟就是官方给的例子。

image.png

既然 ClipPath这么强大,但 ClipRect 还有一席之地,说明它一定有过人之处,没错,ClipRect 的优点就是相对来说比较高效, ClipPath 是比较昂贵的,所以如果是要实现矩形剪裁,优先选用 ClipRect

如何得到 path

既然 ClipPath 是用 path 进行剪裁的,那么如何得到 path 呢?

方法一 自己写

自定义 clipper,clipper 是 CustomClipper 类型,我们只需要扩展一个 CustomClipper 就可以随便写 path了。比如我们可以自己写一个倒立的三角形的 path。

class TrianglePath extends CustomClipper<Path>{
  @override
  Path getClip(Size size) {
    final path = Path();
    path.lineTo(size.width, 0.0);
    path.lineTo(size.width / 2, size.height);
    path.close();
    return path;
  }
  @override
  bool shouldReclip(CustomClipper<Path> oldClipper) {
    return true;
  }
}
复制代码

自己写 path 是比较麻烦的,还得扩展一个类出来。

方法二 通过现有的 ShapeBorder 得到

ShapeBorder 是 形状边框的基类。ShapeBorder 类定义了 getOuterPath 方法,我们可以通过 这个方法得到 path。

Path getOuterPath(Rect rect, { TextDirection? textDirection });
复制代码

为了方便使用, ClipPath 加了一个 命名构造函数 ClipPath.shape 封装了通过 shape 拿 path 的逻辑,我们不用自己动手调用 getOuterPath 方法,只需要提供 shape 即可。

一共有下面几种 shape 可以用。

  1. BeveledRectangleBorder
  2. CircleBorder
  3. ContinuousRectangleBorder
  4. MaterialStateOutlinedBorder
  5. RoundedRectangleBorder
  6. StadiumBorder

拿 CircleBorder 举个例子吧。

ClipPath.shape(
   shape:const CircleBorder(),
   child: Container(
     width: 100,
     height: 100,
     color: Colors.blue[200],
   )),
复制代码

足够简单吧。

应用场景

Flutter ClipPath 的应用场景和 ClipRect 是一样的,不再赘述,也是剪裁 widget 和制作动画。在 【剪裁 widget】Flutter ClipRect 一文中对如何写动画有举例。  ClipPath 做动画会很炫酷,很多效果是其它方式无法实现的。

目录
相关文章
|
1月前
|
容器
Flutter Widget 解析
Flutter Widget 解析
|
1月前
|
存储 容器
Flutter 有状态Widget 和 无状态Widget
Flutter 有状态Widget 和 无状态Widget
|
2月前
深入理解Flutter鸿蒙next版本 中的Widget继承:使用extends获取数据与父类约束
本文详细介绍了Flutter中如何通过继承其他Widget来创建自定义组件。首先解释了Widget继承的基本概念,包括StatelessWidget和StatefulWidget的区别。接着通过具体示例展示了如何继承StatelessWidget和StatefulWidget,并在子类中访问父类的build方法和状态。最后,结合多个自定义Widget展示了如何在实际应用中灵活使用继承和组合来构建复杂的UI。
99 8
|
2月前
|
容器
flutter&鸿蒙next 使用 InheritedWidget 实现跨 Widget 传递状态
在 Flutter 中,状态管理至关重要。本文详细介绍了如何使用 InheritedWidget 实现跨 Widget 的状态传递。InheritedWidget 允许数据在 Widget 树中向下传递,适用于多层嵌套的场景。通过一个简单的计数器示例,展示了如何创建和使用 InheritedWidget,包括其基础概念、工作原理及代码实现。虽然 InheritedWidget 较底层,但它是许多高级状态管理解决方案的基础。
118 2
|
3月前
|
容器
flutter:第一个flutter&Widget的使用 (二)
本文介绍了Flutter框架下的基本组件及其用法,包括简单的 Stateless Widget 如文本和按钮,以及更复杂的 StatefulWidget 示例。详细解释了如何使用 `context` 获取祖先小部件的信息,并展示了 `MaterialApp` 的属性及用途。此外,还探讨了 `StatefulWidget` 与 `StatelessWidget` 的区别,以及 `AppBar` 的常见属性配置方法。适合Flutter初学者参考学习。
|
2月前
|
Dart JavaScript 前端开发
Flutter 的 Widget 概述与常用 Widgets 与鸿蒙 Next 的对比
Flutter 是 Google 开发的开源 UI 框架,用于快速构建高性能的移动、Web 和桌面应用。Flutter 通过 Widget 构建 UI,每个 UI 元素都是 Widget,包括文本、按钮、图片等。Widget 不仅描述外观,还描述行为,是不可变的。常见的 Widget 包括结构型(Container、Column、Row)、呈现型(Text、Image)、交互型(ElevatedButton)和状态管理型(StatefulWidget)。Flutter 与鸿蒙 Next 在组件化架构、开发语言、布局系统、性能和跨平台支持方面各有优势
91 0
|
6月前
Flutter-底部弹出框(Widget层级)
文章描述了如何在Flutter中使用DraggableScrollableSheet创建一个底部弹出框,同时保持其可手势滑动关闭。作者遇到问题并提出对原控件进行扩展,以支持头部和列表布局的滑动关闭功能。
218 0
|
7月前
Flutter StreamBuilder 实现局部刷新 Widget
Flutter StreamBuilder 实现局部刷新 Widget
60 0
|
8月前
|
Android开发
Flutter完整开发实战详解(六、 深入Widget原理),2024百度Android岗面试真题收录解析
Flutter完整开发实战详解(六、 深入Widget原理),2024百度Android岗面试真题收录解析
|
8月前
|
编解码 算法 开发者
Flutter的布局系统:深入探索布局Widget与布局原则
【4月更文挑战第26天】Flutter布局系统详解,涵盖布局Widget(Row/Column、Stack、GridView/ListView、CustomSingleChildLayout)和布局原则(弹性布局、约束优先、流式布局、简洁明了)。文章旨在帮助开发者理解并运用Flutter的布局系统,创建适应性强、用户体验佳的界面。通过选择合适的布局Widget和遵循原则,可实现复杂且高效的UI设计。