Completely transitioned to LALRPOP and new declare macro.

This commit is contained in:
Bodil Stokke 2018-11-14 16:22:07 +00:00
parent cb7e148310
commit 7d1e95f262
9 changed files with 385 additions and 523 deletions

View File

@ -8,8 +8,8 @@ build = "build.rs"
proc-macro = true proc-macro = true
[dependencies] [dependencies]
pom = "2.0.1"
lalrpop-util = "0.16.1" lalrpop-util = "0.16.1"
ansi_term = "0.11.0"
[build-dependencies] [build-dependencies]
lalrpop = "0.16.1" lalrpop = "0.16.1"

View File

@ -1,11 +1,10 @@
use pom::combinator::*; use proc_macro::{quote, Ident, Literal, TokenStream, TokenTree};
use pom::Parser;
use proc_macro::{quote, Group, Ident, Literal, TokenStream, TokenTree};
use config::global_attrs; use config::global_attrs;
use lexer::{Lexer, ParseError, Token}; use error::ParseError;
use lexer::{Lexer, Token};
use map::StringyMap; use map::StringyMap;
use parser::{self, *}; use parser;
// State // State
@ -327,60 +326,6 @@ impl Declare {
} }
} }
// Parser pub fn expand_declare(input: &[Token]) -> Result<Vec<Declare>, ParseError> {
fn declare_attrs<'a>() -> Combinator<impl Parser<'a, TokenTree, Output = Vec<(Ident, TokenStream)>>>
{
group().map(|group: Group| -> Vec<(Ident, TokenStream)> {
let attr = ident() - punct(':') + type_spec() - punct(',').opt();
let parser = attr.repeat(0..);
let input: Vec<TokenTree> = group.stream().into_iter().collect();
// FIXME the borrow checker won't let me use plain &input, it seems like a bug.
// It works in Rust 2018, so please get rid of this unsafe block when it stabilises.
parser
.parse(unsafe { &*(input.as_slice() as *const _) })
.unwrap()
})
}
fn declare_children<'a>() -> Combinator<impl Parser<'a, TokenTree, Output = Vec<Ident>>> {
group().map(|group: Group| {
let input: Vec<TokenTree> = group.stream().into_iter().collect();
let children = (ident() - punct(',').opt()).repeat(0..);
let result = children.parse(&input);
result.unwrap()
})
}
fn declare_traits<'a>() -> Combinator<impl Parser<'a, TokenTree, Output = Vec<TokenStream>>> {
group().map(|group: Group| {
let input: Vec<TokenTree> = group.stream().into_iter().collect();
let traits = (type_spec() - punct(',').opt()).repeat(0..);
let result = traits.parse(&input);
result.unwrap()
})
}
fn declare<'a>() -> Combinator<impl Parser<'a, TokenTree, Output = Declare>> {
(ident() + declare_attrs() + declare_children() + declare_traits().opt() + type_spec().opt())
.map(|((((name, attrs), children), traits), child_type)| {
let mut declare = Declare::new(name);
for (key, value) in attrs {
declare.attrs.insert(key, value);
}
for child in children {
declare.req_children.push(child);
}
declare.opt_children = child_type;
declare.traits = traits.unwrap_or_default();
declare
})
}
pub fn expand_declare(input: &[TokenTree]) -> pom::Result<TokenStream> {
declare().parse(input).map(|decl| decl.into_token_stream())
}
pub fn expand_declare_lalrpop(input: &[Token]) -> Result<Vec<Declare>, ParseError> {
parser::grammar::DeclarationsParser::new().parse(Lexer::new(input)) parser::grammar::DeclarationsParser::new().parse(Lexer::new(input))
} }

88
macros/src/error.rs Normal file
View File

@ -0,0 +1,88 @@
use ansi_term::Style;
use lalrpop_util::ParseError::*;
use lexer::Token;
use proc_macro::{Diagnostic, Ident, Level};
pub type ParseError = lalrpop_util::ParseError<usize, Token, HtmlParseError>;
#[derive(Debug)]
pub enum HtmlParseError {
TagMismatch { open: Ident, close: Ident },
}
fn pprint_token(token: &str) -> &str {
match token {
"BraceGroupToken" => "code block",
"LiteralToken" => "literal",
"IdentToken" => "identifier",
a => a,
}
}
fn pprint_tokens(tokens: &[String]) -> String {
let tokens: Vec<&str> = tokens.iter().map(|s| pprint_token(&s)).collect();
if tokens.len() > 1 {
let start = tokens[..tokens.len() - 1].join(", ");
let end = &tokens[tokens.len() - 1];
format!("{} or {}", start, end)
} else {
tokens[0].to_string()
}
}
fn is_in_node_position(tokens: &[String]) -> bool {
use std::collections::HashSet;
let input: HashSet<&str> = tokens.iter().map(String::as_str).collect();
let output: HashSet<&str> = ["\"<\"", "BraceGroupToken", "LiteralToken"]
.iter()
.cloned()
.collect();
input == output
}
pub fn parse_error(input: &[Token], error: &ParseError) -> Diagnostic {
match error {
InvalidToken { location } => {
let loc = &input[*location];
Diagnostic::spanned(loc.span(), Level::Error, "invalid token")
}
UnrecognizedToken {
token: None,
expected,
} => {
let msg = format!("missing {}", pprint_tokens(&expected));
Diagnostic::spanned(
input[0].span().join(input[input.len() - 1].span()).unwrap(),
Level::Error,
"unexpected end of macro",
)
.help(msg)
}
UnrecognizedToken {
token: Some((_, token, _)),
expected,
} => {
let msg = format!("expected {}", pprint_tokens(&expected));
let mut diag = Diagnostic::spanned(token.span(), Level::Error, msg);
if is_in_node_position(expected) && token.is_ident() {
// special case: you probably meant to quote that text
diag = diag.help(format!(
"text nodes need to be quoted, eg. {}",
Style::new().bold().paint("<p>\"Hello Joe!\"</p>")
))
}
diag
}
ExtraToken {
token: (_, token, _),
} => Diagnostic::spanned(token.span(), Level::Error, "superfluous token"),
User {
error: HtmlParseError::TagMismatch { open, close },
} => Diagnostic::spanned(
close.span(),
Level::Error,
format!("expected closing tag '</{}>', found '</{}>'", open, close),
)
.span_help(open.span(), "opening tag is here:"),
}
}

View File

@ -1,4 +1,5 @@
use lexer::{Token, to_stream, HtmlParseError, Keyword}; use lexer::{self, Token, to_stream};
use error::HtmlParseError;
use html::{Node, Element}; use html::{Node, Element};
use declare::Declare; use declare::Declare;
use map::StringyMap; use map::StringyMap;
@ -130,9 +131,9 @@ ParentTag: Element = <opening:OpeningTag> <children:Node*> <closing:ClosingTag>
children, children,
}) })
} else { } else {
Err(ParseError::User { error: HtmlParseError { Err(ParseError::User { error: HtmlParseError::TagMismatch {
token: closing.into(), open: name.into(),
message: format!("expected closing tag '</{}>', found '</{}>'", name.to_string(), closing_name), close: closing.into(),
}}) }})
} }
}; };
@ -260,7 +261,7 @@ extern {
type Location = usize; type Location = usize;
type Error = HtmlParseError; type Error = HtmlParseError;
enum Token { enum lexer::Token {
"<" => Token::Punct('<', _), "<" => Token::Punct('<', _),
">" => Token::Punct('>', _), ">" => Token::Punct('>', _),
"/" => Token::Punct('/', _), "/" => Token::Punct('/', _),
@ -276,8 +277,8 @@ extern {
"}" => Token::GroupClose(Delimiter::Brace, _), "}" => Token::GroupClose(Delimiter::Brace, _),
"[" => Token::GroupOpen(Delimiter::Bracket, _), "[" => Token::GroupOpen(Delimiter::Bracket, _),
"]" => Token::GroupClose(Delimiter::Bracket, _), "]" => Token::GroupClose(Delimiter::Bracket, _),
"in" => Token::Keyword(Keyword::In, _), "in" => Token::Keyword(lexer::Keyword::In, _),
"with" => Token::Keyword(Keyword::With, _), "with" => Token::Keyword(lexer::Keyword::With, _),
IdentToken => Token::Ident(_), IdentToken => Token::Ident(_),
LiteralToken => Token::Literal(_), LiteralToken => Token::Literal(_),
ParenGroupToken => Token::Group(Delimiter::Parenthesis, _), ParenGroupToken => Token::Group(Delimiter::Parenthesis, _),

View File

@ -3,7 +3,8 @@ use proc_macro::{
}; };
use config::required_children; use config::required_children;
use lexer::{Lexer, ParseError, Token}; use error::ParseError;
use lexer::{Lexer, Token};
use map::StringyMap; use map::StringyMap;
use parser::grammar; use parser::grammar;

View File

@ -1,10 +1,7 @@
use lalrpop_util::ParseError::*; use error::HtmlParseError;
use proc_macro::{ use proc_macro::{Delimiter, Group, Ident, Literal, Punct, Span, TokenStream, TokenTree};
Delimiter, Diagnostic, Group, Ident, Level, Literal, Punct, Span, TokenStream, TokenTree,
};
pub type Spanned<Tok, Loc, Error> = Result<(Loc, Tok, Loc), Error>; pub type Spanned<Tok, Loc, Error> = Result<(Loc, Tok, Loc), Error>;
pub type ParseError = lalrpop_util::ParseError<usize, Token, HtmlParseError>;
#[derive(Clone, Debug)] #[derive(Clone, Debug)]
pub enum Token { pub enum Token {
@ -107,75 +104,6 @@ pub fn keywordise(tokens: Vec<Token>) -> Vec<Token> {
.collect() .collect()
} }
#[derive(Debug)]
pub struct HtmlParseError {
pub token: Token,
pub message: String,
}
fn pprint_token(token: &str) -> &str {
match token {
"BraceGroupToken" => "code block",
"LiteralToken" => "literal",
"IdentToken" => "identifier",
a => a,
}
}
fn pprint_tokens(tokens: &[String]) -> String {
let tokens: Vec<&str> = tokens.iter().map(|s| pprint_token(&s)).collect();
if tokens.len() > 1 {
let start = tokens[..tokens.len() - 1].join(", ");
let end = &tokens[tokens.len() - 1];
format!("{} or {}", start, end)
} else {
tokens[0].to_string()
}
}
fn is_in_node_position(tokens: &[String]) -> bool {
use std::collections::HashSet;
let input: HashSet<&str> = tokens.iter().map(String::as_str).collect();
let output: HashSet<&str> = ["\"<\"", "BraceGroupToken", "LiteralToken"]
.iter()
.cloned()
.collect();
input == output
}
pub fn parse_error(input: &[Token], error: &ParseError) -> Diagnostic {
match error {
InvalidToken { location } => {
let loc = &input[*location];
Diagnostic::spanned(loc.span(), Level::Error, "invalid token")
}
UnrecognizedToken {
token: None,
expected,
} => panic!(
"unexpected end of macro: expecting {}",
pprint_tokens(&expected)
),
UnrecognizedToken {
token: Some((_, token, _)),
expected,
} => {
let mut msg = format!("expected {}", pprint_tokens(&expected));
if is_in_node_position(expected) && token.is_ident() {
// special case: you probably meant to quote that text
msg += "; looks like you forgot to put \"quotes\" around your text nodes";
}
Diagnostic::spanned(token.span(), Level::Error, msg)
}
ExtraToken {
token: (_, token, _),
} => Diagnostic::spanned(token.span(), Level::Error, "superfluous token"),
User { error } => {
Diagnostic::spanned(error.token.span(), Level::Error, error.message.to_owned())
}
}
}
pub fn to_stream<I: IntoIterator<Item = Token>>(tokens: I) -> TokenStream { pub fn to_stream<I: IntoIterator<Item = Token>>(tokens: I) -> TokenStream {
let mut stream = TokenStream::new(); let mut stream = TokenStream::new();
stream.extend(tokens.into_iter().map(TokenTree::from)); stream.extend(tokens.into_iter().map(TokenTree::from));

View File

@ -4,14 +4,15 @@
#![feature(proc_macro_diagnostic)] #![feature(proc_macro_diagnostic)]
#![feature(proc_macro_raw_ident)] #![feature(proc_macro_raw_ident)]
extern crate ansi_term;
extern crate lalrpop_util; extern crate lalrpop_util;
extern crate pom;
extern crate proc_macro; extern crate proc_macro;
use proc_macro::{TokenStream, TokenTree}; use proc_macro::{quote, TokenStream};
mod config; mod config;
mod declare; mod declare;
mod error;
mod html; mod html;
mod lexer; mod lexer;
mod map; mod map;
@ -22,22 +23,22 @@ pub fn html(input: TokenStream) -> TokenStream {
let stream = lexer::unroll_stream(input, false); let stream = lexer::unroll_stream(input, false);
let result = html::expand_html(&stream); let result = html::expand_html(&stream);
match result { match result {
Err(error) => { Err(err) => {
lexer::parse_error(&stream, &error).emit(); error::parse_error(&stream, &err).emit();
panic!("macro expansion produced errors; see above.") quote!(panic!())
} }
Ok(node) => node.into_token_stream(), Ok(node) => node.into_token_stream(),
} }
} }
#[proc_macro] #[proc_macro]
pub fn declalrpop_element(input: TokenStream) -> TokenStream { pub fn declare_elements(input: TokenStream) -> TokenStream {
let stream = lexer::keywordise(lexer::unroll_stream(input, true)); let stream = lexer::keywordise(lexer::unroll_stream(input, true));
let result = declare::expand_declare_lalrpop(&stream); let result = declare::expand_declare(&stream);
match result { match result {
Err(error) => { Err(err) => {
lexer::parse_error(&stream, &error).emit(); error::parse_error(&stream, &err).emit();
panic!("macro expansion produced errors; see above.") quote!(panic!())
} }
Ok(decls) => { Ok(decls) => {
let mut out = TokenStream::new(); let mut out = TokenStream::new();
@ -48,16 +49,3 @@ pub fn declalrpop_element(input: TokenStream) -> TokenStream {
} }
} }
} }
#[proc_macro]
pub fn declare_element(input: TokenStream) -> TokenStream {
let input: Vec<TokenTree> = input.into_iter().collect();
let result = declare::expand_declare(&input);
match result {
Err(error) => {
parser::parse_error(&input, &error).emit();
panic!("macro expansion produced errors; see above.")
}
Ok(ts) => ts,
}
}

View File

@ -1,89 +1,3 @@
use lalrpop_util::lalrpop_mod; use lalrpop_util::lalrpop_mod;
use pom::combinator::*;
use pom::{Error, Parser};
use proc_macro::{Diagnostic, Group, Ident, Level, Punct, TokenStream, TokenTree};
lalrpop_mod!(pub grammar); lalrpop_mod!(pub grammar);
pub fn punct<'a>(punct: char) -> Combinator<impl Parser<'a, TokenTree, Output = Punct>> {
comb(move |input: &[TokenTree], start| match input.get(start) {
Some(TokenTree::Punct(p)) if p.as_char() == punct => Ok((p.clone(), start + 1)),
_ => Err(Error::Mismatch {
message: format!("expected {:?}", punct),
position: start,
}),
})
}
pub fn ident<'a>() -> Combinator<impl Parser<'a, TokenTree, Output = Ident>> {
comb(|input: &[TokenTree], start| match input.get(start) {
Some(TokenTree::Ident(i)) => Ok((i.clone(), start + 1)),
_ => Err(Error::Mismatch {
message: "expected identifier".to_string(),
position: start,
}),
})
}
pub fn group<'a>() -> Combinator<impl Parser<'a, TokenTree, Output = Group>> {
comb(|input: &[TokenTree], start| match input.get(start) {
Some(TokenTree::Group(g)) => Ok((g.clone(), start + 1)),
_ => Err(Error::Mismatch {
message: "expected group".to_string(),
position: start,
}),
})
}
fn to_stream<'a, I: IntoIterator<Item = &'a TokenTree>>(tokens: I) -> TokenStream {
let mut stream = TokenStream::new();
stream.extend(tokens.into_iter().cloned());
stream
}
pub fn type_spec<'a>() -> Combinator<impl Parser<'a, TokenTree, Output = TokenStream>> {
let valid = ident().map(TokenTree::Ident)
| punct(':').map(TokenTree::Punct)
| punct('<').map(TokenTree::Punct)
| punct('>').map(TokenTree::Punct)
| punct('&').map(TokenTree::Punct)
| punct('\'').map(TokenTree::Punct);
valid.repeat(1..).collect().map(to_stream)
}
/// Turn a parser error into a proc_macro diagnostic.
pub fn parse_error(input: &[TokenTree], error: &pom::Error) -> Diagnostic {
match error {
pom::Error::Incomplete => Diagnostic::new(Level::Error, "unexpected end of macro!"),
pom::Error::Mismatch { message, position } => {
Diagnostic::spanned(input[*position].span(), Level::Error, message.as_str())
}
pom::Error::Conversion { message, position } => {
Diagnostic::spanned(input[*position].span(), Level::Error, message.as_str())
}
pom::Error::Expect {
message,
position,
inner,
} => {
let mut diag =
Diagnostic::spanned(input[*position].span(), Level::Error, message.as_str());
let child = parse_error(input, &inner);
diag.span_error(child.spans(), child.message())
}
pom::Error::Custom {
message,
position,
inner,
} => {
let mut diag =
Diagnostic::spanned(input[*position].span(), Level::Error, message.as_str());
if let Some(inner) = inner {
let child = parse_error(input, &inner);
diag.span_error(child.spans(), child.message())
} else {
diag
}
}
}
}

View File

@ -2,7 +2,7 @@
#![allow(dead_code)] #![allow(dead_code)]
use std::fmt::Display; use std::fmt::Display;
use typed_html_macros::{declalrpop_element, declare_element}; use typed_html_macros::declare_elements;
use super::types::*; use super::types::*;
@ -99,7 +99,7 @@ impl Node for TextNode {
impl FlowContent for TextNode {} impl FlowContent for TextNode {}
impl PhrasingContent for TextNode {} impl PhrasingContent for TextNode {}
declalrpop_element!{ declare_elements!{
html { html {
xmlns: Uri, xmlns: Uri,
} with [head, body]; } with [head, body];
@ -196,25 +196,22 @@ declalrpop_element!{
details { details {
open: bool, open: bool,
} in [FlowContent, SectioningContent, InteractiveContent] with [summary] FlowContent; } in [FlowContent, SectioningContent, InteractiveContent] with [summary] FlowContent;
} dfn in [FlowContent, PhrasingContent] with PhrasingContent;
div in [FlowContent] with FlowContent;
// Flow content dl in [FlowContent] with DescriptionListContent;
declare_element!(dfn {} [] [FlowContent, PhrasingContent] PhrasingContent); em in [FlowContent, PhrasingContent] with PhrasingContent;
declare_element!(div {} [] [FlowContent] FlowContent); embed {
declare_element!(dl {} [] [FlowContent] DescriptionListContent);
declare_element!(em {} [] [FlowContent, PhrasingContent] PhrasingContent);
declare_element!(embed {
height: usize, height: usize,
src: Uri, src: Uri,
type: Mime, type: Mime,
width: usize, width: usize,
} [] [FlowContent, PhrasingContent, EmbeddedContent, InteractiveContent]); } in [FlowContent, PhrasingContent, EmbeddedContent, InteractiveContent];
// FIXME the legend attribute should be optional // FIXME the legend attribute should be optional
declare_element!(fieldset {} [legend] [FlowContent, SectioningContent, FormContent] FlowContent); fieldset in [FlowContent, SectioningContent, FormContent] with [legend] FlowContent;
// FIXME the figcaption attribute should be optional // FIXME the figcaption attribute should be optional
declare_element!(figure {} [figcaption] [FlowContent, SectioningContent] FlowContent); figure in [FlowContent, SectioningContent] with [figcaption] FlowContent;
declare_element!(footer {} [] [FlowContent] FlowContent); footer in [FlowContent] with FlowContent;
declare_element!(form { form {
accept-charset: SpacedList<CharacterEncoding>, accept-charset: SpacedList<CharacterEncoding>,
action: Uri, action: Uri,
autocomplete: OnOff, autocomplete: OnOff,
@ -223,18 +220,18 @@ declare_element!(form {
name: Id, name: Id,
novalidate: bool, novalidate: bool,
target: Target, target: Target,
} [] [FlowContent] FlowContent); } in [FlowContent] with FlowContent;
declare_element!(h1 {} [] [FlowContent, HeadingContent, HGroupContent] PhrasingContent); h1 in [FlowContent, HeadingContent, HGroupContent] with PhrasingContent;
declare_element!(h2 {} [] [FlowContent, HeadingContent, HGroupContent] PhrasingContent); h2 in [FlowContent, HeadingContent, HGroupContent] with PhrasingContent;
declare_element!(h3 {} [] [FlowContent, HeadingContent, HGroupContent] PhrasingContent); h3 in [FlowContent, HeadingContent, HGroupContent] with PhrasingContent;
declare_element!(h4 {} [] [FlowContent, HeadingContent, HGroupContent] PhrasingContent); h4 in [FlowContent, HeadingContent, HGroupContent] with PhrasingContent;
declare_element!(h5 {} [] [FlowContent, HeadingContent, HGroupContent] PhrasingContent); h5 in [FlowContent, HeadingContent, HGroupContent] with PhrasingContent;
declare_element!(h6 {} [] [FlowContent, HeadingContent, HGroupContent] PhrasingContent); h6 in [FlowContent, HeadingContent, HGroupContent] with PhrasingContent;
declare_element!(header {} [] [FlowContent] FlowContent); header in [FlowContent] with FlowContent;
declare_element!(hgroup {} [] [FlowContent, HeadingContent] HGroupContent); hgroup in [FlowContent, HeadingContent] with HGroupContent;
declare_element!(hr {} [] [FlowContent]); hr in [FlowContent];
declare_element!(i {} [] [FlowContent, PhrasingContent] PhrasingContent); i in [FlowContent, PhrasingContent] with PhrasingContent;
declare_element!(iframe { iframe {
allow: FeaturePolicy, allow: FeaturePolicy,
allowfullscreen: bool, allowfullscreen: bool,
allowpaymentrequest: bool, allowpaymentrequest: bool,
@ -245,8 +242,8 @@ declare_element!(iframe {
src: Uri, src: Uri,
srcdoc: Uri, srcdoc: Uri,
width: usize, width: usize,
} [] [FlowContent, PhrasingContent, EmbeddedContent, InteractiveContent] FlowContent); } in [FlowContent, PhrasingContent, EmbeddedContent, InteractiveContent] with FlowContent;
declare_element!(img { img {
alt: String, alt: String,
crossorigin: CrossOrigin, crossorigin: CrossOrigin,
decoding: ImageDecoding, decoding: ImageDecoding,
@ -257,8 +254,8 @@ declare_element!(img {
srcset: String, // FIXME this is much more complicated srcset: String, // FIXME this is much more complicated
usemap: String, // FIXME should be a fragment starting with '#' usemap: String, // FIXME should be a fragment starting with '#'
width: usize, width: usize,
} [] [FlowContent, PhrasingContent, EmbeddedContent]); } in [FlowContent, PhrasingContent, EmbeddedContent];
declare_element!(input { input {
autocomplete: String, autocomplete: String,
autofocus: bool, autofocus: bool,
disabled: bool, disabled: bool,
@ -269,23 +266,23 @@ declare_element!(input {
tabindex: usize, tabindex: usize,
type: InputType, type: InputType,
value: String, value: String,
} [] [FlowContent, FormContent, PhrasingContent]); } in [FlowContent, FormContent, PhrasingContent];
declare_element!(ins { ins {
cite: Uri, cite: Uri,
datetime: Datetime, datetime: Datetime,
} [] [FlowContent, PhrasingContent] FlowContent); } in [FlowContent, PhrasingContent] with FlowContent;
declare_element!(kbd {} [] [FlowContent, PhrasingContent] PhrasingContent); kbd in [FlowContent, PhrasingContent] with PhrasingContent;
declare_element!(label { label {
for: Id, for: Id,
form: Id, form: Id,
} [] [FlowContent, PhrasingContent, InteractiveContent, FormContent] PhrasingContent); } in [FlowContent, PhrasingContent, InteractiveContent, FormContent] with PhrasingContent;
declare_element!(main {} [] [FlowContent] FlowContent); main in [FlowContent] with FlowContent;
declare_element!(map { map {
name: Id, name: Id,
} [] [FlowContent, PhrasingContent] MapContent); } in [FlowContent, PhrasingContent] with MapContent;
declare_element!(mark {} [] [FlowContent, PhrasingContent] PhrasingContent); mark in [FlowContent, PhrasingContent] with PhrasingContent;
// TODO the <math> element // TODO the <math> element
declare_element!(meter { meter {
value: isize, value: isize,
min: isize, min: isize,
max: isize, max: isize,
@ -293,10 +290,10 @@ declare_element!(meter {
high: isize, high: isize,
optimum: isize, optimum: isize,
form: Id, form: Id,
} [] [FlowContent, PhrasingContent] PhrasingContent); } in [FlowContent, PhrasingContent] with PhrasingContent;
declare_element!(nav {} [] [FlowContent, SectioningContent] PhrasingContent); nav in [FlowContent, SectioningContent] with PhrasingContent;
declare_element!(noscript {} [] [MetadataContent, FlowContent, PhrasingContent] Node); noscript in [MetadataContent, FlowContent, PhrasingContent] with Node;
declare_element!(object { object {
data: Uri, data: Uri,
form: Id, form: Id,
height: usize, height: usize,
@ -305,30 +302,30 @@ declare_element!(object {
typemustmatch: bool, typemustmatch: bool,
usemap: String, // TODO should be a fragment starting with '#' usemap: String, // TODO should be a fragment starting with '#'
width: usize, width: usize,
} [] [FlowContent, PhrasingContent, EmbeddedContent, InteractiveContent, FormContent] Element_param); } in [FlowContent, PhrasingContent, EmbeddedContent, InteractiveContent, FormContent] with Element_param;
declare_element!(ol { ol {
reversed: bool, reversed: bool,
start: isize, start: isize,
type: OrderedListType, type: OrderedListType,
} [] [FlowContent] Element_li); } in [FlowContent] with Element_li;
declare_element!(output { output {
for: SpacedSet<Id>, for: SpacedSet<Id>,
form: Id, form: Id,
name: Id, name: Id,
} [] [FlowContent, PhrasingContent, FormContent] PhrasingContent); } in [FlowContent, PhrasingContent, FormContent] with PhrasingContent;
declare_element!(p {} [] [FlowContent] PhrasingContent); p in [FlowContent] with PhrasingContent;
declare_element!(pre {} [] [FlowContent] PhrasingContent); pre in [FlowContent] with PhrasingContent;
declare_element!(progress { progress {
max: f64, max: f64,
value: f64, value: f64,
} [] [FlowContent, PhrasingContent] PhrasingContent); } in [FlowContent, PhrasingContent] with PhrasingContent;
declare_element!(q { q {
cite: Uri, cite: Uri,
} [] [FlowContent, PhrasingContent] PhrasingContent); } in [FlowContent, PhrasingContent] with PhrasingContent;
declare_element!(ruby {} [] [FlowContent, PhrasingContent] PhrasingContent); ruby in [FlowContent, PhrasingContent] with PhrasingContent;
declare_element!(s {} [] [FlowContent, PhrasingContent] PhrasingContent); s in [FlowContent, PhrasingContent] with PhrasingContent;
declare_element!(samp {} [] [FlowContent, PhrasingContent] PhrasingContent); samp in [FlowContent, PhrasingContent] with PhrasingContent;
declare_element!(script { script {
async: bool, async: bool,
crossorigin: CrossOrigin, crossorigin: CrossOrigin,
defer: bool, defer: bool,
@ -338,9 +335,9 @@ declare_element!(script {
src: Uri, src: Uri,
text: String, text: String,
type: String, // TODO could be an enum type: String, // TODO could be an enum
} [] [MetadataContent, FlowContent, PhrasingContent, TableColumnContent] TextNode); } in [MetadataContent, FlowContent, PhrasingContent, TableColumnContent] with TextNode;
declare_element!(section {} [] [FlowContent, SectioningContent] FlowContent); section in [FlowContent, SectioningContent] with FlowContent;
declare_element!(select { select {
autocomplete: String, autocomplete: String,
autofocus: bool, autofocus: bool,
disabled: bool, disabled: bool,
@ -349,16 +346,15 @@ declare_element!(select {
name: Id, name: Id,
required: bool, required: bool,
size: usize, size: usize,
} [] [FlowContent, PhrasingContent, InteractiveContent, FormContent] SelectContent); } in [FlowContent, PhrasingContent, InteractiveContent, FormContent] with SelectContent;
declare_element!(small {} [] [FlowContent, PhrasingContent] PhrasingContent); small in [FlowContent, PhrasingContent] with PhrasingContent;
declare_element!(span {} [] [FlowContent, PhrasingContent] PhrasingContent); span in [FlowContent, PhrasingContent] with PhrasingContent;
declare_element!(strong {} [] [FlowContent, PhrasingContent] PhrasingContent); strong in [FlowContent, PhrasingContent] with PhrasingContent;
declare_element!(sub {} [] [FlowContent, PhrasingContent] PhrasingContent); sub in [FlowContent, PhrasingContent] with PhrasingContent;
declare_element!(sup {} [] [FlowContent, PhrasingContent] PhrasingContent); sup in [FlowContent, PhrasingContent] with PhrasingContent;
// TODO the <svg> element table in [FlowContent] with TableContent;
declare_element!(table {} [] [FlowContent] TableContent); template in [MetadataContent, FlowContent, PhrasingContent, TableColumnContent] with Node;
declare_element!(template {} [] [MetadataContent, FlowContent, PhrasingContent, TableColumnContent] Node); textarea {
declare_element!(textarea {
autocomplete: OnOff, autocomplete: OnOff,
autofocus: bool, autofocus: bool,
cols: usize, cols: usize,
@ -373,17 +369,17 @@ declare_element!(textarea {
rows: usize, rows: usize,
spellcheck: BoolOrDefault, spellcheck: BoolOrDefault,
wrap: Wrap, wrap: Wrap,
} [] [FlowContent, PhrasingContent, InteractiveContent, FormContent] TextNode); } in [FlowContent, PhrasingContent, InteractiveContent, FormContent] with TextNode;
declare_element!(time { time {
datetime: Datetime, datetime: Datetime,
} [] [FlowContent, PhrasingContent] PhrasingContent); } in [FlowContent, PhrasingContent] with PhrasingContent;
declare_element!(ul {} [] [FlowContent] Element_li); ul in [FlowContent] with Element_li;
declare_element!(var {} [] [FlowContent, PhrasingContent] PhrasingContent); var in [FlowContent, PhrasingContent] with PhrasingContent;
declare_element!(video {} [] [FlowContent, PhrasingContent, EmbeddedContent] MediaContent); video in [FlowContent, PhrasingContent, EmbeddedContent] with MediaContent;
declare_element!(wbr {} [] [FlowContent, PhrasingContent]); wbr in [FlowContent, PhrasingContent];
// Non-group elements // Non-group elements
declare_element!(area { area {
alt: String, alt: String,
coords: String, // TODO could perhaps be validated coords: String, // TODO could perhaps be validated
download: bool, download: bool,
@ -393,67 +389,67 @@ declare_element!(area {
rel: SpacedSet<LinkType>, rel: SpacedSet<LinkType>,
shape: AreaShape, shape: AreaShape,
target: Target, target: Target,
} [] [MapContent]); } in [MapContent];
declare_element!(caption {} [] [TableContent] FlowContent); caption in [TableContent] with FlowContent;
declare_element!(col { col {
span: usize, span: usize,
} [] []); };
declare_element!(colgroup { colgroup {
span: usize, span: usize,
} [] [TableContent] Element_col); } in [TableContent] with Element_col;
declare_element!(dd {} [] [DescriptionListContent] FlowContent); dd in [DescriptionListContent] with FlowContent;
declare_element!(dt {} [] [DescriptionListContent] FlowContent); dt in [DescriptionListContent] with FlowContent;
declare_element!(figcaption {} [] [] FlowContent); figcaption with FlowContent;
declare_element!(legend {} [] [] PhrasingContent); legend with PhrasingContent;
declare_element!(li { li {
value: isize, value: isize,
} [] [] FlowContent); } with FlowContent;
declare_element!(option { option {
disabled: bool, disabled: bool,
label: String, label: String,
selected: bool, selected: bool,
value: String, value: String,
} [] [SelectContent] TextNode); } in [SelectContent] with TextNode;
declare_element!(optgroup { optgroup {
disabled: bool, disabled: bool,
label: String, label: String,
} [] [SelectContent] Element_option); } in [SelectContent] with Element_option;
declare_element!(param { param {
name: String, name: String,
value: String, value: String,
} [] []); };
declare_element!(source { source {
src: Uri, src: Uri,
type: Mime, type: Mime,
} [] [MediaContent]); } in [MediaContent];
declare_element!(summary {} [] [] PhrasingContent); summary with PhrasingContent;
declare_element!(tbody {} [] [TableContent] Element_tr); tbody in [TableContent] with Element_tr;
declare_element!(td { td {
colspan: usize, colspan: usize,
headers: SpacedSet<Id>, headers: SpacedSet<Id>,
rowspan: usize, rowspan: usize,
} [] [TableColumnContent] FlowContent); } in [TableColumnContent] with FlowContent;
declare_element!(tfoot {} [] [TableContent] Element_tr); tfoot in [TableContent] with Element_tr;
declare_element!(th { th {
abbr: String, abbr: String,
colspan: usize, colspan: usize,
headers: SpacedSet<Id>, headers: SpacedSet<Id>,
rowspan: usize, rowspan: usize,
scope: TableHeaderScope, scope: TableHeaderScope,
} [] [TableColumnContent] FlowContent); } in [TableColumnContent] with FlowContent;
declare_element!(thead {} [] [TableContent] Element_tr); thead in [TableContent] with Element_tr;
declare_element!(tr {} [] [TableContent] TableColumnContent); tr in [TableContent] with TableColumnContent;
declare_element!(track { track {
default: bool, default: bool,
kind: VideoKind, kind: VideoKind,
label: String, label: String,
src: Uri, src: Uri,
srclang: LanguageTag, srclang: LanguageTag,
} [] [MediaContent]); } in [MediaContent];
// Don't @ me // Don't @ me
declare_element!(blink {} [] [FlowContent, PhrasingContent] PhrasingContent); blink in [FlowContent, PhrasingContent] with PhrasingContent;
declare_element!(marquee { marquee {
behavior: String, // FIXME enum behavior: String, // FIXME enum
bgcolor: String, // FIXME colour bgcolor: String, // FIXME colour
direction: String, // FIXME direction enum direction: String, // FIXME direction enum
@ -465,4 +461,5 @@ declare_element!(marquee {
truespeed: bool, truespeed: bool,
vspace: String, // FIXME size vspace: String, // FIXME size
width: String, // FIXME size width: String, // FIXME size
} [] [FlowContent, PhrasingContent] PhrasingContent); } in [FlowContent, PhrasingContent] with PhrasingContent;
}