yew_macro/html_tree/
html_if.rs1use proc_macro2::TokenStream;
2use quote::{quote_spanned, ToTokens};
3use syn::buffer::Cursor;
4use syn::parse::{Parse, ParseStream};
5use syn::spanned::Spanned;
6use syn::{parse_quote, Expr, Token};
7
8use super::HtmlRootBraced;
9use crate::PeekValue;
10
11pub struct HtmlIf {
12    if_token: Token![if],
13    cond: Box<Expr>,
14    then_branch: HtmlRootBraced,
15    else_branch: Option<(Token![else], Box<HtmlRootBracedOrIf>)>,
16}
17
18impl PeekValue<()> for HtmlIf {
19    fn peek(cursor: Cursor) -> Option<()> {
20        let (ident, _) = cursor.ident()?;
21        (ident == "if").then_some(())
22    }
23}
24
25impl Parse for HtmlIf {
26    fn parse(input: ParseStream) -> syn::Result<Self> {
27        let if_token = input.parse()?;
28        let cond = Box::new(input.call(Expr::parse_without_eager_brace)?);
29        match &*cond {
30            Expr::Block(syn::ExprBlock { block, .. }) if block.stmts.is_empty() => {
31                return Err(syn::Error::new(
32                    cond.span(),
33                    "missing condition for `if` expression",
34                ))
35            }
36            _ => {}
37        }
38        if input.is_empty() {
39            return Err(syn::Error::new(
40                cond.span(),
41                "this `if` expression has a condition, but no block",
42            ));
43        }
44
45        let then_branch = input.parse()?;
46        let else_branch = input
47            .parse::<Token![else]>()
48            .ok()
49            .map(|else_token| {
50                if input.is_empty() {
51                    return Err(syn::Error::new(
52                        else_token.span(),
53                        "expected block or `if` after `else`",
54                    ));
55                }
56
57                input.parse().map(|branch| (else_token, branch))
58            })
59            .transpose()?;
60
61        Ok(HtmlIf {
62            if_token,
63            cond,
64            then_branch,
65            else_branch,
66        })
67    }
68}
69
70impl ToTokens for HtmlIf {
71    fn to_tokens(&self, tokens: &mut TokenStream) {
72        let Self {
73            if_token,
74            cond,
75            then_branch,
76            else_branch,
77        } = self;
78        let default_else_branch = parse_quote! { {} };
79        let else_branch = else_branch
80            .as_ref()
81            .map(|(_, branch)| branch)
82            .unwrap_or(&default_else_branch);
83        let new_tokens = quote_spanned! {if_token.span()=>
84            if #cond #then_branch else #else_branch
85        };
86
87        tokens.extend(new_tokens);
88    }
89}
90
91pub enum HtmlRootBracedOrIf {
92    Branch(HtmlRootBraced),
93    If(HtmlIf),
94}
95
96impl PeekValue<()> for HtmlRootBracedOrIf {
97    fn peek(cursor: Cursor) -> Option<()> {
98        HtmlRootBraced::peek(cursor).or_else(|| HtmlIf::peek(cursor))
99    }
100}
101
102impl Parse for HtmlRootBracedOrIf {
103    fn parse(input: ParseStream) -> syn::Result<Self> {
104        if HtmlRootBraced::peek(input.cursor()).is_some() {
105            input.parse().map(Self::Branch)
106        } else {
107            input.parse().map(Self::If)
108        }
109    }
110}
111
112impl ToTokens for HtmlRootBracedOrIf {
113    fn to_tokens(&self, tokens: &mut TokenStream) {
114        match self {
115            Self::Branch(x) => x.to_tokens(tokens),
116            Self::If(x) => x.to_tokens(tokens),
117        }
118    }
119}