文章分享至个人我的技术博客: https://cainluo.github.io/15009611814095.htmlhtml
最近有些变懒了, 学习的步伐放慢了不少, 估计玩懒了身子, 连博客都少写了咯.git
以前有个老铁问我, 怎么去封装一个低耦合可服用的TableViewController
, 那时候没多说啥, 直接把我封装的框架丢给他了, 但他仍是有不少乱七八糟的问题问我, 干脆直接写一篇博文当成教程好了.github
转载声明:如须要转载该文章, 请联系做者, 而且注明出处, 以及不能擅自修改本文.数组
因为这里我是用MVVM
模式来封装的, 这里我就简单点, 通俗点, 易懂点的说说MVVM
吧.微信
不少老铁都习惯于用MVC
, 虽然有听过MVVM
, 但也没去咋了解, 其实MVVM
没有那么复杂, 传统的MVC
是有Model
, Views
, Controller
, 而MVVM
只是在这个得基础上加了一个ViewModel
, 而且弱化了Controller
的职能.框架
MVC:
Model
,Views
,Controller
MVVM:Model
,Views
,ViewModel
学习
那么弱化了的Controller
就负责做为一个粘合剂, 像乐高积木同样, 把Model
, Views
, ViewModel
组装在一块儿, 成为一个模块, 而Model
, Views
, ViewModel
分别又是独立的个体, 谁都不会离不开谁.优化
大概就酱紫吧, 若是有更好的说法, 欢迎各位老铁补充补充哈~~ui
这里说一下, CLTableViewController
是我本身封装的TableViewController
, 因为我比较懒, 因此里面直接集成了MJRefresh
, 各位老铁随意喷哈.atom
这里说一下思路, 因为TableView
还有TableViewDataSource
, TableViewDelegate
, 因此这里咱们须要把两个模块分开, 这样子就不会形成代码臃肿的状况啦.
注意: 这里不包括各位的业务逻辑哈
刚刚其实还说漏了一个, 除去TableViewDataSource
, TableViewDelegate
, 期是还有一个ViewModel
层, 这个是用来请求数据的.
如今咱们先来看TableViewDataSource
:
#import <Foundation/Foundation.h>
#import "CLTableViewBaseModel.h"
@interface CLTableViewDataSource : NSObject <UITableViewDataSource>
@property (nonatomic, strong, readonly) CLTableViewBaseModel *cl_viewModel;
- (instancetype)initTableViewDataSourceWithViewModel:(CLTableViewBaseModel *)viewModel;
@end
复制代码
#import "CLTableViewDataSource.h"
@interface CLTableViewDataSource ()
@property (nonatomic, strong, readwrite) CLTableViewBaseModel *cl_viewModel;
@end
@implementation CLTableViewDataSource
- (instancetype)initTableViewDataSourceWithViewModel:(CLTableViewBaseModel *)viewModel {
self = [super init];
if (self) {
self.cl_viewModel = viewModel;
}
return self;
}
- (NSInteger)tableView:(UITableView *)tableView
numberOfRowsInSection:(NSInteger)section {
return self.cl_viewModel.cl_dataSource.count;
}
- (UITableViewCell *)tableView:(UITableView *)tableView
cellForRowAtIndexPath:(NSIndexPath *)indexPath {
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"UITableViewCell"];
if (!cell) {
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault
reuseIdentifier:@"UITableViewCell"];
}
return cell;
}
@end
复制代码
在.h
文件里, 只提供了一个供给指定ViewModel
的初始化方法, 内部的实现, 所返回的数据源数量也是指定ViewModel
的数组个数, 默认返回一个系统的UITableViewCell
, 这样子就行了.
关于TableViewDelegate
更可能是采用系统的特性, 这里就没写什么内部实现了:
#import <Foundation/Foundation.h>
#import <UIKit/UIKit.h>
#import "CLTableViewBaseModel.h"
@interface CLTableViewDelegate : NSObject <UITableViewDelegate>
@property (nonatomic, strong, readonly) CLTableViewBaseModel *cl_viewModel;
- (instancetype)initTableViewDelegateWithViewModel:(CLTableViewBaseModel *)viewModel;
@end
复制代码
#import "CLTableViewDelegate.h"
@interface CLTableViewDelegate ()
@property (nonatomic, strong, readwrite) CLTableViewBaseModel *cl_viewModel;
@end
@implementation CLTableViewDelegate
- (instancetype)initTableViewDelegateWithViewModel:(CLTableViewBaseModel *)viewModel {
self = [super init];
if (self) {
self.cl_viewModel = viewModel;
}
return self;
}
@end
复制代码
只定义了一个指定ViewModel
的初始化方法.
关于ViewModel
层, 这里我提供了三个方法, 两个属性:
#import <Foundation/Foundation.h>
#import "CLTableViewController.h"
@interface CLTableViewBaseModel : NSObject
@property (nonatomic, strong) NSMutableArray *cl_dataSource;
@property (nonatomic, weak, readonly) CLTableViewController *cl_tableViewController;
- (instancetype)initTableViewBaseModelWithController:(CLTableViewController *)viewController;
/** 经过HTTP请求数据 */
- (void)cl_tableViewHTTPRequest;
/** 配置TableView每一条Cell所显示的分割线 */
- (void)cl_configTableViewWithDataSource;
@end
复制代码
#import "CLTableViewBaseModel.h"
@interface CLTableViewBaseModel()
@property (nonatomic, weak, readwrite) CLTableViewController *cl_tableViewController;
@end
@implementation CLTableViewBaseModel
- (instancetype)initTableViewBaseModelWithController:(CLTableViewController *)viewController {
self = [super init];
if (self) {
self.cl_tableViewController = viewController;
}
return self;
}
- (NSMutableArray *)cl_dataSource {
if (!_cl_dataSource) {
_cl_dataSource = [NSMutableArray array];
}
return _cl_dataSource;
}
- (void)cl_tableViewHTTPRequest {
}
- (void)cl_configTableViewWithDataSource {
if (self.cl_dataSource.count > 0) {
self.cl_tableViewController.cl_tableView.separatorStyle = UITableViewCellSeparatorStyleSingleLine;
}
}
@end
复制代码
这样子就大功告成了, 接下来就是组装起它们就能够了.
刚刚咱们已经把三个模块写好了, 如今就开始组装:
#import "CLViewController.h"
NS_ASSUME_NONNULL_BEGIN
@interface CLTableViewController : CLViewController
@property (nonatomic, strong, readonly) UITableView *cl_tableView;
/** 初始化CLTableViewController @param style UITableViewStyle, 默认是UITableViewStylePlain @return CLTableViewController */
- (instancetype)initTableViewControllerWithStyle:(UITableViewStyle)style;
- (void)cl_removeRefresh;
- (void)cl_removeHeaderRefresh;
- (void)cl_removeFooterRefresh;
/** 下拉刷新方法/上拉加载方法 */
- (void)cl_dropDownRefresh;
- (void)cl_pullUpRefresh;
/** 开始执行下拉操做/结束下拉操做 */
- (void)cl_dropDownBeginRefresh;
- (void)cl_dropDownEndRefresh;
/** 开始执行上拉操做/结束上拉操做 */
- (void)cl_pullUpBeginRefresh;
- (void)cl_pullUpEndRefresh;
- (void)cl_setTableViewDelegate:(_Nullable id <UITableViewDelegate>)delegate
dataSource:(_Nullable id <UITableViewDataSource>)dataSource;
@end
NS_ASSUME_NONNULL_END
复制代码
#import "CLTableViewController.h"
#import "MJRefresh.h"
#import "CLTableViewDelegate.h"
#import "CLTableViewBaseModel.h"
@interface CLTableViewController ()
@property (nonatomic, assign) UITableViewStyle tableViewStyle;
@property (nonatomic, strong, readwrite) UITableView *cl_tableView;
@property (nonatomic, strong) CLTableViewDelegate *cl_tableViewDelegate;
@property (nonatomic, strong) CLTableViewBaseModel *cl_ableViewBaseModel;
@end
@implementation CLTableViewController
- (instancetype)initTableViewControllerWithStyle:(UITableViewStyle)style {
self = [super init];
if (self) {
[self setTableViewStyle:style];
}
return self;
}
#pragma mark - View Did Load
- (void)viewDidLoad {
[super viewDidLoad];
self.view.opaque = YES;
self.automaticallyAdjustsScrollViewInsets = NO;
self.view.backgroundColor = [UIColor whiteColor];
[self cl_addRefresh];
}
- (UITableView *)cl_tableView {
if (!_cl_tableView) {
_cl_tableView = [[UITableView alloc] initWithFrame:self.view.frame
style:self.tableViewStyle];
if (self.tableViewStyle == UITableViewStylePlain) {
_cl_tableView.separatorStyle = UITableViewCellSeparatorStyleNone;
}
_cl_tableView.opaque = YES;
}
return _cl_tableView;
}
- (void)cl_setTableViewDelegate:(id<UITableViewDelegate>)delegate
dataSource:(id<UITableViewDataSource>)dataSource {
self.cl_tableView.delegate = delegate;
self.cl_tableView.dataSource = dataSource;
}
#pragma mark - Table View Delegate
- (CLTableViewDelegate *)cl_tableViewDelegate {
if (!_cl_tableViewDelegate) {
_cl_tableViewDelegate = [[CLTableViewDelegate alloc] initTableViewDelegateWithViewModel:self.cl_ableViewBaseModel];
}
return _cl_tableViewDelegate;
}
#pragma mark - Table View Base Model
- (CLTableViewBaseModel *)cl_ableViewBaseModel {
if (!_cl_ableViewBaseModel) {
_cl_ableViewBaseModel = [[CLTableViewBaseModel alloc] initTableViewBaseModelWithController:self];
}
return _cl_ableViewBaseModel;
}
#pragma mark - Refresh
- (void)cl_addRefresh {
MJRefreshNormalHeader *header = [MJRefreshNormalHeader headerWithRefreshingBlock:^{
[self cl_dropDownRefresh];
}];
self.cl_tableView.mj_header = header;
MJRefreshBackNormalFooter *refreshFooter = [MJRefreshBackNormalFooter footerWithRefreshingBlock:^{
[self cl_pullUpEndRefresh];
}];
self.cl_tableView.mj_footer = refreshFooter;
[self.view addSubview:self.cl_tableView];
}
- (void)cl_dropDownRefresh {}
- (void)cl_pullUpRefresh {}
- (void)cl_dropDownBeginRefresh {
[self.cl_tableView.mj_header beginRefreshing];
}
- (void)cl_dropDownEndRefresh {
[self.cl_tableView.mj_header endRefreshing];
}
- (void)cl_pullUpBeginRefresh {
[self.cl_tableView.mj_footer beginRefreshing];
}
- (void)cl_pullUpEndRefresh {
[self.cl_tableView.mj_footer endRefreshing];
}
- (void)cl_removeRefresh {
self.cl_tableView.mj_header = nil;
self.cl_tableView.mj_footer = nil;
}
- (void)cl_removeHeaderRefresh {
self.cl_tableView.mj_header = nil;
}
- (void)cl_removeFooterRefresh {
self.cl_tableView.mj_footer = nil;
}
@end
复制代码
完成了!!! 如今封装好了一个属于咱们本身的TableViewController
啦.
因为这里封装的比较简单, 若是有更好建议的老铁能够私下留言一下, 我也会跟进优化, 方便你们, 固然我本身封装的框架还有其余, 可是太多了很差一一写出来, 喜欢的话, 能够Star
一下个人CLFramework, 若是有老铁想一块儿维护一下这个开源库的话, 也能够和我说说.
关于详细的用法, 你们能够在里面找到SimpleProject
工程, 里面有个人详细用法.