判断是否输入emoji

// 判断是否输入emoji
    function isEmojiCharacter(substring) {
    for ( var i = 0; i < substring.length; i++) {
        var hs = substring.charCodeAt(i);
        if (0xd800 <= hs && hs <= 0xdbff) {
            if (substring.length > 1) {
                var ls = substring.charCodeAt(i + 1);
                var uc = ((hs - 0xd800) * 0x400) + (ls - 0xdc00) + 0x10000;
                if (0x1d000 <= uc && uc <= 0x1f77f) {
                    return true;
                }
            }
        } else if (substring.length > 1) {
            var ls = substring.charCodeAt(i + 1);
            if (ls == 0x20e3) {
                return true;
            }
        } else {
            if (0x2100 <= hs && hs <= 0x27ff) {
                return true;
            } else if (0x2B05 <= hs && hs <= 0x2b07) {
                return true;
            } else if (0x2934 <= hs && hs <= 0x2935) {
                return true;
            } else if (0x3297 <= hs && hs <= 0x3299) {
                return true;
            } else if (hs == 0xa9 || hs == 0xae || hs == 0x303d || hs == 0x3030
                    || hs == 0x2b55 || hs == 0x2b1c || hs == 0x2b1b
                    || hs == 0x2b50) {
                return true;
            }
        }
    }
}

根据emoji.unified显示emoji表情

/* 利用findSurrogatePair结合String.fromCharCode()来对unicode编码进行处理 */
    function findSurrogatePair(point: any): any {
      // assumes point > 0xffff
      const offset = point - 0x10000,
        lead = 0xd800 + (offset >> 10),
        trail = 0xdc00 + (offset & 0x3ff);
      return [lead.toString(16), trail.toString(16)];
    }
    let afterEmoji: any;
    if (emoji.unified.length === 4) {
      afterEmoji = String.fromCharCode(+'0x' + emoji.unified);
      alert(afterEmoji);
    } else {
      const beforeEmoji = findSurrogatePair('0x' + emoji.unified);
      afterEmoji = beforeEmoji.map(function (item: any, index: any): any {
        const str = new String('0x' + item.trim());
        return str;
      });
      afterEmoji = String.fromCharCode(afterEmoji[0], afterEmoji[1]);
      alert(afterEmoji);
    }
    const expressionContent: any = afterEmoji;

emoji.unified哪些范围的可以在安卓机上显示(只测了部分)

<1f920只测了一部分
在安卓机上可以显示
1f920
1f930~1f93e
1f940~1f94b
1f950~1f95e
1f980~1f991
1f9c0
在安卓机上可以显示
>=20000时在pc端显示汉字等字,但是在安卓机上也没有显示出来,还是带叉的矩形框(只测了部分)
2049-fe0f这个也可以,对应的是一个感叹号加一个问号
四位数的也只测了几个,但是这几个也可以: 2753对应的是一个红色的问号 2754对应的是一个白色的问号,建议最好使用时再测一下,以防有所改动。

上面提到了charCodeAtfromCharCode
charCodeAt() 是普通字符串的方法,返回字符串指定位置的Unicode码点(十进制)表示。

var str=“abc”; console.log(str.charCodeAt(0);
 97

fromCharCode() 它是String对象提供的静态方法(即定义在对象实例的方法),该方法的参数是一个或者多个值,代表unicode码点,返回值是这些码点组成的字符串。

String.fromCharCode(104,101,108,108,111);
 hello
 String.fromCharCode(97);
 a

emoji的使用

import {emojiIndex, Emoji} from 'emoji-mart'
const emINDEX: any = emojiIndex; //用于遍历表情数组,进行高级查询
1、存放表情在对应的图片中的位置posArray
2、表情集合emojiPoformaceArray
3、表情对应名字集合emojiNameArray

发送信息模块
首先得有一个用于高级查询的表情数组(如果明确要哪几个表情,没必要使用这种,可以直接使用,具体参照github对应的文章就可以。如果没有明确要哪个表情,就可以使用这种方式。
public constructor(props: IPropsSendAndShow){
/* 关于表情的部分为了防止初加载时没有显示出来对应的图片位置,对其设置了内容(即表情),后来加载出来后,需要清空之前的内容,显示图片就行 */
    const timer = setInterval(function (): void {
      if ($('#emoticonPackage .EmotionList-con button span')) {
        if ($('#emoticonPackage .EmotionList-con button span').css('background-position')) {
          $('#emoticonPackage .EmotionList-con button span').html('');
          clearInterval(timer);
        }
      }
    }, 1000);
this.state={arrayExpressSearch ['question','high five','hand','bell','+1','smile','good','congra','sad','hug','face_with_thermometer','presnt','book','paper',....]}
const aExprssion: any = this.state.arrayExpressSearch;
aExprssion.map((item: any,index: any): any=>{
const result1=emINDEX.search(item).map((o:any,Index: any): any=>{
//因为有些表情部分安卓手机显示不出来,所以做了处理
  if ((o.unified <= '1f920' || o.unified >= '1f930' && o.unified <= '1f93e' || o.unified >= '1f940' && o.unified <= '1f94b' || o.unified >= '1f950' && o.unified <= '1f95e' || o.unified >= '1f980' && o.unified <= '1f991' || o.unified === '1f9c0') && o.unified < '20000' && o.unified !== "1f91f" && o.unified !== "1f481-200d-2642-fe0f" && o.unified !== "1f64b-200d-2642-fe0f" && o.unified !== "1f481-200d-2640-fe0f" && o.unified !== "1f64b-200d-2640-fe0f" || o.unified.length === 4) {
          emojiPoformaceArray.push(o.native);
          emojiNameArray.push(o.name);
 /*  const emojiProperty: any = { id: o['id'], skin: 6 };
           return (
             <em className="EmotionList-item" key={o['id']}>
               <Emoji emoji={emojiProperty} size={30} tooltip={true} set={'emojione' ? 'emojione' : 'apple' ? 'apple' : 'facebook' ? 'facebook' : 'google' ? 'google' : 'messenger' ? 'messenger' : 'twitter'}
                 backgroundImageFn={((set, sheetSize) => require('./images/appleExpress.png'))} onClick={(emoji, event) => that.expressionHandle(emoji)} />
             </em>
           ); */
        }
      });
    //  resultexpression.push(result1); // 用于获取所有展示出的表情在图片中的位置集合posArray
    });

public componentDidMount(): void{
    const oSpan = $('.EmotionList-con span');
    const oTSpan = Array.prototype.slice.call(oSpan);
    const arr: any = [];
    oTSpan.map((item: any, index: any) => {
      arr.push($(item).css('background-position'));
    });
    console.log(arr);//获取到要展示的表情对应的位置,就可以直接写死了posArray(其值就是arr)
}
       生成的表情块
        <aside id="emoticonPackage">
          <div className="emoticonPackagecontainer">
            <div className="EmotionList">
              <div className="EmotionList-con" style={{ top: '0px' }}>
                {emojiPoformaceArray.map((item: any, index: any) => {
                  return (
                    <button title={emojiNameArray[index]} className="emoji-mart-emoji" type="button" key={index} onClick={() => that.expressionHandle(item)}>
                      <span style={{ backgroundPosition: posArray[index] }}>{item}</span>
                    </button>
                  );
                })}
              </div>
	/*  最初使用下边的dom,是为了在componnetDidMount中获取所有展示出的表情在图片中的位置集合posArray,然后就可以使用上边的了 */
              {/*   <div className="EmotionList-con" style={{ top: '0px' }}>{resultexpression}</div> */} 
            </div>
          </div>
        </aside>
展示消息模块
 searchSubStr查找一个字符串中的某个字串的位置
 posLenIndex信息中的表情的在信息中对应的位置、表情的长度、表情集合中对应的索引位(用作查找到图片位置数组对应的数据)
 三种情况:发的信息中有表情存在、送花、信息中没有表情
 
  public render(): React.ReactNode {
    return (
      <div className="center" id="dialogue">
        {this.state.message.map((item, index) => {
          let content = JSON.parse(item)["content"][0];
          let contentCopy: any = '';
          /**
           *查找一个字符串中的所有子串的位置
           * * */
          function searchSubStr(str: any, subStr: any, positions: any): void {
            let pos = str.indexOf(subStr);
            while (pos > -1) {
              positions.push(pos);
              pos = str.indexOf(subStr, subStr.length + pos);
            }
          }
          const posLenIndex: any = []; // 信息中包含的所有表情对应的在信息中的位置、表情的长度、对应的表情数组的索引位(用作查找到图片位置数组对应的数据)
          emojiPoformaceArray.map((item: any, index: any) => {
            const xh = content.indexOf(item);
            if (xh !== -1) {
              const positions = new Array();
              searchSubStr(content, item, positions);
              posLenIndex.push({
                posArr: positions,
                len: item.length,
                emoitionIndex: index,
              });
            }
          });
          if (posLenIndex.length > 0) { // 发的信息中有表情存在
            posLenIndex.map((a: any, b: any) => {
              const sen = `<span style="background-position:${posArray[a.emoitionIndex]}"></span>`;
              a.posArr.map((c: any, d: any) => { // 遍历是因为在上面emojiPoformaceArray的遍历中用g也不能全部替换,所以就在生成的posLenIndex数组中遍历
                const newItem = new RegExp(emojiPoformaceArray[a.emoitionIndex], 'g'); // 带上g谁知也没有全部替换,所以就在每次变化后重新赋给content
                contentCopy = content.replace(newItem, sen);
                content = contentCopy;
              });
            });
          } else if (content === '?') { // 送花
            contentCopy = `<span class='special' style='background-size: 30px 30px;width: 30px; height: 30px;'></span>`;
          } else { // 没有表情信息
            contentCopy = content;
          }
          return (
            <figure key={index}>
              <img src={JSON.parse(item).user.avatar} alt="头像" onError={(e: any) => { e.currentTarget.src = require("./images/rentou.jpg"); }} /> 
              <figcaption>
                <div className="left">
                  <p dangerouslySetInnerHTML={{ __html: contentCopy }}></p>
                </div>
              </figcaption>
            </figure>
          );
        })}
      </div>
    );
  }