2020-07-04 20:50:53 -04:00
|
|
|
use std::fmt;
|
|
|
|
|
|
|
|
use peekmore::PeekMore;
|
|
|
|
|
|
|
|
use crate::{
|
|
|
|
atrule::keyframes::{Keyframes, KeyframesSelector},
|
|
|
|
error::SassResult,
|
|
|
|
parse::Stmt,
|
|
|
|
utils::eat_whole_number,
|
|
|
|
Token,
|
|
|
|
};
|
|
|
|
|
2020-07-05 10:13:49 -04:00
|
|
|
use super::{common::ContextFlags, Parser};
|
2020-07-04 20:50:53 -04:00
|
|
|
|
|
|
|
impl fmt::Display for KeyframesSelector {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
match self {
|
|
|
|
KeyframesSelector::To => f.write_str("to"),
|
|
|
|
KeyframesSelector::From => f.write_str("from"),
|
|
|
|
KeyframesSelector::Percent(p) => write!(f, "{}%", p),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct KeyframesSelectorParser<'a, 'b> {
|
|
|
|
parser: &'a mut Parser<'b>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a, 'b> KeyframesSelectorParser<'a, 'b> {
|
|
|
|
pub fn new(parser: &'a mut Parser<'b>) -> Self {
|
|
|
|
Self { parser }
|
|
|
|
}
|
|
|
|
|
|
|
|
fn parse_keyframes_selector(&mut self) -> SassResult<Vec<KeyframesSelector>> {
|
|
|
|
let mut selectors = Vec::new();
|
|
|
|
self.parser.whitespace_or_comment();
|
|
|
|
while let Some(tok) = self.parser.toks.peek().cloned() {
|
|
|
|
match tok.kind {
|
|
|
|
't' | 'T' => {
|
|
|
|
let mut ident = self.parser.parse_identifier()?;
|
|
|
|
ident.node.make_ascii_lowercase();
|
|
|
|
if ident.node == "to" {
|
|
|
|
selectors.push(KeyframesSelector::To)
|
|
|
|
} else {
|
|
|
|
return Err(("Expected \"to\" or \"from\".", tok.pos).into());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
'f' | 'F' => {
|
|
|
|
let mut ident = self.parser.parse_identifier()?;
|
|
|
|
ident.node.make_ascii_lowercase();
|
|
|
|
if ident.node == "from" {
|
|
|
|
selectors.push(KeyframesSelector::From)
|
|
|
|
} else {
|
|
|
|
return Err(("Expected \"to\" or \"from\".", tok.pos).into());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
'0'..='9' => {
|
2020-07-29 07:18:47 -04:00
|
|
|
let mut num = eat_whole_number(self.parser.toks);
|
|
|
|
|
|
|
|
if let Some(Token { kind: '.', .. }) = self.parser.toks.peek() {
|
|
|
|
self.parser.toks.next();
|
|
|
|
num.push('.');
|
|
|
|
num.push_str(&eat_whole_number(self.parser.toks));
|
|
|
|
}
|
|
|
|
|
2020-08-06 21:36:11 -04:00
|
|
|
self.parser.expect_char('%')?;
|
|
|
|
|
2020-07-04 20:50:53 -04:00
|
|
|
selectors.push(KeyframesSelector::Percent(num.into_boxed_str()));
|
|
|
|
}
|
|
|
|
'{' => break,
|
|
|
|
'\\' => todo!("escaped chars in @keyframes selector"),
|
|
|
|
_ => return Err(("Expected \"to\" or \"from\".", tok.pos).into()),
|
|
|
|
}
|
|
|
|
self.parser.whitespace_or_comment();
|
|
|
|
if let Some(Token { kind: ',', .. }) = self.parser.toks.peek() {
|
|
|
|
self.parser.toks.next();
|
|
|
|
self.parser.whitespace_or_comment();
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(selectors)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Parser<'a> {
|
|
|
|
fn parse_keyframes_name(&mut self) -> SassResult<String> {
|
|
|
|
let mut name = String::new();
|
|
|
|
self.whitespace_or_comment();
|
|
|
|
while let Some(tok) = self.toks.next() {
|
|
|
|
match tok.kind {
|
|
|
|
'#' => {
|
|
|
|
if let Some(Token { kind: '{', .. }) = self.toks.peek() {
|
|
|
|
self.toks.next();
|
|
|
|
name.push_str(&self.parse_interpolation_as_string()?);
|
|
|
|
} else {
|
|
|
|
name.push('#');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
' ' | '\n' | '\t' => {
|
|
|
|
self.whitespace();
|
|
|
|
name.push(' ');
|
|
|
|
}
|
|
|
|
'{' => {
|
2020-07-05 17:59:48 +08:00
|
|
|
// todo: we can avoid the reallocation by trimming before emitting
|
|
|
|
// (in `output.rs`)
|
|
|
|
return Ok(name.trim().to_string());
|
2020-07-04 20:50:53 -04:00
|
|
|
}
|
|
|
|
_ => name.push(tok.kind),
|
|
|
|
}
|
|
|
|
}
|
2020-07-05 17:59:48 +08:00
|
|
|
Err(("expected \"{\".", self.span_before).into())
|
2020-07-04 20:50:53 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
pub(super) fn parse_keyframes_selector(
|
|
|
|
&mut self,
|
|
|
|
mut string: String,
|
|
|
|
) -> SassResult<Vec<KeyframesSelector>> {
|
|
|
|
let mut span = if let Some(tok) = self.toks.peek() {
|
|
|
|
tok.pos()
|
|
|
|
} else {
|
|
|
|
return Err(("expected \"{\".", self.span_before).into());
|
|
|
|
};
|
|
|
|
|
|
|
|
self.span_before = span;
|
|
|
|
|
|
|
|
while let Some(tok) = self.toks.next() {
|
|
|
|
span = span.merge(tok.pos());
|
|
|
|
match tok.kind {
|
|
|
|
'#' => {
|
|
|
|
if let Some(Token { kind: '{', .. }) = self.toks.peek().cloned() {
|
|
|
|
self.toks.next();
|
|
|
|
string.push_str(&self.parse_interpolation()?.to_css_string(span)?);
|
|
|
|
} else {
|
|
|
|
string.push('#');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
',' => {
|
|
|
|
while let Some(c) = string.pop() {
|
|
|
|
if c == ' ' || c == ',' {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
string.push(c);
|
|
|
|
string.push(',');
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
'/' => {
|
|
|
|
if self.toks.peek().is_none() {
|
|
|
|
return Err(("Expected selector.", tok.pos()).into());
|
|
|
|
}
|
|
|
|
self.parse_comment()?;
|
|
|
|
self.whitespace();
|
|
|
|
string.push(' ');
|
|
|
|
}
|
|
|
|
'{' => {
|
2020-07-05 17:59:48 +08:00
|
|
|
let sel_toks: Vec<Token> =
|
|
|
|
string.chars().map(|x| Token::new(span, x)).collect();
|
|
|
|
|
|
|
|
let selector = KeyframesSelectorParser::new(&mut Parser {
|
|
|
|
toks: &mut sel_toks.into_iter().peekmore(),
|
|
|
|
map: self.map,
|
|
|
|
path: self.path,
|
|
|
|
scopes: self.scopes,
|
|
|
|
global_scope: self.global_scope,
|
|
|
|
super_selectors: self.super_selectors,
|
|
|
|
span_before: self.span_before,
|
|
|
|
content: self.content,
|
2020-07-05 19:16:44 +08:00
|
|
|
flags: self.flags,
|
2020-07-05 17:59:48 +08:00
|
|
|
at_root: self.at_root,
|
|
|
|
at_root_has_selector: self.at_root_has_selector,
|
|
|
|
extender: self.extender,
|
2020-07-08 17:52:37 -04:00
|
|
|
content_scopes: self.content_scopes,
|
2020-07-15 12:37:19 +01:00
|
|
|
options: self.options,
|
2020-07-25 19:22:12 -04:00
|
|
|
modules: self.modules,
|
2020-08-06 21:00:34 -04:00
|
|
|
module_config: self.module_config,
|
2020-07-05 17:59:48 +08:00
|
|
|
})
|
|
|
|
.parse_keyframes_selector()?;
|
|
|
|
|
|
|
|
return Ok(selector);
|
2020-07-04 20:50:53 -04:00
|
|
|
}
|
|
|
|
c => string.push(c),
|
|
|
|
}
|
|
|
|
}
|
2020-07-05 17:59:48 +08:00
|
|
|
Err(("expected \"{\".", span).into())
|
2020-07-04 20:50:53 -04:00
|
|
|
}
|
|
|
|
|
2020-07-29 07:18:47 -04:00
|
|
|
pub(super) fn parse_keyframes(&mut self, rule: String) -> SassResult<Stmt> {
|
2020-08-02 23:00:00 -04:00
|
|
|
if self.flags.in_function() {
|
|
|
|
return Err(("This at-rule is not allowed here.", self.span_before).into());
|
|
|
|
}
|
|
|
|
|
2020-07-04 20:50:53 -04:00
|
|
|
let name = self.parse_keyframes_name()?;
|
|
|
|
|
|
|
|
self.whitespace();
|
|
|
|
|
|
|
|
let body = Parser {
|
|
|
|
toks: self.toks,
|
|
|
|
map: self.map,
|
|
|
|
path: self.path,
|
|
|
|
scopes: self.scopes,
|
|
|
|
global_scope: self.global_scope,
|
|
|
|
super_selectors: self.super_selectors,
|
|
|
|
span_before: self.span_before,
|
|
|
|
content: self.content,
|
2020-07-05 10:13:49 -04:00
|
|
|
flags: self.flags | ContextFlags::IN_KEYFRAMES,
|
2020-07-04 20:50:53 -04:00
|
|
|
at_root: false,
|
|
|
|
at_root_has_selector: self.at_root_has_selector,
|
|
|
|
extender: self.extender,
|
2020-07-08 17:52:37 -04:00
|
|
|
content_scopes: self.content_scopes,
|
2020-07-15 12:37:19 +01:00
|
|
|
options: self.options,
|
2020-07-25 19:22:12 -04:00
|
|
|
modules: self.modules,
|
2020-08-06 21:00:34 -04:00
|
|
|
module_config: self.module_config,
|
2020-07-04 20:50:53 -04:00
|
|
|
}
|
|
|
|
.parse_stmt()?;
|
|
|
|
|
2020-07-29 07:18:47 -04:00
|
|
|
Ok(Stmt::Keyframes(Box::new(Keyframes { rule, name, body })))
|
2020-07-04 20:50:53 -04:00
|
|
|
}
|
|
|
|
}
|