mirror of
https://github.com/bevyengine/bevy
synced 2024-11-22 12:43:34 +00:00
fd329c0426
# Objective > Rust 1.81 released the #[expect(...)] attribute, which works like #[allow(...)] but throws a warning if the lint isn't raised. This is preferred to #[allow(...)] because it tells us when it can be removed. - Adopts the parts of #15118 that are complete, and updates the branch so it can be merged. - There were a few conflicts, let me know if I misjudged any of 'em. Alice's [recommendation](https://github.com/bevyengine/bevy/issues/15059#issuecomment-2349263900) seems well-taken, let's do this crate by crate now that @BD103 has done the lion's share of this! (Relates to, but doesn't yet completely finish #15059.) Crates this _doesn't_ cover: - bevy_input - bevy_gilrs - bevy_window - bevy_winit - bevy_state - bevy_render - bevy_picking - bevy_core_pipeline - bevy_sprite - bevy_text - bevy_pbr - bevy_ui - bevy_gltf - bevy_gizmos - bevy_dev_tools - bevy_internal - bevy_dylib --------- Co-authored-by: BD103 <59022059+BD103@users.noreply.github.com> Co-authored-by: Ben Frankel <ben.frankel7@gmail.com> Co-authored-by: Antony <antony.m.3012@gmail.com>
213 lines
5.5 KiB
Rust
213 lines
5.5 KiB
Rust
// FIXME(15321): solve CI failures, then replace with `#![expect()]`.
|
|
#![allow(missing_docs, reason = "Not all docs are written yet, see #3492.")]
|
|
#![forbid(unsafe_code)]
|
|
#![cfg_attr(docsrs, feature(doc_auto_cfg))]
|
|
#![doc(
|
|
html_logo_url = "https://bevyengine.org/assets/icon.png",
|
|
html_favicon_url = "https://bevyengine.org/assets/icon.png"
|
|
)]
|
|
|
|
extern crate proc_macro;
|
|
|
|
mod bevy_main;
|
|
mod derefs;
|
|
mod enum_variant_meta;
|
|
|
|
use bevy_macro_utils::{derive_label, BevyManifest};
|
|
use proc_macro::TokenStream;
|
|
use quote::format_ident;
|
|
|
|
/// Implements [`Deref`] for structs. This is especially useful when utilizing the [newtype] pattern.
|
|
///
|
|
/// For single-field structs, the implementation automatically uses that field.
|
|
/// For multi-field structs, you must specify which field to use with the `#[deref]` attribute.
|
|
///
|
|
/// If you need [`DerefMut`] as well, consider using the other [derive] macro alongside
|
|
/// this one.
|
|
///
|
|
/// # Example
|
|
///
|
|
/// ## Tuple Structs
|
|
///
|
|
/// Using a single-field struct:
|
|
///
|
|
/// ```
|
|
/// use bevy_derive::Deref;
|
|
///
|
|
/// #[derive(Deref)]
|
|
/// struct MyNewtype(String);
|
|
///
|
|
/// let foo = MyNewtype(String::from("Hello"));
|
|
/// assert_eq!("Hello", *foo);
|
|
/// ```
|
|
///
|
|
/// Using a multi-field struct:
|
|
///
|
|
/// ```
|
|
/// # use std::marker::PhantomData;
|
|
/// use bevy_derive::Deref;
|
|
///
|
|
/// #[derive(Deref)]
|
|
/// struct MyStruct<T>(#[deref] String, PhantomData<T>);
|
|
///
|
|
/// let foo = MyStruct(String::from("Hello"), PhantomData::<usize>);
|
|
/// assert_eq!("Hello", *foo);
|
|
/// ```
|
|
///
|
|
/// ## Named Structs
|
|
///
|
|
/// Using a single-field struct:
|
|
///
|
|
/// ```
|
|
/// use bevy_derive::{Deref, DerefMut};
|
|
///
|
|
/// #[derive(Deref, DerefMut)]
|
|
/// struct MyStruct {
|
|
/// value: String,
|
|
/// }
|
|
///
|
|
/// let foo = MyStruct {
|
|
/// value: String::from("Hello")
|
|
/// };
|
|
/// assert_eq!("Hello", *foo);
|
|
/// ```
|
|
///
|
|
/// Using a multi-field struct:
|
|
///
|
|
/// ```
|
|
/// # use std::marker::PhantomData;
|
|
/// use bevy_derive::{Deref, DerefMut};
|
|
///
|
|
/// #[derive(Deref, DerefMut)]
|
|
/// struct MyStruct<T> {
|
|
/// #[deref]
|
|
/// value: String,
|
|
/// _phantom: PhantomData<T>,
|
|
/// }
|
|
///
|
|
/// let foo = MyStruct {
|
|
/// value:String::from("Hello"),
|
|
/// _phantom:PhantomData::<usize>
|
|
/// };
|
|
/// assert_eq!("Hello", *foo);
|
|
/// ```
|
|
///
|
|
/// [`Deref`]: std::ops::Deref
|
|
/// [newtype]: https://doc.rust-lang.org/rust-by-example/generics/new_types.html
|
|
/// [`DerefMut`]: std::ops::DerefMut
|
|
/// [derive]: crate::derive_deref_mut
|
|
#[proc_macro_derive(Deref, attributes(deref))]
|
|
pub fn derive_deref(input: TokenStream) -> TokenStream {
|
|
derefs::derive_deref(input)
|
|
}
|
|
|
|
/// Implements [`DerefMut`] for structs. This is especially useful when utilizing the [newtype] pattern.
|
|
///
|
|
/// For single-field structs, the implementation automatically uses that field.
|
|
/// For multi-field structs, you must specify which field to use with the `#[deref]` attribute.
|
|
///
|
|
/// [`DerefMut`] requires a [`Deref`] implementation. You can implement it manually or use
|
|
/// Bevy's [derive] macro for convenience.
|
|
///
|
|
/// # Example
|
|
///
|
|
/// ## Tuple Structs
|
|
///
|
|
/// Using a single-field struct:
|
|
///
|
|
/// ```
|
|
/// use bevy_derive::{Deref, DerefMut};
|
|
///
|
|
/// #[derive(Deref, DerefMut)]
|
|
/// struct MyNewtype(String);
|
|
///
|
|
/// let mut foo = MyNewtype(String::from("Hello"));
|
|
/// foo.push_str(" World!");
|
|
/// assert_eq!("Hello World!", *foo);
|
|
/// ```
|
|
///
|
|
/// Using a multi-field struct:
|
|
///
|
|
/// ```
|
|
/// # use std::marker::PhantomData;
|
|
/// use bevy_derive::{Deref, DerefMut};
|
|
///
|
|
/// #[derive(Deref, DerefMut)]
|
|
/// struct MyStruct<T>(#[deref] String, PhantomData<T>);
|
|
///
|
|
/// let mut foo = MyStruct(String::from("Hello"), PhantomData::<usize>);
|
|
/// foo.push_str(" World!");
|
|
/// assert_eq!("Hello World!", *foo);
|
|
/// ```
|
|
///
|
|
/// ## Named Structs
|
|
///
|
|
/// Using a single-field struct:
|
|
///
|
|
/// ```
|
|
/// use bevy_derive::{Deref, DerefMut};
|
|
///
|
|
/// #[derive(Deref, DerefMut)]
|
|
/// struct MyStruct {
|
|
/// value: String,
|
|
/// }
|
|
///
|
|
/// let mut foo = MyStruct {
|
|
/// value: String::from("Hello")
|
|
/// };
|
|
/// foo.push_str(" World!");
|
|
/// assert_eq!("Hello World!", *foo);
|
|
/// ```
|
|
///
|
|
/// Using a multi-field struct:
|
|
///
|
|
/// ```
|
|
/// # use std::marker::PhantomData;
|
|
/// use bevy_derive::{Deref, DerefMut};
|
|
///
|
|
/// #[derive(Deref, DerefMut)]
|
|
/// struct MyStruct<T> {
|
|
/// #[deref]
|
|
/// value: String,
|
|
/// _phantom: PhantomData<T>,
|
|
/// }
|
|
///
|
|
/// let mut foo = MyStruct {
|
|
/// value:String::from("Hello"),
|
|
/// _phantom:PhantomData::<usize>
|
|
/// };
|
|
/// foo.push_str(" World!");
|
|
/// assert_eq!("Hello World!", *foo);
|
|
/// ```
|
|
///
|
|
/// [`DerefMut`]: std::ops::DerefMut
|
|
/// [newtype]: https://doc.rust-lang.org/rust-by-example/generics/new_types.html
|
|
/// [`Deref`]: std::ops::Deref
|
|
/// [derive]: crate::derive_deref
|
|
#[proc_macro_derive(DerefMut, attributes(deref))]
|
|
pub fn derive_deref_mut(input: TokenStream) -> TokenStream {
|
|
derefs::derive_deref_mut(input)
|
|
}
|
|
|
|
#[proc_macro_attribute]
|
|
pub fn bevy_main(attr: TokenStream, item: TokenStream) -> TokenStream {
|
|
bevy_main::bevy_main(attr, item)
|
|
}
|
|
|
|
#[proc_macro_derive(EnumVariantMeta)]
|
|
pub fn derive_enum_variant_meta(input: TokenStream) -> TokenStream {
|
|
enum_variant_meta::derive_enum_variant_meta(input)
|
|
}
|
|
|
|
/// Generates an impl of the `AppLabel` trait.
|
|
///
|
|
/// This does not work for unions.
|
|
#[proc_macro_derive(AppLabel)]
|
|
pub fn derive_app_label(input: TokenStream) -> TokenStream {
|
|
let input = syn::parse_macro_input!(input as syn::DeriveInput);
|
|
let mut trait_path = BevyManifest::default().get_path("bevy_app");
|
|
let mut dyn_eq_path = trait_path.clone();
|
|
trait_path.segments.push(format_ident!("AppLabel").into());
|
|
dyn_eq_path.segments.push(format_ident!("DynEq").into());
|
|
derive_label(input, "AppLabel", &trait_path, &dyn_eq_path)
|
|
}
|