Compare commits
2 Commits
da0cd302bf
...
ed8a160a49
| Author | SHA1 | Date | |
|---|---|---|---|
| ed8a160a49 | |||
| f42de3353c |
@@ -44,6 +44,8 @@ struct TimerView: View {
|
||||
.font(.system(size: 64, weight: .light, design: .monospaced))
|
||||
.foregroundStyle(viewModel.activeTimer != nil ? .primary : .secondary)
|
||||
|
||||
TimerUnitLabels(elapsed: viewModel.elapsedTime)
|
||||
|
||||
if let project = viewModel.selectedProject {
|
||||
ProjectColorBadge(
|
||||
color: project.color,
|
||||
@@ -123,6 +125,32 @@ struct TimerView: View {
|
||||
}
|
||||
}
|
||||
|
||||
/// Displays "h min sec" (or "min sec") column labels aligned under the
|
||||
/// monospaced timer digits.
|
||||
private struct TimerUnitLabels: View {
|
||||
let elapsed: TimeInterval
|
||||
|
||||
private var showHours: Bool { Int(elapsed) >= 3600 }
|
||||
|
||||
var body: some View {
|
||||
HStack(spacing: 0) {
|
||||
if showHours {
|
||||
Text("h")
|
||||
.frame(width: 64)
|
||||
Text("min")
|
||||
.frame(width: 64)
|
||||
} else {
|
||||
Text("min")
|
||||
.frame(width: 64)
|
||||
}
|
||||
Text("sec")
|
||||
.frame(width: 64)
|
||||
}
|
||||
.font(.caption)
|
||||
.foregroundStyle(.secondary)
|
||||
}
|
||||
}
|
||||
|
||||
struct ProjectPickerSheet: View {
|
||||
let projects: [Project]
|
||||
let selectedProject: Project?
|
||||
|
||||
@@ -3,7 +3,7 @@
|
||||
<plist version="1.0">
|
||||
<dict>
|
||||
<key>API_BASE_URL</key>
|
||||
<string>http://localhost:3001</string>
|
||||
<string>https://timetracker.simon-franken.de/api</string>
|
||||
<key>CFBundleDevelopmentRegion</key>
|
||||
<string>$(DEVELOPMENT_LANGUAGE)</string>
|
||||
<key>CFBundleExecutable</key>
|
||||
|
||||
@@ -9,9 +9,7 @@ struct OngoingTimer: Codable, Identifiable, Equatable {
|
||||
let updatedAt: String
|
||||
|
||||
var elapsedTime: TimeInterval {
|
||||
guard let start = ISO8601DateFormatter().date(from: startTime) else {
|
||||
return 0
|
||||
}
|
||||
guard let start = Date.fromISO8601(startTime) else { return 0 }
|
||||
return Date().timeIntervalSince(start)
|
||||
}
|
||||
}
|
||||
|
||||
@@ -11,10 +11,8 @@ struct TimeEntry: Codable, Identifiable, Equatable {
|
||||
let updatedAt: String
|
||||
|
||||
var duration: TimeInterval {
|
||||
guard let start = ISO8601DateFormatter().date(from: startTime),
|
||||
let end = ISO8601DateFormatter().date(from: endTime) else {
|
||||
return 0
|
||||
}
|
||||
guard let start = Date.fromISO8601(startTime),
|
||||
let end = Date.fromISO8601(endTime) else { return 0 }
|
||||
return end.timeIntervalSince(start)
|
||||
}
|
||||
}
|
||||
|
||||
@@ -59,8 +59,14 @@ extension Date {
|
||||
}
|
||||
|
||||
static func fromISO8601(_ string: String) -> Date? {
|
||||
let formatter = ISO8601DateFormatter()
|
||||
formatter.formatOptions = [.withInternetDateTime]
|
||||
return formatter.date(from: string)
|
||||
// Try with fractional seconds first (e.g. "2026-02-20T09:00:00.000Z" from
|
||||
// Prisma/Node.js JSON serialisation), then fall back to whole seconds.
|
||||
let withFractional = ISO8601DateFormatter()
|
||||
withFractional.formatOptions = [.withInternetDateTime, .withFractionalSeconds]
|
||||
if let date = withFractional.date(from: string) { return date }
|
||||
|
||||
let wholeSec = ISO8601DateFormatter()
|
||||
wholeSec.formatOptions = [.withInternetDateTime]
|
||||
return wholeSec.date(from: string)
|
||||
}
|
||||
}
|
||||
|
||||
@@ -2,9 +2,7 @@
|
||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||
<plist version="1.0">
|
||||
<dict>
|
||||
<key>com.apple.security.application-groups</key>
|
||||
<array>
|
||||
<string>group.com.timetracker.app</string>
|
||||
</array>
|
||||
<key>com.apple.security.application-groups</key>
|
||||
<array/>
|
||||
</dict>
|
||||
</plist>
|
||||
|
||||
@@ -2,28 +2,28 @@
|
||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||
<plist version="1.0">
|
||||
<dict>
|
||||
<key>CFBundleDevelopmentRegion</key>
|
||||
<string>$(DEVELOPMENT_LANGUAGE)</string>
|
||||
<key>CFBundleDisplayName</key>
|
||||
<string>Timer</string>
|
||||
<key>CFBundleExecutable</key>
|
||||
<string>$(EXECUTABLE_NAME)</string>
|
||||
<key>CFBundleIdentifier</key>
|
||||
<string>$(PRODUCT_BUNDLE_IDENTIFIER)</string>
|
||||
<key>CFBundleInfoDictionaryVersion</key>
|
||||
<string>6.0</string>
|
||||
<key>CFBundleName</key>
|
||||
<string>$(PRODUCT_NAME)</string>
|
||||
<key>CFBundlePackageType</key>
|
||||
<string>$(PRODUCT_BUNDLE_PACKAGE_TYPE)</string>
|
||||
<key>CFBundleShortVersionString</key>
|
||||
<string>$(MARKETING_VERSION)</string>
|
||||
<key>CFBundleVersion</key>
|
||||
<string>$(CURRENT_PROJECT_VERSION)</string>
|
||||
<key>NSExtension</key>
|
||||
<dict>
|
||||
<key>NSExtensionPointIdentifier</key>
|
||||
<string>com.apple.widgetkit-extension</string>
|
||||
</dict>
|
||||
<key>CFBundleDevelopmentRegion</key>
|
||||
<string>$(DEVELOPMENT_LANGUAGE)</string>
|
||||
<key>CFBundleDisplayName</key>
|
||||
<string>Timer</string>
|
||||
<key>CFBundleExecutable</key>
|
||||
<string>$(EXECUTABLE_NAME)</string>
|
||||
<key>CFBundleIdentifier</key>
|
||||
<string>$(PRODUCT_BUNDLE_IDENTIFIER)</string>
|
||||
<key>CFBundleInfoDictionaryVersion</key>
|
||||
<string>6.0</string>
|
||||
<key>CFBundleName</key>
|
||||
<string>$(PRODUCT_NAME)</string>
|
||||
<key>CFBundlePackageType</key>
|
||||
<string>$(PRODUCT_BUNDLE_PACKAGE_TYPE)</string>
|
||||
<key>CFBundleShortVersionString</key>
|
||||
<string>$(MARKETING_VERSION)</string>
|
||||
<key>CFBundleVersion</key>
|
||||
<string>$(CURRENT_PROJECT_VERSION)</string>
|
||||
<key>NSExtension</key>
|
||||
<dict>
|
||||
<key>NSExtensionPointIdentifier</key>
|
||||
<string>com.apple.widgetkit-extension</string>
|
||||
</dict>
|
||||
</dict>
|
||||
</plist>
|
||||
|
||||
@@ -2,9 +2,7 @@
|
||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||
<plist version="1.0">
|
||||
<dict>
|
||||
<key>com.apple.security.application-groups</key>
|
||||
<array>
|
||||
<string>group.com.timetracker.app</string>
|
||||
</array>
|
||||
<key>com.apple.security.application-groups</key>
|
||||
<array/>
|
||||
</dict>
|
||||
</plist>
|
||||
|
||||
Reference in New Issue
Block a user