1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
use proc_macro::TokenStream;
use proc_macro2::{Span as Span2, TokenStream as TokenStream2};
use syn::punctuated::Punctuated;
use syn::spanned::Spanned;
use syn::{FnArg, Ident, ImplItem, ItemImpl, Path, PathArguments, PathSegment, Type, TypePath};
pub(super) fn generate_widgets_type(
widgets_ty: Option<Type>,
component_impl: &mut ItemImpl,
errors: &mut Vec<syn::Error>,
) -> Option<Type> {
if let Some(ty) = widgets_ty {
Some(ty)
}
else if let Type::Path(self_ty) = &*component_impl.self_ty {
let (path, impl_item) = self_ty_to_widgets_ty(self_ty);
component_impl.items.push(impl_item);
Some(path)
}
else {
let msg = "no `Widgets` type found and the type if `Self` in not a path. \
Please use a path for `Self` or use `type Widgets = WidgetsName;` to name the widgets type.";
errors.push(syn::Error::new(
component_impl
.items
.first()
.map(|i| i.span())
.unwrap_or_else(|| component_impl.self_ty.span()),
msg,
));
None
}
}
pub(super) fn self_ty_to_widgets_ty(self_ty: &TypePath) -> (Type, ImplItem) {
let mut self_path = self_ty.clone();
let last_seg = self_path.path.segments.last_mut().unwrap();
last_seg.arguments = Default::default();
last_seg.ident = Ident::new(&format!("{}Widgets", last_seg.ident), last_seg.span());
let impl_item = syn::parse_quote_spanned! {
self_path.span() => type Widgets = #self_path;
};
(Type::Path(self_path), impl_item)
}
pub(super) fn strings_to_path(strings: &[&str]) -> Path {
let path_segments: Vec<PathSegment> = strings
.iter()
.map(|string| -> PathSegment {
PathSegment {
ident: Ident::new(string, Span2::call_site()),
arguments: PathArguments::None,
}
})
.collect();
Path {
leading_colon: None,
segments: Punctuated::from_iter(path_segments),
}
}
pub(super) fn item_impl_error(original_input: TokenStream) -> TokenStream {
let macro_impls = quote::quote! {
macro_rules! view_output {
() => { () };
}
macro_rules! view {
() => {};
($tt:tt) => {};
($tt:tt $($y:tt)+) => {}
}
}
.into();
vec![macro_impls, original_input].into_iter().collect()
}
pub(super) fn verbatim_impl_item_method(
name: &str,
args: Vec<FnArg>,
ty: Type,
tokens: TokenStream2,
) -> ImplItem {
ImplItem::Method(syn::ImplItemMethod {
attrs: Vec::new(),
vis: syn::Visibility::Inherited,
defaultness: None,
sig: syn::Signature {
constness: None,
asyncness: None,
unsafety: None,
abi: None,
fn_token: syn::token::Fn::default(),
ident: Ident::new(name, Span2::call_site()),
generics: syn::Generics {
lt_token: None,
params: Punctuated::default(),
gt_token: None,
where_clause: None,
},
paren_token: syn::token::Paren::default(),
inputs: Punctuated::from_iter(args.into_iter()),
variadic: None,
output: syn::ReturnType::Type(syn::token::RArrow::default(), Box::new(ty)),
},
block: syn::Block {
brace_token: syn::token::Brace::default(),
stmts: vec![syn::Stmt::Expr(syn::Expr::Verbatim(tokens))],
},
})
}