mirror of
https://github.com/msfjarvis/compose-lobsters
synced 2025-08-18 04:27:02 +05:30
Merge #174
174: Cleanup string abstraction and switch everything to it r=msfjarvis a=msfjarvis bors r+ Co-authored-by: Harsh Shandilya <me@msfjarvis.dev>
This commit is contained in:
commit
f6d8d3c536
13 changed files with 67 additions and 99 deletions
1
app/proguard-rules.pro
vendored
1
app/proguard-rules.pro
vendored
|
@ -3,5 +3,6 @@
|
||||||
-dontwarn androidx.compose.animation.tooling.ComposeAnimation
|
-dontwarn androidx.compose.animation.tooling.ComposeAnimation
|
||||||
-dontwarn android.view.RenderNode
|
-dontwarn android.view.RenderNode
|
||||||
-dontwarn android.view.DisplayListCanvas
|
-dontwarn android.view.DisplayListCanvas
|
||||||
|
-dontwarn sun.misc.Unsafe
|
||||||
-dontobfuscate
|
-dontobfuscate
|
||||||
-dontoptimize
|
-dontoptimize
|
||||||
|
|
|
@ -12,7 +12,6 @@ import androidx.compose.runtime.getValue
|
||||||
import androidx.compose.runtime.rememberCoroutineScope
|
import androidx.compose.runtime.rememberCoroutineScope
|
||||||
import androidx.compose.ui.Modifier
|
import androidx.compose.ui.Modifier
|
||||||
import androidx.compose.ui.platform.testTag
|
import androidx.compose.ui.platform.testTag
|
||||||
import androidx.compose.ui.res.stringResource
|
|
||||||
import androidx.lifecycle.viewmodel.compose.viewModel
|
import androidx.lifecycle.viewmodel.compose.viewModel
|
||||||
import androidx.navigation.compose.KEY_ROUTE
|
import androidx.navigation.compose.KEY_ROUTE
|
||||||
import androidx.navigation.compose.NavHost
|
import androidx.navigation.compose.NavHost
|
||||||
|
@ -27,6 +26,7 @@ import dev.msfjarvis.lobsters.ui.posts.HottestPosts
|
||||||
import dev.msfjarvis.lobsters.ui.posts.SavedPosts
|
import dev.msfjarvis.lobsters.ui.posts.SavedPosts
|
||||||
import dev.msfjarvis.lobsters.ui.viewmodel.LobstersViewModel
|
import dev.msfjarvis.lobsters.ui.viewmodel.LobstersViewModel
|
||||||
import dev.msfjarvis.lobsters.util.IconResource
|
import dev.msfjarvis.lobsters.util.IconResource
|
||||||
|
import dev.msfjarvis.lobsters.utils.get
|
||||||
import kotlinx.coroutines.launch
|
import kotlinx.coroutines.launch
|
||||||
|
|
||||||
@Composable
|
@Composable
|
||||||
|
@ -104,10 +104,10 @@ fun LobstersBottomNav(
|
||||||
icon = {
|
icon = {
|
||||||
IconResource(
|
IconResource(
|
||||||
resourceId = screen.badgeRes,
|
resourceId = screen.badgeRes,
|
||||||
contentDescription = stringResource(screen.labelRes),
|
contentDescription = screen.labelRes.get(),
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
label = { Text(stringResource(id = screen.labelRes)) },
|
label = { Text(screen.labelRes.get()) },
|
||||||
selected = currentDestination == screen,
|
selected = currentDestination == screen,
|
||||||
modifier = Modifier.testTag(screen.name),
|
modifier = Modifier.testTag(screen.name),
|
||||||
alwaysShowLabel = false,
|
alwaysShowLabel = false,
|
||||||
|
|
|
@ -13,6 +13,8 @@ import androidx.compose.ui.unit.dp
|
||||||
import dev.msfjarvis.lobsters.R
|
import dev.msfjarvis.lobsters.R
|
||||||
import dev.msfjarvis.lobsters.ui.navigation.Destination
|
import dev.msfjarvis.lobsters.ui.navigation.Destination
|
||||||
import dev.msfjarvis.lobsters.util.IconResource
|
import dev.msfjarvis.lobsters.util.IconResource
|
||||||
|
import dev.msfjarvis.lobsters.utils.Strings
|
||||||
|
import dev.msfjarvis.lobsters.utils.get
|
||||||
import kotlinx.coroutines.launch
|
import kotlinx.coroutines.launch
|
||||||
|
|
||||||
@OptIn(ExperimentalAnimationApi::class)
|
@OptIn(ExperimentalAnimationApi::class)
|
||||||
|
@ -33,7 +35,7 @@ fun LobstersTopAppBar(
|
||||||
if (currentDestination == Destination.Saved) {
|
if (currentDestination == Destination.Saved) {
|
||||||
IconResource(
|
IconResource(
|
||||||
resourceId = R.drawable.ic_sort_24px,
|
resourceId = R.drawable.ic_sort_24px,
|
||||||
contentDescription = stringResource(id = R.string.change_sorting_order),
|
contentDescription = Strings.ChangeSortingOrder.get(),
|
||||||
modifier = Modifier
|
modifier = Modifier
|
||||||
.padding(horizontal = 8.dp, vertical = 8.dp)
|
.padding(horizontal = 8.dp, vertical = 8.dp)
|
||||||
.clickable { scope.launch { toggleSortingOrder() } },
|
.clickable { scope.launch { toggleSortingOrder() } },
|
||||||
|
|
|
@ -1,19 +1,19 @@
|
||||||
package dev.msfjarvis.lobsters.ui.navigation
|
package dev.msfjarvis.lobsters.ui.navigation
|
||||||
|
|
||||||
import androidx.annotation.DrawableRes
|
import androidx.annotation.DrawableRes
|
||||||
import androidx.annotation.StringRes
|
|
||||||
import dev.msfjarvis.lobsters.R
|
import dev.msfjarvis.lobsters.R
|
||||||
|
import dev.msfjarvis.lobsters.utils.Strings
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Destinations for navigation within the app.
|
* Destinations for navigation within the app.
|
||||||
*/
|
*/
|
||||||
enum class Destination(
|
enum class Destination(
|
||||||
val route: String,
|
val route: String,
|
||||||
@StringRes val labelRes: Int,
|
val labelRes: Strings,
|
||||||
@DrawableRes val badgeRes: Int,
|
@DrawableRes val badgeRes: Int,
|
||||||
) {
|
) {
|
||||||
Hottest("hottest", R.string.hottest_posts, R.drawable.ic_whatshot_24px),
|
Hottest("hottest", Strings.HottestPosts, R.drawable.ic_whatshot_24px),
|
||||||
Saved("saved", R.string.saved_posts, R.drawable.ic_favorite_24px),
|
Saved("saved", Strings.SavedPosts, R.drawable.ic_favorite_24px),
|
||||||
;
|
;
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
|
|
|
@ -36,8 +36,8 @@ import dev.msfjarvis.lobsters.data.local.SavedPost
|
||||||
import dev.msfjarvis.lobsters.ui.theme.LobstersTheme
|
import dev.msfjarvis.lobsters.ui.theme.LobstersTheme
|
||||||
import dev.msfjarvis.lobsters.ui.theme.titleColor
|
import dev.msfjarvis.lobsters.ui.theme.titleColor
|
||||||
import dev.msfjarvis.lobsters.util.IconResource
|
import dev.msfjarvis.lobsters.util.IconResource
|
||||||
import dev.msfjarvis.lobsters.utils.StringEnum
|
import dev.msfjarvis.lobsters.utils.Strings
|
||||||
import dev.msfjarvis.lobsters.utils.stringValue
|
import dev.msfjarvis.lobsters.utils.get
|
||||||
|
|
||||||
val TEST_POST = SavedPost(
|
val TEST_POST = SavedPost(
|
||||||
shortId = "zqyydb",
|
shortId = "zqyydb",
|
||||||
|
@ -142,10 +142,7 @@ fun SubmitterAvatar(
|
||||||
) {
|
) {
|
||||||
CoilImage(
|
CoilImage(
|
||||||
data = "${LobstersApi.BASE_URL}/$avatarUrl",
|
data = "${LobstersApi.BASE_URL}/$avatarUrl",
|
||||||
contentDescription = stringValue(
|
contentDescription = Strings.AvatarContentDescription.get(name),
|
||||||
StringEnum.AvatarContentDescription,
|
|
||||||
name,
|
|
||||||
),
|
|
||||||
fadeIn = true,
|
fadeIn = true,
|
||||||
requestBuilder = {
|
requestBuilder = {
|
||||||
transformations(CircleCropTransformation())
|
transformations(CircleCropTransformation())
|
||||||
|
@ -160,7 +157,7 @@ fun SubmitterNameText(
|
||||||
name: String,
|
name: String,
|
||||||
) {
|
) {
|
||||||
Text(
|
Text(
|
||||||
text = stringValue(StringEnum.SubmittedBy, name),
|
text = Strings.SubmittedBy.get(name),
|
||||||
modifier = Modifier
|
modifier = Modifier
|
||||||
.padding(start = 4.dp),
|
.padding(start = 4.dp),
|
||||||
)
|
)
|
||||||
|
@ -180,12 +177,10 @@ fun SaveButton(
|
||||||
.then(modifier),
|
.then(modifier),
|
||||||
) {
|
) {
|
||||||
Crossfade(targetState = isSaved) { saved ->
|
Crossfade(targetState = isSaved) { saved ->
|
||||||
// Using if (saved) ... else ... throws an IllegalArgumentException
|
|
||||||
val contentDescriptionEnum = if (saved) StringEnum.RemoveFromSavedPosts else StringEnum.AddToSavedPosts
|
|
||||||
IconResource(
|
IconResource(
|
||||||
resourceId = if (saved) R.drawable.ic_favorite_24px else R.drawable.ic_favorite_border_24px,
|
resourceId = if (saved) R.drawable.ic_favorite_24px else R.drawable.ic_favorite_border_24px,
|
||||||
tint = MaterialTheme.colors.secondary,
|
tint = MaterialTheme.colors.secondary,
|
||||||
contentDescription = stringValue(contentDescriptionEnum),
|
contentDescription = if (saved) Strings.RemoveFromSavedPosts.get() else Strings.AddToSavedPosts.get(),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -205,7 +200,7 @@ fun CommentsButton(
|
||||||
IconResource(
|
IconResource(
|
||||||
resourceId = R.drawable.ic_insert_comment_24px,
|
resourceId = R.drawable.ic_insert_comment_24px,
|
||||||
tint = MaterialTheme.colors.secondary,
|
tint = MaterialTheme.colors.secondary,
|
||||||
contentDescription = stringValue(StringEnum.OpenComments),
|
contentDescription = Strings.OpenComments.get(),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,8 +21,8 @@ import dev.msfjarvis.lobsters.data.local.SavedPost
|
||||||
import dev.msfjarvis.lobsters.ui.urllauncher.LocalUrlLauncher
|
import dev.msfjarvis.lobsters.ui.urllauncher.LocalUrlLauncher
|
||||||
import dev.msfjarvis.lobsters.util.IconResource
|
import dev.msfjarvis.lobsters.util.IconResource
|
||||||
import dev.msfjarvis.lobsters.util.asZonedDateTime
|
import dev.msfjarvis.lobsters.util.asZonedDateTime
|
||||||
import dev.msfjarvis.lobsters.utils.StringEnum
|
import dev.msfjarvis.lobsters.utils.Strings
|
||||||
import dev.msfjarvis.lobsters.utils.stringValue
|
import dev.msfjarvis.lobsters.utils.get
|
||||||
import kotlinx.coroutines.flow.Flow
|
import kotlinx.coroutines.flow.Flow
|
||||||
|
|
||||||
@OptIn(ExperimentalFoundationApi::class)
|
@OptIn(ExperimentalFoundationApi::class)
|
||||||
|
@ -47,9 +47,9 @@ fun SavedPosts(
|
||||||
R.drawable.ic_favorite_border_24px,
|
R.drawable.ic_favorite_border_24px,
|
||||||
tint = Color(0xFFD97373),
|
tint = Color(0xFFD97373),
|
||||||
modifier = Modifier.padding(16.dp),
|
modifier = Modifier.padding(16.dp),
|
||||||
contentDescription = stringValue(StringEnum.AddToSavedPosts),
|
contentDescription = Strings.AddToSavedPosts.get(),
|
||||||
)
|
)
|
||||||
Text(stringValue(StringEnum.NoSavedPost))
|
Text(Strings.NoSavedPost.get())
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
LazyColumn(
|
LazyColumn(
|
||||||
|
|
|
@ -1,14 +1,3 @@
|
||||||
<resources>
|
<resources>
|
||||||
<string name="app_name">Claw</string>
|
<string name="app_name">Claw</string>
|
||||||
<string name="loading">Loading posts…</string>
|
|
||||||
<string name="no_saved_posts">You don\'t have any saved posts</string>
|
|
||||||
<string name="hottest_posts">Hottest</string>
|
|
||||||
<string name="saved_posts">Saved</string>
|
|
||||||
<string name="submitted_by">submitted by %1$s</string>
|
|
||||||
<string name="avatar_content_description">%1$s\'s avatar</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="refresh_posts_content_description">Refresh posts</string>
|
|
||||||
<string name="open_comments">Open comments</string>
|
|
||||||
<string name="change_sorting_order">Change sort order</string>
|
|
||||||
</resources>
|
</resources>
|
||||||
|
|
|
@ -68,10 +68,14 @@ kotlin {
|
||||||
}
|
}
|
||||||
|
|
||||||
android {
|
android {
|
||||||
buildFeatures { androidResources = true }
|
buildFeatures {
|
||||||
|
androidResources = true
|
||||||
|
}
|
||||||
|
|
||||||
sourceSets["main"].apply {
|
sourceSets {
|
||||||
|
named("main") {
|
||||||
manifest.srcFile("src/androidMain/AndroidManifest.xml")
|
manifest.srcFile("src/androidMain/AndroidManifest.xml")
|
||||||
res.srcDir("src/androidMain/res")
|
res.srcDirs("src/androidMain/res")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,37 +4,28 @@ import androidx.compose.runtime.Composable
|
||||||
import androidx.compose.ui.res.stringResource
|
import androidx.compose.ui.res.stringResource
|
||||||
import dev.msfjarvis.lobsters.common.R
|
import dev.msfjarvis.lobsters.common.R
|
||||||
|
|
||||||
private fun stringEnumMapper(stringEnum: StringEnum): Int {
|
private fun stringEnumMapper(stringEnum: Strings): Int {
|
||||||
return when (stringEnum) {
|
return when (stringEnum) {
|
||||||
StringEnum.AddToSavedPosts -> R.string.add_to_saved_posts
|
Strings.AddToSavedPosts -> R.string.add_to_saved_posts
|
||||||
StringEnum.AppName -> R.string.app_name
|
Strings.AppName -> R.string.app_name
|
||||||
StringEnum.AvatarContentDescription -> R.string.avatar_content_description
|
Strings.AvatarContentDescription -> R.string.avatar_content_description
|
||||||
StringEnum.HottestPosts -> R.string.hottest_posts
|
Strings.ChangeSortingOrder -> R.string.change_sorting_order
|
||||||
StringEnum.Loading -> R.string.loading
|
Strings.HottestPosts -> R.string.hottest_posts
|
||||||
StringEnum.NoSavedPost -> R.string.no_saved_posts
|
Strings.NoSavedPost -> R.string.no_saved_posts
|
||||||
StringEnum.OpenComments -> R.string.open_comments
|
Strings.OpenComments -> R.string.open_comments
|
||||||
StringEnum.RefreshPostsContentDescription -> R.string.refresh_posts_content_description
|
Strings.RefreshPostsContentDescription -> R.string.refresh_posts_content_description
|
||||||
StringEnum.RemoveFromSavedPosts -> R.string.remove_from_saved_posts
|
Strings.RemoveFromSavedPosts -> R.string.remove_from_saved_posts
|
||||||
StringEnum.SubmittedBy -> R.string.submitted_by
|
Strings.SavedPosts -> R.string.saved_posts
|
||||||
|
Strings.SubmittedBy -> R.string.submitted_by
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Composable
|
@Composable
|
||||||
actual fun stringValue(enum: StringEnum): String {
|
actual fun Strings.get(): String {
|
||||||
return stringResource(stringEnumMapper(enum))
|
return stringResource(stringEnumMapper(this))
|
||||||
}
|
}
|
||||||
|
|
||||||
@Composable
|
@Composable
|
||||||
actual fun stringValue(enum: StringEnum, arg1: Any): String {
|
actual fun Strings.get(fmt: Any): String {
|
||||||
return stringResource(stringEnumMapper(enum), arg1)
|
return stringResource(stringEnumMapper(this), fmt)
|
||||||
}
|
|
||||||
|
|
||||||
@Composable
|
|
||||||
actual fun stringValue(enum: StringEnum, arg1: Any, arg2: Any): String {
|
|
||||||
return stringResource(stringEnumMapper(enum), arg1, arg2)
|
|
||||||
}
|
|
||||||
|
|
||||||
@Composable
|
|
||||||
actual fun stringValue(enum: StringEnum, arg1: Any, arg2: Any, arg3: Any): String {
|
|
||||||
return stringResource(stringEnumMapper(enum), arg1, arg2, arg3)
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,5 @@
|
||||||
<resources>
|
<resources>
|
||||||
<string name="app_name">Claw</string>
|
<string name="app_name">Claw</string>
|
||||||
<string name="loading">Loading posts…</string>
|
|
||||||
<string name="no_saved_posts">You don\'t have any saved posts</string>
|
<string name="no_saved_posts">You don\'t have any saved posts</string>
|
||||||
<string name="hottest_posts">Hottest</string>
|
<string name="hottest_posts">Hottest</string>
|
||||||
<string name="saved_posts">Saved</string>
|
<string name="saved_posts">Saved</string>
|
||||||
|
@ -10,4 +9,5 @@
|
||||||
<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>
|
<string name="open_comments">Open comments</string>
|
||||||
|
<string name="change_sorting_order">Change sort order</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
|
|
@ -3,13 +3,7 @@ package dev.msfjarvis.lobsters.utils
|
||||||
import androidx.compose.runtime.Composable
|
import androidx.compose.runtime.Composable
|
||||||
|
|
||||||
@Composable
|
@Composable
|
||||||
expect fun stringValue(enum: StringEnum): String
|
expect fun Strings.get(): String
|
||||||
|
|
||||||
@Composable
|
@Composable
|
||||||
expect fun stringValue(enum: StringEnum, arg1: Any): String
|
expect fun Strings.get(fmt: Any): String
|
||||||
|
|
||||||
@Composable
|
|
||||||
expect fun stringValue(enum: StringEnum, arg1: Any, arg2: Any): String
|
|
||||||
|
|
||||||
@Composable
|
|
||||||
expect fun stringValue(enum: StringEnum, arg1: Any, arg2: Any, arg3: Any): String
|
|
||||||
|
|
|
@ -1,15 +1,16 @@
|
||||||
package dev.msfjarvis.lobsters.utils
|
package dev.msfjarvis.lobsters.utils
|
||||||
|
|
||||||
enum class StringEnum {
|
enum class Strings {
|
||||||
AddToSavedPosts,
|
AddToSavedPosts,
|
||||||
AppName,
|
AppName,
|
||||||
AvatarContentDescription,
|
AvatarContentDescription,
|
||||||
|
ChangeSortingOrder,
|
||||||
HottestPosts,
|
HottestPosts,
|
||||||
Loading,
|
|
||||||
NoSavedPost,
|
NoSavedPost,
|
||||||
OpenComments,
|
OpenComments,
|
||||||
RefreshPostsContentDescription,
|
RefreshPostsContentDescription,
|
||||||
RemoveFromSavedPosts,
|
RemoveFromSavedPosts,
|
||||||
|
SavedPosts,
|
||||||
SubmittedBy,
|
SubmittedBy,
|
||||||
;
|
;
|
||||||
}
|
}
|
|
@ -2,37 +2,28 @@ package dev.msfjarvis.lobsters.utils
|
||||||
|
|
||||||
import androidx.compose.runtime.Composable
|
import androidx.compose.runtime.Composable
|
||||||
|
|
||||||
private fun stringEnumMapper(stringEnum: StringEnum): String {
|
private fun stringEnumMapper(stringEnum: Strings): String {
|
||||||
return when (stringEnum) {
|
return when (stringEnum) {
|
||||||
StringEnum.AddToSavedPosts -> "Add to saved posts"
|
Strings.AddToSavedPosts -> "Add to saved posts"
|
||||||
StringEnum.AppName -> "Claw"
|
Strings.AppName -> "Claw"
|
||||||
StringEnum.AvatarContentDescription -> "%1s's avatar"
|
Strings.AvatarContentDescription -> "%1s's avatar"
|
||||||
StringEnum.HottestPosts -> "Hottest"
|
Strings.ChangeSortingOrder -> "Change sorting order"
|
||||||
StringEnum.Loading -> "Loading posts…"
|
Strings.HottestPosts -> "Hottest"
|
||||||
StringEnum.NoSavedPost -> "You don't have any saved posts"
|
Strings.NoSavedPost -> "You don't have any saved posts"
|
||||||
StringEnum.OpenComments -> "Open comments"
|
Strings.OpenComments -> "Open comments"
|
||||||
StringEnum.RefreshPostsContentDescription -> "Refresh posts"
|
Strings.RefreshPostsContentDescription -> "Refresh posts"
|
||||||
StringEnum.RemoveFromSavedPosts -> "Remove from saved posts"
|
Strings.RemoveFromSavedPosts -> "Remove from saved posts"
|
||||||
StringEnum.SubmittedBy -> "submitted by %1s"
|
Strings.SavedPosts -> "Saved"
|
||||||
|
Strings.SubmittedBy -> "submitted by %1s"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Composable
|
@Composable
|
||||||
actual fun stringValue(enum: StringEnum): String {
|
actual fun Strings.get(): String {
|
||||||
return stringEnumMapper(enum)
|
return stringEnumMapper(this)
|
||||||
}
|
}
|
||||||
|
|
||||||
@Composable
|
@Composable
|
||||||
actual fun stringValue(enum: StringEnum, arg1: Any): String {
|
actual fun Strings.get(fmt: Any): String {
|
||||||
return stringEnumMapper(enum).format(arg1)
|
return stringEnumMapper(this).format(fmt)
|
||||||
}
|
|
||||||
|
|
||||||
@Composable
|
|
||||||
actual fun stringValue(enum: StringEnum, arg1: Any, arg2: Any): String {
|
|
||||||
return stringEnumMapper(enum).format(arg1, arg2)
|
|
||||||
}
|
|
||||||
|
|
||||||
@Composable
|
|
||||||
actual fun stringValue(enum: StringEnum, arg1: Any, arg2: Any, arg3: Any): String {
|
|
||||||
return stringEnumMapper(enum).format(arg1, arg2, arg3)
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue