Redesign Activity tab with animated rings, monthly calendar, and global stats
Some checks failed
CI / TypeScript (push) Failing after 7s
CI / ESLint (push) Failing after 3s
CI / Tests (push) Failing after 7s
CI / Build Check (push) Has been skipped
CI / Admin Web Tests (push) Successful in 2m6s
CI / Deploy Edge Functions (push) Has been skipped

This commit is contained in:
Millian Lamiaux
2026-04-22 01:18:42 +02:00
parent d74c47b1a8
commit cf096f2068
6 changed files with 389 additions and 152 deletions

View File

@@ -14,6 +14,7 @@
192F8CFFE1888005ABF339E8 /* WorkoutSession.swift in Sources */ = {isa = PBXBuildFile; fileRef = BBBBFC7FC6A52DE9908EE4A6 /* WorkoutSession.swift */; };
1955D0D74D9B09D10705104C /* WorkoutProgram.swift in Sources */ = {isa = PBXBuildFile; fileRef = B6E64CFB210A549AC85F878D /* WorkoutProgram.swift */; };
20FD0BC9A6E01E8EA182E030 /* Supabase in Frameworks */ = {isa = PBXBuildFile; productRef = 2A66A0F120927A9EBC548828 /* Supabase */; };
22669D283A2B7C8D5F4FE19F /* ActivityRingView.swift in Sources */ = {isa = PBXBuildFile; fileRef = ACA8445E547E41DF784C3D2F /* ActivityRingView.swift */; };
29DA1C9905E244CDC316D5AA /* CompletionView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 44A6F4FB39AE902BCED1C2D5 /* CompletionView.swift */; };
2D5CE02211FB67CD2CFDAA11 /* SupabaseService.swift in Sources */ = {isa = PBXBuildFile; fileRef = 4EDF0ED76A8476BF1F80EF8C /* SupabaseService.swift */; };
367B00BF0E8537F9BA15530F /* MusicTrack.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2C6156C6E0E1A543DAC87A90 /* MusicTrack.swift */; };
@@ -21,9 +22,11 @@
3E2E78027B1973F72E05D8D2 /* PlayerViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8206D4F904F61E5685DE369E /* PlayerViewModel.swift */; };
3F0F63E163BBA968C4CEFF81 /* TabataGoWatch.app in Embed Watch Content */ = {isa = PBXBuildFile; fileRef = 484865AEFA8CCD26C4AE7F73 /* TabataGoWatch.app */; settings = {ATTRIBUTES = (RemoveHeadersOnCopy, ); }; };
3FAAAAC1576A7861AB833E39 /* ProfileTab.swift in Sources */ = {isa = PBXBuildFile; fileRef = 9B5DFE227FDB6400C8D7A4A4 /* ProfileTab.swift */; };
41096438BB67B60480460156 /* WatchL10n.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5ACBDB7D81F575AC5370E82F /* WatchL10n.swift */; };
4371D8DD5F2638905606513A /* WatchActivityView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 2D8CA90001C65B27E3B7BE34 /* WatchActivityView.swift */; };
53FDC12EFCD8159045C105C0 /* HealthKitService.swift in Sources */ = {isa = PBXBuildFile; fileRef = 0499FCA348FF1A127C8E4FAE /* HealthKitService.swift */; };
556620C10FA0BC85E1BDE529 /* WatchIdleView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 495E38AB3B412E296F8C3649 /* WatchIdleView.swift */; };
583A4EB0B1D20521D4A37D60 /* GlobalStatsCard.swift in Sources */ = {isa = PBXBuildFile; fileRef = D66750D77B59FCF4F321B36E /* GlobalStatsCard.swift */; };
59B482DEBAA43EE5F24B883D /* HomeTab.swift in Sources */ = {isa = PBXBuildFile; fileRef = FB04FA5E81BD1E52DEFB3AC2 /* HomeTab.swift */; };
5A25DA9A1B21F5EED15BA370 /* ProgramDetailView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6BFF744890571DE314540E16 /* ProgramDetailView.swift */; };
5A402D7E31059AB7107B625C /* MusicPlayerViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5505FBD6E001AE3AFD413ADA /* MusicPlayerViewModel.swift */; };
@@ -32,6 +35,7 @@
6060D95D485E4188EAABDDED /* WatchRootView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 8AEC37E6361DC4C7AE326139 /* WatchRootView.swift */; };
61BD8C313424F89F13FDE92E /* PurchaseViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = B8E005D62F3B3B80A2A53C2C /* PurchaseViewModel.swift */; };
66E87ABBDC5C3B36B3E932FB /* WatchConnectivityManager.swift in Sources */ = {isa = PBXBuildFile; fileRef = 802638FA5E5FDB5B278123AC /* WatchConnectivityManager.swift */; };
6C11E1F61F3ABADC7922C383 /* WeeklySection.swift in Sources */ = {isa = PBXBuildFile; fileRef = 75C8C8AEF5C3E8432CB07861 /* WeeklySection.swift */; };
70C2DAC704F628494A59EF56 /* Theme.swift in Sources */ = {isa = PBXBuildFile; fileRef = 61E5AA44513F793EA7FEBA00 /* Theme.swift */; };
70DEC8E97218C774A46F7CEA /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = CDFE1E10182972315386F9D7 /* Assets.xcassets */; };
725EBACF4CF7BC23D2C476AA /* TabataGoApp.swift in Sources */ = {isa = PBXBuildFile; fileRef = D09EB765FCE6A3EE95E86EB3 /* TabataGoApp.swift */; };
@@ -44,6 +48,7 @@
90728D374B15A38DD9A75E5B /* WatchPlayerView.swift in Sources */ = {isa = PBXBuildFile; fileRef = DBC464C4D17B88E57FB5477C /* WatchPlayerView.swift */; };
93457B73C62C4BEA4329BD4C /* TabataGoWatchWidget.appex in Embed Foundation Extensions */ = {isa = PBXBuildFile; fileRef = 255972F9906563A0921C47C0 /* TabataGoWatchWidget.appex */; settings = {ATTRIBUTES = (RemoveHeadersOnCopy, ); }; };
9633A730F910E47C28A288AC /* WatchConnectivityTypes.swift in Sources */ = {isa = PBXBuildFile; fileRef = D8425C668A3901B0F12DBFCD /* WatchConnectivityTypes.swift */; };
996E613C0A9906AB88D2AEB6 /* WorkoutCalendarView.swift in Sources */ = {isa = PBXBuildFile; fileRef = FAEBB2380DAC8F565F556D41 /* WorkoutCalendarView.swift */; };
9F9695303EEC1516B1845417 /* TabataGoWatchApp.swift in Sources */ = {isa = PBXBuildFile; fileRef = 9006191EE89D06E6558786E3 /* TabataGoWatchApp.swift */; };
AA17AD2E25DF408ECE100F99 /* PreviewData.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7482C05380DE017FF582C28B /* PreviewData.swift */; };
B4CFD4E752EF66F6535AD173 /* SettingsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 815C7C1CC22063B7E27F2F9B /* SettingsView.swift */; };
@@ -57,6 +62,7 @@
D422758C736D40CB0E9C4063 /* NowPlayingView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 31B4E4F3DA1047ACD980F581 /* NowPlayingView.swift */; };
D65673484CBB4DDA03C23225 /* RootView.swift in Sources */ = {isa = PBXBuildFile; fileRef = F8242C26A4F51BE7AA779840 /* RootView.swift */; };
D665638A80E06A7C42019782 /* PlayerView.swift in Sources */ = {isa = PBXBuildFile; fileRef = E93E214AAB0E1CB61B89EC75 /* PlayerView.swift */; };
D708AE54AD57CE932F9880DA /* Localizable.xcstrings in Resources */ = {isa = PBXBuildFile; fileRef = C4C127C41584515D4EF95CB0 /* Localizable.xcstrings */; };
DBFB6F75F59367A957B8F9B9 /* UserProfile.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7FE34000653EE789117CE9D9 /* UserProfile.swift */; };
E21B9936D15D2111807AAAE9 /* OnboardingView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 28A422BCE1A702F8A10951FC /* OnboardingView.swift */; };
E4ED0B8CABBD3502EA468F21 /* HomeViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = AD1AF33C89F42294599C369A /* HomeViewModel.swift */; };
@@ -145,6 +151,7 @@
525C7E8EC6EF89E00D34672E /* PolicyViews.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PolicyViews.swift; sourceTree = "<group>"; };
5505FBD6E001AE3AFD413ADA /* MusicPlayerViewModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MusicPlayerViewModel.swift; sourceTree = "<group>"; };
58DEACB2D18F636B35BB2C48 /* TabataGoSchema.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TabataGoSchema.swift; sourceTree = "<group>"; };
5ACBDB7D81F575AC5370E82F /* WatchL10n.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = WatchL10n.swift; sourceTree = "<group>"; };
5F5D3568A736B7A326874677 /* Localizable.xcstrings */ = {isa = PBXFileReference; lastKnownFileType = text.json.xcstrings; path = Localizable.xcstrings; sourceTree = "<group>"; };
61E5AA44513F793EA7FEBA00 /* Theme.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Theme.swift; sourceTree = "<group>"; };
63599808389B70FC2F6A43C3 /* HealthViewModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = HealthViewModel.swift; sourceTree = "<group>"; };
@@ -152,6 +159,7 @@
6BFF744890571DE314540E16 /* ProgramDetailView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ProgramDetailView.swift; sourceTree = "<group>"; };
6D558DAFE1AD94786AA674A4 /* TabataGoUITests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TabataGoUITests.swift; sourceTree = "<group>"; };
7482C05380DE017FF582C28B /* PreviewData.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PreviewData.swift; sourceTree = "<group>"; };
75C8C8AEF5C3E8432CB07861 /* WeeklySection.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = WeeklySection.swift; sourceTree = "<group>"; };
7FE34000653EE789117CE9D9 /* UserProfile.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UserProfile.swift; sourceTree = "<group>"; };
802638FA5E5FDB5B278123AC /* WatchConnectivityManager.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = WatchConnectivityManager.swift; sourceTree = "<group>"; };
815C7C1CC22063B7E27F2F9B /* SettingsView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SettingsView.swift; sourceTree = "<group>"; };
@@ -165,18 +173,21 @@
9EC19129CD3C493C8B2AEFA8 /* AppState.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AppState.swift; sourceTree = "<group>"; };
A7C07E8AF566483359CE2FEC /* TabataGoTests.xctest */ = {isa = PBXFileReference; includeInIndex = 0; lastKnownFileType = wrapper.cfbundle; path = TabataGoTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; };
A84A0F7F17713D5D0A679122 /* PurchaseService.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PurchaseService.swift; sourceTree = "<group>"; };
ACA8445E547E41DF784C3D2F /* ActivityRingView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ActivityRingView.swift; sourceTree = "<group>"; };
AD1AF33C89F42294599C369A /* HomeViewModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = HomeViewModel.swift; sourceTree = "<group>"; };
B6E64CFB210A549AC85F878D /* WorkoutProgram.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = WorkoutProgram.swift; sourceTree = "<group>"; };
B7EDA5BF7F25E3279A4B1A61 /* TabataGoUITests.xctest */ = {isa = PBXFileReference; includeInIndex = 0; lastKnownFileType = wrapper.cfbundle; path = TabataGoUITests.xctest; sourceTree = BUILT_PRODUCTS_DIR; };
B8E005D62F3B3B80A2A53C2C /* PurchaseViewModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PurchaseViewModel.swift; sourceTree = "<group>"; };
BBBBFC7FC6A52DE9908EE4A6 /* WorkoutSession.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = WorkoutSession.swift; sourceTree = "<group>"; };
BD3DF875E3461305DADB554A /* Secrets.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; path = Secrets.xcconfig; sourceTree = "<group>"; };
C4C127C41584515D4EF95CB0 /* Localizable.xcstrings */ = {isa = PBXFileReference; lastKnownFileType = text.json.xcstrings; path = Localizable.xcstrings; sourceTree = "<group>"; };
C4C9BB1EEE2291A9A23B5F3C /* MusicService.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MusicService.swift; sourceTree = "<group>"; };
CDA5D50FD057EF30BE7915F5 /* TabataGoComplication.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TabataGoComplication.swift; sourceTree = "<group>"; };
CDFE1E10182972315386F9D7 /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = "<group>"; };
D09EB765FCE6A3EE95E86EB3 /* TabataGoApp.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TabataGoApp.swift; sourceTree = "<group>"; };
D168B973B16C94426A15766A /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist; path = Info.plist; sourceTree = "<group>"; };
D593D23B6A2F633DFA166D91 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist; path = Info.plist; sourceTree = "<group>"; };
D66750D77B59FCF4F321B36E /* GlobalStatsCard.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = GlobalStatsCard.swift; sourceTree = "<group>"; };
D8425C668A3901B0F12DBFCD /* WatchConnectivityTypes.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = WatchConnectivityTypes.swift; path = ../../TabataGo/Services/WatchConnectivityTypes.swift; sourceTree = "<group>"; };
D8A69F6B8DC5329436762B50 /* TabataGo.app */ = {isa = PBXFileReference; includeInIndex = 0; lastKnownFileType = wrapper.application; path = TabataGo.app; sourceTree = BUILT_PRODUCTS_DIR; };
D983B6DEDE62A8F0E9E09E66 /* TabataGoWatch.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = TabataGoWatch.entitlements; sourceTree = "<group>"; };
@@ -185,6 +196,7 @@
E93E214AAB0E1CB61B89EC75 /* PlayerView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PlayerView.swift; sourceTree = "<group>"; };
F1DE8A4DAD846A879B8ED379 /* HealthSnapshot.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = HealthSnapshot.swift; sourceTree = "<group>"; };
F8242C26A4F51BE7AA779840 /* RootView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RootView.swift; sourceTree = "<group>"; };
FAEBB2380DAC8F565F556D41 /* WorkoutCalendarView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = WorkoutCalendarView.swift; sourceTree = "<group>"; };
FB04FA5E81BD1E52DEFB3AC2 /* HomeTab.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = HomeTab.swift; sourceTree = "<group>"; };
/* End PBXFileReference section */
@@ -245,6 +257,7 @@
1F1136D2B7FEBD67D18C7679 /* Resources */ = {
isa = PBXGroup;
children = (
C4C127C41584515D4EF95CB0 /* Localizable.xcstrings */,
D983B6DEDE62A8F0E9E09E66 /* TabataGoWatch.entitlements */,
);
path = Resources;
@@ -312,14 +325,27 @@
0FB4710828254C8629904415 /* Complications */,
1F1136D2B7FEBD67D18C7679 /* Resources */,
F514B75119B8194DB1791B95 /* Services */,
CE5E34713E694FAEED0F3480 /* Utilities */,
4B4FB2F47AD5A34A21679372 /* Views */,
);
path = TabataGoWatch;
sourceTree = "<group>";
};
850B47F4DD96E9CD7D6F412A /* Activity */ = {
isa = PBXGroup;
children = (
ACA8445E547E41DF784C3D2F /* ActivityRingView.swift */,
D66750D77B59FCF4F321B36E /* GlobalStatsCard.swift */,
75C8C8AEF5C3E8432CB07861 /* WeeklySection.swift */,
FAEBB2380DAC8F565F556D41 /* WorkoutCalendarView.swift */,
);
path = Activity;
sourceTree = "<group>";
};
8B90DED418BDA3697748C37D /* Tabs */ = {
isa = PBXGroup;
children = (
850B47F4DD96E9CD7D6F412A /* Activity */,
84123E854DE0BF3E0D4F0912 /* ActivityTab.swift */,
FB04FA5E81BD1E52DEFB3AC2 /* HomeTab.swift */,
12715936CAA6BD90A7FBE9D7 /* MainTabView.swift */,
@@ -413,6 +439,14 @@
path = ViewModels;
sourceTree = "<group>";
};
CE5E34713E694FAEED0F3480 /* Utilities */ = {
isa = PBXGroup;
children = (
5ACBDB7D81F575AC5370E82F /* WatchL10n.swift */,
);
path = Utilities;
sourceTree = "<group>";
};
D72E5B5E6AA55AA4DD27D119 /* App */ = {
isa = PBXGroup;
children = (
@@ -495,6 +529,7 @@
buildConfigurationList = E706A289AD3B8CCB1F3310BE /* Build configuration list for PBXNativeTarget "TabataGoWatch" */;
buildPhases = (
688CFA91471FB46E21B9EFB2 /* Sources */,
ABE017E9BF179D97311AB485 /* Resources */,
97F207A5CEE6835FA097805C /* Embed Foundation Extensions */,
);
buildRules = (
@@ -641,6 +676,14 @@
);
runOnlyForDeploymentPostprocessing = 0;
};
ABE017E9BF179D97311AB485 /* Resources */ = {
isa = PBXResourcesBuildPhase;
buildActionMask = 2147483647;
files = (
D708AE54AD57CE932F9880DA /* Localizable.xcstrings in Resources */,
);
runOnlyForDeploymentPostprocessing = 0;
};
/* End PBXResourcesBuildPhase section */
/* Begin PBXSourcesBuildPhase section */
@@ -669,6 +712,7 @@
66E87ABBDC5C3B36B3E932FB /* WatchConnectivityManager.swift in Sources */,
B5591230E8B61A2B18F5DD87 /* WatchConnectivityTypes.swift in Sources */,
556620C10FA0BC85E1BDE529 /* WatchIdleView.swift in Sources */,
41096438BB67B60480460156 /* WatchL10n.swift in Sources */,
850C700B060F46134C2D4569 /* WatchPlayerEngine.swift in Sources */,
90728D374B15A38DD9A75E5B /* WatchPlayerView.swift in Sources */,
6060D95D485E4188EAABDDED /* WatchRootView.swift in Sources */,
@@ -679,6 +723,7 @@
isa = PBXSourcesBuildPhase;
buildActionMask = 2147483647;
files = (
22669D283A2B7C8D5F4FE19F /* ActivityRingView.swift in Sources */,
60503F963221C7FCF719C493 /* ActivityTab.swift in Sources */,
CCCCEFD2D61ED1D7DDB9040C /* AnalyticsService.swift in Sources */,
EE6C591611D52C36ED5E03C6 /* AppState.swift in Sources */,
@@ -686,6 +731,7 @@
CDFA9A56DB6DA111B728FF48 /* BodyZoneView.swift in Sources */,
29DA1C9905E244CDC316D5AA /* CompletionView.swift in Sources */,
C2CB48B999939D3550A50936 /* Environment.swift in Sources */,
583A4EB0B1D20521D4A37D60 /* GlobalStatsCard.swift in Sources */,
53FDC12EFCD8159045C105C0 /* HealthKitService.swift in Sources */,
5CE2F2210BEF17AC304F2AC2 /* HealthSnapshot.swift in Sources */,
FE14257B8CFFDC47C72AE079 /* HealthViewModel.swift in Sources */,
@@ -717,6 +763,8 @@
70C2DAC704F628494A59EF56 /* Theme.swift in Sources */,
DBFB6F75F59367A957B8F9B9 /* UserProfile.swift in Sources */,
9633A730F910E47C28A288AC /* WatchConnectivityTypes.swift in Sources */,
6C11E1F61F3ABADC7922C383 /* WeeklySection.swift in Sources */,
996E613C0A9906AB88D2AEB6 /* WorkoutCalendarView.swift in Sources */,
1955D0D74D9B09D10705104C /* WorkoutProgram.swift in Sources */,
192F8CFFE1888005ABF339E8 /* WorkoutSession.swift in Sources */,
);

View File

@@ -0,0 +1,81 @@
import SwiftUI
struct ActivityRingView: View {
let moveValue: Double
let moveGoal: Double
let exerciseValue: Double
let exerciseGoal: Double
let standValue: Double
let standGoal: Double
@State private var animatedProgress: Bool = false
private let ringWidth: CGFloat = 14
private let spacing: CGFloat = 8
private let size: CGFloat = 160
var body: some View {
VStack(spacing: 16) {
ZStack {
moveRing
exerciseRing
standRing
}
.frame(width: size, height: size)
.onAppear { withAnimation(.easeInOut(duration: 1.2)) { animatedProgress = true } }
HStack(spacing: 0) {
ringLegend(value: Int(moveValue), unit: "kcal", label: L10n.health.move, color: .red)
ringLegend(value: Int(exerciseValue), unit: "min", label: L10n.health.exercise, color: .green)
ringLegend(value: Int(standValue), unit: "hrs", label: L10n.health.stand, color: .cyan)
}
}
}
private var moveRing: some View {
ringTrack(progress: animatedProgress ? min(moveValue / max(moveGoal, 1), 1.0) : 0,
color: .red,
radius: size / 2)
}
private var exerciseRing: some View {
ringTrack(progress: animatedProgress ? min(exerciseValue / max(exerciseGoal, 1), 1.0) : 0,
color: .green,
radius: size / 2 - ringWidth - spacing)
}
private var standRing: some View {
ringTrack(progress: animatedProgress ? min(standValue / max(standGoal, 1), 1.0) : 0,
color: .cyan,
radius: size / 2 - 2 * (ringWidth + spacing))
}
private func ringTrack(progress: Double, color: Color, radius: CGFloat) -> some View {
Circle()
.trim(from: 0, to: progress)
.stroke(color, style: StrokeStyle(lineWidth: ringWidth, lineCap: .round))
.rotationEffect(.degrees(-90))
.frame(width: radius * 2, height: radius * 2)
.background(
Circle()
.stroke(color.opacity(0.15), lineWidth: ringWidth)
.frame(width: radius * 2, height: radius * 2)
)
}
private func ringLegend(value: Int, unit: String, label: LocalizedStringResource, color: Color) -> some View {
VStack(spacing: 2) {
Text("\(value)")
.font(.system(size: 18, weight: .bold, design: .rounded))
.monospacedDigit()
.foregroundStyle(color)
Text(unit)
.font(.caption2.weight(.semibold))
.foregroundStyle(color.opacity(0.7))
Text(label)
.font(.caption2)
.foregroundStyle(.secondary)
}
.frame(maxWidth: .infinity)
}
}

View File

@@ -0,0 +1,68 @@
import SwiftUI
struct GlobalStatsCard: View {
let totalSessions: Int
let totalMinutes: Int
let totalCalories: Double
let currentStreak: Int
let longestStreak: Int
var body: some View {
VStack(spacing: 16) {
HStack(spacing: 0) {
globalStat(value: "\(totalSessions)", label: L10n.activity.workouts, color: Theme.brand, icon: "flame.fill")
globalStat(value: "\(totalMinutes)", label: L10n.activity.minutes, color: Theme.rest, icon: "clock.fill")
globalStat(value: "\(Int(totalCalories))", label: "kcal", color: Theme.success, icon: "bolt.fill")
}
Divider().opacity(0.3)
HStack(spacing: 0) {
streakStat(value: currentStreak, label: L10n.activity.currentStreak, color: Theme.brand)
streakStat(value: longestStreak, label: L10n.activity.bestStreak, color: Theme.success)
}
}
.padding(16)
.glassCard()
}
private func globalStat(value: String, label: LocalizedStringResource, color: Color, icon: String) -> some View {
VStack(spacing: 4) {
Image(systemName: icon)
.font(.system(size: 14, weight: .semibold))
.foregroundStyle(color)
Text(value)
.font(.system(size: 22, weight: .bold, design: .rounded))
.monospacedDigit()
.foregroundStyle(.primary)
Text(label)
.font(.caption)
.foregroundStyle(.secondary)
}
.frame(maxWidth: .infinity)
}
private func streakStat(value: Int, label: LocalizedStringResource, color: Color) -> some View {
HStack(spacing: 8) {
Image(systemName: "flame.fill")
.font(.system(size: 20))
.foregroundStyle(color)
VStack(alignment: .leading, spacing: 2) {
HStack(alignment: .lastTextBaseline, spacing: 2) {
Text("\(value)")
.font(.system(size: 24, weight: .black, design: .rounded))
.monospacedDigit()
.foregroundStyle(color)
Text(L10n.activity.days)
.font(.caption)
.foregroundStyle(.secondary)
}
Text(label)
.font(.caption2)
.foregroundStyle(.secondary)
}
Spacer(minLength: 0)
}
.frame(maxWidth: .infinity)
}
}

View File

@@ -0,0 +1,33 @@
import SwiftUI
struct WeeklySection: View {
let snapshot: HealthSnapshot?
let weeklyWorkouts: Int
let weeklyMinutes: Int
let weeklyCalories: Double
var body: some View {
VStack(alignment: .leading, spacing: 14) {
Text(L10n.home.thisWeek)
.font(.title3.weight(.bold))
.padding(.horizontal)
ActivityRingView(
moveValue: snapshot?.activeCaloricBurn ?? weeklyCalories,
moveGoal: 600,
exerciseValue: snapshot?.exerciseMinutes ?? Double(weeklyMinutes),
exerciseGoal: 30,
standValue: Double(snapshot?.standHours ?? 0),
standGoal: 12
)
.frame(maxWidth: .infinity)
HStack(spacing: 12) {
StatBadge(label: L10n.activity.workouts, value: "\(weeklyWorkouts)", color: Theme.brand, icon: "flame.fill")
StatBadge(label: L10n.activity.minutes, value: "\(weeklyMinutes)", color: Theme.rest, icon: "clock.fill")
StatBadge(label: "kcal", value: "\(Int(weeklyCalories))", color: Theme.success, icon: "bolt.fill")
}
.padding(.horizontal)
}
}
}

View File

@@ -0,0 +1,128 @@
import SwiftUI
struct WorkoutCalendarView: View {
let activeDays: Set<DateComponents>
@State private var displayedMonth = Date()
private let calendar = Calendar.current
private let columns = Array(repeating: GridItem(.flexible(), spacing: 4), count: 7)
var body: some View {
VStack(spacing: 12) {
monthHeader
weekdayHeaders
dayGrid
}
.padding(16)
.glassCard()
}
private var monthHeader: some View {
HStack {
Button { withAnimation { displayedMonth = calendar.date(byAdding: .month, value: -1, to: displayedMonth) ?? displayedMonth } } label: {
Image(systemName: "chevron.left")
.font(.system(size: 14, weight: .semibold))
.foregroundStyle(.secondary)
}
Spacer()
Text(monthName)
.font(.headline)
.foregroundStyle(.primary)
Spacer()
Button { withAnimation { displayedMonth = calendar.date(byAdding: .month, value: 1, to: displayedMonth) ?? displayedMonth } } label: {
Image(systemName: "chevron.right")
.font(.system(size: 14, weight: .semibold))
.foregroundStyle(.secondary)
}
}
}
private var weekdayHeaders: some View {
HStack(spacing: 4) {
ForEach(weekdaySymbols, id: \.self) { symbol in
Text(symbol)
.font(.caption2.weight(.semibold))
.foregroundStyle(.secondary)
.frame(maxWidth: .infinity)
}
}
}
private var dayGrid: some View {
let days = daysInMonth()
let prefix = firstWeekdayOffset()
return LazyVGrid(columns: columns, spacing: 4) {
ForEach(0..<prefix, id: \.self) { _ in
Color.clear.frame(height: 36)
}
ForEach(days, id: \.self) { day in
dayCell(day)
}
}
}
private func dayCell(_ day: Date) -> some View {
let isToday = calendar.isDateInToday(day)
let isActive = activeDays.contains(calendar.dateComponents([.year, .month, .day], from: day))
let dayNum = calendar.component(.day, from: day)
return ZStack {
if isActive {
Circle()
.fill(Theme.brand.gradient)
.frame(width: 32, height: 32)
} else if isToday {
Circle()
.stroke(Theme.brand.opacity(0.5), lineWidth: 1.5)
.frame(width: 32, height: 32)
}
Text("\(dayNum)")
.font(.system(size: 14, weight: isActive ? .bold : .regular, design: .rounded))
.monospacedDigit()
.foregroundStyle(isActive ? .white : (isToday ? Theme.brand : .primary))
}
.frame(height: 36)
}
private var monthName: String {
let formatter = DateFormatter()
formatter.locale = Locale.current
formatter.dateFormat = "MMMM yyyy"
let str = formatter.string(from: displayedMonth)
return str.prefix(1).uppercased() + str.dropFirst()
}
private var weekdaySymbols: [String] {
let formatter = DateFormatter()
formatter.locale = Locale.current
let symbols = formatter.veryShortWeekdaySymbols ?? []
let first = calendar.firstWeekday
let reordered = Array(symbols[(first - 1)..<symbols.count]) + symbols[0..<(first - 1)]
return reordered
}
private func daysInMonth() -> [Date] {
guard let interval = calendar.dateInterval(of: .month, for: displayedMonth) else { return [] }
var days: [Date] = []
var current = interval.start
while current < interval.end {
days.append(current)
current = calendar.date(byAdding: .day, value: 1, to: current) ?? current
}
return days
}
private func firstWeekdayOffset() -> Int {
guard let interval = calendar.dateInterval(of: .month, for: displayedMonth),
let firstDay = calendar.dateInterval(of: .weekOfYear, for: interval.start)?.start else { return 0 }
let diff = calendar.dateComponents([.day], from: firstDay, to: interval.start).day ?? 0
return max(0, diff)
}
}

View File

@@ -1,7 +1,6 @@
import SwiftUI
import SwiftData
/// Activity tab streak, workout history, HealthKit rings summary.
struct ActivityTab: View {
@Query(sort: \WorkoutSession.completedAt, order: .reverse)
private var sessions: [WorkoutSession]
@@ -12,37 +11,35 @@ struct ActivityTab: View {
private var streak: (current: Int, longest: Int) { computeStreak(from: sessions) }
private var snapshot: HealthSnapshot? { snapshots.first }
private var weeklyCount: Int { countThisWeek(sessions) }
private var activeDays: Set<DateComponents> {
let calendar = Calendar.current
return Set(sessions.map { calendar.dateComponents([.year, .month, .day], from: $0.completedAt) })
}
var body: some View {
NavigationStack {
ScrollView {
VStack(spacing: 20) {
VStack(spacing: 24) {
// Streak Banner
StreakBanner(current: streak.current, longest: streak.longest)
GlobalStatsCard(
totalSessions: sessions.count,
totalMinutes: totalMinutes,
totalCalories: totalCalories,
currentStreak: streak.current,
longestStreak: streak.longest
)
.padding(.horizontal)
WeeklySection(
snapshot: snapshot,
weeklyWorkouts: weeklyCount,
weeklyMinutes: weeklyMinutes,
weeklyCalories: weeklyCalories
)
WorkoutCalendarView(activeDays: activeDays)
.padding(.horizontal)
// HealthKit Rings
if let snap = snapshot {
HealthRingsCard(snapshot: snap)
.padding(.horizontal)
}
// Weekly Summary
VStack(alignment: .leading, spacing: 12) {
Text(L10n.home.thisWeek)
.font(.title3.weight(.bold))
.padding(.horizontal)
HStack(spacing: 12) {
StatBadge(label: L10n.activity.workouts, value: "\(weeklyCount)", color: Theme.brand, icon: "flame.fill")
StatBadge(label: L10n.activity.minutes, value: "\(weeklyMinutes)", color: Theme.rest, icon: "clock.fill")
StatBadge(label: L10n.activity.calories, value: "\(Int(weeklyCalories))", color: Theme.success, icon: "bolt.fill")
}
.padding(.horizontal)
}
// Workout History
if !sessions.isEmpty {
VStack(alignment: .leading, spacing: 12) {
Text(L10n.activity.history)
@@ -72,6 +69,14 @@ struct ActivityTab: View {
}
}
private var totalMinutes: Int {
sessions.reduce(0) { $0 + $1.durationSeconds / 60 }
}
private var totalCalories: Double {
sessions.reduce(0) { $0 + $1.caloriesBurned }
}
private var weeklyMinutes: Int {
countThisWeek(sessions, value: { $0.durationSeconds / 60 })
}
@@ -82,139 +87,13 @@ struct ActivityTab: View {
}
}
// Sub-components
struct StreakBanner: View {
let current: Int
let longest: Int
var body: some View {
HStack(spacing: 0) {
// Current streak
VStack(spacing: 4) {
HStack(alignment: .lastTextBaseline, spacing: 4) {
Text("\(current)")
.font(.system(size: 52, weight: .black, design: .rounded))
.monospacedDigit()
.foregroundStyle(Theme.brand)
Text(L10n.activity.days)
.font(.headline)
.foregroundStyle(.secondary)
}
Text(L10n.activity.currentStreak)
.font(.subheadline)
.foregroundStyle(.secondary)
}
.frame(maxWidth: .infinity)
Divider().frame(height: 50)
// Longest streak
VStack(spacing: 4) {
HStack(alignment: .lastTextBaseline, spacing: 4) {
Text("\(longest)")
.font(.system(size: 52, weight: .black, design: .rounded))
.monospacedDigit()
.foregroundStyle(Theme.success)
Text(L10n.activity.days)
.font(.headline)
.foregroundStyle(.secondary)
}
Text(L10n.activity.bestStreak)
.font(.subheadline)
.foregroundStyle(.secondary)
}
.frame(maxWidth: .infinity)
}
.padding(.vertical, 16)
.glassCard()
}
}
struct HealthRingsCard: View {
let snapshot: HealthSnapshot
var body: some View {
VStack(alignment: .leading, spacing: 14) {
HStack {
Image(systemName: "heart.fill")
.foregroundStyle(.red)
Text(L10n.health.appleHealth)
.font(.headline.weight(.semibold))
Spacer()
Text(L10n.health.today)
.font(.caption)
.foregroundStyle(.secondary)
}
HStack(spacing: 12) {
HealthRingStat(
label: L10n.health.move,
value: "\(Int(snapshot.activeCaloricBurn))",
unit: "kcal",
color: .red
)
HealthRingStat(
label: L10n.health.exercise,
value: "\(Int(snapshot.exerciseMinutes))",
unit: "min",
color: .green
)
HealthRingStat(
label: L10n.health.stand,
value: "\(snapshot.standHours)",
unit: "hrs",
color: .cyan
)
}
if let hr = snapshot.restingHeartRate {
Divider()
HStack {
Label("\(Int(hr)) bpm", systemImage: "waveform.path.ecg")
.font(.subheadline)
.foregroundStyle(.secondary)
Spacer()
Text(L10n.health.restingHR)
.font(.caption)
.foregroundStyle(.tertiary)
}
}
}
.padding(16)
.glassCard()
}
}
struct HealthRingStat: View {
let label: LocalizedStringResource
let value: String
let unit: String
let color: Color
var body: some View {
VStack(spacing: 4) {
Text(value)
.font(.system(size: 24, weight: .bold, design: .rounded))
.monospacedDigit()
.foregroundStyle(color)
Text(unit)
.font(.caption2.weight(.semibold))
.foregroundStyle(color.opacity(0.7))
Text(label)
.font(.caption2)
.foregroundStyle(.secondary)
}
.frame(maxWidth: .infinity)
}
}
// Sub-components (kept for history list)
struct SessionHistoryRow: View {
let session: WorkoutSession
var body: some View {
HStack(spacing: 14) {
// Zone indicator
Circle()
.fill(Theme.zoneColor(session.bodyZone))
.frame(width: 10, height: 10)