@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.
Files changed (199) hide show
  1. package/.agents/skills/find-bugs/SKILL.md +75 -0
  2. package/.agents/skills/vercel-react-best-practices/AGENTS.md +2934 -0
  3. package/.agents/skills/vercel-react-best-practices/SKILL.md +136 -0
  4. package/.agents/skills/vercel-react-best-practices/rules/advanced-event-handler-refs.md +55 -0
  5. package/.agents/skills/vercel-react-best-practices/rules/advanced-init-once.md +42 -0
  6. package/.agents/skills/vercel-react-best-practices/rules/advanced-use-latest.md +39 -0
  7. package/.agents/skills/vercel-react-best-practices/rules/async-api-routes.md +38 -0
  8. package/.agents/skills/vercel-react-best-practices/rules/async-defer-await.md +80 -0
  9. package/.agents/skills/vercel-react-best-practices/rules/async-dependencies.md +51 -0
  10. package/.agents/skills/vercel-react-best-practices/rules/async-parallel.md +28 -0
  11. package/.agents/skills/vercel-react-best-practices/rules/async-suspense-boundaries.md +99 -0
  12. package/.agents/skills/vercel-react-best-practices/rules/bundle-barrel-imports.md +59 -0
  13. package/.agents/skills/vercel-react-best-practices/rules/bundle-conditional.md +31 -0
  14. package/.agents/skills/vercel-react-best-practices/rules/bundle-defer-third-party.md +49 -0
  15. package/.agents/skills/vercel-react-best-practices/rules/bundle-dynamic-imports.md +35 -0
  16. package/.agents/skills/vercel-react-best-practices/rules/bundle-preload.md +50 -0
  17. package/.agents/skills/vercel-react-best-practices/rules/client-event-listeners.md +74 -0
  18. package/.agents/skills/vercel-react-best-practices/rules/client-localstorage-schema.md +71 -0
  19. package/.agents/skills/vercel-react-best-practices/rules/client-passive-event-listeners.md +48 -0
  20. package/.agents/skills/vercel-react-best-practices/rules/client-swr-dedup.md +56 -0
  21. package/.agents/skills/vercel-react-best-practices/rules/js-batch-dom-css.md +107 -0
  22. package/.agents/skills/vercel-react-best-practices/rules/js-cache-function-results.md +80 -0
  23. package/.agents/skills/vercel-react-best-practices/rules/js-cache-property-access.md +28 -0
  24. package/.agents/skills/vercel-react-best-practices/rules/js-cache-storage.md +70 -0
  25. package/.agents/skills/vercel-react-best-practices/rules/js-combine-iterations.md +32 -0
  26. package/.agents/skills/vercel-react-best-practices/rules/js-early-exit.md +50 -0
  27. package/.agents/skills/vercel-react-best-practices/rules/js-hoist-regexp.md +45 -0
  28. package/.agents/skills/vercel-react-best-practices/rules/js-index-maps.md +37 -0
  29. package/.agents/skills/vercel-react-best-practices/rules/js-length-check-first.md +49 -0
  30. package/.agents/skills/vercel-react-best-practices/rules/js-min-max-loop.md +82 -0
  31. package/.agents/skills/vercel-react-best-practices/rules/js-set-map-lookups.md +24 -0
  32. package/.agents/skills/vercel-react-best-practices/rules/js-tosorted-immutable.md +57 -0
  33. package/.agents/skills/vercel-react-best-practices/rules/rendering-activity.md +26 -0
  34. package/.agents/skills/vercel-react-best-practices/rules/rendering-animate-svg-wrapper.md +47 -0
  35. package/.agents/skills/vercel-react-best-practices/rules/rendering-conditional-render.md +40 -0
  36. package/.agents/skills/vercel-react-best-practices/rules/rendering-content-visibility.md +38 -0
  37. package/.agents/skills/vercel-react-best-practices/rules/rendering-hoist-jsx.md +46 -0
  38. package/.agents/skills/vercel-react-best-practices/rules/rendering-hydration-no-flicker.md +82 -0
  39. package/.agents/skills/vercel-react-best-practices/rules/rendering-hydration-suppress-warning.md +30 -0
  40. package/.agents/skills/vercel-react-best-practices/rules/rendering-svg-precision.md +28 -0
  41. package/.agents/skills/vercel-react-best-practices/rules/rendering-usetransition-loading.md +75 -0
  42. package/.agents/skills/vercel-react-best-practices/rules/rerender-defer-reads.md +39 -0
  43. package/.agents/skills/vercel-react-best-practices/rules/rerender-dependencies.md +45 -0
  44. package/.agents/skills/vercel-react-best-practices/rules/rerender-derived-state-no-effect.md +40 -0
  45. package/.agents/skills/vercel-react-best-practices/rules/rerender-derived-state.md +29 -0
  46. package/.agents/skills/vercel-react-best-practices/rules/rerender-functional-setstate.md +74 -0
  47. package/.agents/skills/vercel-react-best-practices/rules/rerender-lazy-state-init.md +58 -0
  48. package/.agents/skills/vercel-react-best-practices/rules/rerender-memo-with-default-value.md +38 -0
  49. package/.agents/skills/vercel-react-best-practices/rules/rerender-memo.md +44 -0
  50. package/.agents/skills/vercel-react-best-practices/rules/rerender-move-effect-to-event.md +45 -0
  51. package/.agents/skills/vercel-react-best-practices/rules/rerender-simple-expression-in-memo.md +35 -0
  52. package/.agents/skills/vercel-react-best-practices/rules/rerender-transitions.md +40 -0
  53. package/.agents/skills/vercel-react-best-practices/rules/rerender-use-ref-transient-values.md +73 -0
  54. package/.agents/skills/vercel-react-best-practices/rules/server-after-nonblocking.md +73 -0
  55. package/.agents/skills/vercel-react-best-practices/rules/server-auth-actions.md +96 -0
  56. package/.agents/skills/vercel-react-best-practices/rules/server-cache-lru.md +41 -0
  57. package/.agents/skills/vercel-react-best-practices/rules/server-cache-react.md +76 -0
  58. package/.agents/skills/vercel-react-best-practices/rules/server-dedup-props.md +65 -0
  59. package/.agents/skills/vercel-react-best-practices/rules/server-parallel-fetching.md +83 -0
  60. package/.agents/skills/vercel-react-best-practices/rules/server-serialization.md +38 -0
  61. package/.claude/settings.json +57 -0
  62. package/.claude/settings.local.json +88 -0
  63. package/.claude/skills/agent-prompt/SKILL.md +54 -0
  64. package/.claude/skills/agent-prompt/references/agentic-patterns.md +94 -0
  65. package/.claude/skills/agent-prompt/references/anti-patterns.md +140 -0
  66. package/.claude/skills/agent-prompt/references/context-design.md +124 -0
  67. package/.claude/skills/agent-prompt/references/core-principles.md +75 -0
  68. package/.claude/skills/agent-prompt/references/model-guidance.md +118 -0
  69. package/.claude/skills/agent-prompt/references/output-formats.md +98 -0
  70. package/.claude/skills/agent-prompt/references/skill-structure.md +115 -0
  71. package/.claude/skills/agent-prompt/references/system-prompts.md +115 -0
  72. package/.claude/skills/notseer/SKILL.md +131 -0
  73. package/.claude/skills/skill-writer/SKILL.md +140 -0
  74. package/.claude/skills/testing-guidelines/SKILL.md +132 -0
  75. package/.claude/skills/warden-skill/SKILL.md +250 -0
  76. package/.claude/skills/warden-skill/references/config-schema.md +133 -0
  77. package/.dex/config.toml +2 -0
  78. package/.github/workflows/ci.yml +33 -0
  79. package/.github/workflows/release.yml +54 -0
  80. package/.github/workflows/warden.yml +40 -0
  81. package/AGENTS.md +89 -0
  82. package/CONTRIBUTING.md +60 -0
  83. package/LICENSE +105 -0
  84. package/README.md +43 -0
  85. package/SPEC.md +263 -0
  86. package/action.yml +87 -0
  87. package/assets/favicon.png +0 -0
  88. package/assets/warden-icon-bw.svg +5 -0
  89. package/assets/warden-icon-purple.png +0 -0
  90. package/assets/warden-icon-purple.svg +5 -0
  91. package/docs/.claude/settings.local.json +11 -0
  92. package/docs/astro.config.mjs +43 -0
  93. package/docs/package.json +19 -0
  94. package/docs/pnpm-lock.yaml +4000 -0
  95. package/docs/public/favicon.svg +5 -0
  96. package/docs/src/components/Code.astro +141 -0
  97. package/docs/src/components/PackageManagerTabs.astro +183 -0
  98. package/docs/src/components/Terminal.astro +212 -0
  99. package/docs/src/layouts/Base.astro +380 -0
  100. package/docs/src/pages/cli.astro +167 -0
  101. package/docs/src/pages/config.astro +394 -0
  102. package/docs/src/pages/guide.astro +449 -0
  103. package/docs/src/pages/index.astro +490 -0
  104. package/docs/src/styles/global.css +551 -0
  105. package/docs/tsconfig.json +3 -0
  106. package/docs/vercel.json +5 -0
  107. package/eslint.config.js +33 -0
  108. package/package.json +73 -0
  109. package/src/action/index.ts +1 -0
  110. package/src/action/main.ts +868 -0
  111. package/src/cli/args.test.ts +477 -0
  112. package/src/cli/args.ts +415 -0
  113. package/src/cli/commands/add.ts +447 -0
  114. package/src/cli/commands/init.test.ts +136 -0
  115. package/src/cli/commands/init.ts +132 -0
  116. package/src/cli/commands/setup-app/browser.ts +38 -0
  117. package/src/cli/commands/setup-app/credentials.ts +45 -0
  118. package/src/cli/commands/setup-app/manifest.ts +48 -0
  119. package/src/cli/commands/setup-app/server.ts +172 -0
  120. package/src/cli/commands/setup-app.ts +156 -0
  121. package/src/cli/commands/sync.ts +114 -0
  122. package/src/cli/context.ts +131 -0
  123. package/src/cli/files.test.ts +155 -0
  124. package/src/cli/files.ts +89 -0
  125. package/src/cli/fix.test.ts +310 -0
  126. package/src/cli/fix.ts +387 -0
  127. package/src/cli/git.test.ts +119 -0
  128. package/src/cli/git.ts +318 -0
  129. package/src/cli/index.ts +14 -0
  130. package/src/cli/main.ts +672 -0
  131. package/src/cli/output/box.ts +235 -0
  132. package/src/cli/output/formatters.test.ts +187 -0
  133. package/src/cli/output/formatters.ts +269 -0
  134. package/src/cli/output/icons.ts +13 -0
  135. package/src/cli/output/index.ts +44 -0
  136. package/src/cli/output/ink-runner.tsx +337 -0
  137. package/src/cli/output/jsonl.test.ts +347 -0
  138. package/src/cli/output/jsonl.ts +126 -0
  139. package/src/cli/output/reporter.ts +435 -0
  140. package/src/cli/output/tasks.ts +374 -0
  141. package/src/cli/output/tty.test.ts +117 -0
  142. package/src/cli/output/tty.ts +60 -0
  143. package/src/cli/output/verbosity.test.ts +40 -0
  144. package/src/cli/output/verbosity.ts +31 -0
  145. package/src/cli/terminal.test.ts +148 -0
  146. package/src/cli/terminal.ts +301 -0
  147. package/src/config/index.ts +3 -0
  148. package/src/config/loader.test.ts +313 -0
  149. package/src/config/loader.ts +103 -0
  150. package/src/config/schema.ts +168 -0
  151. package/src/config/writer.test.ts +119 -0
  152. package/src/config/writer.ts +84 -0
  153. package/src/diff/classify.test.ts +162 -0
  154. package/src/diff/classify.ts +92 -0
  155. package/src/diff/coalesce.test.ts +208 -0
  156. package/src/diff/coalesce.ts +133 -0
  157. package/src/diff/context.test.ts +226 -0
  158. package/src/diff/context.ts +201 -0
  159. package/src/diff/index.ts +4 -0
  160. package/src/diff/parser.test.ts +212 -0
  161. package/src/diff/parser.ts +149 -0
  162. package/src/event/context.ts +132 -0
  163. package/src/event/index.ts +2 -0
  164. package/src/event/schedule-context.ts +101 -0
  165. package/src/examples/examples.integration.test.ts +66 -0
  166. package/src/examples/index.test.ts +101 -0
  167. package/src/examples/index.ts +122 -0
  168. package/src/examples/setup.ts +25 -0
  169. package/src/index.ts +115 -0
  170. package/src/output/dedup.test.ts +419 -0
  171. package/src/output/dedup.ts +607 -0
  172. package/src/output/github-checks.test.ts +300 -0
  173. package/src/output/github-checks.ts +476 -0
  174. package/src/output/github-issues.ts +329 -0
  175. package/src/output/index.ts +5 -0
  176. package/src/output/issue-renderer.ts +197 -0
  177. package/src/output/renderer.test.ts +727 -0
  178. package/src/output/renderer.ts +217 -0
  179. package/src/output/stale.test.ts +375 -0
  180. package/src/output/stale.ts +155 -0
  181. package/src/output/types.ts +34 -0
  182. package/src/sdk/index.ts +1 -0
  183. package/src/sdk/runner.test.ts +806 -0
  184. package/src/sdk/runner.ts +1232 -0
  185. package/src/skills/index.ts +36 -0
  186. package/src/skills/loader.test.ts +300 -0
  187. package/src/skills/loader.ts +423 -0
  188. package/src/skills/remote.test.ts +704 -0
  189. package/src/skills/remote.ts +604 -0
  190. package/src/triggers/matcher.test.ts +277 -0
  191. package/src/triggers/matcher.ts +152 -0
  192. package/src/types/index.ts +194 -0
  193. package/src/utils/async.ts +18 -0
  194. package/src/utils/index.test.ts +84 -0
  195. package/src/utils/index.ts +50 -0
  196. package/tsconfig.json +25 -0
  197. package/vitest.config.ts +8 -0
  198. package/vitest.integration.config.ts +11 -0
  199. 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>