「Unity3D」TextMeshPro使用TMP_InputField实现,输入框高度自动扩展与收缩

先看实现效果:

要实现这个效果,有三个方面的问题需要解决:

第一,输入框的高度扩展,内部子元素会随着锚点,拉伸变形------要解决这个问题,需要将内部元素改变父类,然后增加父类高度,再恢复父类,这样内部元素(如图中两个按钮),就不会随着高度增加,而拉伸变形。类似代码如下:

cs 复制代码
private void ChangeInputHeight(float heightChange)
{
    // disable the effect of parent size changes on child layouts
    this.finishRecordRT.SetParent(null);
    this.recordRT      .SetParent(null);

    if (heightChange != 0.0f)
    {
        this.inputFieldRT.SetRelativeSizeY(heightChange);
        this.inputBarHeightChange += heightChange;
    }
    else 
    {
        this.inputFieldRT.SetRelativeSizeY(-this.inputBarHeightChange);
        this.inputBarHeightChange = 0.0f;
    }

    this.finishRecordRT.SetParent(this.inputBarRT);
    this.recordRT      .SetParent(this.inputBarRT);
}

第二,注册inputField.onValueChanged 去监听,输入框的文本变化,并通过inputField.textComponent.textInfo的行变化,去判断换行发生。

cs 复制代码
private void OnInputFieldValueChanged(string text)
{
    var textInfo  = this.inputField.textComponent.textInfo;
    var lineCount = textInfo.lineCount;

    if (this.inputFieldPreTextLineCount != lineCount)
    {
        this.ChangeInputHeight(this.inputFieldLineHeight * (lineCount - this.inputFieldPreTextLineCount));
        this.inputFieldPreTextLineCount = lineCount;
    }
}

这里需要记录之前的行数,即inputFieldPreTextLineCount,然后对比变化后的行数,就可以知道是否换行,以及换了几行。

第三,InputField换行后(输入换行符),无法直接触发行数变化,需要输入一个字符后才行,所以需要自行判断,键入了换行符。但删除换行符后,就会触发行数变化,这个删除逻辑就是需要的。

cs 复制代码
    if (this.inputFieldPreTextLineCount != lineCount)
    {
        this.ChangeInputHeight(this.inputFieldLineHeight * (lineCount - this.inputFieldPreTextLineCount));
        this.inputFieldPreTextLineCount = lineCount;
    }
    else
    {
        // inputField.text equal to text, not equal to inputField.textComponent.text (textInfo)
        // when adds    chars, the textInfo.characterCount equal to the text.Length
        // when removes chars, the textInfo.characterCount is more than text.Length (because adds extra space \u200B)
        if (textInfo.characterCount == text.Length && text[^1] == '\n')
        {
            this.ChangeInputHeight(this.inputFieldLineHeight);
            ++this.inputFieldPreTextLineCount;
        }
    }

这里增加的逻辑就是:当行数没变的时候,需要检测text的最后一个字符,是否是换行符------如果是,就增加高度,而删除则可以通过行数变化处理,包括一次删除多行。

为什么要加一个textInfo.characterCount == text.Length的判断?

因为,只有在输入字符的时候,才会成立------而删除字符的时候不成立,这样就会剔除,删除换行符是最后一个,但还没删除它,此时行数也没变化。

而之所以,删除字符的时候不成立------是因为,在删除字符的时候,InputField 的代码显示,会增加看不见的标识字符,即**\u200B** ------因此,inputField.textComponent.text(textInfo) 与**text(inputField.text)**的内容会不一致。

相关推荐
Yasin Chen7 小时前
Unity TMP_SDF 分析(二)数据来源2
unity·游戏引擎
Mao_Hui7 小时前
Unity3d实时读取Modbus RTU数据
开发语言·嵌入式硬件·unity·c#
心疼你的一切10 小时前
【Unity-MCP完全指南:从零开始构建AI游戏开发助手】
人工智能·unity·ai·游戏引擎·aigc·mcp
示申○言舌11 小时前
基于知识库(RAG)系统打造由大模型(LLM)驱动NPC游戏的技术设想
游戏·unity·大模型·知识库·rag·智能npc·npc记忆
国家一级摸鱼选手1 天前
MCP(Model Context Protocol)学习笔记
unity·ai·mcp
会思考的猴子1 天前
Unity3D发布后软件界面右下角出现Trial Version
unity
ellis19701 天前
Unity资源管理框架Addressables[五] 构建
unity
派葛穆1 天前
Unity-鼠标悬停改变图像位置
unity·计算机外设·交互
avi91111 天前
图例设计软件:Focusky,VisualComponents,图片字风格化等试用和推荐,最终还是回到Unity 游戏代码
unity·ai·aigc·游戏开发·设计工具·信息图
张老师带你学1 天前
unity道具,健身房资源
科技·游戏·unity·游戏引擎·模型