-
-
Save Panajev/f995c97318f66a9085146c14289b1a1f to your computer and use it in GitHub Desktop.
A repeating GCD timer that can run on a background queue
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/// RepeatingTimer mimics the API of DispatchSourceTimer but in a way that prevents | |
/// crashes that occur from calling resume multiple times on a timer that is | |
/// already resumed (noted by https://github.com/SiftScience/sift-ios/issues/52 | |
class RepeatingTimer { | |
let timeInterval: TimeInterval | |
init(timeInterval: TimeInterval) { | |
self.timeInterval = timeInterval | |
} | |
private lazy var timer: DispatchSourceTimer = { | |
let t = DispatchSource.makeTimerSource() | |
t.schedule(deadline: .now() + self.timeInterval, repeating: self.timeInterval) | |
t.setEventHandler(handler: { [weak self] in | |
self?.eventHandler?() | |
}) | |
return t | |
}() | |
var eventHandler: (() -> Void)? | |
private enum State { | |
case suspended | |
case resumed | |
} | |
private var state: State = .suspended | |
deinit { | |
timer.setEventHandler {} | |
timer.cancel() | |
/* | |
If the timer is suspended, calling cancel without resuming | |
triggers a crash. This is documented here https://forums.developer.apple.com/thread/15902 | |
*/ | |
resume() | |
eventHandler = nil | |
} | |
func resume() { | |
if state == .resumed { | |
return | |
} | |
state = .resumed | |
timer.resume() | |
} | |
func suspend() { | |
if state == .suspended { | |
return | |
} | |
state = .suspended | |
timer.suspend() | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment