【绘制 widget】Flutter Transform

简介: 【绘制 widget】Flutter Transform

image.png


绘制 child 之前 Transform 会对 child 进行平移,旋转缩放等变换。Transform 不会对 size 造成影响。

Transform 介绍

Transform 继承自 SingleChildRenderObjectWidget,RenderTransform 继承自 RenderProxyBox,所以 Transform 也能算作是 布局 widget ,但是他并没有 override PerformLayout,对布局方面没有什么建树,完全继承父类的行为, 有 child 的时候  size 和 child 一样大。源码方面就是用 canvas 画出来,对于 Transform 最重要的就是如何使用,理解每个参数的作用。

const Transform({
    super.key,
    required this.transform,
    this.origin,
    this.alignment,
    this.transformHitTests = true,
    this.filterQuality,
    super.child,
 })
复制代码

我们看下这些参数都如何用。

Matrix4 变换 transform

transform 是一个 Matrix4,可以直接使用它来控制如何变换,只是有些复杂。

Matrix4 平移

atrix4.translationValues(offset.dx, offset.dy, 0.0)
复制代码

Matrix4 缩放

Matrix4.diagonal3Values(scaleX, scaleY, 1)
复制代码

Matrix4 旋转

static Matrix4 _createZRotation(double sin, double cos) {
    final Matrix4 result = Matrix4.zero();
    result.storage[0] = cos;
    result.storage[1] = sin;
    result.storage[4] = -sin;
    result.storage[5] = cos;
    result.storage[10] = 1.0;
    result.storage[15] = 1.0;
    return result;
  }
 static Matrix4 _computeRotation(double radians) {
    assert(radians.isFinite, 'Cannot compute the rotation matrix for a non-finite angle: $radians');
    if (radians == 0.0) {
      return Matrix4.identity();
    }
    final double sin = math.sin(radians);
    if (sin == 1.0) {
      return _createZRotation(1.0, 0.0);
    }
    if (sin == -1.0) {
      return _createZRotation(-1.0, 0.0);
    }
    final double cos = math.cos(radians);
    if (cos == -1.0) {
      return _createZRotation(0.0, -1.0);
    }
    return _createZRotation(sin, cos);
  }
  transform = _computeRotation(angle);
复制代码

直接使用还是很麻烦的,所以我们一般用命名构造函数,就不用关心 Matrix4 了,直接用相关参数即可。

Transform.translate(
    offset: const Offset(0.0, 15.0),
    child: Container(
      padding: const EdgeInsets.all(8.0),
      color: const Color(0xFF7F7F7F),
      child: const Text('Quarter'),
    ),
  )
复制代码
Transform.scale(
   scale: 0.5,
   child: Container(
     padding: const EdgeInsets.all(8.0),
     color: const Color(0xFFE8581C),
     child: const Text('Bad Idea Bears'),
   ),
 )
复制代码
Transform.rotate(
   angle: -pi / 12.0,
   child: Container(
     padding: const EdgeInsets.all(8.0),
     color: const Color(0xFFE8581C),
     child: const Text('Apartment for rent!'),
   ),
 )
复制代码

需要注意的是 angle 是一个弧度,一周是 2pi。 用 pi 需要 import 'dart:math';

如果要扭曲还是得直接用 Matrix4.skewX, Matrix4.skewY, Matrix4.skew

Transform origin

origin 是应用变换的原点。默认情况下 scale,roate 的变换原点是 center( alignment 默认是 center )。origin是一个offset。 dx 为正表示原点向左偏移,为负向右偏移;dy 为正向上偏移,为负向下偏移。


image.png


Container(
  decoration: BoxDecoration(border: Border.all(color: Colors.red)),
  child: Transform.scale(
     origin: Offset(50,50),
    scale: 2,
    child: Container(
      width: 100,
      height: 100,
      color: Color.fromRGBO(0, 0, 200, .2),
    ),
  ))
复制代码

image.png

origin: Offset(-50,-50),
复制代码

摆放 child

alignment 和 origin 的效果是一样的,都是平移 child,他们可以同时存在,效果叠加。

比如上面的例子。把默认的 alignment:Alignment.center 改为  alignment:Alignment.topLeft。

image.png

Transform.scale(
          alignment:Alignment.topLeft ,
          origin: Offset(0,0),
复制代码

image.png

Transform.scale(
          alignment:Alignment.topLeft ,
          origin: Offset(50,50),
复制代码

transformHitTests

transformHitTests 默认为true,点击的 local 位置是相对于变换后的 child,否则是相对于变换前的 child。




目录
相关文章
|
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