openuispec 0.1.45 → 0.1.46

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (133) hide show
  1. package/README.md +2 -1
  2. package/cli/init.ts +5 -2
  3. package/examples/social-app/.mcp.json +10 -0
  4. package/examples/social-app/AGENTS.md +105 -0
  5. package/examples/social-app/CLAUDE.md +105 -0
  6. package/examples/social-app/README.md +19 -0
  7. package/examples/social-app/backend/.gitkeep +1 -0
  8. package/examples/social-app/generated/android/social-app/app/build.gradle.kts +92 -0
  9. package/examples/social-app/generated/android/social-app/app/src/main/AndroidManifest.xml +26 -0
  10. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/AppContainer.kt +20 -0
  11. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/MainActivity.kt +35 -0
  12. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/SocialAppApplication.kt +13 -0
  13. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/data/MockData.kt +98 -0
  14. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/data/preferences/AppPreferences.kt +19 -0
  15. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/data/preferences/DataStorePreferencesRepository.kt +68 -0
  16. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/data/preferences/PreferencesRepository.kt +15 -0
  17. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/model/Models.kt +34 -0
  18. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/MainShell.kt +390 -0
  19. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/components/Components.kt +234 -0
  20. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/components/ContractPrimitives.kt +641 -0
  21. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/navigation/RootComponent.kt +113 -0
  22. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/ChatDetailScreen.kt +212 -0
  23. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/CreatePostScreen.kt +113 -0
  24. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/DiscoverScreen.kt +137 -0
  25. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/EditProfileScreen.kt +180 -0
  26. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/HomeFeedScreen.kt +157 -0
  27. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/MessagesInboxScreen.kt +85 -0
  28. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/NotificationsScreen.kt +74 -0
  29. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/PostDetailScreen.kt +293 -0
  30. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/ProfileSelfScreen.kt +116 -0
  31. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/SearchResultsScreen.kt +161 -0
  32. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/SettingsScreen.kt +162 -0
  33. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/SettingsStore.kt +95 -0
  34. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/screens/UserProfileScreen.kt +123 -0
  35. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Color.kt +33 -0
  36. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Shape.kt +41 -0
  37. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Spacing.kt +20 -0
  38. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Theme.kt +82 -0
  39. package/examples/social-app/generated/android/social-app/app/src/main/java/com/social/app/ui/theme/Type.kt +60 -0
  40. package/examples/social-app/generated/android/social-app/app/src/main/res/drawable/ic_launcher_foreground.xml +9 -0
  41. package/examples/social-app/generated/android/social-app/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml +5 -0
  42. package/examples/social-app/generated/android/social-app/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml +5 -0
  43. package/examples/social-app/generated/android/social-app/app/src/main/res/values/strings.xml +91 -0
  44. package/examples/social-app/generated/android/social-app/app/src/main/res/values/themes.xml +10 -0
  45. package/examples/social-app/generated/android/social-app/app/src/main/res/values-ru/strings.xml +79 -0
  46. package/examples/social-app/generated/android/social-app/app/src/main/res/values-uz/strings.xml +79 -0
  47. package/examples/social-app/generated/android/social-app/app/src/main/xml/AndroidManifest.xml +23 -0
  48. package/examples/social-app/generated/android/social-app/build.gradle.kts +6 -0
  49. package/examples/social-app/generated/android/social-app/gradle/libs.versions.toml +48 -0
  50. package/examples/social-app/generated/android/social-app/gradle/wrapper/gradle-wrapper.properties +8 -0
  51. package/examples/social-app/generated/android/social-app/gradle.properties +11 -0
  52. package/examples/social-app/generated/android/social-app/gradlew +25 -0
  53. package/examples/social-app/generated/android/social-app/settings.gradle.kts +23 -0
  54. package/examples/social-app/generated/web/social-app/index.html +12 -0
  55. package/examples/social-app/generated/web/social-app/package-lock.json +2517 -0
  56. package/examples/social-app/generated/web/social-app/package.json +27 -0
  57. package/examples/social-app/generated/web/social-app/src/app/App.tsx +58 -0
  58. package/examples/social-app/generated/web/social-app/src/components/Shell.tsx +247 -0
  59. package/examples/social-app/generated/web/social-app/src/components/cards.tsx +317 -0
  60. package/examples/social-app/generated/web/social-app/src/components/ui.tsx +328 -0
  61. package/examples/social-app/generated/web/social-app/src/flows/CreatePostFlow.tsx +86 -0
  62. package/examples/social-app/generated/web/social-app/src/i18n.tsx +59 -0
  63. package/examples/social-app/generated/web/social-app/src/lib/icons.tsx +85 -0
  64. package/examples/social-app/generated/web/social-app/src/lib/tokens.ts +70 -0
  65. package/examples/social-app/generated/web/social-app/src/lib/utils.ts +97 -0
  66. package/examples/social-app/generated/web/social-app/src/locales/en.json +67 -0
  67. package/examples/social-app/generated/web/social-app/src/locales/ru.json +67 -0
  68. package/examples/social-app/generated/web/social-app/src/locales/uz.json +67 -0
  69. package/examples/social-app/generated/web/social-app/src/main.tsx +16 -0
  70. package/examples/social-app/generated/web/social-app/src/screens/ChatDetailScreen.tsx +90 -0
  71. package/examples/social-app/generated/web/social-app/src/screens/DiscoverScreen.tsx +86 -0
  72. package/examples/social-app/generated/web/social-app/src/screens/EditProfileScreen.tsx +57 -0
  73. package/examples/social-app/generated/web/social-app/src/screens/HomeFeedScreen.tsx +113 -0
  74. package/examples/social-app/generated/web/social-app/src/screens/MessagesInboxScreen.tsx +52 -0
  75. package/examples/social-app/generated/web/social-app/src/screens/NotificationsScreen.tsx +41 -0
  76. package/examples/social-app/generated/web/social-app/src/screens/PostDetailScreen.tsx +115 -0
  77. package/examples/social-app/generated/web/social-app/src/screens/ProfileSelfScreen.tsx +57 -0
  78. package/examples/social-app/generated/web/social-app/src/screens/ProfileUserScreen.tsx +76 -0
  79. package/examples/social-app/generated/web/social-app/src/screens/SearchResultsScreen.tsx +96 -0
  80. package/examples/social-app/generated/web/social-app/src/screens/SettingsScreen.tsx +77 -0
  81. package/examples/social-app/generated/web/social-app/src/state/store.ts +592 -0
  82. package/examples/social-app/generated/web/social-app/src/styles.css +124 -0
  83. package/examples/social-app/generated/web/social-app/src/vite-env.d.ts +1 -0
  84. package/examples/social-app/generated/web/social-app/tsconfig.json +22 -0
  85. package/examples/social-app/generated/web/social-app/tsconfig.node.json +13 -0
  86. package/examples/social-app/generated/web/social-app/tsconfig.node.tsbuildinfo +1 -0
  87. package/examples/social-app/generated/web/social-app/tsconfig.tsbuildinfo +1 -0
  88. package/examples/social-app/generated/web/social-app/vite.config.d.ts +2 -0
  89. package/examples/social-app/generated/web/social-app/vite.config.js +6 -0
  90. package/examples/social-app/generated/web/social-app/vite.config.ts +7 -0
  91. package/examples/social-app/openuispec/README.md +56 -0
  92. package/examples/social-app/openuispec/contracts/.gitkeep +0 -0
  93. package/examples/social-app/openuispec/contracts/action_trigger.yaml +73 -0
  94. package/examples/social-app/openuispec/contracts/collection.yaml +43 -0
  95. package/examples/social-app/openuispec/contracts/data_display.yaml +47 -0
  96. package/examples/social-app/openuispec/contracts/feedback.yaml +49 -0
  97. package/examples/social-app/openuispec/contracts/input_field.yaml +41 -0
  98. package/examples/social-app/openuispec/contracts/nav_container.yaml +34 -0
  99. package/examples/social-app/openuispec/contracts/surface.yaml +41 -0
  100. package/examples/social-app/openuispec/flows/.gitkeep +0 -0
  101. package/examples/social-app/openuispec/flows/create_post.yaml +66 -0
  102. package/examples/social-app/openuispec/locales/.gitkeep +0 -0
  103. package/examples/social-app/openuispec/locales/en.json +67 -0
  104. package/examples/social-app/openuispec/locales/ru.json +67 -0
  105. package/examples/social-app/openuispec/locales/uz.json +67 -0
  106. package/examples/social-app/openuispec/openuispec.yaml +214 -0
  107. package/examples/social-app/openuispec/platform/.gitkeep +0 -0
  108. package/examples/social-app/openuispec/platform/android.yaml +30 -0
  109. package/examples/social-app/openuispec/platform/ios.yaml +19 -0
  110. package/examples/social-app/openuispec/platform/web.yaml +23 -0
  111. package/examples/social-app/openuispec/screens/.gitkeep +0 -0
  112. package/examples/social-app/openuispec/screens/chat_detail.yaml +53 -0
  113. package/examples/social-app/openuispec/screens/discover.yaml +78 -0
  114. package/examples/social-app/openuispec/screens/edit_profile.yaml +78 -0
  115. package/examples/social-app/openuispec/screens/home_feed.yaml +123 -0
  116. package/examples/social-app/openuispec/screens/messages_inbox.yaml +43 -0
  117. package/examples/social-app/openuispec/screens/notifications.yaml +29 -0
  118. package/examples/social-app/openuispec/screens/post_detail.yaml +86 -0
  119. package/examples/social-app/openuispec/screens/profile_self.yaml +53 -0
  120. package/examples/social-app/openuispec/screens/profile_user.yaml +60 -0
  121. package/examples/social-app/openuispec/screens/search_results.yaml +62 -0
  122. package/examples/social-app/openuispec/screens/settings.yaml +94 -0
  123. package/examples/social-app/openuispec/tokens/.gitkeep +0 -0
  124. package/examples/social-app/openuispec/tokens/color.yaml +76 -0
  125. package/examples/social-app/openuispec/tokens/elevation.yaml +31 -0
  126. package/examples/social-app/openuispec/tokens/icons.yaml +147 -0
  127. package/examples/social-app/openuispec/tokens/layout.yaml +37 -0
  128. package/examples/social-app/openuispec/tokens/motion.yaml +28 -0
  129. package/examples/social-app/openuispec/tokens/spacing.yaml +19 -0
  130. package/examples/social-app/openuispec/tokens/themes.yaml +31 -0
  131. package/examples/social-app/openuispec/tokens/typography.yaml +50 -0
  132. package/examples/social-app/package.json +12 -0
  133. package/package.json +1 -1
@@ -0,0 +1,180 @@
1
+ package com.social.app.ui.screens
2
+ import androidx.compose.foundation.layout.Arrangement
3
+ import androidx.compose.foundation.layout.Box
4
+ import androidx.compose.foundation.layout.Column
5
+ import androidx.compose.foundation.layout.fillMaxSize
6
+ import androidx.compose.foundation.layout.fillMaxWidth
7
+ import androidx.compose.foundation.layout.padding
8
+ import androidx.compose.foundation.layout.size
9
+ import androidx.compose.foundation.rememberScrollState
10
+ import androidx.compose.foundation.verticalScroll
11
+ import androidx.compose.material.icons.Icons
12
+ import androidx.compose.material.icons.automirrored.filled.ArrowBack
13
+ import androidx.compose.material.icons.filled.CameraAlt
14
+ import androidx.compose.material3.ExperimentalMaterial3Api
15
+ import androidx.compose.material3.Icon
16
+ import androidx.compose.material3.IconButton
17
+ import androidx.compose.material3.MaterialTheme
18
+ import androidx.compose.material3.Scaffold
19
+ import androidx.compose.material3.SnackbarHostState
20
+ import androidx.compose.material3.Surface
21
+ import androidx.compose.material3.Text
22
+ import androidx.compose.material3.TopAppBar
23
+ import androidx.compose.material3.TopAppBarDefaults
24
+ import androidx.compose.runtime.Composable
25
+ import androidx.compose.runtime.getValue
26
+ import androidx.compose.runtime.mutableStateOf
27
+ import androidx.compose.runtime.remember
28
+ import androidx.compose.runtime.rememberCoroutineScope
29
+ import androidx.compose.runtime.setValue
30
+ import androidx.compose.ui.Alignment
31
+ import androidx.compose.ui.Modifier
32
+ import androidx.compose.ui.layout.ContentScale
33
+ import androidx.compose.ui.res.stringResource
34
+ import androidx.compose.foundation.text.KeyboardOptions
35
+ import androidx.compose.ui.text.style.TextAlign
36
+ import androidx.compose.ui.text.input.KeyboardType
37
+ import androidx.compose.ui.unit.dp
38
+ import coil3.compose.AsyncImage
39
+ import com.social.app.R
40
+ import com.social.app.data.MockData
41
+ import com.social.app.ui.components.ActionTriggerButton
42
+ import com.social.app.ui.components.ActionTriggerVariant
43
+ import com.social.app.ui.components.ContractSnackbarHost
44
+ import com.social.app.ui.components.ContractTextField
45
+ import com.social.app.ui.theme.Shapes
46
+ import com.social.app.ui.theme.Spacing
47
+ import kotlinx.coroutines.delay
48
+ import kotlinx.coroutines.launch
49
+
50
+ @OptIn(ExperimentalMaterial3Api::class)
51
+ @Composable
52
+ fun EditProfileScreen(
53
+ onBackClick: () -> Unit
54
+ ) {
55
+ val user = MockData.users.find { it.id == "user_me" } ?: MockData.users[0]
56
+
57
+ var displayName by remember { mutableStateOf(user.displayName) }
58
+ var handle by remember { mutableStateOf(user.handle) }
59
+ var bio by remember { mutableStateOf(user.bio ?: "") }
60
+ var website by remember { mutableStateOf("https://rustam.design") }
61
+ val savedMessage = stringResource(R.string.edit_profile_saved)
62
+ val snackbarHostState = remember { SnackbarHostState() }
63
+ val scope = rememberCoroutineScope()
64
+
65
+ Scaffold(
66
+ snackbarHost = {
67
+ ContractSnackbarHost(hostState = snackbarHostState)
68
+ },
69
+ topBar = {
70
+ TopAppBar(
71
+ title = { Text(stringResource(R.string.profile_edit_button)) },
72
+ navigationIcon = {
73
+ IconButton(onClick = onBackClick) {
74
+ Icon(Icons.AutoMirrored.Filled.ArrowBack, contentDescription = null)
75
+ }
76
+ },
77
+ colors = TopAppBarDefaults.topAppBarColors(containerColor = MaterialTheme.colorScheme.surface)
78
+ )
79
+ }
80
+ ) { padding ->
81
+ Column(
82
+ modifier = Modifier
83
+ .fillMaxSize()
84
+ .padding(padding)
85
+ .verticalScroll(rememberScrollState())
86
+ .padding(Spacing.MD),
87
+ horizontalAlignment = Alignment.CenterHorizontally,
88
+ verticalArrangement = Arrangement.spacedBy(Spacing.MD),
89
+ ) {
90
+ Surface(
91
+ modifier = Modifier
92
+ .fillMaxWidth(),
93
+ shape = Shapes.CardShape,
94
+ color = MaterialTheme.colorScheme.surfaceVariant,
95
+ ) {
96
+ Column(
97
+ modifier = Modifier
98
+ .fillMaxWidth()
99
+ .padding(Spacing.LG),
100
+ horizontalAlignment = Alignment.CenterHorizontally,
101
+ verticalArrangement = Arrangement.spacedBy(Spacing.SM),
102
+ ) {
103
+ Box(
104
+ modifier = Modifier.size(120.dp),
105
+ contentAlignment = Alignment.Center,
106
+ ) {
107
+ AsyncImage(
108
+ model = user.avatarUrl,
109
+ contentDescription = user.displayName,
110
+ modifier = Modifier.fillMaxSize(),
111
+ contentScale = ContentScale.Crop,
112
+ )
113
+ Surface(
114
+ modifier = Modifier
115
+ .align(Alignment.BottomEnd)
116
+ .size(32.dp),
117
+ shape = Shapes.RoundedCapPrimary,
118
+ color = MaterialTheme.colorScheme.primary,
119
+ ) {
120
+ Box(contentAlignment = Alignment.Center) {
121
+ Icon(
122
+ Icons.Default.CameraAlt,
123
+ contentDescription = null,
124
+ tint = MaterialTheme.colorScheme.onPrimary,
125
+ )
126
+ }
127
+ }
128
+ }
129
+ Text(
130
+ text = stringResource(R.string.edit_profile_avatar),
131
+ style = MaterialTheme.typography.labelLarge,
132
+ color = MaterialTheme.colorScheme.onSurfaceVariant,
133
+ textAlign = TextAlign.Center,
134
+ )
135
+ }
136
+ }
137
+
138
+ ContractTextField(
139
+ value = displayName,
140
+ onValueChange = { displayName = it },
141
+ label = stringResource(R.string.edit_profile_display_name),
142
+ )
143
+ ContractTextField(
144
+ value = handle,
145
+ onValueChange = { handle = it },
146
+ label = stringResource(R.string.edit_profile_handle),
147
+ )
148
+ ContractTextField(
149
+ value = bio,
150
+ onValueChange = { bio = it.take(160) },
151
+ label = stringResource(R.string.edit_profile_bio),
152
+ singleLine = false,
153
+ maxLines = 5,
154
+ )
155
+ ContractTextField(
156
+ value = website,
157
+ onValueChange = { website = it },
158
+ label = stringResource(R.string.edit_profile_website),
159
+ keyboardOptions = KeyboardOptions(keyboardType = KeyboardType.Uri),
160
+ )
161
+
162
+ ActionTriggerButton(
163
+ text = stringResource(R.string.edit_profile_save),
164
+ onClick = {
165
+ scope.launch {
166
+ snackbarHostState.currentSnackbarData?.dismiss()
167
+ snackbarHostState.showSnackbar(message = savedMessage)
168
+ }
169
+ scope.launch {
170
+ delay(3000)
171
+ snackbarHostState.currentSnackbarData?.dismiss()
172
+ onBackClick()
173
+ }
174
+ },
175
+ variant = ActionTriggerVariant.Primary,
176
+ fullWidth = true,
177
+ )
178
+ }
179
+ }
180
+ }
@@ -0,0 +1,157 @@
1
+ package com.social.app.ui.screens
2
+
3
+ import androidx.compose.foundation.ExperimentalFoundationApi
4
+ import androidx.compose.foundation.layout.Arrangement
5
+ import androidx.compose.foundation.layout.PaddingValues
6
+ import androidx.compose.foundation.layout.Spacer
7
+ import androidx.compose.foundation.layout.fillMaxSize
8
+ import androidx.compose.foundation.layout.height
9
+ import androidx.compose.foundation.layout.padding
10
+ import androidx.compose.foundation.lazy.LazyColumn
11
+ import androidx.compose.foundation.lazy.LazyRow
12
+ import androidx.compose.foundation.lazy.items
13
+ import androidx.compose.foundation.lazy.rememberLazyListState
14
+ import androidx.compose.foundation.gestures.snapping.rememberSnapFlingBehavior
15
+ import androidx.compose.material3.CenterAlignedTopAppBar
16
+ import androidx.compose.material3.ExperimentalMaterial3Api
17
+ import androidx.compose.material3.MaterialTheme
18
+ import androidx.compose.material3.Scaffold
19
+ import androidx.compose.material3.Text
20
+ import androidx.compose.material3.TopAppBarDefaults
21
+ import androidx.compose.runtime.Composable
22
+ import androidx.compose.runtime.LaunchedEffect
23
+ import androidx.compose.runtime.getValue
24
+ import androidx.compose.runtime.mutableIntStateOf
25
+ import androidx.compose.runtime.mutableStateOf
26
+ import androidx.compose.runtime.remember
27
+ import androidx.compose.runtime.setValue
28
+ import androidx.compose.runtime.snapshotFlow
29
+ import androidx.compose.ui.Modifier
30
+ import androidx.compose.ui.res.stringResource
31
+ import com.social.app.R
32
+ import com.social.app.data.MockData
33
+ import com.social.app.ui.components.ChipOption
34
+ import com.social.app.ui.components.ContractChipRow
35
+ import com.social.app.ui.components.ContractListCard
36
+ import com.social.app.ui.components.PostItem
37
+ import com.social.app.ui.components.StoryItem
38
+ import com.social.app.ui.theme.Spacing
39
+ import kotlinx.coroutines.flow.distinctUntilChanged
40
+ import kotlinx.coroutines.flow.filter
41
+ import kotlinx.coroutines.flow.map
42
+
43
+ @OptIn(ExperimentalMaterial3Api::class, ExperimentalFoundationApi::class)
44
+ @Composable
45
+ fun HomeFeedScreen(
46
+ onPostClick: (String) -> Unit,
47
+ onUserClick: (String) -> Unit,
48
+ ) {
49
+ var selectedFilter by remember { mutableStateOf("all") }
50
+ val storyListState = rememberLazyListState()
51
+ val feedListState = rememberLazyListState()
52
+ val filteredPosts =
53
+ remember(selectedFilter) {
54
+ when (selectedFilter) {
55
+ "following" -> MockData.posts.filter { it.authorId != "user_me" }
56
+ "popular" -> MockData.posts.filter { it.likeCount >= 1000 }
57
+ else -> MockData.posts
58
+ }
59
+ }
60
+ var visiblePostCount by remember(selectedFilter) { mutableIntStateOf(minOf(2, filteredPosts.size)) }
61
+
62
+ LaunchedEffect(selectedFilter, filteredPosts.size) {
63
+ visiblePostCount = minOf(2, filteredPosts.size)
64
+ }
65
+
66
+ LaunchedEffect(feedListState, filteredPosts.size, visiblePostCount) {
67
+ snapshotFlow { feedListState.layoutInfo.visibleItemsInfo.lastOrNull()?.index ?: -1 }
68
+ .map { lastVisibleIndex -> lastVisibleIndex >= feedListState.layoutInfo.totalItemsCount - 2 }
69
+ .distinctUntilChanged()
70
+ .filter { it }
71
+ .collect {
72
+ if (visiblePostCount < filteredPosts.size) {
73
+ visiblePostCount = minOf(visiblePostCount + 1, filteredPosts.size)
74
+ }
75
+ }
76
+ }
77
+
78
+ Scaffold(
79
+ modifier = Modifier.fillMaxSize(),
80
+ topBar = {
81
+ CenterAlignedTopAppBar(
82
+ title = { Text(stringResource(R.string.nav_home)) },
83
+ colors =
84
+ TopAppBarDefaults.centerAlignedTopAppBarColors(
85
+ containerColor = MaterialTheme.colorScheme.surface,
86
+ ),
87
+ )
88
+ },
89
+ ) { padding ->
90
+ LazyColumn(
91
+ state = feedListState,
92
+ modifier =
93
+ Modifier
94
+ .fillMaxSize()
95
+ .padding(padding),
96
+ contentPadding = PaddingValues(bottom = Spacing.LG),
97
+ ) {
98
+ item {
99
+ LazyRow(
100
+ state = storyListState,
101
+ flingBehavior = rememberSnapFlingBehavior(lazyListState = storyListState),
102
+ contentPadding = PaddingValues(horizontal = Spacing.MD),
103
+ horizontalArrangement = Arrangement.spacedBy(Spacing.SM),
104
+ modifier = Modifier.padding(vertical = Spacing.MD),
105
+ ) {
106
+ items(MockData.stories) { story ->
107
+ val linkedPostId = MockData.posts.find { it.authorId == story.authorId }?.id ?: MockData.posts.first().id
108
+ StoryItem(
109
+ name = story.authorName,
110
+ imageUrl = story.previewUrl,
111
+ onClick = { onPostClick(linkedPostId) },
112
+ )
113
+ }
114
+ }
115
+ }
116
+
117
+ item {
118
+ ContractChipRow(
119
+ options =
120
+ listOf(
121
+ ChipOption("all", stringResource(R.string.home_filter_all)),
122
+ ChipOption("following", stringResource(R.string.home_filter_following)),
123
+ ChipOption("popular", stringResource(R.string.home_filter_popular)),
124
+ ),
125
+ selectedValue = selectedFilter,
126
+ onSelect = { selectedFilter = it.value },
127
+ )
128
+ Spacer(modifier = Modifier.height(Spacing.MD))
129
+ }
130
+
131
+ if (filteredPosts.isEmpty()) {
132
+ item {
133
+ ContractListCard(
134
+ title = stringResource(R.string.home_empty_feed),
135
+ modifier = Modifier.padding(horizontal = Spacing.MD),
136
+ )
137
+ }
138
+ } else {
139
+ items(filteredPosts.take(visiblePostCount), key = { it.id }) { post ->
140
+ PostItem(
141
+ authorName = post.authorName,
142
+ authorHandle = post.authorHandle,
143
+ authorAvatar = post.authorAvatar,
144
+ body = post.body,
145
+ likeCount = post.likeCount,
146
+ commentCount = post.commentCount,
147
+ timestamp = post.timestamp,
148
+ mediaUrl = post.mediaUrl,
149
+ onAuthorClick = { onUserClick(post.authorId) },
150
+ onLikeClick = {},
151
+ onCommentClick = { onPostClick(post.id) },
152
+ )
153
+ }
154
+ }
155
+ }
156
+ }
157
+ }
@@ -0,0 +1,85 @@
1
+ package com.social.app.ui.screens
2
+
3
+ import androidx.compose.foundation.layout.Box
4
+ import androidx.compose.foundation.layout.fillMaxSize
5
+ import androidx.compose.foundation.layout.fillMaxWidth
6
+ import androidx.compose.foundation.layout.padding
7
+ import androidx.compose.foundation.lazy.LazyColumn
8
+ import androidx.compose.foundation.lazy.items
9
+ import androidx.compose.material.icons.Icons
10
+ import androidx.compose.material.icons.filled.Search
11
+ import androidx.compose.material3.CenterAlignedTopAppBar
12
+ import androidx.compose.material3.ExperimentalMaterial3Api
13
+ import androidx.compose.material3.Icon
14
+ import androidx.compose.material3.MaterialTheme
15
+ import androidx.compose.material3.Scaffold
16
+ import androidx.compose.material3.Text
17
+ import androidx.compose.material3.TopAppBarDefaults
18
+ import androidx.compose.runtime.Composable
19
+ import androidx.compose.runtime.getValue
20
+ import androidx.compose.runtime.mutableStateOf
21
+ import androidx.compose.runtime.remember
22
+ import androidx.compose.runtime.setValue
23
+ import androidx.compose.ui.Alignment
24
+ import androidx.compose.ui.Modifier
25
+ import androidx.compose.ui.res.stringResource
26
+ import com.social.app.R
27
+ import com.social.app.data.MockData
28
+ import com.social.app.ui.components.ContractListCard
29
+ import com.social.app.ui.components.ContractTextField
30
+ import com.social.app.ui.theme.Spacing
31
+
32
+ @OptIn(ExperimentalMaterial3Api::class)
33
+ @Composable
34
+ fun MessagesInboxScreen(
35
+ onConversationClick: (String) -> Unit
36
+ ) {
37
+ var searchQuery by remember { mutableStateOf("") }
38
+
39
+ Scaffold(
40
+ topBar = {
41
+ CenterAlignedTopAppBar(
42
+ title = { Text(stringResource(R.string.nav_messages)) },
43
+ colors = TopAppBarDefaults.centerAlignedTopAppBarColors(containerColor = MaterialTheme.colorScheme.surface)
44
+ )
45
+ }
46
+ ) { padding ->
47
+ LazyColumn(
48
+ modifier = Modifier
49
+ .fillMaxSize()
50
+ .padding(padding),
51
+ ) {
52
+ item {
53
+ ContractTextField(
54
+ value = searchQuery,
55
+ onValueChange = { searchQuery = it },
56
+ modifier = Modifier
57
+ .padding(Spacing.MD),
58
+ label = stringResource(R.string.messages_search_placeholder),
59
+ placeholder = stringResource(R.string.messages_search_placeholder),
60
+ leadingIcon = { Icon(Icons.Default.Search, contentDescription = null) },
61
+ )
62
+ }
63
+
64
+ val participants = MockData.users.filter { it.id != "user_me" }
65
+ if (participants.isEmpty()) {
66
+ item {
67
+ Box(modifier = Modifier.fillMaxWidth().padding(Spacing.XL), contentAlignment = Alignment.Center) {
68
+ Text(stringResource(R.string.messages_empty_inbox), style = MaterialTheme.typography.bodyLarge)
69
+ }
70
+ }
71
+ } else {
72
+ items(participants) { user ->
73
+ ContractListCard(
74
+ title = user.displayName,
75
+ subtitle = user.bio ?: user.handle,
76
+ trailing = if (user.id == "u_9921") "1" else null,
77
+ avatarUrl = user.avatarUrl,
78
+ modifier = Modifier.padding(horizontal = Spacing.MD, vertical = Spacing.XS),
79
+ onClick = { onConversationClick(user.id) },
80
+ )
81
+ }
82
+ }
83
+ }
84
+ }
85
+ }
@@ -0,0 +1,74 @@
1
+ package com.social.app.ui.screens
2
+
3
+ import androidx.compose.foundation.layout.Box
4
+ import androidx.compose.foundation.layout.fillMaxSize
5
+ import androidx.compose.foundation.layout.padding
6
+ import androidx.compose.foundation.lazy.LazyColumn
7
+ import androidx.compose.foundation.lazy.items
8
+ import androidx.compose.material3.CenterAlignedTopAppBar
9
+ import androidx.compose.material3.ExperimentalMaterial3Api
10
+ import androidx.compose.material3.MaterialTheme
11
+ import androidx.compose.material3.Scaffold
12
+ import androidx.compose.material3.Text
13
+ import androidx.compose.material3.TopAppBarDefaults
14
+ import androidx.compose.runtime.Composable
15
+ import androidx.compose.runtime.mutableStateListOf
16
+ import androidx.compose.runtime.remember
17
+ import androidx.compose.ui.Alignment
18
+ import androidx.compose.ui.Modifier
19
+ import androidx.compose.ui.res.stringResource
20
+ import com.social.app.R
21
+ import com.social.app.data.MockData
22
+ import com.social.app.ui.components.ContractListCard
23
+ import com.social.app.ui.theme.Spacing
24
+
25
+ @OptIn(ExperimentalMaterial3Api::class)
26
+ @Composable
27
+ fun NotificationsScreen(
28
+ onNotificationRead: (String) -> Unit,
29
+ ) {
30
+ val unreadIds = remember { mutableStateListOf(*MockData.notifications.map { it.id }.toTypedArray()) }
31
+
32
+ Scaffold(
33
+ modifier = Modifier.fillMaxSize(),
34
+ topBar = {
35
+ CenterAlignedTopAppBar(
36
+ title = { Text(stringResource(R.string.nav_notifications)) },
37
+ colors =
38
+ TopAppBarDefaults.centerAlignedTopAppBarColors(
39
+ containerColor = MaterialTheme.colorScheme.surface,
40
+ ),
41
+ )
42
+ },
43
+ ) { padding ->
44
+ if (MockData.notifications.isEmpty()) {
45
+ Box(modifier = Modifier.fillMaxSize(), contentAlignment = Alignment.Center) {
46
+ Text(text = stringResource(R.string.notifications_empty), style = MaterialTheme.typography.bodyLarge)
47
+ }
48
+ } else {
49
+ LazyColumn(
50
+ modifier =
51
+ Modifier
52
+ .fillMaxSize()
53
+ .padding(padding),
54
+ ) {
55
+ items(MockData.notifications) { notification ->
56
+ val isUnread = notification.id in unreadIds
57
+ ContractListCard(
58
+ title = notification.actorName,
59
+ subtitle = notification.message,
60
+ trailing = notification.timestamp,
61
+ avatarUrl = notification.actorAvatar,
62
+ modifier = Modifier.padding(horizontal = Spacing.MD, vertical = Spacing.XS),
63
+ onClick = {
64
+ if (isUnread) {
65
+ unreadIds.remove(notification.id)
66
+ onNotificationRead(notification.id)
67
+ }
68
+ },
69
+ )
70
+ }
71
+ }
72
+ }
73
+ }
74
+ }