class Sass::SCSS::StaticParser
after resolving the interpolation.
This is useful for e.g. {#parse_selector parsing selectors}
but without dynamic SassScript.
Parses with SCSS extensions, like nested rules and parent selectors,
A parser for a static SCSS tree.
def a_n_plus_b
def a_n_plus_b if (parity = tok(/even|odd/i)) return parity end if tok(/[+-]?[0-9]+/) ss return true unless tok(/n/) else return unless tok(/[+-]?n/i) end ss return true unless tok(/[+-]/) ss @expected = "number" tok!(/[0-9]+/) true end
def attrib
def attrib return unless tok(/\[/) ss ns, name = attrib_name! ss op = tok(/=/) || tok(INCLUDES) || tok(DASHMATCH) || tok(PREFIXMATCH) || tok(SUFFIXMATCH) || tok(SUBSTRINGMATCH) if op @expected = "identifier or string" ss val = ident || tok!(STRING) ss end flags = ident || tok(STRING) tok!(/\]/) Selector::Attribute.new(name, ns, op, val, flags) end
def attrib_name!
def attrib_name! if (name_or_ns = ident) # E, E|E if tok(/\|(?!=)/) ns = name_or_ns name = ident else name = name_or_ns end else # *|E or |E ns = tok(/\*/) || "" tok!(/\|/) name = ident! end return ns, name end
def class_selector
def class_selector return unless tok(/\./) @expected = "class name" Selector::Class.new(ident!) end
def combinator
def combinator tok(PLUS) || tok(GREATER) || tok(TILDE) || reference_combinator end
def element_name
def element_name ns, name = Sass::Util.destructure(qualified_name(:allow_star_name)) return unless ns || name if name == '*' Selector::Universal.new(ns) else Selector::Element.new(name, ns) end end
def id_selector
def id_selector return unless tok(/#(?!\{)/) @expected = "id name" Selector::Id.new(name!) end
def initialize(str, filename, importer, line = 1, offset = 1, allow_parent_ref = true)
-
allow_parent_ref
(Boolean
) -- Whether to allow the
Other tags:
- See: Parser#initialize -
def initialize(str, filename, importer, line = 1, offset = 1, allow_parent_ref = true) super(str, filename, importer, line, offset) @allow_parent_ref = allow_parent_ref end
def interp_ident; (s = ident) && [s]; end
def interp_ident; (s = ident) && [s]; end
def interp_string; (s = tok(STRING)) && [s]; end
def interp_string; (s = tok(STRING)) && [s]; end
def interp_uri; (s = tok(URI)) && [s]; end
def interp_uri; (s = tok(URI)) && [s]; end
def interpolation(warn_for_color = false); nil; end
def interpolation(warn_for_color = false); nil; end
def keyframes_selector
def keyframes_selector ss str do return unless keyframes_selector_component ss while tok(/,/) ss expr!(:keyframes_selector_component) ss end end end
def keyframes_selector_component
def keyframes_selector_component ident || tok(PERCENTAGE) end
def moz_document_function
def moz_document_function val = tok(URI) || tok(URL_PREFIX) || tok(DOMAIN) || function(false) return unless val ss [val] end
def parent_selector
def parent_selector return unless @allow_parent_ref && tok(/&/) Selector::Parent.new(name) end
def parse_keyframes_selector
def parse_keyframes_selector init_scanner! sel = expr!(:keyframes_selector) expected("keyframes selector") unless @scanner.eos? sel end
def parse_selector
-
(Sass::SyntaxError)
- if there's a syntax error in the selector
Returns:
-
(Selector::CommaSequence)
- The parsed selector
Parameters:
-
filename
(String, nil
) -- The file in which the selector appears,
def parse_selector init_scanner! seq = expr!(:selector_comma_sequence) expected("selector") unless @scanner.eos? seq.line = @line seq.filename = @filename seq end
def parse_static_at_root_query
-
(Sass::SyntaxError)
- if there's a syntax error in the query,
Returns:
-
((Symbol, Array
- The type of the query))
def parse_static_at_root_query init_scanner! tok!(/\(/); ss type = tok!(/\b(without|with)\b/).to_sym; ss tok!(/:/); ss directives = expr!(:at_root_directive_list); ss tok!(/\)/) expected("@at-root query list") unless @scanner.eos? return type, directives end
def placeholder_selector
def placeholder_selector return unless tok(/%/) @expected = "placeholder name" Selector::Placeholder.new(ident!) end
def prefixed_selector_pseudo
def prefixed_selector_pseudo prefix = str do expr = str {expr!(:a_n_plus_b)} ss return expr, nil unless tok(/of/) ss end return prefix, expr!(:selector_comma_sequence) end
def pseudo
def pseudo s = tok(/::?/) return unless s @expected = "pseudoclass or pseudoelement" name = ident! if tok(/\(/) ss deprefixed = deprefix(name) if s == ':' && SELECTOR_PSEUDO_CLASSES.include?(deprefixed) sel = selector_comma_sequence elsif s == ':' && PREFIXED_SELECTOR_PSEUDO_CLASSES.include?(deprefixed) arg, sel = prefixed_selector_pseudo elsif s == '::' && SELECTOR_PSEUDO_ELEMENTS.include?(deprefixed) sel = selector_comma_sequence else arg = expr!(:declaration_value).join end tok!(/\)/) end Selector::Pseudo.new(s == ':' ? :class : :element, name, arg, sel) end
def qualified_name(allow_star_name = false)
def qualified_name(allow_star_name = false) name = ident || tok(/\*/) || (tok?(/\|/) && "") return unless name return nil, name unless tok(/\|/) return name, ident! unless allow_star_name @expected = "identifier or *" return name, ident || tok!(/\*/) end
def reference_combinator
def reference_combinator return unless tok(%r{/}) res = '/' ns, name = expr!(:qualified_name) res << ns << '|' if ns res << name << tok!(%r{/}) location = " of #{@filename}" if @filename Sass::Util.sass_warn <<MESSAGE ATION WARNING on line #{@line}, column #{@offset}#{location}: ference combinator #{res} is deprecated and will be removed in a future release. E res end
def script_value; nil; end
def script_value; nil; end
def selector
def selector start_pos = source_position # The combinator here allows the "> E" hack val = combinator || simple_selector_sequence return unless val nl = str {ss}.include?("\n") res = [] res << val res << "\n" if nl while (val = combinator || simple_selector_sequence) res << val res << "\n" if str {ss}.include?("\n") end seq = Selector::Sequence.new(res.compact) if seq.members.any? {|sseq| sseq.is_a?(Selector::SimpleSequence) && sseq.subject?} location = " of #{@filename}" if @filename Sass::Util.sass_warn <<MESSAGE ATION WARNING on line #{start_pos.line}, column #{start_pos.offset}#{location}: bject selector operator "!" is deprecated and will be removed in a future release. perator has been replaced by ":has()" in the CSS spec. ample: #{seq.subjectless} E end seq end
def selector_comma_sequence
def selector_comma_sequence sel = selector return unless sel selectors = [sel] ws = '' while tok(/,/) ws << str {ss} next unless (sel = selector) selectors << sel if ws.include?("\n") selectors[-1] = Selector::Sequence.new(["\n"] + selectors.last.members) end ws = '' end Selector::CommaSequence.new(selectors) end
def selector_string
def selector_string sel = selector return unless sel sel.to_s end
def simple_selector_sequence
def simple_selector_sequence start_pos = source_position e = element_name || id_selector || class_selector || placeholder_selector || attrib || pseudo || parent_selector return unless e res = [e] # The tok(/\*/) allows the "E*" hack while (v = id_selector || class_selector || placeholder_selector || attrib || pseudo || (tok(/\*/) && Selector::Universal.new(nil))) res << v end pos = @scanner.pos line = @line if (sel = str? {simple_selector_sequence}) @scanner.pos = pos @line = line begin # If we see "*E", don't force a throw because this could be the # "*prop: val" hack. expected('"{"') if res.length == 1 && res[0].is_a?(Selector::Universal) throw_error {expected('"{"')} rescue Sass::SyntaxError => e e.message << "\n\n\"#{sel}\" may only be used at the beginning of a compound selector." raise e end end Selector::SimpleSequence.new(res, tok(/!/), range(start_pos)) end
def special_directive(name, start_pos)
def special_directive(name, start_pos) return unless %w(media import charset -moz-document).include?(name) super end
def use_css_import?; true; end
def use_css_import?; true; end
def var_expr; nil; end
def var_expr; nil; end
def variable; nil; end
def variable; nil; end