/* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ //! The context within which CSS code is parsed. use context::QuirksMode; use cssparser::{Parser, SourceLocation, UnicodeRange}; use error_reporting::{ContextualParseError, ParseErrorReporter}; use style_traits::{OneOrMoreSeparated, ParseError, ParsingMode, Separator}; use stylesheets::{CssRuleType, Namespaces, Origin, UrlExtraData}; /// Asserts that all ParsingMode flags have a matching ParsingMode value in gecko. #[cfg(feature = "gecko")] #[inline] pub fn assert_parsing_mode_match() { use gecko_bindings::structs; macro_rules! check_parsing_modes { ( $( $a:ident => $b:path ),*, ) => { if cfg!(debug_assertions) { let mut modes = ParsingMode::all(); $( assert_eq!(structs::$a as usize, $b.bits() as usize, stringify!($b)); modes.remove($b); )* assert_eq!(modes, ParsingMode::empty(), "all ParsingMode bits should have an assertion"); } } } check_parsing_modes! { ParsingMode_Default => ParsingMode::DEFAULT, ParsingMode_AllowUnitlessLength => ParsingMode::ALLOW_UNITLESS_LENGTH, ParsingMode_AllowAllNumericValues => ParsingMode::ALLOW_ALL_NUMERIC_VALUES, } } /// The data that the parser needs from outside in order to parse a stylesheet. pub struct ParserContext<'a> { /// The `Origin` of the stylesheet, whether it's a user, author or /// user-agent stylesheet. pub stylesheet_origin: Origin, /// The extra data we need for resolving url values. pub url_data: &'a UrlExtraData, /// The current rule type, if any. pub rule_type: Option, /// The mode to use when parsing. pub parsing_mode: ParsingMode, /// The quirks mode of this stylesheet. pub quirks_mode: QuirksMode, /// The active error reporter, or none if error reporting is disabled. error_reporter: Option<&'a ParseErrorReporter>, /// The currently active namespaces. pub namespaces: Option<&'a Namespaces>, } impl<'a> ParserContext<'a> { /// Create a parser context. #[inline] pub fn new( stylesheet_origin: Origin, url_data: &'a UrlExtraData, rule_type: Option, parsing_mode: ParsingMode, quirks_mode: QuirksMode, error_reporter: Option<&'a ParseErrorReporter>, ) -> Self { ParserContext { stylesheet_origin, url_data, rule_type, parsing_mode, quirks_mode, error_reporter, namespaces: None, } } /// Create a parser context for on-the-fly parsing in CSSOM #[inline] pub fn new_for_cssom( url_data: &'a UrlExtraData, rule_type: Option, parsing_mode: ParsingMode, quirks_mode: QuirksMode, error_reporter: Option<&'a ParseErrorReporter>, ) -> Self { Self::new( Origin::Author, url_data, rule_type, parsing_mode, quirks_mode, error_reporter, ) } /// Create a parser context based on a previous context, but with a modified rule type. #[inline] pub fn new_with_rule_type( context: &'a ParserContext, rule_type: CssRuleType, namespaces: &'a Namespaces, ) -> ParserContext<'a> { ParserContext { stylesheet_origin: context.stylesheet_origin, url_data: context.url_data, rule_type: Some(rule_type), parsing_mode: context.parsing_mode, quirks_mode: context.quirks_mode, namespaces: Some(namespaces), error_reporter: context.error_reporter, } } /// Whether we're in a @page rule. #[inline] pub fn in_page_rule(&self) -> bool { self.rule_type .map_or(false, |rule_type| rule_type == CssRuleType::Page) } /// Get the rule type, which assumes that one is available. pub fn rule_type(&self) -> CssRuleType { self.rule_type .expect("Rule type expected, but none was found.") } /// Record a CSS parse error with this context’s error reporting. pub fn log_css_error( &self, location: SourceLocation, error: ContextualParseError, ) { let error_reporter = match self.error_reporter { Some(r) => r, None => return, }; error_reporter.report_error(self.url_data, location, error) } /// Returns whether chrome-only rules should be parsed. pub fn chrome_rules_enabled(&self) -> bool { self.url_data.is_chrome() || self.stylesheet_origin == Origin::User } } /// A trait to abstract parsing of a specified value given a `ParserContext` and /// CSS input. /// /// This can be derived on keywords with `#[derive(Parse)]`. /// /// The derive code understands the following attributes on each of the variants: /// /// * `#[parse(aliases = "foo,bar")]` can be used to alias a value with another /// at parse-time. /// /// * `#[parse(condition = "function")]` can be used to make the parsing of the /// value conditional on `function`, which needs to fulfill /// `fn(&ParserContext) -> bool`. pub trait Parse: Sized { /// Parse a value of this type. /// /// Returns an error on failure. fn parse<'i, 't>( context: &ParserContext, input: &mut Parser<'i, 't>, ) -> Result>; } impl Parse for Vec where T: Parse + OneOrMoreSeparated, ::S: Separator, { fn parse<'i, 't>( context: &ParserContext, input: &mut Parser<'i, 't>, ) -> Result> { ::S::parse(input, |i| T::parse(context, i)) } } impl Parse for UnicodeRange { fn parse<'i, 't>( _context: &ParserContext, input: &mut Parser<'i, 't>, ) -> Result> { UnicodeRange::parse(input).map_err(|e| e.into()) } }