vue 聊天发送 emoji

在开发项目是我们,有时候会遇到聊天发送emoji ,但是原生的emoji在不同的系统上会有不同,但是保持统一的emoji,所以只有采取一下方式展示,就是发送表情用图片的形式呈现

首先需要将用到的图片根据需要生成一个JSON文件

{
    "[微笑]": {
        "code": "[微笑]",
        "name": "微笑",
        "img_url": ""
    },
}

表情包数据,里面有图片有name还有标识,以键值对的形式,[微笑]是相当于一个表情包标识,此处灵感来源于vx
将emoji展示完成后,就需要一个能输入的并且能展示输入内容的地方 在传统 的输入框 input 或者 textarea 都是不能展示图片的,所以采用在div contenteditable=“true”

  <div class="msg">
        <div contenteditable="true"  ref="inputMessage" class="inputMessage" @keydown="handleKeyDown" @paste="handlePaste" ></div>
        <div class="send" @click="sendMessage" >
          <button >发送&ensp;<i>S</i></button>
        </div>
      </div>

在这里插入图片描述
当咱们点击表情的时候执行一下逻辑

emoji

<script setup>
import Emoji from '/public/images/Emoji.json'
import {onMounted,computed,ref} from "vue";

const emits = defineEmits(['clickFather'])
const emoji = ref()

 const getAssetsFile = (url) => {
   return new URL(`${url}`, import.meta.url).href
}
function emoji_btn(e){
    let target =e.target 
    if(target.className ==='emoji_btn'){
      emits('clickFather',target.children[0])
    }else {
      emits('clickFather',target)
    }
}

function EmojiInit(){
  for (let emojiKey in Emoji) {
    let {code,name,img_url} = Emoji[emojiKey]
    let btn = document.createElement('button')
    btn.className='emoji_btn'
    btn.title=name
    btn.innerHTML = `
      <img data-code="${code}" class="emoji-img"  src="${img_url}" alt="${name}" style="margin: 0px 1px;">
    `
    emoji.value.append(btn)
  }
}


onMounted(()=>{
  EmojiInit()
})
</script>

<template>
  <div class="emoji_box">
    <div class="emoji_body" ref="emoji" @click="emoji_btn"></div>
  </div>
 </template>
 <!--
css 此处就不展示了
-->
// 图标内部
function clickFather(params) {
  const node = params.cloneNode()
  inputMessage.value.focus()
  const selection = window.getSelection()
  if(selection.rangeCount > 0){
    const range = selection.getRangeAt(0)
    range.insertNode(node)
    const newRange = document.createRange()    
    newRange.setStartAfter(node)
    newRange.collapse(true)
    selection.removeAllRanges()
    selection.addRange(newRange)

    // 等待 DOM 更新后滚动到底部
    nextTick(() => {
        inputMessage.scrollTop = inputMessage.scrollHeight
    })
  }
}

在这里插入图片描述

相关代码结构

在这里插入图片描述
实现文字与emoji图片在一起的显示逻辑

// 使用alt+enter换行
const handleKeyDown = (event) => {
    if (event.altKey && event.key === 'Enter'){
        event.preventDefault()
        const div = document.createElement('div') 
        div.setAttribute('data-type', 'br')
        const br = document.createElement('br')
        div.appendChild(br)
        const selection = window.getSelection()
        if (selection.rangeCount > 0) {
            const range = selection.getRangeAt(0)
            range.deleteContents() 
            range.insertNode(div)
            range.setStartAfter(div)
            range.collapse(true)
            selection.removeAllRanges()
            selection.addRange(range)
        }
        nextTick(() => {
            inputMessage.scrollTop = inputMessage.scrollHeight
        })
        return

    }
    if (event.key === 'Enter') {
        event.preventDefault() // 阻止默认的回车换行行为
        sendMessage()
    }
}

发送逻辑

const sendMessage = () => {
  const div = document.createElement('div') 
  div.innerHTML = inputMessage.value.innerHTML
  div.innerHTML = replaceNestedDivs(div.innerHTML)
  div.innerHTML = div.innerHTML.replace(/<img\s+data-code="([^"]+)"[^>]*>/g, '$1')
  message.value = div.innerText
  console.log('传输数据:',message.value);
  if (message.value) {
    const obj = {
          id: 2,
          name:'张三',
          content: initMessage(message.value),
          time: getDateTimer(),
          type: 'system_user',
          avatar: 'https://www.wenpblog.com/cdn/static/header/2812.png',
      }
      messageList.value.push(obj)
      message.value = ''
      inputMessage.value.innerHTML = ''
      initScroll()
  } else {
        if (sendTipTimeout) {
            return
        }
        sendTip = true
        message.value = ''
        inputMessage.value.innerHTML = ''
        sendTipTimeout = setTimeout(() => {
            sendTip = false
            sendTipTimeout = null
        }, 1500)
    }
}

function replaceNestedDivs(html) {
    const regex = /<div\s+data-type="br">(.*?)(<img[^>]*>)?/gi
    return html.replace(regex, function (match, p1, p2) {
        const content = p1 ? p1.trim() : '' 
        const imgTag = p2 ? ` ${p2}` : ''
        return `\n${content}${imgTag}`
    })
}
//处理聊天数据
const initMessage = (text) => {
  return replaceWithEmojiImages(text)
}
// 定义一个函数来将特定格式的字符串替换为图片
function replaceWithEmojiImages(text) {
    const regex = /\[(.*?)\]/g
    return  text.replace(regex, (match, p1) =>{
        const imgSrc = `${emojiJSON[match] ? emojiJSON[match].img_url : ''}`
        return imgSrc?`<img data-code="${match}" src="${imgSrc}" class="emoji-img"/>`:`${match}`
    })
}
// 处理粘贴问题
function handlePaste(e){
    e.stopPropagation();
    e.preventDefault();
    let text = '', event = (e.originalEvent || e);
    if (event.clipboardData && event.clipboardData.getData) {
        text = event.clipboardData.getData('text/plain');
    } else if (window.clipboardData && window.clipboardData.getData) {
        text = window.clipboardData.getData('Text');
    }
    if (document.queryCommandSupported('insertText')) {
        document.execCommand('insertText', false, text);
    } else {
        document.execCommand('paste', false, text);
    }
}


传输带数据

在这里插入图片描述

展示效果

在这里插入图片描述

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值