国产 无码 综合区,色欲AV无码国产永久播放,无码天堂亚洲国产AV,国产日韩欧美女同一区二区

vue.js表情文本輸入框組件

這篇具有很好參考價(jià)值的文章主要介紹了vue.js表情文本輸入框組件。希望對(duì)大家有所幫助。如果存在錯(cuò)誤或未考慮完全的地方,請大家不吝賜教,您也可以點(diǎn)擊"舉報(bào)違法"按鈕提交疑問。

參考鏈接

JS操作文本域獲取光標(biāo)/指定位置插入
vue.js支持表情輸入
ttkwsd博客

效果圖

歡迎來到我的個(gè)人博客留言評(píng)論:www.pscool.fun

vue.js表情文本輸入框組件

代碼

  • 不能換行的bug已處理…

emoji.json

  • 表情圖片放在public的emoji文件夾下面

  • emoji.json放在src/components/EmojiText文件夾下,文件內(nèi)容如下:

    {
      "[酸了]" : "/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",
      "[點(diǎn)贊]" : "/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",
      "[調(diào)皮]" : "/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",
      "[陰險(xiǎn)]" : "/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

里面默認(rèn)用了iconfont的字體,需要自行引入。文章來源地址http://www.zghlxwxcb.cn/news/detail-413714.html

<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">

        <!-- 左側(cè)的頭像 -->
        <div class="comment-avatar">
            <img v-if="avatarUrl" :src="avatarUrl" alt="">
            <i v-else class="iconfont icon-touxiang"></i>
        </div>

        <!-- 文本框 和 評(píng)論按鈕 -->
        <div :class="['comment-right', { height80: height80 }]">

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

            <!-- 評(píng)論按鈕 -->
            <div class="comment-btn" @click="postComment">評(píng)論</div>

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

                <!-- 觸發(fā)表情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>

/* 表情配置數(shù)據(jù) 轉(zhuǎn)為 數(shù)組 */
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: { /* 默認(rèn)占位符 */
            type:String,
            default: '快快來發(fā)表你的觀點(diǎn)吧~~'
        },
        avatarUrl: { /* 頭像 */
            type:String
        },
        emojiSize:{
            type:Number,
            default: null
        },
        afterComment: {  /* 發(fā)表評(píng)論之后,需要執(zhí)行的函數(shù) */
            type: Function
        }
    },
    data() {
        return {

            /* 文本框中有文字 或 無文字但是處于焦點(diǎn)狀態(tài)時(shí) 為true */
            height80: false,

            /* 表情配置數(shù)據(jù) */
            emojiList,

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

            /* 文本框的內(nèi)容 */
            textareaContent: '',
        }
    },
    mounted() {
        let _this = this
        document.addEventListener('click', function (e) { /* 點(diǎn)擊其它地方, 關(guān)閉表情面板;點(diǎn)擊表情面板時(shí),需要阻止事件冒泡 */
            _this.emojiPanelActive = false
        })
    },
    methods: {

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

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

            // 最開始的位置要記錄下,后面要根據(jù)它來設(shè)置插入文本后,設(shè)置光標(biāo)的位置
            let selectionStart1 = textarea.selectionStart

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

            /* 一定要放在$nextTick去執(zhí)行, 上面修改完值后, 還要等vue把修改的數(shù)據(jù)渲染出來之后, 再去定位光標(biāo) */
            this.$nextTick(() => {
                // 替換文本后, 需要把光標(biāo),再次定位到替換后的那個(gè)位置,否則,它會(huì)回到最前面
                textarea.focus()
                textarea.setSelectionRange(selectionStart1 + emoji.title.length, selectionStart1 + emoji.title.length)
            })
        },

        /* 激活表情面板, 第二個(gè)參數(shù): 是否切換 */
        activeEmojiPanel(e, isToggle) {
            if (isToggle) {
                this.emojiPanelActive = !this.emojiPanelActive
            } else {
                this.emojiPanelActive = true
            }
            e.stopPropagation() /* 阻止事件冒泡 */
        },

        /* 文本域失去焦點(diǎn)時(shí) */
        doBlur() {
            if (this.textareaContent.length > 0) {
                this.height80 = true
            } else {
                this.height80 = false
            }
        },

        /* 發(fā)表評(píng)論 */
        postComment() {

            if(!this.textareaContent) {
                return
            }

            let _this = this

			/* 處理換行, 雖然解決了, 但是不知道為什么在文本域里面按enter和手動(dòng)輸入\n有啥區(qū)別?
               哦懂了, \n在正則里面就是表示的換行這一個(gè)字符, 手動(dòng)輸入的\n其實(shí)是2個(gè)字符, 按enter輸入的其實(shí)是一個(gè)字符(雖然它看上去是2個(gè)字符),
               我們程序員習(xí)慣了\n表示換行這個(gè)字符(但這只是在開發(fā)工具里面支持的寫法),
               如果把下面改成 /\\n/ 去替換那就可以匹配到手動(dòng)輸入的\n這2個(gè)字符
            */
            // console.log(this.textareaContent,'textareaContent');
            let result = this.textareaContent.replace(/\n/g, function (str) {
                console.log('檢測到str:' + str);
                return "<br/>"
            })
            // console.log(result,'result');

            /* 處理表情 */
            /* 這個(gè)replace函數(shù), 第一個(gè)參數(shù)是正則表達(dá)式, 他回去匹配文本;第二個(gè)參數(shù)是將匹配的文本傳入進(jìn)行處理的函數(shù),函數(shù)的返回值將會(huì)替換匹配的文本 */
            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. 監(jiān)聽comment事件
	 2. 表情圖的大小,單位:像素 -->
<emoji-text @comment="comment" :emojiSize="20"></emoji-text>

到了這里,關(guān)于vue.js表情文本輸入框組件的文章就介紹完了。如果您還想了解更多內(nèi)容,請?jiān)谟疑辖撬阉鱐OY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!

本文來自互聯(lián)網(wǎng)用戶投稿,該文觀點(diǎn)僅代表作者本人,不代表本站立場。本站僅提供信息存儲(chǔ)空間服務(wù),不擁有所有權(quán),不承擔(dān)相關(guān)法律責(zé)任。如若轉(zhuǎn)載,請注明出處: 如若內(nèi)容造成侵權(quán)/違法違規(guī)/事實(shí)不符,請點(diǎn)擊違法舉報(bào)進(jìn)行投訴反饋,一經(jīng)查實(shí),立即刪除!

領(lǐng)支付寶紅包贊助服務(wù)器費(fèi)用

相關(guān)文章

  • HTML常用表情Emoji?♂?和Emoji參考手冊

    HTML常用表情Emoji?♂?和Emoji參考手冊

    HTML表情可以用來在網(wǎng)頁中插入各種表情符號(hào)圖標(biāo),豐富了網(wǎng)頁表現(xiàn)形式和視覺效果。下面是一些常用HTML表情代碼大全?? ??????????????????????????????????? ???????♂?????????????????????????????????????????????????????????

    2024年02月13日
    瀏覽(22)
  • 【vue】 Tinymce 富文本編輯器 不想讓上傳的圖片轉(zhuǎn)換成base64,而是鏈接

    【vue】 Tinymce 富文本編輯器 不想讓上傳的圖片轉(zhuǎn)換成base64,而是鏈接

    前言:最近項(xiàng)目上需要使用富文本編輯器,覺得tinymce很不錯(cuò)就用了,具體怎么在項(xiàng)目中使用參考 【vue】 vue2 中使用 Tinymce 富文本編輯器 【vue】 Tinymce 數(shù)據(jù) 回顯問題 | 第一次正?;仫@后面,顯示空白bug不能編輯 這兩天又遇到了新的問題,圖片上傳后是看地址欄發(fā)現(xiàn)是base64的

    2024年02月14日
    瀏覽(307)
  • Python爬蟲:抓取表情包的下載鏈接

    Python爬蟲:抓取表情包的下載鏈接

    1. 前言 最近發(fā)現(xiàn)了一個(gè)提供表情包的網(wǎng)址,覺得上面的內(nèi)容不錯(cuò),于是就考慮用Python爬蟲獲取上面表情包的下載鏈接。整體而言,實(shí)現(xiàn)這個(gè)挺簡單的,就是找到提供表情包json數(shù)據(jù)的api接口即可,接口中沒有任何加密操作。網(wǎng)址為:表情包 2. 具體實(shí)現(xiàn) 還是通過搜索功能,找到

    2024年02月13日
    瀏覽(21)
  • 【看表情包學(xué)Linux】軟硬鏈接 | 動(dòng)靜態(tài)庫

    【看表情包學(xué)Linux】軟硬鏈接 | 動(dòng)靜態(tài)庫

    ?? ??? 爆笑 教程 ????《看表情包學(xué)Linux》?? ? 猛戳訂閱 ? ? ?? ?? 寫在前面: 上一章我們講解了 inode,為文件系統(tǒng)收了尾,這幾章我們充分地講解完了文件系統(tǒng)的知識(shí)點(diǎn),現(xiàn)在我們開始開始學(xué)習(xí)軟硬鏈接了。如果沒有文件系統(tǒng)的鋪墊,想直接理解軟硬鏈接難免有些困

    2024年02月16日
    瀏覽(21)
  • vue~封裝一個(gè)文本框標(biāo)簽組件

    vue~封裝一個(gè)文本框標(biāo)簽組件

    父組件向子組件的傳值 類型檢查和默認(rèn)值:您可以為props指定類型檢查和默認(rèn)值。這可以確保傳遞給子組件的數(shù)據(jù)符合期望的類型,以及在沒有傳遞數(shù)據(jù)時(shí)具有合理的默認(rèn)值。例如: 在上述示例中,message 的默認(rèn)值是字符串 \\\'Default Message\\\',count 的默認(rèn)值是數(shù)字 0。 :dic是子組件

    2024年02月08日
    瀏覽(22)
  • Vue組件封裝:基于Vue3+wangeditor富文本組件二次封裝

    Vue組件封裝:基于Vue3+wangeditor富文本組件二次封裝

    1.簡介 ? ? ? ? 開源 Web 富文本編輯器,開箱即用,配置簡單。一個(gè)產(chǎn)品的價(jià)值,就在于解決用戶的問題,提高效率、降低成本、增加穩(wěn)定性和擴(kuò)展性。wangEditor 不是為了做而做,也不是單純的模仿誰,而是經(jīng)過上述問題分析之后,給出一個(gè)系統(tǒng)的解決方案。旨在真正去解決用

    2024年04月08日
    瀏覽(96)
  • iOS輸入限制表情輸入、最大字符、正則過濾

    平時(shí)輸入會(huì)有輸入限制, 1.最大字符數(shù)量 2.限制特殊字符:表情包、數(shù)字中文、大小寫 3.空格換行符等特殊字符 一般的解決辦法在輸入時(shí)或輸入完成時(shí)(根據(jù)業(yè)務(wù)需要),設(shè)置正則過濾就可以。 參考:常用正則表達(dá)式(通用字符過濾) 注意,今天說iOS的輸入限制,很離譜,以

    2024年02月04日
    瀏覽(17)
  • Unity UGUI TextMeshPro實(shí)現(xiàn)輸入中文和表情包(Emoji)表情

    Unity UGUI TextMeshPro實(shí)現(xiàn)輸入中文和表情包(Emoji)表情

    目錄 實(shí)現(xiàn)中文顯示 準(zhǔn)備工作 1、打開Window——TextMeshPro——FontAssetCreator 2、把字體文件放入SourceFont中 3、把CharacterSet改為Characters from File 4、把字體庫文件放入Characters File 5、設(shè)置好參數(shù)點(diǎn)擊Generate Font Atlas等待完成后保存 6、把生成后保存的字體文件退拽到Font Asset即可 效果演

    2024年01月18日
    瀏覽(24)
  • vue 富文本組件(mavon-editor)

    vue 富文本組件(mavon-editor)

    mavon-editor是一款基于vue的markdown編輯器,可以用來做文本的編輯,比如是某種業(yè)務(wù)需要發(fā)送公告、個(gè)人博客等,都可以用到,操作也十分簡單。 官方地址:http://www.mavoneditor.com/ github:https://github.com/hinesboy/mavonEditor 安裝好了之后就需要在全局進(jìn)行注冊 注冊好了就能夠直接使用

    2024年02月16日
    瀏覽(39)
  • vue~封裝一個(gè)文本框添加與刪除的組件

    vue~封裝一個(gè)文本框添加與刪除的組件

    這是一個(gè)div,包含了兩個(gè)文本框,后面是添加和刪除按鈕 添加按鈕復(fù)制出新的div,除了文本框沒有內(nèi)容,其它都上面一樣 刪除按鈕將當(dāng)前行div刪除 提交之后,將出現(xiàn)當(dāng)前你添加的這個(gè)列表的對(duì)象,對(duì)接后端接口,將這個(gè)數(shù)組POST到后端即可,如圖所示:

    2024年02月08日
    瀏覽(18)

覺得文章有用就打賞一下文章作者

支付寶掃一掃打賞

博客贊助

微信掃一掃打賞

請作者喝杯咖啡吧~博客贊助

支付寶掃一掃領(lǐng)取紅包,優(yōu)惠每天領(lǐng)

二維碼1

領(lǐng)取紅包

二維碼2

領(lǐng)紅包