import postcss, { Root } from 'postcss' import selectorParser from 'postcss-selector-parser' export default postcss.plugin('add-id', (options: any) => (root: Root) => { const id: string = options const keyframes = Object.create(null) root.each(function rewriteSelectors(node: any) { if (!node.selector) { // handle media queries if (node.type === 'atrule') { if (node.name === 'media' || node.name === 'supports') { node.each(rewriteSelectors) } else if (/-?keyframes$/.test(node.name)) { // register keyframes keyframes[node.params] = node.params = node.params + '-' + id } } return } node.selector = selectorParser((selectors: any) => { selectors.each(function rewriteSelector( selector: any, _i: number, slotted?: boolean ) { let node: any = null // find the last child node to insert attribute selector selector.each((n: any) => { if (n.type === 'pseudo') { // deep: inject [id] attribute at the node before the ::v-deep // combinator. // .foo ::v-deep .bar -> .foo[xxxxxxx] .bar if (n.value === '::v-deep') { n.value = n.spaces.before = n.spaces.after = '' return false } // slot: use selector inside `::v-slotted` and inject [id + '-s'] // instead. // ::v-slotted(.foo) -> .foo[xxxxxxx-s] if (n.value === '::v-slotted') { rewriteSelector(n.nodes[0], 0, true /* slotted */) selectors.insertAfter(selector, n.nodes[0]) selectors.removeChild(selector) return false } // global: replace with inner selector and do not inject [id]. // ::v-global(.foo) -> .foo if (n.value === '::v-global') { selectors.insertAfter(selector, n.nodes[0]) selectors.removeChild(selector) return false } } if (n.type !== 'pseudo' && n.type !== 'combinator') { node = n } }) if (node) { node.spaces.after = '' } else { // For deep selectors & standalone pseudo selectors, // the attribute selectors are prepended rather than appended. // So all leading spaces must be eliminated to avoid problems. selector.first.spaces.before = '' } const idToAdd = slotted ? id + '-s' : id selector.insertAfter( node, selectorParser.attribute({ attribute: idToAdd, value: idToAdd, raws: {}, quoteMark: `"` }) ) }) }).processSync(node.selector) }) // If keyframes are found in this