@pythoughts/vue-skills-mcp 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (208) hide show
  1. package/README.md +63 -0
  2. package/index.mjs +139 -0
  3. package/package.json +34 -0
  4. package/skills/create-adaptable-composable/SKILL.md +76 -0
  5. package/skills/vue-best-practices/SKILL.md +154 -0
  6. package/skills/vue-best-practices/references/animation-class-based-technique.md +254 -0
  7. package/skills/vue-best-practices/references/animation-state-driven-technique.md +291 -0
  8. package/skills/vue-best-practices/references/component-async.md +97 -0
  9. package/skills/vue-best-practices/references/component-data-flow.md +350 -0
  10. package/skills/vue-best-practices/references/component-fallthrough-attrs.md +174 -0
  11. package/skills/vue-best-practices/references/component-keep-alive.md +137 -0
  12. package/skills/vue-best-practices/references/component-slots.md +216 -0
  13. package/skills/vue-best-practices/references/component-suspense.md +228 -0
  14. package/skills/vue-best-practices/references/component-teleport.md +108 -0
  15. package/skills/vue-best-practices/references/component-transition-group.md +128 -0
  16. package/skills/vue-best-practices/references/component-transition.md +125 -0
  17. package/skills/vue-best-practices/references/composables.md +290 -0
  18. package/skills/vue-best-practices/references/directives.md +162 -0
  19. package/skills/vue-best-practices/references/perf-avoid-component-abstraction-in-lists.md +159 -0
  20. package/skills/vue-best-practices/references/perf-v-once-v-memo-directives.md +182 -0
  21. package/skills/vue-best-practices/references/perf-virtualize-large-lists.md +187 -0
  22. package/skills/vue-best-practices/references/plugins.md +166 -0
  23. package/skills/vue-best-practices/references/reactivity.md +346 -0
  24. package/skills/vue-best-practices/references/render-functions.md +201 -0
  25. package/skills/vue-best-practices/references/sfc.md +310 -0
  26. package/skills/vue-best-practices/references/state-management.md +135 -0
  27. package/skills/vue-best-practices/references/updated-hook-performance.md +187 -0
  28. package/skills/vue-debug-guides/SKILL.md +205 -0
  29. package/skills/vue-debug-guides/reference/animation-key-for-rerender.md +160 -0
  30. package/skills/vue-debug-guides/reference/animation-transitiongroup-performance.md +241 -0
  31. package/skills/vue-debug-guides/reference/async-component-error-handling.md +115 -0
  32. package/skills/vue-debug-guides/reference/async-component-keepalive-ref-issue.md +112 -0
  33. package/skills/vue-debug-guides/reference/async-component-suspense-control.md +84 -0
  34. package/skills/vue-debug-guides/reference/async-component-vue-router.md +109 -0
  35. package/skills/vue-debug-guides/reference/attrs-event-listener-merging.md +205 -0
  36. package/skills/vue-debug-guides/reference/checkbox-true-false-value-form-submission.md +118 -0
  37. package/skills/vue-debug-guides/reference/cleanup-side-effects.md +172 -0
  38. package/skills/vue-debug-guides/reference/click-events-on-components.md +180 -0
  39. package/skills/vue-debug-guides/reference/component-naming-conflicts.md +159 -0
  40. package/skills/vue-debug-guides/reference/component-ref-requires-defineexpose.md +176 -0
  41. package/skills/vue-debug-guides/reference/composable-avoid-hidden-side-effects.md +208 -0
  42. package/skills/vue-debug-guides/reference/composable-call-location-restrictions.md +141 -0
  43. package/skills/vue-debug-guides/reference/composable-naming-return-pattern.md +139 -0
  44. package/skills/vue-debug-guides/reference/composable-tovalue-inside-watcheffect.md +182 -0
  45. package/skills/vue-debug-guides/reference/composition-api-not-functional-programming.md +120 -0
  46. package/skills/vue-debug-guides/reference/composition-api-script-setup-async-context.md +203 -0
  47. package/skills/vue-debug-guides/reference/composition-api-vs-react-hooks-differences.md +156 -0
  48. package/skills/vue-debug-guides/reference/computed-array-mutation.md +148 -0
  49. package/skills/vue-debug-guides/reference/computed-conditional-dependencies.md +147 -0
  50. package/skills/vue-debug-guides/reference/computed-no-parameters.md +159 -0
  51. package/skills/vue-debug-guides/reference/computed-no-side-effects.md +107 -0
  52. package/skills/vue-debug-guides/reference/computed-return-value-readonly.md +160 -0
  53. package/skills/vue-debug-guides/reference/configure-app-before-mount.md +89 -0
  54. package/skills/vue-debug-guides/reference/declare-emits-for-documentation.md +212 -0
  55. package/skills/vue-debug-guides/reference/define-expose-before-await.md +192 -0
  56. package/skills/vue-debug-guides/reference/define-model-default-value-sync.md +139 -0
  57. package/skills/vue-debug-guides/reference/defineEmits-must-be-top-level.md +164 -0
  58. package/skills/vue-debug-guides/reference/defineEmits-no-runtime-and-type-mixed.md +170 -0
  59. package/skills/vue-debug-guides/reference/definemodel-object-mutation-no-emit.md +148 -0
  60. package/skills/vue-debug-guides/reference/dom-update-timing-nexttick.md +90 -0
  61. package/skills/vue-debug-guides/reference/dynamic-argument-constraints.md +146 -0
  62. package/skills/vue-debug-guides/reference/dynamic-component-registration-vite.md +147 -0
  63. package/skills/vue-debug-guides/reference/event-modifier-order-matters.md +101 -0
  64. package/skills/vue-debug-guides/reference/exact-modifier-for-precise-shortcuts.md +155 -0
  65. package/skills/vue-debug-guides/reference/fallthrough-attrs-overwrite-vue3.md +159 -0
  66. package/skills/vue-debug-guides/reference/in-dom-template-parsing-caveats.md +149 -0
  67. package/skills/vue-debug-guides/reference/inheritattrs-false-for-wrapper-components.md +230 -0
  68. package/skills/vue-debug-guides/reference/keepalive-router-nested-double-mount.md +222 -0
  69. package/skills/vue-debug-guides/reference/keepalive-transition-memory-leak.md +144 -0
  70. package/skills/vue-debug-guides/reference/keyup-modifier-timing.md +137 -0
  71. package/skills/vue-debug-guides/reference/lifecycle-dom-access-timing.md +216 -0
  72. package/skills/vue-debug-guides/reference/lifecycle-hooks-synchronous-registration.md +156 -0
  73. package/skills/vue-debug-guides/reference/lifecycle-ssr-awareness.md +184 -0
  74. package/skills/vue-debug-guides/reference/local-components-not-in-descendants.md +151 -0
  75. package/skills/vue-debug-guides/reference/mount-return-value.md +88 -0
  76. package/skills/vue-debug-guides/reference/multi-root-component-class-attrs.md +93 -0
  77. package/skills/vue-debug-guides/reference/native-event-collision-with-emits.md +162 -0
  78. package/skills/vue-debug-guides/reference/no-passive-with-prevent.md +141 -0
  79. package/skills/vue-debug-guides/reference/no-v-if-with-v-for.md +136 -0
  80. package/skills/vue-debug-guides/reference/perf-computed-object-stability.md +157 -0
  81. package/skills/vue-debug-guides/reference/perf-props-stability-update-optimization.md +140 -0
  82. package/skills/vue-debug-guides/reference/plugin-global-properties-sparingly.md +109 -0
  83. package/skills/vue-debug-guides/reference/plugin-install-before-mount.md +124 -0
  84. package/skills/vue-debug-guides/reference/plugin-prefer-provide-inject-over-global-properties.md +120 -0
  85. package/skills/vue-debug-guides/reference/plugin-typescript-type-augmentation.md +157 -0
  86. package/skills/vue-debug-guides/reference/prop-defineprops-scope-limitation.md +161 -0
  87. package/skills/vue-debug-guides/reference/provide-inject-debugging-challenges.md +203 -0
  88. package/skills/vue-debug-guides/reference/provide-inject-default-value-factory.md +244 -0
  89. package/skills/vue-debug-guides/reference/provide-inject-reactivity-not-automatic.md +226 -0
  90. package/skills/vue-debug-guides/reference/provide-inject-synchronous-setup.md +235 -0
  91. package/skills/vue-debug-guides/reference/reactive-destructuring.md +89 -0
  92. package/skills/vue-debug-guides/reference/reactivity-debugging-hooks.md +132 -0
  93. package/skills/vue-debug-guides/reference/reactivity-markraw-for-non-reactive.md +149 -0
  94. package/skills/vue-debug-guides/reference/reactivity-proxy-identity-hazard.md +96 -0
  95. package/skills/vue-debug-guides/reference/reactivity-same-tick-batching.md +166 -0
  96. package/skills/vue-debug-guides/reference/ref-value-access.md +61 -0
  97. package/skills/vue-debug-guides/reference/refs-in-collections-need-value.md +81 -0
  98. package/skills/vue-debug-guides/reference/render-function-avoid-internal-vnode-properties.md +151 -0
  99. package/skills/vue-debug-guides/reference/render-function-vnodes-must-be-unique.md +133 -0
  100. package/skills/vue-debug-guides/reference/rendering-render-function-h-import-vue3.md +148 -0
  101. package/skills/vue-debug-guides/reference/rendering-render-function-return-from-setup.md +148 -0
  102. package/skills/vue-debug-guides/reference/rendering-render-function-slots-as-functions.md +168 -0
  103. package/skills/vue-debug-guides/reference/rendering-resolve-component-for-string-names.md +231 -0
  104. package/skills/vue-debug-guides/reference/select-initial-value-ios-bug.md +91 -0
  105. package/skills/vue-debug-guides/reference/self-referencing-component-name.md +157 -0
  106. package/skills/vue-debug-guides/reference/sfc-named-exports-forbidden.md +184 -0
  107. package/skills/vue-debug-guides/reference/sfc-scoped-css-child-component-styling.md +156 -0
  108. package/skills/vue-debug-guides/reference/sfc-scoped-css-dynamic-content.md +193 -0
  109. package/skills/vue-debug-guides/reference/sfc-scoped-css-slot-content.md +242 -0
  110. package/skills/vue-debug-guides/reference/sfc-script-setup-reactivity.md +195 -0
  111. package/skills/vue-debug-guides/reference/slot-forwarding-to-child-components.md +143 -0
  112. package/skills/vue-debug-guides/reference/slot-implicit-default-content.md +155 -0
  113. package/skills/vue-debug-guides/reference/slot-name-reserved-prop.md +109 -0
  114. package/skills/vue-debug-guides/reference/slot-named-scoped-explicit-default.md +95 -0
  115. package/skills/vue-debug-guides/reference/slot-render-scope-parent-only.md +135 -0
  116. package/skills/vue-debug-guides/reference/slot-v-slot-on-components-or-templates-only.md +122 -0
  117. package/skills/vue-debug-guides/reference/ssr-hydration-mismatch-causes.md +280 -0
  118. package/skills/vue-debug-guides/reference/ssr-platform-specific-apis.md +256 -0
  119. package/skills/vue-debug-guides/reference/state-ssr-cross-request-pollution.md +276 -0
  120. package/skills/vue-debug-guides/reference/suspense-no-builtin-error-handling.md +127 -0
  121. package/skills/vue-debug-guides/reference/suspense-ssr-hydration-issues.md +159 -0
  122. package/skills/vue-debug-guides/reference/tailwind-dynamic-class-generation.md +144 -0
  123. package/skills/vue-debug-guides/reference/teleport-scoped-styles-limitation.md +191 -0
  124. package/skills/vue-debug-guides/reference/teleport-ssr-hydration.md +152 -0
  125. package/skills/vue-debug-guides/reference/teleport-target-must-exist.md +113 -0
  126. package/skills/vue-debug-guides/reference/template-expressions-restrictions.md +114 -0
  127. package/skills/vue-debug-guides/reference/template-functions-no-side-effects.md +187 -0
  128. package/skills/vue-debug-guides/reference/template-ref-null-with-v-if.md +123 -0
  129. package/skills/vue-debug-guides/reference/template-ref-unwrapping-top-level.md +104 -0
  130. package/skills/vue-debug-guides/reference/template-ref-v-for-order.md +172 -0
  131. package/skills/vue-debug-guides/reference/textarea-no-interpolation.md +72 -0
  132. package/skills/vue-debug-guides/reference/transition-group-flip-inline-elements.md +152 -0
  133. package/skills/vue-debug-guides/reference/transition-group-move-animation-position-absolute.md +130 -0
  134. package/skills/vue-debug-guides/reference/transition-group-no-default-wrapper-vue3.md +152 -0
  135. package/skills/vue-debug-guides/reference/transition-js-hooks-done-callback.md +251 -0
  136. package/skills/vue-debug-guides/reference/transition-nested-duration.md +182 -0
  137. package/skills/vue-debug-guides/reference/transition-reusable-scoped-style.md +245 -0
  138. package/skills/vue-debug-guides/reference/transition-router-view-appear.md +193 -0
  139. package/skills/vue-debug-guides/reference/transition-type-when-mixed.md +172 -0
  140. package/skills/vue-debug-guides/reference/transition-unmount-hook-timing.md +149 -0
  141. package/skills/vue-debug-guides/reference/ts-defineprops-boolean-default-false.md +225 -0
  142. package/skills/vue-debug-guides/reference/ts-defineprops-imported-types-limitations.md +281 -0
  143. package/skills/vue-debug-guides/reference/ts-event-handler-explicit-typing.md +213 -0
  144. package/skills/vue-debug-guides/reference/ts-reactive-no-generic-argument.md +196 -0
  145. package/skills/vue-debug-guides/reference/ts-shallowref-for-dynamic-components.md +218 -0
  146. package/skills/vue-debug-guides/reference/ts-template-ref-null-handling.md +249 -0
  147. package/skills/vue-debug-guides/reference/ts-template-type-casting.md +214 -0
  148. package/skills/vue-debug-guides/reference/ts-withdefaults-mutable-factory-function.md +171 -0
  149. package/skills/vue-debug-guides/reference/undeclared-emits-double-firing.md +195 -0
  150. package/skills/vue-debug-guides/reference/use-template-ref-vue35.md +158 -0
  151. package/skills/vue-debug-guides/reference/v-else-must-follow-v-if.md +136 -0
  152. package/skills/vue-debug-guides/reference/v-for-component-props.md +95 -0
  153. package/skills/vue-debug-guides/reference/v-for-computed-reverse-sort.md +86 -0
  154. package/skills/vue-debug-guides/reference/v-for-key-attribute.md +90 -0
  155. package/skills/vue-debug-guides/reference/v-for-range-starts-at-one.md +66 -0
  156. package/skills/vue-debug-guides/reference/v-if-null-check-order.md +171 -0
  157. package/skills/vue-debug-guides/reference/v-model-ignores-html-attributes.md +83 -0
  158. package/skills/vue-debug-guides/reference/v-model-ime-composition.md +83 -0
  159. package/skills/vue-debug-guides/reference/v-model-number-modifier-behavior.md +124 -0
  160. package/skills/vue-debug-guides/reference/v-show-template-limitation.md +124 -0
  161. package/skills/vue-debug-guides/reference/watch-async-cleanup.md +180 -0
  162. package/skills/vue-debug-guides/reference/watch-async-creation-memory-leak.md +176 -0
  163. package/skills/vue-debug-guides/reference/watch-deep-same-object-reference.md +165 -0
  164. package/skills/vue-debug-guides/reference/watch-flush-timing.md +189 -0
  165. package/skills/vue-debug-guides/reference/watch-reactive-property-getter.md +108 -0
  166. package/skills/vue-debug-guides/reference/watcheffect-async-dependency-tracking.md +173 -0
  167. package/skills/vue-debug-guides/reference/watcheffect-flush-post-for-refs.md +176 -0
  168. package/skills/vue-jsx-best-practices/SKILL.md +12 -0
  169. package/skills/vue-jsx-best-practices/reference/render-function-jsx-vue-vs-react.md +141 -0
  170. package/skills/vue-options-api-best-practices/SKILL.md +23 -0
  171. package/skills/vue-options-api-best-practices/reference/no-arrow-functions-in-lifecycle-hooks.md +95 -0
  172. package/skills/vue-options-api-best-practices/reference/no-arrow-functions-in-methods.md +68 -0
  173. package/skills/vue-options-api-best-practices/reference/stateful-methods-lifecycle.md +61 -0
  174. package/skills/vue-options-api-best-practices/reference/ts-options-api-arrow-functions-validators.md +141 -0
  175. package/skills/vue-options-api-best-practices/reference/ts-options-api-computed-return-types.md +192 -0
  176. package/skills/vue-options-api-best-practices/reference/ts-options-api-proptype-complex-types.md +212 -0
  177. package/skills/vue-options-api-best-practices/reference/ts-options-api-provide-inject-limitations.md +135 -0
  178. package/skills/vue-options-api-best-practices/reference/ts-options-api-type-event-handlers.md +202 -0
  179. package/skills/vue-options-api-best-practices/reference/ts-options-api-use-definecomponent.md +172 -0
  180. package/skills/vue-options-api-best-practices/reference/ts-strict-mode-options-api.md +197 -0
  181. package/skills/vue-pinia-best-practices/SKILL.md +21 -0
  182. package/skills/vue-pinia-best-practices/reference/pinia-no-active-pinia-error.md +248 -0
  183. package/skills/vue-pinia-best-practices/reference/pinia-setup-store-return-all-state.md +227 -0
  184. package/skills/vue-pinia-best-practices/reference/pinia-store-destructuring-breaks-reactivity.md +193 -0
  185. package/skills/vue-pinia-best-practices/reference/state-url-for-ephemeral-filters.md +238 -0
  186. package/skills/vue-pinia-best-practices/reference/state-use-pinia-for-large-apps.md +262 -0
  187. package/skills/vue-pinia-best-practices/reference/store-method-binding-parentheses.md +191 -0
  188. package/skills/vue-router-best-practices/SKILL.md +23 -0
  189. package/skills/vue-router-best-practices/reference/router-beforeenter-no-param-trigger.md +167 -0
  190. package/skills/vue-router-best-practices/reference/router-beforerouteenter-no-this.md +176 -0
  191. package/skills/vue-router-best-practices/reference/router-guard-async-await-pattern.md +227 -0
  192. package/skills/vue-router-best-practices/reference/router-navigation-guard-infinite-loop.md +187 -0
  193. package/skills/vue-router-best-practices/reference/router-navigation-guard-next-deprecated.md +150 -0
  194. package/skills/vue-router-best-practices/reference/router-param-change-no-lifecycle.md +181 -0
  195. package/skills/vue-router-best-practices/reference/router-simple-routing-cleanup.md +209 -0
  196. package/skills/vue-router-best-practices/reference/router-use-vue-router-for-production.md +183 -0
  197. package/skills/vue-testing-best-practices/SKILL.md +29 -0
  198. package/skills/vue-testing-best-practices/reference/async-component-testing.md +163 -0
  199. package/skills/vue-testing-best-practices/reference/teleport-testing-complexity.md +158 -0
  200. package/skills/vue-testing-best-practices/reference/testing-async-await-flushpromises.md +175 -0
  201. package/skills/vue-testing-best-practices/reference/testing-browser-vs-node-runners.md +208 -0
  202. package/skills/vue-testing-best-practices/reference/testing-component-blackbox-approach.md +144 -0
  203. package/skills/vue-testing-best-practices/reference/testing-composables-helper-wrapper.md +238 -0
  204. package/skills/vue-testing-best-practices/reference/testing-e2e-playwright-recommended.md +242 -0
  205. package/skills/vue-testing-best-practices/reference/testing-no-snapshot-only.md +197 -0
  206. package/skills/vue-testing-best-practices/reference/testing-pinia-store-setup.md +228 -0
  207. package/skills/vue-testing-best-practices/reference/testing-suspense-async-components.md +229 -0
  208. package/skills/vue-testing-best-practices/reference/testing-vitest-recommended-for-vue.md +204 -0
package/README.md ADDED
@@ -0,0 +1,63 @@
1
+ # vue-skills MCP server
2
+
3
+ Part of [Pythoughts-labs/vue3-best-practices](https://github.com/Pythoughts-labs/vue3-best-practices).
4
+
5
+ Exposes the Vue 3 best-practice skills in this repo as MCP tools, so any MCP
6
+ coding agent can fetch them while working on Vue.
7
+
8
+ ## Tools
9
+
10
+ - **`vue_best_practices`** — primary tool. No args → full index (every skill's
11
+ workflow + its reference filenames). Pass `skill` to scope to one. Its
12
+ description is written to fire on any Vue signal (`.vue`, `<script setup>`,
13
+ Pinia, Vue Router, Vitest, Vue SSR…).
14
+ - **`get_vue_reference`** — full content of one reference file (deep dive).
15
+
16
+ Progressive disclosure: the index returns overviews + a reference list; the
17
+ agent pulls only the references it needs.
18
+
19
+ ## Setup
20
+
21
+ ```bash
22
+ cd mcp && npm install
23
+ ```
24
+
25
+ Register the server with your agent. Replace `<REPO>` with this repo's absolute
26
+ path.
27
+
28
+ ### Claude Code
29
+
30
+ ```bash
31
+ claude mcp add vue-skills -- node <REPO>/mcp/index.mjs
32
+ ```
33
+
34
+ ### Cursor / Windsurf (`~/.cursor/mcp.json` or `mcp_config.json`)
35
+
36
+ ```json
37
+ {
38
+ "mcpServers": {
39
+ "vue-skills": { "command": "node", "args": ["<REPO>/mcp/index.mjs"] }
40
+ }
41
+ }
42
+ ```
43
+
44
+ `VUE_SKILLS_DIR` overrides where skills are read from (defaults to `../skills`).
45
+
46
+ ## About "automatic"
47
+
48
+ MCP has no push trigger — an agent calls a tool when it judges the tool's
49
+ description relevant. There is no way to *force* a call. This server maximizes
50
+ the chance by engineering `vue_best_practices`'s description to match any Vue
51
+ signal. To make it reliable, add one line to your project's agent rules
52
+ (`AGENTS.md`, `.cursorrules`, Claude Code `CLAUDE.md`):
53
+
54
+ > Before any Vue work, call the `vue_best_practices` MCP tool and apply it.
55
+
56
+ That rule is the actual "always fetch on Vue work" guarantee; the MCP server is
57
+ what makes the content callable.
58
+
59
+ ## Test
60
+
61
+ ```bash
62
+ npm test
63
+ ```
package/index.mjs ADDED
@@ -0,0 +1,139 @@
1
+ #!/usr/bin/env node
2
+ // MCP server exposing the Vue 3 best-practice skills in this repo.
3
+ // Any MCP client (Claude Code, Cursor, Windsurf, ...) can call these tools.
4
+ // The agent decides when to call; the tool descriptions are written so a Vue
5
+ // signal (a .vue file, <script setup>, Pinia, Vue Router, Vite+Vue) triggers it.
6
+
7
+ import { readdirSync, readFileSync, statSync, existsSync } from "node:fs";
8
+ import { dirname, join, resolve } from "node:path";
9
+ import { fileURLToPath } from "node:url";
10
+ import { McpServer } from "@modelcontextprotocol/sdk/server/mcp.js";
11
+ import { StdioServerTransport } from "@modelcontextprotocol/sdk/server/stdio.js";
12
+ import { z } from "zod";
13
+
14
+ const __dirname = dirname(fileURLToPath(import.meta.url));
15
+ // Resolution order: VUE_SKILLS_DIR override, then the bundled skills/ shipped
16
+ // inside the published package, then ../skills for in-repo (dev) use.
17
+ function resolveSkillsDir() {
18
+ if (process.env.VUE_SKILLS_DIR) return resolve(process.env.VUE_SKILLS_DIR);
19
+ const bundled = join(__dirname, "skills");
20
+ return existsSync(bundled) ? bundled : join(__dirname, "..", "skills");
21
+ }
22
+ const SKILLS_DIR = resolveSkillsDir();
23
+
24
+ // --- Discover skills at startup -------------------------------------------
25
+ // Build an explicit name -> {description, skillMd, references{name->path}} map.
26
+ // References are served only from this prebuilt map, so an agent-supplied name
27
+ // can never traverse outside the skills directory.
28
+ function loadSkills(root) {
29
+ const skills = {};
30
+ let dirs;
31
+ try {
32
+ dirs = readdirSync(root).filter((d) => statSync(join(root, d)).isDirectory());
33
+ } catch (err) {
34
+ throw new Error(`Cannot read skills dir "${root}": ${err.message}`);
35
+ }
36
+ for (const name of dirs) {
37
+ const skillPath = join(root, name, "SKILL.md");
38
+ let skillMd;
39
+ try {
40
+ skillMd = readFileSync(skillPath, "utf8");
41
+ } catch {
42
+ continue; // not a skill dir
43
+ }
44
+ const references = {};
45
+ for (const refDir of ["references", "reference"]) {
46
+ const full = join(root, name, refDir);
47
+ try {
48
+ for (const f of readdirSync(full)) {
49
+ if (f.endsWith(".md")) references[f] = join(full, f);
50
+ }
51
+ } catch {
52
+ // dir absent; ignore
53
+ }
54
+ }
55
+ skills[name] = { description: parseDescription(skillMd), skillMd, references };
56
+ }
57
+ if (Object.keys(skills).length === 0) {
58
+ throw new Error(`No skills with a SKILL.md found under "${root}"`);
59
+ }
60
+ return skills;
61
+ }
62
+
63
+ function parseDescription(md) {
64
+ const m = md.match(/^description:\s*(.+)$/m);
65
+ return m ? m[1].trim() : "(no description)";
66
+ }
67
+
68
+ const SKILLS = loadSkills(SKILLS_DIR);
69
+ const SKILL_NAMES = Object.keys(SKILLS).sort();
70
+
71
+ // --- Server ---------------------------------------------------------------
72
+ const server = new McpServer({ name: "vue-skills", version: "1.0.0" });
73
+
74
+ const TRIGGER =
75
+ "Call this IMMEDIATELY and BEFORE writing or editing any code when you detect " +
76
+ "Vue work: a .vue file, Composition API, <script setup>, Options API, Vue Router, " +
77
+ "Pinia, Vite + Vue, Vue testing (Vitest/Vue Test Utils/Playwright), Vue JSX, or " +
78
+ "Vue SSR/hydration. Returns the current Vue 3 best-practice index so you apply " +
79
+ "up-to-date patterns instead of relying on training data.";
80
+
81
+ // Primary auto-trigger tool: returns the full best-practice index.
82
+ server.tool(
83
+ "vue_best_practices",
84
+ TRIGGER +
85
+ " With no arguments, returns every skill's overview (the actionable workflow) " +
86
+ "plus the list of deep-dive reference files. Pass `skill` to get one skill's " +
87
+ "overview only. Pull a reference's full content with the `get_vue_reference` tool.",
88
+ { skill: z.enum(SKILL_NAMES).optional().describe("Limit output to one skill.") },
89
+ async ({ skill }) => {
90
+ const names = skill ? [skill] : SKILL_NAMES;
91
+ const sections = names.map((name) => {
92
+ const s = SKILLS[name];
93
+ const refs = Object.keys(s.references).sort();
94
+ const refList = refs.length
95
+ ? `\n\nReference files (fetch via get_vue_reference("${name}", <file>)):\n` +
96
+ refs.map((r) => `- ${r}`).join("\n")
97
+ : "";
98
+ return `\n\n${"=".repeat(72)}\n# SKILL: ${name}\n${"=".repeat(72)}\n\n${s.skillMd}${refList}`;
99
+ });
100
+ const header = skill
101
+ ? `Vue best-practice skill: ${skill}`
102
+ : `Vue 3 best-practice index — ${SKILL_NAMES.length} skills. Read the overviews, ` +
103
+ `then fetch only the reference files relevant to your task.`;
104
+ return { content: [{ type: "text", text: header + sections.join("") }] };
105
+ }
106
+ );
107
+
108
+ // Drill-down tool: full content of one reference file.
109
+ server.tool(
110
+ "get_vue_reference",
111
+ "Return the full content of one Vue best-practice reference file (deep-dive on a " +
112
+ "specific pattern). Use the skill name and reference filename listed by " +
113
+ "vue_best_practices.",
114
+ {
115
+ skill: z.enum(SKILL_NAMES).describe("Skill that owns the reference."),
116
+ reference: z.string().describe("Reference filename, e.g. reactivity.md"),
117
+ },
118
+ async ({ skill, reference }) => {
119
+ const path = SKILLS[skill]?.references[reference];
120
+ if (!path) {
121
+ const available = Object.keys(SKILLS[skill]?.references ?? {}).sort();
122
+ return {
123
+ isError: true,
124
+ content: [
125
+ {
126
+ type: "text",
127
+ text:
128
+ `No reference "${reference}" in skill "${skill}". ` +
129
+ `Available: ${available.join(", ") || "(none)"}`,
130
+ },
131
+ ],
132
+ };
133
+ }
134
+ return { content: [{ type: "text", text: readFileSync(path, "utf8") }] };
135
+ }
136
+ );
137
+
138
+ await server.connect(new StdioServerTransport());
139
+ console.error(`vue-skills MCP ready: ${SKILL_NAMES.length} skills from ${SKILLS_DIR}`);
package/package.json ADDED
@@ -0,0 +1,34 @@
1
+ {
2
+ "name": "@pythoughts/vue-skills-mcp",
3
+ "version": "0.1.0",
4
+ "type": "module",
5
+ "description": "MCP server exposing the Vue 3 best-practice skills so any MCP coding agent can fetch them automatically on Vue work.",
6
+ "author": "Mohamed Elkholy (elkaix)",
7
+ "license": "MIT",
8
+ "homepage": "https://github.com/Pythoughts-labs/vue3-best-practices",
9
+ "repository": {
10
+ "type": "git",
11
+ "url": "git+https://github.com/Pythoughts-labs/vue3-best-practices.git",
12
+ "directory": "mcp"
13
+ },
14
+ "bin": {
15
+ "vue-skills-mcp": "./index.mjs"
16
+ },
17
+ "files": [
18
+ "index.mjs",
19
+ "skills"
20
+ ],
21
+ "publishConfig": {
22
+ "access": "public"
23
+ },
24
+ "scripts": {
25
+ "start": "node index.mjs",
26
+ "test": "node smoke-test.mjs",
27
+ "prepack": "rm -rf skills && cp -R ../skills ./skills",
28
+ "postpack": "rm -rf skills"
29
+ },
30
+ "dependencies": {
31
+ "@modelcontextprotocol/sdk": "^1.12.0",
32
+ "zod": "^3.23.8"
33
+ }
34
+ }
@@ -0,0 +1,76 @@
1
+ ---
2
+ name: create-adaptable-composable
3
+ description: Create a library-grade Vue composable that accepts maybe-reactive inputs (MaybeRef / MaybeRefOrGetter) so callers can pass a plain value, ref, or getter. Normalize inputs with toValue()/toRef() inside reactive effects (watch/watchEffect) to keep behavior predictable and reactive. Use this skill when user asks for creating adaptable or reusable composables.
4
+ license: MIT
5
+ metadata:
6
+ author: github.com/Pythoughts-labs
7
+ version: "17.0.0"
8
+ compatibility: Requires Vue 3 (or above) or Nuxt 3 (or above) project
9
+ ---
10
+
11
+ # Create Adaptable Composable
12
+
13
+ Adaptable composables are reusable functions that can accept both reactive and non-reactive inputs. This allows developers to use the composable in a variety of contexts without worrying about the reactivity of the inputs.
14
+
15
+ Steps to design an adaptable composable in Vue.js:
16
+ 1. Confirm the composable's purpose and API design and expected inputs/outputs.
17
+ 2. Identify inputs params that should be reactive (MaybeRef / MaybeRefOrGetter).
18
+ 3. Use `toValue()` or `toRef()` to normalize inputs inside reactive effects.
19
+ 4. Implement the core logic of the composable using Vue's reactivity APIs.
20
+
21
+ ## Core Type Concepts
22
+
23
+ ### Type Utilities
24
+
25
+ ```ts
26
+ /**
27
+ * value or writable ref (value/ref/shallowRef/writable computed)
28
+ */
29
+ export type MaybeRef<T = any> = T | Ref<T> | ShallowRef<T> | WritableComputedRef<T>;
30
+
31
+ /**
32
+ * MaybeRef<T> + ComputedRef<T> + () => T
33
+ */
34
+ export type MaybeRefOrGetter<T = any> = MaybeRef<T> | ComputedRef<T> | (() => T);
35
+ ```
36
+
37
+ ### Policy and Rules
38
+
39
+ - Read-only, computed-friendly input: use `MaybeRefOrGetter`
40
+ - Needs to be writable / two-way input: use `MaybeRef`
41
+ - Parameter might be a function value (callback/predicate/comparator): do not use `MaybeRefOrGetter`, or you may accidentally invoke it as a getter.
42
+ - DOM/Element targets: if you want computed/derived targets, use `MaybeRefOrGetter`.
43
+
44
+ When `MaybeRefOrGetter` or `MaybeRef` is used:
45
+ - resolve reactive value using `toRef()` (e.g. watcher source)
46
+ - resolve non-reactive value using `toValue()`
47
+
48
+ ### Examples
49
+
50
+ Adaptable `useDocumentTitle` Composable: read-only title parameter
51
+
52
+ ```ts
53
+ import { watch, toRef } from 'vue'
54
+ import type { MaybeRefOrGetter } from 'vue'
55
+
56
+ export function useDocumentTitle(title: MaybeRefOrGetter<string>) {
57
+ watch(toRef(title), (t) => {
58
+ document.title = t
59
+ }, { immediate: true })
60
+ }
61
+ ```
62
+
63
+ Adaptable `useCounter` Composable: two-way writable count parameter
64
+
65
+ ```ts
66
+ import { watch, toRef } from 'vue'
67
+ import type { MaybeRef } from 'vue'
68
+
69
+ function useCounter(count: MaybeRef<number>) {
70
+ const countRef = toRef(count)
71
+ function add() {
72
+ countRef.value++
73
+ }
74
+ return { add }
75
+ }
76
+ ```
@@ -0,0 +1,154 @@
1
+ ---
2
+ name: vue-best-practices
3
+ description: MUST be used for Vue.js tasks. Strongly recommends Composition API with `<script setup>` and TypeScript as the standard approach. Covers Vue 3, SSR, Volar, vue-tsc. Load for any Vue, .vue files, Vue Router, Pinia, or Vite with Vue work. ALWAYS use Composition API unless the project explicitly requires Options API.
4
+ license: MIT
5
+ metadata:
6
+ author: github.com/Pythoughts-labs
7
+ version: "18.1.0"
8
+ ---
9
+
10
+ # Vue Best Practices Workflow
11
+
12
+ Use this skill as an instruction set. Follow the workflow in order unless the user explicitly asks for a different order.
13
+
14
+ ## Core Principles
15
+ - **Keep state predictable:** one source of truth, derive everything else.
16
+ - **Make data flow explicit:** Props down, Events up for most cases.
17
+ - **Favor small, focused components:** easier to test, reuse, and maintain.
18
+ - **Avoid unnecessary re-renders:** use computed properties and watchers wisely.
19
+ - **Readability counts:** write clear, self-documenting code.
20
+
21
+ ## 1) Confirm architecture before coding (required)
22
+
23
+ - Default stack: Vue 3 + Composition API + `<script setup lang="ts">`.
24
+ - If the project explicitly uses Options API, load `vue-options-api-best-practices` skill if available.
25
+ - If the project explicitly uses JSX, load `vue-jsx-best-practices` skill if available.
26
+
27
+ ### 1.1 Must-read core references (required)
28
+
29
+ - Before implementing any Vue task, make sure to read and apply these core references:
30
+ - `references/reactivity.md`
31
+ - `references/sfc.md`
32
+ - `references/component-data-flow.md`
33
+ - `references/composables.md`
34
+ - Keep these references in active working context for the entire task, not only when a specific issue appears.
35
+
36
+ ### 1.2 Plan component boundaries before coding (required)
37
+
38
+ Create a brief component map before implementation for any non-trivial feature.
39
+
40
+ - Define each component's single responsibility in one sentence.
41
+ - Keep entry/root and route-level view components as composition surfaces by default.
42
+ - Move feature UI and feature logic out of entry/root/view components unless the task is intentionally a tiny single-file demo.
43
+ - Define props/emits contracts for each child component in the map.
44
+ - Prefer a feature folder layout (`components/<feature>/...`, `composables/use<Feature>.ts`) when adding more than one component.
45
+
46
+ ## 2) Apply essential Vue foundations (required)
47
+
48
+ These are essential, must-know foundations. Apply all of them in every Vue task using the core references already loaded in section `1.1`.
49
+
50
+ ### Reactivity
51
+
52
+ - Must-read reference from `1.1`: [reactivity](references/reactivity.md)
53
+ - Keep source state minimal (`ref`/`reactive`), derive everything possible with `computed`.
54
+ - Use watchers for side effects if needed.
55
+ - Avoid recomputing expensive logic in templates.
56
+
57
+ ### SFC structure and template safety
58
+
59
+ - Must-read reference from `1.1`: [sfc](references/sfc.md)
60
+ - Keep SFC sections in this order: `<script>` → `<template>` → `<style>`.
61
+ - Keep SFC responsibilities focused; split large components.
62
+ - Keep templates declarative; move branching/derivation to script.
63
+ - Apply Vue template safety rules (`v-html`, list rendering, conditional rendering choices).
64
+
65
+ ### Keep components focused
66
+
67
+ Split a component when it has **more than one clear responsibility** (e.g. data orchestration + UI, or multiple independent UI sections).
68
+
69
+ - Prefer **smaller components + composables** over one “mega component”
70
+ - Move **UI sections** into child components (props in, events out).
71
+ - Move **state/side effects** into composables (`useXxx()`).
72
+
73
+ Apply objective split triggers. Split the component if **any** condition is true:
74
+
75
+ - It owns both orchestration/state and substantial presentational markup for multiple sections.
76
+ - It has 3+ distinct UI sections (for example: form, filters, list, footer/status).
77
+ - A template block is repeated or could become reusable (item rows, cards, list entries).
78
+
79
+ Entry/root and route view rule:
80
+
81
+ - Keep entry/root and route view components thin: app shell/layout, provider wiring, and feature composition.
82
+ - Do not place full feature implementations in entry/root/view components when those features contain independent parts.
83
+ - For CRUD/list features (todo, table, catalog, inbox), split at least into:
84
+ - feature container component
85
+ - input/form component
86
+ - list (and/or item) component
87
+ - footer/actions or filter/status component
88
+ - Allow a single-file implementation only for very small throwaway demos; if chosen, explicitly justify why splitting is unnecessary.
89
+
90
+ ### Component data flow
91
+
92
+ - Must-read reference from `1.1`: [component-data-flow](references/component-data-flow.md)
93
+ - Use props down, events up as the primary model.
94
+ - Use `v-model` only for true two-way component contracts.
95
+ - Use provide/inject only for deep-tree dependencies or shared context.
96
+ - Keep contracts explicit and typed with `defineProps`, `defineEmits`, and `InjectionKey` as needed.
97
+
98
+ ### Composables
99
+
100
+ - Must-read reference from `1.1`: [composables](references/composables.md)
101
+ - Extract logic into composables when it is reused, stateful, or side-effect heavy.
102
+ - Keep composable APIs small, typed, and predictable.
103
+ - Separate feature logic from presentational components.
104
+
105
+ ## 3) Consider optional features only when requirements call for them
106
+
107
+ ### 3.1 Standard optional features
108
+
109
+ Do not add these by default. Load the matching reference only when the requirement exists.
110
+
111
+ - Slots: parent needs to control child content/layout -> [component-slots](references/component-slots.md)
112
+ - Fallthrough attributes: wrapper/base components must forward attrs/events safely -> [component-fallthrough-attrs](references/component-fallthrough-attrs.md)
113
+ - Built-in component `<KeepAlive>` for stateful view caching -> [component-keep-alive](references/component-keep-alive.md)
114
+ - Built-in component `<Teleport>` for overlays/portals -> [component-teleport](references/component-teleport.md)
115
+ - Built-in component `<Suspense>` for async subtree fallback boundaries -> [component-suspense](references/component-suspense.md)
116
+ - Animation-related features: pick the simplest approach that matches the required motion behavior.
117
+ - Built-in component `<Transition>` for enter/leave effects -> [transition](references/component-transition.md)
118
+ - Built-in component `<TransitionGroup>` for animated list mutations -> [transition-group](references/component-transition-group.md)
119
+ - Class-based animation for non-enter/leave effects -> [animation-class-based-technique](references/animation-class-based-technique.md)
120
+ - State-driven animation for user-input-driven animation -> [animation-state-driven-technique](references/animation-state-driven-technique.md)
121
+
122
+ ### 3.2 Less-common optional features
123
+
124
+ Use these only when there is explicit product or technical need.
125
+
126
+ - Directives: behavior is DOM-specific and not a good composable/component fit -> [directives](references/directives.md)
127
+ - Async components: heavy/rarely-used UI should be lazy loaded -> [component-async](references/component-async.md)
128
+ - Render functions only when templates cannot express the requirement -> [render-functions](references/render-functions.md)
129
+ - Plugins when behavior must be installed app-wide -> [plugins](references/plugins.md)
130
+ - State management patterns: app-wide shared state crosses feature boundaries -> [state-management](references/state-management.md)
131
+
132
+ ## 4) Run performance optimization after behavior is correct
133
+
134
+ Performance work is a post-functionality pass. Do not optimize before core behavior is implemented and verified.
135
+
136
+ - Large list rendering bottlenecks -> [perf-virtualize-large-lists](references/perf-virtualize-large-lists.md)
137
+ - Static subtrees re-rendering unnecessarily -> [perf-v-once-v-memo-directives](references/perf-v-once-v-memo-directives.md)
138
+ - Over-abstraction in hot list paths -> [perf-avoid-component-abstraction-in-lists](references/perf-avoid-component-abstraction-in-lists.md)
139
+ - Expensive updates triggered too often -> [updated-hook-performance](references/updated-hook-performance.md)
140
+
141
+ ## 5) Final self-check before finishing
142
+
143
+ - Core behavior works and matches requirements.
144
+ - All must-read references were read and applied.
145
+ - Reactivity model is minimal and predictable.
146
+ - SFC structure and template rules are followed.
147
+ - Components are focused and well-factored, splitting when needed.
148
+ - Entry/root and route view components remain composition surfaces unless there is an explicit small-demo exception.
149
+ - Component split decisions are explicit and defensible (responsibility boundaries are clear).
150
+ - Data flow contracts are explicit and typed.
151
+ - Composables are used where reuse/complexity justifies them.
152
+ - Moved state/side effects into composables if applicable
153
+ - Optional features are used only when requirements demand them.
154
+ - Performance changes were applied only after functionality was complete.