@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.
- package/.jsii +637 -0
- package/API.md +5671 -0
- package/LICENSE +202 -0
- package/README.md +86 -0
- package/lib/ApiConstructFile.d.ts +5 -0
- package/lib/ApiConstructFile.js +175 -0
- package/lib/ApiFunctionFile.d.ts +5 -0
- package/lib/ApiFunctionFile.js +30 -0
- package/lib/Application.d.ts +21 -0
- package/lib/Application.js +72 -0
- package/lib/ApplicationStackFile.d.ts +5 -0
- package/lib/ApplicationStackFile.js +26 -0
- package/lib/BaseApplicationStackFile.d.ts +5 -0
- package/lib/BaseApplicationStackFile.js +172 -0
- package/lib/BatmanProject.d.ts +25 -0
- package/lib/BatmanProject.js +101 -0
- package/lib/GitHubRolesFile.d.ts +9 -0
- package/lib/GitHubRolesFile.js +26 -0
- package/lib/GitHubRolesStackFile.d.ts +8 -0
- package/lib/GitHubRolesStackFile.js +165 -0
- package/lib/LocalDevAppFile.d.ts +10 -0
- package/lib/LocalDevAppFile.js +34 -0
- package/lib/MainFile.d.ts +24 -0
- package/lib/MainFile.js +64 -0
- package/lib/MainTestFile.d.ts +5 -0
- package/lib/MainTestFile.js +33 -0
- package/lib/PrCleanupWorkflow.d.ts +9 -0
- package/lib/PrCleanupWorkflow.js +100 -0
- package/lib/PrDeployWorkflow.d.ts +10 -0
- package/lib/PrDeployWorkflow.js +106 -0
- package/lib/ProductionDeployWorkflow.d.ts +11 -0
- package/lib/ProductionDeployWorkflow.js +68 -0
- package/lib/StagingDeployWorkflow.d.ts +11 -0
- package/lib/StagingDeployWorkflow.js +68 -0
- package/lib/StaticWebsiteConstructFile.d.ts +5 -0
- package/lib/StaticWebsiteConstructFile.js +198 -0
- package/lib/ViteReactProject.d.ts +45 -0
- package/lib/ViteReactProject.js +654 -0
- package/lib/index.d.ts +5 -0
- package/lib/index.js +20 -0
- 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":"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","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"]}
|