UNPKG

@dash-ui/styles

Version:

A tiny, powerful, framework-agnostic CSS-in-JS library.

958 lines (897 loc) 28 kB
import unitless from "@dash-ui/unitless"; import type { HtmlAttributes as CSSHTMLAttributes, PropertiesFallback as CSSProperties, Pseudos as CSSPseudos, SvgAttributes as CSSSvgAttributes, } from "csstype"; import type { JsonValue, PartialDeep, Primitive, ValueOf } from "type-fest"; import { createDash } from "./create-dash"; import type { Dash } from "./create-dash"; import { hash as fnv1aHash, noop, safeHash } from "./utils"; /** * A factory function that returns a new `styles` instance with * your custom configuration options. * * @param options - Configuration options */ export function createStyles< Tokens extends DashTokens = DashTokens, Themes extends DashThemes = DashThemes >(options: CreateStylesOptions<Tokens, Themes> = {}): Styles<Tokens, Themes> { const dash = options.dash || createDash(); const { key, insert, sheets } = dash; const themes = {} as Themes; const tokens = {} as TokensUnion<Tokens, Themes>; const hash = safeHash(key, options.hash || fnv1aHash); let label: (args: any[]) => string; // explicit here on purpose so it's not in every test /* istanbul ignore next */ if ( typeof process !== "undefined" && process.env.NODE_ENV === "development" ) { label = function (args) { // add helpful labels to the name in development return [...args] .reduce((curr, arg) => { if (typeof arg === "string") { curr += "-" + arg; } else if (typeof arg === "object") { const keys = Object.keys(arg).filter( (k) => typeof arg[k] === "number" || arg[k] ); if (keys.length) { curr += "-" + keys.join("-"); } } return curr; }, "") .replace(/[^\w-]/g, "-"); }; } const styles: Styles<Tokens, Themes> = { variants<Variants extends string | number>( styleMap: StyleMap<Variants, Tokens, Themes> ): Style<Variants, Tokens, Themes> { const compiledStyleMap: Record<string | number, string> = {}; let styleKey: keyof typeof styleMap; /* istanbul ignore next */ for (styleKey in styleMap) compiledStyleMap[styleKey] = compileStyles(styleMap[styleKey], tokens); const defaultStyles = compiledStyleMap.default || ""; // style('text', {}) function style(): string { // eslint-disable-next-line prefer-spread const css_ = css.apply(null, arguments as any); if (!css_) return ""; let name = hash(css_); /* istanbul ignore next */ if (label) name += label(arguments as any); const className = key + "-" + name; insert(name, "." + className, css_); return className; } function css(): string { const args = arguments as unknown as StyleArguments<Variants>; const numArgs = args.length; if (numArgs === 1 && typeof args[0] !== "object") { return defaultStyles + (compiledStyleMap[args[0] as any] || ""); } else if (numArgs > 0) { let nextStyles = defaultStyles; for (let i = 0; i < numArgs; i++) { let arg = args[i]; if (typeof arg !== "object") { nextStyles += compiledStyleMap[arg as any] || ""; } else if (arg !== null) { for (const key in arg) if (arg[key]) nextStyles += compiledStyleMap[key] || ""; } } return nextStyles; } return defaultStyles; } style.styles = styleMap; style.css = css; return style; }, one() { const one = compileStyles<Tokens, Themes>( compileLiterals(arguments), tokens ); const name = hash(one); const className = key + "-" + name; const callback: StylesOne = function (createClassName) { if (!createClassName && createClassName !== void 0) return ""; insert(name, "." + className, one); return className; }; callback.css = function (createCss) { return !createCss && createCss !== void 0 ? "" : one; }; return callback; }, cls() { const css = compileStyles<Tokens, Themes>( compileLiterals(arguments), tokens ); const name = hash(css); const className = key + "-" + name; insert(name, "." + className, css); return className; }, lazy<Value extends LazyValue>( lazyFn: ( value: Value ) => string | StyleCallback<Tokens, Themes> | StyleObject ): StylesLazy<Value> { const cache = new Map<string | Value, string>(); function css(value?: Value): string { if (value === void 0) return ""; const key = typeof value === "object" ? JSON.stringify(value) : value; let css = cache.get(key); if (css === void 0) { css = compileStyles<Tokens, Themes>(lazyFn(value), tokens); cache.set(key, css); } return css; } const lazyStyle: StylesLazy<Value> = function (value?: Value) { const css_ = css(value); if (!css_) return ""; const name = hash(css_); const className = key + "-" + name; insert(name, "." + className, css_); return className; }; lazyStyle.css = css; return lazyStyle; }, join() { const css = "".concat(...Array.prototype.slice.call(arguments)); const name = hash(css); const className = key + "-" + name; insert(name, "." + className, css); return className; }, keyframes() { const css = compileStyles<Tokens, Themes>( compileLiterals(arguments), tokens ); const name = hash(css); const animationName = key + "-" + name; // Adding to a cached sheet here rather than the default sheet because // we want this to persist between `clearCache()` calls. insert( name, "", "@keyframes " + animationName + "{" + css + "}", sheets.add(name) ); return animationName; }, insertGlobal() { const css = compileStyles<Tokens, Themes>( compileLiterals(arguments), tokens ); if (!css) return noop; const name = hash(css); insert(name, "", css, sheets.add(name)); return function () { !sheets.delete(name) && dash.inserted.delete(name); }; }, insertTokens(nextTokens, selector = ":root") { const { css, vars } = serializeTokens(nextTokens, options.mangleTokens); if (!css) return noop; mergeTokens<Tokens, Themes>(tokens, vars); return styles.insertGlobal(selector + "{" + css + "}"); }, insertThemes(nextThemes) { const flush: (() => void)[] = []; for (const name in nextThemes) { flush.push( styles.insertTokens( // God the types here are f'ing stupid. Someone should feel free to fix this. // @ts-expect-error (themes[name] = themes[name] === void 0 ? // @ts-expect-error nextThemes[name] : mergeTokens<any>(themes[name], nextThemes[name] as any)), "." + styles.theme(name as Extract<keyof Themes, string>) ) ); } return function () { flush.forEach((e) => e()); }; }, theme(theme) { return key + "-" + theme + "-theme"; }, dash, hash, tokens, }; Object.defineProperty(styles, "tokens", { get() { return tokens; }, configurable: false, }); styles.insertTokens(options.tokens || emptyObj); styles.insertThemes(options.themes || emptyObj); return typeof process !== "undefined" && process.env.NODE_ENV !== "production" ? Object.freeze(styles) : styles; } const emptyObj: any = {}; export interface CreateStylesOptions< Tokens extends DashTokens = DashTokens, Themes extends DashThemes = DashThemes > { /** * An instance of dash created by the `createDash()` factory * * @default createDash() */ dash?: Dash; /** * Inserts CSS tokens into the DOM and makes them available for use in * style callbacks. The name of the CSS tokens is automatically generated * based upon the depth of the mapping i.e. `foo.bar.baz` -> `--foo-bar-baz`. * * @example * const styles = createStyles({ * tokens: { * color: { * // var(--color-light-red) * lightRed: '#c17' * } * } * }) * * const bgRed = styles.one(({color}) => ({ * backgroundColor: color.lightRed * })) * * const Component = () => <div className={bgRed()} /> */ readonly tokens?: Tokens; /** * A mapping of theme name/CSS variable pairs. * * This Creates a CSS variable-based theme by defining tokens within a * class name selector matching the theme name. Apart from that it works * the same way `tokens` does. * * @example * const styles = createStyles({ * themes: { * // .ui-light * light: { * // var(--background-color) * backgroundColor: '#fff' * }, * // .ui-dark * dark: { * // var(--background-color) * backgroundColor: '#000' * } * } * }) * * // CSS tokens in the 'dark' theme take precedence in this component * const App = () => <div className={styles.theme('dark)}/> */ readonly themes?: Themes; /** * When `true` this will mangle CSS variable names. You can also * provide an object with `{key: boolean}` pairs of reserved keys * which will not be mangled. * * @example * const styles = createStyles({ * // All CSS tokens will be mangled in production * mangleTokens: process.env.NODE_ENV === 'production' * }) * @example * const styles = createStyles({ * mangleTokens: { * // --vh will not be mangled * vh: true * } * }) */ readonly mangleTokens?: boolean | Record<string, boolean>; /** * Use your own hash function for creating selector names. By default * Dash uses an fnv1a hashing algorithm. */ readonly hash?: typeof fnv1aHash; } /** * Utility methods that accomplish everything you need to scale an application * using CSS-in-JS. */ export interface Styles< Tokens extends DashTokens = DashTokens, Themes extends DashThemes = DashThemes > { /** * `styles.variants()` is a function for composing styles in a * deterministic way. It returns a function which when called will insert * your styles into the DOM and create a unique class name. * * @param styleMap - A style name/value mapping * @example * const bg = styles({ * // Define styles using an object * blue: { * backgroundColor: 'blue' * }, * // Access stored CSS tokens when a callback is provided as * // the value * red: ({colors}) => ` * background-color: ${colors.red}; * `, * // Define styles using a string * green: ` * background-color: green; * ` * }) * * // This component will have a "red" background * const Component = () => <div className={bg('blue', 'red')}/> * * // This component will have a "blue" background * const Component = () => <div className={bg('red', 'blue')}/> * * // This component will have a "green" background * const Component = () => <div className={bg({red: true, green: true})}/> */ variants<Variants extends string | number>( styleMap: StyleMap<Variants, Tokens, Themes> ): Style<Variants, Tokens, Themes>; /** * A function that accepts a tagged template literal, style object, or style callback, * and returns a function. That function inserts the style into a `<style>` tag and * returns a class name when called. * * @example * const row = styles.one` * display: flex; * flex-wrap: nowrap; * ` * const Row = props => <div {...props} className={row()}/>> * // This will not insert the styles if `isRow` is `false` * const RowSometimes = ({isRow = false}) => <div className={row(isRow)}/>> */ one( literals: | TemplateStringsArray | string | StyleObject | StyleCallback<Tokens, Themes>, ...placeholders: string[] ): StylesOne; /** * A function that accepts a tagged template literal, style object, or style callback. * Calling this will immediately insert the CSS into the DOM and return a unique * class name for the styles. This is a shortcut for `styles.one('display: flex;')()`. * * @example * const Component = () => <div className={styles.cls`display: flex;`}/> */ cls( literals: | TemplateStringsArray | string | StyleObject | StyleCallback<Tokens, Themes>, ...placeholders: string[] ): string; /** * A function that uses lazy evalution to create styles with indeterminate values. * Calling this will immediately insert the CSS into the DOM and return a unique * class name for the styles. * * @example * const lazyWidth = styles.lazy((width) => ({ * width * })) * const Component = ({width = 200}) => <div className={lazyWidth(width)}/>> */ lazy<Value extends LazyValue>( lazyFn: ( value: Value ) => string | StyleCallback<Tokens, Themes> | StyleObject ): StylesLazy<Value>; /** * A function that joins CSS strings, inserts them into the DOM right away, and returns a class name. * * @example * const Component = () => <div * className={styles.join( * button.css('primary'), * transition.css('fade'), * 'display: block;' * )} * /> */ join(...css: string[]): string; /** * A function that accepts a tagged template literal, style object, or style callback. * Using this will immediately insert a global `@keyframes` defintion into the DOM and * return the name of the keyframes instance. * * @example * const fadeIn = styles.keyframes` * from { * opacity: 0; * } * * to { * opactity: 1 * } * ` */ keyframes( literals: | TemplateStringsArray | string | StyleCallback<Tokens, Themes> | StyleObject, ...placeholders: string[] ): string; /** * A function that returns the generated class name for a given theme when * using `insertThemes()` to create CSS variable-based themes. * * @param name - The name of the theme * @example * styles.insertThemes({ * dark: { * color: { * background: '#000' * } * } * }) * * const Component = () => <div className={styles.theme('dark')}/> */ theme(name: Extract<keyof Themes, string>): string; /** * Inserts CSS tokens into the DOM and makes them available for use in * style callbacks. The name of the CSS tokens is automatically generated * based upon the depth of the mapping i.e. `foo.bar.baz` -> `--foo-bar-baz`. * This function returns a function that will flush the styles inserted by * `insertTokens()` when it is called. * * @param tokens - A map of CSS variable name/value pairs * @param selector - Including a selector will only make these CSS variable * definitions take effect within the selector, e.g. a class name or ID. By * default the selector is `":root"`. * @example * // Inserts CSS tokens into the document `:root` * styles.insertTokens({ * color: { * // var(--color-indigo) * indigo: '#5c6ac4', * // var(--color-blue) * blue: '#007ace', * // var(--color-red) * red: '#de3618', * } * }) * * // Overrides the above when they are used within a `.dark` selector * const flushTokens = styles.insertTokens( * { * color: { * // var(--color-indigo) * indigo: '#5c6ac4', * // var(--color-blue) * blue: '#007ace', * // var(--color-red) * red: '#de3618', * } * }, * '.dark' * ) */ insertTokens(tokens: PartialDeep<Tokens>, selector?: string): () => void; /** * Creates a CSS variable-based theme by defining tokens within a * class name selector matching the theme name. Apart from that it works * the same way `insertTokens()` does. This function returns a function * that will flush the styles inserted by `insertTokens()` when it is called. * * @param themes - A mapping of theme name/CSS variable pairs. * @example * const flushThemes = styles.insertThemes({ * // .ui-light * light: { * // var(--background-color) * backgroundColor: '#fff' * }, * // .ui-dark * dark: { * // var(--background-color) * backgroundColor: '#000' * } * }) * * // "dark" css tokens will take precedence within this component * const Component = () => <div className={styles.theme('dark)}/> */ insertThemes( themes: PartialDeep<{ [Name in keyof Themes]: Themes[Name]; }> ): () => void; /** * A function that accepts a tagged template literal, style object, or style callback. * Using this will immediately insert styles into the DOM relative to the root document. * This function returns a function that will flush the styles inserted by * `insertGlobal()` when it is called. * * @example * const flushGlobal = styles.insertGlobal(({color}) => ` * body { * background-color: ${color.primaryBg}; * } * `) */ insertGlobal( literals: | TemplateStringsArray | string | StyleCallback<Tokens, Themes> | StyleObject, ...placeholders: string[] ): () => void; /** * The CSS tokens currently defined in the instance */ tokens: TokensUnion<Tokens, Themes>; /** * A hashing function for creating unique selector names * * @param string - The string you'd like to hash */ hash(string: string): string; /** * The instance of underlying the Dash cache used by this instance. This was * automatically created by `createDash()` when `createStyles()` was called. * Dash controls the caching, style sheets, auto-prefixing, and DOM insertion * that happens in the `styles` instance. */ dash: Dash; } /** * A function that inserts styles from the style map into the DOM when called * with those style names selected. * * @param args - A series of style names or style name/boolean maps which * select the styles from the style map you want to compose into a singular * deterministic style and class name. * @example * const style = styles.variants({ * block: 'display: block', * w100: 'width: 100px;', * h100: 'height: 100px', * }) * * // display: block; height: 100px; width: 100px; * const Component = () => <div className={style('block', 'h100', 'w100')}/> */ export type Style< Variants extends string | number, Tokens extends DashTokens = DashTokens, Themes extends DashThemes = DashThemes > = { (...args: StyleArguments<Variants>): string; /** * A function that returns the raw, CSS string for a given * name in the style map. * * @param names - A series of style names or style name/boolean maps which * select the styles from the style map you want to compose into a singular * CSS string. * @example * const style = styles.variants({ * block: 'display: block', * w100: 'width: 100px;', * h100: 'height: 100px', * }) * * const someOtherStyle = styles.variants({ * // display: block; height: 100px; width: 100px; * default: style.css('block', 'h100', 'w100') * }) */ css(...names: StyleArguments<Variants>): string; /** * The style map that this `style()` instance was instantiated with. */ styles: StyleMap<Variants, Tokens, Themes>; }; /** * A function that inserts styles into the DOM when called without * a falsy value. If the first argument is falsy, the styles will * not be inserted and a class name will not be returned. */ export type StylesOne = { (createClassName?: boolean | number | string | null): string; /** * A method that returns a CSS string if the first argument is not falsy. */ css(createCss?: boolean | number | string | null): string; }; export type StyleMap< Variants extends string | number, Tokens extends DashTokens = DashTokens, Themes extends DashThemes = DashThemes > = { [Name in Variants | "default"]?: StyleValue<Tokens, Themes>; }; export type StyleArguments<Variants extends string | number> = ( | Variants | { [Name in Variants]?: boolean | null | undefined | string | number; } | Exclude<Falsy, 0 | ""> )[]; export type StyleValue< Tokens extends DashTokens = DashTokens, Themes extends DashThemes = DashThemes > = string | StyleCallback<Tokens, Themes> | StyleObject; type KnownStyles = { [property in keyof CSSProperties]?: | CSSProperties[property] // eslint-disable-next-line @typescript-eslint/ban-types | (string & {}) // eslint-disable-next-line @typescript-eslint/ban-types | (number & {}); }; type PseudoStyles = { [property in CSSPseudos | CSSHTMLAttributes | CSSSvgAttributes]?: StyleObject; }; type SelectorStyles = { [property: string]: | string | number | KnownStyles | PseudoStyles | SelectorStyles; }; export type StyleObject = KnownStyles & PseudoStyles & SelectorStyles; export type StyleCallback< Tokens extends DashTokens = DashTokens, Themes extends DashThemes = DashThemes > = (tokens: TokensUnion<Tokens, Themes>) => StyleObject | string; export type LazyValue = JsonValue; /** * A function that inserts indeterminate styles based on the value * into the DOM when called. * * @param value - A JSON serializable value to create indeterminate * styles from */ export type StylesLazy<Value extends LazyValue> = { (value?: Value): string; /** * A method that returns indeterminate CSS strings based on the value * when called. * * @param value - A JSON serializable value to create indeterminate * styles from */ css(value?: Value): string; }; // // Utils export type Falsy = false | null | undefined | "" | 0; /** * A utility function that will compile style objects and callbacks into CSS strings. * * @param styles - A style callback, object, or string * @param tokens - A map of CSS tokens for style callbacks */ export function compileStyles< Tokens extends DashTokens = DashTokens, Themes extends DashThemes = DashThemes >( styles: StyleValue<Tokens, Themes> | Falsy, tokens: TokensUnion<Tokens, Themes> ): string { const value = typeof styles === "function" ? styles(tokens) : styles; return typeof value === "object" && value !== null ? stringifyStyleObject(value) : // TypeScript w/o "strict": true throws here ((value || "") as string); } function stringifyStyleObject(object: StyleObject): string { let string = ""; for (const key in object) { const value = object[key]; if (typeof value !== "object") { const isCustom = key.charCodeAt(1) === 45; string += (isCustom ? key : cssCase(key)) + ":" + (typeof value !== "number" || unitless[key as keyof typeof unitless] || value === 0 || isCustom ? value : value + "px") + ";"; } else { string += key + "{" + stringifyStyleObject(value as StyleObject) + "}"; } } return string; } function compileLiterals(args: IArguments): string { const literals = args[0]; return Array.isArray(literals) ? literals.reduce((curr, next, i) => curr + next + (args[i + 1] || ""), "") : literals; } // // Variable and theme serialization const cssCaseRe = /[A-Z]|^ms/g; const cssDisallowedRe = /[^\w-]/g; // We cache the case transformations below because the cache // will grow to a predictable size and the regex is slowwwww const caseCache: Record<string, string> = {}; function cssCase(string: string): string { return ( caseCache[string] ?? (caseCache[string] = string.replace(cssCaseRe, "-$&").toLowerCase()) ); } function serializeTokens( tokens: Record<string, any>, mangle?: CreateStylesOptions["mangleTokens"], names: string[] = [] ): SerializedTokens { const vars: Record<string, any> = {}; let css = ""; for (let key in tokens) { const value = tokens[key]; if (typeof value === "object") { const result = serializeTokens(value, mangle, names.concat(key)); vars[key] = result.vars; css += result.css; } else { let name = cssCase( names.length > 0 ? names.join("-") + "-" + key : key ).replace(cssDisallowedRe, "-"); vars[key] = "var(" + (name = "--" + (mangle === true || (mangle && !mangle[name]) ? mangled(name) : name)) + ")"; css += name + ":" + value + ";"; } } return { vars, css }; } const mangled = safeHash("", fnv1aHash); type SerializedTokens = { readonly vars: Record<string, Record<string, any> | string | number>; readonly css: string; }; function mergeTokens< Tokens extends DashTokens = DashTokens, Themes extends DashThemes = DashThemes >( target: Record<string, any>, source: Record<string, any> ): TokensUnion<Tokens, Themes> { for (const key in source) { const value = source[key]; target[key] = typeof value === "object" ? mergeTokens(target[key] || {}, value) : value; } return target as TokensUnion<Tokens, Themes>; } /** * A utility function that will convert a camel-cased, dot-notation string * into a dash-cased CSS property variable. * * @param path - A dot-notation string that represents the path to a value */ export function pathToToken< Tokens extends Record<string, unknown> = TokensUnion<DashTokens, DashThemes> >(path: KeysUnion<Tokens>): string { return ( "var(--" + path.replace(/\./g, "-").replace(cssCaseRe, "-$&").toLowerCase() + ")" ); } type Concat<Fst, Scd> = Fst extends string ? Scd extends string | number ? Fst extends "" ? `${Scd}` : `${Fst}.${Scd}` : never : never; type KeysUnion<T, Cache extends string = ""> = T extends Primitive ? Cache : { [P in keyof T]: Concat<Cache, P> | KeysUnion<T[P], Concat<Cache, P>>; }[keyof T]; export type TokensUnion< Tokens extends DashTokens = DashTokens, Themes extends DashThemes = DashThemes > = Tokens & ValueOf<Themes>; // // Creates and exports default `styles` instance export const styles: Styles<DashTokens, DashThemes> = createStyles(); /** * These are CSS variable type definitions that tell functions like * style callbacks which tokens are available. They can be defined * globally in your application like so: * * @example * declare module '@dash-ui/styles' { * export interface DashTokens { * color: { * red: string * } * } * } * * They can also be created automatically when you use a `createStyles()` factory. * @example * const styles = createStyles({ * tokens: { * foo: 'bar', * bar: 'baz' * } * }) * * // "foo" | "bar" * type Level1VariableNames = keyof DashTokens */ export interface DashTokens extends Record<string, unknown> {} /** * These are CSS variable theme type definitions that tell functions like * style callbacks which tokens are available and which themes are available in * `styles.theme()`. They can be defined globally in your application like so: * * @example * declare module '@dash-ui/styles' { * export interface DashThemes { * light: { * color: { * red: string; * } * } * dark: { * color: { * red: string; * } * } * } * } */ export interface DashThemes extends Record<string, Record<string, unknown>> {} /** * The names of the themes defined in the `DashThemes` type */ export type DashThemeNames = Extract<keyof DashThemes, string>;