#MVVMFramework ###Swift版本地址:https://github.com/lovemo/MVVMFramework-Swift ####本项目交流群:474292335 ####欢迎有兴趣的有好的想法的参与到项目中来
#####具体实现思路,请参看博客: ####博客:浅谈MVVM ####地址:点击链接进入
总结整理下一个快速开发MVVM框架(抛砖引玉),主要用于分离控制器中的代码,降低代码耦合程度,可以根据自己使用习惯调整代码。欢迎来喷,提issues。代码加入了cell自适应高度,自动缓存网络请求至sqlite数据库,更加高效的数据库存储库。
==== ####usage: CocoaPods:
pod 'SUIMVVMKit'
- 根据需要继承SMKBaseTableViewManger或SMKBaseCollectionViewManger扩展方法,如需显示多种cell,重写显示cell的数据源方法即可
- 在Controller中,初始化tableView或者collectionView,根据需要实现block代码,将自动根据传入的内容去展现数据。
- 利用xib自定义cell,在- (void)configure:customObj:indexPath:方法中根据模型Model内容配置cell展示的数据。
###部分protocol定义
@protocol SMKViewMangerProtocolDelegate <NSObject>
@optional
/**
* 设置Controller的子视图的管理者为self
*
* @param superView 一般指subView所在控制器的view
*/
- (void)smk_viewMangerWithSuperView:(UIView *)superView;
/**
* 设置subView的管理者为self
*
* @param subView 管理的subView
*/
- (void)smk_viewMangerWithSubView:(UIView *)subView;
/**
* 设置添加subView的事件
*
* @param view 管理的subView
* @param info 附带信息,用于区分调用
*/
- (void)smk_viewMangerWithHandleOfSubView:(UIView *)subView info:(NSString *)info;
/**
* 返回viewManger所管理的视图
*
* @return viewManger所管理的视图
*/
- (__kindof UIView *)smk_viewMangerOfSubView;
/**
* 得到其它viewManger所管理的subView,用于自己内部
*
* @param views 其它的subViews
*/
- (void)smk_viewMangerWithOtherSubViews:(NSDictionary *)viewInfos;
/**
* 需要重新布局subView时,更改subView的frame或者约束
*
* @param block 更新布局完成的block
*/
- (void)smk_viewMangerWithLayoutSubViews:(void (^)( ))updateBlock;
/**
* 使子视图更新到最新的布局约束或者frame
*/
- (void)smk_viewMangerWithUpdateLayoutSubViews;
@end
###配置代码集成展示tableView,cell自适应高度
/**
* tableView的一些初始化工作
*/
- (void)setupTableView
{
self.table.separatorStyle = UITableViewCellSelectionStyleNone;
__weak typeof(self) weakSelf = self;
__weak typeof(self.table) weakTable = self.table;
// 下拉刷新
self.table.mj_header= [MJRefreshNormalHeader headerWithRefreshingBlock:^{
[weakSelf.viewModel smk_viewModelWithGetDataSuccessHandler:^(NSArray *array) {
[weakTable reloadData];
}];
// 结束刷新
[weakTable.mj_header endRefreshing];
}];
[self.table.mj_header beginRefreshing];
// 设置自动切换透明度(在导航栏下面自动隐藏)
self.table.mj_header.automaticallyChangeAlpha = YES;
self.table.tableHander = [[TestViewDelegate alloc]initWithCellIdentifiers:@[MyCellIdentifier] didSelectBlock:^(NSIndexPath *indexPath, id item) {
SecondVC *vc = (SecondVC *)[UIViewController svv_viewControllerWithStoryBoardName:@"Main" identifier:@"SecondVCID"];
[weakSelf.navigationController pushViewController:vc animated:YES];
NSLog(@"click row : %@",@(indexPath.row)) ;
}];
[self.viewModel smk_viewModelWithGetDataSuccessHandler:^(NSArray *array){
[self.table.tableHander getItemsWithModelArray:^NSArray *{
return array;
} completion:^{
[self.table reloadData];
}];
}];
}
###配置代码集成展示collectionView
/**
* collectionView的一些初始化工作
*/
- (void)setupCollectionView
{
// 可用自定义UICollectionViewLayout,默认为UICollectionViewFlowLayout
self.collectionView.contentInset = UIEdgeInsetsMake(-64, 0, 0, 0);
self.collectionView.backgroundColor = [UIColor whiteColor];
self.collectionView.collectionHander = [[SMKBaseCollectionViewManger alloc]initWithCellIdentifier:MyCellIdentifier collectionViewLayout:nil cellItemSizeBlock:^CGSize{
return CGSizeMake(110, 120);
} cellItemMarginBlock:^UIEdgeInsets{
return UIEdgeInsetsMake(0, 20, 0, 20);
} didSelectBlock:^(NSIndexPath *indexPath, id item) {
NSString *strMsg = [NSString stringWithFormat:@"click row : %zd",indexPath.row];
[[[UIAlertView alloc] initWithTitle:@"提示"
message:strMsg
delegate:self
cancelButtonTitle:@"好的"
otherButtonTitles:nil, nil] show];
}];
[self.viewModel smk_viewModelWithGetDataSuccessHandler:^(NSArray *array) {
[self.collectionView.collectionHander getItemsWithModelArray:^NSArray *{
return array;
} completion:^{
[self.collectionView reloadData];
}];
}];
}
###配置viewManger
// UIView的delegate方法 ,两种消息传递方式,开发时任选其一即可 根据传入的events信息处理事件
- (void)smk_view:(__kindof UIView *)view withEvents:(NSDictionary *)events {
NSLog(@"----------%@", events);
if ([[events allKeys] containsObject:@"jump"]) {
FirstVC *firstVC = [UIViewController svv_viewControllerWithStoryBoardName:@"Main" identifier:@"FirstVCID"];
[view.sui_currentVC.navigationController pushViewController:firstVC animated:YES];
}
}
// 得到父视图,添加subView -> superView
- (void)smk_viewMangerWithSuperView:(UIView *)superView {
self.thirdView.frame = CGRectMake(0, 66, [UIScreen mainScreen].bounds.size.width, 200);
[superView addSubview:self.thirdView];
}
// 根据传入的info设置添加subView的事件
- (void)smk_viewMangerWithHandleOfSubView:(UIView *)view info:(NSString *)info {
if ([info isEqualToString:@"click"]) {
[view configureViewWithCustomObj:self.dict[@"model"]];
}
}
// 得到模型数据
- (void)viewMangerWithModel:(NSDictionary *(^)( ))dictBlock {
if (dictBlock) {
self.dict = dictBlock();
}
}
###一句代码实现网络请求,自动缓存网络请求数据 #####具体实现细节,点击进入查看SUIMVVMNetwork
- (void)smk_viewModelWithGetDataSuccessHandler:(void (^)(NSArray *))successHandler {
NSString *url = @"http://news-at.zhihu.com/api/4/news/latest";
[SMKHttp get:url params:nil cachePolicy:SMKHttpReturnCacheDataThenLoad success:^(id responseObj) {
NSArray *array = responseObj[@"stories"];
self.smk_dataArrayList = [ThirdModel mj_objectArrayWithKeyValuesArray:array];
if (successHandler) {
successHandler(self.smk_dataArrayList);
}
} failure:^(NSError *error) {
}];
}
#####具体实现细节,点击进入查看SUIMVVMStore
static NSString *tableName = @"arrarList";
SMKStore *store = [SMKStore sharedStore];
[store db_initWithDBName:@"demo.sqlite" tableName:tableName];
[store db_putObject:array withId:@"arrayID" intoTable:tableName];
- 只需实现加载请求以及配置自定义cell和上述代码,就能轻松实现以下效果,最重要的是代码解耦。
- 如果在使用过程中遇到BUG,希望你能Issues我,谢谢(或者尝试下载最新的代码看看BUG修复没有)
- 如果在使用过程中发现功能不够用,希望你能Issues我,我非常想为这个框架增加更多好用的功能,谢谢
###应用架构文章
- 围观神仙打架,反革命工程师《iOS应用架构谈 组件化方案》和蘑菇街Limboy的《蘑菇街 App 的组件化之路》的阅读指导
- iOS 组件化方案探索
- iOS应用架构谈 组件化方案
- 解耦神器 —— 统跳协议和Rewrite引擎
- 携程移动App架构优化之旅
- 蘑菇街App的组件化之路
- 蘑菇街 App 的组件化之路·续
- 猿题库 iOS 客户端架构设计
- 豆瓣混合开发实践
- 滴滴出行iOS客户端架构演进之路
- 不要写死!天猫App的动态化配置中心实践
- 为移动应用提供离线支持
- 携程App的网络性能优化实践
- QCon旧金山演讲总结:阿里无线技术架构演进
###MVVM学习文章