组件拆分
现在的 App.vue 功能已经完备,但所有代码都塞在一起了。
下一步我们将进行组件拆分,你会学到:
- 如何在 Tailwind v4 环境下编写高度复用的 子组件。
- 如何使用 defineProps 和 defineEmits 在 JS 组件间传递数据。
现在在 src/components 目录下新建以下四个文件:
TaskHeader.vue(纯展示)TaskInput.vue(输入逻辑)TaskFilter.vue(状态切换)TaskItem.vue(单条任务)

TaskHeader.vue —— 静态展示组件
这个组件不需要任何 JS 逻辑,只需展示 UI。
在 Tailwind v4 中,我们继续使用新的渐变语法。
实例
<template>
<header class="bg-linear-to-br from-blue-600 to-indigo-700 p-8 text-white">
<h1 class="text-3xl font-black tracking-tight">TaskHub</h1>
<p class="text-blue-100/80 text-sm">Vue 3 + Tailwind v4 实战</p>
</header>
</template>
<header class="bg-linear-to-br from-blue-600 to-indigo-700 p-8 text-white">
<h1 class="text-3xl font-black tracking-tight">TaskHub</h1>
<p class="text-blue-100/80 text-sm">Vue 3 + Tailwind v4 实战</p>
</header>
</template>
TaskInput.vue —— 使用 defineEmits 发送数据
这个组件负责获取用户输入。当用户点击添加时,它不直接修改数据,而是通过 emit 通知父组件。
实例
<script setup>
import { ref } from 'vue';
// 定义组件向外抛出的事件名
const emit = defineEmits(['add-task']);
const title = ref('');
const handleSubmit = () => {
const value = title.value.trim();
if (!value) return;
// 发送事件,并将数据作为参数传递
emit('add-task', value);
title.value = '';
};
</script>
<template>
<div class="flex gap-2 mb-8">
<input
v-model="title"
@keyup.enter="handleSubmit"
placeholder="今天要完成什么?"
class="flex-1 bg-slate-50 border-none rounded-2xl px-4 py-3 focus:ring-2 focus:ring-brand/50 outline-none transition-all"
/>
<button @click="handleSubmit" class="bg-brand text-white px-6 rounded-2xl font-bold hover:scale-105 active:scale-95 transition-all">
+
</button>
</div>
</template>
import { ref } from 'vue';
// 定义组件向外抛出的事件名
const emit = defineEmits(['add-task']);
const title = ref('');
const handleSubmit = () => {
const value = title.value.trim();
if (!value) return;
// 发送事件,并将数据作为参数传递
emit('add-task', value);
title.value = '';
};
</script>
<template>
<div class="flex gap-2 mb-8">
<input
v-model="title"
@keyup.enter="handleSubmit"
placeholder="今天要完成什么?"
class="flex-1 bg-slate-50 border-none rounded-2xl px-4 py-3 focus:ring-2 focus:ring-brand/50 outline-none transition-all"
/>
<button @click="handleSubmit" class="bg-brand text-white px-6 rounded-2xl font-bold hover:scale-105 active:scale-95 transition-all">
+
</button>
</div>
</template>
TaskFilter.vue —— 组件间的 v-model 通信
在 Vue 3.4+ 中,推荐使用 defineModel() 来实现双向绑定。这在 Tailwind v4 的 UI 切换中非常高效。
实例
<script setup>
// 定义 v-model,父组件传来的值会自动同步
const modelValue = defineModel();
const filters = ['all', 'active', 'completed'];
</script>
<template>
<div class="flex p-1 bg-slate-100 rounded-xl mb-6">
<button
v-for="f in filters"
:key="f"
@click="modelValue = f"
:class="[
'flex-1 py-1.5 text-xs font-bold rounded-lg transition-all capitalize',
modelValue === f ? 'bg-white text-brand shadow-sm' : 'text-slate-500'
]"
>
{{ f }}
</button>
</div>
</template>
// 定义 v-model,父组件传来的值会自动同步
const modelValue = defineModel();
const filters = ['all', 'active', 'completed'];
</script>
<template>
<div class="flex p-1 bg-slate-100 rounded-xl mb-6">
<button
v-for="f in filters"
:key="f"
@click="modelValue = f"
:class="[
'flex-1 py-1.5 text-xs font-bold rounded-lg transition-all capitalize',
modelValue === f ? 'bg-white text-brand shadow-sm' : 'text-slate-500'
]"
>
{{ f }}
</button>
</div>
</template>
TaskItem.vue —— 使用 defineProps 接收数据
它是最核心的子组件,负责展示单条任务并反馈修改/删除操作。
实例
<script setup>
// 接收父组件传递的任务对象
defineProps({
task: {
type: Object,
required: true
}
});
// 定义通知父组件的事件
const emit = defineEmits(['toggle', 'remove']);
</script>
<template>
<li class="group flex items-center justify-between p-4 bg-slate-50 rounded-2xl border border-transparent hover:border-slate-200 hover:bg-white transition-all">
<div class="flex items-center gap-3">
<input
type="checkbox"
:checked="task.isCompleted"
@change="emit('toggle', task.id)"
class="w-5 h-5 accent-brand cursor-pointer"
/>
<span :class="['text-slate-700 font-medium', task.isCompleted ? 'line-through text-slate-400 opacity-50' : '']">
{{ task.title }}
</span>
</div>
<button @click="emit('remove', task.id)" class="opacity-0 group-hover:opacity-100 text-slate-300 hover:text-red-500 transition-all p-1">
✕
</button>
</li>
</template>
// 接收父组件传递的任务对象
defineProps({
task: {
type: Object,
required: true
}
});
// 定义通知父组件的事件
const emit = defineEmits(['toggle', 'remove']);
</script>
<template>
<li class="group flex items-center justify-between p-4 bg-slate-50 rounded-2xl border border-transparent hover:border-slate-200 hover:bg-white transition-all">
<div class="flex items-center gap-3">
<input
type="checkbox"
:checked="task.isCompleted"
@change="emit('toggle', task.id)"
class="w-5 h-5 accent-brand cursor-pointer"
/>
<span :class="['text-slate-700 font-medium', task.isCompleted ? 'line-through text-slate-400 opacity-50' : '']">
{{ task.title }}
</span>
</div>
<button @click="emit('remove', task.id)" class="opacity-0 group-hover:opacity-100 text-slate-300 hover:text-red-500 transition-all p-1">
✕
</button>
</li>
</template>
重构后的 App.vue (逻辑中枢)
现在的 App.vue 变成了一个干净的指挥官,只负责管理原始数据和处理子组件传来的指令。
实例
<script setup>
import { ref, computed } from 'vue';
// 引入子组件
import TaskHeader from './components/TaskHeader.vue';
import TaskInput from './components/TaskInput.vue';
import TaskFilter from './components/TaskFilter.vue';
import TaskItem from './components/TaskItem.vue';
// --- 数据状态 ---
const tasks = ref([
{ id: '1', title: '体验 Tailwind v4 新特性', isCompleted: false },
{ id: '2', title: '掌握 Composition API', isCompleted: true }
]);
const filter = ref('all');
// --- 逻辑处理 ---
const handleAddTask = (title) => {
tasks.value.unshift({ id: crypto.randomUUID(), title, isCompleted: false });
};
const handleRemoveTask = (id) => {
tasks.value = tasks.value.filter(t => t.id !== id);
};
const handleToggleTask = (id) => {
const task = tasks.value.find(t => t.id === id);
if (task) task.isCompleted = !task.isCompleted;
};
// --- 计算属性 ---
const filteredTasks = computed(() => {
if (filter.value === 'active') return tasks.value.filter(t => !t.isCompleted);
if (filter.value === 'completed') return tasks.value.filter(t => t.isCompleted);
return tasks.value;
});
</script>
<template>
<div class="min-h-screen py-12 px-4">
<div class="max-w-md mx-auto bg-white rounded-3xl shadow-xl shadow-slate-200 border border-slate-100 overflow-hidden">
<TaskHeader />
<main class="p-6">
<TaskInput @add-task="handleAddTask" />
<TaskFilter v-model="filter" />
<ul class="space-y-3">
<TransitionGroup name="list">
<TaskItem
v-for="task in filteredTasks"
:key="task.id"
:task="task"
@toggle="handleToggleTask"
@remove="handleRemoveTask"
/>
</TransitionGroup>
</ul>
<div v-if="filteredTasks.length === 0" class="text-center py-12 text-slate-400 text-sm">
暂无相关任务...
</div>
</main>
</div>
</div>
</template>
<style scoped>
.list-enter-active, .list-leave-active { transition: all 0.4s cubic-bezier(0.18, 0.89, 0.32, 1.28); }
.list-enter-from, .list-leave-to { opacity: 0; transform: translateX(30px); }
</style>
import { ref, computed } from 'vue';
// 引入子组件
import TaskHeader from './components/TaskHeader.vue';
import TaskInput from './components/TaskInput.vue';
import TaskFilter from './components/TaskFilter.vue';
import TaskItem from './components/TaskItem.vue';
// --- 数据状态 ---
const tasks = ref([
{ id: '1', title: '体验 Tailwind v4 新特性', isCompleted: false },
{ id: '2', title: '掌握 Composition API', isCompleted: true }
]);
const filter = ref('all');
// --- 逻辑处理 ---
const handleAddTask = (title) => {
tasks.value.unshift({ id: crypto.randomUUID(), title, isCompleted: false });
};
const handleRemoveTask = (id) => {
tasks.value = tasks.value.filter(t => t.id !== id);
};
const handleToggleTask = (id) => {
const task = tasks.value.find(t => t.id === id);
if (task) task.isCompleted = !task.isCompleted;
};
// --- 计算属性 ---
const filteredTasks = computed(() => {
if (filter.value === 'active') return tasks.value.filter(t => !t.isCompleted);
if (filter.value === 'completed') return tasks.value.filter(t => t.isCompleted);
return tasks.value;
});
</script>
<template>
<div class="min-h-screen py-12 px-4">
<div class="max-w-md mx-auto bg-white rounded-3xl shadow-xl shadow-slate-200 border border-slate-100 overflow-hidden">
<TaskHeader />
<main class="p-6">
<TaskInput @add-task="handleAddTask" />
<TaskFilter v-model="filter" />
<ul class="space-y-3">
<TransitionGroup name="list">
<TaskItem
v-for="task in filteredTasks"
:key="task.id"
:task="task"
@toggle="handleToggleTask"
@remove="handleRemoveTask"
/>
</TransitionGroup>
</ul>
<div v-if="filteredTasks.length === 0" class="text-center py-12 text-slate-400 text-sm">
暂无相关任务...
</div>
</main>
</div>
</div>
</template>
<style scoped>
.list-enter-active, .list-leave-active { transition: all 0.4s cubic-bezier(0.18, 0.89, 0.32, 1.28); }
.list-enter-from, .list-leave-to { opacity: 0; transform: translateX(30px); }
</style>
核心知识点总结
defineProps (父传子):
TaskItem通过props接收任务数据。在 Vue 中,Props 是只读的。子组件绝不应该直接修改props的值,这就是"单向数据流"。
defineEmits (子传父):
- 当子组件需要改变数据时(如删除、勾选),它通过
emit发出信号,由父组件执行真正的修改逻辑。这样保证了数据的修改源头只有一个,方便调试。
defineModel (双向绑定神器):
- 在
TaskFilter中,我们使用了 Vue 3.4 引入的defineModel。它极大地简化了父子组件之间状态同步的代码,不再需要手动写props和监听@update:modelValue。
Tailwind v4 的复用性:
- 即使拆分了组件,Tailwind 的原子化类名依然生效。因为 Vite 插件会自动扫描
src下所有的.vue文件并生成对应的样式。
点我分享笔记