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,27 +0,0 @@
|
|
|
1
|
-
{
|
|
2
|
-
"name": "social-app-web",
|
|
3
|
-
"private": true,
|
|
4
|
-
"version": "0.1.0",
|
|
5
|
-
"type": "module",
|
|
6
|
-
"scripts": {
|
|
7
|
-
"dev": "vite",
|
|
8
|
-
"build": "tsc -b && vite build",
|
|
9
|
-
"preview": "vite preview"
|
|
10
|
-
},
|
|
11
|
-
"dependencies": {
|
|
12
|
-
"react": "latest",
|
|
13
|
-
"react-dom": "latest",
|
|
14
|
-
"react-router": "latest",
|
|
15
|
-
"zustand": "latest"
|
|
16
|
-
},
|
|
17
|
-
"devDependencies": {
|
|
18
|
-
"@tailwindcss/vite": "latest",
|
|
19
|
-
"@types/node": "latest",
|
|
20
|
-
"@types/react": "latest",
|
|
21
|
-
"@types/react-dom": "latest",
|
|
22
|
-
"@vitejs/plugin-react": "^5.0.0",
|
|
23
|
-
"tailwindcss": "latest",
|
|
24
|
-
"typescript": "latest",
|
|
25
|
-
"vite": "^7.0.0"
|
|
26
|
-
}
|
|
27
|
-
}
|
|
@@ -1,58 +0,0 @@
|
|
|
1
|
-
import { useEffect } from "react";
|
|
2
|
-
import { BrowserRouter, Navigate, Route, Routes } from "react-router";
|
|
3
|
-
import { I18nProvider } from "../i18n";
|
|
4
|
-
import { AppShell } from "../components/Shell";
|
|
5
|
-
import { ChatDetailScreen } from "../screens/ChatDetailScreen";
|
|
6
|
-
import { DiscoverScreen } from "../screens/DiscoverScreen";
|
|
7
|
-
import { EditProfileScreen } from "../screens/EditProfileScreen";
|
|
8
|
-
import { HomeFeedScreen } from "../screens/HomeFeedScreen";
|
|
9
|
-
import { MessagesInboxScreen } from "../screens/MessagesInboxScreen";
|
|
10
|
-
import { NotificationsScreen } from "../screens/NotificationsScreen";
|
|
11
|
-
import { PostDetailScreen } from "../screens/PostDetailScreen";
|
|
12
|
-
import { ProfileSelfScreen } from "../screens/ProfileSelfScreen";
|
|
13
|
-
import { ProfileUserScreen } from "../screens/ProfileUserScreen";
|
|
14
|
-
import { SearchResultsScreen } from "../screens/SearchResultsScreen";
|
|
15
|
-
import { SettingsScreen } from "../screens/SettingsScreen";
|
|
16
|
-
import { CreatePostFlow } from "../flows/CreatePostFlow";
|
|
17
|
-
import { useAppStore } from "../state/store";
|
|
18
|
-
|
|
19
|
-
function ThemeBridge() {
|
|
20
|
-
const theme = useAppStore((state) => state.preferences.theme);
|
|
21
|
-
useEffect(() => {
|
|
22
|
-
const resolvedTheme =
|
|
23
|
-
theme === "system"
|
|
24
|
-
? window.matchMedia("(prefers-color-scheme: dark)").matches
|
|
25
|
-
? "dark"
|
|
26
|
-
: "light"
|
|
27
|
-
: theme;
|
|
28
|
-
document.documentElement.dataset.theme = resolvedTheme;
|
|
29
|
-
}, [theme]);
|
|
30
|
-
return null;
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
export function App() {
|
|
34
|
-
return (
|
|
35
|
-
<I18nProvider>
|
|
36
|
-
<BrowserRouter>
|
|
37
|
-
<ThemeBridge />
|
|
38
|
-
<Routes>
|
|
39
|
-
<Route element={<AppShell />}>
|
|
40
|
-
<Route index element={<Navigate to="/home" replace />} />
|
|
41
|
-
<Route path="/home" element={<HomeFeedScreen />} />
|
|
42
|
-
<Route path="/discover" element={<DiscoverScreen />} />
|
|
43
|
-
<Route path="/notifications" element={<NotificationsScreen />} />
|
|
44
|
-
<Route path="/messages" element={<MessagesInboxScreen />} />
|
|
45
|
-
<Route path="/profile" element={<ProfileSelfScreen />} />
|
|
46
|
-
<Route path="/profile/edit" element={<EditProfileScreen />} />
|
|
47
|
-
<Route path="/search" element={<SearchResultsScreen />} />
|
|
48
|
-
<Route path="/posts/:postId" element={<PostDetailScreen />} />
|
|
49
|
-
<Route path="/u/:userId" element={<ProfileUserScreen />} />
|
|
50
|
-
<Route path="/chat/:conversationId" element={<ChatDetailScreen />} />
|
|
51
|
-
<Route path="/settings" element={<SettingsScreen />} />
|
|
52
|
-
<Route path="/create" element={<CreatePostFlow />} />
|
|
53
|
-
</Route>
|
|
54
|
-
</Routes>
|
|
55
|
-
</BrowserRouter>
|
|
56
|
-
</I18nProvider>
|
|
57
|
-
);
|
|
58
|
-
}
|
|
@@ -1,259 +0,0 @@
|
|
|
1
|
-
import { type PropsWithChildren, useEffect, useMemo } from "react";
|
|
2
|
-
import { Link, NavLink, Outlet, useLocation, useNavigate } from "react-router";
|
|
3
|
-
import { useI18n } from "../i18n";
|
|
4
|
-
import { Icon } from "../lib/icons";
|
|
5
|
-
import type { SizeClass } from "../lib/tokens";
|
|
6
|
-
import { cn, useSizeClass } from "../lib/utils";
|
|
7
|
-
import { selectUnreadNotifications, useAppStore } from "../state/store";
|
|
8
|
-
import { ActionButton, ActionGroup } from "./ui";
|
|
9
|
-
|
|
10
|
-
const primaryRoutes = [
|
|
11
|
-
{ to: "/home", icon: "home" as const, labelKey: "nav.home" },
|
|
12
|
-
{ to: "/discover", icon: "discover" as const, labelKey: "nav.discover" },
|
|
13
|
-
{ to: "/notifications", icon: "notifications" as const, labelKey: "nav.notifications" },
|
|
14
|
-
{ to: "/profile", icon: "profile" as const, labelKey: "nav.profile" },
|
|
15
|
-
];
|
|
16
|
-
|
|
17
|
-
export function AppShell() {
|
|
18
|
-
const sizeClass = useSizeClass();
|
|
19
|
-
const location = useLocation();
|
|
20
|
-
const navigate = useNavigate();
|
|
21
|
-
const { t, locale, setLocale } = useI18n();
|
|
22
|
-
const toast = useAppStore((state) => state.toast);
|
|
23
|
-
const clearToast = useAppStore((state) => state.clearToast);
|
|
24
|
-
const dialog = useAppStore((state) => state.dialog);
|
|
25
|
-
const closeDialog = useAppStore((state) => state.closeDialog);
|
|
26
|
-
const unreadCount = useAppStore(selectUnreadNotifications);
|
|
27
|
-
|
|
28
|
-
const pageMeta = useMemo(() => resolvePageMeta(location.pathname, t), [location.pathname, t]);
|
|
29
|
-
|
|
30
|
-
useEffect(() => {
|
|
31
|
-
if (!toast) {
|
|
32
|
-
return;
|
|
33
|
-
}
|
|
34
|
-
const timer = window.setTimeout(() => clearToast(), 3000);
|
|
35
|
-
return () => window.clearTimeout(timer);
|
|
36
|
-
}, [toast, clearToast]);
|
|
37
|
-
|
|
38
|
-
return (
|
|
39
|
-
<div className="min-h-screen bg-[var(--color-surface-primary)] text-[var(--color-text-primary)]">
|
|
40
|
-
<div className="pointer-events-none fixed inset-0 -z-10 overflow-hidden">
|
|
41
|
-
<div className="absolute left-[-8rem] top-[-6rem] h-72 w-72 rounded-full bg-[color:rgba(91,82,163,0.12)] blur-3xl" />
|
|
42
|
-
<div className="absolute bottom-[-8rem] right-[-4rem] h-80 w-80 rounded-full bg-[color:rgba(212,146,14,0.12)] blur-3xl" />
|
|
43
|
-
</div>
|
|
44
|
-
|
|
45
|
-
{sizeClass === "expanded" ? <DesktopSidebar unreadCount={unreadCount} /> : null}
|
|
46
|
-
|
|
47
|
-
<div className={cn("min-h-screen", sizeClass === "expanded" && "pl-[304px]")}>
|
|
48
|
-
<header className="sticky top-0 z-30 border-b border-[var(--color-border-default)] bg-[color:rgba(250,248,245,0.88)] backdrop-blur-xl">
|
|
49
|
-
<div className="mx-auto flex max-w-[1180px] items-center gap-3 px-4 py-4 md:px-6 xl:px-8">
|
|
50
|
-
{pageMeta.showBack ? (
|
|
51
|
-
<button
|
|
52
|
-
type="button"
|
|
53
|
-
onClick={() => navigate(-1)}
|
|
54
|
-
className="interactive-press rounded-cap-alternate border border-[var(--color-border-default)] p-2 text-[var(--color-text-primary)]"
|
|
55
|
-
aria-label="Back"
|
|
56
|
-
>
|
|
57
|
-
<Icon name="back" className="h-5 w-5" />
|
|
58
|
-
</button>
|
|
59
|
-
) : null}
|
|
60
|
-
<div className="min-w-0 flex-1">
|
|
61
|
-
<p className="truncate text-base font-semibold">{pageMeta.title}</p>
|
|
62
|
-
{pageMeta.subtitle ? (
|
|
63
|
-
<p className="truncate text-sm text-[var(--color-text-secondary)]">{pageMeta.subtitle}</p>
|
|
64
|
-
) : null}
|
|
65
|
-
</div>
|
|
66
|
-
<label className="hidden items-center gap-2 rounded-cap-primary border border-[var(--color-border-default)] bg-[var(--color-surface-secondary)] px-3 py-2 text-sm text-[var(--color-text-secondary)] sm:flex">
|
|
67
|
-
<span>Lang</span>
|
|
68
|
-
<select
|
|
69
|
-
value={locale}
|
|
70
|
-
onChange={(event) => setLocale(event.target.value as typeof locale)}
|
|
71
|
-
className="bg-transparent font-medium text-[var(--color-text-primary)] outline-none"
|
|
72
|
-
>
|
|
73
|
-
<option value="en">EN</option>
|
|
74
|
-
<option value="ru">RU</option>
|
|
75
|
-
<option value="uz">UZ</option>
|
|
76
|
-
</select>
|
|
77
|
-
</label>
|
|
78
|
-
<Link
|
|
79
|
-
to="/settings"
|
|
80
|
-
className="interactive-press rounded-cap-primary border border-[var(--color-border-default)] bg-[var(--color-surface-secondary)] p-2 text-[var(--color-text-primary)]"
|
|
81
|
-
aria-label="Settings"
|
|
82
|
-
>
|
|
83
|
-
<Icon name="settings" className="h-5 w-5" />
|
|
84
|
-
</Link>
|
|
85
|
-
</div>
|
|
86
|
-
</header>
|
|
87
|
-
|
|
88
|
-
<main>
|
|
89
|
-
<Outlet />
|
|
90
|
-
</main>
|
|
91
|
-
|
|
92
|
-
{sizeClass !== "expanded" ? <BottomTabBar unreadCount={unreadCount} /> : null}
|
|
93
|
-
|
|
94
|
-
</div>
|
|
95
|
-
|
|
96
|
-
{toast ? (
|
|
97
|
-
<div className="pointer-events-none fixed inset-x-0 top-4 z-50 flex justify-center px-4">
|
|
98
|
-
<div className="rounded-cap-primary bg-[var(--color-brand-primary)] px-5 py-3 text-sm font-medium text-[var(--color-brand-primary-on)] shadow-md">
|
|
99
|
-
{toast.message}
|
|
100
|
-
</div>
|
|
101
|
-
</div>
|
|
102
|
-
) : null}
|
|
103
|
-
|
|
104
|
-
{dialog ? (
|
|
105
|
-
<div className="fixed inset-0 z-50 flex items-center justify-center bg-[rgba(28,27,26,0.35)] px-4">
|
|
106
|
-
<div className="w-full max-w-md rounded-surface border border-[var(--color-border-default)] bg-[var(--color-surface-primary)] p-6 shadow-lg">
|
|
107
|
-
<h2 className="text-xl font-semibold">{dialog.title}</h2>
|
|
108
|
-
<p className="mt-2 text-sm leading-6 text-[var(--color-text-secondary)]">{dialog.message}</p>
|
|
109
|
-
<ActionGroup className="mt-6 sm:flex-row sm:items-center sm:justify-end">
|
|
110
|
-
{dialog.actions.map((action) => (
|
|
111
|
-
<ActionButton
|
|
112
|
-
key={action.label}
|
|
113
|
-
variant={action.variant ?? "secondary"}
|
|
114
|
-
onClick={() => {
|
|
115
|
-
action.onPress();
|
|
116
|
-
closeDialog();
|
|
117
|
-
}}
|
|
118
|
-
>
|
|
119
|
-
{action.label}
|
|
120
|
-
</ActionButton>
|
|
121
|
-
))}
|
|
122
|
-
</ActionGroup>
|
|
123
|
-
</div>
|
|
124
|
-
</div>
|
|
125
|
-
) : null}
|
|
126
|
-
</div>
|
|
127
|
-
);
|
|
128
|
-
}
|
|
129
|
-
|
|
130
|
-
function BottomTabBar({ unreadCount }: { unreadCount: number }) {
|
|
131
|
-
const { t } = useI18n();
|
|
132
|
-
return (
|
|
133
|
-
<nav className="fixed inset-x-0 bottom-0 z-30 border-t border-[var(--color-border-default)] bg-[color:rgba(250,248,245,0.94)] px-2 py-2 backdrop-blur-xl">
|
|
134
|
-
<div className="mx-auto grid max-w-xl grid-cols-4 gap-1">
|
|
135
|
-
{primaryRoutes.map((route) => (
|
|
136
|
-
<NavLink
|
|
137
|
-
key={route.to}
|
|
138
|
-
to={route.to}
|
|
139
|
-
className={({ isActive }) =>
|
|
140
|
-
cn(
|
|
141
|
-
"relative flex min-h-14 flex-col items-center justify-center gap-1 rounded-cap-primary px-2 text-[11px] font-medium transition",
|
|
142
|
-
isActive
|
|
143
|
-
? "bg-[var(--color-surface-secondary)] text-[var(--color-brand-primary)]"
|
|
144
|
-
: "text-[var(--color-text-tertiary)]",
|
|
145
|
-
)
|
|
146
|
-
}
|
|
147
|
-
>
|
|
148
|
-
<Icon name={route.icon} className="h-5 w-5" />
|
|
149
|
-
<span>{t(route.labelKey)}</span>
|
|
150
|
-
{route.to === "/notifications" && unreadCount > 0 ? (
|
|
151
|
-
<span className="absolute right-3 top-2 inline-flex min-h-5 min-w-5 items-center justify-center rounded-full bg-[var(--color-brand-accent)] px-1.5 text-[10px] font-semibold text-[var(--color-brand-accent-on)]">
|
|
152
|
-
{unreadCount}
|
|
153
|
-
</span>
|
|
154
|
-
) : null}
|
|
155
|
-
</NavLink>
|
|
156
|
-
))}
|
|
157
|
-
</div>
|
|
158
|
-
</nav>
|
|
159
|
-
);
|
|
160
|
-
}
|
|
161
|
-
|
|
162
|
-
function DesktopSidebar({ unreadCount }: { unreadCount: number }) {
|
|
163
|
-
const { t, locale, setLocale } = useI18n();
|
|
164
|
-
return (
|
|
165
|
-
<aside className="fixed inset-y-0 left-0 z-20 flex w-[280px] flex-col border-r border-[var(--color-border-default)] bg-[var(--color-surface-secondary)] px-5 py-6">
|
|
166
|
-
<div className="mb-10">
|
|
167
|
-
<p className="text-xs uppercase tracking-[0.28em] text-[var(--color-text-tertiary)]">social-app</p>
|
|
168
|
-
<h1 className="mt-2 text-3xl font-semibold leading-tight">Editorial social, reworked for the browser.</h1>
|
|
169
|
-
</div>
|
|
170
|
-
|
|
171
|
-
<nav className="flex flex-1 flex-col gap-2">
|
|
172
|
-
{primaryRoutes.map((route) => (
|
|
173
|
-
<NavLink
|
|
174
|
-
key={route.to}
|
|
175
|
-
to={route.to}
|
|
176
|
-
className={({ isActive }) =>
|
|
177
|
-
cn(
|
|
178
|
-
"relative rounded-cap-primary px-4 py-3 text-sm font-semibold transition",
|
|
179
|
-
isActive
|
|
180
|
-
? "bg-[var(--color-brand-primary)] shadow-sm"
|
|
181
|
-
: "hover:bg-[var(--color-surface-tertiary)]",
|
|
182
|
-
)
|
|
183
|
-
}
|
|
184
|
-
>
|
|
185
|
-
{({ isActive }) => (
|
|
186
|
-
<div className="flex items-center gap-3">
|
|
187
|
-
<Icon
|
|
188
|
-
name={route.icon}
|
|
189
|
-
className={cn(
|
|
190
|
-
"h-5 w-5 shrink-0",
|
|
191
|
-
isActive ? "text-[var(--color-brand-primary-on)]" : "text-[var(--color-text-secondary)]",
|
|
192
|
-
)}
|
|
193
|
-
/>
|
|
194
|
-
<span className={cn(isActive ? "text-[var(--color-brand-primary-on)]" : "text-[var(--color-text-secondary)]")}>
|
|
195
|
-
{t(route.labelKey)}
|
|
196
|
-
</span>
|
|
197
|
-
{route.to === "/notifications" && unreadCount > 0 ? (
|
|
198
|
-
<span className="ml-auto rounded-full bg-[var(--color-brand-accent)] px-2 py-1 text-[10px] font-semibold text-[var(--color-brand-accent-on)]">
|
|
199
|
-
{unreadCount}
|
|
200
|
-
</span>
|
|
201
|
-
) : null}
|
|
202
|
-
</div>
|
|
203
|
-
)}
|
|
204
|
-
</NavLink>
|
|
205
|
-
))}
|
|
206
|
-
</nav>
|
|
207
|
-
|
|
208
|
-
<div className="mt-6 rounded-surface border border-[var(--color-border-default)] bg-[var(--color-surface-primary)] p-4">
|
|
209
|
-
<p className="text-sm font-semibold">Locale</p>
|
|
210
|
-
<select
|
|
211
|
-
value={locale}
|
|
212
|
-
onChange={(event) => setLocale(event.target.value as typeof locale)}
|
|
213
|
-
className="mt-3 w-full rounded-cap-primary border border-[var(--color-border-default)] bg-[var(--color-surface-primary)] px-3 py-3 text-sm outline-none"
|
|
214
|
-
>
|
|
215
|
-
<option value="en">English</option>
|
|
216
|
-
<option value="ru">Русский</option>
|
|
217
|
-
<option value="uz">Oʻzbekcha</option>
|
|
218
|
-
</select>
|
|
219
|
-
</div>
|
|
220
|
-
</aside>
|
|
221
|
-
);
|
|
222
|
-
}
|
|
223
|
-
|
|
224
|
-
function resolvePageMeta(pathname: string, t: (key: string) => string) {
|
|
225
|
-
if (pathname.startsWith("/discover")) {
|
|
226
|
-
return { title: t("nav.discover"), subtitle: "Search, trends, and people worth following.", showBack: false };
|
|
227
|
-
}
|
|
228
|
-
if (pathname.startsWith("/notifications")) {
|
|
229
|
-
return { title: t("nav.notifications"), subtitle: "Your latest activity and responses.", showBack: false };
|
|
230
|
-
}
|
|
231
|
-
if (pathname.startsWith("/messages")) {
|
|
232
|
-
return { title: "Messages", subtitle: "Direct conversations with creators and collaborators.", showBack: false };
|
|
233
|
-
}
|
|
234
|
-
if (pathname.startsWith("/profile/edit")) {
|
|
235
|
-
return { title: "Edit Profile", showBack: true };
|
|
236
|
-
}
|
|
237
|
-
if (pathname.startsWith("/profile")) {
|
|
238
|
-
return { title: t("nav.profile"), subtitle: "Your public identity, posts, and settings surface.", showBack: false };
|
|
239
|
-
}
|
|
240
|
-
if (pathname.startsWith("/search")) {
|
|
241
|
-
return { title: "Search Results", showBack: true };
|
|
242
|
-
}
|
|
243
|
-
if (pathname.startsWith("/posts/")) {
|
|
244
|
-
return { title: "Post Detail", showBack: true };
|
|
245
|
-
}
|
|
246
|
-
if (pathname.startsWith("/u/")) {
|
|
247
|
-
return { title: "Profile", showBack: true };
|
|
248
|
-
}
|
|
249
|
-
if (pathname.startsWith("/chat/")) {
|
|
250
|
-
return { title: "Conversation", showBack: true };
|
|
251
|
-
}
|
|
252
|
-
if (pathname.startsWith("/settings")) {
|
|
253
|
-
return { title: "Settings", subtitle: "Appearance, notifications, and account actions.", showBack: true };
|
|
254
|
-
}
|
|
255
|
-
if (pathname.startsWith("/create")) {
|
|
256
|
-
return { title: "Create Post", subtitle: "Compose something with the same tokenized system as the feed.", showBack: true };
|
|
257
|
-
}
|
|
258
|
-
return { title: t("nav.home"), subtitle: "Stories, filters, and conversations in one warm feed.", showBack: false };
|
|
259
|
-
}
|
|
@@ -1,317 +0,0 @@
|
|
|
1
|
-
import { Link } from "react-router";
|
|
2
|
-
import { Icon } from "../lib/icons";
|
|
3
|
-
import { formatDate, formatNumber, formatRelativeDate, formatTime } from "../lib/utils";
|
|
4
|
-
import type {
|
|
5
|
-
Comment,
|
|
6
|
-
Message,
|
|
7
|
-
NotificationItem,
|
|
8
|
-
Post,
|
|
9
|
-
User,
|
|
10
|
-
} from "../state/store";
|
|
11
|
-
import { ActionButton, Avatar, Surface } from "./ui";
|
|
12
|
-
|
|
13
|
-
export function StoryCard({
|
|
14
|
-
name,
|
|
15
|
-
image,
|
|
16
|
-
to,
|
|
17
|
-
}: {
|
|
18
|
-
name: string;
|
|
19
|
-
image?: string;
|
|
20
|
-
to: string;
|
|
21
|
-
}) {
|
|
22
|
-
return (
|
|
23
|
-
<Link to={to} className="group snap-start">
|
|
24
|
-
<div className="w-28">
|
|
25
|
-
<div className="rounded-surface relative aspect-[4/5] overflow-hidden border border-[var(--color-border-default)] bg-[var(--color-surface-tertiary)] shadow-sm">
|
|
26
|
-
{image ? <img src={image} alt={name} className="h-full w-full object-cover transition duration-300 group-hover:scale-105" /> : null}
|
|
27
|
-
<div className="absolute inset-0 bg-linear-to-t from-[rgba(28,27,26,0.85)] via-transparent to-transparent" />
|
|
28
|
-
<span className="absolute bottom-3 left-3 right-3 text-sm font-semibold text-white">{name}</span>
|
|
29
|
-
</div>
|
|
30
|
-
</div>
|
|
31
|
-
</Link>
|
|
32
|
-
);
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
export function CreatorCard({
|
|
36
|
-
user,
|
|
37
|
-
to,
|
|
38
|
-
}: {
|
|
39
|
-
user: User;
|
|
40
|
-
to: string;
|
|
41
|
-
}) {
|
|
42
|
-
return (
|
|
43
|
-
<Link to={to} className="block w-72 shrink-0 snap-start">
|
|
44
|
-
<Surface className="h-full p-4">
|
|
45
|
-
<div className="flex items-start gap-3">
|
|
46
|
-
<Avatar src={user.avatarUrl} name={user.displayName} size="lg" />
|
|
47
|
-
<div className="min-w-0">
|
|
48
|
-
<p className="truncate text-base font-semibold text-[var(--color-text-primary)]">{user.displayName}</p>
|
|
49
|
-
<p className="truncate text-sm text-[var(--color-text-secondary)]">@{user.handle}</p>
|
|
50
|
-
</div>
|
|
51
|
-
</div>
|
|
52
|
-
<p className="mt-4 line-clamp-3 text-sm text-[var(--color-text-secondary)]">{user.bio}</p>
|
|
53
|
-
</Surface>
|
|
54
|
-
</Link>
|
|
55
|
-
);
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
export function TrendRow({
|
|
59
|
-
title,
|
|
60
|
-
subtitle,
|
|
61
|
-
onClick,
|
|
62
|
-
}: {
|
|
63
|
-
title: string;
|
|
64
|
-
subtitle: string;
|
|
65
|
-
onClick: () => void;
|
|
66
|
-
}) {
|
|
67
|
-
return (
|
|
68
|
-
<button
|
|
69
|
-
type="button"
|
|
70
|
-
onClick={onClick}
|
|
71
|
-
className="flex w-full items-center justify-between gap-4 rounded-card border border-[var(--color-border-default)] bg-[var(--color-surface-secondary)] px-4 py-4 text-left shadow-sm transition hover:-translate-y-0.5"
|
|
72
|
-
>
|
|
73
|
-
<div>
|
|
74
|
-
<p className="font-semibold text-[var(--color-text-primary)]">{title}</p>
|
|
75
|
-
<p className="text-sm text-[var(--color-text-secondary)]">{subtitle}</p>
|
|
76
|
-
</div>
|
|
77
|
-
<Icon name="discover" className="h-5 w-5 text-[var(--color-text-tertiary)]" />
|
|
78
|
-
</button>
|
|
79
|
-
);
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
export function PostCard({
|
|
83
|
-
post,
|
|
84
|
-
author,
|
|
85
|
-
hero,
|
|
86
|
-
onOpen,
|
|
87
|
-
onAuthor,
|
|
88
|
-
onLike,
|
|
89
|
-
onSave,
|
|
90
|
-
}: {
|
|
91
|
-
post: Post;
|
|
92
|
-
author: User;
|
|
93
|
-
hero?: boolean;
|
|
94
|
-
onOpen?: () => void;
|
|
95
|
-
onAuthor?: () => void;
|
|
96
|
-
onLike?: () => void;
|
|
97
|
-
onSave?: () => void;
|
|
98
|
-
}) {
|
|
99
|
-
return (
|
|
100
|
-
<article className={hero ? "rounded-surface border border-[var(--color-border-default)] bg-[var(--color-surface-secondary)] p-5 shadow-sm" : ""}>
|
|
101
|
-
<div className="flex items-center gap-3">
|
|
102
|
-
<button type="button" onClick={onAuthor} className="shrink-0">
|
|
103
|
-
<Avatar src={author.avatarUrl} name={author.displayName} />
|
|
104
|
-
</button>
|
|
105
|
-
<div className="min-w-0">
|
|
106
|
-
<p className={hero ? "truncate text-2xl font-semibold text-[var(--color-text-primary)]" : "truncate font-semibold text-[var(--color-text-primary)]"}>
|
|
107
|
-
{author.displayName}
|
|
108
|
-
</p>
|
|
109
|
-
<p className="truncate text-sm text-[var(--color-text-secondary)]">@{author.handle}</p>
|
|
110
|
-
</div>
|
|
111
|
-
<div className="ml-auto text-right text-xs text-[var(--color-text-tertiary)]">
|
|
112
|
-
{hero ? formatDate(post.publishedAt) : formatRelativeDate(post.publishedAt)}
|
|
113
|
-
</div>
|
|
114
|
-
</div>
|
|
115
|
-
<button type="button" onClick={onOpen} className="mt-4 block w-full text-left">
|
|
116
|
-
<p className="mt-4 whitespace-pre-wrap text-[15px] leading-7 text-[var(--color-text-primary)]">{post.body}</p>
|
|
117
|
-
{post.mediaUrl ? (
|
|
118
|
-
<div className="mt-4 overflow-hidden rounded-surface border border-[var(--color-border-default)]">
|
|
119
|
-
<img src={post.mediaUrl} alt="" className={hero ? "aspect-[4/3] w-full object-cover" : "aspect-[16/10] w-full object-cover"} />
|
|
120
|
-
</div>
|
|
121
|
-
) : null}
|
|
122
|
-
</button>
|
|
123
|
-
<div className="mt-4 flex flex-wrap items-center gap-2 text-sm text-[var(--color-text-secondary)]">
|
|
124
|
-
<span>{formatNumber(post.likeCount)} likes</span>
|
|
125
|
-
<span className="text-[var(--color-border-strong)]">•</span>
|
|
126
|
-
<span>{formatNumber(post.commentCount)} comments</span>
|
|
127
|
-
</div>
|
|
128
|
-
<div className="mt-4 flex flex-wrap gap-2">
|
|
129
|
-
<ActionButton variant="secondary" icon={post.liked ? "like_fill" : "like"} onClick={onLike}>
|
|
130
|
-
Like
|
|
131
|
-
</ActionButton>
|
|
132
|
-
<ActionButton variant="chip" icon={post.saved ? "bookmark_fill" : "bookmark"} selected={post.saved} onClick={onSave}>
|
|
133
|
-
Save
|
|
134
|
-
</ActionButton>
|
|
135
|
-
</div>
|
|
136
|
-
</article>
|
|
137
|
-
);
|
|
138
|
-
}
|
|
139
|
-
|
|
140
|
-
export function CompactRow({
|
|
141
|
-
user,
|
|
142
|
-
title,
|
|
143
|
-
subtitle,
|
|
144
|
-
metadata,
|
|
145
|
-
}: {
|
|
146
|
-
user: User;
|
|
147
|
-
title: string;
|
|
148
|
-
subtitle: string;
|
|
149
|
-
metadata?: string;
|
|
150
|
-
}) {
|
|
151
|
-
return (
|
|
152
|
-
<div className="flex items-center gap-3 border-b border-[var(--color-border-default)] px-1 py-3 last:border-b-0">
|
|
153
|
-
<Avatar src={user.avatarUrl} name={user.displayName} size="sm" />
|
|
154
|
-
<div className="min-w-0 flex-1">
|
|
155
|
-
<p className="truncate text-sm font-semibold text-[var(--color-text-primary)]">{title}</p>
|
|
156
|
-
<p className="truncate text-sm text-[var(--color-text-secondary)]">{subtitle}</p>
|
|
157
|
-
</div>
|
|
158
|
-
{metadata ? <p className="text-xs text-[var(--color-text-tertiary)]">{metadata}</p> : null}
|
|
159
|
-
</div>
|
|
160
|
-
);
|
|
161
|
-
}
|
|
162
|
-
|
|
163
|
-
export function CommentCard({
|
|
164
|
-
comment,
|
|
165
|
-
author,
|
|
166
|
-
onAuthor,
|
|
167
|
-
}: {
|
|
168
|
-
comment: Comment;
|
|
169
|
-
author: User;
|
|
170
|
-
onAuthor: () => void;
|
|
171
|
-
}) {
|
|
172
|
-
return (
|
|
173
|
-
<Surface className="p-4">
|
|
174
|
-
<div className="flex items-start gap-3">
|
|
175
|
-
<button type="button" onClick={onAuthor}>
|
|
176
|
-
<Avatar src={author.avatarUrl} name={author.displayName} size="sm" />
|
|
177
|
-
</button>
|
|
178
|
-
<div className="min-w-0 flex-1">
|
|
179
|
-
<div className="flex items-baseline justify-between gap-3">
|
|
180
|
-
<p className="truncate text-sm font-semibold text-[var(--color-text-primary)]">{author.displayName}</p>
|
|
181
|
-
<p className="text-xs text-[var(--color-text-tertiary)]">{formatRelativeDate(comment.createdAt)}</p>
|
|
182
|
-
</div>
|
|
183
|
-
<p className="mt-2 text-sm leading-6 text-[var(--color-text-secondary)]">{comment.body}</p>
|
|
184
|
-
</div>
|
|
185
|
-
</div>
|
|
186
|
-
</Surface>
|
|
187
|
-
);
|
|
188
|
-
}
|
|
189
|
-
|
|
190
|
-
export function ProfileHero({
|
|
191
|
-
user,
|
|
192
|
-
action,
|
|
193
|
-
}: {
|
|
194
|
-
user: User;
|
|
195
|
-
action?: React.ReactNode;
|
|
196
|
-
}) {
|
|
197
|
-
return (
|
|
198
|
-
<Surface className="overflow-hidden p-6">
|
|
199
|
-
<div className="grid gap-4 md:grid-cols-[auto_1fr_auto] md:items-start">
|
|
200
|
-
<Avatar src={user.avatarUrl} name={user.displayName} size="lg" />
|
|
201
|
-
<div className="min-w-0">
|
|
202
|
-
<h2 className="text-2xl font-semibold text-[var(--color-text-primary)]">{user.displayName}</h2>
|
|
203
|
-
<p className="mt-1 text-sm text-[var(--color-text-secondary)]">@{user.handle}</p>
|
|
204
|
-
<p className="mt-4 max-w-2xl text-sm leading-6 text-[var(--color-text-secondary)]">{user.bio}</p>
|
|
205
|
-
{user.website ? (
|
|
206
|
-
<a href={user.website} className="mt-3 inline-flex text-sm font-medium text-[var(--color-brand-accent)]">
|
|
207
|
-
{user.website.replace(/^https?:\/\//, "")}
|
|
208
|
-
</a>
|
|
209
|
-
) : null}
|
|
210
|
-
</div>
|
|
211
|
-
<div className="flex flex-col gap-3 md:items-end">
|
|
212
|
-
{action}
|
|
213
|
-
<div className="rounded-cap-primary bg-[var(--color-surface-tertiary)] px-4 py-3 text-sm text-[var(--color-text-secondary)]">
|
|
214
|
-
<strong className="mr-1 text-[var(--color-text-primary)]">{formatNumber(user.followers)}</strong>
|
|
215
|
-
followers
|
|
216
|
-
</div>
|
|
217
|
-
</div>
|
|
218
|
-
</div>
|
|
219
|
-
</Surface>
|
|
220
|
-
);
|
|
221
|
-
}
|
|
222
|
-
|
|
223
|
-
export function ConversationCard({
|
|
224
|
-
user,
|
|
225
|
-
excerpt,
|
|
226
|
-
unreadCount,
|
|
227
|
-
timestamp,
|
|
228
|
-
onOpen,
|
|
229
|
-
}: {
|
|
230
|
-
user: User;
|
|
231
|
-
excerpt: string;
|
|
232
|
-
unreadCount: number;
|
|
233
|
-
timestamp?: string;
|
|
234
|
-
onOpen: () => void;
|
|
235
|
-
}) {
|
|
236
|
-
return (
|
|
237
|
-
<button
|
|
238
|
-
type="button"
|
|
239
|
-
onClick={onOpen}
|
|
240
|
-
className="w-full rounded-card border border-[var(--color-border-default)] bg-[var(--color-surface-secondary)] p-4 text-left shadow-sm transition hover:-translate-y-0.5"
|
|
241
|
-
>
|
|
242
|
-
<div className="flex items-center gap-3">
|
|
243
|
-
<Avatar src={user.avatarUrl} name={user.displayName} />
|
|
244
|
-
<div className="min-w-0 flex-1">
|
|
245
|
-
<div className="flex items-center justify-between gap-3">
|
|
246
|
-
<p className="truncate font-semibold text-[var(--color-text-primary)]">{user.displayName}</p>
|
|
247
|
-
{timestamp ? <p className="text-xs text-[var(--color-text-tertiary)]">{formatRelativeDate(timestamp)}</p> : null}
|
|
248
|
-
</div>
|
|
249
|
-
<p className="truncate text-sm text-[var(--color-text-secondary)]">{excerpt}</p>
|
|
250
|
-
</div>
|
|
251
|
-
{unreadCount > 0 ? (
|
|
252
|
-
<span className="rounded-cap-primary bg-[var(--color-brand-primary)] px-2 py-1 text-xs font-semibold text-white">
|
|
253
|
-
{unreadCount}
|
|
254
|
-
</span>
|
|
255
|
-
) : null}
|
|
256
|
-
</div>
|
|
257
|
-
</button>
|
|
258
|
-
);
|
|
259
|
-
}
|
|
260
|
-
|
|
261
|
-
export function NotificationCard({
|
|
262
|
-
item,
|
|
263
|
-
actor,
|
|
264
|
-
onOpen,
|
|
265
|
-
}: {
|
|
266
|
-
item: NotificationItem;
|
|
267
|
-
actor?: User;
|
|
268
|
-
onOpen: () => void;
|
|
269
|
-
}) {
|
|
270
|
-
return (
|
|
271
|
-
<button
|
|
272
|
-
type="button"
|
|
273
|
-
onClick={onOpen}
|
|
274
|
-
className="w-full rounded-card border border-[var(--color-border-default)] bg-[var(--color-surface-secondary)] p-4 text-left shadow-sm transition hover:-translate-y-0.5"
|
|
275
|
-
>
|
|
276
|
-
<div className="flex items-center gap-3">
|
|
277
|
-
<Avatar src={actor?.avatarUrl} name={actor?.displayName ?? "Notification"} size="sm" />
|
|
278
|
-
<div className="min-w-0 flex-1">
|
|
279
|
-
<div className="flex items-center gap-2">
|
|
280
|
-
<p className="truncate font-semibold text-[var(--color-text-primary)]">{actor?.displayName ?? "System"}</p>
|
|
281
|
-
{!item.read ? <span className="h-2 w-2 rounded-full bg-[var(--color-brand-accent)]" /> : null}
|
|
282
|
-
</div>
|
|
283
|
-
<p className="text-sm text-[var(--color-text-secondary)]">{item.message}</p>
|
|
284
|
-
</div>
|
|
285
|
-
<p className="text-xs text-[var(--color-text-tertiary)]">{formatRelativeDate(item.createdAt)}</p>
|
|
286
|
-
</div>
|
|
287
|
-
</button>
|
|
288
|
-
);
|
|
289
|
-
}
|
|
290
|
-
|
|
291
|
-
export function MessageBubble({
|
|
292
|
-
message,
|
|
293
|
-
author,
|
|
294
|
-
isMine,
|
|
295
|
-
}: {
|
|
296
|
-
message: Message;
|
|
297
|
-
author: User;
|
|
298
|
-
isMine: boolean;
|
|
299
|
-
}) {
|
|
300
|
-
return (
|
|
301
|
-
<div className={`flex ${isMine ? "justify-end" : "justify-start"}`}>
|
|
302
|
-
<div
|
|
303
|
-
className={`max-w-[min(32rem,86%)] px-4 py-3 shadow-sm ${
|
|
304
|
-
isMine
|
|
305
|
-
? "rounded-cap-primary bg-[var(--color-brand-primary)] text-white"
|
|
306
|
-
: "rounded-cap-alternate border border-[var(--color-border-default)] bg-[var(--color-surface-secondary)] text-[var(--color-text-primary)]"
|
|
307
|
-
}`}
|
|
308
|
-
>
|
|
309
|
-
<p className={`text-xs ${isMine ? "text-white/70" : "text-[var(--color-text-tertiary)]"}`}>{author.displayName}</p>
|
|
310
|
-
<p className="mt-1 text-sm leading-6">{message.body}</p>
|
|
311
|
-
<p className={`mt-2 text-right text-xs ${isMine ? "text-white/70" : "text-[var(--color-text-tertiary)]"}`}>
|
|
312
|
-
{formatTime(message.createdAt)}
|
|
313
|
-
</p>
|
|
314
|
-
</div>
|
|
315
|
-
</div>
|
|
316
|
-
);
|
|
317
|
-
}
|