Files
RobustToolbox/Robust.Shared/GameObjects/IComponentFactory.cs
metalgearsloth 9837c33de7 Add sourcegenned field deltas (#5155)
* Remove full "delta" states

* Update MapGridComponentState

* abstract ComponentState

* Release notes

* Fix tests

* Fix nullable errors

* A

* Sourcegen component deltas

* Audio deltas + methids

* Also eye

* Optimise out the dictionary

* Minor fixes

* Physics deltas

* Also this

* Fix field deltas

* remove old release notes

* Make IComponentDelta implement IComponent

* add sourcegen launch settings

* make silent error loud

* Review

* UI deltas

* Slimmer

* Sourcegen bandaid

---------

Co-authored-by: ElectroJr <leonsfriedrich@gmail.com>
2024-12-21 15:48:33 +11:00

311 lines
14 KiB
C#

using System;
using System.Collections.Generic;
using System.Diagnostics.CodeAnalysis;
using JetBrains.Annotations;
using Robust.Shared.GameStates;
using Robust.Shared.Prototypes;
namespace Robust.Shared.GameObjects
{
/// <summary>
/// Used by <see cref="EntityPrototype" /> to determine whether a component is available.
/// This distinction is important because prototypes are shared across client and server, but the two might have different components.
/// </summary>
/// <seealso cref="IComponentFactory" />
public enum ComponentAvailability : byte
{
/// <summary>
/// The component is available and can be instantiated.
/// </summary>
Available,
/// <summary>
/// The component is not available, but should be ignored (prevent warnings for missing components).
/// </summary>
Ignore,
/// <summary>
/// The component is unknown entirely. This may warrant a warning or error.
/// </summary>
Unknown
}
/// <summary>
/// Handles the registration and spawning of components.
/// </summary>
/// <remarks>
/// <p>
/// When referring to component names, this is the name the component has been registered as,
/// and what's used in prototypes. However, most commonly the type is referred through by an interface.
/// </p>
/// <p>
/// Before a component can be spawned, it must be registered so things such as name, networking ID, type, etc...
/// are known to the factory.
/// Components are registered into a registry.
/// The relevant methods for writing to this registry are <see cref="RegisterReference" />.
/// The data is exposed for reading through <see cref="GetRegistration" /> and its overloads.
/// This data is returned in the form of a <see cref="IComponentRegistration" />, which represents one component's registration.
/// </p>
/// </remarks>
/// <seealso cref="IComponentRegistration" />
/// <seealso cref="IComponent" />
public interface IComponentFactory
{
event Action<ComponentRegistration[]> ComponentsAdded;
event Action<string> ComponentIgnoreAdded;
/// <summary>
/// All IComponent types that are currently registered to this factory.
/// </summary>
IEnumerable<Type> AllRegisteredTypes { get; }
/// <summary>
/// The subset of all registered components that are networked, so that they can be
/// referenced between the client and the server.
/// </summary>
/// <remarks>
/// This will be null if the network Ids have not been generated yet.
/// </remarks>
/// <seealso cref="GenerateNetIds"/>
IReadOnlyList<ComponentRegistration>? NetworkedComponents { get; }
/// <summary>
/// Get whether a component is available right now.
/// </summary>
/// <param name="componentName">The name of the component to check.</param>
/// <param name="ignoreCase">Whether or not to ignore casing on <see cref="componentName"/></param>
/// <returns>The availability of the component.</returns>
ComponentAvailability GetComponentAvailability(string componentName, bool ignoreCase = false);
public void RegisterNetworkedFields<T>(params string[] fields) where T : IComponent;
public void RegisterNetworkedFields(ComponentRegistration compReg, params string[] fields);
/// <summary>
/// Slow-path for Type -> CompIdx mapping without generics.
/// </summary>
[Pure]
CompIdx GetIndex(Type type);
/// <summary>
/// Slow-path to get the component index for a specified type.
/// </summary>
[Pure]
int GetArrayIndex(Type type);
/// <summary>
/// Registers a component class with the factory.
/// </summary>
/// <param name="overwrite">If the component already exists, will this replace it?</param>
void RegisterClass<T>(bool overwrite = false) where T : IComponent, new();
/// <summary>
/// Registers component types with the factory.
/// </summary>
void RegisterTypes(params Type[] type);
/// <summary>
/// Registers a component name as being ignored.
/// </summary>
/// <param name="names">The names to be ignored.</param>
void RegisterIgnore(params string[] names);
/// <summary>
/// Disables throwing on missing components. Missing components will instead be treated as ignored.
/// </summary>
/// <param name="postfix">If provided, will only ignore components ending with the postfix.</param>
void IgnoreMissingComponents(string postfix = "");
IComponent GetComponent(EntityPrototype.ComponentRegistryEntry entry);
/// <summary>
/// Gets a new component instantiated of the specified type.
/// </summary>
/// <param name="componentType">type of component to make</param>
/// <returns>A Component</returns>
/// <exception cref="InvalidOperationException">
/// Thrown if no component of type <see cref="componentType"/> is registered.
/// </exception>
IComponent GetComponent(Type componentType);
IComponent GetComponent(CompIdx componentType);
/// <summary>
/// Gets a new component instantiated of the specified type.
/// </summary>
/// <typeparam name="T">The type of component to make.</typeparam>
/// <returns>A Component</returns>
/// <exception cref="InvalidOperationException">
/// Thrown if no component of type <see cref="T"/> is registered.
/// </exception>
T GetComponent<T>() where T : IComponent, new();
/// <summary>
/// Gets a new component instantiated from the specified component registration.
/// </summary>
/// <returns>A Component</returns>
IComponent GetComponent(ComponentRegistration reg);
/// <summary>
/// Gets a new component instantiated of the specified <see cref="IComponent.Name"/>.
/// </summary>
/// <param name="componentName">name of component to make</param>
/// <param name="ignoreCase">Whether or not to ignore casing on <see cref="componentName"/></param>
/// <returns>A Component</returns>
/// <exception cref="UnknownComponentException">
/// Thrown if no component exists with the given name <see cref="componentName"/>.
/// </exception>
IComponent GetComponent(string componentName, bool ignoreCase = false);
/// <summary>
/// Gets a new component instantiated of the specified network ID.
/// </summary>
/// <param name="netId">net id of component to make</param>
/// <returns>A Component</returns>
/// <exception cref="UnknownComponentException">
/// Thrown if no component exists with the given id <see cref="netId"/>.
/// </exception>
IComponent GetComponent(ushort netId);
/// <summary>
/// Gets the name of a component, throwing an exception if it does not exist.
/// </summary>
/// <param name="componentType">The type of the component</param>
/// <returns>The registered name of the component</returns>
/// <exception cref="UnknownComponentException">
/// Thrown if no component exists with the given type <see cref="componentType"/>.
/// </exception>
[Pure]
string GetComponentName(Type componentType);
[Pure]
string GetComponentName<T>() where T : IComponent, new();
/// <summary>
/// Gets the name of a component, throwing an exception if it does not exist.
/// </summary>
/// <param name="netID">The network ID corresponding to the component.</param>
/// <returns>The registered name of the component</returns>
/// <exception cref="UnknownComponentException">
/// Thrown if no component with id <see cref="netID"/> exists.
/// </exception>
[Pure]
string GetComponentName(ushort netID);
/// <summary>
/// Gets the registration belonging to a component, throwing an exception if it does not exist.
/// </summary>
/// <param name="componentName">The name of the component.</param>
/// <param name="ignoreCase">Whether or not to ignore casing on <see cref="componentName"/></param>
/// <exception cref="UnknownComponentException">
/// Thrown if no component exists with the given name <see cref="componentName"/>.
/// </exception>
ComponentRegistration GetRegistration(string componentName, bool ignoreCase = false);
/// <summary>
/// Gets the registration belonging to a component, throwing an exception if it does not exist.
/// </summary>
/// <param name="reference">The type of the component to lookup.</param>
/// <exception cref="UnknownComponentException">
/// Thrown if no component exists of type <see cref="reference"/>.
/// </exception>
ComponentRegistration GetRegistration(Type reference);
/// <summary>
/// Gets the registration belonging to a component, throwing an exception if it does not exist.
/// </summary>
/// <typeparam name="T">A type referencing the component.</typeparam>
/// <exception cref="UnknownComponentException">
/// Thrown if no component of type <see cref="T"/> exists.
/// </exception>
ComponentRegistration GetRegistration<T>() where T : IComponent, new();
/// <summary>
/// Gets the registration belonging to a component, throwing an
/// exception if it does not exist.
/// </summary>
/// <param name="netID">The network ID corresponding to the component.</param>
/// <returns></returns>
/// <exception cref="UnknownComponentException">
/// Thrown if no component with id <see cref="netID"/> exists.
/// </exception>
ComponentRegistration GetRegistration(ushort netID);
/// <summary>
/// Gets the registration of a component, throwing an exception if
/// it does not exist.
/// </summary>
/// <param name="component">An instance of the component.</param>
/// <returns></returns>
/// <exception cref="UnknownComponentException">
/// Thrown if no registration exists for component <see cref="component"/>.
/// </exception>
ComponentRegistration GetRegistration(IComponent component);
ComponentRegistration GetRegistration(CompIdx idx);
/// <summary>
/// Returns true if the given component name has been registered as ignored.
/// </summary>
bool IsIgnored(string componentName);
/// <summary>
/// Tries to get the registration belonging to a component.
/// </summary>
/// <param name="componentName">The name of the component.</param>
/// <param name="registration">The registration if found, null otherwise.</param>
/// <param name="ignoreCase">Whether or not to ignore casing on <see cref="componentName"/></param>
/// <returns>true it found, false otherwise.</returns>
bool TryGetRegistration(string componentName, [NotNullWhen(true)] out ComponentRegistration? registration, bool ignoreCase = false);
/// <summary>
/// Tries to get the registration belonging to a component.
/// </summary>
/// <param name="reference">A reference corresponding to the component to look up.</param>
/// <param name="registration">The registration if found, null otherwise.</param>
/// <returns>true it found, false otherwise.</returns>
bool TryGetRegistration(Type reference, [NotNullWhen(true)] out ComponentRegistration? registration);
/// <summary>
/// Tries to get the registration belonging to a component.
/// </summary>
/// <typeparam name="T">A type referencing the component.</typeparam>
/// <param name="registration">The registration if found, null otherwise.</param>
/// <returns>true it found, false otherwise.</returns>
bool TryGetRegistration<T>([NotNullWhen(true)] out ComponentRegistration? registration) where T : IComponent, new();
/// <summary>
/// Tries to get the registration belonging to a component.
/// </summary>
/// <param name="netID">The network ID corresponding to the component.</param>
/// <param name="registration">The registration if found, null otherwise.</param>
/// <returns>true it found, false otherwise.</returns>
bool TryGetRegistration(ushort netID, [NotNullWhen(true)] out ComponentRegistration? registration);
/// <summary>
/// Tries to get the registration of a component.
/// </summary>
/// <param name="component">An instance of the component.</param>
/// <param name="registration">The registration if found, null otherwise.</param>
/// <returns>true it found, false otherwise.</returns>
bool TryGetRegistration(IComponent component, [NotNullWhen(true)] out ComponentRegistration? registration);
/// <summary>
/// Automatically create registrations for all components with a <see cref="RegisterComponentAttribute" />
/// </summary>
void DoAutoRegistrations();
IEnumerable<ComponentRegistration> GetAllRegistrations();
IEnumerable<CompIdx> GetAllRefTypes();
void GenerateNetIds();
Type IdxToType(CompIdx idx);
/// <summary>
/// Gets a hash of all the registered components.
/// </summary>
/// <param name="networkedOnly">Whether to include all components or only networked ones.</param>
byte[] GetHash(bool networkedOnly);
}
}