qualia-framework 2.6.0 → 3.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 (328) hide show
  1. package/CLAUDE.md +63 -0
  2. package/README.md +108 -30
  3. package/agents/builder.md +110 -0
  4. package/agents/planner.md +186 -0
  5. package/agents/qa-browser.md +186 -0
  6. package/agents/verifier.md +369 -0
  7. package/bin/cli.js +691 -492
  8. package/bin/install.js +622 -0
  9. package/bin/qualia-ui.js +284 -0
  10. package/bin/state.js +824 -0
  11. package/bin/statusline.js +252 -0
  12. package/docs/erp-contract.md +161 -0
  13. package/guide.md +63 -0
  14. package/hooks/auto-update.js +117 -0
  15. package/hooks/block-env-edit.js +52 -0
  16. package/hooks/branch-guard.js +68 -0
  17. package/hooks/migration-guard.js +83 -0
  18. package/hooks/pre-compact.js +52 -0
  19. package/hooks/pre-deploy-gate.js +149 -0
  20. package/hooks/pre-push.js +53 -0
  21. package/hooks/session-start.js +126 -0
  22. package/package.json +30 -20
  23. package/rules/design-reference.md +179 -0
  24. package/rules/frontend.md +126 -0
  25. package/rules/infrastructure.md +87 -0
  26. package/skills/qualia/SKILL.md +88 -0
  27. package/skills/qualia-build/SKILL.md +115 -0
  28. package/skills/qualia-debug/SKILL.md +87 -0
  29. package/skills/qualia-design/SKILL.md +99 -0
  30. package/skills/qualia-handoff/SKILL.md +66 -0
  31. package/skills/qualia-help/SKILL.md +60 -0
  32. package/skills/qualia-idk/SKILL.md +8 -0
  33. package/skills/qualia-learn/SKILL.md +111 -0
  34. package/skills/qualia-new/SKILL.md +323 -0
  35. package/skills/qualia-pause/SKILL.md +63 -0
  36. package/skills/qualia-plan/SKILL.md +101 -0
  37. package/skills/qualia-polish/SKILL.md +207 -0
  38. package/skills/qualia-quick/SKILL.md +37 -0
  39. package/skills/qualia-report/SKILL.md +114 -0
  40. package/skills/qualia-resume/SKILL.md +49 -0
  41. package/skills/qualia-review/SKILL.md +161 -0
  42. package/skills/qualia-ship/SKILL.md +90 -0
  43. package/skills/qualia-skill-new/SKILL.md +167 -0
  44. package/skills/qualia-task/SKILL.md +91 -0
  45. package/skills/qualia-test/SKILL.md +134 -0
  46. package/skills/qualia-verify/SKILL.md +113 -0
  47. package/templates/DESIGN.md +475 -0
  48. package/templates/help.html +476 -0
  49. package/templates/plan.md +42 -0
  50. package/templates/project.md +22 -0
  51. package/templates/state.md +27 -0
  52. package/templates/tracking.json +20 -0
  53. package/tests/bin.test.sh +687 -0
  54. package/tests/hooks.test.sh +384 -0
  55. package/tests/runner.js +1956 -0
  56. package/tests/state.test.sh +713 -0
  57. package/tests/statusline.test.sh +243 -0
  58. package/bin/collect-metrics.sh +0 -62
  59. package/framework/.claudeignore +0 -51
  60. package/framework/CLAUDE.md +0 -51
  61. package/framework/MCP_SETUP.md +0 -229
  62. package/framework/agents/architecture-strategist.md +0 -53
  63. package/framework/agents/backend-agent.md +0 -150
  64. package/framework/agents/code-simplicity-reviewer.md +0 -86
  65. package/framework/agents/frontend-agent.md +0 -111
  66. package/framework/agents/kieran-typescript-reviewer.md +0 -96
  67. package/framework/agents/performance-oracle.md +0 -111
  68. package/framework/agents/qualia-codebase-mapper.md +0 -761
  69. package/framework/agents/qualia-debugger.md +0 -1204
  70. package/framework/agents/qualia-executor.md +0 -882
  71. package/framework/agents/qualia-integration-checker.md +0 -424
  72. package/framework/agents/qualia-phase-researcher.md +0 -457
  73. package/framework/agents/qualia-plan-checker.md +0 -700
  74. package/framework/agents/qualia-planner.md +0 -1245
  75. package/framework/agents/qualia-project-researcher.md +0 -603
  76. package/framework/agents/qualia-research-synthesizer.md +0 -200
  77. package/framework/agents/qualia-roadmapper.md +0 -606
  78. package/framework/agents/qualia-verifier.md +0 -686
  79. package/framework/agents/red-team-qa.md +0 -130
  80. package/framework/agents/security-auditor.md +0 -72
  81. package/framework/agents/team-orchestrator.md +0 -229
  82. package/framework/agents/teams/framework-audit-team.md +0 -66
  83. package/framework/agents/teams/full-stack-team.md +0 -48
  84. package/framework/agents/teams/optimize-team.md +0 -53
  85. package/framework/agents/teams/review-team.md +0 -70
  86. package/framework/agents/teams/ship-team.md +0 -86
  87. package/framework/agents/test-agent.md +0 -182
  88. package/framework/hooks/auto-format.sh +0 -54
  89. package/framework/hooks/block-env-edit.sh +0 -42
  90. package/framework/hooks/branch-guard.sh +0 -43
  91. package/framework/hooks/confirm-delete.sh +0 -59
  92. package/framework/hooks/migration-validate.sh +0 -77
  93. package/framework/hooks/notification-speak.sh +0 -16
  94. package/framework/hooks/pre-commit.sh +0 -100
  95. package/framework/hooks/pre-compact.sh +0 -56
  96. package/framework/hooks/pre-deploy-gate.sh +0 -160
  97. package/framework/hooks/qualia-colors.sh +0 -32
  98. package/framework/hooks/retention-cleanup.sh +0 -62
  99. package/framework/hooks/save-session-state.sh +0 -185
  100. package/framework/hooks/session-context-loader.sh +0 -96
  101. package/framework/hooks/session-learn.sh +0 -32
  102. package/framework/hooks/skill-announce.sh +0 -123
  103. package/framework/hooks/tool-error-announce.sh +0 -27
  104. package/framework/install.ps1 +0 -323
  105. package/framework/install.sh +0 -313
  106. package/framework/qualia-framework/VERSION +0 -1
  107. package/framework/qualia-framework/assets/qualia-logo.png +0 -0
  108. package/framework/qualia-framework/bin/collect-metrics.sh +0 -67
  109. package/framework/qualia-framework/bin/generate-report-docx.py +0 -429
  110. package/framework/qualia-framework/bin/qualia-tools.js +0 -2201
  111. package/framework/qualia-framework/bin/qualia-tools.test.js +0 -1054
  112. package/framework/qualia-framework/references/checkpoints.md +0 -775
  113. package/framework/qualia-framework/references/completion-checklists.md +0 -359
  114. package/framework/qualia-framework/references/continuation-format.md +0 -249
  115. package/framework/qualia-framework/references/continuation-prompt.md +0 -97
  116. package/framework/qualia-framework/references/decimal-phase-calculation.md +0 -65
  117. package/framework/qualia-framework/references/design-quality.md +0 -56
  118. package/framework/qualia-framework/references/employee-guide.md +0 -167
  119. package/framework/qualia-framework/references/git-integration.md +0 -254
  120. package/framework/qualia-framework/references/git-planning-commit.md +0 -50
  121. package/framework/qualia-framework/references/model-profile-resolution.md +0 -32
  122. package/framework/qualia-framework/references/model-profiles.md +0 -73
  123. package/framework/qualia-framework/references/phase-argument-parsing.md +0 -61
  124. package/framework/qualia-framework/references/planning-config.md +0 -195
  125. package/framework/qualia-framework/references/questioning.md +0 -141
  126. package/framework/qualia-framework/references/tdd.md +0 -263
  127. package/framework/qualia-framework/references/ui-brand.md +0 -160
  128. package/framework/qualia-framework/references/verification-patterns.md +0 -612
  129. package/framework/qualia-framework/templates/DEBUG.md +0 -159
  130. package/framework/qualia-framework/templates/DESIGN.md +0 -81
  131. package/framework/qualia-framework/templates/UAT.md +0 -247
  132. package/framework/qualia-framework/templates/codebase/architecture.md +0 -255
  133. package/framework/qualia-framework/templates/codebase/concerns.md +0 -310
  134. package/framework/qualia-framework/templates/codebase/conventions.md +0 -307
  135. package/framework/qualia-framework/templates/codebase/integrations.md +0 -280
  136. package/framework/qualia-framework/templates/codebase/stack.md +0 -186
  137. package/framework/qualia-framework/templates/codebase/structure.md +0 -285
  138. package/framework/qualia-framework/templates/codebase/testing.md +0 -480
  139. package/framework/qualia-framework/templates/config.json +0 -35
  140. package/framework/qualia-framework/templates/context.md +0 -283
  141. package/framework/qualia-framework/templates/continue-here.md +0 -78
  142. package/framework/qualia-framework/templates/debug-subagent-prompt.md +0 -91
  143. package/framework/qualia-framework/templates/discovery.md +0 -146
  144. package/framework/qualia-framework/templates/lab-notes.md +0 -16
  145. package/framework/qualia-framework/templates/milestone-archive.md +0 -123
  146. package/framework/qualia-framework/templates/milestone.md +0 -115
  147. package/framework/qualia-framework/templates/phase-prompt.md +0 -567
  148. package/framework/qualia-framework/templates/planner-subagent-prompt.md +0 -117
  149. package/framework/qualia-framework/templates/project.md +0 -184
  150. package/framework/qualia-framework/templates/projects/ai-agent.md +0 -156
  151. package/framework/qualia-framework/templates/projects/mobile-app.md +0 -181
  152. package/framework/qualia-framework/templates/projects/voice-agent.md +0 -134
  153. package/framework/qualia-framework/templates/projects/website.md +0 -137
  154. package/framework/qualia-framework/templates/requirements.md +0 -231
  155. package/framework/qualia-framework/templates/research-project/ARCHITECTURE.md +0 -204
  156. package/framework/qualia-framework/templates/research-project/FEATURES.md +0 -147
  157. package/framework/qualia-framework/templates/research-project/PITFALLS.md +0 -200
  158. package/framework/qualia-framework/templates/research-project/STACK.md +0 -120
  159. package/framework/qualia-framework/templates/research-project/SUMMARY.md +0 -170
  160. package/framework/qualia-framework/templates/research.md +0 -552
  161. package/framework/qualia-framework/templates/roadmap.md +0 -206
  162. package/framework/qualia-framework/templates/state.md +0 -179
  163. package/framework/qualia-framework/templates/summary-complex.md +0 -59
  164. package/framework/qualia-framework/templates/summary-minimal.md +0 -41
  165. package/framework/qualia-framework/templates/summary-standard.md +0 -48
  166. package/framework/qualia-framework/templates/summary.md +0 -246
  167. package/framework/qualia-framework/templates/user-setup.md +0 -311
  168. package/framework/qualia-framework/templates/verification-report.md +0 -322
  169. package/framework/qualia-framework/workflows/add-phase.md +0 -179
  170. package/framework/qualia-framework/workflows/add-todo.md +0 -157
  171. package/framework/qualia-framework/workflows/audit-milestone.md +0 -241
  172. package/framework/qualia-framework/workflows/check-todos.md +0 -176
  173. package/framework/qualia-framework/workflows/complete-milestone.md +0 -858
  174. package/framework/qualia-framework/workflows/diagnose-issues.md +0 -219
  175. package/framework/qualia-framework/workflows/discovery-phase.md +0 -289
  176. package/framework/qualia-framework/workflows/discuss-phase.md +0 -534
  177. package/framework/qualia-framework/workflows/execute-phase.md +0 -559
  178. package/framework/qualia-framework/workflows/execute-plan.md +0 -438
  179. package/framework/qualia-framework/workflows/help.md +0 -470
  180. package/framework/qualia-framework/workflows/insert-phase.md +0 -220
  181. package/framework/qualia-framework/workflows/list-phase-assumptions.md +0 -178
  182. package/framework/qualia-framework/workflows/map-codebase.md +0 -327
  183. package/framework/qualia-framework/workflows/new-milestone.md +0 -363
  184. package/framework/qualia-framework/workflows/new-project.md +0 -982
  185. package/framework/qualia-framework/workflows/pause-work.md +0 -122
  186. package/framework/qualia-framework/workflows/plan-milestone-gaps.md +0 -256
  187. package/framework/qualia-framework/workflows/plan-phase.md +0 -422
  188. package/framework/qualia-framework/workflows/progress.md +0 -389
  189. package/framework/qualia-framework/workflows/quick.md +0 -252
  190. package/framework/qualia-framework/workflows/remove-phase.md +0 -326
  191. package/framework/qualia-framework/workflows/research-phase.md +0 -74
  192. package/framework/qualia-framework/workflows/resume-project.md +0 -306
  193. package/framework/qualia-framework/workflows/set-profile.md +0 -80
  194. package/framework/qualia-framework/workflows/settings.md +0 -145
  195. package/framework/qualia-framework/workflows/transition.md +0 -556
  196. package/framework/qualia-framework/workflows/update.md +0 -197
  197. package/framework/qualia-framework/workflows/verify-phase.md +0 -195
  198. package/framework/qualia-framework/workflows/verify-work.md +0 -625
  199. package/framework/rules/context7.md +0 -14
  200. package/framework/rules/frontend.md +0 -33
  201. package/framework/rules/speed.md +0 -23
  202. package/framework/scripts/__pycache__/say.cpython-314.pyc +0 -0
  203. package/framework/scripts/apply-retention.sh +0 -120
  204. package/framework/scripts/bootstrap-pop-os.sh +0 -354
  205. package/framework/scripts/claude-voice +0 -13
  206. package/framework/scripts/cleanup.sh +0 -131
  207. package/framework/scripts/cowork-mode.sh +0 -141
  208. package/framework/scripts/generate-project-claude-md.sh +0 -153
  209. package/framework/scripts/load-test-webhook.js +0 -172
  210. package/framework/scripts/say.py +0 -236
  211. package/framework/scripts/showcase-video-recorder/ffmpeg-builder.js +0 -167
  212. package/framework/scripts/showcase-video-recorder/playwright-helpers.js +0 -216
  213. package/framework/scripts/speak.py +0 -55
  214. package/framework/scripts/speak.sh +0 -18
  215. package/framework/scripts/status.sh +0 -138
  216. package/framework/scripts/sync-to-framework.sh +0 -65
  217. package/framework/scripts/voice-hotkey.py +0 -227
  218. package/framework/scripts/voice-input.sh +0 -51
  219. package/framework/skills/animate/SKILL.md +0 -202
  220. package/framework/skills/bolder/SKILL.md +0 -144
  221. package/framework/skills/browser-qa/SKILL.md +0 -536
  222. package/framework/skills/clarify/SKILL.md +0 -179
  223. package/framework/skills/client-handoff/SKILL.md +0 -135
  224. package/framework/skills/collab-onboard/SKILL.md +0 -111
  225. package/framework/skills/colorize/SKILL.md +0 -170
  226. package/framework/skills/critique/SKILL.md +0 -126
  227. package/framework/skills/deep-research/SKILL.md +0 -240
  228. package/framework/skills/delight/SKILL.md +0 -329
  229. package/framework/skills/deploy/SKILL.md +0 -261
  230. package/framework/skills/deploy-verify/SKILL.md +0 -377
  231. package/framework/skills/deploy-verify/scripts/canary-check.sh +0 -206
  232. package/framework/skills/deploy-verify/scripts/check-console-errors.js +0 -147
  233. package/framework/skills/deploy-verify/scripts/check-cwv.js +0 -139
  234. package/framework/skills/deploy-verify/scripts/project-detect.sh +0 -84
  235. package/framework/skills/deploy-verify/scripts/verify.sh +0 -548
  236. package/framework/skills/design-quieter/SKILL.md +0 -130
  237. package/framework/skills/distill/SKILL.md +0 -149
  238. package/framework/skills/docs-lookup/SKILL.md +0 -79
  239. package/framework/skills/fcm-notifications/SKILL.md +0 -125
  240. package/framework/skills/financial-ledger/SKILL.md +0 -1039
  241. package/framework/skills/frontend-master/NOTICE.md +0 -4
  242. package/framework/skills/frontend-master/SKILL.md +0 -127
  243. package/framework/skills/frontend-master/reference/color-and-contrast.md +0 -132
  244. package/framework/skills/frontend-master/reference/interaction-design.md +0 -123
  245. package/framework/skills/frontend-master/reference/motion-design.md +0 -99
  246. package/framework/skills/frontend-master/reference/responsive-design.md +0 -114
  247. package/framework/skills/frontend-master/reference/spatial-design.md +0 -100
  248. package/framework/skills/frontend-master/reference/typography.md +0 -131
  249. package/framework/skills/frontend-master/reference/ux-writing.md +0 -107
  250. package/framework/skills/harden/SKILL.md +0 -357
  251. package/framework/skills/i18n-rtl/SKILL.md +0 -752
  252. package/framework/skills/learn/SKILL.md +0 -95
  253. package/framework/skills/memory/SKILL.md +0 -50
  254. package/framework/skills/mobile-expo/SKILL.md +0 -977
  255. package/framework/skills/mobile-expo/references/store-checklist.md +0 -550
  256. package/framework/skills/nestjs-backend/README.md +0 -73
  257. package/framework/skills/nestjs-backend/SKILL.md +0 -446
  258. package/framework/skills/nestjs-backend/references/templates.md +0 -1173
  259. package/framework/skills/normalize/SKILL.md +0 -79
  260. package/framework/skills/onboard/SKILL.md +0 -242
  261. package/framework/skills/openrouter-agent/SKILL.md +0 -922
  262. package/framework/skills/polish/SKILL.md +0 -209
  263. package/framework/skills/pr/SKILL.md +0 -66
  264. package/framework/skills/qualia/SKILL.md +0 -199
  265. package/framework/skills/qualia-add-todo/SKILL.md +0 -68
  266. package/framework/skills/qualia-audit-milestone/SKILL.md +0 -95
  267. package/framework/skills/qualia-check-todos/SKILL.md +0 -55
  268. package/framework/skills/qualia-complete-milestone/SKILL.md +0 -134
  269. package/framework/skills/qualia-debug/SKILL.md +0 -149
  270. package/framework/skills/qualia-design/SKILL.md +0 -203
  271. package/framework/skills/qualia-discuss-phase/SKILL.md +0 -72
  272. package/framework/skills/qualia-evolve/SKILL.md +0 -200
  273. package/framework/skills/qualia-execute-phase/SKILL.md +0 -89
  274. package/framework/skills/qualia-framework-audit/SKILL.md +0 -604
  275. package/framework/skills/qualia-guide/SKILL.md +0 -32
  276. package/framework/skills/qualia-help/SKILL.md +0 -114
  277. package/framework/skills/qualia-idk/SKILL.md +0 -352
  278. package/framework/skills/qualia-list-phase-assumptions/SKILL.md +0 -67
  279. package/framework/skills/qualia-new-milestone/SKILL.md +0 -72
  280. package/framework/skills/qualia-new-project/SKILL.md +0 -232
  281. package/framework/skills/qualia-optimize/SKILL.md +0 -417
  282. package/framework/skills/qualia-pause-work/SKILL.md +0 -96
  283. package/framework/skills/qualia-plan-milestone-gaps/SKILL.md +0 -57
  284. package/framework/skills/qualia-plan-phase/SKILL.md +0 -104
  285. package/framework/skills/qualia-production-check/SKILL.md +0 -0
  286. package/framework/skills/qualia-progress/SKILL.md +0 -53
  287. package/framework/skills/qualia-quick/SKILL.md +0 -89
  288. package/framework/skills/qualia-report/SKILL.md +0 -166
  289. package/framework/skills/qualia-research-phase/SKILL.md +0 -88
  290. package/framework/skills/qualia-resume-work/SKILL.md +0 -62
  291. package/framework/skills/qualia-review/SKILL.md +0 -263
  292. package/framework/skills/qualia-start/SKILL.md +0 -161
  293. package/framework/skills/qualia-verify-work/SKILL.md +0 -132
  294. package/framework/skills/rag/SKILL.md +0 -750
  295. package/framework/skills/responsive/SKILL.md +0 -231
  296. package/framework/skills/retro/SKILL.md +0 -284
  297. package/framework/skills/sakani-conventions/SKILL.md +0 -136
  298. package/framework/skills/sakani-conventions/evals/evals.json +0 -23
  299. package/framework/skills/sakani-conventions/references/entities.md +0 -365
  300. package/framework/skills/sakani-conventions/references/error-codes.md +0 -95
  301. package/framework/skills/seo-master/SKILL.md +0 -490
  302. package/framework/skills/seo-master/references/checklist.md +0 -199
  303. package/framework/skills/seo-master/references/structured-data.md +0 -609
  304. package/framework/skills/ship/SKILL.md +0 -239
  305. package/framework/skills/stack-researcher/SKILL.md +0 -215
  306. package/framework/skills/status/SKILL.md +0 -154
  307. package/framework/skills/status/scripts/health-check.sh +0 -562
  308. package/framework/skills/subscription-payments/SKILL.md +0 -250
  309. package/framework/skills/supabase/SKILL.md +0 -973
  310. package/framework/skills/supabase/references/templates.md +0 -159
  311. package/framework/skills/team/SKILL.md +0 -67
  312. package/framework/skills/test-runner/SKILL.md +0 -202
  313. package/framework/skills/voice-agent/SKILL.md +0 -1312
  314. package/framework/skills/zoho-workflow/SKILL.md +0 -51
  315. package/framework/statusline-command.sh +0 -117
  316. package/framework/teams/default/inboxes/plan-04.json +0 -9
  317. package/framework/teams/review-team.md +0 -75
  318. package/framework/teams/ship-team.md +0 -86
  319. package/profiles/fawzi.json +0 -16
  320. package/profiles/hasan.json +0 -16
  321. package/profiles/moayad.json +0 -16
  322. package/templates/CLAUDE-owner.md +0 -52
  323. package/templates/CLAUDE.md.hbs +0 -58
  324. package/templates/env.claude.template +0 -12
  325. package/templates/settings.json +0 -172
  326. package/uninstall.sh +0 -90
  327. /package/{framework/rules → rules}/deployment.md +0 -0
  328. /package/{framework/rules → rules}/security.md +0 -0
@@ -1,167 +0,0 @@
1
- #!/usr/bin/env node
2
- /**
3
- * FFmpeg Command Builder for Showcase Videos
4
- * Bundled with showcase-video-recorder skill
5
- *
6
- * Usage:
7
- * node ffmpeg-builder.js <input.webm> <output.mp4> [scenes.json]
8
- *
9
- * If scenes.json not provided, reads from stdin.
10
- * scenes.json format: [{ "name": "landing", "time": 0, "speed": 1.0 }, ...]
11
- */
12
-
13
- const fs = require('fs');
14
- const path = require('path');
15
-
16
- const TARGET_DURATION = 60; // seconds
17
-
18
- function buildFfmpegCommand(inputFile, outputFile, scenes) {
19
- // Validate scenes
20
- if (!scenes || scenes.length < 2) {
21
- throw new Error('Need at least 2 scene markers (start + end)');
22
- }
23
-
24
- // Ensure scenes are sorted by time
25
- scenes.sort((a, b) => a.time - b.time);
26
-
27
- // Build segments (each scene runs until the next scene starts)
28
- const segments = [];
29
- for (let i = 0; i < scenes.length - 1; i++) {
30
- const raw = scenes[i + 1].time - scenes[i].time;
31
- const speed = scenes[i].speed || 1.3; // default speed if not specified
32
- const final = raw / speed;
33
- segments.push({
34
- name: scenes[i].name,
35
- start: scenes[i].time,
36
- end: scenes[i + 1].time,
37
- raw,
38
- speed,
39
- final,
40
- });
41
- }
42
-
43
- // Calculate total
44
- const totalFinal = segments.reduce((sum, s) => sum + s.final, 0);
45
-
46
- // Print segment breakdown
47
- console.log('=== SEGMENT BREAKDOWN ===');
48
- for (const s of segments) {
49
- console.log(` ${s.name}: ${s.raw.toFixed(1)}s raw -> ${s.final.toFixed(1)}s @ ${s.speed}x`);
50
- }
51
- console.log(` TOTAL: ${totalFinal.toFixed(1)}s (target: ${TARGET_DURATION}s)`);
52
-
53
- if (totalFinal < 45 || totalFinal > 75) {
54
- console.warn(`\n WARNING: Total duration ${totalFinal.toFixed(1)}s is outside 45-75s range.`);
55
- console.warn(' Adjust speeds and re-run.\n');
56
- }
57
-
58
- // Build ffmpeg filter_complex
59
- const N = segments.length;
60
- const splitOutputs = segments.map((_, i) => `[s${i + 1}]`).join('');
61
- const trimFilters = segments.map((s, i) =>
62
- `[s${i + 1}]trim=start=${s.start.toFixed(3)}:end=${s.end.toFixed(3)},setpts=(PTS-STARTPTS)/${s.speed}[v${i + 1}]`
63
- ).join(';\n ');
64
- const concatInputs = segments.map((_, i) => `[v${i + 1}]`).join('');
65
-
66
- const cmd = `ffmpeg -i "${inputFile}" \\
67
- -filter_complex "
68
- [0:v]split=${N}${splitOutputs};
69
- ${trimFilters};
70
- ${concatInputs}concat=n=${N}:v=1:a=0[out]
71
- " \\
72
- -map "[out]" \\
73
- -c:v libx264 -preset medium -crf 20 \\
74
- -pix_fmt yuv420p -r 30 -s 1920x1080 \\
75
- -movflags +faststart \\
76
- "${outputFile}"`;
77
-
78
- return { cmd, segments, totalFinal };
79
- }
80
-
81
- function autoTuneSpeeds(scenes, targetDuration = TARGET_DURATION) {
82
- const scenesCopy = scenes.map(s => ({ ...s }));
83
-
84
- // Default all to 1.3x if no speed set
85
- for (const s of scenesCopy) {
86
- if (!s.speed) s.speed = 1.3;
87
- }
88
-
89
- // Calculate initial total
90
- let total = 0;
91
- for (let i = 0; i < scenesCopy.length - 1; i++) {
92
- const raw = scenesCopy[i + 1].time - scenesCopy[i].time;
93
- total += raw / scenesCopy[i].speed;
94
- }
95
-
96
- // Iteratively adjust
97
- const maxIterations = 20;
98
- for (let iter = 0; iter < maxIterations; iter++) {
99
- const ratio = total / targetDuration;
100
- if (Math.abs(ratio - 1.0) < 0.05) break; // Within 5% of target
101
-
102
- // Scale all speeds proportionally
103
- for (const s of scenesCopy) {
104
- s.speed = Math.min(4.0, Math.max(0.8, s.speed * ratio));
105
- }
106
-
107
- // Recalculate
108
- total = 0;
109
- for (let i = 0; i < scenesCopy.length - 1; i++) {
110
- const raw = scenesCopy[i + 1].time - scenesCopy[i].time;
111
- total += raw / scenesCopy[i].speed;
112
- }
113
- }
114
-
115
- // Round speeds to 1 decimal
116
- for (const s of scenesCopy) {
117
- s.speed = Math.round(s.speed * 10) / 10;
118
- }
119
-
120
- return scenesCopy;
121
- }
122
-
123
- // CLI entry point
124
- if (require.main === module) {
125
- const args = process.argv.slice(2);
126
-
127
- if (args.length < 2) {
128
- console.log('Usage: node ffmpeg-builder.js <input.webm> <output.mp4> [scenes.json]');
129
- console.log('\nscenes.json example:');
130
- console.log(JSON.stringify([
131
- { name: 'landing', time: 0, speed: 1.0 },
132
- { name: 'dashboard', time: 8, speed: 1.5 },
133
- { name: 'feature-a', time: 20, speed: 1.2 },
134
- { name: 'end', time: 80 },
135
- ], null, 2));
136
- process.exit(1);
137
- }
138
-
139
- const [inputFile, outputFile, scenesFile] = args;
140
-
141
- let scenes;
142
- if (scenesFile) {
143
- scenes = JSON.parse(fs.readFileSync(scenesFile, 'utf-8'));
144
- } else {
145
- // Read from stdin
146
- console.log('Paste scene timestamps JSON (Ctrl+D when done):');
147
- const input = fs.readFileSync('/dev/stdin', 'utf-8');
148
- scenes = JSON.parse(input);
149
- }
150
-
151
- // Auto-tune if speeds not all defined
152
- const hasAllSpeeds = scenes.every(s => s.speed !== undefined);
153
- if (!hasAllSpeeds) {
154
- console.log('Auto-tuning speeds to hit ~60s...');
155
- scenes = autoTuneSpeeds(scenes);
156
- }
157
-
158
- const { cmd, totalFinal } = buildFfmpegCommand(inputFile, outputFile, scenes);
159
-
160
- console.log('\n=== FFMPEG COMMAND ===');
161
- console.log(cmd);
162
-
163
- console.log(`\nEstimated output: ${totalFinal.toFixed(1)}s`);
164
- console.log('Run the command above to process the video.');
165
- }
166
-
167
- module.exports = { buildFfmpegCommand, autoTuneSpeeds };
@@ -1,216 +0,0 @@
1
- /**
2
- * Playwright Recording Helpers
3
- * Bundled with showcase-video-recorder skill
4
- *
5
- * Usage: const helpers = require('./playwright-helpers');
6
- */
7
-
8
- /**
9
- * Mask all input fields to hide credentials.
10
- * Call BEFORE submitting login forms.
11
- */
12
- async function maskInputs(page) {
13
- await page.evaluate(() => {
14
- document.querySelectorAll('input[type="password"], input[type="email"], input[type="text"]')
15
- .forEach(el => {
16
- el.style.webkitTextSecurity = 'disc';
17
- el.style.color = 'transparent';
18
- el.style.textShadow = '0 0 8px rgba(255,255,255,0.5)';
19
- });
20
- });
21
- }
22
-
23
- /**
24
- * Smooth scroll with cubic ease-in-out.
25
- * @param {import('playwright').Page} page
26
- * @param {number} targetY - Target scroll position in pixels
27
- * @param {number} duration - Animation duration in ms (default 1500)
28
- */
29
- async function smoothScroll(page, targetY, duration = 1500) {
30
- await page.evaluate(async ({ targetY, duration }) => {
31
- const startY = window.scrollY;
32
- const distance = targetY - startY;
33
- const startTime = performance.now();
34
- return new Promise(resolve => {
35
- function step(currentTime) {
36
- const elapsed = currentTime - startTime;
37
- const progress = Math.min(elapsed / duration, 1);
38
- // Cubic ease-in-out
39
- const ease = progress < 0.5
40
- ? 4 * progress * progress * progress
41
- : 1 - Math.pow(-2 * progress + 2, 3) / 2;
42
- window.scrollTo(0, startY + distance * ease);
43
- if (progress < 1) requestAnimationFrame(step);
44
- else resolve();
45
- }
46
- requestAnimationFrame(step);
47
- });
48
- }, { targetY, duration });
49
- }
50
-
51
- /**
52
- * Click a button by its accessible name, with fallback to text content.
53
- * @returns {boolean} true if clicked, false if not found
54
- */
55
- async function clickButton(page, text, timeout = 5000) {
56
- try {
57
- const btn = page.getByRole('button', { name: text });
58
- await btn.waitFor({ state: 'visible', timeout });
59
- await btn.scrollIntoViewIfNeeded();
60
- await page.waitForTimeout(300);
61
- await btn.click();
62
- return true;
63
- } catch {
64
- try {
65
- const fallback = page.locator(`text="${text}"`).first();
66
- await fallback.waitFor({ state: 'visible', timeout: 3000 });
67
- await fallback.click();
68
- return true;
69
- } catch {
70
- console.log(`[WARN] Button not found: "${text}" — skipping`);
71
- return false;
72
- }
73
- }
74
- }
75
-
76
- /**
77
- * Type text with natural human-like timing variation.
78
- * @param {number} baseDelay - Base delay between keystrokes in ms
79
- */
80
- async function naturalType(page, selector, text, baseDelay = 80) {
81
- const el = page.locator(selector).first();
82
- await el.waitFor({ state: 'visible', timeout: 5000 });
83
- await el.click();
84
- await page.waitForTimeout(200);
85
- for (const char of text) {
86
- await el.type(char, { delay: 0 });
87
- const delay = baseDelay + Math.random() * 60 - 30;
88
- await page.waitForTimeout(Math.max(30, delay));
89
- }
90
- }
91
-
92
- /**
93
- * Add a custom cursor highlight ring that follows mouse movement.
94
- * Teal border (#00FFD1) with purple click feedback (#8B5CF6).
95
- * Must be called after every page navigation (ring is lost on navigation).
96
- */
97
- async function addCursorHighlight(page) {
98
- await page.evaluate(() => {
99
- // Remove existing ring if present
100
- const existing = document.getElementById('cursor-highlight');
101
- if (existing) existing.remove();
102
-
103
- const ring = document.createElement('div');
104
- ring.id = 'cursor-highlight';
105
- ring.style.cssText = `
106
- position: fixed; width: 28px; height: 28px;
107
- border: 2px solid #00FFD1; border-radius: 50%;
108
- pointer-events: none; z-index: 999999;
109
- transform: translate(-50%, -50%);
110
- transition: all 0.15s ease-out;
111
- box-shadow: 0 0 10px rgba(0,255,209,0.3);
112
- `;
113
- document.body.appendChild(ring);
114
-
115
- document.addEventListener('mousemove', e => {
116
- ring.style.left = e.clientX + 'px';
117
- ring.style.top = e.clientY + 'px';
118
- });
119
-
120
- document.addEventListener('mousedown', () => {
121
- ring.style.borderColor = '#8B5CF6';
122
- ring.style.transform = 'translate(-50%, -50%) scale(0.8)';
123
- ring.style.boxShadow = '0 0 20px rgba(139,92,246,0.5)';
124
- });
125
-
126
- document.addEventListener('mouseup', () => {
127
- ring.style.borderColor = '#00FFD1';
128
- ring.style.transform = 'translate(-50%, -50%) scale(1)';
129
- ring.style.boxShadow = '0 0 10px rgba(0,255,209,0.3)';
130
- });
131
- });
132
- }
133
-
134
- /**
135
- * Click a navigation link by its text content.
136
- * Waits for network idle after navigation.
137
- */
138
- async function clickNavLink(page, text, timeout = 8000) {
139
- try {
140
- const link = page.locator(`a:has-text("${text}")`).first();
141
- await link.waitFor({ state: 'visible', timeout: 5000 });
142
- await link.click();
143
- await page.waitForLoadState('networkidle', { timeout });
144
- await page.waitForTimeout(1500);
145
- return true;
146
- } catch {
147
- console.log(`[WARN] Nav link not found: "${text}" — skipping`);
148
- return false;
149
- }
150
- }
151
-
152
- /**
153
- * Hover over an element to trigger hover states for visual effect.
154
- */
155
- async function hoverElement(page, selector, duration = 1000) {
156
- try {
157
- const el = page.locator(selector).first();
158
- await el.waitFor({ state: 'visible', timeout: 3000 });
159
- await el.hover();
160
- await page.waitForTimeout(duration);
161
- return true;
162
- } catch {
163
- return false;
164
- }
165
- }
166
-
167
- /**
168
- * Scene timestamp tracker for ffmpeg post-processing.
169
- */
170
- class SceneTracker {
171
- constructor() {
172
- this.scenes = [];
173
- this.startTime = null;
174
- }
175
-
176
- start() {
177
- this.startTime = Date.now();
178
- this.mark('start');
179
- }
180
-
181
- mark(name) {
182
- if (!this.startTime) this.startTime = Date.now();
183
- const elapsed = (Date.now() - this.startTime) / 1000;
184
- this.scenes.push({ name, time: elapsed });
185
- console.log(`[SCENE] ${name} @ ${elapsed.toFixed(1)}s`);
186
- }
187
-
188
- getScenes() {
189
- return this.scenes;
190
- }
191
-
192
- printSummary() {
193
- console.log('\n=== SCENE TIMESTAMPS ===');
194
- console.log(JSON.stringify(this.scenes, null, 2));
195
-
196
- // Calculate raw durations
197
- console.log('\n=== SEGMENT DURATIONS ===');
198
- for (let i = 0; i < this.scenes.length - 1; i++) {
199
- const raw = this.scenes[i + 1].time - this.scenes[i].time;
200
- console.log(` ${this.scenes[i].name}: ${raw.toFixed(1)}s`);
201
- }
202
- const total = this.scenes[this.scenes.length - 1].time - this.scenes[0].time;
203
- console.log(` TOTAL RAW: ${total.toFixed(1)}s`);
204
- }
205
- }
206
-
207
- module.exports = {
208
- maskInputs,
209
- smoothScroll,
210
- clickButton,
211
- naturalType,
212
- addCursorHighlight,
213
- clickNavLink,
214
- hoverElement,
215
- SceneTracker,
216
- };
@@ -1,55 +0,0 @@
1
- #!/usr/bin/env python3
2
- """ElevenLabs TTS - pipe text to this script to hear it spoken."""
3
-
4
- import sys
5
- import os
6
- import subprocess
7
- import tempfile
8
-
9
- from elevenlabs import ElevenLabs
10
-
11
- API_KEY = os.environ.get("ELEVEN_API_KEY")
12
- VOICE_ID = os.environ.get("ELEVEN_VOICE_ID", "PB6BdkFkZLbI39GHdnbQ")
13
-
14
- if not API_KEY:
15
- sys.exit(0)
16
-
17
- def speak(text: str):
18
- if not text.strip():
19
- return
20
-
21
- client = ElevenLabs(api_key=API_KEY)
22
-
23
- audio = client.text_to_speech.convert(
24
- voice_id=VOICE_ID,
25
- text=text,
26
- model_id="eleven_turbo_v2_5",
27
- )
28
-
29
- # Write to temp file and play
30
- with tempfile.NamedTemporaryFile(suffix=".mp3", delete=False) as f:
31
- for chunk in audio:
32
- f.write(chunk)
33
- f.flush()
34
-
35
- # Play with mpv, ffplay, or paplay
36
- for cmd in [
37
- ["mpv", "--no-video", "--really-quiet", f.name],
38
- ["ffplay", "-nodisp", "-autoexit", "-loglevel", "quiet", f.name],
39
- ["paplay", f.name],
40
- ]:
41
- try:
42
- subprocess.run(cmd, check=True)
43
- break
44
- except FileNotFoundError:
45
- continue
46
-
47
- os.unlink(f.name)
48
-
49
- if __name__ == "__main__":
50
- if len(sys.argv) > 1:
51
- text = " ".join(sys.argv[1:])
52
- else:
53
- text = sys.stdin.read()
54
-
55
- speak(text)
@@ -1,18 +0,0 @@
1
- #!/bin/bash
2
- # Quick TTS wrapper for hooks
3
- # Requires ELEVEN_API_KEY env var (set in shell profile or .env.claude)
4
- # Cross-platform: detects Windows venv path vs Unix venv path
5
- export PYTHONWARNINGS="ignore::UserWarning"
6
- export ELEVEN_VOICE_ID="${ELEVEN_VOICE_ID:-PB6BdkFkZLbI39GHdnbQ}"
7
-
8
- TEXT="${1:-}"
9
- if [ -n "$TEXT" ] && [ -n "$ELEVEN_API_KEY" ]; then
10
- # Cross-platform venv python detection
11
- VENV_PYTHON="$HOME/.claude/venv/bin/python"
12
- if [ ! -f "$VENV_PYTHON" ]; then
13
- VENV_PYTHON="$HOME/.claude/venv/Scripts/python"
14
- fi
15
- if [ -f "$VENV_PYTHON" ]; then
16
- echo "$TEXT" | "$VENV_PYTHON" "$HOME/.claude/scripts/speak.py" &
17
- fi
18
- fi
@@ -1,138 +0,0 @@
1
- #!/bin/bash
2
- # GSD Status Dashboard - Quick project state overview
3
- # Usage: status.sh [project_path]
4
-
5
- set -e
6
-
7
- PROJECT_DIR="${1:-.}"
8
- cd "$PROJECT_DIR"
9
-
10
- # Colors
11
- RED='\033[0;31m'
12
- GREEN='\033[0;32m'
13
- YELLOW='\033[0;33m'
14
- BLUE='\033[0;34m'
15
- CYAN='\033[0;36m'
16
- BOLD='\033[1m'
17
- DIM='\033[2m'
18
- NC='\033[0m' # No Color
19
-
20
- # Header
21
- echo ""
22
- echo -e "${BOLD}═══════════════════════════════════════════════════════════${NC}"
23
- echo -e "${BOLD} GSD STATUS: $(basename "$PWD")${NC}"
24
- echo -e "${BOLD}═══════════════════════════════════════════════════════════${NC}"
25
-
26
- # Check if GSD project
27
- if [ ! -d ".planning" ]; then
28
- echo -e "\n${YELLOW}⚠ No .planning/ directory - not a GSD project${NC}"
29
- echo -e "${DIM}Run /gsd:new-project to initialize${NC}\n"
30
- exit 0
31
- fi
32
-
33
- # Current Phase from STATE.md
34
- echo -e "\n${CYAN}▸ CURRENT POSITION${NC}"
35
- if [ -f ".planning/STATE.md" ]; then
36
- # Extract current phase line
37
- PHASE_LINE=$(grep -E "^(current_phase|## Current|Phase:)" .planning/STATE.md 2>/dev/null | head -1 || echo "")
38
- if [ -n "$PHASE_LINE" ]; then
39
- echo -e " $PHASE_LINE"
40
- fi
41
-
42
- # Extract status
43
- STATUS_LINE=$(grep -E "^(status:|Status:)" .planning/STATE.md 2>/dev/null | head -1 || echo "")
44
- if [ -n "$STATUS_LINE" ]; then
45
- echo -e " $STATUS_LINE"
46
- fi
47
- else
48
- echo -e " ${YELLOW}No STATE.md found${NC}"
49
- fi
50
-
51
- # Phase Overview from ROADMAP.md
52
- echo -e "\n${CYAN}▸ PHASES${NC}"
53
- if [ -f ".planning/ROADMAP.md" ]; then
54
- # Count phases and show completion
55
- TOTAL_PHASES=$(grep -cE "^## Phase [0-9]" .planning/ROADMAP.md 2>/dev/null || echo "0")
56
- COMPLETE_PHASES=$(grep -cE "\[x\]|✓|complete" .planning/ROADMAP.md 2>/dev/null || echo "0")
57
- echo -e " Progress: ${GREEN}$COMPLETE_PHASES${NC}/${TOTAL_PHASES} phases"
58
-
59
- # List phases
60
- grep -E "^## Phase [0-9]" .planning/ROADMAP.md 2>/dev/null | head -10 | while read -r line; do
61
- echo -e " ${DIM}$line${NC}"
62
- done
63
- else
64
- echo -e " ${YELLOW}No ROADMAP.md found${NC}"
65
- fi
66
-
67
- # Plans in current phase
68
- echo -e "\n${CYAN}▸ PLANS${NC}"
69
- if [ -d ".planning/phases" ]; then
70
- # Find most recent phase directory
71
- LATEST_PHASE=$(ls -d .planning/phases/*/ 2>/dev/null | sort -V | tail -1)
72
- if [ -n "$LATEST_PHASE" ]; then
73
- PHASE_NAME=$(basename "$LATEST_PHASE")
74
- TOTAL_PLANS=$(ls -1 "$LATEST_PHASE"*-PLAN.md 2>/dev/null | wc -l | tr -d ' ')
75
- COMPLETE_PLANS=$(ls -1 "$LATEST_PHASE"*-SUMMARY.md 2>/dev/null | wc -l | tr -d ' ')
76
- echo -e " ${BOLD}$PHASE_NAME${NC}: ${GREEN}$COMPLETE_PLANS${NC}/${TOTAL_PLANS} plans complete"
77
-
78
- # List incomplete plans
79
- for plan in "$LATEST_PHASE"*-PLAN.md; do
80
- if [ -f "$plan" ]; then
81
- PLAN_NAME=$(basename "$plan" -PLAN.md)
82
- SUMMARY="${plan/-PLAN.md/-SUMMARY.md}"
83
- if [ -f "$SUMMARY" ]; then
84
- echo -e " ${GREEN}✓${NC} $PLAN_NAME"
85
- else
86
- echo -e " ${YELLOW}○${NC} $PLAN_NAME ${DIM}(pending)${NC}"
87
- fi
88
- fi
89
- done 2>/dev/null
90
- fi
91
- else
92
- echo -e " ${YELLOW}No phases directory${NC}"
93
- fi
94
-
95
- # Blockers from STATE.md
96
- echo -e "\n${CYAN}▸ BLOCKERS${NC}"
97
- if [ -f ".planning/STATE.md" ]; then
98
- BLOCKERS=$(sed -n '/## Blockers/,/^##/p' .planning/STATE.md 2>/dev/null | grep -E "^-" | head -5)
99
- if [ -n "$BLOCKERS" ]; then
100
- echo -e "${RED}$BLOCKERS${NC}"
101
- else
102
- echo -e " ${GREEN}None${NC}"
103
- fi
104
- else
105
- echo -e " ${DIM}Unknown${NC}"
106
- fi
107
-
108
- # Git status (brief)
109
- echo -e "\n${CYAN}▸ GIT${NC}"
110
- if git rev-parse --git-dir > /dev/null 2>&1; then
111
- BRANCH=$(git branch --show-current 2>/dev/null || echo "detached")
112
- CHANGES=$(git status --porcelain 2>/dev/null | wc -l | tr -d ' ')
113
- COMMITS_AHEAD=$(git rev-list --count @{u}..HEAD 2>/dev/null || echo "?")
114
- echo -e " Branch: ${BOLD}$BRANCH${NC}"
115
- echo -e " Uncommitted: $CHANGES files"
116
- [ "$COMMITS_AHEAD" != "?" ] && echo -e " Ahead of remote: $COMMITS_AHEAD commits"
117
- else
118
- echo -e " ${YELLOW}Not a git repo${NC}"
119
- fi
120
-
121
- # Suggested next action
122
- echo -e "\n${CYAN}▸ NEXT ACTION${NC}"
123
- if [ -f ".planning/STATE.md" ]; then
124
- # Simple heuristic based on state
125
- if grep -qE "needs.*(plan|planning)" .planning/STATE.md 2>/dev/null; then
126
- echo -e " ${BOLD}/gsd:plan-phase${NC}"
127
- elif grep -qE "needs.*(exec|execution)" .planning/STATE.md 2>/dev/null; then
128
- echo -e " ${BOLD}/gsd:execute-phase${NC}"
129
- elif grep -qE "needs.*(verify|verification)" .planning/STATE.md 2>/dev/null; then
130
- echo -e " ${BOLD}/gsd:verify-work${NC}"
131
- else
132
- echo -e " ${BOLD}/gsd:progress${NC} ${DIM}(check state)${NC}"
133
- fi
134
- else
135
- echo -e " ${BOLD}/gsd:new-project${NC}"
136
- fi
137
-
138
- echo -e "\n${DIM}─────────────────────────────────────────────────────────────${NC}\n"
@@ -1,65 +0,0 @@
1
- #!/bin/bash
2
- # Sync ~/.claude/ framework files → ~/Projects/qualia-framework/framework/
3
- # Run this after making changes to ~/.claude/ to prepare for distribution
4
- # Usage: ~/.claude/scripts/sync-to-framework.sh
5
-
6
- source "$HOME/.claude/hooks/qualia-colors.sh"
7
-
8
- SRC="$HOME/.claude"
9
- DEST="$HOME/Projects/qualia-framework/framework"
10
-
11
- if [ ! -d "$DEST" ]; then
12
- printf "${Q_FAIL} ✗ Framework repo not found at ~/Projects/qualia-framework${Q_RESET}\n"
13
- exit 1
14
- fi
15
-
16
- q_header "SYNC TO FRAMEWORK REPO"
17
-
18
- # Sync framework components (these are the distributable parts)
19
- COMPONENTS=(
20
- "skills"
21
- "agents"
22
- "hooks"
23
- "rules"
24
- "knowledge"
25
- "qualia-framework"
26
- "scripts"
27
- )
28
-
29
- for comp in "${COMPONENTS[@]}"; do
30
- if [ -d "$SRC/$comp" ]; then
31
- rsync -a --delete \
32
- --exclude="node_modules" \
33
- --exclude=".DS_Store" \
34
- --exclude="retros/" \
35
- --exclude="project-notes/" \
36
- --exclude="session-digest.md" \
37
- "$SRC/$comp/" "$DEST/$comp/"
38
- COUNT=$(find "$DEST/$comp" -type f | wc -l)
39
- q_pass "$comp ($COUNT files)"
40
- fi
41
- done
42
-
43
- # Sync individual files
44
- cp "$SRC/CLAUDE.md" "$DEST/../templates/CLAUDE-owner.md" 2>/dev/null && q_pass "CLAUDE.md → templates/CLAUDE-owner.md"
45
- cp "$SRC/statusline-command.sh" "$DEST/statusline-command.sh" 2>/dev/null && q_pass "statusline-command.sh"
46
- cp "$SRC/askpass.sh" "$DEST/askpass.sh" 2>/dev/null && q_pass "askpass.sh"
47
- cp "$SRC/install.sh" "$DEST/../install.sh" 2>/dev/null && q_pass "install.sh"
48
- cp "$SRC/install.ps1" "$DEST/../install.ps1" 2>/dev/null && q_pass "install.ps1"
49
-
50
- # Don't sync: secrets, session data, history, plugins, caches, personal memories
51
- # Those stay local to each machine
52
-
53
- # Show what changed in the repo
54
- echo ""
55
- cd "$HOME/Projects/qualia-framework"
56
- CHANGED=$(git status --porcelain | wc -l)
57
- if [ "$CHANGED" -gt 0 ]; then
58
- q_warn "$CHANGED files changed"
59
- git status --short | head -20
60
- echo ""
61
- printf "${Q_DIM} Ready to commit. Run:${Q_RESET}\n"
62
- printf "${Q_TEAL} cd ~/Projects/qualia-framework && git add -A && git commit -m 'chore: sync framework updates' && git push${Q_RESET}\n"
63
- else
64
- q_pass "Already up to date"
65
- fi