Comme la plupart des autres kits d'outils de l'interface utilisateur, Compose affiche un frame à l'aide de plusieurs phases distinctes. Le système de vues Android comporte trois phases principales : la mesure, la mise en page et le dessin. Compose est très similaire, mais inclut une phase supplémentaire clé appelée composition au début.
Celle-ci est décrite dans nos documents Compose, y compris dans les articles Raisonnement dans Compose et États et Jetpack Compose.
Les trois phases d'un frame
Compose compte trois phases principales :
- Composition : quels éléments d'interface utilisateur afficher. Compose exécute des fonctions modulables et crée une description de votre UI.
- Mise en page : où positionner les éléments d'interface utilisateur. Cette phase comprend deux étapes : la mesure et le positionnement. Les éléments de mise en page se mesurent et se placent eux-mêmes ainsi que tous les éléments enfants en coordonnées 2D, pour chaque nœud de l'arborescence de mise en page.
- Dessin : comment effectuer le rendu de l'interface utilisateur. Les éléments de l'interface utilisateur apparaissent dans un canevas, habituellement l'écran d'un d'appareil.
L'ordre de ces phases est généralement le même, ce qui permet aux données de circuler dans un sens (de la composition à la mise en page, en passant par le dessin) afin de générer un frame (également appelé flux de données unidirectionnel).
BoxWithConstraints
, LazyColumn
et LazyRow
constituent des exceptions notables, où la composition des enfants dépend de la phase de mise en page du parent.
Conceptuellement, chacune de ces phases se produit pour chaque frame. Toutefois, pour optimiser les performances, Compose évite de répéter une tâche qui calculerait les mêmes résultats à partir des mêmes entrées dans toutes ces phases. Compose ignore l'exécution d'une fonction modulable si un ancien résultat peut être réutilisé, et l'interface utilisateur de Compose ne recommence pas la mise en page ou le dessin de toute l'arborescence s'il n'y a pas lieu de le faire. Compose ne fournit que les efforts minimums nécessaires pour mettre à jour l'UI. Cette optimisation est possible, car Compose suit les lectures d'état au cours des différentes phases.
Comprendre les phases
Cette section décrit plus en détail comment les trois phases Compose sont exécutées pour les composables.
Composition
Au cours de la phase de composition, l'environnement d'exécution Compose exécute des fonctions modulables et génère une structure arborescente représentant votre UI. Cet arbre d'interface utilisateur se compose de nœuds de mise en page contenant toutes les informations nécessaires pour les phases suivantes, comme illustré dans la vidéo suivante:
Figure 2. Arborescence représentant votre UI créée lors de la phase de composition.
Une sous-section de l'arborescence du code et de l'UI se présente comme suit:
Dans ces exemples, chaque fonction modulable du code est mappée sur un seul nœud de mise en page dans l'arborescence de l'UI. Dans des exemples plus complexes, les composables peuvent contenir de la logique et un flux de contrôle, et produire un arbre différent en fonction des états.
Mise en page
Lors de la phase de mise en page, Compose utilise l'arborescence de l'UI produite lors de la phase de composition comme entrée. La collection de nœuds de mise en page contient toutes les informations nécessaires pour déterminer la taille et l'emplacement de chaque nœud dans l'espace 2D.
Figure 4. Mesure et positionnement de chaque nœud de mise en page dans l'arborescence de l'UI pendant la phase de mise en page.
Au cours de la phase de mise en page, l'arbre est parcouru à l'aide de l'algorithme en trois étapes suivant:
- Mesurer les enfants: un nœud mesure ses enfants, le cas échéant.
- Déterminer sa propre taille: sur la base de ces mesures, un nœud détermine sa propre taille.
- Placer les enfants: chaque nœud enfant est placé par rapport à la position de son propre nœud.
À la fin de cette phase, chaque nœud de mise en page contient:
- Une largeur et une hauteur attribuées
- Coordonnées X, Y où il doit être dessiné
Rappelez-vous l'arborescence de l'UI de la section précédente:
Pour cet arbre, l'algorithme fonctionne comme suit:
Row
mesure ses enfants,Image
etColumn
.- La
Image
est mesurée. Il n'a pas d'enfants. Il décide donc de sa propre taille et la renvoie àRow
. - L'
Column
est ensuite mesuré. Il mesure d'abord ses propres enfants (deux composablesText
). - Le premier
Text
est mesuré. Il n'a pas d'enfants. Il décide donc de sa propre taille et la renvoie àColumn
.- Le deuxième
Text
est mesuré. Il n'a pas d'enfants. Il décide donc de sa propre taille et la renvoie àColumn
.
- Le deuxième
Column
utilise les mesures des enfants pour déterminer sa propre taille. Il utilise la largeur maximale des enfants et la somme de la hauteur de ses enfants.Column
place ses enfants par rapport à lui-même, en les plaçant verticalement les uns sous les autres.Row
utilise les mesures des enfants pour déterminer sa propre taille. Il utilise la hauteur maximale des enfants et la somme des largeurs de ses enfants. Il place ensuite ses enfants.
Notez que chaque nœud n'a été visité qu'une seule fois. L'environnement d'exécution Compose ne nécessite qu'un seul passage dans l'arborescence de l'UI pour mesurer et placer tous les nœuds, ce qui améliore les performances. Lorsque le nombre de nœuds de l'arborescence augmente, le temps passé à la parcourir augmente de manière linéaire. En revanche, si chaque nœud a été visité plusieurs fois, la durée de la traversée augmente de manière exponentielle.
Dessin
Lors de la phase de dessin, l'arbre est à nouveau parcouru de haut en bas, et chaque nœud se dessine à l'écran à son tour.
Figure 5. La phase de dessin dessine les pixels à l'écran.
Dans l'exemple précédent, le contenu de l'arborescence est dessiné comme suit:
Row
dessine tout contenu qu'il peut avoir, comme une couleur d'arrière-plan.Image
s'affiche.Column
s'affiche.- Le premier et le deuxième
Text
s'affichent respectivement.
Figure 6. Arbre de l'UI et sa représentation dessinée.
Lectures d'état
Lorsque vous lisez la valeur d'un état d'instantané au cours de l'une des phases répertoriées ci-dessus, Compose suit automatiquement les actions effectuées lors de la lecture de cette valeur. Ce suivi permet à Compose de réexécuter le lecteur lorsque la valeur de l'état change. Il constitue la base de l'observabilité de l'état dans Compose.
L'état est généralement créé à l'aide de mutableStateOf()
, puis accessible de l'une des deux manières suivantes : directement via la propriété value
ou via un délégué de propriété Kotlin. Pour en savoir plus à ce sujet, consultez la section État dans les composables. Pour les besoins de ce guide, une "lecture d'état" fait référence à l'une ou l'autre de ces méthodes d'accès équivalentes.
// State read without property delegate. val paddingState: MutableState<Dp> = remember { mutableStateOf(8.dp) } Text( text = "Hello", modifier = Modifier.padding(paddingState.value) )
// State read with property delegate. var padding: Dp by remember { mutableStateOf(8.dp) } Text( text = "Hello", modifier = Modifier.padding(padding) )
En arrière-plan du délégué de propriété, les fonctions "getter" et "setter" permettent d'accéder à la valeur (value
) de l'état et de la mettre à jour. Ces fonctions ne sont appelées que lorsque vous référencez la propriété en tant que valeur, et non lorsqu'elle est créée. C'est pourquoi les deux méthodes ci-dessus sont équivalentes.
Chaque bloc de code pouvant être réexécuté lorsqu'un état de lecture est modifié correspond à un champ d'application de redémarrage. Compose effectue le suivi des changements de valeur d'état et redémarre les champs d'application en plusieurs phases.
Lectures d'état par phases
Comme indiqué ci-dessus, Compose se déroule en trois phases principales, lesquelles permettent de suivre les états lus dans chacune d'elles. Cela permet à Compose de notifier uniquement les phases spécifiques pour lesquelles des tâches sont nécessaires pour chaque élément d'UI concerné.
Passons en revue chaque phase et décrivons ce qui se passe lorsqu'une valeur d'état y est lue.
Phase 1 : Composition
Les lectures d'état dans une fonction @Composable
ou un bloc lambda concernent la composition et potentiellement les phases suivantes. Lorsque la valeur d'état change, le recomposeur programme de nouveau l'exécution de toutes les fonctions modulables qui la lisent. Notez que l'environnement d'exécution peut ignorer une partie ou la totalité des fonctions modulables si les entrées n'ont pas changé. Pour en savoir plus, consultez la section Ignorer si les entrées n'ont pas changé.
Selon le résultat de la composition, l'interface utilisateur de Compose exécute les phases de mise en page et de dessin. Elle peut ignorer ces phases si le contenu reste le même, et que la taille et la mise en page ne changent pas.
var padding by remember { mutableStateOf(8.dp) } Text( text = "Hello", // The `padding` state is read in the composition phase // when the modifier is constructed. // Changes in `padding` will invoke recomposition. modifier = Modifier.padding(padding) )
Phase 2 : Mise en page
La phase de mise en page comprend deux étapes : la mesure et le positionnement. L'étape de mesure exécute le lambda de mesure transmis au composable Layout
, à la méthode MeasureScope.measure
de l'interface LayoutModifier
, etc. L'étape de positionnement exécute le bloc de positionnement de la fonction layout
, le bloc lambda de Modifier.offset { … }
, etc.
Les lectures d'état lors de chacune de ces étapes concernent la mise en page et, potentiellement, la phase de dessin. Lorsque l'état change, l'interface utilisateur de Compose planifie la phase de mise en page. Elle exécute également la phase de dessin si la taille ou la position a changé.
Plus précisément, l'étape de mesure et l'étape de positionnement impliquent des champs d'application de redémarrage distincts. Autrement dit, les lectures d'état dans l'étape de positionnement n'appellent pas l'étape de mesure avant cela. Cependant, ces deux étapes sont souvent liées, de sorte qu'un état lu dans l'étape de positionnement peut avoir une incidence sur les autres champs d'application de redémarrage appartenant à l'étape de mesure.
var offsetX by remember { mutableStateOf(8.dp) } Text( text = "Hello", modifier = Modifier.offset { // The `offsetX` state is read in the placement step // of the layout phase when the offset is calculated. // Changes in `offsetX` restart the layout. IntOffset(offsetX.roundToPx(), 0) } )
Phase 3 : Dessin
Les lectures d'état pendant le dessin concernent la phase de dessin. Canvas()
, Modifier.drawBehind
et Modifier.drawWithContent
sont des exemples courants. Lorsque la valeur de l'état change, l'interface utilisateur de Compose n'exécute que la phase de dessin.
var color by remember { mutableStateOf(Color.Red) } Canvas(modifier = modifier) { // The `color` state is read in the drawing phase // when the canvas is rendered. // Changes in `color` restart the drawing. drawRect(color) }
Optimiser les lectures d'état
À mesure que Compose effectue le suivi des lectures d'état localisé, la quantité de travail effectuée peut être réduite en lisant chaque état dans une phase appropriée.
Prenons un exemple. Ici, nous avons Image()
, qui utilise le modificateur de décalage pour décaler sa position finale de mise en page, ce qui entraîne un effet de parallaxe lorsque l'utilisateur fait défiler l'écran.
Box { val listState = rememberLazyListState() Image( // ... // Non-optimal implementation! Modifier.offset( with(LocalDensity.current) { // State read of firstVisibleItemScrollOffset in composition (listState.firstVisibleItemScrollOffset / 2).toDp() } ) ) LazyColumn(state = listState) { // ... } }
Ce code fonctionne, mais génère des performances non optimales. Il lit la valeur de l'état firstVisibleItemScrollOffset
et la transmet à la fonction Modifier.offset(offset: Dp)
. La valeur firstVisibleItemScrollOffset
change à mesure que l'utilisateur fait défiler l'écran. Comme nous le savons, Compose suit toutes les lectures d'état afin de pouvoir relancer (rappeler) le code de lecture qui, dans notre exemple, correspond au contenu de Box
.
Il s'agit d'un exemple d'état lu pendant la phase de composition. Ce n'est pas nécessairement une mauvaise chose. En réalité, c'est là la base de la recomposition : permettre les modifications de données pour émettre une nouvelle UI.
Cet exemple n'est pas optimal, car chaque événement de défilement entraîne la réévaluation de l'intégralité du contenu composable, ainsi que sa mesure, sa mise en page et son dessin. La phase Compose est déclenchée à chaque défilement, même si ce que nous voyons ne change pas. Par contre, la position change. Nous pouvons optimiser la lecture de l'état pour ne déclencher que la phase de mise en page.
Une autre version du modificateur de décalage est disponible : Modifier.offset(offset: Density.() -> IntOffset)
.
Cette version utilise un paramètre lambda, où le décalage obtenu est renvoyé par le bloc lambda. Mettons à jour le code pour l'utiliser :
Box { val listState = rememberLazyListState() Image( // ... Modifier.offset { // State read of firstVisibleItemScrollOffset in Layout IntOffset(x = 0, y = listState.firstVisibleItemScrollOffset / 2) } ) LazyColumn(state = listState) { // ... } }
Pourquoi est-il plus performant ? Le bloc lambda que nous fournissons au modificateur est appelé pendant la phase de mise en page (plus spécifiquement lors de l'étape de positionnement pendant la phase de mise en page). Autrement dit, l'état firstVisibleItemScrollOffset
n'est plus lu pendant la composition. Étant donné que Compose suit la lecture de l'état, cette modification signifie que si la valeur firstVisibleItemScrollOffset
change, Compose doit uniquement redémarrer les phases de mise en page et de dessin.
Cet exemple s'appuie sur les différents modificateurs de décalage pour optimiser le code généré, mais l'idée générale reste valide : essayez de limiter les lectures d'état à la phase la plus basse possible afin de permettre à Compose d'effectuer le moins de travail possible.
Certes, il est souvent absolument nécessaire de lire les états dans la phase de composition. Toutefois, il reste possible de réduire le nombre de recompositions à leur minimum en filtrant les changements d'état. Pour en savoir plus à ce sujet, consultez la page derivedStateOf : convertir un ou plusieurs objets d'état en un autre état.
Recomposition en boucle (dépendance de phase cyclique)
Précédemment, nous avons mentionné que les phases de Compose sont toujours appelées dans le même ordre et qu'il n'existe aucun moyen de revenir en arrière dans le même frame. Toutefois, cela n'empêche pas les applications de réaliser des compositions en boucle sur différents frames. Prenons l'exemple suivant :
Box { var imageHeightPx by remember { mutableStateOf(0) } Image( painter = painterResource(R.drawable.rectangle), contentDescription = "I'm above the text", modifier = Modifier .fillMaxWidth() .onSizeChanged { size -> // Don't do this imageHeightPx = size.height } ) Text( text = "I'm below the image", modifier = Modifier.padding( top = with(LocalDensity.current) { imageHeightPx.toDp() } ) ) }
Dans cet exemple, nous avons (mal) implémenté une colonne verticale, avec l'image en haut, puis le texte en dessous. Nous utilisons Modifier.onSizeChanged()
pour identifier la taille résolue de l'image, puis nous utilisons Modifier.padding()
au niveau du texte pour le décaler vers le bas. La conversion non naturelle de Px
en Dp
indique déjà que le code présente un problème.
Le problème avec cet exemple est que nous ne parvenons pas à la mise en page "finale" dans un seule frame. Le code repose sur l'affichage de plusieurs frames, ce qui entraîne un travail inutile et l'instabilité de l'interface utilisateur à l'écran.
Passons en revue chaque frame pour voir ce qui se passe :
Lors de la phase de composition du premier frame, imageHeightPx
a la valeur 0, et le texte est fourni avec Modifier.padding(top = 0)
. S'ensuit la phase de mise en page. Le rappel du modificateur onSizeChanged
est appelé.
C'est à ce moment que l'attribut imageHeightPx
est mis à jour avec la hauteur réelle de l'image.
Compose planifie la recomposition du frame suivant. Au cours de la phase de dessin, le texte est affiché avec une marge intérieure correspondant à 0, car la modification de valeur n'est pas encore reflétée.
Compose démarre ensuite le deuxième frame planifié par le changement de valeur d'imageHeightPx
. L'état est lu dans le bloc de contenu Box et il est appelé lors de la phase de composition. Cette fois, le texte est fourni avec une marge intérieure correspondant à la hauteur de l'image. Lors de la phase de mise en page, le code définit à nouveau la valeur d'imageHeightPx
, mais aucune recomposition n'est planifiée, car la valeur reste la même.
Au final, nous obtenons la marge intérieure souhaitée sur le texte, mais il n'est pas optimal d'utiliser un frame supplémentaire pour transmettre la valeur de la marge intérieure à une autre phase et de générer ainsi un frame dont le contenu se chevauche.
Cet exemple peut sembler artificiel, mais faites attention à ce schéma général :
Modifier.onSizeChanged()
,onGloballyPositioned()
ou autres opérations de mise en page- Mise à jour d'un état
- Utilisation de cet état comme entrée d'un modificateur de mise en page (
padding()
,height()
ou similaire) - Répétition potentielle
Pour résoudre le problème ci-dessus, la solution consiste à utiliser les primitives de mise en page appropriées. L'exemple ci-dessus peut être implémenté avec un simple attribut Column()
, mais vous pouvez disposer d'un exemple plus complexe nécessitant une personnalisation et, par conséquent, une mise en page personnalisée. Pour en savoir plus, consultez le guide Mises en page personnalisées.
Le principe général ici est d'avoir une source unique et fiable pour tous les éléments d'interface utilisateur qui doivent être mesurés et placés les uns par rapport aux autres. Si vous utilisez une primitive de mise en page appropriée ou si vous créez une mise en page personnalisée, le parent partagé minimal sert de source fiable pouvant coordonner la relation entre divers éléments. L'introduction d'un état dynamique enfreint ce principe.
Recommandations personnalisées
- Remarque : Le texte du lien s'affiche lorsque JavaScript est désactivé.
- États et Jetpack Compose
- Listes et grilles
- Kotlin pour Jetpack Compose