refactor how newlines are emitted between unrelated style rules

this makes our output of bootstrap correct, byte-for-byte
This commit is contained in:
Connor Skees 2021-07-23 22:35:03 -04:00
parent 548a5921e6
commit 6febd161af
3 changed files with 166 additions and 131 deletions

View File

@ -25,7 +25,11 @@ struct ToplevelUnknownAtRule {
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
enum Toplevel { enum Toplevel {
RuleSet(Selector, Vec<BlockEntry>), RuleSet {
selector: Selector,
body: Vec<BlockEntry>,
is_group_end: bool,
},
MultilineComment(String), MultilineComment(String),
UnknownAtRule(Box<ToplevelUnknownAtRule>), UnknownAtRule(Box<ToplevelUnknownAtRule>),
Keyframes(Box<Keyframes>), Keyframes(Box<Keyframes>),
@ -34,15 +38,39 @@ enum Toplevel {
query: String, query: String,
body: Vec<Stmt>, body: Vec<Stmt>,
inside_rule: bool, inside_rule: bool,
is_group_end: bool,
}, },
Supports { Supports {
params: String, params: String,
body: Vec<Stmt>, body: Vec<Stmt>,
}, },
Newline,
// todo: do we actually need a toplevel style variant? // todo: do we actually need a toplevel style variant?
Style(Style), Style(Style),
Import(String), Import(String),
Empty,
}
impl Toplevel {
pub fn is_invisible(&self) -> bool {
match self {
Toplevel::RuleSet { selector, body, .. } => selector.is_empty() || body.is_empty(),
Toplevel::Media { body, .. } => body.is_empty(),
Toplevel::Empty => true,
_ => false,
}
}
pub fn is_group_end(&self) -> bool {
match self {
Toplevel::RuleSet { is_group_end, .. } => *is_group_end,
Toplevel::Media {
inside_rule,
is_group_end,
..
} => *inside_rule && *is_group_end,
_ => false,
}
}
} }
#[derive(Debug, Clone)] #[derive(Debug, Clone)]
@ -61,8 +89,12 @@ impl BlockEntry {
} }
impl Toplevel { impl Toplevel {
const fn new_rule(selector: Selector) -> Self { const fn new_rule(selector: Selector, is_group_end: bool) -> Self {
Toplevel::RuleSet(selector, Vec::new()) Toplevel::RuleSet {
selector,
is_group_end,
body: Vec::new(),
}
} }
fn new_keyframes_rule(selector: Vec<KeyframesSelector>) -> Self { fn new_keyframes_rule(selector: Vec<KeyframesSelector>) -> Self {
@ -73,16 +105,17 @@ impl Toplevel {
if s.value.is_null() { if s.value.is_null() {
return; return;
} }
if let Toplevel::RuleSet(_, entries) | Toplevel::KeyframesRuleSet(_, entries) = self {
entries.push(BlockEntry::Style(s)); if let Toplevel::RuleSet { body, .. } | Toplevel::KeyframesRuleSet(_, body) = self {
body.push(BlockEntry::Style(s));
} else { } else {
panic!(); panic!();
} }
} }
fn push_comment(&mut self, s: String) { fn push_comment(&mut self, s: String) {
if let Toplevel::RuleSet(_, entries) | Toplevel::KeyframesRuleSet(_, entries) = self { if let Toplevel::RuleSet { body, .. } | Toplevel::KeyframesRuleSet(_, body) = self {
entries.push(BlockEntry::MultilineComment(s)); body.push(BlockEntry::MultilineComment(s));
} else { } else {
panic!(); panic!();
} }
@ -119,16 +152,16 @@ impl Css {
Ok(match stmt { Ok(match stmt {
Stmt::RuleSet { selector, body } => { Stmt::RuleSet { selector, body } => {
if body.is_empty() { if body.is_empty() {
return Ok(Vec::new()); return Ok(vec![Toplevel::Empty]);
} }
let selector = selector.into_selector().remove_placeholders(); let selector = selector.into_selector().remove_placeholders();
if selector.is_empty() { if selector.is_empty() {
return Ok(vec![Toplevel::new_rule(selector)]); return Ok(vec![Toplevel::Empty]);
} }
let mut vals = vec![Toplevel::new_rule(selector)]; let mut vals = vec![Toplevel::new_rule(selector, false)];
for rule in body { for rule in body {
match rule { match rule {
@ -141,6 +174,7 @@ impl Css {
query, query,
body, body,
inside_rule: true, inside_rule: true,
is_group_end: false,
}); });
} }
Stmt::Supports(s) => { Stmt::Supports(s) => {
@ -192,6 +226,7 @@ impl Css {
query, query,
body, body,
inside_rule: false, inside_rule: false,
is_group_end: false,
}] }]
} }
Stmt::Supports(s) => { Stmt::Supports(s) => {
@ -230,53 +265,18 @@ impl Css {
} }
fn parse_stylesheet(mut self, stmts: Vec<Stmt>) -> SassResult<Css> { fn parse_stylesheet(mut self, stmts: Vec<Stmt>) -> SassResult<Css> {
#[derive(PartialEq, Eq, Clone, Copy, Debug)]
enum ToplevelKind {
Comment,
Media,
Other,
}
let mut is_first = true;
let mut last_toplevel = ToplevelKind::Other;
for stmt in stmts { for stmt in stmts {
let v = self.parse_stmt(stmt)?; let mut v = self.parse_stmt(stmt)?;
// this is how we print newlines between unrelated styles
// it could probably be refactored
if !v.is_empty() {
match v.first() {
Some(Toplevel::MultilineComment(..)) => {
if last_toplevel != ToplevelKind::Comment && !is_first {
self.blocks.push(Toplevel::Newline);
}
last_toplevel = ToplevelKind::Comment; match v.last_mut() {
} Some(Toplevel::RuleSet { is_group_end, .. })
Some(Toplevel::Media { .. }) => { | Some(Toplevel::Media { is_group_end, .. }) => {
if last_toplevel != ToplevelKind::Media && !is_first { *is_group_end = true;
self.blocks.push(Toplevel::Newline);
}
last_toplevel = ToplevelKind::Media;
}
_ if is_first => {
last_toplevel = ToplevelKind::Other;
}
_ => {
if last_toplevel != ToplevelKind::Comment
&& last_toplevel != ToplevelKind::Media
{
self.blocks.push(Toplevel::Newline);
}
last_toplevel = ToplevelKind::Other;
}
} }
_ => {}
is_first = false;
self.blocks.extend(v);
} }
self.blocks.extend(v);
} }
// move plain imports to top of file // move plain imports to top of file
@ -293,8 +293,15 @@ impl Css {
OutputStyle::Compressed => { OutputStyle::Compressed => {
CompressedFormatter::default().write_css(&mut buf, self, map)?; CompressedFormatter::default().write_css(&mut buf, self, map)?;
} }
OutputStyle::Expanded => ExpandedFormatter::default().write_css(&mut buf, self, map)?, OutputStyle::Expanded => {
ExpandedFormatter::default().write_css(&mut buf, self, map)?;
if !buf.is_empty() {
writeln!(buf)?;
}
}
} }
// TODO: check for this before writing // TODO: check for this before writing
let show_charset = allows_charset && buf.iter().any(|s| !s.is_ascii()); let show_charset = allows_charset && buf.iter().any(|s| !s.is_ascii());
let out = unsafe { String::from_utf8_unchecked(buf) }; let out = unsafe { String::from_utf8_unchecked(buf) };
@ -320,8 +327,8 @@ impl Formatter for CompressedFormatter {
fn write_css(&mut self, buf: &mut Vec<u8>, css: Css, map: &CodeMap) -> SassResult<()> { fn write_css(&mut self, buf: &mut Vec<u8>, css: Css, map: &CodeMap) -> SassResult<()> {
for block in css.blocks { for block in css.blocks {
match block { match block {
Toplevel::RuleSet(selector, styles) => { Toplevel::RuleSet { selector, body, .. } => {
if styles.is_empty() { if body.is_empty() {
continue; continue;
} }
@ -335,7 +342,7 @@ impl Formatter for CompressedFormatter {
} }
write!(buf, "{{")?; write!(buf, "{{")?;
self.write_block_entry(buf, &styles)?; self.write_block_entry(buf, &body)?;
write!(buf, "}}")?; write!(buf, "}}")?;
} }
Toplevel::KeyframesRuleSet(selectors, styles) => { Toplevel::KeyframesRuleSet(selectors, styles) => {
@ -355,7 +362,7 @@ impl Formatter for CompressedFormatter {
self.write_block_entry(buf, &styles)?; self.write_block_entry(buf, &styles)?;
write!(buf, "}}")?; write!(buf, "}}")?;
} }
Toplevel::MultilineComment(..) => continue, Toplevel::Empty | Toplevel::MultilineComment(..) => continue,
Toplevel::Import(s) => { Toplevel::Import(s) => {
write!(buf, "@import {};", s)?; write!(buf, "@import {};", s)?;
} }
@ -428,7 +435,6 @@ impl Formatter for CompressedFormatter {
let value = style.value.node.to_css_string(style.value.span)?; let value = style.value.node.to_css_string(style.value.span)?;
write!(buf, "{}:{};", style.property, value)?; write!(buf, "{}:{};", style.property, value)?;
} }
Toplevel::Newline => {}
} }
} }
Ok(()) Ok(())
@ -484,46 +490,49 @@ struct ExpandedFormatter {
nesting: usize, nesting: usize,
} }
#[derive(Clone, Copy)]
struct Previous {
is_group_end: bool,
}
impl Formatter for ExpandedFormatter { impl Formatter for ExpandedFormatter {
fn write_css(&mut self, buf: &mut Vec<u8>, css: Css, map: &CodeMap) -> SassResult<()> { fn write_css(&mut self, buf: &mut Vec<u8>, css: Css, map: &CodeMap) -> SassResult<()> {
let mut has_written = false;
let padding = " ".repeat(self.nesting); let padding = " ".repeat(self.nesting);
let mut should_emit_newline = false;
self.nesting += 1; self.nesting += 1;
let mut prev: Option<Previous> = None;
for block in css.blocks { for block in css.blocks {
if block.is_invisible() {
continue;
}
let is_group_end = block.is_group_end();
if let Some(prev) = prev {
writeln!(buf)?;
if prev.is_group_end && !css.in_at_rule {
writeln!(buf)?;
}
}
match block { match block {
Toplevel::RuleSet(selector, styles) => { Toplevel::Empty => continue,
if selector.is_empty() { Toplevel::RuleSet { selector, body, .. } => {
has_written = false;
continue;
}
if styles.is_empty() {
continue;
}
has_written = true;
if should_emit_newline && !css.in_at_rule {
should_emit_newline = false;
writeln!(buf)?;
}
writeln!(buf, "{}{} {{", padding, selector)?; writeln!(buf, "{}{} {{", padding, selector)?;
for style in styles { for style in body {
writeln!(buf, "{} {}", padding, style.to_string()?)?; writeln!(buf, "{} {}", padding, style.to_string()?)?;
} }
writeln!(buf, "{}}}", padding)?; write!(buf, "{}}}", padding)?;
} }
Toplevel::KeyframesRuleSet(selector, body) => { Toplevel::KeyframesRuleSet(selector, body) => {
if body.is_empty() { if body.is_empty() {
continue; continue;
} }
has_written = true;
writeln!( writeln!(
buf, buf,
"{}{} {{", "{}{} {{",
@ -537,29 +546,16 @@ impl Formatter for ExpandedFormatter {
for style in body { for style in body {
writeln!(buf, "{} {}", padding, style.to_string()?)?; writeln!(buf, "{} {}", padding, style.to_string()?)?;
} }
writeln!(buf, "{}}}", padding)?; write!(buf, "{}}}", padding)?;
} }
Toplevel::MultilineComment(s) => { Toplevel::MultilineComment(s) => {
if has_written && should_emit_newline { write!(buf, "{}/*{}*/", padding, s)?;
writeln!(buf)?;
}
has_written = true;
should_emit_newline = false;
writeln!(buf, "{}/*{}*/", padding, s)?;
} }
Toplevel::Import(s) => { Toplevel::Import(s) => {
has_written = true; write!(buf, "{}@import {};", padding, s)?;
writeln!(buf, "{}@import {};", padding, s)?;
} }
Toplevel::UnknownAtRule(u) => { Toplevel::UnknownAtRule(u) => {
let ToplevelUnknownAtRule { params, name, body } = *u; let ToplevelUnknownAtRule { params, name, body } = *u;
if should_emit_newline {
should_emit_newline = false;
writeln!(buf)?;
}
if params.is_empty() { if params.is_empty() {
write!(buf, "{}@{}", padding, name)?; write!(buf, "{}@{}", padding, name)?;
@ -568,21 +564,18 @@ impl Formatter for ExpandedFormatter {
} }
if body.is_empty() { if body.is_empty() {
writeln!(buf, ";")?; write!(buf, ";")?;
prev = Some(Previous { is_group_end });
continue; continue;
} }
writeln!(buf, " {{")?; writeln!(buf, " {{")?;
let css = Css::from_stmts(body, true, css.allows_charset)?; let css = Css::from_stmts(body, true, css.allows_charset)?;
self.write_css(buf, css, map)?; self.write_css(buf, css, map)?;
writeln!(buf, "{}}}", padding)?; write!(buf, "\n{}}}", padding)?;
} }
Toplevel::Keyframes(k) => { Toplevel::Keyframes(k) => {
let Keyframes { rule, name, body } = *k; let Keyframes { rule, name, body } = *k;
if should_emit_newline {
should_emit_newline = false;
writeln!(buf)?;
}
write!(buf, "{}@{}", padding, rule)?; write!(buf, "{}@{}", padding, rule)?;
@ -591,21 +584,17 @@ impl Formatter for ExpandedFormatter {
} }
if body.is_empty() { if body.is_empty() {
writeln!(buf, " {{}}")?; write!(buf, " {{}}")?;
prev = Some(Previous { is_group_end });
continue; continue;
} }
writeln!(buf, " {{")?; writeln!(buf, " {{")?;
let css = Css::from_stmts(body, true, css.allows_charset)?; let css = Css::from_stmts(body, true, css.allows_charset)?;
self.write_css(buf, css, map)?; self.write_css(buf, css, map)?;
writeln!(buf, "{}}}", padding)?; write!(buf, "\n{}}}", padding)?;
} }
Toplevel::Supports { params, body } => { Toplevel::Supports { params, body } => {
if should_emit_newline {
should_emit_newline = false;
writeln!(buf)?;
}
if params.is_empty() { if params.is_empty() {
write!(buf, "{}@supports", padding)?; write!(buf, "{}@supports", padding)?;
} else { } else {
@ -613,44 +602,33 @@ impl Formatter for ExpandedFormatter {
} }
if body.is_empty() { if body.is_empty() {
writeln!(buf, ";")?; write!(buf, ";")?;
prev = Some(Previous { is_group_end });
continue; continue;
} }
writeln!(buf, " {{")?; writeln!(buf, " {{")?;
let css = Css::from_stmts(body, true, css.allows_charset)?; let css = Css::from_stmts(body, true, css.allows_charset)?;
self.write_css(buf, css, map)?; self.write_css(buf, css, map)?;
writeln!(buf, "{}}}", padding)?; write!(buf, "\n{}}}", padding)?;
} }
Toplevel::Media { Toplevel::Media {
query, query,
body, body,
inside_rule, inside_rule,
..
} => { } => {
if body.is_empty() {
continue;
}
if should_emit_newline && has_written {
should_emit_newline = false;
writeln!(buf)?;
}
writeln!(buf, "{}@media {} {{", padding, query)?; writeln!(buf, "{}@media {} {{", padding, query)?;
let css = Css::from_stmts(body, inside_rule, css.allows_charset)?; let css = Css::from_stmts(body, inside_rule, css.allows_charset)?;
self.write_css(buf, css, map)?; self.write_css(buf, css, map)?;
writeln!(buf, "{}}}", padding)?; write!(buf, "\n{}}}", padding)?;
} }
Toplevel::Style(s) => { Toplevel::Style(s) => {
writeln!(buf, "{}{}", padding, s.to_string()?)?; write!(buf, "{}{}", padding, s.to_string()?)?;
}
Toplevel::Newline => {
if has_written {
should_emit_newline = true;
}
continue;
} }
} }
prev = Some(Previous { is_group_end });
} }
self.nesting -= 1; self.nesting -= 1;

View File

@ -207,7 +207,7 @@ test!(
"a {\n color: red;\n}\n\n@media (min-width: 0px) {\n b {\n color: red;\n }\n}\nd {\n color: red;\n}\n" "a {\n color: red;\n}\n\n@media (min-width: 0px) {\n b {\n color: red;\n }\n}\nd {\n color: red;\n}\n"
); );
test!( test!(
no_newline_after_media, no_newline_after_media_when_outer,
"@media (min-width: 0px) { "@media (min-width: 0px) {
b { b {
color: red; color: red;
@ -219,6 +219,19 @@ test!(
}", }",
"@media (min-width: 0px) {\n b {\n color: red;\n }\n}\nd {\n color: red;\n}\n" "@media (min-width: 0px) {\n b {\n color: red;\n }\n}\nd {\n color: red;\n}\n"
); );
test!(
newline_after_media_when_inner,
"a {
@media (max-width: 0px) {
color: red;
}
}
a {
color: red;
}",
"@media (max-width: 0px) {\n a {\n color: red;\n }\n}\n\na {\n color: red;\n}\n"
);
error!( error!(
media_feature_missing_closing_paren, media_feature_missing_closing_paren,

44
tests/whitespace.rs Normal file
View File

@ -0,0 +1,44 @@
//! Tests that exist only to verify the printing of whitespace (largely, newlines)
#[macro_use]
mod macros;
test!(
// this is a bug in dart-sass that we must emulate.
no_newline_between_ruleset_when_last_ruleset_is_empty,
"a {
color: red;
b {
color: red;
}
c {
}
}
d {
color: red;
}",
"a {\n color: red;\n}\na b {\n color: red;\n}\nd {\n color: red;\n}\n"
);
test!(
// this is a bug in dart-sass that we must emulate.
no_newline_between_ruleset_when_last_ruleset_is_empty_from_extend,
"a {
color: red;
%b {
color: red;
}
c {
@extend %b;
}
}
d {
color: red;
}",
"a {\n color: red;\n}\na c {\n color: red;\n}\nd {\n color: red;\n}\n"
);