careermate 0.1.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 (124) hide show
  1. package/README.md +256 -0
  2. package/THIRD_PARTY_NOTICES.md +40 -0
  3. package/apps/mcp/src/index.ts +66 -0
  4. package/apps/web/DESIGN_GUIDE.md +105 -0
  5. package/apps/web/UI_CONTRACT.md +44 -0
  6. package/apps/web/public/app.js +118 -0
  7. package/apps/web/public/fonts/PretendardVariable.woff2 +0 -0
  8. package/apps/web/public/index.html +41 -0
  9. package/apps/web/public/lib.js +282 -0
  10. package/apps/web/public/pages/applications.js +98 -0
  11. package/apps/web/public/pages/documents.js +446 -0
  12. package/apps/web/public/pages/home.js +263 -0
  13. package/apps/web/public/pages/interview.js +230 -0
  14. package/apps/web/public/pages/jobs.js +494 -0
  15. package/apps/web/public/pages/profile.js +576 -0
  16. package/apps/web/public/pages/settings.js +233 -0
  17. package/apps/web/public/styles.css +426 -0
  18. package/apps/web/src/exports.ts +68 -0
  19. package/apps/web/src/http.ts +180 -0
  20. package/apps/web/src/index.ts +49 -0
  21. package/apps/web/src/info.ts +50 -0
  22. package/apps/web/src/routes.ts +350 -0
  23. package/apps/web/src/security.ts +102 -0
  24. package/apps/web/src/server.ts +141 -0
  25. package/apps/web/src/settings.ts +88 -0
  26. package/bin/careermate.mjs +74 -0
  27. package/dist/careermate.mcpb +0 -0
  28. package/dist/install-page/index.html +474 -0
  29. package/dist/install-page/style.css +391 -0
  30. package/dist/install-page/vercel.json +20 -0
  31. package/dist/mcp-smoke.err +3 -0
  32. package/dist/mcp.mjs +23704 -0
  33. package/dist/mcpb-stage/README.md +219 -0
  34. package/dist/mcpb-stage/dist/install-page/index.html +434 -0
  35. package/dist/mcpb-stage/dist/install-page/style.css +407 -0
  36. package/dist/mcpb-stage/dist/install-page/vercel.json +20 -0
  37. package/dist/mcpb-stage/dist/mcp.mjs +23704 -0
  38. package/dist/mcpb-stage/dist/public/app.js +118 -0
  39. package/dist/mcpb-stage/dist/public/fonts/PretendardVariable.woff2 +0 -0
  40. package/dist/mcpb-stage/dist/public/index.html +41 -0
  41. package/dist/mcpb-stage/dist/public/lib.js +282 -0
  42. package/dist/mcpb-stage/dist/public/pages/applications.js +98 -0
  43. package/dist/mcpb-stage/dist/public/pages/documents.js +446 -0
  44. package/dist/mcpb-stage/dist/public/pages/home.js +263 -0
  45. package/dist/mcpb-stage/dist/public/pages/interview.js +230 -0
  46. package/dist/mcpb-stage/dist/public/pages/jobs.js +494 -0
  47. package/dist/mcpb-stage/dist/public/pages/profile.js +576 -0
  48. package/dist/mcpb-stage/dist/public/pages/settings.js +233 -0
  49. package/dist/mcpb-stage/dist/public/styles.css +420 -0
  50. package/dist/mcpb-stage/dist/web.mjs +7240 -0
  51. package/dist/mcpb-stage/manifest.json +40 -0
  52. package/dist/public/app.js +118 -0
  53. package/dist/public/fonts/PretendardVariable.woff2 +0 -0
  54. package/dist/public/index.html +41 -0
  55. package/dist/public/lib.js +282 -0
  56. package/dist/public/pages/applications.js +98 -0
  57. package/dist/public/pages/documents.js +446 -0
  58. package/dist/public/pages/home.js +263 -0
  59. package/dist/public/pages/interview.js +230 -0
  60. package/dist/public/pages/jobs.js +494 -0
  61. package/dist/public/pages/profile.js +576 -0
  62. package/dist/public/pages/settings.js +233 -0
  63. package/dist/public/styles.css +426 -0
  64. package/dist/web.mjs +7240 -0
  65. package/docs/ARCHITECTURE.md +208 -0
  66. package/docs/CHANGES_V1.md +103 -0
  67. package/docs/DATA_MODEL.md +460 -0
  68. package/docs/DECISIONS.md +277 -0
  69. package/docs/DEMO.md +242 -0
  70. package/docs/INSTALL.md +148 -0
  71. package/docs/INSTALL_AND_USAGE.md +99 -0
  72. package/docs/MCP_TOOLS.md +233 -0
  73. package/docs/ROADMAP.md +134 -0
  74. package/docs/START_WORKFLOW.md +125 -0
  75. package/docs/SUPPORTED_AI_APPS.md +60 -0
  76. package/docs/TODO.md +57 -0
  77. package/docs/UX_NOTES.md +247 -0
  78. package/docs/WORKFLOWS.md +200 -0
  79. package/install-page/index.html +474 -0
  80. package/install-page/style.css +391 -0
  81. package/install-page/vercel.json +20 -0
  82. package/package.json +68 -0
  83. package/packages/core/src/context.ts +74 -0
  84. package/packages/core/src/index.ts +8 -0
  85. package/packages/core/src/onboarding.ts +81 -0
  86. package/packages/core/src/services.ts +146 -0
  87. package/packages/core/src/summary.ts +104 -0
  88. package/packages/db/src/connection.ts +46 -0
  89. package/packages/db/src/index.ts +22 -0
  90. package/packages/db/src/paths.ts +41 -0
  91. package/packages/db/src/repositories.ts +828 -0
  92. package/packages/db/src/runtime.ts +58 -0
  93. package/packages/db/src/schema.ts +189 -0
  94. package/packages/exporters/src/html.ts +113 -0
  95. package/packages/exporters/src/index.ts +364 -0
  96. package/packages/exporters/src/markdown.ts +178 -0
  97. package/packages/mcp-tools/src/bridge.ts +83 -0
  98. package/packages/mcp-tools/src/index.ts +8 -0
  99. package/packages/mcp-tools/src/result.ts +49 -0
  100. package/packages/mcp-tools/src/tools.ts +455 -0
  101. package/packages/parsers/src/html.ts +86 -0
  102. package/packages/parsers/src/index.ts +228 -0
  103. package/packages/parsers/src/keywords.ts +151 -0
  104. package/packages/prompts/src/humanize.ts +59 -0
  105. package/packages/prompts/src/index.ts +82 -0
  106. package/packages/prompts/src/install.ts +43 -0
  107. package/packages/prompts/src/onboarding.ts +35 -0
  108. package/packages/prompts/src/system.ts +53 -0
  109. package/packages/shared/src/enums.ts +103 -0
  110. package/packages/shared/src/index.ts +18 -0
  111. package/packages/shared/src/schemas.ts +398 -0
  112. package/packages/workflows/src/definitions.ts +107 -0
  113. package/packages/workflows/src/index.ts +39 -0
  114. package/scripts/build-dist.mjs +62 -0
  115. package/scripts/build-mcpb.mjs +70 -0
  116. package/scripts/doctor.ts +81 -0
  117. package/scripts/init.ts +342 -0
  118. package/scripts/mcp-probe.ts +55 -0
  119. package/scripts/migrate.ts +6 -0
  120. package/scripts/run.mjs +33 -0
  121. package/scripts/seed.ts +129 -0
  122. package/scripts/test.ts +117 -0
  123. package/scripts/ui-smoke.ts +73 -0
  124. package/tsconfig.json +29 -0
@@ -0,0 +1,407 @@
1
+ /* =============================================================================
2
+ CareerMate — public landing / install page.
3
+ Self-contained: no external fonts, no CDN, no network (works fully offline).
4
+
5
+ Design language synthesized from established install/registry sites
6
+ (Homebrew, Bun, npm, MCP docs), not copied from any one:
7
+ · restraint over decoration — one flat accent, warm-neutral paper, no
8
+ gradients / glows / gradient-text
9
+ · the "command" (here: the prompt) is the single focal point
10
+ · system sans for prose, monospace for commands → reads as a real tool
11
+ · trust through whitespace, hairlines and consistency
12
+ Light + dark (system preference or manual toggle).
13
+ ========================================================================== */
14
+
15
+ :root {
16
+ /* warm paper neutrals (light) */
17
+ --bg: #faf8f4;
18
+ --bg-elev: #ffffff;
19
+ --bg-subtle: #f2efe8;
20
+ --bg-sunken: #ece8df;
21
+ --surface: #ffffff;
22
+ --surface-hover: #f5f2ec;
23
+ --border: #e6e1d6;
24
+ --border-strong: #d6d0c2;
25
+ --text: #1c1a16;
26
+ --text-secondary: #595449;
27
+ --text-tertiary: #8b867a;
28
+
29
+ /* single flat accent — deep emerald (growth / "proceed"), no gradient */
30
+ --accent: #0f7256;
31
+ --accent-hover: #0b5b44;
32
+ --accent-soft: #e7f1ec;
33
+ --accent-border: #bcdccd;
34
+ --text-on-accent: #ffffff;
35
+
36
+ --code-bg: #f4f1ea;
37
+ --code-text: #2a2722;
38
+ --warn-bg: #fbf3e3;
39
+ --warn-border: #ecd9ac;
40
+ --warn-text: #876214;
41
+
42
+ --shadow-sm: 0 1px 2px rgba(28,26,22,.05);
43
+ --shadow-md: 0 4px 14px rgba(28,26,22,.07);
44
+
45
+ --radius-sm: 6px;
46
+ --radius: 9px;
47
+ --radius-lg: 12px;
48
+ --maxw: 1000px;
49
+
50
+ --font: 'Pretendard', -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Malgun Gothic',
51
+ 'Apple SD Gothic Neo', 'Noto Sans KR', Roboto, sans-serif;
52
+ --mono: 'Cascadia Code', 'SFMono-Regular', 'JetBrains Mono', Consolas, 'D2Coding', ui-monospace, monospace;
53
+ }
54
+
55
+ @media (prefers-color-scheme: dark) {
56
+ :root:not([data-theme='light']) { color-scheme: dark; }
57
+ }
58
+ :root[data-theme='dark'] { color-scheme: dark; }
59
+
60
+ @media (prefers-color-scheme: dark) {
61
+ :root:not([data-theme='light']) {
62
+ --bg: #14130f; --bg-elev: #1b1915; --bg-subtle: #201d17; --bg-sunken: #100f0b;
63
+ --surface: #1b1915; --surface-hover: #262219;
64
+ --border: #2e2a22; --border-strong: #3e3930;
65
+ --text: #f2eee5; --text-secondary: #aaa599; --text-tertiary: #79746a;
66
+ --accent: #34d399; --accent-hover: #6ee7b7; --accent-soft: #11281d; --accent-border: #1f4f3c; --text-on-accent: #05241a;
67
+ --code-bg: #100f0b; --code-text: #d6d1c6;
68
+ --warn-bg: #292008; --warn-border: #574510; --warn-text: #e6c578;
69
+ --shadow-sm: 0 1px 2px rgba(0,0,0,.4); --shadow-md: 0 4px 14px rgba(0,0,0,.45);
70
+ }
71
+ }
72
+ :root[data-theme='dark'] {
73
+ --bg: #14130f; --bg-elev: #1b1915; --bg-subtle: #201d17; --bg-sunken: #100f0b;
74
+ --surface: #1b1915; --surface-hover: #262219;
75
+ --border: #2e2a22; --border-strong: #3e3930;
76
+ --text: #f2eee5; --text-secondary: #aaa599; --text-tertiary: #79746a;
77
+ --accent: #34d399; --accent-hover: #6ee7b7; --accent-soft: #11281d; --accent-border: #1f4f3c;
78
+ --code-bg: #100f0b; --code-text: #d6d1c6;
79
+ --warn-bg: #292008; --warn-border: #574510; --warn-text: #e6c578;
80
+ --shadow-sm: 0 1px 2px rgba(0,0,0,.4); --shadow-md: 0 4px 14px rgba(0,0,0,.45);
81
+ }
82
+
83
+ * { box-sizing: border-box; }
84
+ html { -webkit-text-size-adjust: 100%; scroll-behavior: smooth; }
85
+ body {
86
+ margin: 0; background: var(--bg); color: var(--text);
87
+ font-family: var(--font); font-size: 16px; line-height: 1.65;
88
+ word-break: keep-all; -webkit-font-smoothing: antialiased; text-rendering: optimizeLegibility;
89
+ }
90
+ h1, h2, h3, h4 { margin: 0; font-weight: 650; letter-spacing: -0.01em; line-height: 1.25; }
91
+ p { margin: 0; }
92
+ a { color: var(--accent); text-decoration: none; }
93
+ a:hover { text-decoration: underline; }
94
+ code { font-family: var(--mono); }
95
+ :focus-visible { outline: 2px solid var(--accent); outline-offset: 2px; border-radius: 4px; }
96
+ ::selection { background: color-mix(in srgb, var(--accent) 22%, transparent); }
97
+
98
+ .accent { color: var(--accent); }
99
+
100
+ /* ----------------------------------------------------------------- nav */
101
+ .nav {
102
+ position: sticky; top: 0; z-index: 50;
103
+ display: flex; align-items: center; gap: 18px;
104
+ max-width: var(--maxw); margin: 0 auto; padding: 15px 24px;
105
+ background: color-mix(in srgb, var(--bg) 85%, transparent);
106
+ backdrop-filter: saturate(120%) blur(8px);
107
+ border-bottom: 1px solid var(--border);
108
+ }
109
+ .nav__brand { display: inline-flex; align-items: center; gap: 9px; color: var(--text); font-weight: 700; }
110
+ .nav__brand:hover { text-decoration: none; }
111
+ .nav__logo {
112
+ width: 24px; height: 24px; border-radius: 6px; flex: none;
113
+ background: var(--accent); color: var(--text-on-accent); font-weight: 800; font-size: 13px;
114
+ display: grid; place-items: center;
115
+ }
116
+ .nav__name { font-size: 15.5px; font-family: var(--mono); letter-spacing: -0.02em; }
117
+ .nav__links { display: flex; gap: 2px; margin-left: 12px; }
118
+ .nav__links a {
119
+ color: var(--text-secondary); font-size: 14px; font-weight: 500;
120
+ padding: 6px 10px; border-radius: var(--radius-sm);
121
+ }
122
+ .nav__links a:hover { color: var(--text); background: var(--surface-hover); text-decoration: none; }
123
+ .nav__actions { margin-left: auto; display: flex; align-items: center; gap: 8px; }
124
+
125
+ .theme-toggle {
126
+ width: 34px; height: 34px; display: grid; place-items: center;
127
+ border: 1px solid var(--border-strong); background: var(--surface); border-radius: var(--radius-sm);
128
+ color: var(--text-secondary); cursor: pointer; transition: background .15s, color .15s;
129
+ }
130
+ .theme-toggle:hover { background: var(--surface-hover); color: var(--text); }
131
+ .theme-toggle svg { width: 16px; height: 16px; }
132
+ .theme-toggle .i-moon { display: none; }
133
+ .theme-toggle .i-sun { display: block; }
134
+ :root[data-theme='dark'] .theme-toggle .i-moon { display: block; }
135
+ :root[data-theme='dark'] .theme-toggle .i-sun { display: none; }
136
+ @media (prefers-color-scheme: dark) {
137
+ :root:not([data-theme='light']) .theme-toggle .i-moon { display: block; }
138
+ :root:not([data-theme='light']) .theme-toggle .i-sun { display: none; }
139
+ }
140
+
141
+ /* ----------------------------------------------------------------- buttons */
142
+ .btn {
143
+ display: inline-flex; align-items: center; justify-content: center; gap: 8px;
144
+ height: 38px; padding: 0 15px; border-radius: var(--radius-sm);
145
+ border: 1px solid var(--border-strong); background: var(--surface); color: var(--text);
146
+ font-family: inherit; font-size: 14px; font-weight: 600; cursor: pointer; white-space: nowrap;
147
+ transition: background .15s, border-color .15s;
148
+ }
149
+ .btn:hover { background: var(--surface-hover); text-decoration: none; }
150
+ .btn svg { width: 16px; height: 16px; flex: none; }
151
+ .btn--ghost { background: transparent; border-color: var(--border); color: var(--text-secondary); }
152
+ .btn--ghost:hover { color: var(--text); background: var(--surface-hover); }
153
+ .btn--primary { background: var(--accent); border-color: var(--accent); color: var(--text-on-accent); }
154
+ .btn--primary:hover { background: var(--accent-hover); border-color: var(--accent-hover); }
155
+
156
+ /* ----------------------------------------------------------------- hero */
157
+ /* single centered column — headline on top, the prompt card right below it. */
158
+ .hero {
159
+ max-width: 720px; margin: 0 auto; padding: 66px 24px 8px; text-align: center;
160
+ }
161
+ /* gentle staggered entrance on load */
162
+ @keyframes riseIn { from { opacity: 0; transform: translateY(10px); } to { opacity: 1; transform: none; } }
163
+ .hero > * { animation: riseIn .55s cubic-bezier(.22,.61,.36,1) both; }
164
+ .hero__title { animation-delay: .04s; }
165
+ .hero__lede { animation-delay: .10s; }
166
+ .connect { animation-delay: .18s; }
167
+ .oneclick__hint { animation-delay: .24s; }
168
+ .pill-badge {
169
+ display: inline-flex; align-items: center; gap: 8px;
170
+ padding: 5px 13px; border-radius: 999px; font-size: 12.5px; font-weight: 600;
171
+ color: var(--accent); background: var(--accent-soft); border: 1px solid var(--accent-border);
172
+ }
173
+ .pill-badge__dot { width: 6px; height: 6px; border-radius: 999px; background: var(--accent); }
174
+ .hero__title { font-size: clamp(2.05rem, 5vw, 3rem); margin: 20px 0 0; letter-spacing: -0.028em; }
175
+ .hero__lede {
176
+ font-size: clamp(1rem, 2vw, 1.12rem); color: var(--text-secondary);
177
+ margin: 18px auto 0; max-width: 37rem;
178
+ }
179
+ .hero__lede strong { color: var(--text); font-weight: 650; }
180
+
181
+ /* connect card — two paths: let your AI set it up, or connect directly (MCP std) */
182
+ .connect {
183
+ margin: 30px auto 0; max-width: 600px; text-align: left;
184
+ background: var(--surface); border: 1px solid var(--border);
185
+ border-radius: var(--radius-lg); box-shadow: var(--shadow-sm);
186
+ padding: 18px;
187
+ }
188
+ .connect__lead { font-size: 13px; color: var(--text-secondary); margin-bottom: 11px; }
189
+ .connect__note { margin-top: 12px; font-size: 12px; color: var(--text-tertiary); line-height: 1.55; }
190
+ .connect__note strong { color: var(--text-secondary); }
191
+ .connect__note code { font-family: var(--mono); background: var(--bg-sunken); padding: 1px 4px; border-radius: 4px; }
192
+ .connect .code { margin: 0; }
193
+
194
+ .oneclick__pills { display: flex; flex-wrap: wrap; gap: 5px; margin-bottom: 11px; }
195
+ .ai-pill {
196
+ font-family: inherit; font-size: 13px; font-weight: 600; cursor: pointer;
197
+ padding: 5px 12px; border-radius: var(--radius-sm); color: var(--text-secondary);
198
+ background: transparent; border: 1px solid var(--border-strong);
199
+ transition: background .12s, color .12s, border-color .12s;
200
+ }
201
+ .ai-pill:hover { color: var(--text); background: var(--surface-hover); }
202
+ .ai-pill.is-active { color: var(--accent); background: var(--accent-soft); border-color: var(--accent-border); }
203
+
204
+ .oneclick__box { position: relative; }
205
+ .oneclick__text {
206
+ margin: 0; max-height: 112px; overflow-y: auto;
207
+ background: var(--code-bg); border: 1px solid var(--border);
208
+ border-radius: var(--radius); padding: 15px 15px 54px;
209
+ font-family: var(--font); font-size: 13.5px; line-height: 1.62; color: var(--text);
210
+ white-space: pre-wrap; word-break: keep-all;
211
+ }
212
+ .oneclick__text::-webkit-scrollbar { width: 9px; }
213
+ .oneclick__text::-webkit-scrollbar-thumb { background: var(--border-strong); border-radius: 9px; border: 2px solid var(--code-bg); }
214
+ .oneclick__fade {
215
+ position: absolute; left: 1px; right: 1px; bottom: 1px; height: 58px; border-radius: 0 0 var(--radius) var(--radius);
216
+ background: linear-gradient(to bottom, transparent, var(--code-bg) 80%); pointer-events: none;
217
+ }
218
+ .btn--copy-lg {
219
+ position: absolute; left: 14px; right: 14px; bottom: 12px; width: auto; height: 42px; font-size: 14px;
220
+ }
221
+ .btn--copy-lg.is-copied { background: var(--accent-hover); border-color: var(--accent-hover); }
222
+
223
+ .oneclick__hint { margin-top: 14px; font-size: 12.5px; color: var(--text-tertiary); text-align: center; }
224
+ .oneclick__hint code { font-family: var(--mono); font-size: .92em; background: var(--bg-sunken); padding: 1px 5px; border-radius: 4px; color: var(--text-secondary); }
225
+
226
+ /* ----------------------------------------------------------------- sections */
227
+ .section { max-width: var(--maxw); margin: 0 auto; padding: 112px 24px 0; scroll-margin-top: 80px; }
228
+ .section__head { max-width: 46rem; margin: 0 auto 34px; text-align: center; }
229
+ .eyebrow {
230
+ display: inline-block; font-size: 12px; font-weight: 700; letter-spacing: .09em;
231
+ text-transform: uppercase; color: var(--accent); margin-bottom: 9px;
232
+ }
233
+ .section__head h2 { font-size: clamp(1.4rem, 3vw, 1.9rem); }
234
+ .section__head p { margin-top: 11px; color: var(--text-secondary); font-size: 1rem; }
235
+
236
+ /* how-it-works */
237
+ .cards-3 { display: grid; grid-template-columns: repeat(3, 1fr); gap: 14px; }
238
+ .feature {
239
+ background: var(--surface); border: 1px solid var(--border); border-radius: var(--radius-lg);
240
+ padding: 22px 20px; transition: border-color .15s;
241
+ }
242
+ .feature:hover { border-color: var(--border-strong); }
243
+ .feature__icon {
244
+ width: 40px; height: 40px; border-radius: 9px; display: grid; place-items: center; margin-bottom: 14px;
245
+ color: var(--accent); background: var(--accent-soft); border: 1px solid var(--accent-border);
246
+ }
247
+ .feature__icon svg { width: 20px; height: 20px; }
248
+ .feature h3 { font-size: 16px; margin-bottom: 6px; }
249
+ .feature p { color: var(--text-secondary); font-size: 14px; }
250
+ .feature code { font-family: var(--mono); font-size: .85em; background: var(--bg-sunken); padding: 1px 5px; border-radius: 4px; }
251
+
252
+ /* checklist */
253
+ .checks { display: grid; grid-template-columns: repeat(2, 1fr); gap: 12px; max-width: 840px; margin: 0 auto; }
254
+ .check {
255
+ display: flex; gap: 12px; align-items: flex-start;
256
+ background: var(--surface); border: 1px solid var(--border); border-radius: var(--radius);
257
+ padding: 15px 17px;
258
+ }
259
+ .check > svg { width: 18px; height: 18px; flex: none; color: var(--accent); margin-top: 1px; }
260
+ .check strong { display: block; font-size: 14.5px; font-weight: 650; }
261
+ .check span { display: block; color: var(--text-secondary); font-size: 13px; margin-top: 2px; }
262
+
263
+ /* ----------------------------------------------------------------- manual */
264
+ .note {
265
+ border: 1px solid var(--border); border-left: 3px solid var(--accent);
266
+ background: var(--surface); border-radius: var(--radius);
267
+ padding: 13px 17px; margin: 0 auto 24px; max-width: 720px;
268
+ font-size: 14px; color: var(--text-secondary); line-height: 1.6;
269
+ }
270
+ .note strong { color: var(--text); }
271
+ .note code { font-family: var(--mono); background: var(--bg-sunken); padding: 1px 5px; border-radius: 4px; font-size: .88em; color: var(--text); }
272
+
273
+ .steps { list-style: none; margin: 0 auto; padding: 0; max-width: 720px; }
274
+ .step { display: flex; gap: 15px; padding: 2px 0 24px; position: relative; }
275
+ .step__no {
276
+ width: 30px; height: 30px; flex: none; border-radius: 999px;
277
+ display: grid; place-items: center; font-weight: 700; font-size: 14px;
278
+ color: var(--accent); background: var(--accent-soft); border: 1px solid var(--accent-border); z-index: 1;
279
+ }
280
+ .step:not(:last-of-type)::before {
281
+ content: ''; position: absolute; left: 15px; top: 34px; bottom: 6px; width: 1px;
282
+ background: var(--border); z-index: 0;
283
+ }
284
+ .step__body { flex: 1; min-width: 0; padding-top: 3px; }
285
+ .step__body h3 { font-size: 16px; margin-bottom: 5px; }
286
+ .step__body > p { color: var(--text-secondary); font-size: 14px; margin-bottom: 11px; }
287
+ .step__body code { font-family: var(--mono); background: var(--bg-sunken); padding: 1px 5px; border-radius: 4px; font-size: .86em; }
288
+ .step__tip { margin-top: 11px; font-size: 13px; color: var(--text-secondary); }
289
+ .step__tip code { font-family: var(--mono); background: var(--bg-sunken); padding: 1px 5px; border-radius: 4px; }
290
+
291
+ kbd {
292
+ font-family: inherit; font-size: .82em; background: var(--bg-sunken);
293
+ border: 1px solid var(--border-strong); border-bottom-width: 2px; border-radius: 4px; padding: 1px 6px;
294
+ }
295
+
296
+ /* code blocks */
297
+ .code {
298
+ position: relative; margin: 11px 0; background: var(--code-bg);
299
+ border: 1px solid var(--border); border-radius: var(--radius); overflow: hidden;
300
+ }
301
+ .code__label {
302
+ display: block; font-family: var(--mono); font-size: 11px; letter-spacing: .03em;
303
+ color: var(--text-tertiary); padding: 9px 13px 0;
304
+ }
305
+ .code pre {
306
+ margin: 0; padding: 11px 13px 13px; overflow-x: auto;
307
+ font-family: var(--mono); font-size: 12.5px; line-height: 1.6; color: var(--code-text); white-space: pre;
308
+ }
309
+ .copy-btn {
310
+ position: absolute; top: 7px; right: 7px; font-family: inherit; font-size: 12px; font-weight: 600;
311
+ color: var(--text-secondary); background: var(--surface); border: 1px solid var(--border-strong);
312
+ border-radius: 6px; padding: 3px 10px; cursor: pointer; transition: background .12s, color .12s, border-color .12s;
313
+ }
314
+ .copy-btn:hover { border-color: var(--accent); color: var(--accent); }
315
+ .copy-btn.copied { color: var(--text-on-accent); background: var(--accent); border-color: var(--accent); }
316
+
317
+ /* tabs */
318
+ .tabs { display: flex; gap: 2px; border-bottom: 1px solid var(--border); margin: 4px 0 0; flex-wrap: wrap; }
319
+ .tab {
320
+ font-family: inherit; font-size: 13.5px; font-weight: 600; color: var(--text-secondary);
321
+ background: none; border: none; border-bottom: 2px solid transparent; margin-bottom: -1px;
322
+ padding: 8px 12px; cursor: pointer; transition: color .12s;
323
+ }
324
+ .tab:hover { color: var(--text); }
325
+ .tab.is-active { color: var(--accent); border-bottom-color: var(--accent); }
326
+ .panel { display: none; padding-top: 11px; }
327
+ .panel[data-active='true'] { display: block; }
328
+ .path-note { font-size: 12.5px; color: var(--text-tertiary); margin-bottom: 6px; line-height: 1.6; }
329
+ .path-note code { font-family: var(--mono); font-size: .9em; background: var(--bg-sunken); padding: 1px 5px; border-radius: 4px; }
330
+
331
+ /* troubleshooting accordion */
332
+ .acc {
333
+ max-width: 720px; margin: 4px auto 0; border: 1px solid var(--border);
334
+ border-radius: var(--radius); background: var(--surface); overflow: hidden;
335
+ }
336
+ .acc > summary {
337
+ cursor: pointer; list-style: none; padding: 15px 17px; font-weight: 650; font-size: 14.5px;
338
+ display: flex; align-items: center; gap: 10px;
339
+ }
340
+ .acc > summary::-webkit-details-marker { display: none; }
341
+ .acc > summary::after { content: '+'; margin-left: auto; color: var(--text-tertiary); font-weight: 400; }
342
+ .acc[open] > summary::after { content: '-'; }
343
+ .acc > summary:hover { background: var(--surface-hover); }
344
+ .acc__hint { font-weight: 500; font-size: 12.5px; color: var(--text-tertiary); }
345
+ .acc__body { padding: 0 17px 16px; }
346
+ .trouble { display: grid; grid-template-columns: repeat(2, 1fr); gap: 13px; }
347
+ .trouble__item h4 { font-size: 14px; margin-bottom: 4px; }
348
+ .trouble__item p { color: var(--text-secondary); font-size: 13px; }
349
+ .trouble__item code { font-family: var(--mono); background: var(--bg-sunken); padding: 1px 5px; border-radius: 4px; font-size: .86em; }
350
+
351
+ /* privacy */
352
+ .privacy-card {
353
+ max-width: 800px; margin: 0 auto; display: flex; gap: 20px; align-items: flex-start;
354
+ background: var(--surface); border: 1px solid var(--border); border-radius: var(--radius-lg);
355
+ padding: 28px 30px;
356
+ }
357
+ .privacy-card__icon {
358
+ width: 50px; height: 50px; flex: none; border-radius: 12px; display: grid; place-items: center;
359
+ color: var(--accent); background: var(--accent-soft); border: 1px solid var(--accent-border);
360
+ }
361
+ .privacy-card__icon svg { width: 26px; height: 26px; }
362
+ .privacy-card__text h2 { font-size: 1.3rem; }
363
+ .privacy-card__text > p { color: var(--text-secondary); margin-top: 9px; font-size: 14.5px; }
364
+ .privacy-card__list { list-style: none; margin: 14px 0 0; padding: 0; display: grid; gap: 7px; }
365
+ .privacy-card__list li { font-size: 13.5px; color: var(--text-secondary); }
366
+ .privacy-card__list strong { color: var(--text); margin-right: 8px; }
367
+ .privacy-card__list code { font-family: var(--mono); font-size: .86em; background: var(--bg-sunken); padding: 1px 6px; border-radius: 4px; color: var(--text); }
368
+
369
+ /* footer */
370
+ .foot {
371
+ max-width: var(--maxw); margin: 104px auto 0; padding: 32px 24px 56px;
372
+ border-top: 1px solid var(--border); text-align: center; color: var(--text-tertiary);
373
+ }
374
+ .foot__brand { display: inline-flex; align-items: center; gap: 8px; font-weight: 700; color: var(--text); font-size: 15px; font-family: var(--mono); }
375
+ .foot p { margin: 11px 0 0; font-size: 13px; }
376
+ .foot__links { display: flex; justify-content: center; gap: 16px; margin-top: 14px; font-size: 13px; }
377
+ .foot__links a { color: var(--text-secondary); }
378
+
379
+ /* a11y */
380
+ .sr-only {
381
+ position: absolute; width: 1px; height: 1px; padding: 0; margin: -1px;
382
+ overflow: hidden; clip: rect(0 0 0 0); white-space: nowrap; border: 0;
383
+ }
384
+
385
+ /* scroll reveal — JS adds .reveal only when motion is allowed, so no-JS /
386
+ reduced-motion shows everything immediately (no hidden content risk) */
387
+ .reveal { opacity: 0; transform: translateY(14px); transition: opacity .55s ease, transform .55s ease; }
388
+ .reveal.is-visible { opacity: 1; transform: none; }
389
+ .cards-3 .feature:nth-child(2).reveal { transition-delay: .07s; }
390
+ .cards-3 .feature:nth-child(3).reveal { transition-delay: .14s; }
391
+ .checks .check:nth-child(even).reveal { transition-delay: .05s; }
392
+
393
+ /* responsive */
394
+ @media (max-width: 820px) {
395
+ .cards-3 { grid-template-columns: 1fr; }
396
+ .checks { grid-template-columns: 1fr; }
397
+ .trouble { grid-template-columns: 1fr; }
398
+ .nav__links { display: none; }
399
+ }
400
+ @media (max-width: 540px) {
401
+ .hero { padding-top: 46px; }
402
+ .privacy-card { flex-direction: column; gap: 15px; padding: 22px; }
403
+ }
404
+ @media (prefers-reduced-motion: reduce) {
405
+ html { scroll-behavior: auto; }
406
+ * { animation: none !important; transition: none !important; }
407
+ }
@@ -0,0 +1,20 @@
1
+ {
2
+ "$schema": "https://openapi.vercel.sh/vercel.json",
3
+ "cleanUrls": true,
4
+ "trailingSlash": false,
5
+ "headers": [
6
+ {
7
+ "source": "/style.css",
8
+ "headers": [
9
+ { "key": "Cache-Control", "value": "public, max-age=3600, must-revalidate" }
10
+ ]
11
+ },
12
+ {
13
+ "source": "/(.*)",
14
+ "headers": [
15
+ { "key": "X-Content-Type-Options", "value": "nosniff" },
16
+ { "key": "Referrer-Policy", "value": "strict-origin-when-cross-origin" }
17
+ ]
18
+ }
19
+ ]
20
+ }