@sentry/warden 0.0.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/.agents/skills/find-bugs/SKILL.md +75 -0
- package/.agents/skills/vercel-react-best-practices/AGENTS.md +2934 -0
- package/.agents/skills/vercel-react-best-practices/SKILL.md +136 -0
- package/.agents/skills/vercel-react-best-practices/rules/advanced-event-handler-refs.md +55 -0
- package/.agents/skills/vercel-react-best-practices/rules/advanced-init-once.md +42 -0
- package/.agents/skills/vercel-react-best-practices/rules/advanced-use-latest.md +39 -0
- package/.agents/skills/vercel-react-best-practices/rules/async-api-routes.md +38 -0
- package/.agents/skills/vercel-react-best-practices/rules/async-defer-await.md +80 -0
- package/.agents/skills/vercel-react-best-practices/rules/async-dependencies.md +51 -0
- package/.agents/skills/vercel-react-best-practices/rules/async-parallel.md +28 -0
- package/.agents/skills/vercel-react-best-practices/rules/async-suspense-boundaries.md +99 -0
- package/.agents/skills/vercel-react-best-practices/rules/bundle-barrel-imports.md +59 -0
- package/.agents/skills/vercel-react-best-practices/rules/bundle-conditional.md +31 -0
- package/.agents/skills/vercel-react-best-practices/rules/bundle-defer-third-party.md +49 -0
- package/.agents/skills/vercel-react-best-practices/rules/bundle-dynamic-imports.md +35 -0
- package/.agents/skills/vercel-react-best-practices/rules/bundle-preload.md +50 -0
- package/.agents/skills/vercel-react-best-practices/rules/client-event-listeners.md +74 -0
- package/.agents/skills/vercel-react-best-practices/rules/client-localstorage-schema.md +71 -0
- package/.agents/skills/vercel-react-best-practices/rules/client-passive-event-listeners.md +48 -0
- package/.agents/skills/vercel-react-best-practices/rules/client-swr-dedup.md +56 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-batch-dom-css.md +107 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-cache-function-results.md +80 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-cache-property-access.md +28 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-cache-storage.md +70 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-combine-iterations.md +32 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-early-exit.md +50 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-hoist-regexp.md +45 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-index-maps.md +37 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-length-check-first.md +49 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-min-max-loop.md +82 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-set-map-lookups.md +24 -0
- package/.agents/skills/vercel-react-best-practices/rules/js-tosorted-immutable.md +57 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-activity.md +26 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-animate-svg-wrapper.md +47 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-conditional-render.md +40 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-content-visibility.md +38 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-hoist-jsx.md +46 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-hydration-no-flicker.md +82 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-hydration-suppress-warning.md +30 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-svg-precision.md +28 -0
- package/.agents/skills/vercel-react-best-practices/rules/rendering-usetransition-loading.md +75 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-defer-reads.md +39 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-dependencies.md +45 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-derived-state-no-effect.md +40 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-derived-state.md +29 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-functional-setstate.md +74 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-lazy-state-init.md +58 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-memo-with-default-value.md +38 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-memo.md +44 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-move-effect-to-event.md +45 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-simple-expression-in-memo.md +35 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-transitions.md +40 -0
- package/.agents/skills/vercel-react-best-practices/rules/rerender-use-ref-transient-values.md +73 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-after-nonblocking.md +73 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-auth-actions.md +96 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-cache-lru.md +41 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-cache-react.md +76 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-dedup-props.md +65 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-parallel-fetching.md +83 -0
- package/.agents/skills/vercel-react-best-practices/rules/server-serialization.md +38 -0
- package/.claude/settings.json +57 -0
- package/.claude/settings.local.json +88 -0
- package/.claude/skills/agent-prompt/SKILL.md +54 -0
- package/.claude/skills/agent-prompt/references/agentic-patterns.md +94 -0
- package/.claude/skills/agent-prompt/references/anti-patterns.md +140 -0
- package/.claude/skills/agent-prompt/references/context-design.md +124 -0
- package/.claude/skills/agent-prompt/references/core-principles.md +75 -0
- package/.claude/skills/agent-prompt/references/model-guidance.md +118 -0
- package/.claude/skills/agent-prompt/references/output-formats.md +98 -0
- package/.claude/skills/agent-prompt/references/skill-structure.md +115 -0
- package/.claude/skills/agent-prompt/references/system-prompts.md +115 -0
- package/.claude/skills/notseer/SKILL.md +131 -0
- package/.claude/skills/skill-writer/SKILL.md +140 -0
- package/.claude/skills/testing-guidelines/SKILL.md +132 -0
- package/.claude/skills/warden-skill/SKILL.md +250 -0
- package/.claude/skills/warden-skill/references/config-schema.md +133 -0
- package/.dex/config.toml +2 -0
- package/.github/workflows/ci.yml +33 -0
- package/.github/workflows/release.yml +54 -0
- package/.github/workflows/warden.yml +40 -0
- package/AGENTS.md +89 -0
- package/CONTRIBUTING.md +60 -0
- package/LICENSE +105 -0
- package/README.md +43 -0
- package/SPEC.md +263 -0
- package/action.yml +87 -0
- package/assets/favicon.png +0 -0
- package/assets/warden-icon-bw.svg +5 -0
- package/assets/warden-icon-purple.png +0 -0
- package/assets/warden-icon-purple.svg +5 -0
- package/docs/.claude/settings.local.json +11 -0
- package/docs/astro.config.mjs +43 -0
- package/docs/package.json +19 -0
- package/docs/pnpm-lock.yaml +4000 -0
- package/docs/public/favicon.svg +5 -0
- package/docs/src/components/Code.astro +141 -0
- package/docs/src/components/PackageManagerTabs.astro +183 -0
- package/docs/src/components/Terminal.astro +212 -0
- package/docs/src/layouts/Base.astro +380 -0
- package/docs/src/pages/cli.astro +167 -0
- package/docs/src/pages/config.astro +394 -0
- package/docs/src/pages/guide.astro +449 -0
- package/docs/src/pages/index.astro +490 -0
- package/docs/src/styles/global.css +551 -0
- package/docs/tsconfig.json +3 -0
- package/docs/vercel.json +5 -0
- package/eslint.config.js +33 -0
- package/package.json +73 -0
- package/src/action/index.ts +1 -0
- package/src/action/main.ts +868 -0
- package/src/cli/args.test.ts +477 -0
- package/src/cli/args.ts +415 -0
- package/src/cli/commands/add.ts +447 -0
- package/src/cli/commands/init.test.ts +136 -0
- package/src/cli/commands/init.ts +132 -0
- package/src/cli/commands/setup-app/browser.ts +38 -0
- package/src/cli/commands/setup-app/credentials.ts +45 -0
- package/src/cli/commands/setup-app/manifest.ts +48 -0
- package/src/cli/commands/setup-app/server.ts +172 -0
- package/src/cli/commands/setup-app.ts +156 -0
- package/src/cli/commands/sync.ts +114 -0
- package/src/cli/context.ts +131 -0
- package/src/cli/files.test.ts +155 -0
- package/src/cli/files.ts +89 -0
- package/src/cli/fix.test.ts +310 -0
- package/src/cli/fix.ts +387 -0
- package/src/cli/git.test.ts +119 -0
- package/src/cli/git.ts +318 -0
- package/src/cli/index.ts +14 -0
- package/src/cli/main.ts +672 -0
- package/src/cli/output/box.ts +235 -0
- package/src/cli/output/formatters.test.ts +187 -0
- package/src/cli/output/formatters.ts +269 -0
- package/src/cli/output/icons.ts +13 -0
- package/src/cli/output/index.ts +44 -0
- package/src/cli/output/ink-runner.tsx +337 -0
- package/src/cli/output/jsonl.test.ts +347 -0
- package/src/cli/output/jsonl.ts +126 -0
- package/src/cli/output/reporter.ts +435 -0
- package/src/cli/output/tasks.ts +374 -0
- package/src/cli/output/tty.test.ts +117 -0
- package/src/cli/output/tty.ts +60 -0
- package/src/cli/output/verbosity.test.ts +40 -0
- package/src/cli/output/verbosity.ts +31 -0
- package/src/cli/terminal.test.ts +148 -0
- package/src/cli/terminal.ts +301 -0
- package/src/config/index.ts +3 -0
- package/src/config/loader.test.ts +313 -0
- package/src/config/loader.ts +103 -0
- package/src/config/schema.ts +168 -0
- package/src/config/writer.test.ts +119 -0
- package/src/config/writer.ts +84 -0
- package/src/diff/classify.test.ts +162 -0
- package/src/diff/classify.ts +92 -0
- package/src/diff/coalesce.test.ts +208 -0
- package/src/diff/coalesce.ts +133 -0
- package/src/diff/context.test.ts +226 -0
- package/src/diff/context.ts +201 -0
- package/src/diff/index.ts +4 -0
- package/src/diff/parser.test.ts +212 -0
- package/src/diff/parser.ts +149 -0
- package/src/event/context.ts +132 -0
- package/src/event/index.ts +2 -0
- package/src/event/schedule-context.ts +101 -0
- package/src/examples/examples.integration.test.ts +66 -0
- package/src/examples/index.test.ts +101 -0
- package/src/examples/index.ts +122 -0
- package/src/examples/setup.ts +25 -0
- package/src/index.ts +115 -0
- package/src/output/dedup.test.ts +419 -0
- package/src/output/dedup.ts +607 -0
- package/src/output/github-checks.test.ts +300 -0
- package/src/output/github-checks.ts +476 -0
- package/src/output/github-issues.ts +329 -0
- package/src/output/index.ts +5 -0
- package/src/output/issue-renderer.ts +197 -0
- package/src/output/renderer.test.ts +727 -0
- package/src/output/renderer.ts +217 -0
- package/src/output/stale.test.ts +375 -0
- package/src/output/stale.ts +155 -0
- package/src/output/types.ts +34 -0
- package/src/sdk/index.ts +1 -0
- package/src/sdk/runner.test.ts +806 -0
- package/src/sdk/runner.ts +1232 -0
- package/src/skills/index.ts +36 -0
- package/src/skills/loader.test.ts +300 -0
- package/src/skills/loader.ts +423 -0
- package/src/skills/remote.test.ts +704 -0
- package/src/skills/remote.ts +604 -0
- package/src/triggers/matcher.test.ts +277 -0
- package/src/triggers/matcher.ts +152 -0
- package/src/types/index.ts +194 -0
- package/src/utils/async.ts +18 -0
- package/src/utils/index.test.ts +84 -0
- package/src/utils/index.ts +50 -0
- package/tsconfig.json +25 -0
- package/vitest.config.ts +8 -0
- package/vitest.integration.config.ts +11 -0
- package/warden.toml +19 -0
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
<svg width="32" height="32" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
|
|
2
|
+
<rect width="24" height="24" fill="white"/>
|
|
3
|
+
<path d="M14.2666 4.9061C15.1564 4.09409 16.5921 4.60917 16.7627 5.80162L18.1123 15.2519C18.9476 14.5758 19.6675 14.156 20.2706 13.9423C20.9583 13.6988 21.61 13.6893 22.0606 14.0459C22.5331 14.4202 22.576 15.0232 22.4131 15.5224C22.2471 16.0308 21.8525 16.5326 21.2774 16.916C20.2382 17.6088 19.1848 18.1368 17.7432 18.4863C16.3131 18.833 14.518 19 12 19C9.48202 19 7.68685 18.833 6.25679 18.4863C4.81521 18.1368 3.76183 17.6088 2.72259 16.916C2.14753 16.5326 1.75289 16.0308 1.58684 15.5224C1.42401 15.0233 1.46701 14.4202 1.93938 14.0459C2.38997 13.6893 3.04168 13.6988 3.72943 13.9423C4.33238 14.1559 5.05163 14.5761 5.88667 15.2519L7.23727 5.80162C7.40784 4.60919 8.84362 4.09419 9.73338 4.9061L11.6631 6.66686C11.854 6.84121 12.146 6.84121 12.3369 6.66686L14.2666 4.9061Z" fill="#181225"/>
|
|
4
|
+
<path d="M12 8L13.1226 11.1094H16.7553L13.8164 13.0312L14.9389 16.1406L12 14.2188L9.06107 16.1406L10.1836 13.0312L7.24472 11.1094H10.8774L12 8Z" fill="white"/>
|
|
5
|
+
</svg>
|
|
@@ -0,0 +1,141 @@
|
|
|
1
|
+
---
|
|
2
|
+
interface Props {
|
|
3
|
+
code: string;
|
|
4
|
+
lang?: string;
|
|
5
|
+
title?: string;
|
|
6
|
+
}
|
|
7
|
+
|
|
8
|
+
const { code, lang = 'bash', title } = Astro.props;
|
|
9
|
+
---
|
|
10
|
+
|
|
11
|
+
<div class="code-block">
|
|
12
|
+
<div class="code-header">
|
|
13
|
+
{title && <div class="code-title">{title}</div>}
|
|
14
|
+
<button class="copy-button" aria-label="Copy to clipboard">
|
|
15
|
+
<svg class="copy-icon" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
|
16
|
+
<rect x="9" y="9" width="13" height="13" rx="2" ry="2"></rect>
|
|
17
|
+
<path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"></path>
|
|
18
|
+
</svg>
|
|
19
|
+
<svg class="check-icon" width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
|
20
|
+
<polyline points="20 6 9 17 4 12"></polyline>
|
|
21
|
+
</svg>
|
|
22
|
+
</button>
|
|
23
|
+
</div>
|
|
24
|
+
<pre><code class={`language-${lang}`} set:text={code} /></pre>
|
|
25
|
+
</div>
|
|
26
|
+
|
|
27
|
+
<script>
|
|
28
|
+
function setupCodeCopyButtons() {
|
|
29
|
+
document.querySelectorAll('.code-block').forEach((block) => {
|
|
30
|
+
const button = block.querySelector('.copy-button');
|
|
31
|
+
const code = block.querySelector('code');
|
|
32
|
+
|
|
33
|
+
if (!button || !code) return;
|
|
34
|
+
|
|
35
|
+
button.addEventListener('click', async () => {
|
|
36
|
+
const textToCopy = code.textContent || '';
|
|
37
|
+
|
|
38
|
+
try {
|
|
39
|
+
await navigator.clipboard.writeText(textToCopy.trim());
|
|
40
|
+
|
|
41
|
+
// Show success state
|
|
42
|
+
button.classList.add('copied');
|
|
43
|
+
setTimeout(() => {
|
|
44
|
+
button.classList.remove('copied');
|
|
45
|
+
}, 2000);
|
|
46
|
+
} catch (err) {
|
|
47
|
+
console.error('Failed to copy:', err);
|
|
48
|
+
}
|
|
49
|
+
});
|
|
50
|
+
});
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
// Run on initial load
|
|
54
|
+
setupCodeCopyButtons();
|
|
55
|
+
|
|
56
|
+
// Re-run after page navigation
|
|
57
|
+
document.addEventListener('astro:page-load', setupCodeCopyButtons);
|
|
58
|
+
</script>
|
|
59
|
+
|
|
60
|
+
<style>
|
|
61
|
+
.code-block {
|
|
62
|
+
margin: 1rem 0;
|
|
63
|
+
position: relative;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
.code-header {
|
|
67
|
+
display: flex;
|
|
68
|
+
align-items: center;
|
|
69
|
+
justify-content: space-between;
|
|
70
|
+
gap: 8px;
|
|
71
|
+
background: var(--bg-subtle);
|
|
72
|
+
border: 1px solid var(--border);
|
|
73
|
+
border-bottom: none;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
.code-title {
|
|
77
|
+
font-family: var(--font-mono);
|
|
78
|
+
font-size: 0.75rem;
|
|
79
|
+
color: var(--text-muted);
|
|
80
|
+
padding: 0.5rem 1rem;
|
|
81
|
+
text-transform: uppercase;
|
|
82
|
+
letter-spacing: 0.05em;
|
|
83
|
+
flex: 1;
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
.code-header:has(.code-title) + pre {
|
|
87
|
+
border-top-left-radius: 0;
|
|
88
|
+
border-top-right-radius: 0;
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
.code-header:not(:has(.code-title)) {
|
|
92
|
+
background: transparent;
|
|
93
|
+
border: none;
|
|
94
|
+
position: absolute;
|
|
95
|
+
top: 0;
|
|
96
|
+
right: 0;
|
|
97
|
+
z-index: 1;
|
|
98
|
+
padding: 0.5rem;
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
.copy-button {
|
|
102
|
+
background: rgba(0, 0, 0, 0.3);
|
|
103
|
+
border: 1px solid rgba(255, 255, 255, 0.1);
|
|
104
|
+
border-radius: 4px;
|
|
105
|
+
padding: 4px 8px;
|
|
106
|
+
margin-right: 0.5rem;
|
|
107
|
+
cursor: pointer;
|
|
108
|
+
color: var(--text-muted);
|
|
109
|
+
display: flex;
|
|
110
|
+
align-items: center;
|
|
111
|
+
gap: 4px;
|
|
112
|
+
transition: all 0.2s;
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
.copy-button:hover {
|
|
116
|
+
background: rgba(0, 0, 0, 0.5);
|
|
117
|
+
border-color: rgba(255, 255, 255, 0.2);
|
|
118
|
+
color: var(--text);
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
.copy-button .check-icon {
|
|
122
|
+
display: none;
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
.copy-button.copied {
|
|
126
|
+
border-color: #4ade80;
|
|
127
|
+
color: #4ade80;
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
.copy-button.copied .copy-icon {
|
|
131
|
+
display: none;
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
.copy-button.copied .check-icon {
|
|
135
|
+
display: block;
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
.code-block pre {
|
|
139
|
+
margin: 0;
|
|
140
|
+
}
|
|
141
|
+
</style>
|
|
@@ -0,0 +1,183 @@
|
|
|
1
|
+
---
|
|
2
|
+
interface Props {
|
|
3
|
+
npm: string;
|
|
4
|
+
pnpm: string;
|
|
5
|
+
bun: string;
|
|
6
|
+
}
|
|
7
|
+
|
|
8
|
+
const { npm, pnpm, bun } = Astro.props;
|
|
9
|
+
|
|
10
|
+
// Format multi-line commands with $ prompt on each line
|
|
11
|
+
function formatCommand(cmd: string): string {
|
|
12
|
+
return cmd
|
|
13
|
+
.split('\n')
|
|
14
|
+
.map(line => `<span class="cli-dim">$</span> ${line}`)
|
|
15
|
+
.join('\n');
|
|
16
|
+
}
|
|
17
|
+
---
|
|
18
|
+
|
|
19
|
+
<div class="pm-tabs">
|
|
20
|
+
<div class="terminal">
|
|
21
|
+
<div class="terminal-header">
|
|
22
|
+
<div class="terminal-dots">
|
|
23
|
+
<span class="dot red"></span>
|
|
24
|
+
<span class="dot yellow"></span>
|
|
25
|
+
<span class="dot green"></span>
|
|
26
|
+
</div>
|
|
27
|
+
<div class="tab-buttons">
|
|
28
|
+
<button class="tab-btn active" data-pm="npm">npm</button>
|
|
29
|
+
<button class="tab-btn" data-pm="pnpm">pnpm</button>
|
|
30
|
+
<button class="tab-btn" data-pm="bun">bun</button>
|
|
31
|
+
</div>
|
|
32
|
+
<button class="copy-button" aria-label="Copy to clipboard">
|
|
33
|
+
<svg class="copy-icon" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
|
34
|
+
<rect x="9" y="9" width="13" height="13" rx="2" ry="2"></rect>
|
|
35
|
+
<path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"></path>
|
|
36
|
+
</svg>
|
|
37
|
+
<svg class="check-icon" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
|
38
|
+
<polyline points="20 6 9 17 4 12"></polyline>
|
|
39
|
+
</svg>
|
|
40
|
+
</button>
|
|
41
|
+
</div>
|
|
42
|
+
<div class="terminal-body">
|
|
43
|
+
<pre class="tab-content cli-output" data-pm="npm" set:html={formatCommand(npm)} />
|
|
44
|
+
<pre class="tab-content cli-output hidden" data-pm="pnpm" set:html={formatCommand(pnpm)} />
|
|
45
|
+
<pre class="tab-content cli-output hidden" data-pm="bun" set:html={formatCommand(bun)} />
|
|
46
|
+
</div>
|
|
47
|
+
</div>
|
|
48
|
+
</div>
|
|
49
|
+
|
|
50
|
+
<style>
|
|
51
|
+
.pm-tabs {
|
|
52
|
+
margin: 1.5rem 0;
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
.terminal {
|
|
56
|
+
background: #000;
|
|
57
|
+
border: 1px solid rgba(255, 255, 255, 0.1);
|
|
58
|
+
border-radius: 8px;
|
|
59
|
+
overflow: hidden;
|
|
60
|
+
box-shadow:
|
|
61
|
+
0 0 0 1px rgba(255, 255, 255, 0.05),
|
|
62
|
+
0 20px 50px -10px rgba(0, 0, 0, 0.7);
|
|
63
|
+
position: relative;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
.terminal-header {
|
|
67
|
+
display: flex;
|
|
68
|
+
align-items: center;
|
|
69
|
+
padding: 8px 12px;
|
|
70
|
+
background: #0a0a0a;
|
|
71
|
+
border-bottom: 1px solid rgba(255, 255, 255, 0.06);
|
|
72
|
+
gap: 8px;
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
.terminal-dots {
|
|
76
|
+
display: flex;
|
|
77
|
+
gap: 6px;
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
.dot {
|
|
81
|
+
width: 10px;
|
|
82
|
+
height: 10px;
|
|
83
|
+
border-radius: 50%;
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
.dot.red {
|
|
87
|
+
background: #ff5f57;
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
.dot.yellow {
|
|
91
|
+
background: #febc2e;
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
.dot.green {
|
|
95
|
+
background: #28c840;
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
.tab-buttons {
|
|
99
|
+
display: flex;
|
|
100
|
+
gap: 2px;
|
|
101
|
+
margin-left: 8px;
|
|
102
|
+
background: rgba(255, 255, 255, 0.05);
|
|
103
|
+
border-radius: 4px;
|
|
104
|
+
padding: 2px;
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
.tab-btn {
|
|
108
|
+
font-family: var(--font-mono);
|
|
109
|
+
font-size: 0.7rem;
|
|
110
|
+
color: #888;
|
|
111
|
+
background: transparent;
|
|
112
|
+
border: none;
|
|
113
|
+
padding: 4px 10px;
|
|
114
|
+
cursor: pointer;
|
|
115
|
+
border-radius: 3px;
|
|
116
|
+
transition: all 0.15s;
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
.tab-btn:hover {
|
|
120
|
+
color: #ccc;
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
.tab-btn.active {
|
|
124
|
+
background: rgba(255, 255, 255, 0.1);
|
|
125
|
+
color: #fff;
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
.copy-button {
|
|
129
|
+
margin-left: auto;
|
|
130
|
+
background: transparent;
|
|
131
|
+
border: 1px solid rgba(255, 255, 255, 0.1);
|
|
132
|
+
border-radius: 4px;
|
|
133
|
+
padding: 4px 8px;
|
|
134
|
+
cursor: pointer;
|
|
135
|
+
color: #888;
|
|
136
|
+
display: flex;
|
|
137
|
+
align-items: center;
|
|
138
|
+
gap: 4px;
|
|
139
|
+
transition: all 0.2s;
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
.copy-button:hover {
|
|
143
|
+
background: rgba(255, 255, 255, 0.05);
|
|
144
|
+
border-color: rgba(255, 255, 255, 0.2);
|
|
145
|
+
color: #fff;
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
.copy-button .check-icon {
|
|
149
|
+
display: none;
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
.copy-button.copied {
|
|
153
|
+
border-color: #4ade80;
|
|
154
|
+
color: #4ade80;
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
.copy-button.copied .copy-icon {
|
|
158
|
+
display: none;
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
.copy-button.copied .check-icon {
|
|
162
|
+
display: block;
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
.terminal-body {
|
|
166
|
+
padding: 1rem 1.25rem;
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
/* Override global pre styles */
|
|
170
|
+
.terminal-body pre {
|
|
171
|
+
background: transparent !important;
|
|
172
|
+
border: none !important;
|
|
173
|
+
border-radius: 0 !important;
|
|
174
|
+
padding: 0 !important;
|
|
175
|
+
margin: 0 !important;
|
|
176
|
+
box-shadow: none !important;
|
|
177
|
+
overflow: hidden !important;
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
.tab-content.hidden {
|
|
181
|
+
display: none;
|
|
182
|
+
}
|
|
183
|
+
</style>
|
|
@@ -0,0 +1,212 @@
|
|
|
1
|
+
---
|
|
2
|
+
interface Props {
|
|
3
|
+
title?: string;
|
|
4
|
+
showCopy?: boolean;
|
|
5
|
+
copyText?: string;
|
|
6
|
+
}
|
|
7
|
+
|
|
8
|
+
const { title, showCopy = false, copyText } = Astro.props;
|
|
9
|
+
---
|
|
10
|
+
|
|
11
|
+
<div class="terminal" data-copy-text={copyText}>
|
|
12
|
+
<div class="terminal-header">
|
|
13
|
+
<div class="terminal-dots">
|
|
14
|
+
<span class="dot red"></span>
|
|
15
|
+
<span class="dot yellow"></span>
|
|
16
|
+
<span class="dot green"></span>
|
|
17
|
+
</div>
|
|
18
|
+
{title && <span class="terminal-title">{title}</span>}
|
|
19
|
+
{showCopy && (
|
|
20
|
+
<button class="copy-button" aria-label="Copy to clipboard">
|
|
21
|
+
<svg class="copy-icon" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
|
22
|
+
<rect x="9" y="9" width="13" height="13" rx="2" ry="2"></rect>
|
|
23
|
+
<path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"></path>
|
|
24
|
+
</svg>
|
|
25
|
+
<svg class="check-icon" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
|
|
26
|
+
<polyline points="20 6 9 17 4 12"></polyline>
|
|
27
|
+
</svg>
|
|
28
|
+
</button>
|
|
29
|
+
)}
|
|
30
|
+
</div>
|
|
31
|
+
<div class="terminal-body">
|
|
32
|
+
<slot />
|
|
33
|
+
</div>
|
|
34
|
+
</div>
|
|
35
|
+
|
|
36
|
+
<script>
|
|
37
|
+
function setupCopyButtons() {
|
|
38
|
+
document.querySelectorAll('.terminal').forEach((terminal) => {
|
|
39
|
+
const button = terminal.querySelector('.copy-button');
|
|
40
|
+
const body = terminal.querySelector('.terminal-body');
|
|
41
|
+
|
|
42
|
+
if (!button || !body) return;
|
|
43
|
+
|
|
44
|
+
button.addEventListener('click', async () => {
|
|
45
|
+
// Check for custom copy text
|
|
46
|
+
const customCopyText = terminal.getAttribute('data-copy-text');
|
|
47
|
+
|
|
48
|
+
let textToCopy: string;
|
|
49
|
+
|
|
50
|
+
if (customCopyText) {
|
|
51
|
+
textToCopy = customCopyText;
|
|
52
|
+
} else {
|
|
53
|
+
// Get only visible text content (respects display: none)
|
|
54
|
+
function getVisibleText(element: Element): string {
|
|
55
|
+
if (getComputedStyle(element).display === 'none') return '';
|
|
56
|
+
let text = '';
|
|
57
|
+
for (const node of element.childNodes) {
|
|
58
|
+
if (node.nodeType === Node.TEXT_NODE) {
|
|
59
|
+
text += node.textContent;
|
|
60
|
+
} else if (node.nodeType === Node.ELEMENT_NODE) {
|
|
61
|
+
text += getVisibleText(node as Element);
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
return text;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
const content = getVisibleText(body);
|
|
68
|
+
|
|
69
|
+
// Split into lines and process each line
|
|
70
|
+
const lines = content.split('\n');
|
|
71
|
+
const cleanedLines = lines.map(line => {
|
|
72
|
+
// Remove leading $ or > prompts (with optional spaces before/after)
|
|
73
|
+
return line.replace(/^\s*[$>]\s*/, '');
|
|
74
|
+
});
|
|
75
|
+
|
|
76
|
+
// Join back together and trim
|
|
77
|
+
textToCopy = cleanedLines.join('\n').trim();
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
// Copy to clipboard
|
|
81
|
+
try {
|
|
82
|
+
await navigator.clipboard.writeText(textToCopy);
|
|
83
|
+
|
|
84
|
+
// Show success state
|
|
85
|
+
button.classList.add('copied');
|
|
86
|
+
setTimeout(() => {
|
|
87
|
+
button.classList.remove('copied');
|
|
88
|
+
}, 2000);
|
|
89
|
+
} catch (err) {
|
|
90
|
+
console.error('Failed to copy:', err);
|
|
91
|
+
}
|
|
92
|
+
});
|
|
93
|
+
});
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
// Run on initial load
|
|
97
|
+
setupCopyButtons();
|
|
98
|
+
|
|
99
|
+
// Re-run after page navigation (for SPA-like behavior)
|
|
100
|
+
document.addEventListener('astro:page-load', setupCopyButtons);
|
|
101
|
+
</script>
|
|
102
|
+
|
|
103
|
+
<style>
|
|
104
|
+
.terminal {
|
|
105
|
+
background: #000;
|
|
106
|
+
border: 1px solid rgba(255, 255, 255, 0.1);
|
|
107
|
+
border-radius: 8px;
|
|
108
|
+
overflow: hidden;
|
|
109
|
+
margin: 1.5rem 0;
|
|
110
|
+
box-shadow:
|
|
111
|
+
0 0 0 1px rgba(255, 255, 255, 0.05),
|
|
112
|
+
0 20px 50px -10px rgba(0, 0, 0, 0.7);
|
|
113
|
+
position: relative;
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
.terminal-header {
|
|
117
|
+
display: flex;
|
|
118
|
+
align-items: center;
|
|
119
|
+
padding: 8px 12px;
|
|
120
|
+
background: #0a0a0a;
|
|
121
|
+
border-bottom: 1px solid rgba(255, 255, 255, 0.06);
|
|
122
|
+
gap: 8px;
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
.terminal-dots {
|
|
126
|
+
display: flex;
|
|
127
|
+
gap: 6px;
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
.dot {
|
|
131
|
+
width: 10px;
|
|
132
|
+
height: 10px;
|
|
133
|
+
border-radius: 50%;
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
.dot.red {
|
|
137
|
+
background: #ff5f57;
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
.dot.yellow {
|
|
141
|
+
background: #febc2e;
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
.dot.green {
|
|
145
|
+
background: #28c840;
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
.terminal-title {
|
|
149
|
+
font-family: var(--font-mono);
|
|
150
|
+
font-size: 0.75rem;
|
|
151
|
+
color: #888;
|
|
152
|
+
margin-left: 4px;
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
.copy-button {
|
|
156
|
+
margin-left: auto;
|
|
157
|
+
background: transparent;
|
|
158
|
+
border: 1px solid rgba(255, 255, 255, 0.1);
|
|
159
|
+
border-radius: 4px;
|
|
160
|
+
padding: 4px 8px;
|
|
161
|
+
cursor: pointer;
|
|
162
|
+
color: #888;
|
|
163
|
+
display: flex;
|
|
164
|
+
align-items: center;
|
|
165
|
+
gap: 4px;
|
|
166
|
+
transition: all 0.2s;
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
.copy-button:hover {
|
|
170
|
+
background: rgba(255, 255, 255, 0.05);
|
|
171
|
+
border-color: rgba(255, 255, 255, 0.2);
|
|
172
|
+
color: #fff;
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
.copy-button .check-icon {
|
|
176
|
+
display: none;
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
.copy-button.copied {
|
|
180
|
+
border-color: #4ade80;
|
|
181
|
+
color: #4ade80;
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
.copy-button.copied .copy-icon {
|
|
185
|
+
display: none;
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
.copy-button.copied .check-icon {
|
|
189
|
+
display: block;
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
.terminal-body {
|
|
193
|
+
padding: 1rem 1.25rem;
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
.terminal-body :global(pre),
|
|
197
|
+
.terminal-body :global(pre.astro-code) {
|
|
198
|
+
margin: 0 !important;
|
|
199
|
+
padding: 0 !important;
|
|
200
|
+
background: transparent !important;
|
|
201
|
+
border: none !important;
|
|
202
|
+
box-shadow: none !important;
|
|
203
|
+
border-radius: 0 !important;
|
|
204
|
+
overflow: hidden !important;
|
|
205
|
+
}
|
|
206
|
+
|
|
207
|
+
.terminal-body :global(code) {
|
|
208
|
+
background: transparent !important;
|
|
209
|
+
border: none !important;
|
|
210
|
+
padding: 0 !important;
|
|
211
|
+
}
|
|
212
|
+
</style>
|