50天50个小项目 (Vue3 + Tailwindcss V4) ✨ | RandomChoicePicker(标签生成)

可能你疑惑为啥名字和功能不一样,哈哈哈,不要怀疑,这只是我打错名字懒得改了。。。

📅 我们继续 50 个小项目挑战!------ RandomChoicePicker组件

仓库地址:https://github.com/SunACong/50-vue-projects

项目预览地址:https://50-vue-projects.vercel.app/


使用 Vue 3 的 Composition API 和 <script setup> 语法结合 TailwindCSS 构建一个简单的标签输入组件。用户可以在文本框中输入多个选项,并通过逗号分隔,组件会自动将其拆分成可视化的"标签"展示出来。

🎯 组件目标

  • 接收用户输入的一段文本。
  • 使用逗号 , 分割输入内容。
  • 动态渲染为一组"标签"(Tag)。
  • 使用 TailwindCSS 快速构建美观现代的 UI 界面。
  • 提供清晰的交互反馈。

⚙️ 技术实现点

技术点 描述
Vue 3 Composition API (<script setup>) 使用响应式变量管理组件状态
v-model 双向绑定 绑定 textarea 输入值
watchEffect 副作用监听 自动拆分并更新 tagList
v-for 渲染列表 动态生成多个 Tag 元素
TailwindCSS 样式类 快速构建现代 UI

🧱 组件实现

模板结构 <template>

html 复制代码
<template>
    <div class="flex h-screen items-center justify-center">
        <div class="rounded-2xl bg-gray-400 p-8">
            <h3 class="font-mono text-2xl">
                Enter all of the choices divided by a comma (',').
                <br />
                Press enter when you're done
            </h3>
            <textarea
                class="my-4 h-36 w-full bg-gray-200 p-4"
                placeholder="Enter choices here..."
                v-model="textareaText"></textarea>

            <div class="flex gap-2" v-if="tagList[0] !== ''">
                <div class="h-8 rounded-2xl bg-amber-200 p-1" v-for="item in tagList" :key="item">
                    {{ item }}
                </div>
            </div>
        </div>
    </div>
</template>

脚本逻辑 <script setup>

js 复制代码
<script setup>
import { ref, watchEffect } from 'vue'

const textareaText = ref('')

const tagList = ref([])

const splitTag = () => {
    tagList.value = []
    const tagItems = textareaText.value.split(',')
    tagItems.forEach((item) => {
        tagList.value.push(item.trim()) // 去除前后空格
    })
}

watchEffect(() => {
    splitTag()
})
</script>

🔍 重点效果实现

✅ 实时拆分与渲染

我们使用 v-model<textarea> 的值与 textareaText 进行双向绑定:

html 复制代码
<textarea v-model="textareaText"></textarea>

并通过 watchEffect() 监听其变化,一旦发生变化就执行 splitTag() 方法,将输入内容以逗号分割为数组,并去除空格后推入 tagList

js 复制代码
watchEffect(() => {
    splitTag()
})

💡 条件渲染优化

为了防止页面加载时显示空标签或错误信息,我们添加了一个判断条件:

html 复制代码
<div class="flex gap-2" v-if="tagList[0] !== ''">

这样只有当用户输入了至少一个有效选项后,才会渲染出标签列表。


🎨 TailwindCSS 样式重点讲解

类名 作用
flex, items-center, justify-center 居中布局整个容器
h-screen 容器高度为视口全高
rounded-2xl 圆角大小为 1rem
bg-gray-400bg-gray-200bg-amber-200 设置背景颜色
p-8, p-4, p-1 不同层级的内边距
my-4 上下外边距为 1rem
w-full 宽度为 100%
h-36 高度为 9rem
text-2xl 字体大小为 1.5rem
font-mono 使用等宽字体
gap-2 flex 子元素之间间隔为 0.5rem
h-8 高度为 2rem
rounded-2xl 圆角为 1rem

这些 Tailwind 工具类帮助我们快速构建了一个简洁美观的输入界面和标签展示区域。


📁 常量定义 + 组件路由

constants/index.js 添加组件预览常量:

js 复制代码
{
        id: 13,
        title: 'Random Choice Picker',
        image: 'https://50projects50days.com/img/projects-img/13-random-choice-picker.png',
        link: 'RandomChoicePicker',
    },

router/index.js 中添加路由选项:

js 复制代码
{
        path: '/RandomChoicePicker',
        name: 'RandomChoicePicker',
        component: () => import('@/projects/RandomChoicePicker.vue'),
    },

🏁 总结

标签输入组件涵盖Vue 3 的响应式系统、副作用监听、动态渲染以及 TailwindCSS 的灵活样式组合。

  • 作为表单组件的一部分,用于收集用户输入的多项数据。

👉 下一篇,我们将完成AnimatedNavigation组件,一个非常有意思的动画的导航组件!🚀

相关推荐
威迪斯特6 小时前
Flask:轻量级Web框架的技术本质与工程实践
前端·数据库·后端·python·flask·开发框架·核心架构
Hello.Reader7 小时前
Flink 文件系统通用配置默认文件系统与连接数限制实战
vue.js·flink·npm
wuhen_n7 小时前
JavaScript内置数据结构
开发语言·前端·javascript·数据结构
大鱼前端7 小时前
为什么我说CSS-in-JS是前端“最佳”的糟粕设计?
前端
不爱吃糖的程序媛7 小时前
Capacitor:跨平台Web原生应用开发利器,现已全面适配鸿蒙
前端·华为·harmonyos
AC赳赳老秦7 小时前
2026国产算力新周期:DeepSeek实战适配英伟达H200,引领大模型训练效率跃升
大数据·前端·人工智能·算法·tidb·memcache·deepseek
CHU7290357 小时前
淘宝扭蛋机抽盒小程序前端功能解析:解锁趣味抽盒新体验
前端·小程序
-凌凌漆-7 小时前
【npm】npm的-D选项介绍
前端·npm·node.js
鹿心肺语7 小时前
前端HTML转PDF的两种主流方案深度解析
前端·javascript
海石8 小时前
去到比北方更北的地方—2025年终总结
前端·ai编程·年终总结