@nuxt/ui
Version:
A UI Library for Modern Web Apps, powered by Vue & Tailwind CSS.
326 lines (316 loc) • 10.7 kB
JavaScript
import { fileURLToPath } from 'node:url';
import { join, normalize } from 'pathe';
import { createUnplugin } from 'unplugin';
import { defu } from 'defu';
import tailwind from '@tailwindcss/vite';
import { g as getTemplates, d as defaultOptions, r as resolveColors, a as getDefaultConfig } from './shared/ui.DegC4tw1.mjs';
import fs from 'node:fs';
import path from 'node:path';
import MagicString from 'magic-string';
import { genSafeVariableName } from 'knitwork';
import { resolvePathSync } from 'mlly';
import { globSync } from 'tinyglobby';
import AutoImportComponents from 'unplugin-vue-components';
import AutoImport from 'unplugin-auto-import';
import '../dist/runtime/utils/index.js';
import 'scule';
import 'tailwindcss/colors';
import '@nuxt/kit';
import 'node:fs/promises';
function TemplatePlugin(options, appConfig) {
const templates = getTemplates(options, appConfig.ui);
const templateKeys = new Set(templates.map((t) => `#build/${t.filename}`));
async function writeTemplates(root) {
const map = {};
const dir = path.join(root, "node_modules", ".nuxt-ui");
for (const template of templates) {
if (!template.write || !template.filename) {
continue;
}
const filePath = path.join(dir, template.filename);
if (!fs.existsSync(path.dirname(filePath))) {
fs.mkdirSync(path.dirname(filePath), { recursive: true });
}
fs.writeFileSync(filePath, await template.getContents({}));
map[`#build/${template.filename}`] = filePath;
}
return map;
}
return {
name: "nuxt:ui:templates",
enforce: "pre",
vite: {
async config(config) {
const alias = await writeTemplates(config.root || process.cwd());
return {
resolve: {
alias
}
};
}
},
resolveId(id) {
if (templateKeys.has(id + ".ts")) {
return id.replace("#build/", "virtual:nuxt-ui-templates/") + ".ts";
}
},
loadInclude: (id) => templateKeys.has(id.replace("virtual:nuxt-ui-templates/", "#build/")),
load(id) {
id = id.replace("virtual:nuxt-ui-templates/", "#build/");
return templates.find((t) => `#build/${t.filename}` === id).getContents({});
}
};
}
function PluginsPlugin(options) {
const plugins = globSync(["**/*", "!*.d.ts"], { cwd: join(runtimeDir, "plugins"), absolute: true });
plugins.unshift(resolvePathSync("../runtime/vue/plugins/head", { extensions: [".ts", ".mjs", ".js"], url: import.meta.url }));
if (options.colorMode) {
plugins.push(resolvePathSync("../runtime/vue/plugins/color-mode", { extensions: [".ts", ".mjs", ".js"], url: import.meta.url }));
}
return {
name: "nuxt:ui:plugins",
enforce: "pre",
resolveId(id) {
if (id === "@nuxt/ui/vue-plugin") {
return "virtual:nuxt-ui-plugins";
}
},
transform(code, id) {
if (plugins.some((p) => id.startsWith(p)) && code.includes("import.meta.client")) {
const s = new MagicString(code);
s.replaceAll("import.meta.client", "true");
if (s.hasChanged()) {
return {
code: s.toString(),
map: s.generateMap({ hires: true })
};
}
}
},
loadInclude: (id) => id === "virtual:nuxt-ui-plugins",
load() {
return `
${plugins.map((p) => `import ${genSafeVariableName(p)} from "${p}"`).join("\n")}
export default {
install (app) {
${plugins.map((p) => ` app.use(${genSafeVariableName(p)})`).join("\n")}
}
}
`;
},
// Argument Vite specific configuration
vite: {
config() {
return {
// Opt-out Nuxt UI from Vite's pre-bundling,
// as we need Vite's pipeline to resolve imports like `#imports`
optimizeDeps: {
exclude: ["@nuxt/ui"]
}
};
}
}
};
}
function AppConfigPlugin(_options, appConfig) {
return {
name: "nuxt:ui:app-config",
enforce: "pre",
resolveId(id) {
if (id === "#build/app.config") {
return "virtual:nuxt-ui-app-config";
}
},
loadInclude: (id) => id === "virtual:nuxt-ui-app-config",
load() {
return `
export default ${JSON.stringify(appConfig)}
`;
},
vite: {
config() {
return {
test: {
server: {
deps: {
inline: ["@nuxt/ui"]
}
}
}
};
}
}
};
}
function resolveRouterMode(options) {
if (options.router === false) {
return "none";
}
if (options.router === "inertia") {
return "inertia";
}
if (options.router === void 0 && options.inertia === true) {
return "inertia";
}
return "vue-router";
}
function createComponentSource(cwd, prefix, ignore = []) {
const files = globSync("**/*.vue", { cwd, ignore: ignore.filter(Boolean) });
const names = new Set(files.map((c) => `${prefix}${c.split("/").pop()?.replace(/\.vue$/, "")}`));
const paths = new Map(files.map((c) => {
const componentName = `${prefix}${c.split("/").pop()?.replace(/\.vue$/, "")}`;
return [componentName, c];
}));
return {
has: (name) => names.has(name),
resolve: (componentName) => {
const relativePath = paths.get(componentName);
if (!relativePath) return;
return { name: "default", from: join(cwd, relativePath) };
},
resolveFile: (filename) => {
const componentName = `${prefix}${filename}`;
const relativePath = paths.get(componentName);
if (!relativePath) return;
return join(cwd, relativePath);
}
};
}
function ComponentImportPlugin(options, meta) {
const colorModeIgnore = !options.colorMode ? ["color-mode/**/*.vue"] : [];
const routerMode = resolveRouterMode(options);
const routerOverrides = {
"vue-router": createComponentSource(join(runtimeDir, "vue/overrides/vue-router"), options.prefix),
"inertia": createComponentSource(join(runtimeDir, "vue/overrides/inertia"), options.prefix),
"none": createComponentSource(join(runtimeDir, "vue/overrides/none"), options.prefix)
};
const unpluginComponents = createComponentSource(
join(runtimeDir, "vue/components"),
options.prefix,
colorModeIgnore
);
const defaultComponents = createComponentSource(
join(runtimeDir, "components"),
options.prefix,
[...colorModeIgnore, "content/*.vue", "prose/**/*.vue"]
);
const sources = [routerOverrides[routerMode], unpluginComponents, defaultComponents].filter((s) => !!s);
const packagesToScan = [
"@nuxt/ui",
"@compodium/examples",
...Array.isArray(options.scanPackages) ? options.scanPackages : []
];
const escapeRegex = (str) => str.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
const packagesRegex = packagesToScan.map(escapeRegex).join("|");
const excludeRegex = new RegExp(`[\\\\/]node_modules[\\\\/](?!\\.pnpm|${packagesRegex})`);
const pluginOptions = defu(options.components, {
dts: options.dts ?? true,
exclude: [
excludeRegex,
/[\\/]\.git[\\/]/,
/[\\/]\.nuxt[\\/]/
],
resolvers: [
(componentName) => {
for (const source of sources) {
const resolved = source.resolve(componentName);
if (resolved) return resolved;
}
}
]
});
return [
/**
* This plugin aims to ensure we override certain components with Vue-compatible versions:
* <UIcon> and <ULink> currently.
*/
{
name: "nuxt:ui:components",
enforce: "pre",
resolveId(id, importer) {
if (!importer || !normalize(importer).includes(runtimeDir)) {
return;
}
if (!RELATIVE_IMPORT_RE.test(id)) {
return;
}
const filename = id.match(/([^/]+)\.vue$/)?.[1];
if (filename) {
for (const source of sources) {
const resolved = source.resolveFile(filename);
if (resolved) return resolved;
}
}
}
},
AutoImportComponents.raw(pluginOptions, meta)
];
}
const RELATIVE_IMPORT_RE = /^\.{1,2}\//;
function NuxtEnvironmentPlugin(options) {
const routerMode = resolveRouterMode(options);
const stubsPath = `../runtime/vue/stubs/${routerMode}`;
const stubPath = resolvePathSync(stubsPath, { extensions: [".ts", ".mjs", ".js"], url: import.meta.url });
return {
name: "nuxt:ui",
enforce: "pre",
resolveId(id) {
if (id === "#imports") {
return stubPath;
}
},
transformInclude(id) {
return normalize(id).includes(runtimeDir);
},
transform(code) {
if (code.includes("import.meta.client")) {
const s = new MagicString(code);
s.replaceAll("import.meta.client", "true");
if (s.hasChanged()) {
return {
code: s.toString(),
map: s.generateMap({ hires: true })
};
}
}
}
};
}
function AutoImportPlugin(options, meta) {
const pluginOptions = defu(options.autoImport, {
dts: options.dts ?? true,
dirs: [join(runtimeDir, "composables"), join(runtimeDir, "vue/composables")]
});
return AutoImport.raw(pluginOptions, meta);
}
const runtimeDir = normalize(fileURLToPath(new URL("./runtime", import.meta.url)));
const NuxtUIPlugin = createUnplugin((_options = {}, meta) => {
const options = defu(_options, { fonts: false }, defaultOptions);
options.theme = options.theme || {};
options.theme.colors = resolveColors(options.theme.colors);
const appConfig = defu({ ui: options.ui, colorMode: options.colorMode }, { ui: getDefaultConfig(options.theme) });
return [
NuxtEnvironmentPlugin(options),
ComponentImportPlugin(options, meta),
AutoImportPlugin(options, meta),
tailwind(),
PluginsPlugin(options),
TemplatePlugin(options, appConfig),
AppConfigPlugin(options, appConfig),
{
name: "nuxt:ui:plugins-duplication-detection",
vite: {
configResolved(config) {
const plugins = config.plugins || [];
if (plugins.filter((i) => i.name === "unplugin-auto-import").length > 1) {
throw new Error("[Nuxt UI] Multiple instances of `unplugin-auto-import` detected. Nuxt UI includes `unplugin-auto-import` already, and you can configure it using `autoImport` option in Nuxt UI module options.");
}
if (plugins.filter((i) => i.name === "unplugin-vue-components").length > 1) {
throw new Error("[Nuxt UI] Multiple instances of `unplugin-vue-components` detected. Nuxt UI includes `unplugin-vue-components` already, and you can configure it using `components` option in Nuxt UI module options.");
}
}
}
}
].flat(1);
});
export { NuxtUIPlugin, runtimeDir };