首页 > 其他分享 >Unity使用TextMeshPro实现聊天图文混排

Unity使用TextMeshPro实现聊天图文混排

时间:2024-09-19 09:37:27浏览次数:1  
标签:me render text utils TextMeshPro Unity other 图文混排 local

本文来自:https://developer.aliyun.com/article/1066623

1.文字自适应问题。
2.图文混排问题。

UI界面

1.创建滑动列表

首先创建一个可以上下滑动的列表,命名为chat_scroll
image.png

2.创建聊天预制

因为聊天是两人以上的,自己的聊天显示在右侧,别人的聊天消息显示在左侧。因此需要制作两个聊天框,分别代表自己的其他人,如下图的me和other。
创建的聊天预制如下:
image.png

3.使用TextMeshPro

注意:为了能实现表情加文字的效果,我们需要使用TextMeshPro作为消息显示
image.png

4.添加Layout Element

为聊天预制item添加Layout Element组件,如下:
image.png

Layout Element的Preferred Width和preferred Height值与me和other的长宽保持一致:
image.png

6.设置锚点

接着,我们需要设置游戏对象的锚点,如下图:
34.gif
345.gif

代码控制显示

1.获取游戏对象

首先我们需要获取到需要用到的聊天item的游戏对象:

local render = {}
    render.go = poolManager.AddObj("Modules/friendchat/worldChatItem.prefab", ui.world_content)
    local trans = render.go.transform
    --我
    render.me = utils.findobj(render.go, "me")
    render.me_headFrame = utils.findimage(render.me, "head/headFrame")
    render.me_headImg = utils.findimage(render.me, "head/headImg")
    render.me_lv = utils.findtext(render.me, "level/lv")
    render.me_name = utils.findtext(render.me, "topDes/name")
    render.me_sectionImg = utils.findimage(render.me, "topDes/sectionImg")
    render.me_sectionDes = utils.findtext(render.me, "topDes/sectionImg/sectionDes")
    render.me_msgTxt = utils.findtmp_text(render.me, "msgBg/msgTxt")
    --对方
    render.other = utils.findobj(render.go, "other")
    render.other_headFrame = utils.findimage(render.other, "head/headFrame")
    render.other_headImg = utils.findimage(render.other, "head/headImg")
    render.other_lv = utils.findtext(render.other, "level/lv")
    render.other_name = utils.findtext(render.other, "topDes/name")--名字
    render.other_sectionImg = utils.findimage(render.other, "topDes/sectionImg")--段位img
    render.other_sectionDes = utils.findtext(render.other, "topDes/sectionImg/sectionDes")--段位秒速
    render.other_msgTxt = utils.findtmp_text(render.other, "msgBg/msgTxt")

2.游戏对象刷新

刷新每个item的信息:

if message:GetUid() == playerDataManager.GetUid() then--判断消息是否是本人发送
        --自己
        local sender = message:GetSender()
        local friend = friendData:New(sender)
        render.me_headFrame.sprite = utils.loadsprite("frame", friend:GetFrame())
        render.me_headImg.sprite = utils.loadsprite("head", friend:GetHead())
        render.me_lv.text = friend:GetLevel()
        render.me_name.text = friend:GetName()
        render.me_sectionImg.sprite = utils.loadsprite("friendChat", friend:GetFrameImg())
        render.me_sectionDes.text = friend:GetSectionName()
        render.me_msgTxt.text = message:GetContent()
        CalcWorldChatWidthHight(render.me)
        render.me:SetActive(true)
        render.other:SetActive(false)
    else
        SetActive(render.other_addRoom_Btn,message:GetSType() == 102)--邀请按钮是否显示
        local sender = message:GetSender()
        local friend = friendData:New(sender)
        render.other_headFrame.sprite = utils.loadsprite("frame", friend:GetFrame())
        render.other_headImg.sprite = utils.loadsprite("head", friend:GetHead())
        render.other_lv.text = friend:GetLevel()
        render.other_name.text = friend:GetName()
        render.other_sectionImg.sprite = utils.loadsprite("friendChat", friend:GetFrameImg())
        render.other_sectionDes.text = friend:GetSectionName()
        render.other_msgTxt.text = message:GetContent()
        CalcWorldChatWidthHight(render.other)
        render.me:SetActive(false)
        render.other:SetActive(true)
    end

3.文本自适应

注意上面代码中,使用到了CalcWorldChatWidthHight方法,CalcWorldChatWidthHight方法如下:

--世界聊天计算宽高
local function CalcWorldChatWidthHight(obj)
    local htext = utils.findtmp_text(obj, "msgBg/msgTxt").preferredHeight --22
    local wtext = utils.findtmp_text(obj, "msgBg/msgTxt").preferredWidth--445
    local bg = utils.findobj(obj, "msgBg")
    local msg_rect = utils.findrect(obj, "msgBg/msgTxt")
    local parent_rect = obj.transform.parent
    if wtext < 470 then --一行
        bg:GetComponent("RectTransform").sizeDelta = Vector2.New(wtext+30, 47)
    else
        printlog(utils.findtmp_text(obj, "msgBg/msgTxt").text,"内容")
        local num = math.ceil(wtext/470)--行数
        local hight = (num-2)*30+60
        bg:GetComponent("RectTransform").sizeDelta = Vector2.New(491,hight)--503
    end
    local sizeDelta = bg:GetComponent("RectTransform").sizeDelta
    msg_rect.sizeDelta = Vector2.New(sizeDelta.x-30,sizeDelta.y)
    obj:GetComponent("RectTransform").sizeDelta = Vector2.New(sizeDelta.x-15,sizeDelta.y+40)
    parent_rect:GetComponent("LayoutElement").preferredHeight = sizeDelta.y + 40
end

说明:
470:一行的Text最大宽度为470
30:聊天内容背景宽度= Text实际宽度+30(30是为了让文字两边留有间隙,看着美观)
47:一行文字时,背景高度(可根据实际情况调整)
...
注意:代码中用到的数据都可以根据实际情况调整!

CalcWorldChatWidthHight方法的作用是实现文本宽高随着文字内容的变化而变化。支持一行显示文字是,文本宽度自适应;支持多行时,高度也能自适应。

纯文本聊天预览效果

image.png

加入表情

1.制作表情图集及配置

打开TexturePackerGUI,拖进所有表情,修改Framework为Json(Array)并设置Trim Model为None,输出路径按照实际情况设置:
image.png

2.导入表情

将步骤1生成的两个文件放入工程新建文件夹Emoji中:
image.png

3.生成Assets资源

选中Emjoji—emoji—选中所有表情资源—Create—TextMeshPro—SpriteAsset
image.png
生成的Assets资源如下:
image.png

4.为TextMeshPro指定默认资源

将生成的Assets资源指定给TextMeshPro默认资源,如下:
image.png

5.表情实现

当点击某个表情时,显示方法如下:

_input.text = string.format("<sprite=%s>",dx)

6。运行,完美。

 

标签:me,render,text,utils,TextMeshPro,Unity,other,图文混排,local
From: https://www.cnblogs.com/guangzhiruijie/p/18419835

相关文章

  • Unity UI控件用法汇总
    利用LoopListView实现Banner循环列表,且默认中间节点为默认节点:  1.给ScrollRect节点添加LoopListView组件,并勾选ItemSnapEnable为true。  2.通过LoopListView.InitListView初始化时,totalCount需要传-1.  3.OnGetItemByIndex的回调参数index以(Int32.MinValue,Int32.MaxVa......
  • WORD图文混排复制到XHEDITOR图片不显示
    编辑器:xhEditor前端:vue2,vue3,vue-cli,html5后端:asp,php,jsp,springboot,asp.net,.netcore功能:复制粘贴word内容图片,要求:免费,开源,技术支持最近搞定块挻火的,今天早上又有网友加我微信私聊,也是想了解这块的技术和方案。昨天晚上论坛里面的一个网友给我发私信,想了解一下如何解......
  • 手把手教你写一个Unity对象池
    对象池,我在最初学习时觉得这一定是个非常复杂的东西,但其实从现在看过去,对象池其实非常简单。首先对象池也叫做缓存池,是常见的一种优化内存的手段(划重点,常用,一定要学会哦)再来看,对象池主要用于面对以下问题:1.对象的频繁创建频繁的实例化对象会带来一定的性能开销2.对象的频......
  • 【Unity精品源码】打造甜蜜的三消游戏:Candy Match 3 Kit
    最近总熬夜,肝不好,大家都叫我小心肝。......
  • 在Unity UI中实现UILineRenderer组件绘制线条
    背景介绍        在Unity的UI系统中,绘制线条并不像在3D世界中那样直观(使用Unity自带的LineRender组件在UI中连线并不方便,它在三维中更合适)。没有内置的工具来处理这种需求。如果你希望在UI元素之间绘制连接线(例如在UI上连接不同的图标或控件),需要自己编写逻辑。 ......
  • Unity3D下如何播放RTSP流?
    技术背景在Unity3D中直接播放RTSP(RealTimeStreamingProtocol)流并不直接支持,因为Unity的内置多媒体组件(如AudioSource和VideoPlayer)主要设计用于处理本地文件或HTTP流,而不直接支持RTSP。所以,你可以通过一些间接的方法来实现RTSP流的播放,或者通过比较成熟的第三方插件来播。可选方......
  • Unity3D下如何播放RTSP流?
    技术背景在Unity3D中直接播放RTSP(RealTimeStreamingProtocol)流并不直接支持,因为Unity的内置多媒体组件(如AudioSource和VideoPlayer)主要设计用于处理本地文件或HTTP流,而不直接支持RTSP。所以,你可以通过一些间接的方法来实现RTSP流的播放,或者通过比较成熟的第三方插件来播。......
  • unity人工智能游戏、源码、教程(中秋特别版),完全免费和开源
    任何人不要和我说话,我不想跟任何人说话,因为我对现实世界的人类不感兴趣。谁跟我说话,我都不会理睬的。(一)游戏简介三维虚拟世界的人工智能对话。完全免费、完全开源、完整详细、通俗易懂。我把游戏、游戏源码、教程(三合一)放到了夸克网盘:链接:https://pan.quark.cn/s/65e22d51c1b......
  • Unity实战案例全解析 :PVZ 植物脚本分析
             植物都继承了Pants脚本,但是我因为没注意听讲,把Pants也挂在植物上了,所以子类的PlantEnableUpdate和PlantDisableUpdate抢不过父类,无法正确触发动画,我还找不到哪里出了问题,所以就使用了携程加while强行触发了,但是经过对源码和工程的分析比对,我发现了问题所在,......
  • Unity中的三种渲染路径
    Unity中的渲染路径Unity的渲染路径在Unity里,渲染路径(RenderingPath)决定了光照是如何应用到UnityShader中的。因此,我们只有为Shader正确地选择和设置了需要的渲染路径,该shader的光照计算才可以被正确执行。unity中的渲染路径:ForwardRenderingPath(向前渲染路径)DeferredR......