Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >Vue 表情包输入组件完整实现代码及使用教程

Vue 表情包输入组件完整实现代码及使用教程

原创
作者头像
小焱
发布于 2025-06-08 09:08:13
发布于 2025-06-08 09:08:13
7000
代码可运行
举报
文章被收录于专栏:前端开发前端开发
运行总次数:0
代码可运行

Vue 表情包输入组件的使用方法与封装指南

组件使用方法

要在你的 Vue 项目中使用表情包输入组件,首先需要进行安装和配置,以下是详细步骤:

  1. 安装依赖
代码语言:bash
AI代码解释
复制
npm install emoji-mart @vueuse/core
  1. 全局注册组件main.js 中添加:
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import EmojiPicker from './components/EmojiPicker.vue';

createApp(App)
  .component('EmojiPicker', EmojiPicker)
  .mount('#app');
  1. 在组件中使用
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
<template>
  <div class="chat-container">
    <textarea v-model="message" ref="messageInput" placeholder="输入消息..."></textarea>
    
    <button @click="toggleEmojiPicker">
      <i class="fas fa-smile"></i>
    </button>
    
    <EmojiPicker 
      v-if="showEmojiPicker"
      @select="insertEmoji"
      :recent-emojis="recentEmojis"
    />
    
    <button @click="sendMessage">发送</button>
  </div>
</template>

<script>
import { ref, onMounted } from 'vue';

export default {
  setup() {
    const message = ref('');
    const showEmojiPicker = ref(false);
    const recentEmojis = ref([]);
    const messageInput = ref(null);
    
    const toggleEmojiPicker = () => {
      showEmojiPicker.value = !showEmojiPicker.value;
    };
    
    const insertEmoji = (emoji) => {
      const input = messageInput.value;
      const start = input.selectionStart;
      const end = input.selectionEnd;
      
      message.value = message.value.substring(0, start) + 
                     emoji + 
                     message.value.substring(end);
      
      input.focus();
      input.selectionStart = input.selectionEnd = start + emoji.length;
      
      // 更新最近使用表情
      if (!recentEmojis.value.includes(emoji)) {
        recentEmojis.value.unshift(emoji);
        if (recentEmojis.value.length > 20) {
          recentEmojis.value.pop();
        }
      }
    };
    
    const sendMessage = () => {
      if (message.value.trim()) {
        // 处理消息发送逻辑
        console.log('发送消息:', message.value);
        message.value = '';
      }
    };
    
    return {
      message,
      showEmojiPicker,
      recentEmojis,
      messageInput,
      toggleEmojiPicker,
      insertEmoji,
      sendMessage
    };
  }
}
</script>
组件封装方法

以下是表情包组件的封装实现:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
<!-- EmojiPicker.vue -->
<template>
  <div class="emoji-picker" ref="container">
    <!-- 搜索框 -->
    <div class="search-container">
      <input 
        type="text" 
        v-model="searchText" 
        placeholder="搜索表情..."
        @input="searchEmojis"
      >
    </div>
    
    <!-- 表情分类导航 -->
    <div class="categories">
      <button 
        v-for="category in categories" 
        :key="category.id"
        :class="{ active: activeCategory === category.id }"
        @click="switchCategory(category.id)"
      >
        <i :class="category.icon"></i>
      </button>
    </div>
    
    <!-- 表情内容区域 -->
    <div class="emoji-container">
      <!-- 最近使用表情 -->
      <div v-if="activeCategory === 'recent' && recentEmojis.length > 0" class="emoji-group">
        <div 
          v-for="emoji in recentEmojis" 
          :key="emoji"
          class="emoji-item"
          @click="selectEmoji(emoji)"
        >
          {{ emoji }}
        </div>
      </div>
      
      <!-- 按分类显示表情 -->
      <div v-else class="emoji-group">
        <div 
          v-for="emoji in filteredEmojis" 
          :key="emoji"
          class="emoji-item"
          @click="selectEmoji(emoji)"
        >
          {{ emoji }}
        </div>
      </div>
    </div>
  </div>
</template>

<script>
import { ref, computed, onMounted, watch } from 'vue';
import emojiList from './emojiList.json'; // 表情数据

export default {
  name: 'EmojiPicker',
  props: {
    recentEmojis: {
      type: Array,
      default: () => []
    }
  },
  setup(props, { emit }) {
    const container = ref(null);
    const searchText = ref('');
    const activeCategory = ref('recent');
    const filteredEmojis = ref([]);
    
    // 表情分类
    const categories = ref([
      { id: 'recent', icon: 'fas fa-clock' },
      { id: 'smileys', icon: 'fas fa-smile' },
      { id: 'animals', icon: 'fas fa-paw' },
      { id: 'food', icon: 'fas fa-utensils' },
      { id: 'activities', icon: 'fas fa-futbol' },
      { id: 'travel', icon: 'fas fa-plane' },
      { id: 'objects', icon: 'fas fa-lightbulb' },
      { id: 'symbols', icon: 'fas fa-heart' },
      { id: 'flags', icon: 'fas fa-flag' }
    ]);
    
    // 根据分类过滤表情
    const getCategoryEmojis = (category) => {
      if (category === 'recent') return [];
      return emojiList[category] || [];
    };
    
    // 搜索表情
    const searchEmojis = () => {
      if (!searchText.value.trim()) {
        filteredEmojis.value = getCategoryEmojis(activeCategory.value);
        return;
      }
      
      const searchTerm = searchText.value.toLowerCase();
      const result = [];
      
      Object.values(emojiList).forEach(category => {
        category.forEach(emoji => {
          if (emoji.toLowerCase().includes(searchTerm)) {
            result.push(emoji);
          }
        });
      });
      
      filteredEmojis.value = result;
    };
    
    // 切换分类
    const switchCategory = (category) => {
      activeCategory.value = category;
      filteredEmojis.value = getCategoryEmojis(category);
    };
    
    // 选择表情
    const selectEmoji = (emoji) => {
      emit('select', emoji);
    };
    
    // 初始化
    onMounted(() => {
      filteredEmojis.value = getCategoryEmojis(activeCategory.value);
    });
    
    // 监听搜索文本变化
    watch(searchText, searchEmojis);
    
    return {
      container,
      searchText,
      activeCategory,
      filteredEmojis,
      categories,
      searchEmojis,
      switchCategory,
      selectEmoji
    };
  }
}
</script>

<style scoped>
.emoji-picker {
  width: 320px;
  background: white;
  border-radius: 8px;
  box-shadow: 0 2px 10px rgba(0,0,0,0.1);
  padding: 10px;
  position: absolute;
  z-index: 100;
}

.search-container {
  margin-bottom: 10px;
}

.search-container input {
  width: 100%;
  padding: 8px;
  border: 1px solid #ddd;
  border-radius: 4px;
  outline: none;
}

.categories {
  display: flex;
  justify-content: space-around;
  padding: 8px 0;
  border-bottom: 1px solid #eee;
}

.categories button {
  background: none;
  border: none;
  cursor: pointer;
  font-size: 18px;
  padding: 5px;
  color: #888;
  transition: color 0.2s;
}

.categories button.active {
  color: #007bff;
}

.emoji-container {
  height: 240px;
  overflow-y: auto;
  padding: 5px;
}

.emoji-group {
  display: grid;
  grid-template-columns: repeat(8, 1fr);
  gap: 8px;
}

.emoji-item {
  font-size: 20px;
  padding: 5px;
  text-align: center;
  cursor: pointer;
  transition: transform 0.2s;
}

.emoji-item:hover {
  transform: scale(1.2);
}
</style>
组件高级配置与扩展
  1. 自定义表情列表 可以通过修改 emojiList.json 文件来添加或删除表情,也可以通过 props 传入自定义表情数据。
  2. 主题定制 组件支持通过 CSS 变量进行主题定制:
代码语言:css
AI代码解释
复制
.emoji-picker {
  --primary-color: #007bff;
  --border-radius: 8px;
  --box-shadow: 0 2px 10px rgba(0,0,0,0.1);
}
  1. 添加动画效果 可以为表情选择器添加淡入淡出或滑动动画:
代码语言:css
AI代码解释
复制
.emoji-picker {
  opacity: 0;
  transform: translateY(10px);
  transition: opacity 0.2s ease, transform 0.2s ease;
}

.emoji-picker.active {
  opacity: 1;
  transform: translateY(0);
}
  1. 添加懒加载 对于大型表情库,可以实现懒加载:
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 在 EmojiPicker.vue 中添加
const loadCategoryEmojis = async (category) => {
  if (category === 'recent') return [];
  
  // 模拟异步加载
  if (!emojiList[category]) {
    const response = await fetch(`/emojis/${category}.json`);
    emojiList[category] = await response.json();
  }
  
  return emojiList[category];
};
完整集成示例

下面是一个完整的聊天界面示例,展示如何集成表情包组件:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
<template>
  <div class="chat-app">
    <div class="chat-header">
      <h1>聊天应用</h1>
    </div>
    
    <div class="chat-messages" ref="messagesContainer">
      <div 
        v-for="(message, index) in messages" 
        :key="index"
        class="message"
        :class="{ 'user-message': message.type === 'user', 'bot-message': message.type === 'bot' }"
      >
        <div class="message-content">
          <p>{{ message.text }}</p>
        </div>
      </div>
    </div>
    
    <div class="chat-input">
      <div class="emoji-button" @click="toggleEmojiPicker">
        <i class="fas fa-smile"></i>
      </div>
      
      <textarea 
        v-model="message" 
        ref="messageInput" 
        placeholder="输入消息..."
        @focus="showEmojiPicker = false"
      ></textarea>
      
      <button @click="sendMessage">发送</button>
      
      <EmojiPicker 
        v-if="showEmojiPicker"
        :recent-emojis="recentEmojis"
        @select="insertEmoji"
      />
    </div>
  </div>
</template>

<script>
import { ref, onMounted, nextTick } from 'vue';
import EmojiPicker from './components/EmojiPicker.vue';

export default {
  components: {
    EmojiPicker
  },
  setup() {
    const messagesContainer = ref(null);
    const messageInput = ref(null);
    const message = ref('');
    const showEmojiPicker = ref(false);
    const messages = ref([
      { type: 'bot', text: '你好!我是聊天机器人,有什么可以帮助你的吗?' }
    ]);
    const recentEmojis = ref([]);
    
    const toggleEmojiPicker = () => {
      showEmojiPicker.value = !showEmojiPicker.value;
    };
    
    const insertEmoji = (emoji) => {
      const input = messageInput.value;
      const start = input.selectionStart;
      const end = input.selectionEnd;
      
      message.value = message.value.substring(0, start) + 
                     emoji + 
                     message.value.substring(end);
      
      input.focus();
      input.selectionStart = input.selectionEnd = start + emoji.length;
      
      // 更新最近使用表情
      if (!recentEmojis.value.includes(emoji)) {
        recentEmojis.value.unshift(emoji);
        if (recentEmojis.value.length > 20) {
          recentEmojis.value.pop();
        }
      }
    };
    
    const sendMessage = () => {
      if (!message.value.trim()) return;
      
      // 添加用户消息
      messages.value.push({
        type: 'user',
        text: message.value
      });
      
      // 模拟机器人回复
      setTimeout(() => {
        messages.value.push({
          type: 'bot',
          text: '我收到了你的消息: ' + message.value
        });
        scrollToBottom();
      }, 500);
      
      message.value = '';
      showEmojiPicker.value = false;
      scrollToBottom();
    };
    
    const scrollToBottom = () => {
      nextTick(() => {
        if (messagesContainer.value) {
          messagesContainer.value.scrollTop = messagesContainer.value.scrollHeight;
        }
      });
    };
    
    onMounted(() => {
      scrollToBottom();
    });
    
    return {
      messagesContainer,
      messageInput,
      message,
      showEmojiPicker,
      messages,
      recentEmojis,
      toggleEmojiPicker,
      insertEmoji,
      sendMessage
    };
  }
}
</script>

<style scoped>
.chat-app {
  display: flex;
  flex-direction: column;
  height: 100vh;
  max-width: 800px;
  margin: 0 auto;
}

.chat-header {
  padding: 16px;
  background: #007bff;
  color: white;
  text-align: center;
  border-radius: 8px 8px 0 0;
}

.chat-messages {
  flex: 1;
  overflow-y: auto;
  padding: 16px;
  background: #f8f9fa;
}

.message {
  margin-bottom: 16px;
  display: flex;
}

.user-message {
  justify-content: flex-end;
}

.bot-message {
  justify-content: flex-start;
}

.message-content {
  padding: 12px;
  border-radius: 8px;
  max-width: 70%;
}

.user-message .message-content {
  background: #007bff;
  color: white;
}

.bot-message .message-content {
  background: white;
  box-shadow: 0 1px 3px rgba(0,0,0,0.1);
}

.chat-input {
  display: flex;
  align-items: center;
  padding: 16px;
  background: white;
  border-radius: 0 0 8px 8px;
  box-shadow: 0 -1px 3px rgba(0,0,0,0.1);
  position: relative;
}

.emoji-button {
  font-size: 20px;
  margin-right: 12px;
  cursor: pointer;
  color: #6c757d;
}

textarea {
  flex: 1;
  padding: 12px;
  border: 1px solid #ddd;
  border-radius: 4px;
  resize: none;
  outline: none;
}

button {
  margin-left: 12px;
  padding: 12px 24px;
  background: #007bff;
  color: white;
  border: none;
  border-radius: 4px;
  cursor: pointer;
  transition: background 0.2s;
}

button:hover {
  background: #0069d9;
}

.emoji-picker {
  bottom: 70px;
  right: 20px;
}
</style>

这个表情包输入组件不仅功能完整,而且具有良好的可扩展性和用户体验。你可以根据项目需求进一步定制样式和功能,比如添加 GIF 搜索、自定义表情上传等高级功能。


Vue, 表情包输入组件,完整实现代码,使用教程,Vue 组件开发,前端开发,JavaScript,HTML,CSS, 表情包插件,Vue 实战,前端组件,用户界面设计,交互式组件,前端开发教程



资源地址:

https://pan.quark.cn/s/a85a3b247036


原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
Vue 实现表情包输入组件的完整代码及详细步骤
表情包已成为现代社交应用不可或缺的元素,为用户交流增添了丰富的情感表达。本文将详细介绍如何在 Vue 项目中实现一个功能完善、交互友好的表情包输入组件,并探讨其应用场景。
小焱
2025/06/08
620
Vue 实现表情包输入组件的完整代码及详细步骤
Vue3 封装 AI 问答组件实现 AI 流式回答问题的方法
通过Vue3的Composition API,我们可以方便地封装一个高效、可复用的AI流式问答组件。关键技术点包括:
小焱
2025/05/23
4240
Vue3 封装 AI 问答组件实现 AI 流式回答问题的方法
Vue 项目中 TinyMCE 富文本编辑器的具体使用方法
TinyMCE是一款功能强大、高度可定制的富文本编辑器,被广泛应用于各种Web应用中。在Vue项目中集成TinyMCE可以为用户提供专业的文本编辑体验。本文将详细介绍如何在Vue项目中使用TinyMCE,并提供完整的应用实例。
小焱
2025/05/22
6120
Vue 项目中 TinyMCE 富文本编辑器的具体使用方法
体育赛事直播系统开发:实现用户群聊和发红包功能的关键部分代码
基于“东莞梦幻网络科技”体育赛事直播系统中的聊一聊功能模块的群聊和发红包关键代码实现方案,所用技术栈(后端:PHP-ThinkPHP、安卓-Java、iOS-OC、PC/H5-Vue.js)。
用户10027718
2025/04/09
730
重构谷粒商城11:node快速入门
前言:这个系列将使用最前沿的cursor作为辅助编程工具,来快速开发一些基础的编程项目。目的是为了在真实项目中,帮助初级程序员快速进阶,以最快的速度,效率,快速进阶到中高阶程序员。
半旧518
2025/04/08
930
重构谷粒商城11:node快速入门
springAI初体验 让人人都能跑大模型
他们的设计理念是:为开发人员提供一个抽象接口,为将生成式AI作为独立组件纳入应用奠定基础
用户10143704
2024/04/05
1.1K0
保姆级全栈项目BBS 开发实录 第二章:前端基础页面
https://cdn.jsdelivr.net/gh/Qiu-JW/NotePicture/2024/08/image-20240806215014105.png
Qiuner
2024/11/21
680
保姆级全栈项目BBS 开发实录 第二章:前端基础页面
【AI大模型】使用Python调用DeepSeek的API,原来SDK是调用这个,绝对的一分钟上手和使用
在当今数字化时代,AI大模型技术迅速发展,DeepSeek作为其中的佼佼者,凭借其强大的语言理解和生成能力,受到了众多开发者的关注。作为一名对前沿技术充满热情的程序员,对接DeepSeek的API进行体验和开发是探索其潜力的重要一步。本文将为您提供一份详细且清晰的保姆级教程,帮助您快速上手并使用DeepSeek的API。
全干程序员demo
2025/03/24
6940
【AI大模型】使用Python调用DeepSeek的API,原来SDK是调用这个,绝对的一分钟上手和使用
从个人博客到电商中台:EdgeOne Pages的MCP Server弹性架构×DeepSeek多场景模板实测报告
EdgeOne Pages 是腾讯云推出的一站式边缘开发与部署平台,基于全球边缘节点网络和 Serverless 架构,为开发者提供从代码托管到全球分发的全流程服务。其核心价值在于将边缘计算能力与现代 Web 开发范式深度融合,支持静态站点托管、动态应用开发、边缘函数部署等场景,尤其适合需要快速迭代、全球化加速或低成本运维的项目。
Undoom
2025/04/04
2530
不会小程序设计?10分钟AI工具助你从需求到uniapp开发
之前2025年的年初开年之时,我立下了一个Flag,那就是写一个记账类型的小程序。
半月无霜
2025/03/29
1840
如何用120行代码,实现一个交互完整的拖拽上传组件?
于是我又用React Hooks 重写了一版,除CSS的代码总数 120行。 效果如下:
前端劝退师
2019/08/29
2.1K0
如何用120行代码,实现一个交互完整的拖拽上传组件?
tauri2.0+rust+vue3电脑版Exe聊天软件
经过大半个月爆肝式开发,重磅自研新作tauri2.0+vite.js跨平台桌面端聊天程序,正式完结了。
andy2018
2024/09/29
3103
tauri2.0+rust+vue3电脑版Exe聊天软件
使用Django构建即时通讯应用的最简单方法
模板中包括一个 EventSource 脚本,用于接收来自 Django 的服务器发送的事件。 在sim下创建templates文件夹,在templates下创建chat.html。写入:
一只大鸽子
2024/01/09
4990
使用Django构建即时通讯应用的最简单方法
Vue 2.0实现一个多选下拉框效果
要在上述代码中添加一个确认按钮,用于提交用户选择的选项,你可以按照以下步骤进行修改:
王小婷
2023/10/05
1.7K0
Vue 2.0实现一个多选下拉框效果
基于GPT搭建私有知识库聊天机器人(六)仿chatGPT打字机效果
在前几篇文章中,我们已经了解了如何使用 GPT 模型来搭建一个简单的聊天机器人,并在后端使用私有知识库来提供答案。
夕阳也是醉了
2023/10/16
8170
基于GPT搭建私有知识库聊天机器人(六)仿chatGPT打字机效果
【SpringBoot+Vue】003-创建Login组件、编写Login组件的页面
当写到CSS的时候,真的发现自己的CSS知识不足,仅仅知道一点常规的东西,UP主写的CSS很多我都不记得了,看来自己的路还有很长!加油!
訾博ZiBo
2025/01/06
1380
【SpringBoot+Vue】003-创建Login组件、编写Login组件的页面
【愚公系列】《AIGC辅助软件开发》016-AI辅助前端编程:利用ChatGPT在前端开发中快速生成Vue组件
文章地址:https://cloud.tencent.com/developer/article/2474495
愚公搬代码
2024/12/08
2460
使用Kimi开发自己的问答应用
Kimi是大家常用的一个人工智能助手,本文使用Kimi开发文档,以node作为后端,开发与一个问答系统
牛老师讲GIS
2024/12/30
2430
使用Kimi开发自己的问答应用
调用DeepSeek API增强版纯前端实现方案,支持文件上传和内容解析功能
在现代Web开发中,文件上传和内容解析是常见的需求。随着人工智能技术的发展,DeepSeek API增强版提供了一个强大的解决方案,支持多种文件格式的上传和内容解析。本文将详细介绍如何通过纯前端技术调用DeepSeek API增强版,实现文件上传和内容解析功能,帮助开发者快速构建高效、安全的Web应用。
全干程序员demo
2025/03/27
9000
调用DeepSeek API增强版纯前端实现方案,支持文件上传和内容解析功能
【SpringBoot+Vue】004-Login校验规则、实现登录和重置事件
https://element.eleme.cn/#/zh-CN/component/form
訾博ZiBo
2025/01/06
1110
【SpringBoot+Vue】004-Login校验规则、实现登录和重置事件
推荐阅读
相关推荐
Vue 实现表情包输入组件的完整代码及详细步骤
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验