AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • Início
  • system&network
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • Início
  • system&network
    • Recentes
    • Highest score
    • tags
  • Ubuntu
    • Recentes
    • Highest score
    • tags
  • Unix
    • Recentes
    • tags
  • DBA
    • Recentes
    • tags
  • Computer
    • Recentes
    • tags
  • Coding
    • Recentes
    • tags
Início / coding / Perguntas / 79072449
Accepted
BobDoolittle
BobDoolittle
Asked: 2024-10-10 09:04:23 +0800 CST2024-10-10 09:04:23 +0800 CST 2024-10-10 09:04:23 +0800 CST

Exibindo uma roda de progresso animada no Jetpack Compose

  • 772

Neste Google Codelab (Mars Photos), enquanto o Coil carrega imagens, o aplicativo exibe uma roda de progresso. O código usa este arquivo XML como um drawable:

insira a descrição da imagem aqui

e usa este código para exibi-lo:

@Composable
fun LoadingScreen(modifier: Modifier = Modifier) {
    Box(
        contentAlignment = Alignment.Center,
        modifier = modifier.fillMaxSize()
    ) {
        Image(
            modifier = Modifier.size(200.dp),
            painter = painterResource(R.drawable.loading_img),
            contentDescription = stringResource(R.string.loading)
        )
    }
}

No entanto, isso só exibe uma imagem estática. Quero que ela gire (o que eu certamente esperava como comportamento). Pensei em algumas abordagens, nenhuma das quais eu gosto muito:

  • Defina 12 arquivos XML semelhantes, um para cada grau de rotação, e exiba cada um continuamente em sequência
  • Gere dinamicamente o XML e de alguma forma o converta em um pintor na hora
  • Use Coil para carregar um GIF

Existe uma maneira mais simples de realizar essa tarefa? Existe uma maneira de rotacionar a imagem em si programaticamente, como aplicar um modo de animação predefinido de algum tipo (talvez semelhante ao basicMarquee)?

android-jetpack-compose
  • 1 1 respostas
  • 39 Views

1 respostas

  • Voted
  1. Best Answer
    Raghunandan
    2024-10-10T12:26:59+08:002024-10-10T12:26:59+08:00

    Aqui está o código para o mesmo. Este código foi retirado de https://github.com/SmartToolFactory/Compose-ProgressIndicator e tem muitas outras animações de visualização de progresso circular girando.

    @Composable
    fun SpinningProgressIndicator(
        modifier: Modifier = Modifier,
        @androidx.annotation.IntRange(from = 4, to = 12) staticItemCount: Int = 12,
        dynamicItemCount: Int = staticItemCount / 2,
        staticItemColor: Color = StaticItemColor,
        dynamicItemColor: Color = DynamicItemColor,
        spinnerShape: SpinnerShape = SpinnerShape.RoundedRect,
        durationMillis: Int = 1000
    ) {
    
        // Number of rotating items
        val animatedItemCount = dynamicItemCount.coerceIn(1, staticItemCount)
    
        val coefficient = 360f / staticItemCount
    
        val infiniteTransition = rememberInfiniteTransition()
        val angle by infiniteTransition.animateFloat(
            initialValue = 0f,
            targetValue = staticItemCount.toFloat(),
            animationSpec = infiniteRepeatable(
                animation = tween(
                    durationMillis = durationMillis,
                    easing = LinearEasing
                ),
                repeatMode = RepeatMode.Restart
            )
        )
    
        Canvas(modifier = modifier
            .progressSemantics()
            .size(Size)
        ) {
    
            var canvasWidth = size.width
            var canvasHeight = size.height
    
            if (canvasHeight < canvasWidth) {
                canvasWidth = canvasHeight
            } else {
                canvasHeight = canvasWidth
            }
    
            val itemWidth = canvasWidth * .3f
            val itemHeight = canvasHeight / staticItemCount
    
            val cornerRadius = itemWidth.coerceAtMost(itemHeight) / 2
    
            val horizontalOffset = (size.width - size.height).coerceAtLeast(0f) / 2
            val verticalOffset = (size.height - size.width).coerceAtLeast(0f) / 2
    
            val topLeftOffset = Offset(
                x = horizontalOffset + canvasWidth - itemWidth,
                y = verticalOffset + (canvasHeight - itemHeight) / 2
            )
    
            val size = Size(itemWidth, itemHeight)
    
            // Stationary items
            for (i in 0..360 step 360 / staticItemCount) {
                rotate(i.toFloat()) {
                        drawRect(
                            color = staticItemColor,
                            topLeft = topLeftOffset,
                            size = size,
                        )
                }
            }
    
            // Dynamic items
            for (i in 0..animatedItemCount) {
                // angle is cast to into move in intervals of static items
                rotate((angle.toInt() + i) * coefficient) {
                        drawRect(
                            color = dynamicItemColor.copy(
                                alpha = (0.2f + 0.15f * i).coerceIn(
                                    0f, 1f
                                )
                            ),
                            topLeft = topLeftOffset,
                            size = size,
                        )
                }
            }
        }
    }
    

    val infiniteTransition = rememberInfiniteTransition()é usado para animação de transição infinita e RepeatMode.Restart significa que a animação reinicia, dando uma sensação de animação infinita.

    rotate((angle.toInt() + i) * coefficient) {onde você gira os itens. Todos os cálculos de ângulos são em radianos.

    Você pode personalizar isso conforme suas necessidades. Você não precisa de uma imagem. Desenhe linhas ou retângulos e anime alguns deles girando-os.

    • 3

relate perguntas

  • Desenhando fora do limite do contêiner no jetpack compose

  • Por que o Jetpack Compose está usando o estado antigo no evento onTap

  • Aplicar efeito de pincel a uma palavra inteira enquanto anima cada letra individualmente no Jetpack Compose

  • Problema com o Canvas no Jetpack Compose que não limpa o conteúdo desenhado ao definir StateFlow como nulo

  • Como fazer notas de piano no Jetpack Compose?

Sidebar

Stats

  • Perguntas 205573
  • respostas 270741
  • best respostas 135370
  • utilizador 68524
  • Highest score
  • respostas
  • Marko Smith

    Vue 3: Erro na criação "Identificador esperado, mas encontrado 'import'" [duplicado]

    • 1 respostas
  • Marko Smith

    Por que esse código Java simples e pequeno roda 30x mais rápido em todas as JVMs Graal, mas não em nenhuma JVM Oracle?

    • 1 respostas
  • Marko Smith

    Qual é o propósito de `enum class` com um tipo subjacente especificado, mas sem enumeradores?

    • 1 respostas
  • Marko Smith

    Como faço para corrigir um erro MODULE_NOT_FOUND para um módulo que não importei manualmente?

    • 6 respostas
  • Marko Smith

    `(expression, lvalue) = rvalue` é uma atribuição válida em C ou C++? Por que alguns compiladores aceitam/rejeitam isso?

    • 3 respostas
  • Marko Smith

    Quando devo usar um std::inplace_vector em vez de um std::vector?

    • 3 respostas
  • Marko Smith

    Um programa vazio que não faz nada em C++ precisa de um heap de 204 KB, mas não em C

    • 1 respostas
  • Marko Smith

    PowerBI atualmente quebrado com BigQuery: problema de driver Simba com atualização do Windows

    • 2 respostas
  • Marko Smith

    AdMob: MobileAds.initialize() - "java.lang.Integer não pode ser convertido em java.lang.String" para alguns dispositivos

    • 1 respostas
  • Marko Smith

    Estou tentando fazer o jogo pacman usando apenas o módulo Turtle Random e Math

    • 1 respostas
  • Martin Hope
    Aleksandr Dubinsky Por que a correspondência de padrões com o switch no InetAddress falha com 'não cobre todos os valores de entrada possíveis'? 2024-12-23 06:56:21 +0800 CST
  • Martin Hope
    Phillip Borge Por que esse código Java simples e pequeno roda 30x mais rápido em todas as JVMs Graal, mas não em nenhuma JVM Oracle? 2024-12-12 20:46:46 +0800 CST
  • Martin Hope
    Oodini Qual é o propósito de `enum class` com um tipo subjacente especificado, mas sem enumeradores? 2024-12-12 06:27:11 +0800 CST
  • Martin Hope
    sleeptightAnsiC `(expression, lvalue) = rvalue` é uma atribuição válida em C ou C++? Por que alguns compiladores aceitam/rejeitam isso? 2024-11-09 07:18:53 +0800 CST
  • Martin Hope
    The Mad Gamer Quando devo usar um std::inplace_vector em vez de um std::vector? 2024-10-29 23:01:00 +0800 CST
  • Martin Hope
    Chad Feller O ponto e vírgula agora é opcional em condicionais bash com [[ .. ]] na versão 5.2? 2024-10-21 05:50:33 +0800 CST
  • Martin Hope
    Wrench Por que um traço duplo (--) faz com que esta cláusula MariaDB seja avaliada como verdadeira? 2024-05-05 13:37:20 +0800 CST
  • Martin Hope
    Waket Zheng Por que `dict(id=1, **{'id': 2})` às vezes gera `KeyError: 'id'` em vez de um TypeError? 2024-05-04 14:19:19 +0800 CST
  • Martin Hope
    user924 AdMob: MobileAds.initialize() - "java.lang.Integer não pode ser convertido em java.lang.String" para alguns dispositivos 2024-03-20 03:12:31 +0800 CST
  • Martin Hope
    MarkB Por que o GCC gera código que executa condicionalmente uma implementação SIMD? 2024-02-17 06:17:14 +0800 CST

Hot tag

python javascript c++ c# java typescript sql reactjs html

Explore

  • Início
  • Perguntas
    • Recentes
    • Highest score
  • tag
  • help

Footer

AskOverflow.Dev

About Us

  • About Us
  • Contact Us

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve