首页 > 其他分享 >一统天下 flutter - widget 自定义: 通过组合多个 Widget 的方式实现自定义组件

一统天下 flutter - widget 自定义: 通过组合多个 Widget 的方式实现自定义组件

时间:2023-05-06 14:56:17浏览次数:59  
标签:widget const 自定义 Widget child angle

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

一统天下 flutter - widget 自定义: 通过组合多个 Widget 的方式实现自定义组件

示例如下:

lib\widget\custom\custom_widget.dart

/*
 * 通过组合多个 Widget 的方式实现自定义组件
 */

import 'dart:math';

import 'package:flutter/material.dart';

import '../../helper.dart';

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

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

class _CustomWidgetDemoState extends State<CustomWidgetDemo> {

  double _angle = .0;

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: const Text("title"),),
      backgroundColor: Colors.orange,
      body: Column(
        children: [
          /// 通过继承已有 Widget 的方式实现自定义 Widget
          const _MyText("_MyText"),

          /// 通过继承 StatelessWidget 的方式实现自定义的无状态组件
          _GradientButton(
            colors: const [Colors.red, Colors.green],
            borderRadius: BorderRadius.circular(10),
            width: 300,
            height: 50,
            child: const Text("_GradientButton"),
            onPressed: () {
              setState(() {
                _angle += pi / 2;
              });
            },
          ),

          /// 通过继承 StatefulWidget 的方式实现自定义的有状态组件
          _RotationAnimationBox(
            angle: _angle,
            duration: 500,
            child: const MyText("_RotationAnimationBox"),
          ),
        ],
      ),
    );
  }
}

/// 通过继承已有 Widget 的方式实现自定义 Widget
/// 下面的示例用于实现一个自定义的 Text 组件,其默认文字大小 24,颜色白色,无下划线
class _MyText extends Text {
  const _MyText(super.data) : super(
    style: const TextStyle(
      fontSize: 24.0,
      color: Colors.white,
      decoration: TextDecoration.none,
    ),
  );
}

/// 通过继承 StatelessWidget 的方式实现自定义的无状态组件
/// 下面的示例用于实现一个支持渐变背景色的按钮组件
class _GradientButton extends StatelessWidget {
  const _GradientButton({
    this.colors,        /// 渐变色
    this.borderRadius,
    this.width,
    this.height,
    required this.onPressed,
    required this.child,
  });
  final List<Color>? colors;
  final BorderRadius? borderRadius;
  final double? width;
  final double? height;
  final GestureTapCallback? onPressed;
  final Widget child;

  @override
  Widget build(BuildContext context) {
    ThemeData theme = Theme.of(context);
    List<Color> _colors = colors ?? [theme.primaryColor];
    return DecoratedBox(
      decoration: BoxDecoration(
        gradient: LinearGradient(colors: _colors),
        borderRadius: borderRadius,
      ),
      child: Material(
        type: MaterialType.transparency,
        child: InkWell(
          borderRadius: borderRadius,
          onTap: onPressed,
          child: ConstrainedBox(
            constraints: BoxConstraints.tightFor(height: height, width: width),
            child: Center(
              child: Padding(
                padding: const EdgeInsets.all(8.0),
                child: DefaultTextStyle(
                  style: const TextStyle(fontWeight: FontWeight.bold),
                  child: child,
                ),
              ),
            ),
          ),
        ),
      ),
    );
  }
}

/// 通过继承 StatefulWidget 的方式实现自定义的有状态组件
/// 下面的示例用于实现一个支持旋转动画的组件
class _RotationAnimationBox extends StatefulWidget {
  const _RotationAnimationBox({
    this.angle = .0,      /// 目标旋转弧度
    this.duration = 200,  /// 从当前弧度旋转到目标旋转弧度的动画时长
    required this.child
  });
  final double angle;
  final int duration;
  final Widget child;

  @override
  _RotationAnimationBoxState createState() => _RotationAnimationBoxState();
}
class _RotationAnimationBoxState extends State<_RotationAnimationBox> with SingleTickerProviderStateMixin {
  late AnimationController _controller;

  @override
  void initState() {
    super.initState();
    _controller = AnimationController(vsync: this, lowerBound: -double.infinity, upperBound: double.infinity);
    _controller.value = widget.angle;
  }

  @override
  void dispose() {
    _controller.dispose();
    super.dispose();
  }

  @override
  Widget build(BuildContext context) {
    return RotationTransition(
      /// turns 的意思是旋转的圈数,也就是说 turns = angle / (2 * pi)
      turns: _controller,
      child: widget.child,
    );
  }

  /// 当父组件重绘时
  @override
  void didUpdateWidget(_RotationAnimationBox oldWidget) {
    super.didUpdateWidget(oldWidget);

    /// 如果 _RotationAnimationBox 的 angle 发生变化了则启动动画
    if (oldWidget.angle != widget.angle) {
      /// _controller 会从当前值动画到 widget.angle / (2 * pi)
      _controller.animateTo(widget.angle / (2 * pi), duration: Duration(milliseconds: widget.duration), curve: Curves.ease);
    }
  }
}

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

标签:widget,const,自定义,Widget,child,angle
From: https://www.cnblogs.com/webabcd/p/flutter_lib_widget_custom_custom_widget.html

相关文章

  • 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方法进......
  • djangoadmin后台搜索结果筛选自定义模版
    django-admin对搜索结果进行自定义统计,可参考代码如下:defchangelist_view(self,request,extra_context=None):#cur1_time=datetime.now()data_dict={}value=request.GET.get('q',"")bill_cycle=request.GET.get('bi......
  • Linux配置添加自定义shell脚本需要的PATH
    Linux添加自定义shell脚本记录下,便于之后复习使用。1.确定一个目录e.g.#到达用户目录cd~#创建一个bin文件夹来放脚本文件mkdirbincd./binpwd得到的是/root/bin2.把这个路径放到PATH中cd~#可以用ls-a看一看有没有.branrc文件vim~/.bashrc#编辑最后加入......