derive_app_context.rs

 1use proc_macro::TokenStream;
 2use quote::quote;
 3use syn::{parse_macro_input, DeriveInput};
 4
 5use crate::get_simple_attribute_field;
 6
 7pub fn derive_app_context(input: TokenStream) -> TokenStream {
 8    let ast = parse_macro_input!(input as DeriveInput);
 9
10    let Some(app_variable) = get_simple_attribute_field(&ast, "app") else {
11        return quote! {
12            compile_error!("Derive must have an #[app] attribute to detect the &mut App field");
13        }
14        .into();
15    };
16
17    let type_name = &ast.ident;
18    let (impl_generics, type_generics, where_clause) = ast.generics.split_for_impl();
19
20    let gen = quote! {
21        impl #impl_generics gpui::AppContext for #type_name #type_generics
22        #where_clause
23        {
24            type Result<T> = T;
25
26            fn new<T: 'static>(
27                &mut self,
28                build_model: impl FnOnce(&mut gpui::Context<'_, T>) -> T,
29            ) -> Self::Result<gpui::Entity<T>> {
30                self.#app_variable.new(build_model)
31            }
32
33            fn reserve_entity<T: 'static>(&mut self) -> Self::Result<gpui::Reservation<T>> {
34                self.#app_variable.reserve_entity()
35            }
36
37            fn insert_entity<T: 'static>(
38                &mut self,
39                reservation: gpui::Reservation<T>,
40                build_model: impl FnOnce(&mut gpui::Context<'_, T>) -> T,
41            ) -> Self::Result<gpui::Entity<T>> {
42                self.#app_variable.insert_entity(reservation, build_model)
43            }
44
45            fn update_entity<T, R>(
46                &mut self,
47                handle: &gpui::Entity<T>,
48                update: impl FnOnce(&mut T, &mut gpui::Context<'_, T>) -> R,
49            ) -> Self::Result<R>
50            where
51                T: 'static,
52            {
53                self.#app_variable.update_entity(handle, update)
54            }
55
56            fn read_entity<T, R>(
57                &self,
58                handle: &gpui::Entity<T>,
59                read: impl FnOnce(&T, &gpui::App) -> R,
60            ) -> Self::Result<R>
61            where
62                T: 'static,
63            {
64                self.#app_variable.read_entity(handle, read)
65            }
66
67            fn update_window<T, F>(&mut self, window: gpui::AnyWindowHandle, f: F) -> gpui::Result<T>
68            where
69                F: FnOnce(gpui::AnyView, &mut gpui::Window, &mut gpui::App) -> T,
70            {
71                self.#app_variable.update_window(window, f)
72            }
73
74            fn read_window<T, R>(
75                &self,
76                window: &gpui::WindowHandle<T>,
77                read: impl FnOnce(gpui::Entity<T>, &gpui::App) -> R,
78            ) -> gpui::Result<R>
79            where
80                T: 'static,
81            {
82                self.#app_variable.read_window(window, read)
83            }
84        }
85    };
86
87    gen.into()
88}