setContent { // In here, we can call composables!
MaterialTheme {
Greeting(name = “compose”)
}
}
}
}
@Composable
fun Greeting(name: String) {
Text(text = “Hello $name!”)
}
Use Compose in Fragment
class PureComposeFragment : Fragment() {
override fun onCreateView(
inflater: LayoutInflater,
container: ViewGroup?,
savedInstanceState: Bundle?
): View {
return ComposeView(requireContext()).apply {
setContent {
MaterialTheme {
Text(“Hello Compose!”)
}
}
}
}
}
在View中使用Compose
ComposeView内嵌在Xml中:
一个平平无奇的xml布局文件中加入ComposeView : <?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android=“http://schemas.android.com/apk/res/android”
android:layout_width=“match_parent”
android:layout_height=“match_parent”
android:orientation=“vertical”>
<TextView
android:id="@+id/hello_world"
android:layout_width=“match_parent”
android:layout_height=“wrap_content”
android:text=“Hello from XML layout” />
<androidx.compose.ui.platform.ComposeView
android:id="@+id/compose_view"
android:layout_width=“match_parent”
android:layout_height=“match_parent” />
使用的时候, 先根据id查找出来, 再setContent:
class ComposeViewInXmlActivity : AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_compose_view_in_xml)
findViewById(R.id.compose_view).setContent {
// In Compose world
MaterialTheme {
Text(“Hello Compose!”)
}
}
}
}
动态添加ComposeView
在代码中使用addView() 来添加View对于ComposeView 来说也同样适用:
class ComposeViewInViewActivity : AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(LinearLayout(this).apply {
orientation = VERTICAL
addView(ComposeView(this@ComposeViewInViewActivity).apply {
id = R.id.compose_view_x
setContent {
MaterialTheme {
Text(“Hello Compose View 1”)
}
}
})
addView(TextV iew(context).apply {
text = “I’m am old TextView”
})
addView(ComposeView(context).apply {
id = R.id.compose_view_y
setContent {
MaterialTheme {
Text(“Hello Compose View 2”)
}
}
})
})
}
}
这里在LinearLayout 中添加了三个child: 两个ComposeView 中间还有一个TextView .
起到桥梁作用的ComposeView 是一个ViewGroup , 它本身是一个View, 所以可以混进View的hierarchy tree里占位, 它的setContent() 方法开启了Compose世界的大门, 在这里可以传入composable的方法, 绘制UI.
在Compose中使用View
都用Compose搭建UI了, 什么时候会需要在其中内嵌View呢?
-
要用的View还没有Compose版本, 比如AdView , MapView , WebView . -
有一块之前写好的UI, (暂时或者永远)不想动, 想直接用. -
用Compose实现不了想要的效果, 就得用View.
在Compose中加入Android View
例子:
@Composable
fun CustomView() {
val state = remember { mutableStateOf(0) }
//widget.Button
AndroidView(
factory = { ctx ->
//Here you can construct your View
android.widget.Button(ctx).apply {
text = “My Button”
layoutParams = LinearLayout.LayoutParams(MATCH_PARENT, WRAP_CONTENT)
setOnClickListener {
state.value++
}
}
},
modifier = Modifier.padding(8.dp)
)
//widget.TextView
AndroidView(factory = { ctx ->
//Here you can construct your View
TextView(ctx).apply {
layoutParams = LinearLayout.LayoutParams(MATCH_PARENT, WRAP_CONTENT)
}
}, update = {
it.text = “You have clicked the buttons: " + state.value.toString() + " times”
})
}
这里的桥梁是AndroidView , 它是一个composable方法:
@Composable
fun AndroidView(
factory: (Context) -> T,
modifier: Modifier = Modifier,
update: (T) -> Unit = NoOpUpdate
)
factory接收一个Context参数, 用来构建一个View. update方法是一个callback, inflate之后会执行, 读取的状态state值变化后也会被执行.
在Compose中使用xml布局
上面提到的在Compose中使用AndroidView的方法, 对于少量的UI还行. 如果需要复用一个已经存在的xml布局怎么办? 不用怕, view binding登场了.
使用起来也很简单:
buildFeatures {
compose true
viewBinding true
}
update: (T) -> Unit = NoOpUpdate
)
factory接收一个Context参数, 用来构建一个View. update方法是一个callback, inflate之后会执行, 读取的状态state值变化后也会被执行.
在Compose中使用xml布局
上面提到的在Compose中使用AndroidView的方法, 对于少量的UI还行. 如果需要复用一个已经存在的xml布局怎么办? 不用怕, view binding登场了.
使用起来也很简单:
buildFeatures {
compose true
viewBinding true
}
|