facet-default 0.46.0

Derive Default for facet types with custom field defaults
Documentation
//! # facet-default
//!
//! Derive [`Default`] for your types using facet's plugin system with custom field defaults.
//!
//! ## Usage
//!
//! ```ignore
//! use facet::Facet;
//! use facet_default as _;
//!
//! #[derive(Facet, Debug)]
//! #[facet(derive(Default))]
//! pub struct Config {
//!     #[facet(default = "localhost")]
//!     host: String,
//!     #[facet(default = 8080u16)]
//!     port: u16,
//!     // No attribute = uses Default::default()
//!     debug: bool,
//! }
//! ```
//!
//! ## Attributes
//!
//! ### Field Level
//!
//! - `#[facet(default = literal)]` - Use a literal value
//! - `#[facet(default)]` - Use `Default::default()` for the field type
//!
//! Fields without attributes use `Default::default()`.
//!
//! ## Enums
//!
//! For enums, mark the default variant:
//!
//! ```ignore
//! #[derive(Facet, Debug)]
//! #[facet(derive(Default))]
//! #[repr(u8)]
//! pub enum Status {
//!     #[facet(default::variant)]
//!     Pending,
//!     Active,
//!     Done,
//! }
//! ```

// ============================================================================
// ATTRIBUTE GRAMMAR
// ============================================================================

facet::define_attr_grammar! {
    ns "default";
    crate_path ::facet_default;

    /// Default attribute types for configuring Default implementation.
    pub enum Attr {
        /// Mark an enum variant as the default.
        ///
        /// Usage: `#[facet(default::variant)]`
        Variant,
    }
}

// ============================================================================
// PLUGIN TEMPLATE
// ============================================================================

/// Plugin chain entry point.
///
/// Called by `#[derive(Facet)]` when `#[facet(derive(Default))]` is present.
#[macro_export]
macro_rules! __facet_invoke {
    (
        @tokens { $($tokens:tt)* }
        @remaining { $($remaining:tt)* }
        @plugins { $($plugins:tt)* }
        @facet_crate { $($facet_crate:tt)* }
    ) => {
        $crate::__facet_invoke_internal! {
            @tokens { $($tokens)* }
            @remaining { $($remaining)* }
            @plugins {
                $($plugins)*
                @plugin {
                    @name { "Default" }
                    @template {
                        impl ::core::default::Default for @Self {
                            fn default() -> Self {
                                @if_struct {
                                    Self {
                                        @for_field {
                                            @field_name: @field_default_expr,
                                        }
                                    }
                                }
                                @if_enum {
                                    @for_variant {
                                        @if_attr(default::variant) {
                                            Self::@variant_name @variant_default_construction
                                        }
                                    }
                                }
                            }
                        }
                    }
                }
            }
            @facet_crate { $($facet_crate)* }
        }
    };
}

/// Internal macro that either chains to next plugin or calls finalize
#[doc(hidden)]
#[macro_export]
macro_rules! __facet_invoke_internal {
    // No more plugins - call finalize
    (
        @tokens { $($tokens:tt)* }
        @remaining { }
        @plugins { $($plugins:tt)* }
        @facet_crate { $($facet_crate:tt)* }
    ) => {
        $($facet_crate)*::__facet_finalize! {
            @tokens { $($tokens)* }
            @plugins { $($plugins)* }
            @facet_crate { $($facet_crate)* }
        }
    };

    // More plugins - chain to next
    (
        @tokens { $($tokens:tt)* }
        @remaining { $next:path $(, $rest:path)* $(,)? }
        @plugins { $($plugins:tt)* }
        @facet_crate { $($facet_crate:tt)* }
    ) => {
        $next! {
            @tokens { $($tokens)* }
            @remaining { $($rest),* }
            @plugins { $($plugins)* }
            @facet_crate { $($facet_crate)* }
        }
    };
}