openuispec 0.1.25 → 0.1.28

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (139) hide show
  1. package/README.md +63 -18
  2. package/cli/index.ts +21 -3
  3. package/cli/init.ts +27 -11
  4. package/docs/implementation-notes.md +119 -0
  5. package/docs/release-notes-v0.1.26.md +64 -0
  6. package/docs/release-notes-v0.1.27.md +28 -0
  7. package/docs/release-notes-v0.1.28.md +25 -0
  8. package/docs/stress-test-maturity-report.md +1 -1
  9. package/drift/index.ts +396 -22
  10. package/examples/taskflow/AGENTS.md +112 -0
  11. package/examples/taskflow/CLAUDE.md +112 -0
  12. package/examples/taskflow/generated/android/TaskFlow/README.md +43 -0
  13. package/examples/taskflow/generated/android/TaskFlow/app/build.gradle.kts +76 -0
  14. package/examples/taskflow/generated/android/TaskFlow/app/proguard-rules.pro +1 -0
  15. package/examples/taskflow/generated/android/TaskFlow/app/src/main/AndroidManifest.xml +21 -0
  16. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/MainActivity.kt +19 -0
  17. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/TaskFlowApp.kt +283 -0
  18. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/model/DomainModels.kt +106 -0
  19. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/model/SampleData.kt +57 -0
  20. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/components/Common.kt +109 -0
  21. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/HomeScreen.kt +112 -0
  22. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/ProjectsScreen.kt +61 -0
  23. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/SettingsScreen.kt +82 -0
  24. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/TaskDetailScreen.kt +111 -0
  25. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/sheets/Sheets.kt +77 -0
  26. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Color.kt +30 -0
  27. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Theme.kt +86 -0
  28. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Type.kt +57 -0
  29. package/examples/taskflow/generated/android/TaskFlow/app/src/main/res/values/strings.xml +155 -0
  30. package/examples/taskflow/generated/android/TaskFlow/app/src/main/res/values/themes.xml +4 -0
  31. package/examples/taskflow/generated/android/TaskFlow/build.gradle.kts +5 -0
  32. package/examples/taskflow/generated/android/TaskFlow/gradle/gradle-daemon-jvm.properties +12 -0
  33. package/examples/taskflow/generated/android/TaskFlow/gradle/wrapper/gradle-wrapper.jar +0 -0
  34. package/examples/taskflow/generated/android/TaskFlow/gradle/wrapper/gradle-wrapper.properties +7 -0
  35. package/examples/taskflow/generated/android/TaskFlow/gradle.properties +4 -0
  36. package/examples/taskflow/generated/android/TaskFlow/gradlew +18 -0
  37. package/examples/taskflow/generated/android/TaskFlow/gradlew.bat +12 -0
  38. package/examples/taskflow/generated/android/TaskFlow/settings.gradle.kts +18 -0
  39. package/examples/taskflow/generated/ios/TaskFlow/README.md +21 -0
  40. package/examples/taskflow/generated/ios/TaskFlow/Resources/en.lproj/Localizable.strings +115 -0
  41. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/App/TaskFlowApp.swift +24 -0
  42. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Components/AppChrome.swift +150 -0
  43. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Flows/TaskEditorSheet.swift +220 -0
  44. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Models/DomainModels.swift +122 -0
  45. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/CalendarView.swift +21 -0
  46. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/HomeView.swift +201 -0
  47. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProfileEditView.swift +48 -0
  48. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProjectDetailView.swift +59 -0
  49. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProjectsView.swift +63 -0
  50. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/SettingsView.swift +85 -0
  51. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/TaskDetailView.swift +219 -0
  52. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Support/AppModel.swift +320 -0
  53. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Support/AppSupport.swift +41 -0
  54. package/examples/taskflow/generated/ios/TaskFlow/project.yml +26 -0
  55. package/examples/taskflow/generated/web/TaskFlow/README.md +19 -0
  56. package/examples/taskflow/generated/web/TaskFlow/index.html +12 -0
  57. package/examples/taskflow/generated/web/TaskFlow/package-lock.json +1908 -0
  58. package/examples/taskflow/generated/web/TaskFlow/package.json +24 -0
  59. package/examples/taskflow/generated/web/TaskFlow/src/App.tsx +58 -0
  60. package/examples/taskflow/generated/web/TaskFlow/src/AppShell.tsx +55 -0
  61. package/examples/taskflow/generated/web/TaskFlow/src/components/Common.tsx +82 -0
  62. package/examples/taskflow/generated/web/TaskFlow/src/components/Modals.tsx +191 -0
  63. package/examples/taskflow/generated/web/TaskFlow/src/components/Nav.tsx +41 -0
  64. package/examples/taskflow/generated/web/TaskFlow/src/generated/messages.ts +131 -0
  65. package/examples/taskflow/generated/web/TaskFlow/src/hooks.ts +25 -0
  66. package/examples/taskflow/generated/web/TaskFlow/src/i18n.ts +39 -0
  67. package/examples/taskflow/generated/web/TaskFlow/src/locales.en.json +111 -0
  68. package/examples/taskflow/generated/web/TaskFlow/src/main.tsx +13 -0
  69. package/examples/taskflow/generated/web/TaskFlow/src/screens/HomeScreen.tsx +111 -0
  70. package/examples/taskflow/generated/web/TaskFlow/src/screens/ProjectsScreen.tsx +82 -0
  71. package/examples/taskflow/generated/web/TaskFlow/src/screens/SettingsScreens.tsx +132 -0
  72. package/examples/taskflow/generated/web/TaskFlow/src/screens/TaskDetail.tsx +105 -0
  73. package/examples/taskflow/generated/web/TaskFlow/src/store.ts +216 -0
  74. package/examples/taskflow/generated/web/TaskFlow/src/styles.css +617 -0
  75. package/examples/taskflow/generated/web/TaskFlow/src/types.ts +64 -0
  76. package/examples/taskflow/generated/web/TaskFlow/src/utils.ts +78 -0
  77. package/examples/taskflow/generated/web/TaskFlow/tsconfig.json +21 -0
  78. package/examples/taskflow/generated/web/TaskFlow/vite.config.ts +6 -0
  79. package/examples/taskflow/openuispec/README.md +49 -0
  80. package/examples/todo-orbit/AGENTS.md +46 -14
  81. package/examples/todo-orbit/CLAUDE.md +46 -14
  82. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/components/CommonComponents.kt +69 -18
  83. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/screens/TasksScreen.kt +5 -0
  84. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/sheets/EditorSheets.kt +5 -2
  85. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/res/values/strings.xml +1 -0
  86. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/res/values-ru/strings.xml +1 -0
  87. package/examples/todo-orbit/generated/ios/Todo Orbit/Resources/en.lproj/Localizable.strings +1 -0
  88. package/examples/todo-orbit/generated/ios/Todo Orbit/Resources/ru.lproj/Localizable.strings +1 -0
  89. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Flows/RecurringRuleSheet.swift +3 -0
  90. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Flows/TaskEditorSheet.swift +1 -0
  91. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Screens/SettingsView.swift +2 -0
  92. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Screens/TasksHomeView.swift +1 -0
  93. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Support/AppModel.swift +1 -1
  94. package/examples/todo-orbit/generated/web/Todo Orbit/src/App.tsx +59 -6
  95. package/examples/todo-orbit/generated/web/Todo Orbit/src/styles.css +40 -0
  96. package/examples/todo-orbit/openuispec/README.md +24 -131
  97. package/examples/todo-orbit/openuispec/flows/create_recurring_rule.yaml +3 -0
  98. package/examples/todo-orbit/openuispec/flows/create_task.yaml +1 -0
  99. package/examples/todo-orbit/openuispec/flows/edit_task.yaml +1 -0
  100. package/examples/todo-orbit/openuispec/locales/en.json +1 -0
  101. package/examples/todo-orbit/openuispec/locales/ru.json +1 -0
  102. package/examples/todo-orbit/openuispec/screens/task_detail.yaml +1 -0
  103. package/examples/todo-orbit/openuispec/tokens/icons.yaml +6 -0
  104. package/package.json +6 -1
  105. package/prepare/index.ts +391 -0
  106. package/schema/semantic-lint.ts +592 -0
  107. package/schema/validate.ts +17 -13
  108. package/status/index.ts +200 -0
  109. /package/examples/taskflow/{contracts → openuispec/contracts}/README.md +0 -0
  110. /package/examples/taskflow/{contracts → openuispec/contracts}/action_trigger.yaml +0 -0
  111. /package/examples/taskflow/{contracts → openuispec/contracts}/collection.yaml +0 -0
  112. /package/examples/taskflow/{contracts → openuispec/contracts}/data_display.yaml +0 -0
  113. /package/examples/taskflow/{contracts → openuispec/contracts}/feedback.yaml +0 -0
  114. /package/examples/taskflow/{contracts → openuispec/contracts}/input_field.yaml +0 -0
  115. /package/examples/taskflow/{contracts → openuispec/contracts}/nav_container.yaml +0 -0
  116. /package/examples/taskflow/{contracts → openuispec/contracts}/surface.yaml +0 -0
  117. /package/examples/taskflow/{contracts → openuispec/contracts}/x_media_player.yaml +0 -0
  118. /package/examples/taskflow/{flows → openuispec/flows}/create_task.yaml +0 -0
  119. /package/examples/taskflow/{flows → openuispec/flows}/edit_task.yaml +0 -0
  120. /package/examples/taskflow/{locales → openuispec/locales}/en.json +0 -0
  121. /package/examples/taskflow/{openuispec.yaml → openuispec/openuispec.yaml} +0 -0
  122. /package/examples/taskflow/{platform → openuispec/platform}/android.yaml +0 -0
  123. /package/examples/taskflow/{platform → openuispec/platform}/ios.yaml +0 -0
  124. /package/examples/taskflow/{platform → openuispec/platform}/web.yaml +0 -0
  125. /package/examples/taskflow/{screens → openuispec/screens}/calendar.yaml +0 -0
  126. /package/examples/taskflow/{screens → openuispec/screens}/home.yaml +0 -0
  127. /package/examples/taskflow/{screens → openuispec/screens}/profile_edit.yaml +0 -0
  128. /package/examples/taskflow/{screens → openuispec/screens}/project_detail.yaml +0 -0
  129. /package/examples/taskflow/{screens → openuispec/screens}/projects.yaml +0 -0
  130. /package/examples/taskflow/{screens → openuispec/screens}/settings.yaml +0 -0
  131. /package/examples/taskflow/{screens → openuispec/screens}/task_detail.yaml +0 -0
  132. /package/examples/taskflow/{tokens → openuispec/tokens}/color.yaml +0 -0
  133. /package/examples/taskflow/{tokens → openuispec/tokens}/elevation.yaml +0 -0
  134. /package/examples/taskflow/{tokens → openuispec/tokens}/icons.yaml +0 -0
  135. /package/examples/taskflow/{tokens → openuispec/tokens}/layout.yaml +0 -0
  136. /package/examples/taskflow/{tokens → openuispec/tokens}/motion.yaml +0 -0
  137. /package/examples/taskflow/{tokens → openuispec/tokens}/spacing.yaml +0 -0
  138. /package/examples/taskflow/{tokens → openuispec/tokens}/themes.yaml +0 -0
  139. /package/examples/taskflow/{tokens → openuispec/tokens}/typography.yaml +0 -0
@@ -0,0 +1,150 @@
1
+ import SwiftUI
2
+
3
+ struct AppChrome: View {
4
+ @Bindable var model: AppModel
5
+ @Environment(\.horizontalSizeClass) private var horizontalSizeClass
6
+
7
+ var body: some View {
8
+ Group {
9
+ if horizontalSizeClass == .compact {
10
+ CompactChrome(model: model)
11
+ } else {
12
+ RegularChrome(model: model)
13
+ }
14
+ }
15
+ .tint(AppPalette.brandPrimary)
16
+ .sheet(item: $model.presentedSheet) { sheet in
17
+ switch sheet {
18
+ case .createTask:
19
+ TaskEditorSheet(model: model, mode: .create)
20
+ case let .editTask(taskID):
21
+ TaskEditorSheet(model: model, mode: .edit(taskID))
22
+ case .newProject:
23
+ NewProjectSheet(model: model)
24
+ case let .assignTask(taskID):
25
+ AssignTaskSheet(model: model, taskID: taskID)
26
+ }
27
+ }
28
+ .overlay(alignment: .top) {
29
+ if let message = model.toastMessage {
30
+ ToastView(message: message)
31
+ .padding(.top, 12)
32
+ .transition(.move(edge: .top).combined(with: .opacity))
33
+ .task {
34
+ try? await _Concurrency.Task.sleep(nanoseconds: 2_500_000_000)
35
+ withAnimation {
36
+ model.toastMessage = nil
37
+ }
38
+ }
39
+ }
40
+ }
41
+ }
42
+ }
43
+
44
+ private struct CompactChrome: View {
45
+ @Bindable var model: AppModel
46
+
47
+ var body: some View {
48
+ TabView(selection: $model.selectedSection) {
49
+ NavigationStack {
50
+ HomeView(model: model)
51
+ }
52
+ .tabItem {
53
+ Label(model.localized("nav.tasks"), systemImage: "checklist")
54
+ }
55
+ .tag(AppSection.home)
56
+
57
+ NavigationStack {
58
+ ProjectsView(model: model)
59
+ }
60
+ .tabItem {
61
+ Label(model.localized("nav.projects"), systemImage: "folder")
62
+ }
63
+ .tag(AppSection.projects)
64
+
65
+ NavigationStack {
66
+ CalendarView(model: model)
67
+ }
68
+ .tabItem {
69
+ Label(model.localized("nav.calendar"), systemImage: "calendar")
70
+ }
71
+ .tag(AppSection.calendar)
72
+
73
+ NavigationStack {
74
+ SettingsView(model: model)
75
+ }
76
+ .tabItem {
77
+ Label(model.localized("nav.settings"), systemImage: "gear")
78
+ }
79
+ .tag(AppSection.settings)
80
+ }
81
+ }
82
+ }
83
+
84
+ private struct RegularChrome: View {
85
+ @Bindable var model: AppModel
86
+
87
+ var body: some View {
88
+ NavigationSplitView {
89
+ List {
90
+ ForEach(AppSection.allCases) { section in
91
+ Button {
92
+ model.selectedSection = section
93
+ } label: {
94
+ Label(title(for: section), systemImage: icon(for: section))
95
+ .frame(maxWidth: .infinity, alignment: .leading)
96
+ }
97
+ .buttonStyle(.plain)
98
+ .listRowBackground(model.selectedSection == section ? AppPalette.brandPrimary.opacity(0.12) : Color.clear)
99
+ }
100
+ }
101
+ .navigationTitle("TaskFlow")
102
+ } detail: {
103
+ NavigationStack {
104
+ switch model.selectedSection {
105
+ case .home:
106
+ HomeView(model: model)
107
+ case .projects:
108
+ ProjectsView(model: model)
109
+ case .calendar:
110
+ CalendarView(model: model)
111
+ case .settings:
112
+ SettingsView(model: model)
113
+ }
114
+ }
115
+ }
116
+ }
117
+
118
+ private func title(for section: AppSection) -> String {
119
+ switch section {
120
+ case .home: model.localized("nav.tasks")
121
+ case .projects: model.localized("nav.projects")
122
+ case .calendar: model.localized("nav.calendar")
123
+ case .settings: model.localized("nav.settings")
124
+ }
125
+ }
126
+
127
+ private func icon(for section: AppSection) -> String {
128
+ switch section {
129
+ case .home: "checklist"
130
+ case .projects: "folder"
131
+ case .calendar: "calendar"
132
+ case .settings: "gear"
133
+ }
134
+ }
135
+ }
136
+
137
+ private struct ToastView: View {
138
+ let message: String
139
+
140
+ var body: some View {
141
+ Text(message)
142
+ .font(.subheadline.weight(.semibold))
143
+ .foregroundStyle(.white)
144
+ .padding(.horizontal, 16)
145
+ .padding(.vertical, 10)
146
+ .background(AppPalette.brandPrimary)
147
+ .clipShape(Capsule())
148
+ .shadow(radius: 12, y: 4)
149
+ }
150
+ }
@@ -0,0 +1,220 @@
1
+ import SwiftUI
2
+
3
+ enum TaskEditorMode {
4
+ case create
5
+ case edit(UUID)
6
+
7
+ var titleKey: String {
8
+ switch self {
9
+ case .create: "create_task.title"
10
+ case .edit: "edit_task.title"
11
+ }
12
+ }
13
+
14
+ var saveKey: String {
15
+ switch self {
16
+ case .create: "create_task.save"
17
+ case .edit: "edit_task.save"
18
+ }
19
+ }
20
+ }
21
+
22
+ struct TaskEditorSheet: View {
23
+ @Bindable var model: AppModel
24
+ let mode: TaskEditorMode
25
+ @Environment(\.dismiss) private var dismiss
26
+ @State private var draft = TaskDraft()
27
+
28
+ var body: some View {
29
+ NavigationStack {
30
+ Form {
31
+ Section {
32
+ TextField(model.localized("create_task.field_title_placeholder"), text: $draft.title)
33
+ TextField(model.localized("create_task.field_description_placeholder"), text: $draft.description, axis: .vertical)
34
+ .lineLimit(4, reservesSpace: true)
35
+ } header: {
36
+ Text(model.localized("create_task.field_title"))
37
+ }
38
+
39
+ Section {
40
+ Picker(model.localized("create_task.field_project"), selection: $draft.projectID) {
41
+ Text(model.localized("create_task.field_project_placeholder")).tag(UUID?.none)
42
+ ForEach(model.projects) { project in
43
+ Text(project.name).tag(UUID?.some(project.id))
44
+ }
45
+ }
46
+ Picker(model.localized("create_task.field_priority"), selection: $draft.priority) {
47
+ Text(model.localized("priority.low")).tag(TaskPriority.low)
48
+ Text(model.localized("priority.medium")).tag(TaskPriority.medium)
49
+ Text(model.localized("priority.high")).tag(TaskPriority.high)
50
+ Text(model.localized("priority.urgent")).tag(TaskPriority.urgent)
51
+ }
52
+ }
53
+
54
+ Section {
55
+ Toggle("Include due date", isOn: $draft.dueDateEnabled)
56
+ if draft.dueDateEnabled {
57
+ DatePicker(model.localized("create_task.field_due_date"), selection: $draft.dueDate, displayedComponents: [.date, .hourAndMinute])
58
+ }
59
+ }
60
+
61
+ Section {
62
+ TextField(model.localized("create_task.field_tags_placeholder"), text: $draft.tagsText)
63
+ Toggle(model.localized("create_task.field_assign_to_me"), isOn: $draft.assignToSelf)
64
+ } footer: {
65
+ Text(model.localized("create_task.field_tags_helper"))
66
+ }
67
+ }
68
+ .navigationTitle(model.localized(mode.titleKey))
69
+ .navigationBarTitleDisplayMode(.inline)
70
+ .toolbar {
71
+ ToolbarItem(placement: .topBarLeading) {
72
+ Button(model.localized("common.cancel")) {
73
+ dismiss()
74
+ }
75
+ }
76
+ ToolbarItem(placement: .topBarTrailing) {
77
+ Button(model.localized(mode.saveKey)) {
78
+ save()
79
+ }
80
+ .disabled(draft.title.trimmingCharacters(in: .whitespacesAndNewlines).count < 3)
81
+ .fontWeight(.semibold)
82
+ }
83
+ }
84
+ .onAppear {
85
+ switch mode {
86
+ case .create:
87
+ draft = TaskDraft(priority: model.preferences.defaultPriority)
88
+ case let .edit(taskID):
89
+ let task = model.tasks.first(where: { $0.id == taskID })
90
+ draft = model.makeDraft(task: task)
91
+ }
92
+ }
93
+ }
94
+ .presentationDetents([.medium, .large])
95
+ }
96
+
97
+ private func save() {
98
+ let normalizedTitle = draft.title.trimmingCharacters(in: .whitespacesAndNewlines)
99
+ guard normalizedTitle.count >= 3 else { return }
100
+ draft.title = normalizedTitle
101
+ switch mode {
102
+ case .create:
103
+ model.saveTask(draft, editing: nil)
104
+ case let .edit(taskID):
105
+ model.saveTask(draft, editing: taskID)
106
+ }
107
+ dismiss()
108
+ }
109
+ }
110
+
111
+ struct NewProjectSheet: View {
112
+ @Bindable var model: AppModel
113
+ @Environment(\.dismiss) private var dismiss
114
+ @State private var draft = ProjectDraft()
115
+
116
+ private let colors = ["#5B4FE8", "#E8634F", "#2D9D5E", "#D4920E", "#3B82D4"]
117
+
118
+ var body: some View {
119
+ NavigationStack {
120
+ Form {
121
+ Section {
122
+ TextField(model.localized("projects.field_name_placeholder"), text: $draft.name)
123
+ } header: {
124
+ Text(model.localized("projects.field_name"))
125
+ }
126
+
127
+ Section(header: Text(model.localized("projects.field_color"))) {
128
+ LazyVGrid(columns: [GridItem(.adaptive(minimum: 44))], spacing: 12) {
129
+ ForEach(colors, id: \.self) { hex in
130
+ Circle()
131
+ .fill(Color(hex: hex))
132
+ .frame(width: 36, height: 36)
133
+ .overlay {
134
+ if draft.colorHex == hex {
135
+ Circle().stroke(.white, lineWidth: 3)
136
+ }
137
+ }
138
+ .onTapGesture {
139
+ draft.colorHex = hex
140
+ }
141
+ }
142
+ }
143
+ .padding(.vertical, 8)
144
+ }
145
+ }
146
+ .navigationTitle(model.localized("projects.dialog_title"))
147
+ .navigationBarTitleDisplayMode(.inline)
148
+ .toolbar {
149
+ ToolbarItem(placement: .topBarLeading) {
150
+ Button(model.localized("common.cancel")) {
151
+ dismiss()
152
+ }
153
+ }
154
+ ToolbarItem(placement: .topBarTrailing) {
155
+ Button(model.localized("common.create")) {
156
+ model.createProject(draft)
157
+ dismiss()
158
+ }
159
+ .disabled(draft.name.trimmingCharacters(in: .whitespacesAndNewlines).isEmpty)
160
+ }
161
+ }
162
+ }
163
+ }
164
+ }
165
+
166
+ struct AssignTaskSheet: View {
167
+ @Bindable var model: AppModel
168
+ let taskID: UUID
169
+ @Environment(\.dismiss) private var dismiss
170
+ @State private var query = ""
171
+
172
+ private var filteredUsers: [UserProfile] {
173
+ if query.trimmingCharacters(in: .whitespacesAndNewlines).isEmpty {
174
+ return model.team
175
+ }
176
+ return model.team.filter {
177
+ $0.name.localizedCaseInsensitiveContains(query) ||
178
+ $0.email.localizedCaseInsensitiveContains(query)
179
+ }
180
+ }
181
+
182
+ var body: some View {
183
+ NavigationStack {
184
+ List {
185
+ Section {
186
+ TextField(model.localized("task_detail.search_people_placeholder"), text: $query)
187
+ .textInputAutocapitalization(.never)
188
+ } header: {
189
+ Text(model.localized("task_detail.search_people"))
190
+ }
191
+
192
+ Section {
193
+ ForEach(filteredUsers) { user in
194
+ Button {
195
+ model.assignTask(taskID: taskID, userID: user.id)
196
+ dismiss()
197
+ } label: {
198
+ VStack(alignment: .leading, spacing: 4) {
199
+ Text(user.name)
200
+ Text(user.email)
201
+ .font(.caption)
202
+ .foregroundStyle(AppPalette.textSecondary)
203
+ }
204
+ }
205
+ }
206
+ }
207
+ }
208
+ .navigationTitle(model.localized("task_detail.assign_to"))
209
+ .navigationBarTitleDisplayMode(.inline)
210
+ .toolbar {
211
+ ToolbarItem(placement: .topBarTrailing) {
212
+ Button(model.localized("common.cancel")) {
213
+ dismiss()
214
+ }
215
+ }
216
+ }
217
+ }
218
+ .presentationDetents([.medium, .large])
219
+ }
220
+ }
@@ -0,0 +1,122 @@
1
+ import Foundation
2
+
3
+ enum AppSection: String, CaseIterable, Identifiable {
4
+ case home
5
+ case projects
6
+ case calendar
7
+ case settings
8
+
9
+ var id: String { rawValue }
10
+ }
11
+
12
+ enum TaskStatus: String, CaseIterable, Codable {
13
+ case todo
14
+ case inProgress = "in_progress"
15
+ case done
16
+ }
17
+
18
+ enum TaskPriority: String, CaseIterable, Codable {
19
+ case low
20
+ case medium
21
+ case high
22
+ case urgent
23
+ }
24
+
25
+ enum HomeFilter: String, CaseIterable, Codable {
26
+ case all
27
+ case today
28
+ case upcoming
29
+ case done
30
+ }
31
+
32
+ enum SortOrder: String, CaseIterable, Codable {
33
+ case dueDate = "due_date"
34
+ case priority = "priority"
35
+ case createdAt = "created_at"
36
+ }
37
+
38
+ enum ThemePreference: String, CaseIterable, Codable {
39
+ case system
40
+ case light
41
+ case dark
42
+ case warm
43
+ }
44
+
45
+ struct Project: Identifiable, Hashable, Codable {
46
+ let id: UUID
47
+ var name: String
48
+ var colorHex: String
49
+ var icon: String
50
+ }
51
+
52
+ struct UserProfile: Identifiable, Hashable, Codable {
53
+ let id: UUID
54
+ var name: String
55
+ var firstName: String
56
+ var email: String
57
+ var avatarSymbol: String?
58
+ }
59
+
60
+ struct TaskAttachment: Hashable, Codable {
61
+ var mediaType: String
62
+ var title: String
63
+ var url: URL?
64
+ }
65
+
66
+ struct Task: Identifiable, Hashable, Codable {
67
+ let id: UUID
68
+ var title: String
69
+ var description: String?
70
+ var status: TaskStatus
71
+ var priority: TaskPriority
72
+ var dueDate: Date?
73
+ var projectID: UUID?
74
+ var assigneeID: UUID?
75
+ var tags: [String]
76
+ var createdAt: Date
77
+ var updatedAt: Date
78
+ var attachment: TaskAttachment?
79
+ }
80
+
81
+ struct Preferences: Codable {
82
+ var theme: ThemePreference
83
+ var defaultPriority: TaskPriority
84
+ var notificationsEnabled: Bool
85
+ var remindersEnabled: Bool
86
+ }
87
+
88
+ struct TaskDraft {
89
+ var title = ""
90
+ var description = ""
91
+ var projectID: UUID?
92
+ var priority: TaskPriority = .medium
93
+ var dueDateEnabled = false
94
+ var dueDate = Date()
95
+ var tagsText = ""
96
+ var assignToSelf = true
97
+ }
98
+
99
+ struct ProjectDraft {
100
+ var name = ""
101
+ var colorHex = "#5B4FE8"
102
+ }
103
+
104
+ enum PresentedSheet: Identifiable {
105
+ case createTask
106
+ case editTask(UUID)
107
+ case newProject
108
+ case assignTask(UUID)
109
+
110
+ var id: String {
111
+ switch self {
112
+ case .createTask:
113
+ return "createTask"
114
+ case let .editTask(id):
115
+ return "editTask-\(id.uuidString)"
116
+ case .newProject:
117
+ return "newProject"
118
+ case let .assignTask(id):
119
+ return "assignTask-\(id.uuidString)"
120
+ }
121
+ }
122
+ }
@@ -0,0 +1,21 @@
1
+ import SwiftUI
2
+
3
+ struct CalendarView: View {
4
+ @Bindable var model: AppModel
5
+
6
+ var body: some View {
7
+ VStack(spacing: 18) {
8
+ Image(systemName: "calendar.badge.clock")
9
+ .font(.system(size: 52))
10
+ .foregroundStyle(AppPalette.brandPrimary)
11
+ Text(model.localized("calendar.title"))
12
+ .font(.largeTitle.weight(.bold))
13
+ Text(model.localized("calendar.coming_soon"))
14
+ .foregroundStyle(AppPalette.textSecondary)
15
+ }
16
+ .frame(maxWidth: .infinity, maxHeight: .infinity)
17
+ .padding()
18
+ .background(Color(.systemGroupedBackground))
19
+ .navigationTitle(model.localized("calendar.title"))
20
+ }
21
+ }