No Compose, é possível encadear vários modificadores para mudar a aparência de um elemento combinável. Essas cadeias de modificadores podem afetar as restrições transmitidas aos elementos combináveis, que definem os limites de largura e altura.
Esta página descreve como os modificadores encadeados afetam as restrições e, por sua vez, a medição e o posicionamento dos elementos combináveis.
Modificadores na árvore de IU
Para entender como os modificadores se influenciam, é útil visualizar como eles aparecem na árvore de IU, que é gerada durante a fase de composição. Para mais informações, consulte a seção Composição.
Na árvore de IU, é possível visualizar os modificadores como nós de wrapper para os nós de layout:
Adicionar mais de um modificador a um elemento combinável cria uma cadeia de modificadores. Ao encadear vários modificadores, cada nó de modificador envolve o restante da cadeia e o nó de layout dentro. Por exemplo, ao encadear um clip e um
size modificador, o nó de modificador clip envolve o nó de modificador size,
que envolve o nó de layout Image.
Na fase de layout, o algoritmo que percorre a árvore permanece o mesmo, mas cada nó de modificador também é visitado. Dessa forma, um modificador pode mudar os requisitos de tamanho e o posicionamento do modificador ou do nó de layout que ele envolve.
Como mostrado na Figura 2, a implementação dos elementos combináveis Image e Text consiste em uma cadeia de modificadores que envolvem um único nó de layout.
As implementações de Row e Column são nós de layout que descrevem como dispor os filhos.
Para resumir:
- Os modificadores envolvem um único modificador ou nó de layout.
- Os nós de layout podem dispor vários nós filhos.
As seções a seguir descrevem como usar esse modelo mental para raciocinar sobre o encadeamento de modificadores e como ele influencia o tamanho dos elementos combináveis.
Restrições na fase de layout
A fase de layout segue um algoritmo de três etapas para encontrar a largura, a altura e a coordenada x, y de cada nó de layout:
- Medir filhos: um nó mede os filhos, se houver.
- Decidir o próprio tamanho: com base nessas medições, um nó decide o próprio tamanho.
- Posicionar filhos: cada nó filho é posicionado em relação à posição de um nó.
Constraints ajudam a encontrar os tamanhos certos para os nós durante as duas primeiras
etapas do algoritmo. As restrições definem os limites mínimo e máximo para a largura e a altura de um nó. Quando o nó decide o tamanho, o tamanho medido precisa estar dentro desse intervalo.
Tipos de restrições
Uma restrição pode ser uma das seguintes:
- Limitada: o nó tem uma largura e altura máximas e mínimas.
- Ilimitada: o nó não está restrito a nenhum tamanho. Os limites máximos de largura e altura são definidos como infinitos.
- Exata: o nó precisa seguir um requisito de tamanho exato. Os limites mínimo e máximo são definidos com o mesmo valor.
- Combinação: o nó segue uma combinação dos tipos de restrição anteriores. Por exemplo, uma restrição pode limitar a largura, permitindo uma altura máxima ilimitada, ou definir uma largura exata, mas fornecer uma altura limitada.
A próxima seção descreve como essas restrições são transmitidas de um pai para um filho.
Como as restrições são transmitidas do pai para o filho
Durante a primeira etapa do algoritmo descrito em Restrições na fase de layout, as restrições são transmitidas do pai para o filho na árvore de IU.
Quando um nó pai mede os filhos, ele fornece essas restrições a cada filho para informar o tamanho permitido. Em seguida, ao decidir o próprio tamanho, ele também adere às restrições transmitidas pelos pais.
Em um nível alto, o algoritmo funciona da seguinte maneira:
- Para decidir o tamanho que realmente quer ocupar, o nó raiz na árvore de IU mede os filhos e encaminha as mesmas restrições para o primeiro filho.
- Se o filho for um modificador que não afeta a medição, ele encaminha as restrições para o próximo modificador. As restrições são transmitidas pela cadeia de modificadores como estão, a menos que um modificador que afete a medição seja alcançado. As restrições são redimensionadas de acordo.
- Quando um nó é alcançado e não tem filhos (chamado de "nó folha"), ele decide o tamanho com base nas restrições transmitidas e retorna esse tamanho resolvido para o pai.
- O pai adapta as restrições com base nas medições do filho e chama o próximo filho com essas restrições ajustadas.
- Depois que todos os filhos de um pai são medidos, o nó pai decide o próprio tamanho e comunica isso ao pai.
- Dessa forma, toda a árvore é percorrida em profundidade. Por fim, todos os nós decidem os tamanhos, e a etapa de medição é concluída.
Para um exemplo detalhado, assista ao Constraints and modifier order vídeo (link em inglês).
Modificadores que afetam as restrições
Na seção anterior, você aprendeu que alguns modificadores podem afetar o tamanho da restrição. As seções a seguir descrevem modificadores específicos que afetam as restrições.
O modificador size
O size modificador declara o tamanho preferencial do conteúdo.
Por exemplo, a árvore de IU a seguir precisa ser renderizada em um contêiner de 300dp por 200dp. As restrições são limitadas, permitindo larguras entre 100dp e 300dp e alturas entre 100dp e 200dp:
O modificador size adapta as restrições recebidas para corresponder ao valor transmitido a ele.
Neste exemplo, o valor é 150dp:
size ajustando as restrições para 150dp.Se a largura e a altura forem menores que o limite de restrição menor ou maiores que o limite de restrição maior, o modificador vai corresponder às restrições transmitidas o mais próximo possível, sem deixar de aderir às restrições transmitidas:
size aderindo à restrição transmitida o mais próximo possível.O encadeamento de vários modificadores size não funciona. O primeiro modificador size define as restrições mínima e máxima para um valor fixo. Mesmo que o segundo modificador de tamanho solicite um tamanho menor ou maior, ele ainda precisa aderir aos limites exatos transmitidos. Portanto, ele não vai substituir esses valores:
size, em que o segundo valor transmitido (50dp) não substitui o primeiro valor (100dp).O modificador requiredSize
Use o modificador requiredSize em vez de size se precisar que o seu
nó substitua as restrições recebidas. O modificador requiredSize substitui as restrições recebidas e transmite o tamanho especificado como limites exatos.
Quando o tamanho é transmitido de volta para a árvore, o nó filho é centralizado no espaço disponível:
requiredSize substituindo as restrições recebidas do modificador size.Os modificadores width e height
O modificador size adapta a largura e a altura das restrições. Com o modificador width, é possível definir uma largura fixa, mas deixar a altura indefinida.
Da mesma forma, com o modificador height, é possível definir uma altura fixa, mas deixar a largura indefinida:
width e o modificador height definindo uma largura e altura fixas, respectivamente.O modificador sizeIn
O modificador sizeIn permite definir restrições mínimas e máximas exatas para largura e altura. Use o modificador sizeIn se precisar de um controle refinado sobre as restrições.
sizeIn com minWidth, maxWidth, minHeight e maxHeight definidos.Exemplos
Esta seção mostra e explica a saída de vários snippets de código com modificadores encadeados.
Image( painterResource(R.drawable.hero), contentDescription = null, Modifier .fillMaxSize() .size(50.dp) )
Este snippet produz a seguinte saída:
Image preenche o tamanho máximo como
resultado da cadeia de modificadores.- O modificador
fillMaxSizemuda as restrições para definir a largura e a altura mínimas para o valor máximo:300dpde largura e200dpde altura. - Embora o modificador
sizequeira usar um tamanho de50dp, ele ainda precisa aderir às restrições mínimas recebidas. Portanto, o modificadorsizetambém vai gerar os limites de restrição exatos de300por200, ignorando o valor fornecido no modificadorsize. - A
Imagesegue esses limites e informa um tamanho de300por200, que é transmitido por toda a árvore.
Image( painterResource(R.drawable.hero), contentDescription = null, Modifier .fillMaxSize() .wrapContentSize() .size(50.dp) )
Este snippet produz a seguinte saída:
Image é centralizado e dimensionado para
50dp.- O modificador
fillMaxSizeadapta as restrições para definir a largura e a altura mínimas para o valor máximo:300dpde largura e200dpde altura. - O modificador
wrapContentSizeredefine as restrições mínimas. Portanto, emborafillMaxSizetenha resultado em restrições fixas,wrapContentSizeredefine as restrições limitadas. O nó a seguir agora pode ocupar todo o espaço novamente ou ser menor que todo o espaço. - O modificador
sizedefine as restrições para limites mínimos e máximos de50. - A
Imageé resolvida para um tamanho de50por50, e o modificadorsizeencaminha isso. - O modificador
wrapContentSizetem uma propriedade especial. Ele pega o filho e o coloca no centro dos limites mínimos disponíveis que foram transmitidos a ele. O tamanho que ele comunica aos pais é igual aos limites mínimos que foram transmitidos a ele.
Ao combinar apenas três modificadores, é possível definir um tamanho para o elemento combinável e centralizá-lo no pai.
Image( painterResource(R.drawable.hero), contentDescription = null, Modifier .clip(CircleShape) .padding(10.dp) .size(100.dp) )
Este snippet produz a seguinte saída:
- O modificador
clipnão muda as restrições. - O modificador
paddingdiminui as restrições máximas. - O modificador
sizedefine todas as restrições como100dp. - A
Imageadere a essas restrições e informa um tamanho de100dppor100dp. - O modificador
paddingadiciona10dpem todos os lados ao tamanho informado pelaImage. Portanto, o layout com padding informa uma largura e altura de120dp. - Agora, na fase de desenho, o modificador
clipatua em uma tela de120dppor120dp. Ele cria uma máscara de círculo desse tamanho. - O modificador
paddinginsere o conteúdo em10dpem todos os lados, o que diminui o tamanho da tela para aImagepara100dppor100dp. - A
Imageé desenhada nessa tela menor. A imagem é cortada com base no círculo original de120dp. Portanto, a saída é um resultado não redondo.