yew-macro 0.23.0

A framework for making client-side single-page apps
Documentation
use std::sync::{Arc, Mutex};

use proc_macro_error::emit_error;
use syn::spanned::Spanned;
use syn::visit_mut::VisitMut;
use syn::{
    parse_quote_spanned, visit_mut, Expr, ExprCall, ExprClosure, ExprForLoop, ExprIf, ExprLoop,
    ExprMatch, ExprWhile, Ident, Item,
};

#[derive(Debug)]
pub struct BodyRewriter {
    branch_lock: Arc<Mutex<()>>,
    ctx_ident: Ident,
}

impl BodyRewriter {
    pub fn new(ctx_ident: Ident) -> Self {
        Self {
            branch_lock: Arc::default(),
            ctx_ident,
        }
    }

    fn is_branched(&self) -> bool {
        self.branch_lock.try_lock().is_err()
    }

    fn with_branch<F, O>(&mut self, f: F) -> O
    where
        F: FnOnce(&mut BodyRewriter) -> O,
    {
        let branch_lock = self.branch_lock.clone();
        let _branched = branch_lock.try_lock();
        f(self)
    }
}

impl VisitMut for BodyRewriter {
    fn visit_expr_call_mut(&mut self, i: &mut ExprCall) {
        let ctx_ident = &self.ctx_ident;

        // Only rewrite hook calls.
        if let Expr::Path(ref m) = &*i.func {
            if let Some(m) = m.path.segments.last().as_ref().map(|m| &m.ident) {
                if m.to_string().starts_with("use_") {
                    if self.is_branched() {
                        emit_error!(
                            m,
                            "hooks cannot be called at this position.";
                            help = "move hooks to the top-level of your function.";
                            note = "see: https://yew.rs/docs/next/concepts/function-components/hooks"
                        );
                    } else {
                        *i = parse_quote_spanned! { i.span() => ::yew::functional::Hook::run(#i, #ctx_ident) };
                    }

                    return;
                }
            }
        }

        visit_mut::visit_expr_call_mut(self, i);
    }

    fn visit_expr_mut(&mut self, i: &mut Expr) {
        let ctx_ident = &self.ctx_ident;

        match &mut *i {
            Expr::Macro(m) => {
                if let Some(ident) = m.mac.path.segments.last().as_ref().map(|m| &m.ident) {
                    if ident.to_string().starts_with("use_") {
                        if self.is_branched() {
                            emit_error!(
                                ident,
                                "hooks cannot be called at this position.";
                                help = "move hooks to the top-level of your function.";
                                note = "see: https://yew.rs/docs/next/concepts/function-components/hooks"
                            );
                        } else {
                            *i = parse_quote_spanned! { i.span() => ::yew::functional::Hook::run(#i, #ctx_ident) };
                        }
                    } else {
                        visit_mut::visit_expr_macro_mut(self, m);
                    }
                }
            }
            _ => visit_mut::visit_expr_mut(self, i),
        }
    }

    fn visit_expr_closure_mut(&mut self, i: &mut ExprClosure) {
        self.with_branch(move |m| visit_mut::visit_expr_closure_mut(m, i))
    }

    fn visit_expr_if_mut(&mut self, i: &mut ExprIf) {
        for it in &mut i.attrs {
            visit_mut::visit_attribute_mut(self, it);
        }

        visit_mut::visit_expr_mut(self, &mut i.cond);

        self.with_branch(|m| visit_mut::visit_block_mut(m, &mut i.then_branch));

        if let Some(it) = &mut i.else_branch {
            self.with_branch(|m| visit_mut::visit_expr_mut(m, &mut (it).1));
        }
    }

    fn visit_expr_loop_mut(&mut self, i: &mut ExprLoop) {
        self.with_branch(|m| visit_mut::visit_expr_loop_mut(m, i));
    }

    fn visit_expr_for_loop_mut(&mut self, i: &mut ExprForLoop) {
        for it in &mut i.attrs {
            visit_mut::visit_attribute_mut(self, it);
        }
        if let Some(it) = &mut i.label {
            visit_mut::visit_label_mut(self, it);
        }
        visit_mut::visit_pat_mut(self, &mut i.pat);
        visit_mut::visit_expr_mut(self, &mut i.expr);

        self.with_branch(|m| visit_mut::visit_block_mut(m, &mut i.body));
    }

    fn visit_expr_match_mut(&mut self, i: &mut ExprMatch) {
        for it in &mut i.attrs {
            visit_mut::visit_attribute_mut(self, it);
        }

        visit_mut::visit_expr_mut(self, &mut i.expr);

        self.with_branch(|m| {
            for it in &mut i.arms {
                visit_mut::visit_arm_mut(m, it);
            }
        });
    }

    fn visit_expr_while_mut(&mut self, i: &mut ExprWhile) {
        for it in &mut i.attrs {
            visit_mut::visit_attribute_mut(self, it);
        }
        if let Some(it) = &mut i.label {
            visit_mut::visit_label_mut(self, it);
        }

        self.with_branch(|m| visit_mut::visit_expr_mut(m, &mut i.cond));
        self.with_branch(|m| visit_mut::visit_block_mut(m, &mut i.body));
    }

    fn visit_item_mut(&mut self, _i: &mut Item) {
        // We don't do anything for items.
        // for components / hooks in other components / hooks, apply the attribute again.
    }
}