【Compose】Android Compose 监听TextField粘贴事件

背景

项目中需要监听输入框(也就是TextField组件)的粘贴事件,Compose的TextField不像android.view的EditText中一样,重写onTextContextMenuItem方法就可以做到监听,Compose中而是使用改变provider来实现。

实现

1. 获取本地TextToolbar

kotlin 复制代码
val defaultTextToolbar = LocalTextToolbar.current

2. 重写TextToolbar

kotlin 复制代码
val pasteListeningTextToolbar = remember{
    object : TextToolbar {
        override fun hide() {
            defaultTextToolbar.hide()
        }

        override val status: TextToolbarStatus
            get() = defaultTextToolbar.status

        override fun showMenu(
            rect: Rect,
            onCopyRequested: (() -> Unit)?,
            onPasteRequested: (() -> Unit)?,
            onCutRequested: (() -> Unit)?,
            onSelectAllRequested: (() -> Unit)?
        ) {
            val wrappedPasteCallback = {
                onPasteRequested?.invoke()
                println("paste event happen")
            }

            defaultTextToolbar.showMenu(
                rect,
                onCopyRequested,
                wrappedPasteCallback,
                onCutRequested,
                onSelectAllRequested,
            )
        }
    }
}

关键点

只拦截粘贴事件,其他保持不变

kotlin 复制代码
val wrappedPasteCallback = {
    onPasteRequested?.invoke() // 保持原始粘帖功能
    // 添加自定义逻辑,也可以加自定义的callback,我这里直接输入
    println("paste event happen")
}

其后,再把其他功能回调原封不动的传递给默认TextToolbar

kotlin 复制代码
defaultTextToolbar.showMenu(
    rect,
    onCopyRequested,         // 原始复制功能
    wrappedPasteCallback,    // 包装后的粘贴
    onCutRequested,          // 原始剪切功能
    onSelectAllRequested      // 原始全选功能
)

3. 替换默认的TextToolbar

替换原有的TextToolbar,为自定义的TextToolbar

kotlin 复制代码
CompositionLocalProvider(LocalTextToolbar providers pasteListeningTextToolbar){
    TextField(
        value = text,
        onValueChange = { text = id },
    )
}

完整Composeable代码

kotlin 复制代码
@Composable
fun TextFieldWithPasteCallback(
) {
    var text by remember { mutableStateOf("") }
    var state = rememberTextFieldState()
    var defaultTextToolbar = LocalTextToolbar.current

    var pasteListeningToolbar = remember {
        object : TextToolbar {
            override fun hide() {
                defaultTextToolbar.hide()
            }

            override val status: TextToolbarStatus
                get() = defaultTextToolbar.status

            override fun showMenu(
                rect: Rect,
                onCopyRequested: (() -> Unit)?,
                onPasteRequested: (() -> Unit)?,
                onCutRequested: (() -> Unit)?,
                onSelectAllRequested: (() -> Unit)?
            ) {
                val wrappedPasteCallback = {
                    onPasteRequested?.invoke()
                    println("paste event happen")
                }

                defaultTextToolbar.showMenu(
                    rect,
                    onCopyRequested,
                    wrappedPasteCallback,
                    onCutRequested,
                    onSelectAllRequested,
                )
            }
        }
    }

    CompositionLocalProvider(LocalTextToolbar provides pasteListeningToolbar) {
        TextField(
            placeholder = {
                Text("input placeholder")
            },
            value = text,
            onValueChange = {
                println("text:$it")
                text = it
            }
        )
    }
}
相关推荐
cwzqf2 分钟前
Jectpack Compose项目组件代码分享(1):分页加载组件
android
@北海怪兽25 分钟前
SQL常见函数整理 _ STRING_AGG()
android·数据库·sql
鹏晨互联2 小时前
【Compose vs XML:边框内外间距的实现对比】
android·xml
Android系统攻城狮2 小时前
Android tinyalsa深度解析之pcm_plugin_write调用流程与实战(一百七十九)
android·pcm·tinyalsa·android16·音频进阶·android音频进阶
ID_180079054732 小时前
除了JSON,淘宝店铺商品API接口还支持哪些数据格式?
android·数据库
KillerNoBlood2 小时前
2026移动端跨平台开发面经总结
android·算法·flutter·ios·移动开发·鸿蒙·kmp
消失的旧时光-19433 小时前
Android / IoT 面试复盘总结:从 MQTT、TLS 到 JWT 权限体系(标准答案 + 工程理解 + 延伸知识链)
android·物联网·面试
高林雨露3 小时前
Kotlin 的延迟初始化委托属性 by lazy
kotlin
林多4 小时前
【Android】 GPU过度绘制实现原理
android·gpu·性能·实现原理·过度绘制·overdraw
薄荷椰果抹茶4 小时前
手机端Obsidian安装与同步全攻略
android