android 如何在jetpack compose中的动态按钮上显示loader

uyto3xhc  于 2023-06-20  发布在  Android
关注(0)|答案(2)|浏览(177)

所以我面临的问题是我有一个可以有动态文本的按钮,按钮有一个状态isLoading,当它变成true时,View被重组为显示加载器。现在装载机有自己的宽度和高度。我如何管理装载机遵循相同的尺寸作为文本,这是前面所示。现在,当加载器出现时,按钮大小变大,加载完成后,isLoading再次为false,按钮将达到其实际大小,其中包含文本。
编辑1:@thracian建议的方法工作正常,除了一个奇怪的问题,即按钮获得一些预定义的高度(因此在计算视图高度时导致相同的值),这使得按钮看起来比我计划实现的要高。为ref附加代码,我如何使这个按钮跟随 Package 的内容&同时在视图大小中获得相同的计算值。

  1. @Composable
  2. fun GreenCTA(
  3. modifier: Modifier = Modifier, onClick: () -> Unit, title: String, isShowLoader: Boolean
  4. ) {
  5. var buttonSize by remember {
  6. mutableStateOf(DpSize.Zero)
  7. }
  8. val density = LocalDensity.current
  9. Button(
  10. modifier = modifier
  11. .then(if (buttonSize != DpSize.Zero) modifier.size(buttonSize) else modifier)
  12. .onSizeChanged { newSize ->
  13. if (buttonSize == DpSize.Zero) {
  14. buttonSize = with(density) {
  15. newSize
  16. .toSize()
  17. .toDpSize()
  18. }
  19. }
  20. },
  21. onClick = { onClick.invoke() },
  22. border = BorderStroke(1.dp, color = colorResource(id = R.color.pass_color_c2e6cd)),
  23. colors = ButtonDefaults.buttonColors(backgroundColor = colorResource(id = R.color.pass_color_e3f6ec)),
  24. shape = RoundedCornerShape(6.dp),
  25. contentPadding = PaddingValues(horizontal = 12.dp, vertical = 8.dp),
  26. elevation = ButtonDefaults.elevation(0.dp),
  27. ) {
  28. if (isShowLoader) BlueLoader(
  29. modifier = Modifier
  30. .fillMaxHeight()
  31. .aspectRatio(3f)
  32. )
  33. else {
  34. Row(
  35. verticalAlignment = Alignment.CenterVertically,
  36. horizontalArrangement = Arrangement.spacedBy(8.dp)
  37. ) {
  38. Image(
  39. painter = painterResource(id = R.drawable.ic_add_tag),
  40. contentDescription = "Button Logo"
  41. )
  42. Text(
  43. text = title,
  44. style = MaterialTheme.typography.body2.copy(lineHeight = 20.sp),
  45. fontWeight = FontWeight.SemiBold,
  46. )
  47. }
  48. }
  49. }
  50. }
h7appiyu

h7appiyu1#

如果您希望根据最初合成的文本调整按钮的大小,您可以在显示文本时使用Modifier.onSizeChanged作为按钮尺寸,然后在文本未合成时将这些尺寸设置为按钮的大小,例如

  1. Preview
  2. @Composable
  3. private fun Test() {
  4. var isLoading by remember { mutableStateOf(false) }
  5. var buttonSize by remember { mutableStateOf(DpSize.Zero) }
  6. val density = LocalDensity.current
  7. OutlinedButton(
  8. modifier = Modifier
  9. .then(
  10. if (buttonSize != DpSize.Zero) Modifier.size(buttonSize) else Modifier
  11. )
  12. .onSizeChanged { newSize ->
  13. if (buttonSize == DpSize.Zero) {
  14. buttonSize = with(density) {
  15. newSize
  16. .toSize()
  17. .toDpSize()
  18. }
  19. }
  20. },
  21. onClick = { isLoading = isLoading.not() }) {
  22. if (isLoading) {
  23. CircularProgressIndicator(
  24. modifier = Modifier
  25. .fillMaxHeight()
  26. .aspectRatio(1f)
  27. )
  28. } else {
  29. Text(text = "Click me")
  30. }
  31. }
  32. }

另外,如果你想调整一个Button的大小到一个还没有组合的Composable,比如加载器的大小,你可以使用SubcomposeLayout来预先获得尺寸,就像下面的答案一样
How to adjust size of component to it's child and remain unchanged when it's child size will change? (Jetpack Compose)

展开查看全部
niknxzdl

niknxzdl2#

您可以使用下面的自定义组合来实现您的要求:

  1. @Composable
  2. fun ButtonWithLoader(
  3. buttonText: String,
  4. backgroundColor: Color,
  5. contentColor: Color = White,
  6. showLoader: Boolean = false,
  7. showBorder: Boolean,
  8. onClick: () -> Unit,
  9. ) {
  10. if (showLoader) {
  11. Row(
  12. modifier = Modifier
  13. .widthIn(max = 600.dp)
  14. .fillMaxWidth()
  15. .padding(start = 20.dp, end = 20.dp, top = 20.dp)
  16. .height(48.dp)
  17. .motionClickEvent { onClick() }
  18. .border(
  19. if (showBorder) 1.dp else 0.dp,
  20. colors.textPrimary.copy(0.08f),
  21. RoundedCornerShape(50.dp)
  22. )
  23. .background(backgroundColor, RoundedCornerShape(50.dp)),
  24. horizontalArrangement = Arrangement.Center, verticalAlignment = Alignment.CenterVertically
  25. ) {
  26. CircularProgressIndicator(color = colors.primary, modifier = Modifier.size(28.dp))
  27. }
  28. } else {
  29. Row(
  30. modifier = Modifier
  31. .widthIn(max = 600.dp)
  32. .fillMaxWidth()
  33. .padding(start = 20.dp, end = 20.dp, top = 20.dp)
  34. .height(48.dp)
  35. .motionClickEvent { onClick() } // motionClickEvent is my custom click modifier, use clickable modifier over here
  36. .border(
  37. if (showBorder) 1.dp else 0.dp,
  38. colors.textPrimary.copy(0.08f),
  39. RoundedCornerShape(50.dp)
  40. )
  41. .background(backgroundColor, RoundedCornerShape(50.dp)),
  42. horizontalArrangement = Arrangement.Center, verticalAlignment = Alignment.CenterVertically
  43. ) {
  44. Text(
  45. text = buttonText,
  46. style = AppTheme.typography.buttonStyle,
  47. color = contentColor,
  48. modifier = Modifier.padding(horizontal = 12.dp)
  49. )
  50. }
  51. }
  52. }

用途:

  1. var isGoogleLoginInProgress by remember { mutableStateOf(false) }
  2. ButtonWithLoader(
  3. buttonText = stringResource(R.string.sign_in_method_sign_in_with_google_btn_text),
  4. backgroundColor = White,
  5. contentColor = textColor.copy(0.87f),
  6. showBorder = true,
  7. showLoader = isGoogleLoginInProgress
  8. ) {
  9. isGoogleLoginInProgress = !isGoogleLoginInProgress // Replace this with your logic
  10. }

样本输出:

展开查看全部

相关问题