首页 > 其他分享 >一统天下 flutter - widget 自定义: 通过 SingleChildRenderObjectWidget 实现自定义组件

一统天下 flutter - widget 自定义: 通过 SingleChildRenderObjectWidget 实现自定义组件

时间:2023-05-06 14:57:09浏览次数:52  
标签:widget 自定义 SingleChildRenderObjectWidget context child shadowOffset flutter

源码 https://github.com/webabcd/flutter_demo
作者 webabcd

一统天下 flutter - widget 自定义: 通过 SingleChildRenderObjectWidget 实现自定义组件

示例如下:

lib\widget\custom\single_child_render_object_widget.dart

/*
 * 通过 SingleChildRenderObjectWidget 实现自定义组件
 *
 * RenderBox 继承自 RenderObject,其用于在屏幕上绘制内容
 * SingleChildRenderObjectWidget 继承自 RenderObjectWidget
 * SingleChildRenderObjectWidget 是只有一个 child 的 widget,其可以通过重写 createRenderObject() 让指定的 RenderObject 绘制内容
 */

import 'package:flutter/material.dart';
import 'package:flutter/rendering.dart';

class SingleChildRenderObjectWidgetDemo extends StatefulWidget {
  const SingleChildRenderObjectWidgetDemo({Key? key}) : super(key: key);

  @override
  _SingleChildRenderObjectWidgetDemoState createState() => _SingleChildRenderObjectWidgetDemoState();
}

class _SingleChildRenderObjectWidgetDemoState extends State<SingleChildRenderObjectWidgetDemo> {

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: const Text("title"),),
      backgroundColor: Colors.orange,
      body: Container(
        color: Colors.red,
        /// 使用自定义组件
        child: const ShadowBox(
          child: Icon(
            Icons.accessibility,
            color: Colors.black,
            size: 200,
          ),
          shadowOffset: Offset(60, 60),
        ),
      ),
    );
  }
}

/// 继承 SingleChildRenderObjectWidget 实现一个自带阴影效果的自定义组件
class ShadowBox extends SingleChildRenderObjectWidget {
  /// 阴影的偏移量
  final Offset shadowOffset;
  const ShadowBox({super.key, super.child, required this.shadowOffset});

  /// 创建一个 RenderObject 用于绘制内容
  /// 注:这个 createRenderObject() 方法是由对应的 Element 调用的
  @override
  RenderObject createRenderObject(BuildContext context) {
    return RenderShadowBox(shadowOffset: shadowOffset);
  }

  /// 更新 RenderObject 以便重绘
  /// 注:这个 updateRenderObject() 方法是由对应的 Element 调用的
  @override
  void updateRenderObject(BuildContext context, covariant RenderShadowBox renderObject) {
    renderObject.shadowOffset = shadowOffset;
  }
}

/// 继承 RenderBox 实现自定义绘制逻辑,并通过 with RenderObjectWithChildMixin 简化开发
class RenderShadowBox extends RenderBox with RenderObjectWithChildMixin {
  /// 阴影的偏移量
  late Offset shadowOffset;
  RenderShadowBox({required this.shadowOffset});

  /// 布局
  @override
  void performLayout() {
    /// 布局 child
    /// parentUsesSize 为 true 则自己可以获取到 child 的尺寸
    /// parentUsesSize 为 false 则自己获取不到 child 的尺寸,也就是说自己不需要因 child 的改变而重新布局,从而提高效率
    child!.layout(constraints, parentUsesSize: true);
    /// 设置自己的尺寸与 child 相同(需要 child 在 layout() 的时候设置 parentUsesSize 为 true)
    size = (child as RenderBox).size;
  }

  /// 绘制
  /// 注:绘制是不受布局限制的,也就是说允许在布局之外绘制
  @override
  void paint(PaintingContext context, Offset offset) {
    /// 在指定的位置绘制 child
    context.paintChild(child!, offset);

    /// 在指定的位置添加一个不透明度为 0x40 的图层
    context.pushOpacity(offset, 0x40, (context, offset) {
      /// 在这个图层内的指定位置绘制 child,从而实现阴影效果
      context.paintChild(child!, shadowOffset);
    });
  }
}

源码 https://github.com/webabcd/flutter_demo
作者 webabcd

标签:widget,自定义,SingleChildRenderObjectWidget,context,child,shadowOffset,flutter
From: https://www.cnblogs.com/webabcd/p/flutter_lib_widget_custom_single_child_render_object_wid

相关文章

  • 一统天下 flutter - widget 自定义: 通过 CustomPaint 实现自定义组件
    源码https://github.com/webabcd/flutter_demo作者webabcd一统天下flutter-widget自定义:通过CustomPaint实现自定义组件示例如下:lib\widget\custom\custom_paint.dart/**通过CustomPaint实现自定义组件**CustomPaint继承自SingleChildRenderObjectW......
  • 一统天下 flutter - widget 自定义: 通过组合多个 Widget 的方式实现自定义组件
    源码https://github.com/webabcd/flutter_demo作者webabcd一统天下flutter-widget自定义:通过组合多个Widget的方式实现自定义组件示例如下:lib\widget\custom\custom_widget.dart/**通过组合多个Widget的方式实现自定义组件*/import'dart:math';impo......
  • Material Design UI Widgets
     AndroidL开发者预览支持库提供两个新的Widgets,RecyclerView和CardView。使用这两个Widgets可以显示复杂的Listview和卡片布局,这两个Widgets默认使用Materialdesign。RecyclerView  RecyclerView是一个更高级柔性版本的Listview,RecyclerView是一个能包含很多视图的容器,它能......
  • 【Apache POI】Word文档转换HTML,多级列表自定义处理
    本文使用poi和xdocreport组件,在其基础自定义实现某些功能最近有个需求,文档的转换,需要把Word文档转换为编辑器可识别支持的HTML格式类型,Apache的开源组件poi可以解析docx和doc类型的文档,于是使用该组件实现需求关于Word文档的俩种格式,docx格式是一种压缩文件,由xml格......
  • D365: 实体自定义暂存表到目标表
    当提供的模板文件不能满足D365数据表的数据结构或者模板的数据不能直接通过DMF导入到实际的业务表时,我们按照模板自定义实体后,可以实现如下方法来将暂存表的数据通过一定的逻辑处理,将数据转换到我们的业务表中,sample代码如下publicstaticcontainercopyCustomStagingToTarget(......
  • 自定义mybatis插件之全局数据过滤
    目录一、介绍二、实现三、效果四、源码一、介绍通过开发mybatis的插件来实现对全局的sql查询语句进行拦截,并新增全局的过滤条件做到无感知的数据过滤,比如全局过滤某个租户的数据。二、实现实现思路1、通过mybatis的拦截器拦截所有查询的sql2、使用Druid里面的工具类解析sq......
  • 微信小程序-根据同声传译插件创建语音转文字的自定义插件
    使用了vantweapp组件.js//page/common/components/voice/voice.jsimportToastfrom'../../../../vant-weapp/dist/toast/toast';//引入插件:微信同声传译varplugin=requirePlugin("WechatVoice");//获取全局唯一的语音识别管理器recordRecoManagerletmanager=pl......
  • CesiumWidget.js的作用
    顾名思义,CesiumWidget就是cesium小部件的意思。但是,cesium包含哪些小部件?这些小部件又有哪些意义?这些小部件是不可或缺的吗?看《Cesium原理篇:1最长的一帧之渲染调度》讲,好像cesium的启动就是由widget来触发的?CesiumWidget.jsstartRenderLoop函数需要传入一个widget参数。这个w......
  • .net maui blazor创建存储自定义目录文件
    stringdir="/storage/emulated/0/Android/data/com.example.myapp/data";if(!Directory.Exists(dir)){Directory.CreateDirectory(dir);}stringpath=Path.Combine(dir,"a.txt");File.WriteAllText(path,"abc");//com.example.myapp......
  • 【剪裁 widget】Flutter ClipOval 与 Flutter ClipRRect
    本文是【剪裁widget】系列的第三篇,也是最后一篇,今天介绍一下ClipOval和ClipRRect。ClipOval介绍FlutterClipOval用椭圆形去剪裁child,path以外的部分不显示,还能高效的实现动画。剪裁是在绘制阶段,具体实现是在paint方法中调用PaintingContext类的pushClipPath方法进......