如何在 Jetpack Compose 中使用适当的 size/layout 显示垂直文本
How to show vertical text with proper size/layout in jetpack compose
如何在 Jetpack Compose 中正确旋转文本并使其进行正确的布局?当我在文本对象上使用 rotate
修饰符时,文本会旋转,但布局中占用的大小似乎使用的是旋转前的文本宽度。
这是我想要完成的一个简单示例 - 垂直文本应沿左侧狭窄 space:
@Composable
fun MainBox() {
Column(
modifier = Modifier.fillMaxSize(),
horizontalAlignment = Alignment.CenterHorizontally
) {
Text(
modifier = Modifier.padding(4.dp).background(Color.Gray),
text = "This is at the top"
)
Row(
modifier = Modifier.fillMaxWidth().weight(1f),
verticalAlignment = Alignment.CenterVertically
) {
Text(
modifier = Modifier.padding(4.dp).rotate(-90f),
text = "This should be vertical text on the left side"
)
Text(
modifier = Modifier.fillMaxSize().background(Color.Yellow),
textAlign = TextAlign.Center,
text = "This is a big yellow box that should take up most of the space"
)
}
}
}
然而,这显示的是这个。
如果我缩短垂直文本中的文本,它只会占用一个狭窄的 space,这看起来更像我想要的布局。
有没有办法拦截布局过程或其他一些设置来固定大小,以便垂直文本只占用一个文本行的水平宽度 space,但仍适应用户字体尺寸变化(因此没有固定尺寸)?
类似问题的回答 and here没有解决这个问题或不再有效。
我的版本。经过几次测试,它似乎工作得很好
class ComposeActivity7 : ComponentActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContent {
ComposeTutorialTheme {
Column(
modifier = Modifier.fillMaxSize(),
horizontalAlignment = Alignment.CenterHorizontally
) {
Text(
modifier = Modifier
.padding(4.dp)
.background(Color.Gray),
text = "This is at the top"
)
Row(
modifier = Modifier
.fillMaxWidth()
.weight(1f),
verticalAlignment = Alignment.CenterVertically
) {
Text(
modifier = Modifier
.vertical()
.rotate(-90f)
.background(Color.Gray)
.padding(4.dp),
text = "This should be vertical text on the left side"
)
Text(
modifier = Modifier
.fillMaxSize()
.background(Color.Yellow),
textAlign = TextAlign.Center,
text = "This is a big yellow box that should take up most of the space"
)
}
}
}
}
}
}
fun Modifier.vertical() =
layout { measurable, constraints ->
val placeable = measurable.measure(constraints)
layout(placeable.height, placeable.width) {
placeable.place(
x = -(placeable.width / 2 - placeable.height / 2),
y = -(placeable.height / 2 - placeable.width / 2)
)
}
}
结果
Sergei S 的略微改进变体,适用于具有 Modifier.high(Intrinsic.X)
:
的父级
fun Modifier.rotateVertically(rotation: VerticalRotation) = then(
object : LayoutModifier {
override fun MeasureScope.measure(measurable: Measurable, constraints: Constraints): MeasureResult {
val placeable = measurable.measure(constraints)
return layout(placeable.height, placeable.width) {
placeable.place(
x = -(placeable.width / 2 - placeable.height / 2),
y = -(placeable.height / 2 - placeable.width / 2)
)
}
}
override fun IntrinsicMeasureScope.minIntrinsicHeight(measurable: IntrinsicMeasurable, width: Int): Int {
return measurable.maxIntrinsicWidth(width)
}
override fun IntrinsicMeasureScope.maxIntrinsicHeight(measurable: IntrinsicMeasurable, width: Int): Int {
return measurable.maxIntrinsicWidth(width)
}
override fun IntrinsicMeasureScope.minIntrinsicWidth(measurable: IntrinsicMeasurable, height: Int): Int {
return measurable.minIntrinsicHeight(height)
}
override fun IntrinsicMeasureScope.maxIntrinsicWidth(measurable: IntrinsicMeasurable, height: Int): Int {
return measurable.maxIntrinsicHeight(height)
}
})
.then(rotate(rotation.value))
enum class VerticalRotation(val value: Float) {
CLOCKWISE(90f), COUNTER_CLOCKWISE(270f)
}
如何在 Jetpack Compose 中正确旋转文本并使其进行正确的布局?当我在文本对象上使用 rotate
修饰符时,文本会旋转,但布局中占用的大小似乎使用的是旋转前的文本宽度。
这是我想要完成的一个简单示例 - 垂直文本应沿左侧狭窄 space:
@Composable
fun MainBox() {
Column(
modifier = Modifier.fillMaxSize(),
horizontalAlignment = Alignment.CenterHorizontally
) {
Text(
modifier = Modifier.padding(4.dp).background(Color.Gray),
text = "This is at the top"
)
Row(
modifier = Modifier.fillMaxWidth().weight(1f),
verticalAlignment = Alignment.CenterVertically
) {
Text(
modifier = Modifier.padding(4.dp).rotate(-90f),
text = "This should be vertical text on the left side"
)
Text(
modifier = Modifier.fillMaxSize().background(Color.Yellow),
textAlign = TextAlign.Center,
text = "This is a big yellow box that should take up most of the space"
)
}
}
}
然而,这显示的是这个。
如果我缩短垂直文本中的文本,它只会占用一个狭窄的 space,这看起来更像我想要的布局。
有没有办法拦截布局过程或其他一些设置来固定大小,以便垂直文本只占用一个文本行的水平宽度 space,但仍适应用户字体尺寸变化(因此没有固定尺寸)?
类似问题的回答
我的版本。经过几次测试,它似乎工作得很好
class ComposeActivity7 : ComponentActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContent {
ComposeTutorialTheme {
Column(
modifier = Modifier.fillMaxSize(),
horizontalAlignment = Alignment.CenterHorizontally
) {
Text(
modifier = Modifier
.padding(4.dp)
.background(Color.Gray),
text = "This is at the top"
)
Row(
modifier = Modifier
.fillMaxWidth()
.weight(1f),
verticalAlignment = Alignment.CenterVertically
) {
Text(
modifier = Modifier
.vertical()
.rotate(-90f)
.background(Color.Gray)
.padding(4.dp),
text = "This should be vertical text on the left side"
)
Text(
modifier = Modifier
.fillMaxSize()
.background(Color.Yellow),
textAlign = TextAlign.Center,
text = "This is a big yellow box that should take up most of the space"
)
}
}
}
}
}
}
fun Modifier.vertical() =
layout { measurable, constraints ->
val placeable = measurable.measure(constraints)
layout(placeable.height, placeable.width) {
placeable.place(
x = -(placeable.width / 2 - placeable.height / 2),
y = -(placeable.height / 2 - placeable.width / 2)
)
}
}
结果
Sergei S 的略微改进变体,适用于具有 Modifier.high(Intrinsic.X)
:
fun Modifier.rotateVertically(rotation: VerticalRotation) = then(
object : LayoutModifier {
override fun MeasureScope.measure(measurable: Measurable, constraints: Constraints): MeasureResult {
val placeable = measurable.measure(constraints)
return layout(placeable.height, placeable.width) {
placeable.place(
x = -(placeable.width / 2 - placeable.height / 2),
y = -(placeable.height / 2 - placeable.width / 2)
)
}
}
override fun IntrinsicMeasureScope.minIntrinsicHeight(measurable: IntrinsicMeasurable, width: Int): Int {
return measurable.maxIntrinsicWidth(width)
}
override fun IntrinsicMeasureScope.maxIntrinsicHeight(measurable: IntrinsicMeasurable, width: Int): Int {
return measurable.maxIntrinsicWidth(width)
}
override fun IntrinsicMeasureScope.minIntrinsicWidth(measurable: IntrinsicMeasurable, height: Int): Int {
return measurable.minIntrinsicHeight(height)
}
override fun IntrinsicMeasureScope.maxIntrinsicWidth(measurable: IntrinsicMeasurable, height: Int): Int {
return measurable.maxIntrinsicHeight(height)
}
})
.then(rotate(rotation.value))
enum class VerticalRotation(val value: Float) {
CLOCKWISE(90f), COUNTER_CLOCKWISE(270f)
}