|  |  |  | @ -1,6 +1,6 @@ | 
			
		
	
		
			
				
					|  |  |  |  | import { get_possible_values } from './gather_possible_values.js'; | 
			
		
	
		
			
				
					|  |  |  |  | import { regex_starts_with_whitespace, regex_ends_with_whitespace } from '../../patterns.js'; | 
			
		
	
		
			
				
					|  |  |  |  | import { error } from '../../../errors.js'; | 
			
		
	
		
			
				
					|  |  |  |  | import { get_possible_values } from './utils.js'; | 
			
		
	
		
			
				
					|  |  |  |  | import { regex_starts_with_whitespace, regex_ends_with_whitespace } from '../phases/patterns.js'; | 
			
		
	
		
			
				
					|  |  |  |  | import { error } from '../errors.js'; | 
			
		
	
		
			
				
					|  |  |  |  | import { Stylesheet } from './Stylesheet.js'; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | const NO_MATCH = 'NO_MATCH'; | 
			
		
	
	
		
			
				
					|  |  |  | @ -19,41 +19,48 @@ const whitelist_attribute_selector = new Map([ | 
			
		
	
		
			
				
					|  |  |  |  | 	['dialog', new Set(['open'])] | 
			
		
	
		
			
				
					|  |  |  |  | ]); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | export default class Selector { | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @type {import('#compiler').Css.Selector} */ | 
			
		
	
		
			
				
					|  |  |  |  | export class ComplexSelector { | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @type {import('#compiler').Css.ComplexSelector} */ | 
			
		
	
		
			
				
					|  |  |  |  | 	node; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @type {import('./Stylesheet.js').Stylesheet} */ | 
			
		
	
		
			
				
					|  |  |  |  | 	stylesheet; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @type {Block[]} */ | 
			
		
	
		
			
				
					|  |  |  |  | 	blocks; | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @type {RelativeSelector[]} */ | 
			
		
	
		
			
				
					|  |  |  |  | 	relative_selectors; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @type {Block[]} */ | 
			
		
	
		
			
				
					|  |  |  |  | 	local_blocks; | 
			
		
	
		
			
				
					|  |  |  |  | 	/** | 
			
		
	
		
			
				
					|  |  |  |  | 	 * The `relative_selectors`, minus any trailing global selectors | 
			
		
	
		
			
				
					|  |  |  |  | 	 * (which includes `:root` and `:host`) since we ignore these | 
			
		
	
		
			
				
					|  |  |  |  | 	 * when determining if a selector is used. | 
			
		
	
		
			
				
					|  |  |  |  | 	 * @type {RelativeSelector[]} | 
			
		
	
		
			
				
					|  |  |  |  | 	 */ | 
			
		
	
		
			
				
					|  |  |  |  | 	local_relative_selectors; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	used = false; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	/** | 
			
		
	
		
			
				
					|  |  |  |  | 	 * @param {import('#compiler').Css.Selector} node | 
			
		
	
		
			
				
					|  |  |  |  | 	 * @param {import('#compiler').Css.ComplexSelector} node | 
			
		
	
		
			
				
					|  |  |  |  | 	 * @param {import('./Stylesheet.js').Stylesheet} stylesheet | 
			
		
	
		
			
				
					|  |  |  |  | 	 */ | 
			
		
	
		
			
				
					|  |  |  |  | 	constructor(node, stylesheet) { | 
			
		
	
		
			
				
					|  |  |  |  | 		this.node = node; | 
			
		
	
		
			
				
					|  |  |  |  | 		this.stylesheet = stylesheet; | 
			
		
	
		
			
				
					|  |  |  |  | 		this.blocks = group_selectors(node); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		this.relative_selectors = group_selectors(node); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		// take trailing :global(...) selectors out of consideration
 | 
			
		
	
		
			
				
					|  |  |  |  | 		const i = this.blocks.findLastIndex((block) => !block.can_ignore()); | 
			
		
	
		
			
				
					|  |  |  |  | 		this.local_blocks = this.blocks.slice(0, i + 1); | 
			
		
	
		
			
				
					|  |  |  |  | 		const i = this.relative_selectors.findLastIndex((s) => !s.can_ignore()); | 
			
		
	
		
			
				
					|  |  |  |  | 		this.local_relative_selectors = this.relative_selectors.slice(0, i + 1); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		// if we have a `:root {...}` or `:global(...) {...}` selector, we need to mark
 | 
			
		
	
		
			
				
					|  |  |  |  | 		// this selector as `used` even if the component doesn't contain any nodes
 | 
			
		
	
		
			
				
					|  |  |  |  | 		this.used = this.local_blocks.length === 0; | 
			
		
	
		
			
				
					|  |  |  |  | 		this.used = this.local_relative_selectors.length === 0; | 
			
		
	
		
			
				
					|  |  |  |  | 	} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @param {import('#compiler').RegularElement | import('#compiler').SvelteElement} node */ | 
			
		
	
		
			
				
					|  |  |  |  | 	apply(node) { | 
			
		
	
		
			
				
					|  |  |  |  | 		if (apply_selector(this.local_blocks.slice(), node, this.stylesheet)) { | 
			
		
	
		
			
				
					|  |  |  |  | 		if (apply_selector(this.local_relative_selectors.slice(), node, this.stylesheet)) { | 
			
		
	
		
			
				
					|  |  |  |  | 			this.used = true; | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 	} | 
			
		
	
	
		
			
				
					|  |  |  | @ -71,19 +78,19 @@ export default class Selector { | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		/** | 
			
		
	
		
			
				
					|  |  |  |  | 		 * @param {Block} block | 
			
		
	
		
			
				
					|  |  |  |  | 		 * @param {RelativeSelector} relative_selector | 
			
		
	
		
			
				
					|  |  |  |  | 		 * @param {string} modifier | 
			
		
	
		
			
				
					|  |  |  |  | 		 */ | 
			
		
	
		
			
				
					|  |  |  |  | 		function encapsulate_block(block, modifier) { | 
			
		
	
		
			
				
					|  |  |  |  | 			for (const selector of block.selectors) { | 
			
		
	
		
			
				
					|  |  |  |  | 		function encapsulate_block(relative_selector, modifier) { | 
			
		
	
		
			
				
					|  |  |  |  | 			for (const selector of relative_selector.selectors) { | 
			
		
	
		
			
				
					|  |  |  |  | 				if (selector.type === 'PseudoClassSelector' && selector.name === 'global') { | 
			
		
	
		
			
				
					|  |  |  |  | 					remove_global_pseudo_class(selector); | 
			
		
	
		
			
				
					|  |  |  |  | 				} | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 			let i = block.selectors.length; | 
			
		
	
		
			
				
					|  |  |  |  | 			let i = relative_selector.selectors.length; | 
			
		
	
		
			
				
					|  |  |  |  | 			while (i--) { | 
			
		
	
		
			
				
					|  |  |  |  | 				const selector = block.selectors[i]; | 
			
		
	
		
			
				
					|  |  |  |  | 				const selector = relative_selector.selectors[i]; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 				if (selector.type === 'PseudoElementSelector' || selector.type === 'PseudoClassSelector') { | 
			
		
	
		
			
				
					|  |  |  |  | 					if (selector.name !== 'root' && selector.name !== 'host') { | 
			
		
	
	
		
			
				
					|  |  |  | @ -103,22 +110,22 @@ export default class Selector { | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		let first = true; | 
			
		
	
		
			
				
					|  |  |  |  | 		for (const block of this.blocks) { | 
			
		
	
		
			
				
					|  |  |  |  | 			if (block.global) { | 
			
		
	
		
			
				
					|  |  |  |  | 				remove_global_pseudo_class(block.selectors[0]); | 
			
		
	
		
			
				
					|  |  |  |  | 		for (const relative_selector of this.relative_selectors) { | 
			
		
	
		
			
				
					|  |  |  |  | 			if (relative_selector.is_global) { | 
			
		
	
		
			
				
					|  |  |  |  | 				remove_global_pseudo_class(relative_selector.selectors[0]); | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 			if (block.should_encapsulate) { | 
			
		
	
		
			
				
					|  |  |  |  | 			if (relative_selector.should_encapsulate) { | 
			
		
	
		
			
				
					|  |  |  |  | 				// for the first occurrence, we use a classname selector, so that every
 | 
			
		
	
		
			
				
					|  |  |  |  | 				// encapsulated selector gets a +0-1-0 specificity bump. thereafter,
 | 
			
		
	
		
			
				
					|  |  |  |  | 				// we use a `:where` selector, which does not affect specificity
 | 
			
		
	
		
			
				
					|  |  |  |  | 				encapsulate_block(block, first ? modifier : `:where(${modifier})`); | 
			
		
	
		
			
				
					|  |  |  |  | 				encapsulate_block(relative_selector, first ? modifier : `:where(${modifier})`); | 
			
		
	
		
			
				
					|  |  |  |  | 				first = false; | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 	} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @param {import('../../types.js').ComponentAnalysis} analysis */ | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @param {import('../phases/types.js').ComponentAnalysis} analysis */ | 
			
		
	
		
			
				
					|  |  |  |  | 	validate(analysis) { | 
			
		
	
		
			
				
					|  |  |  |  | 		this.validate_global_placement(); | 
			
		
	
		
			
				
					|  |  |  |  | 		this.validate_global_with_multiple_selectors(); | 
			
		
	
	
		
			
				
					|  |  |  | @ -128,27 +135,27 @@ export default class Selector { | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	validate_global_placement() { | 
			
		
	
		
			
				
					|  |  |  |  | 		let start = 0; | 
			
		
	
		
			
				
					|  |  |  |  | 		let end = this.blocks.length; | 
			
		
	
		
			
				
					|  |  |  |  | 		let end = this.relative_selectors.length; | 
			
		
	
		
			
				
					|  |  |  |  | 		for (; start < end; start += 1) { | 
			
		
	
		
			
				
					|  |  |  |  | 			if (!this.blocks[start].global) break; | 
			
		
	
		
			
				
					|  |  |  |  | 			if (!this.relative_selectors[start].is_global) break; | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 		for (; end > start; end -= 1) { | 
			
		
	
		
			
				
					|  |  |  |  | 			if (!this.blocks[end - 1].global) break; | 
			
		
	
		
			
				
					|  |  |  |  | 			if (!this.relative_selectors[end - 1].is_global) break; | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 		for (let i = start; i < end; i += 1) { | 
			
		
	
		
			
				
					|  |  |  |  | 			if (this.blocks[i].global) { | 
			
		
	
		
			
				
					|  |  |  |  | 				error(this.blocks[i].selectors[0], 'invalid-css-global-placement'); | 
			
		
	
		
			
				
					|  |  |  |  | 			if (this.relative_selectors[i].is_global) { | 
			
		
	
		
			
				
					|  |  |  |  | 				error(this.relative_selectors[i].selectors[0], 'invalid-css-global-placement'); | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 	} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	validate_global_with_multiple_selectors() { | 
			
		
	
		
			
				
					|  |  |  |  | 		if (this.blocks.length === 1 && this.blocks[0].selectors.length === 1) { | 
			
		
	
		
			
				
					|  |  |  |  | 		if (this.relative_selectors.length === 1 && this.relative_selectors[0].selectors.length === 1) { | 
			
		
	
		
			
				
					|  |  |  |  | 			// standalone :global() with multiple selectors is OK
 | 
			
		
	
		
			
				
					|  |  |  |  | 			return; | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 		for (const block of this.blocks) { | 
			
		
	
		
			
				
					|  |  |  |  | 			for (const selector of block.selectors) { | 
			
		
	
		
			
				
					|  |  |  |  | 		for (const relative_selector of this.relative_selectors) { | 
			
		
	
		
			
				
					|  |  |  |  | 			for (const selector of relative_selector.selectors) { | 
			
		
	
		
			
				
					|  |  |  |  | 				if ( | 
			
		
	
		
			
				
					|  |  |  |  | 					selector.type === 'PseudoClassSelector' && | 
			
		
	
		
			
				
					|  |  |  |  | 					selector.name === 'global' && | 
			
		
	
	
		
			
				
					|  |  |  | @ -161,22 +168,22 @@ export default class Selector { | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 	} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @param {import('../../types.js').ComponentAnalysis} analysis */ | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @param {import('../phases/types.js').ComponentAnalysis} analysis */ | 
			
		
	
		
			
				
					|  |  |  |  | 	validate_invalid_combinator_without_selector(analysis) { | 
			
		
	
		
			
				
					|  |  |  |  | 		for (let i = 0; i < this.blocks.length; i++) { | 
			
		
	
		
			
				
					|  |  |  |  | 			const block = this.blocks[i]; | 
			
		
	
		
			
				
					|  |  |  |  | 			if (block.selectors.length === 0) { | 
			
		
	
		
			
				
					|  |  |  |  | 		for (let i = 0; i < this.relative_selectors.length; i++) { | 
			
		
	
		
			
				
					|  |  |  |  | 			const relative_selector = this.relative_selectors[i]; | 
			
		
	
		
			
				
					|  |  |  |  | 			if (relative_selector.selectors.length === 0) { | 
			
		
	
		
			
				
					|  |  |  |  | 				error(this.node, 'invalid-css-selector'); | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 	} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	validate_global_compound_selector() { | 
			
		
	
		
			
				
					|  |  |  |  | 		for (const block of this.blocks) { | 
			
		
	
		
			
				
					|  |  |  |  | 			if (block.selectors.length === 1) continue; | 
			
		
	
		
			
				
					|  |  |  |  | 		for (const relative_selector of this.relative_selectors) { | 
			
		
	
		
			
				
					|  |  |  |  | 			if (relative_selector.selectors.length === 1) continue; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 			for (let i = 0; i < block.selectors.length; i++) { | 
			
		
	
		
			
				
					|  |  |  |  | 				const selector = block.selectors[i]; | 
			
		
	
		
			
				
					|  |  |  |  | 			for (let i = 0; i < relative_selector.selectors.length; i++) { | 
			
		
	
		
			
				
					|  |  |  |  | 				const selector = relative_selector.selectors[i]; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 				if (selector.type === 'PseudoClassSelector' && selector.name === 'global') { | 
			
		
	
		
			
				
					|  |  |  |  | 					const child = selector.args?.children[0].children[0]; | 
			
		
	
	
		
			
				
					|  |  |  | @ -184,7 +191,7 @@ export default class Selector { | 
			
		
	
		
			
				
					|  |  |  |  | 						child?.type === 'TypeSelector' && | 
			
		
	
		
			
				
					|  |  |  |  | 						!/[.:#]/.test(child.name[0]) && | 
			
		
	
		
			
				
					|  |  |  |  | 						(i !== 0 || | 
			
		
	
		
			
				
					|  |  |  |  | 							block.selectors | 
			
		
	
		
			
				
					|  |  |  |  | 							relative_selector.selectors | 
			
		
	
		
			
				
					|  |  |  |  | 								.slice(1) | 
			
		
	
		
			
				
					|  |  |  |  | 								.some( | 
			
		
	
		
			
				
					|  |  |  |  | 									(s) => s.type !== 'PseudoElementSelector' && s.type !== 'PseudoClassSelector' | 
			
		
	
	
		
			
				
					|  |  |  | @ -199,20 +206,22 @@ export default class Selector { | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | /** | 
			
		
	
		
			
				
					|  |  |  |  |  * @param {Block[]} blocks | 
			
		
	
		
			
				
					|  |  |  |  |  * @param {RelativeSelector[]} relative_selectors | 
			
		
	
		
			
				
					|  |  |  |  |  * @param {import('#compiler').RegularElement | import('#compiler').SvelteElement | null} node | 
			
		
	
		
			
				
					|  |  |  |  |  * @param {Stylesheet} stylesheet | 
			
		
	
		
			
				
					|  |  |  |  |  * @returns {boolean} | 
			
		
	
		
			
				
					|  |  |  |  |  */ | 
			
		
	
		
			
				
					|  |  |  |  | function apply_selector(blocks, node, stylesheet) { | 
			
		
	
		
			
				
					|  |  |  |  | 	const block = blocks.pop(); | 
			
		
	
		
			
				
					|  |  |  |  | 	if (!block) return false; | 
			
		
	
		
			
				
					|  |  |  |  | function apply_selector(relative_selectors, node, stylesheet) { | 
			
		
	
		
			
				
					|  |  |  |  | 	const relative_selector = relative_selectors.pop(); | 
			
		
	
		
			
				
					|  |  |  |  | 	if (!relative_selector) return false; | 
			
		
	
		
			
				
					|  |  |  |  | 	if (!node) { | 
			
		
	
		
			
				
					|  |  |  |  | 		return ( | 
			
		
	
		
			
				
					|  |  |  |  | 			(block.global && blocks.every((block) => block.global)) || (block.host && blocks.length === 0) | 
			
		
	
		
			
				
					|  |  |  |  | 			(relative_selector.is_global && | 
			
		
	
		
			
				
					|  |  |  |  | 				relative_selectors.every((relative_selector) => relative_selector.is_global)) || | 
			
		
	
		
			
				
					|  |  |  |  | 			(relative_selector.is_host && relative_selectors.length === 0) | 
			
		
	
		
			
				
					|  |  |  |  | 		); | 
			
		
	
		
			
				
					|  |  |  |  | 	} | 
			
		
	
		
			
				
					|  |  |  |  | 	const applies = block_might_apply_to_node(block, node); | 
			
		
	
		
			
				
					|  |  |  |  | 	const applies = block_might_apply_to_node(relative_selector, node); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	if (applies === NO_MATCH) { | 
			
		
	
		
			
				
					|  |  |  |  | 		return false; | 
			
		
	
	
		
			
				
					|  |  |  | @ -221,27 +230,30 @@ function apply_selector(blocks, node, stylesheet) { | 
			
		
	
		
			
				
					|  |  |  |  | 	/** | 
			
		
	
		
			
				
					|  |  |  |  | 	 * Mark both the compound selector and the node it selects as encapsulated, | 
			
		
	
		
			
				
					|  |  |  |  | 	 * for transformation in a later step | 
			
		
	
		
			
				
					|  |  |  |  | 	 * @param {Block} block | 
			
		
	
		
			
				
					|  |  |  |  | 	 * @param {RelativeSelector} relative_selector | 
			
		
	
		
			
				
					|  |  |  |  | 	 * @param {import('#compiler').RegularElement | import('#compiler').SvelteElement} node | 
			
		
	
		
			
				
					|  |  |  |  | 	 */ | 
			
		
	
		
			
				
					|  |  |  |  | 	function mark(block, node) { | 
			
		
	
		
			
				
					|  |  |  |  | 		block.should_encapsulate = true; | 
			
		
	
		
			
				
					|  |  |  |  | 	function mark(relative_selector, node) { | 
			
		
	
		
			
				
					|  |  |  |  | 		relative_selector.should_encapsulate = true; | 
			
		
	
		
			
				
					|  |  |  |  | 		stylesheet.nodes_with_css_class.add(node); | 
			
		
	
		
			
				
					|  |  |  |  | 		return true; | 
			
		
	
		
			
				
					|  |  |  |  | 	} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	if (applies === UNKNOWN_SELECTOR) { | 
			
		
	
		
			
				
					|  |  |  |  | 		return mark(block, node); | 
			
		
	
		
			
				
					|  |  |  |  | 		return mark(relative_selector, node); | 
			
		
	
		
			
				
					|  |  |  |  | 	} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	if (block.combinator) { | 
			
		
	
		
			
				
					|  |  |  |  | 		if (block.combinator.type === 'Combinator' && block.combinator.name === ' ') { | 
			
		
	
		
			
				
					|  |  |  |  | 			for (const ancestor_block of blocks) { | 
			
		
	
		
			
				
					|  |  |  |  | 				if (ancestor_block.global) { | 
			
		
	
		
			
				
					|  |  |  |  | 	if (relative_selector.combinator) { | 
			
		
	
		
			
				
					|  |  |  |  | 		if ( | 
			
		
	
		
			
				
					|  |  |  |  | 			relative_selector.combinator.type === 'Combinator' && | 
			
		
	
		
			
				
					|  |  |  |  | 			relative_selector.combinator.name === ' ' | 
			
		
	
		
			
				
					|  |  |  |  | 		) { | 
			
		
	
		
			
				
					|  |  |  |  | 			for (const ancestor_block of relative_selectors) { | 
			
		
	
		
			
				
					|  |  |  |  | 				if (ancestor_block.is_global) { | 
			
		
	
		
			
				
					|  |  |  |  | 					continue; | 
			
		
	
		
			
				
					|  |  |  |  | 				} | 
			
		
	
		
			
				
					|  |  |  |  | 				if (ancestor_block.host) { | 
			
		
	
		
			
				
					|  |  |  |  | 					return mark(block, node); | 
			
		
	
		
			
				
					|  |  |  |  | 				if (ancestor_block.is_host) { | 
			
		
	
		
			
				
					|  |  |  |  | 					return mark(relative_selector, node); | 
			
		
	
		
			
				
					|  |  |  |  | 				} | 
			
		
	
		
			
				
					|  |  |  |  | 				/** @type {import('#compiler').RegularElement | import('#compiler').SvelteElement | null} */ | 
			
		
	
		
			
				
					|  |  |  |  | 				let parent = node; | 
			
		
	
	
		
			
				
					|  |  |  | @ -253,35 +265,48 @@ function apply_selector(blocks, node, stylesheet) { | 
			
		
	
		
			
				
					|  |  |  |  | 					} | 
			
		
	
		
			
				
					|  |  |  |  | 				} | 
			
		
	
		
			
				
					|  |  |  |  | 				if (matched) { | 
			
		
	
		
			
				
					|  |  |  |  | 					return mark(block, node); | 
			
		
	
		
			
				
					|  |  |  |  | 					return mark(relative_selector, node); | 
			
		
	
		
			
				
					|  |  |  |  | 				} | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 			if (blocks.every((block) => block.global)) { | 
			
		
	
		
			
				
					|  |  |  |  | 				return mark(block, node); | 
			
		
	
		
			
				
					|  |  |  |  | 			if (relative_selectors.every((relative_selector) => relative_selector.is_global)) { | 
			
		
	
		
			
				
					|  |  |  |  | 				return mark(relative_selector, node); | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 			return false; | 
			
		
	
		
			
				
					|  |  |  |  | 		} else if (block.combinator.name === '>') { | 
			
		
	
		
			
				
					|  |  |  |  | 			const has_global_parent = blocks.every((block) => block.global); | 
			
		
	
		
			
				
					|  |  |  |  | 			if (has_global_parent || apply_selector(blocks, get_element_parent(node), stylesheet)) { | 
			
		
	
		
			
				
					|  |  |  |  | 				return mark(block, node); | 
			
		
	
		
			
				
					|  |  |  |  | 		} else if (relative_selector.combinator.name === '>') { | 
			
		
	
		
			
				
					|  |  |  |  | 			const has_global_parent = relative_selectors.every( | 
			
		
	
		
			
				
					|  |  |  |  | 				(relative_selector) => relative_selector.is_global | 
			
		
	
		
			
				
					|  |  |  |  | 			); | 
			
		
	
		
			
				
					|  |  |  |  | 			if ( | 
			
		
	
		
			
				
					|  |  |  |  | 				has_global_parent || | 
			
		
	
		
			
				
					|  |  |  |  | 				apply_selector(relative_selectors, get_element_parent(node), stylesheet) | 
			
		
	
		
			
				
					|  |  |  |  | 			) { | 
			
		
	
		
			
				
					|  |  |  |  | 				return mark(relative_selector, node); | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 			return false; | 
			
		
	
		
			
				
					|  |  |  |  | 		} else if (block.combinator.name === '+' || block.combinator.name === '~') { | 
			
		
	
		
			
				
					|  |  |  |  | 			const siblings = get_possible_element_siblings(node, block.combinator.name === '+'); | 
			
		
	
		
			
				
					|  |  |  |  | 		} else if ( | 
			
		
	
		
			
				
					|  |  |  |  | 			relative_selector.combinator.name === '+' || | 
			
		
	
		
			
				
					|  |  |  |  | 			relative_selector.combinator.name === '~' | 
			
		
	
		
			
				
					|  |  |  |  | 		) { | 
			
		
	
		
			
				
					|  |  |  |  | 			const siblings = get_possible_element_siblings( | 
			
		
	
		
			
				
					|  |  |  |  | 				node, | 
			
		
	
		
			
				
					|  |  |  |  | 				relative_selector.combinator.name === '+' | 
			
		
	
		
			
				
					|  |  |  |  | 			); | 
			
		
	
		
			
				
					|  |  |  |  | 			let has_match = false; | 
			
		
	
		
			
				
					|  |  |  |  | 			// NOTE: if we have :global(), we couldn't figure out what is selected within `:global` due to the
 | 
			
		
	
		
			
				
					|  |  |  |  | 			// css-tree limitation that does not parse the inner selector of :global
 | 
			
		
	
		
			
				
					|  |  |  |  | 			// so unless we are sure there will be no sibling to match, we will consider it as matched
 | 
			
		
	
		
			
				
					|  |  |  |  | 			const has_global = blocks.some((block) => block.global); | 
			
		
	
		
			
				
					|  |  |  |  | 			const has_global = relative_selectors.some( | 
			
		
	
		
			
				
					|  |  |  |  | 				(relative_selector) => relative_selector.is_global | 
			
		
	
		
			
				
					|  |  |  |  | 			); | 
			
		
	
		
			
				
					|  |  |  |  | 			if (has_global) { | 
			
		
	
		
			
				
					|  |  |  |  | 				if (siblings.size === 0 && get_element_parent(node) !== null) { | 
			
		
	
		
			
				
					|  |  |  |  | 					return false; | 
			
		
	
		
			
				
					|  |  |  |  | 				} | 
			
		
	
		
			
				
					|  |  |  |  | 				return mark(block, node); | 
			
		
	
		
			
				
					|  |  |  |  | 				return mark(relative_selector, node); | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 			for (const possible_sibling of siblings.keys()) { | 
			
		
	
		
			
				
					|  |  |  |  | 				if (apply_selector(blocks.slice(), possible_sibling, stylesheet)) { | 
			
		
	
		
			
				
					|  |  |  |  | 					mark(block, node); | 
			
		
	
		
			
				
					|  |  |  |  | 				if (apply_selector(relative_selectors.slice(), possible_sibling, stylesheet)) { | 
			
		
	
		
			
				
					|  |  |  |  | 					mark(relative_selector, node); | 
			
		
	
		
			
				
					|  |  |  |  | 					has_match = true; | 
			
		
	
		
			
				
					|  |  |  |  | 				} | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
	
		
			
				
					|  |  |  | @ -289,25 +314,25 @@ function apply_selector(blocks, node, stylesheet) { | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		// TODO other combinators
 | 
			
		
	
		
			
				
					|  |  |  |  | 		return mark(block, node); | 
			
		
	
		
			
				
					|  |  |  |  | 		return mark(relative_selector, node); | 
			
		
	
		
			
				
					|  |  |  |  | 	} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	return mark(block, node); | 
			
		
	
		
			
				
					|  |  |  |  | 	return mark(relative_selector, node); | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | const regex_backslash_and_following_character = /\\(.)/g; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | /** | 
			
		
	
		
			
				
					|  |  |  |  |  * @param {Block} block | 
			
		
	
		
			
				
					|  |  |  |  |  * @param {RelativeSelector} relative_selector | 
			
		
	
		
			
				
					|  |  |  |  |  * @param {import('#compiler').RegularElement | import('#compiler').SvelteElement} node | 
			
		
	
		
			
				
					|  |  |  |  |  * @returns {NO_MATCH | POSSIBLE_MATCH | UNKNOWN_SELECTOR} | 
			
		
	
		
			
				
					|  |  |  |  |  */ | 
			
		
	
		
			
				
					|  |  |  |  | function block_might_apply_to_node(block, node) { | 
			
		
	
		
			
				
					|  |  |  |  | 	if (block.host || block.root) return NO_MATCH; | 
			
		
	
		
			
				
					|  |  |  |  | function block_might_apply_to_node(relative_selector, node) { | 
			
		
	
		
			
				
					|  |  |  |  | 	if (relative_selector.is_host || relative_selector.is_root) return NO_MATCH; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	let i = block.selectors.length; | 
			
		
	
		
			
				
					|  |  |  |  | 	let i = relative_selector.selectors.length; | 
			
		
	
		
			
				
					|  |  |  |  | 	while (i--) { | 
			
		
	
		
			
				
					|  |  |  |  | 		const selector = block.selectors[i]; | 
			
		
	
		
			
				
					|  |  |  |  | 		const selector = relative_selector.selectors[i]; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		if (selector.type === 'Percentage' || selector.type === 'Nth') continue; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
	
		
			
				
					|  |  |  | @ -317,7 +342,7 @@ function block_might_apply_to_node(block, node) { | 
			
		
	
		
			
				
					|  |  |  |  | 			return NO_MATCH; | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 		if ( | 
			
		
	
		
			
				
					|  |  |  |  | 			block.selectors.length === 1 && | 
			
		
	
		
			
				
					|  |  |  |  | 			relative_selector.selectors.length === 1 && | 
			
		
	
		
			
				
					|  |  |  |  | 			selector.type === 'PseudoClassSelector' && | 
			
		
	
		
			
				
					|  |  |  |  | 			name === 'global' | 
			
		
	
		
			
				
					|  |  |  |  | 		) { | 
			
		
	
	
		
			
				
					|  |  |  | @ -636,22 +661,22 @@ function get_possible_element_siblings(node, adjacent_only) { | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | /** | 
			
		
	
		
			
				
					|  |  |  |  |  * @param {import('#compiler').EachBlock | import('#compiler').IfBlock | import('#compiler').AwaitBlock} block | 
			
		
	
		
			
				
					|  |  |  |  |  * @param {import('#compiler').EachBlock | import('#compiler').IfBlock | import('#compiler').AwaitBlock} relative_selector | 
			
		
	
		
			
				
					|  |  |  |  |  * @param {boolean} adjacent_only | 
			
		
	
		
			
				
					|  |  |  |  |  * @returns {Map<import('#compiler').RegularElement, NodeExistsValue>} | 
			
		
	
		
			
				
					|  |  |  |  |  */ | 
			
		
	
		
			
				
					|  |  |  |  | function get_possible_last_child(block, adjacent_only) { | 
			
		
	
		
			
				
					|  |  |  |  | function get_possible_last_child(relative_selector, adjacent_only) { | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @typedef {Map<import('#compiler').RegularElement, NodeExistsValue>} NodeMap */ | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @type {NodeMap} */ | 
			
		
	
		
			
				
					|  |  |  |  | 	const result = new Map(); | 
			
		
	
		
			
				
					|  |  |  |  | 	if (block.type === 'EachBlock') { | 
			
		
	
		
			
				
					|  |  |  |  | 	if (relative_selector.type === 'EachBlock') { | 
			
		
	
		
			
				
					|  |  |  |  | 		/** @type {NodeMap} */ | 
			
		
	
		
			
				
					|  |  |  |  | 		const each_result = loop_child(block.body.nodes, adjacent_only); | 
			
		
	
		
			
				
					|  |  |  |  | 		const each_result = loop_child(relative_selector.body.nodes, adjacent_only); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		/** @type {NodeMap} */ | 
			
		
	
		
			
				
					|  |  |  |  | 		const else_result = block.fallback | 
			
		
	
		
			
				
					|  |  |  |  | 			? loop_child(block.fallback.nodes, adjacent_only) | 
			
		
	
		
			
				
					|  |  |  |  | 		const else_result = relative_selector.fallback | 
			
		
	
		
			
				
					|  |  |  |  | 			? loop_child(relative_selector.fallback.nodes, adjacent_only) | 
			
		
	
		
			
				
					|  |  |  |  | 			: new Map(); | 
			
		
	
		
			
				
					|  |  |  |  | 		const not_exhaustive = !has_definite_elements(else_result); | 
			
		
	
		
			
				
					|  |  |  |  | 		if (not_exhaustive) { | 
			
		
	
	
		
			
				
					|  |  |  | @ -660,13 +685,13 @@ function get_possible_last_child(block, adjacent_only) { | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 		add_to_map(each_result, result); | 
			
		
	
		
			
				
					|  |  |  |  | 		add_to_map(else_result, result); | 
			
		
	
		
			
				
					|  |  |  |  | 	} else if (block.type === 'IfBlock') { | 
			
		
	
		
			
				
					|  |  |  |  | 	} else if (relative_selector.type === 'IfBlock') { | 
			
		
	
		
			
				
					|  |  |  |  | 		/** @type {NodeMap} */ | 
			
		
	
		
			
				
					|  |  |  |  | 		const if_result = loop_child(block.consequent.nodes, adjacent_only); | 
			
		
	
		
			
				
					|  |  |  |  | 		const if_result = loop_child(relative_selector.consequent.nodes, adjacent_only); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		/** @type {NodeMap} */ | 
			
		
	
		
			
				
					|  |  |  |  | 		const else_result = block.alternate | 
			
		
	
		
			
				
					|  |  |  |  | 			? loop_child(block.alternate.nodes, adjacent_only) | 
			
		
	
		
			
				
					|  |  |  |  | 		const else_result = relative_selector.alternate | 
			
		
	
		
			
				
					|  |  |  |  | 			? loop_child(relative_selector.alternate.nodes, adjacent_only) | 
			
		
	
		
			
				
					|  |  |  |  | 			: new Map(); | 
			
		
	
		
			
				
					|  |  |  |  | 		const not_exhaustive = !has_definite_elements(if_result) || !has_definite_elements(else_result); | 
			
		
	
		
			
				
					|  |  |  |  | 		if (not_exhaustive) { | 
			
		
	
	
		
			
				
					|  |  |  | @ -675,17 +700,21 @@ function get_possible_last_child(block, adjacent_only) { | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 		add_to_map(if_result, result); | 
			
		
	
		
			
				
					|  |  |  |  | 		add_to_map(else_result, result); | 
			
		
	
		
			
				
					|  |  |  |  | 	} else if (block.type === 'AwaitBlock') { | 
			
		
	
		
			
				
					|  |  |  |  | 	} else if (relative_selector.type === 'AwaitBlock') { | 
			
		
	
		
			
				
					|  |  |  |  | 		/** @type {NodeMap} */ | 
			
		
	
		
			
				
					|  |  |  |  | 		const pending_result = block.pending | 
			
		
	
		
			
				
					|  |  |  |  | 			? loop_child(block.pending.nodes, adjacent_only) | 
			
		
	
		
			
				
					|  |  |  |  | 		const pending_result = relative_selector.pending | 
			
		
	
		
			
				
					|  |  |  |  | 			? loop_child(relative_selector.pending.nodes, adjacent_only) | 
			
		
	
		
			
				
					|  |  |  |  | 			: new Map(); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		/** @type {NodeMap} */ | 
			
		
	
		
			
				
					|  |  |  |  | 		const then_result = block.then ? loop_child(block.then.nodes, adjacent_only) : new Map(); | 
			
		
	
		
			
				
					|  |  |  |  | 		const then_result = relative_selector.then | 
			
		
	
		
			
				
					|  |  |  |  | 			? loop_child(relative_selector.then.nodes, adjacent_only) | 
			
		
	
		
			
				
					|  |  |  |  | 			: new Map(); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		/** @type {NodeMap} */ | 
			
		
	
		
			
				
					|  |  |  |  | 		const catch_result = block.catch ? loop_child(block.catch.nodes, adjacent_only) : new Map(); | 
			
		
	
		
			
				
					|  |  |  |  | 		const catch_result = relative_selector.catch | 
			
		
	
		
			
				
					|  |  |  |  | 			? loop_child(relative_selector.catch.nodes, adjacent_only) | 
			
		
	
		
			
				
					|  |  |  |  | 			: new Map(); | 
			
		
	
		
			
				
					|  |  |  |  | 		const not_exhaustive = | 
			
		
	
		
			
				
					|  |  |  |  | 			!has_definite_elements(pending_result) || | 
			
		
	
		
			
				
					|  |  |  |  | 			!has_definite_elements(then_result) || | 
			
		
	
	
		
			
				
					|  |  |  | @ -774,55 +803,52 @@ function loop_child(children, adjacent_only) { | 
			
		
	
		
			
				
					|  |  |  |  | 	return result; | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | class Block { | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @type {boolean} */ | 
			
		
	
		
			
				
					|  |  |  |  | 	host; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @type {boolean} */ | 
			
		
	
		
			
				
					|  |  |  |  | 	root; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | /** | 
			
		
	
		
			
				
					|  |  |  |  |  * Represents a compound selector (aka an array of simple selectors) plus | 
			
		
	
		
			
				
					|  |  |  |  |  * a preceding combinator (if not the first in the list). Given this... | 
			
		
	
		
			
				
					|  |  |  |  |  * | 
			
		
	
		
			
				
					|  |  |  |  |  * ```css
 | 
			
		
	
		
			
				
					|  |  |  |  |  * .a + .b.c {...} | 
			
		
	
		
			
				
					|  |  |  |  |  * ``` | 
			
		
	
		
			
				
					|  |  |  |  |  * | 
			
		
	
		
			
				
					|  |  |  |  |  * ...both `.a` and `+ .b.c` are relative selectors. | 
			
		
	
		
			
				
					|  |  |  |  |  * Combined, they are a complex selector. | 
			
		
	
		
			
				
					|  |  |  |  |  */ | 
			
		
	
		
			
				
					|  |  |  |  | class RelativeSelector { | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @type {import('#compiler').Css.Combinator | null} */ | 
			
		
	
		
			
				
					|  |  |  |  | 	combinator; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @type {import('#compiler').Css.SimpleSelector[]} */ | 
			
		
	
		
			
				
					|  |  |  |  | 	selectors; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @type {number} */ | 
			
		
	
		
			
				
					|  |  |  |  | 	start; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @type {number} */ | 
			
		
	
		
			
				
					|  |  |  |  | 	end; | 
			
		
	
		
			
				
					|  |  |  |  | 	selectors = []; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @type {boolean} */ | 
			
		
	
		
			
				
					|  |  |  |  | 	should_encapsulate; | 
			
		
	
		
			
				
					|  |  |  |  | 	is_host = false; | 
			
		
	
		
			
				
					|  |  |  |  | 	is_root = false; | 
			
		
	
		
			
				
					|  |  |  |  | 	should_encapsulate = false; | 
			
		
	
		
			
				
					|  |  |  |  | 	start = -1; | 
			
		
	
		
			
				
					|  |  |  |  | 	end = -1; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @param {import('#compiler').Css.Combinator | null} combinator */ | 
			
		
	
		
			
				
					|  |  |  |  | 	constructor(combinator) { | 
			
		
	
		
			
				
					|  |  |  |  | 		this.combinator = combinator; | 
			
		
	
		
			
				
					|  |  |  |  | 		this.host = false; | 
			
		
	
		
			
				
					|  |  |  |  | 		this.root = false; | 
			
		
	
		
			
				
					|  |  |  |  | 		this.selectors = []; | 
			
		
	
		
			
				
					|  |  |  |  | 		this.start = -1; | 
			
		
	
		
			
				
					|  |  |  |  | 		this.end = -1; | 
			
		
	
		
			
				
					|  |  |  |  | 		this.should_encapsulate = false; | 
			
		
	
		
			
				
					|  |  |  |  | 	} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	/** @param {import('#compiler').Css.SimpleSelector} selector */ | 
			
		
	
		
			
				
					|  |  |  |  | 	add(selector) { | 
			
		
	
		
			
				
					|  |  |  |  | 		if (this.selectors.length === 0) { | 
			
		
	
		
			
				
					|  |  |  |  | 			this.start = selector.start; | 
			
		
	
		
			
				
					|  |  |  |  | 			this.host = selector.type === 'PseudoClassSelector' && selector.name === 'host'; | 
			
		
	
		
			
				
					|  |  |  |  | 			this.is_host = selector.type === 'PseudoClassSelector' && selector.name === 'host'; | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 		this.root = this.root || (selector.type === 'PseudoClassSelector' && selector.name === 'root'); | 
			
		
	
		
			
				
					|  |  |  |  | 		this.is_root = | 
			
		
	
		
			
				
					|  |  |  |  | 			this.is_root || (selector.type === 'PseudoClassSelector' && selector.name === 'root'); | 
			
		
	
		
			
				
					|  |  |  |  | 		this.selectors.push(selector); | 
			
		
	
		
			
				
					|  |  |  |  | 		this.end = selector.end; | 
			
		
	
		
			
				
					|  |  |  |  | 	} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	can_ignore() { | 
			
		
	
		
			
				
					|  |  |  |  | 		return this.global || this.host || this.root; | 
			
		
	
		
			
				
					|  |  |  |  | 		return this.is_global || this.is_host || this.is_root; | 
			
		
	
		
			
				
					|  |  |  |  | 	} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	get global() { | 
			
		
	
		
			
				
					|  |  |  |  | 	get is_global() { | 
			
		
	
		
			
				
					|  |  |  |  | 		return ( | 
			
		
	
		
			
				
					|  |  |  |  | 			this.selectors.length >= 1 && | 
			
		
	
		
			
				
					|  |  |  |  | 			this.selectors[0].type === 'PseudoClassSelector' && | 
			
		
	
	
		
			
				
					|  |  |  | @ -835,18 +861,18 @@ class Block { | 
			
		
	
		
			
				
					|  |  |  |  | 	} | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | /** @param {import('#compiler').Css.Selector} selector */ | 
			
		
	
		
			
				
					|  |  |  |  | /** @param {import('#compiler').Css.ComplexSelector} selector */ | 
			
		
	
		
			
				
					|  |  |  |  | function group_selectors(selector) { | 
			
		
	
		
			
				
					|  |  |  |  | 	let block = new Block(null); | 
			
		
	
		
			
				
					|  |  |  |  | 	const blocks = [block]; | 
			
		
	
		
			
				
					|  |  |  |  | 	let relative_selector = new RelativeSelector(null); | 
			
		
	
		
			
				
					|  |  |  |  | 	const relative_selectors = [relative_selector]; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	selector.children.forEach((child) => { | 
			
		
	
		
			
				
					|  |  |  |  | 		if (child.type === 'Combinator') { | 
			
		
	
		
			
				
					|  |  |  |  | 			block = new Block(child); | 
			
		
	
		
			
				
					|  |  |  |  | 			blocks.push(block); | 
			
		
	
		
			
				
					|  |  |  |  | 			relative_selector = new RelativeSelector(child); | 
			
		
	
		
			
				
					|  |  |  |  | 			relative_selectors.push(relative_selector); | 
			
		
	
		
			
				
					|  |  |  |  | 		} else { | 
			
		
	
		
			
				
					|  |  |  |  | 			block.add(child); | 
			
		
	
		
			
				
					|  |  |  |  | 			relative_selector.add(child); | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 	}); | 
			
		
	
		
			
				
					|  |  |  |  | 	return blocks; | 
			
		
	
		
			
				
					|  |  |  |  | 	return relative_selectors; | 
			
		
	
		
			
				
					|  |  |  |  | } |