Compare commits

...

2 Commits

Author SHA1 Message Date
junkfood
2655cb5f43
feat(ui): predictive back animation for Android 14+ 2024-11-13 22:39:59 +08:00
junkfood
c691a106d5
build: update deps & bump targetSdk to 34 2024-11-13 22:39:58 +08:00
5 changed files with 70 additions and 20 deletions

View File

@ -31,7 +31,7 @@ android {
defaultConfig { defaultConfig {
applicationId = "me.ash.reader" applicationId = "me.ash.reader"
minSdk = 26 minSdk = 26
targetSdk = 33 targetSdk = 34
versionCode = 26 versionCode = 26
versionName = "0.11.0" versionName = "0.11.0"

View File

@ -4,8 +4,10 @@
<uses-permission android:name="android.permission.INTERNET" /> <uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.POST_NOTIFICATIONS" /> <uses-permission android:name="android.permission.POST_NOTIFICATIONS" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" <uses-permission
android:name="android.permission.WRITE_EXTERNAL_STORAGE"
android:maxSdkVersion="28" /> android:maxSdkVersion="28" />
<queries> <queries>
<intent> <intent>
<action android:name="android.intent.action.MAIN" /> <action android:name="android.intent.action.MAIN" />
@ -19,6 +21,7 @@
<application <application
android:name=".infrastructure.android.AndroidApp" android:name=".infrastructure.android.AndroidApp"
android:allowBackup="true" android:allowBackup="true"
android:enableOnBackInvokedCallback="true"
android:icon="@mipmap/ic_launcher" android:icon="@mipmap/ic_launcher"
android:label="@string/read_you" android:label="@string/read_you"
android:roundIcon="@mipmap/ic_launcher_round" android:roundIcon="@mipmap/ic_launcher_round"

View File

@ -8,38 +8,88 @@
package me.ash.reader.ui.ext package me.ash.reader.ui.ext
import android.os.Build
import androidx.compose.animation.* import androidx.compose.animation.*
import androidx.compose.animation.core.tween
import androidx.compose.runtime.Composable import androidx.compose.runtime.Composable
import androidx.navigation.NamedNavArgument import androidx.navigation.NamedNavArgument
import androidx.navigation.NavBackStackEntry import androidx.navigation.NavBackStackEntry
import androidx.navigation.NavDeepLink import androidx.navigation.NavDeepLink
import androidx.navigation.NavGraphBuilder import androidx.navigation.NavGraphBuilder
import androidx.navigation.compose.composable import androidx.navigation.compose.composable
import me.ash.reader.ui.motion.EmphasizedAccelerate
import me.ash.reader.ui.motion.EmphasizedDecelerate
import me.ash.reader.ui.motion.materialSharedAxisXIn import me.ash.reader.ui.motion.materialSharedAxisXIn
import me.ash.reader.ui.motion.materialSharedAxisXOut import me.ash.reader.ui.motion.materialSharedAxisXOut
private const val INITIAL_OFFSET_FACTOR = 0.10f private const val INITIAL_OFFSET_FACTOR = 0.10f
private const val INITIAL_SCALE_FACTOR = 0.8f
fun NavGraphBuilder.animatedComposable( fun NavGraphBuilder.animatedComposable(
route: String, route: String,
arguments: List<NamedNavArgument> = emptyList(), arguments: List<NamedNavArgument> = emptyList(),
deepLinks: List<NavDeepLink> = emptyList(), deepLinks: List<NavDeepLink> = emptyList(),
content: @Composable AnimatedVisibilityScope.(NavBackStackEntry) -> Unit usePredictiveBack: Boolean = Build.VERSION.SDK_INT >= 34,
) = composable( content: @Composable AnimatedVisibilityScope.(NavBackStackEntry) -> Unit,
) {
if (usePredictiveBack) {
animatedComposablePredictiveBack(route, arguments, deepLinks, content)
} else {
animatedComposableLegacy(route, arguments, deepLinks, content)
}
}
fun NavGraphBuilder.animatedComposablePredictiveBack(
route: String,
arguments: List<NamedNavArgument> = emptyList(),
deepLinks: List<NavDeepLink> = emptyList(),
content: @Composable AnimatedVisibilityScope.(NavBackStackEntry) -> Unit,
) =
composable(
route = route, route = route,
arguments = arguments, arguments = arguments,
deepLinks = deepLinks, deepLinks = deepLinks,
enterTransition = { enterTransition = { materialSharedAxisXIn(initialOffsetX = { (it * INITIAL_OFFSET_FACTOR).toInt() }) },
materialSharedAxisXIn(initialOffsetX = { (it * INITIAL_OFFSET_FACTOR).toInt() })
},
exitTransition = { exitTransition = {
materialSharedAxisXOut(targetOffsetX = { -(it * INITIAL_OFFSET_FACTOR).toInt() }) materialSharedAxisXOut(targetOffsetX = { -(it * INITIAL_OFFSET_FACTOR).toInt() })
}, },
popEnterTransition = { popEnterTransition = {
materialSharedAxisXIn(initialOffsetX = { -(it * INITIAL_OFFSET_FACTOR).toInt() }) scaleIn(
animationSpec = tween(durationMillis = 350, easing = EmphasizedDecelerate),
initialScale = INITIAL_SCALE_FACTOR,
) + materialSharedAxisXIn(initialOffsetX = { -(it * INITIAL_OFFSET_FACTOR).toInt() })
}, },
popExitTransition = { popExitTransition = {
materialSharedAxisXOut(targetOffsetX = { (it * INITIAL_OFFSET_FACTOR).toInt() }) materialSharedAxisXOut(targetOffsetX = { (it * INITIAL_OFFSET_FACTOR).toInt() }) +
scaleOut(
targetScale = INITIAL_SCALE_FACTOR,
animationSpec = tween(durationMillis = 350, easing = EmphasizedAccelerate),
)
}, },
content = content content = content,
)
fun NavGraphBuilder.animatedComposableLegacy(
route: String,
arguments: List<NamedNavArgument> = emptyList(),
deepLinks: List<NavDeepLink> = emptyList(),
content: @Composable AnimatedVisibilityScope.(NavBackStackEntry) -> Unit
) = composable(
route = route,
arguments = arguments,
deepLinks = deepLinks,
enterTransition = {
materialSharedAxisXIn(initialOffsetX = { (it * INITIAL_OFFSET_FACTOR).toInt() })
},
exitTransition = {
materialSharedAxisXOut(targetOffsetX = { -(it * INITIAL_OFFSET_FACTOR).toInt() })
},
popEnterTransition = {
materialSharedAxisXIn(initialOffsetX = { -(it * INITIAL_OFFSET_FACTOR).toInt() })
},
popExitTransition = {
materialSharedAxisXOut(targetOffsetX = { (it * INITIAL_OFFSET_FACTOR).toInt() })
},
content = content
) )

View File

@ -204,9 +204,6 @@ fun FeedsPage(
} }
} }
BackHandler(true) {
context.findActivity()?.moveTaskToBack(false)
}
RYScaffold( RYScaffold(
topBarTonalElevation = topBarTonalElevation.value.dp, topBarTonalElevation = topBarTonalElevation.value.dp,

View File

@ -13,21 +13,21 @@ aboutLibsRelease = "11.1.1"
accompanist = "0.34.0" accompanist = "0.34.0"
# AndroidX # AndroidX
activityCompose = "1.9.0" activityCompose = "1.9.3"
appcompat = "1.7.0" appcompat = "1.7.0"
browser = "1.8.0" browser = "1.8.0"
coreKtx = "1.13.1" coreKtx = "1.13.1"
datastore = "1.1.1" datastore = "1.1.1"
espresso = "3.5.1" espresso = "3.5.1"
lifecycle = "2.8.1" lifecycle = "2.8.1"
navigation = "2.7.7" navigation = "2.8.3"
paging = "3.3.0" paging = "3.3.0"
profileinstaller = "1.3.1" profileinstaller = "1.4.1"
room = "2.6.1" room = "2.6.1"
work = "2.9.0" work = "2.9.0"
# Compose # Compose
composeBom = "2024.09.02" composeBom = "2024.10.01"
composeCompiler = "1.5.8" composeCompiler = "1.5.8"
composeHtml = "1.0.2" composeHtml = "1.0.2"