The Grid Terminal System

From Space Engineers Wiki
Jump to navigation Jump to search

Introduction

The Grid Terminal System is where you will be able to access your ship’s devices. It provides access to all blocks on the same grid the programmable block is on, as well as any grids connected via rotors, pistons or connectors (not landing gears) - as long as the blocks have the same ownership as the running programmable block.

You access it via the property GridTerminalSystem, which belongs to the script’s base class, MyGridProgram. This means that it will be available to all methods and properties in your script - but it is not directly available in any subclasses. More on this later.

The easiest way to see what the Grid Terminal System has to offer, is to create your script project in Visual Studio. Find your Main method

public void Main() 
{
    // Start typing in here
}

and where the comment indicates above, start typing Grid (mind the casing). Visual Studio will respond by showing you this:

Image showing example of intellisense

This is called “intellisense”. It shows you what is available to you at the current location. In this case, it is showing you GridTerminalSystem, which is the property we want. Press tab or enter and the IDE will fill in the rest for you. This method is available to you for any accessible type, variable, field or property, not just the GridTerminalSystem. This means that if you for instance define a variable with the type of a block you want, you can use this technique to see what this block can do and what it can provide in terms of information - to a certain degree. There’s also (unfortunately) something called Terminal Properties and Terminal Actions which makes this more complicated than it has to be.

Now continue by adding a . at the end. This is what is important. Now it should show you this:

Another image showing example of intellisense

These are the methods available to you in the grid terminal system. All of them are ways to retrieve blocks from your grids in various ways.

Methods in GridTerminalSystem

GetBlockWithName

// method's signature
IMyTerminalBlock GetBlockWithName(string name)

Allows you to retrieve a single block having a specific name. Note that the name must be exact, case sensitive and including any spacing.

// Find a block of a specific name, then check if it's a timer block. If it is, store it in the     
// timer variable, otherwise store null there.     
IMyTimerBlock timer = GridTerminalSystem.GetBlockWithName("Timer Block") as IMyTimerBlock;

GetBlocks

// method's signature
void GetBlocks(List<IMyTerminalBlock> blocks)

Gets all the available blocks the Grid Terminal System has access to. Requires a target list of the base IMyTerminalBlock type.
The method automatically clears the list before filling it.

List<IMyTerminalBlock> blocks = new List<IMyTerminalBlock>(); 

public Program() 
{ 
	// Fill a list with all blocks in the system 
	GridTerminalSystem.GetBlocks(blocks); 
}

GetBlocksOfType

// methods' signatures
void GetBlocksOfType<T>(List<IMyTerminalBlock> blocks, Func<IMyTerminalBlock, bool> collect = null)
void GetBlocksOfType<T>(List<T> blocks, Func<T, bool> collect = null)

Allows you to retrieve a list of blocks of a specific type, optionally filtered by a given collect predicate.
The method automatically clears the list before filling it.

List<IMyInteriorLight> lights = new List<IMyInteriorLight>();     

public Program()     
{         
    // Fill a list with all interior lights in the system         
	GridTerminalSystem.GetBlocksOfType(lights);     
}
List<IMyTerminalBlock> lights = new List<IMyTerminalBlock>();     

public Program()
{
	// Create a list containing all interior lights in the system, but store them    
	// in a generic list containing blocks of the IMyTerminalBlock type     
	GridTerminalSystem.GetBlocksOfType<IMyInteriorLight>(lights);
}
List<IMyInteriorLight> lights = new List<IMyInteriorLight>();     

public Program()     
{         
	// Fill a list with all interior lights that are currently on         
	GridTerminalSystem.GetBlocksOfType(lights, light => light.Enabled);     
}

SearchBlocksOfName

// method's signature
void SearchBlocksOfName(string name, List<IMyTerminalBlock> blocks, Func<IMyTerminalBlock, bool> collect = null)

Searches through all the blocks, returning those whose name contains the entered name. Meaning, a block named “Mynoch” would be returned if you search for “no”. Also provides the ability to use a collect predicate, just like GetBlocksOfType. Unfortunately, this method can only accept a target list of the base IMyTerminalBlock type.
The method automatically clears the list before filling it.

This method is rather inefficiently implemented. You can do everything this method can do, in a better way, by using the filter of GetBlocksOfType.
List<IMyTerminalBlock> lights = new List<IMyTerminalBlock>();
void Main()
{
    // Fill a list with all lights in the system whose name contains the text "no"
    GridTerminalSystem.SearchBlocksOfName("no", lights, light => light is IMyInteriorLight);
}

GetBlockWithId

// method's signature
IMyTerminalBlock GetBlockWithId(long id)

Allows you to retrieve a block by its EntityId. This is a unique id given to a block by the game, and will remain the same even if you rename the block.

GetBlockGroups

// method's signature
void GetBlockGroups(List<IMyBlockGroup> blockGroups, Func<IMyBlockGroup, bool> collect = null)

Fetches a list of block groups, optionally filtered by a given collect predicate.
The method automatically clears the list before filling it.

List<IMyBlockGroup> groups = new List<IMyBlockGroup>();     
public Program()     
{         
	// Fill a list with all the block groups in the system         
	GridTerminalSystem.GetBlockGroups(groups);     
}
List<IMyBlockGroup> groups = new List<IMyBlockGroup>();     
public Program()     
{         
	// Fill a list with all the block groups whose name contains the text "no"
	GridTerminalSystem.GetBlockGroups(groups, group => group.Name.Contains("no"));     
}

GetBlockGroupWithName

// method's signature
IMyBlockGroup GetBlockGroupWithName(string name)

Returns a block group by its name. Note that the name must be exact, case sensitive and including any spacing.

// Retrieve the block group named "Interesting Blocks"
IMyBlockGroup group;     

public Program()
{
	group = GridTerminalSystem.GetBlockGroupWithName("Interesting Blocks");
}

CanAccess

// methods' signatures
bool CanAccess(IMyTerminalBlock block, MyTerminalAccessScope scope = MyTerminalAccessScope.All);
bool CanAccess(IMyCubeGrid grid, MyTerminalAccessScope scope = MyTerminalAccessScope.All);

Checks if the given block or grid is still accessible.
If the grid or block is destroyed, disconnected (see scope) or ownership changed to an enemy then it is no longer accessible.

The scope parameter:

  • MyTerminalAccessScope.All - block/grid is in the same terminal as the PB.
  • MyTerminalAccessScope.Construct - block/grid is linked through rotors/pistons/hinges/wheels (equivalent to IsSameConstructAs()).
  • MyTerminalAccessScope.Grid - block/grid is on the same grid as the PB (equivalent to block.CubeGrid == Me.CubeGrid).

The Grid Terminal System, Docking, and Multiple Constructs

By default, the grid terminal system just grabs everything it has access to, whether it’s a part of your ship or not. From version 1.188 you will have access to a new block method called IsSameConstructAs. This method will check if the two compared blocks belong to the same mechanical group, which means any set of grids connected by rotors and pistons. This mechanical group is called a construct and is usually what your ship, vehicle or station is made out from.

Be aware that you cannot distinguish between grids that are merged with a merge block, since the very purpose of that block is to permanently merge two grids into one.

List<IMyTerminalBlock> blocks = new List<IMyTerminalBlock>();
public Program()
{
    // Fill a list with all blocks in the current construct (ship, vehicle or station),
    // ignoring anything that might be docked to it. "Me" is the Program's reference to
    // the programmable block this script is running in.
    GridTerminalSystem.GetBlocksOfType(blocks, block => block.IsSameConstructAs(Me));
}

The Grid Terminal System and Subclasses

As your scripts increase in complexity, it’s inevitable that you’ll want to split your code into classes. Once you do this you will quickly realize that you lose access to the content of your program inside these classes. This is because the program itself is just another class. This very fact, however, provides you with a solution to this problem.

public class SomeUsefulSubclass
{
    Program _program;

    public SomeUsefulSubclass(Program program) 
    {
        _program = program;
    }

    public void SomeUsefulMethod() 
    {
        var block = _program.GridTerminalSystem.GetBlockWithName("That Block I Want");
        // Do useful stuff with that block
    }
}

SomeUsefulSubclass _subclass;

public Program() 
{
    _subclass = new SomeUsefulSubclass(this);
}

public void Main() {
    _subclass.SomeUsefulMethod();
}

In this example we are leveraging the very fact that your script is just a class like any other, a class which is always named Program. We are passing the main Program instance (this in any of the methods or properties in your main script - not in your subclasses) into our subclass via its constructor. We choose to pass the program itself rather than just the GridTerminalSystem because it will provide access to the other scripting facilities like the Echo, Storage and the Runtime as well as the GridTerminalSystem.



This tutorial was adapted from the original on Malware's MDK wiki, by the author.