ZFJTreeViewKit-高效简单扩展性极强且无限插入子节点的树状列表视图

ZFJTreeViewKit

前言

前几天在写Python的一个Demo的时候,用到一个控件Treeview();

tree = ttk.Treeview(win)
tree.pack()

想到在iOS中没有这个控件,网上看了一下有很多treeView的相关控件,但是都扩展性有点差,有的专为聊天列表设计,有的专为选择列表设计;还有笔者上个月在找工作有个面试官也问到了这个事情,我当时只把方案思路说了一个,因此我做了ZFJTreeViewKit!

ZFJTreeViewKit一款高效简单扩展性极强而且可以无限插入子节点的树状列表视图。

项目地址 https://github.com/zfjsyqk/ZFJTreeViewKit.git
Demo地址 https://gitee.com/zfj1128/ZFJTreeView.git
博客地址 https://zfj1128.blog.csdn.net/article/details/94393642

软件架构

具体结构图如下:


image

项目的主要类就是ZFJTreeView和ZFJNodeModel,在ZFJTreeView中我们主要封装了一个ZFJTreeView的公共方法,需要主要说明的就是我们的节点数据模型了ZFJNodeModel,其结构图如下:


image

ZFJNodeModel包含了节点的关键信息,最重要的就是nodeKey了,这里是ZFJTreeViewKit自动生成,并不需要用户管理和操心,为了方便用户使用和ZFJTreeViewKit的扩展,笔者在这里提供了@property (nonatomic,strong) NSObject *sourceModel;自定义数据源模型,用户可以传入自己自定义的数据模型,方便在CELL中使用;
说到CELL,ZFJTreeViewKit不提供CELL的样式,需要用户自己定义和设计CELL,这也给用户提供了极高的自由度,方便用户根据需求来设计自己的样式功能,但是用户一定要在@property (nonatomic, copy) Class nodeCellCls;中注册自己的CELL。

安装教程

  1. pod 'ZFJTreeViewKit'
  2. pod install
  3. 导入头文件#import "ZFJTreeViewKit.h"

使用说明

创建ZFJTreeView

本控件的主要视图View就是ZFJTreeView,所以用户在使用的时候直接创建ZFJTreeView类就行,示例代码如下:

- (ZFJTreeView *)treeView{
    if(_treeView == nil){
    ZFJTreeViewConfig *model = [[ZFJTreeViewConfig alloc] init];
    model.separatorStyle = UITableViewCellSeparatorStyleNone;
    model.selectionStyle = UITableViewCellSelectionStyleNone;

    _treeView = [[ZFJTreeView alloc] initWithFrame:self.view.bounds config:model];
    _treeView.delegate = self;
    }
return _treeView;
}
创建节点

如果不考虑性能问题,ZFJTreeViewKit是可以无限制添加子节点的,笔者这里面展示了添加十级节点的示例,每个节点都有一个自定义数据模型,比如创建一级节点:

    MyNodeModel *myModel = [[MyNodeModel alloc] init];
    myModel.title = @"自定义Title";
    
    #pragma mark - 添加一级节点
    for (int i = 0; i<25; i++) {
        ZFJNodeModel *model_f1 = [[ZFJNodeModel alloc] initWithParentNodeModel:nil];
        model_f1.nodeName = [NSString stringWithFormat:@"一级节点%d楼",I];
        model_f1.height = 55;//节点高度
        model_f1.sourceModel = myModel;
        model_f1.nodeCellCls = [MyNodeViewCell class];
        [self.treeView insertNode:model_f1 completed:^(ZFJError * _Nonnull error) {
            NSLog(@"%@",error.message);
        }];
        [self.dataArr_1 addObject:model_f1];
    }

从上面的代码可以看到,我们需要设置CELL的高度、自定义数据模型Model(如果有)、注册自定义CELL(必须要有)然后就是调用插入事件了!
效果如图:


image

接着我们添加二级节点:

    #pragma mark - 添加二级节点
    for (ZFJNodeModel *model_f1 in self.dataArr_1) {
        for (int i = 0; i<2; i++) {
            ZFJNodeModel *model_f2 = [[ZFJNodeModel alloc] initWithParentNodeModel:model_f1];
            model_f2.nodeName = [NSString stringWithFormat:@"二级节点%d楼",I];
            model_f2.height = 55;//节点高度
            //model_f2.sourceModel = myModel;
            model_f2.nodeCellCls = [MyNodeViewCell class];
            [self.treeView insertNode:model_f2 completed:^(ZFJError * _Nonnull error) {
                NSLog(@"%@",error.message);
            }];
            [self.dataArr_2 addObject:model_f2];
        }
    }

从上面的代码我们可以看出,我把所有的一级节点存到数组self.dataArr_1中,然后给所有的一级节点都添加了两个二级节点,所有的二级节点都设置了父节点ZFJNodeModel *model_f2 = [[ZFJNodeModel alloc] initWithParentNodeModel:model_f1];,效果图如下:

image

继续添加三级节点:

#pragma mark - 添加三级节点
    for (ZFJNodeModel *model_f2 in self.dataArr_2) {
        for (int i = 0; i<2; i++) {
            ZFJNodeModel *model_f3 = [[ZFJNodeModel alloc] initWithParentNodeModel:model_f2];
            model_f3.nodeName = [NSString stringWithFormat:@"三级节点%d楼",I];
            model_f3.height = 55;//节点高度
            model_f3.sourceModel = myModel;
            model_f3.nodeCellCls = [MyNodeViewCell class];
            [self.treeView insertNode:model_f3 completed:^(ZFJError * _Nonnull error) {
                NSLog(@"%@",error.message);
            }];
            [self.dataArr_3 addObject:model_f3];
        }
    }

同样的道理,需要设置父节点ZFJNodeModel *model_f3 = [[ZFJNodeModel alloc] initWithParentNodeModel:model_f2];,效果图如下:

image

同理,我们可以一直往下添加子节点,为某个节点动态添加子节点,但是一定要设置对父节点?。?!

公共方法

在ZFJTreeView的公共方法里面,我提供了对常用的点击、展开、折叠、插入、删除、查询等操作,具体如下:

//
//  ZFJTreeView.h
//  ZFJTreeViewDemo
//
//  Created by 张福杰 on 2019/6/27.
//  Copyright ? 2019 张福杰. All rights reserved.
//

#import <UIKit/UIKit.h>

@class ZFJTreeView, ZFJTreeViewConfig, ZFJNodeModel, ZFJError;

NS_ASSUME_NONNULL_BEGIN

@protocol ZFJTreeViewDelegate <NSObject>

/**
 节点点击事件代理

 @param listView ZFJTreeView
 @param model 节点model
 @param indexPath indexPath
 */
- (void)treeListView:(ZFJTreeView *)listView didSelectNodeModel:(ZFJNodeModel *)model indexPath:(NSIndexPath *)indexPath;

@end

@interface ZFJTreeView : UIView

/**
 初始化方法

 @param frame frame
 @param config ZFJTreeView配置文件
 @return self
 */
- (instancetype)initWithFrame:(CGRect)frame config:(ZFJTreeViewConfig *)config;

/**
 代理方法
 */
@property (nonatomic, weak) id<ZFJTreeViewDelegate> delegate;

/**
 ZFJTreeView头部视图
 */
@property (nonatomic,strong) UIView *headerView;

/**
 ZFJTreeView尾部视图
 */
@property (nonatomic,strong) UIView *footerView;

/**
 插入某个节点

 @param model 节点model
 */
- (void)insertNode:(ZFJNodeModel *)model completed:(void(^)(ZFJError *error))completed;

/**
 删除某个节点(删除父节点,则子节点全部删除)

 @param model 节点model
 */
- (void)deleteNode:(ZFJNodeModel *)model completed:(void(^)(ZFJError *error))completed;

/**
 展开/折叠某个节点的所以子节点

 @param model 节点model(需要展开/折叠的父节点)
 @param completed 错误信息回调
 */
- (void)expandAllNodes:(ZFJNodeModel *)model completed:(void(^)(ZFJError *error))completed;

/**
 展开/折叠某个节点的下一级子节点

 @param model 节点model(需要展开/折叠的父节点)
 @param completed 错误信息回调
 */
- (void)expandChildNodes:(ZFJNodeModel *)model completed:(void(^)(ZFJError *error))completed;

/**
 展开/折叠全部节点

 @param expand YES:全部展开||NO:全部关闭
 */
- (void)expandAllNodes:(BOOL)expand;

/**
 通过节点Key获取节点model

 @param nodeKey 节点Key
 @return 节点model
 */
- (ZFJNodeModel *)getNodeModelWithNodeKey:(NSString *)nodeKey;

/**
 获取子节点是否全部展开(用于设置Cell样式)
 
 @param nodeModel 节点model
 @return YES:全部展开 || NO:没有全部展开
 */
- (BOOL)getchildNodesExpandState:(ZFJNodeModel *)nodeModel;

/**
 获取节点在父节点中的位置

 @param nodeModel 当前节点model
 @return 在父节点中的下标(-1 未找到)
 */
- (NSInteger)getIndexFromParentNode:(ZFJNodeModel *)nodeModel;

#pragma mark ----------NS_UNAVAILABLE----------

+ (instancetype)new NS_UNAVAILABLE;

- (instancetype)init NS_UNAVAILABLE;

- (instancetype)initWithCoder:(NSCoder *)coder NS_UNAVAILABLE;

- (instancetype)initWithFrame:(CGRect)frame NS_UNAVAILABLE;

- (instancetype)initWithFrame:(CGRect)frame style:(UITableViewStyle)style NS_UNAVAILABLE;

@end

NS_ASSUME_NONNULL_END

从上面的代码可以看到,ZFJTreeView高度灵活自由且扩展性极强,用户还可以设置ZFJTreeView的头部视图和尾部视图,如下:

    self.treeView.backgroundColor = [UIColor groupTableViewBackgroundColor];
    [self.view addSubview:self.treeView];
    
    UIView *headerView = [[UIView alloc] init];
    headerView.frame = CGRectMake(0, 0, ScreenWidth, 100);
    headerView.backgroundColor = [UIColor yellowColor];
    self.treeView.headerView = headerView;
整体效果

这是添加了十级子节点的效果图:


image

使用场景

这种类型的控件使用场景是非常多的,比如地址多级选择、抖音评论列表还有一些社区评论列表等等!

image

--- 更多UI效果等你来设计?。?!---

结束语

这里Demo的样式效果比较丑,大家将就一下吧??????,但是代码绝对是高效而且工工整整注释详细的!??????
欢迎各位大神提出宝贵的意见和建议,也欢迎大家进群交流365152048!

最后编辑于
?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,100评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,308评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事?!?“怎么了?”我有些...
    开封第一讲书人阅读 159,718评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,275评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,376评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,454评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,464评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,248评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,686评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,974评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,150评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,817评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,484评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,140评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,374评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,012评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,041评论 2 351