Parts Pack Production Procedure: Difference between revisions

From KSP 2 Modding Wiki
(Initial version sans graphics)
 
(16 intermediate revisions by 2 users not shown)
Line 1: Line 1:
== '''Preamble''' ==
The aim of this tutorial is to guide you through the process of creating your first '''part mod''' for Kerbal Space Program 2. We will go over everything from creating the part's mesh and textures to setting up the part in Unity to be bundled into a KSP2 mod.
This guide is based on the videos: How to create parts for KSP2 and How to make engines for KSP2, and on other notes and guidance from the KSP2 Modding Society discord.


Also the web pages: KSP2 Part tutorial (almost) from scratch, and Tutorial: My First Part
The way this tutorial will be structured is as a succession of steps which may link to other tutorials on this wiki, do not hesitate to keep this main page open while perusing the other tutorials. We aim at being as comprehensive as possible to enable you to make any part mod, but this page is still work in progress so stuff might be incomplete and/or subject to change in the future.


== '''Prerequisites''' ==
== Magnificent Mesh Modelling (Creating the part asset) ==
You will absolutely need the following things at a bare minimum:
The first step in your part mod creation journey is to create the part's "3D asset", meaning its 3D mesh, textures and eventual animations. You can use any software you're comfortable with to do so but here are some guides for some specific software if you desire:


* Unity Editor
# [[Modeling the mesh in Blender]]
* ThunderKit
# [[Texturing the mesh in Substance 3D Painter]]
* Unity KSP Tools
* Addressables Package (Install via Unity Window menu > Package Manager
** Change Packages to Unity Registry
** Search for “addressables”. Click on it and install.
* LuxShader


You will generally also need the following things:
Here are some resources that can be useful for modelling and texturing, regardless of the software you're using:


* Blender (or other mesh modeling SW that can produce FBX files)
* [[Sizes]]
* Various blender addons such as TexTools, etc. (make a list of the free ones you’re using)
* [[Texturing]]
* Substance Painter (or other texture painting tool – could be Blender or Quixel Mixer, but SP is recommended highly)


You probably need a few other things too:
== Part Pack Prep (Basic Unity part mod project setup) ==
Now that you have created a model and textures for your part, the next step is to set up a Unity project that will be used to configure your part and build the asset bundles that will contain your mod's contents.


* JSON exports of similar stock parts to show you what some things can or should be set to
# '''Creating the project:''' Follow the [[Setting up Unity]] tutorial to create and set up a project for part modding.
# '''Parts Pack folder''': Under Assets, create a folder (R-Click Assets: Create > Folder) and name it the same as your parts pack mod. If your mod is called My Awesome Mod then this would be <code>Assets\MyAwesomeMod</code>.
# '''Materials folder''': Inside your parts pack folder create a folder called Materials. You’ll be storing the textures and materials you need there. E.g., <code>Assets\MyAwesomeMod\Materials</code>. This is just to aid in organization.
# '''Parts folder''': Inside your parts pack folder create a folder called Parts. You’ll be storing the part meshes and related things there. E.g., <code>Assets\MyAwesomeMod\Parts</code>. You can have whatever organization you like here, so if you want to group some parts you might create group folders within Parts (e.g., Methalox Engines, Nuclear Engines, Ion Engines, etc.). This is just to aid in organization and is optional.
# '''Plugin Folder and Content''': In your Unity project's <code>Assets\plugin_template</code> folder create a <code>localizations</code> folder. '''NOTE!''' This is currently required to be exactly this – localizations (plural), not localization (singular). This will be where you put the localization file.
## Create Localization file: Localization files are CSV files following a particular format. These must have lines ending with LF not LF/NL, and they must reference the same <part_name> you use in the Part Production process below. There are some other restrictions for content, particularly that if you want a string that contains a “,” that string needs to be enclosed in quotes or the comma will mess with how the strings are parsed. These files are where the part’s Title, Subtitle, Manufacturer, and Description are configured. Here’s an example:<syntaxhighlight lang="text">
Key,Type,Desc,English,French


== Process ==
Parts/Title/spark_spt100,Text,,SPT-100,SPT-100
This guide covers the part of the process that takes place in the Unity Editor resulting in an assembly you can load as a codeless mod in KSP2, and how to get that result into the game.
Parts/Subtitle/spark_spt100,Text,,Hall Effect Thruster with Xenon Tank,Moteur à Effet Hall et Réservoir de Xénon
Parts/Manufacturer/spark_spt100,Text,,"Stellar Plasma-Assisted Rocket Kinetics, inc.","Stellar Plasma-Assisted Rocket Kinetics, inc."
Parts/Description/spark_spt100,Text,,"The SPT-100 is the pinnacle in tiny (0.625m-class) Ion engines, providing high Isp and low thrust with an integral toroidal xenon tank. Strap this little guy onto a probe core and be sure to bring plenty of electrons because you're in for a long slow ride to anywhere!","Le SPT-100 a le dessus dans la catégorie des très petits (0.625m) moteurs ioniques, avec une haute impulsion spécifique et une poussée faible ainsi qu'un réservoir toroïdal de xénon intégré. Accrochez ce petit gars à vos sondes et assurez-vous d'avoir du stock d'électrons car vous allez aller de partout bien lentement!"


In Unity editor start with a fresh scene. This will hold your entire parts pack.
Parts/Title/spark_x3,Text,,X3 NHT,X3 NHT
Parts/Subtitle/spark_x3,Text,,Three-Channel Nested Hall Effect Thruster,Moteur à Effet Hall à 3 Canaux Concentriques
Parts/Manufacturer/spark_x3,Text,,"Stellar Plasma-Assisted Rocket Kinetics, inc.","Stellar Plasma-Assisted Rocket Kinetics, inc."
Parts/Description/spark_x3,Text,,"The SPARK X3 is the pinnacle in small (1.25m-class) Ion engines, providing high Isp and low thrust. Strap this bad boy onto your large probe or small capsule and be sure to bring plenty of electrons because you're in for a long slow ride to anywhere!","Le SPARK X3 a le dessus dans la catégorie des petits (1.25m) moteurs ioniques, avec une haute impulsion spécifique et une poussée faible ainsi qu'un réservoir toroïdal de xénon intégré. Accrochez ce petit gars à vos grandes sondes et assurez-vous d'avoir du stock d'électrons car vous allez aller de partout bien lentement!"
</syntaxhighlight>


Part Pack Prep
== Primary Part Production (Basic part setup) ==
This part of the tutorial will go over how to create a basic KSP2 part in Unity.
# '''Create Root Part Object''': Create an empty game object under your scene (R-click '''Scene''': '''Game Object''' '''>''' '''Create Empty'''). Name this object the same as your part.
## Recommended naming scheme: <code><mod_name>_<part_name></code>. If your mod is called <code>My Awesome Mod</code> and your part title is <code>My Part</code>, then your part name might be <code>my_awesome_mod_my_part</code> for example. Part names must be unique, though you can have any descriptive title you like (that is done later in Localization). A naming scheme like this helps to prevent naming collisions in case anyone else might make a part they want to call <code>My Part</code>, like yours.
# '''Create Model Object''': Create an empty game object under your root part object and name this one <code>model</code>.
# '''Create Part Folder''': Create a Part folder named for your part inside your Parts folder. E.g., <code>Assets\MyAwesomeMod\Parts\MyPart</code>, or <code>Assets\MyAwesomeMod\Parts\ThisGroup\MyPart</code> if you’re grouping parts.


# '''Create Unity Project''': Create a new Unity project with an empty scene. You can use the sample scene.
=== Mesh setup ===
# '''Install Addressables package''': Open Window > Package Manger and select Packages: Unity Registry. Search for addressables and click Install.
# '''Bring in the Mesh''': Drag a copy of your part’s FBX file into the part folder. If you have baked textures for your part that go with the FBX (meaning, they’re based on the UV Unwrap specific to that FBX), then drag those into this same folder along with the FBX.
# '''Install ThunderKit in Unity'''. See Installing Tunderkit from the Getting Ready section of the KSP2 Part Tutorial linked above for details.
# '''Create Part Mesh Object''': Drag a copy of the part's FBX from the Part folder in Unity to the <code>model</code> object created in step 2. Don’t drag the FBX file from your computer’s files system, you need to use the copy you just placed in step 4. This will create a '''prefab''' for your part as a child of the model object.
# '''Install Unity KSP Tools in Unity'''. See Installing KSP2 Unity Tools from the Getting Ready section of the KSP2 Part Tutorial linked above for details.
# '''Unpack Prefab''': Right-Click part object: '''Prefab > Unpack'''.
## Go to https://github.com/SpaceWarpDev/KSP2UnityTools/releases
# '''Remove Unnecessary Things''': Remove any objects that came in with the FBX that you don’t actually need in the game like lights, cameras, empty nodes, etc. If it’s not an actual object you want the game to render, then delete it.
## Download latest KSP2 Unity Tools version
# '''Orient Part''': Your model will appear in the Unity scene oriented as you built it in Blender, but this may not be the way you want it to be oriented in the game. If you need changes to the position, rotation, or scale of the part do those now using the Transform panel within the Inspector Window with your part object selected. For example, to flip a part over just give a rotation of 180 in Z, etc.
## Double click the KSPUTxxx.unitypackage file to install it in the open unity project
# '''Install LuxShader in Unity'''. Place a copy of “LuxShader” somewhere in your Unity project. I put mine in my Parts Pack folder.
# '''Configure Unity Project''': ''This is essential, or even if you can build a part it won’t load in the game''. You can check your settings like this. Go to the Addressables Window (click the Unity menu for Window > Asset Management > Addressables > Group). Select the Default Local Group in Addressables Groups window. In the Inspector window, check the following properties for your parts pack mod.
## Build Path: Should be set to Library/com.unity.addressables/aa/Windows/StandaloneWindows64
## Load Path: Should be set to {SpaceWarpPaths.<your_mod_id>}/addressables/StandaloneWindows64
### For your_mod_id, it is imperative that you replace any “.” Or “-“ with “_”. Only the very first “.” Between “SpaceWarpPaths” and “<your_mod_id>” is allowed as SpaceWarpPaths is a class, and classes can have variables, but variables can’t have any “.” or “-“ in their name (for reasons that should be obvious). If you’ve got any characters that can’t be used in a C# variable name In your mod ID (e.g. com.github.schlosrat.SPARK, etc.), then just replace those with underscores and you’ll be OK.
## Make sure the Load Path is ''not'' {UnityEngine.AddressableAssets.Addressables.RuntimePath}/StandaloneWindows64 (which it seems to want to be…) If you need to change these it’s best to create a new profile which you can then apply to the Build & Load Paths drop down. To do that go to the Addressables Groups windows and on the Profile: Default dropdown pick Manage Profiles.


This will open up the Addressables Profiles panel. Create a New Profile by clicking the Create dropdown and picking Build and Load Path Variables (All Profiles)
=== Material setup ===
This will give you the following dialog box which you can type into, but be forewarned – it will vanish if it loses focus, so save as soon as you’ve set the Prefix Name (which was set to SpaceWarp above but is in the fresh, clean unmodified form below)
Once you’ve saved it, you can edit the SpaceWarp.BuildPath to match what’s shown above and the SpaceWarp.LoadPath to be {SpaceWarpPaths.<your_mod_id>}/addressables/StandaloneWindows64.


With these setup, you can go back to the Inspector for the Default Local Group and set the Build & Load Paths setting to SpaceWarp in the dropdown menu so that your configuration looks like this
# '''Create the material:''' Create a material in Unity for each material in Substance Painter (right-click on the project window > '''Create > Material''')
# '''Select the KSP2 Shader:''' Select all materials. In the '''Inspector''' window, in '''Shader''', select '''KSP > Parts > Paintable'''.
# '''Apply the textures:''' For each material, assign the textures to the '''Albedo''', '''Metallic''', '''Normal''', '''Emissive''' and '''Paintable''' channels.
## For Normal channels, click the '''Fix Now''' button.
#'''Set properties''': Set '''Metallic/Smoothness Map''' to 1 and check '''Use PaintMask for Paint Smoothness'''.
# '''Apply materials:''' For each mesh, assign the materials to the corresponding locations. Click on '''Apply'''.


7. Create Parts Pack folder: Under Assets, create a folder (L-Click Assets: Create > Folder) and name it the same as your parts pack mod. If your mod is called My Awesome Mod then this would be Assets\MyAwesomeMod.
=== Collider setup ===
8. Create Materials folder: Inside your parts pack folder create a folder called Materials. You’ll be storing the textures and materials you need there. E.g., Assets\MyAwesomeMod\Materials. This is just to aid in organization.
# '''Create Collider Object''': R-Click <code>mesh</code> object: Create Empty. Name this object <code>col</code>.
9. Create Parts folder: Inside your parts pack folder create a folder called Parts. You’ll be storing the part meshes and related things there. E.g., Assets\MyAwesomeMod\Parts. You can have whatever organization you like here, so if you want to group some parts you might create group folders within Parts (e.g., Methalox Engines, Nuclear Engines, Ion Engines, etc.). This is just to aid in organization and is optional.
# '''Create Collider''': In the Inspector window for the col object click “Add Component”. Search for Mesh Collider and pick it. This will create a Mesh Collider component in the col object. Click the arrowhead to the left of it to expand it and see its properties.
10. Create Plugin Folder and Content: In your KSP2 install’s BepInEx\plugins folder create a folder for your mod. You can name this whatever you like, but it should be unique. This will be the “mod” players will install to have your parts pack and we’ll refer to it as the Base Plugin Folder in this guide.
## '''Select Mesh''': In the Mesh Collider properties select the Mesh you want to use. If your FBX is all one object you can pick that, or you can pick a suitable primitive like cube or cylinder, etc.
a. Create swinfo.json: In the base plugin folder create a swinfo.json file with content like this:
## '''Position, Rotation, and Scale''': Set the position, rotation, and scale of the mesh to encompass the part. You should see a green mesh represented in the Scene window to help guide you to make sure you’ve got the right position, rotation, and scale.
## '''Convex''': Check “Convex”


{
=== Core Part Data setup ===
  "spec": "1.3",
# '''Add Core Part Data''': Select your Root Part Object. In the Inspector Window click “Add Component”. Search for “Core Part Data” and pick that. Open it up and configure as follows:
  "mod_id": "com.github.schlosrat.SPARK",
## '''Part Name''': The Part Name needs to be the same as what you’ve used for the Root Part object, i.e., my_awesome_mod_my_part or whatever you used.
  "author": "schlosrat",
## '''Author''': Use what you like here, typically your KSP Forum screen name or whatever you go by as your modding author name.
  "name": "Stellar Plasma-Assisted Rocket Kinetics",
## '''Category''': Select an appropriate category for your part.
  "description": "High ISP engines for your low thrust needs",
## '''Family''': If you wish to identify a “Family” for your part, this needs to be a particular string. You can find examples in the game’s files, or part JSONS, or ask in the KSP2 Modding Society discord to get this information.
  "source": "https://github.com/schlosrat/SPARK",
## '''Co Lift, Co Mass, Co Pressure, etc.''': These parameters allow you to set the Center of Lift, Center of Mass, Center of Pressure, etc. Adjust these to get the markers in the Unity scene where they should be for your part. Typically, Co Pressure and Co Lift are in the same place.
  "version": "0.0.1",
## '''Fuel Cross Feed''': Check if fuel should be able to transit through your part on the way to other parts. Typically set to true, but not always.
  "version_check": "https://raw.githubusercontent.com/schlosrat/SPARK/main/SPARK/swinfo.json",
## '''Mass''': Set this in metric tons, not Kg.
  "ksp2_version": {
## '''Attach Rules''': Check the types of attachment your part should allow. Checking “Stack” or “Srf Attach” will allow your part to attach in a stack or to a surface. Checking “Allow Stack”, “Allow Srf Attach”, etc. will allow other parts to stack attach or surface attach respectively. ''Currently (?) Allow Collision, Allow Dock, Allow Rotate, and Allow Root have no effect in game (check this)''.
    "min": "0.1.3.2",
## '''Attach Nodes''': If Stack is checked above, then you need a “top” and a “bottom” node, if Srf Attach is checked above, then you need a “srfAttach” node. Note, node names are case sensitive and having a node is not enough by itself, you do also need the corresponding Attach Rule set true or the node will have no effect. Under Attach Nodes click the + button to add a blank node and configure as needed.
    "max": "*"
### '''Node ID''': “top”, “bottom”, “srfAttach”, etc. (case sensitive!)
  },
### '''Node Type''': Select as appropriate. (Stack for Stack, Surface for Surface…)
  "dependencies": [
### '''Attach Method''': Select Fixed_Joint for Stack and Hinge_Joint for Surface Attach.
    {
### '''Is Multi Joint''': In general set to True for stack attach to help prevent noodle rockets.
      "id": "com.github.x606.spacewarp",
### '''Multi Joint Max Joint''': ''Set to 3 if you set Is Multi Joint to true?''
      "version": {
### '''Position''': Set as appropriate. Should be on the skin or outside of the part where you would expect to find it on your part in the VAB.
        "min": "1.4.0",
### '''Orientation''': Set as appropriate. The Orientation vector should be a unit vector (length 1) pointing in the direction of the part that will attach to the node, so pointing away from your part.
        "max": "*"
### '''Size''': Affects rigidity of your part. If your part is connected to another part with the same “size” node, then rigidity will be optimal, and otherwise it will be suboptimal.
### '''Visual Size''': Set the same as Size.
### '''Is Resource Crossfeed''': Set as needed for this node.
### '''Is Rigid''': Set as needed for this node.
### ''Rinse and Repeat'': Subsequent nodes created with the + button will inherit settings from the last node made, so this may accelerate the process as you just need to change the Node Id, Position and Orientation for new nodes that are similar to the previous created node.
# '''Add Module_Color''': You need this to be able to paint your part with base and accent colors. As above, click Add Component and search for Module Color.
# '''Add Module_Drag''': All parts need this. As above, click Add Component and search for Module Color.
 
== Crazy Customization (Adding functionalities to the part) ==
# '''Add other modules as needed'''. For example, if your part is an engine you’ll also need:
## '''thrustTransform''' object: Create an empty game object as a child of the root part and name it thrustTransform.
## '''Throttle VFX Manager''': Configure as needed (?). No need to drag anywhere, it just need to be a component for the part.
## '''Flameout VFX Data''': Drag this up to the Flameout VFX property in Module_Engine.
## '''Module_Gimbal''': Configure details as needed (e.g., Gimbal Range and Gimbal Speed, etc.), then drag this up to the Gimbal property in Module_Engine.
## '''Module_Generator''': Configure details as needed, then drag this up to the Alternator property in Module_Engine.
## '''Module_Fairing''': Unless you plan for your engine to only ever be on the very first stage (i.e., a big booster), then you probably want it to have a fairing so it can be used in the second stage and above. Figure 3 shows an example for a working fairing from a Size Small (1.25m-class) engine that was created using this process. Shown below is the corresponding example of the PartComponentModule_Fairing portion of  the part JSON. As can be seen in Figure 3, the data fields that need to be populated in the Unity Editor include only a subset of what's in the auto-generated JSON. That said, everything the editor needs is in the example below and presenting it this way makes copy/paste possible where unity editor graphic would not allow that.[[File:Module Fairing Data.png|thumb|Figure 3: Example Fairing Data in Unity Editor]]<syntaxhighlight lang="json">
      {
        "Name": "PartComponentModule_Fairing",
        "ComponentType": "KSP.Sim.impl.PartComponentModule_Fairing, Assembly-CSharp, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null",
        "BehaviourType": "KSP.Modules.Module_Fairing, Assembly-CSharp, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null",
        "ModuleData": [
          {
            "Name": "Data_Fairing",
            "ModuleType": "KSP.Modules.Module_Fairing, Assembly-CSharp, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null",
            "DataType": "KSP.Modules.Data_Fairing, Assembly-CSharp, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null",
            "Data": null,
            "DataObject": {
              "$type": "KSP.Modules.Data_Fairing, Assembly-CSharp",
              "FairingEnabled": {
                "ContextKey": "FairingEnabled",
                "storedValue": true
              },
              "FairingConstructionType": {
                "ContextKey": "FairingConstructionType",
                "storedValue": "CUSTOM"
              },
              "FloatingNodeEnabled": {
                "ContextKey": "FloatingNodeEnabled",
                "storedValue": true
              },
              "Length": {
                "ContextKey": "Length",
                "storedValue": -1.0
              },
              "DeployType": {
                "ContextKey": "DeployType",
                "storedValue": "Clamshellx4"
              },
              "EjectionForce": {
                "ContextKey": "EjectionForce",
                "storedValue": 100.0
              },
              "IsStagingEnabled": {
                "ContextKey": "IsStagingEnabled",
                "storedValue": true
              },
              "IsDeployed": {
                "ContextKey": "IsDeployed",
                "storedValue": false
              },
              "FloatingNodeSize": 1.0,
              "FloatingAttachNodeTag": "bottom",
              "FloatingNodePosition": {
                "x": 0.0,
                "y": 0.0,
                "z": 0.0
              },
              "FloatingNodeDirection": {
                "x": 0.0,
                "y": -1.0,
                "z": 0.0
              },
              "FloatingNodeIsMultiJoint": false,
              "FloatingNodeMultiJointMaxCount": 3,
              "FloatingNodeMultiJointOffset": 1.0,
              "FairingNode": "top",
              "NoseTip": 0.5,
              "EdgeWarp": 0.02,
              "AberrantNormalLimit": 45.0,
              "LocalUpAxis": {
                "x": 0.0,
                "y": -1.0,
                "z": 0.0
              },
              "Pivot": {
                "x": 0.0,
                "y": 0.0,
                "z": 0.0
              },
              "BaseModelTransformName": "Base",
              "CapRadius": 0.375,
              "BaseRadius": 0.625,
              "CloseRadius": 0.375,
              "MaxRadius": 6.0,
              "SnapThreshold": 0.25,
              "CreateShellColliders": false,
              "NumberOfCollidersPerCrossSection": 12,
              "MinHeightRadiusRatio": 0.07,
              "CrossSectionHeightMin": 0.3,
              "CrossSectionHeightMax": 1.1,
              "AerodynamicallyShieldContents": false,
              "ConeSweepRays": 120,
              "ConeSweepPrecision": 10.0,
              "AmountOfCollidersPerArc": 1,
              "ShouldCapOnAutoGenerate": false,
              "IsCapped": false,
              "IsShroud": true,
              "MassAreaRatio": 0.0,
              "FairingSideCount": 24,
              "FairingLengthSnapIncrement": 0.125,
              "FairingRadiusSnapIncrement": 0.125,
              "FairingSmoothingAngle": 35.0,
              "FairingThickness": 0.025,
              "FairingStartHeight": 0.0,
              "AllowConstructionTypeChange": true,
              "AllowFloatingNodeChange": true,
              "DefaultFairingEnabledToggle": true,
              "DefaultAutoConstruction": true,
              "DefaultDeployType": "Shroud",
              "DefaultFloatingNodeState": true,
              "LengthEditMinimum": 0.0,
              "LengthEditMaximum": 0.001,
              "LengthEditDefault": 1.0,
              "StageToggleDefault": false,
              "MaxAutoFairingTargetRadius": -1,
              "MinAutoFairingTargetRadius": -1,
              "CrossSections": [],
              "MassModifierAmount": 0.0,
              "DragCubeIndex": -1,
              "ModuleType": "KSP.Modules.Module_Fairing, Assembly-CSharp, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null",
              "MassModifier": 0.0,
              "DataType": "KSP.Modules.Data_Fairing, Assembly-CSharp, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null",
              "IsActiveInStagingProp": {
                "ContextKey": null,
                "storedValue": false
              }
            }
          }
        ]
       }
       }
    },
</syntaxhighlight>Of the information show above, the parts you will most likely need to customize for your fairing are Base Radius, Cross Section Height Max, and Fairing Thickness - the rest is likely to be the same for all your engines with one other important thing to note. The example above is for an engine that was made in Blender with the nozzle bell facing upwards. If your engine was made with the nozzle bell facing down, then you will need the '''Local Up Axis''' to be -1 in Y and you will also need set the '''Fairing Node''' to bottom.
    {
      "id": "lfo",
      "version": {
        "min": "0.2.0",
        "max": "*"
      }
    }
  ]
}
b. Create “addressables” folder: In the base plugin folder, create an “addressables” sub-folder. This will be where you put the files created by Unity after a build (discussed below).
c. Create “localizations” folder: In the base plugin folder, create an “localizations” sub-folder (NOTE! This is currently required to be exactly this – localizations (plural), not localization (singular). This will be where you put the localization files (e.g., english.csv, spanish.csv, etc.).
d. Create Localization file(s): Localization files are CSV files following a particular format. These must have lines ending with LF not LF/NL, and they must reference the same <part_name> you use in the Part Production process below. There are some other restrictions for content, particularly that if you want a string that contains a “,” that string needs to be enclosed in quotes or the comma will mess with how the strings are parsed. These files are where the part’s Title, Subtitle, Manufacturer, and Description are configured. Here’s an example:
Key,Type,Desc,English
 
Parts/Title/spark_spt100,Text,,SPT-100
Parts/Subtitle/ spark_spt100,Text,,Hall Effect Thruster with Xenon Tank
Parts/Manufacturer/spark_spt100,Text,,"Stellar Plasma-Assisted Rocket Kinetics, inc."
Parts/Description/spark_spt100,Text,,"The SPT-100 is the pinnacle in tiny (0.625m-class) Ion engines, providing high Isp and low thrust with an integral toroidal xenon tank. Strap this little guy onto a probe core and be sure to bring plenty of electrons because you're in for a long slow ride to anywhere!"


Parts/Title/spark_x3,Text,,X3 NHT
== Final Flourishes (Final steps) ==
Parts/Subtitle/spark_x3,Text,,Three-Channel Nested Hall Effect Thruster
# '''Apply Options''': Select the root part and in the Inspector window near the top on the '''Prefab''' line, click the '''Overrides''' dropdown and pick '''Apply All'''. If this option is not available, then you’ve got nothing you need to do here. Move along, move along.
Parts/Manufacturer/spark_x3,Text,,"Stellar Plasma-Assisted Rocket Kinetics, inc."
# '''Save Part JSON''': Click '''Save Part JSON''' button at the bottom of the '''Core Part Data''' module. This will put the resulting part JSON in the <code>Assets</code> folder for your Unity project. You need to do this any time you’ve edited the '''Core Part Data''' module (or also a module it depends on?).
Parts/Description/spark_x3,Text,,"The SPARK X3 is the pinnacle in small (1.25m-class) Ion engines, providing high Isp and low thrust. Strap this bad boy onto your large probe or small capsule and be sure to bring plenty of electrons because you're in for a long slow ride to anywhere!"
# '''Make Prefab''': Grab the root part object and drag it to the Unity project <code>Assets</code> folder.
e.
# '''Add Part Icon''': Create an icon for your part that the game will use in the parts picker. This needs to be a PNG file with specific dimensions. It should conform to the style used by other parts in the game. However you do this, you need to name the file <code><part_name>_icon.png</code>, and you need to drag that file into the <code>Assets</code> folder in Unity.
Part Production
## '''Convert Icon to Sprite''': Select the part icon in the Assets folder. In the Inspector window, click the pulldown menu for Texture Type and change this from Default to Sprite (2D and UI), then click '''Apply'''. If this is not done, your lovely icon will not display!
1. Create Root Part Object: Create an empty game object under your scene (L-click Scene: Game Object > Create Empty). Name this object the same as your part.
# '''Make Root Part Addressable:''' Select the root part’s prefab in the <code>Assets</code> folder and in the '''Inspector''' window check the box for '''Addressable'''.
a. Recommended naming scheme: <mod_name>_<part_name>. If your mod is called “My Awesome Mod” and your part title is “My Part”, then your part name might be my_awesome_mod_my_part for example. Part names must be unique, though you can have any descriptive title you like (that is done later in Localization). A naming scheme like this helps to prevent naming collisions in case anyone else might make a part they want to call “My Part”, like yours.
# '''Make JSON Addressable''': Select the part’s JSON in the <code>Assets</code> folder and in the '''Inspector''' window check the box for '''Addressable'''.
2. Create Model Object: Create an empty game object under your root part object and name this one “model”.
# '''Make the Icon Addressable''': Select the root part’s icon in the <code>Assets</code> folder and in the '''Inspector''' window check the box for '''Addressable'''.
3. Create Part Folder: Create a Part folder named for your part inside your Parts folder. E.g., Assets\MyAwesomeMod\Parts\MyPart, or Assets\MyAwesomeMod\Parts\ThisGroup\MyPart if you’re grouping parts.
# '''Configure Addressable Properties''': In the '''Addressables Groups''' expand your mod's Group and find your part.
4. Bring in FBX: Drag a copy of your part’s FBX file into the part folder.
## '''Group Name \ Addressable Name''': Change the information in the '''Group Name''' '''\''' '''Addressable Name''' from <code>Assets/<part_name>*</code> to be just <code><part_name>*</code>. So <code>Assets/<part_name>.prefab</code> becomes <code><part_name>.prefab</code>, and so forth. You can leave the <code>Assets/</code> part of the path definition alone for each of these, that’s as it should be. The value for the '''Addressable Name''' needs to be the same as the file name it’s associated with and must not include any path parts. All of these need to be based on the '''Part Name''' established in the '''Core Part Data''' module.
a. If you have baked textures for your part that go with the FBX (meaning, they’re based on the UV Unwrap specific to that FBX), then drag those into this same file with the FBX.
## '''Labels''': For the JSON set this to <code>parts_data</code>. Leave it blank for the prefab. If <code>parts_data</code> is not an option in the dropdown for Labels, then click '''Manage Labels''', click the + button to add a new label, and set the '''Label Name''' to <code>parts_data</code>. Click '''Save'''.
5. Create Part Object: Drag a copy of the part FBX from the Part folder in Unity to the model object created in step 2. Don’t drag the FBX file from your computer’s files system, you need to use the copy you just placed in step 4. This will create a prefab for your part as a child of the model object.
# '''Build Mod or Build And Test''': Select either Build Mod or Build And Test to have KSP2 Unity Tools prepare and deploy your parts pack mod.
6. Unpack Prefab: Left-Click part object: Prefab > Unpack.
# '''Launch Game and Test!'''
7. Remove Unnecessary Things: Remove any parts that came in with the FBX that you don’t actually need in the game like lights, cameras, empty nodes, etc. If it’s not an actual part you want the game to render, then delete it.
# '''''Rinse and Repeat for Additional Parts'''''
8. Orient Part: Your model will appear in the Unity scene oriented as you built it in Blender, but this may not be the way you want it to be oriented in the game. If you need changes to the position, rotation, or scale of the part do those now using the Transform panel within the Inspector Window with your part object selected. For example, to flip a part over just give a rotation of 180 in Z, etc.
== '''Mentions''' ==
9. Create Mesh Object: L-Click part object: Create Empty. Name this object “mesh”.
This guide is based on the videos: How to create parts for KSP2 and How to make engines for KSP2 (see: [[Part modding videos (tutorials)]]), and on other notes and guidance from the KSP2 Modding Society discord.
10. Create Collider Object: L-Click mesh object: Create Empty. Name this object “col”.
11. Create Collider: In the Inspector window for the col object click “Add Component”. Search for Mesh Collider and pick it. This will create a Mesh Collider component in the col object. Click the arrowhead to the left of it to expand it and see its properties.
a. Select Mesh: In the Mesh Collider properties select the Mesh you want to use. If your FBX is all one object you can pick that, or you can pick a suitable primitive like cube or cylinder, etc.
b. Position, Rotation, and Scale: Set the position, rotation, and scale of the mesh to encompass the part. You should see a green mesh represented in the Scene window to help guide you to make sure you’ve got the right position, rotation, and scale.
c. Convex: Check “Convex”
12. Add Core Part Data: Select your Root Part Object. In the Inspector Window click “Add Component”. Search for “Core Part Data” and pick that. Open it up and configure as follows:
a. Part Name: The Part Name needs to be the same as what you’ve used for the Root Part object, i.e., my_awesome_mod_my_part or whatever you used.
b. Author: Use what you like here, typically your KSP Forum screen name or whatever you go by as your modding author name.
c. Category: Select an appropriate category for your part.
d. Family: If you wish to identify a “Family” for your part, this needs to be a particular string. You can find examples in the game’s files, or part JSONS, or ask in the KSP2 Modding Society discord to get this information.
e. Co Lift, Co Mass, Co Pressure, etc.: These parameters allow you to set the Center of Lift, Center of Mass, Center of Pressure, etc. Adjust these to get the markers in the Unity scene where they should be for your part. Typically, Co Pressure and Co Lift are in the same place.
f. Fuel Cross Feed: Check if fuel should be able to transit through your part on the way to other parts. Typically set to true, but not always.
g. Mass: Set this in metric tons, not Kg.
h. Attach Rules: Check the types of attachment your part should allow. Checking “Stack” or “Srf Attach” will allow your part to attach in a stack or to a surface. Checking “Allow Stack”, “Allow Srf Attach”, etc. will allow other parts to stack attach or surface attach respectively. Currently (?) Allow Collision, Allow Dock, Allow Rotate, and Allow Root have no effect in game (check this).
i. Attach Nodes: If Stack is checked above, then you need a “top” and a “bottom” node, if Srf Attach is checked above, then you need a “srfAttach” node. Note, node names are case sensitive and having a node is not enough by itself, you do also need the corresponding Attach Rule set true or the node will have no effect. Under Attach Nodes click the + button to add a blank node and configure as needed.
i. Node ID: “top”, “bottom”, “srfAttach”, etc. (case sensitive!)
ii. Node Type: Select as appropriate. (Stack for Stack, Surface for Surface…)
iii. Attach Method: Select Fixed_Joint for Stack and Hinge_Joint for Surface Attach.
iv. Is Multi Joint: In general set to True for stack attach to help prevent noodle rockets.
v. Multi Joint Max Joint: Set to 3 if you set Is Multi Joint to true?
vi. Position: Set as appropriate. Should be on the skin or outside of the part where you would expect to find it on your part in the VAB.
vii. Orientation: Set as appropriate. The Orientation vector should be a unit vector (length 1) pointing in the direction of the part that will attach to the node, so pointing away from your part.
viii. Size: Affects rigidity of your part. If your part is connected to another part with the same “size” node, then rigidity will be optimal, and otherwise it will be suboptimal.
ix. Visual Size: Set the same as Size.
x. Is Resource Crossfeed: Set as needed for this node.
xi. Is Rigid: Set as needed for this node.
xii. Rinse and Repeat: Subsequent nodes created with the + button will inherit settings from the last node made, so this may accelerate the process as you just need to change the Node Id, Position and Orientation for new nodes that are similar to the previous created node.
13. Add Module_Color: You need this to be able to paint your part with base and accent colors. As above, click Add Component and search for Module Color.
14. Add Module_Drag: All parts need this. As above, click Add Component and search for Module Color.
15. Add other modules as needed. For example, if your part is an engine you’ll also need:
a. thrustTransform object: Create an empty game object as a child of the root part and name it thrustTransform.
b. Throttle VFX Manager: Configure as needed (?). No need to drag anywhere, it just need to be a component for the part.
c. Flameout VFX Data: Drag this up to the Flameout VFX property in Module_Engine.
d. Module_Gimbal: Configure details as needed (e.g., Gimbal Range and Gimbal Speed, etc.), then drag this up to the Gimbal property in Module_Engine.
e. Module_Generator: Configure details as needed, then drag this up to the Alternator property in Module_Engine.
16. Apply Options: Select the root part and in the Inspector window near the top on the Prefab line, click the Overrides dropdown and pick “Apply All”. If this option is not available, then you’ve got nothing you need to do here. Move along, move along.
17. Save Part JSON: Click Save Part JSON button at the bottom of the Core Part Data module. This will put the resulting part JSON in the Assets folder for your Unity project. You need to do this any time you’ve edited the Core Part Data module (or also a module it depends on?).
18. Make Prefab: Grab the root part object and drag it to the Unity project Assets folder.
19. Add Part Icon: Create an icon for your part that the game will use in the parts picker. This needs to be a PNG file with specific dimensions. It should conform to the style used by other parts in the game. However you do this, you need to name the file <part_name>_icon.png, and you need to drag that file into the Assets folder in Unity.
a. Convert Icon to Sprite: Select the part icon in the Assets folder. In the Inspector window, click the pulldown menu for Texture Type and change this from Default to Sprite (2D and UI), then click Apply. If this is not done, your lovely icon will not display!  
20. Make Root Part Addressable: Select the root part’s prefab in the Assets folder and in the Inspector window check the box for Addressable.
21. Make JSON Addressable: Select the part’s JSON in the Assets folder and in the Inspector window check the box for Addressable.
22. Make the Icon Addressable: Select the root part’s icon in the Assets folder and in the Inspector window check the box for Addressable.
23. Configure Addressable Properties: In the Addressables Groups expand the Default Local Group and find your part.
a. Group Name \ Addressable Name: Change the information in the Group Name \ Addressable Name from “Assets/<part_name>*to be just <part_name>*. So “Assets/<part_name>.prefab” becomes <part_name>.prefab”, and so forth. You can leave the “Assets/part of the path definition alone for each of these, that’s as it should be. The value for the Addressable Name needs to be the same as the file name it’s associated with and must not include any path parts. All of these need to be based on the Part Name established in the Core Part Data module.
b. Labels: For the JSON set this to parts_data. Leave it blank for the prefab. If parts_data is not an option in the dropdown for Labels, then click Manage Labels, click the + button to add a new label, and set the Label Name to “parts_data”. Click Save.
24. Build Parts Pack Addressables: In the Addressables Groups window, click the drop down for Build and pick New Build > Default Build Script. If you have any unsaved changes Unity will prompt you to save, and then it will build the addressables package for your mod.
25. Copy Addressables to Plugin Folder: If the Build step above was successful, the do the following.
a. Delete Old Stuff: In Windows Explorer, navigate to your Base Plugin Folder’s addressables folder (<KSP2 Game Dir>\BepInEx\plugins\<your Mod Name>\addressables) and delete any content inside that folder. There may be four things: AddressablesLink (folder), StabaloneWindows64 (folder), catalog.json (file), and setting.json (file).
b. Copy New Stuff: In Windows Explorer, navigate to your Unity project’s folder and from there to Library\com.unity.addressables\aa\Windows folder. If the build process above was successful, then there will be four things in this folder: AddressablesLink (folder), StabaloneWindows64 (folder), catalog.json (file), and setting.json (file). You will need to copy all of these to your Base Plugin Folder’s addressables folder.
26. Launch Game and Test!
27. Rinse and Repeat for Additional Parts


Blender Basics for Beginners
Also the web pages: [https://luxstice.notion.site/KSP2-Part-tutorial-almost-from-scratch-1f336b7c97ae4280afb6a4e3aa6080b2 KSP2 Part tutorial (almost) from scratch], and [https://luxstice.notion.site/Tutorial-My-First-Part-5f0cf456d7f4443d8c92658c7cc58314 Tutorial: My First Part]
1. If you’re making an engine, put an empty single arrow at the world origin facing in the direction your engine will fire, then move it along the axis of the engine so that it’s at the exit of the engine. Name this object thrustTransform, and keep in mind that size does not matter.
2. Make sure all the part objects in your blend have the same material, and that the material has a good name. The material you use for this can be a default material, nothing special is needed.
3. Make sure any Boolean modifiers are applied, but array modifiers can be left unapplied.
4. Make sure you’ve got a good UV Map (use UV Smart Project) or an excellent one (UV Smart Project followed by Pack Islands). Recommended island spacing for both is 0.003
5. When exporting your FBX for use in Substance Painter or any other tool, select only those objects that are actually part of your part and then be sure to check the box for Limit to Selected Objects
Texturing Tips
1. Make sure the Kerbal_Space_Program_2_Parts_Paintable.spt file is in your “Documents>Adobe>Substance Painter>Assets>templates” folder. If the templates folder doesn’t exist yet, then create one under Assets and drop that file in it.
2. Launch Substance Painter and create a New project.
a. In the project creation dialog pick the Kerbal_Space_Program_2_Parts_Paintable template
b. Select your FBX file
c. Make sure the Document Resolution is sufficient for your texture maps (highly recommend 4096! Any less and painted on bolts look like crap)
d. Click Open.


3. Make sure the UV Map looks OK in Substance Painter. If it doesn’t, you need to go back to Blender and re-unwrap then export a new FBX and return to step 2 above.
4. Use the File > Import resources… menu to bring up the Import Resources dialog (shown below) and click on the Add Resource button to bring in the KSP2_Part.spsm file. Repeat this step for the Kerbal Space Program 2 - Standard Damaged.spexp file. Both of these should go into your library as you’re going to need them with every part you make.
a. Use the Add Resources button to find the resource you want to import
b. Select the file and click the Open button
c. Repeat steps (a) and (b) above for as many things as you would like to import
d. Select the destination to Import your resources to (hint: library ‘your_assests’ for these!)
e. Click Import
5. Use the process in Step 4 above to bring in any textures you want to use. There are many you can peruse and download for free here: https://substance3d.adobe.com/community-assets
a. Note: When importing textures, you can either put them in your library so that you can use them in multiple projects, or you can put them directly in your project if you don’t want them in your library. Either way works.
b. You can repeat this step any time, and as often as needed, so don’t worry if you don’t have all the textures you need at the start. It’s also possible to make textures within Substance Painter
6. Apply the KSP2 Part material to your whole object and once done delete the default later that Substance Painter provided.
7. Locate the Bake button (looks like a croissant) in the upper right of Substance Painter and click it 
a. Uncheck the option to make an Opacity Mesh Map
b. Click the large blue Back Selected Textures button
c. When baking is done, click the Return to Painting Mode button
8. Expand the folder by the KSP2_Part smart material in the Layers panel
9. Expand the folder by the Base Material smart material within the KSP2 Part
10. Search for and select the materials you want to paint your part within the Assets panel.
a. Drag a material from the Assets panel onto the Base Material in the Layers panel.
b. Right click on that material and add a black mask to it, making sure the resulting mask is selected (there will be a blue border around it)
c. Pick the Polygon Fill tool from the tool bar and then select a option from Triangle Fill, Polygon Fill, Mesh Fill, and UV Chunk Fill.    Use the triangle to paint tris, the square (polygon) for quads, the cube (mesh) for entire objects, and the checkerboard (UV chunk) for connected faces on a side of surface.
11. Create a simple material: Example white boron nitride ceramic.
a. Click the Add Fill Layer button (paint can) 
b. Drag the new fill layer onto the Base Material folder.
c. In the Propertied panel under Materials, leave only color, roughness, and metal selected.
d. In the Properties panel set the base color to what you need (white in this case)
e. Set the metallic slider to 1.0.
f. Set the roughness slider to 0.3.
g. Add a black mask to the fill layer
h. Paint like a pro
12. Add “Nails”: This is where you add various surface details via the height map, and works for painting on nuts, bolts, fasteners, etc. All of these can be added as “alphas” like this
a. Import your alpha(s)!
b. In Layers, select the black mask for Nails in the Heightmap
c. Select Radial Symmetry (for applications around radially symmetric parts like I've got here)
d. Set your X, Y, Z point about which things will be symmetric, where Y is the vertical. (Switch the Show/Hide Manipulator on to help see where this point is in your model)
e. Set Mirror Y, the count you want, and the angular span the radial pattern should follow (360 for all the way around)
f. With the brush tool active, select your alpha
g. Move your cursor onto the part and see the red dots where the "Nails" will be placed.
h. Adjust the size of the "Nail" using [ and ] to make it smaller or larger respectively
i. Click where you want the first "Nail" to appear, this will place them all.
13. Add “Stickers”: This is where you add signs, stickers, decals, etc., where the sticker has a uniform color (like a radiation symbol, etc.)
a. Import your alpha(s)!
b. In Layers, under Base Material, add a fill layer for the sticker
i. Uncheck all material properties for the layer except color, metal, and rough
ii. Set the fill color to be what you want the sticker to look like
iii. Move the layer to be above any other Base Material layer that impacts the part(s) you’re going to apply stickers to (or it will be overwritten and you won’t see your stickers!)
c. Select Radial Symmetry (for applications around radially symmetric parts like I've got here)
d. Set your X, Y, Z point about which things will be symmetric, where Y is the vertical. (Switch the Show/Hide Manipulator on to help see where this point is in your model)
e. Set Mirror Y, the count you want, and the angular span the radial pattern should follow (360 for all the way around)
f. With the brush tool active, select your alpha
g. Move your cursor onto the part and see the red dots where the "Stickers" will be placed.
h. Adjust the size of the "Sticker" using [ and ] to make it smaller or larger respectively
i. Click where you want the first "Sticker" to appear, this will place them all.
[[Category:Parts and modules]]
[[Category:Parts and modules]]

Latest revision as of 19:41, 8 March 2024

The aim of this tutorial is to guide you through the process of creating your first part mod for Kerbal Space Program 2. We will go over everything from creating the part's mesh and textures to setting up the part in Unity to be bundled into a KSP2 mod.

The way this tutorial will be structured is as a succession of steps which may link to other tutorials on this wiki, do not hesitate to keep this main page open while perusing the other tutorials. We aim at being as comprehensive as possible to enable you to make any part mod, but this page is still work in progress so stuff might be incomplete and/or subject to change in the future.

Magnificent Mesh Modelling (Creating the part asset)

The first step in your part mod creation journey is to create the part's "3D asset", meaning its 3D mesh, textures and eventual animations. You can use any software you're comfortable with to do so but here are some guides for some specific software if you desire:

  1. Modeling the mesh in Blender
  2. Texturing the mesh in Substance 3D Painter

Here are some resources that can be useful for modelling and texturing, regardless of the software you're using:

Part Pack Prep (Basic Unity part mod project setup)

Now that you have created a model and textures for your part, the next step is to set up a Unity project that will be used to configure your part and build the asset bundles that will contain your mod's contents.

  1. Creating the project: Follow the Setting up Unity tutorial to create and set up a project for part modding.
  2. Parts Pack folder: Under Assets, create a folder (R-Click Assets: Create > Folder) and name it the same as your parts pack mod. If your mod is called My Awesome Mod then this would be Assets\MyAwesomeMod.
  3. Materials folder: Inside your parts pack folder create a folder called Materials. You’ll be storing the textures and materials you need there. E.g., Assets\MyAwesomeMod\Materials. This is just to aid in organization.
  4. Parts folder: Inside your parts pack folder create a folder called Parts. You’ll be storing the part meshes and related things there. E.g., Assets\MyAwesomeMod\Parts. You can have whatever organization you like here, so if you want to group some parts you might create group folders within Parts (e.g., Methalox Engines, Nuclear Engines, Ion Engines, etc.). This is just to aid in organization and is optional.
  5. Plugin Folder and Content: In your Unity project's Assets\plugin_template folder create a localizations folder. NOTE! This is currently required to be exactly this – localizations (plural), not localization (singular). This will be where you put the localization file.
    1. Create Localization file: Localization files are CSV files following a particular format. These must have lines ending with LF not LF/NL, and they must reference the same <part_name> you use in the Part Production process below. There are some other restrictions for content, particularly that if you want a string that contains a “,” that string needs to be enclosed in quotes or the comma will mess with how the strings are parsed. These files are where the part’s Title, Subtitle, Manufacturer, and Description are configured. Here’s an example:
      Key,Type,Desc,English,French
      
      Parts/Title/spark_spt100,Text,,SPT-100,SPT-100
      Parts/Subtitle/spark_spt100,Text,,Hall Effect Thruster with Xenon Tank,Moteur à Effet Hall et Réservoir de Xénon
      Parts/Manufacturer/spark_spt100,Text,,"Stellar Plasma-Assisted Rocket Kinetics, inc.","Stellar Plasma-Assisted Rocket Kinetics, inc."
      Parts/Description/spark_spt100,Text,,"The SPT-100 is the pinnacle in tiny (0.625m-class) Ion engines, providing high Isp and low thrust with an integral toroidal xenon tank. Strap this little guy onto a probe core and be sure to bring plenty of electrons because you're in for a long slow ride to anywhere!","Le SPT-100 a le dessus dans la catégorie des très petits (0.625m) moteurs ioniques, avec une haute impulsion spécifique et une poussée faible ainsi qu'un réservoir toroïdal de xénon intégré. Accrochez ce petit gars à vos sondes et assurez-vous d'avoir du stock d'électrons car vous allez aller de partout bien lentement!"
      
      Parts/Title/spark_x3,Text,,X3 NHT,X3 NHT
      Parts/Subtitle/spark_x3,Text,,Three-Channel Nested Hall Effect Thruster,Moteur à Effet Hall à 3 Canaux Concentriques
      Parts/Manufacturer/spark_x3,Text,,"Stellar Plasma-Assisted Rocket Kinetics, inc.","Stellar Plasma-Assisted Rocket Kinetics, inc."
      Parts/Description/spark_x3,Text,,"The SPARK X3 is the pinnacle in small (1.25m-class) Ion engines, providing high Isp and low thrust. Strap this bad boy onto your large probe or small capsule and be sure to bring plenty of electrons because you're in for a long slow ride to anywhere!","Le SPARK X3 a le dessus dans la catégorie des petits (1.25m) moteurs ioniques, avec une haute impulsion spécifique et une poussée faible ainsi qu'un réservoir toroïdal de xénon intégré. Accrochez ce petit gars à vos grandes sondes et assurez-vous d'avoir du stock d'électrons car vous allez aller de partout bien lentement!"

Primary Part Production (Basic part setup)

This part of the tutorial will go over how to create a basic KSP2 part in Unity.

  1. Create Root Part Object: Create an empty game object under your scene (R-click Scene: Game Object > Create Empty). Name this object the same as your part.
    1. Recommended naming scheme: <mod_name>_<part_name>. If your mod is called My Awesome Mod and your part title is My Part, then your part name might be my_awesome_mod_my_part for example. Part names must be unique, though you can have any descriptive title you like (that is done later in Localization). A naming scheme like this helps to prevent naming collisions in case anyone else might make a part they want to call My Part, like yours.
  2. Create Model Object: Create an empty game object under your root part object and name this one model.
  3. Create Part Folder: Create a Part folder named for your part inside your Parts folder. E.g., Assets\MyAwesomeMod\Parts\MyPart, or Assets\MyAwesomeMod\Parts\ThisGroup\MyPart if you’re grouping parts.

Mesh setup

  1. Bring in the Mesh: Drag a copy of your part’s FBX file into the part folder. If you have baked textures for your part that go with the FBX (meaning, they’re based on the UV Unwrap specific to that FBX), then drag those into this same folder along with the FBX.
  2. Create Part Mesh Object: Drag a copy of the part's FBX from the Part folder in Unity to the model object created in step 2. Don’t drag the FBX file from your computer’s files system, you need to use the copy you just placed in step 4. This will create a prefab for your part as a child of the model object.
  3. Unpack Prefab: Right-Click part object: Prefab > Unpack.
  4. Remove Unnecessary Things: Remove any objects that came in with the FBX that you don’t actually need in the game like lights, cameras, empty nodes, etc. If it’s not an actual object you want the game to render, then delete it.
  5. Orient Part: Your model will appear in the Unity scene oriented as you built it in Blender, but this may not be the way you want it to be oriented in the game. If you need changes to the position, rotation, or scale of the part do those now using the Transform panel within the Inspector Window with your part object selected. For example, to flip a part over just give a rotation of 180 in Z, etc.

Material setup

  1. Create the material: Create a material in Unity for each material in Substance Painter (right-click on the project window > Create > Material)
  2. Select the KSP2 Shader: Select all materials. In the Inspector window, in Shader, select KSP > Parts > Paintable.
  3. Apply the textures: For each material, assign the textures to the Albedo, Metallic, Normal, Emissive and Paintable channels.
    1. For Normal channels, click the Fix Now button.
  4. Set properties: Set Metallic/Smoothness Map to 1 and check Use PaintMask for Paint Smoothness.
  5. Apply materials: For each mesh, assign the materials to the corresponding locations. Click on Apply.

Collider setup

  1. Create Collider Object: R-Click mesh object: Create Empty. Name this object col.
  2. Create Collider: In the Inspector window for the col object click “Add Component”. Search for Mesh Collider and pick it. This will create a Mesh Collider component in the col object. Click the arrowhead to the left of it to expand it and see its properties.
    1. Select Mesh: In the Mesh Collider properties select the Mesh you want to use. If your FBX is all one object you can pick that, or you can pick a suitable primitive like cube or cylinder, etc.
    2. Position, Rotation, and Scale: Set the position, rotation, and scale of the mesh to encompass the part. You should see a green mesh represented in the Scene window to help guide you to make sure you’ve got the right position, rotation, and scale.
    3. Convex: Check “Convex”

Core Part Data setup

  1. Add Core Part Data: Select your Root Part Object. In the Inspector Window click “Add Component”. Search for “Core Part Data” and pick that. Open it up and configure as follows:
    1. Part Name: The Part Name needs to be the same as what you’ve used for the Root Part object, i.e., my_awesome_mod_my_part or whatever you used.
    2. Author: Use what you like here, typically your KSP Forum screen name or whatever you go by as your modding author name.
    3. Category: Select an appropriate category for your part.
    4. Family: If you wish to identify a “Family” for your part, this needs to be a particular string. You can find examples in the game’s files, or part JSONS, or ask in the KSP2 Modding Society discord to get this information.
    5. Co Lift, Co Mass, Co Pressure, etc.: These parameters allow you to set the Center of Lift, Center of Mass, Center of Pressure, etc. Adjust these to get the markers in the Unity scene where they should be for your part. Typically, Co Pressure and Co Lift are in the same place.
    6. Fuel Cross Feed: Check if fuel should be able to transit through your part on the way to other parts. Typically set to true, but not always.
    7. Mass: Set this in metric tons, not Kg.
    8. Attach Rules: Check the types of attachment your part should allow. Checking “Stack” or “Srf Attach” will allow your part to attach in a stack or to a surface. Checking “Allow Stack”, “Allow Srf Attach”, etc. will allow other parts to stack attach or surface attach respectively. Currently (?) Allow Collision, Allow Dock, Allow Rotate, and Allow Root have no effect in game (check this).
    9. Attach Nodes: If Stack is checked above, then you need a “top” and a “bottom” node, if Srf Attach is checked above, then you need a “srfAttach” node. Note, node names are case sensitive and having a node is not enough by itself, you do also need the corresponding Attach Rule set true or the node will have no effect. Under Attach Nodes click the + button to add a blank node and configure as needed.
      1. Node ID: “top”, “bottom”, “srfAttach”, etc. (case sensitive!)
      2. Node Type: Select as appropriate. (Stack for Stack, Surface for Surface…)
      3. Attach Method: Select Fixed_Joint for Stack and Hinge_Joint for Surface Attach.
      4. Is Multi Joint: In general set to True for stack attach to help prevent noodle rockets.
      5. Multi Joint Max Joint: Set to 3 if you set Is Multi Joint to true?
      6. Position: Set as appropriate. Should be on the skin or outside of the part where you would expect to find it on your part in the VAB.
      7. Orientation: Set as appropriate. The Orientation vector should be a unit vector (length 1) pointing in the direction of the part that will attach to the node, so pointing away from your part.
      8. Size: Affects rigidity of your part. If your part is connected to another part with the same “size” node, then rigidity will be optimal, and otherwise it will be suboptimal.
      9. Visual Size: Set the same as Size.
      10. Is Resource Crossfeed: Set as needed for this node.
      11. Is Rigid: Set as needed for this node.
      12. Rinse and Repeat: Subsequent nodes created with the + button will inherit settings from the last node made, so this may accelerate the process as you just need to change the Node Id, Position and Orientation for new nodes that are similar to the previous created node.
  2. Add Module_Color: You need this to be able to paint your part with base and accent colors. As above, click Add Component and search for Module Color.
  3. Add Module_Drag: All parts need this. As above, click Add Component and search for Module Color.

Crazy Customization (Adding functionalities to the part)

  1. Add other modules as needed. For example, if your part is an engine you’ll also need:
    1. thrustTransform object: Create an empty game object as a child of the root part and name it thrustTransform.
    2. Throttle VFX Manager: Configure as needed (?). No need to drag anywhere, it just need to be a component for the part.
    3. Flameout VFX Data: Drag this up to the Flameout VFX property in Module_Engine.
    4. Module_Gimbal: Configure details as needed (e.g., Gimbal Range and Gimbal Speed, etc.), then drag this up to the Gimbal property in Module_Engine.
    5. Module_Generator: Configure details as needed, then drag this up to the Alternator property in Module_Engine.
    6. Module_Fairing: Unless you plan for your engine to only ever be on the very first stage (i.e., a big booster), then you probably want it to have a fairing so it can be used in the second stage and above. Figure 3 shows an example for a working fairing from a Size Small (1.25m-class) engine that was created using this process. Shown below is the corresponding example of the PartComponentModule_Fairing portion of the part JSON. As can be seen in Figure 3, the data fields that need to be populated in the Unity Editor include only a subset of what's in the auto-generated JSON. That said, everything the editor needs is in the example below and presenting it this way makes copy/paste possible where unity editor graphic would not allow that.
      Figure 3: Example Fairing Data in Unity Editor
            {
              "Name": "PartComponentModule_Fairing",
              "ComponentType": "KSP.Sim.impl.PartComponentModule_Fairing, Assembly-CSharp, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null",
              "BehaviourType": "KSP.Modules.Module_Fairing, Assembly-CSharp, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null",
              "ModuleData": [
                {
                  "Name": "Data_Fairing",
                  "ModuleType": "KSP.Modules.Module_Fairing, Assembly-CSharp, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null",
                  "DataType": "KSP.Modules.Data_Fairing, Assembly-CSharp, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null",
                  "Data": null,
                  "DataObject": {
                    "$type": "KSP.Modules.Data_Fairing, Assembly-CSharp",
                    "FairingEnabled": {
                      "ContextKey": "FairingEnabled",
                      "storedValue": true
                    },
                    "FairingConstructionType": {
                      "ContextKey": "FairingConstructionType",
                      "storedValue": "CUSTOM"
                    },
                    "FloatingNodeEnabled": {
                      "ContextKey": "FloatingNodeEnabled",
                      "storedValue": true
                    },
                    "Length": {
                      "ContextKey": "Length",
                      "storedValue": -1.0
                    },
                    "DeployType": {
                      "ContextKey": "DeployType",
                      "storedValue": "Clamshellx4"
                    },
                    "EjectionForce": {
                      "ContextKey": "EjectionForce",
                      "storedValue": 100.0
                    },
                    "IsStagingEnabled": {
                      "ContextKey": "IsStagingEnabled",
                      "storedValue": true
                    },
                    "IsDeployed": {
                      "ContextKey": "IsDeployed",
                      "storedValue": false
                    },
                    "FloatingNodeSize": 1.0,
                    "FloatingAttachNodeTag": "bottom",
                    "FloatingNodePosition": {
                      "x": 0.0,
                      "y": 0.0,
                      "z": 0.0
                    },
                    "FloatingNodeDirection": {
                      "x": 0.0,
                      "y": -1.0,
                      "z": 0.0
                    },
                    "FloatingNodeIsMultiJoint": false,
                    "FloatingNodeMultiJointMaxCount": 3,
                    "FloatingNodeMultiJointOffset": 1.0,
                    "FairingNode": "top",
                    "NoseTip": 0.5,
                    "EdgeWarp": 0.02,
                    "AberrantNormalLimit": 45.0,
                    "LocalUpAxis": {
                      "x": 0.0,
                      "y": -1.0,
                      "z": 0.0
                    },
                    "Pivot": {
                      "x": 0.0,
                      "y": 0.0,
                      "z": 0.0
                    },
                    "BaseModelTransformName": "Base",
                    "CapRadius": 0.375,
                    "BaseRadius": 0.625,
                    "CloseRadius": 0.375,
                    "MaxRadius": 6.0,
                    "SnapThreshold": 0.25,
                    "CreateShellColliders": false,
                    "NumberOfCollidersPerCrossSection": 12,
                    "MinHeightRadiusRatio": 0.07,
                    "CrossSectionHeightMin": 0.3,
                    "CrossSectionHeightMax": 1.1,
                    "AerodynamicallyShieldContents": false,
                    "ConeSweepRays": 120,
                    "ConeSweepPrecision": 10.0,
                    "AmountOfCollidersPerArc": 1,
                    "ShouldCapOnAutoGenerate": false,
                    "IsCapped": false,
                    "IsShroud": true,
                    "MassAreaRatio": 0.0,
                    "FairingSideCount": 24,
                    "FairingLengthSnapIncrement": 0.125,
                    "FairingRadiusSnapIncrement": 0.125,
                    "FairingSmoothingAngle": 35.0,
                    "FairingThickness": 0.025,
                    "FairingStartHeight": 0.0,
                    "AllowConstructionTypeChange": true,
                    "AllowFloatingNodeChange": true,
                    "DefaultFairingEnabledToggle": true,
                    "DefaultAutoConstruction": true,
                    "DefaultDeployType": "Shroud",
                    "DefaultFloatingNodeState": true,
                    "LengthEditMinimum": 0.0,
                    "LengthEditMaximum": 0.001,
                    "LengthEditDefault": 1.0,
                    "StageToggleDefault": false,
                    "MaxAutoFairingTargetRadius": -1,
                    "MinAutoFairingTargetRadius": -1,
                    "CrossSections": [],
                    "MassModifierAmount": 0.0,
                    "DragCubeIndex": -1,
                    "ModuleType": "KSP.Modules.Module_Fairing, Assembly-CSharp, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null",
                    "MassModifier": 0.0,
                    "DataType": "KSP.Modules.Data_Fairing, Assembly-CSharp, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null",
                    "IsActiveInStagingProp": {
                      "ContextKey": null,
                      "storedValue": false
                    }
                  }
                }
              ]
            }
      Of the information show above, the parts you will most likely need to customize for your fairing are Base Radius, Cross Section Height Max, and Fairing Thickness - the rest is likely to be the same for all your engines with one other important thing to note. The example above is for an engine that was made in Blender with the nozzle bell facing upwards. If your engine was made with the nozzle bell facing down, then you will need the Local Up Axis to be -1 in Y and you will also need set the Fairing Node to bottom.

Final Flourishes (Final steps)

  1. Apply Options: Select the root part and in the Inspector window near the top on the Prefab line, click the Overrides dropdown and pick Apply All. If this option is not available, then you’ve got nothing you need to do here. Move along, move along.
  2. Save Part JSON: Click Save Part JSON button at the bottom of the Core Part Data module. This will put the resulting part JSON in the Assets folder for your Unity project. You need to do this any time you’ve edited the Core Part Data module (or also a module it depends on?).
  3. Make Prefab: Grab the root part object and drag it to the Unity project Assets folder.
  4. Add Part Icon: Create an icon for your part that the game will use in the parts picker. This needs to be a PNG file with specific dimensions. It should conform to the style used by other parts in the game. However you do this, you need to name the file <part_name>_icon.png, and you need to drag that file into the Assets folder in Unity.
    1. Convert Icon to Sprite: Select the part icon in the Assets folder. In the Inspector window, click the pulldown menu for Texture Type and change this from Default to Sprite (2D and UI), then click Apply. If this is not done, your lovely icon will not display!
  5. Make Root Part Addressable: Select the root part’s prefab in the Assets folder and in the Inspector window check the box for Addressable.
  6. Make JSON Addressable: Select the part’s JSON in the Assets folder and in the Inspector window check the box for Addressable.
  7. Make the Icon Addressable: Select the root part’s icon in the Assets folder and in the Inspector window check the box for Addressable.
  8. Configure Addressable Properties: In the Addressables Groups expand your mod's Group and find your part.
    1. Group Name \ Addressable Name: Change the information in the Group Name \ Addressable Name from Assets/<part_name>* to be just <part_name>*. So Assets/<part_name>.prefab becomes <part_name>.prefab, and so forth. You can leave the Assets/ part of the path definition alone for each of these, that’s as it should be. The value for the Addressable Name needs to be the same as the file name it’s associated with and must not include any path parts. All of these need to be based on the Part Name established in the Core Part Data module.
    2. Labels: For the JSON set this to parts_data. Leave it blank for the prefab. If parts_data is not an option in the dropdown for Labels, then click Manage Labels, click the + button to add a new label, and set the Label Name to parts_data. Click Save.
  9. Build Mod or Build And Test: Select either Build Mod or Build And Test to have KSP2 Unity Tools prepare and deploy your parts pack mod.
  10. Launch Game and Test!
  11. Rinse and Repeat for Additional Parts

Mentions

This guide is based on the videos: How to create parts for KSP2 and How to make engines for KSP2 (see: Part modding videos (tutorials)), and on other notes and guidance from the KSP2 Modding Society discord.

Also the web pages: KSP2 Part tutorial (almost) from scratch, and Tutorial: My First Part