vue.js表情文本输入框组件

news2025/2/27 23:26:46

文章目录

    • 参考链接
    • 效果图
    • 代码
      • emoji.json
      • EmojiText.vue
      • 使用

参考链接

JS操作文本域获取光标/指定位置插入
vue.js支持表情输入
ttkwsd博客

效果图

在这里插入图片描述

代码

  • 不能换行的bug已处理…

emoji.json

  • 表情图片放在public的emoji文件夹下面

  • emoji.json放在src/components/EmojiText文件夹下,文件内容如下:

    {
      "[酸了]" : "/emoji/suanle.png",
      "[捂脸]" : "/emoji/wulian.png",
      "[支持]" : "/emoji/zhichi.png",
      "[生气]" : "/emoji/shengqi.png",
      "[捂眼]" : "/emoji/wuyan.png",
      "[难过]" : "/emoji/nanguo.png",
      "[无语]" : "/emoji/wuyu.png",
      "[偷笑]" : "/emoji/touxiao.png",
      "[tv_微笑]" : "/emoji/tvwx.png",
      "[嗑瓜子]" : "/emoji/kgz.png",
      "[原神_喝茶]" : "/emoji/hecha.png",
      "[笑]" : "/emoji/xiao.png",
      "[撇嘴]" : "/emoji/piezui.png",
      "[点赞]" : "/emoji/dianzan.png",
      "[干杯]" : "/emoji/ganbei.png",
      "[tv_斜眼笑]" : "/emoji/tvxyx.png",
      "[大笑]" : "/emoji/daxiao.png",
      "[拥抱]" : "/emoji/yongbao.png",
      "[歪嘴]" : "/emoji/waizui.png",
      "[星星眼]" : "/emoji/xxy.png",
      "[脱单doge]" : "/emoji/doge.png",
      "[再见]" : "/emoji/zaijian.png",
      "[热]" : "/emoji/re.png",
      "[翻白眼]" : "/emoji/fanby.png",
      "[尴尬]" : "/emoji/ganga.png",
      "[笑哭]" : "/emoji/xiaoku.png",
      "[doge]" : "/emoji/doge.png",
      "[抱拳]" : "/emoji/baoquan.png",
      "[冷]" : "/emoji/leng.png",
      "[喜欢]" : "/emoji/xihuan.png",
      "[委屈]" : "/emoji/weiqu.png",
      "[疑惑]" : "/emoji/yihuo.png",
      "[原神_嗯]" : "/emoji/en.png",
      "[呲牙]" : "/emoji/ciya.png",
      "[调皮]" : "/emoji/tiaopi.png",
      "[疼]" : "/emoji/teng.png",
      "[生病]" : "/emoji/shengbing.png",
      "[嘟嘟]" : "/emoji/dudu.png",
      "[灵魂出窍]" : "/emoji/lhcq.png",
      "[嘘声]" : "/emoji/xusheng.png",
      "[哈欠]" : "/emoji/hqian.png",
      "[大哭]" : "/emoji/daku.png",
      "[原神_生气]" : "/emoji/kqsq.png",
      "[微笑]" : "/emoji/simle.png",
      "[给心心]" : "/emoji/geixx.png",
      "[喜极而泣]" : "/emoji/xjeq.png",
      "[嫌弃]" : "/emoji/xianqi.png",
      "[原神_欸嘿]" : "/emoji/aihei.png",
      "[原神_哇]" : "/emoji/wa.png",
      "[加油]" : "/emoji/jiayou.png",
      "[抠鼻]" : "/emoji/koubi.png",
      "[滑稽]" : "/emoji/guaji.png",
      "[傲娇]" : "/emoji/aojiao.png",
      "[吓]" : "/emoji/xia.png",
      "[惊喜]" : "/emoji/jingxi.png",
      "[保佑]" : "/emoji/baoyou.png",
      "[爱心]" : "/emoji/aixin.png",
      "[惊讶]" : "/emoji/jingya.png",
      "[原神_哼]" : "/emoji/heng.png",
      "[抓狂]" : "/emoji/zhuakuang.png",
      "[打call]" : "/emoji/dacall.png",
      "[阴险]" : "/emoji/yinxian.png",
      "[胜利]" : "/emoji/shengli.png",
      "[吐]" : "/emoji/tu.png",
      "[鼓掌]" : "/emoji/guzhang.png",
      "[脸红]" : "/emoji/lianhong.png",
      "[墨镜]" : "/emoji/mojing.png",
      "[OK]" : "/emoji/ok.png",
      "[辣眼睛]" : "/emoji/lyj.png",
      "[奋斗]" : "/emoji/fendou.png",
      "[妙啊]" : "/emoji/miaoa.png",
      "[呆]" : "/emoji/dai.png",
      "[囧]" : "/emoji/jiong.png",
      "[吃瓜]" : "/emoji/chigua.png",
      "[思考]" : "/emoji/sikao.png",
      "[哦呼]" : "/emoji/ohu.png"
    }
    

EmojiText.vue

里面默认用了iconfont的字体,需要自行引入。

<style lang="scss" scoped>
textarea {
    outline: none;
    border: none;
    background: #f1f2f3;
    resize: none;
    border-radius: 8px;
    padding: 10px 10px;
    font-size: 16px;
    color: #333333;
    border: 1px solid transparent;
}
img {
    -webkit-user-drag: none;
}

.avatar {
    width: 40px;
    height: 40px;
    object-fit: cover;
}

.height80 {
    height: 80px !important;
}

.height80 textarea {
    border: 1px solid #49b1f5;
}

@keyframes scaleUp {
    0% {
        opacity: 0;
        transform: scale(0)
    }
    100% {
        opacity: 1;
        transform: scale(1)
    }
}

.scaleUp {
    animation: scaleUp 0.3s;
    transform-origin: 0 0;
}

.comment-area {
    display: flex;
    align-items: flex-start;

    color: #90949e;

    .comment-avatar {
        width: 48px;
        height: 48px;
        display: flex;
        align-items: center;
        justify-content: center;
        margin-right: 8px;

        i {
            font-size: 40px;
            border: 1px solid #c4c4c4;
            border-radius: 50%;

        }
    }

    .comment-right {
        flex: 1;
        display: flex;
        height: 60px;
        transition: height 0.5s;

        position: relative;

        .edit-area {
            flex: 1;
        }

        .comment-btn {
            background-color: #49b1f5;
            cursor: pointer;
            width: 64px;
            border-radius: 8px;
            margin-left: 8px;
            display: flex;
            align-items: center;
            justify-content: center;
            color: #fff;
        }

        .comment-tips {
            position: absolute;
            bottom: -28px;
            height: 24px;
            width: calc(100% - 72px);
            margin-right: 72px;
            display: flex;
            align-items: center;

            &>span:first-child {
                width: 20px;
                height: 20px;
                cursor: pointer;
                display: flex;
                align-items: center;
                justify-content: center;

                &.active {
                    color: #49b1f5;
                }

            }


            .emoji-wrapper {
                user-select: none;
                position: absolute;
                bottom: 0;
                top: 28px;
                left: 0;
                display: flex;
                flex-wrap: wrap;
                width: 294px;
                height: 146px;
                overflow-y: auto;
                background-color: #fff;
                padding: 5px;
                border-radius: 6px;
                border-radius: 6px;
                box-shadow: 0 3px 6px 0 rgb(0 0 0 / 12%);
                border: 1px solid rgba(0, 0, 0, .06);

                &::before {
                    content: '';
                    position: absolute;
                }

                span.emoji {
                    width: 30px;
                    height: 30px;
                    display: block;
                    margin: 2px;
                    cursor: pointer;
                    padding: 3px;
                    border-radius: 6px;

                    img {
                        width: 100%;
                        height: 100%;
                    }

                    transition: all 0.28s;

                    &:hover {
                        background-color: #dddddd;
                    }
                }
            }

            .triangle {
                content: '';
                position: absolute;
                width: 8px;
                height: 8px;
                top: 25px;
                left: 8px;
                background-color: white;
                border: 1px solid #f0f0f0;
                transform: rotate(45deg);
                border-right-color: transparent;
                border-bottom-color: transparent;
            }
        }
    }

}
</style>

<template>
    <div class="comment-area">

        <!-- 左侧的头像 -->
        <div class="comment-avatar">
            <img v-if="avatarUrl" :src="avatarUrl" alt="">
            <i v-else class="iconfont icon-touxiang"></i>
        </div>

        <!-- 文本框 和 评论按钮 -->
        <div :class="['comment-right', { height80: height80 }]">

            <!-- 文本框 -->
            <textarea ref="textarea" v-model="textareaContent" @focus="height80 = true" @blur="doBlur"
                :placeholder="placeholder" class="edit-area">
            </textarea>

            <!-- 评论按钮 -->
            <div class="comment-btn" @click="postComment">评论</div>

            <!-- 表情面板 -->
            <div class="comment-tips">

                <!-- 触发表情icon -->
                <span @click="activeEmojiPanel($event, true)"
                    :class="['iconfont icon-biaoqing', { active: emojiPanelActive }]">
                </span>

                <!-- 待选择的表情列表 -->
                <div v-show="emojiPanelActive">
                    <div class="emoji-wrapper scaleUp" @click="activeEmojiPanel">
                        <span @click="addEmoji(emoji)" class="emoji" v-for="emoji, idx in emojiList" :key="idx">
                            <img :src="emoji.link" alt="">
                        </span>
                    </div>
                    
                </div>

                <!-- 三角形 -->
                <div v-show="emojiPanelActive" class="triangle"></div>
            </div>
        </div>
    </div>
</template>

<script>

/* 表情配置数据 转为 数组 */
import emojiConfig from './emoji.json'
let emojiList = []
for (let key in emojiConfig) {
    emojiList.push({
        title: key,
        link: emojiConfig[key]
    })
}

export default {
    name: 'EmojiText',
    props: {
        imgPrefix: { /* 图片路径前缀 */
            type:String,
            default:''
        },
        placeholder: { /* 默认占位符 */
            type:String,
            default: '快快来发表你的观点吧~~'
        },
        avatarUrl: { /* 头像 */
            type:String
        },
        emojiSize:{
            type:Number,
            default: null
        },
        afterComment: {  /* 发表评论之后,需要执行的函数 */
            type: Function
        }
    },
    data() {
        return {

            /* 文本框中有文字 或 无文字但是处于焦点状态时 为true */
            height80: false,

            /* 表情配置数据 */
            emojiList,

            /* 是否打开表情面板 */
            emojiPanelActive: false,

            /* 文本框的内容 */
            textareaContent: '',
        }
    },
    mounted() {
        let _this = this
        document.addEventListener('click', function (e) { /* 点击其它地方, 关闭表情面板;点击表情面板时,需要阻止事件冒泡 */
            _this.emojiPanelActive = false
        })
    },
    methods: {

        /* 添加表情 */
        addEmoji(emoji) {
            let textarea = this.$refs['textarea'];

            console.log(textarea.selectionStart, textarea.selectionEnd, 'start,end');

            // 最开始的位置要记录下,后面要根据它来设置插入文本后,设置光标的位置
            let selectionStart1 = textarea.selectionStart

            let txtArr = this.textareaContent.split('')
            txtArr.splice(textarea.selectionStart, textarea.selectionEnd - textarea.selectionStart, emoji.title)
            this.textareaContent = txtArr.join('')

            /* 一定要放在$nextTick去执行, 上面修改完值后, 还要等vue把修改的数据渲染出来之后, 再去定位光标 */
            this.$nextTick(() => {
                // 替换文本后, 需要把光标,再次定位到替换后的那个位置,否则,它会回到最前面
                textarea.focus()
                textarea.setSelectionRange(selectionStart1 + emoji.title.length, selectionStart1 + emoji.title.length)
            })
        },

        /* 激活表情面板, 第二个参数: 是否切换 */
        activeEmojiPanel(e, isToggle) {
            if (isToggle) {
                this.emojiPanelActive = !this.emojiPanelActive
            } else {
                this.emojiPanelActive = true
            }
            e.stopPropagation() /* 阻止事件冒泡 */
        },

        /* 文本域失去焦点时 */
        doBlur() {
            if (this.textareaContent.length > 0) {
                this.height80 = true
            } else {
                this.height80 = false
            }
        },

        /* 发表评论 */
        postComment() {

            if(!this.textareaContent) {
                return
            }

            let _this = this

			/* 处理换行, 虽然解决了, 但是不知道为什么在文本域里面按enter和手动输入\n有啥区别?
               哦懂了, \n在正则里面就是表示的换行这一个字符, 手动输入的\n其实是2个字符, 按enter输入的其实是一个字符(虽然它看上去是2个字符),
               我们程序员习惯了\n表示换行这个字符(但这只是在开发工具里面支持的写法),
               如果把下面改成 /\\n/ 去替换那就可以匹配到手动输入的\n这2个字符
            */
            // console.log(this.textareaContent,'textareaContent');
            let result = this.textareaContent.replace(/\n/g, function (str) {
                console.log('检测到str:' + str);
                return "<br/>"
            })
            // console.log(result,'result');

            /* 处理表情 */
            /* 这个replace函数, 第一个参数是正则表达式, 他回去匹配文本;第二个参数是将匹配的文本传入进行处理的函数,函数的返回值将会替换匹配的文本 */
            result = result.replace(/\[.*?]/g, function (str) {
                if(_this.emojiSize) {
                    return `<img src="${_this.imgPrefix}${emojiConfig[str]}"  style="width:${_this.emojiSize}px;height:${_this.emojiSize}"/>`;
                } else {
                    return `<img src="${_this.imgPrefix}${emojiConfig[str]}" />`;
                }
            })

            this.$emit('comment',result)

            this.textareaContent = ''
            this.doBlur()
            this.afterComment && this.afterComment
        }
    },
}
</script>

使用

<!-- 1. 监听comment事件
	 2. 表情图的大小,单位:像素 -->
<emoji-text @comment="comment" :emojiSize="20"></emoji-text>

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Linux 生成pem文件 用于免密登录

1.在远程机器生成.pem文件 生成密钥对&#xff1a; $ ssh-keygen -t rsa -b 2048 -v 直接确认下一步 查看生成的密钥对&#xff1a; $ ls ~/.ssh/ 将私钥重命名至id_rsa.pem&#xff1a; $ mv ~/.ssh/id_rsa ~/.ssh/id_rsa.pem 修改~/.ssh/目录权限&#xff1a; $ chmod …

力扣sql中等篇练习(三)

力扣sql中等篇练习(三) 1 树节点 1.1 题目内容 1.1.1 基本题目信息 1.1.2 示例输入输出 1.2 示例sql语句 # not in匹配上了返回的值是1,casw when里面也是可以使用not in和子查询的 # 注意去重的时候需要筛选掉null值 SELECT id,case when p_id is null then Rootwhen (id …

暄桐好作业之《临王蒙〈具区林屋图〉》

告诉大家一个好消息“暄桐好作业”栏目上新啦~除了与大家分享正在进行的课程好作业&#xff0c;还会向大家展示来自暄桐学长学姐们的优秀国画作品。希望正在上课的暄桐同学们能够从学长学姐的分享以及暄桐教室专业助教的点评中&#xff0c;从中获益并获得力量&#xff0c;继续努…

Java基础(十七):日期时间API

Java基础系列文章 Java基础(一)&#xff1a;语言概述 Java基础(二)&#xff1a;原码、反码、补码及进制之间的运算 Java基础(三)&#xff1a;数据类型与进制 Java基础(四)&#xff1a;逻辑运算符和位运算符 Java基础(五)&#xff1a;流程控制语句 Java基础(六)&#xff1…

chatGPT开发公司

OpenAI是chatGPT的开发公司&#xff0c;成立于2015年&#xff0c;总部位于美国旧金山。OpenAI致力于开发和推广人工智能技术&#xff0c;包括自然语言处理、视觉识别、机器学习等领域&#xff0c;旨在推动人工智能技术对全球社会和经济的影响和发展。 除了chatGPT之外&…

日撸 Java 三百行day31

文章目录day31 整数矩阵及其运算面向对象思想java异常处理java中的getter和setter方法代码day31 整数矩阵及其运算 面向对象思想 结合之前day7和day8面向过程开发&#xff0c;只关注了矩阵加法和矩阵乘法的功能。而day31是面向对象开发&#xff0c;一个矩阵类&#xff0c;在这…

Python实现驾考自动答题,隔壁老王每次都100分....

人生苦短&#xff0c;我用python 不知道大家都考完驾照没&#xff1f; 这个驾照不管大家有没有&#xff0c;我相信大家都是这个&#xff1a; 朋友最近在考驾照&#xff0c;快考科一了… 我微微一笑当场给他整个活~ 用Python整了几十行代码&#xff0c;给朋友实现一下自动答…

前端面试题 - 基础(HTML、HTTP、WEB)

文章目录1、前端 SEO 需要注意哪些2、img 标签上的 title 和 alt 区别3、浏览器输入 url 到显示过程4、浏览器渲染过程5、常见请求方式6、常见的状态码7、网站性能优化8、语义化理解9、浏览器内核理解10、H5的新特性11、离线存储使用&#xff0c;工作原理12、浏览器是如何管理和…

确保实时操作系统(RTOS)设备中的数据安全

导读1月28日大家庆祝了数据保护日&#xff0c;这是一项旨在促进保护数据隐私和安全的国际活动。为了提高人们对数据保护的意识&#xff0c;讨论实时操作系统中数据安全的问题势在必行。目前非常规操作系统已被广泛使用&#xff0c;所以了解这一系统非常重要&#xff0c;尤其是涉…

做完自动化测试,但别让不会汇报毁了你...

pytest 是一个成熟的全功能Python测试工具&#xff0c;可以帮助您编写更好的程序。它与 python 自带的 unittest 测试框架类似&#xff0c;但 pytest 使用起来更简洁和高效&#xff0c;并且兼容 unittest 框架。pytest 能够支持简单的单元测试和复杂的功能测试&#xff0c;pyte…

Javassist动态生成类使用,以及Arthas 查看动态代理生成类内容

java反射 反射的弊端: 性能开销: 因为反射涉及到动态解析的类型&#xff0c;所以某些Java虚拟机的优化不能被执行(因为它不能真正了解你在做什么)。因此&#xff0c;反射操作的性能比非反射操作的性能要慢&#xff0c;应该避免在对性能敏感的应用程序中频繁调用的代码部分。 …

C++记录总结及面试常见问题

C记录总结及面试常见问题指针和函数内存分区模型访问权限struct & class创建类构造函数静态成员this指针const 修饰成员函数友元friend重载运算符和重载函数继承读文件ifstream/fstreamvector中resize和reserve的区别智能指针左值与右值struct、class区别引用指针和函数 函…

Zabbix6.0升级到Zabbix6.4

1&#xff1a;现在的版本是6.0.3 2&#xff1a;停止Zabbix Server进程。 systemctl stop zabbix-server 若升级Zabbix proxy版本&#xff0c;需也停止Zabbix proxy服务 。 systemctl stop zabbix-proxy 3&#xff1a;我用的是MariaDB数据库名称是zabbix。 4&#xff1a; 查看…

windows11安装pytorch2.0

换了新电脑&#xff0c;好久没用windows了&#xff0c;配置一下环境&#xff0c;顺便记录一下方便帮助AI小白不会浪费时间在装环境上&#xff08;一星期不夸张hah&#xff09; 打开命令行&#xff0c;然后输入 nvidia-smi&#xff0c;检查自己的显卡驱动 也可以通过图形界面查…

Python轻量级Web框架Flask(6)——Flask中的单表操作(增删改“查”)

0、前言&#xff1a;Python轻量级Web框架Flask&#xff08;5&#xff09;中生成的新模板就是包含数据库操作的Flask模板。 在pycharm中用flask写表结构时需用用到数据类型的定义如下&#xff1a; 下面总结一些常用到的数据类型&#xff1a; 1、Python轻量级Web框架Flask&…

传智健康项目总结

耗时一个半月终于把传智健康项目跟着做完了&#xff0c;下面是对项目的一点心得体会。 项目知识点总结 Maven中parent父工程聚合 聚合模块(父工程)作用&#xff1a;父工程是一个pom工程&#xff0c;通常只是用来帮助其子模块构建的工具&#xff0c;本身并没有实质的内容。具体…

【Pytorch】搭建网络模型的实战

【Pytorch】搭建网络模型的实战CIFAR10 model structure搭建网络使用Sequential进行搭建网络模型使用tensorboard查看网络结构对CIFAR10数据集进行分类&#xff0c;根据图片内容识别这是哪一类 CIFAR10 model structure 输入input:3通道的32 x 32 图片卷积操作的通道数不变 那…

C#,码海拾贝(16)——求“矩阵秩”的全选主元“高斯消去法(Gauss Elimination)”C#源代码,《C#数值计算算法编程》源代码升级改进版

1 矩阵的秩 Rank of Matrix 矩阵的秩是线性代数中的一个概念。在线性代数中&#xff0c;一个矩阵A的列秩是A的线性独立的纵列的极大数&#xff0c;通常表示为r(A)&#xff0c;rk(A)或rank A。 在线性代数中&#xff0c;一个矩阵A的列秩是A的线性独立的纵列的极大数目。类似地&…

全面解析反欺诈(羊毛盾)API,助你识别各类欺诈风险

前言 反欺诈&#xff08;羊毛盾&#xff09;反机器欺诈 API&#xff0c;是一种基于大数据分析和模型产品的技术&#xff0c;通过输入手机号、手机 IP 地址进行检测&#xff0c;帮助客户识别大量存在恶意的账号。 反欺诈&#xff08;羊毛盾&#xff09;API 的作用 反欺诈&…

Spring自定义参数解析器~

1. 什么是参数解析器 RequstBody、RequstParam 这些注解是不是很熟悉&#xff1f; 我们在开发 Controller 接口时经常会用到此类参数注解&#xff0c;那这些注解的作用是什么&#xff1f;我们真的了解吗&#xff1f; 简单来说&#xff0c;这些注解就是帮我们将前端传递的参数…