2018-05-22 21:06:32 -07:00
|
|
|
use proc_macro2::TokenStream;
|
2018-05-20 19:34:52 -07:00
|
|
|
use quote::ToTokens;
|
2018-01-08 21:49:09 -08:00
|
|
|
use syn::token;
|
2017-02-20 14:43:51 -08:00
|
|
|
|
|
|
|
pub enum Fragment {
|
|
|
|
/// Tokens that can be used as an expression.
|
2018-05-20 19:34:52 -07:00
|
|
|
Expr(TokenStream),
|
2017-02-20 14:43:51 -08:00
|
|
|
/// Tokens that can be used inside a block. The surrounding curly braces are
|
|
|
|
/// not part of these tokens.
|
2018-05-20 19:34:52 -07:00
|
|
|
Block(TokenStream),
|
2017-02-20 14:43:51 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! quote_expr {
|
|
|
|
($($tt:tt)*) => {
|
|
|
|
$crate::fragment::Fragment::Expr(quote!($($tt)*))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! quote_block {
|
|
|
|
($($tt:tt)*) => {
|
|
|
|
$crate::fragment::Fragment::Block(quote!($($tt)*))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Interpolate a fragment in place of an expression. This involves surrounding
|
|
|
|
/// Block fragments in curly braces.
|
|
|
|
pub struct Expr(pub Fragment);
|
|
|
|
impl ToTokens for Expr {
|
2018-05-20 19:34:52 -07:00
|
|
|
fn to_tokens(&self, out: &mut TokenStream) {
|
2017-02-20 14:43:51 -08:00
|
|
|
match self.0 {
|
|
|
|
Fragment::Expr(ref expr) => expr.to_tokens(out),
|
|
|
|
Fragment::Block(ref block) => {
|
2018-01-08 21:49:09 -08:00
|
|
|
token::Brace::default().surround(out, |out| block.to_tokens(out));
|
2017-02-20 14:43:51 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Interpolate a fragment as the statements of a block.
|
|
|
|
pub struct Stmts(pub Fragment);
|
|
|
|
impl ToTokens for Stmts {
|
2018-05-20 19:34:52 -07:00
|
|
|
fn to_tokens(&self, out: &mut TokenStream) {
|
2017-02-20 14:43:51 -08:00
|
|
|
match self.0 {
|
|
|
|
Fragment::Expr(ref expr) => expr.to_tokens(out),
|
|
|
|
Fragment::Block(ref block) => block.to_tokens(out),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Interpolate a fragment as the value part of a `match` expression. This
|
|
|
|
/// involves putting a comma after expressions and curly braces around blocks.
|
|
|
|
pub struct Match(pub Fragment);
|
|
|
|
impl ToTokens for Match {
|
2018-05-20 19:34:52 -07:00
|
|
|
fn to_tokens(&self, out: &mut TokenStream) {
|
2017-02-20 14:43:51 -08:00
|
|
|
match self.0 {
|
|
|
|
Fragment::Expr(ref expr) => {
|
|
|
|
expr.to_tokens(out);
|
2018-01-08 21:49:09 -08:00
|
|
|
<Token![,]>::default().to_tokens(out);
|
2017-02-20 14:43:51 -08:00
|
|
|
}
|
|
|
|
Fragment::Block(ref block) => {
|
2018-01-08 21:49:09 -08:00
|
|
|
token::Brace::default().surround(out, |out| block.to_tokens(out));
|
2017-02-20 14:43:51 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-12-11 17:55:23 -08:00
|
|
|
|
2018-05-20 19:34:52 -07:00
|
|
|
impl AsRef<TokenStream> for Fragment {
|
|
|
|
fn as_ref(&self) -> &TokenStream {
|
2017-12-11 17:55:23 -08:00
|
|
|
match *self {
|
|
|
|
Fragment::Expr(ref expr) => expr,
|
|
|
|
Fragment::Block(ref block) => block,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|