412 lines
14 KiB
Swift
412 lines
14 KiB
Swift
//
|
|
// TimerEngineTests.swift
|
|
// GazeTests
|
|
//
|
|
// Created by Mike Freno on 1/7/26.
|
|
//
|
|
|
|
import XCTest
|
|
@testable import Gaze
|
|
|
|
@MainActor
|
|
final class TimerEngineTests: XCTestCase {
|
|
|
|
var timerEngine: TimerEngine!
|
|
var settingsManager: SettingsManager!
|
|
|
|
override func setUp() async throws {
|
|
try await super.setUp()
|
|
settingsManager = SettingsManager.shared
|
|
UserDefaults.standard.removeObject(forKey: "gazeAppSettings")
|
|
settingsManager.load()
|
|
timerEngine = TimerEngine(settingsManager: settingsManager)
|
|
}
|
|
|
|
override func tearDown() async throws {
|
|
timerEngine.stop()
|
|
UserDefaults.standard.removeObject(forKey: "gazeAppSettings")
|
|
try await super.tearDown()
|
|
}
|
|
|
|
func testTimerInitialization() {
|
|
timerEngine.start()
|
|
|
|
XCTAssertEqual(timerEngine.timerStates.count, 3)
|
|
XCTAssertNotNil(timerEngine.timerStates[.builtIn(.lookAway)])
|
|
XCTAssertNotNil(timerEngine.timerStates[.builtIn(.blink)])
|
|
XCTAssertNotNil(timerEngine.timerStates[.builtIn(.posture)])
|
|
}
|
|
|
|
func testDisabledTimersNotInitialized() {
|
|
settingsManager.settings.blinkTimer.enabled = false
|
|
|
|
timerEngine.start()
|
|
|
|
XCTAssertEqual(timerEngine.timerStates.count, 2)
|
|
XCTAssertNotNil(timerEngine.timerStates[.builtIn(.lookAway)])
|
|
XCTAssertNil(timerEngine.timerStates[.builtIn(.blink)])
|
|
XCTAssertNotNil(timerEngine.timerStates[.builtIn(.posture)])
|
|
}
|
|
|
|
func testTimerStateInitialValues() {
|
|
timerEngine.start()
|
|
|
|
let lookAwayState = timerEngine.timerStates[.builtIn(.lookAway)]!
|
|
XCTAssertEqual(lookAwayState.identifier, .builtIn(.lookAway))
|
|
XCTAssertEqual(lookAwayState.remainingSeconds, 20 * 60)
|
|
XCTAssertFalse(lookAwayState.isPaused)
|
|
XCTAssertTrue(lookAwayState.isActive)
|
|
}
|
|
|
|
func testPauseAllTimers() {
|
|
timerEngine.start()
|
|
timerEngine.pause()
|
|
|
|
for (_, state) in timerEngine.timerStates {
|
|
XCTAssertTrue(state.isPaused)
|
|
}
|
|
}
|
|
|
|
func testResumeAllTimers() {
|
|
timerEngine.start()
|
|
timerEngine.pause()
|
|
timerEngine.resume()
|
|
|
|
for (_, state) in timerEngine.timerStates {
|
|
XCTAssertFalse(state.isPaused)
|
|
}
|
|
}
|
|
|
|
func testSkipNext() {
|
|
settingsManager.settings.lookAwayTimer.intervalSeconds = 60
|
|
timerEngine.start()
|
|
|
|
timerEngine.timerStates[.builtIn(.lookAway)]?.remainingSeconds = 10
|
|
|
|
timerEngine.skipNext(identifier: .builtIn(.lookAway))
|
|
|
|
XCTAssertEqual(timerEngine.timerStates[.builtIn(.lookAway)]?.remainingSeconds, 60)
|
|
}
|
|
|
|
func testGetTimeRemaining() {
|
|
timerEngine.start()
|
|
|
|
let timeRemaining = timerEngine.getTimeRemaining(for: .builtIn(.lookAway))
|
|
XCTAssertEqual(timeRemaining, TimeInterval(20 * 60))
|
|
}
|
|
|
|
func testGetFormattedTimeRemaining() {
|
|
timerEngine.start()
|
|
timerEngine.timerStates[.builtIn(.lookAway)]?.remainingSeconds = 125
|
|
|
|
let formatted = timerEngine.getFormattedTimeRemaining(for: .builtIn(.lookAway))
|
|
XCTAssertEqual(formatted, "2:05")
|
|
}
|
|
|
|
func testGetFormattedTimeRemainingWithHours() {
|
|
timerEngine.start()
|
|
timerEngine.timerStates[.builtIn(.lookAway)]?.remainingSeconds = 3665
|
|
|
|
let formatted = timerEngine.getFormattedTimeRemaining(for: .builtIn(.lookAway))
|
|
XCTAssertEqual(formatted, "1:01:05")
|
|
}
|
|
|
|
func testStop() {
|
|
timerEngine.start()
|
|
XCTAssertFalse(timerEngine.timerStates.isEmpty)
|
|
|
|
timerEngine.stop()
|
|
XCTAssertTrue(timerEngine.timerStates.isEmpty)
|
|
}
|
|
|
|
func testDismissReminderResetsTimer() {
|
|
timerEngine.start()
|
|
timerEngine.timerStates[.builtIn(.blink)]?.remainingSeconds = 0
|
|
timerEngine.activeReminder = .blinkTriggered
|
|
|
|
timerEngine.dismissReminder()
|
|
|
|
XCTAssertNil(timerEngine.activeReminder)
|
|
XCTAssertEqual(timerEngine.timerStates[.builtIn(.blink)]?.remainingSeconds, 5 * 60)
|
|
}
|
|
|
|
func testDismissLookAwayResumesTimers() {
|
|
timerEngine.start()
|
|
timerEngine.activeReminder = .lookAwayTriggered(countdownSeconds: 20)
|
|
timerEngine.pause()
|
|
|
|
timerEngine.dismissReminder()
|
|
|
|
for (_, state) in timerEngine.timerStates {
|
|
XCTAssertFalse(state.isPaused)
|
|
}
|
|
}
|
|
|
|
func testTriggerReminderForLookAway() {
|
|
timerEngine.start()
|
|
|
|
timerEngine.triggerReminder(for: .builtIn(.lookAway))
|
|
|
|
XCTAssertNotNil(timerEngine.activeReminder)
|
|
if case .lookAwayTriggered(let countdown) = timerEngine.activeReminder {
|
|
XCTAssertEqual(countdown, settingsManager.settings.lookAwayCountdownSeconds)
|
|
} else {
|
|
XCTFail("Expected lookAwayTriggered reminder")
|
|
}
|
|
|
|
for (_, state) in timerEngine.timerStates {
|
|
XCTAssertTrue(state.isPaused)
|
|
}
|
|
}
|
|
|
|
func testTriggerReminderForBlink() {
|
|
timerEngine.start()
|
|
|
|
timerEngine.triggerReminder(for: .builtIn(.blink))
|
|
|
|
XCTAssertNotNil(timerEngine.activeReminder)
|
|
if case .blinkTriggered = timerEngine.activeReminder {
|
|
XCTAssertTrue(true)
|
|
} else {
|
|
XCTFail("Expected blinkTriggered reminder")
|
|
}
|
|
}
|
|
|
|
func testTriggerReminderForPosture() {
|
|
timerEngine.start()
|
|
|
|
timerEngine.triggerReminder(for: .builtIn(.posture))
|
|
|
|
XCTAssertNotNil(timerEngine.activeReminder)
|
|
if case .postureTriggered = timerEngine.activeReminder {
|
|
XCTAssertTrue(true)
|
|
} else {
|
|
XCTFail("Expected postureTriggered reminder")
|
|
}
|
|
}
|
|
|
|
func testGetTimeRemainingForNonExistentTimer() {
|
|
let timeRemaining = timerEngine.getTimeRemaining(for: .builtIn(.lookAway))
|
|
XCTAssertEqual(timeRemaining, 0)
|
|
}
|
|
|
|
func testGetFormattedTimeRemainingZeroSeconds() {
|
|
timerEngine.start()
|
|
timerEngine.timerStates[.builtIn(.lookAway)]?.remainingSeconds = 0
|
|
|
|
let formatted = timerEngine.getFormattedTimeRemaining(for: .builtIn(.lookAway))
|
|
XCTAssertEqual(formatted, "0:00")
|
|
}
|
|
|
|
func testGetFormattedTimeRemainingLessThanMinute() {
|
|
timerEngine.start()
|
|
timerEngine.timerStates[.builtIn(.lookAway)]?.remainingSeconds = 45
|
|
|
|
let formatted = timerEngine.getFormattedTimeRemaining(for: .builtIn(.lookAway))
|
|
XCTAssertEqual(formatted, "0:45")
|
|
}
|
|
|
|
func testGetFormattedTimeRemainingExactHour() {
|
|
timerEngine.start()
|
|
timerEngine.timerStates[.builtIn(.lookAway)]?.remainingSeconds = 3600
|
|
|
|
let formatted = timerEngine.getFormattedTimeRemaining(for: .builtIn(.lookAway))
|
|
XCTAssertEqual(formatted, "1:00:00")
|
|
}
|
|
|
|
func testMultipleStartCallsResetTimers() {
|
|
timerEngine.start()
|
|
timerEngine.timerStates[.builtIn(.lookAway)]?.remainingSeconds = 100
|
|
|
|
timerEngine.start()
|
|
|
|
XCTAssertEqual(timerEngine.timerStates[.builtIn(.lookAway)]?.remainingSeconds, 20 * 60)
|
|
}
|
|
|
|
func testSkipNextPreservesPausedState() {
|
|
timerEngine.start()
|
|
timerEngine.pause()
|
|
|
|
timerEngine.skipNext(identifier: .builtIn(.lookAway))
|
|
|
|
XCTAssertTrue(timerEngine.timerStates[.builtIn(.lookAway)]?.isPaused ?? false)
|
|
}
|
|
|
|
func testSkipNextPreservesActiveState() {
|
|
timerEngine.start()
|
|
|
|
timerEngine.skipNext(identifier: .builtIn(.lookAway))
|
|
|
|
XCTAssertTrue(timerEngine.timerStates[.builtIn(.lookAway)]?.isActive ?? false)
|
|
}
|
|
|
|
func testDismissReminderWithNoActiveReminder() {
|
|
timerEngine.start()
|
|
XCTAssertNil(timerEngine.activeReminder)
|
|
|
|
timerEngine.dismissReminder()
|
|
|
|
XCTAssertNil(timerEngine.activeReminder)
|
|
}
|
|
|
|
func testDismissBlinkReminderDoesNotResumeTimers() {
|
|
timerEngine.start()
|
|
timerEngine.activeReminder = .blinkTriggered
|
|
|
|
timerEngine.dismissReminder()
|
|
|
|
for (_, state) in timerEngine.timerStates {
|
|
XCTAssertFalse(state.isPaused)
|
|
}
|
|
}
|
|
|
|
func testDismissPostureReminderDoesNotResumeTimers() {
|
|
timerEngine.start()
|
|
timerEngine.activeReminder = .postureTriggered
|
|
|
|
timerEngine.dismissReminder()
|
|
|
|
for (_, state) in timerEngine.timerStates {
|
|
XCTAssertFalse(state.isPaused)
|
|
}
|
|
}
|
|
|
|
func testAllTimersStartWhenEnabled() {
|
|
settingsManager.settings.lookAwayTimer.enabled = true
|
|
settingsManager.settings.blinkTimer.enabled = true
|
|
settingsManager.settings.postureTimer.enabled = true
|
|
|
|
timerEngine.start()
|
|
|
|
XCTAssertEqual(timerEngine.timerStates.count, 3)
|
|
for builtInTimer in TimerType.allCases {
|
|
XCTAssertNotNil(timerEngine.timerStates[.builtIn(builtInTimer)])
|
|
}
|
|
}
|
|
|
|
func testAllTimersDisabled() {
|
|
settingsManager.settings.lookAwayTimer.enabled = false
|
|
settingsManager.settings.blinkTimer.enabled = false
|
|
settingsManager.settings.postureTimer.enabled = false
|
|
|
|
timerEngine.start()
|
|
|
|
XCTAssertEqual(timerEngine.timerStates.count, 0)
|
|
}
|
|
|
|
func testPartialTimersEnabled() {
|
|
settingsManager.settings.lookAwayTimer.enabled = true
|
|
settingsManager.settings.blinkTimer.enabled = false
|
|
settingsManager.settings.postureTimer.enabled = true
|
|
|
|
timerEngine.start()
|
|
|
|
XCTAssertEqual(timerEngine.timerStates.count, 2)
|
|
XCTAssertNotNil(timerEngine.timerStates[.builtIn(.lookAway)])
|
|
XCTAssertNil(timerEngine.timerStates[.builtIn(.blink)])
|
|
XCTAssertNotNil(timerEngine.timerStates[.builtIn(.posture)])
|
|
}
|
|
|
|
func testMultipleReminderTypesCanTriggerSimultaneously() {
|
|
// Setup: Create a user timer with overlay type (focus-stealing)
|
|
let overlayTimer = UserTimer(
|
|
title: "Water Break",
|
|
type: .overlay,
|
|
timeOnScreenSeconds: 10,
|
|
intervalMinutes: 1,
|
|
message: "Drink water"
|
|
)
|
|
settingsManager.settings.userTimers = [overlayTimer]
|
|
|
|
timerEngine.start()
|
|
|
|
// Trigger an overlay reminder (look away or user timer overlay)
|
|
timerEngine.triggerReminder(for: .user(id: overlayTimer.id))
|
|
|
|
// Verify overlay reminder is active
|
|
XCTAssertNotNil(timerEngine.activeReminder)
|
|
if case .userTimerTriggered(let timer) = timerEngine.activeReminder {
|
|
XCTAssertEqual(timer.id, overlayTimer.id)
|
|
XCTAssertEqual(timer.type, .overlay)
|
|
} else {
|
|
XCTFail("Expected userTimerTriggered with overlay type")
|
|
}
|
|
|
|
// Verify the overlay timer is paused
|
|
XCTAssertTrue(timerEngine.isTimerPaused(.user(id: overlayTimer.id)))
|
|
|
|
// Now trigger a subtle reminder (blink) while overlay is still active
|
|
let previousActiveReminder = timerEngine.activeReminder
|
|
timerEngine.triggerReminder(for: .builtIn(.blink))
|
|
|
|
// The activeReminder should be replaced with the blink reminder
|
|
// This is expected behavior - TimerEngine only tracks one activeReminder
|
|
XCTAssertNotNil(timerEngine.activeReminder)
|
|
if case .blinkTriggered = timerEngine.activeReminder {
|
|
XCTAssertTrue(true)
|
|
} else {
|
|
XCTFail("Expected blinkTriggered reminder")
|
|
}
|
|
|
|
// Both timers should be paused (the one that triggered their reminder)
|
|
XCTAssertTrue(timerEngine.isTimerPaused(.user(id: overlayTimer.id)))
|
|
XCTAssertTrue(timerEngine.isTimerPaused(.builtIn(.blink)))
|
|
|
|
// The key insight: Even though TimerEngine only tracks one activeReminder,
|
|
// AppDelegate now tracks overlay and subtle windows separately, so both
|
|
// reminders can be displayed simultaneously without interference
|
|
}
|
|
|
|
func testOverlayReminderDoesNotBlockSubtleReminders() {
|
|
// This test verifies the fix for the bug where a subtle reminder
|
|
// would cause an overlay reminder to get stuck
|
|
|
|
// Setup overlay user timer
|
|
let overlayTimer = UserTimer(
|
|
title: "Stand Up",
|
|
type: .overlay,
|
|
timeOnScreenSeconds: 10,
|
|
intervalMinutes: 1
|
|
)
|
|
settingsManager.settings.userTimers = [overlayTimer]
|
|
settingsManager.settings.blinkTimer.enabled = true
|
|
settingsManager.settings.blinkTimer.intervalSeconds = 60
|
|
|
|
timerEngine.start()
|
|
|
|
// Trigger overlay reminder first
|
|
timerEngine.triggerReminder(for: .user(id: overlayTimer.id))
|
|
XCTAssertNotNil(timerEngine.activeReminder)
|
|
XCTAssertTrue(timerEngine.isTimerPaused(.user(id: overlayTimer.id)))
|
|
|
|
// Trigger subtle reminder while overlay is active
|
|
timerEngine.triggerReminder(for: .builtIn(.blink))
|
|
|
|
// The blink reminder should now be active
|
|
if case .blinkTriggered = timerEngine.activeReminder {
|
|
XCTAssertTrue(true)
|
|
} else {
|
|
XCTFail("Expected blinkTriggered reminder")
|
|
}
|
|
|
|
// Both timers should be paused
|
|
XCTAssertTrue(timerEngine.isTimerPaused(.user(id: overlayTimer.id)))
|
|
XCTAssertTrue(timerEngine.isTimerPaused(.builtIn(.blink)))
|
|
|
|
// Dismiss the blink reminder
|
|
timerEngine.dismissReminder()
|
|
|
|
// After dismissing blink, the reminder should be cleared
|
|
XCTAssertNil(timerEngine.activeReminder)
|
|
|
|
// Blink timer should be reset and resumed
|
|
XCTAssertFalse(timerEngine.isTimerPaused(.builtIn(.blink)))
|
|
XCTAssertEqual(timerEngine.timerStates[.builtIn(.blink)]?.remainingSeconds, 60)
|
|
|
|
// The overlay timer should still be paused (user needs to dismiss it manually)
|
|
// Note: In the actual app, AppDelegate tracks this window separately and it
|
|
// remains visible even after the subtle reminder dismisses
|
|
XCTAssertTrue(timerEngine.isTimerPaused(.user(id: overlayTimer.id)))
|
|
}
|
|
}
|