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.
- package/README.md +52 -55
- package/cli/configure-target.ts +416 -0
- package/cli/index.ts +14 -3
- package/cli/init.ts +241 -55
- package/cli/target-presets.json +746 -0
- package/docs/implementation-notes.md +47 -10
- package/docs/release-notes-v0.1.26.md +1 -1
- package/docs/release-notes-v0.1.28.md +25 -0
- package/docs/stress-test-maturity-report.md +1 -1
- package/drift/index.ts +31 -11
- package/examples/taskflow/AGENTS.md +113 -0
- package/examples/taskflow/CLAUDE.md +113 -0
- package/examples/taskflow/backend/.gitkeep +1 -0
- package/examples/taskflow/generated/android/TaskFlow/README.md +43 -0
- package/examples/taskflow/generated/android/TaskFlow/app/build.gradle.kts +76 -0
- package/examples/taskflow/generated/android/TaskFlow/app/proguard-rules.pro +1 -0
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/AndroidManifest.xml +21 -0
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/MainActivity.kt +19 -0
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/TaskFlowApp.kt +283 -0
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/model/DomainModels.kt +106 -0
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/model/SampleData.kt +57 -0
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/components/Common.kt +109 -0
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/HomeScreen.kt +112 -0
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/ProjectsScreen.kt +61 -0
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/SettingsScreen.kt +82 -0
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/TaskDetailScreen.kt +111 -0
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/sheets/Sheets.kt +77 -0
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Color.kt +30 -0
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Theme.kt +86 -0
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Type.kt +57 -0
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/res/values/strings.xml +155 -0
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/res/values/themes.xml +4 -0
- package/examples/taskflow/generated/android/TaskFlow/build.gradle.kts +5 -0
- package/examples/taskflow/generated/android/TaskFlow/gradle/gradle-daemon-jvm.properties +12 -0
- package/examples/taskflow/generated/android/TaskFlow/gradle/wrapper/gradle-wrapper.jar +0 -0
- package/examples/taskflow/generated/android/TaskFlow/gradle/wrapper/gradle-wrapper.properties +7 -0
- package/examples/taskflow/generated/android/TaskFlow/gradle.properties +4 -0
- package/examples/taskflow/generated/android/TaskFlow/gradlew +18 -0
- package/examples/taskflow/generated/android/TaskFlow/gradlew.bat +12 -0
- package/examples/taskflow/generated/android/TaskFlow/settings.gradle.kts +18 -0
- package/examples/taskflow/generated/ios/TaskFlow/README.md +21 -0
- package/examples/taskflow/generated/ios/TaskFlow/Resources/en.lproj/Localizable.strings +115 -0
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/App/TaskFlowApp.swift +24 -0
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Components/AppChrome.swift +150 -0
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Flows/TaskEditorSheet.swift +220 -0
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Models/DomainModels.swift +122 -0
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/CalendarView.swift +21 -0
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/HomeView.swift +201 -0
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProfileEditView.swift +48 -0
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProjectDetailView.swift +59 -0
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProjectsView.swift +63 -0
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/SettingsView.swift +85 -0
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/TaskDetailView.swift +219 -0
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Support/AppModel.swift +320 -0
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Support/AppSupport.swift +41 -0
- package/examples/taskflow/generated/ios/TaskFlow/project.yml +26 -0
- package/examples/taskflow/generated/web/TaskFlow/README.md +19 -0
- package/examples/taskflow/generated/web/TaskFlow/index.html +12 -0
- package/examples/taskflow/generated/web/TaskFlow/package-lock.json +1908 -0
- package/examples/taskflow/generated/web/TaskFlow/package.json +24 -0
- package/examples/taskflow/generated/web/TaskFlow/src/App.tsx +58 -0
- package/examples/taskflow/generated/web/TaskFlow/src/AppShell.tsx +55 -0
- package/examples/taskflow/generated/web/TaskFlow/src/components/Common.tsx +82 -0
- package/examples/taskflow/generated/web/TaskFlow/src/components/Modals.tsx +191 -0
- package/examples/taskflow/generated/web/TaskFlow/src/components/Nav.tsx +41 -0
- package/examples/taskflow/generated/web/TaskFlow/src/generated/messages.ts +131 -0
- package/examples/taskflow/generated/web/TaskFlow/src/hooks.ts +25 -0
- package/examples/taskflow/generated/web/TaskFlow/src/i18n.ts +39 -0
- package/examples/taskflow/generated/web/TaskFlow/src/locales.en.json +111 -0
- package/examples/taskflow/generated/web/TaskFlow/src/main.tsx +13 -0
- package/examples/taskflow/generated/web/TaskFlow/src/screens/HomeScreen.tsx +111 -0
- package/examples/taskflow/generated/web/TaskFlow/src/screens/ProjectsScreen.tsx +82 -0
- package/examples/taskflow/generated/web/TaskFlow/src/screens/SettingsScreens.tsx +132 -0
- package/examples/taskflow/generated/web/TaskFlow/src/screens/TaskDetail.tsx +105 -0
- package/examples/taskflow/generated/web/TaskFlow/src/store.ts +216 -0
- package/examples/taskflow/generated/web/TaskFlow/src/styles.css +617 -0
- package/examples/taskflow/generated/web/TaskFlow/src/types.ts +64 -0
- package/examples/taskflow/generated/web/TaskFlow/src/utils.ts +78 -0
- package/examples/taskflow/generated/web/TaskFlow/tsconfig.json +21 -0
- package/examples/taskflow/generated/web/TaskFlow/vite.config.ts +6 -0
- package/examples/taskflow/openuispec/README.md +54 -0
- package/examples/taskflow/{openuispec.yaml → openuispec/openuispec.yaml} +2 -0
- package/examples/todo-orbit/AGENTS.md +48 -22
- package/examples/todo-orbit/CLAUDE.md +48 -22
- package/examples/todo-orbit/backend/.gitkeep +1 -0
- package/examples/todo-orbit/openuispec/README.md +9 -4
- package/examples/todo-orbit/openuispec/openuispec.yaml +2 -0
- package/package.json +1 -1
- package/prepare/index.ts +811 -25
- package/schema/openuispec.schema.json +10 -0
- package/schema/semantic-lint.ts +36 -12
- package/schema/validate.ts +9 -4
- package/status/index.ts +16 -3
- /package/examples/taskflow/{contracts → openuispec/contracts}/README.md +0 -0
- /package/examples/taskflow/{contracts → openuispec/contracts}/action_trigger.yaml +0 -0
- /package/examples/taskflow/{contracts → openuispec/contracts}/collection.yaml +0 -0
- /package/examples/taskflow/{contracts → openuispec/contracts}/data_display.yaml +0 -0
- /package/examples/taskflow/{contracts → openuispec/contracts}/feedback.yaml +0 -0
- /package/examples/taskflow/{contracts → openuispec/contracts}/input_field.yaml +0 -0
- /package/examples/taskflow/{contracts → openuispec/contracts}/nav_container.yaml +0 -0
- /package/examples/taskflow/{contracts → openuispec/contracts}/surface.yaml +0 -0
- /package/examples/taskflow/{contracts → openuispec/contracts}/x_media_player.yaml +0 -0
- /package/examples/taskflow/{flows → openuispec/flows}/create_task.yaml +0 -0
- /package/examples/taskflow/{flows → openuispec/flows}/edit_task.yaml +0 -0
- /package/examples/taskflow/{locales → openuispec/locales}/en.json +0 -0
- /package/examples/taskflow/{platform → openuispec/platform}/android.yaml +0 -0
- /package/examples/taskflow/{platform → openuispec/platform}/ios.yaml +0 -0
- /package/examples/taskflow/{platform → openuispec/platform}/web.yaml +0 -0
- /package/examples/taskflow/{screens → openuispec/screens}/calendar.yaml +0 -0
- /package/examples/taskflow/{screens → openuispec/screens}/home.yaml +0 -0
- /package/examples/taskflow/{screens → openuispec/screens}/profile_edit.yaml +0 -0
- /package/examples/taskflow/{screens → openuispec/screens}/project_detail.yaml +0 -0
- /package/examples/taskflow/{screens → openuispec/screens}/projects.yaml +0 -0
- /package/examples/taskflow/{screens → openuispec/screens}/settings.yaml +0 -0
- /package/examples/taskflow/{screens → openuispec/screens}/task_detail.yaml +0 -0
- /package/examples/taskflow/{tokens → openuispec/tokens}/color.yaml +0 -0
- /package/examples/taskflow/{tokens → openuispec/tokens}/elevation.yaml +0 -0
- /package/examples/taskflow/{tokens → openuispec/tokens}/icons.yaml +0 -0
- /package/examples/taskflow/{tokens → openuispec/tokens}/layout.yaml +0 -0
- /package/examples/taskflow/{tokens → openuispec/tokens}/motion.yaml +0 -0
- /package/examples/taskflow/{tokens → openuispec/tokens}/spacing.yaml +0 -0
- /package/examples/taskflow/{tokens → openuispec/tokens}/themes.yaml +0 -0
- /package/examples/taskflow/{tokens → openuispec/tokens}/typography.yaml +0 -0
|
@@ -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
|
|
Binary file
|
|
@@ -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
|
+
}
|