Jetpack Compose中使用Android View

使用AndroidView创建日历

Kotlin 复制代码
@Composable
fun AndroidViewPage() {
    AndroidView(
        factory = {
            CalendarView(it)
        },
        modifier = Modifier.fillMaxWidth(),
        update = {
            it.setOnDateChangeListener { view, year, month, day ->
                Toast.makeText(
                    view.context, "${year}年${month + 1}月${day}日",
                    Toast.LENGTH_LONG
                ).show()
            }
        }
    )
}

使用Android原生视图需要引入AndroidView组件,factory属性中加入CalendarView日历视图组件,update属性中设置日历的点击事件。

使用WebView

Kotlin 复制代码
@SuppressLint("UnusedMaterial3ScaffoldPaddingParameter", "SetJavaScriptEnabled")
@OptIn(ExperimentalMaterial3Api::class)
@Composable
fun WebViewPage() {
    val webView = rememberWebViewWithLifecycle()
    Scaffold(
        content = {
            AndroidView(
                factory = {
                    webView
                }, modifier = Modifier
                    .fillMaxSize()
                    .background(Color.Red),
                update = { webView ->
                    // 设置支持JavaScript
                    val webSettings = webView.settings
                    webSettings.javaScriptEnabled = true
                    webView.loadUrl("https://www.baidu.com")
                }
            )
        }
    )
}

@Composable
fun rememberWebViewWithLifecycle(): WebView {
    val context = LocalContext.current
    val webView = remember {
        WebView(context)
    }
    val lifecycleObserver = rememberWebViewLifecycleObserver(webView)
    val lifecycle = LocalLifecycleOwner.current.lifecycle
    DisposableEffect(lifecycle) {
        lifecycle.addObserver(lifecycleObserver)
        onDispose {
            lifecycle.removeObserver(lifecycleObserver)
        }
    }
    return webView
}

@Composable
private fun rememberWebViewLifecycleObserver(webView: WebView): LifecycleEventObserver =
    remember(webView) {
        LifecycleEventObserver { _, event ->
            when (event) {
                Lifecycle.Event.ON_RESUME -> webView.onResume()
                Lifecycle.Event.ON_PAUSE -> webView.onPause()
                Lifecycle.Event.ON_DESTROY -> webView.destroy()
                else -> Log.e("WebView", event.name)
            }
        }
    }

使用Android布局

使用Android布局用到了Viewbinding,这里需要引入ui-viewbinding库

bash 复制代码
implementation "androidx.compose.ui:ui-viewbinding:1.5.0-alpha01"

这是android_view.xml原生页面布局

XML 复制代码
<?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">

    <EditText
        android:id="@+id/editName"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="30dp"
        android:hint="name" />

    <EditText
        android:id="@+id/editPassword"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginHorizontal="30dp"
        android:hint="password"
        android:inputType="textPassword" />

    <Button
        android:id="@+id/btnLogin"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center"
        android:textAllCaps="false"
        android:layout_marginTop="30dp"
        android:text="Login" />
</LinearLayout>

Compose中使用原生布局并操作

Kotlin 复制代码
@Composable
fun AndroidViewBindingPage() {
    val context = LocalContext.current
    AndroidViewBinding(
        factory = { inflate, parent, attachToParent ->
            AndroidViewBinding.inflate(inflate, parent, attachToParent)
        },
        modifier = Modifier.fillMaxSize(),
        update = {
            btnLogin.setOnClickListener {
                val name = editName.text.toString().trim()
                val password = editPassword.text.toString().trim()
                toLogin(context, name, password)
            }
        }
    )
}

fun toLogin(context: Context, name: String, password: String) {
    if (name.isEmpty() || password.isEmpty()) {
        Toast.makeText(context, "请输入完整信息", Toast.LENGTH_SHORT).show()
        return
    }
    Toast.makeText(context, "登录信息为:name:${name}, password:${password}", Toast.LENGTH_SHORT)
        .show()
}

引用非常简单,只要通过ViewBinding组件在factory属性中添加页面的引用就行,update属性下对页面进行操作逻辑。

相关推荐
笔沫拾光4 天前
Flutter 与 Android 原生布局组件对照表(完整版)
android·flutter·widget·android布局
不用89k13 天前
Compose Kotlin Multiplatform跨平台基础运行
compose
缘来的精彩1 个月前
Kotlin与Jetpack Compose的详细使用指南
android·kotlin·android studio·compose·viewmodel
清霜之辰1 个月前
安卓 Compose 相对传统 View 的优势
android·内存·性能·compose
tangweiguo030519871 个月前
Android Compose Activity 页面跳转动画详解
android·compose
tangweiguo030519871 个月前
在 Jetpack Compose 中实现 iOS 风格输入框
android·compose
tangweiguo030519872 个月前
Android Compose 权限申请完整指南
compose
tangweiguo030519872 个月前
androd的XML页面 跳转 Compose Activity 卡顿问题
compose
tangweiguo030519872 个月前
iOS 风格弹框组件集 (Compose版)
compose
tangweiguo030519872 个月前
Android Material Design 3 主题配色终极指南:XML 与 Compose 全解析
compose