iOS开发——带有暂停功能的计时器

上篇博客我跟大家分享了如何在iOS系统中使用原生框架获取步数,又是大半个月过去了,运动模块的全部功能也总算完成了,也打算有始有终的把如何做一个跑步类App跟大家分享了。

运动类应用中,有一个很重要的模块就是计时器,当然,这个计时器不算复杂,只要有简单的开始、暂停以及复位功能即可。那么今天我们从Model层来看看这个计时器的逻辑实现。

我们先自己创建一个时间的Model

class RunningTimer: NSObject {
//MARK: var property
    private var timeLabel: UILabel!
    private var timer: NSTimer?
    //开始和结束时间列表
    lazy private var startTimes = [NSDate]()
    lazy private var endTimes = [NSDate]()
    
    internal var timeNumber = 0 {
        didSet {
            timeString = getTimeStringFromSecond(timeNumber)
        }
    }
    
    internal private(set) var timeString = "00:00:00" {
        didSet {
            timeLabel.text = timeString
        }
    }


}

先从这段声明变量的代码分析开来,首先是定义了一个timeLabel,这个变量主要是为了在初始化时,直接将View层要显示的Label绑定进来,timer即为一个计时器,顺便定义了两个数组,用来记录时间,因为在真实环境中,可能有若干次暂停,所以用数组来存储。timeNumber即为计时器中的总秒数,用SwiftdidSet特性来监听属性的变化,当秒数发送变化时,讲秒数转化成时间的标准格式,并且赋值给timeString,同理,timeString也在属性发送变化时,将自己的值赋值给Labeltext属性用以显示。

到这里我们的变量讲解完毕,接着往下看功能的实现。

    //MARK: - 初始化
    init(timeLabel: UILabel) {
        self.timeLabel = timeLabel
        timeLabel.text = timeString
    }

这是这个Model的初始化,用意一目了然,传入一个外部Label用以显示时间。

//计时开始
    func timingStart(){
        startTimes.append(NSDate())
        timer = NSTimer.scheduledTimerWithTimeInterval(1, target: self, selector: #selector(self.count), userInfo: nil, repeats: true)
    }
    
    //暂停计时
    func timingPause(){
        endTimes.append(NSDate())
        timer?.invalidate()
    }
    
    //暂停后继续计时
    func timingContinue(){
        timingStart()
    }
    
    //重置Timer
    func resetToStart() {
        startTimes = []
        endTimes = []
        timer?.invalidate()
        timeNumber = 0
    }

这里定义了四个方法,对应我们UI界面会出现的Button功能,StartPauseContinueresetToStart。代码很简单,当start时添加当前时间至数组里,并且启动定时器,暂停时,销毁定时器,添加暂停的时间进入暂停数组。继续和重置同理。那么我们来看定时器启动时,对应的selector做了哪些事情。

//MARK: - 计时器
    private func timeCount(){
        if startTimes.count == 1 {
            let currentTime = NSDate()
            timeNumber = Int(CFDateGetTimeIntervalSinceDate(currentTime, startTimes[0]))
        }else{
            if startTimes.count - endTimes.count == 1 {
                endTimes.append(NSDate())
            }
            let index = startTimes.count - 1
            endTimes[index] = NSDate()
            var timeCount = 0
            for startTime in startTimes{
                timeCount += Int(CFDateGetTimeIntervalSinceDate(endTimes[startTimes.indexOf(startTime)!],startTime))
            }
            timeNumber = timeCount
        }
    }
    
    @objc private func count(){
        timeCount()
    }

当计时器的count()方法运行时、调用timeCount()方法。
当我们第一次运行计时器时,获取的秒数就是开始时间与当前时间比对的差值。
而之后,就是跟暂停之后启动时间的对比了。
这里面使用public func CFDateGetTimeIntervalSinceDate(theDate: CFDate!, _ otherDate: CFDate!) -> CFTimeInterval函数获取两个时间之间的时间戳差值。
最后再把前面那个秒数转格式化时间的方法也贴出来吧。

//从以秒计时的时间里获得表示时间的字符串用于显示
func getTimeStringFromSecond(seconds: Int) -> String {
    
    let secondNumber = seconds % 60
    let minuteNumber = (seconds / 60) % 60
    let hourNumber = (seconds / (60*60)) % 24
    
    let secondText = secondNumber < 10 ? "0\(secondNumber)" : "\(secondNumber)"
    let minuteText = minuteNumber < 10 ? "0\(minuteNumber)" : "\(minuteNumber)"
    let hourText = hourNumber < 10 ? "0\(hourNumber)" : "\(hourNumber)"
    
    return "\(hourText):\(minuteText):\(secondText)"
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 136,258评论 19 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 176,909评论 25 709
  • 本是灵珠降陈塘 人间哪识得,好儿郎 一剑劈开见圣婴 授得太乙真传,便有滔天本领 东海一战,千古留名 剥皮抽筋 哪里...
    飞狐119阅读 2,750评论 2 2
  • 隔壁宿舍的 一位同学 和我在校园餐厅 吃饭的时候 告诉我: 从来没有一个学校 像这个大学 让我睡觉 睡到头疼 20...
    道听途LC阅读 1,463评论 0 0
  • 领导叫你去,那是对你的信任。 吃饭比干活强。应该清醒的认识到。 很多事情都是一瞬间的抉择。抉择错误就很难改正过来了...
    当道阅读 4,053评论 0 0

友情链接更多精彩内容