#[derive(Debug, Clone)]
pub struct Stylesheet
{
pub rules: CssRules,
pub source_map_url: Option<String>,
pub source_url: Option<String>,
}
impl HasCssRules for Stylesheet
{
#[inline(always)]
fn css_rules(&self) -> &CssRules
{
&self.rules
}
#[inline(always)]
fn css_rules_mut(&mut self) -> &mut CssRules
{
&mut self.rules
}
#[inline(always)]
fn css_rules_slice(&self) -> &[CssRule]
{
&self.rules.0[..]
}
#[inline(always)]
fn css_rules_vec(&self) -> &Vec<CssRule>
{
&self.rules.0
}
#[inline(always)]
fn css_rules_vec_mut(&mut self) -> &mut Vec<CssRule>
{
&mut self.rules.0
}
}
impl Stylesheet
{
#[inline(always)]
pub fn to_file_path<P: AsRef<Path>>(&self, stylesheet_file_path: P, include_source_urls: bool) -> Result<(), StylesheetError>
{
let path = stylesheet_file_path.as_ref();
let file = File::create(path).context(path)?;
self.to_css(&mut BlockingIoOnlyStdFmtWriteToStdIoWriteAdaptor(file), include_source_urls).context(path)?;
Ok(())
}
#[inline(always)]
pub fn to_css_string(&self, include_source_urls: bool) -> String
{
let mut string = String::new();
self.to_css(&mut string, include_source_urls).unwrap();
string
}
#[inline(always)]
pub fn to_bytes(&self, include_source_urls: bool) -> Vec<u8>
{
self.to_css_string(include_source_urls).into_bytes()
}
pub fn to_css<W: fmt::Write>(&self, destination: &mut W, include_source_urls: bool) -> fmt::Result
{
if include_source_urls
{
if let Some(ref source_map_url) = self.source_map_url
{
write!(destination, "//# sourceMappingURL=<{}>\n", source_map_url)?;
}
if let Some(ref source_url) = self.source_url
{
write!(destination, "//# sourceURL=<{}>\n", source_url)?;
}
}
self.rules.to_css(destination)?;
Ok(())
}
#[inline(always)]
pub fn from_file_path<P: AsRef<Path>>(html_document_file_path: P) -> Result<Self, StylesheetError>
{
let path = html_document_file_path.as_ref();
let metadata = path.metadata().context(path)?;
let mut file = File::open(path).context(path)?;
let mut css = String::with_capacity(metadata.len() as usize);
file.read_to_string(&mut css).context(path)?;
let result = Self::parse(&css);
match result
{
Ok(stylesheet) => Ok(stylesheet),
Err(cause) => Err(StylesheetError::Parse
(
path.to_path_buf(),
cause.location,
format!("{:?}", cause.error),
)),
}
}
pub fn parse<'i>(css: &'i str) -> Result<Self, PreciseParseError<'i, CustomParseError<'i>>>
{
const LineNumberingIsZeroBased: u32 = 0;
let mut parserInput = ParserInput::new_with_line_number_offset(css, LineNumberingIsZeroBased);
let mut input = Parser::new(&mut parserInput);
let mut rules = Vec::new();
let topLevelRuleParser = TopLevelRuleParser
{
context: ParserContext
{
rule_type: None,
parsing_mode: ParsingMode::Default,
},
state: State::Start,
namespaces: Namespaces::empty(),
};
{
let mut iter = RuleListParser::new_for_stylesheet(&mut input, topLevelRuleParser);
while let Some(result) = iter.next()
{
match result
{
Ok(rule) => rules.push(rule),
Err(preciseParseError) => return Err(preciseParseError),
}
}
}
Ok
(
Self
{
rules: CssRules(rules),
source_map_url: input.current_source_map_url().map(String::from),
source_url: input.current_source_url().map(String::from),
}
)
}
}