openuispec 0.2.10 → 0.2.11
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 +3 -1
- package/cli/index.ts +4 -2
- package/cli/init.ts +23 -8
- package/docs/cli.md +13 -8
- package/drift/index.ts +41 -15
- package/mcp-server/index.ts +155 -116
- package/mcp-server/screenshot.ts +19 -4
- package/package.json +5 -2
- package/prepare/index.ts +16 -0
- package/scripts/take-all-screenshots.ts +507 -0
- package/status/index.ts +2 -2
- package/examples/social-app/.mcp.json +0 -10
- package/examples/social-app/AGENTS.md +0 -124
- package/examples/social-app/CLAUDE.md +0 -124
- package/examples/social-app/backend/.gitkeep +0 -1
- package/examples/social-app/generated/android/social-app/app/.paparazzi-hashes.json +0 -3
- package/examples/social-app/generated/android/social-app/app/build.gradle.kts +0 -94
- package/examples/social-app/generated/android/social-app/app/src/main/AndroidManifest.xml +0 -26
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/AppContainer.kt +0 -20
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/MainActivity.kt +0 -35
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/SocialAppApplication.kt +0 -13
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/data/MockData.kt +0 -98
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/data/preferences/AppPreferences.kt +0 -19
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/data/preferences/DataStorePreferencesRepository.kt +0 -68
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/data/preferences/PreferencesRepository.kt +0 -15
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/model/Models.kt +0 -34
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/MainShell.kt +0 -390
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/components/Components.kt +0 -234
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/components/ContractPrimitives.kt +0 -641
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/navigation/RootComponent.kt +0 -113
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/ChatDetailScreen.kt +0 -212
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/CreatePostScreen.kt +0 -113
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/DiscoverScreen.kt +0 -137
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/EditProfileScreen.kt +0 -180
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/HomeFeedScreen.kt +0 -169
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/MessagesInboxScreen.kt +0 -85
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/NotificationsScreen.kt +0 -74
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/PostDetailScreen.kt +0 -293
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/ProfileSelfScreen.kt +0 -116
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/SearchResultsScreen.kt +0 -161
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/SettingsScreen.kt +0 -164
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/SettingsStore.kt +0 -95
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/UserProfileScreen.kt +0 -123
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Color.kt +0 -33
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Shape.kt +0 -41
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Spacing.kt +0 -20
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Theme.kt +0 -82
- package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Type.kt +0 -60
- package/examples/social-app/generated/android/social-app/app/src/main/res/drawable/ic_launcher_foreground.xml +0 -9
- package/examples/social-app/generated/android/social-app/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml +0 -5
- package/examples/social-app/generated/android/social-app/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml +0 -5
- package/examples/social-app/generated/android/social-app/app/src/main/res/values/strings.xml +0 -91
- package/examples/social-app/generated/android/social-app/app/src/main/res/values/themes.xml +0 -10
- package/examples/social-app/generated/android/social-app/app/src/main/res/values-ru/strings.xml +0 -79
- package/examples/social-app/generated/android/social-app/app/src/main/res/values-uz/strings.xml +0 -79
- package/examples/social-app/generated/android/social-app/app/src/main/xml/AndroidManifest.xml +0 -23
- package/examples/social-app/generated/android/social-app/app/src/test/kotlin/com/social/app/screenshots/HomeFeedScreenshotTest.kt +0 -34
- package/examples/social-app/generated/android/social-app/build.gradle.kts +0 -7
- package/examples/social-app/generated/android/social-app/gradle/libs.versions.toml +0 -50
- package/examples/social-app/generated/android/social-app/gradle/wrapper/gradle-wrapper.jar +0 -0
- package/examples/social-app/generated/android/social-app/gradle/wrapper/gradle-wrapper.properties +0 -8
- package/examples/social-app/generated/android/social-app/gradle.properties +0 -11
- package/examples/social-app/generated/android/social-app/gradlew +0 -248
- package/examples/social-app/generated/android/social-app/settings.gradle.kts +0 -27
- package/examples/social-app/generated/web/social-app/index.html +0 -12
- package/examples/social-app/generated/web/social-app/package-lock.json +0 -2517
- package/examples/social-app/generated/web/social-app/package.json +0 -27
- package/examples/social-app/generated/web/social-app/src/app/App.tsx +0 -58
- package/examples/social-app/generated/web/social-app/src/components/Shell.tsx +0 -259
- package/examples/social-app/generated/web/social-app/src/components/cards.tsx +0 -317
- package/examples/social-app/generated/web/social-app/src/components/ui.tsx +0 -340
- package/examples/social-app/generated/web/social-app/src/flows/CreatePostFlow.tsx +0 -86
- package/examples/social-app/generated/web/social-app/src/i18n.tsx +0 -59
- package/examples/social-app/generated/web/social-app/src/lib/icons.tsx +0 -85
- package/examples/social-app/generated/web/social-app/src/lib/tokens.ts +0 -70
- package/examples/social-app/generated/web/social-app/src/lib/utils.ts +0 -97
- package/examples/social-app/generated/web/social-app/src/locales/en.json +0 -67
- package/examples/social-app/generated/web/social-app/src/locales/ru.json +0 -67
- package/examples/social-app/generated/web/social-app/src/locales/uz.json +0 -67
- package/examples/social-app/generated/web/social-app/src/main.tsx +0 -16
- package/examples/social-app/generated/web/social-app/src/screens/ChatDetailScreen.tsx +0 -90
- package/examples/social-app/generated/web/social-app/src/screens/DiscoverScreen.tsx +0 -86
- package/examples/social-app/generated/web/social-app/src/screens/EditProfileScreen.tsx +0 -57
- package/examples/social-app/generated/web/social-app/src/screens/HomeFeedScreen.tsx +0 -103
- package/examples/social-app/generated/web/social-app/src/screens/MessagesInboxScreen.tsx +0 -52
- package/examples/social-app/generated/web/social-app/src/screens/NotificationsScreen.tsx +0 -41
- package/examples/social-app/generated/web/social-app/src/screens/PostDetailScreen.tsx +0 -115
- package/examples/social-app/generated/web/social-app/src/screens/ProfileSelfScreen.tsx +0 -57
- package/examples/social-app/generated/web/social-app/src/screens/ProfileUserScreen.tsx +0 -76
- package/examples/social-app/generated/web/social-app/src/screens/SearchResultsScreen.tsx +0 -96
- package/examples/social-app/generated/web/social-app/src/screens/SettingsScreen.tsx +0 -79
- package/examples/social-app/generated/web/social-app/src/state/store.ts +0 -592
- package/examples/social-app/generated/web/social-app/src/styles.css +0 -125
- package/examples/social-app/generated/web/social-app/src/vite-env.d.ts +0 -1
- package/examples/social-app/generated/web/social-app/tsconfig.json +0 -22
- package/examples/social-app/generated/web/social-app/tsconfig.node.json +0 -13
- package/examples/social-app/generated/web/social-app/tsconfig.node.tsbuildinfo +0 -1
- package/examples/social-app/generated/web/social-app/tsconfig.tsbuildinfo +0 -1
- package/examples/social-app/generated/web/social-app/vite.config.d.ts +0 -2
- package/examples/social-app/generated/web/social-app/vite.config.js +0 -6
- package/examples/social-app/generated/web/social-app/vite.config.ts +0 -7
- package/examples/social-app/package.json +0 -13
- package/examples/social-app/take-web-screenshots.ts +0 -97
- package/examples/taskflow/.codex/config.toml +0 -4
- package/examples/taskflow/.mcp.json +0 -10
- package/examples/taskflow/AGENTS.md +0 -124
- package/examples/taskflow/CLAUDE.md +0 -124
- package/examples/taskflow/backend/.gitkeep +0 -1
- package/examples/taskflow/generated/android/TaskFlow/README.md +0 -43
- package/examples/taskflow/generated/android/TaskFlow/app/build.gradle.kts +0 -76
- package/examples/taskflow/generated/android/TaskFlow/app/proguard-rules.pro +0 -1
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/AndroidManifest.xml +0 -21
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/MainActivity.kt +0 -19
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/TaskFlowApp.kt +0 -283
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/model/DomainModels.kt +0 -106
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/model/SampleData.kt +0 -57
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/components/Common.kt +0 -109
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/HomeScreen.kt +0 -112
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/ProjectsScreen.kt +0 -61
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/SettingsScreen.kt +0 -82
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/screens/TaskDetailScreen.kt +0 -111
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/sheets/Sheets.kt +0 -77
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Color.kt +0 -30
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Theme.kt +0 -86
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/java/uz/rsteam/taskflow/ui/theme/Type.kt +0 -57
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/res/values/strings.xml +0 -155
- package/examples/taskflow/generated/android/TaskFlow/app/src/main/res/values/themes.xml +0 -4
- package/examples/taskflow/generated/android/TaskFlow/build.gradle.kts +0 -5
- package/examples/taskflow/generated/android/TaskFlow/gradle/gradle-daemon-jvm.properties +0 -12
- package/examples/taskflow/generated/android/TaskFlow/gradle/wrapper/gradle-wrapper.jar +0 -0
- package/examples/taskflow/generated/android/TaskFlow/gradle/wrapper/gradle-wrapper.properties +0 -7
- package/examples/taskflow/generated/android/TaskFlow/gradle.properties +0 -4
- package/examples/taskflow/generated/android/TaskFlow/gradlew +0 -18
- package/examples/taskflow/generated/android/TaskFlow/gradlew.bat +0 -12
- package/examples/taskflow/generated/android/TaskFlow/settings.gradle.kts +0 -18
- package/examples/taskflow/generated/ios/TaskFlow/README.md +0 -21
- package/examples/taskflow/generated/ios/TaskFlow/Resources/en.lproj/Localizable.strings +0 -115
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/App/TaskFlowApp.swift +0 -24
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Components/AppChrome.swift +0 -150
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Flows/TaskEditorSheet.swift +0 -220
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Models/DomainModels.swift +0 -122
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/CalendarView.swift +0 -21
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/HomeView.swift +0 -201
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProfileEditView.swift +0 -48
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProjectDetailView.swift +0 -59
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/ProjectsView.swift +0 -63
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/SettingsView.swift +0 -85
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Screens/TaskDetailView.swift +0 -219
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Support/AppModel.swift +0 -320
- package/examples/taskflow/generated/ios/TaskFlow/Sources/TaskFlow/Support/AppSupport.swift +0 -41
- package/examples/taskflow/generated/ios/TaskFlow/project.yml +0 -31
- package/examples/taskflow/generated/web/TaskFlow/README.md +0 -19
- package/examples/taskflow/generated/web/TaskFlow/index.html +0 -12
- package/examples/taskflow/generated/web/TaskFlow/package-lock.json +0 -1908
- package/examples/taskflow/generated/web/TaskFlow/package.json +0 -24
- package/examples/taskflow/generated/web/TaskFlow/src/App.tsx +0 -58
- package/examples/taskflow/generated/web/TaskFlow/src/AppShell.tsx +0 -55
- package/examples/taskflow/generated/web/TaskFlow/src/components/Common.tsx +0 -82
- package/examples/taskflow/generated/web/TaskFlow/src/components/Modals.tsx +0 -191
- package/examples/taskflow/generated/web/TaskFlow/src/components/Nav.tsx +0 -41
- package/examples/taskflow/generated/web/TaskFlow/src/generated/messages.ts +0 -131
- package/examples/taskflow/generated/web/TaskFlow/src/hooks.ts +0 -25
- package/examples/taskflow/generated/web/TaskFlow/src/i18n.ts +0 -39
- package/examples/taskflow/generated/web/TaskFlow/src/locales.en.json +0 -111
- package/examples/taskflow/generated/web/TaskFlow/src/main.tsx +0 -13
- package/examples/taskflow/generated/web/TaskFlow/src/screens/HomeScreen.tsx +0 -111
- package/examples/taskflow/generated/web/TaskFlow/src/screens/ProjectsScreen.tsx +0 -82
- package/examples/taskflow/generated/web/TaskFlow/src/screens/SettingsScreens.tsx +0 -132
- package/examples/taskflow/generated/web/TaskFlow/src/screens/TaskDetail.tsx +0 -105
- package/examples/taskflow/generated/web/TaskFlow/src/store.ts +0 -216
- package/examples/taskflow/generated/web/TaskFlow/src/styles.css +0 -617
- package/examples/taskflow/generated/web/TaskFlow/src/types.ts +0 -64
- package/examples/taskflow/generated/web/TaskFlow/src/utils.ts +0 -78
- package/examples/taskflow/generated/web/TaskFlow/tsconfig.json +0 -21
- package/examples/taskflow/generated/web/TaskFlow/vite.config.ts +0 -6
- package/examples/todo-orbit/.codex/config.toml +0 -4
- package/examples/todo-orbit/.mcp.json +0 -10
- package/examples/todo-orbit/AGENTS.md +0 -124
- package/examples/todo-orbit/CLAUDE.md +0 -124
- package/examples/todo-orbit/backend/.gitkeep +0 -1
- package/examples/todo-orbit/generated/android/Todo Orbit/README.md +0 -14
- package/examples/todo-orbit/generated/android/Todo Orbit/app/build.gradle.kts +0 -58
- package/examples/todo-orbit/generated/android/Todo Orbit/app/proguard-rules.pro +0 -1
- package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/AndroidManifest.xml +0 -20
- package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/MainActivity.kt +0 -14
- package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/TodoOrbitApp.kt +0 -345
- package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/support/AppLogic.kt +0 -231
- package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/support/Models.kt +0 -169
- package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/support/Strings.kt +0 -8
- package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/components/CommonComponents.kt +0 -236
- package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/screens/AnalyticsScreen.kt +0 -193
- package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/screens/SettingsScreen.kt +0 -102
- package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/screens/TasksScreen.kt +0 -347
- package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/sheets/EditorSheets.kt +0 -347
- package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/java/uz/rsteam/todoorbit/ui/theme/TodoOrbitTheme.kt +0 -59
- package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/res/values/strings.xml +0 -149
- package/examples/todo-orbit/generated/android/Todo Orbit/app/src/main/res/values-ru/strings.xml +0 -155
- package/examples/todo-orbit/generated/android/Todo Orbit/build.gradle.kts +0 -4
- package/examples/todo-orbit/generated/android/Todo Orbit/gradle/wrapper/gradle-wrapper.jar +0 -0
- package/examples/todo-orbit/generated/android/Todo Orbit/gradle/wrapper/gradle-wrapper.properties +0 -7
- package/examples/todo-orbit/generated/android/Todo Orbit/gradle.properties +0 -4
- package/examples/todo-orbit/generated/android/Todo Orbit/gradlew +0 -248
- package/examples/todo-orbit/generated/android/Todo Orbit/gradlew.bat +0 -93
- package/examples/todo-orbit/generated/android/Todo Orbit/settings.gradle.kts +0 -18
- package/examples/todo-orbit/generated/ios/Todo Orbit/.screenshot-uitest/Sources/ScreenshotUITest.swift +0 -36
- package/examples/todo-orbit/generated/ios/Todo Orbit/README.md +0 -29
- package/examples/todo-orbit/generated/ios/Todo Orbit/Resources/en.lproj/Localizable.strings +0 -119
- package/examples/todo-orbit/generated/ios/Todo Orbit/Resources/ru.lproj/Localizable.strings +0 -119
- package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/App/TodoOrbitApp.swift +0 -50
- package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Components/OrbitChrome.swift +0 -204
- package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Components/SchedulePreviewView.swift +0 -126
- package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Components/TrendChartView.swift +0 -70
- package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Flows/RecurringRuleSheet.swift +0 -126
- package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Flows/TaskEditorSheet.swift +0 -61
- package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Models/DomainModels.swift +0 -238
- package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Screens/AnalyticsView.swift +0 -94
- package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Screens/SettingsView.swift +0 -76
- package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Screens/TasksHomeView.swift +0 -364
- package/examples/todo-orbit/generated/ios/Todo Orbit/Sources/TodoOrbit/Support/AppModel.swift +0 -324
- package/examples/todo-orbit/generated/ios/Todo Orbit/TodoOrbit.xcodeproj/project.pbxproj +0 -439
- package/examples/todo-orbit/generated/ios/Todo Orbit/TodoOrbit.xcodeproj/project.xcworkspace/contents.xcworkspacedata +0 -7
- package/examples/todo-orbit/generated/ios/Todo Orbit/TodoOrbit.xcodeproj/xcshareddata/xcschemes/TodoOrbit.xcscheme +0 -89
- package/examples/todo-orbit/generated/ios/Todo Orbit/project.yml +0 -32
- package/examples/todo-orbit/generated/web/Todo Orbit/index.html +0 -16
- package/examples/todo-orbit/generated/web/Todo Orbit/package-lock.json +0 -1087
- package/examples/todo-orbit/generated/web/Todo Orbit/package.json +0 -24
- package/examples/todo-orbit/generated/web/Todo Orbit/src/App.tsx +0 -2167
- package/examples/todo-orbit/generated/web/Todo Orbit/src/main.tsx +0 -13
- package/examples/todo-orbit/generated/web/Todo Orbit/src/styles.css +0 -926
- package/examples/todo-orbit/generated/web/Todo Orbit/tsconfig.json +0 -19
- package/examples/todo-orbit/generated/web/Todo Orbit/vite.config.ts +0 -6
|
@@ -1,97 +0,0 @@
|
|
|
1
|
-
import { useEffect, useState } from "react";
|
|
2
|
-
import { useSearchParams } from "react-router";
|
|
3
|
-
import { tokens, type SizeClass } from "./tokens";
|
|
4
|
-
|
|
5
|
-
export function cn(...parts: Array<string | false | null | undefined>) {
|
|
6
|
-
return parts.filter(Boolean).join(" ");
|
|
7
|
-
}
|
|
8
|
-
|
|
9
|
-
export function formatNumber(value: number) {
|
|
10
|
-
return new Intl.NumberFormat().format(value);
|
|
11
|
-
}
|
|
12
|
-
|
|
13
|
-
export function formatDate(value: string) {
|
|
14
|
-
return new Intl.DateTimeFormat(undefined, {
|
|
15
|
-
month: "short",
|
|
16
|
-
day: "numeric",
|
|
17
|
-
year: "numeric",
|
|
18
|
-
}).format(new Date(value));
|
|
19
|
-
}
|
|
20
|
-
|
|
21
|
-
export function formatTime(value: string) {
|
|
22
|
-
return new Intl.DateTimeFormat(undefined, {
|
|
23
|
-
hour: "numeric",
|
|
24
|
-
minute: "2-digit",
|
|
25
|
-
}).format(new Date(value));
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
export function formatRelativeDate(value: string) {
|
|
29
|
-
const now = Date.now();
|
|
30
|
-
const then = new Date(value).getTime();
|
|
31
|
-
const diffHours = Math.round((then - now) / (1000 * 60 * 60));
|
|
32
|
-
const absHours = Math.abs(diffHours);
|
|
33
|
-
const relativeFormat = new Intl.RelativeTimeFormat(undefined, { numeric: "auto" });
|
|
34
|
-
if (absHours < 24) {
|
|
35
|
-
return relativeFormat.format(diffHours, "hour");
|
|
36
|
-
}
|
|
37
|
-
const diffDays = Math.round(diffHours / 24);
|
|
38
|
-
return relativeFormat.format(diffDays, "day");
|
|
39
|
-
}
|
|
40
|
-
|
|
41
|
-
export type UiScenario = "normal" | "loading" | "empty" | "error";
|
|
42
|
-
|
|
43
|
-
export function useUiScenario() {
|
|
44
|
-
const [searchParams] = useSearchParams();
|
|
45
|
-
const ui = searchParams.get("ui");
|
|
46
|
-
if (ui === "loading" || ui === "empty" || ui === "error") {
|
|
47
|
-
return ui;
|
|
48
|
-
}
|
|
49
|
-
return "normal" as const;
|
|
50
|
-
}
|
|
51
|
-
|
|
52
|
-
export function useSimulatedLoading(key: string, scenario: UiScenario) {
|
|
53
|
-
const [loading, setLoading] = useState(true);
|
|
54
|
-
|
|
55
|
-
useEffect(() => {
|
|
56
|
-
if (scenario === "loading") {
|
|
57
|
-
setLoading(true);
|
|
58
|
-
return;
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
setLoading(true);
|
|
62
|
-
const timer = window.setTimeout(() => setLoading(false), 320);
|
|
63
|
-
return () => window.clearTimeout(timer);
|
|
64
|
-
}, [key, scenario]);
|
|
65
|
-
|
|
66
|
-
return loading;
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
export function useSizeClass(): SizeClass {
|
|
70
|
-
const [sizeClass, setSizeClass] = useState<SizeClass>(() => getSizeClass(window.innerWidth));
|
|
71
|
-
|
|
72
|
-
useEffect(() => {
|
|
73
|
-
const onResize = () => setSizeClass(getSizeClass(window.innerWidth));
|
|
74
|
-
window.addEventListener("resize", onResize);
|
|
75
|
-
return () => window.removeEventListener("resize", onResize);
|
|
76
|
-
}, []);
|
|
77
|
-
|
|
78
|
-
return sizeClass;
|
|
79
|
-
}
|
|
80
|
-
|
|
81
|
-
function getSizeClass(width: number): SizeClass {
|
|
82
|
-
if (width <= tokens.breakpoints.compactMax) {
|
|
83
|
-
return "compact";
|
|
84
|
-
}
|
|
85
|
-
if (width <= tokens.breakpoints.regularMax) {
|
|
86
|
-
return "regular";
|
|
87
|
-
}
|
|
88
|
-
return "expanded";
|
|
89
|
-
}
|
|
90
|
-
|
|
91
|
-
export function getInitials(name: string) {
|
|
92
|
-
return name
|
|
93
|
-
.split(/\s+/)
|
|
94
|
-
.slice(0, 2)
|
|
95
|
-
.map((part) => part[0]?.toUpperCase() ?? "")
|
|
96
|
-
.join("");
|
|
97
|
-
}
|
|
@@ -1,67 +0,0 @@
|
|
|
1
|
-
{
|
|
2
|
-
"$locale": "en",
|
|
3
|
-
"$direction": "ltr",
|
|
4
|
-
"nav.home": "Home",
|
|
5
|
-
"nav.discover": "Discover",
|
|
6
|
-
"nav.create": "Create",
|
|
7
|
-
"nav.notifications": "Notifications",
|
|
8
|
-
"nav.profile": "Profile",
|
|
9
|
-
"home.filter_all": "All",
|
|
10
|
-
"home.filter_following": "Following",
|
|
11
|
-
"home.filter_popular": "Popular",
|
|
12
|
-
"home.empty_feed": "No posts yet. Follow people to see their posts here.",
|
|
13
|
-
"discover.search_placeholder": "Search posts, people, tags…",
|
|
14
|
-
"discover.trending": "Trending",
|
|
15
|
-
"discover.popular_tags": "Popular Tags",
|
|
16
|
-
"discover.suggested_creators": "Suggested Creators",
|
|
17
|
-
"search.placeholder": "Search…",
|
|
18
|
-
"search.tab_posts": "Posts",
|
|
19
|
-
"search.tab_people": "People",
|
|
20
|
-
"search.tab_tags": "Tags",
|
|
21
|
-
"search.no_results": "No results found.",
|
|
22
|
-
"post.comments_header": "Comments",
|
|
23
|
-
"post.no_comments": "No comments yet. Be the first to comment.",
|
|
24
|
-
"post.comment_placeholder": "Write a comment…",
|
|
25
|
-
"post.comment_sent": "Comment posted.",
|
|
26
|
-
"post.like_action": "Like",
|
|
27
|
-
"profile.edit_button": "Edit Profile",
|
|
28
|
-
"profile.follow_button": "Follow",
|
|
29
|
-
"profile.posts_header": "Posts",
|
|
30
|
-
"profile.no_posts_self": "You haven't posted anything yet.",
|
|
31
|
-
"profile.no_posts_user": "This user hasn't posted anything yet.",
|
|
32
|
-
"edit_profile.avatar": "Profile Photo",
|
|
33
|
-
"edit_profile.display_name": "Display Name",
|
|
34
|
-
"edit_profile.handle": "Username",
|
|
35
|
-
"edit_profile.bio": "Bio",
|
|
36
|
-
"edit_profile.website": "Website",
|
|
37
|
-
"edit_profile.save": "Save Changes",
|
|
38
|
-
"edit_profile.saved": "Profile updated.",
|
|
39
|
-
"notifications.empty": "No notifications yet.",
|
|
40
|
-
"messages.search_placeholder": "Search conversations…",
|
|
41
|
-
"messages.empty_inbox": "No conversations yet.",
|
|
42
|
-
"chat.message_placeholder": "Type a message…",
|
|
43
|
-
"chat.empty_thread": "Start the conversation.",
|
|
44
|
-
"create_post.body_placeholder": "What's on your mind?",
|
|
45
|
-
"create_post.audience": "Audience",
|
|
46
|
-
"create_post.audience_public": "Public",
|
|
47
|
-
"create_post.audience_followers": "Followers Only",
|
|
48
|
-
"create_post.add_image": "Add Image",
|
|
49
|
-
"create_post.take_photo": "Take Photo",
|
|
50
|
-
"create_post.publish": "Publish",
|
|
51
|
-
"create_post.success": "Post published!",
|
|
52
|
-
"settings.appearance": "Appearance",
|
|
53
|
-
"settings.theme": "Theme",
|
|
54
|
-
"settings.theme_system": "System",
|
|
55
|
-
"settings.theme_light": "Light",
|
|
56
|
-
"settings.theme_dark": "Dark",
|
|
57
|
-
"settings.notifications": "Notifications",
|
|
58
|
-
"settings.push_notifications": "Push Notifications",
|
|
59
|
-
"settings.message_previews": "Message Previews",
|
|
60
|
-
"settings.language": "Language",
|
|
61
|
-
"settings.auto_translate": "Auto-translate Posts",
|
|
62
|
-
"settings.account": "Account",
|
|
63
|
-
"settings.edit_profile": "Edit Profile",
|
|
64
|
-
"settings.logout": "Log Out",
|
|
65
|
-
"settings.logout_confirm": "Are you sure you want to log out?",
|
|
66
|
-
"common.cancel": "Cancel"
|
|
67
|
-
}
|
|
@@ -1,67 +0,0 @@
|
|
|
1
|
-
{
|
|
2
|
-
"$locale": "ru",
|
|
3
|
-
"$direction": "ltr",
|
|
4
|
-
"nav.home": "Главная",
|
|
5
|
-
"nav.discover": "Обзор",
|
|
6
|
-
"nav.create": "Создать",
|
|
7
|
-
"nav.notifications": "Уведомления",
|
|
8
|
-
"nav.profile": "Профиль",
|
|
9
|
-
"home.filter_all": "Все",
|
|
10
|
-
"home.filter_following": "Подписки",
|
|
11
|
-
"home.filter_popular": "Популярное",
|
|
12
|
-
"home.empty_feed": "Пока нет записей. Подпишитесь на людей, чтобы видеть их публикации.",
|
|
13
|
-
"discover.search_placeholder": "Поиск записей, людей, тегов…",
|
|
14
|
-
"discover.trending": "В тренде",
|
|
15
|
-
"discover.popular_tags": "Популярные теги",
|
|
16
|
-
"discover.suggested_creators": "Рекомендуемые авторы",
|
|
17
|
-
"search.placeholder": "Поиск…",
|
|
18
|
-
"search.tab_posts": "Записи",
|
|
19
|
-
"search.tab_people": "Люди",
|
|
20
|
-
"search.tab_tags": "Теги",
|
|
21
|
-
"search.no_results": "Ничего не найдено.",
|
|
22
|
-
"post.comments_header": "Комментарии",
|
|
23
|
-
"post.no_comments": "Пока нет комментариев. Будьте первым.",
|
|
24
|
-
"post.comment_placeholder": "Написать комментарий…",
|
|
25
|
-
"post.comment_sent": "Комментарий опубликован.",
|
|
26
|
-
"post.like_action": "Нравится",
|
|
27
|
-
"profile.edit_button": "Редактировать",
|
|
28
|
-
"profile.follow_button": "Подписаться",
|
|
29
|
-
"profile.posts_header": "Записи",
|
|
30
|
-
"profile.no_posts_self": "У вас пока нет записей.",
|
|
31
|
-
"profile.no_posts_user": "У этого пользователя пока нет записей.",
|
|
32
|
-
"edit_profile.avatar": "Фото профиля",
|
|
33
|
-
"edit_profile.display_name": "Имя",
|
|
34
|
-
"edit_profile.handle": "Имя пользователя",
|
|
35
|
-
"edit_profile.bio": "О себе",
|
|
36
|
-
"edit_profile.website": "Сайт",
|
|
37
|
-
"edit_profile.save": "Сохранить",
|
|
38
|
-
"edit_profile.saved": "Профиль обновлён.",
|
|
39
|
-
"notifications.empty": "Пока нет уведомлений.",
|
|
40
|
-
"messages.search_placeholder": "Поиск переписок…",
|
|
41
|
-
"messages.empty_inbox": "Пока нет переписок.",
|
|
42
|
-
"chat.message_placeholder": "Написать сообщение…",
|
|
43
|
-
"chat.empty_thread": "Начните разговор.",
|
|
44
|
-
"create_post.body_placeholder": "Что у вас нового?",
|
|
45
|
-
"create_post.audience": "Аудитория",
|
|
46
|
-
"create_post.audience_public": "Все",
|
|
47
|
-
"create_post.audience_followers": "Только подписчики",
|
|
48
|
-
"create_post.add_image": "Добавить фото",
|
|
49
|
-
"create_post.take_photo": "Сделать фото",
|
|
50
|
-
"create_post.publish": "Опубликовать",
|
|
51
|
-
"create_post.success": "Запись опубликована!",
|
|
52
|
-
"settings.appearance": "Оформление",
|
|
53
|
-
"settings.theme": "Тема",
|
|
54
|
-
"settings.theme_system": "Системная",
|
|
55
|
-
"settings.theme_light": "Светлая",
|
|
56
|
-
"settings.theme_dark": "Тёмная",
|
|
57
|
-
"settings.notifications": "Уведомления",
|
|
58
|
-
"settings.push_notifications": "Push-уведомления",
|
|
59
|
-
"settings.message_previews": "Превью сообщений",
|
|
60
|
-
"settings.language": "Язык",
|
|
61
|
-
"settings.auto_translate": "Автоперевод записей",
|
|
62
|
-
"settings.account": "Аккаунт",
|
|
63
|
-
"settings.edit_profile": "Редактировать профиль",
|
|
64
|
-
"settings.logout": "Выйти",
|
|
65
|
-
"settings.logout_confirm": "Вы уверены, что хотите выйти?",
|
|
66
|
-
"common.cancel": "Отмена"
|
|
67
|
-
}
|
|
@@ -1,67 +0,0 @@
|
|
|
1
|
-
{
|
|
2
|
-
"$locale": "uz",
|
|
3
|
-
"$direction": "ltr",
|
|
4
|
-
"nav.home": "Bosh sahifa",
|
|
5
|
-
"nav.discover": "Kashfiyot",
|
|
6
|
-
"nav.create": "Yaratish",
|
|
7
|
-
"nav.notifications": "Bildirishnomalar",
|
|
8
|
-
"nav.profile": "Profil",
|
|
9
|
-
"home.filter_all": "Hammasi",
|
|
10
|
-
"home.filter_following": "Obunalar",
|
|
11
|
-
"home.filter_popular": "Ommabop",
|
|
12
|
-
"home.empty_feed": "Hozircha postlar yoʻq. Postlarini koʻrish uchun odamlarga obuna boʻling.",
|
|
13
|
-
"discover.search_placeholder": "Postlar, odamlar, teglar qidirish…",
|
|
14
|
-
"discover.trending": "Trendda",
|
|
15
|
-
"discover.popular_tags": "Ommabop teglar",
|
|
16
|
-
"discover.suggested_creators": "Tavsiya etilgan mualliflar",
|
|
17
|
-
"search.placeholder": "Qidirish…",
|
|
18
|
-
"search.tab_posts": "Postlar",
|
|
19
|
-
"search.tab_people": "Odamlar",
|
|
20
|
-
"search.tab_tags": "Teglar",
|
|
21
|
-
"search.no_results": "Hech narsa topilmadi.",
|
|
22
|
-
"post.comments_header": "Izohlar",
|
|
23
|
-
"post.no_comments": "Hozircha izohlar yoʻq. Birinchi boʻling.",
|
|
24
|
-
"post.comment_placeholder": "Izoh yozish…",
|
|
25
|
-
"post.comment_sent": "Izoh joʻnatildi.",
|
|
26
|
-
"post.like_action": "Yoqtirish",
|
|
27
|
-
"profile.edit_button": "Tahrirlash",
|
|
28
|
-
"profile.follow_button": "Obuna boʻlish",
|
|
29
|
-
"profile.posts_header": "Postlar",
|
|
30
|
-
"profile.no_posts_self": "Sizda hozircha postlar yoʻq.",
|
|
31
|
-
"profile.no_posts_user": "Bu foydalanuvchida hozircha postlar yoʻq.",
|
|
32
|
-
"edit_profile.avatar": "Profil rasmi",
|
|
33
|
-
"edit_profile.display_name": "Ism",
|
|
34
|
-
"edit_profile.handle": "Foydalanuvchi nomi",
|
|
35
|
-
"edit_profile.bio": "Bio",
|
|
36
|
-
"edit_profile.website": "Veb-sayt",
|
|
37
|
-
"edit_profile.save": "Saqlash",
|
|
38
|
-
"edit_profile.saved": "Profil yangilandi.",
|
|
39
|
-
"notifications.empty": "Hozircha bildirishnomalar yoʻq.",
|
|
40
|
-
"messages.search_placeholder": "Suhbatlarni qidirish…",
|
|
41
|
-
"messages.empty_inbox": "Hozircha suhbatlar yoʻq.",
|
|
42
|
-
"chat.message_placeholder": "Xabar yozish…",
|
|
43
|
-
"chat.empty_thread": "Suhbatni boshlang.",
|
|
44
|
-
"create_post.body_placeholder": "Nima yangiliklar?",
|
|
45
|
-
"create_post.audience": "Auditoriya",
|
|
46
|
-
"create_post.audience_public": "Hamma",
|
|
47
|
-
"create_post.audience_followers": "Faqat obunachilar",
|
|
48
|
-
"create_post.add_image": "Rasm qoʻshish",
|
|
49
|
-
"create_post.take_photo": "Suratga olish",
|
|
50
|
-
"create_post.publish": "Eʼlon qilish",
|
|
51
|
-
"create_post.success": "Post eʼlon qilindi!",
|
|
52
|
-
"settings.appearance": "Koʻrinish",
|
|
53
|
-
"settings.theme": "Mavzu",
|
|
54
|
-
"settings.theme_system": "Tizim",
|
|
55
|
-
"settings.theme_light": "Yorugʻ",
|
|
56
|
-
"settings.theme_dark": "Qorongʻu",
|
|
57
|
-
"settings.notifications": "Bildirishnomalar",
|
|
58
|
-
"settings.push_notifications": "Push-bildirishnomalar",
|
|
59
|
-
"settings.message_previews": "Xabar koʻrinishi",
|
|
60
|
-
"settings.language": "Til",
|
|
61
|
-
"settings.auto_translate": "Postlarni avtotarjima qilish",
|
|
62
|
-
"settings.account": "Hisob",
|
|
63
|
-
"settings.edit_profile": "Profilni tahrirlash",
|
|
64
|
-
"settings.logout": "Chiqish",
|
|
65
|
-
"settings.logout_confirm": "Chiqishga ishonchingiz komilmi?",
|
|
66
|
-
"common.cancel": "Bekor qilish"
|
|
67
|
-
}
|
|
@@ -1,16 +0,0 @@
|
|
|
1
|
-
import { StrictMode } from "react";
|
|
2
|
-
import { createRoot } from "react-dom/client";
|
|
3
|
-
import { App } from "./app/App";
|
|
4
|
-
import "./styles.css";
|
|
5
|
-
|
|
6
|
-
const root = document.getElementById("root");
|
|
7
|
-
|
|
8
|
-
if (!root) {
|
|
9
|
-
throw new Error("Missing root element");
|
|
10
|
-
}
|
|
11
|
-
|
|
12
|
-
createRoot(root).render(
|
|
13
|
-
<StrictMode>
|
|
14
|
-
<App />
|
|
15
|
-
</StrictMode>,
|
|
16
|
-
);
|
|
@@ -1,90 +0,0 @@
|
|
|
1
|
-
import { useState } from "react";
|
|
2
|
-
import { useParams } from "react-router";
|
|
3
|
-
import { useI18n } from "../i18n";
|
|
4
|
-
import { MessageBubble } from "../components/cards";
|
|
5
|
-
import { EmptyState, ErrorState, ScreenScaffold, SkeletonList, TextField, ActionButton } from "../components/ui";
|
|
6
|
-
import { selectConversationById, selectMessagesByConversation, selectUserById, useAppStore } from "../state/store";
|
|
7
|
-
import { useSimulatedLoading, useUiScenario } from "../lib/utils";
|
|
8
|
-
|
|
9
|
-
export function ChatDetailScreen() {
|
|
10
|
-
const { t } = useI18n();
|
|
11
|
-
const { conversationId = "" } = useParams();
|
|
12
|
-
const scenario = useUiScenario();
|
|
13
|
-
const loading = useSimulatedLoading(`chat-${conversationId}`, scenario);
|
|
14
|
-
const [messageText, setMessageText] = useState("");
|
|
15
|
-
const state = useAppStore();
|
|
16
|
-
const conversation = scenario === "empty" ? undefined : selectConversationById(state, conversationId);
|
|
17
|
-
const messages = scenario === "empty" ? [] : selectMessagesByConversation(state, conversationId);
|
|
18
|
-
|
|
19
|
-
if (scenario === "error") {
|
|
20
|
-
return (
|
|
21
|
-
<ScreenScaffold title="Conversation">
|
|
22
|
-
<ErrorState title="Thread unavailable" description="The thread request failed. Remove `?ui=error` to restore it." />
|
|
23
|
-
</ScreenScaffold>
|
|
24
|
-
);
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
if (loading) {
|
|
28
|
-
return (
|
|
29
|
-
<ScreenScaffold title="Conversation">
|
|
30
|
-
<SkeletonList count={5} />
|
|
31
|
-
</ScreenScaffold>
|
|
32
|
-
);
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
if (!conversation) {
|
|
36
|
-
return (
|
|
37
|
-
<ScreenScaffold title="Conversation">
|
|
38
|
-
<EmptyState title="No thread found" description="This conversation is not available in the local mock store." />
|
|
39
|
-
</ScreenScaffold>
|
|
40
|
-
);
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
return (
|
|
44
|
-
<ScreenScaffold title="Conversation" subtitle="Alternating message bubble geometry and inline send action.">
|
|
45
|
-
{messages.length === 0 ? (
|
|
46
|
-
<EmptyState title="No messages yet" description={t("chat.empty_thread")} />
|
|
47
|
-
) : (
|
|
48
|
-
<div className="space-y-2">
|
|
49
|
-
{messages.map((message) => {
|
|
50
|
-
const author = selectUserById(state, message.senderId);
|
|
51
|
-
if (!author) {
|
|
52
|
-
return null;
|
|
53
|
-
}
|
|
54
|
-
return (
|
|
55
|
-
<MessageBubble
|
|
56
|
-
key={message.id}
|
|
57
|
-
message={message}
|
|
58
|
-
author={author}
|
|
59
|
-
isMine={message.senderId === state.currentUserId}
|
|
60
|
-
/>
|
|
61
|
-
);
|
|
62
|
-
})}
|
|
63
|
-
</div>
|
|
64
|
-
)}
|
|
65
|
-
|
|
66
|
-
<TextField
|
|
67
|
-
label={t("chat.message_placeholder")}
|
|
68
|
-
value={messageText}
|
|
69
|
-
multiline
|
|
70
|
-
onValueChange={setMessageText}
|
|
71
|
-
placeholder={t("chat.message_placeholder")}
|
|
72
|
-
trailingAction={
|
|
73
|
-
<ActionButton
|
|
74
|
-
variant="primary"
|
|
75
|
-
icon="send"
|
|
76
|
-
onClick={() => {
|
|
77
|
-
if (!messageText.trim()) {
|
|
78
|
-
return;
|
|
79
|
-
}
|
|
80
|
-
state.sendMessage(conversationId, messageText.trim());
|
|
81
|
-
setMessageText("");
|
|
82
|
-
}}
|
|
83
|
-
>
|
|
84
|
-
Send
|
|
85
|
-
</ActionButton>
|
|
86
|
-
}
|
|
87
|
-
/>
|
|
88
|
-
</ScreenScaffold>
|
|
89
|
-
);
|
|
90
|
-
}
|
|
@@ -1,86 +0,0 @@
|
|
|
1
|
-
import { useDeferredValue, useState } from "react";
|
|
2
|
-
import { useNavigate } from "react-router";
|
|
3
|
-
import { useI18n } from "../i18n";
|
|
4
|
-
import { CreatorCard, TrendRow } from "../components/cards";
|
|
5
|
-
import { ActionButton, ErrorState, ScreenScaffold, SectionTitle, SkeletonList, TextField } from "../components/ui";
|
|
6
|
-
import { useSimulatedLoading, useUiScenario } from "../lib/utils";
|
|
7
|
-
import { selectDiscoverCreators, useAppStore } from "../state/store";
|
|
8
|
-
|
|
9
|
-
export function DiscoverScreen() {
|
|
10
|
-
const { t } = useI18n();
|
|
11
|
-
const navigate = useNavigate();
|
|
12
|
-
const scenario = useUiScenario();
|
|
13
|
-
const loading = useSimulatedLoading("discover", scenario);
|
|
14
|
-
const [searchQuery, setSearchQuery] = useState("");
|
|
15
|
-
const deferredQuery = useDeferredValue(searchQuery);
|
|
16
|
-
const state = useAppStore();
|
|
17
|
-
const creators = scenario === "empty" ? [] : selectDiscoverCreators(state);
|
|
18
|
-
const trends = scenario === "empty" ? [] : state.trends;
|
|
19
|
-
const tags = scenario === "empty" ? [] : state.tags;
|
|
20
|
-
|
|
21
|
-
return (
|
|
22
|
-
<ScreenScaffold title={t("nav.discover")} subtitle="Search, trending topics, tags, and suggested creators.">
|
|
23
|
-
<TextField
|
|
24
|
-
label={t("discover.search_placeholder")}
|
|
25
|
-
value={searchQuery}
|
|
26
|
-
onValueChange={setSearchQuery}
|
|
27
|
-
placeholder={t("discover.search_placeholder")}
|
|
28
|
-
trailingAction={
|
|
29
|
-
<ActionButton
|
|
30
|
-
variant="primary"
|
|
31
|
-
icon="search"
|
|
32
|
-
onClick={() => navigate(`/search?query=${encodeURIComponent(deferredQuery || searchQuery)}&tab=posts`)}
|
|
33
|
-
>
|
|
34
|
-
Search
|
|
35
|
-
</ActionButton>
|
|
36
|
-
}
|
|
37
|
-
/>
|
|
38
|
-
|
|
39
|
-
{scenario === "error" ? (
|
|
40
|
-
<ErrorState title="Discover unavailable" description="The discover request failed. Remove `?ui=error` to return to the normal dataset." />
|
|
41
|
-
) : loading ? (
|
|
42
|
-
<SkeletonList count={4} tall />
|
|
43
|
-
) : (
|
|
44
|
-
<>
|
|
45
|
-
<section className="space-y-4">
|
|
46
|
-
<SectionTitle>{t("discover.trending")}</SectionTitle>
|
|
47
|
-
<div className="space-y-3">
|
|
48
|
-
{trends.map((trend) => (
|
|
49
|
-
<TrendRow
|
|
50
|
-
key={trend.id}
|
|
51
|
-
title={trend.label}
|
|
52
|
-
subtitle={`${trend.postCount.toLocaleString()} posts`}
|
|
53
|
-
onClick={() => navigate(`/search?query=${encodeURIComponent(trend.label)}&tab=posts`)}
|
|
54
|
-
/>
|
|
55
|
-
))}
|
|
56
|
-
</div>
|
|
57
|
-
</section>
|
|
58
|
-
|
|
59
|
-
<section className="space-y-4">
|
|
60
|
-
<SectionTitle>{t("discover.popular_tags")}</SectionTitle>
|
|
61
|
-
<div className="no-scrollbar flex gap-2 overflow-x-auto pb-1">
|
|
62
|
-
{tags.map((tag) => (
|
|
63
|
-
<ActionButton
|
|
64
|
-
key={tag.id}
|
|
65
|
-
variant="chip"
|
|
66
|
-
onClick={() => navigate(`/search?query=${encodeURIComponent(`#${tag.name}`)}&tab=posts`)}
|
|
67
|
-
>
|
|
68
|
-
#{tag.name}
|
|
69
|
-
</ActionButton>
|
|
70
|
-
))}
|
|
71
|
-
</div>
|
|
72
|
-
</section>
|
|
73
|
-
|
|
74
|
-
<section className="space-y-4">
|
|
75
|
-
<SectionTitle>{t("discover.suggested_creators")}</SectionTitle>
|
|
76
|
-
<div className="no-scrollbar flex snap-x gap-3 overflow-x-auto pb-2">
|
|
77
|
-
{creators.map((creator) => (
|
|
78
|
-
<CreatorCard key={creator.id} user={creator} to={`/u/${creator.id}`} />
|
|
79
|
-
))}
|
|
80
|
-
</div>
|
|
81
|
-
</section>
|
|
82
|
-
</>
|
|
83
|
-
)}
|
|
84
|
-
</ScreenScaffold>
|
|
85
|
-
);
|
|
86
|
-
}
|
|
@@ -1,57 +0,0 @@
|
|
|
1
|
-
import { useEffect, useState } from "react";
|
|
2
|
-
import { useNavigate } from "react-router";
|
|
3
|
-
import { useI18n } from "../i18n";
|
|
4
|
-
import { Avatar } from "../components/ui";
|
|
5
|
-
import { ActionButton, ScreenScaffold, Surface, TextField } from "../components/ui";
|
|
6
|
-
import { selectCurrentUser, useAppStore } from "../state/store";
|
|
7
|
-
|
|
8
|
-
export function EditProfileScreen() {
|
|
9
|
-
const { t } = useI18n();
|
|
10
|
-
const navigate = useNavigate();
|
|
11
|
-
const state = useAppStore();
|
|
12
|
-
const user = selectCurrentUser(state);
|
|
13
|
-
const [displayName, setDisplayName] = useState(user.displayName);
|
|
14
|
-
const [handle, setHandle] = useState(user.handle);
|
|
15
|
-
const [bio, setBio] = useState(user.bio ?? "");
|
|
16
|
-
const [website, setWebsite] = useState(user.website ?? "");
|
|
17
|
-
|
|
18
|
-
useEffect(() => {
|
|
19
|
-
setDisplayName(user.displayName);
|
|
20
|
-
setHandle(user.handle);
|
|
21
|
-
setBio(user.bio ?? "");
|
|
22
|
-
setWebsite(user.website ?? "");
|
|
23
|
-
}, [user]);
|
|
24
|
-
|
|
25
|
-
return (
|
|
26
|
-
<ScreenScaffold title="Edit Profile" subtitle="Profile photo card, text fields, and save toast.">
|
|
27
|
-
<Surface className="p-6">
|
|
28
|
-
<div className="flex items-center gap-4">
|
|
29
|
-
<Avatar src={user.avatarUrl} name={user.displayName} size="lg" />
|
|
30
|
-
<div>
|
|
31
|
-
<p className="text-sm text-[var(--color-text-secondary)]">{t("edit_profile.avatar")}</p>
|
|
32
|
-
<p className="mt-1 font-semibold text-[var(--color-text-primary)]">{user.displayName}</p>
|
|
33
|
-
</div>
|
|
34
|
-
</div>
|
|
35
|
-
</Surface>
|
|
36
|
-
|
|
37
|
-
<div className="space-y-4">
|
|
38
|
-
<TextField label={t("edit_profile.display_name")} value={displayName} onValueChange={setDisplayName} />
|
|
39
|
-
<TextField label={t("edit_profile.handle")} value={handle} onValueChange={setHandle} />
|
|
40
|
-
<TextField label={t("edit_profile.bio")} value={bio} multiline onValueChange={setBio} maxLength={160} />
|
|
41
|
-
<TextField label={t("edit_profile.website")} value={website} onValueChange={setWebsite} type="url" />
|
|
42
|
-
</div>
|
|
43
|
-
|
|
44
|
-
<ActionButton
|
|
45
|
-
variant="primary"
|
|
46
|
-
fullWidth
|
|
47
|
-
onClick={() => {
|
|
48
|
-
state.updateProfile({ displayName, handle, bio, website });
|
|
49
|
-
state.showToast(t("edit_profile.saved"));
|
|
50
|
-
navigate("/profile");
|
|
51
|
-
}}
|
|
52
|
-
>
|
|
53
|
-
{t("edit_profile.save")}
|
|
54
|
-
</ActionButton>
|
|
55
|
-
</ScreenScaffold>
|
|
56
|
-
);
|
|
57
|
-
}
|
|
@@ -1,103 +0,0 @@
|
|
|
1
|
-
import { useMemo, useState, startTransition } from "react";
|
|
2
|
-
import { useNavigate } from "react-router";
|
|
3
|
-
import { useI18n } from "../i18n";
|
|
4
|
-
import { selectFeed, selectStories, selectUserById, useAppStore } from "../state/store";
|
|
5
|
-
import { PostCard, StoryCard } from "../components/cards";
|
|
6
|
-
import { ActionButton, EmptyState, ErrorState, ScreenScaffold, SectionTitle, SkeletonList } from "../components/ui";
|
|
7
|
-
import { useSimulatedLoading, useUiScenario } from "../lib/utils";
|
|
8
|
-
|
|
9
|
-
export function HomeFeedScreen() {
|
|
10
|
-
const { t } = useI18n();
|
|
11
|
-
const navigate = useNavigate();
|
|
12
|
-
const scenario = useUiScenario();
|
|
13
|
-
const [activeFilter, setActiveFilter] = useState<"all" | "following" | "popular">("all");
|
|
14
|
-
const [searchQuery] = useState("");
|
|
15
|
-
const state = useAppStore();
|
|
16
|
-
const stories = selectStories(state);
|
|
17
|
-
const feed = selectFeed(state, activeFilter, searchQuery);
|
|
18
|
-
const loading = useSimulatedLoading(`home-${activeFilter}`, scenario);
|
|
19
|
-
|
|
20
|
-
const visibleFeed = useMemo(() => (scenario === "empty" ? [] : feed), [feed, scenario]);
|
|
21
|
-
|
|
22
|
-
return (
|
|
23
|
-
<ScreenScaffold title={t("nav.home")} subtitle="Stories, chip-based feed filters, and expressive card actions.">
|
|
24
|
-
<section className="space-y-4">
|
|
25
|
-
<SectionTitle>Stories</SectionTitle>
|
|
26
|
-
<div className="no-scrollbar flex snap-x gap-3 overflow-x-auto pb-2">
|
|
27
|
-
{stories.map((story) => (
|
|
28
|
-
<StoryCard
|
|
29
|
-
key={story.id}
|
|
30
|
-
name={story.author?.displayName ?? "Story"}
|
|
31
|
-
image={story.previewUrl}
|
|
32
|
-
to={`/posts/${story.id}`}
|
|
33
|
-
/>
|
|
34
|
-
))}
|
|
35
|
-
</div>
|
|
36
|
-
</section>
|
|
37
|
-
|
|
38
|
-
<section className="space-y-4">
|
|
39
|
-
<SectionTitle>Feed Filter</SectionTitle>
|
|
40
|
-
<div className="no-scrollbar flex gap-2 overflow-x-auto pb-1">
|
|
41
|
-
{[
|
|
42
|
-
{ value: "all" as const, label: t("home.filter_all") },
|
|
43
|
-
{ value: "following" as const, label: t("home.filter_following") },
|
|
44
|
-
{ value: "popular" as const, label: t("home.filter_popular") },
|
|
45
|
-
].map((option) => (
|
|
46
|
-
<ActionButton
|
|
47
|
-
key={option.value}
|
|
48
|
-
variant="chip"
|
|
49
|
-
selected={activeFilter === option.value}
|
|
50
|
-
onClick={() => startTransition(() => setActiveFilter(option.value))}
|
|
51
|
-
>
|
|
52
|
-
{option.label}
|
|
53
|
-
</ActionButton>
|
|
54
|
-
))}
|
|
55
|
-
</div>
|
|
56
|
-
</section>
|
|
57
|
-
|
|
58
|
-
<section className="space-y-4">
|
|
59
|
-
<SectionTitle>Feed</SectionTitle>
|
|
60
|
-
|
|
61
|
-
{scenario === "error" ? (
|
|
62
|
-
<ErrorState title="Feed unavailable" description="The mock feed request failed. Remove `?ui=error` to restore the normal state." />
|
|
63
|
-
) : loading ? (
|
|
64
|
-
<SkeletonList count={5} tall />
|
|
65
|
-
) : visibleFeed.length === 0 ? (
|
|
66
|
-
<EmptyState title="Feed is empty" description={t("home.empty_feed")} />
|
|
67
|
-
) : (
|
|
68
|
-
<div className="space-y-4">
|
|
69
|
-
{visibleFeed.map((post) => {
|
|
70
|
-
const author = selectUserById(state, post.authorId);
|
|
71
|
-
if (!author) {
|
|
72
|
-
return null;
|
|
73
|
-
}
|
|
74
|
-
|
|
75
|
-
return (
|
|
76
|
-
<div key={post.id} className="rounded-card border border-[var(--color-border-default)] bg-[var(--color-surface-secondary)] p-4 shadow-sm">
|
|
77
|
-
<PostCard
|
|
78
|
-
post={post}
|
|
79
|
-
author={author}
|
|
80
|
-
onOpen={() => navigate(`/posts/${post.id}`)}
|
|
81
|
-
onAuthor={() => navigate(`/u/${author.id}`)}
|
|
82
|
-
onLike={() => state.toggleLike(post.id)}
|
|
83
|
-
onSave={() => state.toggleSave(post.id)}
|
|
84
|
-
/>
|
|
85
|
-
</div>
|
|
86
|
-
);
|
|
87
|
-
})}
|
|
88
|
-
</div>
|
|
89
|
-
)}
|
|
90
|
-
</section>
|
|
91
|
-
|
|
92
|
-
<ActionButton
|
|
93
|
-
variant="fab"
|
|
94
|
-
icon="create_post"
|
|
95
|
-
aria-label={t("nav.create")}
|
|
96
|
-
className="fixed bottom-20 right-4 z-30 lg:bottom-8 lg:right-8"
|
|
97
|
-
onClick={() => navigate("/create")}
|
|
98
|
-
>
|
|
99
|
-
{t("nav.create")}
|
|
100
|
-
</ActionButton>
|
|
101
|
-
</ScreenScaffold>
|
|
102
|
-
);
|
|
103
|
-
}
|