extern crate proc_macro; mod attrs; mod shape; mod symbol; pub use attrs::*; pub use shape::*; pub use symbol::*; use proc_macro::TokenStream; use quote::quote; use std::{env, path::PathBuf}; use toml::{map::Map, Value}; pub struct BevyManifest { manifest: Map, } impl Default for BevyManifest { fn default() -> Self { Self { manifest: env::var_os("CARGO_MANIFEST_DIR") .map(PathBuf::from) .map(|mut path| { path.push("Cargo.toml"); let manifest = std::fs::read_to_string(path).unwrap(); toml::from_str(&manifest).unwrap() }) .unwrap(), } } } impl BevyManifest { pub fn maybe_get_path(&self, name: &str) -> Option { const BEVY: &str = "bevy"; const BEVY_INTERNAL: &str = "bevy_internal"; fn dep_package(dep: &Value) -> Option<&str> { if dep.as_str().is_some() { None } else { dep.as_table() .unwrap() .get("package") .map(|name| name.as_str().unwrap()) } } let find_in_deps = |deps: &Map| -> Option { let package = if let Some(dep) = deps.get(name) { return Some(Self::parse_str(dep_package(dep).unwrap_or(name))); } else if let Some(dep) = deps.get(BEVY) { dep_package(dep).unwrap_or(BEVY) } else if let Some(dep) = deps.get(BEVY_INTERNAL) { dep_package(dep).unwrap_or(BEVY_INTERNAL) } else { return None; }; let mut path = Self::parse_str::(package); if let Some(module) = name.strip_prefix("bevy_") { path.segments.push(Self::parse_str(module)); } Some(path) }; let deps = self .manifest .get("dependencies") .map(|deps| deps.as_table().unwrap()); let deps_dev = self .manifest .get("dev-dependencies") .map(|deps| deps.as_table().unwrap()); deps.and_then(find_in_deps) .or_else(|| deps_dev.and_then(find_in_deps)) } pub fn get_path(&self, name: &str) -> syn::Path { self.maybe_get_path(name) .unwrap_or_else(|| Self::parse_str(name)) } pub fn parse_str(path: &str) -> T { syn::parse(path.parse::().unwrap()).unwrap() } } /// Derive a label trait /// /// # Args /// /// - `input`: The [`syn::DeriveInput`] for struct that is deriving the label trait /// - `trait_path`: The path [`syn::Path`] to the label trait pub fn derive_label(input: syn::DeriveInput, trait_path: &syn::Path) -> TokenStream { let ident = input.ident; let (impl_generics, ty_generics, where_clause) = input.generics.split_for_impl(); let mut where_clause = where_clause.cloned().unwrap_or_else(|| syn::WhereClause { where_token: Default::default(), predicates: Default::default(), }); where_clause.predicates.push(syn::parse2(quote! { Self: Eq + ::std::fmt::Debug + ::std::hash::Hash + Clone + Send + Sync + 'static }).unwrap()); (quote! { impl #impl_generics #trait_path for #ident #ty_generics #where_clause { fn dyn_clone(&self) -> std::boxed::Box { std::boxed::Box::new(std::clone::Clone::clone(self)) } } }) .into() }