reborn-ui 0.1.64 → 0.1.65

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/dist/index.js CHANGED
@@ -715,7 +715,7 @@ function initCommand() {
715
715
  // package.json
716
716
  var package_default = {
717
717
  name: "reborn-ui",
718
- version: "0.1.64",
718
+ version: "0.1.65",
719
719
  description: "A CLI for Reborn UI",
720
720
  author: "1997liuyh-boop",
721
721
  license: "MIT",
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts","../src/commands/add.ts","../src/utils/fs.ts","../src/utils/pm.ts","../src/utils/registry.ts","../src/utils/ui.ts","../src/commands/build.ts","../src/utils/imports.ts","../src/utils/transformers.ts","../src/commands/init.ts","../src/utils/templates.ts","../package.json"],"sourcesContent":["import { Command } from \"commander\";\r\nimport { addCommand } from \"./commands/add.js\";\r\nimport { buildCommand } from \"./commands/build.js\";\r\nimport { initCommand } from \"./commands/init.js\";\r\nimport packageJson from \"../package.json\" assert { type: \"json\" };\r\n\r\nconst program = new Command()\r\n .name(\"reborn-ui\")\r\n .description(\"Reborn UI - 组件安装与 registry 辅助 CLI\")\r\n .version(packageJson.version);\r\n\r\nprogram.addCommand(initCommand());\r\nprogram.addCommand(addCommand());\r\nprogram.addCommand(buildCommand());\r\n\r\nawait program.parseAsync(process.argv);\r\n\r\n\r\n","import path from \"node:path\";\r\nimport { Command } from \"commander\";\r\nimport prompts from \"prompts\";\r\nimport type { PackageManager, RegistryComponent } from \"../types.js\";\r\nimport { ensureDir, pathExists, writeTextFile } from \"../utils/fs.js\";\r\nimport {\r\n detectPackageManager,\r\n readPackageJson,\r\n} from \"../utils/pm.js\";\r\nimport { defaultConfig, loadConfigCompat, loadRegistry } from \"../utils/registry.js\";\r\nimport cliProgress from \"cli-progress\";\r\nimport chalk from \"chalk\";\r\nimport { successLog } from \"../utils/ui.js\";\r\n\r\nfunction rewriteImports(params: {\r\n content: string;\r\n aliasSymbol: string;\r\n}) {\r\n const { content, aliasSymbol } = params;\r\n if (!aliasSymbol || aliasSymbol === \"@\") return content;\r\n\r\n // 仅替换项目根目录映射符号:把 \"@/xxx\" -> \"<aliasSymbol>/xxx\"\r\n // 注意:不要影响 scoped package(@scope/name),这里只替换 \"@/\" 前缀。\r\n return content.replaceAll(\"@/\", `${aliasSymbol}/`);\r\n}\r\n\r\nasync function writeComponentFiles(params: {\r\n cwd: string;\r\n componentsDir: string;\r\n aliasSymbol: string;\r\n component: RegistryComponent;\r\n overwrite?: boolean;\r\n onProgress?: () => void;\r\n platform: \"web\" | \"uniapp\";\r\n}) {\r\n const { cwd, componentsDir, aliasSymbol, component, overwrite, onProgress, platform } = params;\r\n const baseDir = path.join(cwd, componentsDir, component.name);\r\n await ensureDir(baseDir);\r\n\r\n for (const f of component.files) {\r\n // 过滤逻辑:\r\n // 1. 如果 f.target 存在且不等于 platform,跳过\r\n // 2. 如果 f.target 不存在,视为通用文件,不跳过\r\n if (f.target && f.target !== platform) {\r\n continue;\r\n }\r\n\r\n const target = path.join(baseDir, ...f.path.split(\"/\"));\r\n if (!overwrite && (await pathExists(target))) {\r\n onProgress?.();\r\n continue;\r\n }\r\n const nextContent = rewriteImports({ content: f.content, aliasSymbol });\r\n await writeTextFile(target, nextContent);\r\n onProgress?.();\r\n // 增加一个极小的延迟,让进度条看起来在“跑”\r\n await new Promise(r => setTimeout(r, 10));\r\n }\r\n\r\n return baseDir;\r\n}\r\n\r\nexport function addCommand() {\r\n const cmd = new Command(\"add\")\r\n .description(\"向项目中添加组件与相关依赖\")\r\n .argument(\"[components...]\", \"组件名(可多个)\")\r\n .option(\"--cwd <path>\", \"目标项目目录\", process.cwd())\r\n .option(\"--pm <pm>\", \"包管理器:pnpm|npm|yarn|bun\")\r\n .option(\"--yes\", \"跳过交互\", false)\r\n .option(\"--overwrite\", \"覆盖已存在文件\", false)\r\n .option(\"--config <path>\", \"配置文件路径(相对 cwd)\", \"components.json\")\r\n .option(\"--registry <pkgOrPath>\", \"覆盖配置里的 registry\")\r\n .option(\"--components-dir <path>\", \"覆盖配置里的 componentsDir\")\r\n .option(\"--lib-dir <path>\", \"覆盖配置里的 libDir\")\r\n .option(\"--alias-symbol <symbol>\", \"覆盖配置里的 aliasSymbol(默认 @)\")\r\n .action(async (components: string[], opts) => {\r\n const cwd = path.resolve(opts.cwd);\r\n const pm: PackageManager =\r\n opts.pm ?? (await detectPackageManager(cwd));\r\n\r\n const cfg = (await loadConfigCompat(cwd, opts.config)) ?? defaultConfig();\r\n if (opts.registry) cfg.registry = opts.registry;\r\n if (opts.componentsDir) cfg.componentsDir = opts.componentsDir;\r\n if (opts.libDir) cfg.libDir = opts.libDir;\r\n if (opts.aliasSymbol) cfg.aliasSymbol = opts.aliasSymbol;\r\n\r\n const registry = await loadRegistry({ cwd, registry: cfg.registry });\r\n\r\n let targets = components ?? [];\r\n if (!targets.length) {\r\n if (opts.yes) {\r\n throw new Error(\"未指定组件名;请传入组件参数或去掉 --yes 以交互选择。\");\r\n }\r\n const choices = registry.components.map((c) => ({\r\n title: c.name,\r\n value: c.name,\r\n }));\r\n const res = await prompts(\r\n [\r\n {\r\n type: \"multiselect\",\r\n name: \"selected\",\r\n message: \"选择要添加的组件\",\r\n choices,\r\n min: 1,\r\n },\r\n ],\r\n {\r\n onCancel: () => {\r\n throw new Error(\"已取消\");\r\n },\r\n },\r\n );\r\n targets = res.selected ?? [];\r\n }\r\n\r\n // 询问平台 (Web / UniApp)\r\n const { platform } = await prompts({\r\n type: \"select\",\r\n name: \"platform\",\r\n message: \"选择目标平台\",\r\n choices: [\r\n { title: \"Web (默认)\", value: \"web\" },\r\n { title: \"UniApp\", value: \"uniapp\" },\r\n ],\r\n initial: 0,\r\n });\r\n\r\n if (!platform) {\r\n throw new Error(\"已取消\");\r\n }\r\n\r\n const totalFiles = targets.reduce((acc, name) => {\r\n const c = registry.components.find((x) => x.name === name);\r\n return acc + (c?.files.length ?? 0);\r\n }, 0);\r\n\r\n const bar = new cliProgress.SingleBar({\r\n format: '正在写入文件 |' + chalk.cyan('{bar}') + '| {percentage}% || {value}/{total} 文件',\r\n barCompleteChar: '\\u2588',\r\n barIncompleteChar: '\\u2591',\r\n hideCursor: true\r\n });\r\n\r\n bar.start(totalFiles, 0);\r\n\r\n for (const name of targets) {\r\n const c = registry.components.find((x) => x.name === name);\r\n if (!c) {\r\n bar.stop();\r\n throw new Error(`registry 中不存在该组件:${name}`);\r\n }\r\n\r\n await writeComponentFiles({\r\n cwd,\r\n componentsDir: cfg.componentsDir,\r\n aliasSymbol: cfg.aliasSymbol ?? \"@\",\r\n component: c,\r\n overwrite: opts.overwrite,\r\n onProgress: () => bar.increment(),\r\n platform,\r\n });\r\n }\r\n bar.stop();\r\n\r\n console.log(\"\");\r\n for (const name of targets) {\r\n successLog(`组件 ${chalk.bold(name)} 已成功添加到项目`);\r\n }\r\n\r\n // eslint-disable-next-line no-console\r\n console.log(`\\n${chalk.bold.green('DONE')} 已完成 ${targets.length} 个组件的添加(pm=${pm})`);\r\n });\r\n\r\n return cmd;\r\n}\r\n\r\n\r\n","import { createHash } from \"node:crypto\";\nimport fs from \"node:fs/promises\";\nimport path from \"node:path\";\n\nexport async function pathExists(p: string) {\n try {\n await fs.access(p);\n return true;\n } catch {\n return false;\n }\n}\n\nexport async function ensureDir(dirPath: string) {\n await fs.mkdir(dirPath, { recursive: true });\n}\n\nexport async function readJsonFile<T>(filePath: string): Promise<T> {\n const raw = await fs.readFile(filePath, \"utf8\");\n return JSON.parse(raw) as T;\n}\n\nexport async function writeJsonFile(filePath: string, data: unknown) {\n await ensureDir(path.dirname(filePath));\n await fs.writeFile(filePath, JSON.stringify(data, null, 2) + \"\\n\", \"utf8\");\n}\n\nexport async function writeTextFile(filePath: string, content: string) {\n await ensureDir(path.dirname(filePath));\n await fs.writeFile(filePath, content, \"utf8\");\n}\n\nexport async function listFilesRecursive(\n dirPath: string,\n opts?: { ignoreDirNames?: Set<string> },\n) {\n const ignoreDirNames =\n opts?.ignoreDirNames ??\n new Set([\"node_modules\", \".git\", \".nuxt\", \"dist\", \".output\"]);\n\n const out: string[] = [];\n\n async function walk(current: string) {\n const entries = await fs.readdir(current, { withFileTypes: true });\n for (const entry of entries) {\n const p = path.join(current, entry.name);\n if (entry.isDirectory()) {\n if (ignoreDirNames.has(entry.name)) continue;\n await walk(p);\n } else if (entry.isFile()) {\n out.push(p);\n }\n }\n }\n\n await walk(dirPath);\n return out;\n}\n\nexport async function copyDirRecursive(params: {\n fromDir: string;\n toDir: string;\n overwrite?: boolean;\n ignoreFileNames?: Set<string>;\n}) {\n const { fromDir, toDir, overwrite } = params;\n const ignoreFileNames = params.ignoreFileNames ?? new Set([\".DS_Store\"]);\n\n const entries = await fs.readdir(fromDir, { withFileTypes: true });\n await ensureDir(toDir);\n\n for (const entry of entries) {\n if (ignoreFileNames.has(entry.name)) continue;\n const from = path.join(fromDir, entry.name);\n const to = path.join(toDir, entry.name);\n\n if (entry.isDirectory()) {\n await copyDirRecursive({ fromDir: from, toDir: to, overwrite, ignoreFileNames });\n continue;\n }\n\n if (!entry.isFile()) continue;\n\n if (!overwrite) {\n try {\n await fs.access(to);\n continue; // 已存在则跳过\n } catch {\n // not exists\n }\n }\n\n await ensureDir(path.dirname(to));\n await fs.copyFile(from, to);\n }\n}\n\nexport function sha1(text: string) {\n return createHash(\"sha1\").update(text).digest(\"hex\");\n}\n\n\n","import fs from \"node:fs/promises\";\r\nimport path from \"node:path\";\r\nimport { execa } from \"execa\";\r\nimport type { PackageManager } from \"../types.js\";\r\nimport { pathExists, readJsonFile, writeJsonFile } from \"./fs.js\";\r\n\r\nexport async function detectPackageManager(cwd: string): Promise<PackageManager> {\r\n if (await pathExists(path.join(cwd, \"pnpm-lock.yaml\"))) return \"pnpm\";\r\n if (await pathExists(path.join(cwd, \"yarn.lock\"))) return \"yarn\";\r\n if (await pathExists(path.join(cwd, \"package-lock.json\"))) return \"npm\";\r\n if (await pathExists(path.join(cwd, \"bun.lockb\"))) return \"bun\";\r\n return \"pnpm\";\r\n}\r\n\r\nexport async function readPackageJson(\r\n cwd: string,\r\n): Promise<Record<string, any> & { dependencies?: Record<string, string>; devDependencies?: Record<string, string> }> {\r\n return await readJsonFile(path.join(cwd, \"package.json\"));\r\n}\r\n\r\nexport async function writePackageJson(cwd: string, pkg: unknown) {\r\n await writeJsonFile(path.join(cwd, \"package.json\"), pkg);\r\n}\r\n\r\nexport function getMissingDeps(\r\n pkg: { dependencies?: Record<string, string>; devDependencies?: Record<string, string> },\r\n deps: string[],\r\n) {\r\n const existing = new Set([\r\n ...Object.keys(pkg.dependencies ?? {}),\r\n ...Object.keys(pkg.devDependencies ?? {}),\r\n ]);\r\n return deps.filter((d) => !existing.has(d));\r\n}\r\n\r\nexport async function installDeps(params: {\r\n cwd: string;\r\n pm: PackageManager;\r\n deps: string[];\r\n dev?: boolean;\r\n}) {\r\n const { cwd, pm, deps, dev } = params;\r\n if (!deps.length) return;\r\n\r\n const args: string[] = [];\r\n if (pm === \"pnpm\") args.push(\"add\");\r\n else if (pm === \"npm\") args.push(\"install\");\r\n else if (pm === \"yarn\") args.push(\"add\");\r\n else if (pm === \"bun\") args.push(\"add\");\r\n\r\n if (dev) {\r\n if (pm === \"npm\") args.push(\"--save-dev\");\r\n else args.push(\"-D\");\r\n }\r\n\r\n args.push(...deps);\r\n\r\n await execa(pm, args, { cwd, stdio: \"inherit\" });\r\n}\r\n\r\nexport async function ensureJsonFile(cwd: string, relPath: string, defaultJson: unknown) {\r\n const p = path.join(cwd, relPath);\r\n if (await pathExists(p)) return;\r\n await fs.mkdir(path.dirname(p), { recursive: true });\r\n await fs.writeFile(p, JSON.stringify(defaultJson, null, 2) + \"\\n\", \"utf8\");\r\n}\r\n\r\n\r\n","import path from \"node:path\";\nimport fs from \"node:fs\";\nimport { createRequire } from \"node:module\";\nimport { fileURLToPath } from \"node:url\";\nimport type { CliConfig, RegistryFile } from \"../types.js\";\nimport { pathExists, readJsonFile } from \"./fs.js\";\n\nexport const DEFAULT_CONFIG_PATH = \"components.json\";\n\nfunction findNearestPackageRoot(from: string) {\n // from: import.meta.url\n let dir = path.dirname(fileURLToPath(from));\n while (true) {\n if (fs.existsSync(path.join(dir, \"package.json\"))) return dir;\n const parent = path.dirname(dir);\n if (parent === dir) return dir;\n dir = parent;\n }\n}\n\nexport function defaultConfig(): CliConfig {\n return {\n schemaVersion: 1,\n componentsDir: \"components\",\n libDir: \"lib\",\n composablesDir: \"composables\",\n // 默认使用 CLI 包内置的 registry(随 reborn-ui 一起发布)\n registry: \"builtin\",\n // 项目根目录映射别名符号(默认 @)\n aliasSymbol: \"@\",\n };\n}\n\nexport async function loadConfig(cwd: string, configPath = DEFAULT_CONFIG_PATH) {\n const abs = path.isAbsolute(configPath)\n ? configPath\n : path.join(cwd, configPath);\n if (!(await pathExists(abs))) return null;\n return await readJsonFile<CliConfig>(abs);\n}\n\nexport function resolveRegistryJsonPath(params: { cwd: string; registry: string }) {\n const { cwd, registry } = params;\n\n // 0) 内置 registry\n if (!registry || registry === \"builtin\") {\n // 注意:CLI 经过打包后可能变成单文件(例如 dist/index.js),\n // import.meta.url 的相对基准会变化,不能用固定的 ../../registry/...\n // 这里通过向上查找最近的 package.json 来定位包根目录。\n const pkgRoot = findNearestPackageRoot(import.meta.url);\n return path.join(pkgRoot, \"registry\", \"registry.json\");\n }\n\n // 1) 如果是文件路径(包含分隔符或 .json),优先当成路径处理\n const looksLikePath =\n registry.includes(\"/\") ||\n registry.includes(\"\\\\\") ||\n registry.endsWith(\".json\");\n\n if (looksLikePath) {\n const abs = path.isAbsolute(registry) ? registry : path.join(cwd, registry);\n return abs;\n }\n\n // 2) 当成包名处理:默认读取 <pkg>/registry/registry.json\n const require = createRequire(import.meta.url);\n // 不显式绑定 cwd,避免 npx 场景下找不到 registry 包\n return require.resolve(`${registry}/registry/registry.json`);\n}\n\nexport async function loadRegistry(params: { cwd: string; registry: string }) {\n const jsonPath = resolveRegistryJsonPath(params);\n return await readJsonFile<RegistryFile>(jsonPath);\n}\n\n// 兼容旧配置文件名:shadcn-docs.json\nexport async function loadConfigCompat(cwd: string, configPath?: string) {\n if (configPath) return await loadConfig(cwd, configPath);\n const primary = await loadConfig(cwd, DEFAULT_CONFIG_PATH);\n if (primary) return primary;\n return await loadConfig(cwd, \"shadcn-docs.json\");\n}\n\n\n","import chalk from 'chalk';\r\nimport figlet from 'figlet';\r\n\r\n/**\r\n * 模拟打字机效果\r\n * @param text 要显示的文本\r\n * @param delay 每字符延迟时间 (ms)\r\n * @param newline 是否在结束时换行\r\n */\r\nexport async function typewriter(text: string, delay: number = 15, newline: boolean = true) {\r\n for (const char of text) {\r\n process.stdout.write(char);\r\n if (delay > 0) {\r\n await new Promise((resolve) => setTimeout(resolve, delay));\r\n }\r\n }\r\n if (newline) {\r\n process.stdout.write('\\n');\r\n }\r\n}\r\n\r\n/**\r\n * 显示 ASCII Logo\r\n */\r\nexport async function showLogo(text: string = 'Reborn UI') {\r\n return new Promise<void>((resolve) => {\r\n figlet.text(text, {\r\n font: 'Slant',\r\n horizontalLayout: 'default',\r\n verticalLayout: 'default',\r\n width: 80,\r\n whitespaceBreak: true\r\n }, async (err, data) => {\r\n if (err) {\r\n console.log(gradientText(text));\r\n resolve();\r\n return;\r\n }\r\n if (data) {\r\n const lines = data.split('\\n');\r\n for (const line of lines) {\r\n console.log(gradientText(line));\r\n await new Promise(r => setTimeout(r, 20)); // 每行显示间隔,增加动感\r\n }\r\n }\r\n resolve();\r\n });\r\n });\r\n}\r\n\r\n/**\r\n * 带有样式的成功提示\r\n */\r\nexport function successLog(message: string) {\r\n console.log(`${chalk.green('✔')} ${message}`);\r\n}\r\n\r\n/**\r\n * 带有样式的警告提示\r\n */\r\nexport function warnLog(message: string) {\r\n console.log(`${chalk.yellow('⚠')} ${message}`);\r\n}\r\n\r\n/**\r\n * 带有样式的错误提示\r\n */\r\nexport function errorLog(message: string) {\r\n console.log(`${chalk.red('✘')} ${message}`);\r\n}\r\n\r\n/**\r\n * 渐变色文本 (简单模拟,不依赖第三方复杂渐变库)\r\n */\r\nexport function gradientText(text: string) {\r\n const colors = [\r\n chalk.hex('#FF0080'),\r\n chalk.hex('#FF8C00'),\r\n chalk.hex('#40E0D0'),\r\n chalk.hex('#0080FF'),\r\n chalk.hex('#7B68EE'),\r\n ];\r\n\r\n let result = '';\r\n for (let i = 0; i < text.length; i++) {\r\n const color = colors[i % colors.length];\r\n result += color!(text[i]);\r\n }\r\n return result;\r\n}\r\n","import path from \"node:path\";\r\nimport fs from \"node:fs/promises\";\r\nimport fssync from \"node:fs\";\r\nimport { Command } from \"commander\";\r\nimport type { RegistryComponent, RegistryFile } from \"../types.js\";\r\nimport { listFilesRecursive, sha1, writeJsonFile } from \"../utils/fs.js\";\r\nimport { extractNpmDependenciesFromText } from \"../utils/imports.js\";\r\n\r\nfunction isAllowedFile(filePath: string) {\r\n const ext = path.extname(filePath).toLowerCase();\r\n return [\r\n \".vue\",\r\n \".ts\",\r\n \".js\",\r\n \".json\",\r\n \".css\",\r\n \".md\",\r\n \".svg\",\r\n ].includes(ext);\r\n}\r\n\r\nimport { transformToUniapp } from \"../utils/transformers.js\";\r\n\r\nfunction findWorkspaceRoot(startDir: string) {\r\n let current = path.resolve(startDir);\r\n for (; ;) {\r\n const marker = path.join(current, \"pnpm-workspace.yaml\");\r\n if (fssync.existsSync(marker)) return current;\r\n const parent = path.dirname(current);\r\n if (parent === current) return startDir;\r\n current = parent;\r\n }\r\n}\r\n\r\nexport function buildCommand() {\r\n const cmd = new Command(\"build\")\r\n .description(\"(内部)扫描组件源码并生成 registry JSON\")\r\n .option(\"--root <path>\", \"仓库根目录(默认自动向上查找 pnpm-workspace.yaml)\")\r\n .option(\r\n \"--source <path>\",\r\n \"组件源码目录(相对 root)\",\r\n \"app/components/reborn/ui\",\r\n )\r\n .option(\r\n \"--out <path>\",\r\n \"输出 registry.json 路径(相对 root)\",\r\n \"packages/cli/registry/registry.json\",\r\n )\r\n .option(\r\n \"--also-out <path>\",\r\n \"额外再输出一份 registry.json(可重复传参)\",\r\n (val, acc: string[]) => {\r\n acc.push(val);\r\n return acc;\r\n },\r\n [],\r\n )\r\n .action(async (opts) => {\r\n const rootDir = opts.root\r\n ? path.resolve(opts.root)\r\n : findWorkspaceRoot(process.cwd());\r\n const sourceDir = path.join(rootDir, opts.source);\r\n const outPath = path.join(rootDir, opts.out);\r\n const alsoOutPaths: string[] = (opts.alsoOut ?? []).map((p: string) =>\r\n path.join(rootDir, p),\r\n );\r\n\r\n const dirents = await fs.readdir(sourceDir, { withFileTypes: true });\r\n const componentDirs = dirents\r\n .filter((d) => d.isDirectory())\r\n .map((d) => path.join(sourceDir, d.name))\r\n .sort((a, b) => a.localeCompare(b));\r\n\r\n const components: RegistryComponent[] = [];\r\n\r\n for (const absComponentDir of componentDirs) {\r\n const name = path.basename(absComponentDir);\r\n const absFiles = (await listFilesRecursive(absComponentDir)).filter(\r\n isAllowedFile,\r\n );\r\n\r\n\r\n const files: RegistryComponent[\"files\"] = [];\r\n const depSet = new Set<string>();\r\n\r\n for (const absFile of absFiles) {\r\n const rel = path\r\n .relative(absComponentDir, absFile)\r\n .split(path.sep)\r\n .join(\"/\");\r\n const content = await fs.readFile(absFile, \"utf8\");\r\n\r\n const ext = path.extname(absFile).toLowerCase();\r\n\r\n // 1. Web 版本(原样)\r\n // 如果是 .vue,明确标记 target=\"web\"\r\n // 如果是其他文件(.ts/.js),默认不标记(表示通用),或者也可以标记 web。\r\n // 策略:.vue 必须要分 web/uniapp;.ts/.js 视情况,这里先默认通用(不标记 target),\r\n // 除非你需要 uniapp 特有的 js。\r\n // 简化起见:.vue 文件生成两份,一份 target=web,一份 target=uniapp\r\n // 其他文件生成一份,无 target(通用)。\r\n\r\n if (ext === \".vue\") {\r\n files.push({ path: rel, content, target: \"web\" });\r\n files.push({\r\n path: rel,\r\n content: transformToUniapp(content),\r\n target: \"uniapp\",\r\n });\r\n } else {\r\n files.push({ path: rel, content });\r\n }\r\n\r\n // 只从代码文件里抽依赖\r\n if (ext === \".ts\" || ext === \".js\" || ext === \".vue\") {\r\n for (const dep of extractNpmDependenciesFromText(content)) {\r\n depSet.add(dep);\r\n }\r\n }\r\n }\r\n\r\n components.push({\r\n name,\r\n dependencies: [...depSet].sort(),\r\n files,\r\n });\r\n }\r\n\r\n const registry: RegistryFile = {\r\n schemaVersion: 1,\r\n generatedAt: new Date().toISOString(),\r\n source: {\r\n rootDir: rootDir.split(path.sep).join(\"/\"),\r\n componentsDir: opts.source,\r\n },\r\n components,\r\n };\r\n\r\n async function writeOut(targetRegistryPath: string) {\r\n const outComponentsDir = path.join(\r\n path.dirname(targetRegistryPath),\r\n \"components\",\r\n );\r\n await writeJsonFile(targetRegistryPath, registry);\r\n\r\n // 额外输出每个组件的 json,方便调试/按需读取\r\n await fs.mkdir(outComponentsDir, { recursive: true });\r\n for (const c of components) {\r\n await writeJsonFile(\r\n path.join(outComponentsDir, `${c.name}.json`),\r\n {\r\n ...c,\r\n fileCount: c.files.length,\r\n contentHash: sha1(JSON.stringify(c.files.map((f) => f.content))),\r\n },\r\n );\r\n }\r\n }\r\n\r\n await writeOut(outPath);\r\n for (const p of alsoOutPaths) await writeOut(p);\r\n\r\n // eslint-disable-next-line no-console\r\n console.log(\r\n `registry 已生成:${path.relative(process.cwd(), outPath)}(${components.length} 个组件)`,\r\n );\r\n });\r\n\r\n return cmd;\r\n}\r\n\r\n\r\n","const IMPORT_RE =\r\n /\\bfrom\\s+[\"']([^\"']+)[\"']|\\bimport\\(\\s*[\"']([^\"']+)[\"']\\s*\\)|\\brequire\\(\\s*[\"']([^\"']+)[\"']\\s*\\)/g;\r\n\r\nfunction normalizePackageName(specifier: string) {\r\n // ignore relative/alias/virtual\r\n if (\r\n specifier.startsWith(\".\") ||\r\n specifier.startsWith(\"/\") ||\r\n specifier.startsWith(\"@/\") ||\r\n specifier.startsWith(\"~/\") ||\r\n specifier.startsWith(\"#\") ||\r\n specifier.startsWith(\"virtual:\")\r\n ) {\r\n return null;\r\n }\r\n\r\n // scoped package: @scope/name[/...]\r\n if (specifier.startsWith(\"@\")) {\r\n const parts = specifier.split(\"/\");\r\n if (parts.length >= 2) return `${parts[0]}/${parts[1]}`;\r\n return specifier;\r\n }\r\n\r\n // normal package: name[/...]\r\n return specifier.split(\"/\")[0] ?? null;\r\n}\r\n\r\nexport function extractNpmDependenciesFromText(text: string) {\r\n const out = new Set<string>();\r\n\r\n for (const match of text.matchAll(IMPORT_RE)) {\r\n const spec = match[1] ?? match[2] ?? match[3];\r\n if (!spec) continue;\r\n const pkg = normalizePackageName(spec);\r\n if (!pkg) continue;\r\n out.add(pkg);\r\n }\r\n\r\n // 常见“框架自带/不应自动安装”的虚拟模块\r\n out.delete(\"nuxt\");\r\n out.delete(\"vue\");\r\n\r\n return [...out].sort();\r\n}\r\n\r\n\r\n","\r\nexport function transformToUniapp(code: string): string {\r\n // 1. 标签替换\r\n // div -> view\r\n // span, font -> text(注意:span可能嵌套,正则要注意)\r\n // ul, li -> view\r\n // img -> image\r\n // a -> navigator\r\n // select -> picker\r\n // iframe -> web-view\r\n\r\n let newCode = code;\r\n\r\n // 简单正则替换标签名\r\n const tagMap: Record<string, string> = {\r\n div: \"view\",\r\n span: \"text\",\r\n font: \"text\",\r\n ul: \"view\",\r\n li: \"view\",\r\n img: \"image\",\r\n a: \"navigator\",\r\n select: \"picker\",\r\n iframe: \"web-view\",\r\n };\r\n\r\n for (const [k, v] of Object.entries(tagMap)) {\r\n // 匹配 <tag ...> 和 </tag>\r\n // 1. <tag\r\n newCode = newCode.replace(new RegExp(`<${k}(\\\\s+|>)`, \"g\"), `<${v}$1`);\r\n // 2. </tag>\r\n newCode = newCode.replace(new RegExp(`<\\\\/${k}>`, \"g\"), `</${v}>`);\r\n }\r\n\r\n // 2. 特殊处理\r\n // input[type=\"search\"] -> type=\"text\" confirm-type=\"search\"\r\n newCode = newCode.replace(/type=[\"']search[\"']/g, 'confirm-type=\"search\"');\r\n\r\n return newCode;\r\n}\r\n","import path from \"node:path\";\r\nimport fs from \"node:fs/promises\";\r\nimport { Command } from \"commander\";\r\nimport prompts from \"prompts\";\r\nimport type { CliConfig, PackageManager } from \"../types.js\";\r\nimport { ensureDir, listFilesRecursive, pathExists, writeJsonFile, writeTextFile } from \"../utils/fs.js\";\r\nimport { detectPackageManager, readPackageJson } from \"../utils/pm.js\";\r\nimport { DEFAULT_CONFIG_PATH, defaultConfig, loadConfigCompat } from \"../utils/registry.js\";\r\nimport { copyTemplateSubdir } from \"../utils/templates.js\";\r\nimport { gradientText, showLogo, typewriter } from \"../utils/ui.js\";\r\n\r\nfunction looksLikeNuxtProject(cwd: string) {\r\n return Promise.all([\r\n pathExists(path.join(cwd, \"nuxt.config.ts\")),\r\n pathExists(path.join(cwd, \"nuxt.config.js\")),\r\n pathExists(path.join(cwd, \"nuxt.config.mjs\")),\r\n ]).then((arr) => arr.some(Boolean));\r\n}\r\n\r\n\r\n// function makeDefaultTailwindConfig removed in favor of template file\r\n\r\n\r\n\r\n\r\nfunction cnUtilsTs(params: { importPath?: string }) {\r\n // importPath 预留:如果用户想从别处导入 clsx/twMerge\r\n void params;\r\n return `import { type ClassValue, clsx } from \"clsx\";\\nimport { twMerge } from \"tailwind-merge\";\\n\\nexport function cn(...inputs: ClassValue[]) {\\n return twMerge(clsx(inputs));\\n}\\n`;\r\n}\r\n\r\nasync function rewriteAliasInDir(params: {\r\n cwd: string;\r\n targetDir: string;\r\n aliasSymbol: string;\r\n}) {\r\n const { cwd, targetDir, aliasSymbol } = params;\r\n if (!aliasSymbol || aliasSymbol === \"@\") return;\r\n const absDir = path.join(cwd, targetDir);\r\n if (!(await pathExists(absDir))) return;\r\n\r\n const files = await listFilesRecursive(absDir);\r\n for (const f of files) {\r\n const ext = path.extname(f).toLowerCase();\r\n // 只处理常见文本文件,避免误改二进制\r\n if (![\".ts\", \".tsx\", \".js\", \".jsx\", \".vue\", \".mjs\", \".cjs\"].includes(ext)) continue;\r\n const raw = await fs.readFile(f, \"utf8\");\r\n const next = raw.replaceAll(\"@/\", `${aliasSymbol}/`);\r\n if (next !== raw) await fs.writeFile(f, next, \"utf8\");\r\n }\r\n}\r\n\r\nexport function initCommand() {\r\n const cmd = new Command(\"init\")\r\n .description(\"初始化新项目\")\r\n .option(\"--cwd <path>\", \"目标项目目录\", process.cwd())\r\n .option(\"--pm <pm>\", \"包管理器:pnpm|npm|yarn|bun\")\r\n .option(\"--yes\", \"跳过交互,使用默认值\", false)\r\n .option(\"--overwrite\", \"覆盖已存在的模板文件(lib/composables)\", false)\r\n .option(\r\n \"--config <path>\",\r\n \"配置文件路径(相对 cwd)\",\r\n DEFAULT_CONFIG_PATH,\r\n )\r\n .option(\r\n \"--components-dir <path>\",\r\n \"组件写入目录\",\r\n defaultConfig().componentsDir,\r\n )\r\n .option(\r\n \"--lib-dir <path>\",\r\n \"lib 目录(cn/utils)\",\r\n defaultConfig().libDir,\r\n )\r\n .option(\r\n \"--composables-dir <path>\",\r\n \"composables 目录\",\r\n defaultConfig().composablesDir,\r\n )\r\n .option(\r\n \"--alias-symbol <symbol>\",\r\n \"项目根目录映射别名符号(默认 @)\",\r\n defaultConfig().aliasSymbol,\r\n )\r\n .option(\"--registry <pkgOrPath>\", \"registry 来源(默认 builtin)\", defaultConfig().registry)\r\n .action(async (opts) => {\r\n const cwd = path.resolve(opts.cwd);\r\n\r\n // eslint-disable-next-line no-console\r\n console.log(\r\n \"提示:项目中有部分依赖需要使用@/ 路径进行配置,请您安装到项目中配置好的路径下,请慎重选择文件位置\",\r\n );\r\n\r\n const pkgPath = path.join(cwd, \"package.json\");\r\n if (!(await pathExists(pkgPath))) {\r\n throw new Error(`未找到 package.json:${pkgPath}`);\r\n }\r\n\r\n const pm: PackageManager =\r\n opts.pm ?? (await detectPackageManager(cwd));\r\n\r\n const cfg: CliConfig = {\r\n schemaVersion: 1,\r\n componentsDir: opts.componentsDir,\r\n libDir: opts.libDir,\r\n composablesDir: opts.composablesDir,\r\n aliasSymbol: opts.aliasSymbol,\r\n registry: opts.registry,\r\n };\r\n\r\n // 如果已存在 components.json,默认沿用(除非用户显式传参覆盖)\r\n const existing = await loadConfigCompat(cwd);\r\n if (existing) {\r\n cfg.componentsDir = opts.componentsDir ?? existing.componentsDir ?? cfg.componentsDir;\r\n cfg.libDir = opts.libDir ?? existing.libDir ?? cfg.libDir;\r\n cfg.composablesDir = opts.composablesDir ?? existing.composablesDir ?? cfg.composablesDir;\r\n cfg.aliasSymbol = opts.aliasSymbol ?? existing.aliasSymbol ?? cfg.aliasSymbol;\r\n cfg.registry = opts.registry ?? existing.registry ?? cfg.registry;\r\n }\r\n\r\n if (!opts.yes) {\r\n const nuxt = await looksLikeNuxtProject(cwd);\r\n const res = await prompts(\r\n [\r\n {\r\n type: \"text\",\r\n name: \"componentsDir\",\r\n message: \"组件目录(componentsDir)\",\r\n initial: cfg.componentsDir,\r\n },\r\n {\r\n type: \"text\",\r\n name: \"libDir\",\r\n message: \"lib 目录(libDir,用于 cn/utils)\",\r\n initial: cfg.libDir,\r\n },\r\n {\r\n type: \"text\",\r\n name: \"composablesDir\",\r\n message: \"composables 目录(composablesDir)\",\r\n initial: cfg.composablesDir,\r\n },\r\n {\r\n type: \"text\",\r\n name: \"aliasSymbol\",\r\n message: \"基于项目根目录的“映射”的符号是?(aliasSymbol)\",\r\n initial: cfg.aliasSymbol ?? \"@\",\r\n },\r\n ],\r\n {\r\n onCancel: () => {\r\n throw new Error(\"已取消\");\r\n },\r\n },\r\n );\r\n\r\n cfg.componentsDir = res.componentsDir ?? cfg.componentsDir;\r\n cfg.libDir = res.libDir ?? cfg.libDir;\r\n cfg.composablesDir = res.composablesDir ?? cfg.composablesDir;\r\n cfg.aliasSymbol = res.aliasSymbol ?? cfg.aliasSymbol;\r\n\r\n // 写配置\r\n const cfgPath = path.join(cwd, opts.config);\r\n await writeJsonFile(cfgPath, cfg);\r\n\r\n // 复制模板:lib/ + composables/\r\n await copyTemplateSubdir({\r\n subdir: \"lib\",\r\n cwd,\r\n targetDir: cfg.libDir,\r\n overwrite: opts.overwrite,\r\n });\r\n await copyTemplateSubdir({\r\n subdir: \"composables\",\r\n cwd,\r\n targetDir: cfg.composablesDir,\r\n overwrite: opts.overwrite,\r\n });\r\n // 仅替换 \"@/...\" 的别名符号,不改其它路径内容\r\n await rewriteAliasInDir({ cwd, targetDir: cfg.libDir, aliasSymbol: cfg.aliasSymbol ?? \"@\" });\r\n await rewriteAliasInDir({\r\n cwd,\r\n targetDir: cfg.composablesDir,\r\n aliasSymbol: cfg.aliasSymbol ?? \"@\",\r\n });\r\n\r\n // 兜底:如果用户模板里没有 utils.ts,则生成 cn\r\n const libUtilsPath = path.join(cwd, cfg.libDir, \"utils.ts\");\r\n if (!(await pathExists(libUtilsPath))) await writeTextFile(libUtilsPath, cnUtilsTs({}));\r\n\r\n console.log(`已写入配置:${path.relative(process.cwd(), cfgPath)};并生成 cn/utils(pm=${pm})`);\r\n console.log(\"请按照 https://tw.icebreaker.top/docs/quick-start/v4/uni-app-vite 指引进行项目配置初始化\");\r\n\r\n console.log(\"\");\r\n await showLogo();\r\n await typewriter(gradientText(\"感谢使用 Reborn UI ! ✨\"), 20);\r\n return;\r\n }\r\n\r\n // --yes:直接写默认配置\r\n const cfgPath = path.join(cwd, opts.config);\r\n await ensureDir(path.dirname(cfgPath));\r\n await fs.writeFile(cfgPath, JSON.stringify(cfg, null, 2) + \"\\n\", \"utf8\");\r\n\r\n // 复制模板:lib/ + composables/\r\n await copyTemplateSubdir({\r\n subdir: \"lib\",\r\n cwd,\r\n targetDir: cfg.libDir,\r\n overwrite: opts.overwrite,\r\n });\r\n await copyTemplateSubdir({\r\n subdir: \"composables\",\r\n cwd,\r\n targetDir: cfg.composablesDir,\r\n overwrite: opts.overwrite,\r\n });\r\n // 仅替换 \"@/...\" 的别名符号,不改其它路径内容\r\n await rewriteAliasInDir({ cwd, targetDir: cfg.libDir, aliasSymbol: cfg.aliasSymbol ?? \"@\" });\r\n await rewriteAliasInDir({\r\n cwd,\r\n targetDir: cfg.composablesDir,\r\n aliasSymbol: cfg.aliasSymbol ?? \"@\",\r\n });\r\n\r\n // 兜底:如果用户模板里没有 utils.ts,则生成 cn\r\n const libUtilsPath = path.join(cwd, cfg.libDir, \"utils.ts\");\r\n if (!(await pathExists(libUtilsPath))) await writeTextFile(libUtilsPath, cnUtilsTs({}));\r\n\r\n // eslint-disable-next-line no-console\r\n console.log(`已初始化:${path.relative(process.cwd(), cfgPath)};并生成 cn/utils(pm=${pm})`);\r\n console.log(\"请按照 https://tw.icebreaker.top/docs/quick-start/v4/uni-app-vite 指引进行项目配置初始化\");\r\n\r\n console.log(\"\");\r\n await showLogo();\r\n await typewriter(gradientText(\"感谢使用 Reborn UI ! ✨\"), 2);\r\n });\r\n\r\n return cmd;\r\n}\r\n\r\n\r\n","import path from \"node:path\";\nimport fs from \"node:fs\";\nimport { fileURLToPath } from \"node:url\";\nimport { pathExists } from \"./fs.js\";\nimport { copyDirRecursive } from \"./fs.js\";\n\nfunction findNearestPackageRoot(from: string) {\n // from: import.meta.url\n let dir = path.dirname(fileURLToPath(from));\n while (true) {\n if (fs.existsSync(path.join(dir, \"package.json\"))) return dir;\n const parent = path.dirname(dir);\n if (parent === dir) return dir;\n dir = parent;\n }\n}\n\nexport function getTemplatesRootDir() {\n // 注意:CLI 打包后可能变成单文件(dist/index.js),此时 import.meta.url 的相对基准会变化。\n // 通过向上查找最近的 package.json 来定位包根目录。\n const pkgRoot = findNearestPackageRoot(import.meta.url);\n return path.join(pkgRoot, \"templates\");\n}\n\nexport async function copyTemplateSubdir(params: {\n subdir: \"lib\" | \"composables\";\n cwd: string;\n targetDir: string;\n overwrite?: boolean;\n}) {\n const templatesRoot = getTemplatesRootDir();\n const fromDir = path.join(templatesRoot, params.subdir);\n if (!(await pathExists(fromDir))) return false;\n\n const toDir = path.join(params.cwd, params.targetDir);\n await copyDirRecursive({ fromDir, toDir, overwrite: params.overwrite });\n return true;\n}\n\n\n\n\n\n\n","{\n \"name\": \"reborn-ui\",\n \"version\": \"0.1.64\",\n \"description\": \"A CLI for Reborn UI\",\n \"author\": \"1997liuyh-boop\",\n \"license\": \"MIT\",\n \"repository\": {\n \"type\": \"git\",\n \"url\": \"https://github.com/1997liuyh-boop/Reborn-UI.git\"\n },\n \"type\": \"module\",\n \"bin\": {\n \"reborn-ui\": \"dist/index.js\"\n },\n \"files\": [\n \"dist\",\n \"registry\",\n \"templates\",\n \"README.md\"\n ],\n \"publishConfig\": {\n \"access\": \"public\"\n },\n \"scripts\": {\n \"cli\": \"tsx src/index.ts\",\n \"build\": \"tsup\",\n \"dev\": \"tsx src/index.ts\",\n \"clean\": \"rimraf dist\",\n \"prepare\": \"tsup\",\n \"registry:build\": \"tsx src/index.ts build --root ../..\",\n \"prepack\": \"npm run build && npm run registry:build\",\n \"prepublishOnly\": \"npm run build && npm run registry:build\"\n },\n \"dependencies\": {\n \"chalk\": \"^5.6.2\",\n \"cli-progress\": \"^3.12.0\",\n \"commander\": \"^14.0.0\",\n \"execa\": \"^9.6.0\",\n \"figlet\": \"^1.9.4\",\n \"ora\": \"^9.0.0\",\n \"prompts\": \"^2.4.2\"\n },\n \"devDependencies\": {\n \"@types/cli-progress\": \"^3.11.6\",\n \"@types/figlet\": \"^1.7.0\",\n \"@types/node\": \"^22.10.7\",\n \"@types/prompts\": \"^2.4.9\",\n \"rimraf\": \"^6.0.1\",\n \"tsup\": \"^8.5.0\",\n \"tsx\": \"^4.20.0\",\n \"typescript\": \"^5.8.3\"\n }\n}\n"],"mappings":";;;AAAA,SAAS,WAAAA,gBAAe;;;ACAxB,OAAOC,WAAU;AACjB,SAAS,eAAe;AACxB,OAAO,aAAa;;;ACFpB,SAAS,kBAAkB;AAC3B,OAAO,QAAQ;AACf,OAAO,UAAU;AAEjB,eAAsB,WAAW,GAAW;AAC1C,MAAI;AACF,UAAM,GAAG,OAAO,CAAC;AACjB,WAAO;AAAA,EACT,QAAQ;AACN,WAAO;AAAA,EACT;AACF;AAEA,eAAsB,UAAU,SAAiB;AAC/C,QAAM,GAAG,MAAM,SAAS,EAAE,WAAW,KAAK,CAAC;AAC7C;AAEA,eAAsB,aAAgB,UAA8B;AAClE,QAAM,MAAM,MAAM,GAAG,SAAS,UAAU,MAAM;AAC9C,SAAO,KAAK,MAAM,GAAG;AACvB;AAEA,eAAsB,cAAc,UAAkB,MAAe;AACnE,QAAM,UAAU,KAAK,QAAQ,QAAQ,CAAC;AACtC,QAAM,GAAG,UAAU,UAAU,KAAK,UAAU,MAAM,MAAM,CAAC,IAAI,MAAM,MAAM;AAC3E;AAEA,eAAsB,cAAc,UAAkB,SAAiB;AACrE,QAAM,UAAU,KAAK,QAAQ,QAAQ,CAAC;AACtC,QAAM,GAAG,UAAU,UAAU,SAAS,MAAM;AAC9C;AAEA,eAAsB,mBACpB,SACA,MACA;AACA,QAAM,iBACJ,MAAM,kBACN,oBAAI,IAAI,CAAC,gBAAgB,QAAQ,SAAS,QAAQ,SAAS,CAAC;AAE9D,QAAM,MAAgB,CAAC;AAEvB,iBAAe,KAAK,SAAiB;AACnC,UAAM,UAAU,MAAM,GAAG,QAAQ,SAAS,EAAE,eAAe,KAAK,CAAC;AACjE,eAAW,SAAS,SAAS;AAC3B,YAAM,IAAI,KAAK,KAAK,SAAS,MAAM,IAAI;AACvC,UAAI,MAAM,YAAY,GAAG;AACvB,YAAI,eAAe,IAAI,MAAM,IAAI,EAAG;AACpC,cAAM,KAAK,CAAC;AAAA,MACd,WAAW,MAAM,OAAO,GAAG;AACzB,YAAI,KAAK,CAAC;AAAA,MACZ;AAAA,IACF;AAAA,EACF;AAEA,QAAM,KAAK,OAAO;AAClB,SAAO;AACT;AAEA,eAAsB,iBAAiB,QAKpC;AACD,QAAM,EAAE,SAAS,OAAO,UAAU,IAAI;AACtC,QAAM,kBAAkB,OAAO,mBAAmB,oBAAI,IAAI,CAAC,WAAW,CAAC;AAEvE,QAAM,UAAU,MAAM,GAAG,QAAQ,SAAS,EAAE,eAAe,KAAK,CAAC;AACjE,QAAM,UAAU,KAAK;AAErB,aAAW,SAAS,SAAS;AAC3B,QAAI,gBAAgB,IAAI,MAAM,IAAI,EAAG;AACrC,UAAM,OAAO,KAAK,KAAK,SAAS,MAAM,IAAI;AAC1C,UAAM,KAAK,KAAK,KAAK,OAAO,MAAM,IAAI;AAEtC,QAAI,MAAM,YAAY,GAAG;AACvB,YAAM,iBAAiB,EAAE,SAAS,MAAM,OAAO,IAAI,WAAW,gBAAgB,CAAC;AAC/E;AAAA,IACF;AAEA,QAAI,CAAC,MAAM,OAAO,EAAG;AAErB,QAAI,CAAC,WAAW;AACd,UAAI;AACF,cAAM,GAAG,OAAO,EAAE;AAClB;AAAA,MACF,QAAQ;AAAA,MAER;AAAA,IACF;AAEA,UAAM,UAAU,KAAK,QAAQ,EAAE,CAAC;AAChC,UAAM,GAAG,SAAS,MAAM,EAAE;AAAA,EAC5B;AACF;AAEO,SAAS,KAAK,MAAc;AACjC,SAAO,WAAW,MAAM,EAAE,OAAO,IAAI,EAAE,OAAO,KAAK;AACrD;;;ACnGA,OAAOC,SAAQ;AACf,OAAOC,WAAU;AACjB,SAAS,aAAa;AAItB,eAAsB,qBAAqB,KAAsC;AAC/E,MAAI,MAAM,WAAWC,MAAK,KAAK,KAAK,gBAAgB,CAAC,EAAG,QAAO;AAC/D,MAAI,MAAM,WAAWA,MAAK,KAAK,KAAK,WAAW,CAAC,EAAG,QAAO;AAC1D,MAAI,MAAM,WAAWA,MAAK,KAAK,KAAK,mBAAmB,CAAC,EAAG,QAAO;AAClE,MAAI,MAAM,WAAWA,MAAK,KAAK,KAAK,WAAW,CAAC,EAAG,QAAO;AAC1D,SAAO;AACT;;;ACZA,OAAOC,WAAU;AACjB,OAAOC,SAAQ;AACf,SAAS,qBAAqB;AAC9B,SAAS,qBAAqB;AAIvB,IAAM,sBAAsB;AAEnC,SAAS,uBAAuB,MAAc;AAE5C,MAAI,MAAMC,MAAK,QAAQ,cAAc,IAAI,CAAC;AAC1C,SAAO,MAAM;AACX,QAAIC,IAAG,WAAWD,MAAK,KAAK,KAAK,cAAc,CAAC,EAAG,QAAO;AAC1D,UAAM,SAASA,MAAK,QAAQ,GAAG;AAC/B,QAAI,WAAW,IAAK,QAAO;AAC3B,UAAM;AAAA,EACR;AACF;AAEO,SAAS,gBAA2B;AACzC,SAAO;AAAA,IACL,eAAe;AAAA,IACf,eAAe;AAAA,IACf,QAAQ;AAAA,IACR,gBAAgB;AAAA;AAAA,IAEhB,UAAU;AAAA;AAAA,IAEV,aAAa;AAAA,EACf;AACF;AAEA,eAAsB,WAAW,KAAa,aAAa,qBAAqB;AAC9E,QAAM,MAAMA,MAAK,WAAW,UAAU,IAClC,aACAA,MAAK,KAAK,KAAK,UAAU;AAC7B,MAAI,CAAE,MAAM,WAAW,GAAG,EAAI,QAAO;AACrC,SAAO,MAAM,aAAwB,GAAG;AAC1C;AAEO,SAAS,wBAAwB,QAA2C;AACjF,QAAM,EAAE,KAAK,SAAS,IAAI;AAG1B,MAAI,CAAC,YAAY,aAAa,WAAW;AAIvC,UAAM,UAAU,uBAAuB,YAAY,GAAG;AACtD,WAAOA,MAAK,KAAK,SAAS,YAAY,eAAe;AAAA,EACvD;AAGA,QAAM,gBACJ,SAAS,SAAS,GAAG,KACrB,SAAS,SAAS,IAAI,KACtB,SAAS,SAAS,OAAO;AAE3B,MAAI,eAAe;AACjB,UAAM,MAAMA,MAAK,WAAW,QAAQ,IAAI,WAAWA,MAAK,KAAK,KAAK,QAAQ;AAC1E,WAAO;AAAA,EACT;AAGA,QAAME,WAAU,cAAc,YAAY,GAAG;AAE7C,SAAOA,SAAQ,QAAQ,GAAG,QAAQ,yBAAyB;AAC7D;AAEA,eAAsB,aAAa,QAA2C;AAC5E,QAAM,WAAW,wBAAwB,MAAM;AAC/C,SAAO,MAAM,aAA2B,QAAQ;AAClD;AAGA,eAAsB,iBAAiB,KAAa,YAAqB;AACvE,MAAI,WAAY,QAAO,MAAM,WAAW,KAAK,UAAU;AACvD,QAAM,UAAU,MAAM,WAAW,KAAK,mBAAmB;AACzD,MAAI,QAAS,QAAO;AACpB,SAAO,MAAM,WAAW,KAAK,kBAAkB;AACjD;;;AHvEA,OAAO,iBAAiB;AACxB,OAAOC,YAAW;;;AIXlB,OAAO,WAAW;AAClB,OAAO,YAAY;AAQnB,eAAsB,WAAW,MAAc,QAAgB,IAAI,UAAmB,MAAM;AACxF,aAAW,QAAQ,MAAM;AACrB,YAAQ,OAAO,MAAM,IAAI;AACzB,QAAI,QAAQ,GAAG;AACX,YAAM,IAAI,QAAQ,CAAC,YAAY,WAAW,SAAS,KAAK,CAAC;AAAA,IAC7D;AAAA,EACJ;AACA,MAAI,SAAS;AACT,YAAQ,OAAO,MAAM,IAAI;AAAA,EAC7B;AACJ;AAKA,eAAsB,SAAS,OAAe,aAAa;AACvD,SAAO,IAAI,QAAc,CAAC,YAAY;AAClC,WAAO,KAAK,MAAM;AAAA,MACd,MAAM;AAAA,MACN,kBAAkB;AAAA,MAClB,gBAAgB;AAAA,MAChB,OAAO;AAAA,MACP,iBAAiB;AAAA,IACrB,GAAG,OAAO,KAAK,SAAS;AACpB,UAAI,KAAK;AACL,gBAAQ,IAAI,aAAa,IAAI,CAAC;AAC9B,gBAAQ;AACR;AAAA,MACJ;AACA,UAAI,MAAM;AACN,cAAM,QAAQ,KAAK,MAAM,IAAI;AAC7B,mBAAW,QAAQ,OAAO;AACtB,kBAAQ,IAAI,aAAa,IAAI,CAAC;AAC9B,gBAAM,IAAI,QAAQ,OAAK,WAAW,GAAG,EAAE,CAAC;AAAA,QAC5C;AAAA,MACJ;AACA,cAAQ;AAAA,IACZ,CAAC;AAAA,EACL,CAAC;AACL;AAKO,SAAS,WAAW,SAAiB;AACxC,UAAQ,IAAI,GAAG,MAAM,MAAM,QAAG,CAAC,IAAI,OAAO,EAAE;AAChD;AAmBO,SAAS,aAAa,MAAc;AACvC,QAAM,SAAS;AAAA,IACX,MAAM,IAAI,SAAS;AAAA,IACnB,MAAM,IAAI,SAAS;AAAA,IACnB,MAAM,IAAI,SAAS;AAAA,IACnB,MAAM,IAAI,SAAS;AAAA,IACnB,MAAM,IAAI,SAAS;AAAA,EACvB;AAEA,MAAI,SAAS;AACb,WAAS,IAAI,GAAG,IAAI,KAAK,QAAQ,KAAK;AAClC,UAAM,QAAQ,OAAO,IAAI,OAAO,MAAM;AACtC,cAAU,MAAO,KAAK,CAAC,CAAC;AAAA,EAC5B;AACA,SAAO;AACX;;;AJ3EA,SAAS,eAAe,QAGrB;AACD,QAAM,EAAE,SAAS,YAAY,IAAI;AACjC,MAAI,CAAC,eAAe,gBAAgB,IAAK,QAAO;AAIhD,SAAO,QAAQ,WAAW,MAAM,GAAG,WAAW,GAAG;AACnD;AAEA,eAAe,oBAAoB,QAQhC;AACD,QAAM,EAAE,KAAK,eAAe,aAAa,WAAW,WAAW,YAAY,SAAS,IAAI;AACxF,QAAM,UAAUC,MAAK,KAAK,KAAK,eAAe,UAAU,IAAI;AAC5D,QAAM,UAAU,OAAO;AAEvB,aAAW,KAAK,UAAU,OAAO;AAI/B,QAAI,EAAE,UAAU,EAAE,WAAW,UAAU;AACrC;AAAA,IACF;AAEA,UAAM,SAASA,MAAK,KAAK,SAAS,GAAG,EAAE,KAAK,MAAM,GAAG,CAAC;AACtD,QAAI,CAAC,aAAc,MAAM,WAAW,MAAM,GAAI;AAC5C,mBAAa;AACb;AAAA,IACF;AACA,UAAM,cAAc,eAAe,EAAE,SAAS,EAAE,SAAS,YAAY,CAAC;AACtE,UAAM,cAAc,QAAQ,WAAW;AACvC,iBAAa;AAEb,UAAM,IAAI,QAAQ,OAAK,WAAW,GAAG,EAAE,CAAC;AAAA,EAC1C;AAEA,SAAO;AACT;AAEO,SAAS,aAAa;AAC3B,QAAM,MAAM,IAAI,QAAQ,KAAK,EAC1B,YAAY,gFAAe,EAC3B,SAAS,mBAAmB,kDAAU,EACtC,OAAO,gBAAgB,wCAAU,QAAQ,IAAI,CAAC,EAC9C,OAAO,aAAa,iDAAwB,EAC5C,OAAO,SAAS,4BAAQ,KAAK,EAC7B,OAAO,eAAe,8CAAW,KAAK,EACtC,OAAO,mBAAmB,oEAAkB,iBAAiB,EAC7D,OAAO,0BAA0B,+CAAiB,EAClD,OAAO,2BAA2B,oDAAsB,EACxD,OAAO,oBAAoB,6CAAe,EAC1C,OAAO,2BAA2B,4EAA0B,EAC5D,OAAO,OAAO,YAAsB,SAAS;AAC5C,UAAM,MAAMA,MAAK,QAAQ,KAAK,GAAG;AACjC,UAAM,KACJ,KAAK,MAAO,MAAM,qBAAqB,GAAG;AAE5C,UAAM,MAAO,MAAM,iBAAiB,KAAK,KAAK,MAAM,KAAM,cAAc;AACxE,QAAI,KAAK,SAAU,KAAI,WAAW,KAAK;AACvC,QAAI,KAAK,cAAe,KAAI,gBAAgB,KAAK;AACjD,QAAI,KAAK,OAAQ,KAAI,SAAS,KAAK;AACnC,QAAI,KAAK,YAAa,KAAI,cAAc,KAAK;AAE7C,UAAM,WAAW,MAAM,aAAa,EAAE,KAAK,UAAU,IAAI,SAAS,CAAC;AAEnE,QAAI,UAAU,cAAc,CAAC;AAC7B,QAAI,CAAC,QAAQ,QAAQ;AACnB,UAAI,KAAK,KAAK;AACZ,cAAM,IAAI,MAAM,mJAAgC;AAAA,MAClD;AACA,YAAM,UAAU,SAAS,WAAW,IAAI,CAAC,OAAO;AAAA,QAC9C,OAAO,EAAE;AAAA,QACT,OAAO,EAAE;AAAA,MACX,EAAE;AACF,YAAM,MAAM,MAAM;AAAA,QAChB;AAAA,UACE;AAAA,YACE,MAAM;AAAA,YACN,MAAM;AAAA,YACN,SAAS;AAAA,YACT;AAAA,YACA,KAAK;AAAA,UACP;AAAA,QACF;AAAA,QACA;AAAA,UACE,UAAU,MAAM;AACd,kBAAM,IAAI,MAAM,oBAAK;AAAA,UACvB;AAAA,QACF;AAAA,MACF;AACA,gBAAU,IAAI,YAAY,CAAC;AAAA,IAC7B;AAGA,UAAM,EAAE,SAAS,IAAI,MAAM,QAAQ;AAAA,MACjC,MAAM;AAAA,MACN,MAAM;AAAA,MACN,SAAS;AAAA,MACT,SAAS;AAAA,QACP,EAAE,OAAO,sBAAY,OAAO,MAAM;AAAA,QAClC,EAAE,OAAO,UAAU,OAAO,SAAS;AAAA,MACrC;AAAA,MACA,SAAS;AAAA,IACX,CAAC;AAED,QAAI,CAAC,UAAU;AACb,YAAM,IAAI,MAAM,oBAAK;AAAA,IACvB;AAEA,UAAM,aAAa,QAAQ,OAAO,CAAC,KAAK,SAAS;AAC/C,YAAM,IAAI,SAAS,WAAW,KAAK,CAAC,MAAM,EAAE,SAAS,IAAI;AACzD,aAAO,OAAO,GAAG,MAAM,UAAU;AAAA,IACnC,GAAG,CAAC;AAEJ,UAAM,MAAM,IAAI,YAAY,UAAU;AAAA,MACpC,QAAQ,2CAAaC,OAAM,KAAK,OAAO,IAAI;AAAA,MAC3C,iBAAiB;AAAA,MACjB,mBAAmB;AAAA,MACnB,YAAY;AAAA,IACd,CAAC;AAED,QAAI,MAAM,YAAY,CAAC;AAEvB,eAAW,QAAQ,SAAS;AAC1B,YAAM,IAAI,SAAS,WAAW,KAAK,CAAC,MAAM,EAAE,SAAS,IAAI;AACzD,UAAI,CAAC,GAAG;AACN,YAAI,KAAK;AACT,cAAM,IAAI,MAAM,4DAAoB,IAAI,EAAE;AAAA,MAC5C;AAEA,YAAM,oBAAoB;AAAA,QACxB;AAAA,QACA,eAAe,IAAI;AAAA,QACnB,aAAa,IAAI,eAAe;AAAA,QAChC,WAAW;AAAA,QACX,WAAW,KAAK;AAAA,QAChB,YAAY,MAAM,IAAI,UAAU;AAAA,QAChC;AAAA,MACF,CAAC;AAAA,IACH;AACA,QAAI,KAAK;AAET,YAAQ,IAAI,EAAE;AACd,eAAW,QAAQ,SAAS;AAC1B,iBAAW,gBAAMA,OAAM,KAAK,IAAI,CAAC,mDAAW;AAAA,IAC9C;AAGA,YAAQ,IAAI;AAAA,EAAKA,OAAM,KAAK,MAAM,MAAM,CAAC,uBAAQ,QAAQ,MAAM,iDAAc,EAAE,QAAG;AAAA,EACpF,CAAC;AAEH,SAAO;AACT;;;AK/KA,OAAOC,WAAU;AACjB,OAAOC,SAAQ;AACf,OAAO,YAAY;AACnB,SAAS,WAAAC,gBAAe;;;ACHxB,IAAM,YACJ;AAEF,SAAS,qBAAqB,WAAmB;AAE/C,MACE,UAAU,WAAW,GAAG,KACxB,UAAU,WAAW,GAAG,KACxB,UAAU,WAAW,IAAI,KACzB,UAAU,WAAW,IAAI,KACzB,UAAU,WAAW,GAAG,KACxB,UAAU,WAAW,UAAU,GAC/B;AACA,WAAO;AAAA,EACT;AAGA,MAAI,UAAU,WAAW,GAAG,GAAG;AAC7B,UAAM,QAAQ,UAAU,MAAM,GAAG;AACjC,QAAI,MAAM,UAAU,EAAG,QAAO,GAAG,MAAM,CAAC,CAAC,IAAI,MAAM,CAAC,CAAC;AACrD,WAAO;AAAA,EACT;AAGA,SAAO,UAAU,MAAM,GAAG,EAAE,CAAC,KAAK;AACpC;AAEO,SAAS,+BAA+B,MAAc;AAC3D,QAAM,MAAM,oBAAI,IAAY;AAE5B,aAAW,SAAS,KAAK,SAAS,SAAS,GAAG;AAC5C,UAAM,OAAO,MAAM,CAAC,KAAK,MAAM,CAAC,KAAK,MAAM,CAAC;AAC5C,QAAI,CAAC,KAAM;AACX,UAAM,MAAM,qBAAqB,IAAI;AACrC,QAAI,CAAC,IAAK;AACV,QAAI,IAAI,GAAG;AAAA,EACb;AAGA,MAAI,OAAO,MAAM;AACjB,MAAI,OAAO,KAAK;AAEhB,SAAO,CAAC,GAAG,GAAG,EAAE,KAAK;AACvB;;;AC1CO,SAAS,kBAAkB,MAAsB;AAUpD,MAAI,UAAU;AAGd,QAAM,SAAiC;AAAA,IACnC,KAAK;AAAA,IACL,MAAM;AAAA,IACN,MAAM;AAAA,IACN,IAAI;AAAA,IACJ,IAAI;AAAA,IACJ,KAAK;AAAA,IACL,GAAG;AAAA,IACH,QAAQ;AAAA,IACR,QAAQ;AAAA,EACZ;AAEA,aAAW,CAAC,GAAG,CAAC,KAAK,OAAO,QAAQ,MAAM,GAAG;AAGzC,cAAU,QAAQ,QAAQ,IAAI,OAAO,IAAI,CAAC,YAAY,GAAG,GAAG,IAAI,CAAC,IAAI;AAErE,cAAU,QAAQ,QAAQ,IAAI,OAAO,OAAO,CAAC,KAAK,GAAG,GAAG,KAAK,CAAC,GAAG;AAAA,EACrE;AAIA,YAAU,QAAQ,QAAQ,wBAAwB,uBAAuB;AAEzE,SAAO;AACX;;;AF/BA,SAAS,cAAc,UAAkB;AACvC,QAAM,MAAMC,MAAK,QAAQ,QAAQ,EAAE,YAAY;AAC/C,SAAO;AAAA,IACL;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EACF,EAAE,SAAS,GAAG;AAChB;AAIA,SAAS,kBAAkB,UAAkB;AAC3C,MAAI,UAAUA,MAAK,QAAQ,QAAQ;AACnC,aAAU;AACR,UAAM,SAASA,MAAK,KAAK,SAAS,qBAAqB;AACvD,QAAI,OAAO,WAAW,MAAM,EAAG,QAAO;AACtC,UAAM,SAASA,MAAK,QAAQ,OAAO;AACnC,QAAI,WAAW,QAAS,QAAO;AAC/B,cAAU;AAAA,EACZ;AACF;AAEO,SAAS,eAAe;AAC7B,QAAM,MAAM,IAAIC,SAAQ,OAAO,EAC5B,YAAY,8FAA6B,EACzC,OAAO,iBAAiB,gHAAqC,EAC7D;AAAA,IACC;AAAA,IACA;AAAA,IACA;AAAA,EACF,EACC;AAAA,IACC;AAAA,IACA;AAAA,IACA;AAAA,EACF,EACC;AAAA,IACC;AAAA,IACA;AAAA,IACA,CAAC,KAAK,QAAkB;AACtB,UAAI,KAAK,GAAG;AACZ,aAAO;AAAA,IACT;AAAA,IACA,CAAC;AAAA,EACH,EACC,OAAO,OAAO,SAAS;AACtB,UAAM,UAAU,KAAK,OACjBD,MAAK,QAAQ,KAAK,IAAI,IACtB,kBAAkB,QAAQ,IAAI,CAAC;AACnC,UAAM,YAAYA,MAAK,KAAK,SAAS,KAAK,MAAM;AAChD,UAAM,UAAUA,MAAK,KAAK,SAAS,KAAK,GAAG;AAC3C,UAAM,gBAA0B,KAAK,WAAW,CAAC,GAAG;AAAA,MAAI,CAAC,MACvDA,MAAK,KAAK,SAAS,CAAC;AAAA,IACtB;AAEA,UAAM,UAAU,MAAME,IAAG,QAAQ,WAAW,EAAE,eAAe,KAAK,CAAC;AACnE,UAAM,gBAAgB,QACnB,OAAO,CAAC,MAAM,EAAE,YAAY,CAAC,EAC7B,IAAI,CAAC,MAAMF,MAAK,KAAK,WAAW,EAAE,IAAI,CAAC,EACvC,KAAK,CAAC,GAAG,MAAM,EAAE,cAAc,CAAC,CAAC;AAEpC,UAAM,aAAkC,CAAC;AAEzC,eAAW,mBAAmB,eAAe;AAC3C,YAAM,OAAOA,MAAK,SAAS,eAAe;AAC1C,YAAM,YAAY,MAAM,mBAAmB,eAAe,GAAG;AAAA,QAC3D;AAAA,MACF;AAGA,YAAM,QAAoC,CAAC;AAC3C,YAAM,SAAS,oBAAI,IAAY;AAE/B,iBAAW,WAAW,UAAU;AAC9B,cAAM,MAAMA,MACT,SAAS,iBAAiB,OAAO,EACjC,MAAMA,MAAK,GAAG,EACd,KAAK,GAAG;AACX,cAAM,UAAU,MAAME,IAAG,SAAS,SAAS,MAAM;AAEjD,cAAM,MAAMF,MAAK,QAAQ,OAAO,EAAE,YAAY;AAU9C,YAAI,QAAQ,QAAQ;AAClB,gBAAM,KAAK,EAAE,MAAM,KAAK,SAAS,QAAQ,MAAM,CAAC;AAChD,gBAAM,KAAK;AAAA,YACT,MAAM;AAAA,YACN,SAAS,kBAAkB,OAAO;AAAA,YAClC,QAAQ;AAAA,UACV,CAAC;AAAA,QACH,OAAO;AACL,gBAAM,KAAK,EAAE,MAAM,KAAK,QAAQ,CAAC;AAAA,QACnC;AAGA,YAAI,QAAQ,SAAS,QAAQ,SAAS,QAAQ,QAAQ;AACpD,qBAAW,OAAO,+BAA+B,OAAO,GAAG;AACzD,mBAAO,IAAI,GAAG;AAAA,UAChB;AAAA,QACF;AAAA,MACF;AAEA,iBAAW,KAAK;AAAA,QACd;AAAA,QACA,cAAc,CAAC,GAAG,MAAM,EAAE,KAAK;AAAA,QAC/B;AAAA,MACF,CAAC;AAAA,IACH;AAEA,UAAM,WAAyB;AAAA,MAC7B,eAAe;AAAA,MACf,cAAa,oBAAI,KAAK,GAAE,YAAY;AAAA,MACpC,QAAQ;AAAA,QACN,SAAS,QAAQ,MAAMA,MAAK,GAAG,EAAE,KAAK,GAAG;AAAA,QACzC,eAAe,KAAK;AAAA,MACtB;AAAA,MACA;AAAA,IACF;AAEA,mBAAe,SAAS,oBAA4B;AAClD,YAAM,mBAAmBA,MAAK;AAAA,QAC5BA,MAAK,QAAQ,kBAAkB;AAAA,QAC/B;AAAA,MACF;AACA,YAAM,cAAc,oBAAoB,QAAQ;AAGhD,YAAME,IAAG,MAAM,kBAAkB,EAAE,WAAW,KAAK,CAAC;AACpD,iBAAW,KAAK,YAAY;AAC1B,cAAM;AAAA,UACJF,MAAK,KAAK,kBAAkB,GAAG,EAAE,IAAI,OAAO;AAAA,UAC5C;AAAA,YACE,GAAG;AAAA,YACH,WAAW,EAAE,MAAM;AAAA,YACnB,aAAa,KAAK,KAAK,UAAU,EAAE,MAAM,IAAI,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;AAAA,UACjE;AAAA,QACF;AAAA,MACF;AAAA,IACF;AAEA,UAAM,SAAS,OAAO;AACtB,eAAW,KAAK,aAAc,OAAM,SAAS,CAAC;AAG9C,YAAQ;AAAA,MACN,oCAAgBA,MAAK,SAAS,QAAQ,IAAI,GAAG,OAAO,CAAC,SAAI,WAAW,MAAM;AAAA,IAC5E;AAAA,EACF,CAAC;AAEH,SAAO;AACT;;;AGzKA,OAAOG,WAAU;AACjB,OAAOC,SAAQ;AACf,SAAS,WAAAC,gBAAe;AACxB,OAAOC,cAAa;;;ACHpB,OAAOC,WAAU;AACjB,OAAOC,SAAQ;AACf,SAAS,iBAAAC,sBAAqB;AAI9B,SAASC,wBAAuB,MAAc;AAE5C,MAAI,MAAMC,MAAK,QAAQC,eAAc,IAAI,CAAC;AAC1C,SAAO,MAAM;AACX,QAAIC,IAAG,WAAWF,MAAK,KAAK,KAAK,cAAc,CAAC,EAAG,QAAO;AAC1D,UAAM,SAASA,MAAK,QAAQ,GAAG;AAC/B,QAAI,WAAW,IAAK,QAAO;AAC3B,UAAM;AAAA,EACR;AACF;AAEO,SAAS,sBAAsB;AAGpC,QAAM,UAAUD,wBAAuB,YAAY,GAAG;AACtD,SAAOC,MAAK,KAAK,SAAS,WAAW;AACvC;AAEA,eAAsB,mBAAmB,QAKtC;AACD,QAAM,gBAAgB,oBAAoB;AAC1C,QAAM,UAAUA,MAAK,KAAK,eAAe,OAAO,MAAM;AACtD,MAAI,CAAE,MAAM,WAAW,OAAO,EAAI,QAAO;AAEzC,QAAM,QAAQA,MAAK,KAAK,OAAO,KAAK,OAAO,SAAS;AACpD,QAAM,iBAAiB,EAAE,SAAS,OAAO,WAAW,OAAO,UAAU,CAAC;AACtE,SAAO;AACT;;;AD1BA,SAAS,qBAAqB,KAAa;AACzC,SAAO,QAAQ,IAAI;AAAA,IACjB,WAAWG,MAAK,KAAK,KAAK,gBAAgB,CAAC;AAAA,IAC3C,WAAWA,MAAK,KAAK,KAAK,gBAAgB,CAAC;AAAA,IAC3C,WAAWA,MAAK,KAAK,KAAK,iBAAiB,CAAC;AAAA,EAC9C,CAAC,EAAE,KAAK,CAAC,QAAQ,IAAI,KAAK,OAAO,CAAC;AACpC;AAQA,SAAS,UAAU,QAAiC;AAElD,OAAK;AACL,SAAO;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACT;AAEA,eAAe,kBAAkB,QAI9B;AACD,QAAM,EAAE,KAAK,WAAW,YAAY,IAAI;AACxC,MAAI,CAAC,eAAe,gBAAgB,IAAK;AACzC,QAAM,SAASA,MAAK,KAAK,KAAK,SAAS;AACvC,MAAI,CAAE,MAAM,WAAW,MAAM,EAAI;AAEjC,QAAM,QAAQ,MAAM,mBAAmB,MAAM;AAC7C,aAAW,KAAK,OAAO;AACrB,UAAM,MAAMA,MAAK,QAAQ,CAAC,EAAE,YAAY;AAExC,QAAI,CAAC,CAAC,OAAO,QAAQ,OAAO,QAAQ,QAAQ,QAAQ,MAAM,EAAE,SAAS,GAAG,EAAG;AAC3E,UAAM,MAAM,MAAMC,IAAG,SAAS,GAAG,MAAM;AACvC,UAAM,OAAO,IAAI,WAAW,MAAM,GAAG,WAAW,GAAG;AACnD,QAAI,SAAS,IAAK,OAAMA,IAAG,UAAU,GAAG,MAAM,MAAM;AAAA,EACtD;AACF;AAEO,SAAS,cAAc;AAC5B,QAAM,MAAM,IAAIC,SAAQ,MAAM,EAC3B,YAAY,sCAAQ,EACpB,OAAO,gBAAgB,wCAAU,QAAQ,IAAI,CAAC,EAC9C,OAAO,aAAa,iDAAwB,EAC5C,OAAO,SAAS,gEAAc,KAAK,EACnC,OAAO,eAAe,2FAA+B,KAAK,EAC1D;AAAA,IACC;AAAA,IACA;AAAA,IACA;AAAA,EACF,EACC;AAAA,IACC;AAAA,IACA;AAAA,IACA,cAAc,EAAE;AAAA,EAClB,EACC;AAAA,IACC;AAAA,IACA;AAAA,IACA,cAAc,EAAE;AAAA,EAClB,EACC;AAAA,IACC;AAAA,IACA;AAAA,IACA,cAAc,EAAE;AAAA,EAClB,EACC;AAAA,IACC;AAAA,IACA;AAAA,IACA,cAAc,EAAE;AAAA,EAClB,EACC,OAAO,0BAA0B,yDAA2B,cAAc,EAAE,QAAQ,EACpF,OAAO,OAAO,SAAS;AACtB,UAAM,MAAMF,MAAK,QAAQ,KAAK,GAAG;AAGjC,YAAQ;AAAA,MACN;AAAA,IACF;AAEA,UAAM,UAAUA,MAAK,KAAK,KAAK,cAAc;AAC7C,QAAI,CAAE,MAAM,WAAW,OAAO,GAAI;AAChC,YAAM,IAAI,MAAM,wCAAoB,OAAO,EAAE;AAAA,IAC/C;AAEA,UAAM,KACJ,KAAK,MAAO,MAAM,qBAAqB,GAAG;AAE5C,UAAM,MAAiB;AAAA,MACrB,eAAe;AAAA,MACf,eAAe,KAAK;AAAA,MACpB,QAAQ,KAAK;AAAA,MACb,gBAAgB,KAAK;AAAA,MACrB,aAAa,KAAK;AAAA,MAClB,UAAU,KAAK;AAAA,IACjB;AAGA,UAAM,WAAW,MAAM,iBAAiB,GAAG;AAC3C,QAAI,UAAU;AACZ,UAAI,gBAAgB,KAAK,iBAAiB,SAAS,iBAAiB,IAAI;AACxE,UAAI,SAAS,KAAK,UAAU,SAAS,UAAU,IAAI;AACnD,UAAI,iBAAiB,KAAK,kBAAkB,SAAS,kBAAkB,IAAI;AAC3E,UAAI,cAAc,KAAK,eAAe,SAAS,eAAe,IAAI;AAClE,UAAI,WAAW,KAAK,YAAY,SAAS,YAAY,IAAI;AAAA,IAC3D;AAEA,QAAI,CAAC,KAAK,KAAK;AACb,YAAM,OAAO,MAAM,qBAAqB,GAAG;AAC3C,YAAM,MAAM,MAAMG;AAAA,QAChB;AAAA,UACE;AAAA,YACE,MAAM;AAAA,YACN,MAAM;AAAA,YACN,SAAS;AAAA,YACT,SAAS,IAAI;AAAA,UACf;AAAA,UACA;AAAA,YACE,MAAM;AAAA,YACN,MAAM;AAAA,YACN,SAAS;AAAA,YACT,SAAS,IAAI;AAAA,UACf;AAAA,UACA;AAAA,YACE,MAAM;AAAA,YACN,MAAM;AAAA,YACN,SAAS;AAAA,YACT,SAAS,IAAI;AAAA,UACf;AAAA,UACA;AAAA,YACE,MAAM;AAAA,YACN,MAAM;AAAA,YACN,SAAS;AAAA,YACT,SAAS,IAAI,eAAe;AAAA,UAC9B;AAAA,QACF;AAAA,QACA;AAAA,UACE,UAAU,MAAM;AACd,kBAAM,IAAI,MAAM,oBAAK;AAAA,UACvB;AAAA,QACF;AAAA,MACF;AAEA,UAAI,gBAAgB,IAAI,iBAAiB,IAAI;AAC7C,UAAI,SAAS,IAAI,UAAU,IAAI;AAC/B,UAAI,iBAAiB,IAAI,kBAAkB,IAAI;AAC/C,UAAI,cAAc,IAAI,eAAe,IAAI;AAGzC,YAAMC,WAAUJ,MAAK,KAAK,KAAK,KAAK,MAAM;AAC1C,YAAM,cAAcI,UAAS,GAAG;AAGhC,YAAM,mBAAmB;AAAA,QACvB,QAAQ;AAAA,QACR;AAAA,QACA,WAAW,IAAI;AAAA,QACf,WAAW,KAAK;AAAA,MAClB,CAAC;AACD,YAAM,mBAAmB;AAAA,QACvB,QAAQ;AAAA,QACR;AAAA,QACA,WAAW,IAAI;AAAA,QACf,WAAW,KAAK;AAAA,MAClB,CAAC;AAED,YAAM,kBAAkB,EAAE,KAAK,WAAW,IAAI,QAAQ,aAAa,IAAI,eAAe,IAAI,CAAC;AAC3F,YAAM,kBAAkB;AAAA,QACtB;AAAA,QACA,WAAW,IAAI;AAAA,QACf,aAAa,IAAI,eAAe;AAAA,MAClC,CAAC;AAGD,YAAMC,gBAAeL,MAAK,KAAK,KAAK,IAAI,QAAQ,UAAU;AAC1D,UAAI,CAAE,MAAM,WAAWK,aAAY,EAAI,OAAM,cAAcA,eAAc,UAAU,CAAC,CAAC,CAAC;AAEtF,cAAQ,IAAI,uCAASL,MAAK,SAAS,QAAQ,IAAI,GAAGI,QAAO,CAAC,6CAAoB,EAAE,QAAG;AACnF,cAAQ,IAAI,kJAA4E;AAExF,cAAQ,IAAI,EAAE;AACd,YAAM,SAAS;AACf,YAAM,WAAW,aAAa,6CAAoB,GAAG,EAAE;AACvD;AAAA,IACF;AAGA,UAAM,UAAUJ,MAAK,KAAK,KAAK,KAAK,MAAM;AAC1C,UAAM,UAAUA,MAAK,QAAQ,OAAO,CAAC;AACrC,UAAMC,IAAG,UAAU,SAAS,KAAK,UAAU,KAAK,MAAM,CAAC,IAAI,MAAM,MAAM;AAGvE,UAAM,mBAAmB;AAAA,MACvB,QAAQ;AAAA,MACR;AAAA,MACA,WAAW,IAAI;AAAA,MACf,WAAW,KAAK;AAAA,IAClB,CAAC;AACD,UAAM,mBAAmB;AAAA,MACvB,QAAQ;AAAA,MACR;AAAA,MACA,WAAW,IAAI;AAAA,MACf,WAAW,KAAK;AAAA,IAClB,CAAC;AAED,UAAM,kBAAkB,EAAE,KAAK,WAAW,IAAI,QAAQ,aAAa,IAAI,eAAe,IAAI,CAAC;AAC3F,UAAM,kBAAkB;AAAA,MACtB;AAAA,MACA,WAAW,IAAI;AAAA,MACf,aAAa,IAAI,eAAe;AAAA,IAClC,CAAC;AAGD,UAAM,eAAeD,MAAK,KAAK,KAAK,IAAI,QAAQ,UAAU;AAC1D,QAAI,CAAE,MAAM,WAAW,YAAY,EAAI,OAAM,cAAc,cAAc,UAAU,CAAC,CAAC,CAAC;AAGtF,YAAQ,IAAI,iCAAQA,MAAK,SAAS,QAAQ,IAAI,GAAG,OAAO,CAAC,6CAAoB,EAAE,QAAG;AAClF,YAAQ,IAAI,kJAA4E;AAExF,YAAQ,IAAI,EAAE;AACd,UAAM,SAAS;AACf,UAAM,WAAW,aAAa,6CAAoB,GAAG,CAAC;AAAA,EACxD,CAAC;AAEH,SAAO;AACT;;;AE/OA;AAAA,EACE,MAAQ;AAAA,EACR,SAAW;AAAA,EACX,aAAe;AAAA,EACf,QAAU;AAAA,EACV,SAAW;AAAA,EACX,YAAc;AAAA,IACZ,MAAQ;AAAA,IACR,KAAO;AAAA,EACT;AAAA,EACA,MAAQ;AAAA,EACR,KAAO;AAAA,IACL,aAAa;AAAA,EACf;AAAA,EACA,OAAS;AAAA,IACP;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EACF;AAAA,EACA,eAAiB;AAAA,IACf,QAAU;AAAA,EACZ;AAAA,EACA,SAAW;AAAA,IACT,KAAO;AAAA,IACP,OAAS;AAAA,IACT,KAAO;AAAA,IACP,OAAS;AAAA,IACT,SAAW;AAAA,IACX,kBAAkB;AAAA,IAClB,SAAW;AAAA,IACX,gBAAkB;AAAA,EACpB;AAAA,EACA,cAAgB;AAAA,IACd,OAAS;AAAA,IACT,gBAAgB;AAAA,IAChB,WAAa;AAAA,IACb,OAAS;AAAA,IACT,QAAU;AAAA,IACV,KAAO;AAAA,IACP,SAAW;AAAA,EACb;AAAA,EACA,iBAAmB;AAAA,IACjB,uBAAuB;AAAA,IACvB,iBAAiB;AAAA,IACjB,eAAe;AAAA,IACf,kBAAkB;AAAA,IAClB,QAAU;AAAA,IACV,MAAQ;AAAA,IACR,KAAO;AAAA,IACP,YAAc;AAAA,EAChB;AACF;;;AX9CA,IAAM,UAAU,IAAIM,SAAQ,EACzB,KAAK,WAAW,EAChB,YAAY,sEAAmC,EAC/C,QAAQ,gBAAY,OAAO;AAE9B,QAAQ,WAAW,YAAY,CAAC;AAChC,QAAQ,WAAW,WAAW,CAAC;AAC/B,QAAQ,WAAW,aAAa,CAAC;AAEjC,MAAM,QAAQ,WAAW,QAAQ,IAAI;","names":["Command","path","fs","path","path","path","fs","path","fs","require","chalk","path","chalk","path","fs","Command","path","Command","fs","path","fs","Command","prompts","path","fs","fileURLToPath","findNearestPackageRoot","path","fileURLToPath","fs","path","fs","Command","prompts","cfgPath","libUtilsPath","Command"]}
1
+ {"version":3,"sources":["../src/index.ts","../src/commands/add.ts","../src/utils/fs.ts","../src/utils/pm.ts","../src/utils/registry.ts","../src/utils/ui.ts","../src/commands/build.ts","../src/utils/imports.ts","../src/utils/transformers.ts","../src/commands/init.ts","../src/utils/templates.ts","../package.json"],"sourcesContent":["import { Command } from \"commander\";\r\nimport { addCommand } from \"./commands/add.js\";\r\nimport { buildCommand } from \"./commands/build.js\";\r\nimport { initCommand } from \"./commands/init.js\";\r\nimport packageJson from \"../package.json\" assert { type: \"json\" };\r\n\r\nconst program = new Command()\r\n .name(\"reborn-ui\")\r\n .description(\"Reborn UI - 组件安装与 registry 辅助 CLI\")\r\n .version(packageJson.version);\r\n\r\nprogram.addCommand(initCommand());\r\nprogram.addCommand(addCommand());\r\nprogram.addCommand(buildCommand());\r\n\r\nawait program.parseAsync(process.argv);\r\n\r\n\r\n","import path from \"node:path\";\r\nimport { Command } from \"commander\";\r\nimport prompts from \"prompts\";\r\nimport type { PackageManager, RegistryComponent } from \"../types.js\";\r\nimport { ensureDir, pathExists, writeTextFile } from \"../utils/fs.js\";\r\nimport {\r\n detectPackageManager,\r\n readPackageJson,\r\n} from \"../utils/pm.js\";\r\nimport { defaultConfig, loadConfigCompat, loadRegistry } from \"../utils/registry.js\";\r\nimport cliProgress from \"cli-progress\";\r\nimport chalk from \"chalk\";\r\nimport { successLog } from \"../utils/ui.js\";\r\n\r\nfunction rewriteImports(params: {\r\n content: string;\r\n aliasSymbol: string;\r\n}) {\r\n const { content, aliasSymbol } = params;\r\n if (!aliasSymbol || aliasSymbol === \"@\") return content;\r\n\r\n // 仅替换项目根目录映射符号:把 \"@/xxx\" -> \"<aliasSymbol>/xxx\"\r\n // 注意:不要影响 scoped package(@scope/name),这里只替换 \"@/\" 前缀。\r\n return content.replaceAll(\"@/\", `${aliasSymbol}/`);\r\n}\r\n\r\nasync function writeComponentFiles(params: {\r\n cwd: string;\r\n componentsDir: string;\r\n aliasSymbol: string;\r\n component: RegistryComponent;\r\n overwrite?: boolean;\r\n onProgress?: () => void;\r\n platform: \"web\" | \"uniapp\";\r\n}) {\r\n const { cwd, componentsDir, aliasSymbol, component, overwrite, onProgress, platform } = params;\r\n const baseDir = path.join(cwd, componentsDir, component.name);\r\n await ensureDir(baseDir);\r\n\r\n for (const f of component.files) {\r\n // 过滤逻辑:\r\n // 1. 如果 f.target 存在且不等于 platform,跳过\r\n // 2. 如果 f.target 不存在,视为通用文件,不跳过\r\n if (f.target && f.target !== platform) {\r\n continue;\r\n }\r\n\r\n const target = path.join(baseDir, ...f.path.split(\"/\"));\r\n if (!overwrite && (await pathExists(target))) {\r\n onProgress?.();\r\n continue;\r\n }\r\n const nextContent = rewriteImports({ content: f.content, aliasSymbol });\r\n await writeTextFile(target, nextContent);\r\n onProgress?.();\r\n // 增加一个极小的延迟,让进度条看起来在“跑”\r\n await new Promise(r => setTimeout(r, 10));\r\n }\r\n\r\n return baseDir;\r\n}\r\n\r\nexport function addCommand() {\r\n const cmd = new Command(\"add\")\r\n .description(\"向项目中添加组件与相关依赖\")\r\n .argument(\"[components...]\", \"组件名(可多个)\")\r\n .option(\"--cwd <path>\", \"目标项目目录\", process.cwd())\r\n .option(\"--pm <pm>\", \"包管理器:pnpm|npm|yarn|bun\")\r\n .option(\"--yes\", \"跳过交互\", false)\r\n .option(\"--overwrite\", \"覆盖已存在文件\", false)\r\n .option(\"--config <path>\", \"配置文件路径(相对 cwd)\", \"components.json\")\r\n .option(\"--registry <pkgOrPath>\", \"覆盖配置里的 registry\")\r\n .option(\"--components-dir <path>\", \"覆盖配置里的 componentsDir\")\r\n .option(\"--lib-dir <path>\", \"覆盖配置里的 libDir\")\r\n .option(\"--alias-symbol <symbol>\", \"覆盖配置里的 aliasSymbol(默认 @)\")\r\n .action(async (components: string[], opts) => {\r\n const cwd = path.resolve(opts.cwd);\r\n const pm: PackageManager =\r\n opts.pm ?? (await detectPackageManager(cwd));\r\n\r\n const cfg = (await loadConfigCompat(cwd, opts.config)) ?? defaultConfig();\r\n if (opts.registry) cfg.registry = opts.registry;\r\n if (opts.componentsDir) cfg.componentsDir = opts.componentsDir;\r\n if (opts.libDir) cfg.libDir = opts.libDir;\r\n if (opts.aliasSymbol) cfg.aliasSymbol = opts.aliasSymbol;\r\n\r\n const registry = await loadRegistry({ cwd, registry: cfg.registry });\r\n\r\n let targets = components ?? [];\r\n if (!targets.length) {\r\n if (opts.yes) {\r\n throw new Error(\"未指定组件名;请传入组件参数或去掉 --yes 以交互选择。\");\r\n }\r\n const choices = registry.components.map((c) => ({\r\n title: c.name,\r\n value: c.name,\r\n }));\r\n const res = await prompts(\r\n [\r\n {\r\n type: \"multiselect\",\r\n name: \"selected\",\r\n message: \"选择要添加的组件\",\r\n choices,\r\n min: 1,\r\n },\r\n ],\r\n {\r\n onCancel: () => {\r\n throw new Error(\"已取消\");\r\n },\r\n },\r\n );\r\n targets = res.selected ?? [];\r\n }\r\n\r\n // 询问平台 (Web / UniApp)\r\n const { platform } = await prompts({\r\n type: \"select\",\r\n name: \"platform\",\r\n message: \"选择目标平台\",\r\n choices: [\r\n { title: \"Web (默认)\", value: \"web\" },\r\n { title: \"UniApp\", value: \"uniapp\" },\r\n ],\r\n initial: 0,\r\n });\r\n\r\n if (!platform) {\r\n throw new Error(\"已取消\");\r\n }\r\n\r\n const totalFiles = targets.reduce((acc, name) => {\r\n const c = registry.components.find((x) => x.name === name);\r\n return acc + (c?.files.length ?? 0);\r\n }, 0);\r\n\r\n const bar = new cliProgress.SingleBar({\r\n format: '正在写入文件 |' + chalk.cyan('{bar}') + '| {percentage}% || {value}/{total} 文件',\r\n barCompleteChar: '\\u2588',\r\n barIncompleteChar: '\\u2591',\r\n hideCursor: true\r\n });\r\n\r\n bar.start(totalFiles, 0);\r\n\r\n for (const name of targets) {\r\n const c = registry.components.find((x) => x.name === name);\r\n if (!c) {\r\n bar.stop();\r\n throw new Error(`registry 中不存在该组件:${name}`);\r\n }\r\n\r\n await writeComponentFiles({\r\n cwd,\r\n componentsDir: cfg.componentsDir,\r\n aliasSymbol: cfg.aliasSymbol ?? \"@\",\r\n component: c,\r\n overwrite: opts.overwrite,\r\n onProgress: () => bar.increment(),\r\n platform,\r\n });\r\n }\r\n bar.stop();\r\n\r\n console.log(\"\");\r\n for (const name of targets) {\r\n successLog(`组件 ${chalk.bold(name)} 已成功添加到项目`);\r\n }\r\n\r\n // eslint-disable-next-line no-console\r\n console.log(`\\n${chalk.bold.green('DONE')} 已完成 ${targets.length} 个组件的添加(pm=${pm})`);\r\n });\r\n\r\n return cmd;\r\n}\r\n\r\n\r\n","import { createHash } from \"node:crypto\";\nimport fs from \"node:fs/promises\";\nimport path from \"node:path\";\n\nexport async function pathExists(p: string) {\n try {\n await fs.access(p);\n return true;\n } catch {\n return false;\n }\n}\n\nexport async function ensureDir(dirPath: string) {\n await fs.mkdir(dirPath, { recursive: true });\n}\n\nexport async function readJsonFile<T>(filePath: string): Promise<T> {\n const raw = await fs.readFile(filePath, \"utf8\");\n return JSON.parse(raw) as T;\n}\n\nexport async function writeJsonFile(filePath: string, data: unknown) {\n await ensureDir(path.dirname(filePath));\n await fs.writeFile(filePath, JSON.stringify(data, null, 2) + \"\\n\", \"utf8\");\n}\n\nexport async function writeTextFile(filePath: string, content: string) {\n await ensureDir(path.dirname(filePath));\n await fs.writeFile(filePath, content, \"utf8\");\n}\n\nexport async function listFilesRecursive(\n dirPath: string,\n opts?: { ignoreDirNames?: Set<string> },\n) {\n const ignoreDirNames =\n opts?.ignoreDirNames ??\n new Set([\"node_modules\", \".git\", \".nuxt\", \"dist\", \".output\"]);\n\n const out: string[] = [];\n\n async function walk(current: string) {\n const entries = await fs.readdir(current, { withFileTypes: true });\n for (const entry of entries) {\n const p = path.join(current, entry.name);\n if (entry.isDirectory()) {\n if (ignoreDirNames.has(entry.name)) continue;\n await walk(p);\n } else if (entry.isFile()) {\n out.push(p);\n }\n }\n }\n\n await walk(dirPath);\n return out;\n}\n\nexport async function copyDirRecursive(params: {\n fromDir: string;\n toDir: string;\n overwrite?: boolean;\n ignoreFileNames?: Set<string>;\n}) {\n const { fromDir, toDir, overwrite } = params;\n const ignoreFileNames = params.ignoreFileNames ?? new Set([\".DS_Store\"]);\n\n const entries = await fs.readdir(fromDir, { withFileTypes: true });\n await ensureDir(toDir);\n\n for (const entry of entries) {\n if (ignoreFileNames.has(entry.name)) continue;\n const from = path.join(fromDir, entry.name);\n const to = path.join(toDir, entry.name);\n\n if (entry.isDirectory()) {\n await copyDirRecursive({ fromDir: from, toDir: to, overwrite, ignoreFileNames });\n continue;\n }\n\n if (!entry.isFile()) continue;\n\n if (!overwrite) {\n try {\n await fs.access(to);\n continue; // 已存在则跳过\n } catch {\n // not exists\n }\n }\n\n await ensureDir(path.dirname(to));\n await fs.copyFile(from, to);\n }\n}\n\nexport function sha1(text: string) {\n return createHash(\"sha1\").update(text).digest(\"hex\");\n}\n\n\n","import fs from \"node:fs/promises\";\r\nimport path from \"node:path\";\r\nimport { execa } from \"execa\";\r\nimport type { PackageManager } from \"../types.js\";\r\nimport { pathExists, readJsonFile, writeJsonFile } from \"./fs.js\";\r\n\r\nexport async function detectPackageManager(cwd: string): Promise<PackageManager> {\r\n if (await pathExists(path.join(cwd, \"pnpm-lock.yaml\"))) return \"pnpm\";\r\n if (await pathExists(path.join(cwd, \"yarn.lock\"))) return \"yarn\";\r\n if (await pathExists(path.join(cwd, \"package-lock.json\"))) return \"npm\";\r\n if (await pathExists(path.join(cwd, \"bun.lockb\"))) return \"bun\";\r\n return \"pnpm\";\r\n}\r\n\r\nexport async function readPackageJson(\r\n cwd: string,\r\n): Promise<Record<string, any> & { dependencies?: Record<string, string>; devDependencies?: Record<string, string> }> {\r\n return await readJsonFile(path.join(cwd, \"package.json\"));\r\n}\r\n\r\nexport async function writePackageJson(cwd: string, pkg: unknown) {\r\n await writeJsonFile(path.join(cwd, \"package.json\"), pkg);\r\n}\r\n\r\nexport function getMissingDeps(\r\n pkg: { dependencies?: Record<string, string>; devDependencies?: Record<string, string> },\r\n deps: string[],\r\n) {\r\n const existing = new Set([\r\n ...Object.keys(pkg.dependencies ?? {}),\r\n ...Object.keys(pkg.devDependencies ?? {}),\r\n ]);\r\n return deps.filter((d) => !existing.has(d));\r\n}\r\n\r\nexport async function installDeps(params: {\r\n cwd: string;\r\n pm: PackageManager;\r\n deps: string[];\r\n dev?: boolean;\r\n}) {\r\n const { cwd, pm, deps, dev } = params;\r\n if (!deps.length) return;\r\n\r\n const args: string[] = [];\r\n if (pm === \"pnpm\") args.push(\"add\");\r\n else if (pm === \"npm\") args.push(\"install\");\r\n else if (pm === \"yarn\") args.push(\"add\");\r\n else if (pm === \"bun\") args.push(\"add\");\r\n\r\n if (dev) {\r\n if (pm === \"npm\") args.push(\"--save-dev\");\r\n else args.push(\"-D\");\r\n }\r\n\r\n args.push(...deps);\r\n\r\n await execa(pm, args, { cwd, stdio: \"inherit\" });\r\n}\r\n\r\nexport async function ensureJsonFile(cwd: string, relPath: string, defaultJson: unknown) {\r\n const p = path.join(cwd, relPath);\r\n if (await pathExists(p)) return;\r\n await fs.mkdir(path.dirname(p), { recursive: true });\r\n await fs.writeFile(p, JSON.stringify(defaultJson, null, 2) + \"\\n\", \"utf8\");\r\n}\r\n\r\n\r\n","import path from \"node:path\";\nimport fs from \"node:fs\";\nimport { createRequire } from \"node:module\";\nimport { fileURLToPath } from \"node:url\";\nimport type { CliConfig, RegistryFile } from \"../types.js\";\nimport { pathExists, readJsonFile } from \"./fs.js\";\n\nexport const DEFAULT_CONFIG_PATH = \"components.json\";\n\nfunction findNearestPackageRoot(from: string) {\n // from: import.meta.url\n let dir = path.dirname(fileURLToPath(from));\n while (true) {\n if (fs.existsSync(path.join(dir, \"package.json\"))) return dir;\n const parent = path.dirname(dir);\n if (parent === dir) return dir;\n dir = parent;\n }\n}\n\nexport function defaultConfig(): CliConfig {\n return {\n schemaVersion: 1,\n componentsDir: \"components\",\n libDir: \"lib\",\n composablesDir: \"composables\",\n // 默认使用 CLI 包内置的 registry(随 reborn-ui 一起发布)\n registry: \"builtin\",\n // 项目根目录映射别名符号(默认 @)\n aliasSymbol: \"@\",\n };\n}\n\nexport async function loadConfig(cwd: string, configPath = DEFAULT_CONFIG_PATH) {\n const abs = path.isAbsolute(configPath)\n ? configPath\n : path.join(cwd, configPath);\n if (!(await pathExists(abs))) return null;\n return await readJsonFile<CliConfig>(abs);\n}\n\nexport function resolveRegistryJsonPath(params: { cwd: string; registry: string }) {\n const { cwd, registry } = params;\n\n // 0) 内置 registry\n if (!registry || registry === \"builtin\") {\n // 注意:CLI 经过打包后可能变成单文件(例如 dist/index.js),\n // import.meta.url 的相对基准会变化,不能用固定的 ../../registry/...\n // 这里通过向上查找最近的 package.json 来定位包根目录。\n const pkgRoot = findNearestPackageRoot(import.meta.url);\n return path.join(pkgRoot, \"registry\", \"registry.json\");\n }\n\n // 1) 如果是文件路径(包含分隔符或 .json),优先当成路径处理\n const looksLikePath =\n registry.includes(\"/\") ||\n registry.includes(\"\\\\\") ||\n registry.endsWith(\".json\");\n\n if (looksLikePath) {\n const abs = path.isAbsolute(registry) ? registry : path.join(cwd, registry);\n return abs;\n }\n\n // 2) 当成包名处理:默认读取 <pkg>/registry/registry.json\n const require = createRequire(import.meta.url);\n // 不显式绑定 cwd,避免 npx 场景下找不到 registry 包\n return require.resolve(`${registry}/registry/registry.json`);\n}\n\nexport async function loadRegistry(params: { cwd: string; registry: string }) {\n const jsonPath = resolveRegistryJsonPath(params);\n return await readJsonFile<RegistryFile>(jsonPath);\n}\n\n// 兼容旧配置文件名:shadcn-docs.json\nexport async function loadConfigCompat(cwd: string, configPath?: string) {\n if (configPath) return await loadConfig(cwd, configPath);\n const primary = await loadConfig(cwd, DEFAULT_CONFIG_PATH);\n if (primary) return primary;\n return await loadConfig(cwd, \"shadcn-docs.json\");\n}\n\n\n","import chalk from 'chalk';\r\nimport figlet from 'figlet';\r\n\r\n/**\r\n * 模拟打字机效果\r\n * @param text 要显示的文本\r\n * @param delay 每字符延迟时间 (ms)\r\n * @param newline 是否在结束时换行\r\n */\r\nexport async function typewriter(text: string, delay: number = 15, newline: boolean = true) {\r\n for (const char of text) {\r\n process.stdout.write(char);\r\n if (delay > 0) {\r\n await new Promise((resolve) => setTimeout(resolve, delay));\r\n }\r\n }\r\n if (newline) {\r\n process.stdout.write('\\n');\r\n }\r\n}\r\n\r\n/**\r\n * 显示 ASCII Logo\r\n */\r\nexport async function showLogo(text: string = 'Reborn UI') {\r\n return new Promise<void>((resolve) => {\r\n figlet.text(text, {\r\n font: 'Slant',\r\n horizontalLayout: 'default',\r\n verticalLayout: 'default',\r\n width: 80,\r\n whitespaceBreak: true\r\n }, async (err, data) => {\r\n if (err) {\r\n console.log(gradientText(text));\r\n resolve();\r\n return;\r\n }\r\n if (data) {\r\n const lines = data.split('\\n');\r\n for (const line of lines) {\r\n console.log(gradientText(line));\r\n await new Promise(r => setTimeout(r, 20)); // 每行显示间隔,增加动感\r\n }\r\n }\r\n resolve();\r\n });\r\n });\r\n}\r\n\r\n/**\r\n * 带有样式的成功提示\r\n */\r\nexport function successLog(message: string) {\r\n console.log(`${chalk.green('✔')} ${message}`);\r\n}\r\n\r\n/**\r\n * 带有样式的警告提示\r\n */\r\nexport function warnLog(message: string) {\r\n console.log(`${chalk.yellow('⚠')} ${message}`);\r\n}\r\n\r\n/**\r\n * 带有样式的错误提示\r\n */\r\nexport function errorLog(message: string) {\r\n console.log(`${chalk.red('✘')} ${message}`);\r\n}\r\n\r\n/**\r\n * 渐变色文本 (简单模拟,不依赖第三方复杂渐变库)\r\n */\r\nexport function gradientText(text: string) {\r\n const colors = [\r\n chalk.hex('#FF0080'),\r\n chalk.hex('#FF8C00'),\r\n chalk.hex('#40E0D0'),\r\n chalk.hex('#0080FF'),\r\n chalk.hex('#7B68EE'),\r\n ];\r\n\r\n let result = '';\r\n for (let i = 0; i < text.length; i++) {\r\n const color = colors[i % colors.length];\r\n result += color!(text[i]);\r\n }\r\n return result;\r\n}\r\n","import path from \"node:path\";\r\nimport fs from \"node:fs/promises\";\r\nimport fssync from \"node:fs\";\r\nimport { Command } from \"commander\";\r\nimport type { RegistryComponent, RegistryFile } from \"../types.js\";\r\nimport { listFilesRecursive, sha1, writeJsonFile } from \"../utils/fs.js\";\r\nimport { extractNpmDependenciesFromText } from \"../utils/imports.js\";\r\n\r\nfunction isAllowedFile(filePath: string) {\r\n const ext = path.extname(filePath).toLowerCase();\r\n return [\r\n \".vue\",\r\n \".ts\",\r\n \".js\",\r\n \".json\",\r\n \".css\",\r\n \".md\",\r\n \".svg\",\r\n ].includes(ext);\r\n}\r\n\r\nimport { transformToUniapp } from \"../utils/transformers.js\";\r\n\r\nfunction findWorkspaceRoot(startDir: string) {\r\n let current = path.resolve(startDir);\r\n for (; ;) {\r\n const marker = path.join(current, \"pnpm-workspace.yaml\");\r\n if (fssync.existsSync(marker)) return current;\r\n const parent = path.dirname(current);\r\n if (parent === current) return startDir;\r\n current = parent;\r\n }\r\n}\r\n\r\nexport function buildCommand() {\r\n const cmd = new Command(\"build\")\r\n .description(\"(内部)扫描组件源码并生成 registry JSON\")\r\n .option(\"--root <path>\", \"仓库根目录(默认自动向上查找 pnpm-workspace.yaml)\")\r\n .option(\r\n \"--source <path>\",\r\n \"组件源码目录(相对 root)\",\r\n \"app/components/reborn/ui\",\r\n )\r\n .option(\r\n \"--out <path>\",\r\n \"输出 registry.json 路径(相对 root)\",\r\n \"packages/cli/registry/registry.json\",\r\n )\r\n .option(\r\n \"--also-out <path>\",\r\n \"额外再输出一份 registry.json(可重复传参)\",\r\n (val, acc: string[]) => {\r\n acc.push(val);\r\n return acc;\r\n },\r\n [],\r\n )\r\n .action(async (opts) => {\r\n const rootDir = opts.root\r\n ? path.resolve(opts.root)\r\n : findWorkspaceRoot(process.cwd());\r\n const sourceDir = path.join(rootDir, opts.source);\r\n const outPath = path.join(rootDir, opts.out);\r\n const alsoOutPaths: string[] = (opts.alsoOut ?? []).map((p: string) =>\r\n path.join(rootDir, p),\r\n );\r\n\r\n const dirents = await fs.readdir(sourceDir, { withFileTypes: true });\r\n const componentDirs = dirents\r\n .filter((d) => d.isDirectory())\r\n .map((d) => path.join(sourceDir, d.name))\r\n .sort((a, b) => a.localeCompare(b));\r\n\r\n const components: RegistryComponent[] = [];\r\n\r\n for (const absComponentDir of componentDirs) {\r\n const name = path.basename(absComponentDir);\r\n const absFiles = (await listFilesRecursive(absComponentDir)).filter(\r\n isAllowedFile,\r\n );\r\n\r\n\r\n const files: RegistryComponent[\"files\"] = [];\r\n const depSet = new Set<string>();\r\n\r\n for (const absFile of absFiles) {\r\n const rel = path\r\n .relative(absComponentDir, absFile)\r\n .split(path.sep)\r\n .join(\"/\");\r\n const content = await fs.readFile(absFile, \"utf8\");\r\n\r\n const ext = path.extname(absFile).toLowerCase();\r\n\r\n // 1. Web 版本(原样)\r\n // 如果是 .vue,明确标记 target=\"web\"\r\n // 如果是其他文件(.ts/.js),默认不标记(表示通用),或者也可以标记 web。\r\n // 策略:.vue 必须要分 web/uniapp;.ts/.js 视情况,这里先默认通用(不标记 target),\r\n // 除非你需要 uniapp 特有的 js。\r\n // 简化起见:.vue 文件生成两份,一份 target=web,一份 target=uniapp\r\n // 其他文件生成一份,无 target(通用)。\r\n\r\n if (ext === \".vue\") {\r\n files.push({ path: rel, content, target: \"web\" });\r\n files.push({\r\n path: rel,\r\n content: transformToUniapp(content),\r\n target: \"uniapp\",\r\n });\r\n } else {\r\n files.push({ path: rel, content });\r\n }\r\n\r\n // 只从代码文件里抽依赖\r\n if (ext === \".ts\" || ext === \".js\" || ext === \".vue\") {\r\n for (const dep of extractNpmDependenciesFromText(content)) {\r\n depSet.add(dep);\r\n }\r\n }\r\n }\r\n\r\n components.push({\r\n name,\r\n dependencies: [...depSet].sort(),\r\n files,\r\n });\r\n }\r\n\r\n const registry: RegistryFile = {\r\n schemaVersion: 1,\r\n generatedAt: new Date().toISOString(),\r\n source: {\r\n rootDir: rootDir.split(path.sep).join(\"/\"),\r\n componentsDir: opts.source,\r\n },\r\n components,\r\n };\r\n\r\n async function writeOut(targetRegistryPath: string) {\r\n const outComponentsDir = path.join(\r\n path.dirname(targetRegistryPath),\r\n \"components\",\r\n );\r\n await writeJsonFile(targetRegistryPath, registry);\r\n\r\n // 额外输出每个组件的 json,方便调试/按需读取\r\n await fs.mkdir(outComponentsDir, { recursive: true });\r\n for (const c of components) {\r\n await writeJsonFile(\r\n path.join(outComponentsDir, `${c.name}.json`),\r\n {\r\n ...c,\r\n fileCount: c.files.length,\r\n contentHash: sha1(JSON.stringify(c.files.map((f) => f.content))),\r\n },\r\n );\r\n }\r\n }\r\n\r\n await writeOut(outPath);\r\n for (const p of alsoOutPaths) await writeOut(p);\r\n\r\n // eslint-disable-next-line no-console\r\n console.log(\r\n `registry 已生成:${path.relative(process.cwd(), outPath)}(${components.length} 个组件)`,\r\n );\r\n });\r\n\r\n return cmd;\r\n}\r\n\r\n\r\n","const IMPORT_RE =\r\n /\\bfrom\\s+[\"']([^\"']+)[\"']|\\bimport\\(\\s*[\"']([^\"']+)[\"']\\s*\\)|\\brequire\\(\\s*[\"']([^\"']+)[\"']\\s*\\)/g;\r\n\r\nfunction normalizePackageName(specifier: string) {\r\n // ignore relative/alias/virtual\r\n if (\r\n specifier.startsWith(\".\") ||\r\n specifier.startsWith(\"/\") ||\r\n specifier.startsWith(\"@/\") ||\r\n specifier.startsWith(\"~/\") ||\r\n specifier.startsWith(\"#\") ||\r\n specifier.startsWith(\"virtual:\")\r\n ) {\r\n return null;\r\n }\r\n\r\n // scoped package: @scope/name[/...]\r\n if (specifier.startsWith(\"@\")) {\r\n const parts = specifier.split(\"/\");\r\n if (parts.length >= 2) return `${parts[0]}/${parts[1]}`;\r\n return specifier;\r\n }\r\n\r\n // normal package: name[/...]\r\n return specifier.split(\"/\")[0] ?? null;\r\n}\r\n\r\nexport function extractNpmDependenciesFromText(text: string) {\r\n const out = new Set<string>();\r\n\r\n for (const match of text.matchAll(IMPORT_RE)) {\r\n const spec = match[1] ?? match[2] ?? match[3];\r\n if (!spec) continue;\r\n const pkg = normalizePackageName(spec);\r\n if (!pkg) continue;\r\n out.add(pkg);\r\n }\r\n\r\n // 常见“框架自带/不应自动安装”的虚拟模块\r\n out.delete(\"nuxt\");\r\n out.delete(\"vue\");\r\n\r\n return [...out].sort();\r\n}\r\n\r\n\r\n","\r\nexport function transformToUniapp(code: string): string {\r\n // 1. 标签替换\r\n // div -> view\r\n // span, font -> text(注意:span可能嵌套,正则要注意)\r\n // ul, li -> view\r\n // img -> image\r\n // a -> navigator\r\n // select -> picker\r\n // iframe -> web-view\r\n\r\n let newCode = code;\r\n\r\n // 简单正则替换标签名\r\n const tagMap: Record<string, string> = {\r\n div: \"view\",\r\n span: \"text\",\r\n font: \"text\",\r\n ul: \"view\",\r\n li: \"view\",\r\n img: \"image\",\r\n a: \"navigator\",\r\n select: \"picker\",\r\n iframe: \"web-view\",\r\n };\r\n\r\n for (const [k, v] of Object.entries(tagMap)) {\r\n // 匹配 <tag ...> 和 </tag>\r\n // 1. <tag\r\n newCode = newCode.replace(new RegExp(`<${k}(\\\\s+|>)`, \"g\"), `<${v}$1`);\r\n // 2. </tag>\r\n newCode = newCode.replace(new RegExp(`<\\\\/${k}>`, \"g\"), `</${v}>`);\r\n }\r\n\r\n // 2. 特殊处理\r\n // input[type=\"search\"] -> type=\"text\" confirm-type=\"search\"\r\n newCode = newCode.replace(/type=[\"']search[\"']/g, 'confirm-type=\"search\"');\r\n\r\n return newCode;\r\n}\r\n","import path from \"node:path\";\r\nimport fs from \"node:fs/promises\";\r\nimport { Command } from \"commander\";\r\nimport prompts from \"prompts\";\r\nimport type { CliConfig, PackageManager } from \"../types.js\";\r\nimport { ensureDir, listFilesRecursive, pathExists, writeJsonFile, writeTextFile } from \"../utils/fs.js\";\r\nimport { detectPackageManager, readPackageJson } from \"../utils/pm.js\";\r\nimport { DEFAULT_CONFIG_PATH, defaultConfig, loadConfigCompat } from \"../utils/registry.js\";\r\nimport { copyTemplateSubdir } from \"../utils/templates.js\";\r\nimport { gradientText, showLogo, typewriter } from \"../utils/ui.js\";\r\n\r\nfunction looksLikeNuxtProject(cwd: string) {\r\n return Promise.all([\r\n pathExists(path.join(cwd, \"nuxt.config.ts\")),\r\n pathExists(path.join(cwd, \"nuxt.config.js\")),\r\n pathExists(path.join(cwd, \"nuxt.config.mjs\")),\r\n ]).then((arr) => arr.some(Boolean));\r\n}\r\n\r\n\r\n// function makeDefaultTailwindConfig removed in favor of template file\r\n\r\n\r\n\r\n\r\nfunction cnUtilsTs(params: { importPath?: string }) {\r\n // importPath 预留:如果用户想从别处导入 clsx/twMerge\r\n void params;\r\n return `import { type ClassValue, clsx } from \"clsx\";\\nimport { twMerge } from \"tailwind-merge\";\\n\\nexport function cn(...inputs: ClassValue[]) {\\n return twMerge(clsx(inputs));\\n}\\n`;\r\n}\r\n\r\nasync function rewriteAliasInDir(params: {\r\n cwd: string;\r\n targetDir: string;\r\n aliasSymbol: string;\r\n}) {\r\n const { cwd, targetDir, aliasSymbol } = params;\r\n if (!aliasSymbol || aliasSymbol === \"@\") return;\r\n const absDir = path.join(cwd, targetDir);\r\n if (!(await pathExists(absDir))) return;\r\n\r\n const files = await listFilesRecursive(absDir);\r\n for (const f of files) {\r\n const ext = path.extname(f).toLowerCase();\r\n // 只处理常见文本文件,避免误改二进制\r\n if (![\".ts\", \".tsx\", \".js\", \".jsx\", \".vue\", \".mjs\", \".cjs\"].includes(ext)) continue;\r\n const raw = await fs.readFile(f, \"utf8\");\r\n const next = raw.replaceAll(\"@/\", `${aliasSymbol}/`);\r\n if (next !== raw) await fs.writeFile(f, next, \"utf8\");\r\n }\r\n}\r\n\r\nexport function initCommand() {\r\n const cmd = new Command(\"init\")\r\n .description(\"初始化新项目\")\r\n .option(\"--cwd <path>\", \"目标项目目录\", process.cwd())\r\n .option(\"--pm <pm>\", \"包管理器:pnpm|npm|yarn|bun\")\r\n .option(\"--yes\", \"跳过交互,使用默认值\", false)\r\n .option(\"--overwrite\", \"覆盖已存在的模板文件(lib/composables)\", false)\r\n .option(\r\n \"--config <path>\",\r\n \"配置文件路径(相对 cwd)\",\r\n DEFAULT_CONFIG_PATH,\r\n )\r\n .option(\r\n \"--components-dir <path>\",\r\n \"组件写入目录\",\r\n defaultConfig().componentsDir,\r\n )\r\n .option(\r\n \"--lib-dir <path>\",\r\n \"lib 目录(cn/utils)\",\r\n defaultConfig().libDir,\r\n )\r\n .option(\r\n \"--composables-dir <path>\",\r\n \"composables 目录\",\r\n defaultConfig().composablesDir,\r\n )\r\n .option(\r\n \"--alias-symbol <symbol>\",\r\n \"项目根目录映射别名符号(默认 @)\",\r\n defaultConfig().aliasSymbol,\r\n )\r\n .option(\"--registry <pkgOrPath>\", \"registry 来源(默认 builtin)\", defaultConfig().registry)\r\n .action(async (opts) => {\r\n const cwd = path.resolve(opts.cwd);\r\n\r\n // eslint-disable-next-line no-console\r\n console.log(\r\n \"提示:项目中有部分依赖需要使用@/ 路径进行配置,请您安装到项目中配置好的路径下,请慎重选择文件位置\",\r\n );\r\n\r\n const pkgPath = path.join(cwd, \"package.json\");\r\n if (!(await pathExists(pkgPath))) {\r\n throw new Error(`未找到 package.json:${pkgPath}`);\r\n }\r\n\r\n const pm: PackageManager =\r\n opts.pm ?? (await detectPackageManager(cwd));\r\n\r\n const cfg: CliConfig = {\r\n schemaVersion: 1,\r\n componentsDir: opts.componentsDir,\r\n libDir: opts.libDir,\r\n composablesDir: opts.composablesDir,\r\n aliasSymbol: opts.aliasSymbol,\r\n registry: opts.registry,\r\n };\r\n\r\n // 如果已存在 components.json,默认沿用(除非用户显式传参覆盖)\r\n const existing = await loadConfigCompat(cwd);\r\n if (existing) {\r\n cfg.componentsDir = opts.componentsDir ?? existing.componentsDir ?? cfg.componentsDir;\r\n cfg.libDir = opts.libDir ?? existing.libDir ?? cfg.libDir;\r\n cfg.composablesDir = opts.composablesDir ?? existing.composablesDir ?? cfg.composablesDir;\r\n cfg.aliasSymbol = opts.aliasSymbol ?? existing.aliasSymbol ?? cfg.aliasSymbol;\r\n cfg.registry = opts.registry ?? existing.registry ?? cfg.registry;\r\n }\r\n\r\n if (!opts.yes) {\r\n const nuxt = await looksLikeNuxtProject(cwd);\r\n const res = await prompts(\r\n [\r\n {\r\n type: \"text\",\r\n name: \"componentsDir\",\r\n message: \"组件目录(componentsDir)\",\r\n initial: cfg.componentsDir,\r\n },\r\n {\r\n type: \"text\",\r\n name: \"libDir\",\r\n message: \"lib 目录(libDir,用于 cn/utils)\",\r\n initial: cfg.libDir,\r\n },\r\n {\r\n type: \"text\",\r\n name: \"composablesDir\",\r\n message: \"composables 目录(composablesDir)\",\r\n initial: cfg.composablesDir,\r\n },\r\n {\r\n type: \"text\",\r\n name: \"aliasSymbol\",\r\n message: \"基于项目根目录的“映射”的符号是?(aliasSymbol)\",\r\n initial: cfg.aliasSymbol ?? \"@\",\r\n },\r\n ],\r\n {\r\n onCancel: () => {\r\n throw new Error(\"已取消\");\r\n },\r\n },\r\n );\r\n\r\n cfg.componentsDir = res.componentsDir ?? cfg.componentsDir;\r\n cfg.libDir = res.libDir ?? cfg.libDir;\r\n cfg.composablesDir = res.composablesDir ?? cfg.composablesDir;\r\n cfg.aliasSymbol = res.aliasSymbol ?? cfg.aliasSymbol;\r\n\r\n // 写配置\r\n const cfgPath = path.join(cwd, opts.config);\r\n await writeJsonFile(cfgPath, cfg);\r\n\r\n // 复制模板:lib/ + composables/\r\n await copyTemplateSubdir({\r\n subdir: \"lib\",\r\n cwd,\r\n targetDir: cfg.libDir,\r\n overwrite: opts.overwrite,\r\n });\r\n await copyTemplateSubdir({\r\n subdir: \"composables\",\r\n cwd,\r\n targetDir: cfg.composablesDir,\r\n overwrite: opts.overwrite,\r\n });\r\n // 仅替换 \"@/...\" 的别名符号,不改其它路径内容\r\n await rewriteAliasInDir({ cwd, targetDir: cfg.libDir, aliasSymbol: cfg.aliasSymbol ?? \"@\" });\r\n await rewriteAliasInDir({\r\n cwd,\r\n targetDir: cfg.composablesDir,\r\n aliasSymbol: cfg.aliasSymbol ?? \"@\",\r\n });\r\n\r\n // 兜底:如果用户模板里没有 utils.ts,则生成 cn\r\n const libUtilsPath = path.join(cwd, cfg.libDir, \"utils.ts\");\r\n if (!(await pathExists(libUtilsPath))) await writeTextFile(libUtilsPath, cnUtilsTs({}));\r\n\r\n console.log(`已写入配置:${path.relative(process.cwd(), cfgPath)};并生成 cn/utils(pm=${pm})`);\r\n console.log(\"请按照 https://tw.icebreaker.top/docs/quick-start/v4/uni-app-vite 指引进行项目配置初始化\");\r\n\r\n console.log(\"\");\r\n await showLogo();\r\n await typewriter(gradientText(\"感谢使用 Reborn UI ! ✨\"), 20);\r\n return;\r\n }\r\n\r\n // --yes:直接写默认配置\r\n const cfgPath = path.join(cwd, opts.config);\r\n await ensureDir(path.dirname(cfgPath));\r\n await fs.writeFile(cfgPath, JSON.stringify(cfg, null, 2) + \"\\n\", \"utf8\");\r\n\r\n // 复制模板:lib/ + composables/\r\n await copyTemplateSubdir({\r\n subdir: \"lib\",\r\n cwd,\r\n targetDir: cfg.libDir,\r\n overwrite: opts.overwrite,\r\n });\r\n await copyTemplateSubdir({\r\n subdir: \"composables\",\r\n cwd,\r\n targetDir: cfg.composablesDir,\r\n overwrite: opts.overwrite,\r\n });\r\n // 仅替换 \"@/...\" 的别名符号,不改其它路径内容\r\n await rewriteAliasInDir({ cwd, targetDir: cfg.libDir, aliasSymbol: cfg.aliasSymbol ?? \"@\" });\r\n await rewriteAliasInDir({\r\n cwd,\r\n targetDir: cfg.composablesDir,\r\n aliasSymbol: cfg.aliasSymbol ?? \"@\",\r\n });\r\n\r\n // 兜底:如果用户模板里没有 utils.ts,则生成 cn\r\n const libUtilsPath = path.join(cwd, cfg.libDir, \"utils.ts\");\r\n if (!(await pathExists(libUtilsPath))) await writeTextFile(libUtilsPath, cnUtilsTs({}));\r\n\r\n // eslint-disable-next-line no-console\r\n console.log(`已初始化:${path.relative(process.cwd(), cfgPath)};并生成 cn/utils(pm=${pm})`);\r\n console.log(\"请按照 https://tw.icebreaker.top/docs/quick-start/v4/uni-app-vite 指引进行项目配置初始化\");\r\n\r\n console.log(\"\");\r\n await showLogo();\r\n await typewriter(gradientText(\"感谢使用 Reborn UI ! ✨\"), 2);\r\n });\r\n\r\n return cmd;\r\n}\r\n\r\n\r\n","import path from \"node:path\";\nimport fs from \"node:fs\";\nimport { fileURLToPath } from \"node:url\";\nimport { pathExists } from \"./fs.js\";\nimport { copyDirRecursive } from \"./fs.js\";\n\nfunction findNearestPackageRoot(from: string) {\n // from: import.meta.url\n let dir = path.dirname(fileURLToPath(from));\n while (true) {\n if (fs.existsSync(path.join(dir, \"package.json\"))) return dir;\n const parent = path.dirname(dir);\n if (parent === dir) return dir;\n dir = parent;\n }\n}\n\nexport function getTemplatesRootDir() {\n // 注意:CLI 打包后可能变成单文件(dist/index.js),此时 import.meta.url 的相对基准会变化。\n // 通过向上查找最近的 package.json 来定位包根目录。\n const pkgRoot = findNearestPackageRoot(import.meta.url);\n return path.join(pkgRoot, \"templates\");\n}\n\nexport async function copyTemplateSubdir(params: {\n subdir: \"lib\" | \"composables\";\n cwd: string;\n targetDir: string;\n overwrite?: boolean;\n}) {\n const templatesRoot = getTemplatesRootDir();\n const fromDir = path.join(templatesRoot, params.subdir);\n if (!(await pathExists(fromDir))) return false;\n\n const toDir = path.join(params.cwd, params.targetDir);\n await copyDirRecursive({ fromDir, toDir, overwrite: params.overwrite });\n return true;\n}\n\n\n\n\n\n\n","{\n \"name\": \"reborn-ui\",\n \"version\": \"0.1.65\",\n \"description\": \"A CLI for Reborn UI\",\n \"author\": \"1997liuyh-boop\",\n \"license\": \"MIT\",\n \"repository\": {\n \"type\": \"git\",\n \"url\": \"https://github.com/1997liuyh-boop/Reborn-UI.git\"\n },\n \"type\": \"module\",\n \"bin\": {\n \"reborn-ui\": \"dist/index.js\"\n },\n \"files\": [\n \"dist\",\n \"registry\",\n \"templates\",\n \"README.md\"\n ],\n \"publishConfig\": {\n \"access\": \"public\"\n },\n \"scripts\": {\n \"cli\": \"tsx src/index.ts\",\n \"build\": \"tsup\",\n \"dev\": \"tsx src/index.ts\",\n \"clean\": \"rimraf dist\",\n \"prepare\": \"tsup\",\n \"registry:build\": \"tsx src/index.ts build --root ../..\",\n \"prepack\": \"npm run build && npm run registry:build\",\n \"prepublishOnly\": \"npm run build && npm run registry:build\"\n },\n \"dependencies\": {\n \"chalk\": \"^5.6.2\",\n \"cli-progress\": \"^3.12.0\",\n \"commander\": \"^14.0.0\",\n \"execa\": \"^9.6.0\",\n \"figlet\": \"^1.9.4\",\n \"ora\": \"^9.0.0\",\n \"prompts\": \"^2.4.2\"\n },\n \"devDependencies\": {\n \"@types/cli-progress\": \"^3.11.6\",\n \"@types/figlet\": \"^1.7.0\",\n \"@types/node\": \"^22.10.7\",\n \"@types/prompts\": \"^2.4.9\",\n \"rimraf\": \"^6.0.1\",\n \"tsup\": \"^8.5.0\",\n \"tsx\": \"^4.20.0\",\n \"typescript\": \"^5.8.3\"\n }\n}\n"],"mappings":"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","names":["Command","path","fs","path","path","path","fs","path","fs","require","chalk","path","chalk","path","fs","Command","path","Command","fs","path","fs","Command","prompts","path","fs","fileURLToPath","findNearestPackageRoot","path","fileURLToPath","fs","path","fs","Command","prompts","cfgPath","libUtilsPath","Command"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "reborn-ui",
3
- "version": "0.1.64",
3
+ "version": "0.1.65",
4
4
  "description": "A CLI for Reborn UI",
5
5
  "author": "1997liuyh-boop",
6
6
  "license": "MIT",
@@ -0,0 +1,32 @@
1
+ {
2
+ "name": "badge",
3
+ "dependencies": [
4
+ "class-variance-authority"
5
+ ],
6
+ "files": [
7
+ {
8
+ "path": "badge.ts",
9
+ "content": "const color = [\r\n \"primary\",\r\n \"secondary\",\r\n \"success\",\r\n \"info\",\r\n \"warning\",\r\n \"error\",\r\n \"neutral\"\r\n] as const\r\n\r\nconst variant = [\r\n \"solid\",\r\n \"outline\",\r\n \"soft\",\r\n \"subtle\"\r\n] as const\r\n\r\nconst size = [\r\n \"xs\",\r\n \"sm\",\r\n \"md\",\r\n \"lg\",\r\n \"xl\"\r\n] as const\r\n\r\nexport default {\r\n \"slots\": {\r\n \"base\": \"font-medium inline-flex items-center\",\r\n \"label\": \"truncate\",\r\n \"leadingIcon\": \"shrink-0\",\r\n \"leadingAvatar\": \"shrink-0\",\r\n \"leadingAvatarSize\": \"\",\r\n \"trailingIcon\": \"shrink-0\",\r\n \"closeButton\": \"ms-1 inline-flex items-center justify-center rounded-full transition-colors hover:bg-black/10 dark:hover:bg-white/10 cursor-pointer focus:outline-none focus-visible:ring-1 focus-visible:ring-current\"\r\n },\r\n \"variants\": {\r\n \"fieldGroup\": {\r\n \"horizontal\": \"not-only:first:rounded-e-none not-only:last:rounded-s-none not-last:not-first:rounded-none focus-visible:z-[1]\",\r\n \"vertical\": \"not-only:first:rounded-b-none not-only:last:rounded-t-none not-last:not-first:rounded-none focus-visible:z-[1]\"\r\n },\r\n \"color\": {\r\n \"primary\": \"\",\r\n \"secondary\": \"\",\r\n \"success\": \"\",\r\n \"info\": \"\",\r\n \"warning\": \"\",\r\n \"error\": \"\",\r\n \"neutral\": \"\"\r\n },\r\n \"variant\": {\r\n \"solid\": \"\",\r\n \"outline\": \"\",\r\n \"soft\": \"\",\r\n \"subtle\": \"\"\r\n },\r\n \"size\": {\r\n \"xs\": {\r\n \"base\": \"text-[8px]/3 px-1 py-0.5 gap-1 rounded-sm\",\r\n \"leadingIcon\": \"size-3\",\r\n \"leadingAvatarSize\": \"3xs\",\r\n \"trailingIcon\": \"size-3\"\r\n },\r\n \"sm\": {\r\n \"base\": \"text-[10px]/3 px-1.5 py-1 gap-1 rounded-sm\",\r\n \"leadingIcon\": \"size-3\",\r\n \"leadingAvatarSize\": \"3xs\",\r\n \"trailingIcon\": \"size-3\"\r\n },\r\n \"md\": {\r\n \"base\": \"text-xs px-2 py-1 gap-1 rounded-md\",\r\n \"leadingIcon\": \"size-4\",\r\n \"leadingAvatarSize\": \"3xs\",\r\n \"trailingIcon\": \"size-4\"\r\n },\r\n \"lg\": {\r\n \"base\": \"text-sm px-2 py-1 gap-1.5 rounded-md\",\r\n \"leadingIcon\": \"size-5\",\r\n \"leadingAvatarSize\": \"2xs\",\r\n \"trailingIcon\": \"size-5\"\r\n },\r\n \"xl\": {\r\n \"base\": \"text-base px-2.5 py-1 gap-1.5 rounded-md\",\r\n \"leadingIcon\": \"size-6\",\r\n \"leadingAvatarSize\": \"2xs\",\r\n \"trailingIcon\": \"size-6\"\r\n }\r\n },\r\n \"square\": {\r\n \"true\": \"\"\r\n }\r\n },\r\n \"compoundVariants\": [\r\n {\r\n \"color\": \"primary\" as typeof color[number],\r\n \"variant\": \"solid\" as typeof variant[number],\r\n \"class\": \"bg-primary text-inverted\"\r\n },\r\n {\r\n \"color\": \"secondary\" as typeof color[number],\r\n \"variant\": \"solid\" as typeof variant[number],\r\n \"class\": \"bg-secondary text-inverted\"\r\n },\r\n {\r\n \"color\": \"success\" as typeof color[number],\r\n \"variant\": \"solid\" as typeof variant[number],\r\n \"class\": \"bg-success text-inverted\"\r\n },\r\n {\r\n \"color\": \"info\" as typeof color[number],\r\n \"variant\": \"solid\" as typeof variant[number],\r\n \"class\": \"bg-info text-inverted\"\r\n },\r\n {\r\n \"color\": \"warning\" as typeof color[number],\r\n \"variant\": \"solid\" as typeof variant[number],\r\n \"class\": \"bg-warning text-inverted\"\r\n },\r\n {\r\n \"color\": \"error\" as typeof color[number],\r\n \"variant\": \"solid\" as typeof variant[number],\r\n \"class\": \"bg-error text-inverted\"\r\n },\r\n {\r\n \"color\": \"primary\" as typeof color[number],\r\n \"variant\": \"outline\" as typeof variant[number],\r\n \"class\": \"text-primary ring ring-inset ring-primary/50\"\r\n },\r\n {\r\n \"color\": \"secondary\" as typeof color[number],\r\n \"variant\": \"outline\" as typeof variant[number],\r\n \"class\": \"text-secondary ring ring-inset ring-secondary/50\"\r\n },\r\n {\r\n \"color\": \"success\" as typeof color[number],\r\n \"variant\": \"outline\" as typeof variant[number],\r\n \"class\": \"text-success ring ring-inset ring-success/50\"\r\n },\r\n {\r\n \"color\": \"info\" as typeof color[number],\r\n \"variant\": \"outline\" as typeof variant[number],\r\n \"class\": \"text-info ring ring-inset ring-info/50\"\r\n },\r\n {\r\n \"color\": \"warning\" as typeof color[number],\r\n \"variant\": \"outline\" as typeof variant[number],\r\n \"class\": \"text-warning ring ring-inset ring-warning/50\"\r\n },\r\n {\r\n \"color\": \"error\" as typeof color[number],\r\n \"variant\": \"outline\" as typeof variant[number],\r\n \"class\": \"text-error ring ring-inset ring-error/50\"\r\n },\r\n {\r\n \"color\": \"primary\" as typeof color[number],\r\n \"variant\": \"soft\" as typeof variant[number],\r\n \"class\": \"bg-primary/10 text-primary\"\r\n },\r\n {\r\n \"color\": \"secondary\" as typeof color[number],\r\n \"variant\": \"soft\" as typeof variant[number],\r\n \"class\": \"bg-secondary/10 text-secondary\"\r\n },\r\n {\r\n \"color\": \"success\" as typeof color[number],\r\n \"variant\": \"soft\" as typeof variant[number],\r\n \"class\": \"bg-success/10 text-success\"\r\n },\r\n {\r\n \"color\": \"info\" as typeof color[number],\r\n \"variant\": \"soft\" as typeof variant[number],\r\n \"class\": \"bg-info/10 text-info\"\r\n },\r\n {\r\n \"color\": \"warning\" as typeof color[number],\r\n \"variant\": \"soft\" as typeof variant[number],\r\n \"class\": \"bg-warning/10 text-warning\"\r\n },\r\n {\r\n \"color\": \"error\" as typeof color[number],\r\n \"variant\": \"soft\" as typeof variant[number],\r\n \"class\": \"bg-error/10 text-error\"\r\n },\r\n {\r\n \"color\": \"primary\" as typeof color[number],\r\n \"variant\": \"subtle\" as typeof variant[number],\r\n \"class\": \"bg-primary/10 text-primary ring ring-inset ring-primary/25\"\r\n },\r\n {\r\n \"color\": \"secondary\" as typeof color[number],\r\n \"variant\": \"subtle\" as typeof variant[number],\r\n \"class\": \"bg-secondary/10 text-secondary ring ring-inset ring-secondary/25\"\r\n },\r\n {\r\n \"color\": \"success\" as typeof color[number],\r\n \"variant\": \"subtle\" as typeof variant[number],\r\n \"class\": \"bg-success/10 text-success ring ring-inset ring-success/25\"\r\n },\r\n {\r\n \"color\": \"info\" as typeof color[number],\r\n \"variant\": \"subtle\" as typeof variant[number],\r\n \"class\": \"bg-info/10 text-info ring ring-inset ring-info/25\"\r\n },\r\n {\r\n \"color\": \"warning\" as typeof color[number],\r\n \"variant\": \"subtle\" as typeof variant[number],\r\n \"class\": \"bg-warning/10 text-warning ring ring-inset ring-warning/25\"\r\n },\r\n {\r\n \"color\": \"error\" as typeof color[number],\r\n \"variant\": \"subtle\" as typeof variant[number],\r\n \"class\": \"bg-error/10 text-error ring ring-inset ring-error/25\"\r\n },\r\n {\r\n \"color\": \"neutral\" as typeof color[number],\r\n \"variant\": \"solid\" as typeof variant[number],\r\n \"class\": \"text-inverted bg-inverted\"\r\n },\r\n {\r\n \"color\": \"neutral\" as typeof color[number],\r\n \"variant\": \"outline\" as typeof variant[number],\r\n \"class\": \"ring ring-inset ring-accented text-default bg-default\"\r\n },\r\n {\r\n \"color\": \"neutral\" as typeof color[number],\r\n \"variant\": \"soft\" as typeof variant[number],\r\n \"class\": \"text-default bg-elevated\"\r\n },\r\n {\r\n \"color\": \"neutral\" as typeof color[number],\r\n \"variant\": \"subtle\" as typeof variant[number],\r\n \"class\": \"ring ring-inset ring-accented text-default bg-elevated\"\r\n },\r\n {\r\n \"size\": \"xs\" as typeof size[number],\r\n \"square\": true,\r\n \"class\": \"p-0.5\"\r\n },\r\n {\r\n \"size\": \"sm\" as typeof size[number],\r\n \"square\": true,\r\n \"class\": \"p-1\"\r\n },\r\n {\r\n \"size\": \"md\" as typeof size[number],\r\n \"square\": true,\r\n \"class\": \"p-1\"\r\n },\r\n {\r\n \"size\": \"lg\" as typeof size[number],\r\n \"square\": true,\r\n \"class\": \"p-1\"\r\n },\r\n {\r\n \"size\": \"xl\" as typeof size[number],\r\n \"square\": true,\r\n \"class\": \"p-1\"\r\n }\r\n ],\r\n \"defaultVariants\": {\r\n \"color\": \"primary\" as typeof color[number],\r\n \"variant\": \"solid\" as typeof variant[number],\r\n \"size\": \"md\" as typeof size[number]\r\n }\r\n}"
10
+ },
11
+ {
12
+ "path": "Badge.vue",
13
+ "content": "<script setup lang=\"ts\">\r\nimport { computed } from 'vue'\r\nimport theme from './badge'\r\nimport { useFieldGroup } from '~/composables/useFieldGroup'\r\nimport { tv } from '~/lib/tv'\r\n\r\n// Define the variant builder\r\nconst b = tv({ extend: tv(theme) })\r\n\r\n// Derive types directly from the theme object to ensure static analysis works\r\ntype Theme = typeof theme\r\ntype BadgeColor = keyof Theme['variants']['color']\r\ntype BadgeVariant = keyof Theme['variants']['variant']\r\ntype BadgeSize = keyof Theme['variants']['size']\r\n\r\nexport interface BadgeProps {\r\n /**\r\n * The element or component this component should render as.\r\n * @defaultValue 'span'\r\n */\r\n as?: any\r\n label?: string | number\r\n /**\r\n * @defaultValue 'primary'\r\n */\r\n color?: BadgeColor | (string & {})\r\n /**\r\n * @defaultValue 'solid'\r\n */\r\n variant?: BadgeVariant | (string & {})\r\n /**\r\n * @defaultValue 'md'\r\n */\r\n size?: BadgeSize | (string & {})\r\n /** Render the badge with equal padding on all sides. */\r\n square?: boolean\r\n /** Whether the badge is closable. */\r\n closable?: boolean\r\n /** The icon for the close button. */\r\n closeIcon?: string\r\n class?: any\r\n ui?: any\r\n}\r\n\r\nexport interface BadgeSlots {\r\n leading(props: { ui: any }): any\r\n default(props: { ui: any }): any\r\n trailing(props: { ui: any }): any\r\n close(props: { ui: any; close: (e: MouseEvent) => void }): any\r\n}\r\n\r\nconst props = withDefaults(defineProps<BadgeProps>(), {\r\n as: 'span',\r\n color: 'primary',\r\n variant: 'solid',\r\n size: 'md',\r\n closeIcon: 'i-lucide-x'\r\n})\r\n\r\nconst slots = defineSlots<BadgeSlots>()\r\n\r\nconst emit = defineEmits<{\r\n close: [payload: MouseEvent]\r\n}>()\r\n\r\nconst show = defineModel<boolean>('show', { default: true })\r\n\r\nconst { orientation, size: fieldGroupSize } = useFieldGroup(props)\r\n\r\nconst ui = computed(() => b({\r\n color: props.color as BadgeColor,\r\n variant: props.variant as BadgeVariant,\r\n size: (fieldGroupSize.value || props.size) as BadgeSize,\r\n square: props.square || (!slots.default && !props.label),\r\n fieldGroup: orientation.value\r\n}))\r\n\r\nconst handleClose = (e: MouseEvent) => {\r\n show.value = false\r\n emit('close', e)\r\n}\r\n</script>\r\n\r\n<template>\r\n <component :is=\"props.as\" v-if=\"show\" :class=\"ui.base({ class: props.class })\">\r\n <slot name=\"leading\" :ui=\"ui\" />\r\n\r\n <slot :ui=\"ui\">\r\n <span v-if=\"label\" :class=\"ui.label()\">\r\n {{ label }}\r\n </span>\r\n </slot>\r\n\r\n <slot name=\"trailing\" :ui=\"ui\" />\r\n\r\n <span v-if=\"closable\" @click.stop=\"handleClose\" :class=\"ui.closeButton()\">\r\n <slot name=\"close\" :ui=\"ui\" :close=\"handleClose\">\r\n <Icon :name=\"closeIcon\" class=\"size-3\" />\r\n </slot>\r\n </span>\r\n </component>\r\n</template>\r\n",
14
+ "target": "web"
15
+ },
16
+ {
17
+ "path": "Badge.vue",
18
+ "content": "<script setup lang=\"ts\">\r\nimport { computed } from 'vue'\r\nimport theme from './badge'\r\nimport { useFieldGroup } from '~/composables/useFieldGroup'\r\nimport { tv } from '~/lib/tv'\r\n\r\n// Define the variant builder\r\nconst b = tv({ extend: tv(theme) })\r\n\r\n// Derive types directly from the theme object to ensure static analysis works\r\ntype Theme = typeof theme\r\ntype BadgeColor = keyof Theme['variants']['color']\r\ntype BadgeVariant = keyof Theme['variants']['variant']\r\ntype BadgeSize = keyof Theme['variants']['size']\r\n\r\nexport interface BadgeProps {\r\n /**\r\n * The element or component this component should render as.\r\n * @defaultValue 'span'\r\n */\r\n as?: any\r\n label?: string | number\r\n /**\r\n * @defaultValue 'primary'\r\n */\r\n color?: BadgeColor | (string & {})\r\n /**\r\n * @defaultValue 'solid'\r\n */\r\n variant?: BadgeVariant | (string & {})\r\n /**\r\n * @defaultValue 'md'\r\n */\r\n size?: BadgeSize | (string & {})\r\n /** Render the badge with equal padding on all sides. */\r\n square?: boolean\r\n /** Whether the badge is closable. */\r\n closable?: boolean\r\n /** The icon for the close button. */\r\n closeIcon?: string\r\n class?: any\r\n ui?: any\r\n}\r\n\r\nexport interface BadgeSlots {\r\n leading(props: { ui: any }): any\r\n default(props: { ui: any }): any\r\n trailing(props: { ui: any }): any\r\n close(props: { ui: any; close: (e: MouseEvent) => void }): any\r\n}\r\n\r\nconst props = withDefaults(defineProps<BadgeProps>(), {\r\n as: 'span',\r\n color: 'primary',\r\n variant: 'solid',\r\n size: 'md',\r\n closeIcon: 'i-lucide-x'\r\n})\r\n\r\nconst slots = defineSlots<BadgeSlots>()\r\n\r\nconst emit = defineEmits<{\r\n close: [payload: MouseEvent]\r\n}>()\r\n\r\nconst show = defineModel<boolean>('show', { default: true })\r\n\r\nconst { orientation, size: fieldGroupSize } = useFieldGroup(props)\r\n\r\nconst ui = computed(() => b({\r\n color: props.color as BadgeColor,\r\n variant: props.variant as BadgeVariant,\r\n size: (fieldGroupSize.value || props.size) as BadgeSize,\r\n square: props.square || (!slots.default && !props.label),\r\n fieldGroup: orientation.value\r\n}))\r\n\r\nconst handleClose = (e: MouseEvent) => {\r\n show.value = false\r\n emit('close', e)\r\n}\r\n</script>\r\n\r\n<template>\r\n <component :is=\"props.as\" v-if=\"show\" :class=\"ui.base({ class: props.class })\">\r\n <slot name=\"leading\" :ui=\"ui\" />\r\n\r\n <slot :ui=\"ui\">\r\n <text v-if=\"label\" :class=\"ui.label()\">\r\n {{ label }}\r\n </text>\r\n </slot>\r\n\r\n <slot name=\"trailing\" :ui=\"ui\" />\r\n\r\n <text v-if=\"closable\" @click.stop=\"handleClose\" :class=\"ui.closeButton()\">\r\n <slot name=\"close\" :ui=\"ui\" :close=\"handleClose\">\r\n <Icon :name=\"closeIcon\" class=\"size-3\" />\r\n </slot>\r\n </text>\r\n </component>\r\n</template>\r\n",
19
+ "target": "uniapp"
20
+ },
21
+ {
22
+ "path": "index.ts",
23
+ "content": "export { default as Badge } from \"./Badge.vue\";\r\n"
24
+ },
25
+ {
26
+ "path": "useBadge.ts",
27
+ "content": "import type { VariantProps } from \"class-variance-authority\"\r\nimport { cva } from \"class-variance-authority\"\r\n\r\nexport { default as Badge } from \"./Badge.vue\"\r\n\r\nexport const badgeVariants = cva(\r\n \"inline-flex items-center justify-center rounded-full border px-2 py-0.5 text-xs font-medium w-fit whitespace-nowrap shrink-0 [&>svg]:size-3 gap-1 [&>svg]:pointer-events-none focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive transition-[color,box-shadow] overflow-hidden\",\r\n {\r\n variants: {\r\n variant: {\r\n default:\r\n \"border-transparent bg-primary text-primary-foreground [a&]:hover:bg-primary/90\",\r\n secondary:\r\n \"border-transparent bg-secondary text-secondary-foreground [a&]:hover:bg-secondary/90\",\r\n destructive:\r\n \"border-transparent bg-destructive text-white [a&]:hover:bg-destructive/90 focus-visible:ring-destructive/20 dark:focus-visible:ring-destructive/40 dark:bg-destructive/60\",\r\n outline:\r\n \"text-foreground [a&]:hover:bg-accent [a&]:hover:text-accent-foreground\",\r\n },\r\n },\r\n defaultVariants: {\r\n variant: \"default\",\r\n },\r\n },\r\n)\r\nexport type BadgeVariants = VariantProps<typeof badgeVariants>\r\n"
28
+ }
29
+ ],
30
+ "fileCount": 5,
31
+ "contentHash": "6c42e42daf68a49c11d6dce945dc05fd13a89c26"
32
+ }
@@ -8,15 +8,15 @@
8
8
  },
9
9
  {
10
10
  "path": "iPhone15ProMockup.vue",
11
- "content": "<!-- eslint-disable check-file/filename-naming-convention -->\r\n<script lang=\"ts\" setup>\r\n// eslint-disable-next-line check-file/filename-naming-convention\r\ninterface Props {\r\n width?: number;\r\n height?: number;\r\n src?: string;\r\n}\r\n\r\nwithDefaults(defineProps<Props>(), {\r\n width: 433,\r\n height: 882,\r\n});\r\n</script>\r\n\r\n<template>\r\n <svg\r\n fill=\"none\"\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n :width=\"width\"\r\n :height=\"height\"\r\n :viewBox=\"`0 0 ${width} ${height}`\"\r\n >\r\n <path\r\n d=\"M2 73C2 32.6832 34.6832 0 75 0H357C397.317 0 430 32.6832 430 73V809C430 849.317 397.317 882 357 882H75C34.6832 882 2 849.317 2 809V73Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\"\r\n />\r\n <path\r\n d=\"M0 171C0 170.448 0.447715 170 1 170H3V204H1C0.447715 204 0 203.552 0 203V171Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\"\r\n />\r\n <path\r\n d=\"M1 234C1 233.448 1.44772 233 2 233H3.5V300H2C1.44772 300 1 299.552 1 299V234Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\"\r\n />\r\n <path\r\n d=\"M1 319C1 318.448 1.44772 318 2 318H3.5V385H2C1.44772 385 1 384.552 1 384V319Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\"\r\n />\r\n <path\r\n d=\"M430 279H432C432.552 279 433 279.448 433 280V384C433 384.552 432.552 385 432 385H430V279Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\"\r\n />\r\n <path\r\n d=\"M6 74C6 35.3401 37.3401 4 76 4H356C394.66 4 426 35.3401 426 74V808C426 846.66 394.66 878 356 878H76C37.3401 878 6 846.66 6 808V74Z\"\r\n class=\"fill-white dark:fill-[#262626]\"\r\n />\r\n <path\r\n opacity=\"0.5\"\r\n d=\"M174 5H258V5.5C258 6.60457 257.105 7.5 256 7.5H176C174.895 7.5 174 6.60457 174 5.5V5Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\"\r\n />\r\n <path\r\n d=\"M21.25 75C21.25 44.2101 46.2101 19.25 77 19.25H355C385.79 19.25 410.75 44.2101 410.75 75V807C410.75 837.79 385.79 862.75 355 862.75H77C46.2101 862.75 21.25 837.79 21.25 807V75Z\"\r\n class=\"fill-[#E5E5E5] stroke-[#E5E5E5] stroke-[0.5] dark:fill-[#404040] dark:stroke-[#404040]\"\r\n />\r\n <image\r\n v-if=\"src\"\r\n x=\"21.25\"\r\n y=\"19.25\"\r\n width=\"389.5\"\r\n height=\"843.5\"\r\n preserveAspectRatio=\"xMidYMid slice\"\r\n style=\"clip-path: url(#roundedCorners)\"\r\n :href=\"src\"\r\n />\r\n\r\n <path\r\n d=\"M154 48.5C154 38.2827 162.283 30 172.5 30H259.5C269.717 30 278 38.2827 278 48.5C278 58.7173 269.717 67 259.5 67H172.5C162.283 67 154 58.7173 154 48.5Z\"\r\n class=\"fill-[#F5F5F5] dark:fill-[#262626]\"\r\n />\r\n <path\r\n d=\"M249 48.5C249 42.701 253.701 38 259.5 38C265.299 38 270 42.701 270 48.5C270 54.299 265.299 59 259.5 59C253.701 59 249 54.299 249 48.5Z\"\r\n class=\"fill-[#F5F5F5] dark:fill-[#262626]\"\r\n />\r\n <path\r\n d=\"M254 48.5C254 45.4624 256.462 43 259.5 43C262.538 43 265 45.4624 265 48.5C265 51.5376 262.538 54 259.5 54C256.462 54 254 51.5376 254 48.5Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\"\r\n />\r\n <defs>\r\n <clipPath id=\"roundedCorners\">\r\n <rect\r\n x=\"21.25\"\r\n y=\"19.25\"\r\n width=\"389.5\"\r\n height=\"843.5\"\r\n rx=\"55.75\"\r\n ry=\"55.75\"\r\n />\r\n </clipPath>\r\n </defs>\r\n </svg>\r\n</template>\r\n",
11
+ "content": "<!-- eslint-disable check-file/filename-naming-convention -->\r\n<script lang=\"ts\" setup>\r\n// eslint-disable-next-line check-file/filename-naming-convention\r\ninterface Props {\r\n width?: number;\r\n height?: number;\r\n src?: string;\r\n}\r\n\r\nwithDefaults(defineProps<Props>(), {\r\n width: 433,\r\n height: 882,\r\n});\r\n</script>\r\n\r\n<template>\r\n <svg fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" :width=\"width\" :height=\"height\"\r\n :viewBox=\"`0 0 ${width} ${height}`\" class=\"group\">\r\n <path\r\n d=\"M2 73C2 32.6832 34.6832 0 75 0H357C397.317 0 430 32.6832 430 73V809C430 849.317 397.317 882 357 882H75C34.6832 882 2 849.317 2 809V73Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\" />\r\n <path d=\"M0 171C0 170.448 0.447715 170 1 170H3V204H1C0.447715 204 0 203.552 0 203V171Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\" />\r\n <path d=\"M1 234C1 233.448 1.44772 233 2 233H3.5V300H2C1.44772 300 1 299.552 1 299V234Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\" />\r\n <path d=\"M1 319C1 318.448 1.44772 318 2 318H3.5V385H2C1.44772 385 1 384.552 1 384V319Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\" />\r\n <path d=\"M430 279H432C432.552 279 433 279.448 433 280V384C433 384.552 432.552 385 432 385H430V279Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\" />\r\n <path\r\n d=\"M6 74C6 35.3401 37.3401 4 76 4H356C394.66 4 426 35.3401 426 74V808C426 846.66 394.66 878 356 878H76C37.3401 878 6 846.66 6 808V74Z\"\r\n class=\"fill-white dark:fill-[#262626]\" />\r\n <path opacity=\"0.5\" d=\"M174 5H258V5.5C258 6.60457 257.105 7.5 256 7.5H176C174.895 7.5 174 6.60457 174 5.5V5Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\" />\r\n <path\r\n d=\"M21.25 75C21.25 44.2101 46.2101 19.25 77 19.25H355C385.79 19.25 410.75 44.2101 410.75 75V807C410.75 837.79 385.79 862.75 355 862.75H77C46.2101 862.75 21.25 837.79 21.25 807V75Z\"\r\n class=\"fill-[#E5E5E5] stroke-[#E5E5E5] stroke-[0.5] dark:fill-[#404040] dark:stroke-[#404040]\" />\r\n <foreignObject v-if=\"src\" x=\"21.25\" y=\"19.25\" width=\"389.5\" height=\"843.5\" style=\"clip-path: url(#roundedCorners)\">\r\n <iframe :src=\"src\" class=\"size-full bg-white border-none\" />\r\n </foreignObject>\r\n\r\n <path\r\n d=\"M154 48.5C154 38.2827 162.283 30 172.5 30H259.5C269.717 30 278 38.2827 278 48.5C278 58.7173 269.717 67 259.5 67H172.5C162.283 67 154 58.7173 154 48.5Z\"\r\n class=\"fill-[#F5F5F5] transition-opacity duration-300 group-hover:opacity-0 dark:fill-[#262626]\" />\r\n <path\r\n d=\"M249 48.5C249 42.701 253.701 38 259.5 38C265.299 38 270 42.701 270 48.5C270 54.299 265.299 59 259.5 59C253.701 59 249 54.299 249 48.5Z\"\r\n class=\"fill-[#F5F5F5] transition-opacity duration-300 group-hover:opacity-0 dark:fill-[#262626]\" />\r\n <path\r\n d=\"M254 48.5C254 45.4624 256.462 43 259.5 43C262.538 43 265 45.4624 265 48.5C265 51.5376 262.538 54 259.5 54C256.462 54 254 51.5376 254 48.5Z\"\r\n class=\"fill-[#E5E5E5] transition-opacity duration-300 group-hover:opacity-0 dark:fill-[#404040]\" />\r\n <defs>\r\n <clipPath id=\"roundedCorners\">\r\n <rect x=\"21.25\" y=\"19.25\" width=\"389.5\" height=\"843.5\" rx=\"55.75\" ry=\"55.75\" />\r\n </clipPath>\r\n </defs>\r\n </svg>\r\n</template>\r\n",
12
12
  "target": "web"
13
13
  },
14
14
  {
15
15
  "path": "iPhone15ProMockup.vue",
16
- "content": "<!-- eslint-disable check-file/filename-naming-convention -->\r\n<script lang=\"ts\" setup>\r\n// eslint-disable-next-line check-file/filename-naming-convention\r\ninterface Props {\r\n width?: number;\r\n height?: number;\r\n src?: string;\r\n}\r\n\r\nwithDefaults(defineProps<Props>(), {\r\n width: 433,\r\n height: 882,\r\n});\r\n</script>\r\n\r\n<template>\r\n <svg\r\n fill=\"none\"\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n :width=\"width\"\r\n :height=\"height\"\r\n :viewBox=\"`0 0 ${width} ${height}`\"\r\n >\r\n <path\r\n d=\"M2 73C2 32.6832 34.6832 0 75 0H357C397.317 0 430 32.6832 430 73V809C430 849.317 397.317 882 357 882H75C34.6832 882 2 849.317 2 809V73Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\"\r\n />\r\n <path\r\n d=\"M0 171C0 170.448 0.447715 170 1 170H3V204H1C0.447715 204 0 203.552 0 203V171Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\"\r\n />\r\n <path\r\n d=\"M1 234C1 233.448 1.44772 233 2 233H3.5V300H2C1.44772 300 1 299.552 1 299V234Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\"\r\n />\r\n <path\r\n d=\"M1 319C1 318.448 1.44772 318 2 318H3.5V385H2C1.44772 385 1 384.552 1 384V319Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\"\r\n />\r\n <path\r\n d=\"M430 279H432C432.552 279 433 279.448 433 280V384C433 384.552 432.552 385 432 385H430V279Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\"\r\n />\r\n <path\r\n d=\"M6 74C6 35.3401 37.3401 4 76 4H356C394.66 4 426 35.3401 426 74V808C426 846.66 394.66 878 356 878H76C37.3401 878 6 846.66 6 808V74Z\"\r\n class=\"fill-white dark:fill-[#262626]\"\r\n />\r\n <path\r\n opacity=\"0.5\"\r\n d=\"M174 5H258V5.5C258 6.60457 257.105 7.5 256 7.5H176C174.895 7.5 174 6.60457 174 5.5V5Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\"\r\n />\r\n <path\r\n d=\"M21.25 75C21.25 44.2101 46.2101 19.25 77 19.25H355C385.79 19.25 410.75 44.2101 410.75 75V807C410.75 837.79 385.79 862.75 355 862.75H77C46.2101 862.75 21.25 837.79 21.25 807V75Z\"\r\n class=\"fill-[#E5E5E5] stroke-[#E5E5E5] stroke-[0.5] dark:fill-[#404040] dark:stroke-[#404040]\"\r\n />\r\n <image\r\n v-if=\"src\"\r\n x=\"21.25\"\r\n y=\"19.25\"\r\n width=\"389.5\"\r\n height=\"843.5\"\r\n preserveAspectRatio=\"xMidYMid slice\"\r\n style=\"clip-path: url(#roundedCorners)\"\r\n :href=\"src\"\r\n />\r\n\r\n <path\r\n d=\"M154 48.5C154 38.2827 162.283 30 172.5 30H259.5C269.717 30 278 38.2827 278 48.5C278 58.7173 269.717 67 259.5 67H172.5C162.283 67 154 58.7173 154 48.5Z\"\r\n class=\"fill-[#F5F5F5] dark:fill-[#262626]\"\r\n />\r\n <path\r\n d=\"M249 48.5C249 42.701 253.701 38 259.5 38C265.299 38 270 42.701 270 48.5C270 54.299 265.299 59 259.5 59C253.701 59 249 54.299 249 48.5Z\"\r\n class=\"fill-[#F5F5F5] dark:fill-[#262626]\"\r\n />\r\n <path\r\n d=\"M254 48.5C254 45.4624 256.462 43 259.5 43C262.538 43 265 45.4624 265 48.5C265 51.5376 262.538 54 259.5 54C256.462 54 254 51.5376 254 48.5Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\"\r\n />\r\n <defs>\r\n <clipPath id=\"roundedCorners\">\r\n <rect\r\n x=\"21.25\"\r\n y=\"19.25\"\r\n width=\"389.5\"\r\n height=\"843.5\"\r\n rx=\"55.75\"\r\n ry=\"55.75\"\r\n />\r\n </clipPath>\r\n </defs>\r\n </svg>\r\n</template>\r\n",
16
+ "content": "<!-- eslint-disable check-file/filename-naming-convention -->\r\n<script lang=\"ts\" setup>\r\n// eslint-disable-next-line check-file/filename-naming-convention\r\ninterface Props {\r\n width?: number;\r\n height?: number;\r\n src?: string;\r\n}\r\n\r\nwithDefaults(defineProps<Props>(), {\r\n width: 433,\r\n height: 882,\r\n});\r\n</script>\r\n\r\n<template>\r\n <svg fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\" :width=\"width\" :height=\"height\"\r\n :viewBox=\"`0 0 ${width} ${height}`\" class=\"group\">\r\n <path\r\n d=\"M2 73C2 32.6832 34.6832 0 75 0H357C397.317 0 430 32.6832 430 73V809C430 849.317 397.317 882 357 882H75C34.6832 882 2 849.317 2 809V73Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\" />\r\n <path d=\"M0 171C0 170.448 0.447715 170 1 170H3V204H1C0.447715 204 0 203.552 0 203V171Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\" />\r\n <path d=\"M1 234C1 233.448 1.44772 233 2 233H3.5V300H2C1.44772 300 1 299.552 1 299V234Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\" />\r\n <path d=\"M1 319C1 318.448 1.44772 318 2 318H3.5V385H2C1.44772 385 1 384.552 1 384V319Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\" />\r\n <path d=\"M430 279H432C432.552 279 433 279.448 433 280V384C433 384.552 432.552 385 432 385H430V279Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\" />\r\n <path\r\n d=\"M6 74C6 35.3401 37.3401 4 76 4H356C394.66 4 426 35.3401 426 74V808C426 846.66 394.66 878 356 878H76C37.3401 878 6 846.66 6 808V74Z\"\r\n class=\"fill-white dark:fill-[#262626]\" />\r\n <path opacity=\"0.5\" d=\"M174 5H258V5.5C258 6.60457 257.105 7.5 256 7.5H176C174.895 7.5 174 6.60457 174 5.5V5Z\"\r\n class=\"fill-[#E5E5E5] dark:fill-[#404040]\" />\r\n <path\r\n d=\"M21.25 75C21.25 44.2101 46.2101 19.25 77 19.25H355C385.79 19.25 410.75 44.2101 410.75 75V807C410.75 837.79 385.79 862.75 355 862.75H77C46.2101 862.75 21.25 837.79 21.25 807V75Z\"\r\n class=\"fill-[#E5E5E5] stroke-[#E5E5E5] stroke-[0.5] dark:fill-[#404040] dark:stroke-[#404040]\" />\r\n <foreignObject v-if=\"src\" x=\"21.25\" y=\"19.25\" width=\"389.5\" height=\"843.5\" style=\"clip-path: url(#roundedCorners)\">\r\n <web-view :src=\"src\" class=\"size-full bg-white border-none\" />\r\n </foreignObject>\r\n\r\n <path\r\n d=\"M154 48.5C154 38.2827 162.283 30 172.5 30H259.5C269.717 30 278 38.2827 278 48.5C278 58.7173 269.717 67 259.5 67H172.5C162.283 67 154 58.7173 154 48.5Z\"\r\n class=\"fill-[#F5F5F5] transition-opacity duration-300 group-hover:opacity-0 dark:fill-[#262626]\" />\r\n <path\r\n d=\"M249 48.5C249 42.701 253.701 38 259.5 38C265.299 38 270 42.701 270 48.5C270 54.299 265.299 59 259.5 59C253.701 59 249 54.299 249 48.5Z\"\r\n class=\"fill-[#F5F5F5] transition-opacity duration-300 group-hover:opacity-0 dark:fill-[#262626]\" />\r\n <path\r\n d=\"M254 48.5C254 45.4624 256.462 43 259.5 43C262.538 43 265 45.4624 265 48.5C265 51.5376 262.538 54 259.5 54C256.462 54 254 51.5376 254 48.5Z\"\r\n class=\"fill-[#E5E5E5] transition-opacity duration-300 group-hover:opacity-0 dark:fill-[#404040]\" />\r\n <defs>\r\n <clipPath id=\"roundedCorners\">\r\n <rect x=\"21.25\" y=\"19.25\" width=\"389.5\" height=\"843.5\" rx=\"55.75\" ry=\"55.75\" />\r\n </clipPath>\r\n </defs>\r\n </svg>\r\n</template>\r\n",
17
17
  "target": "uniapp"
18
18
  }
19
19
  ],
20
20
  "fileCount": 3,
21
- "contentHash": "663e0bef752dbcb208fe876d9b2be047417f53fc"
21
+ "contentHash": "a38fa2410d058b57c2ab0ba73e2e8c9cbcecd988"
22
22
  }
@@ -0,0 +1,26 @@
1
+ {
2
+ "name": "reborn-button",
3
+ "dependencies": [],
4
+ "files": [
5
+ {
6
+ "path": "index.ts",
7
+ "content": "export { default as RebornButton } from \"./RebornButton.vue\";\r\n"
8
+ },
9
+ {
10
+ "path": "reborn-button.config.ts",
11
+ "content": "const color = [\"primary\", \"secondary\", \"success\", \"info\", \"warning\", \"error\", \"neutral\"] as const;\r\n\r\nconst variant = [\"solid\", \"outline\", \"soft\", \"subtle\"] as const;\r\n\r\nconst size = [\r\n \"xs\",\r\n \"sm\",\r\n \"default\", // Mapped to md\r\n \"md\",\r\n \"lg\",\r\n \"xl\",\r\n \"2xl\",\r\n \"icon-xs\",\r\n \"icon-sm\",\r\n \"icon\", // Mapped to icon-md\r\n \"icon-md\",\r\n \"icon-lg\",\r\n \"icon-xl\",\r\n \"icon-2xl\",\r\n] as const;\r\n\r\nexport { color as buttonColors, variant as buttonVariants, size as buttonSizes };\r\n\r\nexport default {\r\n slots: {\r\n base: \"inline-flex items-center justify-center gap-2 whitespace-nowrap transition-all cursor-pointer disabled:cursor-not-allowed disabled:opacity-70 disabled:bg-gray-4 [&_svg]:pointer-events-none [&_svg:not([class*='size-'])]:size-4 shrink-0 [&_svg]:shrink-0 outline-none focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:ring-[3px] aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 aria-invalid:border-destructive\",\r\n label: \"truncate\",\r\n leadingIcon: \"shrink-0\",\r\n leadingAvatar: \"shrink-0\",\r\n leadingAvatarSize: \"\",\r\n trailingIcon: \"shrink-0\",\r\n },\r\n variants: {\r\n fieldGroup: {\r\n horizontal:\r\n \"not-only:first:rounded-e-none not-only:last:rounded-s-none not-last:not-first:rounded-none focus-visible:z-[1]\",\r\n vertical:\r\n \"not-only:first:rounded-b-none not-only:last:rounded-t-none not-last:not-first:rounded-none focus-visible:z-[1]\",\r\n },\r\n color: {\r\n primary: \"\",\r\n secondary: \"\",\r\n success: \"\",\r\n info: \"\",\r\n warning: \"\",\r\n error: \"\",\r\n neutral: \"\",\r\n },\r\n variant: {\r\n solid: \"\",\r\n outline: \"\",\r\n soft: \"\",\r\n subtle: \"\",\r\n },\r\n size: {\r\n xs: {\r\n base: \"h-[var(--button-xs-height)] text-24 leading-[1.5] gap-1.5 px-3 has-[>svg]:px-2.5\",\r\n },\r\n sm: {\r\n base: \"h-[var(--button-sm-height)] text-24 leading-[1.5] gap-1.5 px-3 has-[>svg]:px-2.5\",\r\n },\r\n default: {\r\n base: \"h-[var(--button-base-height)] text-26 leading-[1.5] px-4 py-4 has-[>svg]:px-3\",\r\n },\r\n md: { base: \"h-[var(--button-base-height)] text-26 leading-[1.5] px-4 py-4 has-[>svg]:px-3\" },\r\n lg: { base: \"h-[var(--button-lg-height)] text-28 leading-[1.5] px-6 has-[>svg]:px-4\" },\r\n xl: { base: \"h-[var(--button-xl-height)] text-30 leading-[1.5] px-6 has-[>svg]:px-4\" },\r\n \"2xl\": { base: \"h-[var(--button-2xl-height)] text-32 leading-[1.5] px-6 has-[>svg]:px-4\" },\r\n \"icon-xs\": { base: \"\" },\r\n \"icon-sm\": { base: \"size-[var(--button-sm-height)]\" },\r\n icon: { base: \"size-[var(--button-base-height)]\" },\r\n \"icon-md\": { base: \"size-[var(--button-base-height)]\" },\r\n \"icon-lg\": { base: \"size-[var(--button-lg-height)]\" },\r\n \"icon-xl\": { base: \"size-[var(--button-xl-height)]\" },\r\n \"icon-2xl\": { base: \"size-[var(--button-2xl-height)]\" },\r\n },\r\n square: {\r\n true: { base: \"p-0\" },\r\n },\r\n },\r\n compoundVariants: [\r\n // Solid Variants\r\n {\r\n color: \"primary\" as (typeof color)[number],\r\n variant: \"solid\" as (typeof variant)[number],\r\n class: \"bg-primary text-white hover:bg-primary/75\",\r\n },\r\n {\r\n color: \"secondary\" as (typeof color)[number], // Secondary not in original, defaulting to Purple/Violet or Gray? User had Info=Gray. I'll make Secondary=Violet\r\n variant: \"solid\" as (typeof variant)[number],\r\n class: \"bg-secondary text-white hover:bg-secondary/75\",\r\n },\r\n {\r\n color: \"success\" as (typeof color)[number],\r\n variant: \"solid\" as (typeof variant)[number],\r\n class: \"bg-success text-white hover:bg-success/75\",\r\n },\r\n {\r\n color: \"info\" as (typeof color)[number], // User's original info was gray-8. I will keep that as explicit choice.\r\n variant: \"solid\" as (typeof variant)[number],\r\n class: \"bg-info text-white hover:bg-info/75\",\r\n },\r\n {\r\n color: \"warning\" as (typeof color)[number],\r\n variant: \"solid\" as (typeof variant)[number],\r\n class: \"bg-warning text-white hover:bg-warning/75\",\r\n },\r\n {\r\n color: \"error\" as (typeof color)[number], // User's original default was red.\r\n variant: \"solid\" as (typeof variant)[number],\r\n class: \"bg-error text-white hover:bg-error/75\",\r\n },\r\n {\r\n color: \"neutral\" as (typeof color)[number],\r\n variant: \"solid\" as (typeof variant)[number],\r\n class: \"bg-neutral text-white hover:bg-neutral/75\",\r\n },\r\n\r\n // Outline Variants\r\n {\r\n color: \"primary\" as (typeof color)[number],\r\n variant: \"outline\" as (typeof variant)[number],\r\n class:\r\n \"bg-transparent text-primary border border-primary hover:bg-primary/10 disabled:bg-gray-2 disabled:border-gray-4 disabled:text-gray-6\",\r\n },\r\n {\r\n color: \"secondary\" as (typeof color)[number],\r\n variant: \"outline\" as (typeof variant)[number],\r\n class:\r\n \"bg-transparent text-secondary border border-secondary hover:bg-secondary/10 disabled:bg-gray-2 disabled:border-gray-4 disabled:text-gray-6\",\r\n },\r\n {\r\n color: \"success\" as (typeof color)[number],\r\n variant: \"outline\" as (typeof variant)[number],\r\n class:\r\n \"bg-transparent text-success border border-success hover:bg-success/10 disabled:bg-gray-2 disabled:border-gray-4 disabled:text-gray-6\",\r\n },\r\n {\r\n color: \"info\" as (typeof color)[number],\r\n variant: \"outline\" as (typeof variant)[number],\r\n class:\r\n \"bg-transparent text-info border border-info hover:bg-info/10 disabled:bg-gray-2 disabled:border-gray-4 disabled:text-gray-6\",\r\n },\r\n {\r\n color: \"warning\" as (typeof color)[number],\r\n variant: \"outline\" as (typeof variant)[number],\r\n class:\r\n \"bg-transparent text-warning border border-warning hover:bg-warning/10 disabled:bg-gray-2 disabled:border-gray-4 disabled:text-gray-6\",\r\n },\r\n {\r\n color: \"error\" as (typeof color)[number],\r\n variant: \"outline\" as (typeof variant)[number],\r\n class:\r\n \"bg-transparent text-error border border-error hover:bg-error/10 disabled:bg-gray-2 disabled:border-gray-4 disabled:text-gray-6\",\r\n },\r\n {\r\n color: \"neutral\" as (typeof color)[number],\r\n variant: \"outline\" as (typeof variant)[number],\r\n class:\r\n \"bg-transparent text-neutral border border-neutral hover:bg-neutral/10 disabled:bg-gray-2 disabled:border-gray-4 disabled:text-gray-6\",\r\n },\r\n\r\n // Soft Variants (Freestyling as requested, using lighter 1/2 shades)\r\n {\r\n color: \"primary\" as (typeof color)[number],\r\n variant: \"soft\" as (typeof variant)[number],\r\n class: \"bg-primary/10 text-primary hover:bg-primary/20\",\r\n },\r\n {\r\n color: \"secondary\" as (typeof color)[number],\r\n variant: \"soft\" as (typeof variant)[number],\r\n class: \"bg-secondary/10 text-secondary hover:bg-secondary/20\",\r\n },\r\n {\r\n color: \"success\" as (typeof color)[number],\r\n variant: \"soft\" as (typeof variant)[number],\r\n class: \"bg-success/10 text-success hover:bg-success/20\",\r\n },\r\n {\r\n color: \"info\" as (typeof color)[number],\r\n variant: \"soft\" as (typeof variant)[number],\r\n class: \"bg-info/10 text-info hover:bg-info/20\",\r\n },\r\n {\r\n color: \"warning\" as (typeof color)[number],\r\n variant: \"soft\" as (typeof variant)[number],\r\n class: \"bg-warning/10 text-warning hover:bg-warning/20\",\r\n },\r\n {\r\n color: \"error\" as (typeof color)[number],\r\n variant: \"soft\" as (typeof variant)[number],\r\n class: \"bg-error/10 text-error hover:bg-error/20\",\r\n },\r\n {\r\n color: \"neutral\" as (typeof color)[number],\r\n variant: \"soft\" as (typeof variant)[number],\r\n class: \"bg-neutral/10 text-neutral hover:bg-neutral/20\",\r\n },\r\n\r\n // Subtle Variants (Transparent bg, colored text, hover light bg)\r\n {\r\n color: \"primary\" as (typeof color)[number],\r\n variant: \"subtle\" as (typeof variant)[number],\r\n // class: \"bg-transparent text-primary hover:bg-primary/10\",\r\n class: \"bg-primary/10 ring-1 ring-inset ring-primary text-primary hover:bg-primary/20\",\r\n },\r\n {\r\n color: \"secondary\" as (typeof color)[number],\r\n variant: \"subtle\" as (typeof variant)[number],\r\n class:\r\n \"bg-secondary/10 ring-1 ring-inset ring-secondary text-secondary hover:bg-secondary/20\",\r\n },\r\n {\r\n color: \"success\" as (typeof color)[number],\r\n variant: \"subtle\" as (typeof variant)[number],\r\n class: \"bg-success/10 ring-1 ring-inset ring-success text-success hover:bg-success/20\",\r\n },\r\n {\r\n color: \"info\" as (typeof color)[number],\r\n variant: \"subtle\" as (typeof variant)[number],\r\n class: \"bg-info/10 ring-1 ring-inset ring-info text-info hover:bg-info/20\",\r\n },\r\n {\r\n color: \"warning\" as (typeof color)[number],\r\n variant: \"subtle\" as (typeof variant)[number],\r\n class: \"bg-warning/10 ring-1 ring-inset ring-warning text-warning hover:bg-warning/20\",\r\n },\r\n {\r\n color: \"error\" as (typeof color)[number],\r\n variant: \"subtle\" as (typeof variant)[number],\r\n class: \"bg-error/10 ring-1 ring-inset ring-error text-error hover:bg-error/20\",\r\n },\r\n {\r\n color: \"neutral\" as (typeof color)[number],\r\n variant: \"subtle\" as (typeof variant)[number],\r\n class: \"bg-neutral/10 ring-1 ring-inset ring-neutral text-neutral hover:bg-neutral/20\",\r\n },\r\n ],\r\n defaultVariants: {\r\n color: \"primary\" as (typeof color)[number],\r\n variant: \"solid\" as (typeof variant)[number],\r\n size: \"md\" as (typeof size)[number],\r\n },\r\n};\r\n"
12
+ },
13
+ {
14
+ "path": "RebornButton.vue",
15
+ "content": "<script setup lang=\"ts\">\r\nimport { computed, toRef } from 'vue'\r\nimport theme, { buttonColors, buttonVariants, buttonSizes } from './reborn-button.config'\r\nimport { useFieldGroup } from '~/composables/useFieldGroup'\r\nimport { tv } from '~/lib/tv'\r\n\r\nconst b = tv(theme)\r\n\r\nexport interface ButtonProps {\r\n label?: string\r\n color?: typeof buttonColors[number]\r\n variant?: typeof buttonVariants[number]\r\n size?: typeof buttonSizes[number]\r\n loading?: boolean\r\n disabled?: boolean\r\n square?: boolean\r\n class?: any\r\n ui?: any\r\n}\r\n\r\nconst props = withDefaults(defineProps<ButtonProps>(), {\r\n color: 'primary',\r\n variant: 'solid',\r\n size: 'md',\r\n loading: false,\r\n disabled: false,\r\n square: false\r\n})\r\n\r\nconst slots = defineSlots<{\r\n leading(props: { ui: any }): any\r\n default(props: { ui: any }): any\r\n trailing(props: { ui: any }): any\r\n}>()\r\n\r\nconst { orientation, size: fieldGroupSize } = useFieldGroup(props)\r\n\r\nconst isDisabled = computed(() => props.disabled || props.loading)\r\n\r\nconst color = toRef(props, 'color')\r\nconst variant = toRef(props, 'variant')\r\nconst size = toRef(props, 'size')\r\nconst square = toRef(props, 'square')\r\n\r\nconst ui = computed(() => b({\r\n color: color.value,\r\n variant: variant.value,\r\n size: (fieldGroupSize.value || size.value) as any,\r\n square: square.value,\r\n fieldGroup: orientation.value\r\n}))\r\n</script>\r\n\r\n<template>\r\n <button :disabled=\"isDisabled\" :class=\"ui.base({ class: props.class })\" v-bind=\"$attrs\">\r\n <slot name=\"leading\" :ui=\"ui\">\r\n <Icon name=\"svg-spinners:270-ring\" v-if=\"props.loading\" :class=\"ui.leadingIcon()\" />\r\n </slot>\r\n\r\n <slot :ui=\"ui\">\r\n <span v-if=\"label\" :class=\"ui.base()\">\r\n {{ label }}\r\n </span>\r\n <slot v-else :ui=\"ui\" />\r\n </slot>\r\n\r\n\r\n <slot name=\"trailing\" :ui=\"ui\" />\r\n </button>\r\n</template>\r\n",
16
+ "target": "web"
17
+ },
18
+ {
19
+ "path": "RebornButton.vue",
20
+ "content": "<script setup lang=\"ts\">\r\nimport { computed, toRef } from 'vue'\r\nimport theme, { buttonColors, buttonVariants, buttonSizes } from './reborn-button.config'\r\nimport { useFieldGroup } from '~/composables/useFieldGroup'\r\nimport { tv } from '~/lib/tv'\r\n\r\nconst b = tv(theme)\r\n\r\nexport interface ButtonProps {\r\n label?: string\r\n color?: typeof buttonColors[number]\r\n variant?: typeof buttonVariants[number]\r\n size?: typeof buttonSizes[number]\r\n loading?: boolean\r\n disabled?: boolean\r\n square?: boolean\r\n class?: any\r\n ui?: any\r\n}\r\n\r\nconst props = withDefaults(defineProps<ButtonProps>(), {\r\n color: 'primary',\r\n variant: 'solid',\r\n size: 'md',\r\n loading: false,\r\n disabled: false,\r\n square: false\r\n})\r\n\r\nconst slots = defineSlots<{\r\n leading(props: { ui: any }): any\r\n default(props: { ui: any }): any\r\n trailing(props: { ui: any }): any\r\n}>()\r\n\r\nconst { orientation, size: fieldGroupSize } = useFieldGroup(props)\r\n\r\nconst isDisabled = computed(() => props.disabled || props.loading)\r\n\r\nconst color = toRef(props, 'color')\r\nconst variant = toRef(props, 'variant')\r\nconst size = toRef(props, 'size')\r\nconst square = toRef(props, 'square')\r\n\r\nconst ui = computed(() => b({\r\n color: color.value,\r\n variant: variant.value,\r\n size: (fieldGroupSize.value || size.value) as any,\r\n square: square.value,\r\n fieldGroup: orientation.value\r\n}))\r\n</script>\r\n\r\n<template>\r\n <button :disabled=\"isDisabled\" :class=\"ui.base({ class: props.class })\" v-bind=\"$attrs\">\r\n <slot name=\"leading\" :ui=\"ui\">\r\n <Icon name=\"svg-spinners:270-ring\" v-if=\"props.loading\" :class=\"ui.leadingIcon()\" />\r\n </slot>\r\n\r\n <slot :ui=\"ui\">\r\n <text v-if=\"label\" :class=\"ui.base()\">\r\n {{ label }}\r\n </text>\r\n <slot v-else :ui=\"ui\" />\r\n </slot>\r\n\r\n\r\n <slot name=\"trailing\" :ui=\"ui\" />\r\n </button>\r\n</template>\r\n",
21
+ "target": "uniapp"
22
+ }
23
+ ],
24
+ "fileCount": 4,
25
+ "contentHash": "b09bf4831bf6c79d34c24b969d87180b8d35839b"
26
+ }
@@ -0,0 +1,28 @@
1
+ {
2
+ "name": "reborn-checkbox",
3
+ "dependencies": [
4
+ "clsx"
5
+ ],
6
+ "files": [
7
+ {
8
+ "path": "index.ts",
9
+ "content": "export { default as RebornCheckbox } from \"./RebornCheckbox.vue\";\r\n"
10
+ },
11
+ {
12
+ "path": "reborn-checkbox.config.ts",
13
+ "content": "const size = [\"sm\", \"md\", \"lg\"] as const;\r\nconst color = [\"primary\", \"secondary\", \"success\", \"info\", \"warning\", \"error\", \"neutral\"] as const;\r\n\r\nexport { size as checkboxSizes, color as checkboxColors };\r\n\r\nexport default {\r\n slots: {\r\n wrapper: \"inline-flex items-center gap-3 cursor-pointer select-none\",\r\n input: \"peer sr-only\",\r\n control:\r\n \"flex items-center justify-center rounded-md border border-gray-4 bg-white text-white transition-colors ring-1 ring-transparent peer-focus-visible:ring-2 peer-focus-visible:ring-primary/40 peer-disabled:cursor-not-allowed peer-disabled:bg-gray-2 peer-disabled:border-gray-3 peer-checked:[&>*]:opacity-100 peer-checked:[&>*]:scale-100\",\r\n icon: \"size-4 opacity-0 scale-75 transition-all\",\r\n label: \"text-gray-8 dark:text-gray-2\",\r\n },\r\n variants: {\r\n size: {\r\n sm: {\r\n control: \"size-4\",\r\n label: \"text-[length:var(--text-size-24)]\",\r\n },\r\n md: {\r\n control: \"size-5\",\r\n label: \"text-[length:var(--text-size-26)]\",\r\n },\r\n lg: {\r\n control: \"size-6\",\r\n label: \"text-[length:var(--text-size-28)]\",\r\n },\r\n },\r\n color: {\r\n primary: {\r\n control: \"peer-checked:bg-primary peer-checked:border-primary\",\r\n icon: \"text-white\",\r\n },\r\n secondary: {\r\n control: \"peer-checked:bg-secondary peer-checked:border-secondary\",\r\n icon: \"text-white\",\r\n },\r\n success: {\r\n control: \"peer-checked:bg-success peer-checked:border-success\",\r\n icon: \"text-white\",\r\n },\r\n info: {\r\n control: \"peer-checked:bg-info peer-checked:border-info\",\r\n icon: \"text-white\",\r\n },\r\n warning: {\r\n control: \"peer-checked:bg-warning peer-checked:border-warning\",\r\n icon: \"text-white\",\r\n },\r\n error: {\r\n control: \"peer-checked:bg-error peer-checked:border-error\",\r\n icon: \"text-white\",\r\n },\r\n neutral: {\r\n control: \"peer-checked:bg-neutral peer-checked:border-neutral\",\r\n icon: \"text-white\",\r\n },\r\n },\r\n },\r\n defaultVariants: {\r\n size: \"md\" as (typeof size)[number],\r\n color: \"primary\" as (typeof color)[number],\r\n },\r\n};\r\n"
14
+ },
15
+ {
16
+ "path": "RebornCheckbox.vue",
17
+ "content": "<script setup lang=\"ts\">\r\nimport { computed, ref, useAttrs, watch } from \"vue\";\r\nimport type { ClassValue } from \"clsx\";\r\nimport { cn } from \"~/lib/utils\";\r\nimport theme, { checkboxColors, checkboxSizes } from \"./reborn-checkbox.config\";\r\nimport { tv } from \"~/lib/tv\";\r\n\r\nconst b = tv(theme);\r\n\r\ndefineOptions({\r\n inheritAttrs: false,\r\n});\r\n\r\nexport type CheckboxValue = string | number;\r\n\r\nexport interface CheckboxProps {\r\n modelValue?: boolean | CheckboxValue[];\r\n defaultValue?: boolean | CheckboxValue[];\r\n value?: CheckboxValue;\r\n label?: string;\r\n disabled?: boolean;\r\n size?: typeof checkboxSizes[number];\r\n color?: typeof checkboxColors[number];\r\n class?: any;\r\n ui?: Partial<{\r\n wrapper: ClassValue;\r\n input: ClassValue;\r\n control: ClassValue;\r\n icon: ClassValue;\r\n label: ClassValue;\r\n }>;\r\n}\r\n\r\nconst props = withDefaults(defineProps<CheckboxProps>(), {\r\n disabled: false,\r\n size: \"md\",\r\n color: \"primary\",\r\n});\r\n\r\nconst emit = defineEmits<{\r\n (e: \"update:modelValue\", value: boolean | CheckboxValue[]): void;\r\n}>();\r\n\r\nconst attrs = useAttrs();\r\n\r\nconst localValue = ref<boolean | CheckboxValue[]>(props.defaultValue ?? false);\r\nconst currentValue = computed(() => (props.modelValue !== undefined ? props.modelValue : localValue.value));\r\nconst optionValue = computed<CheckboxValue>(() => props.value ?? props.label ?? \"\");\r\n\r\nconst isChecked = computed(() => {\r\n if (Array.isArray(currentValue.value)) {\r\n return currentValue.value.includes(optionValue.value);\r\n }\r\n\r\n return Boolean(currentValue.value);\r\n});\r\n\r\nconst uiOverrides = computed(() => props.ui || {});\r\n\r\nconst ui = computed(() => {\r\n const styles = b({\r\n size: props.size,\r\n color: props.color,\r\n });\r\n\r\n return {\r\n wrapper: (opts?: { class?: any }) => styles.wrapper({ class: cn(opts?.class, uiOverrides.value.wrapper) }),\r\n input: (opts?: { class?: any }) => styles.input({ class: cn(opts?.class, uiOverrides.value.input) }),\r\n control: (opts?: { class?: any }) => styles.control({ class: cn(opts?.class, uiOverrides.value.control) }),\r\n icon: (opts?: { class?: any }) => styles.icon({ class: cn(opts?.class, uiOverrides.value.icon) }),\r\n label: (opts?: { class?: any }) => styles.label({ class: cn(opts?.class, uiOverrides.value.label) }),\r\n };\r\n});\r\n\r\nconst inputAttrs = computed(() => {\r\n const { class: _class, ...rest } = attrs;\r\n return rest;\r\n});\r\n\r\nfunction updateValue(nextValue: boolean | CheckboxValue[]) {\r\n if (props.modelValue === undefined) {\r\n localValue.value = nextValue;\r\n }\r\n emit(\"update:modelValue\", nextValue);\r\n}\r\n\r\nfunction handleChange(event: Event) {\r\n const target = event.target as HTMLInputElement;\r\n const checked = target.checked;\r\n\r\n if (Array.isArray(currentValue.value)) {\r\n const next = new Set(currentValue.value);\r\n if (checked) {\r\n next.add(optionValue.value);\r\n } else {\r\n next.delete(optionValue.value);\r\n }\r\n updateValue(Array.from(next));\r\n return;\r\n }\r\n\r\n updateValue(checked);\r\n}\r\n\r\nwatch(\r\n () => props.modelValue,\r\n (value) => {\r\n if (value !== undefined) {\r\n localValue.value = value;\r\n }\r\n },\r\n);\r\n</script>\r\n\r\n<template>\r\n <label :class=\"ui.wrapper({ class: props.class })\" :data-disabled=\"props.disabled\">\r\n <input v-bind=\"inputAttrs\" type=\"checkbox\" :value=\"optionValue\" :checked=\"isChecked\" :disabled=\"props.disabled\"\r\n :class=\"ui.input()\" @change=\"handleChange\" />\r\n\r\n <span :class=\"ui.control()\">\r\n <slot name=\"icon\" :checked=\"isChecked\">\r\n <Icon name=\"lucide:check\" :class=\"ui.icon()\" />\r\n </slot>\r\n </span>\r\n\r\n <span v-if=\"props.label || $slots.default\" :class=\"ui.label()\">\r\n <slot>{{ props.label }}</slot>\r\n </span>\r\n </label>\r\n</template>\r\n",
18
+ "target": "web"
19
+ },
20
+ {
21
+ "path": "RebornCheckbox.vue",
22
+ "content": "<script setup lang=\"ts\">\r\nimport { computed, ref, useAttrs, watch } from \"vue\";\r\nimport type { ClassValue } from \"clsx\";\r\nimport { cn } from \"~/lib/utils\";\r\nimport theme, { checkboxColors, checkboxSizes } from \"./reborn-checkbox.config\";\r\nimport { tv } from \"~/lib/tv\";\r\n\r\nconst b = tv(theme);\r\n\r\ndefineOptions({\r\n inheritAttrs: false,\r\n});\r\n\r\nexport type CheckboxValue = string | number;\r\n\r\nexport interface CheckboxProps {\r\n modelValue?: boolean | CheckboxValue[];\r\n defaultValue?: boolean | CheckboxValue[];\r\n value?: CheckboxValue;\r\n label?: string;\r\n disabled?: boolean;\r\n size?: typeof checkboxSizes[number];\r\n color?: typeof checkboxColors[number];\r\n class?: any;\r\n ui?: Partial<{\r\n wrapper: ClassValue;\r\n input: ClassValue;\r\n control: ClassValue;\r\n icon: ClassValue;\r\n label: ClassValue;\r\n }>;\r\n}\r\n\r\nconst props = withDefaults(defineProps<CheckboxProps>(), {\r\n disabled: false,\r\n size: \"md\",\r\n color: \"primary\",\r\n});\r\n\r\nconst emit = defineEmits<{\r\n (e: \"update:modelValue\", value: boolean | CheckboxValue[]): void;\r\n}>();\r\n\r\nconst attrs = useAttrs();\r\n\r\nconst localValue = ref<boolean | CheckboxValue[]>(props.defaultValue ?? false);\r\nconst currentValue = computed(() => (props.modelValue !== undefined ? props.modelValue : localValue.value));\r\nconst optionValue = computed<CheckboxValue>(() => props.value ?? props.label ?? \"\");\r\n\r\nconst isChecked = computed(() => {\r\n if (Array.isArray(currentValue.value)) {\r\n return currentValue.value.includes(optionValue.value);\r\n }\r\n\r\n return Boolean(currentValue.value);\r\n});\r\n\r\nconst uiOverrides = computed(() => props.ui || {});\r\n\r\nconst ui = computed(() => {\r\n const styles = b({\r\n size: props.size,\r\n color: props.color,\r\n });\r\n\r\n return {\r\n wrapper: (opts?: { class?: any }) => styles.wrapper({ class: cn(opts?.class, uiOverrides.value.wrapper) }),\r\n input: (opts?: { class?: any }) => styles.input({ class: cn(opts?.class, uiOverrides.value.input) }),\r\n control: (opts?: { class?: any }) => styles.control({ class: cn(opts?.class, uiOverrides.value.control) }),\r\n icon: (opts?: { class?: any }) => styles.icon({ class: cn(opts?.class, uiOverrides.value.icon) }),\r\n label: (opts?: { class?: any }) => styles.label({ class: cn(opts?.class, uiOverrides.value.label) }),\r\n };\r\n});\r\n\r\nconst inputAttrs = computed(() => {\r\n const { class: _class, ...rest } = attrs;\r\n return rest;\r\n});\r\n\r\nfunction updateValue(nextValue: boolean | CheckboxValue[]) {\r\n if (props.modelValue === undefined) {\r\n localValue.value = nextValue;\r\n }\r\n emit(\"update:modelValue\", nextValue);\r\n}\r\n\r\nfunction handleChange(event: Event) {\r\n const target = event.target as HTMLInputElement;\r\n const checked = target.checked;\r\n\r\n if (Array.isArray(currentValue.value)) {\r\n const next = new Set(currentValue.value);\r\n if (checked) {\r\n next.add(optionValue.value);\r\n } else {\r\n next.delete(optionValue.value);\r\n }\r\n updateValue(Array.from(next));\r\n return;\r\n }\r\n\r\n updateValue(checked);\r\n}\r\n\r\nwatch(\r\n () => props.modelValue,\r\n (value) => {\r\n if (value !== undefined) {\r\n localValue.value = value;\r\n }\r\n },\r\n);\r\n</script>\r\n\r\n<template>\r\n <label :class=\"ui.wrapper({ class: props.class })\" :data-disabled=\"props.disabled\">\r\n <input v-bind=\"inputAttrs\" type=\"checkbox\" :value=\"optionValue\" :checked=\"isChecked\" :disabled=\"props.disabled\"\r\n :class=\"ui.input()\" @change=\"handleChange\" />\r\n\r\n <text :class=\"ui.control()\">\r\n <slot name=\"icon\" :checked=\"isChecked\">\r\n <Icon name=\"lucide:check\" :class=\"ui.icon()\" />\r\n </slot>\r\n </text>\r\n\r\n <text v-if=\"props.label || $slots.default\" :class=\"ui.label()\">\r\n <slot>{{ props.label }}</slot>\r\n </text>\r\n </label>\r\n</template>\r\n",
23
+ "target": "uniapp"
24
+ }
25
+ ],
26
+ "fileCount": 4,
27
+ "contentHash": "eee5a72eaee4187dadac6673b695fc655eee6406"
28
+ }