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:
bors[bot] 2021-03-24 06:55:17 +00:00 committed by GitHub
commit f6d8d3c536
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
13 changed files with 67 additions and 99 deletions

View file

@ -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

View file

@ -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,

View file

@ -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() } },

View file

@ -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 {

View file

@ -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(),
) )
} }
} }

View file

@ -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(

View file

@ -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>

View file

@ -68,10 +68,14 @@ kotlin {
} }
android { android {
buildFeatures { androidResources = true } buildFeatures {
androidResources = true
}
sourceSets["main"].apply { sourceSets {
manifest.srcFile("src/androidMain/AndroidManifest.xml") named("main") {
res.srcDir("src/androidMain/res") manifest.srcFile("src/androidMain/AndroidManifest.xml")
res.srcDirs("src/androidMain/res")
}
} }
} }

View file

@ -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)
} }

View file

@ -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>

View file

@ -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

View file

@ -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,
; ;
} }

View file

@ -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)
} }