@salty-css/core 0.0.1-alpha.28 → 0.0.1-alpha.280
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/.saltyrc.schema.json +48 -0
- package/README.md +469 -26
- package/bin/logger.d.ts +1 -0
- package/bin/main.cjs +8 -5
- package/bin/main.js +230 -100
- package/cache/resolve-dynamic-config-cache.cjs +1 -0
- package/cache/resolve-dynamic-config-cache.d.ts +1 -0
- package/cache/resolve-dynamic-config-cache.js +10 -0
- package/compiler/get-files.d.ts +3 -0
- package/compiler/get-function-range.d.ts +1 -0
- package/compiler/index.cjs +1 -11
- package/compiler/index.d.ts +19 -11
- package/compiler/index.js +19 -172
- package/config/define-config.d.ts +1 -1
- package/config/index.cjs +1 -1
- package/config/index.d.ts +2 -1
- package/config/index.js +12 -2
- package/css/index.cjs +1 -1
- package/css/index.d.ts +4 -1
- package/css/index.js +10 -2
- package/css/keyframes.cjs +1 -0
- package/css/keyframes.d.ts +22 -0
- package/css/keyframes.js +34 -0
- package/css/media.cjs +1 -0
- package/css/media.d.ts +87 -0
- package/css/media.js +88 -0
- package/css/merge.cjs +1 -0
- package/css/merge.d.ts +7 -0
- package/css/merge.js +5 -0
- package/css/token.cjs +1 -0
- package/css/token.d.ts +1 -0
- package/css/token.js +4 -0
- package/{dash-case-DKzpenwY.cjs → dash-case-BJEkFEGQ.cjs} +1 -1
- package/{dash-case-DMQMcCO6.js → dash-case-DBThphLm.js} +2 -2
- package/define-templates-4A2yHcMF.js +52 -0
- package/define-templates-Cunsb_Tr.cjs +1 -0
- package/factories/define-global-styles.d.ts +7 -0
- package/factories/define-media-query.d.ts +8 -0
- package/factories/define-templates.d.ts +27 -0
- package/factories/define-variables.d.ts +12 -0
- package/factories/index.cjs +1 -0
- package/factories/index.d.ts +4 -0
- package/factories/index.js +30 -0
- package/generators/class-name-generator.d.ts +6 -0
- package/generators/index.cjs +1 -0
- package/generators/index.d.ts +2 -0
- package/generators/index.js +88 -0
- package/generators/styled-generator.d.ts +20 -0
- package/generators/styles-generator.d.ts +22 -0
- package/helpers/color.d.ts +18 -0
- package/helpers/index.cjs +1 -0
- package/helpers/index.d.ts +2 -0
- package/helpers/index.js +1154 -0
- package/helpers/viewport-clamp.d.ts +8 -0
- package/index-BgHtTuZf.cjs +41 -0
- package/index-CXjwv_Dc.js +519 -0
- package/package.json +54 -6
- package/parse-styles-Bdgw_4ME.cjs +5 -0
- package/parse-styles-C1E6ETeH.js +154 -0
- package/parsers/index.cjs +2 -0
- package/parsers/index.d.ts +5 -0
- package/parsers/index.js +33 -0
- package/parsers/parse-modifiers.d.ts +3 -0
- package/parsers/parse-styles.d.ts +13 -0
- package/parsers/parse-templates.d.ts +4 -0
- package/parsers/parse-tokens.d.ts +3 -0
- package/parsers/parser-types.d.ts +8 -0
- package/parsers/property-name-check.d.ts +1 -0
- package/parsers/unit-check.d.ts +7 -0
- package/react-vanilla-file-CCXbsjIb.js +18 -0
- package/react-vanilla-file-CG_WJLam.cjs +15 -0
- package/{salty.config-D9ANEDiH.js → salty.config-BhBY_oOk.js} +1 -0
- package/{salty.config-BupieCfE.cjs → salty.config-Dk6ZcCxI.cjs} +3 -2
- package/server/index.cjs +1 -0
- package/server/index.d.ts +1 -0
- package/server/index.js +4 -0
- package/server/should-restart.d.ts +1 -0
- package/should-restart-BELKpXEw.js +12 -0
- package/should-restart-CBwAaej8.cjs +1 -0
- package/templates/salty-reset.d.ts +2 -0
- package/types/cli-types.d.ts +10 -0
- package/types/config-types.d.ts +85 -0
- package/types/index.d.ts +53 -22
- package/util/dot-case.d.ts +1 -0
- package/util/index.cjs +1 -1
- package/util/index.js +1 -1
- package/util/module-type.d.ts +1 -0
- package/viewport-clamp-BOc-8Oph.js +7 -0
- package/viewport-clamp-kY8JqYzm.cjs +1 -0
- package/config/config-types.d.ts +0 -59
- package/generator/index.cjs +0 -1
- package/generator/index.d.ts +0 -1
- package/generator/index.js +0 -61
- package/generator/parse-modifiers.d.ts +0 -3
- package/generator/parse-styles.d.ts +0 -2
- package/generator/parse-templates.d.ts +0 -2
- package/generator/parse-tokens.d.ts +0 -2
- package/generator/parser-types.d.ts +0 -4
- package/generator/style-generator.d.ts +0 -28
- package/parse-templates-D4p3pgQR.js +0 -92
- package/parse-templates-W0YfTmOT.cjs +0 -8
@@ -0,0 +1,519 @@
|
|
1
|
+
import * as Xg from "esbuild";
|
2
|
+
import { execSync as Hg } from "child_process";
|
3
|
+
import { t as Q, d as E } from "./dash-case-DBThphLm.js";
|
4
|
+
import { join as b, parse as gg } from "path";
|
5
|
+
import { existsSync as ag, writeFileSync as p, mkdirSync as f, readFileSync as F, statSync as Kg, readdirSync as zg } from "fs";
|
6
|
+
import { readFile as pg } from "fs/promises";
|
7
|
+
import { p as Yg, a as wg } from "./parse-styles-C1E6ETeH.js";
|
8
|
+
import { parseTemplates as Sg, getTemplateTypes as vg } from "./parsers/index.js";
|
9
|
+
import { createLogger as kg, format as ng, transports as jg } from "winston";
|
10
|
+
import { mergeObjects as U, mergeFactories as xg } from "./css/merge.js";
|
11
|
+
import { d as fg } from "./define-templates-4A2yHcMF.js";
|
12
|
+
import tg from "typescript";
|
13
|
+
const Vg = (g) => {
|
14
|
+
if (!g || g === "/") throw new Error("Could not find package.json file");
|
15
|
+
const I = b(g, "package.json");
|
16
|
+
return ag(I) ? I : Vg(b(g, ".."));
|
17
|
+
}, Ug = async (g) => {
|
18
|
+
const I = Vg(g);
|
19
|
+
return await pg(I, "utf-8").then(JSON.parse).catch(() => {
|
20
|
+
});
|
21
|
+
}, Qg = async (g) => {
|
22
|
+
const I = await Ug(g);
|
23
|
+
if (I)
|
24
|
+
return I.type;
|
25
|
+
};
|
26
|
+
let x;
|
27
|
+
const Ng = async (g) => {
|
28
|
+
if (x) return x;
|
29
|
+
const I = await Qg(g);
|
30
|
+
return I === "module" ? x = "esm" : (I === "commonjs" || import.meta.url.endsWith(".cjs")) && (x = "cjs"), x || "esm";
|
31
|
+
}, _ = kg({
|
32
|
+
level: "debug",
|
33
|
+
format: ng.combine(ng.colorize(), ng.cli()),
|
34
|
+
transports: [new jg.Console({})]
|
35
|
+
}), tI = (g) => {
|
36
|
+
_.error(g);
|
37
|
+
};
|
38
|
+
function hg(g) {
|
39
|
+
return g ? typeof g != "string" ? hg(String(g)) : g.replace(/[^\d\w]/g, ".") : "";
|
40
|
+
}
|
41
|
+
const Tg = {
|
42
|
+
/** Set box model to border-box */
|
43
|
+
"*, *::before, *::after": {
|
44
|
+
boxSizing: "border-box"
|
45
|
+
},
|
46
|
+
/** Remove default margin and padding */
|
47
|
+
"*": {
|
48
|
+
margin: 0
|
49
|
+
},
|
50
|
+
/** Remove adjust font properties */
|
51
|
+
html: {
|
52
|
+
lineHeight: 1.15,
|
53
|
+
textSizeAdjust: "100%",
|
54
|
+
WebkitFontSmoothing: "antialiased"
|
55
|
+
},
|
56
|
+
/** Make media elements responsive */
|
57
|
+
"img, picture, video, canvas, svg": {
|
58
|
+
display: "block",
|
59
|
+
maxWidth: "100%"
|
60
|
+
},
|
61
|
+
/** Avoid overflow of text */
|
62
|
+
"p, h1, h2, h3, h4, h5, h6": {
|
63
|
+
overflowWrap: "break-word"
|
64
|
+
},
|
65
|
+
/** Improve text wrapping */
|
66
|
+
p: {
|
67
|
+
textWrap: "pretty"
|
68
|
+
},
|
69
|
+
"h1, h2, h3, h4, h5, h6": {
|
70
|
+
textWrap: "balance"
|
71
|
+
},
|
72
|
+
/** Improve link color */
|
73
|
+
a: {
|
74
|
+
color: "currentColor"
|
75
|
+
},
|
76
|
+
/** Improve button line height */
|
77
|
+
button: {
|
78
|
+
lineHeight: "1em",
|
79
|
+
color: "currentColor"
|
80
|
+
},
|
81
|
+
/** Improve form elements */
|
82
|
+
"input, optgroup, select, textarea": {
|
83
|
+
fontFamily: "inherit",
|
84
|
+
fontSize: "100%",
|
85
|
+
lineHeight: "1.15em"
|
86
|
+
}
|
87
|
+
}, Lg = (g, I) => new Promise((c, l) => {
|
88
|
+
const Z = setTimeout(() => {
|
89
|
+
l(new Error("Timeout"));
|
90
|
+
}, 100), n = tg.createSourceFile("temp.ts", g, tg.ScriptTarget.Latest, !0);
|
91
|
+
function C(G) {
|
92
|
+
if (tg.isVariableDeclaration(G) && G.name.getText() === I) {
|
93
|
+
const W = G.getStart(), e = G.getEnd();
|
94
|
+
clearTimeout(Z), c([W, e]);
|
95
|
+
}
|
96
|
+
G.forEachChild(C);
|
97
|
+
}
|
98
|
+
C(n);
|
99
|
+
}), Dg = new URL("data:video/mp2t;base64,/* eslint-disable prefer-const */
/* eslint-disable @typescript-eslint/no-explicit-any */
import * as esbuild from 'esbuild';
import { execSync } from 'child_process';
import { toHash } from '../util/to-hash';
import { join, parse as parsePath } from 'path';
import { statSync, existsSync, mkdirSync, readdirSync, writeFileSync, readFileSync } from 'fs';
import { dashCase } from '../util/dash-case';
import { readFile } from 'fs/promises';
import { parseAndJoinStyles } from '../parsers/parse-styles';
import { getTemplateTypes, parseTemplates } from '../parsers/parse-templates';
import { CachedConfig, CssConditionalVariables, CssResponsiveVariables, CssTemplates, SaltyConfig, SaltyVariables } from '../config';
import { parseVariableTokens } from '../parsers/parse-tokens';
import { detectCurrentModuleType } from '../util/module-type';
import { logger } from '../bin/logger';
import { dotCase } from '../util/dot-case';
import { saltyReset } from '../templates/salty-reset';
import { RCFile } from '../types/cli-types';
import { mergeFactories, mergeObjects } from '../css';
import { defineTemplates, GlobalStylesFactory, TemplatesFactory, VariablesFactory } from '../factories';
import { StyledGenerator, ClassNameGenerator } from '../generators';
import { StylesGenerator } from '../generators/styles-generator';
import { getFunctionRange } from './get-function-range';

interface GeneratorResult<V extends StylesGenerator> {
  generator: V;
  src: string;
  name: string;
}

interface FunctionResult<V extends object> {
  value: V;
  src: string;
  name: string;
}

interface ConfigGenerationResults {
  mediaQueries: [string, string][];
  globalStyles: GlobalStylesFactory[];
  variables: VariablesFactory[];
  templates: TemplatesFactory[];
}

interface StylesGenerationResults {
  components: GeneratorResult<StyledGenerator>[];
  classNames: GeneratorResult<ClassNameGenerator>[];
  keyframes: FunctionResult<{ animationName: string; css: string }>[];
}

interface Cache {
  externalModules: string[];
  rcFile?: RCFile;
  destDir?: string;
}

const _dirname = new URL('.', import.meta.url).pathname;

const cache: Cache = {
  externalModules: [],
  rcFile: undefined,
  destDir: undefined,
};

const getExternalModules = (coreConfigPath: string) => {
  if (cache.externalModules.length > 0) return cache.externalModules;
  const content = readFileSync(coreConfigPath, 'utf8');
  const match = content.match(/externalModules:\s?\[(.*)\]/);
  if (!match) return [];
  const externalModules = match[1].split(',').map((d) => d.replace(/['"`]/g, '').trim());
  cache.externalModules = externalModules;
  return externalModules;
};

const getDestDir = async (dirname: string) => {
  if (cache.destDir) return cache.destDir;
  const projectConfig = await getRCProjectConfig(dirname);
  const destDir = join(dirname, projectConfig?.saltygenDir || 'saltygen');
  cache.destDir = destDir;
  return destDir;
};

export const saltyFileExtensions = ['salty', 'css', 'styles', 'styled'];
export const saltyFileRegExp = (additional: string[] = []) => new RegExp(`\\.(${[...saltyFileExtensions, ...additional].join('|')})\\.`);
export const isSaltyFile = (file: string, additional: string[] = []) => saltyFileRegExp(additional).test(file);

const readRCFile = async (currentDir: string) => {
  if (cache.rcFile) return cache.rcFile;
  if (currentDir === '/') throw new Error('Could not find .saltyrc.json file');
  const rcPath = join(currentDir, '.saltyrc.json');
  const rcContent = await readFile(rcPath, 'utf-8')
    .then(JSON.parse)
    .catch(() => undefined);

  if (!rcContent) return readRCFile(join(currentDir, '..'));
  cache.rcFile = rcContent;
  return rcContent as RCFile;
};

const getRCProjectConfig = async (dirname: string) => {
  const rcFile = await readRCFile(dirname);
  const projectConfig = rcFile.projects?.find((project) => dirname.endsWith(project.dir || ''));
  if (!projectConfig) return rcFile.projects?.find((project) => project.dir === rcFile.defaultProject);
  return projectConfig;
};

const generateConfig = async (dirname: string) => {
  const rcProject = await getRCProjectConfig(dirname);
  const destDir = await getDestDir(dirname);
  const coreConfigPath = join(dirname, rcProject?.configDir || '', 'salty.config.ts');
  const coreConfigDest = join(destDir, 'salty.config.js');

  const moduleType = await detectCurrentModuleType(dirname);
  const externalModules = getExternalModules(coreConfigPath);
  await esbuild.build({
    entryPoints: [coreConfigPath],
    minify: true,
    treeShaking: true,
    bundle: true,
    outfile: coreConfigDest,
    format: moduleType,
    external: externalModules,
  });

  const now = Date.now();
  const { config } = await import(`${coreConfigDest}?t=${now}`);
  return { config, path: coreConfigDest };
};

export const generateConfigStyles = async (dirname: string, configFiles: Set<string>) => {
  const destDir = await getDestDir(dirname);

  const generationResults: ConfigGenerationResults = {
    mediaQueries: [],
    globalStyles: [],
    variables: [],
    templates: [],
  };

  await Promise.all(
    [...configFiles].map(async (src) => {
      const { contents, outputFilePath } = await compileSaltyFile(dirname, src, destDir);
      Object.entries(contents).forEach(([name, value]) => {
        if (value.isMedia) generationResults.mediaQueries.push([name, value as any]);
        else if (value.isGlobalDefine) generationResults.globalStyles.push(value as any);
        else if (value.isDefineVariables) generationResults.variables.push(value as any);
        else if (value.isDefineTemplates) generationResults.templates.push((value as any)._setPath(`${name};;${outputFilePath}`));
      });
    })
  );

  // Generate the config files
  const { config, path: configPath } = await generateConfig(dirname);

  // Cache the config content
  const configCacheContent = { ...config } as CachedConfig;

  // Generate media query helpers
  const { mediaQueries } = generationResults;
  configCacheContent.mediaQueries = Object.fromEntries(mediaQueries.map(([name, value]) => [`@${name}`, value]));
  const mediaQueryKeys = mediaQueries.map(([name]) => `'@${name}'`).join(' | ');

  // Generate variables css file
  const variableTokens = new Set<string>();

  type Variables = string | undefined;
  const parseVariables = async <T extends object>(obj: T, path: PropertyKey[] = []): Promise<Variables[]> => {
    if (!obj) return [];
    const promises = Object.entries(obj).map(async ([key, value]): Promise<Variables | Variables[]> => {
      const parseVariable = async (value: unknown) => {
        if (!value) return undefined;
        if (value instanceof Promise) return await parseVariable(await value);
        if (typeof value === 'function') return await parseVariable(await value());
        if (typeof value === 'object') return await parseVariables(value, [...path, key]);

        const dottedKey = dotCase(key);
        const dashedKey = dashCase(key);

        const tsName = [...path, dottedKey].join('.');
        variableTokens.add(`"${tsName}"`);

        const cssName = [...path.map(dashCase), dashedKey].join('-');
        const result = parseVariableTokens(value);
        if (!result) return `--${cssName}: ${value};`;
        return `--${cssName}: ${result.transformed};`;
      };
      return await parseVariable(value);
    });

    const results = await Promise.all(promises);
    return results.flat();
  };

  const parseResponsiveVariables = async <T extends CssResponsiveVariables>(obj: T): Promise<Variables[]> => {
    if (!obj) return [];

    const promises = Object.entries(obj).map(async ([mediaQuery, values]): Promise<Variables | Variables[]> => {
      const variables = await parseVariables(values);
      if (mediaQuery === 'base') return variables.join('');
      if (configCacheContent.mediaQueries[mediaQuery]) {
        const mediaQueryValue = configCacheContent.mediaQueries[mediaQuery];
        return `${mediaQueryValue} { ${variables.join('')} }`;
      }
      return `${mediaQuery} { ${variables.join('')} }`;
    });
    const results = await Promise.all(promises);
    return results.flat();
  };

  const parseConditionalVariables = async <T extends CssConditionalVariables>(obj: T): Promise<Variables[]> => {
    if (!obj) return [];

    const promises = Object.entries(obj).map(async ([property, conditions]): Promise<Variables | Variables[]> => {
      const promises = Object.entries(conditions).map(async ([condition, values]): Promise<Variables | Variables[]> => {
        const variables = await parseVariables(values, [property]);
        const conditionScope = `.${property}-${condition}, [data-${property}="${condition}"]`;
        const combined = variables.join('');
        return `${conditionScope} { ${combined} }`;
      });
      const result = await Promise.all(promises);
      return result.flat();
    });
    const results = await Promise.all(promises);
    return results.flat();
  };

  const getStaticVariables = (variables: SaltyVariables): Record<string, any> => {
    return { ...variables, responsive: undefined, conditional: undefined };
  };

  const getGeneratedVariables = (type: 'static' | 'responsive' | 'conditional') => {
    return generationResults.variables.map((factory) => {
      if (type === 'static') return getStaticVariables(factory._current);
      return factory._current[type];
    });
  };

  const _staticVariables = mergeObjects(getStaticVariables(config.variables), getGeneratedVariables('static'));
  const staticVariables = await parseVariables(_staticVariables);
  const _responsiveVariables = mergeObjects<CssResponsiveVariables>(config.variables?.responsive, getGeneratedVariables('responsive'));
  const responsiveVariables = await parseResponsiveVariables(_responsiveVariables);
  const _conditionalVariables = mergeObjects(config.variables?.conditional, getGeneratedVariables('conditional'));
  const conditionalVariables = await parseConditionalVariables(_conditionalVariables);

  const variablesPath = join(destDir, 'css/_variables.css');
  const variablesCss = `:root { ${staticVariables.join('')} ${responsiveVariables.join('')} } ${conditionalVariables.join('')}`;
  writeFileSync(variablesPath, variablesCss);
  configCacheContent.staticVariables = _staticVariables;

  // Generate global styles
  const globalStylesPath = join(destDir, 'css/_global.css');
  const mergedGlobalStyles = mergeObjects(config.global, generationResults.globalStyles);
  const globalStylesString = await parseAndJoinStyles(mergedGlobalStyles, '');

  writeFileSync(globalStylesPath, `@layer global { ${globalStylesString} }`);

  // Generate reset styles
  const resetStylesPath = join(destDir, 'css/_reset.css');

  const getResetStyles = () => {
    if (config.reset === 'none') return {};
    if (typeof config.reset === 'object') return config.reset;
    return saltyReset;
  };

  const resetStyles = getResetStyles();
  const resetStylesString = await parseAndJoinStyles(resetStyles, '');

  writeFileSync(resetStylesPath, `@layer reset { ${resetStylesString} }`);

  // Generate templates
  const templateStylesPath = join(destDir, 'css/_templates.css');
  const templates = mergeObjects<CssTemplates>(config.templates, generationResults.templates);

  const templateStylesString = await parseTemplates(templates);
  const templateTokens = getTemplateTypes(templates);

  writeFileSync(templateStylesPath, `@layer templates { ${templateStylesString} }`);
  configCacheContent.templates = templates;

  const configTemplateFactories = config.templates ? [defineTemplates(config.templates)._setPath(`config;;${configPath}`)] : [];
  const templateFactories = mergeFactories(generationResults.templates, configTemplateFactories);

  configCacheContent.templatePaths = Object.fromEntries(Object.entries(templateFactories).map(([key, faktory]) => [key, faktory._path]));

  // Generate types

  const tsTokensPath = join(destDir, 'types/css-tokens.d.ts');
  const tsVariableTokens = [...variableTokens].join('|');

  const tsTokensTypes = `
  // Variable types
  type VariableTokens = ${tsVariableTokens}; 
  type PropertyValueToken = \`{\${VariableTokens}}\`;

  // Template types
  type TemplateTokens = {
    ${Object.entries(templateTokens)
      .map(([key, value]) => `${key}?: ${value}`)
      .join('\n')}
  }

  // Media query types
  type MediaQueryKeys = ${mediaQueryKeys || `''`};
  `;

  writeFileSync(tsTokensPath, tsTokensTypes);

  // Save config cache file
  const configCachePath = join(destDir, 'cache/config-cache.json');
  writeFileSync(configCachePath, JSON.stringify(configCacheContent, null, 2));

  const configCacheSecondaryPath = join(_dirname, '../cache/config-cache.json');
  writeFileSync(configCacheSecondaryPath, JSON.stringify(configCacheContent, null, 2));
};

const replaceStyledTag = (currentFile: string) => {
  return currentFile.replace(/styled\(([^"'`{,]+),/g, (match, tag) => {
    // Check if the tag is a string
    const isString = /^['"`]/.test(tag);
    if (isString) return match;

    // Check if the tag is imported from somewhere else
    const isImportedRegExp = new RegExp(`import[^;]*${tag}[,\\s{][^;]*from\\s?([^{};]+);`);
    const isImported = isImportedRegExp.test(currentFile);
    if (!isImported) return match;

    // Check if the import is a salty file
    const importResult = isImportedRegExp.exec(currentFile);
    if (importResult) {
      const importPath = importResult.at(1);
      const isSaltyImport = saltyFileExtensions.some((ext) => importPath?.includes(ext));
      if (isSaltyImport) return match;
    }

    // To avoid unnecessary imports, we will replace the styled call with a string when import is not a salty file
    return "styled('div',";
  });
};

const addConfigCache = (currentFile: string, dirname: string) => {
  try {
    const saltyCachedConfig = readFileSync(join(dirname, 'saltygen/cache/config-cache.json'), 'utf8');
    if (!saltyCachedConfig) return `globalThis.saltyConfig = {};\n\n${currentFile}`;
    return `globalThis.saltyConfig = ${saltyCachedConfig};\n\n${currentFile}`;
  } catch {
    return currentFile;
  }
};

export const compileSaltyFile = async (dirname: string, sourceFilePath: string, outputDirectory: string) => {
  const hashedName = toHash(sourceFilePath);
  const tempDir = join(outputDirectory, './temp');

  if (!existsSync(tempDir)) mkdirSync(tempDir);

  const parsed = parsePath(sourceFilePath);
  let currentFile = readFileSync(sourceFilePath, 'utf8');

  currentFile = replaceStyledTag(currentFile);
  currentFile = addConfigCache(currentFile, dirname);

  const outputFilePath = join(outputDirectory, 'js', hashedName + '.js');
  const rcProject = await getRCProjectConfig(dirname);
  const coreConfigPath = join(dirname, rcProject?.configDir || '', 'salty.config.ts');
  const externalModules = getExternalModules(coreConfigPath);
  const moduleType = await detectCurrentModuleType(dirname);

  await esbuild.build({
    stdin: {
      contents: currentFile,
      sourcefile: parsed.base,
      resolveDir: parsed.dir,
      loader: 'tsx',
    },
    minify: false,
    treeShaking: true,
    bundle: true,
    outfile: outputFilePath,
    format: moduleType,
    target: ['node20'],
    keepNames: true,
    external: externalModules,
    packages: 'external',
    plugins: [
      {
        name: 'test',
        setup: (build) => {
          build.onLoad({ filter: /.*\.css|salty|styles|styled\.ts/ }, (args) => {
            const original = readFileSync(args.path, 'utf8');
            const modified = replaceStyledTag(original);
            return { contents: modified, loader: 'ts' };
          });
        },
      },
    ],
  });

  type Contents = {
    [key: string]: {
      generator: any;
      isClassName?: boolean;
      isMedia?: boolean;
      isGlobalDefine?: boolean;
      isDefineVariables?: boolean;
      isDefineTemplates?: boolean;
      isKeyframes?: boolean;
      animationName?: string;
      css?: Promise<string>;
      styles?: any;
    };
  };

  const now = Date.now();
  const contents = (await import(`${outputFilePath}?t=${now}`)) as Contents;

  return { contents, outputFilePath };
};

const getConfigCache = async (dirname: string) => {
  const destDir = await getDestDir(dirname);
  const coreConfigDest = join(destDir, 'cache/config-cache.json');
  const contents = readFileSync(coreConfigDest, 'utf8');
  if (!contents) throw new Error('Could not find config cache file');
  return JSON.parse(contents);
};

const getConfig = async (dirname: string) => {
  const cached = await getConfigCache(dirname);
  const destDir = await getDestDir(dirname);
  const coreConfigDest = join(destDir, 'salty.config.js');
  const now = Date.now();
  const { config } = await import(`${coreConfigDest}?t=${now}`);
  return mergeObjects<SaltyConfig & CachedConfig>(config, cached);
};

const isProduction = () => {
  try {
    return process.env['NODE_ENV'] === 'production';
  } catch {
    return false;
  }
};

export const generateCss = async (dirname: string, prod = isProduction(), clean = true) => {
  try {
    const start = Date.now();
    if (prod) logger.info('Generating CSS in production mode! 🔥');
    else logger.info('Generating CSS in development mode! 🚀');

    const globalCssFiles: string[] = [];
    const cssFiles: string[][] = [];
    const destDir = await getDestDir(dirname);
    const cssFile = join(destDir, 'index.css');

    const clearDistDir = () => {
      if (existsSync(destDir)) execSync('rm -rf ' + destDir);
      mkdirSync(destDir, { recursive: true });
      mkdirSync(join(destDir, 'css'));
      mkdirSync(join(destDir, 'types'));
      mkdirSync(join(destDir, 'js'));
      mkdirSync(join(destDir, 'cache'));
    };

    // Clear the dist directory

    if (clean) clearDistDir();

    // Collect salty css files
    const files = new Set<string>();
    const configFiles = new Set<string>();

    async function collectFiles(src: string) {
      const foldersToSkip = ['node_modules', 'saltygen'];
      const stats = statSync(src);

      if (stats.isDirectory()) {
        const files = readdirSync(src);
        const shouldSkip = foldersToSkip.some((folder) => src.includes(folder));
        if (shouldSkip) return;
        await Promise.all(files.map((file) => collectFiles(join(src, file))));
      } else if (stats.isFile()) {
        const validFile = isSaltyFile(src);

        if (validFile) {
          files.add(src);
          const contents = readFileSync(src, 'utf8');
          const hasDefineFunction = /define[\w\d]+\(/.test(contents);
          if (hasDefineFunction) configFiles.add(src);
        }
      }
    }
    // Start the copying process
    await collectFiles(dirname);

    // Generate variables
    await generateConfigStyles(dirname, configFiles);

    const generationResults: StylesGenerationResults = {
      keyframes: [],
      components: [],
      classNames: [],
    };

    await Promise.all(
      [...files].map(async (src) => {
        const { contents } = await compileSaltyFile(dirname, src, destDir);
        for (let [name, value] of Object.entries(contents)) {
          if (value instanceof Promise) value = await value;

          if (value.isKeyframes) {
            generationResults.keyframes.push({
              value: value as any,
              src,
              name,
            });
          } else if (value.isClassName) {
            generationResults.classNames.push({
              ...value,
              src,
              name,
            });
          } else if (value.generator) {
            generationResults.components.push({
              ...value,
              src,
              name,
            });
          }
        }
      })
    );

    // Get config
    const config = await getConfig(dirname);

    // Generate CSS for keyframe animations
    for (const keyframes of generationResults.keyframes) {
      const { value } = keyframes;
      const fileName = `a_${value.animationName}.css`;
      const filePath = `css/${fileName}`;
      const cssPath = join(destDir, filePath);
      globalCssFiles.push(fileName);

      writeFileSync(cssPath, value.css);
    }

    // Start gathering CSS files for components
    const localCssFiles: Record<string, string[]> = {};

    // Generate CSS for components
    for (const componentResult of generationResults.components) {
      const { src, name } = componentResult;
      if (!localCssFiles[src]) localCssFiles[src] = [];

      const generator = componentResult.generator._withBuildContext({
        callerName: name,
        isProduction: prod,
        config,
      });

      if (!cssFiles[generator.priority]) cssFiles[generator.priority] = [];
      const styles = await generator.css;
      if (!styles) continue;

      cssFiles[generator.priority].push(generator.cssFileName);

      const filePath = `css/${generator.cssFileName}`;
      const cssPath = join(destDir, filePath);

      writeFileSync(cssPath, styles);

      if (config.importStrategy === 'component') {
        localCssFiles[src].push(generator.cssFileName);
      }
    }

    // Generate CSS for class names
    for (const classNameResult of generationResults.classNames) {
      const { src, name } = classNameResult;
      if (!localCssFiles[src]) localCssFiles[src] = [];

      const generator = classNameResult.generator._withBuildContext({
        callerName: name,
        isProduction: prod,
        config,
      });

      const styles = await generator.css;
      if (!styles) continue;

      if (!cssFiles[generator.priority]) cssFiles[generator.priority] = [];
      cssFiles[generator.priority].push(generator.cssFileName);

      const filePath = `css/${generator.cssFileName}`;
      const cssPath = join(destDir, filePath);

      writeFileSync(cssPath, styles);

      if (config.importStrategy === 'component') {
        localCssFiles[src].push(generator.cssFileName);
      }
    }

    // Generate CSS files for component import
    if (config.importStrategy === 'component') {
      Object.entries(localCssFiles).forEach(([src, localCssFile]) => {
        const cssContent = localCssFile.map((file) => `@import url('./${file}');`).join('\n');

        const hashName = toHash(src, 6);
        const parsedPath = parsePath(src);
        const dasherized = dashCase(parsedPath.name);

        const cssFile = join(destDir, `css/f_${dasherized}-${hashName}.css`);
        writeFileSync(cssFile, cssContent || `/* Empty file */`);
      });
    }

    const otherGlobalCssFiles = globalCssFiles.map((file) => `@import url('./css/${file}');`).join('\n');

    const globalCssFilenames = ['_variables.css', '_reset.css', '_global.css', '_templates.css'];
    const importsWithData = globalCssFilenames.filter((file) => {
      try {
        const data = readFileSync(join(destDir, 'css', file), 'utf8');
        return data.length > 0;
      } catch {
        return false;
      }
    });
    const globalImports = importsWithData.map((file) => `@import url('./css/${file}');`);
    const generatorText = '/*!\n * Generated with Salty CSS (https://salty-css.dev)\n * Do not edit this file directly\n */\n';
    let cssContent = `${generatorText}@layer reset, global, templates, l0, l1, l2, l3, l4, l5, l6, l7, l8;\n\n${globalImports.join(
      '\n'
    )}\n${otherGlobalCssFiles}`;

    if (config.importStrategy !== 'component') {
      const mergedContent = cssFiles.reduce((acc, val, layer) => {
        const layerContent = val.reduce((layerAcc, file) => {
          const filepath = join(destDir, 'css', file);
          const css = readFileSync(filepath, 'utf8');
          const filepathHash = /.*-([^-]+)-\d+.css/.exec(file)?.at(1) || toHash(filepath, 6);
          if (layerAcc.includes(filepathHash)) return layerAcc;
          return `${layerAcc}\n/*start:${filepathHash}-${file}*/\n${css}\n/*end:${filepathHash}*/\n`;
        }, '');

        const layerFileName = `l_${layer}.css`;
        const layerFilePath = join(destDir, 'css', layerFileName);
        const layerContentWithLayer = `@layer l${layer} { ${layerContent}\n }`;
        writeFileSync(layerFilePath, layerContentWithLayer);

        return `${acc}\n@import url('./css/${layerFileName}');`;
      }, '');

      cssContent += mergedContent;
    }

    writeFileSync(cssFile, cssContent);

    const end = Date.now();
    const time = end - start;
    const emoji = time < 200 ? '🔥' : time < 500 ? '🚀' : time < 1000 ? '🎉' : time < 2000 ? '🚗' : time < 5000 ? '🤔' : '🥴';
    logger.info(`Generated CSS in ${time}ms! ${emoji}`);
  } catch (e) {
    console.error(e);
  }
};

export const generateFile = async (dirname: string, file: string, prod = isProduction()) => {
  try {
    const destDir = await getDestDir(dirname);
    const validFile = isSaltyFile(file);

    if (validFile) {
      const cssFiles: string[][] = [];
      const config = await getConfig(dirname);
      const { contents } = await compileSaltyFile(dirname, file, destDir);
      for (const [name, value] of Object.entries(contents)) {
        // Object.entries(contents).forEach(([name, value]: [string, any]) => {
        if (value.isKeyframes && value.css) {
          const fileName = `a_${value.animationName}.css`;
          const filePath = `css/${fileName}`;
          const cssPath = join(destDir, filePath);

          writeFileSync(cssPath, await value.css);
          continue;
        }

        if (value.isClassName) {
          const generator = value.generator._withBuildContext({
            callerName: name,
            isProduction: prod,
            config,
          });

          const styles = await generator.css;
          if (!styles) continue;

          if (!cssFiles[generator.priority]) cssFiles[generator.priority] = [];
          cssFiles[generator.priority].push(generator.cssFileName);

          const filePath = `css/${generator.cssFileName}`;
          const cssPath = join(destDir, filePath);
          writeFileSync(cssPath, styles);
          continue;
        }

        if (!value.generator) continue;

        const generator = value.generator._withBuildContext({
          callerName: name,
          isProduction: prod,
          config,
        });

        const styles = await generator.css;
        if (!styles) continue;

        const filePath = `css/${generator.cssFileName}`;
        const cssPath = join(destDir, filePath);

        writeFileSync(cssPath, styles);

        if (!cssFiles[generator.priority]) cssFiles[generator.priority] = [];
        cssFiles[generator.priority].push(generator.cssFileName);
      }

      if (config.importStrategy !== 'component') {
        cssFiles.forEach((val, layer) => {
          const layerFileName = `l_${layer}.css`;
          const layerFilePath = join(destDir, 'css', layerFileName);
          let currentLayerFileContent = readFileSync(layerFilePath, 'utf8');
          val.forEach((file) => {
            const filepath = join(destDir, 'css', file);
            const filepathHash = /.*-([^-]+)-\d+.css/.exec(file)?.at(1) || toHash(filepath, 6);
            const found = currentLayerFileContent.includes(filepathHash);
            if (!found) {
              const css = readFileSync(filepath, 'utf8');
              const newContent = `/*start:${filepathHash}-${file}*/\n${css}\n/*end:${filepathHash}*/\n`;
              currentLayerFileContent = `${currentLayerFileContent.replace(/\}$/, '')}\n${newContent}\n}`;
            }
          });
          writeFileSync(layerFilePath, currentLayerFileContent);
        });
      } else {
        const cssContent = cssFiles
          .flat()
          .map((file) => `@import url('./${file}');`)
          .join('\n');

        const hashName = toHash(file, 6);
        const parsedPath = parsePath(file);
        const dasherized = dashCase(parsedPath.name);

        const cssFile = join(destDir, `css/f_${dasherized}-${hashName}.css`);
        writeFileSync(cssFile, cssContent || `/* Empty file */`);
      }
    }
  } catch (e) {
    console.error(e);
  }
};

export const minimizeFile = async (dirname: string, file: string, prod = isProduction()) => {
  try {
    const destDir = await getDestDir(dirname);
    const validFile = isSaltyFile(file);

    if (validFile) {
      const original = readFileSync(file, 'utf8');

      const config = await getConfig(dirname);
      const { contents } = await compileSaltyFile(dirname, file, destDir);

      let current = original;

      for (const [name, value] of Object.entries(contents)) {
        if (value.isKeyframes) continue;

        if (!value.generator) continue;
        const generator = value.generator._withBuildContext({
          callerName: name,
          isProduction: prod,
          config,
        });

        const [start, end] = await getFunctionRange(current, name);
        const range = current.slice(start, end);

        if (value.isClassName) {
          const copy = current;
          const clientVersion = ` ${name} = className("${generator.classNames}")`;
          current = current.replace(range, clientVersion);

          if (copy === current) console.error('Minimize file failed to change content', { name });
        }

        if (range.includes('styled')) {
          const tagName = /styled\(([^,]+),/.exec(range)?.at(1)?.trim();

          // Replace the styled call with the client version
          const copy = current;
          const clientVersion = ` ${name} = styled(${tagName}, "${generator.classNames}", ${JSON.stringify(generator.clientProps)})`;
          current = current.replace(range, clientVersion);

          if (copy === current) console.error('Minimize file failed to change content', { name, tagName });
        }
      }

      if (config.importStrategy === 'component') {
        const fileHash = toHash(file, 6);
        const parsed = parsePath(file);
        const dasherized = dashCase(parsed.name);
        const cssFileName = `f_${dasherized}-${fileHash}.css`;
        current = `import '../../saltygen/css/${cssFileName}';\n${current}`;
      }

      current = current.replace(`@salty-css/react/class-name`, `@salty-css/react/class-name-client`);

      current = current.replace(`{ styled }`, `{ styledClient as styled }`);
      current = current.replace(`@salty-css/react/styled`, `@salty-css/react/styled-client`);

      return current;
    }
  } catch (e) {
    console.error('Error in minimizeFile:', e);
  }
  return undefined;
};
", import.meta.url).pathname, K = {
|
100
|
+
externalModules: [],
|
101
|
+
rcFile: void 0,
|
102
|
+
destDir: void 0
|
103
|
+
}, Bg = (g) => {
|
104
|
+
if (K.externalModules.length > 0) return K.externalModules;
|
105
|
+
const c = F(g, "utf8").match(/externalModules:\s?\[(.*)\]/);
|
106
|
+
if (!c) return [];
|
107
|
+
const l = c[1].split(",").map((Z) => Z.replace(/['"`]/g, "").trim());
|
108
|
+
return K.externalModules = l, l;
|
109
|
+
}, k = async (g) => {
|
110
|
+
if (K.destDir) return K.destDir;
|
111
|
+
const I = await eg(g), c = b(g, (I == null ? void 0 : I.saltygenDir) || "saltygen");
|
112
|
+
return K.destDir = c, c;
|
113
|
+
}, ug = ["salty", "css", "styles", "styled"], Og = (g = []) => new RegExp(`\\.(${[...ug, ...g].join("|")})\\.`), mg = (g, I = []) => Og(I).test(g), Rg = async (g) => {
|
114
|
+
if (K.rcFile) return K.rcFile;
|
115
|
+
if (g === "/") throw new Error("Could not find .saltyrc.json file");
|
116
|
+
const I = b(g, ".saltyrc.json"), c = await pg(I, "utf-8").then(JSON.parse).catch(() => {
|
117
|
+
});
|
118
|
+
return c ? (K.rcFile = c, c) : Rg(b(g, ".."));
|
119
|
+
}, eg = async (g) => {
|
120
|
+
var l, Z;
|
121
|
+
const I = await Rg(g), c = (l = I.projects) == null ? void 0 : l.find((n) => g.endsWith(n.dir || ""));
|
122
|
+
return c || ((Z = I.projects) == null ? void 0 : Z.find((n) => n.dir === I.defaultProject));
|
123
|
+
}, Pg = async (g) => {
|
124
|
+
const I = await eg(g), c = await k(g), l = b(g, (I == null ? void 0 : I.configDir) || "", "salty.config.ts"), Z = b(c, "salty.config.js"), n = await Ng(g), C = Bg(l);
|
125
|
+
await Xg.build({
|
126
|
+
entryPoints: [l],
|
127
|
+
minify: !0,
|
128
|
+
treeShaking: !0,
|
129
|
+
bundle: !0,
|
130
|
+
outfile: Z,
|
131
|
+
format: n,
|
132
|
+
external: C
|
133
|
+
});
|
134
|
+
const G = Date.now(), { config: W } = await import(`${Z}?t=${G}`);
|
135
|
+
return { config: W, path: Z };
|
136
|
+
}, Eg = async (g, I) => {
|
137
|
+
var dg, Ag;
|
138
|
+
const c = await k(g), l = {
|
139
|
+
mediaQueries: [],
|
140
|
+
globalStyles: [],
|
141
|
+
variables: [],
|
142
|
+
templates: []
|
143
|
+
};
|
144
|
+
await Promise.all(
|
145
|
+
[...I].map(async (A) => {
|
146
|
+
const { contents: X, outputFilePath: O } = await Ig(g, A, c);
|
147
|
+
Object.entries(X).forEach(([J, h]) => {
|
148
|
+
h.isMedia ? l.mediaQueries.push([J, h]) : h.isGlobalDefine ? l.globalStyles.push(h) : h.isDefineVariables ? l.variables.push(h) : h.isDefineTemplates && l.templates.push(h._setPath(`${J};;${O}`));
|
149
|
+
});
|
150
|
+
})
|
151
|
+
);
|
152
|
+
const { config: Z, path: n } = await Pg(g), C = { ...Z }, { mediaQueries: G } = l;
|
153
|
+
C.mediaQueries = Object.fromEntries(G.map(([A, X]) => [`@${A}`, X]));
|
154
|
+
const W = G.map(([A]) => `'@${A}'`).join(" | "), e = /* @__PURE__ */ new Set(), s = async (A, X = []) => {
|
155
|
+
if (!A) return [];
|
156
|
+
const O = Object.entries(A).map(async ([h, v]) => {
|
157
|
+
const P = async (r) => {
|
158
|
+
if (!r) return;
|
159
|
+
if (r instanceof Promise) return await P(await r);
|
160
|
+
if (typeof r == "function") return await P(await r());
|
161
|
+
if (typeof r == "object") return await s(r, [...X, h]);
|
162
|
+
const sg = hg(h), bg = E(h), Zg = [...X, sg].join(".");
|
163
|
+
e.add(`"${Zg}"`);
|
164
|
+
const q = [...X.map(E), bg].join("-"), Wg = wg(r);
|
165
|
+
return Wg ? `--${q}: ${Wg.transformed};` : `--${q}: ${r};`;
|
166
|
+
};
|
167
|
+
return await P(v);
|
168
|
+
});
|
169
|
+
return (await Promise.all(O)).flat();
|
170
|
+
}, i = async (A) => {
|
171
|
+
if (!A) return [];
|
172
|
+
const X = Object.entries(A).map(async ([J, h]) => {
|
173
|
+
const v = await s(h);
|
174
|
+
return J === "base" ? v.join("") : C.mediaQueries[J] ? `${C.mediaQueries[J]} { ${v.join("")} }` : `${J} { ${v.join("")} }`;
|
175
|
+
});
|
176
|
+
return (await Promise.all(X)).flat();
|
177
|
+
}, o = async (A) => {
|
178
|
+
if (!A) return [];
|
179
|
+
const X = Object.entries(A).map(async ([J, h]) => {
|
180
|
+
const v = Object.entries(h).map(async ([r, sg]) => {
|
181
|
+
const bg = await s(sg, [J]), Zg = `.${J}-${r}, [data-${J}="${r}"]`, q = bg.join("");
|
182
|
+
return `${Zg} { ${q} }`;
|
183
|
+
});
|
184
|
+
return (await Promise.all(v)).flat();
|
185
|
+
});
|
186
|
+
return (await Promise.all(X)).flat();
|
187
|
+
}, d = (A) => ({ ...A, responsive: void 0, conditional: void 0 }), m = (A) => l.variables.map((X) => A === "static" ? d(X._current) : X._current[A]), V = U(d(Z.variables), m("static")), u = await s(V), R = U((dg = Z.variables) == null ? void 0 : dg.responsive, m("responsive")), z = await i(R), j = U((Ag = Z.variables) == null ? void 0 : Ag.conditional, m("conditional")), M = await o(j), ig = b(c, "css/_variables.css"), w = `:root { ${u.join("")} ${z.join("")} } ${M.join("")}`;
|
188
|
+
p(ig, w), C.staticVariables = V;
|
189
|
+
const cg = b(c, "css/_global.css"), a = U(Z.global, l.globalStyles), Y = await Yg(a, "");
|
190
|
+
p(cg, `@layer global { ${Y} }`);
|
191
|
+
const y = b(c, "css/_reset.css"), N = Z.reset === "none" ? {} : typeof Z.reset == "object" ? Z.reset : Tg, B = await Yg(N, "");
|
192
|
+
p(y, `@layer reset { ${B} }`);
|
193
|
+
const H = b(c, "css/_templates.css"), S = U(Z.templates, l.templates), T = await Sg(S), L = vg(S);
|
194
|
+
p(H, `@layer templates { ${T} }`), C.templates = S;
|
195
|
+
const $ = Z.templates ? [fg(Z.templates)._setPath(`config;;${n}`)] : [], lg = xg(l.templates, $);
|
196
|
+
C.templatePaths = Object.fromEntries(Object.entries(lg).map(([A, X]) => [A, X._path]));
|
197
|
+
const D = b(c, "types/css-tokens.d.ts"), Jg = `
|
198
|
+
// Variable types
|
199
|
+
type VariableTokens = ${[...e].join("|")};
|
200
|
+
type PropertyValueToken = \`{\${VariableTokens}}\`;
|
201
|
+
|
202
|
+
// Template types
|
203
|
+
type TemplateTokens = {
|
204
|
+
${Object.entries(L).map(([A, X]) => `${A}?: ${X}`).join(`
|
205
|
+
`)}
|
206
|
+
}
|
207
|
+
|
208
|
+
// Media query types
|
209
|
+
type MediaQueryKeys = ${W || "''"};
|
210
|
+
`;
|
211
|
+
p(D, Jg);
|
212
|
+
const rg = b(c, "cache/config-cache.json");
|
213
|
+
p(rg, JSON.stringify(C, null, 2));
|
214
|
+
const Fg = b(Dg, "../cache/config-cache.json");
|
215
|
+
p(Fg, JSON.stringify(C, null, 2));
|
216
|
+
}, yg = (g) => g.replace(/styled\(([^"'`{,]+),/g, (I, c) => {
|
217
|
+
if (/^['"`]/.test(c)) return I;
|
218
|
+
const Z = new RegExp(`import[^;]*${c}[,\\s{][^;]*from\\s?([^{};]+);`);
|
219
|
+
if (!Z.test(g)) return I;
|
220
|
+
const C = Z.exec(g);
|
221
|
+
if (C) {
|
222
|
+
const G = C.at(1);
|
223
|
+
if (ug.some((e) => G == null ? void 0 : G.includes(e))) return I;
|
224
|
+
}
|
225
|
+
return "styled('div',";
|
226
|
+
}), Mg = (g, I) => {
|
227
|
+
try {
|
228
|
+
const c = F(b(I, "saltygen/cache/config-cache.json"), "utf8");
|
229
|
+
return c ? `globalThis.saltyConfig = ${c};
|
230
|
+
|
231
|
+
${g}` : `globalThis.saltyConfig = {};
|
232
|
+
|
233
|
+
${g}`;
|
234
|
+
} catch {
|
235
|
+
return g;
|
236
|
+
}
|
237
|
+
}, Ig = async (g, I, c) => {
|
238
|
+
const l = Q(I), Z = b(c, "./temp");
|
239
|
+
ag(Z) || f(Z);
|
240
|
+
const n = gg(I);
|
241
|
+
let C = F(I, "utf8");
|
242
|
+
C = yg(C), C = Mg(C, g);
|
243
|
+
const G = b(c, "js", l + ".js"), W = await eg(g), e = b(g, (W == null ? void 0 : W.configDir) || "", "salty.config.ts"), s = Bg(e), i = await Ng(g);
|
244
|
+
await Xg.build({
|
245
|
+
stdin: {
|
246
|
+
contents: C,
|
247
|
+
sourcefile: n.base,
|
248
|
+
resolveDir: n.dir,
|
249
|
+
loader: "tsx"
|
250
|
+
},
|
251
|
+
minify: !1,
|
252
|
+
treeShaking: !0,
|
253
|
+
bundle: !0,
|
254
|
+
outfile: G,
|
255
|
+
format: i,
|
256
|
+
target: ["node20"],
|
257
|
+
keepNames: !0,
|
258
|
+
external: s,
|
259
|
+
packages: "external",
|
260
|
+
plugins: [
|
261
|
+
{
|
262
|
+
name: "test",
|
263
|
+
setup: (m) => {
|
264
|
+
m.onLoad({ filter: /.*\.css|salty|styles|styled\.ts/ }, (V) => {
|
265
|
+
const u = F(V.path, "utf8");
|
266
|
+
return { contents: yg(u), loader: "ts" };
|
267
|
+
});
|
268
|
+
}
|
269
|
+
}
|
270
|
+
]
|
271
|
+
});
|
272
|
+
const o = Date.now();
|
273
|
+
return { contents: await import(`${G}?t=${o}`), outputFilePath: G };
|
274
|
+
}, $g = async (g) => {
|
275
|
+
const I = await k(g), c = b(I, "cache/config-cache.json"), l = F(c, "utf8");
|
276
|
+
if (!l) throw new Error("Could not find config cache file");
|
277
|
+
return JSON.parse(l);
|
278
|
+
}, og = async (g) => {
|
279
|
+
const I = await $g(g), c = await k(g), l = b(c, "salty.config.js"), Z = Date.now(), { config: n } = await import(`${l}?t=${Z}`);
|
280
|
+
return U(n, I);
|
281
|
+
}, Gg = () => {
|
282
|
+
try {
|
283
|
+
return process.env.NODE_ENV === "production";
|
284
|
+
} catch {
|
285
|
+
return !1;
|
286
|
+
}
|
287
|
+
}, aI = async (g, I = Gg(), c = !0) => {
|
288
|
+
try {
|
289
|
+
const l = Date.now();
|
290
|
+
I ? _.info("Generating CSS in production mode! 🔥") : _.info("Generating CSS in development mode! 🚀");
|
291
|
+
const Z = [], n = [], C = await k(g), G = b(C, "index.css");
|
292
|
+
c && (() => {
|
293
|
+
ag(C) && Hg("rm -rf " + C), f(C, { recursive: !0 }), f(b(C, "css")), f(b(C, "types")), f(b(C, "js")), f(b(C, "cache"));
|
294
|
+
})();
|
295
|
+
const e = /* @__PURE__ */ new Set(), s = /* @__PURE__ */ new Set();
|
296
|
+
async function i(a) {
|
297
|
+
const Y = ["node_modules", "saltygen"], y = Kg(a);
|
298
|
+
if (y.isDirectory()) {
|
299
|
+
const t = zg(a);
|
300
|
+
if (Y.some((B) => a.includes(B))) return;
|
301
|
+
await Promise.all(t.map((B) => i(b(a, B))));
|
302
|
+
} else if (y.isFile() && mg(a)) {
|
303
|
+
e.add(a);
|
304
|
+
const N = F(a, "utf8");
|
305
|
+
/define[\w\d]+\(/.test(N) && s.add(a);
|
306
|
+
}
|
307
|
+
}
|
308
|
+
await i(g), await Eg(g, s);
|
309
|
+
const o = {
|
310
|
+
keyframes: [],
|
311
|
+
components: [],
|
312
|
+
classNames: []
|
313
|
+
};
|
314
|
+
await Promise.all(
|
315
|
+
[...e].map(async (a) => {
|
316
|
+
const { contents: Y } = await Ig(g, a, C);
|
317
|
+
for (let [y, t] of Object.entries(Y))
|
318
|
+
t instanceof Promise && (t = await t), t.isKeyframes ? o.keyframes.push({
|
319
|
+
value: t,
|
320
|
+
src: a,
|
321
|
+
name: y
|
322
|
+
}) : t.isClassName ? o.classNames.push({
|
323
|
+
...t,
|
324
|
+
src: a,
|
325
|
+
name: y
|
326
|
+
}) : t.generator && o.components.push({
|
327
|
+
...t,
|
328
|
+
src: a,
|
329
|
+
name: y
|
330
|
+
});
|
331
|
+
})
|
332
|
+
);
|
333
|
+
const d = await og(g);
|
334
|
+
for (const a of o.keyframes) {
|
335
|
+
const { value: Y } = a, y = `a_${Y.animationName}.css`, t = `css/${y}`, N = b(C, t);
|
336
|
+
Z.push(y), p(N, Y.css);
|
337
|
+
}
|
338
|
+
const m = {};
|
339
|
+
for (const a of o.components) {
|
340
|
+
const { src: Y, name: y } = a;
|
341
|
+
m[Y] || (m[Y] = []);
|
342
|
+
const t = a.generator._withBuildContext({
|
343
|
+
callerName: y,
|
344
|
+
isProduction: I,
|
345
|
+
config: d
|
346
|
+
});
|
347
|
+
n[t.priority] || (n[t.priority] = []);
|
348
|
+
const N = await t.css;
|
349
|
+
if (!N) continue;
|
350
|
+
n[t.priority].push(t.cssFileName);
|
351
|
+
const B = `css/${t.cssFileName}`, H = b(C, B);
|
352
|
+
p(H, N), d.importStrategy === "component" && m[Y].push(t.cssFileName);
|
353
|
+
}
|
354
|
+
for (const a of o.classNames) {
|
355
|
+
const { src: Y, name: y } = a;
|
356
|
+
m[Y] || (m[Y] = []);
|
357
|
+
const t = a.generator._withBuildContext({
|
358
|
+
callerName: y,
|
359
|
+
isProduction: I,
|
360
|
+
config: d
|
361
|
+
}), N = await t.css;
|
362
|
+
if (!N) continue;
|
363
|
+
n[t.priority] || (n[t.priority] = []), n[t.priority].push(t.cssFileName);
|
364
|
+
const B = `css/${t.cssFileName}`, H = b(C, B);
|
365
|
+
p(H, N), d.importStrategy === "component" && m[Y].push(t.cssFileName);
|
366
|
+
}
|
367
|
+
d.importStrategy === "component" && Object.entries(m).forEach(([a, Y]) => {
|
368
|
+
const y = Y.map((S) => `@import url('./${S}');`).join(`
|
369
|
+
`), t = Q(a, 6), N = gg(a), B = E(N.name), H = b(C, `css/f_${B}-${t}.css`);
|
370
|
+
p(H, y || "/* Empty file */");
|
371
|
+
});
|
372
|
+
const V = Z.map((a) => `@import url('./css/${a}');`).join(`
|
373
|
+
`);
|
374
|
+
let M = `/*!
|
375
|
+
* Generated with Salty CSS (https://salty-css.dev)
|
376
|
+
* Do not edit this file directly
|
377
|
+
*/
|
378
|
+
@layer reset, global, templates, l0, l1, l2, l3, l4, l5, l6, l7, l8;
|
379
|
+
|
380
|
+
${["_variables.css", "_reset.css", "_global.css", "_templates.css"].filter((a) => {
|
381
|
+
try {
|
382
|
+
return F(b(C, "css", a), "utf8").length > 0;
|
383
|
+
} catch {
|
384
|
+
return !1;
|
385
|
+
}
|
386
|
+
}).map((a) => `@import url('./css/${a}');`).join(
|
387
|
+
`
|
388
|
+
`
|
389
|
+
)}
|
390
|
+
${V}`;
|
391
|
+
if (d.importStrategy !== "component") {
|
392
|
+
const a = n.reduce((Y, y, t) => {
|
393
|
+
const N = y.reduce((T, L) => {
|
394
|
+
var Cg;
|
395
|
+
const $ = b(C, "css", L), lg = F($, "utf8"), D = ((Cg = /.*-([^-]+)-\d+.css/.exec(L)) == null ? void 0 : Cg.at(1)) || Q($, 6);
|
396
|
+
return T.includes(D) ? T : `${T}
|
397
|
+
/*start:${D}-${L}*/
|
398
|
+
${lg}
|
399
|
+
/*end:${D}*/
|
400
|
+
`;
|
401
|
+
}, ""), B = `l_${t}.css`, H = b(C, "css", B), S = `@layer l${t} { ${N}
|
402
|
+
}`;
|
403
|
+
return p(H, S), `${Y}
|
404
|
+
@import url('./css/${B}');`;
|
405
|
+
}, "");
|
406
|
+
M += a;
|
407
|
+
}
|
408
|
+
p(G, M);
|
409
|
+
const w = Date.now() - l, cg = w < 200 ? "🔥" : w < 500 ? "🚀" : w < 1e3 ? "🎉" : w < 2e3 ? "🚗" : w < 5e3 ? "🤔" : "🥴";
|
410
|
+
_.info(`Generated CSS in ${w}ms! ${cg}`);
|
411
|
+
} catch (l) {
|
412
|
+
console.error(l);
|
413
|
+
}
|
414
|
+
}, mI = async (g, I, c = Gg()) => {
|
415
|
+
try {
|
416
|
+
const l = await k(g);
|
417
|
+
if (mg(I)) {
|
418
|
+
const n = [], C = await og(g), { contents: G } = await Ig(g, I, l);
|
419
|
+
for (const [W, e] of Object.entries(G)) {
|
420
|
+
if (e.isKeyframes && e.css) {
|
421
|
+
const V = `css/${`a_${e.animationName}.css`}`, u = b(l, V);
|
422
|
+
p(u, await e.css);
|
423
|
+
continue;
|
424
|
+
}
|
425
|
+
if (e.isClassName) {
|
426
|
+
const m = e.generator._withBuildContext({
|
427
|
+
callerName: W,
|
428
|
+
isProduction: c,
|
429
|
+
config: C
|
430
|
+
}), V = await m.css;
|
431
|
+
if (!V) continue;
|
432
|
+
n[m.priority] || (n[m.priority] = []), n[m.priority].push(m.cssFileName);
|
433
|
+
const u = `css/${m.cssFileName}`, R = b(l, u);
|
434
|
+
p(R, V);
|
435
|
+
continue;
|
436
|
+
}
|
437
|
+
if (!e.generator) continue;
|
438
|
+
const s = e.generator._withBuildContext({
|
439
|
+
callerName: W,
|
440
|
+
isProduction: c,
|
441
|
+
config: C
|
442
|
+
}), i = await s.css;
|
443
|
+
if (!i) continue;
|
444
|
+
const o = `css/${s.cssFileName}`, d = b(l, o);
|
445
|
+
p(d, i), n[s.priority] || (n[s.priority] = []), n[s.priority].push(s.cssFileName);
|
446
|
+
}
|
447
|
+
if (C.importStrategy !== "component")
|
448
|
+
n.forEach((W, e) => {
|
449
|
+
const s = `l_${e}.css`, i = b(l, "css", s);
|
450
|
+
let o = F(i, "utf8");
|
451
|
+
W.forEach((d) => {
|
452
|
+
var R;
|
453
|
+
const m = b(l, "css", d), V = ((R = /.*-([^-]+)-\d+.css/.exec(d)) == null ? void 0 : R.at(1)) || Q(m, 6);
|
454
|
+
if (!o.includes(V)) {
|
455
|
+
const z = F(m, "utf8"), j = `/*start:${V}-${d}*/
|
456
|
+
${z}
|
457
|
+
/*end:${V}*/
|
458
|
+
`;
|
459
|
+
o = `${o.replace(/\}$/, "")}
|
460
|
+
${j}
|
461
|
+
}`;
|
462
|
+
}
|
463
|
+
}), p(i, o);
|
464
|
+
});
|
465
|
+
else {
|
466
|
+
const W = n.flat().map((d) => `@import url('./${d}');`).join(`
|
467
|
+
`), e = Q(I, 6), s = gg(I), i = E(s.name), o = b(l, `css/f_${i}-${e}.css`);
|
468
|
+
p(o, W || "/* Empty file */");
|
469
|
+
}
|
470
|
+
}
|
471
|
+
} catch (l) {
|
472
|
+
console.error(l);
|
473
|
+
}
|
474
|
+
}, eI = async (g, I, c = Gg()) => {
|
475
|
+
var l, Z;
|
476
|
+
try {
|
477
|
+
const n = await k(g);
|
478
|
+
if (mg(I)) {
|
479
|
+
const G = F(I, "utf8"), W = await og(g), { contents: e } = await Ig(g, I, n);
|
480
|
+
let s = G;
|
481
|
+
for (const [i, o] of Object.entries(e)) {
|
482
|
+
if (o.isKeyframes || !o.generator) continue;
|
483
|
+
const d = o.generator._withBuildContext({
|
484
|
+
callerName: i,
|
485
|
+
isProduction: c,
|
486
|
+
config: W
|
487
|
+
}), [m, V] = await Lg(s, i), u = s.slice(m, V);
|
488
|
+
if (o.isClassName) {
|
489
|
+
const R = s, z = ` ${i} = className("${d.classNames}")`;
|
490
|
+
s = s.replace(u, z), R === s && console.error("Minimize file failed to change content", { name: i });
|
491
|
+
}
|
492
|
+
if (u.includes("styled")) {
|
493
|
+
const R = (Z = (l = /styled\(([^,]+),/.exec(u)) == null ? void 0 : l.at(1)) == null ? void 0 : Z.trim(), z = s, j = ` ${i} = styled(${R}, "${d.classNames}", ${JSON.stringify(d.clientProps)})`;
|
494
|
+
s = s.replace(u, j), z === s && console.error("Minimize file failed to change content", { name: i, tagName: R });
|
495
|
+
}
|
496
|
+
}
|
497
|
+
if (W.importStrategy === "component") {
|
498
|
+
const i = Q(I, 6), o = gg(I);
|
499
|
+
s = `import '../../saltygen/css/${`f_${E(o.name)}-${i}.css`}';
|
500
|
+
${s}`;
|
501
|
+
}
|
502
|
+
return s = s.replace("@salty-css/react/class-name", "@salty-css/react/class-name-client"), s = s.replace("{ styled }", "{ styledClient as styled }"), s = s.replace("@salty-css/react/styled", "@salty-css/react/styled-client"), s;
|
503
|
+
}
|
504
|
+
} catch (n) {
|
505
|
+
console.error("Error in minimizeFile:", n);
|
506
|
+
}
|
507
|
+
};
|
508
|
+
export {
|
509
|
+
tI as a,
|
510
|
+
mI as b,
|
511
|
+
Og as c,
|
512
|
+
Eg as d,
|
513
|
+
Ig as e,
|
514
|
+
aI as g,
|
515
|
+
mg as i,
|
516
|
+
_ as l,
|
517
|
+
eI as m,
|
518
|
+
ug as s
|
519
|
+
};
|