fix(ui): menu position jitters when animating content height

This commit is contained in:
junkfood 2024-04-28 16:24:50 +08:00
parent 1199c6850b
commit 1bf597d32e
No known key found for this signature in database
GPG Key ID: 2EA5B648DB112A34
2 changed files with 28 additions and 21 deletions

View File

@ -1,5 +1,6 @@
package me.ash.reader.ui.component.menu package me.ash.reader.ui.component.menu
import android.util.Log
import androidx.compose.animation.AnimatedVisibility import androidx.compose.animation.AnimatedVisibility
import androidx.compose.animation.EnterTransition import androidx.compose.animation.EnterTransition
import androidx.compose.animation.core.LinearEasing import androidx.compose.animation.core.LinearEasing
@ -26,7 +27,6 @@ import androidx.compose.runtime.Stable
import androidx.compose.ui.AbsoluteAlignment import androidx.compose.ui.AbsoluteAlignment
import androidx.compose.ui.Alignment import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier import androidx.compose.ui.Modifier
import androidx.compose.ui.graphics.TransformOrigin
import androidx.compose.ui.unit.Density import androidx.compose.ui.unit.Density
import androidx.compose.ui.unit.DpOffset import androidx.compose.ui.unit.DpOffset
import androidx.compose.ui.unit.IntOffset import androidx.compose.ui.unit.IntOffset
@ -38,12 +38,14 @@ import androidx.compose.ui.util.fastFirstOrNull
import androidx.compose.ui.util.fastMap import androidx.compose.ui.util.fastMap
import androidx.compose.ui.window.PopupPositionProvider import androidx.compose.ui.window.PopupPositionProvider
import me.ash.reader.domain.model.constant.ElevationTokens import me.ash.reader.domain.model.constant.ElevationTokens
import me.ash.reader.ui.component.menu.MenuPosition.Horizontal
import me.ash.reader.ui.component.menu.MenuPosition.Vertical
import me.ash.reader.ui.motion.EmphasizedAccelerate import me.ash.reader.ui.motion.EmphasizedAccelerate
import me.ash.reader.ui.motion.EmphasizedDecelerate import me.ash.reader.ui.motion.EmphasizedDecelerate
import me.ash.reader.ui.motion.EnterDuration import me.ash.reader.ui.motion.EnterDuration
import me.ash.reader.ui.motion.ExitDuration import me.ash.reader.ui.motion.ExitDuration
import kotlin.math.max
import kotlin.math.min private const val TAG = "DropdownMenuImpl"
/** /**
* Interfaces for positioning a menu within a window. This is the same purpose as the interface * Interfaces for positioning a menu within a window. This is the same purpose as the interface
@ -385,25 +387,32 @@ internal data class DropdownMenuPositionProvider(
it >= 0 && it + popupContentSize.width <= windowSize.width it >= 0 && it + popupContentSize.width <= windowSize.width
} ?: xCandidates.last() } ?: xCandidates.last()
val yCandidates = listOf( /* val yCandidates = listOf(
topToAnchorBottom, topToAnchorBottom,
bottomToAnchorTop, bottomToAnchorTop,
centerToAnchorTop, centerToAnchorTop,
if (anchorBounds.center.y < windowSize.height / 2) { if (anchorBounds.center.y < windowSize.height / 2) {
topToWindowTop topToWindowTop
} else { } else {
bottomToWindowBottom bottomToWindowBottom
} }
).fastMap { ).fastMap {
it.position( it.position(
anchorBounds = anchorBounds,
windowSize = windowSize,
menuHeight = popupContentSize.height
)
}
val y = yCandidates.fastFirstOrNull {
it >= verticalMargin && it + popupContentSize.height <= windowSize.height - verticalMargin
} ?: yCandidates.last()*/
val y =
(if (anchorBounds.top < windowSize.height / 2) topToAnchorBottom else bottomToAnchorTop).position(
anchorBounds = anchorBounds, anchorBounds = anchorBounds,
windowSize = windowSize, windowSize = windowSize,
menuHeight = popupContentSize.height menuHeight = popupContentSize.height
) )
}
val y = yCandidates.fastFirstOrNull {
it >= verticalMargin && it + popupContentSize.height <= windowSize.height - verticalMargin
} ?: yCandidates.last()
val menuOffset = IntOffset(x, y) val menuOffset = IntOffset(x, y)
onPositionCalculated(/* anchorBounds = */anchorBounds,/* menuBounds = */ onPositionCalculated(/* anchorBounds = */anchorBounds,/* menuBounds = */

View File

@ -377,9 +377,7 @@ fun SwipeableArticleItem(
expanded = expanded, expanded = expanded,
onDismissRequest = { expanded = false }, onDismissRequest = { expanded = false },
offset = density.run { offset = density.run {
if (LocalLayoutDirection.current == LayoutDirection.Ltr) DpOffset(menuOffset.x.toDp(), 0.dp)
DpOffset(menuOffset.x.toDp(), 0.dp)
else DpOffset(0.dp, 0.dp)
}, },
) { ) {
ArticleItemMenuContent( ArticleItemMenuContent(