openuispec 0.2.10 → 0.2.12

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 (237) hide show
  1. package/README.md +3 -1
  2. package/check/index.ts +17 -0
  3. package/cli/index.ts +21 -3
  4. package/cli/init.ts +224 -10
  5. package/docs/cli.md +13 -8
  6. package/docs/file-formats.md +36 -0
  7. package/docs/implementation-notes.md +7 -0
  8. package/drift/index.ts +281 -40
  9. package/mcp-server/index.ts +179 -119
  10. package/mcp-server/screenshot.ts +19 -4
  11. package/package.json +5 -2
  12. package/prepare/index.ts +155 -18
  13. package/schema/openuispec.schema.json +59 -0
  14. package/schema/semantic-lint.ts +25 -1
  15. package/scripts/take-all-screenshots.ts +507 -0
  16. package/spec/openuispec-v0.1.md +13 -0
  17. package/status/index.ts +72 -2
  18. package/examples/social-app/.mcp.json +0 -10
  19. package/examples/social-app/AGENTS.md +0 -124
  20. package/examples/social-app/CLAUDE.md +0 -124
  21. package/examples/social-app/backend/.gitkeep +0 -1
  22. package/examples/social-app/generated/android/social-app/app/.paparazzi-hashes.json +0 -3
  23. package/examples/social-app/generated/android/social-app/app/build.gradle.kts +0 -94
  24. package/examples/social-app/generated/android/social-app/app/src/main/AndroidManifest.xml +0 -26
  25. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/AppContainer.kt +0 -20
  26. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/MainActivity.kt +0 -35
  27. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/SocialAppApplication.kt +0 -13
  28. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/data/MockData.kt +0 -98
  29. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/data/preferences/AppPreferences.kt +0 -19
  30. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/data/preferences/DataStorePreferencesRepository.kt +0 -68
  31. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/data/preferences/PreferencesRepository.kt +0 -15
  32. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/model/Models.kt +0 -34
  33. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/MainShell.kt +0 -390
  34. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/components/Components.kt +0 -234
  35. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/components/ContractPrimitives.kt +0 -641
  36. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/navigation/RootComponent.kt +0 -113
  37. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/ChatDetailScreen.kt +0 -212
  38. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/CreatePostScreen.kt +0 -113
  39. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/DiscoverScreen.kt +0 -137
  40. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/EditProfileScreen.kt +0 -180
  41. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/HomeFeedScreen.kt +0 -169
  42. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/MessagesInboxScreen.kt +0 -85
  43. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/NotificationsScreen.kt +0 -74
  44. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/PostDetailScreen.kt +0 -293
  45. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/ProfileSelfScreen.kt +0 -116
  46. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/SearchResultsScreen.kt +0 -161
  47. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/SettingsScreen.kt +0 -164
  48. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/SettingsStore.kt +0 -95
  49. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/UserProfileScreen.kt +0 -123
  50. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Color.kt +0 -33
  51. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Shape.kt +0 -41
  52. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Spacing.kt +0 -20
  53. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Theme.kt +0 -82
  54. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Type.kt +0 -60
  55. package/examples/social-app/generated/android/social-app/app/src/main/res/drawable/ic_launcher_foreground.xml +0 -9
  56. package/examples/social-app/generated/android/social-app/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml +0 -5
  57. package/examples/social-app/generated/android/social-app/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml +0 -5
  58. package/examples/social-app/generated/android/social-app/app/src/main/res/values/strings.xml +0 -91
  59. package/examples/social-app/generated/android/social-app/app/src/main/res/values/themes.xml +0 -10
  60. package/examples/social-app/generated/android/social-app/app/src/main/res/values-ru/strings.xml +0 -79
  61. package/examples/social-app/generated/android/social-app/app/src/main/res/values-uz/strings.xml +0 -79
  62. package/examples/social-app/generated/android/social-app/app/src/main/xml/AndroidManifest.xml +0 -23
  63. package/examples/social-app/generated/android/social-app/app/src/test/kotlin/com/social/app/screenshots/HomeFeedScreenshotTest.kt +0 -34
  64. package/examples/social-app/generated/android/social-app/build.gradle.kts +0 -7
  65. package/examples/social-app/generated/android/social-app/gradle/libs.versions.toml +0 -50
  66. package/examples/social-app/generated/android/social-app/gradle/wrapper/gradle-wrapper.jar +0 -0
  67. package/examples/social-app/generated/android/social-app/gradle/wrapper/gradle-wrapper.properties +0 -8
  68. package/examples/social-app/generated/android/social-app/gradle.properties +0 -11
  69. package/examples/social-app/generated/android/social-app/gradlew +0 -248
  70. package/examples/social-app/generated/android/social-app/settings.gradle.kts +0 -27
  71. package/examples/social-app/generated/web/social-app/index.html +0 -12
  72. package/examples/social-app/generated/web/social-app/package-lock.json +0 -2517
  73. package/examples/social-app/generated/web/social-app/package.json +0 -27
  74. package/examples/social-app/generated/web/social-app/src/app/App.tsx +0 -58
  75. package/examples/social-app/generated/web/social-app/src/components/Shell.tsx +0 -259
  76. package/examples/social-app/generated/web/social-app/src/components/cards.tsx +0 -317
  77. package/examples/social-app/generated/web/social-app/src/components/ui.tsx +0 -340
  78. package/examples/social-app/generated/web/social-app/src/flows/CreatePostFlow.tsx +0 -86
  79. package/examples/social-app/generated/web/social-app/src/i18n.tsx +0 -59
  80. package/examples/social-app/generated/web/social-app/src/lib/icons.tsx +0 -85
  81. package/examples/social-app/generated/web/social-app/src/lib/tokens.ts +0 -70
  82. package/examples/social-app/generated/web/social-app/src/lib/utils.ts +0 -97
  83. package/examples/social-app/generated/web/social-app/src/locales/en.json +0 -67
  84. package/examples/social-app/generated/web/social-app/src/locales/ru.json +0 -67
  85. package/examples/social-app/generated/web/social-app/src/locales/uz.json +0 -67
  86. package/examples/social-app/generated/web/social-app/src/main.tsx +0 -16
  87. package/examples/social-app/generated/web/social-app/src/screens/ChatDetailScreen.tsx +0 -90
  88. package/examples/social-app/generated/web/social-app/src/screens/DiscoverScreen.tsx +0 -86
  89. package/examples/social-app/generated/web/social-app/src/screens/EditProfileScreen.tsx +0 -57
  90. package/examples/social-app/generated/web/social-app/src/screens/HomeFeedScreen.tsx +0 -103
  91. package/examples/social-app/generated/web/social-app/src/screens/MessagesInboxScreen.tsx +0 -52
  92. package/examples/social-app/generated/web/social-app/src/screens/NotificationsScreen.tsx +0 -41
  93. package/examples/social-app/generated/web/social-app/src/screens/PostDetailScreen.tsx +0 -115
  94. package/examples/social-app/generated/web/social-app/src/screens/ProfileSelfScreen.tsx +0 -57
  95. package/examples/social-app/generated/web/social-app/src/screens/ProfileUserScreen.tsx +0 -76
  96. package/examples/social-app/generated/web/social-app/src/screens/SearchResultsScreen.tsx +0 -96
  97. package/examples/social-app/generated/web/social-app/src/screens/SettingsScreen.tsx +0 -79
  98. package/examples/social-app/generated/web/social-app/src/state/store.ts +0 -592
  99. package/examples/social-app/generated/web/social-app/src/styles.css +0 -125
  100. package/examples/social-app/generated/web/social-app/src/vite-env.d.ts +0 -1
  101. package/examples/social-app/generated/web/social-app/tsconfig.json +0 -22
  102. package/examples/social-app/generated/web/social-app/tsconfig.node.json +0 -13
  103. package/examples/social-app/generated/web/social-app/tsconfig.node.tsbuildinfo +0 -1
  104. package/examples/social-app/generated/web/social-app/tsconfig.tsbuildinfo +0 -1
  105. package/examples/social-app/generated/web/social-app/vite.config.d.ts +0 -2
  106. package/examples/social-app/generated/web/social-app/vite.config.js +0 -6
  107. package/examples/social-app/generated/web/social-app/vite.config.ts +0 -7
  108. package/examples/social-app/package.json +0 -13
  109. package/examples/social-app/take-web-screenshots.ts +0 -97
  110. package/examples/taskflow/.codex/config.toml +0 -4
  111. package/examples/taskflow/.mcp.json +0 -10
  112. package/examples/taskflow/AGENTS.md +0 -124
  113. package/examples/taskflow/CLAUDE.md +0 -124
  114. package/examples/taskflow/backend/.gitkeep +0 -1
  115. package/examples/taskflow/generated/android/TaskFlow/README.md +0 -43
  116. package/examples/taskflow/generated/android/TaskFlow/app/build.gradle.kts +0 -76
  117. package/examples/taskflow/generated/android/TaskFlow/app/proguard-rules.pro +0 -1
  118. package/examples/taskflow/generated/android/TaskFlow/app/src/main/AndroidManifest.xml +0 -21
  119. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/MainActivity.kt +0 -19
  120. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/TaskFlowApp.kt +0 -283
  121. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/model/DomainModels.kt +0 -106
  122. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/model/SampleData.kt +0 -57
  123. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/components/Common.kt +0 -109
  124. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/HomeScreen.kt +0 -112
  125. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/ProjectsScreen.kt +0 -61
  126. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/SettingsScreen.kt +0 -82
  127. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/TaskDetailScreen.kt +0 -111
  128. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/sheets/Sheets.kt +0 -77
  129. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Color.kt +0 -30
  130. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Theme.kt +0 -86
  131. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Type.kt +0 -57
  132. package/examples/taskflow/generated/android/TaskFlow/app/src/main/res/values/strings.xml +0 -155
  133. package/examples/taskflow/generated/android/TaskFlow/app/src/main/res/values/themes.xml +0 -4
  134. package/examples/taskflow/generated/android/TaskFlow/build.gradle.kts +0 -5
  135. package/examples/taskflow/generated/android/TaskFlow/gradle/gradle-daemon-jvm.properties +0 -12
  136. package/examples/taskflow/generated/android/TaskFlow/gradle/wrapper/gradle-wrapper.jar +0 -0
  137. package/examples/taskflow/generated/android/TaskFlow/gradle/wrapper/gradle-wrapper.properties +0 -7
  138. package/examples/taskflow/generated/android/TaskFlow/gradle.properties +0 -4
  139. package/examples/taskflow/generated/android/TaskFlow/gradlew +0 -18
  140. package/examples/taskflow/generated/android/TaskFlow/gradlew.bat +0 -12
  141. package/examples/taskflow/generated/android/TaskFlow/settings.gradle.kts +0 -18
  142. package/examples/taskflow/generated/ios/TaskFlow/README.md +0 -21
  143. package/examples/taskflow/generated/ios/TaskFlow/Resources/en.lproj/Localizable.strings +0 -115
  144. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/App/TaskFlowApp.swift +0 -24
  145. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Components/AppChrome.swift +0 -150
  146. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Flows/TaskEditorSheet.swift +0 -220
  147. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Models/DomainModels.swift +0 -122
  148. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/CalendarView.swift +0 -21
  149. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/HomeView.swift +0 -201
  150. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProfileEditView.swift +0 -48
  151. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProjectDetailView.swift +0 -59
  152. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProjectsView.swift +0 -63
  153. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/SettingsView.swift +0 -85
  154. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/TaskDetailView.swift +0 -219
  155. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Support/AppModel.swift +0 -320
  156. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Support/AppSupport.swift +0 -41
  157. package/examples/taskflow/generated/ios/TaskFlow/project.yml +0 -31
  158. package/examples/taskflow/generated/web/TaskFlow/README.md +0 -19
  159. package/examples/taskflow/generated/web/TaskFlow/index.html +0 -12
  160. package/examples/taskflow/generated/web/TaskFlow/package-lock.json +0 -1908
  161. package/examples/taskflow/generated/web/TaskFlow/package.json +0 -24
  162. package/examples/taskflow/generated/web/TaskFlow/src/App.tsx +0 -58
  163. package/examples/taskflow/generated/web/TaskFlow/src/AppShell.tsx +0 -55
  164. package/examples/taskflow/generated/web/TaskFlow/src/components/Common.tsx +0 -82
  165. package/examples/taskflow/generated/web/TaskFlow/src/components/Modals.tsx +0 -191
  166. package/examples/taskflow/generated/web/TaskFlow/src/components/Nav.tsx +0 -41
  167. package/examples/taskflow/generated/web/TaskFlow/src/generated/messages.ts +0 -131
  168. package/examples/taskflow/generated/web/TaskFlow/src/hooks.ts +0 -25
  169. package/examples/taskflow/generated/web/TaskFlow/src/i18n.ts +0 -39
  170. package/examples/taskflow/generated/web/TaskFlow/src/locales.en.json +0 -111
  171. package/examples/taskflow/generated/web/TaskFlow/src/main.tsx +0 -13
  172. package/examples/taskflow/generated/web/TaskFlow/src/screens/HomeScreen.tsx +0 -111
  173. package/examples/taskflow/generated/web/TaskFlow/src/screens/ProjectsScreen.tsx +0 -82
  174. package/examples/taskflow/generated/web/TaskFlow/src/screens/SettingsScreens.tsx +0 -132
  175. package/examples/taskflow/generated/web/TaskFlow/src/screens/TaskDetail.tsx +0 -105
  176. package/examples/taskflow/generated/web/TaskFlow/src/store.ts +0 -216
  177. package/examples/taskflow/generated/web/TaskFlow/src/styles.css +0 -617
  178. package/examples/taskflow/generated/web/TaskFlow/src/types.ts +0 -64
  179. package/examples/taskflow/generated/web/TaskFlow/src/utils.ts +0 -78
  180. package/examples/taskflow/generated/web/TaskFlow/tsconfig.json +0 -21
  181. package/examples/taskflow/generated/web/TaskFlow/vite.config.ts +0 -6
  182. package/examples/todo-orbit/.codex/config.toml +0 -4
  183. package/examples/todo-orbit/.mcp.json +0 -10
  184. package/examples/todo-orbit/AGENTS.md +0 -124
  185. package/examples/todo-orbit/CLAUDE.md +0 -124
  186. package/examples/todo-orbit/backend/.gitkeep +0 -1
  187. package/examples/todo-orbit/generated/android/Todo Orbit/README.md +0 -14
  188. package/examples/todo-orbit/generated/android/Todo Orbit/app/build.gradle.kts +0 -58
  189. package/examples/todo-orbit/generated/android/Todo Orbit/app/proguard-rules.pro +0 -1
  190. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/AndroidManifest.xml +0 -20
  191. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/MainActivity.kt +0 -14
  192. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/TodoOrbitApp.kt +0 -345
  193. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/support/AppLogic.kt +0 -231
  194. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/support/Models.kt +0 -169
  195. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/support/Strings.kt +0 -8
  196. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/components/CommonComponents.kt +0 -236
  197. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/screens/AnalyticsScreen.kt +0 -193
  198. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/screens/SettingsScreen.kt +0 -102
  199. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/screens/TasksScreen.kt +0 -347
  200. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/sheets/EditorSheets.kt +0 -347
  201. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/theme/TodoOrbitTheme.kt +0 -59
  202. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/res/values/strings.xml +0 -149
  203. package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/res/values-ru/strings.xml +0 -155
  204. package/examples/todo-orbit/generated/android/Todo Orbit/build.gradle.kts +0 -4
  205. package/examples/todo-orbit/generated/android/Todo Orbit/gradle/wrapper/gradle-wrapper.jar +0 -0
  206. package/examples/todo-orbit/generated/android/Todo Orbit/gradle/wrapper/gradle-wrapper.properties +0 -7
  207. package/examples/todo-orbit/generated/android/Todo Orbit/gradle.properties +0 -4
  208. package/examples/todo-orbit/generated/android/Todo Orbit/gradlew +0 -248
  209. package/examples/todo-orbit/generated/android/Todo Orbit/gradlew.bat +0 -93
  210. package/examples/todo-orbit/generated/android/Todo Orbit/settings.gradle.kts +0 -18
  211. package/examples/todo-orbit/generated/ios/Todo Orbit/.screenshot-uitest/Sources/ScreenshotUITest.swift +0 -36
  212. package/examples/todo-orbit/generated/ios/Todo Orbit/README.md +0 -29
  213. package/examples/todo-orbit/generated/ios/Todo Orbit/Resources/en.lproj/Localizable.strings +0 -119
  214. package/examples/todo-orbit/generated/ios/Todo Orbit/Resources/ru.lproj/Localizable.strings +0 -119
  215. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/App/TodoOrbitApp.swift +0 -50
  216. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Components/OrbitChrome.swift +0 -204
  217. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Components/SchedulePreviewView.swift +0 -126
  218. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Components/TrendChartView.swift +0 -70
  219. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Flows/RecurringRuleSheet.swift +0 -126
  220. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Flows/TaskEditorSheet.swift +0 -61
  221. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Models/DomainModels.swift +0 -238
  222. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Screens/AnalyticsView.swift +0 -94
  223. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Screens/SettingsView.swift +0 -76
  224. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Screens/TasksHomeView.swift +0 -364
  225. package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Support/AppModel.swift +0 -324
  226. package/examples/todo-orbit/generated/ios/Todo Orbit/TodoOrbit.xcodeproj/project.pbxproj +0 -439
  227. package/examples/todo-orbit/generated/ios/Todo Orbit/TodoOrbit.xcodeproj/project.xcworkspace/contents.xcworkspacedata +0 -7
  228. package/examples/todo-orbit/generated/ios/Todo Orbit/TodoOrbit.xcodeproj/xcshareddata/xcschemes/TodoOrbit.xcscheme +0 -89
  229. package/examples/todo-orbit/generated/ios/Todo Orbit/project.yml +0 -32
  230. package/examples/todo-orbit/generated/web/Todo Orbit/index.html +0 -16
  231. package/examples/todo-orbit/generated/web/Todo Orbit/package-lock.json +0 -1087
  232. package/examples/todo-orbit/generated/web/Todo Orbit/package.json +0 -24
  233. package/examples/todo-orbit/generated/web/Todo Orbit/src/App.tsx +0 -2167
  234. package/examples/todo-orbit/generated/web/Todo Orbit/src/main.tsx +0 -13
  235. package/examples/todo-orbit/generated/web/Todo Orbit/src/styles.css +0 -926
  236. package/examples/todo-orbit/generated/web/Todo Orbit/tsconfig.json +0 -19
  237. package/examples/todo-orbit/generated/web/Todo Orbit/vite.config.ts +0 -6
@@ -1,63 +0,0 @@
1
- import SwiftUI
2
-
3
- struct ProjectsView: View {
4
- @Bindable var model: AppModel
5
-
6
- private let gridColumns = [
7
- GridItem(.adaptive(minimum: 220), spacing: 16)
8
- ]
9
-
10
- var body: some View {
11
- ScrollView {
12
- LazyVGrid(columns: gridColumns, spacing: 16) {
13
- ForEach(model.projects) { project in
14
- NavigationLink {
15
- ProjectDetailView(model: model, projectID: project.id)
16
- } label: {
17
- ProjectCard(project: project, taskCount: model.tasks(for: project).count)
18
- }
19
- .buttonStyle(.plain)
20
- }
21
- }
22
- .padding(24)
23
- }
24
- .background(Color(.systemGroupedBackground))
25
- .navigationTitle(model.localized("projects.title"))
26
- .toolbar {
27
- ToolbarItem(placement: .topBarTrailing) {
28
- Button {
29
- model.presentedSheet = .newProject
30
- } label: {
31
- Label(model.localized("projects.new_project"), systemImage: "plus.circle")
32
- }
33
- }
34
- }
35
- }
36
- }
37
-
38
- private struct ProjectCard: View {
39
- let project: Project
40
- let taskCount: Int
41
-
42
- var body: some View {
43
- VStack(alignment: .leading, spacing: 14) {
44
- Image(systemName: project.icon)
45
- .font(.title2.weight(.semibold))
46
- .foregroundStyle(Color(hex: project.colorHex))
47
- .frame(width: 48, height: 48)
48
- .background(Color(hex: project.colorHex).opacity(0.12))
49
- .clipShape(RoundedRectangle(cornerRadius: 16))
50
- Text(project.name)
51
- .font(.headline)
52
- .multilineTextAlignment(.leading)
53
- Text(taskCount == 1 ? "1 task" : "\(taskCount) tasks")
54
- .font(.subheadline)
55
- .foregroundStyle(AppPalette.textSecondary)
56
- }
57
- .frame(maxWidth: .infinity, alignment: .leading)
58
- .padding(20)
59
- .background(.background)
60
- .clipShape(RoundedRectangle(cornerRadius: 24))
61
- .shadow(color: .black.opacity(0.05), radius: 12, y: 4)
62
- }
63
- }
@@ -1,85 +0,0 @@
1
- import SwiftUI
2
-
3
- struct SettingsView: View {
4
- @Bindable var model: AppModel
5
- @State private var showDeleteAlert = false
6
-
7
- var body: some View {
8
- List {
9
- Section {
10
- NavigationLink {
11
- ProfileEditView(model: model)
12
- } label: {
13
- HStack(spacing: 12) {
14
- Image(systemName: model.currentUser.avatarSymbol ?? "person.crop.circle.fill")
15
- .font(.largeTitle)
16
- .foregroundStyle(AppPalette.brandPrimary)
17
- VStack(alignment: .leading) {
18
- Text(model.currentUser.name)
19
- .font(.headline)
20
- Text(model.currentUser.email)
21
- .font(.subheadline)
22
- .foregroundStyle(AppPalette.textSecondary)
23
- }
24
- }
25
- .padding(.vertical, 4)
26
- }
27
- }
28
-
29
- Section {
30
- Picker(model.localized("settings.theme"), selection: $model.preferences.theme) {
31
- Text(model.localized("settings.theme_system")).tag(ThemePreference.system)
32
- Text(model.localized("settings.theme_light")).tag(ThemePreference.light)
33
- Text(model.localized("settings.theme_dark")).tag(ThemePreference.dark)
34
- Text(model.localized("settings.theme_warm")).tag(ThemePreference.warm)
35
- }
36
-
37
- Picker(model.localized("settings.default_priority"), selection: $model.preferences.defaultPriority) {
38
- Text(model.localized("priority.low")).tag(TaskPriority.low)
39
- Text(model.localized("priority.medium")).tag(TaskPriority.medium)
40
- Text(model.localized("priority.high")).tag(TaskPriority.high)
41
- Text(model.localized("priority.urgent")).tag(TaskPriority.urgent)
42
- }
43
-
44
- Toggle(model.localized("settings.notifications"), isOn: $model.preferences.notificationsEnabled)
45
- Toggle(model.localized("settings.reminders"), isOn: $model.preferences.remindersEnabled)
46
- } header: {
47
- Text(model.localized("settings.preferences"))
48
- } footer: {
49
- Text(model.localized("settings.reminders_helper"))
50
- }
51
-
52
- Section {
53
- Button(model.localized("settings.export")) {
54
- model.exportData()
55
- }
56
-
57
- Button(model.localized("settings.delete_account"), role: .destructive) {
58
- showDeleteAlert = true
59
- }
60
- } header: {
61
- Text(model.localized("settings.data"))
62
- }
63
-
64
- Section {
65
- VStack(spacing: 4) {
66
- Text(model.localized("settings.app_version"))
67
- Text(model.localized("settings.app_credit"))
68
- }
69
- .font(.caption)
70
- .foregroundStyle(AppPalette.textSecondary)
71
- .frame(maxWidth: .infinity)
72
- }
73
- .listRowBackground(Color.clear)
74
- }
75
- .navigationTitle(model.localized("nav.settings"))
76
- .alert(model.localized("settings.delete_title"), isPresented: $showDeleteAlert) {
77
- Button(model.localized("common.cancel"), role: .cancel) {}
78
- Button(model.localized("settings.delete_confirm"), role: .destructive) {
79
- model.toastMessage = model.localized("settings.delete_title")
80
- }
81
- } message: {
82
- Text(model.localized("settings.delete_message"))
83
- }
84
- }
85
- }
@@ -1,219 +0,0 @@
1
- import AVKit
2
- import SwiftUI
3
-
4
- struct TaskDetailView: View {
5
- @Bindable var model: AppModel
6
- let taskID: UUID
7
- @State private var showDeletePrompt = false
8
-
9
- private var task: Task? {
10
- model.tasks.first(where: { $0.id == taskID })
11
- }
12
-
13
- var body: some View {
14
- Group {
15
- if let task {
16
- ScrollView {
17
- TaskDetailContent(model: model, task: task) {
18
- showDeletePrompt = true
19
- }
20
- .padding(24)
21
- }
22
- .navigationTitle(task.title)
23
- .navigationBarTitleDisplayMode(.inline)
24
- .alert(model.localized("task_detail.delete_title"), isPresented: $showDeletePrompt) {
25
- Button(model.localized("common.cancel"), role: .cancel) {}
26
- Button(model.localized("common.delete"), role: .destructive) {
27
- model.delete(task: task)
28
- }
29
- } message: {
30
- Text(String(format: model.localized("task_detail.delete_message"), task.title))
31
- }
32
- } else {
33
- ContentUnavailableView("Task missing", systemImage: "exclamationmark.triangle")
34
- }
35
- }
36
- }
37
- }
38
-
39
- struct TaskDetailPanel: View {
40
- @Bindable var model: AppModel
41
- let taskID: UUID
42
- @State private var showDeletePrompt = false
43
-
44
- private var task: Task? {
45
- model.tasks.first(where: { $0.id == taskID })
46
- }
47
-
48
- var body: some View {
49
- Group {
50
- if let task {
51
- TaskDetailContent(model: model, task: task) {
52
- showDeletePrompt = true
53
- }
54
- .padding(24)
55
- .background(.background)
56
- .clipShape(RoundedRectangle(cornerRadius: 28))
57
- .alert(model.localized("task_detail.delete_title"), isPresented: $showDeletePrompt) {
58
- Button(model.localized("common.cancel"), role: .cancel) {}
59
- Button(model.localized("common.delete"), role: .destructive) {
60
- model.delete(task: task)
61
- }
62
- } message: {
63
- Text(String(format: model.localized("task_detail.delete_message"), task.title))
64
- }
65
- }
66
- }
67
- }
68
- }
69
-
70
- private struct TaskDetailContent: View {
71
- @Bindable var model: AppModel
72
- let task: Task
73
- let requestDelete: () -> Void
74
-
75
- var body: some View {
76
- VStack(alignment: .leading, spacing: 24) {
77
- VStack(alignment: .leading, spacing: 12) {
78
- Text(task.title)
79
- .font(.system(size: 28, weight: .bold, design: .rounded))
80
- HStack(spacing: 12) {
81
- badge(model.statusLabel(task.status), color: statusColor(task.status))
82
- badge(model.priorityLabel(task.priority), color: priorityColor(task.priority))
83
- }
84
- }
85
-
86
- HStack(spacing: 12) {
87
- statCard(model.localized("task_detail.status"), value: model.statusLabel(task.status), color: statusColor(task.status))
88
- statCard(model.localized("task_detail.priority"), value: model.priorityLabel(task.priority), color: priorityColor(task.priority))
89
- statCard(model.localized("task_detail.due"), value: task.dueDate?.formatted(date: .abbreviated, time: .omitted) ?? "No due date", color: AppPalette.info)
90
- }
91
-
92
- if let description = task.description, !description.isEmpty {
93
- detailBlock(title: model.localized("task_detail.description")) {
94
- Text(description)
95
- .foregroundStyle(AppPalette.textSecondary)
96
- }
97
- }
98
-
99
- if let attachment = task.attachment {
100
- detailBlock(title: "Media") {
101
- VStack(alignment: .leading, spacing: 10) {
102
- Label(attachment.title, systemImage: attachment.mediaType == "video" ? "play.rectangle.fill" : "waveform")
103
- .font(.headline)
104
- if attachment.mediaType == "video", let mediaURL = attachment.url {
105
- VideoPlayer(player: AVPlayer(url: mediaURL))
106
- .frame(minHeight: 220)
107
- .clipShape(RoundedRectangle(cornerRadius: 16))
108
- } else {
109
- Text(model.localized("media_player.error"))
110
- .font(.subheadline)
111
- .foregroundStyle(AppPalette.textSecondary)
112
- }
113
- }
114
- .padding()
115
- .frame(maxWidth: .infinity, alignment: .leading)
116
- .background(AppPalette.surfaceSecondary)
117
- .clipShape(RoundedRectangle(cornerRadius: 20))
118
- }
119
- }
120
-
121
- detailBlock(title: model.localized("task_detail.details")) {
122
- VStack(spacing: 12) {
123
- detailRow(model.localized("task_detail.project"), value: model.project(for: task)?.name ?? "No project", symbol: "folder")
124
- detailRow(model.localized("task_detail.assignee"), value: model.assignee(for: task)?.name ?? model.localized("task_detail.unassigned"), symbol: "person")
125
- detailRow(model.localized("task_detail.tags"), value: task.tags.isEmpty ? "No tags" : task.tags.joined(separator: ", "), symbol: "tag")
126
- detailRow(model.localized("task_detail.created"), value: task.createdAt.formatted(date: .abbreviated, time: .shortened), symbol: "clock")
127
- }
128
- }
129
-
130
- HStack(spacing: 12) {
131
- Button(model.localized("task_detail.edit")) {
132
- model.presentedSheet = .editTask(task.id)
133
- }
134
- .buttonStyle(.borderedProminent)
135
-
136
- Button(model.localized("task_detail.assign_to")) {
137
- model.presentedSheet = .assignTask(task.id)
138
- }
139
- .buttonStyle(.bordered)
140
-
141
- Button(task.status == .done ? model.localized("task_detail.reopen") : model.localized("task_detail.complete")) {
142
- model.toggle(task: task)
143
- }
144
- .buttonStyle(.bordered)
145
-
146
- Button(model.localized("task_detail.delete"), role: .destructive) {
147
- requestDelete()
148
- }
149
- .buttonStyle(.bordered)
150
- }
151
- }
152
- }
153
-
154
- private func badge(_ text: String, color: Color) -> some View {
155
- Text(text)
156
- .font(.subheadline.weight(.semibold))
157
- .padding(.horizontal, 12)
158
- .padding(.vertical, 8)
159
- .background(color.opacity(0.14))
160
- .foregroundStyle(color)
161
- .clipShape(Capsule())
162
- }
163
-
164
- private func statCard(_ title: String, value: String, color: Color) -> some View {
165
- VStack(alignment: .leading, spacing: 8) {
166
- Text(title)
167
- .font(.caption.weight(.semibold))
168
- .foregroundStyle(AppPalette.textSecondary)
169
- Text(value)
170
- .font(.headline)
171
- Circle()
172
- .fill(color)
173
- .frame(width: 10, height: 10)
174
- }
175
- .frame(maxWidth: .infinity, alignment: .leading)
176
- .padding()
177
- .background(AppPalette.surfaceSecondary)
178
- .clipShape(RoundedRectangle(cornerRadius: 20))
179
- }
180
-
181
- private func detailBlock<Content: View>(title: String, @ViewBuilder content: () -> Content) -> some View {
182
- VStack(alignment: .leading, spacing: 10) {
183
- Text(title.uppercased())
184
- .font(.caption.weight(.bold))
185
- .foregroundStyle(AppPalette.textTertiary)
186
- content()
187
- }
188
- }
189
-
190
- private func detailRow(_ title: String, value: String, symbol: String) -> some View {
191
- HStack {
192
- Label(title, systemImage: symbol)
193
- .foregroundStyle(AppPalette.textSecondary)
194
- Spacer()
195
- Text(value)
196
- .multilineTextAlignment(.trailing)
197
- }
198
- .padding()
199
- .background(AppPalette.surfaceSecondary)
200
- .clipShape(RoundedRectangle(cornerRadius: 18))
201
- }
202
-
203
- private func priorityColor(_ priority: TaskPriority) -> Color {
204
- switch priority {
205
- case .low: Color(hex: "#9CA3AF")
206
- case .medium: AppPalette.info
207
- case .high: AppPalette.warning
208
- case .urgent: AppPalette.danger
209
- }
210
- }
211
-
212
- private func statusColor(_ status: TaskStatus) -> Color {
213
- switch status {
214
- case .todo: Color(hex: "#9CA3AF")
215
- case .inProgress: AppPalette.brandPrimary
216
- case .done: AppPalette.success
217
- }
218
- }
219
- }
@@ -1,320 +0,0 @@
1
- import Foundation
2
- import Observation
3
- import SwiftUI
4
-
5
- @MainActor
6
- @Observable
7
- final class AppModel {
8
- var selectedSection: AppSection = .home
9
- var homeFilter: HomeFilter = .today
10
- var sortOrder: SortOrder = .dueDate
11
- var searchQuery = ""
12
- var selectedTaskID: UUID?
13
- var selectedProjectID: UUID?
14
- var presentedSheet: PresentedSheet?
15
- var preferences = Preferences(
16
- theme: .system,
17
- defaultPriority: .medium,
18
- notificationsEnabled: true,
19
- remindersEnabled: true
20
- )
21
- var currentUser: UserProfile
22
- var team: [UserProfile]
23
- var projects: [Project]
24
- var tasks: [Task]
25
- var toastMessage: String?
26
-
27
- private let calendar = Calendar.current
28
-
29
- init() {
30
- let me = UserProfile(
31
- id: UUID(),
32
- name: "Nora Malik",
33
- firstName: "Nora",
34
- email: "nora@taskflow.app",
35
- avatarSymbol: "person.crop.circle.fill"
36
- )
37
- currentUser = me
38
-
39
- let inbox = Project(id: UUID(), name: "Inbox Zero", colorHex: "#5B4FE8", icon: "tray.full")
40
- let launch = Project(id: UUID(), name: "Product Launch", colorHex: "#E8634F", icon: "rocket")
41
- let studio = Project(id: UUID(), name: "Studio Refresh", colorHex: "#2D9D5E", icon: "paintbrush.pointed")
42
- projects = [inbox, launch, studio]
43
-
44
- let leo = UserProfile(id: UUID(), name: "Leo Park", firstName: "Leo", email: "leo@taskflow.app", avatarSymbol: "person.crop.circle")
45
- let maya = UserProfile(id: UUID(), name: "Maya Chen", firstName: "Maya", email: "maya@taskflow.app", avatarSymbol: "person.crop.circle.badge.checkmark")
46
- team = [me, leo, maya]
47
-
48
- let now = Date()
49
- tasks = [
50
- Task(
51
- id: UUID(),
52
- title: "Finalize keynote outline",
53
- description: "Align story arc, confirm metrics slide, and leave time for live demo rehearsal.",
54
- status: .inProgress,
55
- priority: .urgent,
56
- dueDate: calendar.date(byAdding: .hour, value: 6, to: now),
57
- projectID: launch.id,
58
- assigneeID: me.id,
59
- tags: ["launch", "slides"],
60
- createdAt: calendar.date(byAdding: .day, value: -4, to: now) ?? now,
61
- updatedAt: now,
62
- attachment: TaskAttachment(mediaType: "video", title: "Preview reel", url: URL(string: "https://example.com/reel.mp4"))
63
- ),
64
- Task(
65
- id: UUID(),
66
- title: "Book photographer",
67
- description: "Shortlist candidates and secure availability for the launch event.",
68
- status: .todo,
69
- priority: .high,
70
- dueDate: calendar.date(byAdding: .day, value: 2, to: now),
71
- projectID: launch.id,
72
- assigneeID: leo.id,
73
- tags: ["launch", "vendor"],
74
- createdAt: calendar.date(byAdding: .day, value: -2, to: now) ?? now,
75
- updatedAt: now,
76
- attachment: nil
77
- ),
78
- Task(
79
- id: UUID(),
80
- title: "Refine onboarding checklist",
81
- description: "Add copy tweaks from user testing and reduce setup friction.",
82
- status: .done,
83
- priority: .medium,
84
- dueDate: calendar.date(byAdding: .day, value: -1, to: now),
85
- projectID: inbox.id,
86
- assigneeID: maya.id,
87
- tags: ["ux", "copy"],
88
- createdAt: calendar.date(byAdding: .day, value: -5, to: now) ?? now,
89
- updatedAt: calendar.date(byAdding: .day, value: -1, to: now) ?? now,
90
- attachment: nil
91
- ),
92
- Task(
93
- id: UUID(),
94
- title: "Source new desk lamps",
95
- description: "Collect options that fit the warm material palette for the studio.",
96
- status: .todo,
97
- priority: .low,
98
- dueDate: calendar.date(byAdding: .day, value: 5, to: now),
99
- projectID: studio.id,
100
- assigneeID: nil,
101
- tags: ["studio"],
102
- createdAt: calendar.date(byAdding: .day, value: -3, to: now) ?? now,
103
- updatedAt: now,
104
- attachment: nil
105
- )
106
- ]
107
-
108
- selectedTaskID = tasks.first?.id
109
- selectedProjectID = projects.first?.id
110
- }
111
-
112
- var selectedTask: Task? {
113
- tasks.first(where: { $0.id == selectedTaskID }) ?? filteredTasks.first
114
- }
115
-
116
- var selectedProject: Project? {
117
- projects.first(where: { $0.id == selectedProjectID }) ?? projects.first
118
- }
119
-
120
- var filteredTasks: [Task] {
121
- let filtered = tasks.filter { task in
122
- let queryMatches: Bool
123
- if searchQuery.isEmpty {
124
- queryMatches = true
125
- } else {
126
- let haystack = [
127
- task.title,
128
- task.description ?? "",
129
- project(for: task)?.name ?? "",
130
- task.tags.joined(separator: " ")
131
- ].joined(separator: " ").localizedCaseInsensitiveContains(searchQuery)
132
- queryMatches = haystack
133
- }
134
-
135
- let filterMatches: Bool
136
- switch homeFilter {
137
- case .all:
138
- filterMatches = true
139
- case .today:
140
- filterMatches = task.dueDate.map(calendar.isDateInToday) ?? false
141
- case .upcoming:
142
- filterMatches = task.status != .done && (task.dueDate.map { $0 > Date() } ?? false)
143
- case .done:
144
- filterMatches = task.status == .done
145
- }
146
- return queryMatches && filterMatches
147
- }
148
-
149
- switch sortOrder {
150
- case .dueDate:
151
- return filtered.sorted { ($0.dueDate ?? .distantFuture) < ($1.dueDate ?? .distantFuture) }
152
- case .priority:
153
- return filtered.sorted { $0.priority.rank > $1.priority.rank }
154
- case .createdAt:
155
- return filtered.sorted { $0.createdAt > $1.createdAt }
156
- }
157
- }
158
-
159
- func project(for task: Task) -> Project? {
160
- guard let projectID = task.projectID else { return nil }
161
- return projects.first(where: { $0.id == projectID })
162
- }
163
-
164
- func assignee(for task: Task) -> UserProfile? {
165
- guard let assigneeID = task.assigneeID else { return nil }
166
- return team.first(where: { $0.id == assigneeID })
167
- }
168
-
169
- func tasks(for project: Project) -> [Task] {
170
- tasks.filter { $0.projectID == project.id }
171
- }
172
-
173
- func count(for filter: HomeFilter) -> Int {
174
- switch filter {
175
- case .all:
176
- return tasks.count
177
- case .today:
178
- return tasks.filter { $0.dueDate.map(calendar.isDateInToday) ?? false }.count
179
- case .upcoming:
180
- return tasks.filter { $0.status != .done && ($0.dueDate.map { $0 > Date() } ?? false) }.count
181
- case .done:
182
- return tasks.filter { $0.status == .done }.count
183
- }
184
- }
185
-
186
- func greeting() -> String {
187
- let hour = calendar.component(.hour, from: Date())
188
- let key = hour < 12 ? "home.greeting.morning" : (hour < 18 ? "home.greeting.afternoon" : "home.greeting.evening")
189
- return String(format: localized(key), currentUser.firstName)
190
- }
191
-
192
- func todayCountText() -> String {
193
- let count = count(for: .today)
194
- if count == 0 { return localized("home.task_count.none") }
195
- if count == 1 { return String(format: localized("home.task_count.one"), count) }
196
- return String(format: localized("home.task_count.other"), count)
197
- }
198
-
199
- func statusLabel(_ status: TaskStatus) -> String {
200
- localized("status.\(status.rawValue)")
201
- }
202
-
203
- func priorityLabel(_ priority: TaskPriority) -> String {
204
- localized("priority.\(priority.rawValue)")
205
- }
206
-
207
- func localized(_ key: String) -> String {
208
- NSLocalizedString(key, comment: "")
209
- }
210
-
211
- func toggle(task: Task) {
212
- guard let index = tasks.firstIndex(where: { $0.id == task.id }) else { return }
213
- tasks[index].status = task.status == .done ? .todo : .done
214
- tasks[index].updatedAt = Date()
215
- toastMessage = localized("task_detail.task_updated")
216
- }
217
-
218
- func delete(task: Task) {
219
- tasks.removeAll { $0.id == task.id }
220
- if selectedTaskID == task.id {
221
- selectedTaskID = filteredTasks.first?.id
222
- }
223
- toastMessage = localized("task_detail.task_deleted")
224
- }
225
-
226
- func saveTask(_ draft: TaskDraft, editing taskID: UUID?) {
227
- let tags = draft.tagsText
228
- .split(separator: ",")
229
- .map { $0.trimmingCharacters(in: .whitespacesAndNewlines) }
230
- .filter { !$0.isEmpty }
231
- let dueDate = draft.dueDateEnabled ? draft.dueDate : nil
232
- let assignee = draft.assignToSelf ? currentUser.id : nil
233
-
234
- if let taskID, let index = tasks.firstIndex(where: { $0.id == taskID }) {
235
- tasks[index].title = draft.title
236
- tasks[index].description = draft.description.isEmpty ? nil : draft.description
237
- tasks[index].projectID = draft.projectID
238
- tasks[index].priority = draft.priority
239
- tasks[index].dueDate = dueDate
240
- tasks[index].assigneeID = assignee
241
- tasks[index].tags = tags
242
- tasks[index].updatedAt = Date()
243
- selectedTaskID = taskID
244
- toastMessage = localized("edit_task.success")
245
- } else {
246
- let task = Task(
247
- id: UUID(),
248
- title: draft.title,
249
- description: draft.description.isEmpty ? nil : draft.description,
250
- status: .todo,
251
- priority: draft.priority,
252
- dueDate: dueDate,
253
- projectID: draft.projectID,
254
- assigneeID: assignee,
255
- tags: tags,
256
- createdAt: Date(),
257
- updatedAt: Date(),
258
- attachment: nil
259
- )
260
- tasks.insert(task, at: 0)
261
- selectedTaskID = task.id
262
- toastMessage = localized("create_task.success")
263
- }
264
- }
265
-
266
- func makeDraft(task: Task?) -> TaskDraft {
267
- guard let task else { return TaskDraft() }
268
- return TaskDraft(
269
- title: task.title,
270
- description: task.description ?? "",
271
- projectID: task.projectID,
272
- priority: task.priority,
273
- dueDateEnabled: task.dueDate != nil,
274
- dueDate: task.dueDate ?? Date(),
275
- tagsText: task.tags.joined(separator: ", "),
276
- assignToSelf: task.assigneeID == currentUser.id
277
- )
278
- }
279
-
280
- func createProject(_ draft: ProjectDraft) {
281
- let project = Project(
282
- id: UUID(),
283
- name: draft.name,
284
- colorHex: draft.colorHex,
285
- icon: "folder"
286
- )
287
- projects.append(project)
288
- selectedProjectID = project.id
289
- toastMessage = localized("projects.created")
290
- }
291
-
292
- func updateProfile(name: String, email: String) {
293
- currentUser.name = name
294
- currentUser.firstName = name.split(separator: " ").first.map(String.init) ?? name
295
- currentUser.email = email
296
- toastMessage = localized("profile.success")
297
- }
298
-
299
- func assignTask(taskID: UUID, userID: UUID) {
300
- guard let index = tasks.firstIndex(where: { $0.id == taskID }) else { return }
301
- tasks[index].assigneeID = userID
302
- tasks[index].updatedAt = Date()
303
- toastMessage = localized("task_detail.task_updated")
304
- }
305
-
306
- func exportData() {
307
- toastMessage = localized("settings.export_success")
308
- }
309
- }
310
-
311
- private extension TaskPriority {
312
- var rank: Int {
313
- switch self {
314
- case .low: 1
315
- case .medium: 2
316
- case .high: 3
317
- case .urgent: 4
318
- }
319
- }
320
- }