claude-agent-skills 1.4.1 → 1.5.1

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.
package/commands/hub.js CHANGED
@@ -1,43 +1,75 @@
1
- import { select, isCancel, cancel, outro } from '@clack/prompts';
2
1
  import updateNotifier from 'update-notifier';
3
2
  import { createRequire } from 'node:module';
4
3
  import { fileURLToPath } from 'node:url';
5
4
  import { dirname, join } from 'node:path';
5
+ import ansis from 'ansis';
6
6
  import { showIntro } from '../lib/banner.js';
7
7
  import { CliCancel } from '../lib/prompts.js';
8
+ import { brand, muted, white } from '../lib/theme.js';
9
+ import { inlineSelect } from '../lib/inlineSelect.js';
8
10
 
9
11
  const req = createRequire(import.meta.url);
10
12
  const pkg = req(join(dirname(fileURLToPath(import.meta.url)), '..', 'package.json'));
11
13
  updateNotifier({ pkg }).notify();
12
- import { runAdd } from './add.js';
14
+
15
+ import { runAdd } from './add.js';
13
16
  import { runUpdate } from './update.js';
14
17
  import { runRemove } from './remove.js';
15
- import { runList } from './list.js';
16
- import { runSync } from './sync.js';
17
- import { runCheck } from './check.js';
18
+ import { runList } from './list.js';
19
+ import { runSync } from './sync.js';
20
+ import { runCheck } from './check.js';
18
21
 
19
22
  const SKIP = { skipIntro: true };
20
23
 
21
24
  const MENU = [
22
- { value: 'add', label: 'Add Skill(s)' },
23
- { value: 'update', label: 'Update Existing Skill(s)' },
24
- { value: 'remove', label: 'Remove Existing Skill(s)' },
25
- { value: 'list', label: 'List Installed Skill(s)' },
26
- { value: 'sync', label: 'Sync/Restore Skills from Lockfile' },
27
- { value: 'check', label: 'Check Skill(s)' },
25
+ { value: 'add', label: 'Add Skill(s)', hint: 'install new skills' },
26
+ { value: 'update', label: 'Update Existing Skill(s)', hint: 'pull latest versions' },
27
+ { value: 'remove', label: 'Remove Existing Skill(s)', hint: 'uninstall skills' },
28
+ { value: 'list', label: 'List Installed Skill(s)', hint: "show what's installed" },
29
+ { value: 'sync', label: 'Sync/Restore from Lockfile', hint: 'restore from claude-skills-lock.json' },
30
+ { value: 'check', label: 'Check Skill(s)', hint: 'verify hashes & lockfile' },
28
31
  { value: 'quit', label: 'Quit' },
29
32
  ];
30
33
 
34
+ function printCompactHeader() {
35
+ const silver = s => ansis.rgb(190, 190, 190)(s);
36
+ process.stdout.write('\n');
37
+ process.stdout.write(brand('◈ CLAUDE SKILLS') + ' ' + silver('Agent Skills for Claude Code') + '\n');
38
+ process.stdout.write('\n');
39
+ }
40
+
41
+ function restoreScreen() {
42
+ process.stdout.write('\x1b[?1049l');
43
+ }
44
+
31
45
  export async function runHub() {
46
+ // Enter alternate screen buffer — isolated viewport with no scrollback accumulation.
47
+ // Original terminal content is restored when we exit (same as vim/less/claude-code).
48
+ process.stdout.write('\x1b[?1049h\x1b[2J\x1b[H');
49
+ process.on('exit', restoreScreen); // covers Ctrl+C, process.exit(), uncaught errors
50
+
32
51
  await showIntro();
33
52
 
53
+ let first = true;
34
54
  for (;;) {
35
- const choice = await select({ message: 'What do you want to do?', options: MENU });
36
-
37
- if (isCancel(choice)) { cancel('Cancelled.'); return; }
38
- if (choice === 'quit') { outro('Goodbye.'); return; }
55
+ if (!first) {
56
+ process.stdout.write('\x1b[2J\x1b[H');
57
+ printCompactHeader();
58
+ }
59
+ first = false;
39
60
 
40
61
  try {
62
+ const choice = await inlineSelect({
63
+ message: 'What do you want to do?',
64
+ hint: '↑↓ navigate · enter select · esc quit',
65
+ options: MENU,
66
+ });
67
+
68
+ if (choice === 'quit') {
69
+ restoreScreen();
70
+ return;
71
+ }
72
+
41
73
  if (choice === 'add') await runAdd(SKIP);
42
74
  if (choice === 'update') await runUpdate(SKIP);
43
75
  if (choice === 'remove') await runRemove(SKIP);
@@ -45,7 +77,12 @@ export async function runHub() {
45
77
  if (choice === 'sync') await runSync(SKIP);
46
78
  if (choice === 'check') await runCheck(SKIP);
47
79
  } catch (e) {
48
- if (e instanceof CliCancel) continue;
80
+ if (e instanceof CliCancel) continue; // sub-command ESC → back to menu
81
+ if (e?.isCancel) { // hub menu ESC → quit
82
+ restoreScreen();
83
+ return;
84
+ }
85
+ restoreScreen();
49
86
  throw e;
50
87
  }
51
88
  }
@@ -0,0 +1,88 @@
1
+ /**
2
+ * Single-select in-place picker — renders in-place like skillPicker.
3
+ * Resolves with selected value. Throws { isCancel: true } on ESC.
4
+ */
5
+ import ansis from 'ansis';
6
+ import { brand, white, muted } from './theme.js';
7
+
8
+ const KEY = {
9
+ UP: '\x1b[A',
10
+ DOWN: '\x1b[B',
11
+ ENTER: '\r',
12
+ CTRL_C: '\x03',
13
+ ESC: '\x1b',
14
+ };
15
+
16
+ export async function inlineSelect({ message, hint = '↑↓ navigate · enter select · esc back', options }) {
17
+ let cursor = 0;
18
+ let lastLines = 0;
19
+
20
+ function renderLines() {
21
+ const out = [];
22
+ out.push(brand('◆') + ' ' + ansis.bold(white(message)) + ' ' + muted(hint));
23
+ out.push(muted('│'));
24
+ for (let i = 0; i < options.length; i++) {
25
+ const focused = i === cursor;
26
+ const arrow = focused ? ansis.bold(ansis.white('▶')) : ' ';
27
+ const label = focused
28
+ ? ansis.bold(ansis.white(options[i].label))
29
+ : muted(options[i].label);
30
+ const desc = options[i].hint ? ' ' + muted(options[i].hint) : '';
31
+ out.push(muted('│') + ' ' + arrow + ' ' + label + desc);
32
+ }
33
+ out.push(muted('│'));
34
+ return out;
35
+ }
36
+
37
+ function draw() {
38
+ const lines = renderLines();
39
+ if (lastLines > 0) process.stdout.write(`\x1b[${lastLines}A`);
40
+ for (const l of lines) process.stdout.write('\x1b[2K' + l + '\n');
41
+ process.stdout.write('\x1b[0J');
42
+ lastLines = lines.length;
43
+ }
44
+
45
+ function clearBlock() {
46
+ if (lastLines > 0) {
47
+ process.stdout.write(`\x1b[${lastLines}A\x1b[0J`);
48
+ lastLines = 0;
49
+ }
50
+ }
51
+
52
+ draw();
53
+
54
+ return new Promise((resolve, reject) => {
55
+ process.stdin.setRawMode(true);
56
+ process.stdin.resume();
57
+ process.stdin.setEncoding('utf8');
58
+
59
+ function cleanup() {
60
+ process.stdin.setRawMode(false);
61
+ process.stdin.pause();
62
+ process.stdin.removeAllListeners('data');
63
+ }
64
+
65
+ process.stdin.on('data', key => {
66
+ if (key === KEY.CTRL_C) { cleanup(); process.exit(0); }
67
+
68
+ if (key === KEY.ESC) {
69
+ cleanup();
70
+ clearBlock();
71
+ reject(Object.assign(new Error('cancel'), { isCancel: true }));
72
+ return;
73
+ }
74
+
75
+ if (key === KEY.UP) cursor = Math.max(0, cursor - 1);
76
+ if (key === KEY.DOWN) cursor = Math.min(options.length - 1, cursor + 1);
77
+
78
+ if (key === KEY.ENTER) {
79
+ cleanup();
80
+ clearBlock();
81
+ resolve(options[cursor].value);
82
+ return;
83
+ }
84
+
85
+ draw();
86
+ });
87
+ });
88
+ }
package/lib/picker.js CHANGED
@@ -157,7 +157,9 @@ export async function skillPicker({ message, options }) {
157
157
  if (key === KEY.ENTER) {
158
158
  if (!sel.size) return;
159
159
  cleanup();
160
- process.stdout.write('\x1b[2K' + success('◆') + ' ' +
160
+ // Clear entire picker block then write compact confirmation
161
+ if (lastLines > 0) process.stdout.write(`\x1b[${lastLines}A\x1b[0J`);
162
+ process.stdout.write(success('◆') + ' ' +
161
163
  white(`${sel.size} skill(s) selected`) + '\n');
162
164
  resolve([...sel].sort((a, b) => a - b).map(i => options[i].value));
163
165
  return;
package/lib/prompts.js CHANGED
@@ -1,12 +1,11 @@
1
- import { select, confirm, isCancel } from '@clack/prompts';
1
+ import { confirm, isCancel } from '@clack/prompts';
2
2
  import ansis from 'ansis';
3
3
  import { skillColor, white, muted, divider } from './theme.js';
4
4
  import { skillPicker } from './picker.js';
5
+ import { inlineSelect } from './inlineSelect.js';
5
6
 
6
7
  export class CliCancel extends Error {}
7
8
 
8
- // Escape/Ctrl+C at any prompt throws CliCancel — hub catches it silently and
9
- // returns to the main menu (no "Cancelled." noise printed).
10
9
  function guard(v) {
11
10
  if (isCancel(v)) throw new CliCancel();
12
11
  return v;
@@ -15,13 +14,14 @@ function guard(v) {
15
14
  export async function pickScope(flags = {}) {
16
15
  if (flags.global) return { global: true };
17
16
  if (flags.project) return { project: true };
18
- const v = guard(await select({
17
+ const v = await inlineSelect({
19
18
  message: 'Select scope',
19
+ hint: '↑↓ navigate · enter select · esc back',
20
20
  options: [
21
- { value: 'global', label: 'Global (~/.claude/skills + ~/.agents/skills)' },
22
- { value: 'project', label: 'Project (.claude/skills in current directory)' },
21
+ { value: 'global', label: 'Global', hint: '~/.claude/skills + ~/.agents/skills' },
22
+ { value: 'project', label: 'Project', hint: '.claude/skills in current directory' },
23
23
  ],
24
- }));
24
+ }).catch(e => { if (e?.isCancel) throw new CliCancel(); throw e; });
25
25
  return { global: v === 'global' };
26
26
  }
27
27
 
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "claude-agent-skills",
3
- "version": "1.4.1",
3
+ "version": "1.5.1",
4
4
  "description": "Install and manage Pavi's Claude Code agent skills",
5
5
  "license": "MIT",
6
6
  "type": "module",
package/skills.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "schema_version": 1,
3
3
  "name": "claude-agent-skills",
4
- "version": "1.4.1",
4
+ "version": "1.5.1",
5
5
  "skills": [
6
6
  "ask-matt",
7
7
  "brainstorming",