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' && n.value === '::v-deep') { n.value = n.spaces.before = n.spaces.after = '' return false } if (n.type === 'pseudo' && n.value === '::v-slotted') { rewriteSelector(n.nodes[0], 0, true) 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