Compose TextField详解

news2025/1/12 23:02:20

首先明确Compose TextField的底层依赖是:

TextField \rightarrow BasicTextField \rightarrow CoreTextField

相较于Text,TextField需要提供可输入的能力以及渲染效果动态更新的能力。 

  // CompositionLocals
    // If the text field is disabled or read-only, we should not deal with the input service

// !!!用于请求焦点的对象。通过调用 focusRequester.requestFocus(),可以请求将焦点设置到与之关联的 TextField 上,但如何设置的???
    val textInputService = LocalTextInputService.current

// 输入框显示的内容的密度
    val density = LocalDensity.current

// 字体的管理
    val fontFamilyResolver = LocalFontFamilyResolver.current


    val selectionBackgroundColor = LocalTextSelectionColors.current.backgroundColor


// 管理焦点的对象
    val focusManager = LocalFocusManager.current

// 获取当前窗口的信息。窗口是什么???
    val windowInfo = LocalWindowInfo.current

// 用于控制键盘的对象
    val keyboardController = LocalSoftwareKeyboardController.current

针对所谓的文本内容超过TextField的显示区域,会出现Scroller滚动条来动态查看所输入的内容;默认情况下,次效果不显示;且Scroller不算TextField的核心的内容。

 // Scroll state
    val singleLine = maxLines == 1 && !softWrap && imeOptions.singleLine
    val orientation = if (singleLine) Orientation.Horizontal else Orientation.Vertical
    val scrollerPosition = textScrollerPosition ?: rememberSaveable(
        orientation,
        saver = TextFieldScrollerPosition.Saver
    ) { TextFieldScrollerPosition(orientation) }
    if (scrollerPosition.orientation != orientation) {
        throw IllegalArgumentException(
            "Mismatching scroller orientation; " + (
                if (orientation == Orientation.Vertical)
                    "only single-line, non-wrap text fields can scroll horizontally"
                else
                    "single-line, non-wrap text fields can only scroll horizontally"
                )
        )
    }

代码的作用是创建和初始化一个 TextFieldState 对象,并更新其属性。首先通过currentRecomposeScope获取当前重新组合的作用域(recompose scope)。

使用 remember 函数创建一个 TextFieldState 对象,并将其赋值给变量 state;其中keyboardController 是一个键盘控制器对象,它被传递给 remember 函数作为键,以确保在重新组合时 TextFieldState 对象可与输入内容保持一致。

state.update 方法,更新 TextFieldState 对象的属性。这些属性包括文本内容、样式、软换行、密度等,以及一些回调函数和其他对象。通过更新这些属性,可以确保 TextFieldState 对象的状态与传递的参数保持一致。

    val scope = currentRecomposeScope
    // 产生一个关键的对象 state
    val state = remember(keyboardController) {
        TextFieldState(
            TextDelegate(
                text = visualText,
                style = textStyle,
                softWrap = softWrap,
                density = density,
                fontFamilyResolver = fontFamilyResolver
            ),
            recomposeScope = scope,
            keyboardController = keyboardController
        )
    }
    state.update(
        value.annotatedString,
        visualText,
        textStyle,
        softWrap,
        density,
        fontFamilyResolver,
        onValueChange,
        keyboardActions,
        focusManager,
        selectionBackgroundColor
    )

TextFieldState是CoreTextField完成构建各类组件的对象以及声明剩余输入框样式及内容的状态state之后开始构建支持TextField样式渲染的各种Modifier。以此可以理解,Compose布局的过程是先测量到布局到渲染,所以构建各类对象再构建各种Modifier的顺序理应如此。

// Focus
    val focusModifier = Modifier.textFieldFocusModifier(
        enabled = enabled,
        focusRequester = focusRequester,
        interactionSource = interactionSource
    ) {
        if (state.hasFocus == it.isFocused) {
            return@textFieldFocusModifier
        }
        state.hasFocus = it.isFocused


        if (textInputService != null) {
            if (state.hasFocus && enabled && !readOnly) {
                startInputSession(
                    textInputService,
                    state,
                    value,
                    imeOptions,
                    offsetMapping
                )
            } else {
                endInputSession(state)
            }


            // The focusable modifier itself will request the entire focusable be brought into view
            // when it gains focus – in this case, that's the decoration box. However, since text
            // fields may have their own internal scrolling, and the decoration box can do anything,
            // we also need to specifically request that the cursor itself be brought into view.
            // TODO(b/216790855) If this request happens after the focusable's request, the field
            //  will only be scrolled far enough to show the cursor, _not_ the entire decoration
            //  box.
            if (it.isFocused) {
                state.layoutResult?.let { layoutResult ->
                    coroutineScope.launch {
                        bringIntoViewRequester.bringSelectionEndIntoView(
                            value,
                            state.textDelegate,
                            layoutResult.value,
                            offsetMapping
                        )
                    }
                }
            }
        }
        if (!it.isFocused) manager.deselect()
    }

1️⃣ 为 TextField 添加一个修饰符(modifier):focusModifier,用于处理焦点相关的逻辑。

接下来,继续执行多个if语句,旨在判断是否提供可交互的TextField输入框组件。如果存在 textInputService(文本输入服务),并且 TextFieldState 对象具有焦点、启用且不是只读模式,那么会调用 startInputSession 函数来启动输入会话。这个函数会与文本输入服务进行交互,以处理文本输入和键盘事件。如果 TextFieldState 对象失去焦点,会调用 endInputSession 函数来结束输入会话。

coroutineScope.launchstartInputSession 函数分别用于在协程中异步执行将光标位置带入视图的请求和启动输入会话的操作。

因此focusModifier可以总结出必须具备这样的参数条件,页面中的TextField组件才是可输入进行交互的:

  1. 存在 textInputService(文本输入服务)。
  2. TextFieldState 对象具有焦点(state.hasFocus 为 true)。
  3. TextField 启用(enabled 为 true)。
  4. TextField 不是只读模式(readOnly 为 false)。
val pointerModifier = Modifier.textFieldPointer(
        manager,
        enabled,
        interactionSource,
        state,
        focusRequester,
        readOnly,
        offsetMapping
    )

 2️⃣ pointerModifier 是一个修饰符,用于处理与指针(例如鼠标)交互相关的逻辑。

它接受多个参数,包括 managerenabledinteractionSourcestatefocusRequesterreadOnlyoffsetMapping

这个修饰符的作用是为 TextField 添加指针交互的功能。它处理与指针相关的事件,例如鼠标点击、拖动等。通过使用这个修饰符,可以实现对 TextField 的指针交互进行控制和处理。

3️⃣ drawModifier 是一个修饰符,用于在 TextField 的绘制过程中进行自定义绘制操作。

这个修饰符使用了 drawBehind 函数,允许在组件的背后进行绘制操作。调用drawIntoCanvas 函数,获取 Canvas 对象,然后在这个 Canvas 上进行绘制操作。canvasvalueoffsetMappinglayoutResult.valuestate.selectionPaint。这些参数用于确定绘制的位置、内容和样式。

所以我们可以明确TextField不仅是会基于ResumableComposeNode作为渲染的基础,而且还是基于canvas进行渲染的。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2060765.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

智能菜谱推荐系统_ct3p7

TOC springboot575智能菜谱推荐系统_ct3p7--论文 第一章 概述 1.1 研究背景 近些年,随着中国经济发展,人民的生活质量逐渐提高,对网络的依赖性越来越高,通过网络处理的事务越来越多。随着智能菜谱推荐管理的常态化&#xff0c…

F1 F4 Fn lock 指示灯不亮 联想笔记本 thinkpad

问题描述:F1 F4 Fn lock 指示灯开机的时候亮,但是使用的时候虽然能够发挥正常功能,但是指示灯一直熄灭,指示灯不亮。 电脑型号:联想笔记本 thinkpad E14 Gen 2 。本方案应该适用于所有联想电脑。 解决方法:…

嵌入式和单片机有什么区别?

目录 (1)什么是嵌入式? (2)什么是单片机? (3)嵌入式和单片机的共同点 (4)嵌入式和单片机的区别 (1)什么是嵌入式? 关…

掉毛不愁!浮毛怎么去掉比较干净?这宠物空气净化器用上真能解决

这阵子天气热得让人只想宅家,门窗紧闭,空调一开就是一整天。室内凉爽宜人,但一出门再回来,那满屋的浮毛和异味简直让人措手不及,仿佛从天堂跌入地狱。幸好,我家有台宠物空气净化器这位“救星”,…

【Linux】线程控制|POSIX线程库|多线程创建|线程终止|等待|线程分离|线程空间布局

目录 ​编辑 POSIX线程库 多线程创建 独立栈结构 获取线程ID pthread_self 线程终止 return终止线程 pthread_exit pthread_cancel 线程等待 退出码问题 线程分离 测试 线程ID及地址空间布局 ​编辑 POSIX线程库 pthread线程库是 POSIX线程库的一部分&#xf…

MySQL运维学习(1):4种日志

1.错误日志 mysql错误日志记录了mysql发生任何严重错误时的信息,若数据库无法正常使用时,可以先查看错误日志 默认情况下错误日志是开启的,文件名为/var/log/mysqld.log,如果文件不在默认位置,可以通过下面的命令查看…

【Java】Junit的使用

Java系列文章目录 补充内容 Windows通过SSH连接Linux 第一章 Linux基本命令的学习与Linux历史 文章目录 Java系列文章目录一、前言二、学习内容:三、问题描述四、解决方案:4.1 Junit测试方法的使用4.2 测试规范 五、总结 一、前言 学习测试JunitMock后…

LLM如何理解图数据? Graph+LLM综述

对图推理(RoG):忠实可解释的大语言模型推理方法(ICLR2024) https://github.com/RManLuo/reasoning-on-graphs 推理图(Reasoning on Graphs, RoG)提出了一个计划-检索-推理框架,该…

『基础』OS-1计算机系统概述_操作系统发展历程及它的运行环境

操作系统发展历程 常考的三种操作系统对比 批操作系统脱机使用计算机;作业是分批处理的;系统内多道程序并发执行;交互能力差分时操作系统多个用户同时使用计算机;人机交互强;具有每个用户独立使用计算机的独占性&…

学习大数据DAY42 hive 分桶表

目录 分桶表 分桶表注意事项 hive 分桶表-创建分桶表 hive 排序关键字 hive 排序语句 上机练习 分桶表 分区提供一个隔离数据和优化查询的便利方式。不过,并非所有的数据集都可形 成合理的分区。对于一张表或者分区,Hive 可以进一步组织成桶&…

8.21T1 草莓蛋糕(拆max + 权值线段树)

http://cplusoj.com/d/senior/p/NODSX2302A 看到式子: 我们就应该想到拆max 若 我们可以整理推出: 记: 由 L L L 算 C C C,我们满足 h a ≤ h b h_a\le h_b ha​≤hb​,找 c c c 的最小值 C C C 算 L L L 同…

05、Redis实战:优惠券秒杀、全局唯一ID、超买乐观悲观锁、一人一单逻辑、分布式锁、分布式锁的原子性

3、优惠卷秒杀 3.1 -全局唯一ID 每个店铺都可以发布优惠券: 当用户抢购时,就会生成订单并保存到tb_voucher_order这张表中,而订单表如果使用数据库自增ID就存在一些问题: id的规律性太明显受单表数据量的限制 场景分析&#x…

第2章-02-网页中的Document元素

🏆作者简介,黑夜开发者,CSDN领军人物,全栈领域优质创作者✌,CSDN博客专家,阿里云社区专家博主,2023年CSDN全站百大博主。 🏆数年电商行业从业经验,历任核心研发工程师,项目技术负责人。 🏆本文已收录于专栏:Web爬虫入门与实战精讲,后续完整更新内容如下。 文章…

2024思维导图工具评测:性能、易用性全面对比

现在工作日常要处理的数据纷繁复杂,如何高效地组织、理解和记忆这些信息,成为了每个人都需要面对的挑战。不知道你有没有尝试过使用思维导图软件呢?这次我们看看它们是如何帮助我们优化思维、提升效率的。 1.福晰思维导图 链接一下&#xf…

数据结构day04(队列 Queue 循环队列、链式队列)

目录 【1】队列 Queue 1》 队列的定义 2》循环队列 3》链式队列 【1】队列 Queue 1》 队列的定义 队列(queue)是只允许在一端进行插入操作,而在另一端进行删除操作的线性表。 队列是一种先进先出(First In First Out&#xf…

Day23 第十站 文件IO的多路复用

#include <myhead.h>void insert_client(int *client_arr,int *len,int client) {//client_arr[n]{3,4} len&client_count,client_count2;//添加 5 client_arr[2(*len)]5(client)client_arr[*len]client;(*len); } int find_client(int *client_arr,int len,int clie…

Spring DI 数据类型—— set 方法注入

首先新建项目&#xff0c;可参考 初识IDEA、模拟三层--控制层、业务层和数据访问层 一、spring 环境搭建 &#xff08;一&#xff09;pom.xml 导相关坐标 <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.or…

代码随想录算法训练营第二十二天| 77. 组合 216.组合总和III 17.电话号码的字母组合

77. 组合 题目&#xff1a; 给定两个整数 n 和 k&#xff0c;返回范围 [1, n] 中所有可能的 k 个数的组合。 你可以按 任何顺序 返回答案。 示例 1&#xff1a; 输入&#xff1a;n 4, k 2 输出&#xff1a; [[2,4],[3,4],[2,3],[1,2],[1,3],[1,4], ] 示例 2&#xff1a…

VAuditDemo安装漏洞

目录 VAuditDemo安装漏洞 index.php header.php config.php lib.php install.php 分析结果 漏洞利用 第一步&#xff1a;删除install.lock文件&#xff0c;访问 install.php 抓包 第二步&#xff1a;通过审计构造payload 第三步&#xff1a;修改抓包请求内容&#x…

客户分级管理系统设计与实现(源码+lw+部署文档+讲解等)

文章目录 前言具体实现截图详细视频演示为什么选择我官方认证闲鱼玩家&#xff0c;服务很多代码文档&#xff0c;百分百好评&#xff0c;战绩可查&#xff01;&#xff01;入职于互联网大厂&#xff0c;可以交流&#xff0c;共同进步。有保障的售后 代码参考数据库参考源码获取…