@claude-collective/cli 0.8.0 → 0.13.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 (217) hide show
  1. package/CHANGELOG.md +127 -0
  2. package/README.md +26 -9
  3. package/dist/{chunk-TOPAIL5W.js → chunk-3U3R4NCG.js} +2 -2
  4. package/dist/chunk-3U3R4NCG.js.map +1 -0
  5. package/dist/{chunk-YKXBGCFD.js → chunk-57Y5RALO.js} +10 -10
  6. package/dist/chunk-57Y5RALO.js.map +1 -0
  7. package/dist/{chunk-ED73HCW2.js → chunk-6DCSSORF.js} +37 -88
  8. package/dist/chunk-6DCSSORF.js.map +1 -0
  9. package/dist/{chunk-6LS7XO3H.js → chunk-6Q3Y7KVB.js} +2 -2
  10. package/dist/chunk-6Q3Y7KVB.js.map +1 -0
  11. package/dist/{chunk-A3J6IAXK.js → chunk-76DWXGQE.js} +4 -2
  12. package/dist/chunk-76DWXGQE.js.map +1 -0
  13. package/dist/{chunk-Q6DR5QUH.js → chunk-7Q44DMSP.js} +62 -27
  14. package/dist/chunk-7Q44DMSP.js.map +1 -0
  15. package/dist/chunk-ACNBKXXJ.js +321 -0
  16. package/dist/chunk-ACNBKXXJ.js.map +1 -0
  17. package/dist/{chunk-QGGSLMO3.js → chunk-B7CCVP6Q.js} +41 -9
  18. package/dist/chunk-B7CCVP6Q.js.map +1 -0
  19. package/dist/{chunk-LVKRVFYR.js → chunk-BDLUZVKU.js} +2 -2
  20. package/dist/chunk-BDLUZVKU.js.map +1 -0
  21. package/dist/{chunk-HNDT5QRB.js → chunk-CDX4W4DM.js} +3 -3
  22. package/dist/chunk-CDX4W4DM.js.map +1 -0
  23. package/dist/{chunk-MYAVQ23U.js → chunk-CJEHB4TB.js} +23 -9
  24. package/dist/chunk-CJEHB4TB.js.map +1 -0
  25. package/dist/{chunk-DKGL77IY.js → chunk-CPZOTVCI.js} +15 -14
  26. package/dist/chunk-CPZOTVCI.js.map +1 -0
  27. package/dist/{chunk-K7PTOVX4.js → chunk-D237EVNB.js} +32 -3
  28. package/dist/chunk-D237EVNB.js.map +1 -0
  29. package/dist/{chunk-Q2LH2DAB.js → chunk-DRXPNNPB.js} +19 -18
  30. package/dist/chunk-DRXPNNPB.js.map +1 -0
  31. package/dist/{chunk-Y3V43XCU.js → chunk-E3FJH4TF.js} +12 -8
  32. package/dist/chunk-E3FJH4TF.js.map +1 -0
  33. package/dist/{chunk-3HBTELJN.js → chunk-ED4E6Q2T.js} +10 -10
  34. package/dist/chunk-ED4E6Q2T.js.map +1 -0
  35. package/dist/{chunk-SYQ7R2JO.js → chunk-EHS3TWWP.js} +3 -3
  36. package/dist/chunk-EHS3TWWP.js.map +1 -0
  37. package/dist/{chunk-LQTST4WY.js → chunk-GDH553MV.js} +6 -6
  38. package/dist/chunk-GDH553MV.js.map +1 -0
  39. package/dist/{chunk-A65SBAAJ.js → chunk-HLJX2FTL.js} +31 -5
  40. package/dist/chunk-HLJX2FTL.js.map +1 -0
  41. package/dist/chunk-I2DSLOXZ.js +75 -0
  42. package/dist/chunk-I2DSLOXZ.js.map +1 -0
  43. package/dist/{chunk-SEBPPFUW.js → chunk-I4TPKIYX.js} +33 -18
  44. package/dist/chunk-I4TPKIYX.js.map +1 -0
  45. package/dist/{chunk-NGBFJJ7Q.js → chunk-IMDW5ZUP.js} +19 -11
  46. package/dist/chunk-IMDW5ZUP.js.map +1 -0
  47. package/dist/{chunk-U4VYHKPM.js → chunk-JIPWV2FX.js} +6 -6
  48. package/dist/chunk-JIPWV2FX.js.map +1 -0
  49. package/dist/{chunk-G2FBJOZG.js → chunk-K7EVM5LY.js} +4 -4
  50. package/dist/chunk-K7EVM5LY.js.map +1 -0
  51. package/dist/{chunk-MJSFR562.js → chunk-KAAEN2PO.js} +3 -3
  52. package/dist/chunk-KAAEN2PO.js.map +1 -0
  53. package/dist/{chunk-FNOYEXUE.js → chunk-LE6IY6IT.js} +22 -17
  54. package/dist/chunk-LE6IY6IT.js.map +1 -0
  55. package/dist/{chunk-CIY5UBRB.js → chunk-NDY25DTL.js} +6 -6
  56. package/dist/chunk-NDY25DTL.js.map +1 -0
  57. package/dist/{chunk-OLBOTK3O.js → chunk-P26A2K5N.js} +7 -7
  58. package/dist/chunk-P26A2K5N.js.map +1 -0
  59. package/dist/{chunk-DHFFRMF6.js → chunk-RTE64SJA.js} +2 -2
  60. package/dist/chunk-RTE64SJA.js.map +1 -0
  61. package/dist/{chunk-3ZCB5K33.js → chunk-SGJ23HIP.js} +14 -11
  62. package/dist/chunk-SGJ23HIP.js.map +1 -0
  63. package/dist/{chunk-C4ZTIYFR.js → chunk-SVYPSDWY.js} +10 -10
  64. package/dist/chunk-SVYPSDWY.js.map +1 -0
  65. package/dist/{chunk-MMDXNZPF.js → chunk-TKFPKEV3.js} +2 -2
  66. package/dist/chunk-TKFPKEV3.js.map +1 -0
  67. package/dist/{chunk-M7YCPFIX.js → chunk-UQTEPWU7.js} +2 -2
  68. package/dist/chunk-UQTEPWU7.js.map +1 -0
  69. package/dist/{chunk-QESUUPOE.js → chunk-V46GGCCI.js} +80 -27
  70. package/dist/chunk-V46GGCCI.js.map +1 -0
  71. package/dist/{chunk-UOWHJ6BE.js → chunk-X6QONICW.js} +6 -3
  72. package/dist/chunk-X6QONICW.js.map +1 -0
  73. package/dist/chunk-Y2LW7R3Y.js +23 -0
  74. package/dist/chunk-Y2LW7R3Y.js.map +1 -0
  75. package/dist/chunk-Z2CWURZ6.js +78 -0
  76. package/dist/chunk-Z2CWURZ6.js.map +1 -0
  77. package/dist/chunk-Z7G4B5HJ.js +377 -0
  78. package/dist/chunk-Z7G4B5HJ.js.map +1 -0
  79. package/dist/{chunk-XKEG3SCV.js → chunk-ZENYS6KW.js} +13 -9
  80. package/dist/chunk-ZENYS6KW.js.map +1 -0
  81. package/dist/{cli-v2 → cli}/defaults/agent-mappings.yaml +5 -5
  82. package/dist/commands/build/marketplace.js +9 -9
  83. package/dist/commands/build/marketplace.js.map +1 -1
  84. package/dist/commands/build/plugins.js +12 -12
  85. package/dist/commands/build/plugins.js.map +1 -1
  86. package/dist/commands/build/stack.js +15 -15
  87. package/dist/commands/build/stack.js.map +1 -1
  88. package/dist/commands/compile.js +21 -21
  89. package/dist/commands/compile.js.map +1 -1
  90. package/dist/commands/config/get.js +9 -8
  91. package/dist/commands/config/get.js.map +1 -1
  92. package/dist/commands/config/index.js +7 -6
  93. package/dist/commands/config/index.js.map +1 -1
  94. package/dist/commands/config/path.js +8 -7
  95. package/dist/commands/config/path.js.map +1 -1
  96. package/dist/commands/config/set-project.js +9 -8
  97. package/dist/commands/config/set-project.js.map +1 -1
  98. package/dist/commands/config/set.js +9 -8
  99. package/dist/commands/config/set.js.map +1 -1
  100. package/dist/commands/config/show.js +6 -5
  101. package/dist/commands/config/unset-project.js +9 -8
  102. package/dist/commands/config/unset-project.js.map +1 -1
  103. package/dist/commands/config/unset.js +9 -8
  104. package/dist/commands/config/unset.js.map +1 -1
  105. package/dist/commands/diff.js +12 -12
  106. package/dist/commands/diff.js.map +1 -1
  107. package/dist/commands/doctor.js +10 -10
  108. package/dist/commands/doctor.js.map +1 -1
  109. package/dist/commands/edit.js +52 -49
  110. package/dist/commands/edit.js.map +1 -1
  111. package/dist/commands/eject.js +180 -97
  112. package/dist/commands/eject.js.map +1 -1
  113. package/dist/commands/import/skill.js +339 -0
  114. package/dist/commands/import/skill.js.map +1 -0
  115. package/dist/commands/info.js +9 -9
  116. package/dist/commands/info.js.map +1 -1
  117. package/dist/commands/init.js +205 -78
  118. package/dist/commands/init.js.map +1 -1
  119. package/dist/commands/list.js +9 -9
  120. package/dist/commands/list.js.map +1 -1
  121. package/dist/commands/new/agent.js +19 -21
  122. package/dist/commands/new/agent.js.map +1 -1
  123. package/dist/commands/new/skill.js +11 -12
  124. package/dist/commands/new/skill.js.map +1 -1
  125. package/dist/commands/outdated.js +12 -12
  126. package/dist/commands/outdated.js.map +1 -1
  127. package/dist/commands/search.js +205 -17
  128. package/dist/commands/search.js.map +1 -1
  129. package/dist/commands/test-imports.js +18 -18
  130. package/dist/commands/test-imports.js.map +1 -1
  131. package/dist/commands/uninstall.js +58 -78
  132. package/dist/commands/uninstall.js.map +1 -1
  133. package/dist/commands/update.js +21 -21
  134. package/dist/commands/update.js.map +1 -1
  135. package/dist/commands/validate.js +9 -9
  136. package/dist/commands/validate.js.map +1 -1
  137. package/dist/commands/version/bump.js +8 -8
  138. package/dist/commands/version/bump.js.map +1 -1
  139. package/dist/commands/version/index.js +8 -8
  140. package/dist/commands/version/index.js.map +1 -1
  141. package/dist/commands/version/set.js +7 -7
  142. package/dist/commands/version/set.js.map +1 -1
  143. package/dist/commands/version/show.js +8 -8
  144. package/dist/commands/version/show.js.map +1 -1
  145. package/dist/components/common/confirm.js +1 -1
  146. package/dist/components/common/message.js +1 -1
  147. package/dist/components/common/message.js.map +1 -1
  148. package/dist/components/common/spinner.js +1 -1
  149. package/dist/components/common/spinner.js.map +1 -1
  150. package/dist/components/skill-search/skill-search.js +10 -0
  151. package/dist/components/skill-search/skill-search.js.map +1 -0
  152. package/dist/components/wizard/category-grid.js +1 -1
  153. package/dist/components/wizard/category-grid.test.js +213 -80
  154. package/dist/components/wizard/category-grid.test.js.map +1 -1
  155. package/dist/components/wizard/section-progress.js +1 -1
  156. package/dist/components/wizard/section-progress.test.js +2 -2
  157. package/dist/components/wizard/section-progress.test.js.map +1 -1
  158. package/dist/components/wizard/step-approach.js +4 -3
  159. package/dist/components/wizard/step-build.js +4 -3
  160. package/dist/components/wizard/step-build.test.js +29 -17
  161. package/dist/components/wizard/step-build.test.js.map +1 -1
  162. package/dist/components/wizard/step-confirm.js +2 -1
  163. package/dist/components/wizard/step-refine.js +2 -1
  164. package/dist/components/wizard/step-refine.test.js +4 -3
  165. package/dist/components/wizard/step-refine.test.js.map +1 -1
  166. package/dist/components/wizard/step-stack-options.js +3 -3
  167. package/dist/components/wizard/step-stack.js +3 -3
  168. package/dist/components/wizard/wizard-footer.js +9 -0
  169. package/dist/components/wizard/wizard-footer.js.map +1 -0
  170. package/dist/components/wizard/wizard-tabs.js +1 -1
  171. package/dist/components/wizard/wizard.js +14 -12
  172. package/dist/hooks/init.js +5 -4
  173. package/dist/hooks/init.js.map +1 -1
  174. package/dist/index.js +1 -1
  175. package/dist/index.js.map +1 -1
  176. package/dist/stores/wizard-store.js +2 -2
  177. package/dist/stores/wizard-store.test.js +3 -3
  178. package/dist/stores/wizard-store.test.js.map +1 -1
  179. package/package.json +3 -2
  180. package/dist/chunk-3HBTELJN.js.map +0 -1
  181. package/dist/chunk-3ZCB5K33.js.map +0 -1
  182. package/dist/chunk-6LS7XO3H.js.map +0 -1
  183. package/dist/chunk-A3J6IAXK.js.map +0 -1
  184. package/dist/chunk-A65SBAAJ.js.map +0 -1
  185. package/dist/chunk-ALEPJ6YN.js +0 -80
  186. package/dist/chunk-ALEPJ6YN.js.map +0 -1
  187. package/dist/chunk-C4ZTIYFR.js.map +0 -1
  188. package/dist/chunk-CIY5UBRB.js.map +0 -1
  189. package/dist/chunk-DHFFRMF6.js.map +0 -1
  190. package/dist/chunk-DKGL77IY.js.map +0 -1
  191. package/dist/chunk-ED73HCW2.js.map +0 -1
  192. package/dist/chunk-FNOYEXUE.js.map +0 -1
  193. package/dist/chunk-G2FBJOZG.js.map +0 -1
  194. package/dist/chunk-HNDT5QRB.js.map +0 -1
  195. package/dist/chunk-K7PTOVX4.js.map +0 -1
  196. package/dist/chunk-LQTST4WY.js.map +0 -1
  197. package/dist/chunk-LVKRVFYR.js.map +0 -1
  198. package/dist/chunk-M7YCPFIX.js.map +0 -1
  199. package/dist/chunk-MJSFR562.js.map +0 -1
  200. package/dist/chunk-MMDXNZPF.js.map +0 -1
  201. package/dist/chunk-MYAVQ23U.js.map +0 -1
  202. package/dist/chunk-NGBFJJ7Q.js.map +0 -1
  203. package/dist/chunk-OLBOTK3O.js.map +0 -1
  204. package/dist/chunk-PPNTD5LO.js +0 -330
  205. package/dist/chunk-PPNTD5LO.js.map +0 -1
  206. package/dist/chunk-Q2LH2DAB.js.map +0 -1
  207. package/dist/chunk-Q6DR5QUH.js.map +0 -1
  208. package/dist/chunk-QESUUPOE.js.map +0 -1
  209. package/dist/chunk-QGGSLMO3.js.map +0 -1
  210. package/dist/chunk-SEBPPFUW.js.map +0 -1
  211. package/dist/chunk-SYQ7R2JO.js.map +0 -1
  212. package/dist/chunk-TOPAIL5W.js.map +0 -1
  213. package/dist/chunk-U4VYHKPM.js.map +0 -1
  214. package/dist/chunk-UOWHJ6BE.js.map +0 -1
  215. package/dist/chunk-XKEG3SCV.js.map +0 -1
  216. package/dist/chunk-Y3V43XCU.js.map +0 -1
  217. package/dist/chunk-YKXBGCFD.js.map +0 -1
@@ -1,27 +1,30 @@
1
1
  #!/usr/bin/env node
2
- import {
3
- loadProjectConfig
4
- } from "./chunk-MYAVQ23U.js";
5
2
  import {
6
3
  getCollectivePluginDir
7
- } from "./chunk-3HBTELJN.js";
4
+ } from "./chunk-ED4E6Q2T.js";
8
5
  import {
9
- CLAUDE_DIR
10
- } from "./chunk-A3J6IAXK.js";
6
+ loadProjectConfig
7
+ } from "./chunk-CJEHB4TB.js";
11
8
  import {
12
9
  directoryExists,
13
10
  fileExists
14
- } from "./chunk-MMDXNZPF.js";
11
+ } from "./chunk-TKFPKEV3.js";
12
+ import {
13
+ CLAUDE_DIR,
14
+ CLAUDE_SRC_DIR
15
+ } from "./chunk-76DWXGQE.js";
15
16
  import {
16
17
  init_esm_shims
17
18
  } from "./chunk-DHET7RCE.js";
18
19
 
19
- // src/cli-v2/lib/installation.ts
20
+ // src/cli/lib/installation.ts
20
21
  init_esm_shims();
21
22
  import path from "path";
22
23
  async function detectInstallation(projectDir = process.cwd()) {
23
- const localConfigPath = path.join(projectDir, CLAUDE_DIR, "config.yaml");
24
- if (await fileExists(localConfigPath)) {
24
+ const srcConfigPath = path.join(projectDir, CLAUDE_SRC_DIR, "config.yaml");
25
+ const legacyConfigPath = path.join(projectDir, CLAUDE_DIR, "config.yaml");
26
+ const localConfigPath = await fileExists(srcConfigPath) ? srcConfigPath : await fileExists(legacyConfigPath) ? legacyConfigPath : null;
27
+ if (localConfigPath) {
25
28
  const loaded = await loadProjectConfig(projectDir);
26
29
  const mode = loaded?.config?.installMode ?? "local";
27
30
  if (mode === "local") {
@@ -51,4 +54,4 @@ async function detectInstallation(projectDir = process.cwd()) {
51
54
  export {
52
55
  detectInstallation
53
56
  };
54
- //# sourceMappingURL=chunk-3ZCB5K33.js.map
57
+ //# sourceMappingURL=chunk-SGJ23HIP.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/cli/lib/installation.ts"],"sourcesContent":["/**\n * Installation detection utilities for Claude Collective.\n *\n * Detects whether a project uses local mode (.claude-src/config.yaml) or\n * plugin mode (.claude/plugins/claude-collective/).\n */\nimport path from \"path\";\nimport { directoryExists, fileExists } from \"../utils/fs\";\nimport { loadProjectConfig } from \"./project-config\";\nimport { getCollectivePluginDir } from \"./plugin-finder\";\nimport { CLAUDE_DIR, CLAUDE_SRC_DIR } from \"../consts\";\n\nexport type InstallMode = \"local\" | \"plugin\";\n\nexport interface Installation {\n mode: InstallMode;\n configPath: string;\n agentsDir: string;\n skillsDir: string;\n projectDir: string;\n}\n\n/**\n * Detect the current installation mode by checking for local config first.\n * Priority: Local (.claude-src/config.yaml with installMode: local) > Plugin\n */\nexport async function detectInstallation(\n projectDir: string = process.cwd(),\n): Promise<Installation | null> {\n // 1. Check for local installation first\n // Check .claude-src/config.yaml first (new location)\n const srcConfigPath = path.join(projectDir, CLAUDE_SRC_DIR, \"config.yaml\");\n // Fall back to .claude/config.yaml (legacy location)\n const legacyConfigPath = path.join(projectDir, CLAUDE_DIR, \"config.yaml\");\n\n const localConfigPath = (await fileExists(srcConfigPath))\n ? srcConfigPath\n : (await fileExists(legacyConfigPath))\n ? legacyConfigPath\n : null;\n\n if (localConfigPath) {\n const loaded = await loadProjectConfig(projectDir);\n\n // If config exists and has installMode: local (or no installMode, defaults to local)\n // treat it as local mode\n const mode: InstallMode = loaded?.config?.installMode ?? \"local\";\n\n if (mode === \"local\") {\n return {\n mode: \"local\",\n configPath: localConfigPath,\n agentsDir: path.join(projectDir, CLAUDE_DIR, \"agents\"),\n skillsDir: path.join(projectDir, CLAUDE_DIR, \"skills\"),\n projectDir,\n };\n }\n }\n\n // 2. Check for plugin installation\n const pluginDir = getCollectivePluginDir(projectDir);\n const pluginConfigPath = path.join(pluginDir, \"config.yaml\");\n\n if (await directoryExists(pluginDir)) {\n return {\n mode: \"plugin\",\n configPath: pluginConfigPath,\n agentsDir: path.join(pluginDir, \"agents\"),\n skillsDir: path.join(pluginDir, \"skills\"),\n projectDir,\n };\n }\n\n // No installation found\n return null;\n}\n\n/**\n * Get installation or throw with helpful error message\n */\nexport async function getInstallationOrThrow(\n projectDir: string = process.cwd(),\n): Promise<Installation> {\n const installation = await detectInstallation(projectDir);\n\n if (!installation) {\n throw new Error(\n \"No Claude Collective installation found.\\n\" +\n \"Run 'cc init' to create one.\",\n );\n }\n\n return installation;\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;AAAA;AAMA,OAAO,UAAU;AAoBjB,eAAsB,mBACpB,aAAqB,QAAQ,IAAI,GACH;AAG9B,QAAM,gBAAgB,KAAK,KAAK,YAAY,gBAAgB,aAAa;AAEzE,QAAM,mBAAmB,KAAK,KAAK,YAAY,YAAY,aAAa;AAExE,QAAM,kBAAmB,MAAM,WAAW,aAAa,IACnD,gBACC,MAAM,WAAW,gBAAgB,IAChC,mBACA;AAEN,MAAI,iBAAiB;AACnB,UAAM,SAAS,MAAM,kBAAkB,UAAU;AAIjD,UAAM,OAAoB,QAAQ,QAAQ,eAAe;AAEzD,QAAI,SAAS,SAAS;AACpB,aAAO;AAAA,QACL,MAAM;AAAA,QACN,YAAY;AAAA,QACZ,WAAW,KAAK,KAAK,YAAY,YAAY,QAAQ;AAAA,QACrD,WAAW,KAAK,KAAK,YAAY,YAAY,QAAQ;AAAA,QACrD;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAGA,QAAM,YAAY,uBAAuB,UAAU;AACnD,QAAM,mBAAmB,KAAK,KAAK,WAAW,aAAa;AAE3D,MAAI,MAAM,gBAAgB,SAAS,GAAG;AACpC,WAAO;AAAA,MACL,MAAM;AAAA,MACN,YAAY;AAAA,MACZ,WAAW,KAAK,KAAK,WAAW,QAAQ;AAAA,MACxC,WAAW,KAAK,KAAK,WAAW,QAAQ;AAAA,MACxC;AAAA,IACF;AAAA,EACF;AAGA,SAAO;AACT;","names":[]}
@@ -1,23 +1,23 @@
1
1
  #!/usr/bin/env node
2
2
  import {
3
3
  fetchFromSource
4
- } from "./chunk-NGBFJJ7Q.js";
5
- import {
6
- CLAUDE_DIR,
7
- DIRS,
8
- PROJECT_ROOT
9
- } from "./chunk-A3J6IAXK.js";
4
+ } from "./chunk-IMDW5ZUP.js";
10
5
  import {
11
6
  verbose
12
- } from "./chunk-TOPAIL5W.js";
7
+ } from "./chunk-3U3R4NCG.js";
13
8
  import {
14
9
  directoryExists
15
- } from "./chunk-MMDXNZPF.js";
10
+ } from "./chunk-TKFPKEV3.js";
11
+ import {
12
+ CLAUDE_DIR,
13
+ DIRS,
14
+ PROJECT_ROOT
15
+ } from "./chunk-76DWXGQE.js";
16
16
  import {
17
17
  init_esm_shims
18
18
  } from "./chunk-DHET7RCE.js";
19
19
 
20
- // src/cli-v2/lib/agent-fetcher.ts
20
+ // src/cli/lib/agent-fetcher.ts
21
21
  init_esm_shims();
22
22
  import path from "path";
23
23
  async function getAgentDefinitions(remoteSource, options = {}) {
@@ -81,4 +81,4 @@ async function fetchAgentDefinitionsFromRemote(source, options = {}) {
81
81
  export {
82
82
  getAgentDefinitions
83
83
  };
84
- //# sourceMappingURL=chunk-C4ZTIYFR.js.map
84
+ //# sourceMappingURL=chunk-SVYPSDWY.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/cli/lib/agent-fetcher.ts"],"sourcesContent":["import path from \"path\";\nimport { directoryExists } from \"../utils/fs\";\nimport { verbose } from \"../utils/logger\";\nimport { PROJECT_ROOT, DIRS, CLAUDE_DIR } from \"../consts\";\nimport { fetchFromSource, type FetchOptions } from \"./source-fetcher\";\nimport type { AgentSourcePaths } from \"../../types\";\n\nexport interface AgentDefinitionOptions extends FetchOptions {\n /** Project directory to check for local templates */\n projectDir?: string;\n}\n\nexport async function getAgentDefinitions(\n remoteSource?: string,\n options: AgentDefinitionOptions = {},\n): Promise<AgentSourcePaths> {\n if (remoteSource) {\n return fetchAgentDefinitionsFromRemote(remoteSource, options);\n }\n return getLocalAgentDefinitions(options);\n}\n\nexport async function getLocalAgentDefinitions(\n options: AgentDefinitionOptions = {},\n): Promise<AgentSourcePaths> {\n const agentsDir = path.join(PROJECT_ROOT, DIRS.agents);\n let templatesDir = path.join(PROJECT_ROOT, DIRS.templates);\n\n if (!(await directoryExists(agentsDir))) {\n throw new Error(\n `Agent partials not found at: ${agentsDir}. ` +\n `Ensure the CLI is properly installed.`,\n );\n }\n\n // Check for local templates first (from eject templates)\n if (options.projectDir) {\n const localTemplatesDir = path.join(\n options.projectDir,\n CLAUDE_DIR,\n \"templates\",\n );\n if (await directoryExists(localTemplatesDir)) {\n verbose(`Using local templates from: ${localTemplatesDir}`);\n templatesDir = localTemplatesDir;\n }\n }\n\n if (!(await directoryExists(templatesDir))) {\n verbose(`Templates directory not found: ${templatesDir}`);\n }\n\n verbose(`Agent partials loaded from CLI: ${agentsDir}`);\n verbose(`Templates directory: ${templatesDir}`);\n\n return {\n agentsDir,\n templatesDir,\n sourcePath: PROJECT_ROOT,\n };\n}\n\nexport async function fetchAgentDefinitionsFromRemote(\n source: string,\n options: FetchOptions = {},\n): Promise<AgentSourcePaths> {\n verbose(`Fetching agent partials from remote: ${source}`);\n\n const result = await fetchFromSource(source, {\n forceRefresh: options.forceRefresh,\n subdir: \"\",\n });\n\n const agentsDir = path.join(result.path, \"src\", \"agents\");\n const templatesDir = path.join(agentsDir, \"_templates\");\n\n if (!(await directoryExists(agentsDir))) {\n throw new Error(`Agent partials not found at: ${agentsDir}`);\n }\n\n if (!(await directoryExists(templatesDir))) {\n verbose(`Templates directory not found: ${templatesDir}`);\n }\n\n verbose(`Agent partials fetched from: ${result.path}`);\n\n return {\n agentsDir,\n templatesDir,\n sourcePath: result.path,\n };\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;AAAA;AAAA,OAAO,UAAU;AAYjB,eAAsB,oBACpB,cACA,UAAkC,CAAC,GACR;AAC3B,MAAI,cAAc;AAChB,WAAO,gCAAgC,cAAc,OAAO;AAAA,EAC9D;AACA,SAAO,yBAAyB,OAAO;AACzC;AAEA,eAAsB,yBACpB,UAAkC,CAAC,GACR;AAC3B,QAAM,YAAY,KAAK,KAAK,cAAc,KAAK,MAAM;AACrD,MAAI,eAAe,KAAK,KAAK,cAAc,KAAK,SAAS;AAEzD,MAAI,CAAE,MAAM,gBAAgB,SAAS,GAAI;AACvC,UAAM,IAAI;AAAA,MACR,gCAAgC,SAAS;AAAA,IAE3C;AAAA,EACF;AAGA,MAAI,QAAQ,YAAY;AACtB,UAAM,oBAAoB,KAAK;AAAA,MAC7B,QAAQ;AAAA,MACR;AAAA,MACA;AAAA,IACF;AACA,QAAI,MAAM,gBAAgB,iBAAiB,GAAG;AAC5C,cAAQ,+BAA+B,iBAAiB,EAAE;AAC1D,qBAAe;AAAA,IACjB;AAAA,EACF;AAEA,MAAI,CAAE,MAAM,gBAAgB,YAAY,GAAI;AAC1C,YAAQ,kCAAkC,YAAY,EAAE;AAAA,EAC1D;AAEA,UAAQ,mCAAmC,SAAS,EAAE;AACtD,UAAQ,wBAAwB,YAAY,EAAE;AAE9C,SAAO;AAAA,IACL;AAAA,IACA;AAAA,IACA,YAAY;AAAA,EACd;AACF;AAEA,eAAsB,gCACpB,QACA,UAAwB,CAAC,GACE;AAC3B,UAAQ,wCAAwC,MAAM,EAAE;AAExD,QAAM,SAAS,MAAM,gBAAgB,QAAQ;AAAA,IAC3C,cAAc,QAAQ;AAAA,IACtB,QAAQ;AAAA,EACV,CAAC;AAED,QAAM,YAAY,KAAK,KAAK,OAAO,MAAM,OAAO,QAAQ;AACxD,QAAM,eAAe,KAAK,KAAK,WAAW,YAAY;AAEtD,MAAI,CAAE,MAAM,gBAAgB,SAAS,GAAI;AACvC,UAAM,IAAI,MAAM,gCAAgC,SAAS,EAAE;AAAA,EAC7D;AAEA,MAAI,CAAE,MAAM,gBAAgB,YAAY,GAAI;AAC1C,YAAQ,kCAAkC,YAAY,EAAE;AAAA,EAC1D;AAEA,UAAQ,gCAAgC,OAAO,IAAI,EAAE;AAErD,SAAO;AAAA,IACL;AAAA,IACA;AAAA,IACA,YAAY,OAAO;AAAA,EACrB;AACF;","names":[]}
@@ -3,7 +3,7 @@ import {
3
3
  init_esm_shims
4
4
  } from "./chunk-DHET7RCE.js";
5
5
 
6
- // src/cli-v2/utils/fs.ts
6
+ // src/cli/utils/fs.ts
7
7
  init_esm_shims();
8
8
  import fs from "fs-extra";
9
9
  import fg from "fast-glob";
@@ -66,4 +66,4 @@ export {
66
66
  remove,
67
67
  copy
68
68
  };
69
- //# sourceMappingURL=chunk-MMDXNZPF.js.map
69
+ //# sourceMappingURL=chunk-TKFPKEV3.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/cli/utils/fs.ts"],"sourcesContent":["import fs from \"fs-extra\";\nimport fg from \"fast-glob\";\nimport path from \"path\";\n\nexport async function readFile(filePath: string): Promise<string> {\n return fs.readFile(filePath, \"utf-8\");\n}\n\nexport async function readFileOptional(\n filePath: string,\n fallback = \"\",\n): Promise<string> {\n try {\n return await fs.readFile(filePath, \"utf-8\");\n } catch {\n return fallback;\n }\n}\n\nexport async function fileExists(filePath: string): Promise<boolean> {\n return fs.pathExists(filePath);\n}\n\nexport async function directoryExists(dirPath: string): Promise<boolean> {\n try {\n const stat = await fs.stat(dirPath);\n return stat.isDirectory();\n } catch {\n return false;\n }\n}\n\nexport async function listDirectories(dirPath: string): Promise<string[]> {\n try {\n const entries = await fs.readdir(dirPath, { withFileTypes: true });\n return entries.filter((e) => e.isDirectory()).map((e) => e.name);\n } catch {\n return [];\n }\n}\n\nexport async function glob(pattern: string, cwd: string): Promise<string[]> {\n return fg(pattern, { cwd, onlyFiles: true });\n}\n\nexport async function writeFile(\n filePath: string,\n content: string,\n): Promise<void> {\n await fs.ensureDir(path.dirname(filePath));\n await fs.writeFile(filePath, content, \"utf-8\");\n}\n\nexport async function ensureDir(dirPath: string): Promise<void> {\n await fs.ensureDir(dirPath);\n}\n\nexport async function remove(filePath: string): Promise<void> {\n await fs.remove(filePath);\n}\n\nexport async function copy(src: string, dest: string): Promise<void> {\n await fs.copy(src, dest);\n}\n"],"mappings":";;;;;;AAAA;AAAA,OAAO,QAAQ;AACf,OAAO,QAAQ;AACf,OAAO,UAAU;AAEjB,eAAsB,SAAS,UAAmC;AAChE,SAAO,GAAG,SAAS,UAAU,OAAO;AACtC;AAEA,eAAsB,iBACpB,UACA,WAAW,IACM;AACjB,MAAI;AACF,WAAO,MAAM,GAAG,SAAS,UAAU,OAAO;AAAA,EAC5C,QAAQ;AACN,WAAO;AAAA,EACT;AACF;AAEA,eAAsB,WAAW,UAAoC;AACnE,SAAO,GAAG,WAAW,QAAQ;AAC/B;AAEA,eAAsB,gBAAgB,SAAmC;AACvE,MAAI;AACF,UAAM,OAAO,MAAM,GAAG,KAAK,OAAO;AAClC,WAAO,KAAK,YAAY;AAAA,EAC1B,QAAQ;AACN,WAAO;AAAA,EACT;AACF;AAEA,eAAsB,gBAAgB,SAAoC;AACxE,MAAI;AACF,UAAM,UAAU,MAAM,GAAG,QAAQ,SAAS,EAAE,eAAe,KAAK,CAAC;AACjE,WAAO,QAAQ,OAAO,CAAC,MAAM,EAAE,YAAY,CAAC,EAAE,IAAI,CAAC,MAAM,EAAE,IAAI;AAAA,EACjE,QAAQ;AACN,WAAO,CAAC;AAAA,EACV;AACF;AAEA,eAAsB,KAAK,SAAiB,KAAgC;AAC1E,SAAO,GAAG,SAAS,EAAE,KAAK,WAAW,KAAK,CAAC;AAC7C;AAEA,eAAsB,UACpB,UACA,SACe;AACf,QAAM,GAAG,UAAU,KAAK,QAAQ,QAAQ,CAAC;AACzC,QAAM,GAAG,UAAU,UAAU,SAAS,OAAO;AAC/C;AAEA,eAAsB,UAAU,SAAgC;AAC9D,QAAM,GAAG,UAAU,OAAO;AAC5B;AAEA,eAAsB,OAAO,UAAiC;AAC5D,QAAM,GAAG,OAAO,QAAQ;AAC1B;AAEA,eAAsB,KAAK,KAAa,MAA6B;AACnE,QAAM,GAAG,KAAK,KAAK,IAAI;AACzB;","names":[]}
@@ -3,7 +3,7 @@ import {
3
3
  init_esm_shims
4
4
  } from "./chunk-DHET7RCE.js";
5
5
 
6
- // src/cli-v2/utils/exec.ts
6
+ // src/cli/utils/exec.ts
7
7
  init_esm_shims();
8
8
  import { spawn } from "child_process";
9
9
  async function execCommand(command, args, options) {
@@ -105,4 +105,4 @@ export {
105
105
  claudePluginMarketplaceAdd,
106
106
  claudePluginUninstall
107
107
  };
108
- //# sourceMappingURL=chunk-M7YCPFIX.js.map
108
+ //# sourceMappingURL=chunk-UQTEPWU7.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/cli/utils/exec.ts"],"sourcesContent":["import { spawn } from \"child_process\";\n\nexport interface ExecResult {\n stdout: string;\n stderr: string;\n exitCode: number;\n}\n\n/**\n * Execute a command and return the result\n */\nexport async function execCommand(\n command: string,\n args: string[],\n options?: { cwd?: string; env?: NodeJS.ProcessEnv },\n): Promise<ExecResult> {\n return new Promise((resolve, reject) => {\n const proc = spawn(command, args, {\n cwd: options?.cwd,\n env: { ...process.env, ...options?.env },\n stdio: [\"ignore\", \"pipe\", \"pipe\"],\n });\n\n let stdout = \"\";\n let stderr = \"\";\n\n proc.stdout.on(\"data\", (data) => {\n stdout += data.toString();\n });\n\n proc.stderr.on(\"data\", (data) => {\n stderr += data.toString();\n });\n\n proc.on(\"close\", (code) => {\n resolve({\n stdout,\n stderr,\n exitCode: code ?? 1,\n });\n });\n\n proc.on(\"error\", (err) => {\n reject(err);\n });\n });\n}\n\n/**\n * Install a plugin using the native claude CLI\n */\nexport async function claudePluginInstall(\n pluginPath: string,\n scope: \"project\" | \"user\",\n projectDir: string,\n): Promise<void> {\n const args = [\"plugin\", \"install\", pluginPath, \"--scope\", scope];\n const result = await execCommand(\"claude\", args, { cwd: projectDir });\n\n if (result.exitCode !== 0) {\n const errorMessage = result.stderr || result.stdout || \"Unknown error\";\n throw new Error(`Plugin installation failed: ${errorMessage.trim()}`);\n }\n}\n\n/**\n * Check if the claude CLI is available\n */\nexport async function isClaudeCLIAvailable(): Promise<boolean> {\n try {\n const result = await execCommand(\"claude\", [\"--version\"], {});\n return result.exitCode === 0;\n } catch {\n return false;\n }\n}\n\nexport interface MarketplaceInfo {\n name: string;\n source: string;\n repo?: string;\n path?: string;\n}\n\n/**\n * List configured marketplaces in Claude Code\n */\nexport async function claudePluginMarketplaceList(): Promise<\n MarketplaceInfo[]\n> {\n try {\n const result = await execCommand(\n \"claude\",\n [\"plugin\", \"marketplace\", \"list\", \"--json\"],\n {},\n );\n\n if (result.exitCode !== 0) {\n return [];\n }\n\n return JSON.parse(result.stdout);\n } catch {\n // Returns empty array if claude CLI is not available or parsing fails\n return [];\n }\n}\n\n/**\n * Check if a marketplace with the given name exists\n */\nexport async function claudePluginMarketplaceExists(\n name: string,\n): Promise<boolean> {\n const marketplaces = await claudePluginMarketplaceList();\n return marketplaces.some((m) => m.name === name);\n}\n\n/**\n * Add a marketplace to Claude Code from a GitHub repository\n */\nexport async function claudePluginMarketplaceAdd(\n githubRepo: string,\n name: string,\n): Promise<void> {\n const args = [\"plugin\", \"marketplace\", \"add\", githubRepo, \"--name\", name];\n let result;\n try {\n result = await execCommand(\"claude\", args, {});\n } catch (err) {\n throw new Error(\n `Failed to add marketplace: ${err instanceof Error ? err.message : \"Unknown error\"}`,\n );\n }\n\n if (result.exitCode !== 0) {\n const errorMessage = result.stderr || result.stdout || \"Unknown error\";\n if (errorMessage.includes(\"already installed\")) {\n return;\n }\n throw new Error(`Failed to add marketplace: ${errorMessage.trim()}`);\n }\n}\n\n/**\n * Uninstall a plugin using the native claude CLI\n */\nexport async function claudePluginUninstall(\n pluginName: string,\n scope: \"project\" | \"user\",\n projectDir: string,\n): Promise<void> {\n const args = [\"plugin\", \"uninstall\", pluginName, \"--scope\", scope];\n const result = await execCommand(\"claude\", args, { cwd: projectDir });\n\n if (result.exitCode !== 0) {\n const errorMessage = result.stderr || result.stdout || \"Unknown error\";\n // Ignore \"not installed\" errors - plugin may already be removed\n if (\n errorMessage.includes(\"not installed\") ||\n errorMessage.includes(\"not found\")\n ) {\n return;\n }\n throw new Error(`Plugin uninstall failed: ${errorMessage.trim()}`);\n }\n}\n"],"mappings":"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","names":[]}
@@ -1,23 +1,26 @@
1
1
  #!/usr/bin/env node
2
2
  import {
3
3
  verbose
4
- } from "./chunk-TOPAIL5W.js";
4
+ } from "./chunk-3U3R4NCG.js";
5
5
  import {
6
6
  ensureDir,
7
7
  fileExists,
8
8
  readFile,
9
9
  writeFile
10
- } from "./chunk-MMDXNZPF.js";
10
+ } from "./chunk-TKFPKEV3.js";
11
+ import {
12
+ CLAUDE_DIR,
13
+ CLAUDE_SRC_DIR
14
+ } from "./chunk-76DWXGQE.js";
11
15
  import {
12
16
  init_esm_shims
13
17
  } from "./chunk-DHET7RCE.js";
14
18
 
15
- // src/cli-v2/lib/config.ts
19
+ // src/cli/lib/config.ts
16
20
  init_esm_shims();
17
21
  import path from "path";
18
22
  import os from "os";
19
23
  import { parse as parseYaml, stringify as stringifyYaml } from "yaml";
20
- var PROJECT_CONFIG_DIR = ".claude-collective";
21
24
  var DEFAULT_SOURCE = "github:claude-collective/skills";
22
25
  var SOURCE_ENV_VAR = "CC_SOURCE";
23
26
  var CONFIG_HOME_ENV_VAR = "CC_CONFIG_HOME";
@@ -27,6 +30,19 @@ function getGlobalConfigDir() {
27
30
  return process.env[CONFIG_HOME_ENV_VAR] || path.join(os.homedir(), ".claude-collective");
28
31
  }
29
32
  var GLOBAL_CONFIG_DIR = path.join(os.homedir(), ".claude-collective");
33
+ function isValidSourceEntry(entry) {
34
+ if (typeof entry !== "object" || entry === null) return false;
35
+ const e = entry;
36
+ if (typeof e.name !== "string" || typeof e.url !== "string") return false;
37
+ if (e.description !== void 0 && typeof e.description !== "string")
38
+ return false;
39
+ if (e.ref !== void 0 && typeof e.ref !== "string") return false;
40
+ return true;
41
+ }
42
+ function isValidSourcesArray(arr) {
43
+ if (!Array.isArray(arr)) return false;
44
+ return arr.every(isValidSourceEntry);
45
+ }
30
46
  function isValidGlobalConfig(obj) {
31
47
  if (typeof obj !== "object" || obj === null) return false;
32
48
  const config = obj;
@@ -38,6 +54,8 @@ function isValidGlobalConfig(obj) {
38
54
  return false;
39
55
  if (config.agents_source !== void 0 && typeof config.agents_source !== "string")
40
56
  return false;
57
+ if (config.sources !== void 0 && !isValidSourcesArray(config.sources))
58
+ return false;
41
59
  return true;
42
60
  }
43
61
  function isValidProjectConfig(obj) {
@@ -45,17 +63,21 @@ function isValidProjectConfig(obj) {
45
63
  const config = obj;
46
64
  if (config.source !== void 0 && typeof config.source !== "string")
47
65
  return false;
66
+ if (config.author !== void 0 && typeof config.author !== "string")
67
+ return false;
48
68
  if (config.marketplace !== void 0 && typeof config.marketplace !== "string")
49
69
  return false;
50
70
  if (config.agents_source !== void 0 && typeof config.agents_source !== "string")
51
71
  return false;
72
+ if (config.sources !== void 0 && !isValidSourcesArray(config.sources))
73
+ return false;
52
74
  return true;
53
75
  }
54
76
  function getGlobalConfigPath() {
55
77
  return path.join(getGlobalConfigDir(), GLOBAL_CONFIG_FILE);
56
78
  }
57
79
  function getProjectConfigPath(projectDir) {
58
- return path.join(projectDir, PROJECT_CONFIG_DIR, PROJECT_CONFIG_FILE);
80
+ return path.join(projectDir, CLAUDE_SRC_DIR, PROJECT_CONFIG_FILE);
59
81
  }
60
82
  async function loadGlobalConfig() {
61
83
  const configPath = getGlobalConfigPath();
@@ -78,10 +100,19 @@ async function loadGlobalConfig() {
78
100
  }
79
101
  }
80
102
  async function loadProjectConfig(projectDir) {
81
- const configPath = getProjectConfigPath(projectDir);
82
- if (!await fileExists(configPath)) {
83
- verbose(`Project config not found at ${configPath}`);
84
- return null;
103
+ const srcConfigPath = getProjectConfigPath(projectDir);
104
+ const legacyConfigPath = path.join(projectDir, CLAUDE_DIR, "config.yaml");
105
+ let configPath = srcConfigPath;
106
+ if (!await fileExists(srcConfigPath)) {
107
+ if (await fileExists(legacyConfigPath)) {
108
+ configPath = legacyConfigPath;
109
+ verbose(`Using legacy config location: ${legacyConfigPath}`);
110
+ } else {
111
+ verbose(
112
+ `Project config not found at ${srcConfigPath} or ${legacyConfigPath}`
113
+ );
114
+ return null;
115
+ }
85
116
  }
86
117
  try {
87
118
  const content = await readFile(configPath);
@@ -106,15 +137,14 @@ async function saveGlobalConfig(config) {
106
137
  }
107
138
  async function saveProjectConfig(projectDir, config) {
108
139
  const configPath = getProjectConfigPath(projectDir);
109
- await ensureDir(path.join(projectDir, PROJECT_CONFIG_DIR));
140
+ await ensureDir(path.join(projectDir, CLAUDE_SRC_DIR));
110
141
  const content = stringifyYaml(config, { lineWidth: 0 });
111
142
  await writeFile(configPath, content);
112
143
  verbose(`Saved project config to ${configPath}`);
113
144
  }
114
145
  async function resolveSource(flagValue, projectDir) {
115
146
  const projectConfig = projectDir ? await loadProjectConfig(projectDir) : null;
116
- const globalConfig = await loadGlobalConfig();
117
- const marketplace = projectConfig?.marketplace || globalConfig?.marketplace;
147
+ const marketplace = projectConfig?.marketplace;
118
148
  if (flagValue !== void 0) {
119
149
  if (flagValue === "" || flagValue.trim() === "") {
120
150
  throw new Error("--source flag cannot be empty");
@@ -135,10 +165,6 @@ async function resolveSource(flagValue, projectDir) {
135
165
  marketplace
136
166
  };
137
167
  }
138
- if (globalConfig?.source) {
139
- verbose(`Source from global config: ${globalConfig.source}`);
140
- return { source: globalConfig.source, sourceOrigin: "global", marketplace };
141
- }
142
168
  verbose(`Using default source: ${DEFAULT_SOURCE}`);
143
169
  return { source: DEFAULT_SOURCE, sourceOrigin: "default", marketplace };
144
170
  }
@@ -160,14 +186,6 @@ async function resolveAgentsSource(flagValue, projectDir) {
160
186
  agentsSourceOrigin: "project"
161
187
  };
162
188
  }
163
- const globalConfig = await loadGlobalConfig();
164
- if (globalConfig?.agents_source) {
165
- verbose(`Agents source from global config: ${globalConfig.agents_source}`);
166
- return {
167
- agentsSource: globalConfig.agents_source,
168
- agentsSourceOrigin: "global"
169
- };
170
- }
171
189
  verbose("Using default agents source (local CLI)");
172
190
  return { agentsSource: void 0, agentsSourceOrigin: "default" };
173
191
  }
@@ -176,13 +194,17 @@ function formatAgentsSourceOrigin(origin) {
176
194
  case "flag":
177
195
  return "--agent-source flag";
178
196
  case "project":
179
- return "project config (.claude-collective/config.yaml)";
197
+ return "project config (.claude-src/config.yaml)";
180
198
  case "global":
181
199
  return "global config (~/.claude-collective/config.yaml)";
182
200
  case "default":
183
201
  return "default (local CLI)";
184
202
  }
185
203
  }
204
+ async function resolveAuthor(projectDir) {
205
+ const projectConfig = projectDir ? await loadProjectConfig(projectDir) : null;
206
+ return projectConfig?.author;
207
+ }
186
208
  function formatSourceOrigin(origin) {
187
209
  switch (origin) {
188
210
  case "flag":
@@ -190,13 +212,42 @@ function formatSourceOrigin(origin) {
190
212
  case "env":
191
213
  return `${SOURCE_ENV_VAR} environment variable`;
192
214
  case "project":
193
- return "project config (.claude-collective/config.yaml)";
215
+ return "project config (.claude-src/config.yaml)";
194
216
  case "global":
195
217
  return "global config (~/.claude-collective/config.yaml)";
196
218
  case "default":
197
219
  return "default";
198
220
  }
199
221
  }
222
+ async function resolveAllSources(projectDir) {
223
+ const projectConfig = projectDir ? await loadProjectConfig(projectDir) : null;
224
+ const globalConfig = await loadGlobalConfig();
225
+ const resolvedConfig = await resolveSource(void 0, projectDir);
226
+ const primary = {
227
+ name: "marketplace",
228
+ url: resolvedConfig.source,
229
+ description: "Primary skills marketplace"
230
+ };
231
+ const extras = [];
232
+ const seenNames = /* @__PURE__ */ new Set();
233
+ if (projectConfig?.sources) {
234
+ for (const source of projectConfig.sources) {
235
+ if (!seenNames.has(source.name)) {
236
+ seenNames.add(source.name);
237
+ extras.push(source);
238
+ }
239
+ }
240
+ }
241
+ if (globalConfig?.sources) {
242
+ for (const source of globalConfig.sources) {
243
+ if (!seenNames.has(source.name)) {
244
+ seenNames.add(source.name);
245
+ extras.push(source);
246
+ }
247
+ }
248
+ }
249
+ return { primary, extras };
250
+ }
200
251
  function isLocalSource(source) {
201
252
  if (source.startsWith("/") || source.startsWith(".")) {
202
253
  return true;
@@ -235,7 +286,9 @@ export {
235
286
  resolveSource,
236
287
  resolveAgentsSource,
237
288
  formatAgentsSourceOrigin,
289
+ resolveAuthor,
238
290
  formatSourceOrigin,
291
+ resolveAllSources,
239
292
  isLocalSource
240
293
  };
241
- //# sourceMappingURL=chunk-QESUUPOE.js.map
294
+ //# sourceMappingURL=chunk-V46GGCCI.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/cli/lib/config.ts"],"sourcesContent":["import path from \"path\";\nimport os from \"os\";\nimport { parse as parseYaml, stringify as stringifyYaml } from \"yaml\";\nimport { readFile, writeFile, fileExists, ensureDir } from \"../utils/fs\";\nimport { verbose } from \"../utils/logger\";\nimport { CLAUDE_DIR, CLAUDE_SRC_DIR } from \"../consts\";\n\nexport const DEFAULT_SOURCE = \"github:claude-collective/skills\";\nexport const SOURCE_ENV_VAR = \"CC_SOURCE\";\n/** Environment variable to override global config directory (used for testing) */\nexport const CONFIG_HOME_ENV_VAR = \"CC_CONFIG_HOME\";\nexport const GLOBAL_CONFIG_FILE = \"config.yaml\";\nexport const PROJECT_CONFIG_FILE = \"config.yaml\";\n\n/**\n * Get global config directory path.\n * Can be overridden via CC_CONFIG_HOME environment variable for testing isolation.\n * This is a function (not constant) to allow tests to set the env var after module load.\n */\nexport function getGlobalConfigDir(): string {\n return (\n process.env[CONFIG_HOME_ENV_VAR] ||\n path.join(os.homedir(), \".claude-collective\")\n );\n}\n\n/**\n * @deprecated Use getGlobalConfigDir() instead. This constant is kept for backwards compatibility\n * but won't respect CC_CONFIG_HOME set after module load.\n */\nexport const GLOBAL_CONFIG_DIR = path.join(os.homedir(), \".claude-collective\");\n\n/**\n * Extra source entry for third-party skill repositories.\n * Used in both global and project configs.\n */\nexport interface SourceEntry {\n /** Short name for the source (e.g., \"company\", \"team\") */\n name: string;\n /** GitHub URL or path (e.g., \"github:owner/repo\") */\n url: string;\n /** Optional description */\n description?: string;\n /** Optional ref to pin to (branch, tag, or commit) */\n ref?: string;\n}\n\nexport interface GlobalConfig {\n source?: string;\n author?: string;\n marketplace?: string;\n agents_source?: string;\n /** Extra sources for third-party skills */\n sources?: SourceEntry[];\n}\n\nexport interface ProjectConfig {\n source?: string;\n author?: string;\n marketplace?: string;\n agents_source?: string;\n /** Extra sources for third-party skills */\n sources?: SourceEntry[];\n}\n\nexport interface ResolvedConfig {\n source: string;\n sourceOrigin: \"flag\" | \"env\" | \"project\" | \"global\" | \"default\";\n marketplace?: string;\n}\n\nfunction isValidSourceEntry(entry: unknown): entry is SourceEntry {\n if (typeof entry !== \"object\" || entry === null) return false;\n const e = entry as Record<string, unknown>;\n if (typeof e.name !== \"string\" || typeof e.url !== \"string\") return false;\n if (e.description !== undefined && typeof e.description !== \"string\")\n return false;\n if (e.ref !== undefined && typeof e.ref !== \"string\") return false;\n return true;\n}\n\nfunction isValidSourcesArray(arr: unknown): arr is SourceEntry[] {\n if (!Array.isArray(arr)) return false;\n return arr.every(isValidSourceEntry);\n}\n\nfunction isValidGlobalConfig(obj: unknown): obj is GlobalConfig {\n if (typeof obj !== \"object\" || obj === null) return false;\n const config = obj as Record<string, unknown>;\n if (config.source !== undefined && typeof config.source !== \"string\")\n return false;\n if (config.author !== undefined && typeof config.author !== \"string\")\n return false;\n if (\n config.marketplace !== undefined &&\n typeof config.marketplace !== \"string\"\n )\n return false;\n if (\n config.agents_source !== undefined &&\n typeof config.agents_source !== \"string\"\n )\n return false;\n if (config.sources !== undefined && !isValidSourcesArray(config.sources))\n return false;\n return true;\n}\n\nfunction isValidProjectConfig(obj: unknown): obj is ProjectConfig {\n if (typeof obj !== \"object\" || obj === null) return false;\n const config = obj as Record<string, unknown>;\n if (config.source !== undefined && typeof config.source !== \"string\")\n return false;\n if (config.author !== undefined && typeof config.author !== \"string\")\n return false;\n if (\n config.marketplace !== undefined &&\n typeof config.marketplace !== \"string\"\n )\n return false;\n if (\n config.agents_source !== undefined &&\n typeof config.agents_source !== \"string\"\n )\n return false;\n if (config.sources !== undefined && !isValidSourcesArray(config.sources))\n return false;\n return true;\n}\n\nexport function getGlobalConfigPath(): string {\n return path.join(getGlobalConfigDir(), GLOBAL_CONFIG_FILE);\n}\n\nexport function getProjectConfigPath(projectDir: string): string {\n return path.join(projectDir, CLAUDE_SRC_DIR, PROJECT_CONFIG_FILE);\n}\n\n/**\n * @deprecated Global config is deprecated. Use project-level config at .claude/config.yaml instead.\n * This function is kept for backwards compatibility during migration.\n */\nexport async function loadGlobalConfig(): Promise<GlobalConfig | null> {\n const configPath = getGlobalConfigPath();\n\n if (!(await fileExists(configPath))) {\n verbose(`Global config not found at ${configPath}`);\n return null;\n }\n\n try {\n const content = await readFile(configPath);\n const parsed = parseYaml(content);\n if (!isValidGlobalConfig(parsed)) {\n verbose(`Invalid global config structure at ${configPath}`);\n return null;\n }\n verbose(`Loaded global config from ${configPath}`);\n return parsed;\n } catch (error) {\n verbose(`Failed to parse global config: ${error}`);\n return null;\n }\n}\n\nexport async function loadProjectConfig(\n projectDir: string,\n): Promise<ProjectConfig | null> {\n // Check .claude-src/config.yaml first (new location)\n const srcConfigPath = getProjectConfigPath(projectDir);\n // Fall back to .claude/config.yaml (legacy location)\n const legacyConfigPath = path.join(projectDir, CLAUDE_DIR, \"config.yaml\");\n\n let configPath = srcConfigPath;\n if (!(await fileExists(srcConfigPath))) {\n if (await fileExists(legacyConfigPath)) {\n configPath = legacyConfigPath;\n verbose(`Using legacy config location: ${legacyConfigPath}`);\n } else {\n verbose(\n `Project config not found at ${srcConfigPath} or ${legacyConfigPath}`,\n );\n return null;\n }\n }\n\n try {\n const content = await readFile(configPath);\n const parsed = parseYaml(content);\n if (!isValidProjectConfig(parsed)) {\n verbose(`Invalid project config structure at ${configPath}`);\n return null;\n }\n verbose(`Loaded project config from ${configPath}`);\n return parsed;\n } catch (error) {\n verbose(`Failed to parse project config: ${error}`);\n return null;\n }\n}\n\n/**\n * @deprecated Global config is deprecated. Use project-level config at .claude/config.yaml instead.\n * This function is kept for backwards compatibility during migration.\n */\nexport async function saveGlobalConfig(config: GlobalConfig): Promise<void> {\n const configPath = getGlobalConfigPath();\n await ensureDir(getGlobalConfigDir());\n const content = stringifyYaml(config, { lineWidth: 0 });\n await writeFile(configPath, content);\n verbose(`Saved global config to ${configPath}`);\n}\n\nexport async function saveProjectConfig(\n projectDir: string,\n config: ProjectConfig,\n): Promise<void> {\n const configPath = getProjectConfigPath(projectDir);\n await ensureDir(path.join(projectDir, CLAUDE_SRC_DIR));\n const content = stringifyYaml(config, { lineWidth: 0 });\n await writeFile(configPath, content);\n verbose(`Saved project config to ${configPath}`);\n}\n\n/** Resolve source with precedence: flag > env > project > default */\nexport async function resolveSource(\n flagValue?: string,\n projectDir?: string,\n): Promise<ResolvedConfig> {\n // Load project config for marketplace (marketplace is resolved separately from source)\n const projectConfig = projectDir ? await loadProjectConfig(projectDir) : null;\n\n // Resolve marketplace: project config only (no flag/env support for marketplace)\n const marketplace = projectConfig?.marketplace;\n\n if (flagValue !== undefined) {\n if (flagValue === \"\" || flagValue.trim() === \"\") {\n throw new Error(\"--source flag cannot be empty\");\n }\n verbose(`Source from --source flag: ${flagValue}`);\n return { source: flagValue, sourceOrigin: \"flag\", marketplace };\n }\n\n const envValue = process.env[SOURCE_ENV_VAR];\n if (envValue) {\n verbose(`Source from ${SOURCE_ENV_VAR} env var: ${envValue}`);\n return { source: envValue, sourceOrigin: \"env\", marketplace };\n }\n\n if (projectConfig?.source) {\n verbose(`Source from project config: ${projectConfig.source}`);\n return {\n source: projectConfig.source,\n sourceOrigin: \"project\",\n marketplace,\n };\n }\n\n verbose(`Using default source: ${DEFAULT_SOURCE}`);\n return { source: DEFAULT_SOURCE, sourceOrigin: \"default\", marketplace };\n}\n\nexport type AgentsSourceOrigin = \"flag\" | \"project\" | \"global\" | \"default\";\n\nexport interface ResolvedAgentsSource {\n agentsSource?: string;\n agentsSourceOrigin: AgentsSourceOrigin;\n}\n\n/** Resolve agents_source with precedence: flag > project > default (undefined) */\nexport async function resolveAgentsSource(\n flagValue?: string,\n projectDir?: string,\n): Promise<ResolvedAgentsSource> {\n if (flagValue !== undefined) {\n if (flagValue === \"\" || flagValue.trim() === \"\") {\n throw new Error(\"--agent-source flag cannot be empty\");\n }\n verbose(`Agents source from --agent-source flag: ${flagValue}`);\n return { agentsSource: flagValue, agentsSourceOrigin: \"flag\" };\n }\n\n const projectConfig = projectDir ? await loadProjectConfig(projectDir) : null;\n if (projectConfig?.agents_source) {\n verbose(\n `Agents source from project config: ${projectConfig.agents_source}`,\n );\n return {\n agentsSource: projectConfig.agents_source,\n agentsSourceOrigin: \"project\",\n };\n }\n\n verbose(\"Using default agents source (local CLI)\");\n return { agentsSource: undefined, agentsSourceOrigin: \"default\" };\n}\n\nexport function formatAgentsSourceOrigin(origin: AgentsSourceOrigin): string {\n switch (origin) {\n case \"flag\":\n return \"--agent-source flag\";\n case \"project\":\n return \"project config (.claude-src/config.yaml)\";\n case \"global\":\n return \"global config (~/.claude-collective/config.yaml)\";\n case \"default\":\n return \"default (local CLI)\";\n }\n}\n\n/** Resolve author from project config */\nexport async function resolveAuthor(\n projectDir?: string,\n): Promise<string | undefined> {\n const projectConfig = projectDir ? await loadProjectConfig(projectDir) : null;\n return projectConfig?.author;\n}\n\nexport function formatSourceOrigin(\n origin: ResolvedConfig[\"sourceOrigin\"],\n): string {\n switch (origin) {\n case \"flag\":\n return \"--source flag\";\n case \"env\":\n return `${SOURCE_ENV_VAR} environment variable`;\n case \"project\":\n return \"project config (.claude-src/config.yaml)\";\n case \"global\":\n return \"global config (~/.claude-collective/config.yaml)\";\n case \"default\":\n return \"default\";\n }\n}\n\n/**\n * Resolve all configured sources for skill search.\n * Returns primary source plus any extra sources from project/global config.\n */\nexport async function resolveAllSources(\n projectDir?: string,\n): Promise<{ primary: SourceEntry; extras: SourceEntry[] }> {\n const projectConfig = projectDir ? await loadProjectConfig(projectDir) : null;\n const globalConfig = await loadGlobalConfig();\n\n // Get primary source\n const resolvedConfig = await resolveSource(undefined, projectDir);\n const primary: SourceEntry = {\n name: \"marketplace\",\n url: resolvedConfig.source,\n description: \"Primary skills marketplace\",\n };\n\n // Merge extra sources: project takes precedence, then global\n const extras: SourceEntry[] = [];\n const seenNames = new Set<string>();\n\n // Add project sources first (higher priority)\n if (projectConfig?.sources) {\n for (const source of projectConfig.sources) {\n if (!seenNames.has(source.name)) {\n seenNames.add(source.name);\n extras.push(source);\n }\n }\n }\n\n // Add global sources (lower priority, skip duplicates)\n if (globalConfig?.sources) {\n for (const source of globalConfig.sources) {\n if (!seenNames.has(source.name)) {\n seenNames.add(source.name);\n extras.push(source);\n }\n }\n }\n\n return { primary, extras };\n}\n\nexport function isLocalSource(source: string): boolean {\n if (source.startsWith(\"/\") || source.startsWith(\".\")) {\n return true;\n }\n\n const remoteProtocols = [\n \"github:\",\n \"gh:\",\n \"gitlab:\",\n \"bitbucket:\",\n \"sourcehut:\",\n \"https://\",\n \"http://\",\n ];\n\n const hasRemoteProtocol = remoteProtocols.some((prefix) =>\n source.startsWith(prefix),\n );\n\n if (!hasRemoteProtocol) {\n if (source.includes(\"..\") || source.includes(\"~\")) {\n throw new Error(\n `Invalid source path: ${source}. Path traversal patterns are not allowed.`,\n );\n }\n }\n\n return !hasRemoteProtocol;\n}\n"],"mappings":"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","names":[]}
@@ -1,9 +1,12 @@
1
1
  #!/usr/bin/env node
2
+ import {
3
+ WizardFooter
4
+ } from "./chunk-Y2LW7R3Y.js";
2
5
  import {
3
6
  init_esm_shims
4
7
  } from "./chunk-DHET7RCE.js";
5
8
 
6
- // src/cli-v2/components/wizard/step-confirm.tsx
9
+ // src/cli/components/wizard/step-confirm.tsx
7
10
  init_esm_shims();
8
11
  import { Box, Text, useInput } from "ink";
9
12
  import { jsx, jsxs } from "react/jsx-runtime";
@@ -73,11 +76,11 @@ var StepConfirm = ({
73
76
  ] })
74
77
  ] }),
75
78
  /* @__PURE__ */ jsx(Text, { children: " " }),
76
- /* @__PURE__ */ jsx(Text, { dimColor: true, children: "ENTER confirm ESC back" })
79
+ /* @__PURE__ */ jsx(WizardFooter, { navigation: "ENTER confirm", action: "ESC back" })
77
80
  ] });
78
81
  };
79
82
 
80
83
  export {
81
84
  StepConfirm
82
85
  };
83
- //# sourceMappingURL=chunk-UOWHJ6BE.js.map
86
+ //# sourceMappingURL=chunk-X6QONICW.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/cli/components/wizard/step-confirm.tsx"],"sourcesContent":["import React from \"react\";\nimport { Box, Text, useInput } from \"ink\";\nimport type { MergedSkillsMatrix } from \"../../types-matrix.js\";\nimport { WizardFooter } from \"./wizard-footer.js\";\n\n// =============================================================================\n// Types\n// =============================================================================\n\nexport interface StepConfirmProps {\n matrix: MergedSkillsMatrix;\n onComplete: () => void;\n stackName?: string;\n selectedDomains?: string[];\n domainSelections?: Record<string, Record<string, string[]>>;\n technologyCount?: number;\n skillCount?: number;\n installMode?: \"plugin\" | \"local\";\n onBack?: () => void;\n}\n\n// =============================================================================\n// Helper Functions\n// =============================================================================\n\n/**\n * Format domain name for display.\n */\nconst formatDomainName = (domain: string): string => {\n const names: Record<string, string> = {\n web: \"Web\",\n api: \"API\",\n cli: \"CLI\",\n mobile: \"Mobile\",\n };\n return names[domain] || domain.charAt(0).toUpperCase() + domain.slice(1);\n};\n\n// =============================================================================\n// Component\n// =============================================================================\n\nexport const StepConfirm: React.FC<StepConfirmProps> = ({\n matrix,\n onComplete,\n stackName,\n selectedDomains,\n domainSelections,\n technologyCount,\n skillCount,\n installMode,\n onBack,\n}) => {\n useInput((input, key) => {\n if (key.return) {\n onComplete();\n }\n if (key.escape && onBack) {\n onBack();\n }\n });\n\n // Build title based on stack vs scratch path\n const domainsText = selectedDomains?.map(formatDomainName).join(\" + \") || \"\";\n const title = stackName\n ? `Ready to install ${stackName}`\n : `Ready to install your custom stack${domainsText ? ` (${domainsText})` : \"\"}`;\n\n return (\n <Box flexDirection=\"column\" paddingX={2}>\n <Text bold color=\"green\">\n {title}\n </Text>\n <Text> </Text>\n\n {/* Domain breakdown (for scratch path) */}\n {domainSelections && selectedDomains && !stackName && (\n <Box flexDirection=\"column\" marginBottom={1}>\n {selectedDomains.map((domain) => {\n const selections = domainSelections[domain] || {};\n const techs = Object.values(selections).flat();\n if (techs.length === 0) return null;\n return (\n <Text key={domain}>\n <Text bold>{formatDomainName(domain)}:</Text>{\" \"}\n <Text>{techs.join(\", \")}</Text>\n </Text>\n );\n })}\n </Box>\n )}\n\n {/* Summary stats */}\n <Box flexDirection=\"column\" marginY={1}>\n {technologyCount !== undefined && (\n <Text>\n <Text dimColor>Technologies:</Text>{\" \"}\n <Text bold>{technologyCount}</Text>\n </Text>\n )}\n {skillCount !== undefined && (\n <Text>\n <Text dimColor>Skills:</Text> <Text bold>{skillCount}</Text>{\" \"}\n <Text color=\"green\">(all verified)</Text>\n </Text>\n )}\n {installMode && (\n <Text>\n <Text dimColor>Install mode:</Text>{\" \"}\n <Text bold>{installMode === \"plugin\" ? \"Plugin\" : \"Local\"}</Text>\n </Text>\n )}\n </Box>\n\n <Text> </Text>\n <WizardFooter navigation=\"ENTER confirm\" action=\"ESC back\" />\n </Box>\n );\n};\n"],"mappings":";;;;;;;;;AAAA;AACA,SAAS,KAAK,MAAM,gBAAgB;AAqE9B,cAcU,YAdV;AA1CN,IAAM,mBAAmB,CAAC,WAA2B;AACnD,QAAM,QAAgC;AAAA,IACpC,KAAK;AAAA,IACL,KAAK;AAAA,IACL,KAAK;AAAA,IACL,QAAQ;AAAA,EACV;AACA,SAAO,MAAM,MAAM,KAAK,OAAO,OAAO,CAAC,EAAE,YAAY,IAAI,OAAO,MAAM,CAAC;AACzE;AAMO,IAAM,cAA0C,CAAC;AAAA,EACtD;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF,MAAM;AACJ,WAAS,CAAC,OAAO,QAAQ;AACvB,QAAI,IAAI,QAAQ;AACd,iBAAW;AAAA,IACb;AACA,QAAI,IAAI,UAAU,QAAQ;AACxB,aAAO;AAAA,IACT;AAAA,EACF,CAAC;AAGD,QAAM,cAAc,iBAAiB,IAAI,gBAAgB,EAAE,KAAK,KAAK,KAAK;AAC1E,QAAM,QAAQ,YACV,oBAAoB,SAAS,KAC7B,qCAAqC,cAAc,KAAK,WAAW,MAAM,EAAE;AAE/E,SACE,qBAAC,OAAI,eAAc,UAAS,UAAU,GACpC;AAAA,wBAAC,QAAK,MAAI,MAAC,OAAM,SACd,iBACH;AAAA,IACA,oBAAC,QAAK,eAAC;AAAA,IAGN,oBAAoB,mBAAmB,CAAC,aACvC,oBAAC,OAAI,eAAc,UAAS,cAAc,GACvC,0BAAgB,IAAI,CAAC,WAAW;AAC/B,YAAM,aAAa,iBAAiB,MAAM,KAAK,CAAC;AAChD,YAAM,QAAQ,OAAO,OAAO,UAAU,EAAE,KAAK;AAC7C,UAAI,MAAM,WAAW,EAAG,QAAO;AAC/B,aACE,qBAAC,QACC;AAAA,6BAAC,QAAK,MAAI,MAAE;AAAA,2BAAiB,MAAM;AAAA,UAAE;AAAA,WAAC;AAAA,QAAQ;AAAA,QAC9C,oBAAC,QAAM,gBAAM,KAAK,IAAI,GAAE;AAAA,WAFf,MAGX;AAAA,IAEJ,CAAC,GACH;AAAA,IAIF,qBAAC,OAAI,eAAc,UAAS,SAAS,GAClC;AAAA,0BAAoB,UACnB,qBAAC,QACC;AAAA,4BAAC,QAAK,UAAQ,MAAC,2BAAa;AAAA,QAAQ;AAAA,QACpC,oBAAC,QAAK,MAAI,MAAE,2BAAgB;AAAA,SAC9B;AAAA,MAED,eAAe,UACd,qBAAC,QACC;AAAA,4BAAC,QAAK,UAAQ,MAAC,qBAAO;AAAA,QAAO;AAAA,QAAC,oBAAC,QAAK,MAAI,MAAE,sBAAW;AAAA,QAAQ;AAAA,QAC7D,oBAAC,QAAK,OAAM,SAAQ,4BAAc;AAAA,SACpC;AAAA,MAED,eACC,qBAAC,QACC;AAAA,4BAAC,QAAK,UAAQ,MAAC,2BAAa;AAAA,QAAQ;AAAA,QACpC,oBAAC,QAAK,MAAI,MAAE,0BAAgB,WAAW,WAAW,SAAQ;AAAA,SAC5D;AAAA,OAEJ;AAAA,IAEA,oBAAC,QAAK,eAAC;AAAA,IACP,oBAAC,gBAAa,YAAW,iBAAgB,QAAO,YAAW;AAAA,KAC7D;AAEJ;","names":[]}
@@ -0,0 +1,23 @@
1
+ #!/usr/bin/env node
2
+ import {
3
+ init_esm_shims
4
+ } from "./chunk-DHET7RCE.js";
5
+
6
+ // src/cli/components/wizard/wizard-footer.tsx
7
+ init_esm_shims();
8
+ import { Box, Text } from "ink";
9
+ import { jsx, jsxs } from "react/jsx-runtime";
10
+ var WizardFooter = ({
11
+ navigation,
12
+ action
13
+ }) => {
14
+ return /* @__PURE__ */ jsxs(Box, { flexDirection: "row", justifyContent: "space-between", marginTop: 1, children: [
15
+ /* @__PURE__ */ jsx(Text, { dimColor: true, children: navigation }),
16
+ /* @__PURE__ */ jsx(Text, { dimColor: true, children: action })
17
+ ] });
18
+ };
19
+
20
+ export {
21
+ WizardFooter
22
+ };
23
+ //# sourceMappingURL=chunk-Y2LW7R3Y.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/cli/components/wizard/wizard-footer.tsx"],"sourcesContent":["/**\n * WizardFooter component - Shared footer with split layout for navigation hints.\n *\n * Uses `justifyContent=\"space-between\"` pattern for left/right alignment:\n * - Left side: navigation controls (arrows, SPACE, etc.)\n * - Right side: action hints (ESC, ENTER)\n */\nimport React from \"react\";\nimport { Box, Text } from \"ink\";\n\n// =============================================================================\n// Types\n// =============================================================================\n\nexport interface WizardFooterProps {\n /** Left side: navigation controls (e.g., \"up/down navigate ENTER select\") */\n navigation: string;\n /** Right side: action hints (e.g., \"ESC cancel\") */\n action: string;\n}\n\n// =============================================================================\n// Component\n// =============================================================================\n\nexport const WizardFooter: React.FC<WizardFooterProps> = ({\n navigation,\n action,\n}) => {\n return (\n <Box flexDirection=\"row\" justifyContent=\"space-between\" marginTop={1}>\n <Text dimColor>{navigation}</Text>\n <Text dimColor>{action}</Text>\n </Box>\n );\n};\n"],"mappings":";;;;;;AAAA;AAQA,SAAS,KAAK,YAAY;AAsBtB,SACE,KADF;AALG,IAAM,eAA4C,CAAC;AAAA,EACxD;AAAA,EACA;AACF,MAAM;AACJ,SACE,qBAAC,OAAI,eAAc,OAAM,gBAAe,iBAAgB,WAAW,GACjE;AAAA,wBAAC,QAAK,UAAQ,MAAE,sBAAW;AAAA,IAC3B,oBAAC,QAAK,UAAQ,MAAE,kBAAO;AAAA,KACzB;AAEJ;","names":[]}
@@ -0,0 +1,78 @@
1
+ #!/usr/bin/env node
2
+ import {
3
+ init_esm_shims
4
+ } from "./chunk-DHET7RCE.js";
5
+
6
+ // src/cli/components/themes/default.ts
7
+ init_esm_shims();
8
+ import { extendTheme, defaultTheme } from "@inkjs/ui";
9
+ var cliTheme = extendTheme(defaultTheme, {
10
+ components: {
11
+ Spinner: {
12
+ styles: {
13
+ frame: () => ({ color: "cyan" }),
14
+ label: () => ({ color: "gray" })
15
+ }
16
+ },
17
+ Select: {
18
+ styles: {
19
+ focusIndicator: () => ({ color: "cyan" }),
20
+ label: ({ isFocused }) => ({
21
+ color: isFocused ? "cyan" : void 0
22
+ })
23
+ }
24
+ },
25
+ MultiSelect: {
26
+ styles: {
27
+ focusIndicator: () => ({ color: "cyan" }),
28
+ label: ({ isFocused, isSelected }) => ({
29
+ color: isFocused ? "cyan" : isSelected ? "green" : void 0
30
+ }),
31
+ checkboxChecked: () => ({ color: "green" })
32
+ }
33
+ },
34
+ StatusMessage: {
35
+ styles: {
36
+ container: ({ variant }) => ({
37
+ borderStyle: "round",
38
+ borderColor: variant === "error" ? "red" : variant === "warning" ? "yellow" : variant === "success" ? "green" : "blue"
39
+ })
40
+ }
41
+ },
42
+ Alert: {
43
+ styles: {
44
+ container: ({ variant }) => ({
45
+ borderColor: variant === "error" ? "red" : variant === "warning" ? "yellow" : variant === "success" ? "green" : "blue"
46
+ }),
47
+ icon: ({ variant }) => ({
48
+ color: variant === "error" ? "red" : variant === "warning" ? "yellow" : variant === "success" ? "green" : "blue"
49
+ })
50
+ }
51
+ },
52
+ TextInput: {
53
+ styles: {
54
+ container: ({ isFocused }) => ({
55
+ borderColor: isFocused ? "cyan" : "gray"
56
+ }),
57
+ cursor: () => ({ color: "cyan" })
58
+ }
59
+ },
60
+ ConfirmInput: {
61
+ styles: {
62
+ highlightedChoice: () => ({ color: "cyan" })
63
+ }
64
+ },
65
+ Badge: {
66
+ styles: {
67
+ container: ({ variant }) => ({
68
+ color: variant === "error" ? "red" : variant === "warning" ? "yellow" : variant === "success" ? "green" : variant === "info" ? "blue" : "cyan"
69
+ })
70
+ }
71
+ }
72
+ }
73
+ });
74
+
75
+ export {
76
+ cliTheme
77
+ };
78
+ //# sourceMappingURL=chunk-Z2CWURZ6.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/cli/components/themes/default.ts"],"sourcesContent":["import { extendTheme, defaultTheme } from \"@inkjs/ui\";\n\n/**\n * CLI theme matching existing picocolors styling\n *\n * Color scheme:\n * - Cyan: Focus/primary (headings, prompts, selected items)\n * - Green: Success states\n * - Red: Errors\n * - Yellow: Warnings\n * - Blue: Info messages\n */\nexport const cliTheme = extendTheme(defaultTheme, {\n components: {\n Spinner: {\n styles: {\n frame: () => ({ color: \"cyan\" }),\n label: () => ({ color: \"gray\" }),\n },\n },\n Select: {\n styles: {\n focusIndicator: () => ({ color: \"cyan\" }),\n label: ({ isFocused }) => ({\n color: isFocused ? \"cyan\" : undefined,\n }),\n },\n },\n MultiSelect: {\n styles: {\n focusIndicator: () => ({ color: \"cyan\" }),\n label: ({ isFocused, isSelected }) => ({\n color: isFocused ? \"cyan\" : isSelected ? \"green\" : undefined,\n }),\n checkboxChecked: () => ({ color: \"green\" }),\n },\n },\n StatusMessage: {\n styles: {\n container: ({ variant }) => ({\n borderStyle: \"round\",\n borderColor:\n variant === \"error\"\n ? \"red\"\n : variant === \"warning\"\n ? \"yellow\"\n : variant === \"success\"\n ? \"green\"\n : \"blue\",\n }),\n },\n },\n Alert: {\n styles: {\n container: ({ variant }) => ({\n borderColor:\n variant === \"error\"\n ? \"red\"\n : variant === \"warning\"\n ? \"yellow\"\n : variant === \"success\"\n ? \"green\"\n : \"blue\",\n }),\n icon: ({ variant }) => ({\n color:\n variant === \"error\"\n ? \"red\"\n : variant === \"warning\"\n ? \"yellow\"\n : variant === \"success\"\n ? \"green\"\n : \"blue\",\n }),\n },\n },\n TextInput: {\n styles: {\n container: ({ isFocused }) => ({\n borderColor: isFocused ? \"cyan\" : \"gray\",\n }),\n cursor: () => ({ color: \"cyan\" }),\n },\n },\n ConfirmInput: {\n styles: {\n highlightedChoice: () => ({ color: \"cyan\" }),\n },\n },\n Badge: {\n styles: {\n container: ({ variant }) => ({\n color:\n variant === \"error\"\n ? \"red\"\n : variant === \"warning\"\n ? \"yellow\"\n : variant === \"success\"\n ? \"green\"\n : variant === \"info\"\n ? \"blue\"\n : \"cyan\",\n }),\n },\n },\n },\n});\n"],"mappings":";;;;;;AAAA;AAAA,SAAS,aAAa,oBAAoB;AAYnC,IAAM,WAAW,YAAY,cAAc;AAAA,EAChD,YAAY;AAAA,IACV,SAAS;AAAA,MACP,QAAQ;AAAA,QACN,OAAO,OAAO,EAAE,OAAO,OAAO;AAAA,QAC9B,OAAO,OAAO,EAAE,OAAO,OAAO;AAAA,MAChC;AAAA,IACF;AAAA,IACA,QAAQ;AAAA,MACN,QAAQ;AAAA,QACN,gBAAgB,OAAO,EAAE,OAAO,OAAO;AAAA,QACvC,OAAO,CAAC,EAAE,UAAU,OAAO;AAAA,UACzB,OAAO,YAAY,SAAS;AAAA,QAC9B;AAAA,MACF;AAAA,IACF;AAAA,IACA,aAAa;AAAA,MACX,QAAQ;AAAA,QACN,gBAAgB,OAAO,EAAE,OAAO,OAAO;AAAA,QACvC,OAAO,CAAC,EAAE,WAAW,WAAW,OAAO;AAAA,UACrC,OAAO,YAAY,SAAS,aAAa,UAAU;AAAA,QACrD;AAAA,QACA,iBAAiB,OAAO,EAAE,OAAO,QAAQ;AAAA,MAC3C;AAAA,IACF;AAAA,IACA,eAAe;AAAA,MACb,QAAQ;AAAA,QACN,WAAW,CAAC,EAAE,QAAQ,OAAO;AAAA,UAC3B,aAAa;AAAA,UACb,aACE,YAAY,UACR,QACA,YAAY,YACV,WACA,YAAY,YACV,UACA;AAAA,QACZ;AAAA,MACF;AAAA,IACF;AAAA,IACA,OAAO;AAAA,MACL,QAAQ;AAAA,QACN,WAAW,CAAC,EAAE,QAAQ,OAAO;AAAA,UAC3B,aACE,YAAY,UACR,QACA,YAAY,YACV,WACA,YAAY,YACV,UACA;AAAA,QACZ;AAAA,QACA,MAAM,CAAC,EAAE,QAAQ,OAAO;AAAA,UACtB,OACE,YAAY,UACR,QACA,YAAY,YACV,WACA,YAAY,YACV,UACA;AAAA,QACZ;AAAA,MACF;AAAA,IACF;AAAA,IACA,WAAW;AAAA,MACT,QAAQ;AAAA,QACN,WAAW,CAAC,EAAE,UAAU,OAAO;AAAA,UAC7B,aAAa,YAAY,SAAS;AAAA,QACpC;AAAA,QACA,QAAQ,OAAO,EAAE,OAAO,OAAO;AAAA,MACjC;AAAA,IACF;AAAA,IACA,cAAc;AAAA,MACZ,QAAQ;AAAA,QACN,mBAAmB,OAAO,EAAE,OAAO,OAAO;AAAA,MAC5C;AAAA,IACF;AAAA,IACA,OAAO;AAAA,MACL,QAAQ;AAAA,QACN,WAAW,CAAC,EAAE,QAAQ,OAAO;AAAA,UAC3B,OACE,YAAY,UACR,QACA,YAAY,YACV,WACA,YAAY,YACV,UACA,YAAY,SACV,SACA;AAAA,QACd;AAAA,MACF;AAAA,IACF;AAAA,EACF;AACF,CAAC;","names":[]}