mirror of
https://github.com/tailwindlabs/tailwindcss.git
synced 2025-12-08 21:36:08 +00:00
761 lines
21 KiB
JavaScript
761 lines
21 KiB
JavaScript
import fs from 'fs'
|
|
import url from 'url'
|
|
import postcss from 'postcss'
|
|
import dlv from 'dlv'
|
|
import selectorParser from 'postcss-selector-parser'
|
|
|
|
import transformThemeValue from '../util/transformThemeValue'
|
|
import parseObjectStyles from '../util/parseObjectStyles'
|
|
import prefixSelector from '../util/prefixSelector'
|
|
import isPlainObject from '../util/isPlainObject'
|
|
import escapeClassName from '../util/escapeClassName'
|
|
import nameClass, { formatClass } from '../util/nameClass'
|
|
import { coerceValue } from '../util/pluginUtils'
|
|
import bigSign from '../util/bigSign'
|
|
import corePlugins from '../corePlugins'
|
|
import * as sharedState from './sharedState'
|
|
import { env } from './sharedState'
|
|
import { toPath } from '../util/toPath'
|
|
import log from '../util/log'
|
|
|
|
function insertInto(list, value, { before = [] } = {}) {
|
|
before = [].concat(before)
|
|
|
|
if (before.length <= 0) {
|
|
list.push(value)
|
|
return
|
|
}
|
|
|
|
let idx = list.length - 1
|
|
for (let other of before) {
|
|
let iidx = list.indexOf(other)
|
|
if (iidx === -1) continue
|
|
idx = Math.min(idx, iidx)
|
|
}
|
|
|
|
list.splice(idx, 0, value)
|
|
}
|
|
|
|
function parseStyles(styles) {
|
|
if (!Array.isArray(styles)) {
|
|
return parseStyles([styles])
|
|
}
|
|
|
|
return styles.flatMap((style) => {
|
|
let isNode = !Array.isArray(style) && !isPlainObject(style)
|
|
return isNode ? style : parseObjectStyles(style)
|
|
})
|
|
}
|
|
|
|
function getClasses(selector) {
|
|
let parser = selectorParser((selectors) => {
|
|
let allClasses = []
|
|
selectors.walkClasses((classNode) => {
|
|
allClasses.push(classNode.value)
|
|
})
|
|
return allClasses
|
|
})
|
|
return parser.transformSync(selector)
|
|
}
|
|
|
|
function extractCandidates(node) {
|
|
let classes = []
|
|
|
|
if (node.type === 'rule') {
|
|
for (let selector of node.selectors) {
|
|
let classCandidates = getClasses(selector)
|
|
// At least one of the selectors contains non-"on-demandable" candidates.
|
|
if (classCandidates.length === 0) return []
|
|
|
|
classes = [...classes, ...classCandidates]
|
|
}
|
|
return classes
|
|
}
|
|
|
|
if (node.type === 'atrule') {
|
|
node.walkRules((rule) => {
|
|
classes = [...classes, ...rule.selectors.flatMap((selector) => getClasses(selector))]
|
|
})
|
|
}
|
|
|
|
return classes
|
|
}
|
|
|
|
function withIdentifiers(styles) {
|
|
return parseStyles(styles).flatMap((node) => {
|
|
let nodeMap = new Map()
|
|
let candidates = extractCandidates(node)
|
|
|
|
// If this isn't "on-demandable", assign it a universal candidate.
|
|
if (candidates.length === 0) {
|
|
return [['*', node]]
|
|
}
|
|
|
|
return candidates.map((c) => {
|
|
if (!nodeMap.has(node)) {
|
|
nodeMap.set(node, node)
|
|
}
|
|
return [c, nodeMap.get(node)]
|
|
})
|
|
})
|
|
}
|
|
|
|
let matchingBrackets = new Map([
|
|
['{', '}'],
|
|
['[', ']'],
|
|
['(', ')'],
|
|
])
|
|
let inverseMatchingBrackets = new Map(
|
|
Array.from(matchingBrackets.entries()).map(([k, v]) => [v, k])
|
|
)
|
|
|
|
let quotes = new Set(['"', "'", '`'])
|
|
|
|
// Arbitrary values must contain balanced brackets (), [] and {}. Escaped
|
|
// values don't count, and brackets inside quotes also don't count.
|
|
//
|
|
// E.g.: w-[this-is]w-[weird-and-invalid]
|
|
// E.g.: w-[this-is\\]w-\\[weird-but-valid]
|
|
// E.g.: content-['this-is-also-valid]-weirdly-enough']
|
|
function isValidArbitraryValue(value) {
|
|
let stack = []
|
|
let inQuotes = false
|
|
|
|
for (let i = 0; i < value.length; i++) {
|
|
let char = value[i]
|
|
|
|
// Non-escaped quotes allow us to "allow" anything in between
|
|
if (quotes.has(char) && value[i - 1] !== '\\') {
|
|
inQuotes = !inQuotes
|
|
}
|
|
|
|
if (inQuotes) continue
|
|
if (value[i - 1] === '\\') continue // Escaped
|
|
|
|
if (matchingBrackets.has(char)) {
|
|
stack.push(char)
|
|
} else if (inverseMatchingBrackets.has(char)) {
|
|
let inverse = inverseMatchingBrackets.get(char)
|
|
|
|
// Nothing to pop from, therefore it is unbalanced
|
|
if (stack.length <= 0) {
|
|
return false
|
|
}
|
|
|
|
// Popped value must match the inverse value, otherwise it is unbalanced
|
|
if (stack.pop() !== inverse) {
|
|
return false
|
|
}
|
|
}
|
|
}
|
|
|
|
// If there is still something on the stack, it is also unbalanced
|
|
if (stack.length > 0) {
|
|
return false
|
|
}
|
|
|
|
// All good, totally balanced!
|
|
return true
|
|
}
|
|
|
|
function buildPluginApi(tailwindConfig, context, { variantList, variantMap, offsets, classList }) {
|
|
function getConfigValue(path, defaultValue) {
|
|
return path ? dlv(tailwindConfig, path, defaultValue) : tailwindConfig
|
|
}
|
|
|
|
function applyConfiguredPrefix(selector) {
|
|
return prefixSelector(tailwindConfig.prefix, selector)
|
|
}
|
|
|
|
function prefixIdentifier(identifier, options) {
|
|
if (identifier === '*') {
|
|
return '*'
|
|
}
|
|
|
|
if (!options.respectPrefix) {
|
|
return identifier
|
|
}
|
|
|
|
if (typeof context.tailwindConfig.prefix === 'function') {
|
|
return prefixSelector(context.tailwindConfig.prefix, `.${identifier}`).substr(1)
|
|
}
|
|
|
|
return context.tailwindConfig.prefix + identifier
|
|
}
|
|
|
|
return {
|
|
addVariant(variantName, variantFunctions, options = {}) {
|
|
variantFunctions = [].concat(variantFunctions)
|
|
|
|
insertInto(variantList, variantName, options)
|
|
variantMap.set(variantName, variantFunctions)
|
|
},
|
|
postcss,
|
|
prefix: applyConfiguredPrefix,
|
|
e: escapeClassName,
|
|
config: getConfigValue,
|
|
theme(path, defaultValue) {
|
|
const [pathRoot, ...subPaths] = toPath(path)
|
|
const value = getConfigValue(['theme', pathRoot, ...subPaths], defaultValue)
|
|
return transformThemeValue(pathRoot)(value)
|
|
},
|
|
corePlugins: (path) => {
|
|
if (Array.isArray(tailwindConfig.corePlugins)) {
|
|
return tailwindConfig.corePlugins.includes(path)
|
|
}
|
|
|
|
return getConfigValue(['corePlugins', path], true)
|
|
},
|
|
variants: () => {
|
|
// Preserved for backwards compatibility but not used in v3.0+
|
|
return []
|
|
},
|
|
addUserCss(userCss) {
|
|
for (let [identifier, rule] of withIdentifiers(userCss)) {
|
|
let offset = offsets.user++
|
|
|
|
if (!context.candidateRuleMap.has(identifier)) {
|
|
context.candidateRuleMap.set(identifier, [])
|
|
}
|
|
|
|
context.candidateRuleMap.get(identifier).push([{ sort: offset, layer: 'user' }, rule])
|
|
}
|
|
},
|
|
addBase(base) {
|
|
for (let [identifier, rule] of withIdentifiers(base)) {
|
|
let prefixedIdentifier = prefixIdentifier(identifier, {})
|
|
let offset = offsets.base++
|
|
|
|
if (!context.candidateRuleMap.has(prefixedIdentifier)) {
|
|
context.candidateRuleMap.set(prefixedIdentifier, [])
|
|
}
|
|
|
|
context.candidateRuleMap
|
|
.get(prefixedIdentifier)
|
|
.push([{ sort: offset, layer: 'base' }, rule])
|
|
}
|
|
},
|
|
addComponents(components, options) {
|
|
let defaultOptions = {
|
|
variants: [],
|
|
respectPrefix: true,
|
|
respectImportant: false,
|
|
respectVariants: true,
|
|
}
|
|
|
|
options = Object.assign(
|
|
{},
|
|
defaultOptions,
|
|
Array.isArray(options) ? { variants: options } : options
|
|
)
|
|
|
|
for (let [identifier, rule] of withIdentifiers(components)) {
|
|
let prefixedIdentifier = prefixIdentifier(identifier, options)
|
|
let offset = offsets.components++
|
|
|
|
classList.add(prefixedIdentifier)
|
|
|
|
if (!context.candidateRuleMap.has(prefixedIdentifier)) {
|
|
context.candidateRuleMap.set(prefixedIdentifier, [])
|
|
}
|
|
|
|
context.candidateRuleMap
|
|
.get(prefixedIdentifier)
|
|
.push([{ sort: offset, layer: 'components', options }, rule])
|
|
}
|
|
},
|
|
addUtilities(utilities, options) {
|
|
let defaultOptions = {
|
|
variants: [],
|
|
respectPrefix: true,
|
|
respectImportant: true,
|
|
respectVariants: true,
|
|
}
|
|
|
|
options = Object.assign(
|
|
{},
|
|
defaultOptions,
|
|
Array.isArray(options) ? { variants: options } : options
|
|
)
|
|
|
|
for (let [identifier, rule] of withIdentifiers(utilities)) {
|
|
let prefixedIdentifier = prefixIdentifier(identifier, options)
|
|
let offset = offsets.utilities++
|
|
|
|
classList.add(prefixedIdentifier)
|
|
|
|
if (!context.candidateRuleMap.has(prefixedIdentifier)) {
|
|
context.candidateRuleMap.set(prefixedIdentifier, [])
|
|
}
|
|
|
|
context.candidateRuleMap
|
|
.get(prefixedIdentifier)
|
|
.push([{ sort: offset, layer: 'utilities', options }, rule])
|
|
}
|
|
},
|
|
matchUtilities: function (utilities, options) {
|
|
let defaultOptions = {
|
|
variants: [],
|
|
respectPrefix: true,
|
|
respectImportant: true,
|
|
respectVariants: true,
|
|
}
|
|
|
|
options = { ...defaultOptions, ...options }
|
|
|
|
let offset = offsets.utilities++
|
|
|
|
for (let identifier in utilities) {
|
|
let prefixedIdentifier = prefixIdentifier(identifier, options)
|
|
let rule = utilities[identifier]
|
|
|
|
classList.add([prefixedIdentifier, options])
|
|
|
|
function wrapped(modifier, { isOnlyPlugin }) {
|
|
let { type = 'any' } = options
|
|
type = [].concat(type)
|
|
let [value, coercedType] = coerceValue(type, modifier, options.values, tailwindConfig)
|
|
|
|
if (value === undefined) {
|
|
return []
|
|
}
|
|
|
|
if (!type.includes(coercedType)) {
|
|
if (isOnlyPlugin) {
|
|
log.warn([
|
|
`Unnecessary typehint \`${coercedType}\` in \`${identifier}-${modifier}\`.`,
|
|
`You can safely update it to \`${identifier}-${modifier.replace(
|
|
coercedType + ':',
|
|
''
|
|
)}\`.`,
|
|
])
|
|
} else {
|
|
return []
|
|
}
|
|
}
|
|
|
|
if (!isValidArbitraryValue(value)) {
|
|
return []
|
|
}
|
|
|
|
let includedRules = []
|
|
let ruleSets = []
|
|
.concat(
|
|
rule(value, {
|
|
includeRules(rules) {
|
|
includedRules.push(...rules)
|
|
},
|
|
})
|
|
)
|
|
.filter(Boolean)
|
|
.map((declaration) => ({
|
|
[nameClass(identifier, modifier)]: declaration,
|
|
}))
|
|
|
|
return [...includedRules, ...ruleSets]
|
|
}
|
|
|
|
let withOffsets = [{ sort: offset, layer: 'utilities', options }, wrapped]
|
|
|
|
if (!context.candidateRuleMap.has(prefixedIdentifier)) {
|
|
context.candidateRuleMap.set(prefixedIdentifier, [])
|
|
}
|
|
|
|
context.candidateRuleMap.get(prefixedIdentifier).push(withOffsets)
|
|
}
|
|
},
|
|
}
|
|
}
|
|
|
|
let fileModifiedMapCache = new WeakMap()
|
|
export function getFileModifiedMap(context) {
|
|
if (!fileModifiedMapCache.has(context)) {
|
|
fileModifiedMapCache.set(context, new Map())
|
|
}
|
|
return fileModifiedMapCache.get(context)
|
|
}
|
|
|
|
function trackModified(files, fileModifiedMap) {
|
|
let changed = false
|
|
|
|
for (let file of files) {
|
|
if (!file) continue
|
|
|
|
let parsed = url.parse(file)
|
|
let pathname = parsed.hash ? parsed.href.replace(parsed.hash, '') : parsed.href
|
|
pathname = parsed.search ? pathname.replace(parsed.search, '') : pathname
|
|
let newModified = fs.statSync(decodeURIComponent(pathname)).mtimeMs
|
|
|
|
if (!fileModifiedMap.has(file) || newModified > fileModifiedMap.get(file)) {
|
|
changed = true
|
|
}
|
|
|
|
fileModifiedMap.set(file, newModified)
|
|
}
|
|
|
|
return changed
|
|
}
|
|
|
|
function extractVariantAtRules(node) {
|
|
node.walkAtRules((atRule) => {
|
|
if (['responsive', 'variants'].includes(atRule.name)) {
|
|
extractVariantAtRules(atRule)
|
|
atRule.before(atRule.nodes)
|
|
atRule.remove()
|
|
}
|
|
})
|
|
}
|
|
|
|
function collectLayerPlugins(root) {
|
|
let layerPlugins = []
|
|
|
|
root.each((node) => {
|
|
if (node.type === 'atrule' && ['responsive', 'variants'].includes(node.name)) {
|
|
node.name = 'layer'
|
|
node.params = 'utilities'
|
|
}
|
|
})
|
|
|
|
// Walk @layer rules and treat them like plugins
|
|
root.walkAtRules('layer', (layerRule) => {
|
|
extractVariantAtRules(layerRule)
|
|
|
|
if (layerRule.params === 'base') {
|
|
for (let node of layerRule.nodes) {
|
|
layerPlugins.push(function ({ addBase }) {
|
|
addBase(node, { respectPrefix: false })
|
|
})
|
|
}
|
|
layerRule.remove()
|
|
} else if (layerRule.params === 'components') {
|
|
for (let node of layerRule.nodes) {
|
|
layerPlugins.push(function ({ addComponents }) {
|
|
addComponents(node, { respectPrefix: false })
|
|
})
|
|
}
|
|
layerRule.remove()
|
|
} else if (layerRule.params === 'utilities') {
|
|
for (let node of layerRule.nodes) {
|
|
layerPlugins.push(function ({ addUtilities }) {
|
|
addUtilities(node, { respectPrefix: false })
|
|
})
|
|
}
|
|
layerRule.remove()
|
|
}
|
|
})
|
|
|
|
root.walkRules((rule) => {
|
|
// At this point it is safe to include all the left-over css from the
|
|
// user's css file. This is because the `@tailwind` and `@layer` directives
|
|
// will already be handled and will be removed from the css tree.
|
|
layerPlugins.push(function ({ addUserCss }) {
|
|
addUserCss(rule, { respectPrefix: false })
|
|
})
|
|
})
|
|
|
|
return layerPlugins
|
|
}
|
|
|
|
function resolvePlugins(context, root) {
|
|
let corePluginList = Object.entries(corePlugins)
|
|
.map(([name, plugin]) => {
|
|
if (!context.tailwindConfig.corePlugins.includes(name)) {
|
|
return null
|
|
}
|
|
|
|
return plugin
|
|
})
|
|
.filter(Boolean)
|
|
|
|
let userPlugins = context.tailwindConfig.plugins.map((plugin) => {
|
|
if (plugin.__isOptionsFunction) {
|
|
plugin = plugin()
|
|
}
|
|
|
|
return typeof plugin === 'function' ? plugin : plugin.handler
|
|
})
|
|
|
|
let layerPlugins = collectLayerPlugins(root)
|
|
|
|
// TODO: This is a workaround for backwards compatibility, since custom variants
|
|
// were historically sorted before screen/stackable variants.
|
|
let beforeVariants = [corePlugins['pseudoElementVariants'], corePlugins['pseudoClassVariants']]
|
|
let afterVariants = [
|
|
corePlugins['directionVariants'],
|
|
corePlugins['reducedMotionVariants'],
|
|
corePlugins['darkVariants'],
|
|
corePlugins['screenVariants'],
|
|
]
|
|
|
|
return [...corePluginList, ...beforeVariants, ...userPlugins, ...afterVariants, ...layerPlugins]
|
|
}
|
|
|
|
function registerPlugins(plugins, context) {
|
|
let variantList = []
|
|
let variantMap = new Map()
|
|
let offsets = {
|
|
base: 0n,
|
|
components: 0n,
|
|
utilities: 0n,
|
|
user: 0n,
|
|
}
|
|
|
|
let classList = new Set()
|
|
|
|
let pluginApi = buildPluginApi(context.tailwindConfig, context, {
|
|
variantList,
|
|
variantMap,
|
|
offsets,
|
|
classList,
|
|
})
|
|
|
|
for (let plugin of plugins) {
|
|
if (Array.isArray(plugin)) {
|
|
for (let pluginItem of plugin) {
|
|
pluginItem(pluginApi)
|
|
}
|
|
} else {
|
|
plugin?.(pluginApi)
|
|
}
|
|
}
|
|
|
|
let highestOffset = ((args) => args.reduce((m, e) => (e > m ? e : m)))([
|
|
offsets.base,
|
|
offsets.components,
|
|
offsets.utilities,
|
|
offsets.user,
|
|
])
|
|
let reservedBits = BigInt(highestOffset.toString(2).length)
|
|
|
|
context.layerOrder = {
|
|
base: (1n << reservedBits) << 0n,
|
|
components: (1n << reservedBits) << 1n,
|
|
utilities: (1n << reservedBits) << 2n,
|
|
user: (1n << reservedBits) << 3n,
|
|
}
|
|
|
|
reservedBits += 4n
|
|
|
|
let offset = 0
|
|
context.variantOrder = new Map(
|
|
variantList
|
|
.map((variant, i) => {
|
|
let variantFunctions = variantMap.get(variant).length
|
|
let bits = (1n << BigInt(i + offset)) << reservedBits
|
|
offset += variantFunctions - 1
|
|
return [variant, bits]
|
|
})
|
|
.sort(([, a], [, z]) => bigSign(a - z))
|
|
)
|
|
|
|
context.minimumScreen = [...context.variantOrder.values()].shift()
|
|
|
|
// Build variantMap
|
|
for (let [variantName, variantFunctions] of variantMap.entries()) {
|
|
let sort = context.variantOrder.get(variantName)
|
|
context.variantMap.set(
|
|
variantName,
|
|
variantFunctions.map((variantFunction, idx) => [sort << BigInt(idx), variantFunction])
|
|
)
|
|
}
|
|
|
|
context.safelist = function () {
|
|
let safelist = (context.tailwindConfig.safelist ?? []).filter(Boolean)
|
|
if (safelist.length <= 0) return []
|
|
|
|
let output = []
|
|
let checks = []
|
|
|
|
for (let value of safelist) {
|
|
if (typeof value === 'string') {
|
|
output.push(value)
|
|
continue
|
|
}
|
|
|
|
if (value instanceof RegExp) {
|
|
log.warn('root-regex', [
|
|
// TODO: Improve this warning message
|
|
'RegExp in the safelist option is not supported.',
|
|
'Please use the object syntax instead: https://tailwindcss.com/docs/...',
|
|
])
|
|
continue
|
|
}
|
|
|
|
checks.push(value)
|
|
}
|
|
|
|
if (checks.length <= 0) return output.map((value) => ({ raw: value, extension: 'html' }))
|
|
|
|
let patternMatchingCount = new Map()
|
|
|
|
for (let util of classList) {
|
|
let utils = Array.isArray(util)
|
|
? (() => {
|
|
let [utilName, options] = util
|
|
return Object.keys(options?.values ?? {}).map((value) => formatClass(utilName, value))
|
|
})()
|
|
: [util]
|
|
|
|
for (let util of utils) {
|
|
for (let { pattern, variants = [] } of checks) {
|
|
// RegExp with the /g flag are stateful, so let's reset the last
|
|
// index pointer to reset the state.
|
|
pattern.lastIndex = 0
|
|
|
|
if (!patternMatchingCount.has(pattern)) {
|
|
patternMatchingCount.set(pattern, 0)
|
|
}
|
|
|
|
if (!pattern.test(util)) continue
|
|
|
|
patternMatchingCount.set(pattern, patternMatchingCount.get(pattern) + 1)
|
|
|
|
output.push(util)
|
|
for (let variant of variants) {
|
|
output.push(variant + context.tailwindConfig.separator + util)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
for (let [regex, count] of patternMatchingCount.entries()) {
|
|
if (count !== 0) continue
|
|
|
|
log.warn([
|
|
// TODO: Improve this warning message
|
|
`You have a regex pattern in your "safelist" config (${regex}) that doesn't match any utilities.`,
|
|
'For more info, visit https://tailwindcss.com/docs/...',
|
|
])
|
|
}
|
|
|
|
return output.map((value) => ({ raw: value, extension: 'html' }))
|
|
}
|
|
|
|
// Generate a list of strings for autocompletion purposes, e.g.
|
|
// ['uppercase', 'lowercase', ...]
|
|
context.getClassList = function () {
|
|
let output = []
|
|
|
|
for (let util of classList) {
|
|
if (Array.isArray(util)) {
|
|
let [utilName, options] = util
|
|
|
|
for (let value of Object.keys(options?.values ?? {})) {
|
|
output.push(formatClass(utilName, value))
|
|
}
|
|
} else {
|
|
output.push(util)
|
|
}
|
|
}
|
|
|
|
return output
|
|
}
|
|
}
|
|
|
|
export function createContext(tailwindConfig, changedContent = [], root = postcss.root()) {
|
|
let context = {
|
|
disposables: [],
|
|
ruleCache: new Set(),
|
|
classCache: new Map(),
|
|
applyClassCache: new Map(),
|
|
notClassCache: new Set(),
|
|
postCssNodeCache: new Map(),
|
|
candidateRuleMap: new Map(),
|
|
tailwindConfig,
|
|
changedContent: changedContent,
|
|
variantMap: new Map(),
|
|
stylesheetCache: null,
|
|
}
|
|
|
|
let resolvedPlugins = resolvePlugins(context, root)
|
|
registerPlugins(resolvedPlugins, context)
|
|
|
|
return context
|
|
}
|
|
|
|
let contextMap = sharedState.contextMap
|
|
let configContextMap = sharedState.configContextMap
|
|
let contextSourcesMap = sharedState.contextSourcesMap
|
|
|
|
export function getContext(
|
|
root,
|
|
result,
|
|
tailwindConfig,
|
|
userConfigPath,
|
|
tailwindConfigHash,
|
|
contextDependencies
|
|
) {
|
|
let sourcePath = result.opts.from
|
|
let isConfigFile = userConfigPath !== null
|
|
|
|
env.DEBUG && console.log('Source path:', sourcePath)
|
|
|
|
let existingContext
|
|
|
|
if (isConfigFile && contextMap.has(sourcePath)) {
|
|
existingContext = contextMap.get(sourcePath)
|
|
} else if (configContextMap.has(tailwindConfigHash)) {
|
|
let context = configContextMap.get(tailwindConfigHash)
|
|
contextSourcesMap.get(context).add(sourcePath)
|
|
contextMap.set(sourcePath, context)
|
|
|
|
existingContext = context
|
|
}
|
|
|
|
// If there's already a context in the cache and we don't need to
|
|
// reset the context, return the cached context.
|
|
if (existingContext) {
|
|
let contextDependenciesChanged = trackModified(
|
|
[...contextDependencies],
|
|
getFileModifiedMap(existingContext)
|
|
)
|
|
if (!contextDependenciesChanged) {
|
|
return [existingContext, false]
|
|
}
|
|
}
|
|
|
|
// If this source is in the context map, get the old context.
|
|
// Remove this source from the context sources for the old context,
|
|
// and clean up that context if no one else is using it. This can be
|
|
// called by many processes in rapid succession, so we check for presence
|
|
// first because the first process to run this code will wipe it out first.
|
|
if (contextMap.has(sourcePath)) {
|
|
let oldContext = contextMap.get(sourcePath)
|
|
if (contextSourcesMap.has(oldContext)) {
|
|
contextSourcesMap.get(oldContext).delete(sourcePath)
|
|
if (contextSourcesMap.get(oldContext).size === 0) {
|
|
contextSourcesMap.delete(oldContext)
|
|
for (let [tailwindConfigHash, context] of configContextMap) {
|
|
if (context === oldContext) {
|
|
configContextMap.delete(tailwindConfigHash)
|
|
}
|
|
}
|
|
for (let disposable of oldContext.disposables.splice(0)) {
|
|
disposable(oldContext)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
env.DEBUG && console.log('Setting up new context...')
|
|
|
|
let context = createContext(tailwindConfig, [], root)
|
|
|
|
trackModified([...contextDependencies], getFileModifiedMap(context))
|
|
|
|
// ---
|
|
|
|
// Update all context tracking state
|
|
|
|
configContextMap.set(tailwindConfigHash, context)
|
|
contextMap.set(sourcePath, context)
|
|
|
|
if (!contextSourcesMap.has(context)) {
|
|
contextSourcesMap.set(context, new Set())
|
|
}
|
|
|
|
contextSourcesMap.get(context).add(sourcePath)
|
|
|
|
return [context, true]
|
|
}
|