@salty-css/core 0.0.1-alpha.279 → 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/bin/main.cjs +1 -1
- package/bin/main.js +2 -2
- 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/index.cjs +1 -1
- package/compiler/index.js +1 -1
- package/index-BgHtTuZf.cjs +41 -0
- package/index-CXjwv_Dc.js +519 -0
- package/package.json +4 -4
- package/parsers/parse-styles.d.ts +1 -1
- package/server/index.cjs +1 -1
- package/server/index.js +1 -1
- package/{should-restart-CfoDl2WZ.js → should-restart-BELKpXEw.js} +1 -1
- package/{should-restart-CapzIWDV.cjs → should-restart-CBwAaej8.cjs} +1 -1
- package/types/config-types.d.ts +1 -0
- package/cache/get-config-cache.cjs +0 -1
- package/cache/get-config-cache.d.ts +0 -3
- package/cache/get-config-cache.js +0 -6
- package/index-B5Z3hZJn.cjs +0 -41
- package/index-CI0AEQ1k.js +0 -517
package/bin/main.cjs
CHANGED
@@ -1,4 +1,4 @@
|
|
1
|
-
"use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const te=require("commander"),A=require("fs"),s=require("fs/promises"),n=require("path"),ne=require("ejs"),e=require("../index-
|
1
|
+
"use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const te=require("commander"),A=require("fs"),s=require("fs/promises"),n=require("path"),ne=require("ejs"),e=require("../index-BgHtTuZf.cjs"),ie=require("../pascal-case-iWoaJWwT.cjs"),se=require("child_process"),oe=require("ora"),ae=require("../should-restart-CBwAaej8.cjs");var G=typeof document<"u"?document.currentScript:null;const X=y=>new Promise((C,P)=>{se.exec(y,x=>{if(x)return P(x);C()})}),I=async(...y)=>{const C=y.map(b=>b.replace("-D","").split("@").slice(0,-1).join("@").trim()).join(", "),P=oe(`Installing packages: ${C}`).start(),x=y.join(" ");await X(`npm install ${x}`),P.succeed(`Installed packages: ${C}`)},re=()=>A.existsSync(n.join(process.cwd(),"node_modules",".bin","prettier"));async function S(y){try{if(!re())return;await X(`./node_modules/.bin/prettier --write "${y}"`),e.logger.info(`Formatted ${y} with Prettier`)}catch(C){e.logger.error(`Error formatting ${y} with Prettier:`,C)}}async function ce(){const y=new te.Command;y.name("salty-css").description("Salty-CSS CLI tool to help with annoying configuration tasks.");const C={"salty.config.ts":Promise.resolve().then(()=>require("../salty.config-Dk6ZcCxI.cjs")),"saltygen/index.css":Promise.resolve().then(()=>require("../index-84Wroia-.cjs")),"react/react-styled-file.ts":Promise.resolve().then(()=>require("../react-styled-file-Dkubsz-U.cjs")),"react/react-vanilla-file.ts":Promise.resolve().then(()=>require("../react-vanilla-file-CG_WJLam.cjs"))},P=async(i,a)=>{const{default:f}=await C[i],d=ne.render(f,a);return{fileName:i,content:d}},x=async()=>{const i=n.join(process.cwd(),".saltyrc.json");return await s.readFile(i,"utf-8").then(JSON.parse).catch(()=>({}))},b=n.join(process.cwd(),"package.json"),N=async(i=b)=>{const a=await s.readFile(i,"utf-8").then(JSON.parse).catch(()=>{});if(!a)throw"Could not read package.json file!";return a},Y=async(i,a=b)=>{typeof i=="object"&&(i=JSON.stringify(i,null,2)),await s.writeFile(a,i)},B=async()=>{const i=new URL("../package.json",typeof document>"u"?require("url").pathToFileURL(__filename).href:G&&G.tagName.toUpperCase()==="SCRIPT"&&G.src||new URL("bin/main.cjs",document.baseURI).href);return N(i)},V=await(async()=>(await x()).defaultProject)(),$=await B(),D={core:`@salty-css/core@${$.version}`,react:`@salty-css/react@${$.version}`,eslintConfigCore:`@salty-css/eslint-config-core@${$.version}`,vite:`@salty-css/vite@${$.version}`,next:`@salty-css/next@${$.version}`},H=i=>{const a=i==="."?"":i,f=process.cwd();return n.join(f,a)};y.command("init [directory]").description("Initialize a new Salty-CSS project.").option("-d, --dir <dir>","Project directory to initialize the project in.").option("--css-file <css-file>","Existing CSS file where to import the generated CSS. Path must be relative to the given project directory.").option("--skip-install","Skip installing dependencies.").action(async function(i="."){if(!await N().catch(()=>{}))return e.logError("Salty CSS project must be initialized in a directory with a package.json file.");e.logger.info("Initializing a new Salty-CSS project!");const{dir:f=i,cssFile:d,skipInstall:p}=this.opts();if(!f)return e.logError("Project directory must be provided. Add it as the first argument after init command or use the --dir option.");p||await I(D.core,D.react);const c=process.cwd(),r=H(f),q=await Promise.all([P("salty.config.ts"),P("saltygen/index.css")]);await s.mkdir(r,{recursive:!0});const m=q.map(async({fileName:t,content:u})=>{const l=n.join(r,t);if(await s.readFile(l,"utf-8").catch(()=>{})!==void 0){e.logger.debug("File already exists: "+l);return}const k=t.split("/").slice(0,-1).join("/");k&&await s.mkdir(n.join(r,k),{recursive:!0}),e.logger.info("Creating file: "+l),await s.writeFile(l,u),await S(l)});await Promise.all(m);const w=n.relative(c,r)||".",h=n.join(c,".saltyrc.json"),g=await s.readFile(h,"utf-8").catch(()=>{});if(g===void 0){e.logger.info("Creating file: "+h);const u=JSON.stringify({$schema:"./node_modules/@salty-css/core/.saltyrc.schema.json",info:"This file is used to define projects and their configurations for Salty CSS cli. Do not delete, modify or add this file to .gitignore.",defaultProject:w,projects:[{dir:w,framework:"react"}]},null,2);await s.writeFile(h,u),await S(h)}else{const t=JSON.parse(g),u=(t==null?void 0:t.projects)||[];if(u.findIndex(o=>o.dir===w)===-1){u.push({dir:w,framework:"react"}),t.projects=[...u];const o=JSON.stringify(t,null,2);o!==g&&(e.logger.info("Edit file: "+h),await s.writeFile(h,o),await S(h))}}const j=n.join(c,".gitignore"),T=await s.readFile(j,"utf-8").catch(()=>{});T!==void 0&&(T.includes("saltygen")||(e.logger.info("Edit file: "+j),await s.writeFile(j,T+`
|
2
2
|
|
3
3
|
# Salty-CSS
|
4
4
|
saltygen
|
package/bin/main.js
CHANGED
@@ -3,11 +3,11 @@ import { existsSync as Q, watch as re } from "fs";
|
|
3
3
|
import { mkdir as q, readFile as S, writeFile as C } from "fs/promises";
|
4
4
|
import { join as a, relative as z, parse as ce, format as te } from "path";
|
5
5
|
import { render as le } from "ejs";
|
6
|
-
import { l as t, a as h, g as K, i as de, b as fe } from "../index-
|
6
|
+
import { l as t, a as h, g as K, i as de, b as fe } from "../index-CXjwv_Dc.js";
|
7
7
|
import { p as pe } from "../pascal-case-BQpR5PdN.js";
|
8
8
|
import { exec as ue } from "child_process";
|
9
9
|
import ge from "ora";
|
10
|
-
import { c as ye } from "../should-restart-
|
10
|
+
import { c as ye } from "../should-restart-BELKpXEw.js";
|
11
11
|
const ne = (g) => new Promise((P, x) => {
|
12
12
|
ue(g, ($) => {
|
13
13
|
if ($) return x($);
|
@@ -0,0 +1 @@
|
|
1
|
+
"use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const o=require("fs/promises"),t=require("path");var e=typeof document<"u"?document.currentScript:null;const r=new URL(".",typeof document>"u"?require("url").pathToFileURL(__filename).href:e&&e.tagName.toUpperCase()==="SCRIPT"&&e.src||new URL("cache/resolve-dynamic-config-cache.cjs",document.baseURI).href).pathname,i=async()=>{const c=t.join(r,"config-cache.json"),n=await o.readFile(c,"utf8");if(!n)throw new Error("Could not find config cache file");return JSON.parse(n)};exports.resolveDynamicConfigCache=i;
|
@@ -0,0 +1 @@
|
|
1
|
+
export declare const resolveDynamicConfigCache: () => Promise<any>;
|
@@ -0,0 +1,10 @@
|
|
1
|
+
import { readFile as e } from "fs/promises";
|
2
|
+
import { join as r } from "path";
|
3
|
+
const t = new URL(".", import.meta.url).pathname, a = async () => {
|
4
|
+
const n = r(t, "config-cache.json"), o = await e(n, "utf8");
|
5
|
+
if (!o) throw new Error("Could not find config cache file");
|
6
|
+
return JSON.parse(o);
|
7
|
+
};
|
8
|
+
export {
|
9
|
+
a as resolveDynamicConfigCache
|
10
|
+
};
|
package/compiler/index.cjs
CHANGED
@@ -1 +1 @@
|
|
1
|
-
"use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});require("esbuild");require("child_process");require("../dash-case-BJEkFEGQ.cjs");require("path");require("fs");require("fs/promises");require("../parse-styles-Bdgw_4ME.cjs");require("../parsers/index.cjs");const e=require("../index-
|
1
|
+
"use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});require("esbuild");require("child_process");require("../dash-case-BJEkFEGQ.cjs");require("path");require("fs");require("fs/promises");require("../parse-styles-Bdgw_4ME.cjs");require("../parsers/index.cjs");const e=require("../index-BgHtTuZf.cjs");require("../css/merge.cjs");require("../define-templates-Cunsb_Tr.cjs");exports.compileSaltyFile=e.compileSaltyFile;exports.generateConfigStyles=e.generateConfigStyles;exports.generateCss=e.generateCss;exports.generateFile=e.generateFile;exports.isSaltyFile=e.isSaltyFile;exports.minimizeFile=e.minimizeFile;exports.saltyFileExtensions=e.saltyFileExtensions;exports.saltyFileRegExp=e.saltyFileRegExp;
|
package/compiler/index.js
CHANGED
@@ -6,7 +6,7 @@ import "fs";
|
|
6
6
|
import "fs/promises";
|
7
7
|
import "../parse-styles-C1E6ETeH.js";
|
8
8
|
import "../parsers/index.js";
|
9
|
-
import { e as F, d as y, g as x, b as S, i as c, m as f, s as C, c as E } from "../index-
|
9
|
+
import { e as F, d as y, g as x, b as S, i as c, m as f, s as C, c as E } from "../index-CXjwv_Dc.js";
|
10
10
|
import "../css/merge.js";
|
11
11
|
import "../define-templates-4A2yHcMF.js";
|
12
12
|
export {
|
@@ -0,0 +1,41 @@
|
|
1
|
+
"use strict";const Jg=require("esbuild"),Hg=require("child_process"),J=require("./dash-case-BJEkFEGQ.cjs"),s=require("path"),n=require("fs"),yg=require("fs/promises"),ng=require("./parse-styles-Bdgw_4ME.cjs"),Ag=require("./parsers/index.cjs"),P=require("winston"),j=require("./css/merge.cjs"),Kg=require("./define-templates-Cunsb_Tr.cjs"),eg=require("typescript");var U=typeof document<"u"?document.currentScript:null;function zg(g){const c=Object.create(null,{[Symbol.toStringTag]:{value:"Module"}});if(g){for(const l in g)if(l!=="default"){const I=Object.getOwnPropertyDescriptor(g,l);Object.defineProperty(c,l,I.get?I:{enumerable:!0,get:()=>g[l]})}}return c.default=g,Object.freeze(c)}const Yg=zg(Jg),pg=g=>{if(!g||g==="/")throw new Error("Could not find package.json file");const c=s.join(g,"package.json");return n.existsSync(c)?c:pg(s.join(g,".."))},wg=async g=>{const c=pg(g);return await yg.readFile(c,"utf-8").then(JSON.parse).catch(()=>{})},Sg=async g=>{const c=await wg(g);if(c)return c.type};let f;const Xg=async g=>{if(f)return f;const c=await Sg(g);return c==="module"?f="esm":(c==="commonjs"||(typeof document>"u"?require("url").pathToFileURL(__filename).href:U&&U.tagName.toUpperCase()==="SCRIPT"&&U.src||new URL("index-BgHtTuZf.cjs",document.baseURI).href).endsWith(".cjs"))&&(f="cjs"),f||"esm"},E=P.createLogger({level:"debug",format:P.format.combine(P.format.colorize(),P.format.cli()),transports:[new P.transports.Console({})]}),vg=g=>{E.error(g)};function Vg(g){return g?typeof g!="string"?Vg(String(g)):g.replace(/[^\d\w]/g,"."):""}const jg={"*, *::before, *::after":{boxSizing:"border-box"},"*":{margin:0},html:{lineHeight:1.15,textSizeAdjust:"100%",WebkitFontSmoothing:"antialiased"},"img, picture, video, canvas, svg":{display:"block",maxWidth:"100%"},"p, h1, h2, h3, h4, h5, h6":{overflowWrap:"break-word"},p:{textWrap:"pretty"},"h1, h2, h3, h4, h5, h6":{textWrap:"balance"},a:{color:"currentColor"},button:{lineHeight:"1em",color:"currentColor"},"input, optgroup, select, textarea":{fontFamily:"inherit",fontSize:"100%",lineHeight:"1.15em"}},kg=(g,c)=>new Promise((l,I)=>{const e=setTimeout(()=>{I(new Error("Timeout"))},100),Z=eg.createSourceFile("temp.ts",g,eg.ScriptTarget.Latest,!0);function C(G){if(eg.isVariableDeclaration(G)&&G.name.getText()===c){const y=G.getStart(),m=G.getEnd();clearTimeout(e),l([y,m])}G.forEachChild(C)}C(Z)}),xg=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;
};
",typeof document>"u"?require("url").pathToFileURL(__filename).href:U&&U.tagName.toUpperCase()==="SCRIPT"&&U.src||new URL("index-BgHtTuZf.cjs",document.baseURI).href).pathname,K={externalModules:[],rcFile:void 0,destDir:void 0},Ng=g=>{if(K.externalModules.length>0)return K.externalModules;const l=n.readFileSync(g,"utf8").match(/externalModules:\s?\[(.*)\]/);if(!l)return[];const I=l[1].split(",").map(e=>e.replace(/['"`]/g,"").trim());return K.externalModules=I,I},k=async g=>{if(K.destDir)return K.destDir;const c=await tg(g),l=s.join(g,(c==null?void 0:c.saltygenDir)||"saltygen");return K.destDir=l,l},Zg=["salty","css","styles","styled"],ug=(g=[])=>new RegExp(`\\.(${[...Zg,...g].join("|")})\\.`),gg=(g,c=[])=>ug(c).test(g),hg=async g=>{if(K.rcFile)return K.rcFile;if(g==="/")throw new Error("Could not find .saltyrc.json file");const c=s.join(g,".saltyrc.json"),l=await yg.readFile(c,"utf-8").then(JSON.parse).catch(()=>{});return l?(K.rcFile=l,l):hg(s.join(g,".."))},tg=async g=>{var I,e;const c=await hg(g),l=(I=c.projects)==null?void 0:I.find(Z=>g.endsWith(Z.dir||""));return l||((e=c.projects)==null?void 0:e.find(Z=>Z.dir===c.defaultProject))},fg=async g=>{const c=await tg(g),l=await k(g),I=s.join(g,(c==null?void 0:c.configDir)||"","salty.config.ts"),e=s.join(l,"salty.config.js"),Z=await Xg(g),C=Ng(I);await Yg.build({entryPoints:[I],minify:!0,treeShaking:!0,bundle:!0,outfile:e,format:Z,external:C});const G=Date.now(),{config:y}=await import(`${e}?t=${G}`);return{config:y,path:e}},Bg=async(g,c)=>{var ig,Gg;const l=await k(g),I={mediaQueries:[],globalStyles:[],variables:[],templates:[]};await Promise.all([...c].map(async W=>{const{contents:X,outputFilePath:D}=await M(g,W,l);Object.entries(X).forEach(([r,u])=>{u.isMedia?I.mediaQueries.push([r,u]):u.isGlobalDefine?I.globalStyles.push(u):u.isDefineVariables?I.variables.push(u):u.isDefineTemplates&&I.templates.push(u._setPath(`${r};;${D}`))})}));const{config:e,path:Z}=await fg(g),C={...e},{mediaQueries:G}=I;C.mediaQueries=Object.fromEntries(G.map(([W,X])=>[`@${W}`,X]));const y=G.map(([W])=>`'@${W}'`).join(" | "),m=new Set,b=async(W,X=[])=>{if(!W)return[];const D=Object.entries(W).map(async([u,v])=>{const O=async F=>{if(!F)return;if(F instanceof Promise)return await O(await F);if(typeof F=="function")return await O(await F());if(typeof F=="object")return await b(F,[...X,u]);const Cg=Vg(u),sg=J.dashCase(u),bg=[...X,Cg].join(".");m.add(`"${bg}"`);const _=[...X.map(J.dashCase),sg].join("-"),dg=ng.parseVariableTokens(F);return dg?`--${_}: ${dg.transformed};`:`--${_}: ${F};`};return await O(v)});return(await Promise.all(D)).flat()},d=async W=>{if(!W)return[];const X=Object.entries(W).map(async([r,u])=>{const v=await b(u);return r==="base"?v.join(""):C.mediaQueries[r]?`${C.mediaQueries[r]} { ${v.join("")} }`:`${r} { ${v.join("")} }`});return(await Promise.all(X)).flat()},i=async W=>{if(!W)return[];const X=Object.entries(W).map(async([r,u])=>{const v=Object.entries(u).map(async([F,Cg])=>{const sg=await b(Cg,[r]),bg=`.${r}-${F}, [data-${r}="${F}"]`,_=sg.join("");return`${bg} { ${_} }`});return(await Promise.all(v)).flat()});return(await Promise.all(X)).flat()},A=W=>({...W,responsive:void 0,conditional:void 0}),o=W=>I.variables.map(X=>W==="static"?A(X._current):X._current[W]),V=j.mergeObjects(A(e.variables),o("static")),B=await b(V),R=j.mergeObjects((ig=e.variables)==null?void 0:ig.responsive,o("responsive")),z=await d(R),x=j.mergeObjects((Gg=e.variables)==null?void 0:Gg.conditional,o("conditional")),$=await i(x),mg=s.join(l,"css/_variables.css"),w=`:root { ${B.join("")} ${z.join("")} } ${$.join("")}`;n.writeFileSync(mg,w),C.staticVariables=V;const cg=s.join(l,"css/_global.css"),a=j.mergeObjects(e.global,I.globalStyles),Y=await ng.parseAndJoinStyles(a,"");n.writeFileSync(cg,`@layer global { ${Y} }`);const p=s.join(l,"css/_reset.css"),N=e.reset==="none"?{}:typeof e.reset=="object"?e.reset:jg,h=await ng.parseAndJoinStyles(N,"");n.writeFileSync(p,`@layer reset { ${h} }`);const H=s.join(l,"css/_templates.css"),S=j.mergeObjects(e.templates,I.templates),Q=await Ag.parseTemplates(S),T=Ag.getTemplateTypes(S);n.writeFileSync(H,`@layer templates { ${Q} }`),C.templates=S;const q=e.templates?[Kg.defineTemplates(e.templates)._setPath(`config;;${Z}`)]:[],lg=j.mergeFactories(I.templates,q);C.templatePaths=Object.fromEntries(Object.entries(lg).map(([W,X])=>[W,X._path]));const L=s.join(l,"types/css-tokens.d.ts"),Rg=`
|
2
|
+
// Variable types
|
3
|
+
type VariableTokens = ${[...m].join("|")};
|
4
|
+
type PropertyValueToken = \`{\${VariableTokens}}\`;
|
5
|
+
|
6
|
+
// Template types
|
7
|
+
type TemplateTokens = {
|
8
|
+
${Object.entries(T).map(([W,X])=>`${W}?: ${X}`).join(`
|
9
|
+
`)}
|
10
|
+
}
|
11
|
+
|
12
|
+
// Media query types
|
13
|
+
type MediaQueryKeys = ${y||"''"};
|
14
|
+
`;n.writeFileSync(L,Rg);const rg=s.join(l,"cache/config-cache.json");n.writeFileSync(rg,JSON.stringify(C,null,2));const Fg=s.join(xg,"../cache/config-cache.json");n.writeFileSync(Fg,JSON.stringify(C,null,2))},Wg=g=>g.replace(/styled\(([^"'`{,]+),/g,(c,l)=>{if(/^['"`]/.test(l))return c;const e=new RegExp(`import[^;]*${l}[,\\s{][^;]*from\\s?([^{};]+);`);if(!e.test(g))return c;const C=e.exec(g);if(C){const G=C.at(1);if(Zg.some(m=>G==null?void 0:G.includes(m)))return c}return"styled('div',"}),Ug=(g,c)=>{try{const l=n.readFileSync(s.join(c,"saltygen/cache/config-cache.json"),"utf8");return l?`globalThis.saltyConfig = ${l};
|
15
|
+
|
16
|
+
${g}`:`globalThis.saltyConfig = {};
|
17
|
+
|
18
|
+
${g}`}catch{return g}},M=async(g,c,l)=>{const I=J.toHash(c),e=s.join(l,"./temp");n.existsSync(e)||n.mkdirSync(e);const Z=s.parse(c);let C=n.readFileSync(c,"utf8");C=Wg(C),C=Ug(C,g);const G=s.join(l,"js",I+".js"),y=await tg(g),m=s.join(g,(y==null?void 0:y.configDir)||"","salty.config.ts"),b=Ng(m),d=await Xg(g);await Yg.build({stdin:{contents:C,sourcefile:Z.base,resolveDir:Z.dir,loader:"tsx"},minify:!1,treeShaking:!0,bundle:!0,outfile:G,format:d,target:["node20"],keepNames:!0,external:b,packages:"external",plugins:[{name:"test",setup:o=>{o.onLoad({filter:/.*\.css|salty|styles|styled\.ts/},V=>{const B=n.readFileSync(V.path,"utf8");return{contents:Wg(B),loader:"ts"}})}}]});const i=Date.now();return{contents:await import(`${G}?t=${i}`),outputFilePath:G}},Qg=async g=>{const c=await k(g),l=s.join(c,"cache/config-cache.json"),I=n.readFileSync(l,"utf8");if(!I)throw new Error("Could not find config cache file");return JSON.parse(I)},ag=async g=>{const c=await Qg(g),l=await k(g),I=s.join(l,"salty.config.js"),e=Date.now(),{config:Z}=await import(`${I}?t=${e}`);return j.mergeObjects(Z,c)},og=()=>{try{return process.env.NODE_ENV==="production"}catch{return!1}},Tg=async(g,c=og(),l=!0)=>{try{const I=Date.now();c?E.info("Generating CSS in production mode! 🔥"):E.info("Generating CSS in development mode! 🚀");const e=[],Z=[],C=await k(g),G=s.join(C,"index.css");l&&(()=>{n.existsSync(C)&&Hg.execSync("rm -rf "+C),n.mkdirSync(C,{recursive:!0}),n.mkdirSync(s.join(C,"css")),n.mkdirSync(s.join(C,"types")),n.mkdirSync(s.join(C,"js")),n.mkdirSync(s.join(C,"cache"))})();const m=new Set,b=new Set;async function d(a){const Y=["node_modules","saltygen"],p=n.statSync(a);if(p.isDirectory()){const t=n.readdirSync(a);if(Y.some(h=>a.includes(h)))return;await Promise.all(t.map(h=>d(s.join(a,h))))}else if(p.isFile()&&gg(a)){m.add(a);const N=n.readFileSync(a,"utf8");/define[\w\d]+\(/.test(N)&&b.add(a)}}await d(g),await Bg(g,b);const i={keyframes:[],components:[],classNames:[]};await Promise.all([...m].map(async a=>{const{contents:Y}=await M(g,a,C);for(let[p,t]of Object.entries(Y))t instanceof Promise&&(t=await t),t.isKeyframes?i.keyframes.push({value:t,src:a,name:p}):t.isClassName?i.classNames.push({...t,src:a,name:p}):t.generator&&i.components.push({...t,src:a,name:p})}));const A=await ag(g);for(const a of i.keyframes){const{value:Y}=a,p=`a_${Y.animationName}.css`,t=`css/${p}`,N=s.join(C,t);e.push(p),n.writeFileSync(N,Y.css)}const o={};for(const a of i.components){const{src:Y,name:p}=a;o[Y]||(o[Y]=[]);const t=a.generator._withBuildContext({callerName:p,isProduction:c,config:A});Z[t.priority]||(Z[t.priority]=[]);const N=await t.css;if(!N)continue;Z[t.priority].push(t.cssFileName);const h=`css/${t.cssFileName}`,H=s.join(C,h);n.writeFileSync(H,N),A.importStrategy==="component"&&o[Y].push(t.cssFileName)}for(const a of i.classNames){const{src:Y,name:p}=a;o[Y]||(o[Y]=[]);const t=a.generator._withBuildContext({callerName:p,isProduction:c,config:A}),N=await t.css;if(!N)continue;Z[t.priority]||(Z[t.priority]=[]),Z[t.priority].push(t.cssFileName);const h=`css/${t.cssFileName}`,H=s.join(C,h);n.writeFileSync(H,N),A.importStrategy==="component"&&o[Y].push(t.cssFileName)}A.importStrategy==="component"&&Object.entries(o).forEach(([a,Y])=>{const p=Y.map(S=>`@import url('./${S}');`).join(`
|
19
|
+
`),t=J.toHash(a,6),N=s.parse(a),h=J.dashCase(N.name),H=s.join(C,`css/f_${h}-${t}.css`);n.writeFileSync(H,p||"/* Empty file */")});const V=e.map(a=>`@import url('./css/${a}');`).join(`
|
20
|
+
`);let $=`/*!
|
21
|
+
* Generated with Salty CSS (https://salty-css.dev)
|
22
|
+
* Do not edit this file directly
|
23
|
+
*/
|
24
|
+
@layer reset, global, templates, l0, l1, l2, l3, l4, l5, l6, l7, l8;
|
25
|
+
|
26
|
+
${["_variables.css","_reset.css","_global.css","_templates.css"].filter(a=>{try{return n.readFileSync(s.join(C,"css",a),"utf8").length>0}catch{return!1}}).map(a=>`@import url('./css/${a}');`).join(`
|
27
|
+
`)}
|
28
|
+
${V}`;if(A.importStrategy!=="component"){const a=Z.reduce((Y,p,t)=>{const N=p.reduce((Q,T)=>{var Ig;const q=s.join(C,"css",T),lg=n.readFileSync(q,"utf8"),L=((Ig=/.*-([^-]+)-\d+.css/.exec(T))==null?void 0:Ig.at(1))||J.toHash(q,6);return Q.includes(L)?Q:`${Q}
|
29
|
+
/*start:${L}-${T}*/
|
30
|
+
${lg}
|
31
|
+
/*end:${L}*/
|
32
|
+
`},""),h=`l_${t}.css`,H=s.join(C,"css",h),S=`@layer l${t} { ${N}
|
33
|
+
}`;return n.writeFileSync(H,S),`${Y}
|
34
|
+
@import url('./css/${h}');`},"");$+=a}n.writeFileSync(G,$);const w=Date.now()-I,cg=w<200?"🔥":w<500?"🚀":w<1e3?"🎉":w<2e3?"🚗":w<5e3?"🤔":"🥴";E.info(`Generated CSS in ${w}ms! ${cg}`)}catch(I){console.error(I)}},Lg=async(g,c,l=og())=>{try{const I=await k(g);if(gg(c)){const Z=[],C=await ag(g),{contents:G}=await M(g,c,I);for(const[y,m]of Object.entries(G)){if(m.isKeyframes&&m.css){const V=`css/${`a_${m.animationName}.css`}`,B=s.join(I,V);n.writeFileSync(B,await m.css);continue}if(m.isClassName){const o=m.generator._withBuildContext({callerName:y,isProduction:l,config:C}),V=await o.css;if(!V)continue;Z[o.priority]||(Z[o.priority]=[]),Z[o.priority].push(o.cssFileName);const B=`css/${o.cssFileName}`,R=s.join(I,B);n.writeFileSync(R,V);continue}if(!m.generator)continue;const b=m.generator._withBuildContext({callerName:y,isProduction:l,config:C}),d=await b.css;if(!d)continue;const i=`css/${b.cssFileName}`,A=s.join(I,i);n.writeFileSync(A,d),Z[b.priority]||(Z[b.priority]=[]),Z[b.priority].push(b.cssFileName)}if(C.importStrategy!=="component")Z.forEach((y,m)=>{const b=`l_${m}.css`,d=s.join(I,"css",b);let i=n.readFileSync(d,"utf8");y.forEach(A=>{var R;const o=s.join(I,"css",A),V=((R=/.*-([^-]+)-\d+.css/.exec(A))==null?void 0:R.at(1))||J.toHash(o,6);if(!i.includes(V)){const z=n.readFileSync(o,"utf8"),x=`/*start:${V}-${A}*/
|
35
|
+
${z}
|
36
|
+
/*end:${V}*/
|
37
|
+
`;i=`${i.replace(/\}$/,"")}
|
38
|
+
${x}
|
39
|
+
}`}}),n.writeFileSync(d,i)});else{const y=Z.flat().map(A=>`@import url('./${A}');`).join(`
|
40
|
+
`),m=J.toHash(c,6),b=s.parse(c),d=J.dashCase(b.name),i=s.join(I,`css/f_${d}-${m}.css`);n.writeFileSync(i,y||"/* Empty file */")}}}catch(I){console.error(I)}},Dg=async(g,c,l=og())=>{var I,e;try{const Z=await k(g);if(gg(c)){const G=n.readFileSync(c,"utf8"),y=await ag(g),{contents:m}=await M(g,c,Z);let b=G;for(const[d,i]of Object.entries(m)){if(i.isKeyframes||!i.generator)continue;const A=i.generator._withBuildContext({callerName:d,isProduction:l,config:y}),[o,V]=await kg(b,d),B=b.slice(o,V);if(i.isClassName){const R=b,z=` ${d} = className("${A.classNames}")`;b=b.replace(B,z),R===b&&console.error("Minimize file failed to change content",{name:d})}if(B.includes("styled")){const R=(e=(I=/styled\(([^,]+),/.exec(B))==null?void 0:I.at(1))==null?void 0:e.trim(),z=b,x=` ${d} = styled(${R}, "${A.classNames}", ${JSON.stringify(A.clientProps)})`;b=b.replace(B,x),z===b&&console.error("Minimize file failed to change content",{name:d,tagName:R})}}if(y.importStrategy==="component"){const d=J.toHash(c,6),i=s.parse(c);b=`import '../../saltygen/css/${`f_${J.dashCase(i.name)}-${d}.css`}';
|
41
|
+
${b}`}return b=b.replace("@salty-css/react/class-name","@salty-css/react/class-name-client"),b=b.replace("{ styled }","{ styledClient as styled }"),b=b.replace("@salty-css/react/styled","@salty-css/react/styled-client"),b}}catch(Z){console.error("Error in minimizeFile:",Z)}};exports.compileSaltyFile=M;exports.generateConfigStyles=Bg;exports.generateCss=Tg;exports.generateFile=Lg;exports.isSaltyFile=gg;exports.logError=vg;exports.logger=E;exports.minimizeFile=Dg;exports.saltyFileExtensions=Zg;exports.saltyFileRegExp=ug;
|