(已解决)Unicode高位码点emoji表情无法解析

问题描述

我在制作游戏论坛项目,希望制作一个表情库,我参考了菜鸟的emoji手册,并且使fromCharCode函数来进行字符串转换,但是经过我的测试,对于超过5位数的高位码点,无法正常解析。

源码:

<template>
  <div class="container">
    <!-- 网页底部的表情按钮 -->
    <el-button
      type="text"
      @click="drawerVisible = true"
    >表情</el-button>

    <!-- 使用Element-UI的抽屉组件 -->
    <el-drawer
      :visible.sync="drawerVisible"
      size="300px"
      position="right"
      :before-close="handleClose"
    >
      <!-- 遍历并显示可点击复制的特殊Unicode字符 -->
      <div
        v-for="(char, index) in specialChars"
        :key="index"
        @click="copyToClipboard(char)"
        class="emoji-item"
      >{{ charDisplay[index] }}</div>
    </el-drawer>

    <!-- 输入框 -->
    <input
      type="text"
      v-model="copiedText"
      ref="input"
    />
  </div>
</template>

<script>
export default {
  data() {
    return {
      drawerVisible: false,
      copiedText: "",
      // 新增数组存储多个Unicode码点表示的表情
      specialChars: [
        String.fromCharCode(8986),
        String.fromCharCode(8987),
        String.fromCharCode(9193),
        String.fromCharCode(9194),
        String.fromCharCode(9195),
        String.fromCharCode(128155),
      ],
      // 将码点转换为显示字符的数组
      charDisplay: [
        String.fromCharCode(8986),
        String.fromCharCode(8987),
        String.fromCharCode(9193),
        String.fromCharCode(9194),
        String.fromCharCode(9195),
        String.fromCharCode(128155),
      ],
    };
  },
  methods: {
    handleClose(done) {
      // 关闭前的处理逻辑(如果需要的话)
      done();
    },

    copyToClipboard(text) {
      this.copiedText = text;
      const input = this.$refs.input;
      input.focus();
      input.select();
      document.execCommand("copy");
      input.blur();
    },
  },
};
</script>

<style>
.emoji-item {
  display: inline-block;
  width: 40px;
  height: 40px;
  text-align: center;
  line-height: 40px;
  margin: 5px;
}
</style>

问题产生原理及解决方法

fromCharCode函数可以把Unicode码点转换成对应的字符,以便在页面上显示表情符号。
但是对于超过 \ufff 的高位码点,就是我说的六位数码点,实际上对于六位数的Unicode已经不说属于码点范围,而是一个 Unicode 字符簇。需要我们用两个UTF-16编码单元,也就是四个十六进制数字来表示,查询资料后,这个被叫做代理对。

解决办法:通过次转换拆分后再进行字符串转换,就可以正常渲染了
 

let highSurrogate = Math.floor((codePoint - 0x10000) / 0x400) + 0xD800;
let lowSurrogate = (codePoint - 0x10000) % 0x400 + 0xDC00;
let emojiCharacter = String.fromCharCode(highSurrogate, lowSurrogate);

例:126980高位编码的红中表情已被正常显示

全部修改后源码:

<template>
  <div class="container">
    <!-- 网页底部的表情按钮 -->
    <el-button
      type="text"
      @click="drawerVisible = true"
    >表情</el-button>

    <!-- 使用Element-UI的抽屉组件 -->
    <el-drawer
      :visible.sync="drawerVisible"
      size="300px"
      position="right"
      :before-close="handleClose"
    >
      <!-- 遍历并显示可点击复制的特殊Unicode字符 -->
      <div
        v-for="(char, index) in renderedSpecialChars"
        :key="index"
        @click="copyToClipboard(char)"
        class="emoji-item"
      >{{ char }}</div>
    </el-drawer>

    <!-- 输入框 -->
    <input
      type="text"
      v-model="copiedText"
      ref="input"
    />
  </div>
</template>

<script>
export default {
  data() {
    const highCodePointsDec = [
      126980, // 示例:笑脸表情
      // 其他高位Unicode码点...
    ];

    // 将十进制Unicode码点转换为可以渲染的字符串
    const specialChars = highCodePointsDec.map((codePoint) => {
      if (codePoint > 0xffff) {
        // 处理高位码点
        const highSurrogate =
          Math.floor((codePoint - 0x10000) / 0x400) + 0xd800;
        const lowSurrogate = ((codePoint - 0x10000) % 0x400) + 0xdc00;
        return String.fromCharCode(highSurrogate, lowSurrogate);
      } else {
        return String.fromCharCode(codePoint); // BMP内直接转换
      }
    });

    return {
      drawerVisible: false,
      copiedText: "",
      renderedSpecialChars: specialChars,
      charDisplay: specialChars, // 在此情况下,由于已经是渲染后字符,无需额外转换
    };
  },
  methods: {
    handleClose(done) {
      // 关闭前的处理逻辑(如果需要的话)
      done();
    },

    copyToClipboard(text) {
      this.copiedText = text;
      const input = this.$refs.input;
      input.focus();
      input.select();
      document.execCommand("copy");
      input.blur();
    },
  },
};
</script>

<style>
.emoji-item {
  display: inline-block;
  width: 40px;
  height: 40px;
  text-align: center;
  line-height: 40px;
  margin: 5px;
}
</style>

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

WiFi模块引领零售数字化转型:智能零售体验再定义

随着科技的不断发展&#xff0c;零售业正迎来一场数字化转型的浪潮。在这个变革过程中&#xff0c;WiFi模块成为零售业中的关键技术&#xff0c;为商家提供了丰富的数字化工具&#xff0c;打造了更智能、便捷、个性化的零售体验。本文将深入探讨WiFi模块在零售数字化转型中的关…

学习使用paddle来构造hrnet网络模型

1、首先阅读了hrnet的网络结构分析&#xff0c;了解到了网络构造如下&#xff1a; 参考博文姿态估计之2D人体姿态估计 - &#xff08;HRNet&#xff09;Deep High-Resolution Representation Learning for Human Pose Estimation&#xff08;多家综合&#xff09;-CSDN博客 最…

Python的With...As 语句:优雅管理资源的技术探索【第116篇—With...As 语句】

Python的With…As 语句&#xff1a;优雅管理资源的技术探索 在Python编程中&#xff0c;with...as语句是一项强大而优雅的功能&#xff0c;用于管理资源&#xff0c;如文件、网络连接、数据库连接等。本文将深入介绍with...as语句的用法、其工作原理&#xff0c;并通过代码示例…

光子嫩肤仪面罩控制器PCB电路中升压恒流芯片FP7208的应用

护肤已经成为现代人日常生活中不可或缺的一部分&#xff0c;尤其对于追求美丽肌肤的人来说&#xff0c;寻找一款适合自己的护肤利器至关重要。 光子嫩肤仪作为一种高科技美容仪器&#xff0c;受到越来越多人的追捧。其中&#xff0c;FP7208LED升压驱动IC作为其核心部件之一&am…

TQ15EG开发板教程:创建运行petalinux2019.1

工程网盘链接&#xff1a;https://pan.baidu.com/s/1vFRpzmbifXt7GypU9aKjeg 提取码&#xff1a;0ylh 首先需要使用与petalinux相同版本的vivado创建工程&#xff0c;与之前不同的是在创建硬件设计时需要勾选上添加bit文件&#xff0c;所以要在生成bit文件之后再创建硬件设计…

谷粒商城【成神路】-【8】——商品上架

目录 1.数据模型封装 1.es数据模型 2.将es数据模型封装为JAVA bean 3.根据前端发送请求,编写controller 2.模型实现 2.1服务controller 2.2服务service 2.3服务远程调用接口 2.4检索服务controller 2.5检索服务保存到es 2.6库存查询服务 1.数据模型封装 1.es数据模…

银河麒麟之Workstation安装

一、VMware Workstation简介 VMware Workstation是一款由VMware公司开发的虚拟化软件&#xff0c;它允许用户在一台物理计算机上运行多个操作系统&#xff0c;并在每个操作系统中运行多个虚拟机。VMware Workstation提供了一个可视化的用户界面&#xff0c;使用户可以轻松创建、…

纵行科技荣登“中国物联网企业投资价值50强”、“中国物联网行业创新产品榜”

近日&#xff0c;由深圳市物联传媒有限公司、AIoT星图研究院、IOTE组委会、深圳市物联网产业协会主办的“2023‘物联之星’中国物联网行业年度榜单”评选结果正式公布。厦门纵行信息科技有限公司&#xff08;以下简称“纵行科技”&#xff09;最终从500多家参评企业中脱颖而出&…

数据库-ODBC操作

一、ODBC 数据源配置 打开ODBC数据源管理器&#xff1a; 在Windows搜索栏中键入“ODBC数据源”并选择“ODBC数据源(64位)”&#xff08;如果你的系统是64位的&#xff09;。如果你的系统是32位的&#xff0c;你可以选择“ODBC数据源(32位)”。或者&#xff0c;你可以在控制面…

使用DockerFile构建Tomcat镜像

1、准备镜像文件tomcat压缩包&#xff0c;jdk的压缩包 tomcat链接&#xff1a;https://pan.baidu.com/s/1Xpecb-BSGR2sdxSL7FDtBw?pwd1234 提取码&#xff1a;1234 jdk链接&#xff1a;https://pan.baidu.com/s/1mQHInn27j1I9uuuicBsyAA?pwd1234 提取码&#xff1a;1234 …

网工学习 DHCP配置-接口模式

网工学习 DHCP配置-接口模式 学习DHCP总是看到&#xff0c;接口模式、全局模式、中继模式。理解起来也不困难&#xff0c;但是自己动手操作起来全是问号。跟着老师视频配置啥问题没有&#xff0c;自己组建网络环境配置就是不通&#xff0c;悲催。今天总结一下我学习接口模式的…

信息系统安全与对抗-作业2

目录 1、使用自己姓名拼音创建一个账户&#xff0c; 并使用命令和图形化查看 2、使用自己拼音打头字母创建一个隐藏账户 &#xff0c;并使用命令和图形化查看 3、使用命令启动 telnet 服务 4、使用命令打开防火墙 23 端口 5、熟悉LINUX系统&#xff0c;使用命令行创建用户…

docker基线安全修复和容器逃逸修复

一、docker安全基线存在的问题和修复建议 1、将容器的根文件系统挂载为只读 修复建议&#xff1a; 添加“ --read-only”标志&#xff0c;以允许将容器的根文件系统挂载为只读。 可以将其与卷结合使用&#xff0c;以强制容器的过程仅写入要保留的位置。 可以使用命令&#x…

不同控制方式下的无人机二维码识别降落对比

无人机技术的快速发展正在推动众多行业的革新&#xff0c;从农业监测、灾害响应到城市规划和物流配送&#xff0c;无人机的应用前景无限广阔。随着应用场景的多样化&#xff0c;无人机精准降落成为一大挑战。基于PX4飞控固件和ROS系统的开源自主无人机平台Prometheus应运而生。…

【Linux】进程间通信之共享内存

文章目录 引入共享内存的原理共享内存的相关接口shmget()shmat()shmdt()shmctl() 共享内存的简单使用共享内存的特点 引入 进程间通信&#xff0c;顾名思义就是一个进程和另一个进程之间进行对话&#xff0c;以此完成数据传输、资源共享、通知事件或进程控制等。 众所周知&am…

Vscode安装,ssh插件与配置

原因 发现很多新人在练习linux&#xff0c;可是只有windows机的时候&#xff0c;一般都是下载虚拟机&#xff0c;然后在虚拟机上安装ubuntu等linux平台。每次需要在linux中写代码&#xff0c;就打开ubuntu&#xff0c;然后在终端上用vim写代码&#xff0c;或者先编辑代码文本&…

hook函数——useReducer

目录 1.useReducer定义2.useReducer用法3.useState和useReducer区别 1.useReducer定义 const [state, dispatch] useReducer(reducer, initialArg, init?) reducer&#xff1a;用于更新 state 的纯函数。参数为 state 和 action&#xff0c;返回值是更新后的 state。state …

JVM相关问题

JVM相关问题 一、Java继承时父子类的初始化顺序是怎样的&#xff1f;二、JVM类加载的双亲委派模型&#xff1f;三、JDK为什么要设计双亲委派模型&#xff0c;有什么好处&#xff1f;四、可以打破JVM双亲委派模型吗&#xff1f;如何打破JVM双亲委派模型&#xff1f;五、什么是内…

Matlab|基于Logistic函数负荷需求响应

目录 1 基于Logistic函数的负荷转移率模型 2 程序示例 3 效果图 4 下载链接 负荷需求响应模型种类较多&#xff0c;有电价型和激励型等类型&#xff0c;本次和大家分享一个基于Logistic函数的负荷转移率模型&#xff0c;该模型属于电价型&#xff0c;由于该方法使用的较少&a…

C++ //练习 10.24 给定一个string,使用bind和check_size在一个int的vector中查找第一个大于string长度的值。

C Primer&#xff08;第5版&#xff09; 练习 10.24 练习 10.24 给定一个string&#xff0c;使用bind和check_size在一个int的vector中查找第一个大于string长度的值。。 环境&#xff1a;Linux Ubuntu&#xff08;云服务器&#xff09; 工具&#xff1a;vim 代码块 /*****…