MJRefresh源码知识点解析
一、公共类
这里主要来说说UIScrollView+MJRefresh
(1) 交换函数exchangeInstanceMethod1:(SEL)method1 method2:(SEL)method2
@implementation UITableView (MJRefresh)
+ (void)load
{
[self exchangeInstanceMethod1:@selector(reloadData) method2:@selector(mj_reloadData)];
}
- (void)mj_reloadData
{
[self mj_reloadData];
[self executeReloadDataBlock];
}
@end
此方法是在load
方法中调用。 load
方法是在程序一加载就会最先调用的,之后走main函数
,再然后才是我们熟悉的didFinishLaunchingWithOptions
。load
方法只会走一次,一般跟hook 机制相关的方法都会在load
方法中执行。 hook(钩子),可以理解为把方法钩住,然后做些自己想做的事情。
比如 mj_reloadData
里面的[self executeReloadDataBlock]
。有些人可能有这样的疑问,在mj_reloadData
中调用[self mj_reloadData]
不会产生死循环吗? 当然不会的!别忘了此时mj_reloadData
已经与reloadData
交换了,所以[self mj_reloadData]
实际上执行的是reloadData
方法~ 事实证明把[self mj_reloadData]
改为[self reloadData]
才会出现死循环呢。
说到load
方法,就不得不提一提initialize
方法。两个方法全局都只执行一次。只不过initialize
是在类即将初始化的时候执行的。
(2)分类中重写set和get方法
static const char MJRefreshHeaderKey = '\0';
- (void)setMj_header:(MJRefreshHeader *)mj_header
{
if (mj_header != self.mj_header) {
// 删除旧的,添加新的
[self.mj_header removeFromSuperview];
[self insertSubview:mj_header atIndex:0];
// 存储新的
[self willChangeValueForKey:@"mj_header"]; // KVO
objc_setAssociatedObject(self, &MJRefreshHeaderKey,
mj_header, OBJC_ASSOCIATION_ASSIGN);
[self didChangeValueForKey:@"mj_header"]; // KVO
}
}
- (MJRefreshHeader *)mj_header
{
return objc_getAssociatedObject(self, &MJRefreshHeaderKey);
}
分类中是不能直接增加属性的,即使我们在.h文件中写了@property (strong, nonatomic) MJRefreshHeader *mj_header;
,实际上也是不生成set和get方法的。分类中不能访问成员变量。
但是,我们可以通过关联的方法来重写set和get方法,也就可以给分类增加属性啦。有一点值得注意,就是这两个KVOwillChangeValueForKey
和didChangeValueForKey
。如果是这样的代码_name = name;
,系统会自动给name增加KVO,但由于分类中不能访问成员变量,也只能手动添加咯
二、MJRefresh层级
里面的代码逻辑小码哥注释的很详细,相信认真看的童鞋都杠杠的! 我就从中总结几点我特别受益的地方。
(1)高内聚低耦合。 全篇看下来没有一行的废代码,一层一层的很清晰。
(2)让外界可以一行代码轻松调用,比如+ (instancetype)headerWithRefreshingBlock:(MJRefreshComponentRefreshingBlock)refreshingBlock
。 从此以后我自己定义的控件也都使用这样的方法了
(3)命名很规范。
(4)一些方面要提前想到哪些种情况不允许,要提前return掉。防止外界进行一些不规范的操作导致代码崩溃或执行不必要的操作。
(5)通过数据的改变来改变UI。比如endRefreshing
方法
- (void)endRefreshing
{
dispatch_async(dispatch_get_main_queue(), ^{
self.state = MJRefreshStateIdle;
});
}
并没有在endRefreshing
方法中直接恢复inset和offset,而是将状态(数据)先改变,并在状态中做事情
- (void)setState:(MJRefreshState)state
{
MJRefreshCheckState
// 根据状态做事情
if (state == MJRefreshStateIdle) {
if (oldState != MJRefreshStateRefreshing) return;
// 保存刷新时间
[[NSUserDefaults standardUserDefaults] setObject:[NSDate date] forKey:self.lastUpdatedTimeKey];
[[NSUserDefaults standardUserDefaults] synchronize];
// 恢复inset和offset
[UIView animateWithDuration:MJRefreshSlowAnimationDuration animations:^{
self.scrollView.mj_insetT += self.insetTDelta;
// 自动调整透明度
if (self.isAutomaticallyChangeAlpha) self.alpha = 0.0;
} completion:^(BOOL finished) {
self.pullingPercent = 0.0;
if (self.endRefreshingCompletionBlock) {
self.endRefreshingCompletionBlock();
}
}];
} else if (state == MJRefreshStateRefreshing) {
dispatch_async(dispatch_get_main_queue(), ^{
[UIView animateWithDuration:MJRefreshFastAnimationDuration animations:^{
CGFloat top = self.scrollViewOriginalInset.top + self.mj_h;
// 增加滚动区域top
self.scrollView.mj_insetT = top;
// 设置滚动位置
[self.scrollView setContentOffset:CGPointMake(0, -top) animated:NO];
} completion:^(BOOL finished) {
[self executeRefreshingCallback];
}];
});
}
}