|
|
|
// Copyright © 2015 Abhishek Banthia
|
|
|
|
|
|
|
|
import Foundation
|
|
|
|
|
|
|
|
open class Repeater: Equatable {
|
|
|
|
/// State of the timer
|
|
|
|
///
|
|
|
|
/// - paused: idle (never started yet or paused)
|
|
|
|
/// - running: timer is running
|
|
|
|
/// - executing: the observers are being executed
|
|
|
|
/// - finished: timer lifetime is finished
|
|
|
|
public enum State: Equatable, CustomStringConvertible {
|
|
|
|
case paused
|
|
|
|
case running
|
|
|
|
case executing
|
|
|
|
case finished
|
|
|
|
|
|
|
|
public static func == (lhs: State, rhs: State) -> Bool {
|
|
|
|
switch (lhs, rhs) {
|
|
|
|
case (.paused, .paused),
|
|
|
|
(.running, .running),
|
|
|
|
(.executing, .executing),
|
|
|
|
(.finished, .finished):
|
|
|
|
return true
|
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Return `true` if timer is currently running, including when the observers are being executed.
|
|
|
|
public var isRunning: Bool {
|
|
|
|
guard self == .running || self == .executing else { return false }
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Return `true` if the observers are being executed.
|
|
|
|
public var isExecuting: Bool {
|
|
|
|
guard case .executing = self else { return false }
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Is timer finished its lifetime?
|
|
|
|
/// It return always `false` for infinite timers.
|
|
|
|
/// It return `true` for `.once` mode timer after the first fire,
|
|
|
|
/// and when `.remainingIterations` is zero for `.finite` mode timers
|
|
|
|
public var isFinished: Bool {
|
|
|
|
guard case .finished = self else { return false }
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
/// State description
|
|
|
|
public var description: String {
|
|
|
|
switch self {
|
|
|
|
case .paused: return "idle/paused"
|
|
|
|
case .finished: return "finished"
|
|
|
|
case .running: return "running"
|
|
|
|
case .executing: return "executing"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Repeat interval
|
|
|
|
public enum Interval {
|
|
|
|
case nanoseconds(_: Int)
|
|
|
|
case microseconds(_: Int)
|
|
|
|
case milliseconds(_: Int)
|
|
|
|
case minutes(_: Int)
|
|
|
|
case seconds(_: Double)
|
|
|
|
case hours(_: Int)
|
|
|
|
case days(_: Int)
|
|
|
|
|
|
|
|
internal var value: DispatchTimeInterval {
|
|
|
|
switch self {
|
|
|
|
case let .nanoseconds(value): return .nanoseconds(value)
|
|
|
|
case let .microseconds(value): return .microseconds(value)
|
|
|
|
case let .milliseconds(value): return .milliseconds(value)
|
|
|
|
case let .seconds(value): return .milliseconds(Int(Double(value) * Double(1000)))
|
|
|
|
case let .minutes(value): return .seconds(value * 60)
|
|
|
|
case let .hours(value): return .seconds(value * 3600)
|
|
|
|
case let .days(value): return .seconds(value * 86400)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Mode of the timer.
|
|
|
|
///
|
|
|
|
/// - infinite: infinite number of repeats.
|
|
|
|
/// - finite: finite number of repeats.
|
|
|
|
/// - once: single repeat.
|
|
|
|
public enum Mode {
|
|
|
|
case infinite
|
|
|
|
case finite(_: Int)
|
|
|
|
case once
|
|
|
|
|
|
|
|
/// Is timer a repeating timer?
|
|
|
|
internal var isRepeating: Bool {
|
|
|
|
switch self {
|
|
|
|
case .once: return false
|
|
|
|
default: return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Number of repeats, if applicable. Otherwise `nil`
|
|
|
|
public var countIterations: Int? {
|
|
|
|
switch self {
|
|
|
|
case let .finite(counts): return counts
|
|
|
|
default: return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Is infinite timer
|
|
|
|
public var isInfinite: Bool {
|
|
|
|
guard case .infinite = self else {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Handler typealias
|
|
|
|
public typealias Observer = ((Repeater) -> Void)
|
|
|
|
|
|
|
|
/// Token assigned to the observer
|
|
|
|
public typealias ObserverToken = UInt64
|
|
|
|
|
|
|
|
/// Current state of the timer
|
|
|
|
public private(set) var state: State = .paused {
|
|
|
|
didSet {
|
|
|
|
onStateChanged?(self, state)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Callback called to intercept state's change of the timer
|
|
|
|
public var onStateChanged: ((_ timer: Repeater, _ state: State) -> Void)?
|
|
|
|
|
|
|
|
/// List of the observer of the timer
|
|
|
|
private var observers = [ObserverToken: Observer]()
|
|
|
|
|
|
|
|
/// Next token of the timer
|
|
|
|
private var nextObserverID: UInt64 = 0
|
|
|
|
|
|
|
|
/// Internal GCD Timer
|
|
|
|
private var timer: DispatchSourceTimer?
|
|
|
|
|
|
|
|
/// Is timer a repeat timer
|
|
|
|
public private(set) var mode: Mode
|
|
|
|
|
|
|
|
/// Number of remaining repeats count
|
|
|
|
public private(set) var remainingIterations: Int?
|
|
|
|
|
|
|
|
/// Interval of the timer
|
|
|
|
private var interval: Interval
|
|
|
|
|
|
|
|
/// Accuracy of the timer
|
|
|
|
private var tolerance: DispatchTimeInterval
|
|
|
|
|
|
|
|
/// Dispatch queue parent of the timer
|
|
|
|
private var queue: DispatchQueue?
|
|
|
|
|
|
|
|
/// Initialize a new timer.
|
|
|
|
///
|
|
|
|
/// - Parameters:
|
|
|
|
/// - interval: interval of the timer
|
|
|
|
/// - mode: mode of the timer
|
|
|
|
/// - tolerance: tolerance of the timer, 0 is default.
|
|
|
|
/// - queue: queue in which the timer should be executed; if `nil` a new queue is created automatically.
|
|
|
|
/// - observer: observer
|
|
|
|
public init(interval: Interval, mode: Mode = .infinite, tolerance: DispatchTimeInterval = .nanoseconds(3), queue: DispatchQueue? = nil, observer: @escaping Observer) {
|
|
|
|
self.mode = mode
|
|
|
|
self.interval = interval
|
|
|
|
self.tolerance = tolerance
|
|
|
|
remainingIterations = mode.countIterations
|
|
|
|
self.queue = (queue ?? DispatchQueue(label: "com.abhishek.Clocker"))
|
|
|
|
timer = configureTimer()
|
|
|
|
observe(observer)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Add new a listener to the timer.
|
|
|
|
///
|
|
|
|
/// - Parameter callback: callback to call for fire events.
|
|
|
|
/// - Returns: token used to remove the handler
|
|
|
|
@discardableResult
|
|
|
|
public func observe(_ observer: @escaping Observer) -> ObserverToken {
|
|
|
|
var (new, overflow) = nextObserverID.addingReportingOverflow(1)
|
|
|
|
if overflow { // you need to add an incredible number of offset...sure you can't
|
|
|
|
nextObserverID = 0
|
|
|
|
new = 0
|
|
|
|
}
|
|
|
|
nextObserverID = new
|
|
|
|
observers[new] = observer
|
|
|
|
return new
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Remove an observer of the timer.
|
|
|
|
///
|
|
|
|
/// - Parameter id: id of the observer to remove
|
|
|
|
public func remove(observer identifier: ObserverToken) {
|
|
|
|
observers.removeValue(forKey: identifier)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Remove all observers of the timer.
|
|
|
|
///
|
|
|
|
/// - Parameter stopTimer: `true` to also stop timer by calling `pause()` function.
|
|
|
|
public func removeAllObservers(thenStop stopTimer: Bool = false) {
|
|
|
|
observers.removeAll()
|
|
|
|
|
|
|
|
if stopTimer {
|
|
|
|
pause()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Configure a new timer session.
|
|
|
|
///
|
|
|
|
/// - Returns: dispatch timer
|
|
|
|
private func configureTimer() -> DispatchSourceTimer {
|
|
|
|
let associatedQueue = (queue ?? DispatchQueue(label: "com.repeat.\(NSUUID().uuidString)"))
|
|
|
|
let timer = DispatchSource.makeTimerSource(queue: associatedQueue)
|
|
|
|
let repeatInterval = interval.value
|
|
|
|
let deadline: DispatchTime = (DispatchTime.now() + repeatInterval)
|
|
|
|
if mode.isRepeating {
|
|
|
|
timer.schedule(deadline: deadline, repeating: repeatInterval, leeway: tolerance)
|
|
|
|
} else {
|
|
|
|
timer.schedule(deadline: deadline, leeway: tolerance)
|
|
|
|
}
|
|
|
|
|
|
|
|
timer.setEventHandler { [weak self] in
|
|
|
|
if let unwrapped = self {
|
|
|
|
unwrapped.timeFired()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return timer
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Destroy current timer
|
|
|
|
private func destroyTimer() {
|
|
|
|
timer?.setEventHandler(handler: nil)
|
|
|
|
timer?.cancel()
|
|
|
|
|
|
|
|
if state == .paused || state == .finished {
|
|
|
|
timer?.resume()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create and schedule a timer that will call `handler` once after the specified time.
|
|
|
|
///
|
|
|
|
/// - Parameters:
|
|
|
|
/// - interval: interval delay for single fire
|
|
|
|
/// - queue: destination queue, if `nil` a new `DispatchQueue` is created automatically.
|
|
|
|
/// - observer: handler to call when timer fires.
|
|
|
|
/// - Returns: timer instance
|
|
|
|
@discardableResult
|
|
|
|
public class func once(after interval: Interval, queue: DispatchQueue? = nil, _ observer: @escaping Observer) -> Repeater {
|
|
|
|
let timer = Repeater(interval: interval, mode: .once, queue: queue, observer: observer)
|
|
|
|
timer.start()
|
|
|
|
return timer
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create and schedule a timer that will fire every interval optionally by limiting the number of fires.
|
|
|
|
///
|
|
|
|
/// - Parameters:
|
|
|
|
/// - interval: interval of fire
|
|
|
|
/// - count: a non `nil` and > 0 value to limit the number of fire, `nil` to set it as infinite.
|
|
|
|
/// - queue: destination queue, if `nil` a new `DispatchQueue` is created automatically.
|
|
|
|
/// - handler: handler to call on fire
|
|
|
|
/// - Returns: timer
|
|
|
|
@discardableResult
|
|
|
|
public class func every(_ interval: Interval, count: Int? = nil, queue: DispatchQueue? = nil, _ handler: @escaping Observer) -> Repeater {
|
|
|
|
let mode: Mode = (count != nil ? .finite(count!) : .infinite)
|
|
|
|
let timer = Repeater(interval: interval, mode: mode, queue: queue, observer: handler)
|
|
|
|
timer.start()
|
|
|
|
return timer
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Force fire.
|
|
|
|
///
|
|
|
|
/// - Parameter pause: `true` to pause after fire, `false` to continue the regular firing schedule.
|
|
|
|
public func fire(andPause pause: Bool = false) {
|
|
|
|
timeFired()
|
|
|
|
if pause == true {
|
|
|
|
self.pause()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Reset the state of the timer, optionally changing the fire interval.
|
|
|
|
///
|
|
|
|
/// - Parameters:
|
|
|
|
/// - interval: new fire interval; pass `nil` to keep the latest interval set.
|
|
|
|
/// - restart: `true` to automatically restart the timer, `false` to keep it stopped after configuration.
|
|
|
|
public func reset(_ interval: Interval?, restart: Bool = true) {
|
|
|
|
if state.isRunning {
|
|
|
|
setPause(from: state)
|
|
|
|
}
|
|
|
|
|
|
|
|
// For finite counter we want to also reset the repeat count
|
|
|
|
if case let .finite(count) = mode {
|
|
|
|
self.remainingIterations = count
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a new instance of timer configured
|
|
|
|
if let newInterval = interval {
|
|
|
|
self.interval = newInterval
|
|
|
|
} // update interval
|
|
|
|
destroyTimer()
|
|
|
|
timer = configureTimer()
|
|
|
|
state = .paused
|
|
|
|
|
|
|
|
if restart {
|
|
|
|
timer?.resume()
|
|
|
|
state = .running
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Start timer. If timer is already running it does nothing.
|
|
|
|
@discardableResult
|
|
|
|
public func start() -> Bool {
|
|
|
|
guard state.isRunning == false else {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// If timer has not finished its lifetime we want simply
|
|
|
|
// restart it from the current state.
|
|
|
|
guard state.isFinished == true else {
|
|
|
|
state = .running
|
|
|
|
timer?.resume()
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise we need to reset the state based upon the mode
|
|
|
|
// and start it again.
|
|
|
|
reset(nil, restart: true)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Pause a running timer. If timer is paused it does nothing.
|
|
|
|
@discardableResult
|
|
|
|
public func pause() -> Bool {
|
|
|
|
guard state != .paused, state != .finished else {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return setPause(from: state)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Pause a running timer optionally changing the state with regard to the current state.
|
|
|
|
///
|
|
|
|
/// - Parameters:
|
|
|
|
/// - from: the state which the timer should only be paused if it is the current state
|
|
|
|
/// - to: the new state to change to if the timer is paused
|
|
|
|
/// - Returns: `true` if timer is paused
|
|
|
|
@discardableResult
|
|
|
|
private func setPause(from currentState: State, to newState: State = .paused) -> Bool {
|
|
|
|
guard state == currentState else {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
timer?.suspend()
|
|
|
|
state = newState
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Called when timer is fired
|
|
|
|
private func timeFired() {
|
|
|
|
state = .executing
|
|
|
|
|
|
|
|
// dispatch to observers
|
|
|
|
observers.values.forEach { $0(self) }
|
|
|
|
|
|
|
|
// manage lifetime
|
|
|
|
switch mode {
|
|
|
|
case .once:
|
|
|
|
// once timer's lifetime is finished after the first fire
|
|
|
|
// you can reset it by calling `reset()` function.
|
|
|
|
setPause(from: .executing, to: .finished)
|
|
|
|
case .finite:
|
|
|
|
// for finite intervals we decrement the left iterations count...
|
|
|
|
remainingIterations! -= 1
|
|
|
|
if remainingIterations! == 0 {
|
|
|
|
// ...if left count is zero we just pause the timer and stop
|
|
|
|
setPause(from: .executing, to: .finished)
|
|
|
|
}
|
|
|
|
case .infinite:
|
|
|
|
// infinite timer does nothing special on the state machine
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
deinit {
|
|
|
|
self.observers.removeAll()
|
|
|
|
self.destroyTimer()
|
|
|
|
}
|
|
|
|
|
|
|
|
public static func == (lhs: Repeater, rhs: Repeater) -> Bool {
|
|
|
|
return lhs === rhs
|
|
|
|
}
|
|
|
|
}
|