ndomo 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 (247) hide show
  1. package/.bun-version +1 -0
  2. package/.dockerignore +79 -0
  3. package/.editorconfig +18 -0
  4. package/.env.example +19 -0
  5. package/.github/CODEOWNERS +8 -0
  6. package/.github/ISSUE_TEMPLATE/bug_report.yml +62 -0
  7. package/.github/ISSUE_TEMPLATE/config.yml +2 -0
  8. package/.github/ISSUE_TEMPLATE/feature_request.yml +34 -0
  9. package/.github/dependabot.yml +36 -0
  10. package/.github/pull_request_template.md +24 -0
  11. package/.github/release.yml +30 -0
  12. package/.github/workflows/gitleaks.yml +28 -0
  13. package/.github/workflows/release-please.yml +27 -0
  14. package/.github/workflows/smoke.yml +29 -0
  15. package/.husky/commit-msg +1 -0
  16. package/CHANGELOG.md +114 -0
  17. package/Dockerfile +32 -0
  18. package/README.es.md +174 -0
  19. package/README.md +187 -0
  20. package/agents/chronicler.md +98 -0
  21. package/agents/ci-smith.md +136 -0
  22. package/agents/craftsman.md +341 -0
  23. package/agents/deploy-smith.md +138 -0
  24. package/agents/foreman.md +377 -0
  25. package/agents/go-smith.md +164 -0
  26. package/agents/guild.md +188 -0
  27. package/agents/inspector.md +83 -0
  28. package/agents/js-smith.md +127 -0
  29. package/agents/ops-scout.md +173 -0
  30. package/agents/painter.md +200 -0
  31. package/agents/python-smith.md +120 -0
  32. package/agents/ranger.md +307 -0
  33. package/agents/release-smith.md +165 -0
  34. package/agents/rust-smith.md +159 -0
  35. package/agents/sage.md +178 -0
  36. package/agents/scout.md +144 -0
  37. package/agents/scribe.md +156 -0
  38. package/agents/smith.md +201 -0
  39. package/agents/vue-smith.md +155 -0
  40. package/agents/warden.md +216 -0
  41. package/agents/zig-smith.md +156 -0
  42. package/bin/ndomo-analyses.ts +4 -0
  43. package/bin/ndomo-status.ts +4 -0
  44. package/biome.json +57 -0
  45. package/bun.lock +514 -0
  46. package/commitlint.config.js +3 -0
  47. package/config/ndomo.config.json +258 -0
  48. package/config/ndomo.schema.json +166 -0
  49. package/docs/agents.md +375 -0
  50. package/docs/bugs/plan-create-orphan-fk.md +131 -0
  51. package/docs/bugs/task_create_batch-order-index-collision.md +158 -0
  52. package/docs/configuration.md +276 -0
  53. package/docs/database.md +364 -0
  54. package/docs/features/feature-flexible-builder-v1.md +724 -0
  55. package/docs/features/feature-flexible-builder-v2.md +882 -0
  56. package/docs/features/feature-flexible-builder.md +974 -0
  57. package/docs/http-server.md +244 -0
  58. package/docs/installation.md +259 -0
  59. package/docs/integrations.md +129 -0
  60. package/docs/operations/anti-pattern-sub-agent-verify-2026-06-21.md +32 -0
  61. package/docs/operations/audit-v1.md +417 -0
  62. package/docs/operations/audit-v2.md +197 -0
  63. package/docs/operations/audit-v3.md +306 -0
  64. package/docs/operations/db-optimize-foundations.md +123 -0
  65. package/docs/operations/verify-gate-architecture.md +82 -0
  66. package/docs/workflows.md +448 -0
  67. package/opencode.json +5 -0
  68. package/package.json +65 -0
  69. package/release-please-config.json +11 -0
  70. package/scripts/dev-bust-cache.sh +164 -0
  71. package/scripts/install.sh +688 -0
  72. package/scripts/smoke-e2e.ts +704 -0
  73. package/scripts/smoke-hot.ts +417 -0
  74. package/scripts/smoke-http.sh +228 -0
  75. package/scripts/smoke-v4.ts +256 -0
  76. package/scripts/smoke-v5.ts +397 -0
  77. package/scripts/smoke.sh +9 -0
  78. package/scripts/uninstall.sh +224 -0
  79. package/skills/api-security-best-practices/SKILL.md +915 -0
  80. package/skills/bash-scripting/SKILL.md +201 -0
  81. package/skills/bun/SKILL.md +313 -0
  82. package/skills/cavecrew/SKILL.md +82 -0
  83. package/skills/caveman/SKILL.md +74 -0
  84. package/skills/caveman-review/README.md +33 -0
  85. package/skills/caveman-review/SKILL.md +55 -0
  86. package/skills/find-skills/SKILL.md +142 -0
  87. package/skills/frontend-design/LICENSE.txt +177 -0
  88. package/skills/frontend-design/SKILL.md +55 -0
  89. package/skills/golang-patterns/SKILL.md +674 -0
  90. package/skills/golang-security/SKILL.md +185 -0
  91. package/skills/golang-security/evals/evals.json +595 -0
  92. package/skills/golang-security/references/architecture.md +268 -0
  93. package/skills/golang-security/references/checklist.md +80 -0
  94. package/skills/golang-security/references/cookies.md +200 -0
  95. package/skills/golang-security/references/cryptography.md +424 -0
  96. package/skills/golang-security/references/filesystem.md +285 -0
  97. package/skills/golang-security/references/injection.md +315 -0
  98. package/skills/golang-security/references/logging.md +163 -0
  99. package/skills/golang-security/references/memory-safety.md +241 -0
  100. package/skills/golang-security/references/network.md +253 -0
  101. package/skills/golang-security/references/secrets.md +189 -0
  102. package/skills/golang-security/references/third-party.md +159 -0
  103. package/skills/golang-security/references/threat-modeling.md +189 -0
  104. package/skills/golang-testing/SKILL.md +720 -0
  105. package/skills/grill-me/SKILL.md +7 -0
  106. package/skills/javascript-testing-patterns/SKILL.md +537 -0
  107. package/skills/javascript-testing-patterns/references/advanced-testing-patterns.md +513 -0
  108. package/skills/modern-javascript-patterns/SKILL.md +43 -0
  109. package/skills/modern-javascript-patterns/references/advanced-patterns.md +487 -0
  110. package/skills/modern-javascript-patterns/references/details.md +457 -0
  111. package/skills/python-anti-patterns/SKILL.md +349 -0
  112. package/skills/python-design-patterns/SKILL.md +85 -0
  113. package/skills/python-design-patterns/references/details.md +353 -0
  114. package/skills/python-error-handling/SKILL.md +193 -0
  115. package/skills/python-error-handling/references/details.md +171 -0
  116. package/skills/python-testing-patterns/SKILL.md +278 -0
  117. package/skills/python-testing-patterns/references/advanced-patterns.md +411 -0
  118. package/skills/python-testing-patterns/references/details.md +349 -0
  119. package/skills/rust-patterns/SKILL.md +500 -0
  120. package/skills/rust-testing/SKILL.md +501 -0
  121. package/skills/security-review/SKILL.md +504 -0
  122. package/skills/security-review/cloud-infrastructure-security.md +361 -0
  123. package/skills/vue-best-practices/SKILL.md +154 -0
  124. package/skills/vue-best-practices/references/animation-class-based-technique.md +254 -0
  125. package/skills/vue-best-practices/references/animation-state-driven-technique.md +291 -0
  126. package/skills/vue-best-practices/references/component-async.md +97 -0
  127. package/skills/vue-best-practices/references/component-data-flow.md +307 -0
  128. package/skills/vue-best-practices/references/component-fallthrough-attrs.md +174 -0
  129. package/skills/vue-best-practices/references/component-keep-alive.md +137 -0
  130. package/skills/vue-best-practices/references/component-slots.md +216 -0
  131. package/skills/vue-best-practices/references/component-suspense.md +228 -0
  132. package/skills/vue-best-practices/references/component-teleport.md +108 -0
  133. package/skills/vue-best-practices/references/component-transition-group.md +128 -0
  134. package/skills/vue-best-practices/references/component-transition.md +125 -0
  135. package/skills/vue-best-practices/references/composables.md +290 -0
  136. package/skills/vue-best-practices/references/directives.md +162 -0
  137. package/skills/vue-best-practices/references/perf-avoid-component-abstraction-in-lists.md +159 -0
  138. package/skills/vue-best-practices/references/perf-v-once-v-memo-directives.md +182 -0
  139. package/skills/vue-best-practices/references/perf-virtualize-large-lists.md +187 -0
  140. package/skills/vue-best-practices/references/plugins.md +166 -0
  141. package/skills/vue-best-practices/references/reactivity.md +344 -0
  142. package/skills/vue-best-practices/references/render-functions.md +201 -0
  143. package/skills/vue-best-practices/references/sfc.md +310 -0
  144. package/skills/vue-best-practices/references/state-management.md +135 -0
  145. package/skills/vue-best-practices/references/updated-hook-performance.md +187 -0
  146. package/skills/vue-pinia-best-practices/SKILL.md +21 -0
  147. package/skills/vue-pinia-best-practices/reference/pinia-no-active-pinia-error.md +248 -0
  148. package/skills/vue-pinia-best-practices/reference/pinia-setup-store-return-all-state.md +227 -0
  149. package/skills/vue-pinia-best-practices/reference/pinia-store-destructuring-breaks-reactivity.md +193 -0
  150. package/skills/vue-pinia-best-practices/reference/state-url-for-ephemeral-filters.md +238 -0
  151. package/skills/vue-pinia-best-practices/reference/state-use-pinia-for-large-apps.md +262 -0
  152. package/skills/vue-pinia-best-practices/reference/store-method-binding-parentheses.md +191 -0
  153. package/skills/zig-0.16/SKILL.md +840 -0
  154. package/skills/zig-0.16/scripts/check-zig-version.sh +21 -0
  155. package/src/cli/analyses.ts +280 -0
  156. package/src/cli/index.ts +108 -0
  157. package/src/cli/serve.ts +192 -0
  158. package/src/cli/smoke.ts +131 -0
  159. package/src/cli/status.test.ts +204 -0
  160. package/src/cli/status.ts +263 -0
  161. package/src/cli/vacuum.test.ts +82 -0
  162. package/src/cli/vacuum.ts +96 -0
  163. package/src/config/schema.test.ts +88 -0
  164. package/src/config/schema.ts +64 -0
  165. package/src/db/analyses-migration.test.ts +210 -0
  166. package/src/db/analyses.test.ts +466 -0
  167. package/src/db/analyses.ts +375 -0
  168. package/src/db/auto-checkpoint.ts +131 -0
  169. package/src/db/client.test.ts +129 -0
  170. package/src/db/client.ts +55 -0
  171. package/src/db/fts-escape.ts +20 -0
  172. package/src/db/incidents.test.ts +201 -0
  173. package/src/db/incidents.ts +93 -0
  174. package/src/db/index.ts +86 -0
  175. package/src/db/migrations-v13.test.ts +141 -0
  176. package/src/db/migrations-v8.test.ts +301 -0
  177. package/src/db/migrations.ts +147 -0
  178. package/src/db/plan-archive.test.ts +180 -0
  179. package/src/db/plan-archive.ts +274 -0
  180. package/src/db/plan-create.test.ts +276 -0
  181. package/src/db/plan-create.ts +78 -0
  182. package/src/db/plan-files.test.ts +289 -0
  183. package/src/db/plan-update-status.ts +287 -0
  184. package/src/db/plans.test.ts +490 -0
  185. package/src/db/plans.ts +534 -0
  186. package/src/db/resolve-project-dir.test.ts +143 -0
  187. package/src/db/resolve-project-dir.ts +75 -0
  188. package/src/db/rollbacks.test.ts +150 -0
  189. package/src/db/rollbacks.ts +67 -0
  190. package/src/db/schema.ts +907 -0
  191. package/src/db/sessions.test.ts +80 -0
  192. package/src/db/sessions.ts +135 -0
  193. package/src/db/shutdown.test.ts +147 -0
  194. package/src/db/shutdown.ts +45 -0
  195. package/src/db/tasks.test.ts +921 -0
  196. package/src/db/tasks.ts +747 -0
  197. package/src/db/types.ts +619 -0
  198. package/src/http/__tests__/auth.test.ts +196 -0
  199. package/src/http/__tests__/routes.test.ts +465 -0
  200. package/src/http/__tests__/sse.test.ts +317 -0
  201. package/src/http/auth.ts +72 -0
  202. package/src/http/middleware/cors.ts +53 -0
  203. package/src/http/middleware/security-headers.ts +21 -0
  204. package/src/http/routes/events.ts +112 -0
  205. package/src/http/routes/health.ts +51 -0
  206. package/src/http/routes/plans.ts +66 -0
  207. package/src/http/routes/sessions.ts +50 -0
  208. package/src/http/routes/tasks.ts +60 -0
  209. package/src/http/server.ts +95 -0
  210. package/src/http/sse.ts +116 -0
  211. package/src/index.ts +37 -0
  212. package/src/lib.ts +65 -0
  213. package/src/mem/scoped.ts +65 -0
  214. package/src/orchestrator/background.test.ts +268 -0
  215. package/src/orchestrator/background.ts +293 -0
  216. package/src/orchestrator/memory-hook.ts +182 -0
  217. package/src/orchestrator/reconciler.ts +123 -0
  218. package/src/orchestrator/scheduler.test.ts +300 -0
  219. package/src/orchestrator/scheduler.ts +243 -0
  220. package/src/plugin.test.ts +2574 -0
  221. package/src/plugin.ts +1690 -0
  222. package/src/sdk/client.ts +66 -0
  223. package/src/worktrees/manager.ts +236 -0
  224. package/src/worktrees/state.ts +87 -0
  225. package/tests/integration/ranger-flow.test.ts +257 -0
  226. package/tools/analysis_archive.ts +28 -0
  227. package/tools/analysis_create.ts +55 -0
  228. package/tools/analysis_get.ts +33 -0
  229. package/tools/analysis_link_plan.ts +44 -0
  230. package/tools/analysis_list.ts +48 -0
  231. package/tools/analysis_search.ts +36 -0
  232. package/tools/analysis_update.ts +44 -0
  233. package/tools/plan_approve.ts +31 -0
  234. package/tools/plan_create.ts +58 -0
  235. package/tools/plan_get.ts +40 -0
  236. package/tools/plan_list.ts +37 -0
  237. package/tools/plan_search.ts +34 -0
  238. package/tools/plan_update_status.ts +71 -0
  239. package/tools/session_checkpoint.ts +31 -0
  240. package/tools/session_end.ts +26 -0
  241. package/tools/session_start.ts +43 -0
  242. package/tools/task_create_batch.ts +70 -0
  243. package/tools/task_list.ts +35 -0
  244. package/tools/task_next_for_agent.ts +30 -0
  245. package/tools/task_search.ts +34 -0
  246. package/tools/task_update_status.ts +37 -0
  247. package/tsconfig.json +31 -0
@@ -0,0 +1,254 @@
1
+ ---
2
+ title: Use Class-based Animations for Non-Enter/Leave Effects
3
+ impact: LOW
4
+ impactDescription: Class-based animations are simpler and more performant for elements that remain in the DOM
5
+ type: best-practice
6
+ tags: [vue3, animation, css, class-binding, state]
7
+ ---
8
+
9
+ # Use Class-based Animations for Non-Enter/Leave Effects
10
+
11
+ **Impact: LOW** - For animations on elements that are not entering or leaving the DOM, use CSS class-based animations triggered by Vue's reactive state. This is simpler than `<Transition>` and more appropriate for feedback animations like shake, pulse, or highlight effects.
12
+
13
+ ## Task List
14
+
15
+ - Use class-based animations for elements staying in the DOM
16
+ - Use `<Transition>` only for enter/leave animations
17
+ - Combine CSS animations with Vue's class bindings (`:class`)
18
+ - Consider using `setTimeout` to auto-remove animation classes
19
+
20
+ **When to Use Class-based Animations:**
21
+ - User feedback (shake on error, pulse on success)
22
+ - Attention-grabbing effects (highlight changes)
23
+ - Hover/focus states that need more than CSS transitions
24
+ - Any animation where the element stays mounted
25
+
26
+ **When to Use Transition Component:**
27
+ - Elements entering/leaving the DOM (v-if/v-show)
28
+ - Route transitions
29
+ - List item additions/removals
30
+
31
+ ## Basic Pattern
32
+
33
+ ```vue
34
+ <template>
35
+ <div :class="{ shake: showError }">
36
+ <button @click="submitForm">Submit</button>
37
+ <span v-if="showError">This feature is disabled!</span>
38
+ </div>
39
+ </template>
40
+
41
+ <script setup>
42
+ import { ref } from 'vue'
43
+
44
+ const showError = ref(false)
45
+
46
+ function submitForm() {
47
+ if (!isValid()) {
48
+ // Trigger shake animation
49
+ showError.value = true
50
+
51
+ // Auto-remove class after animation completes
52
+ setTimeout(() => {
53
+ showError.value = false
54
+ }, 820) // Match animation duration
55
+ }
56
+ }
57
+ </script>
58
+
59
+ <style>
60
+ .shake {
61
+ animation: shake 0.82s cubic-bezier(0.36, 0.07, 0.19, 0.97) both;
62
+ transform: translate3d(0, 0, 0); /* Enable GPU acceleration */
63
+ }
64
+
65
+ @keyframes shake {
66
+ 10%, 90% { transform: translate3d(-1px, 0, 0); }
67
+ 20%, 80% { transform: translate3d(2px, 0, 0); }
68
+ 30%, 50%, 70% { transform: translate3d(-4px, 0, 0); }
69
+ 40%, 60% { transform: translate3d(4px, 0, 0); }
70
+ }
71
+ </style>
72
+ ```
73
+
74
+ ## Common Animation Patterns
75
+
76
+ ### Pulse on Success
77
+
78
+ ```vue
79
+ <template>
80
+ <button
81
+ @click="save"
82
+ :class="{ pulse: saved }"
83
+ >
84
+ {{ saved ? 'Saved!' : 'Save' }}
85
+ </button>
86
+ </template>
87
+
88
+ <script setup>
89
+ import { ref } from 'vue'
90
+
91
+ const saved = ref(false)
92
+
93
+ async function save() {
94
+ await saveData()
95
+ saved.value = true
96
+ setTimeout(() => saved.value = false, 1000)
97
+ }
98
+ </script>
99
+
100
+ <style>
101
+ .pulse {
102
+ animation: pulse 0.5s ease-in-out;
103
+ }
104
+
105
+ @keyframes pulse {
106
+ 0%, 100% { transform: scale(1); }
107
+ 50% { transform: scale(1.05); }
108
+ }
109
+ </style>
110
+ ```
111
+
112
+ ### Highlight on Change
113
+
114
+ ```vue
115
+ <template>
116
+ <div
117
+ :class="{ highlight: justUpdated }"
118
+ >
119
+ Value: {{ value }}
120
+ </div>
121
+ </template>
122
+
123
+ <script setup>
124
+ import { ref, watch } from 'vue'
125
+
126
+ const value = ref(0)
127
+ const justUpdated = ref(false)
128
+
129
+ watch(value, () => {
130
+ justUpdated.value = true
131
+ setTimeout(() => justUpdated.value = false, 1000)
132
+ })
133
+ </script>
134
+
135
+ <style>
136
+ .highlight {
137
+ animation: highlight 1s ease-out;
138
+ }
139
+
140
+ @keyframes highlight {
141
+ 0% { background-color: yellow; }
142
+ 100% { background-color: transparent; }
143
+ }
144
+ </style>
145
+ ```
146
+
147
+ ### Bounce Attention
148
+
149
+ ```vue
150
+ <template>
151
+ <div
152
+ :class="{ bounce: needsAttention }"
153
+ @animationend="needsAttention = false"
154
+ >
155
+ <BellIcon />
156
+ </div>
157
+ </template>
158
+
159
+ <script setup>
160
+ import { ref } from 'vue'
161
+
162
+ const needsAttention = ref(false)
163
+
164
+ function notifyUser() {
165
+ needsAttention.value = true
166
+ // No setTimeout needed - using animationend event
167
+ }
168
+ </script>
169
+
170
+ <style>
171
+ .bounce {
172
+ animation: bounce 0.5s ease;
173
+ }
174
+
175
+ @keyframes bounce {
176
+ 0%, 100% { transform: translateY(0); }
177
+ 50% { transform: translateY(-10px); }
178
+ }
179
+ </style>
180
+ ```
181
+
182
+ ## Using animationend Event
183
+
184
+ Instead of `setTimeout`, use the `animationend` event for cleaner code:
185
+
186
+ ```vue
187
+ <template>
188
+ <div
189
+ :class="{ animate: isAnimating }"
190
+ @animationend="isAnimating = false"
191
+ >
192
+ Content
193
+ </div>
194
+ </template>
195
+
196
+ <script setup>
197
+ import { ref } from 'vue'
198
+
199
+ const isAnimating = ref(false)
200
+
201
+ function triggerAnimation() {
202
+ isAnimating.value = true
203
+ // Class is automatically removed when animation ends
204
+ }
205
+ </script>
206
+ ```
207
+
208
+ ## Composable for Reusable Animations
209
+
210
+ ```javascript
211
+ // composables/useAnimation.js
212
+ import { ref } from 'vue'
213
+
214
+ export function useAnimation(duration = 500) {
215
+ const isAnimating = ref(false)
216
+
217
+ function trigger() {
218
+ isAnimating.value = true
219
+ setTimeout(() => {
220
+ isAnimating.value = false
221
+ }, duration)
222
+ }
223
+
224
+ return {
225
+ isAnimating,
226
+ trigger
227
+ }
228
+ }
229
+ ```
230
+
231
+ ```vue
232
+ <script setup>
233
+ import { useAnimation } from '@/composables/useAnimation'
234
+
235
+ const shake = useAnimation(820)
236
+ const pulse = useAnimation(500)
237
+ </script>
238
+
239
+ <template>
240
+ <button
241
+ :class="{ shake: shake.isAnimating.value }"
242
+ @click="shake.trigger()"
243
+ >
244
+ Shake me
245
+ </button>
246
+
247
+ <button
248
+ :class="{ pulse: pulse.isAnimating.value }"
249
+ @click="pulse.trigger()"
250
+ >
251
+ Pulse me
252
+ </button>
253
+ </template>
254
+ ```
@@ -0,0 +1,291 @@
1
+ ---
2
+ title: State-driven Animations with CSS Transitions and Style Bindings
3
+ impact: LOW
4
+ impactDescription: Combining Vue's reactive style bindings with CSS transitions creates smooth, interactive animations
5
+ type: best-practice
6
+ tags: [vue3, animation, css, transition, style-binding, state, interactive]
7
+ ---
8
+
9
+ # State-driven Animations with CSS Transitions and Style Bindings
10
+
11
+ **Impact: LOW** - For responsive, interactive animations that react to user input or state changes, combine Vue's dynamic style bindings with CSS transitions. This creates smooth animations that interpolate values in real-time based on state.
12
+
13
+ ## Task List
14
+
15
+ - Use `:style` binding for dynamic properties that change frequently
16
+ - Add CSS `transition` property to smoothly animate between values
17
+ - Consider using `transform` and `opacity` for GPU-accelerated animations
18
+ - For complex value interpolation, use watchers with animation libraries
19
+
20
+ ## Basic Pattern
21
+
22
+ ```vue
23
+ <template>
24
+ <div
25
+ @mousemove="onMousemove"
26
+ :style="{ backgroundColor: `hsl(${hue}, 80%, 50%)` }"
27
+ class="interactive-area"
28
+ >
29
+ <p>Move your mouse across this div...</p>
30
+ <p>Hue: {{ hue }}</p>
31
+ </div>
32
+ </template>
33
+
34
+ <script setup>
35
+ import { ref } from 'vue'
36
+
37
+ const hue = ref(0)
38
+
39
+ function onMousemove(e) {
40
+ // Map mouse X position to hue (0-360)
41
+ const rect = e.currentTarget.getBoundingClientRect()
42
+ hue.value = Math.round((e.clientX - rect.left) / rect.width * 360)
43
+ }
44
+ </script>
45
+
46
+ <style>
47
+ .interactive-area {
48
+ transition: background-color 0.3s ease;
49
+ height: 200px;
50
+ display: flex;
51
+ flex-direction: column;
52
+ align-items: center;
53
+ justify-content: center;
54
+ }
55
+ </style>
56
+ ```
57
+
58
+ ## Common Use Cases
59
+
60
+ ### Following Mouse Position
61
+
62
+ ```vue
63
+ <template>
64
+ <div
65
+ class="container"
66
+ @mousemove="onMousemove"
67
+ >
68
+ <div
69
+ class="follower"
70
+ :style="{
71
+ transform: `translate(${x}px, ${y}px)`
72
+ }"
73
+ />
74
+ </div>
75
+ </template>
76
+
77
+ <script setup>
78
+ import { ref } from 'vue'
79
+
80
+ const x = ref(0)
81
+ const y = ref(0)
82
+
83
+ function onMousemove(e) {
84
+ const rect = e.currentTarget.getBoundingClientRect()
85
+ x.value = e.clientX - rect.left
86
+ y.value = e.clientY - rect.top
87
+ }
88
+ </script>
89
+
90
+ <style>
91
+ .container {
92
+ position: relative;
93
+ height: 300px;
94
+ }
95
+
96
+ .follower {
97
+ position: absolute;
98
+ width: 20px;
99
+ height: 20px;
100
+ background: blue;
101
+ border-radius: 50%;
102
+ /* Smooth following with transition */
103
+ transition: transform 0.1s ease-out;
104
+ /* Prevent the follower from triggering mousemove */
105
+ pointer-events: none;
106
+ }
107
+ </style>
108
+ ```
109
+
110
+ ### Progress Animation
111
+
112
+ ```vue
113
+ <template>
114
+ <div class="progress-container">
115
+ <div
116
+ class="progress-bar"
117
+ :style="{ width: `${progress}%` }"
118
+ />
119
+ </div>
120
+ <input
121
+ type="range"
122
+ v-model.number="progress"
123
+ min="0"
124
+ max="100"
125
+ />
126
+ </template>
127
+
128
+ <script setup>
129
+ import { ref } from 'vue'
130
+
131
+ const progress = ref(0)
132
+ </script>
133
+
134
+ <style>
135
+ .progress-container {
136
+ height: 20px;
137
+ background: #e0e0e0;
138
+ border-radius: 10px;
139
+ overflow: hidden;
140
+ }
141
+
142
+ .progress-bar {
143
+ height: 100%;
144
+ background: linear-gradient(90deg, #4CAF50, #8BC34A);
145
+ transition: width 0.3s ease;
146
+ }
147
+ </style>
148
+ ```
149
+
150
+ ### Scroll-based Animation
151
+
152
+ ```vue
153
+ <template>
154
+ <div
155
+ class="hero"
156
+ :style="{
157
+ opacity: heroOpacity,
158
+ transform: `translateY(${scrollOffset}px)`
159
+ }"
160
+ >
161
+ <h1>Scroll Down</h1>
162
+ </div>
163
+ </template>
164
+
165
+ <script setup>
166
+ import { ref, computed, onMounted, onUnmounted } from 'vue'
167
+
168
+ const scrollY = ref(0)
169
+
170
+ const heroOpacity = computed(() => {
171
+ return Math.max(0, 1 - scrollY.value / 300)
172
+ })
173
+
174
+ const scrollOffset = computed(() => {
175
+ return scrollY.value * 0.5 // Parallax effect
176
+ })
177
+
178
+ function handleScroll() {
179
+ scrollY.value = window.scrollY
180
+ }
181
+
182
+ onMounted(() => {
183
+ window.addEventListener('scroll', handleScroll, { passive: true })
184
+ })
185
+
186
+ onUnmounted(() => {
187
+ window.removeEventListener('scroll', handleScroll)
188
+ })
189
+ </script>
190
+
191
+ <style>
192
+ .hero {
193
+ height: 100vh;
194
+ display: flex;
195
+ align-items: center;
196
+ justify-content: center;
197
+ /* Note: No transition for scroll-based animations - they should be instant */
198
+ }
199
+ </style>
200
+ ```
201
+
202
+ ### Color Theme Transition
203
+
204
+ ```vue
205
+ <template>
206
+ <div
207
+ class="app"
208
+ :style="themeStyles"
209
+ >
210
+ <button @click="toggleTheme">Toggle Theme</button>
211
+ <p>Current theme: {{ isDark ? 'Dark' : 'Light' }}</p>
212
+ </div>
213
+ </template>
214
+
215
+ <script setup>
216
+ import { ref, computed } from 'vue'
217
+
218
+ const isDark = ref(false)
219
+
220
+ const themeStyles = computed(() => ({
221
+ '--bg-color': isDark.value ? '#1a1a1a' : '#ffffff',
222
+ '--text-color': isDark.value ? '#ffffff' : '#1a1a1a',
223
+ backgroundColor: 'var(--bg-color)',
224
+ color: 'var(--text-color)'
225
+ }))
226
+
227
+ function toggleTheme() {
228
+ isDark.value = !isDark.value
229
+ }
230
+ </script>
231
+
232
+ <style>
233
+ .app {
234
+ min-height: 100vh;
235
+ transition: background-color 0.5s ease, color 0.5s ease;
236
+ }
237
+ </style>
238
+ ```
239
+
240
+ ## Advanced: Numerical Tweening with Watchers
241
+
242
+ For smooth number animations (counters, stats), use watchers with animation libraries:
243
+
244
+ ```vue
245
+ <template>
246
+ <div>
247
+ <input v-model.number="targetNumber" type="number" />
248
+ <p class="counter">{{ displayNumber.toFixed(0) }}</p>
249
+ </div>
250
+ </template>
251
+
252
+ <script setup>
253
+ import { computed, ref, reactive, watch } from 'vue'
254
+ import gsap from 'gsap'
255
+
256
+ const targetNumber = ref(0)
257
+ const tweened = reactive({ value: 0 })
258
+
259
+ // Computed for display
260
+ const displayNumber = computed(() => tweened.value)
261
+
262
+ watch(targetNumber, (newValue) => {
263
+ gsap.to(tweened, {
264
+ duration: 0.5,
265
+ value: Number(newValue) || 0,
266
+ ease: 'power2.out'
267
+ })
268
+ })
269
+ </script>
270
+ ```
271
+
272
+ ## Performance Considerations
273
+
274
+ ```vue
275
+ <style>
276
+ /* GOOD: GPU-accelerated properties */
277
+ .element {
278
+ transition: transform 0.3s ease, opacity 0.3s ease;
279
+ }
280
+
281
+ /* AVOID: Properties that trigger layout recalculation */
282
+ .element {
283
+ transition: width 0.3s ease, height 0.3s ease, margin 0.3s ease;
284
+ }
285
+
286
+ /* For high-frequency updates, consider will-change */
287
+ .frequently-animated {
288
+ will-change: transform;
289
+ }
290
+ </style>
291
+ ```
@@ -0,0 +1,97 @@
1
+ ---
2
+ title: Async Component Best Practices
3
+ impact: MEDIUM
4
+ impactDescription: Poor async component strategy can delay interactivity in SSR apps and create loading UI flicker
5
+ type: best-practice
6
+ tags: [vue3, async-components, ssr, hydration, performance, ux]
7
+ ---
8
+
9
+ # Async Component Best Practices
10
+
11
+ **Impact: MEDIUM** - Async components should reduce JavaScript cost without degrading perceived performance. Focus on hydration timing in SSR and stable loading UX.
12
+
13
+ ## Task List
14
+
15
+ - Use lazy hydration strategies for non-critical SSR component trees
16
+ - Import only the hydration helpers you actually use
17
+ - Keep `loadingComponent` delay near the default `200ms` unless real UX data suggests otherwise
18
+ - Configure `delay` and `timeout` together for predictable loading behavior
19
+
20
+ ## Use Lazy Hydration Strategies in SSR
21
+
22
+ In Vue 3.5+, async components can delay hydration until idle time, visibility, media query match, or user interaction.
23
+
24
+ **BAD:**
25
+ ```vue
26
+ <script setup lang="ts">
27
+ import { defineAsyncComponent } from 'vue'
28
+
29
+ const AsyncComments = defineAsyncComponent({
30
+ loader: () => import('./Comments.vue')
31
+ })
32
+ </script>
33
+ ```
34
+
35
+ **GOOD:**
36
+ ```vue
37
+ <script setup lang="ts">
38
+ import {
39
+ defineAsyncComponent,
40
+ hydrateOnVisible,
41
+ hydrateOnIdle
42
+ } from 'vue'
43
+
44
+ const AsyncComments = defineAsyncComponent({
45
+ loader: () => import('./Comments.vue'),
46
+ hydrate: hydrateOnVisible({ rootMargin: '100px' })
47
+ })
48
+
49
+ const AsyncFooter = defineAsyncComponent({
50
+ loader: () => import('./Footer.vue'),
51
+ hydrate: hydrateOnIdle(5000)
52
+ })
53
+ </script>
54
+ ```
55
+
56
+ ## Prevent Loading Spinner Flicker
57
+
58
+ Avoid showing loading UI immediately for components that usually resolve quickly.
59
+
60
+ **BAD:**
61
+ ```vue
62
+ <script setup lang="ts">
63
+ import { defineAsyncComponent } from 'vue'
64
+ import LoadingSpinner from './LoadingSpinner.vue'
65
+
66
+ const AsyncDashboard = defineAsyncComponent({
67
+ loader: () => import('./Dashboard.vue'),
68
+ loadingComponent: LoadingSpinner,
69
+ delay: 0
70
+ })
71
+ </script>
72
+ ```
73
+
74
+ **GOOD:**
75
+ ```vue
76
+ <script setup lang="ts">
77
+ import { defineAsyncComponent } from 'vue'
78
+ import LoadingSpinner from './LoadingSpinner.vue'
79
+ import ErrorDisplay from './ErrorDisplay.vue'
80
+
81
+ const AsyncDashboard = defineAsyncComponent({
82
+ loader: () => import('./Dashboard.vue'),
83
+ loadingComponent: LoadingSpinner,
84
+ errorComponent: ErrorDisplay,
85
+ delay: 200,
86
+ timeout: 30000
87
+ })
88
+ </script>
89
+ ```
90
+
91
+ ## Delay Guidelines
92
+
93
+ | Scenario | Recommended Delay |
94
+ |----------|-------------------|
95
+ | Small component, fast network | `200ms` |
96
+ | Known heavy component | `100ms` |
97
+ | Background or non-critical UI | `300-500ms` |