@macedon-technologies/batman 1.0.16

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 (41) hide show
  1. package/.jsii +637 -0
  2. package/API.md +5671 -0
  3. package/LICENSE +202 -0
  4. package/README.md +86 -0
  5. package/lib/ApiConstructFile.d.ts +5 -0
  6. package/lib/ApiConstructFile.js +175 -0
  7. package/lib/ApiFunctionFile.d.ts +5 -0
  8. package/lib/ApiFunctionFile.js +30 -0
  9. package/lib/Application.d.ts +21 -0
  10. package/lib/Application.js +72 -0
  11. package/lib/ApplicationStackFile.d.ts +5 -0
  12. package/lib/ApplicationStackFile.js +26 -0
  13. package/lib/BaseApplicationStackFile.d.ts +5 -0
  14. package/lib/BaseApplicationStackFile.js +172 -0
  15. package/lib/BatmanProject.d.ts +25 -0
  16. package/lib/BatmanProject.js +101 -0
  17. package/lib/GitHubRolesFile.d.ts +9 -0
  18. package/lib/GitHubRolesFile.js +26 -0
  19. package/lib/GitHubRolesStackFile.d.ts +8 -0
  20. package/lib/GitHubRolesStackFile.js +165 -0
  21. package/lib/LocalDevAppFile.d.ts +10 -0
  22. package/lib/LocalDevAppFile.js +34 -0
  23. package/lib/MainFile.d.ts +24 -0
  24. package/lib/MainFile.js +64 -0
  25. package/lib/MainTestFile.d.ts +5 -0
  26. package/lib/MainTestFile.js +33 -0
  27. package/lib/PrCleanupWorkflow.d.ts +9 -0
  28. package/lib/PrCleanupWorkflow.js +100 -0
  29. package/lib/PrDeployWorkflow.d.ts +10 -0
  30. package/lib/PrDeployWorkflow.js +106 -0
  31. package/lib/ProductionDeployWorkflow.d.ts +11 -0
  32. package/lib/ProductionDeployWorkflow.js +68 -0
  33. package/lib/StagingDeployWorkflow.d.ts +11 -0
  34. package/lib/StagingDeployWorkflow.js +68 -0
  35. package/lib/StaticWebsiteConstructFile.d.ts +5 -0
  36. package/lib/StaticWebsiteConstructFile.js +198 -0
  37. package/lib/ViteReactProject.d.ts +45 -0
  38. package/lib/ViteReactProject.js +654 -0
  39. package/lib/index.d.ts +5 -0
  40. package/lib/index.js +20 -0
  41. package/package.json +125 -0
@@ -0,0 +1,654 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ViteReactProject = void 0;
4
+ const path_1 = require("path");
5
+ const projen_1 = require("projen");
6
+ const typescript_1 = require("projen/lib/typescript");
7
+ /**
8
+ * Vite + React TypeScript project
9
+ */
10
+ class ViteReactProject extends typescript_1.TypeScriptAppProject {
11
+ constructor(options) {
12
+ const { sampleCode = true, srcdir = 'src', publicDir = 'public', title, ...restOptions } = options;
13
+ super({
14
+ ...restOptions,
15
+ sampleCode: false, // We'll handle sample code ourselves
16
+ eslint: false, // We'll configure eslint ourselves
17
+ jest: false, // Vite uses vitest typically
18
+ package: true,
19
+ disableTsconfig: true, // We'll manage tsconfig ourselves
20
+ deps: [
21
+ '@clerk/clerk-react',
22
+ '@tanstack/react-query',
23
+ '@tailwindcss/postcss',
24
+ 'autoprefixer',
25
+ 'postcss',
26
+ 'tailwindcss',
27
+ ],
28
+ });
29
+ // Initialize config fields storage
30
+ this.configFields = new Map();
31
+ this.srcdir = srcdir;
32
+ this.publicDir = publicDir;
33
+ // Add Vite component
34
+ new ViteComponent(this, {
35
+ srcdir: this.srcdir,
36
+ });
37
+ // Add ESLint configuration
38
+ new ViteEslintConfig(this);
39
+ // Add index.html
40
+ new IndexHtml(this, {
41
+ title: title ?? this.name,
42
+ srcdir: this.srcdir,
43
+ });
44
+ // Add TypeScript configs
45
+ new ViteTsConfigs(this, {
46
+ srcdir: this.srcdir,
47
+ });
48
+ // Add .gitignore entries
49
+ this.gitignore?.addPatterns('dist', 'dist-ssr', '*.local');
50
+ // Add vite-env.d.ts for type declarations
51
+ new ViteEnvDts(this, {
52
+ srcdir: this.srcdir,
53
+ });
54
+ // Add PostCSS config for Tailwind
55
+ new PostCSSConfigFile(this);
56
+ // Add config.ts for runtime configuration
57
+ new ConfigFile(this, {
58
+ srcdir: this.srcdir,
59
+ });
60
+ // Add sample config.json
61
+ new ConfigJsonFile(this, {});
62
+ // Add sample code if requested
63
+ if (sampleCode) {
64
+ new ViteSampleCode(this, {
65
+ srcdir: this.srcdir,
66
+ publicDir: this.publicDir,
67
+ });
68
+ }
69
+ // Use relative path for devCommand
70
+ const relativeOutdir = this.parent
71
+ ? (0, path_1.relative)(this.parent.outdir, this.outdir)
72
+ : (0, path_1.basename)(this.outdir);
73
+ this.devCommand = `cd ${relativeOutdir} && yarn dev`;
74
+ }
75
+ /**
76
+ * Add a custom field to the Config type
77
+ * @param fieldName The name of the field
78
+ * @param fieldType The TypeScript type of the field (e.g., 'string', 'number', 'boolean', 'string[]')
79
+ * @returns this project instance for chaining
80
+ */
81
+ addConfigField(fieldName, fieldType) {
82
+ this.configFields.set(fieldName, fieldType);
83
+ return this;
84
+ }
85
+ /**
86
+ * Get the config fields map (for internal use by ConfigFile)
87
+ * @internal
88
+ */
89
+ getConfigFields() {
90
+ return this.configFields;
91
+ }
92
+ }
93
+ exports.ViteReactProject = ViteReactProject;
94
+ /**
95
+ * Configures Vite dependencies and scripts
96
+ */
97
+ class ViteComponent extends projen_1.Component {
98
+ constructor(project, _options) {
99
+ super(project);
100
+ const tsProject = project;
101
+ // Set package type to module for ESM support
102
+ tsProject.package.addField('type', 'module');
103
+ // Add React dependencies
104
+ tsProject.addDeps('react@^19.2.0', 'react-dom@^19.2.0');
105
+ // Add dev dependencies
106
+ tsProject.addDevDeps('@types/react@^19.2.2', '@types/react-dom@^19.2.2', '@vitejs/plugin-react@^5.1.0', 'vite@^5.4.0', 'typescript@~5.9.3', '@eslint/js@^9.39.1', 'eslint@^9.39.1', 'eslint-plugin-react-hooks@^5.2.0', 'eslint-plugin-react-refresh@^0.4.24', 'globals@^16.5.0', 'typescript-eslint@^8.46.3');
107
+ // Create vite.config.ts as a managed file
108
+ new ViteConfigFile(project);
109
+ // Add scripts
110
+ tsProject.addTask('dev', {
111
+ description: 'Start development server',
112
+ exec: 'vite',
113
+ });
114
+ // Remove and recreate build task to customize it for Vite
115
+ tsProject.removeTask('build');
116
+ tsProject.addTask('build', {
117
+ description: 'Build for production',
118
+ exec: 'tsc -b && vite build',
119
+ });
120
+ tsProject.addTask('preview', {
121
+ description: 'Preview production build',
122
+ exec: 'vite preview',
123
+ });
124
+ tsProject.addTask('lint', {
125
+ description: 'Lint code',
126
+ exec: 'eslint .',
127
+ });
128
+ }
129
+ }
130
+ /**
131
+ * Creates the ESLint configuration for Vite + React
132
+ */
133
+ class ViteEslintConfig extends projen_1.FileBase {
134
+ constructor(project) {
135
+ super(project, 'eslint.config.js', {
136
+ marker: true,
137
+ readonly: true,
138
+ });
139
+ }
140
+ synthesizeContent() {
141
+ return `import js from '@eslint/js'
142
+ import globals from 'globals'
143
+ import reactHooks from 'eslint-plugin-react-hooks'
144
+ import reactRefresh from 'eslint-plugin-react-refresh'
145
+ import tseslint from 'typescript-eslint'
146
+
147
+ export default [
148
+ { ignores: ['dist'] },
149
+ {
150
+ files: ['**/*.{ts,tsx}'],
151
+ languageOptions: {
152
+ ecmaVersion: 2020,
153
+ globals: globals.browser,
154
+ },
155
+ plugins: {
156
+ 'react-hooks': reactHooks,
157
+ 'react-refresh': reactRefresh,
158
+ },
159
+ rules: {
160
+ ...reactHooks.configs.recommended.rules,
161
+ 'react-refresh/only-export-components': [
162
+ 'warn',
163
+ { allowConstantExport: true },
164
+ ],
165
+ },
166
+ },
167
+ js.configs.recommended,
168
+ ...tseslint.configs.recommended,
169
+ ]
170
+ `;
171
+ }
172
+ }
173
+ /**
174
+ * Creates the vite.config.ts file
175
+ */
176
+ class ViteConfigFile extends projen_1.FileBase {
177
+ constructor(project) {
178
+ super(project, 'vite.config.ts', {
179
+ marker: true,
180
+ readonly: true,
181
+ });
182
+ }
183
+ synthesizeContent() {
184
+ return `import { defineConfig } from 'vite'
185
+ import react from '@vitejs/plugin-react'
186
+
187
+ // https://vite.dev/config/
188
+ export default defineConfig({
189
+ plugins: [react()],
190
+ })
191
+ `;
192
+ }
193
+ }
194
+ /**
195
+ * Creates the postcss.config.js file for Tailwind CSS v4
196
+ */
197
+ class PostCSSConfigFile extends projen_1.FileBase {
198
+ constructor(project) {
199
+ super(project, 'postcss.config.js', {
200
+ marker: true,
201
+ readonly: true,
202
+ });
203
+ }
204
+ synthesizeContent() {
205
+ return `export default {
206
+ plugins: {
207
+ "@tailwindcss/postcss": {},
208
+ autoprefixer: {},
209
+ },
210
+ }
211
+ `;
212
+ }
213
+ }
214
+ /**
215
+ * Creates the vite-env.d.ts file for type declarations
216
+ */
217
+ class ViteEnvDts extends projen_1.FileBase {
218
+ constructor(project, options) {
219
+ super(project, `${options.srcdir}/vite-env.d.ts`, {
220
+ marker: true,
221
+ readonly: true,
222
+ });
223
+ }
224
+ synthesizeContent() {
225
+ return `/// <reference types="vite/client" />
226
+ `;
227
+ }
228
+ }
229
+ /**
230
+ * Creates the config.ts file for runtime configuration
231
+ */
232
+ class ConfigFile extends projen_1.FileBase {
233
+ constructor(project, options) {
234
+ super(project, `${options.srcdir}/config.ts`, {
235
+ marker: true,
236
+ readonly: true,
237
+ });
238
+ }
239
+ synthesizeContent() {
240
+ // Get custom config fields from parent project
241
+ const viteProject = this.project;
242
+ const customFields = viteProject.getConfigFields();
243
+ // Generate base config type
244
+ const baseConfigType = `type BaseConfig = {
245
+ clerkPublishableKey: string;
246
+ apiBaseUrl: string;
247
+ timeout: number;
248
+ }`;
249
+ // Generate custom fields type if any exist
250
+ let configTypeDefinition;
251
+ if (customFields.size > 0) {
252
+ const fieldDefinitions = Array.from(customFields.entries())
253
+ .map(([name, type]) => ` ${name}: ${type};`)
254
+ .join('\n');
255
+ const customFieldsType = `type CustomConfig = {
256
+ ${fieldDefinitions}
257
+ }`;
258
+ configTypeDefinition = `${customFieldsType}
259
+
260
+ export type Config = BaseConfig & CustomConfig;`;
261
+ }
262
+ else {
263
+ configTypeDefinition = 'export type Config = BaseConfig;';
264
+ }
265
+ return `import { useEffect, useState } from 'react';
266
+
267
+ ${baseConfigType}
268
+
269
+ ${configTypeDefinition}
270
+
271
+ export function useConfig() {
272
+ const [loadedConfig, setLoadedConfig] = useState<Config|undefined>();
273
+ const [isLoading, setIsLoading] = useState(false);
274
+
275
+ useEffect(() => {
276
+ if (!loadedConfig) {
277
+ fetch('/config.json', { cache: 'no-store' }).then(response=>{
278
+ response.json().then((config) => {
279
+ setIsLoading(false)
280
+ setLoadedConfig(config)
281
+ });
282
+ });
283
+
284
+ }
285
+ }, [loadedConfig]);
286
+
287
+ return { config: loadedConfig, isLoading };
288
+ }
289
+ `;
290
+ }
291
+ }
292
+ /**
293
+ * Creates the config.json sample file for runtime configuration
294
+ */
295
+ class ConfigJsonFile extends projen_1.SampleFile {
296
+ constructor(project, _options) {
297
+ super(project, `${project.publicDir}/config.json`, {
298
+ contents: JSON.stringify({
299
+ apiBaseUrl: 'http://localhost:3000/api',
300
+ clerkPublishableKey: 'tbd',
301
+ }, null, 2),
302
+ });
303
+ this.path = `${project.publicDir}/config.json`;
304
+ }
305
+ }
306
+ /**
307
+ * Creates the index.html file
308
+ */
309
+ class IndexHtml extends projen_1.FileBase {
310
+ constructor(project, options) {
311
+ super(project, 'index.html', {
312
+ marker: true,
313
+ readonly: true,
314
+ });
315
+ this.title = options.title;
316
+ this.srcdir = options.srcdir;
317
+ }
318
+ synthesizeContent() {
319
+ return `<!doctype html>
320
+ <html lang="en">
321
+ <head>
322
+ <meta charset="UTF-8" />
323
+ <link rel="icon" type="image/svg+xml" href="/vite.svg" />
324
+ <meta name="viewport" content="width=device-width, initial-scale=1.0" />
325
+ <title>${this.title}</title>
326
+ </head>
327
+ <body>
328
+ <div id="root"></div>
329
+ <script type="module" src="/${this.srcdir}/main.tsx"></script>
330
+ </body>
331
+ </html>
332
+ `;
333
+ }
334
+ }
335
+ /**
336
+ * Creates the TypeScript configuration files for Vite
337
+ */
338
+ class ViteTsConfigs extends projen_1.Component {
339
+ constructor(project, options) {
340
+ super(project);
341
+ // Main tsconfig.json (project references)
342
+ new projen_1.JsonFile(project, 'tsconfig.json', {
343
+ marker: true,
344
+ readonly: true,
345
+ obj: {
346
+ files: [],
347
+ references: [
348
+ { path: './tsconfig.app.json' },
349
+ { path: './tsconfig.node.json' },
350
+ ],
351
+ },
352
+ });
353
+ // tsconfig.app.json (for application code)
354
+ new projen_1.JsonFile(project, 'tsconfig.app.json', {
355
+ marker: true,
356
+ readonly: true,
357
+ obj: {
358
+ compilerOptions: {
359
+ target: 'ES2020',
360
+ useDefineForClassFields: true,
361
+ lib: ['ES2020', 'DOM', 'DOM.Iterable'],
362
+ module: 'ESNext',
363
+ skipLibCheck: true,
364
+ // Bundler mode
365
+ moduleResolution: 'bundler',
366
+ allowImportingTsExtensions: true,
367
+ isolatedModules: true,
368
+ moduleDetection: 'force',
369
+ noEmit: true,
370
+ jsx: 'react-jsx',
371
+ // Linting
372
+ strict: true,
373
+ noUnusedLocals: true,
374
+ noUnusedParameters: true,
375
+ noFallthroughCasesInSwitch: true,
376
+ noUncheckedSideEffectImports: true,
377
+ },
378
+ include: [options.srcdir],
379
+ },
380
+ });
381
+ // tsconfig.node.json (for Vite config)
382
+ new projen_1.JsonFile(project, 'tsconfig.node.json', {
383
+ marker: true,
384
+ readonly: true,
385
+ obj: {
386
+ compilerOptions: {
387
+ target: 'ES2022',
388
+ lib: ['ES2023'],
389
+ module: 'ESNext',
390
+ skipLibCheck: true,
391
+ // Bundler mode
392
+ moduleResolution: 'bundler',
393
+ allowImportingTsExtensions: true,
394
+ isolatedModules: true,
395
+ moduleDetection: 'force',
396
+ noEmit: true,
397
+ // Linting
398
+ strict: true,
399
+ noUnusedLocals: true,
400
+ noUnusedParameters: true,
401
+ noFallthroughCasesInSwitch: true,
402
+ },
403
+ include: ['vite.config.ts'],
404
+ },
405
+ });
406
+ }
407
+ }
408
+ /**
409
+ * Generates sample code for a Vite + React project
410
+ */
411
+ class ViteSampleCode extends projen_1.Component {
412
+ constructor(project, options) {
413
+ super(project);
414
+ // Create App.tsx
415
+ new projen_1.SampleFile(project, `${options.srcdir}/App.tsx`, {
416
+ contents: `import './App.css'
417
+ import { SignedIn, UserButton } from '@clerk/clerk-react';
418
+ import { useBaseQuery } from "./services/BaseService.ts";
419
+
420
+
421
+ function App() {
422
+ const {data, isLoading} = useBaseQuery('/', {retryOnMount: true});
423
+
424
+
425
+ return (
426
+ <>
427
+ <header>
428
+
429
+ <SignedIn>
430
+ <UserButton/>
431
+ </SignedIn>
432
+ </header>
433
+ <div className="card">
434
+ { isLoading ? 'Loading...' : JSON.stringify(data, null ,)}
435
+ </div>
436
+ </>
437
+ )
438
+ }
439
+
440
+ export default App
441
+ `,
442
+ });
443
+ // Create BaseService.ts
444
+ new projen_1.TextFile(project, `${options.srcdir}/services/BaseService.ts`, {
445
+ marker: true,
446
+ readonly: true,
447
+ lines: [
448
+ 'import { useQuery, UseQueryOptions } from \'@tanstack/react-query\';',
449
+ 'import { useAuth } from \'@clerk/clerk-react\';',
450
+ 'import { useConfig } from \'../config\';',
451
+ '',
452
+ 'export function useBaseQuery<TData = unknown>(',
453
+ ' path: string,',
454
+ ' options?: Omit<UseQueryOptions<TData>, \'queryKey\' | \'queryFn\'>',
455
+ ') {',
456
+ ' const { config } = useConfig();',
457
+ ' const { getToken } = useAuth();',
458
+ '',
459
+ ' return useQuery<TData>({',
460
+ ' queryKey: [\'base\'],',
461
+ ' queryFn: async () => {',
462
+ ' if (!config?.apiBaseUrl) {',
463
+ ' throw new Error(\'API base URL not configured\');',
464
+ ' }',
465
+ '',
466
+ ' const token = await getToken();',
467
+ '',
468
+ ' // Normalize URL by removing trailing slash from baseUrl and ensuring path starts with slash',
469
+ ' const baseUrl = config.apiBaseUrl.replace(/\\/+$/, \'\');',
470
+ ' const normalizedPath = path.startsWith(\'/\') ? path : `/${path}`;',
471
+ ' ',
472
+ ' const response = await fetch(`${baseUrl}${normalizedPath}`, {',
473
+ ' headers: {',
474
+ ' \'Authorization\': `Bearer ${token}`,',
475
+ ' },',
476
+ ' });',
477
+ '',
478
+ ' if (!response.ok) {',
479
+ ' throw new Error(`HTTP error! status: ${response.status}`);',
480
+ ' }',
481
+ '',
482
+ ' return response.json();',
483
+ ' },',
484
+ ' enabled: !!config?.apiBaseUrl,',
485
+ ' ...options,',
486
+ ' });',
487
+ '}',
488
+ ],
489
+ });
490
+ // Create RootService.ts
491
+ new projen_1.SampleFile(project, `${options.srcdir}/services/RootService.ts`, {
492
+ contents: `import { UseQueryOptions } from '@tanstack/react-query';
493
+ import { useBaseQuery } from './BaseService';
494
+
495
+ export function useRootQuery<TData = unknown>(
496
+ options?: Omit<UseQueryOptions<TData>, 'queryKey' | 'queryFn'>
497
+ ) {
498
+ return useBaseQuery<TData>('/', options);
499
+ }
500
+ `,
501
+ });
502
+ // Create App.css
503
+ new projen_1.SampleFile(project, `${options.srcdir}/App.css`, {
504
+ contents: `#root {
505
+ width: 100%;
506
+ height: 100%;
507
+ }
508
+
509
+ .logo {
510
+ height: 6em;
511
+ padding: 1.5em;
512
+ will-change: filter;
513
+ transition: filter 300ms;
514
+ }
515
+ .logo:hover {
516
+ filter: drop-shadow(0 0 2em #646cffaa);
517
+ }
518
+ .logo.react:hover {
519
+ filter: drop-shadow(0 0 2em #61dafbaa);
520
+ }
521
+
522
+ @keyframes logo-spin {
523
+ from {
524
+ transform: rotate(0deg);
525
+ }
526
+ to {
527
+ transform: rotate(360deg);
528
+ }
529
+ }
530
+
531
+ @media (prefers-reduced-motion: no-preference) {
532
+ a:nth-of-type(2) .logo {
533
+ animation: logo-spin infinite 20s linear;
534
+ }
535
+ }
536
+
537
+ .card {
538
+ padding: 2em;
539
+ }
540
+
541
+ .read-the-docs {
542
+ color: #888;
543
+ }
544
+ `,
545
+ });
546
+ // Create main.tsx
547
+ new projen_1.SampleFile(project, `${options.srcdir}/main.tsx`, {
548
+ contents: `import { StrictMode } from 'react'
549
+ import { createRoot } from 'react-dom/client'
550
+ import './index.css'
551
+ import App from './App.tsx'
552
+ import { ClerkProvider, SignedIn, SignedOut, SignInButton } from '@clerk/clerk-react'
553
+ import { useConfig } from './config'
554
+ import { QueryClient, QueryClientProvider } from "@tanstack/react-query"
555
+
556
+ function AppWrapper() {
557
+ const { config, isLoading } = useConfig();
558
+ const queryClient = new QueryClient()
559
+ if (isLoading || !config) {
560
+ return (
561
+ <div className="loading-screen">
562
+ Loading...
563
+ </div>
564
+ );
565
+ }
566
+
567
+ return (
568
+ <ClerkProvider publishableKey={config.clerkPublishableKey}>
569
+ <QueryClientProvider client={queryClient}>
570
+ <div className="w-full h-full">
571
+ <SignedOut>
572
+ <div className="flex items-center justify-center min-h-screen">
573
+ <SignInButton/>
574
+ </div>
575
+ </SignedOut>
576
+ <SignedIn>
577
+ <App/>
578
+ </SignedIn>
579
+ </div>
580
+ </QueryClientProvider>
581
+ </ClerkProvider>
582
+ );
583
+ }
584
+
585
+ createRoot(document.getElementById('root')!).render(
586
+ <StrictMode>
587
+ <AppWrapper/>
588
+ </StrictMode>
589
+ )
590
+ `,
591
+ });
592
+ // Create index.css
593
+ new projen_1.SampleFile(project, `${options.srcdir}/index.css`, {
594
+ contents: `@import "tailwindcss";
595
+
596
+ @theme {
597
+ --color-scheme: light dark;
598
+ }
599
+
600
+ :root {
601
+ font-family: Inter, system-ui, Avenir, Helvetica, Arial, sans-serif;
602
+ line-height: 1.5;
603
+ font-weight: 400;
604
+
605
+ color-scheme: light dark;
606
+
607
+ font-synthesis: none;
608
+ text-rendering: optimizeLegibility;
609
+ -webkit-font-smoothing: antialiased;
610
+ -moz-osx-font-smoothing: grayscale;
611
+ }
612
+
613
+ a {
614
+ font-weight: 500;
615
+ color: #646cff;
616
+ text-decoration: inherit;
617
+ }
618
+ a:hover {
619
+ color: #535bf2;
620
+ }
621
+
622
+ body {
623
+ margin: 0;
624
+ min-width: 320px;
625
+ min-height: 100vh;
626
+ }
627
+
628
+ h1 {
629
+ font-size: 3.2em;
630
+ line-height: 1.1;
631
+ }
632
+
633
+ button {
634
+ font-family: inherit;
635
+ cursor: pointer;
636
+ }
637
+
638
+ .loading-screen {
639
+ display: flex;
640
+ justify-content: center;
641
+ align-items: center;
642
+ height: 100vh;
643
+ font-size: 1.5rem;
644
+ }
645
+
646
+ html, body, #root {
647
+ width: 100%;
648
+ height: 100%;
649
+ }
650
+ `,
651
+ });
652
+ }
653
+ }
654
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ViteReactProject.js","sourceRoot":"","sources":["../src/ViteReactProject.ts"],"names":[],"mappings":";;;AAAA,+BAA0C;AAC1C,mCAAsF;AACtF,sDAAuF;AA4BvF;;GAEG;AACH,MAAa,gBAAiB,SAAQ,iCAAoB;IAMxD,YAAY,OAAgC;QAC1C,MAAM,EACJ,UAAU,GAAG,IAAI,EACjB,MAAM,GAAG,KAAK,EACd,SAAS,GAAG,QAAQ,EACpB,KAAK,EACL,GAAG,WAAW,EACf,GAAG,OAAO,CAAC;QACZ,KAAK,CAAC;YACJ,GAAG,WAAW;YACd,UAAU,EAAE,KAAK,EAAE,qCAAqC;YACxD,MAAM,EAAE,KAAK,EAAE,mCAAmC;YAClD,IAAI,EAAE,KAAK,EAAE,6BAA6B;YAC1C,OAAO,EAAE,IAAI;YACb,eAAe,EAAE,IAAI,EAAE,kCAAkC;YACzD,IAAI,EAAE;gBACJ,oBAAoB;gBACpB,uBAAuB;gBACvB,sBAAsB;gBACtB,cAAc;gBACd,SAAS;gBACT,aAAa;aACd;SACF,CAAC,CAAC;QAEH,mCAAmC;QACnC,IAAI,CAAC,YAAY,GAAG,IAAI,GAAG,EAAkB,CAAC;QAE9C,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;QACrB,IAAI,CAAC,SAAS,GAAG,SAAS,CAAC;QAE3B,qBAAqB;QACrB,IAAI,aAAa,CAAC,IAAI,EAAE;YACtB,MAAM,EAAE,IAAI,CAAC,MAAM;SACpB,CAAC,CAAC;QAEH,2BAA2B;QAC3B,IAAI,gBAAgB,CAAC,IAAI,CAAC,CAAC;QAE3B,iBAAiB;QACjB,IAAI,SAAS,CAAC,IAAI,EAAE;YAClB,KAAK,EAAE,KAAK,IAAI,IAAI,CAAC,IAAI;YACzB,MAAM,EAAE,IAAI,CAAC,MAAM;SACpB,CAAC,CAAC;QAEH,yBAAyB;QACzB,IAAI,aAAa,CAAC,IAAI,EAAE;YACtB,MAAM,EAAE,IAAI,CAAC,MAAM;SACpB,CAAC,CAAC;QAEH,yBAAyB;QACzB,IAAI,CAAC,SAAS,EAAE,WAAW,CAAC,MAAM,EAAE,UAAU,EAAE,SAAS,CAAC,CAAC;QAE3D,0CAA0C;QAC1C,IAAI,UAAU,CAAC,IAAI,EAAE;YACnB,MAAM,EAAE,IAAI,CAAC,MAAM;SACpB,CAAC,CAAC;QAEH,kCAAkC;QAClC,IAAI,iBAAiB,CAAC,IAAI,CAAC,CAAC;QAE5B,0CAA0C;QAC1C,IAAI,UAAU,CAAC,IAAI,EAAE;YACnB,MAAM,EAAE,IAAI,CAAC,MAAM;SACpB,CAAC,CAAC;QAEH,yBAAyB;QACzB,IAAI,cAAc,CAAC,IAAI,EAAE,EAAG,CAAC,CAAC;QAG9B,+BAA+B;QAC/B,IAAI,UAAU,EAAE,CAAC;YACf,IAAI,cAAc,CAAC,IAAI,EAAE;gBACvB,MAAM,EAAE,IAAI,CAAC,MAAM;gBACnB,SAAS,EAAE,IAAI,CAAC,SAAS;aAC1B,CAAC,CAAC;QACL,CAAC;QAED,mCAAmC;QACnC,MAAM,cAAc,GAAG,IAAI,CAAC,MAAM;YAChC,CAAC,CAAC,IAAA,eAAQ,EAAC,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE,IAAI,CAAC,MAAM,CAAC;YAC3C,CAAC,CAAC,IAAA,eAAQ,EAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAC1B,IAAI,CAAC,UAAU,GAAG,MAAM,cAAc,cAAc,CAAC;IACvD,CAAC;IAED;;;;;OAKG;IACH,cAAc,CAAC,SAAiB,EAAE,SAAiB;QACjD,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,SAAS,EAAE,SAAS,CAAC,CAAC;QAC5C,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;OAGG;IACH,eAAe;QACb,OAAO,IAAI,CAAC,YAAY,CAAC;IAC3B,CAAC;CACF;AA7GD,4CA6GC;AAMD;;GAEG;AACH,MAAM,aAAc,SAAQ,kBAAS;IACnC,YAAY,OAA6B,EAAE,QAA8B;QACvE,KAAK,CAAC,OAAO,CAAC,CAAC;QAEf,MAAM,SAAS,GAAG,OAA+B,CAAC;QAElD,6CAA6C;QAC7C,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAC;QAE7C,yBAAyB;QACzB,SAAS,CAAC,OAAO,CAAC,eAAe,EAAE,mBAAmB,CAAC,CAAC;QAExD,uBAAuB;QACvB,SAAS,CAAC,UAAU,CAClB,sBAAsB,EACtB,0BAA0B,EAC1B,6BAA6B,EAC7B,aAAa,EACb,mBAAmB,EACnB,oBAAoB,EACpB,gBAAgB,EAChB,kCAAkC,EAClC,qCAAqC,EACrC,iBAAiB,EACjB,2BAA2B,CAC5B,CAAC;QAEF,0CAA0C;QAC1C,IAAI,cAAc,CAAC,OAAO,CAAC,CAAC;QAE5B,cAAc;QACd,SAAS,CAAC,OAAO,CAAC,KAAK,EAAE;YACvB,WAAW,EAAE,0BAA0B;YACvC,IAAI,EAAE,MAAM;SACb,CAAC,CAAC;QAEH,0DAA0D;QAC1D,SAAS,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC;QAC9B,SAAS,CAAC,OAAO,CAAC,OAAO,EAAE;YACzB,WAAW,EAAE,sBAAsB;YACnC,IAAI,EAAE,sBAAsB;SAC7B,CAAC,CAAC;QAEH,SAAS,CAAC,OAAO,CAAC,SAAS,EAAE;YAC3B,WAAW,EAAE,0BAA0B;YACvC,IAAI,EAAE,cAAc;SACrB,CAAC,CAAC;QAEH,SAAS,CAAC,OAAO,CAAC,MAAM,EAAE;YACxB,WAAW,EAAE,WAAW;YACxB,IAAI,EAAE,UAAU;SACjB,CAAC,CAAC;IACL,CAAC;CACF;AAED;;GAEG;AACH,MAAM,gBAAiB,SAAQ,iBAAQ;IACrC,YAAY,OAAgB;QAC1B,KAAK,CAAC,OAAO,EAAE,kBAAkB,EAAE;YACjC,MAAM,EAAE,IAAI;YACZ,QAAQ,EAAE,IAAI;SACf,CAAC,CAAC;IACL,CAAC;IAED,iBAAiB;QACf,OAAO;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CA6BV,CAAC;IACA,CAAC;CACF;AAED;;GAEG;AACH,MAAM,cAAe,SAAQ,iBAAQ;IACnC,YAAY,OAAgB;QAC1B,KAAK,CAAC,OAAO,EAAE,gBAAgB,EAAE;YAC/B,MAAM,EAAE,IAAI;YACZ,QAAQ,EAAE,IAAI;SACf,CAAC,CAAC;IACL,CAAC;IAED,iBAAiB;QACf,OAAO;;;;;;;CAOV,CAAC;IACA,CAAC;CACF;AAED;;GAEG;AACH,MAAM,iBAAkB,SAAQ,iBAAQ;IACtC,YAAY,OAAgB;QAC1B,KAAK,CAAC,OAAO,EAAE,mBAAmB,EAAE;YAClC,MAAM,EAAE,IAAI;YACZ,QAAQ,EAAE,IAAI;SACf,CAAC,CAAC;IACL,CAAC;IAED,iBAAiB;QACf,OAAO;;;;;;CAMV,CAAC;IACA,CAAC;CACF;AAMD;;GAEG;AACH,MAAM,UAAW,SAAQ,iBAAQ;IAC/B,YAAY,OAAgB,EAAE,OAA0B;QACtD,KAAK,CAAC,OAAO,EAAE,GAAG,OAAO,CAAC,MAAM,gBAAgB,EAAE;YAChD,MAAM,EAAE,IAAI;YACZ,QAAQ,EAAE,IAAI;SACf,CAAC,CAAC;IACL,CAAC;IAED,iBAAiB;QACf,OAAO;CACV,CAAC;IACA,CAAC;CACF;AAMD;;GAEG;AACH,MAAM,UAAW,SAAQ,iBAAQ;IAC/B,YAAY,OAAgB,EAAE,OAA0B;QACtD,KAAK,CAAC,OAAO,EAAE,GAAG,OAAO,CAAC,MAAM,YAAY,EAAE;YAC5C,MAAM,EAAE,IAAI;YACZ,QAAQ,EAAE,IAAI;SACf,CAAC,CAAC;IACL,CAAC;IAED,iBAAiB;QACf,+CAA+C;QAC/C,MAAM,WAAW,GAAG,IAAI,CAAC,OAA2B,CAAC;QACrD,MAAM,YAAY,GAAG,WAAW,CAAC,eAAe,EAAE,CAAC;QAEnD,4BAA4B;QAC5B,MAAM,cAAc,GAAG;;;;EAIzB,CAAC;QAEC,2CAA2C;QAC3C,IAAI,oBAA4B,CAAC;QACjC,IAAI,YAAY,CAAC,IAAI,GAAG,CAAC,EAAE,CAAC;YAC1B,MAAM,gBAAgB,GAAG,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,OAAO,EAAE,CAAC;iBACxD,GAAG,CAAC,CAAC,CAAC,IAAI,EAAE,IAAI,CAAC,EAAE,EAAE,CAAC,KAAK,IAAI,KAAK,IAAI,GAAG,CAAC;iBAC5C,IAAI,CAAC,IAAI,CAAC,CAAC;YAEd,MAAM,gBAAgB,GAAG;EAC7B,gBAAgB;EAChB,CAAC;YAEG,oBAAoB,GAAG,GAAG,gBAAgB;;gDAEA,CAAC;QAC7C,CAAC;aAAM,CAAC;YACN,oBAAoB,GAAG,kCAAkC,CAAC;QAC5D,CAAC;QAED,OAAO;;EAET,cAAc;;EAEd,oBAAoB;;;;;;;;;;;;;;;;;;;;CAoBrB,CAAC;IACA,CAAC;CACF;AAID;;GAEG;AACH,MAAM,cAAe,SAAQ,mBAAU;IAErC,YAAY,OAAyB,EAAE,QAA+B;QACpE,KAAK,CAAC,OAAO,EAAE,GAAG,OAAO,CAAC,SAAS,cAAc,EAAE;YACjD,QAAQ,EAAE,IAAI,CAAC,SAAS,CAAC;gBACvB,UAAU,EAAE,2BAA2B;gBACvC,mBAAmB,EAAE,KAAK;aAC3B,EAAE,IAAI,EAAE,CAAC,CAAC;SACZ,CAAC,CAAC;QACH,IAAI,CAAC,IAAI,GAAG,GAAG,OAAO,CAAC,SAAS,cAAc,CAAC;IACjD,CAAC;CACF;AAOD;;GAEG;AACH,MAAM,SAAU,SAAQ,iBAAQ;IAI9B,YAAY,OAAgB,EAAE,OAAyB;QACrD,KAAK,CAAC,OAAO,EAAE,YAAY,EAAE;YAC3B,MAAM,EAAE,IAAI;YACZ,QAAQ,EAAE,IAAI;SACf,CAAC,CAAC;QAEH,IAAI,CAAC,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC;QAC3B,IAAI,CAAC,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;IAC/B,CAAC;IAED,iBAAiB;QACf,OAAO;;;;;;aAME,IAAI,CAAC,KAAK;;;;kCAIW,IAAI,CAAC,MAAM;;;CAG5C,CAAC;IACA,CAAC;CACF;AAMD;;GAEG;AACH,MAAM,aAAc,SAAQ,kBAAS;IACnC,YAAY,OAAgB,EAAE,OAA6B;QACzD,KAAK,CAAC,OAAO,CAAC,CAAC;QAEf,0CAA0C;QAC1C,IAAI,iBAAQ,CAAC,OAAO,EAAE,eAAe,EAAE;YACrC,MAAM,EAAE,IAAI;YACZ,QAAQ,EAAE,IAAI;YACd,GAAG,EAAE;gBACH,KAAK,EAAE,EAAE;gBACT,UAAU,EAAE;oBACV,EAAE,IAAI,EAAE,qBAAqB,EAAE;oBAC/B,EAAE,IAAI,EAAE,sBAAsB,EAAE;iBACjC;aACF;SACF,CAAC,CAAC;QAEH,2CAA2C;QAC3C,IAAI,iBAAQ,CAAC,OAAO,EAAE,mBAAmB,EAAE;YACzC,MAAM,EAAE,IAAI;YACZ,QAAQ,EAAE,IAAI;YACd,GAAG,EAAE;gBACH,eAAe,EAAE;oBACf,MAAM,EAAE,QAAQ;oBAChB,uBAAuB,EAAE,IAAI;oBAC7B,GAAG,EAAE,CAAC,QAAQ,EAAE,KAAK,EAAE,cAAc,CAAC;oBACtC,MAAM,EAAE,QAAQ;oBAChB,YAAY,EAAE,IAAI;oBAElB,eAAe;oBACf,gBAAgB,EAAE,SAAS;oBAC3B,0BAA0B,EAAE,IAAI;oBAChC,eAAe,EAAE,IAAI;oBACrB,eAAe,EAAE,OAAO;oBACxB,MAAM,EAAE,IAAI;oBACZ,GAAG,EAAE,WAAW;oBAEhB,UAAU;oBACV,MAAM,EAAE,IAAI;oBACZ,cAAc,EAAE,IAAI;oBACpB,kBAAkB,EAAE,IAAI;oBACxB,0BAA0B,EAAE,IAAI;oBAChC,4BAA4B,EAAE,IAAI;iBACnC;gBACD,OAAO,EAAE,CAAC,OAAO,CAAC,MAAM,CAAC;aAC1B;SACF,CAAC,CAAC;QAEH,uCAAuC;QACvC,IAAI,iBAAQ,CAAC,OAAO,EAAE,oBAAoB,EAAE;YAC1C,MAAM,EAAE,IAAI;YACZ,QAAQ,EAAE,IAAI;YACd,GAAG,EAAE;gBACH,eAAe,EAAE;oBACf,MAAM,EAAE,QAAQ;oBAChB,GAAG,EAAE,CAAC,QAAQ,CAAC;oBACf,MAAM,EAAE,QAAQ;oBAChB,YAAY,EAAE,IAAI;oBAElB,eAAe;oBACf,gBAAgB,EAAE,SAAS;oBAC3B,0BAA0B,EAAE,IAAI;oBAChC,eAAe,EAAE,IAAI;oBACrB,eAAe,EAAE,OAAO;oBACxB,MAAM,EAAE,IAAI;oBAEZ,UAAU;oBACV,MAAM,EAAE,IAAI;oBACZ,cAAc,EAAE,IAAI;oBACpB,kBAAkB,EAAE,IAAI;oBACxB,0BAA0B,EAAE,IAAI;iBACjC;gBACD,OAAO,EAAE,CAAC,gBAAgB,CAAC;aAC5B;SACF,CAAC,CAAC;IACL,CAAC;CACF;AAOD;;GAEG;AACH,MAAM,cAAe,SAAQ,kBAAS;IACpC,YAAY,OAAgB,EAAE,OAA8B;QAC1D,KAAK,CAAC,OAAO,CAAC,CAAC;QAEf,iBAAiB;QACjB,IAAI,mBAAU,CAAC,OAAO,EAAE,GAAG,OAAO,CAAC,MAAM,UAAU,EAAE;YACnD,QAAQ,EAAE;;;;;;;;;;;;;;;;;;;;;;;;;CAyBf;SACI,CAAC,CAAC;QAEH,wBAAwB;QACxB,IAAI,iBAAQ,CAAC,OAAO,EAAE,GAAG,OAAO,CAAC,MAAM,0BAA0B,EAAE;YACjE,MAAM,EAAE,IAAI;YACZ,QAAQ,EAAE,IAAI;YACd,KAAK,EAAE;gBACL,sEAAsE;gBACtE,iDAAiD;gBACjD,0CAA0C;gBAC1C,EAAE;gBACF,gDAAgD;gBAChD,iBAAiB;gBACjB,sEAAsE;gBACtE,KAAK;gBACL,mCAAmC;gBACnC,mCAAmC;gBACnC,EAAE;gBACF,4BAA4B;gBAC5B,2BAA2B;gBAC3B,4BAA4B;gBAC5B,kCAAkC;gBAClC,2DAA2D;gBAC3D,SAAS;gBACT,EAAE;gBACF,uCAAuC;gBACvC,EAAE;gBACF,oGAAoG;gBACpG,iEAAiE;gBACjE,0EAA0E;gBAC1E,QAAQ;gBACR,qEAAqE;gBACrE,oBAAoB;gBACpB,iDAAiD;gBACjD,YAAY;gBACZ,WAAW;gBACX,EAAE;gBACF,2BAA2B;gBAC3B,oEAAoE;gBACpE,SAAS;gBACT,EAAE;gBACF,+BAA+B;gBAC/B,QAAQ;gBACR,oCAAoC;gBACpC,iBAAiB;gBACjB,OAAO;gBACP,GAAG;aACJ;SACF,CAAC,CAAC;QAEH,wBAAwB;QACxB,IAAI,mBAAU,CAAC,OAAO,EAAE,GAAG,OAAO,CAAC,MAAM,0BAA0B,EAAE;YACnE,QAAQ,EAAE;;;;;;;;CAQf;SACI,CAAC,CAAC;QAEH,iBAAiB;QACjB,IAAI,mBAAU,CAAC,OAAO,EAAE,GAAG,OAAO,CAAC,MAAM,UAAU,EAAE;YACnD,QAAQ,EAAE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CAwCf;SACI,CAAC,CAAC;QAEH,kBAAkB;QAClB,IAAI,mBAAU,CAAC,OAAO,EAAE,GAAG,OAAO,CAAC,MAAM,WAAW,EAAE;YACpD,QAAQ,EAAE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CA0Cf;SACI,CAAC,CAAC;QAEH,mBAAmB;QACnB,IAAI,mBAAU,CAAC,OAAO,EAAE,GAAG,OAAO,CAAC,MAAM,YAAY,EAAE;YACrD,QAAQ,EAAE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CAwDf;SACI,CAAC,CAAC;IAEL,CAAC;CACF","sourcesContent":["import { relative, basename } from 'path';\nimport { Component, FileBase, JsonFile, Project, SampleFile, TextFile } from 'projen';\nimport { TypeScriptAppProject, TypeScriptProjectOptions } from 'projen/lib/typescript';\n\nexport interface ViteReactProjectOptions extends TypeScriptProjectOptions {\n  /**\n   * Whether to generate sample code\n   * @default true\n   */\n  readonly sampleCode?: boolean;\n\n  /**\n   * Source directory\n   * @default \"src\"\n   */\n  readonly srcdir?: string;\n\n  /**\n   * Public directory for static assets\n   * @default \"public\"\n   */\n  readonly publicDir?: string;\n\n  /**\n   * Title for the HTML page\n   * @default project name\n   */\n  readonly title?: string;\n}\n\n/**\n * Vite + React TypeScript project\n */\nexport class ViteReactProject extends TypeScriptAppProject {\n  public readonly srcdir: string;\n  public readonly publicDir: string;\n  public readonly devCommand: string;\n  private readonly configFields: Map<string, string>;\n\n  constructor(options: ViteReactProjectOptions) {\n    const {\n      sampleCode = true,\n      srcdir = 'src',\n      publicDir = 'public',\n      title,\n      ...restOptions\n    } = options;\n    super({\n      ...restOptions,\n      sampleCode: false, // We'll handle sample code ourselves\n      eslint: false, // We'll configure eslint ourselves\n      jest: false, // Vite uses vitest typically\n      package: true,\n      disableTsconfig: true, // We'll manage tsconfig ourselves\n      deps: [\n        '@clerk/clerk-react',\n        '@tanstack/react-query',\n        '@tailwindcss/postcss',\n        'autoprefixer',\n        'postcss',\n        'tailwindcss',\n      ],\n    });\n\n    // Initialize config fields storage\n    this.configFields = new Map<string, string>();\n\n    this.srcdir = srcdir;\n    this.publicDir = publicDir;\n\n    // Add Vite component\n    new ViteComponent(this, {\n      srcdir: this.srcdir,\n    });\n\n    // Add ESLint configuration\n    new ViteEslintConfig(this);\n\n    // Add index.html\n    new IndexHtml(this, {\n      title: title ?? this.name,\n      srcdir: this.srcdir,\n    });\n\n    // Add TypeScript configs\n    new ViteTsConfigs(this, {\n      srcdir: this.srcdir,\n    });\n\n    // Add .gitignore entries\n    this.gitignore?.addPatterns('dist', 'dist-ssr', '*.local');\n\n    // Add vite-env.d.ts for type declarations\n    new ViteEnvDts(this, {\n      srcdir: this.srcdir,\n    });\n\n    // Add PostCSS config for Tailwind\n    new PostCSSConfigFile(this);\n\n    // Add config.ts for runtime configuration\n    new ConfigFile(this, {\n      srcdir: this.srcdir,\n    });\n\n    // Add sample config.json\n    new ConfigJsonFile(this, { });\n\n\n    // Add sample code if requested\n    if (sampleCode) {\n      new ViteSampleCode(this, {\n        srcdir: this.srcdir,\n        publicDir: this.publicDir,\n      });\n    }\n\n    // Use relative path for devCommand\n    const relativeOutdir = this.parent\n      ? relative(this.parent.outdir, this.outdir)\n      : basename(this.outdir);\n    this.devCommand = `cd ${relativeOutdir} && yarn dev`;\n  }\n\n  /**\n   * Add a custom field to the Config type\n   * @param fieldName The name of the field\n   * @param fieldType The TypeScript type of the field (e.g., 'string', 'number', 'boolean', 'string[]')\n   * @returns this project instance for chaining\n   */\n  addConfigField(fieldName: string, fieldType: string): this {\n    this.configFields.set(fieldName, fieldType);\n    return this;\n  }\n\n  /**\n   * Get the config fields map (for internal use by ConfigFile)\n   * @internal\n   */\n  getConfigFields(): Map<string, string> {\n    return this.configFields;\n  }\n}\n\ninterface ViteComponentOptions {\n  readonly srcdir: string;\n}\n\n/**\n * Configures Vite dependencies and scripts\n */\nclass ViteComponent extends Component {\n  constructor(project: TypeScriptAppProject, _options: ViteComponentOptions) {\n    super(project);\n\n    const tsProject = project as TypeScriptAppProject;\n\n    // Set package type to module for ESM support\n    tsProject.package.addField('type', 'module');\n\n    // Add React dependencies\n    tsProject.addDeps('react@^19.2.0', 'react-dom@^19.2.0');\n\n    // Add dev dependencies\n    tsProject.addDevDeps(\n      '@types/react@^19.2.2',\n      '@types/react-dom@^19.2.2',\n      '@vitejs/plugin-react@^5.1.0',\n      'vite@^5.4.0',\n      'typescript@~5.9.3',\n      '@eslint/js@^9.39.1',\n      'eslint@^9.39.1',\n      'eslint-plugin-react-hooks@^5.2.0',\n      'eslint-plugin-react-refresh@^0.4.24',\n      'globals@^16.5.0',\n      'typescript-eslint@^8.46.3',\n    );\n\n    // Create vite.config.ts as a managed file\n    new ViteConfigFile(project);\n\n    // Add scripts\n    tsProject.addTask('dev', {\n      description: 'Start development server',\n      exec: 'vite',\n    });\n\n    // Remove and recreate build task to customize it for Vite\n    tsProject.removeTask('build');\n    tsProject.addTask('build', {\n      description: 'Build for production',\n      exec: 'tsc -b && vite build',\n    });\n\n    tsProject.addTask('preview', {\n      description: 'Preview production build',\n      exec: 'vite preview',\n    });\n\n    tsProject.addTask('lint', {\n      description: 'Lint code',\n      exec: 'eslint .',\n    });\n  }\n}\n\n/**\n * Creates the ESLint configuration for Vite + React\n */\nclass ViteEslintConfig extends FileBase {\n  constructor(project: Project) {\n    super(project, 'eslint.config.js', {\n      marker: true,\n      readonly: true,\n    });\n  }\n\n  synthesizeContent(): string | undefined {\n    return `import js from '@eslint/js'\nimport globals from 'globals'\nimport reactHooks from 'eslint-plugin-react-hooks'\nimport reactRefresh from 'eslint-plugin-react-refresh'\nimport tseslint from 'typescript-eslint'\n\nexport default [\n  { ignores: ['dist'] },\n  {\n    files: ['**/*.{ts,tsx}'],\n    languageOptions: {\n      ecmaVersion: 2020,\n      globals: globals.browser,\n    },\n    plugins: {\n      'react-hooks': reactHooks,\n      'react-refresh': reactRefresh,\n    },\n    rules: {\n      ...reactHooks.configs.recommended.rules,\n      'react-refresh/only-export-components': [\n        'warn',\n        { allowConstantExport: true },\n      ],\n    },\n  },\n  js.configs.recommended,\n  ...tseslint.configs.recommended,\n]\n`;\n  }\n}\n\n/**\n * Creates the vite.config.ts file\n */\nclass ViteConfigFile extends FileBase {\n  constructor(project: Project) {\n    super(project, 'vite.config.ts', {\n      marker: true,\n      readonly: true,\n    });\n  }\n\n  synthesizeContent(): string | undefined {\n    return `import { defineConfig } from 'vite'\nimport react from '@vitejs/plugin-react'\n\n// https://vite.dev/config/\nexport default defineConfig({\n  plugins: [react()],\n})\n`;\n  }\n}\n\n/**\n * Creates the postcss.config.js file for Tailwind CSS v4\n */\nclass PostCSSConfigFile extends FileBase {\n  constructor(project: Project) {\n    super(project, 'postcss.config.js', {\n      marker: true,\n      readonly: true,\n    });\n  }\n\n  synthesizeContent(): string | undefined {\n    return `export default {\n  plugins: {\n    \"@tailwindcss/postcss\": {},\n    autoprefixer: {},\n  },\n}\n`;\n  }\n}\n\ninterface ViteEnvDtsOptions {\n  readonly srcdir: string;\n}\n\n/**\n * Creates the vite-env.d.ts file for type declarations\n */\nclass ViteEnvDts extends FileBase {\n  constructor(project: Project, options: ViteEnvDtsOptions) {\n    super(project, `${options.srcdir}/vite-env.d.ts`, {\n      marker: true,\n      readonly: true,\n    });\n  }\n\n  synthesizeContent(): string | undefined {\n    return `/// <reference types=\"vite/client\" />\n`;\n  }\n}\n\ninterface ConfigFileOptions {\n  readonly srcdir: string;\n}\n\n/**\n * Creates the config.ts file for runtime configuration\n */\nclass ConfigFile extends FileBase {\n  constructor(project: Project, options: ConfigFileOptions) {\n    super(project, `${options.srcdir}/config.ts`, {\n      marker: true,\n      readonly: true,\n    });\n  }\n\n  synthesizeContent(): string | undefined {\n    // Get custom config fields from parent project\n    const viteProject = this.project as ViteReactProject;\n    const customFields = viteProject.getConfigFields();\n\n    // Generate base config type\n    const baseConfigType = `type BaseConfig = {\n  clerkPublishableKey: string;\n  apiBaseUrl: string;\n  timeout: number;\n}`;\n\n    // Generate custom fields type if any exist\n    let configTypeDefinition: string;\n    if (customFields.size > 0) {\n      const fieldDefinitions = Array.from(customFields.entries())\n        .map(([name, type]) => `  ${name}: ${type};`)\n        .join('\\n');\n\n      const customFieldsType = `type CustomConfig = {\n${fieldDefinitions}\n}`;\n\n      configTypeDefinition = `${customFieldsType}\n\nexport type Config = BaseConfig & CustomConfig;`;\n    } else {\n      configTypeDefinition = 'export type Config = BaseConfig;';\n    }\n\n    return `import { useEffect, useState } from 'react';\n\n${baseConfigType}\n\n${configTypeDefinition}\n\nexport function useConfig() {\n  const [loadedConfig, setLoadedConfig] = useState<Config|undefined>();\n  const [isLoading, setIsLoading] = useState(false);\n\n  useEffect(() => {\n    if (!loadedConfig) {\n      fetch('/config.json', { cache: 'no-store' }).then(response=>{\n        response.json().then((config) => {\n          setIsLoading(false)\n          setLoadedConfig(config)\n        });\n      });\n\n    }\n  }, [loadedConfig]);\n\n  return { config: loadedConfig, isLoading };\n}\n`;\n  }\n}\n\ninterface ConfigJsonFileOptions {}\n\n/**\n * Creates the config.json sample file for runtime configuration\n */\nclass ConfigJsonFile extends SampleFile {\n  path: string;\n  constructor(project: ViteReactProject, _options: ConfigJsonFileOptions) {\n    super(project, `${project.publicDir}/config.json`, {\n      contents: JSON.stringify({\n        apiBaseUrl: 'http://localhost:3000/api',\n        clerkPublishableKey: 'tbd',\n      }, null, 2),\n    });\n    this.path = `${project.publicDir}/config.json`;\n  }\n}\n\ninterface IndexHtmlOptions {\n  readonly title: string;\n  readonly srcdir: string;\n}\n\n/**\n * Creates the index.html file\n */\nclass IndexHtml extends FileBase {\n  private readonly title: string;\n  private readonly srcdir: string;\n\n  constructor(project: Project, options: IndexHtmlOptions) {\n    super(project, 'index.html', {\n      marker: true,\n      readonly: true,\n    });\n\n    this.title = options.title;\n    this.srcdir = options.srcdir;\n  }\n\n  synthesizeContent(): string | undefined {\n    return `<!doctype html>\n<html lang=\"en\">\n  <head>\n    <meta charset=\"UTF-8\" />\n    <link rel=\"icon\" type=\"image/svg+xml\" href=\"/vite.svg\" />\n    <meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0\" />\n    <title>${this.title}</title>\n  </head>\n  <body>\n    <div id=\"root\"></div>\n    <script type=\"module\" src=\"/${this.srcdir}/main.tsx\"></script>\n  </body>\n</html>\n`;\n  }\n}\n\ninterface ViteTsConfigsOptions {\n  readonly srcdir: string;\n}\n\n/**\n * Creates the TypeScript configuration files for Vite\n */\nclass ViteTsConfigs extends Component {\n  constructor(project: Project, options: ViteTsConfigsOptions) {\n    super(project);\n\n    // Main tsconfig.json (project references)\n    new JsonFile(project, 'tsconfig.json', {\n      marker: true,\n      readonly: true,\n      obj: {\n        files: [],\n        references: [\n          { path: './tsconfig.app.json' },\n          { path: './tsconfig.node.json' },\n        ],\n      },\n    });\n\n    // tsconfig.app.json (for application code)\n    new JsonFile(project, 'tsconfig.app.json', {\n      marker: true,\n      readonly: true,\n      obj: {\n        compilerOptions: {\n          target: 'ES2020',\n          useDefineForClassFields: true,\n          lib: ['ES2020', 'DOM', 'DOM.Iterable'],\n          module: 'ESNext',\n          skipLibCheck: true,\n\n          // Bundler mode\n          moduleResolution: 'bundler',\n          allowImportingTsExtensions: true,\n          isolatedModules: true,\n          moduleDetection: 'force',\n          noEmit: true,\n          jsx: 'react-jsx',\n\n          // Linting\n          strict: true,\n          noUnusedLocals: true,\n          noUnusedParameters: true,\n          noFallthroughCasesInSwitch: true,\n          noUncheckedSideEffectImports: true,\n        },\n        include: [options.srcdir],\n      },\n    });\n\n    // tsconfig.node.json (for Vite config)\n    new JsonFile(project, 'tsconfig.node.json', {\n      marker: true,\n      readonly: true,\n      obj: {\n        compilerOptions: {\n          target: 'ES2022',\n          lib: ['ES2023'],\n          module: 'ESNext',\n          skipLibCheck: true,\n\n          // Bundler mode\n          moduleResolution: 'bundler',\n          allowImportingTsExtensions: true,\n          isolatedModules: true,\n          moduleDetection: 'force',\n          noEmit: true,\n\n          // Linting\n          strict: true,\n          noUnusedLocals: true,\n          noUnusedParameters: true,\n          noFallthroughCasesInSwitch: true,\n        },\n        include: ['vite.config.ts'],\n      },\n    });\n  }\n}\n\ninterface ViteSampleCodeOptions {\n  readonly srcdir: string;\n  readonly publicDir: string;\n}\n\n/**\n * Generates sample code for a Vite + React project\n */\nclass ViteSampleCode extends Component {\n  constructor(project: Project, options: ViteSampleCodeOptions) {\n    super(project);\n\n    // Create App.tsx\n    new SampleFile(project, `${options.srcdir}/App.tsx`, {\n      contents: `import './App.css'\nimport { SignedIn, UserButton } from '@clerk/clerk-react';\nimport { useBaseQuery } from \"./services/BaseService.ts\";\n\n\nfunction App() {\n  const {data, isLoading} = useBaseQuery('/', {retryOnMount: true});\n\n\n  return (\n    <>\n        <header>\n\n          <SignedIn>\n            <UserButton/>\n          </SignedIn>\n        </header>\n        <div className=\"card\">\n          { isLoading ? 'Loading...' : JSON.stringify(data, null ,)}\n        </div>\n    </>\n  )\n}\n\nexport default App\n`,\n    });\n\n    // Create BaseService.ts\n    new TextFile(project, `${options.srcdir}/services/BaseService.ts`, {\n      marker: true,\n      readonly: true,\n      lines: [\n        'import { useQuery, UseQueryOptions } from \\'@tanstack/react-query\\';',\n        'import { useAuth } from \\'@clerk/clerk-react\\';',\n        'import { useConfig } from \\'../config\\';',\n        '',\n        'export function useBaseQuery<TData = unknown>(',\n        '  path: string,',\n        '  options?: Omit<UseQueryOptions<TData>, \\'queryKey\\' | \\'queryFn\\'>',\n        ') {',\n        '  const { config } = useConfig();',\n        '  const { getToken } = useAuth();',\n        '',\n        '  return useQuery<TData>({',\n        '    queryKey: [\\'base\\'],',\n        '    queryFn: async () => {',\n        '      if (!config?.apiBaseUrl) {',\n        '        throw new Error(\\'API base URL not configured\\');',\n        '      }',\n        '',\n        '      const token = await getToken();',\n        '',\n        '      // Normalize URL by removing trailing slash from baseUrl and ensuring path starts with slash',\n        '      const baseUrl = config.apiBaseUrl.replace(/\\\\/+$/, \\'\\');',\n        '      const normalizedPath = path.startsWith(\\'/\\') ? path : `/${path}`;',\n        '      ',\n        '      const response = await fetch(`${baseUrl}${normalizedPath}`, {',\n        '        headers: {',\n        '          \\'Authorization\\': `Bearer ${token}`,',\n        '        },',\n        '      });',\n        '',\n        '      if (!response.ok) {',\n        '        throw new Error(`HTTP error! status: ${response.status}`);',\n        '      }',\n        '',\n        '      return response.json();',\n        '    },',\n        '    enabled: !!config?.apiBaseUrl,',\n        '    ...options,',\n        '  });',\n        '}',\n      ],\n    });\n\n    // Create RootService.ts\n    new SampleFile(project, `${options.srcdir}/services/RootService.ts`, {\n      contents: `import { UseQueryOptions } from '@tanstack/react-query';\nimport { useBaseQuery } from './BaseService';\n\nexport function useRootQuery<TData = unknown>(\n  options?: Omit<UseQueryOptions<TData>, 'queryKey' | 'queryFn'>\n) {\n  return useBaseQuery<TData>('/', options);\n}\n`,\n    });\n\n    // Create App.css\n    new SampleFile(project, `${options.srcdir}/App.css`, {\n      contents: `#root {\n  width: 100%;\n  height: 100%;\n}\n\n.logo {\n  height: 6em;\n  padding: 1.5em;\n  will-change: filter;\n  transition: filter 300ms;\n}\n.logo:hover {\n  filter: drop-shadow(0 0 2em #646cffaa);\n}\n.logo.react:hover {\n  filter: drop-shadow(0 0 2em #61dafbaa);\n}\n\n@keyframes logo-spin {\n  from {\n    transform: rotate(0deg);\n  }\n  to {\n    transform: rotate(360deg);\n  }\n}\n\n@media (prefers-reduced-motion: no-preference) {\n  a:nth-of-type(2) .logo {\n    animation: logo-spin infinite 20s linear;\n  }\n}\n\n.card {\n  padding: 2em;\n}\n\n.read-the-docs {\n  color: #888;\n}\n`,\n    });\n\n    // Create main.tsx\n    new SampleFile(project, `${options.srcdir}/main.tsx`, {\n      contents: `import { StrictMode } from 'react'\nimport { createRoot } from 'react-dom/client'\nimport './index.css'\nimport App from './App.tsx'\nimport { ClerkProvider, SignedIn, SignedOut, SignInButton } from '@clerk/clerk-react'\nimport { useConfig } from './config'\nimport { QueryClient, QueryClientProvider } from \"@tanstack/react-query\"\n\nfunction AppWrapper() {\n  const { config, isLoading } = useConfig();\n  const queryClient = new QueryClient()\n  if (isLoading || !config) {\n    return (\n      <div className=\"loading-screen\">\n        Loading...\n      </div>\n    );\n  }\n\n  return (\n    <ClerkProvider publishableKey={config.clerkPublishableKey}>\n      <QueryClientProvider client={queryClient}>\n        <div className=\"w-full h-full\">\n          <SignedOut>\n            <div className=\"flex items-center justify-center min-h-screen\">\n              <SignInButton/>\n            </div>\n          </SignedOut>\n          <SignedIn>\n            <App/>\n          </SignedIn>\n        </div>\n      </QueryClientProvider>\n    </ClerkProvider>\n  );\n}\n\ncreateRoot(document.getElementById('root')!).render(\n  <StrictMode>\n    <AppWrapper/>\n  </StrictMode>\n)\n`,\n    });\n\n    // Create index.css\n    new SampleFile(project, `${options.srcdir}/index.css`, {\n      contents: `@import \"tailwindcss\";\n\n@theme {\n  --color-scheme: light dark;\n}\n\n:root {\n  font-family: Inter, system-ui, Avenir, Helvetica, Arial, sans-serif;\n  line-height: 1.5;\n  font-weight: 400;\n\n  color-scheme: light dark;\n\n  font-synthesis: none;\n  text-rendering: optimizeLegibility;\n  -webkit-font-smoothing: antialiased;\n  -moz-osx-font-smoothing: grayscale;\n}\n\na {\n  font-weight: 500;\n  color: #646cff;\n  text-decoration: inherit;\n}\na:hover {\n  color: #535bf2;\n}\n\nbody {\n  margin: 0;\n  min-width: 320px;\n  min-height: 100vh;\n}\n\nh1 {\n  font-size: 3.2em;\n  line-height: 1.1;\n}\n\nbutton {\n  font-family: inherit;\n  cursor: pointer;\n}\n\n.loading-screen {\n  display: flex;\n  justify-content: center;\n  align-items: center;\n  height: 100vh;\n  font-size: 1.5rem;\n}\n\nhtml, body, #root {\n  width: 100%;\n  height: 100%;\n}\n`,\n    });\n\n  }\n}\n"]}