grass/src/css.rs

144 lines
4.3 KiB
Rust
Raw Normal View History

2020-01-05 12:52:50 -05:00
//! # Convert from SCSS AST to CSS
2020-01-26 15:27:38 -05:00
use crate::atrule::AtRule;
use crate::{RuleSet, SassResult, Selector, Stmt, Style, StyleSheet};
use std::fmt;
use std::io::Write;
#[derive(Debug, Clone)]
enum Toplevel {
RuleSet(Selector, Vec<BlockEntry>),
MultilineComment(String),
2020-01-26 15:27:38 -05:00
AtRule(AtRule),
Newline,
}
#[derive(Debug, Clone)]
enum BlockEntry {
Style(Style),
MultilineComment(String),
2020-01-26 15:27:38 -05:00
AtRule(AtRule),
}
impl fmt::Display for BlockEntry {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
BlockEntry::Style(s) => writeln!(f, " {}", s),
BlockEntry::MultilineComment(s) => writeln!(f, " /*{}*/", s),
2020-01-26 15:27:38 -05:00
BlockEntry::AtRule(r) => writeln!(f, "{}", r),
}
}
}
impl Toplevel {
const fn new_rule(selector: Selector) -> Self {
Toplevel::RuleSet(selector, Vec::new())
}
fn push_style(&mut self, s: Style) {
if let Toplevel::RuleSet(_, entries) = self {
entries.push(BlockEntry::Style(s));
}
}
fn push_comment(&mut self, s: String) {
if let Toplevel::RuleSet(_, entries) = self {
entries.push(BlockEntry::MultilineComment(s));
}
}
}
#[derive(Debug, Clone)]
pub struct Css {
blocks: Vec<Toplevel>,
}
impl Css {
2020-01-05 12:52:50 -05:00
pub const fn new() -> Self {
2020-01-19 19:27:52 -05:00
Css { blocks: Vec::new() }
}
pub fn from_stylesheet(s: StyleSheet) -> Self {
Css::new().parse_stylesheet(s)
}
2020-01-19 19:27:52 -05:00
fn parse_stmt(&mut self, stmt: Stmt) -> Vec<Toplevel> {
match stmt {
Stmt::RuleSet(RuleSet {
selector,
super_selector,
rules,
}) => {
2020-01-19 19:27:52 -05:00
let mut vals = vec![Toplevel::new_rule(super_selector.zip(&selector))];
for rule in rules {
match rule {
Stmt::RuleSet(_) => vals.extend(self.parse_stmt(rule)),
Stmt::Style(s) => vals
.get_mut(0)
.expect("expected block to exist")
.push_style(s),
Stmt::MultilineComment(s) => vals
.get_mut(0)
.expect("expected block to exist")
.push_comment(s),
2020-01-26 15:27:38 -05:00
Stmt::AtRule(_) => todo!("at rule inside css block"),
2020-01-19 19:27:52 -05:00
};
}
2020-01-19 19:27:52 -05:00
vals
}
2020-01-19 19:27:52 -05:00
Stmt::MultilineComment(s) => vec![Toplevel::MultilineComment(s)],
Stmt::Style(_) => panic!("expected toplevel element, found style"),
2020-01-26 15:27:38 -05:00
Stmt::AtRule(r) => vec![Toplevel::AtRule(r)],
}
}
fn parse_stylesheet(mut self, s: StyleSheet) -> Css {
let mut is_first = true;
2020-01-18 14:57:56 -05:00
for stmt in s.0 {
2020-01-19 19:27:52 -05:00
let v = self.parse_stmt(stmt);
// this is how we print newlines between unrelated styles
// it could probably be refactored
if !v.is_empty() {
if let Toplevel::MultilineComment(..) = v[0] {}
else if is_first {
is_first = false;
} else {
self.blocks.push(Toplevel::Newline);
}
}
2020-01-19 19:27:52 -05:00
self.blocks.extend(v);
}
self
}
pub fn pretty_print<W: Write>(self, buf: &mut W) -> SassResult<()> {
let mut has_written = false;
for block in self.blocks {
match block {
Toplevel::RuleSet(selector, styles) => {
if styles.is_empty() {
continue;
}
has_written = true;
writeln!(buf, "{} {{", selector)?;
for style in styles {
write!(buf, "{}", style)?;
}
writeln!(buf, "}}")?;
}
Toplevel::MultilineComment(s) => {
has_written = true;
writeln!(buf, "/*{}*/", s)?;
}
2020-01-26 15:27:38 -05:00
Toplevel::AtRule(r) => {
has_written = true;
2020-01-26 15:27:38 -05:00
writeln!(buf, "{}", r)?;
}
Toplevel::Newline => if has_written {
writeln!(buf)?
}
}
}
Ok(())
}
2020-01-05 12:52:50 -05:00
}