【剪裁 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中使用DraggableScrollableSheet创建一个底部弹出框,同时保持其可手势滑动关闭。作者遇到问题并提出对原控件进行扩展,以支持头部和列表布局的滑动关闭功能。
70 0
|
2月前
Flutter StreamBuilder 实现局部刷新 Widget
Flutter StreamBuilder 实现局部刷新 Widget
22 0
|
3月前
|
Android开发
Flutter完整开发实战详解(六、 深入Widget原理),2024百度Android岗面试真题收录解析
Flutter完整开发实战详解(六、 深入Widget原理),2024百度Android岗面试真题收录解析
|
3月前
|
开发框架 前端开发 搜索推荐
【Flutter前端技术开发专栏】Flutter中的自定义Widget与渲染流程
【4月更文挑战第30天】探索Flutter的自定义Widget与渲染流程。自定义Widget是实现复杂UI设计的关键,优点在于个性化设计、功能扩展和代码复用,但也面临性能优化和复杂性管理的挑战。创建步骤包括设计结构、定义Widget类、实现构建逻辑和处理交互。Flutter渲染流程涉及渲染对象树、布局、绘制和合成阶段。实践案例展示如何创建带渐变背景和阴影的自定义按钮。了解这些知识能提升应用体验并应对开发挑战。查阅官方文档以深入学习。
49 0
【Flutter前端技术开发专栏】Flutter中的自定义Widget与渲染流程
|
3月前
|
JavaScript 前端开发 开发者
【Flutter前端技术开发专栏】Flutter中的Widget与状态管理
【4月更文挑战第30天】本文探讨了Flutter的Widget和状态管理。Widget是Flutter构建UI的基础,分为有状态和无状态两种。状态管理确保UI随应用状态变化更新,影响应用性能和可维护性。文章介绍了`setState`、`Provider`、`Riverpod`、`Bloc`和`Redux`等状态管理方法,并通过计数器应用展示了其实现。选择合适的状态管理策略对高效开发至关重要。
37 0
【Flutter前端技术开发专栏】Flutter中的Widget与状态管理
|
3月前
|
编解码 算法 开发者
Flutter的布局系统:深入探索布局Widget与布局原则
【4月更文挑战第26天】Flutter布局系统详解,涵盖布局Widget(Row/Column、Stack、GridView/ListView、CustomSingleChildLayout)和布局原则(弹性布局、约束优先、流式布局、简洁明了)。文章旨在帮助开发者理解并运用Flutter的布局系统,创建适应性强、用户体验佳的界面。通过选择合适的布局Widget和遵循原则,可实现复杂且高效的UI设计。
|
3月前
|
前端开发 开发者 UED
Flutter的自定义Painter:深入探索自定义绘制Widget的技术实现
【4月更文挑战第26天】Flutter的自定义Painter允许开发者根据需求绘制独特UI,通过继承`CustomPaint`类和重写`paint`方法实现。在`paint`中使用`Canvas`绘制图形、路径等。创建自定义Painter类后,将其作为`CustomPaint` Widget的`painter`属性使用。此技术可实现自定义形状、渐变、动画等复杂效果,提升应用视觉体验。随着Flutter的进化,自定义Painter将提供更丰富的功能。
|
3月前
|
开发框架 搜索推荐 Android开发
Flutter的Widget基础:概念、分类与深入探索
【4月更文挑战第26天】Flutter Widget详解:基础、分类与工作原理。Widget是Flutter UI的核心,描述界面外观而非直接渲染。分为基础、布局、可滚动及状态管理四大类。基于响应式编程,状态变化时自动更新UI。了解其概念、分类和原理,能助开发者高效构建精美应用。随着Flutter生态发展,Widget系统潜力无限。
Flutter源码分析笔记:Widget类源码分析
本文记录阅读与分析Flutter源码 - Widget类源码分析。
80 0
Flutter源码分析笔记:Widget类源码分析
|
11月前
|
Dart 前端开发 开发工具
谷歌移动UI框架Flutter教程之Widget
谷歌移动UI框架Flutter教程之Widget