-
Notifications
You must be signed in to change notification settings - Fork 125
The Anatomy of a Script
If you're not using MDK, you can skip to the next section
Ok, so you've learned the basics of C# and know where to get help, and you've opened your brand new project and you are looking at this (comments and using removed for brevity):
namespace IngameScript
{
partial class Program : MyGridProgram
{
public Program()
{
}
public void Save()
{
}
public void Main(string argument)
{
}
}
}
The first you need to be aware of, is that the namespace and the partial class Program : MyGridProgram
parts is not a part of the programmable block script. Only the parts inside the Program class is actually used when this script project is deployed to Space Engineers. The reason is that Space Engineers need strict control of what a script class is. A script class is always named Program
, and it's always inherited from MyGridProgram
. Space Engineers will add these parts to your script behind the scene. If you're just using the ingame editor, a normal text editor or something like that, you won't need these additions. See also "The Program Wrapper And Why You Need It".
This is what you'll see the first time you open a newly placed programmable block (comments removed for brevity):
public Program()
{
}
public void Save()
{
}
public void Main(string argument)
{
}
public Program()
{
}
It is used for one-time initialization of your script, and is run once every time your script is instantiated. This means it's run once after your game is loaded, or after you recompile your script. Recompiling happens when you have edited your script or when you press the Recompile
button.
The constructor is optional, you don't need it to have a working script.
public void Save()
{
}
The Save method is called whenever the game is saved, or right before you recompile your script. Here you can store any data you need to persist between play sessions. See The Storage String for more details on how to do this.
This save method is also optional.
public void Main(string argument)
{
}
or simply
public void Main()
{
}
The Main method is the main entry point of your script. It might be called many times during a script's lifetime, depending on what your script is and how it is set up. This method is required, obviously. The argument is passed from the game (depending on your toolbar setup) to your script, enabling a single script to do a bunch of different things. You can set this argument by dragging your programmable block onto a button panel, sensor or timer toolbar (or any other device toolbar except your suit) and selecting "Run". You will be presented with an input box where you can type in what you want to be passed to your script. It will be passed verbatim, Keep that in mind.
This is a game, obviously, so there needs to be limitations. There are three distinct limitations in place for a programmable block script.
The first is the the size of the script itself. No scripts can have any more than 100 000 characters.
The second is what is called the "instruction counter". You may only have a maximum of 50000 code junctions per single run in your script. A code junction includes things like method calls, switches, conditional, loops and similar. Simple statements like 1+1 does not increase this count. note that this is only a tool. You should always strive to execute as little as possible code per run as possible. Remember, the game has only about 16 milliseconds per frame to do everything, including game logic, physics, rendering and all scripts in your world. Every nanosecond counts!
The third limitation is how much of the .NET framework you have access to. Obviously giving full access to the framework is dangerous, because then people could do bad things to others' computers. This is why Space Engineers uses a whitelist to restrict which parts of the framework you have access to.
If you use MDK, it will tell you if you try to reference a type or member that is not allowed. You can try this now by creating a new MDK project, and add this:
public void Main()
{
System.Threading.Thread thread;
}
With the help of the MDK extension, Visual Studio will mark this with an error telling you that the type System.Threading.Thread
is prohibited.
Do you have questions, comments, suggestions for improvements? Is there something I can do better? Did I make a mistake? Please add an issue here, and prefix your issue title with Wiki. Thank you, your help will be very appreciated!