@assistkick/create 1.23.0 → 1.24.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.
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@assistkick/create",
3
- "version": "1.23.0",
3
+ "version": "1.24.0",
4
4
  "description": "Scaffold assistkick-product-system into any project",
5
5
  "type": "module",
6
6
  "bin": {
@@ -261,6 +261,124 @@ const AgentDetailView = ({ input, result }: Omit<ToolDetailViewProps, 'toolName'
261
261
  );
262
262
  };
263
263
 
264
+ /* ── Generic (MCP & other tools) ──────────────────── */
265
+
266
+ /** Format an MCP-style tool name for display: mcp__jetbrains__list_directory_tree → jetbrains / list_directory_tree */
267
+ const formatToolName = (name: string): { provider: string | null; action: string } => {
268
+ if (name.startsWith('mcp__')) {
269
+ const parts = name.slice(5).split('__');
270
+ if (parts.length >= 2) {
271
+ return { provider: parts[0], action: parts.slice(1).join(' / ') };
272
+ }
273
+ return { provider: null, action: parts[0] };
274
+ }
275
+ return { provider: null, action: name };
276
+ };
277
+
278
+ /** Try to pretty-print a value — if JSON, format it; otherwise return as string. */
279
+ const tryFormatValue = (value: unknown): { text: string; isJson: boolean } => {
280
+ if (value == null) return { text: '', isJson: false };
281
+ if (typeof value === 'string') {
282
+ // Try parsing as JSON
283
+ try {
284
+ const parsed = JSON.parse(value);
285
+ if (typeof parsed === 'object' && parsed !== null) {
286
+ return { text: JSON.stringify(parsed, null, 2), isJson: true };
287
+ }
288
+ } catch {
289
+ // Not JSON — return as plain text
290
+ }
291
+ return { text: value, isJson: false };
292
+ }
293
+ if (typeof value === 'object') {
294
+ return { text: JSON.stringify(value, null, 2), isJson: true };
295
+ }
296
+ return { text: String(value), isJson: false };
297
+ };
298
+
299
+ const GenericDetailView = ({ toolName, input, result }: ToolDetailViewProps) => {
300
+ const { provider, action } = formatToolName(toolName);
301
+ const content = result ? contentToString(result.content) : '';
302
+ const inputEntries = Object.entries(input);
303
+
304
+ return (
305
+ <div className="flex flex-col gap-2">
306
+ {/* Tool name badge */}
307
+ <div className="flex items-center gap-2 px-1">
308
+ {provider && (
309
+ <span className="text-[10px] font-mono px-2 py-0.5 rounded-full bg-accent/15 text-accent">
310
+ {provider}
311
+ </span>
312
+ )}
313
+ <span className="text-[11px] font-mono text-content-secondary">{action}</span>
314
+ </div>
315
+
316
+ {/* Input parameters */}
317
+ {inputEntries.length > 0 && (
318
+ <Panel label="Input">
319
+ <div className="px-3 py-2 flex flex-col gap-1.5">
320
+ {inputEntries.map(([key, value]) => {
321
+ const { text, isJson } = tryFormatValue(value);
322
+ const isLong = text.length > 120;
323
+ return (
324
+ <div key={key} className="flex flex-col gap-0.5">
325
+ <span className="text-[10px] font-mono text-content-muted uppercase tracking-wider">{key}</span>
326
+ {isJson ? (
327
+ <pre className="text-[12px] font-mono text-content-secondary whitespace-pre-wrap break-words m-0 max-h-[200px] overflow-y-auto bg-surface-raised/50 rounded px-2 py-1">
328
+ {text}
329
+ </pre>
330
+ ) : isLong ? (
331
+ <pre className="text-[12px] font-mono text-content-secondary whitespace-pre-wrap break-words m-0 max-h-[200px] overflow-y-auto">
332
+ {text}
333
+ </pre>
334
+ ) : (
335
+ <span className="text-[12px] font-mono text-content-secondary">{text || '(empty)'}</span>
336
+ )}
337
+ </div>
338
+ );
339
+ })}
340
+ </div>
341
+ </Panel>
342
+ )}
343
+
344
+ {/* Output */}
345
+ {result && (
346
+ <Panel label={result.isError ? 'Error' : 'Output'} labelColor={result.isError ? 'text-red-400' : undefined}>
347
+ <div className="max-h-[400px] overflow-y-auto">
348
+ {(() => {
349
+ const { text: formatted, isJson } = tryFormatValue(content);
350
+ if (result.isError) {
351
+ return (
352
+ <pre className="text-[12px] font-mono text-error whitespace-pre-wrap break-words px-3 py-2 m-0">
353
+ {content || '(empty)'}
354
+ </pre>
355
+ );
356
+ }
357
+ if (isJson) {
358
+ return (
359
+ <SyntaxHighlighter
360
+ language="json"
361
+ style={oneDark}
362
+ customStyle={codeStyle}
363
+ wrapLongLines
364
+ >
365
+ {formatted}
366
+ </SyntaxHighlighter>
367
+ );
368
+ }
369
+ return (
370
+ <pre className="text-[12px] font-mono text-content-secondary whitespace-pre-wrap break-words px-3 py-2 m-0">
371
+ {content || '(empty)'}
372
+ </pre>
373
+ );
374
+ })()}
375
+ </div>
376
+ </Panel>
377
+ )}
378
+ </div>
379
+ );
380
+ };
381
+
264
382
  /* ── Router ────────────────────────────────────────── */
265
383
 
266
384
  export const ToolDetailView = ({ toolName, input, result }: ToolDetailViewProps) => {
@@ -268,6 +386,6 @@ export const ToolDetailView = ({ toolName, input, result }: ToolDetailViewProps)
268
386
  case 'Read': return <ReadDetailView input={input} result={result} />;
269
387
  case 'Bash': return <BashDetailView input={input} result={result} />;
270
388
  case 'Agent': return <AgentDetailView input={input} result={result} />;
271
- default: return null;
389
+ default: return <GenericDetailView toolName={toolName} input={input} result={result} />;
272
390
  }
273
391
  };
@@ -53,12 +53,16 @@ const contentToString = (content: ToolResultBlock['content']): string => {
53
53
 
54
54
  type ViewTab = 'humanized' | 'raw';
55
55
 
56
- const hasHumanizedView = (name: string): boolean =>
57
- name === 'Edit' || name === 'Write' || name === 'Read' || name === 'Bash' || name === 'Agent';
56
+ /** All tools now support humanized + raw tabs. */
57
+ const hasHumanizedView = (_name: string): boolean => true;
58
58
 
59
59
  /** Tools where the humanized view renders both input and result together. */
60
60
  const hasIntegratedView = (name: string): boolean =>
61
- name === 'Read' || name === 'Bash' || name === 'Agent';
61
+ name === 'Read' || name === 'Bash' || name === 'Agent' || hasGenericView(name);
62
+
63
+ /** Tools that use the generic humanized view (MCP tools and other non-built-in tools). */
64
+ const hasGenericView = (name: string): boolean =>
65
+ !['Edit', 'Write', 'Read', 'Bash', 'Agent'].includes(name);
62
66
 
63
67
  export const ToolUseCard = ({ name, input, isStreaming, result }: ToolUseCardProps) => {
64
68
  const [expanded, setExpanded] = useState(false);
@@ -120,10 +124,10 @@ export const ToolUseCard = ({ name, input, isStreaming, result }: ToolUseCardPro
120
124
  {/* Tool input — humanized or raw */}
121
125
  {hasHumanizedView(name) && viewTab === 'humanized' ? (
122
126
  <div className="px-3 py-2">
123
- {hasIntegratedView(name) ? (
124
- <ToolDetailView toolName={name} input={input} result={result} />
125
- ) : (
127
+ {name === 'Edit' || name === 'Write' ? (
126
128
  <ToolDiffView toolName={name as 'Edit' | 'Write'} input={input} />
129
+ ) : (
130
+ <ToolDetailView toolName={name} input={input} result={result} />
127
131
  )}
128
132
  </div>
129
133
  ) : (
@@ -76,8 +76,18 @@ export const summarizeToolUse = (
76
76
  return url ? truncate(`Fetch: ${url}`, MAX_SUMMARY_LEN) : 'Web fetch';
77
77
  }
78
78
 
79
- default:
79
+ default: {
80
+ // MCP tools: mcp__provider__action → "provider: action"
81
+ if (name.startsWith('mcp__')) {
82
+ const parts = name.slice(5).split('__');
83
+ if (parts.length >= 2) {
84
+ const provider = parts[0];
85
+ const action = parts.slice(1).join(' ');
86
+ return truncate(`${provider}: ${action}`, MAX_SUMMARY_LEN);
87
+ }
88
+ }
80
89
  return name;
90
+ }
81
91
  }
82
92
  };
83
93