Flutter 封装 —— 树形组件 NTree
一、需求来源
最近开发遇到一种树形组件功能,周末花时间琢磨了一下,基本实现了要求: 层级缩进、折叠展开、选择功能;缩进为30,效果图下:

二、使用示例
import 'package:flutter/material.dart';
import 'package:flutter_templet_project/basicWidget/NTree/NTree.dart';
class NTreeDemo extends StatefulWidget {
  NTreeDemo({ Key? key, this.title}) : super(key: key);
  final String? title;
  @override
  _NTreeDemoState createState() => _NTreeDemoState();
}
class _NTreeDemoState extends State<NTreeDemo> {
  final _list = [
    NTreeNodeModel(
      name:'1 一级菜单',
      isExpand: true,//是否展开子项
      enabled: false,//是否可以响应事件
      items:[
        NTreeNodeModel(
          name:'1.1 二级菜单',
          isExpand: true,
          items:[
            NTreeNodeModel(
              name:'1.1.1 三级菜单',
              isExpand: true,
              items: [
                NTreeNodeModel(
                  name:'1.1.1.1 四级菜单',
                  isExpand: true,
                ),
              ]
            ),
          ]
        ),
        NTreeNodeModel(
          name:'1.2 二级菜单',
          isExpand: true,
        ),
      ]
    ),
    NTreeNodeModel(
      name:'2 一级菜单',
      // isExpand: true,
        items:[
        NTreeNodeModel(
          name:'2.1 二级菜单',
          // isExpand: true,
        ),
        NTreeNodeModel(
          name:'2.2 二级菜单',
          isExpand: false,
          items:[
            NTreeNodeModel(
              name:'2.2.1 三级菜单',
              // isExpand: true,
            ),
          ]
        ),
      ]
    ),
  ];
  @override
  void initState() {
    super.initState();
  }
   @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text(widget.title ?? "$widget"),
        actions: ['done',].map((e) => TextButton(
          child: Text(e,
            style: TextStyle(color: Colors.white),
          ),
          onPressed: onPressed,)
        ).toList(),
      ),
      body: _buildBody(),
    );
  }
  _buildBody() {
    dynamic arguments = ModalRoute.of(context)!.settings.arguments;
    return CustomScrollView(
      slivers: [
        Text(arguments.toString()),
        NTree(
          list: _list,
        ),
      ].map((e) => SliverToBoxAdapter(child: e,)).toList(),
    );
  }
  onPressed(){
  }
}
三、源码
import 'package:flutter/material.dart';
import 'package:flutter_templet_project/basicWidget/enhance/enhance_expansion/enhance_expansion_tile.dart';
import 'package:flutter_templet_project/extension/color_ext.dart';
class NTree extends StatefulWidget {
  NTree({
    Key? key,
    required this.list,
    this.color = Colors.black87,
    this.iconColor = Colors.blueAccent,
    this.indent = 30,
  }) : super(key: key);
  /// 数据源
  List<NTreeNodeModel> list;
  /// 标题颜色
  Color color;
  /// 文字颜色
  Color iconColor;
  /// 层级缩进
  double indent;
  @override
  _NTreeState createState() => _NTreeState();
}
class _NTreeState extends State<NTree> {
  @override
  void initState() {
    super.initState();
  }
   @override
  Widget build(BuildContext context) {
    return Column(
      children: widget.list.map((e) {
        return Column(
          children: [
            _buildNode(
              e: e,
              color: widget.color,
              iconColor: widget.iconColor,
            ),
            if(e.isExpand)Padding(
              padding: EdgeInsets.only(left: widget.indent),
              child: NTree(
                color: widget.color,
                list: e.items,
              ),
            ),
          ],
        );
      }).toList(),
    );
  }
  Widget _buildNode({
    required NTreeNodeModel e,
    Color? color,
    Color? iconColor,
  }){
    final leadingIcon = e.isSelected ? Icon(Icons.check_box, color: iconColor,)
        : Icon(Icons.check_box_outline_blank, color: iconColor,);
    final trailing = e.items.isEmpty ? SizedBox() :
    (e.isExpand ? Icon(Icons.keyboard_arrow_down, color: iconColor,)
        : Icon(Icons.keyboard_arrow_right, color: iconColor,));
    final leading = IconButton(
      onPressed: () {
        e.isSelected = !e.isSelected;
        recursion(e: e, cb: (item) => item.isSelected = e.isSelected,);
        setState((){});
      },
      icon: leadingIcon,
    );
    return Theme(
      data: ThemeData(
        dividerColor: Colors.transparent,
      ),
      child: EnhanceExpansionTile(
        backgroundColor: ColorExt.random,
        // tilePadding: EdgeInsets.symmetric(horizontal: 100),
        // leading: leading,
        leading: leading,
        trailing: trailing,
        title: Text("${e.name}",
          style: TextStyle(
            color: color,
          ),
        ),
        // title: Text("${e.name}"),
        initiallyExpanded: e.isExpand,
        onExpansionChanged: (val){
          e.isExpand = val;
          e.onClick?.call(e);
          setState((){});
        },
        header: (onTap) => InkWell(
          onTap: (){
            onTap();
            e.isExpand = !e.isExpand;
            setState((){});
          },
          child: Container(
            padding: EdgeInsets.only(
              left: 0,
              top: 4,
              bottom: 4,
              right: 16,
            ),
            child: Row(
              children: [
                leading,
                Expanded(
                  child: Text("${e.name}",
                    style: TextStyle(
                      color: color,
                    ),
                  ),
                ),
                trailing,
              ],
            ),
          ),
        ),
      ),
    );
  }
  recursion({
     required NTreeNodeModel e,
     required void Function(NTreeNodeModel e) cb
   }) {
     cb(e);
     debugPrint("item:${e.name} ${e.isSelected}");
     e.items.forEach((item) {
       recursion(e: item, cb: cb);
    });
  }
}
class NTreeNodeModel{
  NTreeNodeModel({
    this.name,
    this.isExpand = false,
    this.isSelected = false,
    this.enabled = true,
    this.onClick,
    this.data,
    this.items = const [],
  });
  /// 标题
  String? name;
  /// 是否展开
  bool isExpand;
  /// 是否已选择
  bool isSelected;
  ///
  bool enabled;
  /// 模型
  dynamic data;
  /// 子元素
  List<NTreeNodeModel> items;
  /// 点击事件
  void Function(NTreeNodeModel e)? onClick;
  /// 遍历子树
  recursion(void Function(NTreeNodeModel e)? cb) {
    cb?.call(this);
    debugPrint("item:$name $isSelected");
    items.forEach((item) {
      recursion(cb);
    });
  }
}
总结
1、局部折叠展开不会引起外部重绘,性能比较优;
2、实现简单,任何人都可以在此基础上做二次开发,毕竟需求场景各有不同;
转载自:https://juejin.cn/post/7255561917682974779




