Phases 1-6 of fixes.md — closes all 13 issues from codex_issues_2.md re-validation: KMP Architecture: - Fix subscription purchase/restore response contract (VerificationResponse aligned) - Add feature benefits auth token + APILayer init flow - Remove ResidenceFormScreen direct API bypass (use APILayer) - Wire paywall purchase/restore to real SubscriptionApi calls iOS Platform: - Add iOS Keychain token storage via Swift KeychainHelper - Implement Google Sign-In via ASWebAuthenticationSession (GoogleSignInManager) - DocumentViewModelWrapper observes DataManager for auto-updates - Add missing accessibility identifiers (document, task columns, Google Sign-In) XCUITest Rewrite: - Rewrite test infrastructure: zero sleep() calls, accessibility ID lookups - Create AuthCriticalPathTests and NavigationCriticalPathTests - Delete 14 legacy brittle test files (Suite0-10, templates) - Fix CaseraTests module import (@testable import Casera) All platforms build clean. TEST BUILD SUCCEEDED. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
82 lines
3.1 KiB
Swift
82 lines
3.1 KiB
Swift
import SwiftUI
|
|
import ComposeApp
|
|
|
|
/// Dynamic task column view that adapts based on the column configuration
|
|
struct DynamicTaskColumnView: View {
|
|
let column: TaskColumn
|
|
let onEditTask: (TaskResponse) -> Void
|
|
let onCancelTask: (TaskResponse) -> Void
|
|
let onUncancelTask: (Int32) -> Void
|
|
let onMarkInProgress: (Int32) -> Void
|
|
let onCompleteTask: (TaskResponse) -> Void
|
|
let onArchiveTask: (TaskResponse) -> Void
|
|
let onUnarchiveTask: (Int32) -> Void
|
|
|
|
// Get icon from API response, with fallback
|
|
private var columnIcon: String {
|
|
column.icons["ios"] ?? "list.bullet"
|
|
}
|
|
|
|
private var columnColor: Color {
|
|
Color(hex: column.color) ?? Color.appTextPrimary
|
|
}
|
|
|
|
var body: some View {
|
|
VStack(spacing: 0) {
|
|
ScrollView {
|
|
VStack(spacing: 16) {
|
|
// Header
|
|
HStack(spacing: 8) {
|
|
Image(systemName: columnIcon)
|
|
.font(.headline)
|
|
.foregroundColor(columnColor)
|
|
|
|
Text(column.displayName)
|
|
.font(.headline)
|
|
.foregroundColor(columnColor)
|
|
|
|
Spacer()
|
|
|
|
Text("\(column.count)")
|
|
.font(.system(size: 12, weight: .semibold))
|
|
.foregroundColor(Color.appTextOnPrimary)
|
|
.padding(.horizontal, 10)
|
|
.padding(.vertical, 5)
|
|
.background(columnColor)
|
|
.clipShape(Capsule())
|
|
}
|
|
|
|
if column.tasks.isEmpty {
|
|
VStack(spacing: 8) {
|
|
Image(systemName: columnIcon)
|
|
.font(.system(size: 40))
|
|
.foregroundColor(columnColor.opacity(0.3))
|
|
|
|
Text(L10n.Tasks.noTasks)
|
|
.font(.caption)
|
|
.foregroundColor(Color.appTextSecondary)
|
|
}
|
|
.frame(maxWidth: .infinity)
|
|
.padding(.top, 40)
|
|
} else {
|
|
ForEach(column.tasks, id: \.id) { task in
|
|
DynamicTaskCard(
|
|
task: task,
|
|
buttonTypes: column.buttonTypes,
|
|
onEdit: { onEditTask(task) },
|
|
onCancel: { onCancelTask(task) },
|
|
onUncancel: { onUncancelTask(task.id) },
|
|
onMarkInProgress: { onMarkInProgress(task.id) },
|
|
onComplete: { onCompleteTask(task) },
|
|
onArchive: { onArchiveTask(task) },
|
|
onUnarchive: { onUnarchiveTask(task.id) }
|
|
)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
.accessibilityIdentifier("Task.Column.\(column.name)")
|
|
}
|
|
}
|