mirror of
https://github.com/msfjarvis/compose-lobsters
synced 2025-08-17 22:37:03 +05:30
Merge #162
162: Refactor LobstersItem and add a dedicated comments button r=msfjarvis a=msfjarvis Refer to updated screenshot test for the UI changes. bors r+ Co-authored-by: Harsh Shandilya <me@msfjarvis.dev>
This commit is contained in:
commit
4cd00a4b44
7 changed files with 129 additions and 68 deletions
Binary file not shown.
Before Width: | Height: | Size: 56 KiB After Width: | Height: | Size: 57 KiB |
|
@ -20,9 +20,9 @@ class LobstersItemTest : ScreenshotTest {
|
||||||
DarkTestTheme {
|
DarkTestTheme {
|
||||||
LobstersItem(
|
LobstersItem(
|
||||||
post = TEST_POST,
|
post = TEST_POST,
|
||||||
onClick = { /*TODO*/ },
|
viewPost = { /*TODO*/ },
|
||||||
onLongClick = { /*TODO*/ },
|
viewComments = { /*TODO*/ },
|
||||||
onSaveButtonClick = { /*TODO*/ },
|
toggleSave = { /*TODO*/ },
|
||||||
isSaved = true,
|
isSaved = true,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
|
@ -58,9 +58,9 @@ fun HottestPosts(
|
||||||
LobstersItem(
|
LobstersItem(
|
||||||
post = item,
|
post = item,
|
||||||
isSaved = isSaved,
|
isSaved = isSaved,
|
||||||
onClick = { urlLauncher.launch(item.url.ifEmpty { item.commentsUrl }) },
|
viewPost = { urlLauncher.launch(item.url.ifEmpty { item.commentsUrl }) },
|
||||||
onLongClick = { urlLauncher.launch(item.commentsUrl) },
|
viewComments = { urlLauncher.launch(item.commentsUrl) },
|
||||||
onSaveButtonClick = {
|
toggleSave = {
|
||||||
isSaved = isSaved.not()
|
isSaved = isSaved.not()
|
||||||
saveAction.invoke(item)
|
saveAction.invoke(item)
|
||||||
},
|
},
|
||||||
|
|
|
@ -3,17 +3,17 @@ package dev.msfjarvis.lobsters.ui.posts
|
||||||
import androidx.compose.animation.Crossfade
|
import androidx.compose.animation.Crossfade
|
||||||
import androidx.compose.foundation.ExperimentalFoundationApi
|
import androidx.compose.foundation.ExperimentalFoundationApi
|
||||||
import androidx.compose.foundation.background
|
import androidx.compose.foundation.background
|
||||||
import androidx.compose.foundation.combinedClickable
|
import androidx.compose.foundation.clickable
|
||||||
import androidx.compose.foundation.layout.Arrangement
|
import androidx.compose.foundation.layout.Arrangement
|
||||||
|
import androidx.compose.foundation.layout.Box
|
||||||
import androidx.compose.foundation.layout.Column
|
import androidx.compose.foundation.layout.Column
|
||||||
import androidx.compose.foundation.layout.Row
|
import androidx.compose.foundation.layout.Row
|
||||||
import androidx.compose.foundation.layout.fillMaxWidth
|
|
||||||
import androidx.compose.foundation.layout.padding
|
import androidx.compose.foundation.layout.padding
|
||||||
import androidx.compose.foundation.layout.requiredSize
|
import androidx.compose.foundation.layout.requiredSize
|
||||||
import androidx.compose.foundation.layout.requiredWidth
|
|
||||||
import androidx.compose.foundation.lazy.LazyColumn
|
import androidx.compose.foundation.lazy.LazyColumn
|
||||||
import androidx.compose.foundation.lazy.items
|
import androidx.compose.foundation.lazy.items
|
||||||
import androidx.compose.foundation.shape.RoundedCornerShape
|
import androidx.compose.foundation.shape.RoundedCornerShape
|
||||||
|
import androidx.compose.material.IconButton
|
||||||
import androidx.compose.material.IconToggleButton
|
import androidx.compose.material.IconToggleButton
|
||||||
import androidx.compose.material.MaterialTheme
|
import androidx.compose.material.MaterialTheme
|
||||||
import androidx.compose.material.Surface
|
import androidx.compose.material.Surface
|
||||||
|
@ -52,77 +52,128 @@ val TEST_POST = SavedPost(
|
||||||
fun LobstersItem(
|
fun LobstersItem(
|
||||||
post: SavedPost,
|
post: SavedPost,
|
||||||
isSaved: Boolean,
|
isSaved: Boolean,
|
||||||
onClick: () -> Unit,
|
viewPost: () -> Unit,
|
||||||
onLongClick: () -> Unit,
|
viewComments: () -> Unit,
|
||||||
onSaveButtonClick: () -> Unit,
|
toggleSave: () -> Unit,
|
||||||
) {
|
) {
|
||||||
Surface(
|
Surface(
|
||||||
modifier = Modifier
|
modifier = Modifier
|
||||||
.combinedClickable(
|
.clickable { viewPost.invoke() },
|
||||||
onClick = onClick,
|
|
||||||
onLongClick = onLongClick,
|
|
||||||
),
|
|
||||||
) {
|
) {
|
||||||
Row(
|
Row(
|
||||||
modifier = Modifier.padding(start = 12.dp, end = 24.dp),
|
modifier = Modifier.padding(start = 12.dp),
|
||||||
verticalAlignment = Alignment.CenterVertically,
|
verticalAlignment = Alignment.CenterVertically,
|
||||||
horizontalArrangement = Arrangement.SpaceBetween,
|
horizontalArrangement = Arrangement.SpaceBetween,
|
||||||
) {
|
) {
|
||||||
Column(
|
Box(
|
||||||
modifier = Modifier.fillMaxWidth(),
|
modifier = Modifier.weight(0.8f),
|
||||||
) {
|
) {
|
||||||
Text(
|
PostDetails(
|
||||||
text = post.title,
|
post,
|
||||||
color = titleColor,
|
|
||||||
fontWeight = FontWeight.Bold,
|
|
||||||
modifier = Modifier
|
|
||||||
.padding(top = 4.dp),
|
|
||||||
)
|
)
|
||||||
TagRow(
|
}
|
||||||
tags = post.tags,
|
Box(
|
||||||
modifier = Modifier
|
modifier = Modifier.weight(0.1f),
|
||||||
.padding(top = 8.dp, bottom = 8.dp, end = 16.dp),
|
) {
|
||||||
|
SaveButton(
|
||||||
|
isSaved,
|
||||||
|
toggleSave,
|
||||||
)
|
)
|
||||||
Row {
|
|
||||||
CoilImage(
|
|
||||||
data = "${LobstersApi.BASE_URL}/${post.submitterAvatarUrl}",
|
|
||||||
contentDescription = stringResource(
|
|
||||||
R.string.avatar_content_description,
|
|
||||||
post.submitterName
|
|
||||||
),
|
|
||||||
fadeIn = true,
|
|
||||||
requestBuilder = {
|
|
||||||
transformations(CircleCropTransformation())
|
|
||||||
},
|
|
||||||
modifier = Modifier
|
|
||||||
.requiredWidth(30.dp)
|
|
||||||
.padding(4.dp),
|
|
||||||
)
|
|
||||||
Text(
|
|
||||||
text = stringResource(id = R.string.submitted_by, post.submitterName),
|
|
||||||
modifier = Modifier
|
|
||||||
.padding(4.dp),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
IconToggleButton(
|
Box(
|
||||||
checked = isSaved,
|
modifier = Modifier.weight(0.1f),
|
||||||
onCheckedChange = { onSaveButtonClick.invoke() },
|
|
||||||
modifier = Modifier
|
|
||||||
.requiredSize(24.dp),
|
|
||||||
) {
|
) {
|
||||||
Crossfade(targetState = isSaved) { saved ->
|
CommentsButton(
|
||||||
IconResource(
|
onClick = viewComments,
|
||||||
resourceId = if (saved) R.drawable.ic_favorite_24px else R.drawable.ic_favorite_border_24px,
|
)
|
||||||
tint = MaterialTheme.colors.secondary,
|
|
||||||
contentDescription = stringResource(if (saved) R.string.remove_from_saved_posts else R.string.add_to_saved_posts),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
fun PostDetails(
|
||||||
|
post: SavedPost,
|
||||||
|
) {
|
||||||
|
Column(
|
||||||
|
modifier = Modifier.padding(top = 8.dp, bottom = 8.dp),
|
||||||
|
) {
|
||||||
|
Text(
|
||||||
|
text = post.title,
|
||||||
|
color = titleColor,
|
||||||
|
fontWeight = FontWeight.Bold,
|
||||||
|
modifier = Modifier
|
||||||
|
.padding(bottom = 4.dp),
|
||||||
|
)
|
||||||
|
TagRow(
|
||||||
|
tags = post.tags,
|
||||||
|
modifier = Modifier
|
||||||
|
.padding(bottom = 4.dp),
|
||||||
|
)
|
||||||
|
Row(
|
||||||
|
verticalAlignment = Alignment.CenterVertically,
|
||||||
|
) {
|
||||||
|
CoilImage(
|
||||||
|
data = "${LobstersApi.BASE_URL}/${post.submitterAvatarUrl}",
|
||||||
|
contentDescription = stringResource(
|
||||||
|
R.string.avatar_content_description,
|
||||||
|
post.submitterName
|
||||||
|
),
|
||||||
|
fadeIn = true,
|
||||||
|
requestBuilder = {
|
||||||
|
transformations(CircleCropTransformation())
|
||||||
|
},
|
||||||
|
modifier = Modifier
|
||||||
|
.requiredSize(24.dp)
|
||||||
|
.padding(bottom = 4.dp),
|
||||||
|
)
|
||||||
|
Text(
|
||||||
|
text = stringResource(id = R.string.submitted_by, post.submitterName),
|
||||||
|
modifier = Modifier
|
||||||
|
.padding(start = 4.dp),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
fun SaveButton(
|
||||||
|
isSaved: Boolean,
|
||||||
|
onSaveButtonClick: () -> Unit,
|
||||||
|
) {
|
||||||
|
IconToggleButton(
|
||||||
|
checked = isSaved,
|
||||||
|
onCheckedChange = { onSaveButtonClick.invoke() },
|
||||||
|
modifier = Modifier
|
||||||
|
.requiredSize(24.dp),
|
||||||
|
) {
|
||||||
|
Crossfade(targetState = isSaved) { saved ->
|
||||||
|
IconResource(
|
||||||
|
resourceId = if (saved) R.drawable.ic_favorite_24px else R.drawable.ic_favorite_border_24px,
|
||||||
|
tint = MaterialTheme.colors.secondary,
|
||||||
|
contentDescription = stringResource(if (saved) R.string.remove_from_saved_posts else R.string.add_to_saved_posts),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
fun CommentsButton(
|
||||||
|
onClick: () -> Unit,
|
||||||
|
) {
|
||||||
|
IconButton(
|
||||||
|
onClick = onClick,
|
||||||
|
modifier = Modifier
|
||||||
|
.requiredSize(24.dp),
|
||||||
|
) {
|
||||||
|
IconResource(
|
||||||
|
resourceId = R.drawable.ic_insert_comment_24px,
|
||||||
|
tint = MaterialTheme.colors.secondary,
|
||||||
|
contentDescription = stringResource(R.string.open_comments),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Composable
|
@Composable
|
||||||
fun TagRow(
|
fun TagRow(
|
||||||
tags: List<String>,
|
tags: List<String>,
|
||||||
|
@ -154,9 +205,9 @@ fun Preview() {
|
||||||
LobstersItem(
|
LobstersItem(
|
||||||
post = item,
|
post = item,
|
||||||
isSaved = false,
|
isSaved = false,
|
||||||
onClick = {},
|
viewPost = {},
|
||||||
onLongClick = {},
|
viewComments = {},
|
||||||
onSaveButtonClick = {},
|
toggleSave = {},
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -59,9 +59,9 @@ fun SavedPosts(
|
||||||
LobstersItem(
|
LobstersItem(
|
||||||
post = item,
|
post = item,
|
||||||
isSaved = true,
|
isSaved = true,
|
||||||
onClick = { urlLauncher.launch(item.url.ifEmpty { item.commentsUrl }) },
|
viewPost = { urlLauncher.launch(item.url.ifEmpty { item.commentsUrl }) },
|
||||||
onLongClick = { urlLauncher.launch(item.commentsUrl) },
|
viewComments = { urlLauncher.launch(item.commentsUrl) },
|
||||||
onSaveButtonClick = { saveAction.invoke(item) },
|
toggleSave = { saveAction.invoke(item) },
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
9
app/src/main/res/drawable/ic_insert_comment_24px.xml
Normal file
9
app/src/main/res/drawable/ic_insert_comment_24px.xml
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
<vector xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
android:width="24dp"
|
||||||
|
android:height="24dp"
|
||||||
|
android:viewportWidth="24"
|
||||||
|
android:viewportHeight="24">
|
||||||
|
<path
|
||||||
|
android:fillColor="#000000"
|
||||||
|
android:pathData="M20,2L4,2c-1.1,0 -2,0.9 -2,2v12c0,1.1 0.9,2 2,2h14l4,4L22,4c0,-1.1 -0.9,-2 -2,-2zM18,14L6,14v-2h12v2zM18,11L6,11L6,9h12v2zM18,8L6,8L6,6h12v2z" />
|
||||||
|
</vector>
|
|
@ -9,4 +9,5 @@
|
||||||
<string name="add_to_saved_posts">Add to saved posts</string>
|
<string name="add_to_saved_posts">Add to saved posts</string>
|
||||||
<string name="remove_from_saved_posts">Remove from saved posts</string>
|
<string name="remove_from_saved_posts">Remove from saved posts</string>
|
||||||
<string name="refresh_posts_content_description">Refresh posts</string>
|
<string name="refresh_posts_content_description">Refresh posts</string>
|
||||||
|
<string name="open_comments">Open comments</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue