Di Compose, Anda dapat menggabungkan beberapa pengubah untuk mengubah tampilan dan nuansa composable. Rantai pengubah ini dapat memengaruhi batasan yang diteruskan ke composable, yang menentukan batas lebar dan tinggi.
Halaman ini menjelaskan pengaruh pengubah berantai terhadap batasan dan, pada gilirannya, pengukuran dan penempatan composable.
Pengubah di hierarki UI
Untuk memahami pengaruh pengubah satu sama lain, sebaiknya visualisasikan tampilannya di hierarki UI, yang dibuat selama fase komposisi. Untuk mengetahui informasi selengkapnya, lihat bagian Komposisi.
Di hierarki UI, Anda dapat memvisualisasikan pengubah sebagai node wrapper untuk node tata letak:
Menambahkan lebih dari satu pengubah ke composable akan membuat rantai pengubah. Saat Anda menggabungkan beberapa pengubah, setiap node pengubah menggabungkan sisa rantai dan node tata letak di dalamnya. Misalnya, saat Anda menggabungkan pengubah clip dan
size, node pengubah clip akan menggabungkan node pengubah size,
yang kemudian menggabungkan node tata letak Image.
Pada fase tata letak, algoritma yang menelusuri hierarki tetap sama, tetapi setiap node pengubah juga dikunjungi. Dengan cara ini, pengubah dapat mengubah persyaratan ukuran dan penempatan pengubah atau node tata letak yang digabungkannya.
Seperti ditunjukkan pada Gambar 2, penerapan composable Image dan Text itu sendiri terdiri dari rantai pengubah yang menggabungkan satu node tata letak.
Penerapan Row dan Column adalah node tata letak yang menjelaskan cara menata letak turunannya.
Ringkasnya:
- Pengubah menggabungkan satu pengubah atau node tata letak.
- Node tata letak dapat menata letak beberapa node turunan.
Bagian berikut menjelaskan cara menggunakan model mental ini untuk memahami rantai pengubah dan pengaruhnya terhadap ukuran composable.
Batasan dalam fase tata letak
Fase tata letak mengikuti algoritma tiga langkah untuk menemukan lebar, tinggi, dan koordinat x, y setiap node tata letak:
- Mengukur turunan: Node mengukur turunannya, jika ada.
- Menentukan ukuran sendiri: Berdasarkan pengukuran tersebut, node menentukan ukurannya sendiri.
- Menempatkan turunan: Setiap node turunan ditempatkan relatif terhadap posisi node itu sendiri.
Constraints membantu menemukan ukuran yang tepat untuk node selama dua langkah pertama algoritma. Batasan menentukan batas minimum dan maksimum untuk lebar dan tinggi node. Saat node menentukan ukurannya, ukuran yang diukur harus berada dalam rentang ukuran ini.
Jenis batasan
Batasan dapat berupa salah satu hal berikut:
- Terbatas: Node memiliki lebar dan tinggi maksimum dan minimum.
- Tidak terbatas: Node tidak dibatasi ke ukuran apa pun. Batas lebar dan tinggi maksimum ditetapkan ke tak terbatas.
- Tepat: Node diminta untuk mengikuti persyaratan ukuran yang tepat. Batas minimum dan maksimum ditetapkan ke nilai yang sama.
- Kombinasi: Node mengikuti kombinasi jenis batasan sebelumnya. Misalnya, batasan dapat membatasi lebar sekaligus memungkinkan tinggi maksimum yang tidak terbatas, atau menetapkan lebar yang tepat tetapi memberikan tinggi yang terbatas.
Bagian berikutnya menjelaskan cara batasan ini diteruskan dari induk ke turunan.
Cara batasan diteruskan dari induk ke turunan
Selama langkah pertama algoritma yang dijelaskan dalam Batasan dalam fase tata letak, batasan diteruskan dari induk ke turunan di hierarki UI.
Saat node induk mengukur turunannya, node induk akan memberikan batasan ini kepada setiap turunan untuk memberi tahu seberapa besar atau kecil ukuran yang diizinkan. Kemudian, saat menentukan ukurannya sendiri, node induk juga mematuhi batasan yang diteruskan oleh induknya sendiri.
Pada tingkat tinggi, algoritma berfungsi dengan cara berikut:
- Untuk menentukan ukuran yang sebenarnya ingin ditempati, node root di hierarki UI mengukur turunannya dan meneruskan batasan yang sama ke turunan pertamanya.
- Jika turunan adalah pengubah yang tidak memengaruhi pengukuran, turunan akan meneruskan batasan ke pengubah berikutnya. Batasan diteruskan ke rantai pengubah apa adanya kecuali jika pengubah yang memengaruhi pengukuran tercapai. Kemudian, batasan akan diubah ukurannya sesuai dengan itu.
- Setelah node yang tidak memiliki turunan tercapai (disebut sebagai "node leaf"), node tersebut akan menentukan ukurannya berdasarkan batasan yang diteruskan, dan menampilkan ukuran yang diatasi ini ke induknya.
- Induk menyesuaikan batasannya berdasarkan pengukuran turunan ini, dan memanggil turunan berikutnya dengan batasan yang disesuaikan ini.
- Setelah semua turunan induk diukur, node induk akan menentukan ukurannya sendiri dan mengomunikasikannya ke induknya sendiri.
- Dengan cara ini, seluruh hierarki akan dilalui secara depth-first. Pada akhirnya, semua node telah menentukan ukurannya, dan langkah pengukuran selesai.
Untuk contoh mendalam, lihat video Batasan dan urutan pengubah.
Pengubah yang memengaruhi batasan
Anda telah mempelajari di bagian sebelumnya bahwa beberapa pengubah dapat memengaruhi ukuran batasan. Bagian berikut menjelaskan pengubah tertentu yang memengaruhi batasan.
Pengubah size
Pengubah size mendeklarasikan ukuran konten yang diinginkan.
Misalnya, hierarki UI berikut harus dirender dalam penampung 300dp x 200dp. Batasannya terbatas, sehingga memungkinkan lebar antara 100dp dan 300dp, serta tinggi antara 100dp dan 200dp:
Pengubah size menyesuaikan batasan masuk agar sesuai dengan nilai yang diteruskan ke pengubah tersebut.
Dalam contoh ini, nilainya adalah 150dp:
size yang menyesuaikan batasan ke 150dp.Jika lebar dan tinggi lebih kecil dari batas batasan terkecil, atau lebih besar dari batas batasan terbesar, pengubah akan mencocokkan batasan yang diteruskan sedekat mungkin sambil tetap mematuhi batasan yang diteruskan:
size yang mematuhi batasan yang diteruskan sedekat mungkin.Perhatikan bahwa menggabungkan beberapa pengubah size tidak berfungsi. Pengubah size pertama menetapkan batasan minimum dan maksimum ke nilai tetap. Meskipun pengubah ukuran kedua meminta ukuran yang lebih kecil atau lebih besar, pengubah tersebut tetap harus mematuhi batas yang tepat yang diteruskan, sehingga tidak akan mengganti nilai tersebut:
size, yang nilai kedua yang diteruskan (50dp) tidak mengganti nilai pertama (100dp).Pengubah requiredSize
Gunakan pengubah requiredSize dan bukan size jika Anda ingin
node mengganti batasan masuk. Pengubah requiredSize mengganti batasan masuk dan meneruskan ukuran yang Anda tentukan sebagai batas yang tepat.
Saat ukuran diteruskan kembali ke hierarki, node turunan akan berada di tengah ruang yang tersedia:
requiredSize yang mengganti batasan masuk dari pengubah size.Pengubah width dan height
Pengubah size menyesuaikan lebar dan tinggi batasan. Dengan pengubah width, Anda dapat menetapkan lebar tetap tetapi membiarkan tinggi tidak ditentukan.
Demikian pula, dengan pengubah height, Anda dapat menetapkan tinggi tetap, tetapi membiarkan lebar tidak ditentukan:
width dan pengubah height yang menetapkan lebar dan tinggi tetap.Pengubah sizeIn
Pengubah sizeIn memungkinkan Anda menetapkan batasan minimum dan maksimum yang tepat untuk lebar dan tinggi. Gunakan pengubah sizeIn jika Anda memerlukan kontrol terperinci atas batasan.
sizeIn dengan minWidth, maxWidth, minHeight, dan maxHeight ditetapkan.Contoh
Bagian ini menunjukkan dan menjelaskan output dari beberapa cuplikan kode dengan pengubah berantai.
Image( painterResource(R.drawable.hero), contentDescription = null, Modifier .fillMaxSize() .size(50.dp) )
Cuplikan ini menghasilkan output berikut:
Image mengisi ukuran maksimum sebagai
hasil dari rantai pengubah.- Pengubah
fillMaxSizemengubah batasan untuk menetapkan lebar dan tinggi minimum ke nilai maksimum —300dpuntuk lebar dan200dpuntuk tinggi. - Meskipun pengubah
sizeingin menggunakan ukuran50dp, pengubah tersebut tetap harus mematuhi batasan minimum yang masuk. Jadi, pengubahsizejuga akan menampilkan batas batasan yang tepat sebesar300x200, yang secara efektif mengabaikan nilai yang diberikan dalam pengubahsize. Imagemengikuti batas ini dan melaporkan ukuran300x200, yang diteruskan hingga ke hierarki.
Image( painterResource(R.drawable.hero), contentDescription = null, Modifier .fillMaxSize() .wrapContentSize() .size(50.dp) )
Cuplikan ini menghasilkan output berikut:
Image berada di tengah dan berukuran
50dp.- Pengubah
fillMaxSizemenyesuaikan batasan untuk menetapkan lebar dan tinggi minimum ke nilai maksimum —300dpuntuk lebar dan200dpuntuk tinggi. - Pengubah
wrapContentSizemereset batasan minimum. Jadi, meskipunfillMaxSizemenghasilkan batasan tetap,wrapContentSizemeresetnya kembali ke batasan terbatas. Node berikut kini dapat menempati seluruh ruang lagi, atau lebih kecil dari seluruh ruang. - Pengubah
sizemenetapkan batasan ke batas minimum dan maksimum50. Imagediatasi ke ukuran50x50, dan pengubahsizemeneruskannya.- Pengubah
wrapContentSizememiliki properti khusus. Pengubah ini mengambil turunannya dan menempatkannya di tengah batas minimum yang tersedia yang diteruskan ke pengubah tersebut. Oleh karena itu, ukuran yang dikomunikasikan ke induknya sama dengan batas minimum yang diteruskan ke pengubah tersebut.
Dengan menggabungkan tiga pengubah saja, Anda dapat menentukan ukuran untuk composable dan menempatkannya di tengah induknya.
Image( painterResource(R.drawable.hero), contentDescription = null, Modifier .clip(CircleShape) .padding(10.dp) .size(100.dp) )
Cuplikan ini menghasilkan output berikut:
- Pengubah
cliptidak mengubah batasan. - Pengubah
paddingmenurunkan batasan maksimum. - Pengubah
sizemenetapkan semua batasan ke100dp. Imagemematuhi batasan tersebut dan melaporkan ukuran100dpx100dp.- Pengubah
paddingmenambahkan10dpdi semua sisi ke ukuran yang dilaporkan olehImage, sehingga tata letak dengan padding melaporkan lebar dan tinggi120dp. - Sekarang, pada fase gambar, pengubah
clipbertindak pada kanvas120dpx120dp. Pengubah ini membuat mask lingkaran dengan ukuran tersebut. - Kemudian, pengubah
paddingmenyisipkan kontennya sebesar10dpdi semua sisi, yang menurunkan ukuran kanvas untukImagemenjadi100dpx100dp. Imagedigambar di kanvas yang lebih kecil tersebut. Gambar dipangkas berdasarkan lingkaran asli120dp, sehingga outputnya adalah hasil yang tidak bulat.