HowTo/Hide- & Unhide Meshes
|  (Created page with "In HowTo/Your first Trainz Script we created the script:   ; myscript.gs  '''<font color="#0000C0">include</font>''' <font color="#808080">"MapObject.gs"</font>    '''<fon...") | Tonyhilliam  (Talk | contribs)  | ||
| (9 intermediate revisions by 2 users not shown) | |||
| Line 14: | Line 14: | ||
| − | Today we're gonna use this little example for a vehicle. There is an ice cream truck on the DLS by misterairbrush (<kuid:414943:100987>). This ice cream truck has a door that can be  | + | Today we're gonna use this little example for a vehicle. There is an ice cream truck on the DLS by misterairbrush (<kuid:414943:100987>). This ice cream truck has a door that can be opened or closed. | 
| It looks like: | It looks like: | ||
| [[File:Scripting_02_Truck_1.jpg]] [[File:Scripting_02_Truck_2.jpg]] | [[File:Scripting_02_Truck_1.jpg]] [[File:Scripting_02_Truck_2.jpg]] | ||
| − | We'll learn today how to hide the "wrong" and show the right mesh. Also we want  | + | The truck comes equipped with two versions of the door, called meshes. One or the other mesh is visible at any one time but never both together.  | 
| − | + | We'll learn today how to hide the "wrong" and show the right mesh. Also we want users to be able to select an option in the "?"-Window of the scenery object to determine whether the door is to be open or closed. | |
| + | First we need a member variable to store the (user-selected) open/closed state of the door, then we need functions to save and load the information every time the session starts or is saved. | ||
| − | Variables  | + | |
| + | Variables store values used in the game. These values can be changed by script and are stored in the computer's memory until needed. | ||
| In Trainz Script there are four (well, really there are five, but we will ignore one this time!) data [[Types]] we can use: | In Trainz Script there are four (well, really there are five, but we will ignore one this time!) data [[Types]] we can use: | ||
| Line 30: | Line 32: | ||
| * '''int''' - every number between -2147483647 and 2147483647 (a 32 Bit integer) | * '''int''' - every number between -2147483647 and 2147483647 (a 32 Bit integer) | ||
| * '''string''' - characters | * '''string''' - characters | ||
| − | * '''float''' -  | + | * '''float''' - numbers with a decimal point (32 Bit) | 
| − | Today we'll have a closer look  | + | Today we'll have a closer look at the '''bool''' type. The others will be explained another time :) | 
| In our case, we only need a true/false value. | In our case, we only need a true/false value. | ||
| − | '''True''': The user wants the door to be  | + | '''True''': The user wants the door to be open | 
| '''False''': The user wants the door to be closed | '''False''': The user wants the door to be closed | ||
| Line 60: | Line 62: | ||
| − | + | We begin by stating the data-type of our variable and then give it a name. Now we can use it. | |
| − | This variable now is available in the whole CIceCreamTruck-Class, because we declared it  | + | This variable now is available in the whole CIceCreamTruck-Class, because we declared it before any other methods, so we may say it's a global variable (that isn't true, but it's easier to explain). | 
| − | We  | + | We may also declare variables within methods/functions. These then will only be accessible inside the function or method and not in other parts of the script. | 
| − | In Trainz Script there are now real global variables (that's why I think we may call the members global, even if it's not true). We  | + | In Trainz Script there are now real global variables (that's why I think we may call the members global, even if it's not true). We may only declare variables inside classes. | 
| − | Next, we will have a look  | + | Next, we will have a look at two new methods: | 
| * '''[[Class_PropertyObject#GetProperties]] (GetProperties)''' - Save data to a soup | * '''[[Class_PropertyObject#GetProperties]] (GetProperties)''' - Save data to a soup | ||
| * '''[[Class_PropertyObject#SetProperties]] (SetProperties)''' - Fetch data from a Soup | * '''[[Class_PropertyObject#SetProperties]] (SetProperties)''' - Fetch data from a Soup | ||
| − | These methods are implemented in native code. That means we don't have to  | + | These methods are implemented in native code. That means we don't have to worry about when they are called. We just have to override them. Trainz will call them whenever needed.  To override a class means to replace the built-in class with a new one that extends the functions of the original (the parent) while retaining all the original functionality. | 
| Let's override them: | Let's override them: | ||
| Line 101: | Line 103: | ||
| ; GetProperties: | ; GetProperties: | ||
| − | As I said above, this  | + | As I said above, the purpose of this function is to save data into a Soup. It expects a Soup-Object to be returned. You'll see that Soup is a class, not a data type. | 
| But we're able to declare class instances as variables too. (Look at: '''<font color="#800000">Soup</font>''' <font color="#008000">pSoup</font> = <font color="#0000C0">'''inherited'''</font>();) | But we're able to declare class instances as variables too. (Look at: '''<font color="#800000">Soup</font>''' <font color="#008000">pSoup</font> = <font color="#0000C0">'''inherited'''</font>();) | ||
| − | And again: The inherited is in  | + | And again: The inherited is in there! | 
| − | Every function/method we override has in most cases to inherit the parent classes method. The inherited  | + | Every function/method we override has in most cases to inherit the parent classes method. The inherited statement calls GetProperties of Class MapObject. This makes sure, that every operation inside a method of the parent class is called too. Remember: We overrode the method. If we don't inherit it here again, the parent classes GetProperties-Method will never be called! | 
| − | Only miss the call to inherited if you want to disable  | + | Only miss the call to inherited if you want to disable functionality for a class, if not, don't forget it. | 
| − | In the script above, we return the Soup directly, without any  | + | In the script above, we return the Soup directly, without any changes. What exactly "return types" are will be explained another time :) (it is too much detail to go into now!) | 
| ; SetProperties: | ; SetProperties: | ||
| − | In this method we get a  | + | In this method we get a parameter of type soup. We inherit the MapObject-Setproprties. This time we don't have to return any value. Here we will load the data from the Soup we got as parameter. | 
| ; '''Saving data''' | ; '''Saving data''' | ||
| − | I already explained that Soup is a class. So it may  | + | Think of soup as a container into which you throw pieces of paper with names and values. When you want to find out what you wrote on the paper you look through all the pieces until you find the name and then read off the value.    | 
| + | I already explained that Soup is a class. So it may have members, and in fact it has. Look at: [[Class_Soup]] | ||
| − | + | I use the variable name as the soup tag name, because it's convenient. The method "SetNamedTag" in class Soup is used to assign a value to each tag in the soup. We can access all public methods inside a class with the dereference operator, that | |
| − | + | ||
| is a single dot ('''.'''). So, we call the function like this: | is a single dot ('''.'''). So, we call the function like this: | ||
| Line 144: | Line 146: | ||
|       { |       { | ||
|           '''<font color="#800000">Soup</font>''' <font color="#008000">pSoup</font> = <font color="#0000C0">'''inherited'''</font>(); |           '''<font color="#800000">Soup</font>''' <font color="#008000">pSoup</font> = <font color="#0000C0">'''inherited'''</font>(); | ||
| − |           <font color="#008000">pSoup</font>.<font color="#0000C0">SetNamedTag</font>(<font color="#808080">" | + |           <font color="#008000">pSoup</font>.<font color="#0000C0">SetNamedTag</font>(<font color="#808080">"m_bDoorOpened"</font>, <font color="#008000">m_bDoorOpened</font>); | 
|           <font color="#0000C0">'''return'''</font> <font color="#008000">pSoup</font>; |           <font color="#0000C0">'''return'''</font> <font color="#008000">pSoup</font>; | ||
|       } |       } | ||
| Line 155: | Line 157: | ||
| − | Now you'll understand, why we have to return the Soup back to Trainz. We manipulate the values inside the Soup and Trainz has to  | + | Now you'll understand, why we have to return the Soup back to Trainz. We manipulate the values inside the Soup and Trainz has to be kept informed, if Trainz wants to save the Soup for us. | 
| ; '''Loading data''' | ; '''Loading data''' | ||
| − | The  | + | The opposite of "Setting data" (SetNamedTag) is "Getting data" (GetNamedTag). And, yes, Soup has a method called "GetNamedTag", but it'll return the value as a string. We need a boolean value. | 
| − | So, we  | + | So, we could use the method "GetNamedTagAsBool". It expects only one parameter, the name of our value inside the Soup. We can directly assign the value of "GetNamedTagAsBool" to our variable. | 
| − | But this  | + | But this function has another variant which expects two parameters. The first one is the name of our value, and the other one is a default value, if the selected | 
| − | value is not set in the Soup. We'll need that  | + | value is not set in the Soup. We'll need to use that method because, if we place an object in surveyor, there are no values in the Soup. So, we can set an initial value for the door state. In our case we will set the initial door state to closed (so: m_bDoorOpened has to be false). | 
|   <font color="#008000">m_bDoorOpened</font> = <font color="#008000">pSoup</font>.<font color="#0000C0">GetNamedTagAsBool</font>(<font color="#808080">"m_bDoorOpened"</font>, <font color="#0000C0">'''false'''</font>); |   <font color="#008000">m_bDoorOpened</font> = <font color="#008000">pSoup</font>.<font color="#0000C0">GetNamedTagAsBool</font>(<font color="#808080">"m_bDoorOpened"</font>, <font color="#0000C0">'''false'''</font>); | ||
| − | If we  | + | If we add that part into our script it'll look like: | 
| Line 184: | Line 186: | ||
|       { |       { | ||
|           '''<font color="#800000">Soup</font>''' <font color="#008000">pSoup</font> = <font color="#0000C0">'''inherited'''</font>(); |           '''<font color="#800000">Soup</font>''' <font color="#008000">pSoup</font> = <font color="#0000C0">'''inherited'''</font>(); | ||
| − |           <font color="#008000">pSoup</font>.<font color="#0000C0">SetNamedTag</font>(<font color="#808080">" | + |           <font color="#008000">pSoup</font>.<font color="#0000C0">SetNamedTag</font>(<font color="#808080">"m_bDoorOpened"</font>, <font color="#008000">m_bDoorOpened</font>); | 
|           <font color="#0000C0">'''return'''</font> <font color="#008000">pSoup</font>; |           <font color="#0000C0">'''return'''</font> <font color="#008000">pSoup</font>; | ||
|       } |       } | ||
| Line 196: | Line 198: | ||
| − | Now we can display and hide the meshes. But the object itself has to be prepared for it. The Ice cream truck has three meshes | + | Now we can display and hide the meshes. But the object itself has to be prepared for it. The Ice cream truck has three meshes: The truck with a hole where the door should be, a door in the opened state and a door in the closed state. You also may achieve this by using animations, but this way is much easier. | 
| [[File:Scripting_2_Truck_whole.jpg]]<br> | [[File:Scripting_2_Truck_whole.jpg]]<br> | ||
| − | Truck with  | + | Truck with hole, where the door should be | 
| Line 244: | Line 246: | ||
| All meshes we want to control by script may never have an auto-create value of 1! It has to be 0, otherwise we can't take the control over the meshes. | All meshes we want to control by script may never have an auto-create value of 1! It has to be 0, otherwise we can't take the control over the meshes. | ||
| − | Our CIceCreamTruck's parent class is MapObject. And MapObject's parent class is MeshObject, and there we will find a method  | + | Our CIceCreamTruck's parent class is MapObject. And MapObject's parent class is MeshObject, and there we will find a method "SetMeshVisible". | 
| We can use it directly, because we automatically inherit from MeshObject, too. | We can use it directly, because we automatically inherit from MeshObject, too. | ||
| − | SetMeshVsible expects three parameters | + | SetMeshVsible expects to receive three parameters: The name of our mesh-table entry, a state (true or false - [visible, invisible]) and a floating-point number that sets a fading time in seconds. | 
| We'll leave the last parameter 0.0f. | We'll leave the last parameter 0.0f. | ||
| Line 252: | Line 254: | ||
| − | We have two meshes | + | We have two meshes that have to be set. In script we'll hide/show them like this: | 
| Line 270: | Line 272: | ||
|       { |       { | ||
|           '''<font color="#800000">Soup</font>''' <font color="#008000">pSoup</font> = <font color="#0000C0">'''inherited'''</font>(); |           '''<font color="#800000">Soup</font>''' <font color="#008000">pSoup</font> = <font color="#0000C0">'''inherited'''</font>(); | ||
| − |           <font color="#008000">pSoup</font>.<font color="#0000C0">SetNamedTag</font>(<font color="#808080">" | + |           <font color="#008000">pSoup</font>.<font color="#0000C0">SetNamedTag</font>(<font color="#808080">"m_bDoorOpened"</font>, <font color="#008000">m_bDoorOpened</font>); | 
|           <font color="#0000C0">'''return'''</font> <font color="#008000">pSoup</font>; |           <font color="#0000C0">'''return'''</font> <font color="#008000">pSoup</font>; | ||
|       } |       } | ||
| Line 285: | Line 287: | ||
| − | We  | + | We use our bool variable for indicating the mesh state. The opened mesh is set with the value of m_bDoorOpened. If m_bDoorOpened is true, the mesh will display, otherwise it'll be hidden. | 
| − | The closed state mesh is set with the  | + | The closed state mesh is set with the opposite value of m_bDoorOpened. This can be easily done by using the 'not' operator "!". Just place it in front of a boolean variable, to use the opposite value. | 
| Now you know how to control meshes. Because it will be too much for now, I'll give the last part of the script without explanation and move it to another time :) | Now you know how to control meshes. Because it will be too much for now, I'll give the last part of the script without explanation and move it to another time :) | ||
| − | |||
| Line 308: | Line 309: | ||
|       { |       { | ||
|           '''<font color="#800000">Soup</font>''' <font color="#008000">pSoup</font> = <font color="#0000C0">'''inherited'''</font>(); |           '''<font color="#800000">Soup</font>''' <font color="#008000">pSoup</font> = <font color="#0000C0">'''inherited'''</font>(); | ||
| − |           <font color="#008000">pSoup</font>.<font color="#0000C0">SetNamedTag</font>(<font color="#808080">" | + |           <font color="#008000">pSoup</font>.<font color="#0000C0">SetNamedTag</font>(<font color="#808080">"m_bDoorOpened"</font>, <font color="#008000">m_bDoorOpened</font>); | 
|           <font color="#0000C0">'''return'''</font> <font color="#008000">pSoup</font>; |           <font color="#0000C0">'''return'''</font> <font color="#008000">pSoup</font>; | ||
|       } |       } | ||
| Line 346: | Line 347: | ||
|   }; |   }; | ||
| − | Hope you enjoyed  | + | Hope you enjoyed and learned!<br> | 
| + | Full example here: [[Media:TrainzDevIceCreamTruck.zip]] (Thanks to misterairbursh for the permission!) | ||
| If you want to contact me, you may use this e-mail: callavsg@gmx.de | If you want to contact me, you may use this e-mail: callavsg@gmx.de | ||
| callavsg | callavsg | ||
| + | |||
| + | Edited by --[[User:Trev999|Trev999]] ([[User talk:Trev999|talk]]) 02:28, 20 August 2017 (AEST) | ||
| + | |||
| + | ==Return to Index== | ||
| + | |||
| + | [[HowToGuides|<< How To Guides]] | ||
| + | |||
| + | [[Category:How-to guides]] | ||
Latest revision as of 09:18, 17 January 2018
In HowTo/Your first Trainz Script we created the script:
- myscript.gs
include "MapObject.gs"
class CMyClass isclass MapObject
{
    public void Init(Asset pAsset)
    {
        inherited(pAsset);
    }
};
Today we're gonna use this little example for a vehicle. There is an ice cream truck on the DLS by misterairbrush (<kuid:414943:100987>). This ice cream truck has a door that can be opened or closed.
It looks like:
The truck comes equipped with two versions of the door, called meshes. One or the other mesh is visible at any one time but never both together. We'll learn today how to hide the "wrong" and show the right mesh. Also we want users to be able to select an option in the "?"-Window of the scenery object to determine whether the door is to be open or closed.
First we need a member variable to store the (user-selected) open/closed state of the door, then we need functions to save and load the information every time the session starts or is saved.
Variables store values used in the game. These values can be changed by script and are stored in the computer's memory until needed.
In Trainz Script there are four (well, really there are five, but we will ignore one this time!) data Types we can use:
- bool - boolean constants true/false
- int - every number between -2147483647 and 2147483647 (a 32 Bit integer)
- string - characters
- float - numbers with a decimal point (32 Bit)
Today we'll have a closer look at the bool type. The others will be explained another time :)
In our case, we only need a true/false value.
True: The user wants the door to be open
False: The user wants the door to be closed
How do we declare a member variable? This is pretty easy:
- icecreamtruck.gs
include "MapObject.gs"
class CIceCreamTruck isclass MapObject
{
    bool        m_bDoorOpened;
    public void Init(Asset pAsset)
    {
        inherited(pAsset);
    }
};
We begin by stating the data-type of our variable and then give it a name. Now we can use it.
This variable now is available in the whole CIceCreamTruck-Class, because we declared it before any other methods, so we may say it's a global variable (that isn't true, but it's easier to explain).
We may also declare variables within methods/functions. These then will only be accessible inside the function or method and not in other parts of the script.
In Trainz Script there are now real global variables (that's why I think we may call the members global, even if it's not true). We may only declare variables inside classes.
Next, we will have a look at two new methods:
- Class_PropertyObject#GetProperties (GetProperties) - Save data to a soup
- Class_PropertyObject#SetProperties (SetProperties) - Fetch data from a Soup
These methods are implemented in native code. That means we don't have to worry about when they are called. We just have to override them. Trainz will call them whenever needed. To override a class means to replace the built-in class with a new one that extends the functions of the original (the parent) while retaining all the original functionality. Let's override them:
- icecreamtruck.gs
include "MapObject.gs"
class CIceCreamTruck isclass MapObject
{
    bool        m_bDoorOpened;
    public void Init(Asset pAsset)
    {
        inherited(pAsset);
    }
    public Soup GetProperties(void)
    {
        Soup pSoup = inherited();
        return pSoup;
    }
    public void SetProperties(Soup pSoup)
    {
        inherited(pSoup);
    }
};
- GetProperties
As I said above, the purpose of this function is to save data into a Soup. It expects a Soup-Object to be returned. You'll see that Soup is a class, not a data type.
But we're able to declare class instances as variables too. (Look at: Soup pSoup = inherited();)
And again: The inherited is in there! Every function/method we override has in most cases to inherit the parent classes method. The inherited statement calls GetProperties of Class MapObject. This makes sure, that every operation inside a method of the parent class is called too. Remember: We overrode the method. If we don't inherit it here again, the parent classes GetProperties-Method will never be called! Only miss the call to inherited if you want to disable functionality for a class, if not, don't forget it.
In the script above, we return the Soup directly, without any changes. What exactly "return types" are will be explained another time :) (it is too much detail to go into now!)
- SetProperties
In this method we get a parameter of type soup. We inherit the MapObject-Setproprties. This time we don't have to return any value. Here we will load the data from the Soup we got as parameter.
- Saving data
Think of soup as a container into which you throw pieces of paper with names and values. When you want to find out what you wrote on the paper you look through all the pieces until you find the name and then read off the value. I already explained that Soup is a class. So it may have members, and in fact it has. Look at: Class_Soup
I use the variable name as the soup tag name, because it's convenient. The method "SetNamedTag" in class Soup is used to assign a value to each tag in the soup. We can access all public methods inside a class with the dereference operator, that is a single dot (.). So, we call the function like this:
pSoup.SetNamedTag("NAME", [VALUE]);
Now in script, it looks like:
- icecreamtruck.gs
include "MapObject.gs"
class CIceCreamTruck isclass MapObject
{
    bool        m_bDoorOpened;
    public void Init(Asset pAsset)
    {
        inherited(pAsset);
    }
    public Soup GetProperties(void)
    {
        Soup pSoup = inherited();
        pSoup.SetNamedTag("m_bDoorOpened", m_bDoorOpened);
        return pSoup;
    }
    public void SetProperties(Soup pSoup)
    {
        inherited(pSoup);
    }
};
Now you'll understand, why we have to return the Soup back to Trainz. We manipulate the values inside the Soup and Trainz has to be kept informed, if Trainz wants to save the Soup for us.
- Loading data
The opposite of "Setting data" (SetNamedTag) is "Getting data" (GetNamedTag). And, yes, Soup has a method called "GetNamedTag", but it'll return the value as a string. We need a boolean value. So, we could use the method "GetNamedTagAsBool". It expects only one parameter, the name of our value inside the Soup. We can directly assign the value of "GetNamedTagAsBool" to our variable. But this function has another variant which expects two parameters. The first one is the name of our value, and the other one is a default value, if the selected value is not set in the Soup. We'll need to use that method because, if we place an object in surveyor, there are no values in the Soup. So, we can set an initial value for the door state. In our case we will set the initial door state to closed (so: m_bDoorOpened has to be false).
m_bDoorOpened = pSoup.GetNamedTagAsBool("m_bDoorOpened", false);
If we add that part into our script it'll look like:
- icecreamtruck.gs
include "MapObject.gs"
class CIceCreamTruck isclass MapObject
{
    bool        m_bDoorOpened;
    public void Init(Asset pAsset)
    {
        inherited(pAsset);
    }
    public Soup GetProperties(void)
    {
        Soup pSoup = inherited();
        pSoup.SetNamedTag("m_bDoorOpened", m_bDoorOpened);
        return pSoup;
    }
    public void SetProperties(Soup pSoup)
    {
        inherited(pSoup);
        m_bDoorOpened = pSoup.GetNamedTagAsBool("m_bDoorOpened", false);
    }
};
Now we can display and hide the meshes. But the object itself has to be prepared for it. The Ice cream truck has three meshes: The truck with a hole where the door should be, a door in the opened state and a door in the closed state. You also may achieve this by using animations, but this way is much easier.

Truck with hole, where the door should be
In script we reference the mesh-table entries of config.txt.
It may look like:
[...]
mesh-table
{  
   default
   {
     mesh                                "truck.im"
     auto-create                         1
   }
  
   door-opened
   {
     mesh                                "door_opened.im"
     auto-create                         0
   }
  
   door-closed
   {
     mesh                                "door_closed.im"
     auto-create                         0
   }
   [...]
}
[...]
All meshes we want to control by script may never have an auto-create value of 1! It has to be 0, otherwise we can't take the control over the meshes.
Our CIceCreamTruck's parent class is MapObject. And MapObject's parent class is MeshObject, and there we will find a method "SetMeshVisible". We can use it directly, because we automatically inherit from MeshObject, too. SetMeshVsible expects to receive three parameters: The name of our mesh-table entry, a state (true or false - [visible, invisible]) and a floating-point number that sets a fading time in seconds. We'll leave the last parameter 0.0f.
SetMeshVisible("NAME", true/false, 0.0f);
We have two meshes that have to be set. In script we'll hide/show them like this:
- icecreamtruck.gs
include "MapObject.gs"
class CIceCreamTruck isclass MapObject
{
    bool        m_bDoorOpened;
    public void Init(Asset pAsset)
    {
        inherited(pAsset);
    }
    public Soup GetProperties(void)
    {
        Soup pSoup = inherited();
        pSoup.SetNamedTag("m_bDoorOpened", m_bDoorOpened);
        return pSoup;
    }
    public void SetProperties(Soup pSoup)
    {
        inherited(pSoup);
        m_bDoorOpened = pSoup.GetNamedTagAsBool("m_bDoorOpened", false);
        SetMeshVisible("door-opened", m_bDoorOpened, 0.0f);
        SetMeshVisible("door-closed", !m_bDoorOpened, 0.0f);
    }
};
We use our bool variable for indicating the mesh state. The opened mesh is set with the value of m_bDoorOpened. If m_bDoorOpened is true, the mesh will display, otherwise it'll be hidden.
The closed state mesh is set with the opposite value of m_bDoorOpened. This can be easily done by using the 'not' operator "!". Just place it in front of a boolean variable, to use the opposite value.
Now you know how to control meshes. Because it will be too much for now, I'll give the last part of the script without explanation and move it to another time :)
- icecreamtruck.gs
include "MapObject.gs"
class CIceCreamTruck isclass MapObject
{
    bool        m_bDoorOpened;
    public void Init(Asset pAsset)
    {
        inherited(pAsset);
    }
    public Soup GetProperties(void)
    {
        Soup pSoup = inherited();
        pSoup.SetNamedTag("m_bDoorOpened", m_bDoorOpened);
        return pSoup;
    }
    public void SetProperties(Soup pSoup)
    {
        inherited(pSoup);
        m_bDoorOpened = pSoup.GetNamedTagAsBool("m_bDoorOpened", false);
        SetMeshVisible("door-opened", m_bDoorOpened, 0.0f);
        SetMeshVisible("door-closed", !m_bDoorOpened, 0.0f);
    }
    //////////////////////////////////////////////////////////////////////////////////////////////////////////////
    public string GetDescriptionHTML(void)
    {
        string sHtml = inherited();
        sHtml = sHtml + HTMLWindow.CheckBox("live://property/m_bDoorOpened", m_bDoorOpened) + " Open door";
        return sHtml;
    }
    public string GetPropertyType(string sPropertyId)
    {
        string sRet = "link";
        if(sPropertyId == "m_bDoorOpened")sRet = "link";
        return sRet;
    }
    public void LinkPropertyValue(string sPropertyId)
    {
        if(sPropertyId == "m_bDoorOpened")
        {
            m_bDoorOpened = !m_bDoorOpened;
        }
    }
};
Hope you enjoyed and learned!
Full example here: Media:TrainzDevIceCreamTruck.zip (Thanks to misterairbursh for the permission!)
If you want to contact me, you may use this e-mail: callavsg@gmx.de
callavsg
Edited by --Trev999 (talk) 02:28, 20 August 2017 (AEST)




