import fs from 'fs' import * as path from 'path' import postcss from 'postcss' import createUtilityPlugin from './util/createUtilityPlugin' import buildMediaQuery from './util/buildMediaQuery' import escapeClassName from './util/escapeClassName' import parseAnimationValue from './util/parseAnimationValue' import flattenColorPalette from './util/flattenColorPalette' import withAlphaVariable, { withAlphaValue } from './util/withAlphaVariable' import toColorValue from './util/toColorValue' import isPlainObject from './util/isPlainObject' import transformThemeValue from './util/transformThemeValue' import { version as tailwindVersion } from '../package.json' import log from './util/log' import { normalizeScreens, isScreenSortable, compareScreens, toScreen, } from './util/normalizeScreens' import { formatBoxShadowValue, parseBoxShadowValue } from './util/parseBoxShadowValue' import { removeAlphaVariables } from './util/removeAlphaVariables' import { flagEnabled } from './featureFlags' import { normalize } from './util/dataTypes' import { INTERNAL_FEATURES } from './lib/setupContextUtils' export let variantPlugins = { childVariant: ({ addVariant }) => { addVariant('*', '& > *') }, pseudoElementVariants: ({ addVariant }) => { addVariant('first-letter', '&::first-letter') addVariant('first-line', '&::first-line') addVariant('marker', [ ({ container }) => { removeAlphaVariables(container, ['--tw-text-opacity']) return '& *::marker' }, ({ container }) => { removeAlphaVariables(container, ['--tw-text-opacity']) return '&::marker' }, ]) addVariant('selection', ['& *::selection', '&::selection']) addVariant('file', '&::file-selector-button') addVariant('placeholder', '&::placeholder') addVariant('backdrop', '&::backdrop') addVariant('before', ({ container }) => { container.walkRules((rule) => { let foundContent = false rule.walkDecls('content', () => { foundContent = true }) if (!foundContent) { rule.prepend(postcss.decl({ prop: 'content', value: 'var(--tw-content)' })) } }) return '&::before' }) addVariant('after', ({ container }) => { container.walkRules((rule) => { let foundContent = false rule.walkDecls('content', () => { foundContent = true }) if (!foundContent) { rule.prepend(postcss.decl({ prop: 'content', value: 'var(--tw-content)' })) } }) return '&::after' }) }, pseudoClassVariants: ({ addVariant, matchVariant, config, prefix }) => { let pseudoVariants = [ // Positional ['first', '&:first-child'], ['last', '&:last-child'], ['only', '&:only-child'], ['odd', '&:nth-child(odd)'], ['even', '&:nth-child(even)'], 'first-of-type', 'last-of-type', 'only-of-type', // State [ 'visited', ({ container }) => { removeAlphaVariables(container, [ '--tw-text-opacity', '--tw-border-opacity', '--tw-bg-opacity', ]) return '&:visited' }, ], 'target', ['open', '&[open]'], // Forms 'default', 'checked', 'indeterminate', 'placeholder-shown', 'autofill', 'optional', 'required', 'valid', 'invalid', 'in-range', 'out-of-range', 'read-only', // Content 'empty', // Interactive 'focus-within', [ 'hover', !flagEnabled(config(), 'hoverOnlyWhenSupported') ? '&:hover' : '@media (hover: hover) and (pointer: fine) { &:hover }', ], 'focus', 'focus-visible', 'active', 'enabled', 'disabled', ].map((variant) => (Array.isArray(variant) ? variant : [variant, `&:${variant}`])) for (let [variantName, state] of pseudoVariants) { addVariant(variantName, (ctx) => { let result = typeof state === 'function' ? state(ctx) : state return result }) } let variants = { group: (_, { modifier }) => modifier ? [`:merge(${prefix('.group')}\\/${escapeClassName(modifier)})`, ' &'] : [`:merge(${prefix('.group')})`, ' &'], peer: (_, { modifier }) => modifier ? [`:merge(${prefix('.peer')}\\/${escapeClassName(modifier)})`, ' ~ &'] : [`:merge(${prefix('.peer')})`, ' ~ &'], } for (let [name, fn] of Object.entries(variants)) { matchVariant( name, (value = '', extra) => { let result = normalize(typeof value === 'function' ? value(extra) : value) if (!result.includes('&')) result = '&' + result let [a, b] = fn('', extra) let start = null let end = null let quotes = 0 for (let i = 0; i < result.length; ++i) { let c = result[i] if (c === '&') { start = i } else if (c === "'" || c === '"') { quotes += 1 } else if (start !== null && c === ' ' && !quotes) { end = i } } if (start !== null && end === null) { end = result.length } // Basically this but can handle quotes: // result.replace(/&(\S+)?/g, (_, pseudo = '') => a + pseudo + b) return result.slice(0, start) + a + result.slice(start + 1, end) + b + result.slice(end) }, { values: Object.fromEntries(pseudoVariants), [INTERNAL_FEATURES]: { respectPrefix: false, }, } ) } }, directionVariants: ({ addVariant }) => { addVariant('ltr', '&:where([dir="ltr"], [dir="ltr"] *)') addVariant('rtl', '&:where([dir="rtl"], [dir="rtl"] *)') }, reducedMotionVariants: ({ addVariant }) => { addVariant('motion-safe', '@media (prefers-reduced-motion: no-preference)') addVariant('motion-reduce', '@media (prefers-reduced-motion: reduce)') }, darkVariants: ({ config, addVariant }) => { let [mode, selector = '.dark'] = [].concat(config('darkMode', 'media')) if (mode === false) { mode = 'media' log.warn('darkmode-false', [ 'The `darkMode` option in your Tailwind CSS configuration is set to `false`, which now behaves the same as `media`.', 'Change `darkMode` to `media` or remove it entirely.', 'https://tailwindcss.com/docs/upgrade-guide#remove-dark-mode-configuration', ]) } if (mode === 'variant') { let formats if (Array.isArray(selector)) { formats = selector } else if (typeof selector === 'function') { formats = selector } else if (typeof selector === 'string') { formats = [selector] } // TODO: We could also add these warnings if the user passes a function that returns string | string[] // But this is an advanced enough use case that it's probably not necessary if (Array.isArray(formats)) { for (let format of formats) { if (format === '.dark') { mode = false log.warn('darkmode-variant-without-selector', [ 'When using `variant` for `darkMode`, you must provide a selector.', 'Example: `darkMode: ["variant", ".your-selector &"]`', ]) } else if (!format.includes('&')) { mode = false log.warn('darkmode-variant-without-ampersand', [ 'When using `variant` for `darkMode`, your selector must contain `&`.', 'Example `darkMode: ["variant", ".your-selector &"]`', ]) } } } selector = formats } if (mode === 'selector') { // New preferred behavior addVariant('dark', `&:where(${selector}, ${selector} *)`) } else if (mode === 'media') { addVariant('dark', '@media (prefers-color-scheme: dark)') } else if (mode === 'variant') { addVariant('dark', selector) } else if (mode === 'class') { // Old behavior addVariant('dark', `:is(${selector} &)`) } }, printVariant: ({ addVariant }) => { addVariant('print', '@media print') }, screenVariants: ({ theme, addVariant, matchVariant }) => { let rawScreens = theme('screens') ?? {} let areSimpleScreens = Object.values(rawScreens).every((v) => typeof v === 'string') let screens = normalizeScreens(theme('screens')) /** @type {Set} */ let unitCache = new Set([]) /** @param {string} value */ function units(value) { return value.match(/(\D+)$/)?.[1] ?? '(none)' } /** @param {string} value */ function recordUnits(value) { if (value !== undefined) { unitCache.add(units(value)) } } /** @param {string} value */ function canUseUnits(value) { recordUnits(value) // If the cache was empty it'll become 1 because we've just added the current unit // If the cache was not empty and the units are the same the size doesn't change // Otherwise, if the units are different from what is already known the size will always be > 1 return unitCache.size === 1 } for (const screen of screens) { for (const value of screen.values) { recordUnits(value.min) recordUnits(value.max) } } let screensUseConsistentUnits = unitCache.size <= 1 /** * @typedef {import('./util/normalizeScreens').Screen} Screen */ /** * @param {'min' | 'max'} type * @returns {Record} */ function buildScreenValues(type) { return Object.fromEntries( screens .filter((screen) => isScreenSortable(screen).result) .map((screen) => { let { min, max } = screen.values[0] if (type === 'min' && min !== undefined) { return screen } else if (type === 'min' && max !== undefined) { return { ...screen, not: !screen.not } } else if (type === 'max' && max !== undefined) { return screen } else if (type === 'max' && min !== undefined) { return { ...screen, not: !screen.not } } }) .map((screen) => [screen.name, screen]) ) } /** * @param {'min' | 'max'} type * @returns {(a: { value: string | Screen }, z: { value: string | Screen }) => number} */ function buildSort(type) { return (a, z) => compareScreens(type, a.value, z.value) } let maxSort = buildSort('max') let minSort = buildSort('min') /** @param {'min'|'max'} type */ function buildScreenVariant(type) { return (value) => { if (!areSimpleScreens) { log.warn('complex-screen-config', [ 'The `min-*` and `max-*` variants are not supported with a `screens` configuration containing objects.', ]) return [] } else if (!screensUseConsistentUnits) { log.warn('mixed-screen-units', [ 'The `min-*` and `max-*` variants are not supported with a `screens` configuration containing mixed units.', ]) return [] } else if (typeof value === 'string' && !canUseUnits(value)) { log.warn('minmax-have-mixed-units', [ 'The `min-*` and `max-*` variants are not supported with a `screens` configuration containing mixed units.', ]) return [] } return [`@media ${buildMediaQuery(toScreen(value, type))}`] } } matchVariant('max', buildScreenVariant('max'), { sort: maxSort, values: areSimpleScreens ? buildScreenValues('max') : {}, }) // screens and min-* are sorted together when they can be let id = 'min-screens' for (let screen of screens) { addVariant(screen.name, `@media ${buildMediaQuery(screen)}`, { id, sort: areSimpleScreens && screensUseConsistentUnits ? minSort : undefined, value: screen, }) } matchVariant('min', buildScreenVariant('min'), { id, sort: minSort, }) }, supportsVariants: ({ matchVariant, theme }) => { matchVariant( 'supports', (value = '') => { let check = normalize(value) let isRaw = /^\w*\s*\(/.test(check) // Chrome has a bug where `(condtion1)or(condition2)` is not valid // But `(condition1) or (condition2)` is supported. check = isRaw ? check.replace(/\b(and|or|not)\b/g, ' $1 ') : check if (isRaw) { return `@supports ${check}` } if (!check.includes(':')) { check = `${check}: var(--tw)` } if (!(check.startsWith('(') && check.endsWith(')'))) { check = `(${check})` } return `@supports ${check}` }, { values: theme('supports') ?? {} } ) }, hasVariants: ({ matchVariant }) => { matchVariant('has', (value) => `&:has(${normalize(value)})`, { values: {} }) matchVariant( 'group-has', (value, { modifier }) => modifier ? `:merge(.group\\/${modifier}):has(${normalize(value)}) &` : `:merge(.group):has(${normalize(value)}) &`, { values: {} } ) matchVariant( 'peer-has', (value, { modifier }) => modifier ? `:merge(.peer\\/${modifier}):has(${normalize(value)}) ~ &` : `:merge(.peer):has(${normalize(value)}) ~ &`, { values: {} } ) }, ariaVariants: ({ matchVariant, theme }) => { matchVariant('aria', (value) => `&[aria-${normalize(value)}]`, { values: theme('aria') ?? {} }) matchVariant( 'group-aria', (value, { modifier }) => modifier ? `:merge(.group\\/${modifier})[aria-${normalize(value)}] &` : `:merge(.group)[aria-${normalize(value)}] &`, { values: theme('aria') ?? {} } ) matchVariant( 'peer-aria', (value, { modifier }) => modifier ? `:merge(.peer\\/${modifier})[aria-${normalize(value)}] ~ &` : `:merge(.peer)[aria-${normalize(value)}] ~ &`, { values: theme('aria') ?? {} } ) }, dataVariants: ({ matchVariant, theme }) => { matchVariant('data', (value) => `&[data-${normalize(value)}]`, { values: theme('data') ?? {} }) matchVariant( 'group-data', (value, { modifier }) => modifier ? `:merge(.group\\/${modifier})[data-${normalize(value)}] &` : `:merge(.group)[data-${normalize(value)}] &`, { values: theme('data') ?? {} } ) matchVariant( 'peer-data', (value, { modifier }) => modifier ? `:merge(.peer\\/${modifier})[data-${normalize(value)}] ~ &` : `:merge(.peer)[data-${normalize(value)}] ~ &`, { values: theme('data') ?? {} } ) }, orientationVariants: ({ addVariant }) => { addVariant('portrait', '@media (orientation: portrait)') addVariant('landscape', '@media (orientation: landscape)') }, prefersContrastVariants: ({ addVariant }) => { addVariant('contrast-more', '@media (prefers-contrast: more)') addVariant('contrast-less', '@media (prefers-contrast: less)') }, forcedColorsVariants: ({ addVariant }) => { addVariant('forced-colors', '@media (forced-colors: active)') }, } let cssTransformValue = [ 'translate(var(--tw-translate-x), var(--tw-translate-y))', 'rotate(var(--tw-rotate))', 'skewX(var(--tw-skew-x))', 'skewY(var(--tw-skew-y))', 'scaleX(var(--tw-scale-x))', 'scaleY(var(--tw-scale-y))', ].join(' ') let cssFilterValue = [ 'var(--tw-blur)', 'var(--tw-brightness)', 'var(--tw-contrast)', 'var(--tw-grayscale)', 'var(--tw-hue-rotate)', 'var(--tw-invert)', 'var(--tw-saturate)', 'var(--tw-sepia)', 'var(--tw-drop-shadow)', ].join(' ') let cssBackdropFilterValue = [ 'var(--tw-backdrop-blur)', 'var(--tw-backdrop-brightness)', 'var(--tw-backdrop-contrast)', 'var(--tw-backdrop-grayscale)', 'var(--tw-backdrop-hue-rotate)', 'var(--tw-backdrop-invert)', 'var(--tw-backdrop-opacity)', 'var(--tw-backdrop-saturate)', 'var(--tw-backdrop-sepia)', ].join(' ') export let corePlugins = { preflight: ({ addBase }) => { let preflightStyles = postcss.parse( fs.readFileSync(path.join(__dirname, './css/preflight.css'), 'utf8') ) addBase([ postcss.comment({ text: `! tailwindcss v${tailwindVersion} | MIT License | https://tailwindcss.com`, }), ...preflightStyles.nodes, ]) }, container: (() => { function extractMinWidths(breakpoints = []) { return breakpoints .flatMap((breakpoint) => breakpoint.values.map((breakpoint) => breakpoint.min)) .filter((v) => v !== undefined) } function mapMinWidthsToPadding(minWidths, screens, paddings) { if (typeof paddings === 'undefined') { return [] } if (!(typeof paddings === 'object' && paddings !== null)) { return [ { screen: 'DEFAULT', minWidth: 0, padding: paddings, }, ] } let mapping = [] if (paddings.DEFAULT) { mapping.push({ screen: 'DEFAULT', minWidth: 0, padding: paddings.DEFAULT, }) } for (let minWidth of minWidths) { for (let screen of screens) { for (let { min } of screen.values) { if (min === minWidth) { mapping.push({ minWidth, padding: paddings[screen.name] }) } } } } return mapping } return function ({ addComponents, theme }) { let screens = normalizeScreens(theme('container.screens', theme('screens'))) let minWidths = extractMinWidths(screens) let paddings = mapMinWidthsToPadding(minWidths, screens, theme('container.padding')) let generatePaddingFor = (minWidth) => { let paddingConfig = paddings.find((padding) => padding.minWidth === minWidth) if (!paddingConfig) { return {} } return { paddingRight: paddingConfig.padding, paddingLeft: paddingConfig.padding, } } let atRules = Array.from( new Set(minWidths.slice().sort((a, z) => parseInt(a) - parseInt(z))) ).map((minWidth) => ({ [`@media (min-width: ${minWidth})`]: { '.container': { 'max-width': minWidth, ...generatePaddingFor(minWidth), }, }, })) addComponents([ { '.container': Object.assign( { width: '100%' }, theme('container.center', false) ? { marginRight: 'auto', marginLeft: 'auto' } : {}, generatePaddingFor(0) ), }, ...atRules, ]) } })(), accessibility: ({ addUtilities }) => { addUtilities({ '.sr-only': { position: 'absolute', width: '1px', height: '1px', padding: '0', margin: '-1px', overflow: 'hidden', clip: 'rect(0, 0, 0, 0)', whiteSpace: 'nowrap', borderWidth: '0', }, '.not-sr-only': { position: 'static', width: 'auto', height: 'auto', padding: '0', margin: '0', overflow: 'visible', clip: 'auto', whiteSpace: 'normal', }, }) }, pointerEvents: ({ addUtilities }) => { addUtilities({ '.pointer-events-none': { 'pointer-events': 'none' }, '.pointer-events-auto': { 'pointer-events': 'auto' }, }) }, visibility: ({ addUtilities }) => { addUtilities({ '.visible': { visibility: 'visible' }, '.invisible': { visibility: 'hidden' }, '.collapse': { visibility: 'collapse' }, }) }, position: ({ addUtilities }) => { addUtilities({ '.static': { position: 'static' }, '.fixed': { position: 'fixed' }, '.absolute': { position: 'absolute' }, '.relative': { position: 'relative' }, '.sticky': { position: 'sticky' }, }) }, inset: createUtilityPlugin( 'inset', [ ['inset', ['inset']], [ ['inset-x', ['left', 'right']], ['inset-y', ['top', 'bottom']], ], [ ['start', ['inset-inline-start']], ['end', ['inset-inline-end']], ['top', ['top']], ['right', ['right']], ['bottom', ['bottom']], ['left', ['left']], ], ], { supportsNegativeValues: true } ), isolation: ({ addUtilities }) => { addUtilities({ '.isolate': { isolation: 'isolate' }, '.isolation-auto': { isolation: 'auto' }, }) }, zIndex: createUtilityPlugin('zIndex', [['z', ['zIndex']]], { supportsNegativeValues: true }), order: createUtilityPlugin('order', undefined, { supportsNegativeValues: true }), gridColumn: createUtilityPlugin('gridColumn', [['col', ['gridColumn']]]), gridColumnStart: createUtilityPlugin('gridColumnStart', [['col-start', ['gridColumnStart']]]), gridColumnEnd: createUtilityPlugin('gridColumnEnd', [['col-end', ['gridColumnEnd']]]), gridRow: createUtilityPlugin('gridRow', [['row', ['gridRow']]]), gridRowStart: createUtilityPlugin('gridRowStart', [['row-start', ['gridRowStart']]]), gridRowEnd: createUtilityPlugin('gridRowEnd', [['row-end', ['gridRowEnd']]]), float: ({ addUtilities }) => { addUtilities({ '.float-start': { float: 'inline-start' }, '.float-end': { float: 'inline-end' }, '.float-right': { float: 'right' }, '.float-left': { float: 'left' }, '.float-none': { float: 'none' }, }) }, clear: ({ addUtilities }) => { addUtilities({ '.clear-start': { clear: 'inline-start' }, '.clear-end': { clear: 'inline-end' }, '.clear-left': { clear: 'left' }, '.clear-right': { clear: 'right' }, '.clear-both': { clear: 'both' }, '.clear-none': { clear: 'none' }, }) }, margin: createUtilityPlugin( 'margin', [ ['m', ['margin']], [ ['mx', ['margin-left', 'margin-right']], ['my', ['margin-top', 'margin-bottom']], ], [ ['ms', ['margin-inline-start']], ['me', ['margin-inline-end']], ['mt', ['margin-top']], ['mr', ['margin-right']], ['mb', ['margin-bottom']], ['ml', ['margin-left']], ], ], { supportsNegativeValues: true } ), boxSizing: ({ addUtilities }) => { addUtilities({ '.box-border': { 'box-sizing': 'border-box' }, '.box-content': { 'box-sizing': 'content-box' }, }) }, lineClamp: ({ matchUtilities, addUtilities, theme }) => { matchUtilities( { 'line-clamp': (value) => ({ overflow: 'hidden', display: '-webkit-box', '-webkit-box-orient': 'vertical', '-webkit-line-clamp': `${value}`, }), }, { values: theme('lineClamp') } ) addUtilities({ '.line-clamp-none': { overflow: 'visible', display: 'block', '-webkit-box-orient': 'horizontal', '-webkit-line-clamp': 'none', }, }) }, display: ({ addUtilities }) => { addUtilities({ '.block': { display: 'block' }, '.inline-block': { display: 'inline-block' }, '.inline': { display: 'inline' }, '.flex': { display: 'flex' }, '.inline-flex': { display: 'inline-flex' }, '.table': { display: 'table' }, '.inline-table': { display: 'inline-table' }, '.table-caption': { display: 'table-caption' }, '.table-cell': { display: 'table-cell' }, '.table-column': { display: 'table-column' }, '.table-column-group': { display: 'table-column-group' }, '.table-footer-group': { display: 'table-footer-group' }, '.table-header-group': { display: 'table-header-group' }, '.table-row-group': { display: 'table-row-group' }, '.table-row': { display: 'table-row' }, '.flow-root': { display: 'flow-root' }, '.grid': { display: 'grid' }, '.inline-grid': { display: 'inline-grid' }, '.contents': { display: 'contents' }, '.list-item': { display: 'list-item' }, '.hidden': { display: 'none' }, }) }, aspectRatio: createUtilityPlugin('aspectRatio', [['aspect', ['aspect-ratio']]]), size: createUtilityPlugin('size', [['size', ['width', 'height']]]), height: createUtilityPlugin('height', [['h', ['height']]]), maxHeight: createUtilityPlugin('maxHeight', [['max-h', ['maxHeight']]]), minHeight: createUtilityPlugin('minHeight', [['min-h', ['minHeight']]]), width: createUtilityPlugin('width', [['w', ['width']]]), minWidth: createUtilityPlugin('minWidth', [['min-w', ['minWidth']]]), maxWidth: createUtilityPlugin('maxWidth', [['max-w', ['maxWidth']]]), flex: createUtilityPlugin('flex'), flexShrink: createUtilityPlugin('flexShrink', [ ['flex-shrink', ['flex-shrink']], // Deprecated ['shrink', ['flex-shrink']], ]), flexGrow: createUtilityPlugin('flexGrow', [ ['flex-grow', ['flex-grow']], // Deprecated ['grow', ['flex-grow']], ]), flexBasis: createUtilityPlugin('flexBasis', [['basis', ['flex-basis']]]), tableLayout: ({ addUtilities }) => { addUtilities({ '.table-auto': { 'table-layout': 'auto' }, '.table-fixed': { 'table-layout': 'fixed' }, }) }, captionSide: ({ addUtilities }) => { addUtilities({ '.caption-top': { 'caption-side': 'top' }, '.caption-bottom': { 'caption-side': 'bottom' }, }) }, borderCollapse: ({ addUtilities }) => { addUtilities({ '.border-collapse': { 'border-collapse': 'collapse' }, '.border-separate': { 'border-collapse': 'separate' }, }) }, borderSpacing: ({ addDefaults, matchUtilities, theme }) => { addDefaults('border-spacing', { '--tw-border-spacing-x': 0, '--tw-border-spacing-y': 0, }) matchUtilities( { 'border-spacing': (value) => { return { '--tw-border-spacing-x': value, '--tw-border-spacing-y': value, '@defaults border-spacing': {}, 'border-spacing': 'var(--tw-border-spacing-x) var(--tw-border-spacing-y)', } }, 'border-spacing-x': (value) => { return { '--tw-border-spacing-x': value, '@defaults border-spacing': {}, 'border-spacing': 'var(--tw-border-spacing-x) var(--tw-border-spacing-y)', } }, 'border-spacing-y': (value) => { return { '--tw-border-spacing-y': value, '@defaults border-spacing': {}, 'border-spacing': 'var(--tw-border-spacing-x) var(--tw-border-spacing-y)', } }, }, { values: theme('borderSpacing') } ) }, transformOrigin: createUtilityPlugin('transformOrigin', [['origin', ['transformOrigin']]]), translate: createUtilityPlugin( 'translate', [ [ [ 'translate-x', [['@defaults transform', {}], '--tw-translate-x', ['transform', cssTransformValue]], ], [ 'translate-y', [['@defaults transform', {}], '--tw-translate-y', ['transform', cssTransformValue]], ], ], ], { supportsNegativeValues: true } ), rotate: createUtilityPlugin( 'rotate', [['rotate', [['@defaults transform', {}], '--tw-rotate', ['transform', cssTransformValue]]]], { supportsNegativeValues: true } ), skew: createUtilityPlugin( 'skew', [ [ ['skew-x', [['@defaults transform', {}], '--tw-skew-x', ['transform', cssTransformValue]]], ['skew-y', [['@defaults transform', {}], '--tw-skew-y', ['transform', cssTransformValue]]], ], ], { supportsNegativeValues: true } ), scale: createUtilityPlugin( 'scale', [ [ 'scale', [ ['@defaults transform', {}], '--tw-scale-x', '--tw-scale-y', ['transform', cssTransformValue], ], ], [ [ 'scale-x', [['@defaults transform', {}], '--tw-scale-x', ['transform', cssTransformValue]], ], [ 'scale-y', [['@defaults transform', {}], '--tw-scale-y', ['transform', cssTransformValue]], ], ], ], { supportsNegativeValues: true } ), transform: ({ addDefaults, addUtilities }) => { addDefaults('transform', { '--tw-translate-x': '0', '--tw-translate-y': '0', '--tw-rotate': '0', '--tw-skew-x': '0', '--tw-skew-y': '0', '--tw-scale-x': '1', '--tw-scale-y': '1', }) addUtilities({ '.transform': { '@defaults transform': {}, transform: cssTransformValue }, '.transform-cpu': { transform: cssTransformValue, }, '.transform-gpu': { transform: cssTransformValue.replace( 'translate(var(--tw-translate-x), var(--tw-translate-y))', 'translate3d(var(--tw-translate-x), var(--tw-translate-y), 0)' ), }, '.transform-none': { transform: 'none' }, }) }, animation: ({ matchUtilities, theme, config }) => { let prefixName = (name) => escapeClassName(config('prefix') + name) let keyframes = Object.fromEntries( Object.entries(theme('keyframes') ?? {}).map(([key, value]) => { return [key, { [`@keyframes ${prefixName(key)}`]: value }] }) ) matchUtilities( { animate: (value) => { let animations = parseAnimationValue(value) return [ ...animations.flatMap((animation) => keyframes[animation.name]), { animation: animations .map(({ name, value }) => { if (name === undefined || keyframes[name] === undefined) { return value } return value.replace(name, prefixName(name)) }) .join(', '), }, ] }, }, { values: theme('animation') } ) }, cursor: createUtilityPlugin('cursor'), touchAction: ({ addDefaults, addUtilities }) => { addDefaults('touch-action', { '--tw-pan-x': ' ', '--tw-pan-y': ' ', '--tw-pinch-zoom': ' ', }) let cssTouchActionValue = 'var(--tw-pan-x) var(--tw-pan-y) var(--tw-pinch-zoom)' addUtilities({ '.touch-auto': { 'touch-action': 'auto' }, '.touch-none': { 'touch-action': 'none' }, '.touch-pan-x': { '@defaults touch-action': {}, '--tw-pan-x': 'pan-x', 'touch-action': cssTouchActionValue, }, '.touch-pan-left': { '@defaults touch-action': {}, '--tw-pan-x': 'pan-left', 'touch-action': cssTouchActionValue, }, '.touch-pan-right': { '@defaults touch-action': {}, '--tw-pan-x': 'pan-right', 'touch-action': cssTouchActionValue, }, '.touch-pan-y': { '@defaults touch-action': {}, '--tw-pan-y': 'pan-y', 'touch-action': cssTouchActionValue, }, '.touch-pan-up': { '@defaults touch-action': {}, '--tw-pan-y': 'pan-up', 'touch-action': cssTouchActionValue, }, '.touch-pan-down': { '@defaults touch-action': {}, '--tw-pan-y': 'pan-down', 'touch-action': cssTouchActionValue, }, '.touch-pinch-zoom': { '@defaults touch-action': {}, '--tw-pinch-zoom': 'pinch-zoom', 'touch-action': cssTouchActionValue, }, '.touch-manipulation': { 'touch-action': 'manipulation' }, }) }, userSelect: ({ addUtilities }) => { addUtilities({ '.select-none': { 'user-select': 'none' }, '.select-text': { 'user-select': 'text' }, '.select-all': { 'user-select': 'all' }, '.select-auto': { 'user-select': 'auto' }, }) }, resize: ({ addUtilities }) => { addUtilities({ '.resize-none': { resize: 'none' }, '.resize-y': { resize: 'vertical' }, '.resize-x': { resize: 'horizontal' }, '.resize': { resize: 'both' }, }) }, scrollSnapType: ({ addDefaults, addUtilities }) => { addDefaults('scroll-snap-type', { '--tw-scroll-snap-strictness': 'proximity', }) addUtilities({ '.snap-none': { 'scroll-snap-type': 'none' }, '.snap-x': { '@defaults scroll-snap-type': {}, 'scroll-snap-type': 'x var(--tw-scroll-snap-strictness)', }, '.snap-y': { '@defaults scroll-snap-type': {}, 'scroll-snap-type': 'y var(--tw-scroll-snap-strictness)', }, '.snap-both': { '@defaults scroll-snap-type': {}, 'scroll-snap-type': 'both var(--tw-scroll-snap-strictness)', }, '.snap-mandatory': { '--tw-scroll-snap-strictness': 'mandatory' }, '.snap-proximity': { '--tw-scroll-snap-strictness': 'proximity' }, }) }, scrollSnapAlign: ({ addUtilities }) => { addUtilities({ '.snap-start': { 'scroll-snap-align': 'start' }, '.snap-end': { 'scroll-snap-align': 'end' }, '.snap-center': { 'scroll-snap-align': 'center' }, '.snap-align-none': { 'scroll-snap-align': 'none' }, }) }, scrollSnapStop: ({ addUtilities }) => { addUtilities({ '.snap-normal': { 'scroll-snap-stop': 'normal' }, '.snap-always': { 'scroll-snap-stop': 'always' }, }) }, scrollMargin: createUtilityPlugin( 'scrollMargin', [ ['scroll-m', ['scroll-margin']], [ ['scroll-mx', ['scroll-margin-left', 'scroll-margin-right']], ['scroll-my', ['scroll-margin-top', 'scroll-margin-bottom']], ], [ ['scroll-ms', ['scroll-margin-inline-start']], ['scroll-me', ['scroll-margin-inline-end']], ['scroll-mt', ['scroll-margin-top']], ['scroll-mr', ['scroll-margin-right']], ['scroll-mb', ['scroll-margin-bottom']], ['scroll-ml', ['scroll-margin-left']], ], ], { supportsNegativeValues: true } ), scrollPadding: createUtilityPlugin('scrollPadding', [ ['scroll-p', ['scroll-padding']], [ ['scroll-px', ['scroll-padding-left', 'scroll-padding-right']], ['scroll-py', ['scroll-padding-top', 'scroll-padding-bottom']], ], [ ['scroll-ps', ['scroll-padding-inline-start']], ['scroll-pe', ['scroll-padding-inline-end']], ['scroll-pt', ['scroll-padding-top']], ['scroll-pr', ['scroll-padding-right']], ['scroll-pb', ['scroll-padding-bottom']], ['scroll-pl', ['scroll-padding-left']], ], ]), listStylePosition: ({ addUtilities }) => { addUtilities({ '.list-inside': { 'list-style-position': 'inside' }, '.list-outside': { 'list-style-position': 'outside' }, }) }, listStyleType: createUtilityPlugin('listStyleType', [['list', ['listStyleType']]]), listStyleImage: createUtilityPlugin('listStyleImage', [['list-image', ['listStyleImage']]]), appearance: ({ addUtilities }) => { addUtilities({ '.appearance-none': { appearance: 'none' }, '.appearance-auto': { appearance: 'auto' }, }) }, columns: createUtilityPlugin('columns', [['columns', ['columns']]]), breakBefore: ({ addUtilities }) => { addUtilities({ '.break-before-auto': { 'break-before': 'auto' }, '.break-before-avoid': { 'break-before': 'avoid' }, '.break-before-all': { 'break-before': 'all' }, '.break-before-avoid-page': { 'break-before': 'avoid-page' }, '.break-before-page': { 'break-before': 'page' }, '.break-before-left': { 'break-before': 'left' }, '.break-before-right': { 'break-before': 'right' }, '.break-before-column': { 'break-before': 'column' }, }) }, breakInside: ({ addUtilities }) => { addUtilities({ '.break-inside-auto': { 'break-inside': 'auto' }, '.break-inside-avoid': { 'break-inside': 'avoid' }, '.break-inside-avoid-page': { 'break-inside': 'avoid-page' }, '.break-inside-avoid-column': { 'break-inside': 'avoid-column' }, }) }, breakAfter: ({ addUtilities }) => { addUtilities({ '.break-after-auto': { 'break-after': 'auto' }, '.break-after-avoid': { 'break-after': 'avoid' }, '.break-after-all': { 'break-after': 'all' }, '.break-after-avoid-page': { 'break-after': 'avoid-page' }, '.break-after-page': { 'break-after': 'page' }, '.break-after-left': { 'break-after': 'left' }, '.break-after-right': { 'break-after': 'right' }, '.break-after-column': { 'break-after': 'column' }, }) }, gridAutoColumns: createUtilityPlugin('gridAutoColumns', [['auto-cols', ['gridAutoColumns']]]), gridAutoFlow: ({ addUtilities }) => { addUtilities({ '.grid-flow-row': { gridAutoFlow: 'row' }, '.grid-flow-col': { gridAutoFlow: 'column' }, '.grid-flow-dense': { gridAutoFlow: 'dense' }, '.grid-flow-row-dense': { gridAutoFlow: 'row dense' }, '.grid-flow-col-dense': { gridAutoFlow: 'column dense' }, }) }, gridAutoRows: createUtilityPlugin('gridAutoRows', [['auto-rows', ['gridAutoRows']]]), gridTemplateColumns: createUtilityPlugin('gridTemplateColumns', [ ['grid-cols', ['gridTemplateColumns']], ]), gridTemplateRows: createUtilityPlugin('gridTemplateRows', [['grid-rows', ['gridTemplateRows']]]), flexDirection: ({ addUtilities }) => { addUtilities({ '.flex-row': { 'flex-direction': 'row' }, '.flex-row-reverse': { 'flex-direction': 'row-reverse' }, '.flex-col': { 'flex-direction': 'column' }, '.flex-col-reverse': { 'flex-direction': 'column-reverse' }, }) }, flexWrap: ({ addUtilities }) => { addUtilities({ '.flex-wrap': { 'flex-wrap': 'wrap' }, '.flex-wrap-reverse': { 'flex-wrap': 'wrap-reverse' }, '.flex-nowrap': { 'flex-wrap': 'nowrap' }, }) }, placeContent: ({ addUtilities }) => { addUtilities({ '.place-content-center': { 'place-content': 'center' }, '.place-content-start': { 'place-content': 'start' }, '.place-content-end': { 'place-content': 'end' }, '.place-content-between': { 'place-content': 'space-between' }, '.place-content-around': { 'place-content': 'space-around' }, '.place-content-evenly': { 'place-content': 'space-evenly' }, '.place-content-baseline': { 'place-content': 'baseline' }, '.place-content-stretch': { 'place-content': 'stretch' }, }) }, placeItems: ({ addUtilities }) => { addUtilities({ '.place-items-start': { 'place-items': 'start' }, '.place-items-end': { 'place-items': 'end' }, '.place-items-center': { 'place-items': 'center' }, '.place-items-baseline': { 'place-items': 'baseline' }, '.place-items-stretch': { 'place-items': 'stretch' }, }) }, alignContent: ({ addUtilities }) => { addUtilities({ '.content-normal': { 'align-content': 'normal' }, '.content-center': { 'align-content': 'center' }, '.content-start': { 'align-content': 'flex-start' }, '.content-end': { 'align-content': 'flex-end' }, '.content-between': { 'align-content': 'space-between' }, '.content-around': { 'align-content': 'space-around' }, '.content-evenly': { 'align-content': 'space-evenly' }, '.content-baseline': { 'align-content': 'baseline' }, '.content-stretch': { 'align-content': 'stretch' }, }) }, alignItems: ({ addUtilities }) => { addUtilities({ '.items-start': { 'align-items': 'flex-start' }, '.items-end': { 'align-items': 'flex-end' }, '.items-center': { 'align-items': 'center' }, '.items-baseline': { 'align-items': 'baseline' }, '.items-stretch': { 'align-items': 'stretch' }, }) }, justifyContent: ({ addUtilities }) => { addUtilities({ '.justify-normal': { 'justify-content': 'normal' }, '.justify-start': { 'justify-content': 'flex-start' }, '.justify-end': { 'justify-content': 'flex-end' }, '.justify-center': { 'justify-content': 'center' }, '.justify-between': { 'justify-content': 'space-between' }, '.justify-around': { 'justify-content': 'space-around' }, '.justify-evenly': { 'justify-content': 'space-evenly' }, '.justify-stretch': { 'justify-content': 'stretch' }, }) }, justifyItems: ({ addUtilities }) => { addUtilities({ '.justify-items-start': { 'justify-items': 'start' }, '.justify-items-end': { 'justify-items': 'end' }, '.justify-items-center': { 'justify-items': 'center' }, '.justify-items-stretch': { 'justify-items': 'stretch' }, }) }, gap: createUtilityPlugin('gap', [ ['gap', ['gap']], [ ['gap-x', ['columnGap']], ['gap-y', ['rowGap']], ], ]), space: ({ matchUtilities, addUtilities, theme }) => { matchUtilities( { 'space-x': (value) => { value = value === '0' ? '0px' : value if (__OXIDE__) { return { '& > :not([hidden]) ~ :not([hidden])': { '--tw-space-x-reverse': '0', 'margin-inline-end': `calc(${value} * var(--tw-space-x-reverse))`, 'margin-inline-start': `calc(${value} * calc(1 - var(--tw-space-x-reverse)))`, }, } } return { '& > :not([hidden]) ~ :not([hidden])': { '--tw-space-x-reverse': '0', 'margin-right': `calc(${value} * var(--tw-space-x-reverse))`, 'margin-left': `calc(${value} * calc(1 - var(--tw-space-x-reverse)))`, }, } }, 'space-y': (value) => { value = value === '0' ? '0px' : value return { '& > :not([hidden]) ~ :not([hidden])': { '--tw-space-y-reverse': '0', 'margin-top': `calc(${value} * calc(1 - var(--tw-space-y-reverse)))`, 'margin-bottom': `calc(${value} * var(--tw-space-y-reverse))`, }, } }, }, { values: theme('space'), supportsNegativeValues: true } ) addUtilities({ '.space-y-reverse > :not([hidden]) ~ :not([hidden])': { '--tw-space-y-reverse': '1' }, '.space-x-reverse > :not([hidden]) ~ :not([hidden])': { '--tw-space-x-reverse': '1' }, }) }, divideWidth: ({ matchUtilities, addUtilities, theme }) => { matchUtilities( { 'divide-x': (value) => { value = value === '0' ? '0px' : value if (__OXIDE__) { return { '& > :not([hidden]) ~ :not([hidden])': { '@defaults border-width': {}, '--tw-divide-x-reverse': '0', 'border-inline-end-width': `calc(${value} * var(--tw-divide-x-reverse))`, 'border-inline-start-width': `calc(${value} * calc(1 - var(--tw-divide-x-reverse)))`, }, } } return { '& > :not([hidden]) ~ :not([hidden])': { '@defaults border-width': {}, '--tw-divide-x-reverse': '0', 'border-right-width': `calc(${value} * var(--tw-divide-x-reverse))`, 'border-left-width': `calc(${value} * calc(1 - var(--tw-divide-x-reverse)))`, }, } }, 'divide-y': (value) => { value = value === '0' ? '0px' : value return { '& > :not([hidden]) ~ :not([hidden])': { '@defaults border-width': {}, '--tw-divide-y-reverse': '0', 'border-top-width': `calc(${value} * calc(1 - var(--tw-divide-y-reverse)))`, 'border-bottom-width': `calc(${value} * var(--tw-divide-y-reverse))`, }, } }, }, { values: theme('divideWidth'), type: ['line-width', 'length', 'any'] } ) addUtilities({ '.divide-y-reverse > :not([hidden]) ~ :not([hidden])': { '@defaults border-width': {}, '--tw-divide-y-reverse': '1', }, '.divide-x-reverse > :not([hidden]) ~ :not([hidden])': { '@defaults border-width': {}, '--tw-divide-x-reverse': '1', }, }) }, divideStyle: ({ addUtilities }) => { addUtilities({ '.divide-solid > :not([hidden]) ~ :not([hidden])': { 'border-style': 'solid' }, '.divide-dashed > :not([hidden]) ~ :not([hidden])': { 'border-style': 'dashed' }, '.divide-dotted > :not([hidden]) ~ :not([hidden])': { 'border-style': 'dotted' }, '.divide-double > :not([hidden]) ~ :not([hidden])': { 'border-style': 'double' }, '.divide-none > :not([hidden]) ~ :not([hidden])': { 'border-style': 'none' }, }) }, divideColor: ({ matchUtilities, theme, corePlugins }) => { matchUtilities( { divide: (value) => { if (!corePlugins('divideOpacity')) { return { ['& > :not([hidden]) ~ :not([hidden])']: { 'border-color': toColorValue(value), }, } } return { ['& > :not([hidden]) ~ :not([hidden])']: withAlphaVariable({ color: value, property: 'border-color', variable: '--tw-divide-opacity', }), } }, }, { values: (({ DEFAULT: _, ...colors }) => colors)(flattenColorPalette(theme('divideColor'))), type: ['color', 'any'], } ) }, divideOpacity: ({ matchUtilities, theme }) => { matchUtilities( { 'divide-opacity': (value) => { return { [`& > :not([hidden]) ~ :not([hidden])`]: { '--tw-divide-opacity': value } } }, }, { values: theme('divideOpacity') } ) }, placeSelf: ({ addUtilities }) => { addUtilities({ '.place-self-auto': { 'place-self': 'auto' }, '.place-self-start': { 'place-self': 'start' }, '.place-self-end': { 'place-self': 'end' }, '.place-self-center': { 'place-self': 'center' }, '.place-self-stretch': { 'place-self': 'stretch' }, }) }, alignSelf: ({ addUtilities }) => { addUtilities({ '.self-auto': { 'align-self': 'auto' }, '.self-start': { 'align-self': 'flex-start' }, '.self-end': { 'align-self': 'flex-end' }, '.self-center': { 'align-self': 'center' }, '.self-stretch': { 'align-self': 'stretch' }, '.self-baseline': { 'align-self': 'baseline' }, }) }, justifySelf: ({ addUtilities }) => { addUtilities({ '.justify-self-auto': { 'justify-self': 'auto' }, '.justify-self-start': { 'justify-self': 'start' }, '.justify-self-end': { 'justify-self': 'end' }, '.justify-self-center': { 'justify-self': 'center' }, '.justify-self-stretch': { 'justify-self': 'stretch' }, }) }, overflow: ({ addUtilities }) => { addUtilities({ '.overflow-auto': { overflow: 'auto' }, '.overflow-hidden': { overflow: 'hidden' }, '.overflow-clip': { overflow: 'clip' }, '.overflow-visible': { overflow: 'visible' }, '.overflow-scroll': { overflow: 'scroll' }, '.overflow-x-auto': { 'overflow-x': 'auto' }, '.overflow-y-auto': { 'overflow-y': 'auto' }, '.overflow-x-hidden': { 'overflow-x': 'hidden' }, '.overflow-y-hidden': { 'overflow-y': 'hidden' }, '.overflow-x-clip': { 'overflow-x': 'clip' }, '.overflow-y-clip': { 'overflow-y': 'clip' }, '.overflow-x-visible': { 'overflow-x': 'visible' }, '.overflow-y-visible': { 'overflow-y': 'visible' }, '.overflow-x-scroll': { 'overflow-x': 'scroll' }, '.overflow-y-scroll': { 'overflow-y': 'scroll' }, }) }, overscrollBehavior: ({ addUtilities }) => { addUtilities({ '.overscroll-auto': { 'overscroll-behavior': 'auto' }, '.overscroll-contain': { 'overscroll-behavior': 'contain' }, '.overscroll-none': { 'overscroll-behavior': 'none' }, '.overscroll-y-auto': { 'overscroll-behavior-y': 'auto' }, '.overscroll-y-contain': { 'overscroll-behavior-y': 'contain' }, '.overscroll-y-none': { 'overscroll-behavior-y': 'none' }, '.overscroll-x-auto': { 'overscroll-behavior-x': 'auto' }, '.overscroll-x-contain': { 'overscroll-behavior-x': 'contain' }, '.overscroll-x-none': { 'overscroll-behavior-x': 'none' }, }) }, scrollBehavior: ({ addUtilities }) => { addUtilities({ '.scroll-auto': { 'scroll-behavior': 'auto' }, '.scroll-smooth': { 'scroll-behavior': 'smooth' }, }) }, textOverflow: ({ addUtilities }) => { addUtilities({ '.truncate': { overflow: 'hidden', 'text-overflow': 'ellipsis', 'white-space': 'nowrap' }, '.overflow-ellipsis': { 'text-overflow': 'ellipsis' }, // Deprecated '.text-ellipsis': { 'text-overflow': 'ellipsis' }, '.text-clip': { 'text-overflow': 'clip' }, }) }, hyphens: ({ addUtilities }) => { addUtilities({ '.hyphens-none': { hyphens: 'none' }, '.hyphens-manual': { hyphens: 'manual' }, '.hyphens-auto': { hyphens: 'auto' }, }) }, whitespace: ({ addUtilities }) => { addUtilities({ '.whitespace-normal': { 'white-space': 'normal' }, '.whitespace-nowrap': { 'white-space': 'nowrap' }, '.whitespace-pre': { 'white-space': 'pre' }, '.whitespace-pre-line': { 'white-space': 'pre-line' }, '.whitespace-pre-wrap': { 'white-space': 'pre-wrap' }, '.whitespace-break-spaces': { 'white-space': 'break-spaces' }, }) }, textWrap: ({ addUtilities }) => { addUtilities({ '.text-wrap': { 'text-wrap': 'wrap' }, '.text-nowrap': { 'text-wrap': 'nowrap' }, '.text-balance': { 'text-wrap': 'balance' }, '.text-pretty': { 'text-wrap': 'pretty' }, }) }, wordBreak: ({ addUtilities }) => { addUtilities({ '.break-normal': { 'overflow-wrap': 'normal', 'word-break': 'normal' }, '.break-words': { 'overflow-wrap': 'break-word' }, '.break-all': { 'word-break': 'break-all' }, '.break-keep': { 'word-break': 'keep-all' }, }) }, borderRadius: createUtilityPlugin('borderRadius', [ ['rounded', ['border-radius']], [ ['rounded-s', ['border-start-start-radius', 'border-end-start-radius']], ['rounded-e', ['border-start-end-radius', 'border-end-end-radius']], ['rounded-t', ['border-top-left-radius', 'border-top-right-radius']], ['rounded-r', ['border-top-right-radius', 'border-bottom-right-radius']], ['rounded-b', ['border-bottom-right-radius', 'border-bottom-left-radius']], ['rounded-l', ['border-top-left-radius', 'border-bottom-left-radius']], ], [ ['rounded-ss', ['border-start-start-radius']], ['rounded-se', ['border-start-end-radius']], ['rounded-ee', ['border-end-end-radius']], ['rounded-es', ['border-end-start-radius']], ['rounded-tl', ['border-top-left-radius']], ['rounded-tr', ['border-top-right-radius']], ['rounded-br', ['border-bottom-right-radius']], ['rounded-bl', ['border-bottom-left-radius']], ], ]), borderWidth: createUtilityPlugin( 'borderWidth', [ ['border', [['@defaults border-width', {}], 'border-width']], [ ['border-x', [['@defaults border-width', {}], 'border-left-width', 'border-right-width']], ['border-y', [['@defaults border-width', {}], 'border-top-width', 'border-bottom-width']], ], [ ['border-s', [['@defaults border-width', {}], 'border-inline-start-width']], ['border-e', [['@defaults border-width', {}], 'border-inline-end-width']], ['border-t', [['@defaults border-width', {}], 'border-top-width']], ['border-r', [['@defaults border-width', {}], 'border-right-width']], ['border-b', [['@defaults border-width', {}], 'border-bottom-width']], ['border-l', [['@defaults border-width', {}], 'border-left-width']], ], ], { type: ['line-width', 'length'] } ), borderStyle: ({ addUtilities }) => { addUtilities({ '.border-solid': { 'border-style': 'solid' }, '.border-dashed': { 'border-style': 'dashed' }, '.border-dotted': { 'border-style': 'dotted' }, '.border-double': { 'border-style': 'double' }, '.border-hidden': { 'border-style': 'hidden' }, '.border-none': { 'border-style': 'none' }, }) }, borderColor: ({ matchUtilities, theme, corePlugins }) => { matchUtilities( { border: (value) => { if (!corePlugins('borderOpacity')) { return { 'border-color': toColorValue(value), } } return withAlphaVariable({ color: value, property: 'border-color', variable: '--tw-border-opacity', }) }, }, { values: (({ DEFAULT: _, ...colors }) => colors)(flattenColorPalette(theme('borderColor'))), type: ['color', 'any'], } ) matchUtilities( { 'border-x': (value) => { if (!corePlugins('borderOpacity')) { return { 'border-left-color': toColorValue(value), 'border-right-color': toColorValue(value), } } return withAlphaVariable({ color: value, property: ['border-left-color', 'border-right-color'], variable: '--tw-border-opacity', }) }, 'border-y': (value) => { if (!corePlugins('borderOpacity')) { return { 'border-top-color': toColorValue(value), 'border-bottom-color': toColorValue(value), } } return withAlphaVariable({ color: value, property: ['border-top-color', 'border-bottom-color'], variable: '--tw-border-opacity', }) }, }, { values: (({ DEFAULT: _, ...colors }) => colors)(flattenColorPalette(theme('borderColor'))), type: ['color', 'any'], } ) matchUtilities( { 'border-s': (value) => { if (!corePlugins('borderOpacity')) { return { 'border-inline-start-color': toColorValue(value), } } return withAlphaVariable({ color: value, property: 'border-inline-start-color', variable: '--tw-border-opacity', }) }, 'border-e': (value) => { if (!corePlugins('borderOpacity')) { return { 'border-inline-end-color': toColorValue(value), } } return withAlphaVariable({ color: value, property: 'border-inline-end-color', variable: '--tw-border-opacity', }) }, 'border-t': (value) => { if (!corePlugins('borderOpacity')) { return { 'border-top-color': toColorValue(value), } } return withAlphaVariable({ color: value, property: 'border-top-color', variable: '--tw-border-opacity', }) }, 'border-r': (value) => { if (!corePlugins('borderOpacity')) { return { 'border-right-color': toColorValue(value), } } return withAlphaVariable({ color: value, property: 'border-right-color', variable: '--tw-border-opacity', }) }, 'border-b': (value) => { if (!corePlugins('borderOpacity')) { return { 'border-bottom-color': toColorValue(value), } } return withAlphaVariable({ color: value, property: 'border-bottom-color', variable: '--tw-border-opacity', }) }, 'border-l': (value) => { if (!corePlugins('borderOpacity')) { return { 'border-left-color': toColorValue(value), } } return withAlphaVariable({ color: value, property: 'border-left-color', variable: '--tw-border-opacity', }) }, }, { values: (({ DEFAULT: _, ...colors }) => colors)(flattenColorPalette(theme('borderColor'))), type: ['color', 'any'], } ) }, borderOpacity: createUtilityPlugin('borderOpacity', [ ['border-opacity', ['--tw-border-opacity']], ]), backgroundColor: ({ matchUtilities, theme, corePlugins }) => { matchUtilities( { bg: (value) => { if (!corePlugins('backgroundOpacity')) { return { 'background-color': toColorValue(value), } } return withAlphaVariable({ color: value, property: 'background-color', variable: '--tw-bg-opacity', }) }, }, { values: flattenColorPalette(theme('backgroundColor')), type: ['color', 'any'] } ) }, backgroundOpacity: createUtilityPlugin('backgroundOpacity', [ ['bg-opacity', ['--tw-bg-opacity']], ]), backgroundImage: createUtilityPlugin('backgroundImage', [['bg', ['background-image']]], { type: ['lookup', 'image', 'url'], }), gradientColorStops: (() => { function transparentTo(value) { return withAlphaValue(value, 0, 'rgb(255 255 255 / 0)') } return function ({ matchUtilities, theme, addDefaults }) { addDefaults('gradient-color-stops', { '--tw-gradient-from-position': ' ', '--tw-gradient-via-position': ' ', '--tw-gradient-to-position': ' ', }) let options = { values: flattenColorPalette(theme('gradientColorStops')), type: ['color', 'any'], } let positionOptions = { values: theme('gradientColorStopPositions'), type: ['length', 'percentage'], } matchUtilities( { from: (value) => { let transparentToValue = transparentTo(value) return { '@defaults gradient-color-stops': {}, '--tw-gradient-from': `${toColorValue(value)} var(--tw-gradient-from-position)`, '--tw-gradient-to': `${transparentToValue} var(--tw-gradient-to-position)`, '--tw-gradient-stops': `var(--tw-gradient-from), var(--tw-gradient-to)`, } }, }, options ) matchUtilities( { from: (value) => { return { '--tw-gradient-from-position': value, } }, }, positionOptions ) matchUtilities( { via: (value) => { let transparentToValue = transparentTo(value) return { '@defaults gradient-color-stops': {}, '--tw-gradient-to': `${transparentToValue} var(--tw-gradient-to-position)`, '--tw-gradient-stops': `var(--tw-gradient-from), ${toColorValue( value )} var(--tw-gradient-via-position), var(--tw-gradient-to)`, } }, }, options ) matchUtilities( { via: (value) => { return { '--tw-gradient-via-position': value, } }, }, positionOptions ) matchUtilities( { to: (value) => ({ '@defaults gradient-color-stops': {}, '--tw-gradient-to': `${toColorValue(value)} var(--tw-gradient-to-position)`, }), }, options ) matchUtilities( { to: (value) => { return { '--tw-gradient-to-position': value, } }, }, positionOptions ) } })(), boxDecorationBreak: ({ addUtilities }) => { addUtilities({ '.decoration-slice': { 'box-decoration-break': 'slice' }, // Deprecated '.decoration-clone': { 'box-decoration-break': 'clone' }, // Deprecated '.box-decoration-slice': { 'box-decoration-break': 'slice' }, '.box-decoration-clone': { 'box-decoration-break': 'clone' }, }) }, backgroundSize: createUtilityPlugin('backgroundSize', [['bg', ['background-size']]], { type: ['lookup', 'length', 'percentage', 'size'], }), backgroundAttachment: ({ addUtilities }) => { addUtilities({ '.bg-fixed': { 'background-attachment': 'fixed' }, '.bg-local': { 'background-attachment': 'local' }, '.bg-scroll': { 'background-attachment': 'scroll' }, }) }, backgroundClip: ({ addUtilities }) => { addUtilities({ '.bg-clip-border': { 'background-clip': 'border-box' }, '.bg-clip-padding': { 'background-clip': 'padding-box' }, '.bg-clip-content': { 'background-clip': 'content-box' }, '.bg-clip-text': { 'background-clip': 'text' }, }) }, backgroundPosition: createUtilityPlugin('backgroundPosition', [['bg', ['background-position']]], { type: ['lookup', ['position', { preferOnConflict: true }]], }), backgroundRepeat: ({ addUtilities }) => { addUtilities({ '.bg-repeat': { 'background-repeat': 'repeat' }, '.bg-no-repeat': { 'background-repeat': 'no-repeat' }, '.bg-repeat-x': { 'background-repeat': 'repeat-x' }, '.bg-repeat-y': { 'background-repeat': 'repeat-y' }, '.bg-repeat-round': { 'background-repeat': 'round' }, '.bg-repeat-space': { 'background-repeat': 'space' }, }) }, backgroundOrigin: ({ addUtilities }) => { addUtilities({ '.bg-origin-border': { 'background-origin': 'border-box' }, '.bg-origin-padding': { 'background-origin': 'padding-box' }, '.bg-origin-content': { 'background-origin': 'content-box' }, }) }, fill: ({ matchUtilities, theme }) => { matchUtilities( { fill: (value) => { return { fill: toColorValue(value) } }, }, { values: flattenColorPalette(theme('fill')), type: ['color', 'any'] } ) }, stroke: ({ matchUtilities, theme }) => { matchUtilities( { stroke: (value) => { return { stroke: toColorValue(value) } }, }, { values: flattenColorPalette(theme('stroke')), type: ['color', 'url', 'any'] } ) }, strokeWidth: createUtilityPlugin('strokeWidth', [['stroke', ['stroke-width']]], { type: ['length', 'number', 'percentage'], }), objectFit: ({ addUtilities }) => { addUtilities({ '.object-contain': { 'object-fit': 'contain' }, '.object-cover': { 'object-fit': 'cover' }, '.object-fill': { 'object-fit': 'fill' }, '.object-none': { 'object-fit': 'none' }, '.object-scale-down': { 'object-fit': 'scale-down' }, }) }, objectPosition: createUtilityPlugin('objectPosition', [['object', ['object-position']]]), padding: createUtilityPlugin('padding', [ ['p', ['padding']], [ ['px', ['padding-left', 'padding-right']], ['py', ['padding-top', 'padding-bottom']], ], [ ['ps', ['padding-inline-start']], ['pe', ['padding-inline-end']], ['pt', ['padding-top']], ['pr', ['padding-right']], ['pb', ['padding-bottom']], ['pl', ['padding-left']], ], ]), textAlign: ({ addUtilities }) => { addUtilities({ '.text-left': { 'text-align': 'left' }, '.text-center': { 'text-align': 'center' }, '.text-right': { 'text-align': 'right' }, '.text-justify': { 'text-align': 'justify' }, '.text-start': { 'text-align': 'start' }, '.text-end': { 'text-align': 'end' }, }) }, textIndent: createUtilityPlugin('textIndent', [['indent', ['text-indent']]], { supportsNegativeValues: true, }), verticalAlign: ({ addUtilities, matchUtilities }) => { addUtilities({ '.align-baseline': { 'vertical-align': 'baseline' }, '.align-top': { 'vertical-align': 'top' }, '.align-middle': { 'vertical-align': 'middle' }, '.align-bottom': { 'vertical-align': 'bottom' }, '.align-text-top': { 'vertical-align': 'text-top' }, '.align-text-bottom': { 'vertical-align': 'text-bottom' }, '.align-sub': { 'vertical-align': 'sub' }, '.align-super': { 'vertical-align': 'super' }, }) matchUtilities({ align: (value) => ({ 'vertical-align': value }) }) }, fontFamily: ({ matchUtilities, theme }) => { matchUtilities( { font: (value) => { let [families, options = {}] = Array.isArray(value) && isPlainObject(value[1]) ? value : [value] let { fontFeatureSettings, fontVariationSettings } = options return { 'font-family': Array.isArray(families) ? families.join(', ') : families, ...(fontFeatureSettings === undefined ? {} : { 'font-feature-settings': fontFeatureSettings }), ...(fontVariationSettings === undefined ? {} : { 'font-variation-settings': fontVariationSettings }), } }, }, { values: theme('fontFamily'), type: ['lookup', 'generic-name', 'family-name'], } ) }, fontSize: ({ matchUtilities, theme }) => { matchUtilities( { text: (value, { modifier }) => { let [fontSize, options] = Array.isArray(value) ? value : [value] if (modifier) { return { 'font-size': fontSize, 'line-height': modifier, } } let { lineHeight, letterSpacing, fontWeight } = isPlainObject(options) ? options : { lineHeight: options } return { 'font-size': fontSize, ...(lineHeight === undefined ? {} : { 'line-height': lineHeight }), ...(letterSpacing === undefined ? {} : { 'letter-spacing': letterSpacing }), ...(fontWeight === undefined ? {} : { 'font-weight': fontWeight }), } }, }, { values: theme('fontSize'), modifiers: theme('lineHeight'), type: ['absolute-size', 'relative-size', 'length', 'percentage'], } ) }, fontWeight: createUtilityPlugin('fontWeight', [['font', ['fontWeight']]], { type: ['lookup', 'number', 'any'], }), textTransform: ({ addUtilities }) => { addUtilities({ '.uppercase': { 'text-transform': 'uppercase' }, '.lowercase': { 'text-transform': 'lowercase' }, '.capitalize': { 'text-transform': 'capitalize' }, '.normal-case': { 'text-transform': 'none' }, }) }, fontStyle: ({ addUtilities }) => { addUtilities({ '.italic': { 'font-style': 'italic' }, '.not-italic': { 'font-style': 'normal' }, }) }, fontVariantNumeric: ({ addDefaults, addUtilities }) => { let cssFontVariantNumericValue = 'var(--tw-ordinal) var(--tw-slashed-zero) var(--tw-numeric-figure) var(--tw-numeric-spacing) var(--tw-numeric-fraction)' addDefaults('font-variant-numeric', { '--tw-ordinal': ' ', '--tw-slashed-zero': ' ', '--tw-numeric-figure': ' ', '--tw-numeric-spacing': ' ', '--tw-numeric-fraction': ' ', }) addUtilities({ '.normal-nums': { 'font-variant-numeric': 'normal' }, '.ordinal': { '@defaults font-variant-numeric': {}, '--tw-ordinal': 'ordinal', 'font-variant-numeric': cssFontVariantNumericValue, }, '.slashed-zero': { '@defaults font-variant-numeric': {}, '--tw-slashed-zero': 'slashed-zero', 'font-variant-numeric': cssFontVariantNumericValue, }, '.lining-nums': { '@defaults font-variant-numeric': {}, '--tw-numeric-figure': 'lining-nums', 'font-variant-numeric': cssFontVariantNumericValue, }, '.oldstyle-nums': { '@defaults font-variant-numeric': {}, '--tw-numeric-figure': 'oldstyle-nums', 'font-variant-numeric': cssFontVariantNumericValue, }, '.proportional-nums': { '@defaults font-variant-numeric': {}, '--tw-numeric-spacing': 'proportional-nums', 'font-variant-numeric': cssFontVariantNumericValue, }, '.tabular-nums': { '@defaults font-variant-numeric': {}, '--tw-numeric-spacing': 'tabular-nums', 'font-variant-numeric': cssFontVariantNumericValue, }, '.diagonal-fractions': { '@defaults font-variant-numeric': {}, '--tw-numeric-fraction': 'diagonal-fractions', 'font-variant-numeric': cssFontVariantNumericValue, }, '.stacked-fractions': { '@defaults font-variant-numeric': {}, '--tw-numeric-fraction': 'stacked-fractions', 'font-variant-numeric': cssFontVariantNumericValue, }, }) }, lineHeight: createUtilityPlugin('lineHeight', [['leading', ['lineHeight']]]), letterSpacing: createUtilityPlugin('letterSpacing', [['tracking', ['letterSpacing']]], { supportsNegativeValues: true, }), textColor: ({ matchUtilities, theme, corePlugins }) => { matchUtilities( { text: (value) => { if (!corePlugins('textOpacity')) { return { color: toColorValue(value) } } return withAlphaVariable({ color: value, property: 'color', variable: '--tw-text-opacity', }) }, }, { values: flattenColorPalette(theme('textColor')), type: ['color', 'any'] } ) }, textOpacity: createUtilityPlugin('textOpacity', [['text-opacity', ['--tw-text-opacity']]]), textDecoration: ({ addUtilities }) => { addUtilities({ '.underline': { 'text-decoration-line': 'underline' }, '.overline': { 'text-decoration-line': 'overline' }, '.line-through': { 'text-decoration-line': 'line-through' }, '.no-underline': { 'text-decoration-line': 'none' }, }) }, textDecorationColor: ({ matchUtilities, theme }) => { matchUtilities( { decoration: (value) => { return { 'text-decoration-color': toColorValue(value) } }, }, { values: flattenColorPalette(theme('textDecorationColor')), type: ['color', 'any'] } ) }, textDecorationStyle: ({ addUtilities }) => { addUtilities({ '.decoration-solid': { 'text-decoration-style': 'solid' }, '.decoration-double': { 'text-decoration-style': 'double' }, '.decoration-dotted': { 'text-decoration-style': 'dotted' }, '.decoration-dashed': { 'text-decoration-style': 'dashed' }, '.decoration-wavy': { 'text-decoration-style': 'wavy' }, }) }, textDecorationThickness: createUtilityPlugin( 'textDecorationThickness', [['decoration', ['text-decoration-thickness']]], { type: ['length', 'percentage'] } ), textUnderlineOffset: createUtilityPlugin( 'textUnderlineOffset', [['underline-offset', ['text-underline-offset']]], { type: ['length', 'percentage', 'any'] } ), fontSmoothing: ({ addUtilities }) => { addUtilities({ '.antialiased': { '-webkit-font-smoothing': 'antialiased', '-moz-osx-font-smoothing': 'grayscale', }, '.subpixel-antialiased': { '-webkit-font-smoothing': 'auto', '-moz-osx-font-smoothing': 'auto', }, }) }, placeholderColor: ({ matchUtilities, theme, corePlugins }) => { matchUtilities( { placeholder: (value) => { if (!corePlugins('placeholderOpacity')) { return { '&::placeholder': { color: toColorValue(value), }, } } return { '&::placeholder': withAlphaVariable({ color: value, property: 'color', variable: '--tw-placeholder-opacity', }), } }, }, { values: flattenColorPalette(theme('placeholderColor')), type: ['color', 'any'] } ) }, placeholderOpacity: ({ matchUtilities, theme }) => { matchUtilities( { 'placeholder-opacity': (value) => { return { ['&::placeholder']: { '--tw-placeholder-opacity': value } } }, }, { values: theme('placeholderOpacity') } ) }, caretColor: ({ matchUtilities, theme }) => { matchUtilities( { caret: (value) => { return { 'caret-color': toColorValue(value) } }, }, { values: flattenColorPalette(theme('caretColor')), type: ['color', 'any'] } ) }, accentColor: ({ matchUtilities, theme }) => { matchUtilities( { accent: (value) => { return { 'accent-color': toColorValue(value) } }, }, { values: flattenColorPalette(theme('accentColor')), type: ['color', 'any'] } ) }, opacity: createUtilityPlugin('opacity', [['opacity', ['opacity']]]), backgroundBlendMode: ({ addUtilities }) => { addUtilities({ '.bg-blend-normal': { 'background-blend-mode': 'normal' }, '.bg-blend-multiply': { 'background-blend-mode': 'multiply' }, '.bg-blend-screen': { 'background-blend-mode': 'screen' }, '.bg-blend-overlay': { 'background-blend-mode': 'overlay' }, '.bg-blend-darken': { 'background-blend-mode': 'darken' }, '.bg-blend-lighten': { 'background-blend-mode': 'lighten' }, '.bg-blend-color-dodge': { 'background-blend-mode': 'color-dodge' }, '.bg-blend-color-burn': { 'background-blend-mode': 'color-burn' }, '.bg-blend-hard-light': { 'background-blend-mode': 'hard-light' }, '.bg-blend-soft-light': { 'background-blend-mode': 'soft-light' }, '.bg-blend-difference': { 'background-blend-mode': 'difference' }, '.bg-blend-exclusion': { 'background-blend-mode': 'exclusion' }, '.bg-blend-hue': { 'background-blend-mode': 'hue' }, '.bg-blend-saturation': { 'background-blend-mode': 'saturation' }, '.bg-blend-color': { 'background-blend-mode': 'color' }, '.bg-blend-luminosity': { 'background-blend-mode': 'luminosity' }, }) }, mixBlendMode: ({ addUtilities }) => { addUtilities({ '.mix-blend-normal': { 'mix-blend-mode': 'normal' }, '.mix-blend-multiply': { 'mix-blend-mode': 'multiply' }, '.mix-blend-screen': { 'mix-blend-mode': 'screen' }, '.mix-blend-overlay': { 'mix-blend-mode': 'overlay' }, '.mix-blend-darken': { 'mix-blend-mode': 'darken' }, '.mix-blend-lighten': { 'mix-blend-mode': 'lighten' }, '.mix-blend-color-dodge': { 'mix-blend-mode': 'color-dodge' }, '.mix-blend-color-burn': { 'mix-blend-mode': 'color-burn' }, '.mix-blend-hard-light': { 'mix-blend-mode': 'hard-light' }, '.mix-blend-soft-light': { 'mix-blend-mode': 'soft-light' }, '.mix-blend-difference': { 'mix-blend-mode': 'difference' }, '.mix-blend-exclusion': { 'mix-blend-mode': 'exclusion' }, '.mix-blend-hue': { 'mix-blend-mode': 'hue' }, '.mix-blend-saturation': { 'mix-blend-mode': 'saturation' }, '.mix-blend-color': { 'mix-blend-mode': 'color' }, '.mix-blend-luminosity': { 'mix-blend-mode': 'luminosity' }, '.mix-blend-plus-lighter': { 'mix-blend-mode': 'plus-lighter' }, }) }, boxShadow: (() => { let transformValue = transformThemeValue('boxShadow') let defaultBoxShadow = [ `var(--tw-ring-offset-shadow, 0 0 #0000)`, `var(--tw-ring-shadow, 0 0 #0000)`, `var(--tw-shadow)`, ].join(', ') return function ({ matchUtilities, addDefaults, theme }) { addDefaults(' box-shadow', { '--tw-ring-offset-shadow': '0 0 #0000', '--tw-ring-shadow': '0 0 #0000', '--tw-shadow': '0 0 #0000', '--tw-shadow-colored': '0 0 #0000', }) matchUtilities( { shadow: (value) => { value = transformValue(value) let ast = parseBoxShadowValue(value) for (let shadow of ast) { // Don't override color if the whole shadow is a variable if (!shadow.valid) { continue } shadow.color = 'var(--tw-shadow-color)' } return { '@defaults box-shadow': {}, '--tw-shadow': value === 'none' ? '0 0 #0000' : value, '--tw-shadow-colored': value === 'none' ? '0 0 #0000' : formatBoxShadowValue(ast), 'box-shadow': defaultBoxShadow, } }, }, { values: theme('boxShadow'), type: ['shadow'] } ) } })(), boxShadowColor: ({ matchUtilities, theme }) => { matchUtilities( { shadow: (value) => { return { '--tw-shadow-color': toColorValue(value), '--tw-shadow': 'var(--tw-shadow-colored)', } }, }, { values: flattenColorPalette(theme('boxShadowColor')), type: ['color', 'any'] } ) }, outlineStyle: ({ addUtilities }) => { addUtilities({ '.outline-none': { outline: '2px solid transparent', 'outline-offset': '2px', }, '.outline': { 'outline-style': 'solid' }, '.outline-dashed': { 'outline-style': 'dashed' }, '.outline-dotted': { 'outline-style': 'dotted' }, '.outline-double': { 'outline-style': 'double' }, }) }, outlineWidth: createUtilityPlugin('outlineWidth', [['outline', ['outline-width']]], { type: ['length', 'number', 'percentage'], }), outlineOffset: createUtilityPlugin('outlineOffset', [['outline-offset', ['outline-offset']]], { type: ['length', 'number', 'percentage', 'any'], supportsNegativeValues: true, }), outlineColor: ({ matchUtilities, theme }) => { matchUtilities( { outline: (value) => { return { 'outline-color': toColorValue(value) } }, }, { values: flattenColorPalette(theme('outlineColor')), type: ['color', 'any'] } ) }, ringWidth: ({ matchUtilities, addDefaults, addUtilities, theme, config }) => { let ringColorDefault = (() => { if (flagEnabled(config(), 'respectDefaultRingColorOpacity')) { return theme('ringColor.DEFAULT') } let ringOpacityDefault = theme('ringOpacity.DEFAULT', '0.5') if (!theme('ringColor')?.DEFAULT) { return `rgb(147 197 253 / ${ringOpacityDefault})` } return withAlphaValue( theme('ringColor')?.DEFAULT, ringOpacityDefault, `rgb(147 197 253 / ${ringOpacityDefault})` ) })() addDefaults('ring-width', { '--tw-ring-inset': ' ', '--tw-ring-offset-width': theme('ringOffsetWidth.DEFAULT', '0px'), '--tw-ring-offset-color': theme('ringOffsetColor.DEFAULT', '#fff'), '--tw-ring-color': ringColorDefault, '--tw-ring-offset-shadow': '0 0 #0000', '--tw-ring-shadow': '0 0 #0000', '--tw-shadow': '0 0 #0000', '--tw-shadow-colored': '0 0 #0000', }) matchUtilities( { ring: (value) => { return { '@defaults ring-width': {}, '--tw-ring-offset-shadow': `var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color)`, '--tw-ring-shadow': `var(--tw-ring-inset) 0 0 0 calc(${value} + var(--tw-ring-offset-width)) var(--tw-ring-color)`, 'box-shadow': [ `var(--tw-ring-offset-shadow)`, `var(--tw-ring-shadow)`, `var(--tw-shadow, 0 0 #0000)`, ].join(', '), } }, }, { values: theme('ringWidth'), type: 'length' } ) addUtilities({ '.ring-inset': { '@defaults ring-width': {}, '--tw-ring-inset': 'inset' }, }) }, ringColor: ({ matchUtilities, theme, corePlugins }) => { matchUtilities( { ring: (value) => { if (!corePlugins('ringOpacity')) { return { '--tw-ring-color': toColorValue(value), } } return withAlphaVariable({ color: value, property: '--tw-ring-color', variable: '--tw-ring-opacity', }) }, }, { values: Object.fromEntries( Object.entries(flattenColorPalette(theme('ringColor'))).filter( ([modifier]) => modifier !== 'DEFAULT' ) ), type: ['color', 'any'], } ) }, ringOpacity: (helpers) => { let { config } = helpers return createUtilityPlugin('ringOpacity', [['ring-opacity', ['--tw-ring-opacity']]], { filterDefault: !flagEnabled(config(), 'respectDefaultRingColorOpacity'), })(helpers) }, ringOffsetWidth: createUtilityPlugin( 'ringOffsetWidth', [['ring-offset', ['--tw-ring-offset-width']]], { type: 'length' } ), ringOffsetColor: ({ matchUtilities, theme }) => { matchUtilities( { 'ring-offset': (value) => { return { '--tw-ring-offset-color': toColorValue(value), } }, }, { values: flattenColorPalette(theme('ringOffsetColor')), type: ['color', 'any'] } ) }, blur: ({ matchUtilities, theme }) => { matchUtilities( { blur: (value) => { return { '--tw-blur': `blur(${value})`, '@defaults filter': {}, filter: cssFilterValue, } }, }, { values: theme('blur') } ) }, brightness: ({ matchUtilities, theme }) => { matchUtilities( { brightness: (value) => { return { '--tw-brightness': `brightness(${value})`, '@defaults filter': {}, filter: cssFilterValue, } }, }, { values: theme('brightness') } ) }, contrast: ({ matchUtilities, theme }) => { matchUtilities( { contrast: (value) => { return { '--tw-contrast': `contrast(${value})`, '@defaults filter': {}, filter: cssFilterValue, } }, }, { values: theme('contrast') } ) }, dropShadow: ({ matchUtilities, theme }) => { matchUtilities( { 'drop-shadow': (value) => { return { '--tw-drop-shadow': Array.isArray(value) ? value.map((v) => `drop-shadow(${v})`).join(' ') : `drop-shadow(${value})`, '@defaults filter': {}, filter: cssFilterValue, } }, }, { values: theme('dropShadow') } ) }, grayscale: ({ matchUtilities, theme }) => { matchUtilities( { grayscale: (value) => { return { '--tw-grayscale': `grayscale(${value})`, '@defaults filter': {}, filter: cssFilterValue, } }, }, { values: theme('grayscale') } ) }, hueRotate: ({ matchUtilities, theme }) => { matchUtilities( { 'hue-rotate': (value) => { return { '--tw-hue-rotate': `hue-rotate(${value})`, '@defaults filter': {}, filter: cssFilterValue, } }, }, { values: theme('hueRotate'), supportsNegativeValues: true } ) }, invert: ({ matchUtilities, theme }) => { matchUtilities( { invert: (value) => { return { '--tw-invert': `invert(${value})`, '@defaults filter': {}, filter: cssFilterValue, } }, }, { values: theme('invert') } ) }, saturate: ({ matchUtilities, theme }) => { matchUtilities( { saturate: (value) => { return { '--tw-saturate': `saturate(${value})`, '@defaults filter': {}, filter: cssFilterValue, } }, }, { values: theme('saturate') } ) }, sepia: ({ matchUtilities, theme }) => { matchUtilities( { sepia: (value) => { return { '--tw-sepia': `sepia(${value})`, '@defaults filter': {}, filter: cssFilterValue, } }, }, { values: theme('sepia') } ) }, filter: ({ addDefaults, addUtilities }) => { addDefaults('filter', { '--tw-blur': ' ', '--tw-brightness': ' ', '--tw-contrast': ' ', '--tw-grayscale': ' ', '--tw-hue-rotate': ' ', '--tw-invert': ' ', '--tw-saturate': ' ', '--tw-sepia': ' ', '--tw-drop-shadow': ' ', }) addUtilities({ '.filter': { '@defaults filter': {}, filter: cssFilterValue }, '.filter-none': { filter: 'none' }, }) }, backdropBlur: ({ matchUtilities, theme }) => { matchUtilities( { 'backdrop-blur': (value) => { return { '--tw-backdrop-blur': `blur(${value})`, '@defaults backdrop-filter': {}, 'backdrop-filter': cssBackdropFilterValue, } }, }, { values: theme('backdropBlur') } ) }, backdropBrightness: ({ matchUtilities, theme }) => { matchUtilities( { 'backdrop-brightness': (value) => { return { '--tw-backdrop-brightness': `brightness(${value})`, '@defaults backdrop-filter': {}, 'backdrop-filter': cssBackdropFilterValue, } }, }, { values: theme('backdropBrightness') } ) }, backdropContrast: ({ matchUtilities, theme }) => { matchUtilities( { 'backdrop-contrast': (value) => { return { '--tw-backdrop-contrast': `contrast(${value})`, '@defaults backdrop-filter': {}, 'backdrop-filter': cssBackdropFilterValue, } }, }, { values: theme('backdropContrast') } ) }, backdropGrayscale: ({ matchUtilities, theme }) => { matchUtilities( { 'backdrop-grayscale': (value) => { return { '--tw-backdrop-grayscale': `grayscale(${value})`, '@defaults backdrop-filter': {}, 'backdrop-filter': cssBackdropFilterValue, } }, }, { values: theme('backdropGrayscale') } ) }, backdropHueRotate: ({ matchUtilities, theme }) => { matchUtilities( { 'backdrop-hue-rotate': (value) => { return { '--tw-backdrop-hue-rotate': `hue-rotate(${value})`, '@defaults backdrop-filter': {}, 'backdrop-filter': cssBackdropFilterValue, } }, }, { values: theme('backdropHueRotate'), supportsNegativeValues: true } ) }, backdropInvert: ({ matchUtilities, theme }) => { matchUtilities( { 'backdrop-invert': (value) => { return { '--tw-backdrop-invert': `invert(${value})`, '@defaults backdrop-filter': {}, 'backdrop-filter': cssBackdropFilterValue, } }, }, { values: theme('backdropInvert') } ) }, backdropOpacity: ({ matchUtilities, theme }) => { matchUtilities( { 'backdrop-opacity': (value) => { return { '--tw-backdrop-opacity': `opacity(${value})`, '@defaults backdrop-filter': {}, 'backdrop-filter': cssBackdropFilterValue, } }, }, { values: theme('backdropOpacity') } ) }, backdropSaturate: ({ matchUtilities, theme }) => { matchUtilities( { 'backdrop-saturate': (value) => { return { '--tw-backdrop-saturate': `saturate(${value})`, '@defaults backdrop-filter': {}, 'backdrop-filter': cssBackdropFilterValue, } }, }, { values: theme('backdropSaturate') } ) }, backdropSepia: ({ matchUtilities, theme }) => { matchUtilities( { 'backdrop-sepia': (value) => { return { '--tw-backdrop-sepia': `sepia(${value})`, '@defaults backdrop-filter': {}, 'backdrop-filter': cssBackdropFilterValue, } }, }, { values: theme('backdropSepia') } ) }, backdropFilter: ({ addDefaults, addUtilities }) => { addDefaults('backdrop-filter', { '--tw-backdrop-blur': ' ', '--tw-backdrop-brightness': ' ', '--tw-backdrop-contrast': ' ', '--tw-backdrop-grayscale': ' ', '--tw-backdrop-hue-rotate': ' ', '--tw-backdrop-invert': ' ', '--tw-backdrop-opacity': ' ', '--tw-backdrop-saturate': ' ', '--tw-backdrop-sepia': ' ', }) addUtilities({ '.backdrop-filter': { '@defaults backdrop-filter': {}, 'backdrop-filter': cssBackdropFilterValue, }, '.backdrop-filter-none': { 'backdrop-filter': 'none' }, }) }, transitionProperty: ({ matchUtilities, theme }) => { let defaultTimingFunction = theme('transitionTimingFunction.DEFAULT') let defaultDuration = theme('transitionDuration.DEFAULT') matchUtilities( { transition: (value) => { return { 'transition-property': value, ...(value === 'none' ? {} : { 'transition-timing-function': defaultTimingFunction, 'transition-duration': defaultDuration, }), } }, }, { values: theme('transitionProperty') } ) }, transitionDelay: createUtilityPlugin('transitionDelay', [['delay', ['transitionDelay']]]), transitionDuration: createUtilityPlugin( 'transitionDuration', [['duration', ['transitionDuration']]], { filterDefault: true } ), transitionTimingFunction: createUtilityPlugin( 'transitionTimingFunction', [['ease', ['transitionTimingFunction']]], { filterDefault: true } ), willChange: createUtilityPlugin('willChange', [['will-change', ['will-change']]]), content: createUtilityPlugin('content', [ ['content', ['--tw-content', ['content', 'var(--tw-content)']]], ]), forcedColorAdjust: ({ addUtilities }) => { addUtilities({ '.forced-color-adjust-auto': { 'forced-color-adjust': 'auto' }, '.forced-color-adjust-none': { 'forced-color-adjust': 'none' }, }) }, }