MJRefresh

介绍:A easy way to use pull-to-refresh、用法简单的下拉刷新框架:一行代码搞定
支持控件刷新:UIScrollView、UITableView、UICollectionView、UIWebView
如何使用:手动导入:①将MJRefresh文件夹中的所有文件拽入项目中 ,②导入主头文件:#import “MJRefresh.h”

//包含:
Base                        Custom
MJRefresh.bundle            MJRefresh.h
MJRefreshConst.h            MJRefreshConst.m
UIScrollView+MJExtension.h  UIScrollView+MJExtension.m
UIScrollView+MJRefresh.h    UIScrollView+MJRefresh.m
UIView+MJExtension.h        UIView+MJExtension.m

结构图:
这里写图片描述

<1.>MJRefreshComponent.h

/** 刷新控件的基类 */
@interface MJRefreshComponent : UIView
#pragma mark - 刷新状态控制
/** 进入刷新状态 */
- (void)beginRefreshing;
/** 结束刷新状态 */
- (void)endRefreshing;
/** 是否正在刷新 */
- (BOOL)isRefreshing;
- 
#pragma mark - 其他
/** 根据拖拽比例自动切换透明度 */
@property (assign, nonatomic, getter=isAutomaticallyChangeAlpha) BOOL automaticallyChangeAlpha;
@end

<2.>MJRefreshHeader.h

@interface MJRefreshHeader : MJRefreshComponent
/** 创建header */
+ (instancetype)headerWithRefreshingBlock:(MJRefreshComponentRefreshingBlock)refreshingBlock;
/** 创建header */
+ (instancetype)headerWithRefreshingTarget:(id)target refreshingAction:(SEL)action;

/** 这个key用来存储上一次下拉刷新成功的时间 */
@property (copy, nonatomic) NSString *lastUpdatedTimeKey;
/** 上一次下拉刷新成功的时间 */
@property (strong, nonatomic, readonly) NSDate *lastUpdatedTime;

/** 忽略多少scrollView的contentInset的top */
@property (assign, nonatomic) CGFloat ignoredScrollViewContentInsetTop;
@end

<3.>MJRefreshFooter.h

@interface MJRefreshFooter : MJRefreshComponent
/** 创建footer */
+ (instancetype)footerWithRefreshingBlock:(MJRefreshComponentRefreshingBlock)refreshingBlock;
/** 创建footer */
+ (instancetype)footerWithRefreshingTarget:(id)target refreshingAction:(SEL)action;

/** 提示没有更多的数据 */
- (void)noticeNoMoreData;
/** 重置没有更多的数据(消除没有更多数据的状态) */
- (void)resetNoMoreData;

/** 忽略多少scrollView的contentInset的bottom */
@property (assign, nonatomic) CGFloat ignoredScrollViewContentInsetBottom;

/** 自动根据有无数据来显示和隐藏(有数据就显示,没有数据隐藏) */
@property (assign, nonatomic) BOOL automaticallyHidden;
@end

<4.>MJRefreshAutoFooter.h

@interface MJRefreshAutoFooter : MJRefreshFooter
/** 是否自动刷新(默认为YES) */
@property (assign, nonatomic, getter=isAutomaticallyRefresh) BOOL automaticallyRefresh;

/** 当底部控件出现多少时就自动刷新(默认为1.0,也就是底部控件完全出现时,才会自动刷新) */
@property (assign, nonatomic) CGFloat triggerAutomaticallyRefreshPercent;
@end

参考事例————————-

<1.>下拉刷新01-默认

self.tableView.header = [MJRefreshNormalHeader headerWithRefreshingBlock:^{
   // 进入刷新状态后会自动调用这个block
}];
或
// 设置回调(一旦进入刷新状态,就调用target的action,也就是调用self的loadNewData方法)
self.tableView.header = [MJRefreshNormalHeader headerWithRefreshingTarget:self refreshingAction:@selector(loadNewData)];

// 马上进入刷新状态
[self.tableView.header beginRefreshing];

这里写图片描述

<2.>下拉刷新02-动画图片

// 设置回调(一旦进入刷新状态,就调用target的action,也就是调用self的loadNewData方法)
MJRefreshGifHeader *header = [MJRefreshGifHeader headerWithRefreshingTarget:self refreshingAction:@selector(loadNewData)];
// 设置普通状态的动画图片
[header setImages:idleImages forState:MJRefreshStateIdle];
// 设置即将刷新状态的动画图片(一松开就会刷新的状态)
[header setImages:pullingImages forState:MJRefreshStatePulling];
// 设置正在刷新状态的动画图片
[header setImages:refreshingImages forState:MJRefreshStateRefreshing];
// 设置header
self.tableView.header = header;

这里写图片描述

<3.>下拉刷新03-隐藏时间

// 隐藏时间
header.lastUpdatedTimeLabel.hidden = YES;

这里写图片描述

<4.>下拉刷新04-隐藏状态和时间

// 隐藏时间
header.lastUpdatedTimeLabel.hidden = YES;

// 隐藏状态
header.stateLabel.hidden = YES;

这里写图片描述

<5.>下拉刷新05-自定义文字

// 设置文字
[header setTitle:@"Pull down to refresh" forState:MJRefreshStateIdle];
[header setTitle:@"Release to refresh" forState:MJRefreshStatePulling];
[header setTitle:@"Loading ..." forState:MJRefreshStateRefreshing];

// 设置字体
header.stateLabel.font = [UIFont systemFontOfSize:15];
header.lastUpdatedTimeLabel.font = [UIFont systemFontOfSize:14];

// 设置颜色
header.stateLabel.textColor = [UIColor redColor];
header.lastUpdatedTimeLabel.textColor = [UIColor blueColor];

这里写图片描述

<6.>上拉刷新01-默认

self.tableView.footer = [MJRefreshAutoNormalFooter footerWithRefreshingBlock:^{
   // 进入刷新状态后会自动调用这个block
}];
或
// 设置回调(一旦进入刷新状态,就调用target的action,也就是调用self的loadMoreData方法)
self.tableView.footer = [MJRefreshAutoNormalFooter footerWithRefreshingTarget:self refreshingAction:

这里写图片描述

<7.>上拉刷新02-动画图片

// 设置回调(一旦进入刷新状态,就调用target的action,也就是调用self的loadMoreData方法)
MJRefreshAutoGifFooter *footer = [MJRefreshAutoGifFooter footerWithRefreshingTarget:self refreshingAction:@selector(loadMoreData)];

// 设置刷新图片
[footer setImages:refreshingImages forState:MJRefreshStateRefreshing];

// 隐藏刷新状态的文字
footer.refreshingTitleHidden = YES;
// 如果没有上面的方法,就用footer.stateLabel.hidden = YES;

// 设置尾部
self.tableView.footer = footer;

这里写图片描述

<8.>上拉刷新05-自定义文字

// 设置文字
[footer setTitle:@"Click or drag up to refresh" forState:MJRefreshStateIdle];
[footer setTitle:@"Loading more ..." forState:MJRefreshStateRefreshing];
[footer setTitle:@"No more data" forState:MJRefreshStateNoMoreData];
// 设置字体
footer.stateLabel.font = [UIFont systemFontOfSize:17];
// 设置颜色
footer.stateLabel.textColor = [UIColor blueColor];

这里写图片描述

//测试加载最新数据
- (void)loadNewData
{
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        [self.tableView.header endRefreshing];
    });
}

//测试加载更多
- (void)loadMoreData
{
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        [self.tableView.footer noticeNoMoreData];// 变为没有更多数据的状态

        //self.tableView.footer.hidden = YES;// 隐藏当前的上拉刷新控件
        //[footer noticeNoMoreData]; // 变为没有更多数据的状态
        [self.tableView.footer endRefreshing];

    });
}

UICollectionView01-上下拉刷新

// 下拉刷新
self.collectionView.header = [MJRefreshNormalHeader headerWithRefreshingBlock:^{
   // 进入刷新状态后会自动调用这个block
}];

// 上拉刷新
self.collectionView.footer = [MJRefreshAutoNormalFooter footerWithRefreshingBlock:^{
   // 进入刷新状态后会自动调用这个block
}];

这里写图片描述

UIWebView01-下拉刷新

// 添加下拉刷新控件
self.webView.scrollView.header = [MJRefreshNormalHeader headerWithRefreshingBlock:^{
   // 进入刷新状态后会自动调用这个block
}];

Contents:
常见API
MJRefreshComponent.h
MJRefreshHeader.h
MJRefreshFooter.h
MJRefreshAutoFooter.h
Examples
Reference【参考】
下拉刷新01-默认
下拉刷新02-动画图片
下拉刷新03-隐藏时间
下拉刷新04-隐藏状态和时间
下拉刷新05-自定义文字
下拉刷新06-自定义刷新控件
上拉刷新01-默认
上拉刷新02-动画图片
上拉刷新03-隐藏刷新状态的文字
上拉刷新04-全部加载完毕
上拉刷新05-自定义文字
上拉刷新06-加载后隐藏
上拉刷新07-自动回弹的上拉01
上拉刷新08-自动回弹的上拉02
上拉刷新09-自定义刷新控件(自动刷新)
上拉刷新10-自定义刷新控件(自动回弹)
UICollectionView01-上下拉刷新
UIWebView01-下拉刷新

内容转载并整理自MJ:https://github.com/CoderMJLee/MJRefresh

### 关于 MJRefresh 的适配方案 MJRefresh 是一款流行的下拉刷新和上拉加载更多组件库,在不同版本的 iOS 和不同的 UI 控件(如 `UITableView` 和 `UICollectionView`)中可能会遇到一些兼容性和适配问题。以下是针对 MJRefreshiOS 不同版本中的常见适配方法以及解决方案。 #### 1. **iOS 11 中的内容偏移调整** 在 iOS 11 中,苹果弃用了 `automaticallyAdjustsScrollViewInsets` 属性,并引入了新的 `contentInsetAdjustmentBehavior` 来控制滚动视图的内容偏移行为。这可能导致使用 MJRefresh 的表格视图或集合视图在升级到 iOS 11 后出现头部刷新区域错乱的情况[^1]。 可以通过以下方式解决此问题: ```swift if #available(iOS 11.0, *) { tableView.contentInsetAdjustmentBehavior = .never } else { automaticallyAdjustsScrollViewInsets = false } ``` 这段代码确保无论设备运行的是哪个版本的 iOS,都能正确处理内容偏移的行为[^3]。 --- #### 2. **UITableViewCell 自适应高度与 MJRefresh 冲突** 当使用 `UITableViewAutomaticDimension` 实现单元格自适应高度时,可能引发 MJRefresh 上拉加载过程中无限触发刷新的问题[^2]。这是因为 MJRefresh 需要精确计算表视图的高度变化,而动态高度的单元格可能导致不一致的结果。 一种常见的解决方案是在初始化 MJRefresh 组件之前手动设置表视图的预估行高和实际行高: ```objc self.tableView.rowHeight = UITableViewAutomaticDimension; self.tableView.estimatedRowHeight = 44; // 设置合理的默认值 ``` 此外,还可以尝试禁用 MJRefresh 的某些特性以减少冲突的可能性。例如,关闭自动进入刷新状态的功能: ```objc refreshControl.beginRefreshingAutomaticallyEnabled = NO; ``` --- #### 3. **上拉加载完成后表视图闪烁或跳动** 部分开发者报告说,在调用 MJRefresh 的完成回调函数后,表视图会出现短暂的闪烁或跳动现象[^4]。这种问题通常发生在使用 Auto Layout 动态布局单元格的情况下。 为了缓解这一问题,可以在结束刷新操作之后重新调整表视图的内容偏移量: ```objc [weakTableView reloadData]; dispatch_async(dispatch_get_main_queue(), ^{ [weakRefresh endRefreshing]; weakTableView.contentOffset = CGPointMake(0, -weakRefresh.header.frame.size.height); }); ``` 这种方法强制将滚动位置恢复到刷新头所在的位置,从而避免视觉上的异常效果。 --- #### 4. **全局配置 Scroll View 行为** 对于大规模项目而言,逐一手动修改每一个 `UIScrollView` 子类实例可能是低效且容易遗漏的做法。因此,建议在应用启动阶段通过外观代理统一设定所有 scroll view 的内容嵌入调整策略[^5]: ```objc if (@available(iOS 11.0, *)) { [[UIScrollView appearance] setContentInsetAdjustmentBehavior:UIScrollViewContentInsetAdjustmentNever]; } ``` 这样不仅简化了开发流程,还能够有效防止因个别页面未做特殊处理而导致的显示错误。 --- #### 总结 综上所述,MJRefresh 库虽然功能强大但在跨平台支持方面仍需注意细节差异。具体来说: - 对于 iOS 11+ 设备应优先采用新推出的 API 替代旧版机制; - 当涉及复杂交互逻辑比如动态尺寸管理时则更应该谨慎对待边界条件以免引起意外状况发生; - 考虑到用户体验的一致性最好能尽早规划好整体架构以便后期维护更加便捷高效。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值