plugins/languages/javascript/skills/vue/SKILL.md
Vue 3.5开发规范:Composition API组合式API、Vapor mode高性能模式、Pinia状态管理、Nuxt 4全栈框架。开发Vue组件、SFC单文件组件、响应式数据、路由页面时加载。
npx skillsauth add lazygophers/ccplugin plugins/languages/javascript/skills/vueInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
| Agent | 说明 | | ----- | ---- | | dev | JavaScript 开发专家 | | test | JavaScript 测试专家 |
| 场景 | Skill | 说明 | |------|-------|------| | 核心规范 | Skills(javascript:core) | ES2025-2026 标准、ESM、工具链 | | 异步编程 | Skills(javascript:async) | async/await、Promise | | 安全编码 | Skills(javascript:security) | XSS 防护、Zod 验证 |
<script setup><script setup>
import { ref, computed, onMounted, watch } from 'vue';
const users = ref([]);
const loading = ref(true);
const searchQuery = ref('');
const filteredUsers = computed(() =>
users.value.filter(user =>
user.name.toLowerCase().includes(searchQuery.value.toLowerCase())
)
);
onMounted(async () => {
try {
const response = await fetch('/api/users');
users.value = await response.json();
} catch (error) {
console.error('Failed to fetch users:', error);
} finally {
loading.value = false;
}
});
</script>
<template>
<input v-model="searchQuery" placeholder="Search users..." />
<div v-if="loading">Loading...</div>
<UserList v-else :users="filteredUsers" />
</template>
<script setup>
// defineProps 解构(Vue 3.5)- 保持响应式
const { name, count = 0 } = defineProps({
name: String,
count: { type: Number, default: 0 },
});
// useTemplateRef(Vue 3.5)- 类型安全的模板引用
import { useTemplateRef, onMounted } from 'vue';
const inputRef = useTemplateRef('input');
onMounted(() => {
inputRef.value?.focus();
});
</script>
<template>
<input ref="input" />
<p>{{ name }}: {{ count }}</p>
</template>
// composables/useUser.js
import { ref, watch } from 'vue';
export function useUser(userId) {
const user = ref(null);
const loading = ref(true);
const error = ref(null);
watch(userId, async (id) => {
if (!id) return;
loading.value = true;
error.value = null;
try {
const response = await fetch(`/api/users/${id}`);
if (!response.ok) throw new Error(`HTTP ${response.status}`);
user.value = await response.json();
} catch (e) {
error.value = e;
} finally {
loading.value = false;
}
}, { immediate: true });
return { user, loading, error };
}
// composables/useAbortFetch.js
import { ref, onUnmounted } from 'vue';
export function useAbortFetch() {
let controller = null;
async function fetchData(url) {
controller?.abort();
controller = new AbortController();
const response = await fetch(url, { signal: controller.signal });
return response.json();
}
onUnmounted(() => controller?.abort());
return { fetchData };
}
// stores/user.js
import { defineStore } from 'pinia';
import { ref, computed } from 'vue';
export const useUserStore = defineStore('user', () => {
// State
const users = ref([]);
const currentUser = ref(null);
// Getters
const activeUsers = computed(() =>
users.value.filter(u => u.isActive)
);
// Actions
async function fetchUsers() {
const response = await fetch('/api/users');
users.value = await response.json();
}
async function login(credentials) {
const response = await fetch('/api/login', {
method: 'POST',
body: JSON.stringify(credentials),
});
currentUser.value = await response.json();
}
return { users, currentUser, activeUsers, fetchUsers, login };
});
// nuxt.config.js
export default defineNuxtConfig({
compatibilityDate: '2024-11-01',
future: { compatibilityVersion: 4 },
modules: ['@pinia/nuxt'],
});
// pages/users/[id].vue - 自动路由
<script setup>
const route = useRoute();
const { data: user } = await useFetch(`/api/users/${route.params.id}`);
</script>
// server/api/users/[id].get.js - Server API
export default defineEventHandler(async (event) => {
const id = getRouterParam(event, 'id');
return await db.users.findById(id);
});
import { createRouter, createWebHistory } from 'vue-router';
const router = createRouter({
history: createWebHistory(),
routes: [
{
path: '/',
component: () => import('./pages/Home.vue'), // 懒加载
},
{
path: '/users/:id',
component: () => import('./pages/UserDetail.vue'),
props: true, // 路由参数作为 props
},
],
});
// 导航守卫
router.beforeEach((to, from) => {
const store = useUserStore();
if (to.meta.requiresAuth && !store.currentUser) {
return { name: 'login' };
}
});
| 现象 | 问题 | 严重程度 |
|------|------|---------|
| Options API | 应使用 Composition API + <script setup> | 高 |
| Vuex | 应迁移到 Pinia | 中 |
| this.$refs | 应使用 useTemplateRef()(Vue 3.5)| 中 |
| Mixin | 应使用组合式函数替代 | 高 |
| v-html 无清理 | 必须使用 DOMPurify 清理 | 高 |
| 无 AbortController | 应在 onUnmounted 中取消请求 | 中 |
| Vue 2 语法 | 应迁移到 Vue 3.5 | 高 |
<script setup>defineProps 解构保持响应式useTemplateRef() 替代字符串 ref() => import())onUnmounted 清理 AbortController 和计时器v-html 必须配合 DOMPurifytools
--- name: trellisx-workspace description: 维护 `.trellis/task.md` 任务看板 —— trellis 缺的跨任务总览。**一个表格, 一行一个任务**, 列为 id/名称/描述/状态/阶段/进度/worktree (状态/阶段中文显示)。在 task create/start/阶段切换/archive 后**及时更新**对应行; 并**自动清理超 7 天的已完成行**防膨胀。保持看板与 task.json 实时一致。 when_to_use: 维护 / 创建 / 更新 `.trellis/task.md` 任务看板时; task 生命周期任一节点 (create/start/阶段推进/archive) 之后同步看板时; 用户问"当前有哪些任务 / 任务进度 / 任务看板"时。被 trellisx-flow 与 trellisx-apply 注入的流程引用。 user-invocable: true argument-hint: [show|update|sync|cleanup ...] [task id] arguments:
testing
强制以 Trellis task 闭环处理用户指定的请求 (自判新建/并入 → plan→exec→check→finish 全程不跳步)。**仅用户显式主动调用** (/trellisx-flow 或明确要求"强制走 task 处理这个"); **禁止自动 / 被动 / 推断式调用** —— 不要因为某个请求"看起来该建 task"就自动触发本 skill, 那是 apply 注入的 no_task 倾向的职责。
testing
把 强推task + subtask拆分 + worktree隔离 + 闭环收尾 四维度增量注入当前项目 .trellis/ (workflow.md 的 no_task/planning/in_progress 块 + spec 背书文档 + trellis 生命周期 hook worktree 自动化)。强推 task 与闭环为纯 prompt 软约束 (非平台 hook 硬拦截)。**纯增量追加, 绝不替换 trellis 原生文本** (no_task 分类+征同意/check/finish/前缀全保留)。幂等 (marker 包裹)。
development
Claude Code 会话历史整理 — 扫 ~/.claude/projects/**/*.jsonl 全部 session transcripts, 提取学习增量 (用户校正/决策/踩坑/L0 规则) → 全局记忆库 ~/.cortex/.wiki/memory/. 默认 --apply 落盘 (--dry-run opt-in 仅出 JSON plan 预览). 与 cortex-extract (L4-inbox 内部) 互补.