Toggle menu
Toggle personal menu
Not logged in
Your IP address will be publicly visible if you make any edits.

Class descriptions for custom modules

From Kerbal Space Program 2 Modding Wiki



While you're mostly free to define how your classes would be set up, there are some guidelines that need to be followed, restrictions to be aware of and inner workings to understand.

Data class[edit | edit source]

Defining your class[edit | edit source]

[Serializable]
public class Data_MyCustomModule : ModuleData
{ .. }

Set your ParthBehaviourModule type reference[edit | edit source]

public override Type ModuleType => typeof(Module_MyCustomModule);

Defining module properties[edit | edit source]

  • these are entries shown in the PAM
// Toggle (true/false) property
[KSPState] // KSPState attribute tells the game to save the state of this property in the save game file
[LocalizedField("Path/To/Your/Localization/String1")] // localization string for this attribute (see 'localization' paragraph)
[PAMDisplayControl(SortIndex = 2)] // sets the sorting index for this property. Lower values are placed first
public ModuleProperty<bool> SomeToggleProperty = new(false); // value in the parentheses defines the initial value

// String property
[LocalizedField("Path/To/Your/Localization/String2")]
[PAMDisplayControl(SortIndex = 4)]
[KSPDefinition] // KSPDefinition tells this property that its value is set from the part definition json (set by Patch Manager)
public ModuleProperty<string> SomeStringProperty = new ("");

// Float property - readonly
[LocalizedField("Path/To/Your/Localization/String3")]
[PAMDisplayControl(SortIndex = 7)]
[JsonIgnore] // either [KSPState] or [JsonIgnore] is needed if you want this property to be readonly
public ModuleProperty<float> SomeReadOnlyFloat = new (0, true, val => $"{val:N0} m");

// Float property – editable by players, will be built as a slider
[LocalizedField("Path/To/Your/Localization/String4")]
[PAMDisplayControl(SortIndex = 1)]
[SteppedRange(1f, 45f, 1f)] // minimum, maximum and step values
public ModuleProperty<float> SomeEditableProperty = new (1f, false, val => $"{val:N0}°"); // initial value, isReadOnly, ToStringDelegate

Defining a dropdown list property[edit | edit source]

  • dropdown list properties are string properties for which you define dropdown values in OnPartBehaviourModuleInit()
// Dropdown property
[LocalizedField("Path/To/Your/Localization/String5")]
public ModuleProperty<string> DropdownProperty = new ModuleProperty<string>("Some value");

public override void OnPartBehaviourModuleInit()
{
    var dropdownList = new DropdownItemList();
    dropdownList.Add("some key", new DropdownItem() { key = "some key", text = "Some value" });
    dropdownList.Add("another key", new DropdownItem() { key = "another key", text = "Another value" });
    SetDropdownData(DropdownProperty, dropdownList);
}

OnPartBehaviourModuleInit()[edit | edit source]

  • runs when this module is initialized when entering Flight/OAB state
public override void OnPartBehaviourModuleInit()
{ /* use this to initialize some values for your module, if needed */ }

OAB module description[edit | edit source]

  • set the description of your module for all parts it’s being attached to
  • description is shown in OAB and R&D while hovering over the part after pressing SHIFT
public override List<OABPartData.PartInfoModuleEntry> GetPartInfoEntries(Type partBehaviourModuleType, List<OABPartData.PartInfoModuleEntry> delegateList)
{
    if (partBehaviourModuleType == ModuleType)
    {
        // add module description
        delegateList.Add(new OABPartData.PartInfoModuleEntry("", (_) => „Path/To/Your/Localization/String5“));

        // entry header
        var entry = new OABPartData.PartInfoModuleEntry(„Path/To/Your/Localization/String6“,
            _ =>
            {
                // subentries
                var subEntries = new List<OABPartData.PartInfoModuleSubEntry>();
       
                // first subentry
                subEntries.Add(new OABPartData.PartInfoModuleSubEntry(
                    "Path/To/Your/Localization/String7", // subentry NAME
                    "subentry value"
                ));

                // second subentry
                subEntries.Add(new OABPartData.PartInfoModuleSubEntry(
                    "Path/To/Your/Localization/String8", // subentry NAME
                    "subentry value"
                ));

                // if your module is using resources, you can add them to the description
                // this doesn't set the value, it's just used to display it to the player
                if (UseResources)
                {
                    subEntries.Add(new OABPartData.PartInfoModuleSubEntry(
                        "Path/To/Your/Localization/String/ResourceName",
                        $"{RequiredResource.Rate.ToString("N3")} /s"
                    ));
                }

                return subEntries;
            });
        delegateList.Add(entry);
    }
    return delegateList;
}

Setting up resource consumptions[edit | edit source]

  • Note: trigger this from OnStart() in the Part Component class
public override void SetupResourceRequest(ResourceFlowRequestBroker resourceFlowRequestBroker)
{
    if (UseResources)
    {
        ResourceDefinitionID resourceIDFromName =
            GameManager.Instance.Game.ResourceDefinitionDatabase.GetResourceIDFromName(this.RequiredResource.ResourceName);
        if (resourceIDFromName == ResourceDefinitionID.InvalidID)
        {
            _LOGGER.LogError($"There are no resources with name {this.RequiredResource.ResourceName}");
            return;
        }
        RequestConfig = new ResourceFlowRequestCommandConfig();
        RequestConfig.FlowResource = resourceIDFromName;
        RequestConfig.FlowDirection = FlowDirection.FLOW_OUTBOUND;
        RequestConfig.FlowUnits = 0.0;
        RequestHandle = resourceFlowRequestBroker.AllocateOrGetRequest("MyCustomModule", default(ResourceFlowRequestHandle));
        resourceFlowRequestBroker.SetCommands(this.RequestHandle, 1.0, new ResourceFlowRequestCommandConfig[] { this.RequestConfig });
    }
}

[KSPDefinition]
[Tooltip("Whether the module consumes resources")]
public bool UseResources = true;

public bool HasResourcesToOperate = true;

[KSPDefinition]
[Tooltip("Resource required to operate this module if it consumes resources")]
public PartModuleResourceSetting RequiredResource;

public ResourceFlowRequestCommandConfig RequestConfig;

   

Part Behaviour class[edit | edit source]

Defining your class[edit | edit source]

[DisallowMultipleComponent]
public class Module_OrbitalSurvey : PartBehaviourModule
{ .. }

Set your PartComponentModule type reference[edit | edit source]

public override Type PartComponentModuleType => typeof(PartComponentModule_MyCustomModule);

Create Data module instance[edit | edit source]

[SerializeField]
protected Data_MyCustomModule _dataMyCustomModule;

public override void AddDataModules()
{
    base.AddDataModules();
    _dataMyCustomModule ??= new Data_MyCustomModule();
    DataModules.TryAddUnique(_dataMyCustomModule, out _dataMyCustomModule);
}

Initialize the module behaviour[edit | edit source]

private ModuleAction _myCustomAction;

public override void OnInitialize()
{
    base.OnInitialize();

    // module actions are triggered when players press a button on the PAM property
    _myCustomAction = new ModuleAction(MethodThatWillHandleTheAction);
    _dataMyCustomModule.AddAction("Path/To/Your/Localization/String/X", _myCustomAction, 1);

    if (PartBackingMode == PartBackingModes.Flight)
    {
        /* do stuff that's only needed in Flight view */

        // example1: hide or show PAM properties depending on the Flight/OAB view
        UpdateFlightPAMVisibility(); 

        // example2: subscribe to the enabled toggle
        _dataMyCustomModule.EnabledToggleProperty.OnChangedValue += MethodThatWillHandleThis;
    } 

    if (PartBackingMode == PartBackingModes.OAB)
    { /* do stuff that's only needed in the OAB*/ }
}

private void MethodThatWillHandleTheAction()
{ /* do stuff here */}

FixedUpdate loop - Flight/Map[edit | edit source]

  • define stuff that needs to be executed continuously on every FixedUpdate loop. Be careful not to do expensive stuff here
  • this triggers when the vessel is loaded, in Flight/Map view only
// This triggers in flight
public override void OnModuleFixedUpdate(float fixedDeltaTime)
{   
    // example1: do stuff only if the module is enabled
    if (_dataMyCustomModule.EnabledToggleProperty.GetValue())
    { .. }

    // example2: update PAM items
    if (someConditionMet)
    {
        UpdateFlightPAMVisibility();
    }
}

Update loop[edit | edit source]

  • similar to FixedUpdate, but this is a regular Update loop independent of game time
  • this triggers in Flight/Map when the vessel is loaded and in OAB when the part is attached to the assembly
public override void OnUpdate(float deltaTime)
{ .. }

FixedUpdate loop - OAB[edit | edit source]

  • same as OnModuleFixedUpdate but it triggers only in OAB
public override void OnModuleOABFixedUpdate(float deltaTime)
{ .. }

Define behaviour when the behaviour module instance will be destroyed[edit | edit source]

  • cases: exiting Flight view, part has been destroyed, exiting the game
public override void OnShutdown()
{
    // example: unsubscribe from events
    _dataMyCustomModule.EnabledToggleProperty.OnChangedValue -= OnToggleChangedValue;
}

Setting visibility for PAM properties[edit | edit source]

private void UpdateFlightPAMVisibility(bool state)
{
    _dataMyCustomModule.SetVisible(_dataMyCustomModule.SomeProperty, state);
    _dataMyCustomModule.SetVisible(_dataMyCustomModule.SomeOtherProperty, true);
    _dataMyCustomModule.SetVisible(_dataMyCustomModule.YetAnotherProperty, false);
}

OnEnable[edit | edit source]

  • triggers when Flight view is loaded (only for the loaded vessel) and in OAB when part is added to the assembly
protected void OnEnable()
{ .. }

Part Component class[edit | edit source]

Defining your class[edit | edit source]

public class PartComponentModule_MyCustomModule : PartBehaviourModule
{ .. }

Set your PartBehaviourModule type reference[edit | edit source]

public override Type PartComponentModuleType => typeof(PartComponentModule_MyCustomModule);

OnStart(double universalTime)[edit | edit source]

  • for new vessels this will run when the Flight view is loaded
  • also runs on load for every vessel currently in Flight (don't need to be loaded).
  • best used for any kind of initialization of backend tasks this vessel/module needs to go through
private Data_MyCustomModule _dataMyCustomModule;

public override void OnStart(double universalTime)
{
    // set a reference to the Data class
    if (!DataModules.TryGetByType<Data_MyCustomModule>(out _dataMyCustomModule))
    {
        _LOGGER.LogError("Unable to find a Data_MyCustomModule in the PartComponentModule for " + base.Part.PartName);
        return;
    }

    // initialize resource requests
    _dataMyCustomModule.SetupResourceRequest(base.resourceFlowRequestBroker);
}

OnUpdate[edit | edit source]

  • this starts triggering when the vessel is first placed in Flight. Doesn't trigger in the OAB before that
  • once the vessel is in Flight, it will always trigger, in any view, until the part is destroyed/recovered
  • use this for tasks that need to continually run, even when vessel is unloaded. Be careful not to put expensive tasks here.
public override void OnUpdate(double universalTime, double deltaUniversalTime)
{
    ResourceConsumptionUpdate(deltaUniversalTime); // example1: trigger resources consumption
    UpdateStatusAndState(); // example2: do general status updates if needed
}

Resource consumption[edit | edit source]

private void ResourceConsumptionUpdate(double deltaTime)
{
    if (_dataMyCustomModule.UseResources)
    {
        if (GameManager.Instance.Game.SessionManager.IsDifficultyOptionEnabled("InfinitePower"))
        {
            _dataMyCustomModule.HasResourcesToOperate = true;
            if (base.resourceFlowRequestBroker.IsRequestActive(_dataMyCustomModule.RequestHandle))
            {
                base.resourceFlowRequestBroker.SetRequestInactive(_dataMyCustomModule.RequestHandle);
                return;
            }
        }
        else
        {
            if (this._hasOutstandingRequest)
            {
                this._returnedRequestResolutionState =
                    base.resourceFlowRequestBroker.GetRequestState(_dataMyCustomModule.RequestHandle);
                _dataMyCustomModule.HasResourcesToOperate = this._returnedRequestResolutionState.WasLastTickDeliveryAccepted;
            }
            this._hasOutstandingRequest = false;
            if (!_dataMyCustomModule.EnabledToggleProperty.GetValue() &&
                base.resourceFlowRequestBroker.IsRequestActive(_dataMyCustomModule.RequestHandle))
            {
                base.resourceFlowRequestBroker.SetRequestInactive(_dataMyCustomModule.RequestHandle);
                _dataMyCustomModule.HasResourcesToOperate = false;
            }
            else if (_dataMyCustomModule.EnabledToggle.GetValue() &&
                base.resourceFlowRequestBroker.IsRequestInactive(_dataMyCustomModule.RequestHandle))
            {
                base.resourceFlowRequestBroker.SetRequestActive(_dataMyCustomModule.RequestHandle);
            }
            if (_dataMyCustomModule.EnabledToggleProperty.GetValue())
            {
                _dataMyCustomModule.RequestConfig.FlowUnits = (double)_dataMyCustomModule.RequiredResource.Rate;
                base.resourceFlowRequestBroker.SetCommands(_dataMyCustomModule.RequestHandle, 1.0,
                    new ResourceFlowRequestCommandConfig[] { _dataMyCustomModule.RequestConfig });
                this._hasOutstandingRequest = true;
                return;
            }
        }
    }
    else
    {
        _dataMyCustomModule.HasResourcesToOperate = true;
    }
}

OnShutdown[edit | edit source]

  • define behaviour when the Part Component instance will be destroyed
  • cases: part has been destroyed, exiting the game
public override void OnShutdown()
{ .. }