我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Flutter重构属性透传及函数透传使用示例

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Flutter重构属性透传及函数透传使用示例

一、来源

今天在研究 flutter 相册库 wechat_assets_picker 遇到一个问题:(我需要在第三方库基础上封装一个组件,供项目内部调用,组件内封装一些公共逻辑。)但是 AssetPicker.pickAssets 的属性太多了,一个个传递实在太麻烦,就想是否有 vue 中那种数据透传的解决方法呢(已知 flutter 中目前不支持这种属性透传)?苦苦思索5分钟之后,灵光一闪:

函数透传

/// pickAssets 方法源码:
static Future<List<AssetEntity>?> pickAssets(
  BuildContext context, {
  List<AssetEntity>? selectedAssets,
  int maxAssets = 9,
  int pageSize = 80,
  int gridThumbSize = Constants.defaultGridThumbSize,
  int pathThumbSize = 80,
  int gridCount = 4,
  RequestType requestType = RequestType.image,
  List<int>? previewThumbSize,
  SpecialPickerType? specialPickerType,
  Color? themeColor,
  ThemeData? pickerTheme,
  SortPathDelegate<AssetPathEntity>? sortPathDelegate,
  AssetsPickerTextDelegate? textDelegate,
  FilterOptionGroup? filterOptions,
  WidgetBuilder? specialItemBuilder,
  IndicatorBuilder? loadingIndicatorBuilder,
  SpecialItemPosition specialItemPosition = SpecialItemPosition.none,
  bool allowSpecialItemWhenEmpty = false,
  AssetSelectPredicate<AssetEntity>? selectPredicate,
  bool? shouldRevertGrid,
  bool useRootNavigator = true,
  Curve routeCurve = Curves.easeIn,
  Duration routeDuration = const Duration(milliseconds: 300),
}) async {
...

二、 WechatPhotoPicker 使用示例

class WechatPhotoPickerDemo extends StatefulWidget {
  WechatPhotoPickerDemo({ Key? key, this.title}) : super(key: key);
  final String? title;
  @override
  _WechatPhotoPickerDemoState createState() => _WechatPhotoPickerDemoState();
}
class _WechatPhotoPickerDemoState extends State<WechatPhotoPickerDemo> {
  int maxCount = 9;
  List<AssetEntity> entitys = [];
  GlobalKey<WechatPhotoPickerState> _globalKey = GlobalKey();
  @override
  Widget build(BuildContext context) {
    return Scaffold(
        appBar: AppBar(
          title: Text(widget.title ?? "$widget"),
          actions: ['选择',].map((e) => TextButton(
            child: Text(e,
              style: TextStyle(color: Colors.white),
            ),
            onPressed: onPicker,
          )).toList(),
        ),
        body: Column(
          children: [
            WechatPhotoPicker(
              key: _globalKey,
              selectedAssets: entitys,
              onChanged: (List<AssetEntity> selectedAssets) {
                print("selectedAssets: ${selectedAssets.length}");
              },
              onPicker: () => AssetPicker.pickAssets(
                  context,
                  maxAssets: 8,
                  selectedAssets: entitys,
                ),
            )
          ],
        )
    );
  }
  onPicker() async {
    _globalKey.currentState?.onPicker();
    print(entitys.length);
  }
}

二、 WechatPhotoPicker 组件源码

/// 基于 wechat_assets_picker 的图片选择组件
class WechatPhotoPicker extends StatefulWidget {
  WechatPhotoPicker({
    Key? key,
    this.selectedAssets = const [],
    this.maxCount = 9,
    this.rowCount = 3,
    this.spacing = 10,
    this.decoration,
    this.addBuilder,
    required this.onChanged,
    this.onPicker,
  }) : super(key: key);
  /// 媒体对象数组
  List<AssetEntity> selectedAssets;
  /// 最大个数
  int maxCount;
  /// 每行元素个数
  int rowCount;
  /// 元素间距
  double spacing;
  /// 元素修饰器
  BoxDecoration? decoration;
  /// 添加图片
  Widget Function(BuildContext context, double itemWidth)? addBuilder;
  /// 确认选择回调函数
  void Function(List<AssetEntity> selectedAssets) onChanged;
  /// 解决flutter数据无法透传的问题(透传 AssetPicker.pickAssets 方法)
  Future<List<AssetEntity>?> Function()? onPicker;
  @override
  WechatPhotoPickerState createState() => WechatPhotoPickerState();
}
class WechatPhotoPickerState extends State<WechatPhotoPicker> {
  @override
  Widget build(BuildContext context) {
    return photoSection(
      selectedAssets: widget.selectedAssets,
      maxCount: widget.maxCount,
      rowCount: widget.rowCount,
      spacing: widget.spacing,
    );
  }
  photoSection({
    List<AssetEntity> selectedAssets = const [],
    int maxCount = 9,
    int rowCount = 3,
    double spacing = 10,
  }) {
    return LayoutBuilder(
        builder: (BuildContext context, BoxConstraints constraints){
          double itemWidth = ((constraints.maxWidth - spacing * (rowCount - 1))/rowCount).truncateToDouble();
          // print("itemWidth: $itemWidth");
          return Wrap(
              spacing: spacing,
              runSpacing: spacing,
              children: [
                ...selectedAssets.map((e) => Container(
                  clipBehavior: Clip.antiAlias,
                  decoration: widget.decoration ?? BoxDecoration(
                    // border: Border.all(width: 2),
                    borderRadius: BorderRadius.all(Radius.circular(4)),
                  ),
                  child: FadeInImage(
                    width: itemWidth,
                    height: itemWidth,
                    placeholder: AssetImage('images/img_placeholder.png'),
                    image: AssetEntityImageProvider(e, isOriginal: false),
                    fit: BoxFit.cover,
                  ),
                )).toList(),
                if (selectedAssets.length < maxCount)
                  InkWell(
                    onTap: () {
                      onPicker();
                    },
                    child: Container(
                      width: itemWidth,
                      height: itemWidth,
                      decoration: BoxDecoration(
                        color: Colors.black.withOpacity(0.1),
                        // border: Border.all(width: 1),
                        borderRadius: BorderRadius.all(Radius.circular(4)),
                      ),
                      child: widget.addBuilder != null ? widget.addBuilder!(context, itemWidth) : Icon(
                        Icons.add, 
                        size: itemWidth/3, 
                        color: Colors.black.withOpacity(0.3),
                      ),
                    ),
                  )
              ]
          );
        }
    );
  }
  onPicker() async {
    List<AssetEntity>? result = widget.onPicker != null ? await widget.onPicker!() :
    await AssetPicker.pickAssets(
      context,
      maxAssets: widget.maxCount,
      selectedAssets: widget.selectedAssets,
    );
    widget.selectedAssets = result ?? [];
    widget.onChanged(widget.selectedAssets);
    setState(() { });
  }
}

总结

1、onPicker 参数需要和调用方法搭配使用,即实现了函数透传,函数里的参数直接暴露给外部使用者,做二次定制开发;如果默认参数(可以适量添加)能够满足通用需求,则无需使用 onPicker 可选参数;

  onPicker: () => AssetPicker.pickAssets(
      context,
      maxAssets: 8,
      selectedAssets: entitys,
    ),
    List<AssetEntity>? result = widget.onPicker != null ? await widget.onPicker!() :
    await AssetPicker.pickAssets(
      context,
      maxAssets: widget.maxCount,
      selectedAssets: widget.selectedAssets,
    );

2、WechatPhotoPickerState,没有使用下换线(私有)实现是为了向外部暴露 State, 可以通过 GlobalKey 获取 State 实例对象,进而调用一些封装方法;达到更高的代码复用;

声明 GlobalKey:

 GlobalKey:<WechatPhotoPickerState> _globalKey = GlobalKey();

调用 State 方法:

_globalKey.currentState?.onPicker();

3、所有自定义组件原则上都要支持 key 属性,才是一个完整的组件 Widget;

无论是移动原生、前端 h5 或者 flutter 跨平台,各种数据透传的思想是相近,在一端取得突破之后,其他端基本都是平移实现,这些可以减少代码量又不损失功能,而且维护性和扩展性更优的实现方式就是代码重构的本质。

以上就是Flutter 重构属性透传及函数透传使用示例的详细内容,更多关于Flutter 重构属性函数透传的资料请关注编程网其它相关文章!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Flutter重构属性透传及函数透传使用示例

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Flutter重构属性透传及函数透传使用示例

这篇文章主要为大家介绍了Flutter重构属性透传及函数透传使用示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
2023-01-08

Flutter重构属性透传及函数透传如何使用

这篇文章主要介绍“Flutter重构属性透传及函数透传如何使用”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Flutter重构属性透传及函数透传如何使用”文章能帮助大家解决问题。一、来源今天在研究
2023-07-04

Vue3重构函数透传示例解析

这篇文章主要为大家介绍了Vue3重构函数透传示例解析
2023-02-17

编程热搜

  • Android:VolumeShaper
    VolumeShaper(支持版本改一下,minsdkversion:26,android8.0(api26)进一步学习对声音的编辑,可以让音频的声音有变化的播放 VolumeShaper.Configuration的三个参数 durati
    Android:VolumeShaper
  • Android崩溃异常捕获方法
    开发中最让人头疼的是应用突然爆炸,然后跳回到桌面。而且我们常常不知道这种状况会何时出现,在应用调试阶段还好,还可以通过调试工具的日志查看错误出现在哪里。但平时使用的时候给你闹崩溃,那你就欲哭无泪了。 那么今天主要讲一下如何去捕捉系统出现的U
    Android崩溃异常捕获方法
  • android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
    系统的设置–>电池–>使用情况中,统计的能耗的使用情况也是以power_profile.xml的value作为基础参数的1、我的手机中power_profile.xml的内容: HTC t328w代码如下:
    android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
  • Android SQLite数据库基本操作方法
    程序的最主要的功能在于对数据进行操作,通过对数据进行操作来实现某个功能。而数据库就是很重要的一个方面的,Android中内置了小巧轻便,功能却很强的一个数据库–SQLite数据库。那么就来看一下在Android程序中怎么去操作SQLite数
    Android SQLite数据库基本操作方法
  • ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
    工作的时候为了方便直接打开编辑文件,一些常用的软件或者文件我们会放在桌面,但是在ubuntu20.04下直接直接拖拽文件到桌面根本没有效果,在进入桌面后发现软件列表中的软件只能收藏到面板,无法复制到桌面使用,不知道为什么会这样,似乎并不是很
    ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
  • android获取当前手机号示例程序
    代码如下: public String getLocalNumber() { TelephonyManager tManager =
    android获取当前手机号示例程序
  • Android音视频开发(三)TextureView
    简介 TextureView与SurfaceView类似,可用于显示视频或OpenGL场景。 与SurfaceView的区别 SurfaceView不能使用变换和缩放等操作,不能叠加(Overlay)两个SurfaceView。 Textu
    Android音视频开发(三)TextureView
  • android获取屏幕高度和宽度的实现方法
    本文实例讲述了android获取屏幕高度和宽度的实现方法。分享给大家供大家参考。具体分析如下: 我们需要获取Android手机或Pad的屏幕的物理尺寸,以便于界面的设计或是其他功能的实现。下面就介绍讲一讲如何获取屏幕的物理尺寸 下面的代码即
    android获取屏幕高度和宽度的实现方法
  • Android自定义popupwindow实例代码
    先来看看效果图:一、布局
  • Android第一次实验
    一、实验原理 1.1实验目标 编程实现用户名与密码的存储与调用。 1.2实验要求 设计用户登录界面、登录成功界面、用户注册界面,用户注册时,将其用户名、密码保存到SharedPreference中,登录时输入用户名、密码,读取SharedP
    Android第一次实验

目录