openuispec 0.1.27 → 0.1.29

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 (123) hide show
  1. package/README.md +52 -55
  2. package/cli/configure-target.ts +416 -0
  3. package/cli/index.ts +14 -3
  4. package/cli/init.ts +241 -55
  5. package/cli/target-presets.json +746 -0
  6. package/docs/implementation-notes.md +47 -10
  7. package/docs/release-notes-v0.1.26.md +1 -1
  8. package/docs/release-notes-v0.1.28.md +25 -0
  9. package/docs/stress-test-maturity-report.md +1 -1
  10. package/drift/index.ts +31 -11
  11. package/examples/taskflow/AGENTS.md +113 -0
  12. package/examples/taskflow/CLAUDE.md +113 -0
  13. package/examples/taskflow/backend/.gitkeep +1 -0
  14. package/examples/taskflow/generated/android/TaskFlow/README.md +43 -0
  15. package/examples/taskflow/generated/android/TaskFlow/app/build.gradle.kts +76 -0
  16. package/examples/taskflow/generated/android/TaskFlow/app/proguard-rules.pro +1 -0
  17. package/examples/taskflow/generated/android/TaskFlow/app/src/main/AndroidManifest.xml +21 -0
  18. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/MainActivity.kt +19 -0
  19. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/TaskFlowApp.kt +283 -0
  20. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/model/DomainModels.kt +106 -0
  21. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/model/SampleData.kt +57 -0
  22. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/components/Common.kt +109 -0
  23. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/HomeScreen.kt +112 -0
  24. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/ProjectsScreen.kt +61 -0
  25. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/SettingsScreen.kt +82 -0
  26. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/TaskDetailScreen.kt +111 -0
  27. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/sheets/Sheets.kt +77 -0
  28. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Color.kt +30 -0
  29. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Theme.kt +86 -0
  30. package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Type.kt +57 -0
  31. package/examples/taskflow/generated/android/TaskFlow/app/src/main/res/values/strings.xml +155 -0
  32. package/examples/taskflow/generated/android/TaskFlow/app/src/main/res/values/themes.xml +4 -0
  33. package/examples/taskflow/generated/android/TaskFlow/build.gradle.kts +5 -0
  34. package/examples/taskflow/generated/android/TaskFlow/gradle/gradle-daemon-jvm.properties +12 -0
  35. package/examples/taskflow/generated/android/TaskFlow/gradle/wrapper/gradle-wrapper.jar +0 -0
  36. package/examples/taskflow/generated/android/TaskFlow/gradle/wrapper/gradle-wrapper.properties +7 -0
  37. package/examples/taskflow/generated/android/TaskFlow/gradle.properties +4 -0
  38. package/examples/taskflow/generated/android/TaskFlow/gradlew +18 -0
  39. package/examples/taskflow/generated/android/TaskFlow/gradlew.bat +12 -0
  40. package/examples/taskflow/generated/android/TaskFlow/settings.gradle.kts +18 -0
  41. package/examples/taskflow/generated/ios/TaskFlow/README.md +21 -0
  42. package/examples/taskflow/generated/ios/TaskFlow/Resources/en.lproj/Localizable.strings +115 -0
  43. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/App/TaskFlowApp.swift +24 -0
  44. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Components/AppChrome.swift +150 -0
  45. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Flows/TaskEditorSheet.swift +220 -0
  46. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Models/DomainModels.swift +122 -0
  47. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/CalendarView.swift +21 -0
  48. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/HomeView.swift +201 -0
  49. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProfileEditView.swift +48 -0
  50. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProjectDetailView.swift +59 -0
  51. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProjectsView.swift +63 -0
  52. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/SettingsView.swift +85 -0
  53. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/TaskDetailView.swift +219 -0
  54. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Support/AppModel.swift +320 -0
  55. package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Support/AppSupport.swift +41 -0
  56. package/examples/taskflow/generated/ios/TaskFlow/project.yml +26 -0
  57. package/examples/taskflow/generated/web/TaskFlow/README.md +19 -0
  58. package/examples/taskflow/generated/web/TaskFlow/index.html +12 -0
  59. package/examples/taskflow/generated/web/TaskFlow/package-lock.json +1908 -0
  60. package/examples/taskflow/generated/web/TaskFlow/package.json +24 -0
  61. package/examples/taskflow/generated/web/TaskFlow/src/App.tsx +58 -0
  62. package/examples/taskflow/generated/web/TaskFlow/src/AppShell.tsx +55 -0
  63. package/examples/taskflow/generated/web/TaskFlow/src/components/Common.tsx +82 -0
  64. package/examples/taskflow/generated/web/TaskFlow/src/components/Modals.tsx +191 -0
  65. package/examples/taskflow/generated/web/TaskFlow/src/components/Nav.tsx +41 -0
  66. package/examples/taskflow/generated/web/TaskFlow/src/generated/messages.ts +131 -0
  67. package/examples/taskflow/generated/web/TaskFlow/src/hooks.ts +25 -0
  68. package/examples/taskflow/generated/web/TaskFlow/src/i18n.ts +39 -0
  69. package/examples/taskflow/generated/web/TaskFlow/src/locales.en.json +111 -0
  70. package/examples/taskflow/generated/web/TaskFlow/src/main.tsx +13 -0
  71. package/examples/taskflow/generated/web/TaskFlow/src/screens/HomeScreen.tsx +111 -0
  72. package/examples/taskflow/generated/web/TaskFlow/src/screens/ProjectsScreen.tsx +82 -0
  73. package/examples/taskflow/generated/web/TaskFlow/src/screens/SettingsScreens.tsx +132 -0
  74. package/examples/taskflow/generated/web/TaskFlow/src/screens/TaskDetail.tsx +105 -0
  75. package/examples/taskflow/generated/web/TaskFlow/src/store.ts +216 -0
  76. package/examples/taskflow/generated/web/TaskFlow/src/styles.css +617 -0
  77. package/examples/taskflow/generated/web/TaskFlow/src/types.ts +64 -0
  78. package/examples/taskflow/generated/web/TaskFlow/src/utils.ts +78 -0
  79. package/examples/taskflow/generated/web/TaskFlow/tsconfig.json +21 -0
  80. package/examples/taskflow/generated/web/TaskFlow/vite.config.ts +6 -0
  81. package/examples/taskflow/openuispec/README.md +54 -0
  82. package/examples/taskflow/{openuispec.yaml → openuispec/openuispec.yaml} +2 -0
  83. package/examples/todo-orbit/AGENTS.md +48 -22
  84. package/examples/todo-orbit/CLAUDE.md +48 -22
  85. package/examples/todo-orbit/backend/.gitkeep +1 -0
  86. package/examples/todo-orbit/openuispec/README.md +9 -4
  87. package/examples/todo-orbit/openuispec/openuispec.yaml +2 -0
  88. package/package.json +1 -1
  89. package/prepare/index.ts +811 -25
  90. package/schema/openuispec.schema.json +10 -0
  91. package/schema/semantic-lint.ts +36 -12
  92. package/schema/validate.ts +9 -4
  93. package/status/index.ts +16 -3
  94. /package/examples/taskflow/{contracts → openuispec/contracts}/README.md +0 -0
  95. /package/examples/taskflow/{contracts → openuispec/contracts}/action_trigger.yaml +0 -0
  96. /package/examples/taskflow/{contracts → openuispec/contracts}/collection.yaml +0 -0
  97. /package/examples/taskflow/{contracts → openuispec/contracts}/data_display.yaml +0 -0
  98. /package/examples/taskflow/{contracts → openuispec/contracts}/feedback.yaml +0 -0
  99. /package/examples/taskflow/{contracts → openuispec/contracts}/input_field.yaml +0 -0
  100. /package/examples/taskflow/{contracts → openuispec/contracts}/nav_container.yaml +0 -0
  101. /package/examples/taskflow/{contracts → openuispec/contracts}/surface.yaml +0 -0
  102. /package/examples/taskflow/{contracts → openuispec/contracts}/x_media_player.yaml +0 -0
  103. /package/examples/taskflow/{flows → openuispec/flows}/create_task.yaml +0 -0
  104. /package/examples/taskflow/{flows → openuispec/flows}/edit_task.yaml +0 -0
  105. /package/examples/taskflow/{locales → openuispec/locales}/en.json +0 -0
  106. /package/examples/taskflow/{platform → openuispec/platform}/android.yaml +0 -0
  107. /package/examples/taskflow/{platform → openuispec/platform}/ios.yaml +0 -0
  108. /package/examples/taskflow/{platform → openuispec/platform}/web.yaml +0 -0
  109. /package/examples/taskflow/{screens → openuispec/screens}/calendar.yaml +0 -0
  110. /package/examples/taskflow/{screens → openuispec/screens}/home.yaml +0 -0
  111. /package/examples/taskflow/{screens → openuispec/screens}/profile_edit.yaml +0 -0
  112. /package/examples/taskflow/{screens → openuispec/screens}/project_detail.yaml +0 -0
  113. /package/examples/taskflow/{screens → openuispec/screens}/projects.yaml +0 -0
  114. /package/examples/taskflow/{screens → openuispec/screens}/settings.yaml +0 -0
  115. /package/examples/taskflow/{screens → openuispec/screens}/task_detail.yaml +0 -0
  116. /package/examples/taskflow/{tokens → openuispec/tokens}/color.yaml +0 -0
  117. /package/examples/taskflow/{tokens → openuispec/tokens}/elevation.yaml +0 -0
  118. /package/examples/taskflow/{tokens → openuispec/tokens}/icons.yaml +0 -0
  119. /package/examples/taskflow/{tokens → openuispec/tokens}/layout.yaml +0 -0
  120. /package/examples/taskflow/{tokens → openuispec/tokens}/motion.yaml +0 -0
  121. /package/examples/taskflow/{tokens → openuispec/tokens}/spacing.yaml +0 -0
  122. /package/examples/taskflow/{tokens → openuispec/tokens}/themes.yaml +0 -0
  123. /package/examples/taskflow/{tokens → openuispec/tokens}/typography.yaml +0 -0
@@ -0,0 +1,5 @@
1
+ plugins {
2
+ id("com.android.application") version "8.7.3" apply false
3
+ id("org.jetbrains.kotlin.android") version "2.0.21" apply false
4
+ id("org.jetbrains.kotlin.plugin.compose") version "2.0.21" apply false
5
+ }
@@ -0,0 +1,12 @@
1
+ #This file is generated by updateDaemonJvm
2
+ toolchainUrl.FREE_BSD.AARCH64=https\://api.foojay.io/disco/v3.0/ids/ec7520a1e057cd116f9544c42142a16b/redirect
3
+ toolchainUrl.FREE_BSD.X86_64=https\://api.foojay.io/disco/v3.0/ids/4c4f879899012ff0a8b2e2117df03b0e/redirect
4
+ toolchainUrl.LINUX.AARCH64=https\://api.foojay.io/disco/v3.0/ids/ec7520a1e057cd116f9544c42142a16b/redirect
5
+ toolchainUrl.LINUX.X86_64=https\://api.foojay.io/disco/v3.0/ids/4c4f879899012ff0a8b2e2117df03b0e/redirect
6
+ toolchainUrl.MAC_OS.AARCH64=https\://api.foojay.io/disco/v3.0/ids/73bcfb608d1fde9fb62e462f834a3299/redirect
7
+ toolchainUrl.MAC_OS.X86_64=https\://api.foojay.io/disco/v3.0/ids/846ee0d876d26a26f37aa1ce8de73224/redirect
8
+ toolchainUrl.UNIX.AARCH64=https\://api.foojay.io/disco/v3.0/ids/ec7520a1e057cd116f9544c42142a16b/redirect
9
+ toolchainUrl.UNIX.X86_64=https\://api.foojay.io/disco/v3.0/ids/4c4f879899012ff0a8b2e2117df03b0e/redirect
10
+ toolchainUrl.WINDOWS.AARCH64=https\://api.foojay.io/disco/v3.0/ids/9482ddec596298c84656d31d16652665/redirect
11
+ toolchainUrl.WINDOWS.X86_64=https\://api.foojay.io/disco/v3.0/ids/39701d92e1756bb2f141eb67cd4c660e/redirect
12
+ toolchainVersion=21
@@ -0,0 +1,7 @@
1
+ distributionBase=GRADLE_USER_HOME
2
+ distributionPath=wrapper/dists
3
+ distributionUrl=https\://services.gradle.org/distributions/gradle-9.4.0-bin.zip
4
+ networkTimeout=10000
5
+ validateDistributionUrl=true
6
+ zipStoreBase=GRADLE_USER_HOME
7
+ zipStorePath=wrapper/dists
@@ -0,0 +1,4 @@
1
+ org.gradle.jvmargs=-Xmx2048m -Dfile.encoding=UTF-8
2
+ android.useAndroidX=true
3
+ kotlin.code.style=official
4
+ android.nonTransitiveRClass=true
@@ -0,0 +1,18 @@
1
+ #!/bin/sh
2
+
3
+ ##############################################################################
4
+ ##
5
+ ## Gradle start up script for POSIX generated by assistant
6
+ ##
7
+ ##############################################################################
8
+
9
+ APP_HOME=$(cd "${0%/*}" >/dev/null 2>&1; pwd -P)
10
+ CLASSPATH="$APP_HOME/gradle/wrapper/gradle-wrapper.jar"
11
+
12
+ if [ -n "$JAVA_HOME" ] ; then
13
+ JAVACMD="$JAVA_HOME/bin/java"
14
+ else
15
+ JAVACMD="java"
16
+ fi
17
+
18
+ exec "$JAVACMD" -Dorg.gradle.appname=gradlew -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@"
@@ -0,0 +1,12 @@
1
+ @ECHO OFF
2
+ SET DIRNAME=%~dp0
3
+ SET APP_HOME=%DIRNAME%
4
+ SET CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
5
+
6
+ IF DEFINED JAVA_HOME (
7
+ SET JAVA_EXE=%JAVA_HOME%\bin\java.exe
8
+ ) ELSE (
9
+ SET JAVA_EXE=java.exe
10
+ )
11
+
12
+ "%JAVA_EXE%" -Dorg.gradle.appname=gradlew -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %*
@@ -0,0 +1,18 @@
1
+ pluginManagement {
2
+ repositories {
3
+ google()
4
+ mavenCentral()
5
+ gradlePluginPortal()
6
+ }
7
+ }
8
+
9
+ dependencyResolutionManagement {
10
+ repositoriesMode.set(RepositoriesMode.FAIL_ON_PROJECT_REPOS)
11
+ repositories {
12
+ google()
13
+ mavenCentral()
14
+ }
15
+ }
16
+
17
+ rootProject.name = "TaskFlow"
18
+ include(":app")
@@ -0,0 +1,21 @@
1
+ # TaskFlow iOS Target
2
+
3
+ This directory contains a lightweight SwiftUI interpretation of the local `openuispec/` source for TaskFlow.
4
+
5
+ What is included:
6
+ - SwiftUI app shell with adaptive compact vs regular navigation
7
+ - Spec-derived screens: home, task detail, projects, project detail, calendar, settings, profile edit
8
+ - Spec-derived create/edit task sheets
9
+ - Sample data matching the TaskFlow entities in `openuispec/openuispec.yaml`
10
+ - English localization from the local OpenUISpec locale file
11
+ - `project.yml` for XcodeGen-based project creation
12
+
13
+ Notes:
14
+ - This target is generated only from the local TaskFlow spec and does not depend on other sample outputs.
15
+ - The implementation is intentionally lean: API calls are represented with local sample state mutations.
16
+ - `NavigationSplitView` is used for regular-width devices to reflect the adaptive nav and home split-view intent from the spec.
17
+
18
+ Suggested next steps:
19
+ 1. Install XcodeGen if needed.
20
+ 2. Run `xcodegen generate` in this directory.
21
+ 3. Open the generated Xcode project in Xcode 16+ and run on iOS 17+.
@@ -0,0 +1,115 @@
1
+ "nav.tasks" = "Tasks";
2
+ "nav.projects" = "Projects";
3
+ "nav.calendar" = "Calendar";
4
+ "nav.settings" = "Settings";
5
+ "home.greeting.morning" = "Good morning, %@";
6
+ "home.greeting.afternoon" = "Good afternoon, %@";
7
+ "home.greeting.evening" = "Good evening, %@";
8
+ "home.task_count.none" = "No tasks today";
9
+ "home.task_count.one" = "%d task today";
10
+ "home.task_count.other" = "%d tasks today";
11
+ "home.search_label" = "Search tasks";
12
+ "home.search_placeholder" = "Search by title, tag, or project...";
13
+ "home.filter.all" = "All";
14
+ "home.filter.today" = "Today";
15
+ "home.filter.upcoming" = "Upcoming";
16
+ "home.filter.done" = "Done";
17
+ "home.empty_title" = "All caught up!";
18
+ "home.empty_body" = "No tasks match this filter. Tap + to add one.";
19
+ "home.new_task" = "New task";
20
+ "home.mark_complete" = "Mark %@ complete";
21
+ "task_detail.status" = "Status";
22
+ "task_detail.priority" = "Priority";
23
+ "task_detail.due" = "Due";
24
+ "task_detail.description" = "Description";
25
+ "task_detail.details" = "Details";
26
+ "task_detail.project" = "Project";
27
+ "task_detail.assignee" = "Assignee";
28
+ "task_detail.unassigned" = "Unassigned";
29
+ "task_detail.tags" = "Tags";
30
+ "task_detail.created" = "Created";
31
+ "task_detail.edit" = "Edit task";
32
+ "task_detail.reopen" = "Reopen task";
33
+ "task_detail.complete" = "Mark complete";
34
+ "task_detail.delete" = "Delete task";
35
+ "task_detail.delete_title" = "Delete task?";
36
+ "task_detail.delete_message" = "This action cannot be undone. The task \"%@\" will be permanently removed.";
37
+ "task_detail.task_updated" = "Task updated";
38
+ "task_detail.task_deleted" = "Task deleted";
39
+ "task_detail.assign_to" = "Assign to";
40
+ "task_detail.search_people" = "Search people";
41
+ "task_detail.search_people_placeholder" = "Name or email...";
42
+ "create_task.title" = "New task";
43
+ "create_task.save" = "Save";
44
+ "create_task.saving" = "Saving...";
45
+ "create_task.field_title" = "Title";
46
+ "create_task.field_title_placeholder" = "What needs to be done?";
47
+ "create_task.field_description" = "Description";
48
+ "create_task.field_description_placeholder" = "Add details, notes, or context...";
49
+ "create_task.field_project" = "Project";
50
+ "create_task.field_project_placeholder" = "Select a project";
51
+ "create_task.field_priority" = "Priority";
52
+ "create_task.field_due_date" = "Due date";
53
+ "create_task.field_due_date_placeholder" = "No due date";
54
+ "create_task.field_tags" = "Tags";
55
+ "create_task.field_tags_placeholder" = "Add tags separated by commas";
56
+ "create_task.field_tags_helper" = "Press comma or Enter to add a tag";
57
+ "create_task.field_assign_to_me" = "Assign to me";
58
+ "create_task.success" = "Task created";
59
+ "edit_task.title" = "Edit task";
60
+ "edit_task.save" = "Save";
61
+ "edit_task.saving" = "Saving...";
62
+ "edit_task.field_title" = "Title";
63
+ "edit_task.field_description" = "Description";
64
+ "edit_task.field_priority" = "Priority";
65
+ "edit_task.field_due_date" = "Due date";
66
+ "edit_task.success" = "Task updated";
67
+ "projects.title" = "Projects";
68
+ "projects.new_project" = "New project";
69
+ "projects.empty_title" = "No projects yet";
70
+ "projects.empty_body" = "Create a project to organize your tasks.";
71
+ "projects.dialog_title" = "New project";
72
+ "projects.field_name" = "Project name";
73
+ "projects.field_name_placeholder" = "e.g., Product Launch";
74
+ "projects.field_color" = "Color";
75
+ "projects.created" = "Project created";
76
+ "project_detail.empty_title" = "No tasks in this project";
77
+ "project_detail.empty_body" = "Add a task to get started.";
78
+ "settings.preferences" = "Preferences";
79
+ "settings.theme" = "Theme";
80
+ "settings.theme_system" = "System";
81
+ "settings.theme_light" = "Light";
82
+ "settings.theme_dark" = "Dark";
83
+ "settings.theme_warm" = "Warm";
84
+ "settings.default_priority" = "Default priority";
85
+ "settings.notifications" = "Push notifications";
86
+ "settings.reminders" = "Due date reminders";
87
+ "settings.reminders_helper" = "Get notified 1 hour before a task is due";
88
+ "settings.data" = "Data";
89
+ "settings.export" = "Export data";
90
+ "settings.export_success" = "Export sent to your email";
91
+ "settings.delete_account" = "Delete account";
92
+ "settings.delete_title" = "Delete your account?";
93
+ "settings.delete_message" = "This will permanently delete your account and all your data. This action cannot be undone.";
94
+ "settings.delete_confirm" = "Delete my account";
95
+ "settings.app_version" = "TaskFlow v1.0.0";
96
+ "settings.app_credit" = "Built with OpenUISpec";
97
+ "profile.change_photo" = "Change photo";
98
+ "profile.field_name" = "Name";
99
+ "profile.field_email" = "Email";
100
+ "profile.save" = "Save changes";
101
+ "profile.success" = "Profile updated";
102
+ "calendar.title" = "Calendar";
103
+ "calendar.coming_soon" = "Coming in a future version";
104
+ "priority.low" = "Low";
105
+ "priority.medium" = "Medium";
106
+ "priority.high" = "High";
107
+ "priority.urgent" = "Urgent";
108
+ "status.todo" = "To do";
109
+ "status.in_progress" = "In progress";
110
+ "status.done" = "Done";
111
+ "media_player.error" = "Unable to play media";
112
+ "common.cancel" = "Cancel";
113
+ "common.delete" = "Delete";
114
+ "common.create" = "Create";
115
+ "common.save" = "Save";
@@ -0,0 +1,24 @@
1
+ import SwiftUI
2
+
3
+ @main
4
+ struct TaskFlowApp: App {
5
+ @State private var model = AppModel()
6
+
7
+ var body: some Scene {
8
+ WindowGroup {
9
+ AppChrome(model: model)
10
+ .preferredColorScheme(colorScheme(for: model.preferences.theme))
11
+ }
12
+ }
13
+
14
+ private func colorScheme(for theme: ThemePreference) -> ColorScheme? {
15
+ switch theme {
16
+ case .system:
17
+ nil
18
+ case .light, .warm:
19
+ .light
20
+ case .dark:
21
+ .dark
22
+ }
23
+ }
24
+ }
@@ -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
+ }