@tomjs/create-app 2.1.1 → 4.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +8 -7
- package/README.zh_CN.md +8 -7
- package/dist/index.mjs +10 -9
- package/locales/en-US.json +4 -3
- package/locales/zh-CN.json +4 -3
- package/package.json +9 -8
- package/templates/config/base/_eslint.config.mjs +1 -4
- package/templates/config/base/package.json +4 -4
- package/templates/config/package.json +57 -0
- package/templates/electron-react/package.json +2 -2
- package/templates/electron-vue/package.json +2 -2
- package/templates/node-cli/package.json +1 -1
- package/templates/node-vite/examples/react/package.json +1 -0
- package/templates/node-vite/examples/react/vite.config.ts +2 -1
- package/templates/node-vite/examples/vue/package.json +1 -0
- package/templates/node-vite/examples/vue/vite.config.ts +2 -1
- package/templates/node-vite/package.json +4 -2
- package/templates/node-vite/tsdown.config.ts +1 -0
- package/templates/vscode-base/package.json +1 -1
- package/templates/vscode-react/extension/views/helper.ts +2 -1
- package/templates/vscode-react/package.json +1 -1
- package/templates/vscode-vue/extension/views/helper.ts +2 -1
- package/templates/vscode-vue/package.json +2 -2
- package/templates/web-react/package.json +2 -2
- package/templates/web-vue/package.json +1 -1
- package/templates/node-vite/pnpm-lock.yaml +0 -2293
package/README.md
CHANGED
|
@@ -26,19 +26,20 @@ Then generate the project based on the prompts.
|
|
|
26
26
|
```bash
|
|
27
27
|
$ pnpm create @tomjs/app -h
|
|
28
28
|
|
|
29
|
-
Create a
|
|
29
|
+
Create a node/web/electron/vscode project based on tomjs
|
|
30
30
|
|
|
31
31
|
Usage
|
|
32
|
-
$ create-app [options] <
|
|
32
|
+
$ create-app [options] <package-name>
|
|
33
33
|
|
|
34
|
-
|
|
34
|
+
package-name Package name
|
|
35
35
|
|
|
36
36
|
Options
|
|
37
|
-
-o
|
|
38
|
-
-p
|
|
37
|
+
--overwrite, -o Overwrite existing project
|
|
38
|
+
--package, -p Create a package for the workspace project
|
|
39
|
+
--private Set as private project
|
|
39
40
|
--verbose Show verbose logs
|
|
40
|
-
-h
|
|
41
|
-
-v
|
|
41
|
+
--help, -h Show help information
|
|
42
|
+
--version, -v Show version information
|
|
42
43
|
```
|
|
43
44
|
|
|
44
45
|
Support templates:
|
package/README.zh_CN.md
CHANGED
|
@@ -26,19 +26,20 @@ npm create @tomjs/app@latest
|
|
|
26
26
|
```bash
|
|
27
27
|
$ pnpm create @tomjs/app -h
|
|
28
28
|
|
|
29
|
-
创建基于 tomjs 的 node/web/electron/vscode
|
|
29
|
+
创建基于 tomjs 的 node/web/electron/vscode 项目
|
|
30
30
|
|
|
31
31
|
Usage
|
|
32
|
-
$ create-app [options] <
|
|
32
|
+
$ create-app [options] <package-name>
|
|
33
33
|
|
|
34
|
-
|
|
34
|
+
package-name 包名称
|
|
35
35
|
|
|
36
36
|
Options
|
|
37
|
-
-o
|
|
38
|
-
-p
|
|
37
|
+
--overwrite, -o 覆盖已存在的目录
|
|
38
|
+
--package, -p 为 workspace 项目创建包
|
|
39
|
+
--private 设为私有项目
|
|
39
40
|
--verbose 显示详细日志
|
|
40
|
-
-h
|
|
41
|
-
-v
|
|
41
|
+
--help, -h 显示帮助信息
|
|
42
|
+
--version, -v 显示版本信息
|
|
42
43
|
```
|
|
43
44
|
|
|
44
45
|
支持模板:
|
package/dist/index.mjs
CHANGED
|
@@ -1,13 +1,14 @@
|
|
|
1
|
-
import e from"meow";import t from"node:fs";import n from"node:fs/promises";import r from"node:os";import i from"node:path";import{fileURLToPath as a}from"node:url";import*as o from"@clack/prompts";import{copy as s,mkdir as c,mkdirp as l,readFile as u,readJson as d,rm as f,rmSync as p,writeFile as m,writeJson as h}from"@tomjs/node";import{merge as g,omit as ee}from"lodash-es";import{glob as _}from"tinyglobby";import v from"chalk";import{$ as y}from"execa";import b from"@tomjs/logger";const x=a(new URL(`..`,import.meta.url)),S=[{name:`node`,display:`Node`,color:v.green,children:[{name:`node-base`,display:`Base`,color:v.green},{name:`node-cli`,display:`CLI`,color:v.green},{name:`node-vite`,display:`Vite Plugin`,color:v.blue,hasStyle:!0}]},{name:`web`,display:`Web`,color:v.magenta,children:[{name:`web-vue`,display:`Vue`,color:v.green,hasStyle:!0,isPublic:!1},{name:`web-react`,display:`React`,color:v.blue,hasStyle:!0,isPublic:!1}]},{name:`vscode`,display:`VSCode`,color:v.blue,children:[{name:`vscode-base`,display:`Base`,color:v.yellow,isPublic:`public`},{name:`vscode-vue`,display:`Vue`,color:v.green,hasStyle:!0,isPublic:`public`},{name:`vscode-react`,display:`React`,color:v.blue,hasStyle:!0,isPublic:`public`}]},{name:`electron`,display:`Electron`,color:v.cyan,children:[{name:`electron-vue`,display:`Vue`,color:v.green,hasStyle:!0,isPublic:!1},{name:`electron-react`,display:`React`,color:v.blue,hasStyle:!0,isPublic:!1}]}],C=[{id:`github`,name:`Github`,url:`https://github.com`},{id:`gitcode`,name:`GitCode`,url:`https://gitcode.com`},{id:`gitee`,name:`码云`,url:`https://gitee.com`}],w=`publisher.name.displayName.type.version.private.packageManager.description.author.contributors.license.funding.homepage.repository.publishConfig.bugs.keywords.categories.sideEffects.imports.exports.main.module.unpkg.jsdelivr.types.typesVersions.bin.icon.files.engines.activationEvents.contributes.scripts.peerDependencies.peerDependenciesMeta.dependencies.optionalDependencies.devDependencies.pnpm.overrides.resolutions.husky.simple-git-hooks.lint-staged.eslintConfig`.split(`.`),T=[`dev`,`dev:`,`debug`,`debug:`,`start`,`build`,`build:`,`preview`,`release`,`release:`,`clean`,`clean:`,`test`,`lint`,`lint:`],E=new b({directory:`create-app/logs`}),te=process.platform===`win32`;let D={};function O(e){D=e}function k(){return D}async function A(e,t){let{trim:n,...r}=Object.assign({stdio:`pipe`,shell:!0},t);Array.isArray(e)&&(e=e.join(` `));let i=e=>{k().verbose?E.debug(e):E.write(e)};i(`$ ${e}`);try{let t=(await y(r)`${e}`).stdout;return i(t),n?t.trim().replace(/\n|\r/g,``):t.trim()}catch(e){let t=e.stderr||e.message;throw i(t),Error(t)}}function j(e){if(e===`C`)return`en-US`;let t=``;try{t=Intl.getCanonicalLocales(e)[0]}catch(t){console.log(`${t.toString()}, invalid language tag: "${e}"\n`)}switch(t){case`zh-TW`:case`zh-HK`:case`zh-MO`:case`zh-CN`:case`zh-SG`:t=`zh-CN`;break;default:t=e}return t}function M(){return j((process.env.LC_ALL||process.env.LC_MESSAGES||process.env.LANG||Intl.DateTimeFormat().resolvedOptions().locale||`en-US`).split(`.`)[0].replace(`_`,`-`))}async function N(e){return await t.promises.readFile(e,`utf-8`).then(e=>JSON.parse(e)||{})}async function ne(e){let n=M(),r=i.resolve(e,`${n}.json`),a=i.resolve(e,`en-US.json`);return t.existsSync(r)?await N(r):await N(a)}const P=await ne(i.join(x,`locales`));function F(...e){if(e.length===0||!P)return``;let[t,...n]=e,r=P[t]??``;if(n[0]===null||n[0]===void 0||n[0]===``)return r;let i=typeof n[0]==`object`?n[0]:n;return r.replace(/\{([^}]+)\}/g,(e,t)=>i[t]??e)}const re=i.join(a(import.meta.url),`../../`),I=i.join(re,`templates`),L=()=>o.cancel(F(`prompt.cancel`));async function R(e){O(e);let t=await z(e);t&&await B(t)}async function z(e){let n=async function(){let e=await o.select({message:F(`prompt.templateType.message`),options:S.map(e=>({label:e.color(e.display||e.name),value:e.name}))});if(o.isCancel(e))return L();let t=S.find(t=>t.name===e),r=await o.select({message:F(`prompt.template.message`),options:t.children.map(e=>({label:e.color(e.display||e.name),value:e.name})).concat({label:F(`prompt.template.back`),value:`back`})});return o.isCancel(r)?L():r===`back`?await n():{template:r,templateOptions:t.children.find(e=>e.name===r)}},a=await n();if(!a)return;let{template:s,templateOptions:c}=a,l=V(e.dir);if(!l||[`.`,`~`].includes(l)){let e=await o.text({message:F(`prompt.project.message`),defaultValue:s,placeholder:s,validate:e=>e.length===0||V(e).length>0?void 0:F(`prompt.project.invalid`)});if(o.isCancel(e))return L();l=i.resolve(l===`~`&&!te?r.homedir():process.cwd(),V(e))}if(t.existsSync(l)&&!H(l)){let t=e.overwrite;if(!t){let e=await o.select({message:F(`prompt.exist.message`,l),options:[{label:F(`prompt.exist.option.no`),value:`no`},{label:F(`prompt.exist.option.yes`),value:`yes`},{label:F(`prompt.exist.option.ignore`),value:`ignore`}]});if(o.isCancel(e))return L();t=e}switch(t){case`yes`:U(l);break;case`no`:L();return}}let u=i.basename(i.resolve(l)),d=await o.text({message:F(`prompt.package.message`),defaultValue:K(u),placeholder:K(u),validate:e=>e.length===0||W(e)?void 0:F(`prompt.package.invalid`)});if(o.isCancel(d))return L();u=d;let f;if(e.private!==!0&&c.isPublic!==`public`){let e=await o.confirm({message:F(`prompt.public.message`),active:F(`prompt.confirm.yes`),inactive:F(`prompt.confirm.no`),initialValue:c.isPublic??!0});if(o.isCancel(e))return L();f=e}else f=!e.private;let p;if(f){let e=await o.select({message:F(`prompt.gitUrl.message`),options:C.map(e=>({label:`${e.name} (${e.url})`,value:e.id}))});if(o.isCancel(e))return L();p=C.find(t=>t.id===e)?.url}let m;if(f){let e=J(u);e||await q(`user.email`)===`tom@tomgao.cc`&&(e=`tomjs`);let t=await o.text({message:F(`prompt.gitOrg.message`),defaultValue:e,placeholder:e,validate:t=>e&&!t||G(t)?void 0:F(`prompt.gitOrg.invalid`)});if(o.isCancel(t))return L();m=t}return{targetDir:l,packageName:u,template:s,templateOptions:c,isPublic:f,gitUrl:p,gitOrg:m}}async function B(e){E.debug(`projectOptions:`,ee(e,`templateOptions`));let{targetDir:t,packageName:a,template:o,isPublic:p,gitUrl:v,gitOrg:y,templateOptions:b}=e;await c(t);let x=o.includes(`vscode`),S={},C=i.join(t,`.temp`);if(await l(C),await Y(S,`base`,C),b.hasStyle&&await Y(S,`style`,C),p&&!x&&await Y(S,`public`,C),await s(C,t),await f(C),await s(i.join(I,o),t),g(S,await d(i.join(i.join(I,o),`package.json`))),S.name=a,p){delete S.private,S.name=a,S.license=`MIT`;let e=await q(`user.name`)||r.userInfo().username,n=await q(`user.email`);e&&n?S.author={name:e,email:n}:e&&(S.author=e);let o=`${v}/${y}/${i.basename(t)}`;x?(S.publisher=y,S.homepage=`${o}/blob/main/README.md`,S.bugs={url:`${o}/issues`}):delete S.publishConfig,S.repository={type:`git`,url:`git+${o}.git`}}else S.private=!0,delete S.publishConfig,S.devDependencies&&delete S.devDependencies.publint;if(await h(i.join(t,`package.json`),ae(S)),p){let{author:e}=S,r=``;e&&(typeof e==`string`?r=e:typeof e==`object`&&(e.name&&e.email?r=`${e.name}<${e.email}>`:e.name?r=e.name:e.email&&(r=e.email)));let o={pkgName:a,dateYear:new Date().getFullYear(),gitUserName:r},s=(await n.readdir(t)).filter(e=>[`README.md`,`README.zh_CN.md`,`LICENSE`].includes(e));for(let e of s){let n=await u(i.join(t,e));Object.keys(o).forEach(e=>{n=n.replace(RegExp(`{{${e}}}`,`g`),o[e])}),await m(i.join(t,e),n)}}let w=await _(`**/_*`,{ignore:[`**/node_modules/**`,`**/.*`],cwd:t});for(let e of w){let r=e.split(`/`);r[r.length-1]=r[r.length-1].substring(1),await n.rename(i.join(t,e),i.join(t,r.join(`/`)))}await A(`git init`,{cwd:t})}function V(e){return(e||``).trim().replace(/\/+$/g,``)}function H(e){return t.readdirSync(e).length===0}function U(e){t.existsSync(e)&&p(e)}function W(e){return/^(?:@[a-z\d\-*~][a-z\d\-*._~]*\/)?[a-z\d\-~][a-z\d\-._~]*$/.test(e)}function G(e){return/^[a-z][a-z\d\-._]*$/.test(e)}function K(e){return e.trim().toLowerCase().replace(/\s+/g,`-`).replace(/^[._]/,``).replace(/[^a-z\d\-~]+/g,`-`)}async function q(e){return A(`git config --get ${e}`,{trim:!0})}function J(e){if(e.startsWith(`@`))return e.split(`/`)[0].substring(1)}async function Y(e,t,r){let a=i.join(I,`config/${t}`),o=await n.readdir(a);for(let t of o)[`node_modules`,`.git`].includes(t)||(t===`package.json`?g(e,await d(i.join(a,`package.json`))):await s(i.join(a,t),i.join(r,t)))}function X(e,t){if(!e)return e;let n={};return Array.isArray(t)&&t.length>0?(t.filter(t=>t in e).forEach(t=>{n[t]=e[t]}),Object.keys(e).filter(e=>!t.includes(e)).sort().forEach(t=>{n[t]=e[t]})):Object.keys(e).sort().forEach(t=>{n[t]=e[t]}),n}function ie(e,t,n){let r={},i=Object.keys(e);{let e=i.filter(e=>e.includes(`:`));i=i.filter(e=>!e.includes(`:`)).sort().concat(e)}let a=s(t),o=s(n);return a.concat(i.filter(e=>!a.includes(e)&&!o.includes(e))).concat(o).forEach(t=>{r[t]=e[t]}),r;function s(e){let t=Array.isArray(e)?e:[];return t.length>0&&(t=t.reduce((e,t)=>{if(t.endsWith(`:`)){let n=i.filter(n=>!e.includes(n)&&n.startsWith(t));return e.concat(n)}return i.includes(t)?e.concat(t):e},[])),t}}function ae(e){return e=X(e,w),e.scripts&&=ie(e.scripts,T,[`prepare`,`prepublishOnly`]),[`dependencies`,`devDependencies`,`peerDependencies`].forEach(t=>{e[t]&&(e[t]=X(e[t]))}),e}const Z=e(`
|
|
1
|
+
import e from"meow";import t from"node:fs";import n from"node:fs/promises";import r from"node:os";import i from"node:path";import{fileURLToPath as a}from"node:url";import*as o from"@clack/prompts";import{copy as s,mkdir as c,mkdirp as l,readFile as u,readJson as d,readJsonSync as f,rm as p,rmSync as m,writeFile as h,writeJson as g}from"@tomjs/node";import{camelCase as _,merge as v,upperFirst as y}from"lodash-es";import{glob as b}from"tinyglobby";import x from"chalk";import{$ as S}from"execa";import C from"@tomjs/logger";const w=a(new URL(`..`,import.meta.url)),T=[{name:`node`,display:`Node`,color:x.green,children:[{name:`node-base`,display:`Base`,color:x.green},{name:`node-cli`,display:`CLI`,color:x.green},{name:`node-vite`,display:`Vite Plugin`,value:`vite-plugin`,color:x.blue,hasStyle:!0}]},{name:`web`,display:`Web`,color:x.magenta,children:[{name:`web-vue`,display:`Vue`,color:x.green,hasStyle:!0,isPublic:!1},{name:`web-react`,display:`React`,color:x.blue,hasStyle:!0,isPublic:!1}]},{name:`vscode`,display:`VSCode`,color:x.blue,children:[{name:`vscode-base`,display:`Base`,color:x.yellow,isPublic:`public`},{name:`vscode-vue`,display:`Vue`,color:x.green,hasStyle:!0,isPublic:`public`},{name:`vscode-react`,display:`React`,color:x.blue,hasStyle:!0,isPublic:`public`}]},{name:`electron`,display:`Electron`,color:x.cyan,children:[{name:`electron-vue`,display:`Vue`,color:x.green,hasStyle:!0,isPublic:!1},{name:`electron-react`,display:`React`,color:x.blue,hasStyle:!0,isPublic:!1}]}],E=[{id:`github`,name:`Github`,url:`https://github.com`},{id:`gitcode`,name:`GitCode`,url:`https://gitcode.com`},{id:`gitee`,name:`码云`,url:`https://gitee.com`}],ee=`publisher.name.displayName.type.version.private.packageManager.description.author.contributors.license.funding.homepage.repository.publishConfig.bugs.keywords.categories.sideEffects.imports.exports.main.module.unpkg.jsdelivr.types.typesVersions.bin.icon.files.engines.activationEvents.contributes.scripts.peerDependencies.peerDependenciesMeta.dependencies.optionalDependencies.devDependencies.pnpm.overrides.resolutions.husky.simple-git-hooks.lint-staged.eslintConfig`.split(`.`),te=[`dev`,`dev:`,`debug`,`debug:`,`start`,`build`,`build:`,`preview`,`release`,`release:`,`clean`,`clean:`,`test`,`lint`,`lint:`],D=new C({directory:`create-app/logs`}),ne=process.platform===`win32`;let O={};function re(e){O=e}function k(){return O}async function A(e,t){let{trim:n,...r}=Object.assign({stdio:`pipe`,shell:!0},t);Array.isArray(e)&&(e=e.join(` `));let i=e=>{k().verbose?D.debug(e):D.write(e)};i(`$ ${e}`);try{let t=(await S(r)`${e}`).stdout;return i(t),n?t.trim().replace(/\n|\r/g,``):t.trim()}catch(e){let t=e.stderr||e.message;throw i(t),Error(t)}}function ie(e){if(e===`C`)return`en-US`;let t=``;try{t=Intl.getCanonicalLocales(e)[0]}catch(t){console.log(`${t.toString()}, invalid language tag: "${e}"\n`)}switch(t){case`zh-TW`:case`zh-HK`:case`zh-MO`:case`zh-CN`:case`zh-SG`:t=`zh-CN`;break;default:t=e}return t}function j(){return ie((process.env.LC_ALL||process.env.LC_MESSAGES||process.env.LANG||Intl.DateTimeFormat().resolvedOptions().locale||`en-US`).split(`.`)[0].replace(`_`,`-`))}async function M(e){return await t.promises.readFile(e,`utf-8`).then(e=>JSON.parse(e)||{})}async function ae(e){let n=j(),r=i.resolve(e,`${n}.json`),a=i.resolve(e,`en-US.json`);return t.existsSync(r)?await M(r):await M(a)}const N=await ae(i.join(w,`locales`));function P(...e){if(e.length===0||!N)return``;let[t,...n]=e,r=N[t]??``;if(n[0]===null||n[0]===void 0||n[0]===``)return r;let i=typeof n[0]==`object`?n[0]:n;return r.replace(/\{([^}]+)\}/g,(e,t)=>i[t]??e)}const oe=i.join(a(import.meta.url),`../../`),F=i.join(oe,`templates`),I=()=>o.cancel(P(`prompt.cancel`));async function L(e){re(e);let t=await R(e);t&&(await z(t),e.package&&await le(t),t.template===`node-vite`&&(await ce(t.targetDir),await X(t.targetDir,t.packageName)),e.package||await A(`git init`,{cwd:t.targetDir}))}async function R(e){let n=async function(){let e=await o.select({message:P(`prompt.templateType.message`),options:T.map(e=>({label:e.color(e.display||e.name),value:e.name}))});if(o.isCancel(e))return I();let t=T.find(t=>t.name===e),r=await o.select({message:P(`prompt.template.message`),options:t.children.map(e=>({label:e.color(e.display||e.name),value:e.name})).concat({label:P(`prompt.template.back`),value:`back`})});return o.isCancel(r)?I():r===`back`?await n():{template:r,templateOptions:t.children.find(e=>e.name===r)}},a=await n();if(!a)return;let s=``;await G(`user.email`)===`tom@tomgao.cc`&&(s=`tomjs`);let{template:c,templateOptions:l}=a,u=c.startsWith(`node-`),d=e.packageName;d||(d=l.value||c,s&&u&&(d=`@${s}/${d}`));let f=await o.text({message:P(`prompt.package.message`),initialValue:d,defaultValue:d,placeholder:d,validate:e=>e.length===0||U(e)?void 0:P(`prompt.package.invalid`)});if(o.isCancel(f))return I();d=f;let p=B(d),m=await o.text({message:P(`prompt.project.message`),defaultValue:p,initialValue:p,placeholder:p,validate:e=>e.length===0||e.length>0?void 0:P(`prompt.project.invalid`)});if(o.isCancel(m))return I();if(p=i.resolve(p===`~`&&!ne?r.homedir():process.cwd(),m),t.existsSync(p)&&!V(p)){let t=e.overwrite;if(!t){let e=await o.select({message:P(`prompt.exist.message`,p),options:[{label:P(`prompt.exist.option.no`),value:`no`},{label:P(`prompt.exist.option.yes`),value:`yes`},{label:P(`prompt.exist.option.ignore`),value:`ignore`}]});if(o.isCancel(e))return I();t=e}switch(t){case`yes`:H(p);break;case`no`:I();return}}let h;if(e.private!==!0&&l.isPublic!==`public`){let e=await o.confirm({message:P(`prompt.public.message`),active:P(`prompt.confirm.yes`),inactive:P(`prompt.confirm.no`),initialValue:l.isPublic??!0});if(o.isCancel(e))return I();h=e}else h=!e.private;let g;if(h){let e=await o.select({message:P(`prompt.gitUrl.message`),options:E.map(e=>({label:`${e.name} (${e.url})`,value:e.id}))});if(o.isCancel(e))return I();g=E.find(t=>t.id===e)?.url}let _;if(h){s=K(d);let e=await o.text({message:P(`prompt.orgName.message`),defaultValue:s,placeholder:s,validate:e=>s&&!e||W(e)?void 0:P(`prompt.orgName.invalid`)});if(o.isCancel(e))return I();_=e}return delete l.color,{targetDir:p,orgName:_,packageName:d,template:c,templateOptions:l,isPublic:h,gitUrl:g}}async function z(e){D.debug(`projectOptions:`,e);let{targetDir:t,packageName:a,template:o,isPublic:f,gitUrl:m,orgName:_,templateOptions:y}=e;await c(t);let x=o.includes(`vscode`),S={},C=i.join(t,`.temp`);if(await l(C),await q(S,`base`,C),y.hasStyle&&await q(S,`style`,C),f&&!x&&await q(S,`public`,C),await s(C,t),await p(C),await s(i.join(F,o),t),v(S,await d(i.join(i.join(F,o),`package.json`))),S.name=a,f){delete S.private,S.name=a,S.license=`MIT`;let e=await G(`user.name`)||r.userInfo().username,n=await G(`user.email`);e&&n?S.author={name:e,email:n}:e&&(S.author=e);let o=`${m}/${_}/${i.basename(t)}`;x&&(S.publisher=_,S.homepage=`${o}/blob/main/README.md`,S.bugs={url:`${o}/issues`},delete S.publishConfig),S.repository={type:`git`,url:`${o}.git`,directory:k().package?`packages/${i.basename(t)}`:void 0}}else S.private=!0,delete S.publishConfig,S.devDependencies&&delete S.devDependencies.publint;if(await g(i.join(t,`package.json`),Y(S)),f){let{author:e}=S,r=``;e&&(typeof e==`string`?r=e:typeof e==`object`&&(e.name&&e.email?r=`${e.name}<${e.email}>`:e.name?r=e.name:e.email&&(r=e.email)));let o={pkgName:a,dateYear:new Date().getFullYear(),gitUserName:r},s=(await n.readdir(t)).filter(e=>[`README.md`,`README.zh_CN.md`,`LICENSE`].includes(e));for(let e of s){let n=await u(i.join(t,e));Object.keys(o).forEach(e=>{n=n.replace(RegExp(`{{${e}}}`,`g`),o[e])}),await h(i.join(t,e),n)}}let w=await b(`**/_*`,{ignore:[`**/node_modules/**`,`**/.*`],cwd:t});for(let e of w){let r=e.split(`/`);r[r.length-1]=r[r.length-1].substring(1),await n.rename(i.join(t,e),i.join(t,r.join(`/`)))}}function B(e){return(e||``).trim().replace(/^@[^/]+\//,``)}function V(e){return t.readdirSync(e).length===0}function H(e){t.existsSync(e)&&m(e)}function U(e){return/^(?:@[a-z\d\-*~][a-z\d\-*._~]*\/)?[a-z\d\-~][a-z\d\-._~]*$/.test(e)}function W(e){return/^[a-z][a-z\d\-._]*$/.test(e)}async function G(e){return A(`git config --get ${e}`,{trim:!0})}function K(e){if(e.startsWith(`@`))return e.split(`/`)[0].substring(1)}async function q(e,t,r){let a=i.join(F,`config/${t}`),o=await n.readdir(a);for(let t of o)[`node_modules`,`.git`].includes(t)||(t===`package.json`?v(e,await d(i.join(a,`package.json`))):await s(i.join(a,t),i.join(r,t)))}function J(e,t){if(!e)return e;let n={};return Array.isArray(t)&&t.length>0?(t.filter(t=>t in e).forEach(t=>{n[t]=e[t]}),Object.keys(e).filter(e=>!t.includes(e)).sort().forEach(t=>{n[t]=e[t]})):Object.keys(e).sort().forEach(t=>{n[t]=e[t]}),n}function se(e,t,n){let r={},i=Object.keys(e);{let e=i.filter(e=>e.includes(`:`));i=i.filter(e=>!e.includes(`:`)).sort().concat(e)}let a=s(t),o=s(n);return a.concat(i.filter(e=>!a.includes(e)&&!o.includes(e))).concat(o).forEach(t=>{r[t]=e[t]}),r;function s(e){let t=Array.isArray(e)?e:[];return t.length>0&&(t=t.reduce((e,t)=>{if(t.endsWith(`:`)){let n=i.filter(n=>!e.includes(n)&&n.startsWith(t));return e.concat(n)}return i.includes(t)?e.concat(t):e},[])),t}}function Y(e){return e=J(e,ee),e.scripts&&=se(e.scripts,te,[`prepare`,`prepublishOnly`]),[`dependencies`,`devDependencies`,`peerDependencies`].forEach(t=>{e[t]&&(e[t]=J(e[t]))}),e}async function ce(e){let t=f(i.join(F,`config/package.json`))?.dependencies||{},n=await b(`**/package.json`,{ignore:[`**/node_modules/**`,`**/.*`],cwd:e});for(let r of n){let n=i.join(e,r),a=await d(n);[`dependencies`,`devDependencies`].forEach(e=>{let n=a[e];n&&Object.keys(n).forEach(e=>{n[e]=t[e]||n[e]})}),g(n,Y(a))}}async function le(e){let n=e.templateOptions.hasStyle,r=[`.editorconfig`,`.gitignore`,`.gitattributes`,`commitlint.config.mjs`,`simple-git-hooks.mjs`,`pnpm-workspace.yaml`];n||r.push(`lint-staged.config.mjs`),[`vscode`,`electron`].find(t=>e.template.startsWith(t))||r.push(`.vscode`),r.forEach(n=>{let r=i.join(e.targetDir,n);t.existsSync(r)&&m(r)});let a=await d(i.join(e.targetDir,`package.json`));if(a.scripts&&delete a.scripts.prepare,!n)delete a.devDependencies;else if(a.devDependencies){let e=await d(i.join(F,`config/style/package.json`)),t=Object.keys(e.devDependencies||{});Object.keys(a.devDependencies).forEach(e=>{t.includes(e)||delete a.devDependencies[e]})}await g(i.join(e.targetDir,`package.json`),a)}async function X(e,t){let n=await b([`examples/*/package.json`,`examples/*/vite.config.ts`],{ignore:[`**/node_modules/**`,`**/.*`],cwd:e});for(let r of n){let n=i.join(e,r);await h(n,(await u(n)).replaceAll(`@tomjs/vite-plugin-template`,t))}let r=_((t.split(`/`).pop()||``).replace(`vite-plugin-`,``));if(r){let t=await b([`src/**/*.ts`],{ignore:[`**/node_modules/**`,`**/.*`],cwd:e});for(let n of t){let t=i.join(e,n);await h(t,(await u(t)).replaceAll(`@tomjs:xxx`,`@tomjs/${r}`).replaceAll(`useXxxPlugin`,`use${y(r)}Plugin`).replaceAll(`XxxPluginOptions`,`${y(r)}PluginOptions`))}}}const Z=e(`
|
|
2
2
|
Usage
|
|
3
|
-
$ create-app [options] <
|
|
3
|
+
$ create-app [options] <package-name>
|
|
4
4
|
|
|
5
|
-
|
|
5
|
+
package-name ${P(`options.packageName`)}
|
|
6
6
|
|
|
7
7
|
Options
|
|
8
|
-
-o
|
|
9
|
-
-p
|
|
10
|
-
--
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
8
|
+
--overwrite, -o ${P(`options.overwrite`)}
|
|
9
|
+
--package, -p ${P(`options.package`)}
|
|
10
|
+
--private ${P(`options.private`)}
|
|
11
|
+
--verbose ${P(`options.verbose`)}
|
|
12
|
+
--help, -h ${P(`options.help`)}
|
|
13
|
+
--version, -v ${P(`options.version`)}
|
|
14
|
+
`,{importMeta:import.meta,booleanDefault:void 0,helpIndent:0,description:P(`app.description`),flags:{overwrite:{type:`string`},private:{type:`boolean`},package:{type:`boolean`,shortFlag:`p`},verbose:{type:`boolean`,default:!1},help:{type:`boolean`,shortFlag:`h`},version:{type:`boolean`,shortFlag:`v`}}}),{input:Q,flags:$}=Z;$.help?Z.showHelp(0):$.version?Z.showVersion():(D.enableDebug($.verbose),D.debug(`package name:`,Q.join()),D.debug(`cli options:`,$),L(Object.assign({packageName:Q[0]},$)));export{};
|
package/locales/en-US.json
CHANGED
|
@@ -1,9 +1,10 @@
|
|
|
1
1
|
{
|
|
2
2
|
"app.description": "Create a node/web/electron/vscode project based on tomjs",
|
|
3
|
-
"options.dir": "Project directory",
|
|
4
3
|
"options.help": "Show help information",
|
|
5
4
|
"options.interactive": "Force interactive",
|
|
6
5
|
"options.overwrite": "Overwrite existing project",
|
|
6
|
+
"options.package": "Create a package for the workspace project",
|
|
7
|
+
"options.packageName": "Package name",
|
|
7
8
|
"options.private": "Set as private project",
|
|
8
9
|
"options.verbose": "Show verbose logs",
|
|
9
10
|
"options.version": "Show version information",
|
|
@@ -14,9 +15,9 @@
|
|
|
14
15
|
"prompt.exist.option.ignore": "Ignore files and continue",
|
|
15
16
|
"prompt.exist.option.no": "Cancel operation",
|
|
16
17
|
"prompt.exist.option.yes": "Delete existing files and continue",
|
|
17
|
-
"prompt.gitOrg.invalid": "Invalid organization",
|
|
18
|
-
"prompt.gitOrg.message": "Please enter git repository organization:",
|
|
19
18
|
"prompt.gitUrl.message": "Please select git repository URL:",
|
|
19
|
+
"prompt.orgName.invalid": "Invalid organization",
|
|
20
|
+
"prompt.orgName.message": "Please enter git repository organization:",
|
|
20
21
|
"prompt.package.invalid": "Invalid package.json name",
|
|
21
22
|
"prompt.package.message": "Please enter package name:",
|
|
22
23
|
"prompt.project.invalid": "Invalid project name",
|
package/locales/zh-CN.json
CHANGED
|
@@ -1,9 +1,10 @@
|
|
|
1
1
|
{
|
|
2
2
|
"app.description": "创建基于 tomjs 的 node/web/electron/vscode 项目",
|
|
3
|
-
"options.dir": "项目目录",
|
|
4
3
|
"options.help": "显示帮助信息",
|
|
5
4
|
"options.interactive": "强制互动",
|
|
6
5
|
"options.overwrite": "覆盖已存在的目录",
|
|
6
|
+
"options.package": "为 workspace 项目创建包",
|
|
7
|
+
"options.packageName": "包名称",
|
|
7
8
|
"options.private": "设为私有项目",
|
|
8
9
|
"options.verbose": "显示详细日志",
|
|
9
10
|
"options.version": "显示版本信息",
|
|
@@ -14,9 +15,9 @@
|
|
|
14
15
|
"prompt.exist.option.ignore": "忽略文件并继续",
|
|
15
16
|
"prompt.exist.option.no": "取消操作",
|
|
16
17
|
"prompt.exist.option.yes": "删除现有文件并继续",
|
|
17
|
-
"prompt.gitOrg.invalid": "无效的组织",
|
|
18
|
-
"prompt.gitOrg.message": "请输入git仓库组织:",
|
|
19
18
|
"prompt.gitUrl.message": "请选择git仓库地址:",
|
|
19
|
+
"prompt.orgName.invalid": "无效的组织",
|
|
20
|
+
"prompt.orgName.message": "请输入git仓库组织:",
|
|
20
21
|
"prompt.package.invalid": "无效的 package.json name",
|
|
21
22
|
"prompt.package.message": "请输入包名称:",
|
|
22
23
|
"prompt.project.invalid": "无效的项目名称",
|
package/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@tomjs/create-app",
|
|
3
3
|
"type": "module",
|
|
4
|
-
"version": "
|
|
4
|
+
"version": "4.0.0",
|
|
5
5
|
"description": "Create a node/web/electron/vscode project based on tomjs",
|
|
6
6
|
"author": {
|
|
7
7
|
"name": "Tom Gao",
|
|
@@ -38,8 +38,8 @@
|
|
|
38
38
|
"create-app": "dist/index.mjs"
|
|
39
39
|
},
|
|
40
40
|
"files": [
|
|
41
|
-
"locales",
|
|
42
41
|
"dist",
|
|
42
|
+
"locales",
|
|
43
43
|
"templates",
|
|
44
44
|
"!templates/*/node_modules/.bin/*"
|
|
45
45
|
],
|
|
@@ -58,12 +58,12 @@
|
|
|
58
58
|
"tinyglobby": "^0.2.15"
|
|
59
59
|
},
|
|
60
60
|
"devDependencies": {
|
|
61
|
-
"@antfu/eslint-config": "^6.7.
|
|
61
|
+
"@antfu/eslint-config": "^6.7.3",
|
|
62
62
|
"@commitlint/cli": "^20.2.0",
|
|
63
63
|
"@tomjs/commitlint": "^5.0.0",
|
|
64
|
-
"@tomjs/eslint": "^6.
|
|
64
|
+
"@tomjs/eslint": "^6.3.0",
|
|
65
65
|
"@tomjs/stylelint": "^7.0.0",
|
|
66
|
-
"@tomjs/tsconfig": "^2.
|
|
66
|
+
"@tomjs/tsconfig": "^2.2.0",
|
|
67
67
|
"@tsconfig/node20": "^20.1.8",
|
|
68
68
|
"@types/lodash-es": "^4.17.12",
|
|
69
69
|
"@types/node": "^20.19.27",
|
|
@@ -75,15 +75,16 @@
|
|
|
75
75
|
"rimraf": "^6.1.2",
|
|
76
76
|
"simple-git-hooks": "^2.13.1",
|
|
77
77
|
"stylelint": "^16.26.1",
|
|
78
|
-
"tsdown": "~0.18.
|
|
79
|
-
"tsx": "^4.
|
|
80
|
-
"type-fest": "^5.
|
|
78
|
+
"tsdown": "~0.18.4",
|
|
79
|
+
"tsx": "^4.21.0",
|
|
80
|
+
"type-fest": "^5.3.1",
|
|
81
81
|
"typescript": "~5.9.3"
|
|
82
82
|
},
|
|
83
83
|
"scripts": {
|
|
84
84
|
"dev": "tsdown --watch",
|
|
85
85
|
"debug": "tsx -r dotenv/config src/index.ts",
|
|
86
86
|
"build": "tsdown --minify",
|
|
87
|
+
"file": "tsx ./scripts/file.ts",
|
|
87
88
|
"lint": "run-s lint:stylelint lint:eslint",
|
|
88
89
|
"lint:eslint": "eslint --fix",
|
|
89
90
|
"lint:stylelint": "stylelint \"templates/**/*.{css,scss,vue,html}\" --fix --cache"
|
|
@@ -2,17 +2,17 @@
|
|
|
2
2
|
|
|
3
3
|
"type": "module",
|
|
4
4
|
"version": "0.0.0",
|
|
5
|
-
"packageManager": "pnpm@10.
|
|
5
|
+
"packageManager": "pnpm@10.26.2",
|
|
6
6
|
"scripts": {
|
|
7
7
|
"lint": "eslint --fix",
|
|
8
8
|
"prepare": "simple-git-hooks"
|
|
9
9
|
},
|
|
10
10
|
"devDependencies": {
|
|
11
|
-
"@antfu/eslint-config": "^6.7.
|
|
11
|
+
"@antfu/eslint-config": "^6.7.3",
|
|
12
12
|
"@commitlint/cli": "^20.2.0",
|
|
13
13
|
"@tomjs/commitlint": "^5.0.0",
|
|
14
|
-
"@tomjs/eslint": "^6.
|
|
15
|
-
"@tomjs/tsconfig": "^2.
|
|
14
|
+
"@tomjs/eslint": "^6.1.0",
|
|
15
|
+
"@tomjs/tsconfig": "^2.2.0",
|
|
16
16
|
"@types/node": "^20.19.27",
|
|
17
17
|
"eslint": "^9.39.2",
|
|
18
18
|
"lint-staged": "^16.2.7",
|
|
@@ -0,0 +1,57 @@
|
|
|
1
|
+
{
|
|
2
|
+
"dependencies": {
|
|
3
|
+
"@antfu/eslint-config": "^6.7.3",
|
|
4
|
+
"@clack/prompts": "^0.11.0",
|
|
5
|
+
"@commitlint/cli": "^20.2.0",
|
|
6
|
+
"@eslint-react/eslint-plugin": "^2.5.0",
|
|
7
|
+
"@tomjs/commitlint": "^5.0.0",
|
|
8
|
+
"@tomjs/eslint": "^6.1.0",
|
|
9
|
+
"@tomjs/logger": "^1.4.0",
|
|
10
|
+
"@tomjs/node": "^2.2.3",
|
|
11
|
+
"@tomjs/stylelint": "^7.0.0",
|
|
12
|
+
"@tomjs/tsconfig": "^2.2.0",
|
|
13
|
+
"@tomjs/vite-plugin-electron": "^2.1.0",
|
|
14
|
+
"@tomjs/vite-plugin-vscode": "^6.0.1",
|
|
15
|
+
"@tomjs/vscode": "^2.5.0",
|
|
16
|
+
"@tomjs/vscode-dev": "^3.0.2",
|
|
17
|
+
"@tomjs/vscode-webview": "^2.0.2",
|
|
18
|
+
"@types/node": "^20.19.27",
|
|
19
|
+
"@types/react": "^19.2.7",
|
|
20
|
+
"@types/react-dom": "^19.2.3",
|
|
21
|
+
"@types/vscode": "^1.56.0",
|
|
22
|
+
"@types/vscode-webview": "^1.57.5",
|
|
23
|
+
"@vitejs/plugin-react": "^5.1.2",
|
|
24
|
+
"@vitejs/plugin-react-swc": "^4.2.2",
|
|
25
|
+
"@vitejs/plugin-vue": "^6.0.3",
|
|
26
|
+
"@vscode/webview-ui-toolkit": "^1.4.0",
|
|
27
|
+
"@vue/tsconfig": "^0.8.1",
|
|
28
|
+
"chalk": "^5.6.2",
|
|
29
|
+
"cosmiconfig": "^9.0.0",
|
|
30
|
+
"cross-env": "^10.1.0",
|
|
31
|
+
"electron": "^39.2.7",
|
|
32
|
+
"electron-builder": "^26.0.12",
|
|
33
|
+
"eslint": "^9.39.2",
|
|
34
|
+
"eslint-plugin-react-hooks": "^7.0.1",
|
|
35
|
+
"eslint-plugin-react-refresh": "^0.4.26",
|
|
36
|
+
"execa": "^9.6.1",
|
|
37
|
+
"lint-staged": "^16.2.7",
|
|
38
|
+
"meow": "^14.0.0",
|
|
39
|
+
"npm-run-all2": "^8.0.4",
|
|
40
|
+
"ora": "^9.0.0",
|
|
41
|
+
"publint": "0.3.16",
|
|
42
|
+
"react": "^19.2.3",
|
|
43
|
+
"react-dom": "^19.2.3",
|
|
44
|
+
"sass": "^1.97.1",
|
|
45
|
+
"simple-git-hooks": "^2.13.1",
|
|
46
|
+
"stylelint": "^16.26.1",
|
|
47
|
+
"tsdown": "^0.18.4",
|
|
48
|
+
"tsx": "^4.21.0",
|
|
49
|
+
"typescript": "~5.9.3",
|
|
50
|
+
"vite": "^7.3.0",
|
|
51
|
+
"vite-plugin-electron-renderer": "^0.14.6",
|
|
52
|
+
"vite-plugin-vue-devtools": "^8.0.5",
|
|
53
|
+
"vue": "^3.5.26",
|
|
54
|
+
"vue-i18n": "^11.2.8",
|
|
55
|
+
"vue-tsc": "^3.2.1"
|
|
56
|
+
}
|
|
57
|
+
}
|
|
@@ -19,9 +19,9 @@
|
|
|
19
19
|
"react-dom": "^19.2.3"
|
|
20
20
|
},
|
|
21
21
|
"devDependencies": {
|
|
22
|
-
"@eslint-react/eslint-plugin": "^2.
|
|
22
|
+
"@eslint-react/eslint-plugin": "^2.4.0",
|
|
23
23
|
"@tomjs/vite-plugin-electron": "^2.1.0",
|
|
24
|
-
"@types/node": "^
|
|
24
|
+
"@types/node": "^20.19.27",
|
|
25
25
|
"@types/react": "^19.2.7",
|
|
26
26
|
"@types/react-dom": "^19.2.3",
|
|
27
27
|
"@vitejs/plugin-react-swc": "^4.2.2",
|
|
@@ -20,7 +20,7 @@
|
|
|
20
20
|
},
|
|
21
21
|
"devDependencies": {
|
|
22
22
|
"@tomjs/vite-plugin-electron": "^2.1.0",
|
|
23
|
-
"@types/node": "^
|
|
23
|
+
"@types/node": "^20.19.27",
|
|
24
24
|
"@vitejs/plugin-vue": "^6.0.3",
|
|
25
25
|
"@vue/tsconfig": "^0.8.1",
|
|
26
26
|
"cross-env": "^10.1.0",
|
|
@@ -30,6 +30,6 @@
|
|
|
30
30
|
"vite": "^7.3.0",
|
|
31
31
|
"vite-plugin-electron-renderer": "^0.14.6",
|
|
32
32
|
"vite-plugin-vue-devtools": "^8.0.5",
|
|
33
|
-
"vue-tsc": "^3.2.
|
|
33
|
+
"vue-tsc": "^3.2.1"
|
|
34
34
|
}
|
|
35
35
|
}
|
|
@@ -1,4 +1,5 @@
|
|
|
1
1
|
import path from 'node:path';
|
|
2
|
+
import usePlugin from '@tomjs/vite-plugin-template';
|
|
2
3
|
import react from '@vitejs/plugin-react-swc';
|
|
3
4
|
import { defineConfig } from 'vite';
|
|
4
5
|
|
|
@@ -10,5 +11,5 @@ export default defineConfig({
|
|
|
10
11
|
react: path.resolve(process.cwd(), 'node_modules/react'),
|
|
11
12
|
},
|
|
12
13
|
},
|
|
13
|
-
plugins: [react()],
|
|
14
|
+
plugins: [react(), usePlugin()],
|
|
14
15
|
});
|
|
@@ -1,5 +1,7 @@
|
|
|
1
1
|
{
|
|
2
|
-
"name": "
|
|
2
|
+
"name": "@tomjs/vite-plugin-template",
|
|
3
|
+
"main": "./dist/index.js",
|
|
4
|
+
"types": "./dist/index.d.ts",
|
|
3
5
|
"scripts": {
|
|
4
6
|
"dev": "tsdown --watch",
|
|
5
7
|
"build": "tsdown",
|
|
@@ -10,7 +12,7 @@
|
|
|
10
12
|
},
|
|
11
13
|
"devDependencies": {
|
|
12
14
|
"publint": "0.3.16",
|
|
13
|
-
"tsdown": "^0.18.
|
|
15
|
+
"tsdown": "^0.18.3",
|
|
14
16
|
"vite": "^7.3.0"
|
|
15
17
|
}
|
|
16
18
|
}
|
|
@@ -1,9 +1,10 @@
|
|
|
1
1
|
import type { Disposable, ExtensionContext, Webview } from 'vscode';
|
|
2
|
+
import { getWebviewHtml } from 'virtual:vscode';
|
|
2
3
|
import { window } from 'vscode';
|
|
3
4
|
|
|
4
5
|
export class WebviewHelper {
|
|
5
6
|
public static setupHtml(webview: Webview, context: ExtensionContext) {
|
|
6
|
-
return
|
|
7
|
+
return getWebviewHtml({
|
|
7
8
|
serverUrl: process.env.VITE_DEV_SERVER_URL,
|
|
8
9
|
webview,
|
|
9
10
|
context,
|
|
@@ -1,9 +1,10 @@
|
|
|
1
1
|
import type { Disposable, ExtensionContext, Webview } from 'vscode';
|
|
2
|
+
import { getWebviewHtml } from 'virtual:vscode';
|
|
2
3
|
import { window } from 'vscode';
|
|
3
4
|
|
|
4
5
|
export class WebviewHelper {
|
|
5
6
|
public static setupHtml(webview: Webview, context: ExtensionContext) {
|
|
6
|
-
return
|
|
7
|
+
return getWebviewHtml({
|
|
7
8
|
serverUrl: process.env.VITE_DEV_SERVER_URL,
|
|
8
9
|
webview,
|
|
9
10
|
context,
|
|
@@ -52,13 +52,13 @@
|
|
|
52
52
|
"vue-i18n": "^11.2.7"
|
|
53
53
|
},
|
|
54
54
|
"devDependencies": {
|
|
55
|
-
"@tomjs/vite-plugin-vscode": "^
|
|
55
|
+
"@tomjs/vite-plugin-vscode": "^6.0.1",
|
|
56
56
|
"@tomjs/vscode-dev": "^3.0.2",
|
|
57
57
|
"@types/vscode": "^1.56.0",
|
|
58
58
|
"@types/vscode-webview": "^1.57.5",
|
|
59
59
|
"@vitejs/plugin-vue": "^6.0.3",
|
|
60
60
|
"vite": "^7.3.0",
|
|
61
61
|
"vite-plugin-vue-devtools": "^8.0.5",
|
|
62
|
-
"vue-tsc": "^3.2.
|
|
62
|
+
"vue-tsc": "^3.2.1"
|
|
63
63
|
}
|
|
64
64
|
}
|
|
@@ -11,10 +11,10 @@
|
|
|
11
11
|
},
|
|
12
12
|
"dependencies": {
|
|
13
13
|
"react": "^19.2.3",
|
|
14
|
-
"react-dom": "^19.2.
|
|
14
|
+
"react-dom": "^19.2.3"
|
|
15
15
|
},
|
|
16
16
|
"devDependencies": {
|
|
17
|
-
"@eslint-react/eslint-plugin": "^2.
|
|
17
|
+
"@eslint-react/eslint-plugin": "^2.4.0",
|
|
18
18
|
"@types/node": "^22.19.3",
|
|
19
19
|
"@types/react": "^19.2.7",
|
|
20
20
|
"@types/react-dom": "^19.2.3",
|