qualia-framework 2.6.0 → 3.2.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 (321) hide show
  1. package/CLAUDE.md +64 -0
  2. package/README.md +103 -30
  3. package/agents/builder.md +110 -0
  4. package/agents/planner.md +134 -0
  5. package/agents/qa-browser.md +186 -0
  6. package/agents/verifier.md +221 -0
  7. package/bin/cli.js +336 -531
  8. package/bin/install.js +570 -0
  9. package/bin/qualia-ui.js +299 -0
  10. package/bin/state.js +630 -0
  11. package/bin/statusline.js +252 -0
  12. package/guide.md +63 -0
  13. package/hooks/auto-update.js +139 -0
  14. package/hooks/branch-guard.js +47 -0
  15. package/hooks/migration-guard.js +60 -0
  16. package/hooks/pre-compact.js +32 -0
  17. package/hooks/pre-deploy-gate.js +110 -0
  18. package/hooks/pre-push.js +33 -0
  19. package/hooks/session-start.js +170 -0
  20. package/package.json +29 -20
  21. package/rules/design-reference.md +179 -0
  22. package/rules/frontend.md +126 -0
  23. package/skills/qualia/SKILL.md +87 -0
  24. package/skills/qualia-build/SKILL.md +97 -0
  25. package/skills/qualia-debug/SKILL.md +87 -0
  26. package/skills/qualia-design/SKILL.md +93 -0
  27. package/skills/qualia-handoff/SKILL.md +66 -0
  28. package/skills/qualia-idk/SKILL.md +8 -0
  29. package/skills/qualia-learn/SKILL.md +88 -0
  30. package/skills/qualia-new/SKILL.md +323 -0
  31. package/{framework/skills → skills}/qualia-optimize/SKILL.md +1 -1
  32. package/skills/qualia-pause/SKILL.md +63 -0
  33. package/skills/qualia-plan/SKILL.md +101 -0
  34. package/skills/qualia-polish/SKILL.md +157 -0
  35. package/skills/qualia-quick/SKILL.md +37 -0
  36. package/skills/qualia-report/SKILL.md +105 -0
  37. package/skills/qualia-resume/SKILL.md +49 -0
  38. package/skills/qualia-review/SKILL.md +76 -0
  39. package/skills/qualia-ship/SKILL.md +90 -0
  40. package/skills/qualia-skill-new/SKILL.md +167 -0
  41. package/skills/qualia-task/SKILL.md +91 -0
  42. package/skills/qualia-verify/SKILL.md +113 -0
  43. package/templates/DESIGN.md +137 -0
  44. package/templates/plan.md +28 -0
  45. package/templates/project.md +22 -0
  46. package/templates/state.md +27 -0
  47. package/templates/tracking.json +20 -0
  48. package/tests/bin.test.sh +673 -0
  49. package/tests/hooks.test.sh +315 -0
  50. package/tests/state.test.sh +535 -0
  51. package/tests/statusline.test.sh +243 -0
  52. package/bin/collect-metrics.sh +0 -62
  53. package/framework/.claudeignore +0 -51
  54. package/framework/CLAUDE.md +0 -51
  55. package/framework/MCP_SETUP.md +0 -229
  56. package/framework/agents/architecture-strategist.md +0 -53
  57. package/framework/agents/backend-agent.md +0 -150
  58. package/framework/agents/code-simplicity-reviewer.md +0 -86
  59. package/framework/agents/frontend-agent.md +0 -111
  60. package/framework/agents/kieran-typescript-reviewer.md +0 -96
  61. package/framework/agents/performance-oracle.md +0 -111
  62. package/framework/agents/qualia-codebase-mapper.md +0 -761
  63. package/framework/agents/qualia-debugger.md +0 -1204
  64. package/framework/agents/qualia-executor.md +0 -882
  65. package/framework/agents/qualia-integration-checker.md +0 -424
  66. package/framework/agents/qualia-phase-researcher.md +0 -457
  67. package/framework/agents/qualia-plan-checker.md +0 -700
  68. package/framework/agents/qualia-planner.md +0 -1245
  69. package/framework/agents/qualia-project-researcher.md +0 -603
  70. package/framework/agents/qualia-research-synthesizer.md +0 -200
  71. package/framework/agents/qualia-roadmapper.md +0 -606
  72. package/framework/agents/qualia-verifier.md +0 -686
  73. package/framework/agents/red-team-qa.md +0 -130
  74. package/framework/agents/security-auditor.md +0 -72
  75. package/framework/agents/team-orchestrator.md +0 -229
  76. package/framework/agents/teams/framework-audit-team.md +0 -66
  77. package/framework/agents/teams/full-stack-team.md +0 -48
  78. package/framework/agents/teams/optimize-team.md +0 -53
  79. package/framework/agents/teams/review-team.md +0 -70
  80. package/framework/agents/teams/ship-team.md +0 -86
  81. package/framework/agents/test-agent.md +0 -182
  82. package/framework/hooks/auto-format.sh +0 -54
  83. package/framework/hooks/block-env-edit.sh +0 -42
  84. package/framework/hooks/branch-guard.sh +0 -43
  85. package/framework/hooks/confirm-delete.sh +0 -59
  86. package/framework/hooks/migration-validate.sh +0 -77
  87. package/framework/hooks/notification-speak.sh +0 -16
  88. package/framework/hooks/pre-commit.sh +0 -100
  89. package/framework/hooks/pre-compact.sh +0 -56
  90. package/framework/hooks/pre-deploy-gate.sh +0 -160
  91. package/framework/hooks/qualia-colors.sh +0 -32
  92. package/framework/hooks/retention-cleanup.sh +0 -62
  93. package/framework/hooks/save-session-state.sh +0 -185
  94. package/framework/hooks/session-context-loader.sh +0 -96
  95. package/framework/hooks/session-learn.sh +0 -32
  96. package/framework/hooks/skill-announce.sh +0 -123
  97. package/framework/hooks/tool-error-announce.sh +0 -27
  98. package/framework/install.ps1 +0 -323
  99. package/framework/install.sh +0 -313
  100. package/framework/qualia-framework/VERSION +0 -1
  101. package/framework/qualia-framework/assets/qualia-logo.png +0 -0
  102. package/framework/qualia-framework/bin/collect-metrics.sh +0 -67
  103. package/framework/qualia-framework/bin/generate-report-docx.py +0 -429
  104. package/framework/qualia-framework/bin/qualia-tools.js +0 -2201
  105. package/framework/qualia-framework/bin/qualia-tools.test.js +0 -1054
  106. package/framework/qualia-framework/references/checkpoints.md +0 -775
  107. package/framework/qualia-framework/references/completion-checklists.md +0 -359
  108. package/framework/qualia-framework/references/continuation-format.md +0 -249
  109. package/framework/qualia-framework/references/continuation-prompt.md +0 -97
  110. package/framework/qualia-framework/references/decimal-phase-calculation.md +0 -65
  111. package/framework/qualia-framework/references/design-quality.md +0 -56
  112. package/framework/qualia-framework/references/employee-guide.md +0 -167
  113. package/framework/qualia-framework/references/git-integration.md +0 -254
  114. package/framework/qualia-framework/references/git-planning-commit.md +0 -50
  115. package/framework/qualia-framework/references/model-profile-resolution.md +0 -32
  116. package/framework/qualia-framework/references/model-profiles.md +0 -73
  117. package/framework/qualia-framework/references/phase-argument-parsing.md +0 -61
  118. package/framework/qualia-framework/references/planning-config.md +0 -195
  119. package/framework/qualia-framework/references/questioning.md +0 -141
  120. package/framework/qualia-framework/references/tdd.md +0 -263
  121. package/framework/qualia-framework/references/ui-brand.md +0 -160
  122. package/framework/qualia-framework/references/verification-patterns.md +0 -612
  123. package/framework/qualia-framework/templates/DEBUG.md +0 -159
  124. package/framework/qualia-framework/templates/DESIGN.md +0 -81
  125. package/framework/qualia-framework/templates/UAT.md +0 -247
  126. package/framework/qualia-framework/templates/codebase/architecture.md +0 -255
  127. package/framework/qualia-framework/templates/codebase/concerns.md +0 -310
  128. package/framework/qualia-framework/templates/codebase/conventions.md +0 -307
  129. package/framework/qualia-framework/templates/codebase/integrations.md +0 -280
  130. package/framework/qualia-framework/templates/codebase/stack.md +0 -186
  131. package/framework/qualia-framework/templates/codebase/structure.md +0 -285
  132. package/framework/qualia-framework/templates/codebase/testing.md +0 -480
  133. package/framework/qualia-framework/templates/config.json +0 -35
  134. package/framework/qualia-framework/templates/context.md +0 -283
  135. package/framework/qualia-framework/templates/continue-here.md +0 -78
  136. package/framework/qualia-framework/templates/debug-subagent-prompt.md +0 -91
  137. package/framework/qualia-framework/templates/discovery.md +0 -146
  138. package/framework/qualia-framework/templates/lab-notes.md +0 -16
  139. package/framework/qualia-framework/templates/milestone-archive.md +0 -123
  140. package/framework/qualia-framework/templates/milestone.md +0 -115
  141. package/framework/qualia-framework/templates/phase-prompt.md +0 -567
  142. package/framework/qualia-framework/templates/planner-subagent-prompt.md +0 -117
  143. package/framework/qualia-framework/templates/project.md +0 -184
  144. package/framework/qualia-framework/templates/projects/ai-agent.md +0 -156
  145. package/framework/qualia-framework/templates/projects/mobile-app.md +0 -181
  146. package/framework/qualia-framework/templates/projects/voice-agent.md +0 -134
  147. package/framework/qualia-framework/templates/projects/website.md +0 -137
  148. package/framework/qualia-framework/templates/requirements.md +0 -231
  149. package/framework/qualia-framework/templates/research-project/ARCHITECTURE.md +0 -204
  150. package/framework/qualia-framework/templates/research-project/FEATURES.md +0 -147
  151. package/framework/qualia-framework/templates/research-project/PITFALLS.md +0 -200
  152. package/framework/qualia-framework/templates/research-project/STACK.md +0 -120
  153. package/framework/qualia-framework/templates/research-project/SUMMARY.md +0 -170
  154. package/framework/qualia-framework/templates/research.md +0 -552
  155. package/framework/qualia-framework/templates/roadmap.md +0 -206
  156. package/framework/qualia-framework/templates/state.md +0 -179
  157. package/framework/qualia-framework/templates/summary-complex.md +0 -59
  158. package/framework/qualia-framework/templates/summary-minimal.md +0 -41
  159. package/framework/qualia-framework/templates/summary-standard.md +0 -48
  160. package/framework/qualia-framework/templates/summary.md +0 -246
  161. package/framework/qualia-framework/templates/user-setup.md +0 -311
  162. package/framework/qualia-framework/templates/verification-report.md +0 -322
  163. package/framework/qualia-framework/workflows/add-phase.md +0 -179
  164. package/framework/qualia-framework/workflows/add-todo.md +0 -157
  165. package/framework/qualia-framework/workflows/audit-milestone.md +0 -241
  166. package/framework/qualia-framework/workflows/check-todos.md +0 -176
  167. package/framework/qualia-framework/workflows/complete-milestone.md +0 -858
  168. package/framework/qualia-framework/workflows/diagnose-issues.md +0 -219
  169. package/framework/qualia-framework/workflows/discovery-phase.md +0 -289
  170. package/framework/qualia-framework/workflows/discuss-phase.md +0 -534
  171. package/framework/qualia-framework/workflows/execute-phase.md +0 -559
  172. package/framework/qualia-framework/workflows/execute-plan.md +0 -438
  173. package/framework/qualia-framework/workflows/help.md +0 -470
  174. package/framework/qualia-framework/workflows/insert-phase.md +0 -220
  175. package/framework/qualia-framework/workflows/list-phase-assumptions.md +0 -178
  176. package/framework/qualia-framework/workflows/map-codebase.md +0 -327
  177. package/framework/qualia-framework/workflows/new-milestone.md +0 -363
  178. package/framework/qualia-framework/workflows/new-project.md +0 -982
  179. package/framework/qualia-framework/workflows/pause-work.md +0 -122
  180. package/framework/qualia-framework/workflows/plan-milestone-gaps.md +0 -256
  181. package/framework/qualia-framework/workflows/plan-phase.md +0 -422
  182. package/framework/qualia-framework/workflows/progress.md +0 -389
  183. package/framework/qualia-framework/workflows/quick.md +0 -252
  184. package/framework/qualia-framework/workflows/remove-phase.md +0 -326
  185. package/framework/qualia-framework/workflows/research-phase.md +0 -74
  186. package/framework/qualia-framework/workflows/resume-project.md +0 -306
  187. package/framework/qualia-framework/workflows/set-profile.md +0 -80
  188. package/framework/qualia-framework/workflows/settings.md +0 -145
  189. package/framework/qualia-framework/workflows/transition.md +0 -556
  190. package/framework/qualia-framework/workflows/update.md +0 -197
  191. package/framework/qualia-framework/workflows/verify-phase.md +0 -195
  192. package/framework/qualia-framework/workflows/verify-work.md +0 -625
  193. package/framework/rules/context7.md +0 -14
  194. package/framework/rules/frontend.md +0 -33
  195. package/framework/rules/speed.md +0 -23
  196. package/framework/scripts/__pycache__/say.cpython-314.pyc +0 -0
  197. package/framework/scripts/apply-retention.sh +0 -120
  198. package/framework/scripts/bootstrap-pop-os.sh +0 -354
  199. package/framework/scripts/claude-voice +0 -13
  200. package/framework/scripts/cleanup.sh +0 -131
  201. package/framework/scripts/cowork-mode.sh +0 -141
  202. package/framework/scripts/generate-project-claude-md.sh +0 -153
  203. package/framework/scripts/load-test-webhook.js +0 -172
  204. package/framework/scripts/say.py +0 -236
  205. package/framework/scripts/showcase-video-recorder/ffmpeg-builder.js +0 -167
  206. package/framework/scripts/showcase-video-recorder/playwright-helpers.js +0 -216
  207. package/framework/scripts/speak.py +0 -55
  208. package/framework/scripts/speak.sh +0 -18
  209. package/framework/scripts/status.sh +0 -138
  210. package/framework/scripts/sync-to-framework.sh +0 -65
  211. package/framework/scripts/voice-hotkey.py +0 -227
  212. package/framework/scripts/voice-input.sh +0 -51
  213. package/framework/skills/animate/SKILL.md +0 -202
  214. package/framework/skills/bolder/SKILL.md +0 -144
  215. package/framework/skills/browser-qa/SKILL.md +0 -536
  216. package/framework/skills/clarify/SKILL.md +0 -179
  217. package/framework/skills/client-handoff/SKILL.md +0 -135
  218. package/framework/skills/collab-onboard/SKILL.md +0 -111
  219. package/framework/skills/colorize/SKILL.md +0 -170
  220. package/framework/skills/critique/SKILL.md +0 -126
  221. package/framework/skills/deep-research/SKILL.md +0 -240
  222. package/framework/skills/delight/SKILL.md +0 -329
  223. package/framework/skills/deploy/SKILL.md +0 -261
  224. package/framework/skills/deploy-verify/SKILL.md +0 -377
  225. package/framework/skills/deploy-verify/scripts/canary-check.sh +0 -206
  226. package/framework/skills/deploy-verify/scripts/check-console-errors.js +0 -147
  227. package/framework/skills/deploy-verify/scripts/check-cwv.js +0 -139
  228. package/framework/skills/deploy-verify/scripts/project-detect.sh +0 -84
  229. package/framework/skills/deploy-verify/scripts/verify.sh +0 -548
  230. package/framework/skills/design-quieter/SKILL.md +0 -130
  231. package/framework/skills/distill/SKILL.md +0 -149
  232. package/framework/skills/docs-lookup/SKILL.md +0 -79
  233. package/framework/skills/fcm-notifications/SKILL.md +0 -125
  234. package/framework/skills/financial-ledger/SKILL.md +0 -1039
  235. package/framework/skills/frontend-master/NOTICE.md +0 -4
  236. package/framework/skills/frontend-master/SKILL.md +0 -127
  237. package/framework/skills/frontend-master/reference/color-and-contrast.md +0 -132
  238. package/framework/skills/frontend-master/reference/interaction-design.md +0 -123
  239. package/framework/skills/frontend-master/reference/motion-design.md +0 -99
  240. package/framework/skills/frontend-master/reference/responsive-design.md +0 -114
  241. package/framework/skills/frontend-master/reference/spatial-design.md +0 -100
  242. package/framework/skills/frontend-master/reference/typography.md +0 -131
  243. package/framework/skills/frontend-master/reference/ux-writing.md +0 -107
  244. package/framework/skills/harden/SKILL.md +0 -357
  245. package/framework/skills/i18n-rtl/SKILL.md +0 -752
  246. package/framework/skills/learn/SKILL.md +0 -95
  247. package/framework/skills/memory/SKILL.md +0 -50
  248. package/framework/skills/mobile-expo/SKILL.md +0 -977
  249. package/framework/skills/mobile-expo/references/store-checklist.md +0 -550
  250. package/framework/skills/nestjs-backend/README.md +0 -73
  251. package/framework/skills/nestjs-backend/SKILL.md +0 -446
  252. package/framework/skills/nestjs-backend/references/templates.md +0 -1173
  253. package/framework/skills/normalize/SKILL.md +0 -79
  254. package/framework/skills/onboard/SKILL.md +0 -242
  255. package/framework/skills/openrouter-agent/SKILL.md +0 -922
  256. package/framework/skills/polish/SKILL.md +0 -209
  257. package/framework/skills/pr/SKILL.md +0 -66
  258. package/framework/skills/qualia/SKILL.md +0 -199
  259. package/framework/skills/qualia-add-todo/SKILL.md +0 -68
  260. package/framework/skills/qualia-audit-milestone/SKILL.md +0 -95
  261. package/framework/skills/qualia-check-todos/SKILL.md +0 -55
  262. package/framework/skills/qualia-complete-milestone/SKILL.md +0 -134
  263. package/framework/skills/qualia-debug/SKILL.md +0 -149
  264. package/framework/skills/qualia-design/SKILL.md +0 -203
  265. package/framework/skills/qualia-discuss-phase/SKILL.md +0 -72
  266. package/framework/skills/qualia-evolve/SKILL.md +0 -200
  267. package/framework/skills/qualia-execute-phase/SKILL.md +0 -89
  268. package/framework/skills/qualia-framework-audit/SKILL.md +0 -604
  269. package/framework/skills/qualia-guide/SKILL.md +0 -32
  270. package/framework/skills/qualia-help/SKILL.md +0 -114
  271. package/framework/skills/qualia-idk/SKILL.md +0 -352
  272. package/framework/skills/qualia-list-phase-assumptions/SKILL.md +0 -67
  273. package/framework/skills/qualia-new-milestone/SKILL.md +0 -72
  274. package/framework/skills/qualia-new-project/SKILL.md +0 -232
  275. package/framework/skills/qualia-pause-work/SKILL.md +0 -96
  276. package/framework/skills/qualia-plan-milestone-gaps/SKILL.md +0 -57
  277. package/framework/skills/qualia-plan-phase/SKILL.md +0 -104
  278. package/framework/skills/qualia-production-check/SKILL.md +0 -0
  279. package/framework/skills/qualia-progress/SKILL.md +0 -53
  280. package/framework/skills/qualia-quick/SKILL.md +0 -89
  281. package/framework/skills/qualia-report/SKILL.md +0 -166
  282. package/framework/skills/qualia-research-phase/SKILL.md +0 -88
  283. package/framework/skills/qualia-resume-work/SKILL.md +0 -62
  284. package/framework/skills/qualia-review/SKILL.md +0 -263
  285. package/framework/skills/qualia-start/SKILL.md +0 -161
  286. package/framework/skills/qualia-verify-work/SKILL.md +0 -132
  287. package/framework/skills/rag/SKILL.md +0 -750
  288. package/framework/skills/responsive/SKILL.md +0 -231
  289. package/framework/skills/retro/SKILL.md +0 -284
  290. package/framework/skills/sakani-conventions/SKILL.md +0 -136
  291. package/framework/skills/sakani-conventions/evals/evals.json +0 -23
  292. package/framework/skills/sakani-conventions/references/entities.md +0 -365
  293. package/framework/skills/sakani-conventions/references/error-codes.md +0 -95
  294. package/framework/skills/seo-master/SKILL.md +0 -490
  295. package/framework/skills/seo-master/references/checklist.md +0 -199
  296. package/framework/skills/seo-master/references/structured-data.md +0 -609
  297. package/framework/skills/ship/SKILL.md +0 -239
  298. package/framework/skills/stack-researcher/SKILL.md +0 -215
  299. package/framework/skills/status/SKILL.md +0 -154
  300. package/framework/skills/status/scripts/health-check.sh +0 -562
  301. package/framework/skills/subscription-payments/SKILL.md +0 -250
  302. package/framework/skills/supabase/SKILL.md +0 -973
  303. package/framework/skills/supabase/references/templates.md +0 -159
  304. package/framework/skills/team/SKILL.md +0 -67
  305. package/framework/skills/test-runner/SKILL.md +0 -202
  306. package/framework/skills/voice-agent/SKILL.md +0 -1312
  307. package/framework/skills/zoho-workflow/SKILL.md +0 -51
  308. package/framework/statusline-command.sh +0 -117
  309. package/framework/teams/default/inboxes/plan-04.json +0 -9
  310. package/framework/teams/review-team.md +0 -75
  311. package/framework/teams/ship-team.md +0 -86
  312. package/profiles/fawzi.json +0 -16
  313. package/profiles/hasan.json +0 -16
  314. package/profiles/moayad.json +0 -16
  315. package/templates/CLAUDE-owner.md +0 -52
  316. package/templates/CLAUDE.md.hbs +0 -58
  317. package/templates/env.claude.template +0 -12
  318. package/templates/settings.json +0 -172
  319. package/uninstall.sh +0 -90
  320. /package/{framework/rules → rules}/deployment.md +0 -0
  321. /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