godot-core 0.5.1

Internal crate used by godot-rust
Documentation
/*
 * Copyright (c) godot-rust; Bromeon and contributors.
 * This Source Code Form is subject to the terms of the Mozilla Public
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 * file, You can obtain one at https://mozilla.org/MPL/2.0/.
 */

use std::fmt::{self, Debug, Formatter};
use std::mem;

use crate::builtin::{GString, NodePath};
use crate::classes::{Node, Resource};
use crate::meta::shape::GodotShape;
use crate::meta::{AsArg, FromGodot, GodotConvert, arg_into_owned};
use crate::obj::{Gd, Inherits};
use crate::registry::property::Var;

/// Ergonomic late-initialization container with `ready()` support.
///
/// While deferred initialization is generally seen as bad practice, it is often inevitable in game development.
/// Godot in particular encourages initialization inside `ready()`, e.g. to access the scene tree after a node is inserted into it.
/// The alternative to using this pattern is [`Option<T>`][option], which needs to be explicitly unwrapped with `unwrap()` or `expect()` each time.
///
/// If you have a value that you expect to be initialized in the Godot editor, use [`OnEditor<T>`][crate::obj::OnEditor] instead.
/// As a general "maybe initialized" type, `Option<Gd<T>>` is always available, even if more verbose.
///
/// # Late-init semantics
///
/// `OnReady<T>` should always be used as a struct field. There are two modes to use it:
///
/// 1. **Automatic mode, using [`new()`](OnReady::new), [`from_base_fn()`](OnReady::from_base_fn),
///    [`from_node()`][Self::from_node] or [`from_loaded()`][Self::from_loaded].**<br>
///    Before `ready()` is called, all `OnReady` fields constructed with the above methods are automatically initialized,
///    in the order of declaration. This means that you can safely access them in `ready()`.<br>
/// 2. **Manual mode, using [`manual()`](Self::manual).**<br>
///    These fields are left uninitialized until you call [`init()`][Self::init] on them. This is useful if you need more complex
///    initialization scenarios than a closure allows. If you forget initialization, a panic will occur on first access.
///
/// Conceptually, `OnReady<T>` is very close to [once_cell's `Lazy<T>`][lazy], with additional hooks into the Godot lifecycle.
/// The absence of methods to check initialization state is deliberate: you don't need them if you follow the above two patterns.
/// This container is not designed as a general late-initialization solution, but tailored to the `ready()` semantics of Godot.
///
/// `OnReady<T>` cannot be used with `#[export]` fields, because `ready()` is typically not called in the editor (unless `#[class(tool)]`
/// is specified). You can however use it with `#[var]` -- just make sure to access the fields in GDScript after `ready()`.
///
/// This type is not thread-safe. `ready()` runs on the main thread, and you are expected to access its value on the main thread, as well.
///
/// [option]: std::option::Option
/// [lazy]: https://docs.rs/once_cell/1/once_cell/unsync/struct.Lazy.html
///
/// # Requirements
/// - The class must have an explicit `Base` field (i.e. `base: Base<Node>`).
/// - The class must inherit `Node` (otherwise `ready()` would not exist anyway).
///
/// # Example - user-defined `init`
/// ```
/// use godot::prelude::*;
///
/// #[derive(GodotClass)]
/// #[class(base = Node)]
/// struct MyClass {
///    base: Base<Node>,
///    auto: OnReady<i32>,
///    manual: OnReady<i32>,
/// }
///
/// #[godot_api]
/// impl INode for MyClass {
///     fn init(base: Base<Node>) -> Self {
///        Self {
///            base,
///            auto: OnReady::new(|| 11),
///            manual: OnReady::manual(),
///        }
///     }
///
///     fn ready(&mut self) {
///        // self.auto is now ready with value 11.
///        assert_eq!(*self.auto, 11);
///
///        // self.manual needs to be initialized manually.
///        self.manual.init(22);
///        assert_eq!(*self.manual, 22);
///     }
/// }
/// ```
///
/// # Example - macro-generated `init`
/// ```
/// use godot::prelude::*;
///
/// #[derive(GodotClass)]
/// #[class(init, base = Node)]
/// struct MyClass {
///    base: Base<Node>,
///
///    #[init(node = "ChildPath")]
///    auto: OnReady<Gd<Node2D>>,
///
///    #[init(val = OnReady::manual())]
///    manual: OnReady<i32>,
/// }
///
/// #[godot_api]
/// impl INode for MyClass {
///     fn ready(&mut self) {
///        // self.node is now ready with the node found at path `ChildPath`.
///        assert_eq!(self.auto.get_name(), "ChildPath");
///
///        // self.manual needs to be initialized manually.
///        self.manual.init(22);
///        assert_eq!(*self.manual, 22);
///     }
/// }
/// ```
#[derive(Debug)]
pub struct OnReady<T> {
    state: InitState<T>,
}

impl<T: Inherits<Node>> OnReady<Gd<T>> {
    /// Variant of [`OnReady::new()`], fetching the node located at `path` before `ready()`.
    ///
    /// This is the functional equivalent of:
    /// - the GDScript pattern `@onready var node = $NODE_PATH`.
    /// - the Rust method [`Node::get_node_as()`].
    ///
    /// When used with `#[class(init)]`, the field can be annotated with `#[init(node = "NODE_PATH")]` to call this constructor.
    ///
    /// # Panics (deferred)
    /// - If `path` does not point to a valid node, or its type is not a `T` or a subclass.
    ///
    /// Note that the panic will only happen if and when the node enters the SceneTree for the first time
    /// (i.e. it receives the `READY` notification).
    pub fn from_node(path: impl AsArg<NodePath>) -> Self {
        arg_into_owned!(path);

        Self::from_base_fn(move |base| base.get_node_as(&path))
    }
}

impl<T: Inherits<Resource>> OnReady<Gd<T>> {
    /// Variant of [`OnReady::new()`], loading the resource stored at `path` before `ready()`.
    ///
    /// This is the functional equivalent of:
    /// - the GDScript pattern `@onready var res = load(...)`.
    /// - the Rust function [`tools::load()`][crate::tools::load].
    ///
    /// When used with `#[class(init)]`, the field can be annotated with `#[init(load = "FILE_PATH")]` to call this constructor.
    ///
    /// # Panics (deferred)
    /// - If the resource does not exist at `path`, cannot be loaded or is not compatible with type `T`.
    ///
    /// Note that the panic will only happen if and when the node enters the SceneTree for the first time
    /// (i.e. it receives the `READY` notification).
    pub fn from_loaded(path: impl AsArg<GString>) -> Self {
        arg_into_owned!(path);

        Self::new(move || crate::tools::load(&path))
    }
}

impl<T> OnReady<T> {
    /// Schedule automatic initialization before `ready()`.
    ///
    /// This guarantees that the value is initialized once `ready()` starts running.
    /// Until then, accessing the object may panic. In particular, the object is _not_ initialized on first use.
    ///
    /// The value is also initialized when you don't override `ready()`.
    ///
    /// For more control over initialization, use the [`OnReady::manual()`] constructor, followed by a [`self.init()`][OnReady::init]
    /// call during `ready()`.
    pub fn new<F>(init_fn: F) -> Self
    where
        F: FnOnce() -> T + 'static,
    {
        Self::from_base_fn(|_| init_fn())
    }

    /// Variant of [`OnReady::new()`], allowing access to `Base` when initializing.
    pub fn from_base_fn<F>(init_fn: F) -> Self
    where
        F: FnOnce(&Gd<Node>) -> T + 'static,
    {
        Self {
            state: InitState::AutoPrepared {
                initializer: Box::new(init_fn),
            },
        }
    }

    /// Leave uninitialized, expects manual initialization during `ready()`.
    ///
    /// If you use this method, you _must_ call [`init()`][Self::init] during the `ready()` callback, otherwise a panic will occur.
    pub fn manual() -> Self {
        Self {
            state: InitState::ManualUninitialized,
        }
    }

    /// Runs manual initialization.
    ///
    /// # Panics
    /// - If `init()` was called before.
    /// - If this object was already provided with a closure during construction, in [`Self::new()`] or any other automatic constructor.
    pub fn init(&mut self, value: T) {
        match &self.state {
            InitState::ManualUninitialized => {
                self.state = InitState::Initialized { value };
            }
            InitState::AutoPrepared { .. } | InitState::AutoInitializationFailed => {
                panic!("cannot call init() on auto-initialized OnReady objects")
            }
            InitState::Initialized { .. } => {
                panic!("already initialized; did you call init() more than once?")
            }
        };
    }

    /// Runs initialization.
    ///
    /// # Panics
    /// - If the value is already initialized.
    /// - If previous auto initialization failed.
    pub(crate) fn init_auto(&mut self, base: &Gd<Node>) {
        // Two branches needed, because mem::replace() could accidentally overwrite an already initialized value.
        match &self.state {
            InitState::ManualUninitialized => return, // skipped
            InitState::AutoPrepared { .. } => {}      // handled below
            InitState::AutoInitializationFailed => {
                panic!("OnReady automatic value initialization has already failed")
            }
            InitState::Initialized { .. } => panic!("OnReady object already initialized"),
        };

        // Temporarily replace with AutoInitializationFailed state which will be left in iff initialization fails.
        let InitState::AutoPrepared { initializer } =
            mem::replace(&mut self.state, InitState::AutoInitializationFailed)
        else {
            // SAFETY: condition checked above.
            unsafe { std::hint::unreachable_unchecked() }
        };

        self.state = InitState::Initialized {
            value: initializer(base),
        };
    }
}

// Panicking Deref is not best practice according to Rust, but constant get() calls are significantly less ergonomic and make it harder to
// migrate between T and LateInit<T>, because all the accesses need to change.
impl<T> std::ops::Deref for OnReady<T> {
    type Target = T;

    /// Returns a shared reference to the value.
    ///
    /// # Panics
    /// If the value is not yet initialized.
    fn deref(&self) -> &Self::Target {
        match &self.state {
            InitState::ManualUninitialized => {
                panic!("OnReady manual value uninitialized, did you call init()?")
            }
            InitState::AutoPrepared { .. } => {
                panic!("OnReady automatic value uninitialized, is only available in ready()")
            }
            InitState::AutoInitializationFailed => {
                panic!("OnReady automatic value initialization failed")
            }
            InitState::Initialized { value } => value,
        }
    }
}

impl<T> std::ops::DerefMut for OnReady<T> {
    /// Returns an exclusive reference to the value.
    ///     
    /// # Panics
    /// If the value is not yet initialized.
    fn deref_mut(&mut self) -> &mut Self::Target {
        match &mut self.state {
            InitState::Initialized { value } => value,
            InitState::ManualUninitialized | InitState::AutoPrepared { .. } => {
                panic!("value not yet initialized")
            }
            InitState::AutoInitializationFailed => {
                panic!("OnReady automatic value initialization failed")
            }
        }
    }
}

impl<T: GodotConvert> GodotConvert for OnReady<T> {
    type Via = T::Via;

    fn godot_shape() -> GodotShape {
        T::godot_shape()
    }
}

impl<T> Var for OnReady<T>
where
    T: Var<PubType = T> + FromGodot,
{
    type PubType = T;

    // All following functions: Deref/DerefMut panics if not initialized, preserving the "single init point" invariant.

    fn var_get(field: &Self) -> Self::Via {
        let deref: &T = field;
        T::var_get(deref)
    }

    fn var_set(field: &mut Self, value: Self::Via) {
        let deref: &mut T = field;
        T::var_set(deref, value);
    }

    fn var_pub_get(field: &Self) -> Self::PubType {
        let deref: &T = field;
        T::var_pub_get(deref)
    }

    fn var_pub_set(field: &mut Self, value: Self::PubType) {
        let deref: &mut T = field;
        T::var_pub_set(deref, value);
    }
}

// ----------------------------------------------------------------------------------------------------------------------------------------------
// Implementation

type InitFn<T> = dyn FnOnce(&Gd<Node>) -> T;

enum InitState<T> {
    ManualUninitialized,
    AutoPrepared { initializer: Box<InitFn<T>> },
    AutoInitializationFailed,
    Initialized { value: T },
}

impl<T: Debug> Debug for InitState<T> {
    fn fmt(&self, fmt: &mut Formatter<'_>) -> fmt::Result {
        match self {
            InitState::ManualUninitialized => fmt.debug_struct("ManualUninitialized").finish(),
            InitState::AutoPrepared { .. } => {
                fmt.debug_struct("AutoPrepared").finish_non_exhaustive()
            }
            InitState::AutoInitializationFailed => {
                fmt.debug_struct("AutoInitializationFailed").finish()
            }
            InitState::Initialized { value } => fmt
                .debug_struct("Initialized")
                .field("value", value)
                .finish(),
        }
    }
}