Utilizzare Visualizzazioni in Scrivi

Puoi includere una gerarchia Android View in una UI di Compose. Questo approccio è particolarmente utile se vuoi utilizzare elementi UI non ancora disponibili in Compose, come AdView. Questo approccio ti consente anche di riutilizzare le visualizzazioni personalizzate che potresti aver progettato.

Per includere un elemento o una gerarchia della visualizzazione, utilizza il composable AndroidView . AndroidView viene passato un lambda che restituisce View. AndroidView fornisce anche un callback di update che viene chiamato quando la visualizzazione è gonfiata. AndroidView si ricomponie ogni volta che un valore State letto all'interno del callback cambia. AndroidView, come molti altri composabili integrati, accetta un parametro Modifier che può essere utilizzato, ad esempio, per impostarne la posizione nel composable principale.

@Composable
fun CustomView() {
    var selectedItem by remember { mutableStateOf(0) }

    // Adds view to Compose
    AndroidView(
        modifier = Modifier.fillMaxSize(), // Occupy the max size in the Compose UI tree
        factory = { context ->
            // Creates view
            MyView(context).apply {
                // Sets up listeners for View -> Compose communication
                setOnClickListener {
                    selectedItem = 1
                }
            }
        },
        update = { view ->
            // View's been inflated or state read in this block has been updated
            // Add logic here if necessary

            // As selectedItem is read here, AndroidView will recompose
            // whenever the state changes
            // Example of Compose -> View communication
            view.selectedItem = selectedItem
        }
    )
}

@Composable
fun ContentExample() {
    Column(Modifier.fillMaxSize()) {
        Text("Look at this CustomView!")
        CustomView()
    }
}

AndroidView con associazione delle visualizzazioni

Per incorporare un layout XML, utilizza l'API AndroidViewBinding, fornita dalla libreria androidx.compose.ui:ui-viewbinding. Per farlo, il progetto deve attivare il collegamento delle visualizzazioni.

@Composable
fun AndroidViewBindingExample() {
    AndroidViewBinding(ExampleLayoutBinding::inflate) {
        exampleView.setBackgroundColor(Color.GRAY)
    }
}

AndroidView negli elenchi lazy

Se utilizzi un AndroidView in un elenco lazy (LazyColumn, LazyRow, Pager e così via), ti consigliamo di utilizzare l'overload AndroidView introdotto nella versione 1.4.0-rc01. Questo sovraccarico consente a Compose di riutilizzare l'istanza View sottostante quando la composizione contenitore viene riutilizzata, come avviene per gli elenchi Lazy.

Questo sovraccarico di AndroidView aggiunge due parametri aggiuntivi:

  • onReset: un callback invocato per segnalare che View sta per essere riutilizzato. Deve essere diverso da null per consentire il riutilizzo della visualizzazione.
  • onRelease (facoltativo) - Un callback invocato per segnalare che View è uscito dalla composizione e non verrà riutilizzato.

@OptIn(ExperimentalComposeUiApi::class)
@Composable
fun AndroidViewInLazyList() {
    LazyColumn {
        items(100) { index ->
            AndroidView(
                modifier = Modifier.fillMaxSize(), // Occupy the max size in the Compose UI tree
                factory = { context ->
                    MyView(context)
                },
                update = { view ->
                    view.selectedItem = index
                },
                onReset = { view ->
                    view.clear()
                }
            )
        }
    }
}

Frammenti in Compose

Utilizza il composable AndroidViewBinding per aggiungere un Fragment in Compose. AndroidViewBinding ha un trattamento specifico per i frammenti, ad esempio la rimozione del frammento quando il composable esce dalla composizione.

A tal fine, esegui il pompaggio di un file XML contenente un FragmentContainerView come contenitore del tuo Fragment.

Ad esempio, se hai definito my_fragment_layout.xml, puoi utilizzare un codice come questo sostituendo l'attributo XML android:name con il nome della classe Fragment:

<androidx.fragment.app.FragmentContainerView xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/fragment_container_view"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:name="com.example.compose.snippets.interop.MyFragment" />

Ingrandisci questo frammento in Componi come segue:

@Composable
fun FragmentInComposeExample() {
    AndroidViewBinding(MyFragmentLayoutBinding::inflate) {
        val myFragment = fragmentContainerView.getFragment<MyFragment>()
        // ...
    }
}

Se devi utilizzare più frammenti nello stesso layout, assicurati di avere definito un ID univoco per ogni FragmentContainerView.

Chiamare il framework Android da Compose

Compose opera all'interno delle classi del framework Android. Ad esempio, è ospitato su classi View di Android, come Activity o Fragment, e potrebbe utilizzare classi del framework Android come Context, risorse di sistema, Service o BroadcastReceiver.

Per scoprire di più sulle risorse di sistema, consulta Risorse in Compose.

Impostazioni internazionali della composizione

Le classi CompositionLocal consentono di passare i dati in modo implicito tramite funzioni componibili. Generalmente vengono forniti con un valore in un determinato nodo dell'albero dell'interfaccia utente. Questo valore può essere utilizzato dai suoi discendenti componibili senza dichiarare CompositionLocal come parametro nella funzione componibile.

CompositionLocal viene utilizzato per propagare i valori per i tipi di framework Android in Compose, come Context, Configuration o View, in cui il codice Compose è ospitato con il corrispondente LocalContext, LocalConfiguration o LocalView. Tieni presente che le classi CompositionLocal hanno il prefisso Local per una migliore scoperta con il completamento automatico nell'IDE.

Accedi al valore corrente di un CompositionLocal utilizzando la relativa proprietà current. Ad esempio, il codice seguente mostra un messaggio di notifica passando LocalContext.current al metodo Toast.makeToast.

@Composable
fun ToastGreetingButton(greeting: String) {
    val context = LocalContext.current
    Button(onClick = {
        Toast.makeText(context, greeting, Toast.LENGTH_SHORT).show()
    }) {
        Text("Greet")
    }
}

Per un esempio più completo, consulta la sezione Case study: BroadcastReceiver alla fine di questo documento.

Altre interazioni

Se non è stata definita un'utilità per l'interazione di cui hai bisogno, la best practice è seguire la linea guida generale di Compose, i dati scorrono verso il basso, gli eventi verso l'alto (discussa più dettagliatamente in Pensare in Compose). Ad esempio, questo composable avvia un'attività diversa:

class OtherInteractionsActivity : ComponentActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        // get data from savedInstanceState
        setContent {
            MaterialTheme {
                ExampleComposable(data, onButtonClick = {
                    startActivity(Intent(this, MyActivity::class.java))
                })
            }
        }
    }
}

@Composable
fun ExampleComposable(data: DataExample, onButtonClick: () -> Unit) {
    Button(onClick = onButtonClick) {
        Text(data.title)
    }
}

Case study: broadcast receiver

Per un esempio più realistico delle funzionalità di cui potresti voler eseguire la migrazione o l'implementazione in Compose e per mostrare CompositionLocal e gli effetti collaterali, supponiamo che un BroadcastReceiver debba essere registrato da una funzione componibile.

La soluzione utilizza LocalContext per utilizzare il contesto corrente e gli effetti collaterali rememberUpdatedState e DisposableEffect.

@Composable
fun SystemBroadcastReceiver(
    systemAction: String,
    onSystemEvent: (intent: Intent?) -> Unit
) {
    // Grab the current context in this part of the UI tree
    val context = LocalContext.current

    // Safely use the latest onSystemEvent lambda passed to the function
    val currentOnSystemEvent by rememberUpdatedState(onSystemEvent)

    // If either context or systemAction changes, unregister and register again
    DisposableEffect(context, systemAction) {
        val intentFilter = IntentFilter(systemAction)
        val broadcast = object : BroadcastReceiver() {
            override fun onReceive(context: Context?, intent: Intent?) {
                currentOnSystemEvent(intent)
            }
        }

        context.registerReceiver(broadcast, intentFilter)

        // When the effect leaves the Composition, remove the callback
        onDispose {
            context.unregisterReceiver(broadcast)
        }
    }
}

@Composable
fun HomeScreen() {

    SystemBroadcastReceiver(Intent.ACTION_BATTERY_CHANGED) { batteryStatus ->
        val isCharging = /* Get from batteryStatus ... */ true
        /* Do something if the device is charging */
    }

    /* Rest of the HomeScreen */
}

Passaggi successivi

Ora che conosci le API di interoperabilità quando utilizzi Compose in Visualizzazioni e viceversa, consulta la pagina Altre considerazioni per saperne di più.