前言
最近开发中,用到了UITableViewCell
倒计时功能,这里将这部分功能分离出来,供大家参考。
1.原理
考虑到APP性能,这里只创建一个定时器,定时刷新当前正在显示的UITableViewCell
,使用Model
记录剩余倒计时时间和当前UITableViewCell
是否暂停。
2.核心代码
创建定时器
考虑到方便和不需要销毁,这里定时器使用GCD—>GCD定时器封装OC&Swift
self.timer = [[CLGCDTimer alloc] initWithInterval:1 delaySecs:0 queue:dispatch_get_main_queue() repeats:YES action:^(NSInteger actionTimes) {
__typeof(&*weakSelf) strongSelf = weakSelf;
strongSelf.actionTimes = actionTimes;
[strongSelf reloadVisibleCells];
}];
[self.timer start];
刷新当前正在显示的UITableViewCell
这里只对正在显示的UITableViewCell
进行操作,找出当前正在显示的UITableViewCell
对应的数据Model
,对数据源进行修改后刷新UITableView
。
- (void)reloadVisibleCells {
for (CLCountdownCell *cell in self.tableView.visibleCells) {
NSIndexPath *indexPath = [self.tableView indexPathForCell:cell];
CLCountdownModel *model = [self.arrayDS objectAtIndex:indexPath.row];
model.actionTimes = self.actionTimes;
model.leaveTime = self.leaveTime;
if (model.isPause) {
continue;
}
cell.model = model;
}
}
Model数据修正
因为只修改了当前正在显示的UITableViewCell
对应Model
的数据源,所以滑动出来的UITableViewCell
对应Model
数据源并不正确,这里在UITableViewCell
即将显示的时候修正对应数据源。
- (void)tableView:(UITableView *)tableView willDisplayCell:(UITableViewCell *)cell forRowAtIndexPath:(NSIndexPath *)indexPath {
CLCountdownModel *model = [self.arrayDS objectAtIndex:indexPath.row];
model.actionTimes = self.actionTimes;
model.leaveTime = self.leaveTime;
if (!model.isPause) {
CLCountdownCell *countdownCell = (CLCountdownCell *)cell;
countdownCell.model = model;
}
}
记录暂停状态
通过修改对应UITableViewCell
所在数据Model
,记录当前UITableViewCell
暂停状态,达到暂停效果。这里需要注意记录暂停当时的时间以及重新开始的时间。
- (void)setIsPause:(BOOL)isPause {
if (isPause) {
self.pauseTime = self.remainingTime;
self.startTime = 0;
}else {
_isPause = isPause;
self.startTime = self.remainingTime;
}
_isPause = isPause;
}
- (NSInteger)remainingTime {
if (_isPause) {
return self.pauseTime;
}else {
if (self.pauseTime != 0 && self.startTime != 0) {
return self.countdownTime - self.actionTimes + self.pauseTime - self.startTime - self.leaveTime;
}else {
return self.countdownTime - self.actionTimes - self.leaveTime;
}
}
}
APP进入后台记录
当APP进入后台,需要记录当前时间,当再次进入前台的时候,需要减去离开时间,这样即使进入后台也不会影响到倒计时,这里考虑到进入后台期间修改时间等操作,直接使用系统运行时间进行记录。
///系统当前运行了多长时间
///因为两个参数都会受用户修改时间的影响,因此它们想减的值是不变的
+ (NSTimeInterval)uptimeSinceLastBoot {
//获取当前设备时间时间戳 受用户修改时间影响
struct timeval now;
struct timezone tz;
gettimeofday(&now, &tz);
//获取系统上次重启的时间戳 受用户修改时间影响
struct timeval boottime;
int mib[2] = {CTL_KERN, KERN_BOOTTIME};
size_t size = sizeof(boottime);
double uptime = -1;
if (sysctl(mib, 2, &boottime, &size, NULL, 0) != -1 && boottime.tv_sec != 0) {
//获取上次启动时间成功
//秒
uptime = now.tv_sec - boottime.tv_sec;
//微秒
uptime += (double)(now.tv_usec - boottime.tv_usec) / 1000000.0;
}
return uptime;
}
监听APP进入后台和进入前台通知,进行记录。
- (void)addNotification {
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(applicationWillResignActive:)
name:UIApplicationWillResignActiveNotification object:nil];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(applicationDidBecomeActive:)
name:UIApplicationDidBecomeActiveNotification object:nil];
}
- (void)applicationWillResignActive:(NSNotification *)notification {
self.resignSystemUpTime = [NSDate uptimeSinceLastBoot];
[self.timer suspend];
}
- (void)applicationDidBecomeActive:(NSNotification *)notification {
self.becomeSystemUpTime = [NSDate uptimeSinceLastBoot];
self.leaveTime += (NSInteger)floor(self.becomeSystemUpTime - self.resignSystemUpTime);
[self.timer resume];
}
3.效果图
这里数据源创建了10万,因为只有一个定时器,并且只刷新当前正在显示的UITableViewCell
,所以滑动起来并不会有任何卡顿。
4.总结
UITableViewCell
倒计时代码并不多,只是需要注意一些细节,记录对应时间和状态。更多细节请参考文章对应Demo---->CLDemo如有帮助,欢迎Star。