Class Inheritance: Difference between revisions
(Almost complete rewrite to improve grammatical phrasing, spelling, and legibility, as well as to fix incorrectly written examples and provide further elaboration.) |
Lou Montana (talk | contribs) m (Some wiki formatting) |
||
Line 2: | Line 2: | ||
Inheriting classes within the config is a concept that can easily go wrong. | Inheriting classes within the config is a concept that can easily go wrong. | ||
The purpose of this document is to clarify the cause of some common errors and contains instructions on how to resolve these mistakes. | The purpose of this document is to clarify the cause of some common errors and contains instructions on how to resolve these mistakes. | ||
== Terms == | == Terms == | ||
Line 7: | Line 8: | ||
* A '''config''' refers to a config.cpp or config.bin file that is loaded during the game start. It does ''not'' refer to a [[Model Config]]. | * A '''config''' refers to a config.cpp or config.bin file that is loaded during the game start. It does ''not'' refer to a [[Model Config]]. | ||
* The '''bin\config.bin''' is the father of all other configs. It forms the first state of the master config.bin and consists of the base game's root classes. | * The '''bin\config.bin''' is the father of all other configs. It forms the first state of the master config.bin and consists of the base game's root classes. | ||
* The '''master''' config is built from the merging of ''all'' configs to the bin\config.bin, and it is what is seen through the Splendid Config Viewer. This includes both the base game addons as well as user-made addons. | * The '''master''' config is built from the merging of ''all'' configs to the bin\config.bin, and it is what is seen through the Splendid Config Viewer. | ||
* '''Child configs''' are the configs that are merged into the master config.bin during game load. The order in which they are added is defined by the requiredAddons array within each CfgPatches (see | This includes both the base game addons as well as user-made addons. | ||
*CfgPatches is a necessary prerequisite for '''all''' child configs so that its addon name and required addons (if any) are known. | * '''Child configs''' are the configs that are merged into the master config.bin during game load. The order in which they are added is defined by the requiredAddons array within each CfgPatches (see {{Link|#Addon loading order}} below). | ||
* CfgPatches is a necessary prerequisite for '''all''' child configs so that its addon name and required addons (if any) are known. | |||
* Missions and campaigns are not configs, as the pbo they are in may or may not contain a config. This makes them mission pbos or mission addons respectively. | * Missions and campaigns are not configs, as the pbo they are in may or may not contain a config. This makes them mission pbos or mission addons respectively. | ||
Line 19: | Line 21: | ||
Classes can contain either child classes (see below) or properties. Properties work similarly to script variables, they have a name and are assigned a value. | Classes can contain either child classes (see below) or properties. Properties work similarly to script variables, they have a name and are assigned a value. | ||
However, unlike script variables, properties can only have number values, string values, or (one or multidimensional) arrays that are made up of number or string values. | However, unlike script variables, properties can only have number values, string values, or (one or multidimensional) arrays that are made up of number or string values. | ||
=== Parent and Child Classes === | === Parent and Child Classes === | ||
Line 34: | Line 35: | ||
// ... | // ... | ||
}; | }; | ||
class Jacob | class Jacob | ||
{ | { | ||
Line 42: | Line 44: | ||
}; | }; | ||
</syntaxhighlight> | </syntaxhighlight> | ||
In this example, the classes "Esau" and "Jacob" are '''child''' classes of "Isaac". This makes them '''sibling''' classes. | In this example, the classes "Esau" and "Jacob" are '''child''' classes of "Isaac". This makes them '''sibling''' classes. | ||
Line 53: | Line 56: | ||
// ... | // ... | ||
}; | }; | ||
class arifle_Katiba_F | class arifle_Katiba_F | ||
{ | { | ||
Line 59: | Line 63: | ||
}; | }; | ||
</syntaxhighlight> | </syntaxhighlight> | ||
=== Basic Inheritance === | === Basic Inheritance === | ||
A class can inherit properties from sibling classes by defining it as a base class. All properties of the inherited base class will also exist within your class and, unless you overwrite them, they will have the same values as the base class. This is extremely useful for quickly writing sibling classes that share properties, or to build a class on top of another class that has the same properties with only minor value changes. | A class can inherit properties from sibling classes by defining it as a base class. | ||
All properties of the inherited base class will also exist within your class and, unless you overwrite them, they will have the same values as the base class. | |||
This is extremely useful for quickly writing sibling classes that share properties, or to build a class on top of another class that has the same properties with only minor value changes. | |||
Taking the example from above we can create a common "ChildMaleBase" base class. | Taking the example from above we can create a common "ChildMaleBase" base class. | ||
<syntaxhighlight lang="cpp"> | <syntaxhighlight lang="cpp"> | ||
class Isaac | class Isaac | ||
Line 74: | Line 80: | ||
gender = "male"; | gender = "male"; | ||
}; | }; | ||
class Esau: ChildMaleBase | |||
class Esau : ChildMaleBase | |||
{ | { | ||
firstborn = 1; | firstborn = 1; | ||
// ... | // ... | ||
}; | }; | ||
class Jacob: ChildMaleBase | |||
class Jacob : ChildMaleBase | |||
{ | { | ||
firstborn = 0; | firstborn = 0; | ||
Line 86: | Line 94: | ||
}; | }; | ||
</syntaxhighlight> | </syntaxhighlight> | ||
In this example, both "Esau" and "Jacob" will inherit the gender property from "ChildMaleBase". | In this example, both "Esau" and "Jacob" will inherit the gender property from "ChildMaleBase". | ||
Line 99: | Line 108: | ||
// ... | // ... | ||
}; | }; | ||
class Jacob: Esau | |||
class Jacob : Esau | |||
{ | { | ||
firstborn = 0; | firstborn = 0; | ||
// ... | // ... | ||
}; | }; | ||
class Mary: Esau | |||
class Mary : Esau | |||
{ | { | ||
gender = "Female"; | gender = "Female"; | ||
Line 111: | Line 122: | ||
}; | }; | ||
</syntaxhighlight> | </syntaxhighlight> | ||
In the above example, while "Mary" overwrites the gender property, the class would still inherit the firstborn value of 1 from sibling class "Esau". | In the above example, while "Mary" overwrites the gender property, the class would still inherit the firstborn value of 1 from sibling class "Esau". | ||
Either method is stored, as written, in the master config, and either one achieves the same result for any other classes that access "Isaac". | |||
== External Base Classes == | == External Base Classes == | ||
Line 121: | Line 133: | ||
{{Feature|important|The concept of external base classes only applies to addon configs. For mission configs, the [[import]] keyword fulfills a similar purpose.}} | {{Feature|important|The concept of external base classes only applies to addon configs. For mission configs, the [[import]] keyword fulfills a similar purpose.}} | ||
External base classes are essentially base classes that are first defined ''outside'' of your config, either by the base game or another addon. It | External base classes are essentially base classes that are first defined ''outside'' of your config, either by the base game or another addon. | ||
It is a simple concept to understand but not so simple to implement correctly. | |||
When using an external base class the first thing you have to do is declare it in what is termed a "template" or "skeleton". These do NOT affect those classes, they merely tell the compiler how they are constructed. If the classes you declare are not yet discovered by the engine, it will build empty classes waiting to be filled later, based on what inheritance you | When using an external base class the first thing you have to do is declare it in what is termed a "template" or "skeleton". | ||
These do NOT affect those classes, they merely tell the compiler how they are constructed. | |||
If the classes you declare are not yet discovered by the engine, it will build empty classes waiting to be filled later, based on what inheritance you have written. | |||
As a result, it is important to get the inheritance (if any) right! | |||
<syntaxhighlight lang="cpp"> | <syntaxhighlight lang="cpp"> | ||
class externalBaseClass; // | class externalBaseClass; // declare an external base class "skeleton" | ||
class myClass: externalBaseClass // | class myClass : externalBaseClass // define that your class inherits from it | ||
{ | { | ||
// ... // | // ... // start using it | ||
}; | }; | ||
</syntaxhighlight> | </syntaxhighlight> | ||
Line 139: | Line 155: | ||
class CfgWeapons | class CfgWeapons | ||
{ | { | ||
class arifle_MX_F; // | class arifle_MX_F; // declare an external base class "skeleton" | ||
class arifle_MX_Black_F: arifle_MX_F // | class arifle_MX_Black_F : arifle_MX_F // define that your class inherits from the external base class | ||
{ | { | ||
hiddenSelections[] = {"camo1", "camo2"}; | hiddenSelections[] = { "camo1", "camo2" }; | ||
hiddenSelectionsTextures[] = {"\A3\Weapons_F_EPB\Rifles\MX_Black\Data\XMX_Base_Black_co.paa","\A3\Weapons_F_EPB\Rifles\MX_Black\Data\XMX_short_Black_co.paa"}; | hiddenSelectionsTextures[] = { "\A3\Weapons_F_EPB\Rifles\MX_Black\Data\XMX_Base_Black_co.paa", "\A3\Weapons_F_EPB\Rifles\MX_Black\Data\XMX_short_Black_co.paa" }; | ||
// ... | // ... | ||
}; | }; | ||
class arifle_MX_khk_F: arifle_MX_Black_F | class arifle_MX_khk_F : arifle_MX_Black_F | ||
{ | { | ||
hiddenSelectionsTextures[] = {"\A3\Weapons_F_Exp\Rifles\MX\Data\XMX_Base_khk_co.paa","\A3\Weapons_F_Exp\Rifles\MX\Data\XMX_Short_khk_co.paa"}; | hiddenSelectionsTextures[] = { "\A3\Weapons_F_Exp\Rifles\MX\Data\XMX_Base_khk_co.paa", "\A3\Weapons_F_Exp\Rifles\MX\Data\XMX_Short_khk_co.paa" }; | ||
// ... | // ... | ||
}; | }; | ||
}; | }; | ||
</syntaxhighlight> | </syntaxhighlight> | ||
"CfgWeapons" serves as the parent class, while "arifle_MX_F", "arifle_MX_Black_F", and "arifle_MX_khk_F" serve as sibling classes. In this instance, the "arifle_MX_Black_F" is inheriting all of its values from the "arifle_MX_F" external base class skeleton, but it changes the value for the "hiddenSelections" and "hiddenSelectionsTextures" arrays. When defining "arifle_MX_khk_F" later, instead of changing the "hiddenSelections" array value again, it inherits the changed value from its sibling class, "arifle_MX_Black_F", and changes the necessary values in the "hiddenSelectionsTextures" array for the new textures to appear. | |||
"CfgWeapons" serves as the parent class, while "arifle_MX_F", "arifle_MX_Black_F", and "arifle_MX_khk_F" serve as sibling classes. | |||
In this instance, the "arifle_MX_Black_F" is inheriting all of its values from the "arifle_MX_F" external base class skeleton, but it changes the value for the "hiddenSelections" and "hiddenSelectionsTextures" arrays. | |||
When defining "arifle_MX_khk_F" later, instead of changing the "hiddenSelections" array value again, it inherits the changed value from its sibling class, "arifle_MX_Black_F", and changes the necessary values in the "hiddenSelectionsTextures" array for the new textures to appear. | |||
It is not necessary to declare the base class' inheritance tree (if any) ''if'' you have the correct inherited class' addon in the "requiredAddons" array. | It is not necessary to declare the base class' inheritance tree (if any) ''if'' you have the correct inherited class' addon in the "requiredAddons" array. | ||
=== External Base Child Classes === | |||
In the event that you want to access a child class of a base class, you must declare the child class as a part of the base class. | |||
In the event that you want to access a child class of a base class, you must declare the child class as a part of the base class. In order to open the base class, however, you must declare what it inherits from as well. Opening the base class to declare the child class without defining what the base class inherits from will declare the base class as if it is the first time it is being defined. | In order to open the base class, however, you must declare what it inherits from as well. | ||
Opening the base class to declare the child class without defining what the base class inherits from will declare the base class as if it is the first time it is being defined. | |||
<syntaxhighlight lang="cpp"> | <syntaxhighlight lang="cpp"> | ||
class externalRootClass; // | class externalRootClass; // declare an external base class "skeleton" | ||
class externalBaseClass: externalRootClass // | class externalBaseClass : externalRootClass // define that the external parent base class inherits from the external base class | ||
{ | { | ||
class externalChildClass; // | class externalChildClass; // define the external child base class "skeleton" | ||
}; | }; | ||
class myClass: externalBaseClass // | class myClass : externalBaseClass // define that your class inherits from the external parent base class | ||
{ | { | ||
class myChildClass: externalChildClass // | class myChildClass : externalChildClass // define that your child class inherits from the external child base class | ||
{ | { | ||
// ... add or make changes | // ... add or make changes | ||
Line 179: | Line 200: | ||
</syntaxhighlight> | </syntaxhighlight> | ||
=== Implied Child Classes === | |||
Once an external child class is defined somewhere in the inheritance tree, you will not need to redefine it later, as it is implied by the game engine that that child class is the class you are looking for. | Once an external child class is defined somewhere in the inheritance tree, you will not need to redefine it later, as it is implied by the game engine that that child class is the class you are looking for. | ||
<syntaxhighlight lang="cpp"> | <syntaxhighlight lang="cpp"> | ||
Line 187: | Line 208: | ||
class B | class B | ||
{ | { | ||
// | // whatever | ||
}; | }; | ||
}; | }; | ||
class C: A | class C : A | ||
{ | { | ||
class D | class D | ||
{ | { | ||
// | // whatever | ||
}; | }; | ||
}; | }; | ||
class E: C | class E : C | ||
{ | { | ||
class D: D // | class D : D // fairly standard | ||
{ | { | ||
// | // change things | ||
}; | }; | ||
class B: B // | class B : B // fairly strange! | ||
{ | { | ||
// | // change things | ||
}; | }; | ||
}; | }; | ||
</syntaxhighlight> | </syntaxhighlight> | ||
The reason ''why'' you can access, and use, "B" (which is not a direct child of "E") is due to inheritance. "B" does indeed become a child of "E"! | The reason ''why'' you can access, and use, "B" (which is not a direct child of "E") is due to inheritance. "B" does indeed become a child of "E"! | ||
Line 221: | Line 243: | ||
}; | }; | ||
class C: A | class C : A | ||
{ | { | ||
class D; | class D; | ||
Line 229: | Line 251: | ||
== Addon Loading Order == | == Addon Loading Order == | ||
When you are using external base classes it is extremely important to define what addons your addon depends on, or in other words, which external base classes need to be already loaded when your config gets loaded. | When you are using external base classes it is extremely important to define what addons your addon depends on, or in other words, which external base classes need to be already loaded when your config gets loaded. | ||
Line 240: | Line 263: | ||
{ | { | ||
// ... | // ... | ||
requiredAddons[] = {"ExternalAddonA", "ExternalAddonB", ...}; | requiredAddons[] = { "ExternalAddonA", "ExternalAddonB", ... }; | ||
// ... | // ... | ||
}; | }; | ||
Line 249: | Line 272: | ||
Without defining the required addons and load order, there is no guarantee that your config gets applied correctly. | Without defining the required addons and load order, there is no guarantee that your config gets applied correctly. | ||
{{Feature| | {{Feature|informative|{{hl|A3_Data_F_Oldman_Loadorder}} is the last vanilla CfgPatches entry in {{arma3}} as of 2.12, so you can use this to overwrite some vanilla configs.}} | ||
== Empty == | == Empty == | ||
The <syntaxhighlight lang="cpp" inline>class Empty{};</syntaxhighlight> syntax can be easily misunderstood for defining an external base class. Using the empty syntax removes everything inside of the class (including embedded classes ''within the scope'' of the parent class it is found in, defining it as if it was being defined for the very first time. It cannot be used for any class that has inheritance, because that is simply seen as a skeleton tree. | The <syntaxhighlight lang="cpp" inline>class Empty{};</syntaxhighlight> syntax can be easily misunderstood for defining an external base class. | ||
Using the empty syntax removes everything inside of the class (including embedded classes ''within the scope'' of the parent class it is found in, defining it as if it was being defined for the very first time. | |||
It cannot be used for any class that has inheritance, because that is simply seen as a skeleton tree. | |||
<syntaxhighlight lang="cpp"> | <syntaxhighlight lang="cpp"> | ||
class MyAddon; // | class MyAddon; // defines a skeleton class | ||
</syntaxhighlight> | </syntaxhighlight> | ||
<syntaxhighlight lang="cpp"> | <syntaxhighlight lang="cpp"> | ||
class MyAddon{}; // | class MyAddon{}; // removes everything inside the class | ||
</syntaxhighlight> | </syntaxhighlight> | ||
Line 266: | Line 291: | ||
== Delete == | == Delete == | ||
Within configs, the <syntaxhighlight lang="cpp" inline>delete Classname;</syntaxhighlight> keyword is available. It can be used to delete already existing classes. | Within configs, the <syntaxhighlight lang="cpp" inline>delete Classname;</syntaxhighlight> keyword is available. It can be used to delete already existing classes. | ||
{{Feature|important|Classes from which other classes derive cannot be deleted unless the child classes are deleted first.}} | {{Feature|important|Classes from which other classes derive cannot be deleted unless the child classes are deleted first.}} | ||
<syntaxhighlight lang="cpp"> | <syntaxhighlight lang="cpp"> | ||
Line 285: | Line 310: | ||
<syntaxhighlight lang="cpp"> | <syntaxhighlight lang="cpp"> | ||
class Intel | class Intel | ||
{ | { | ||
class Attributes | class Attributes | ||
{ | { | ||
class Fog; | class Fog; | ||
class TimeMultiplier: Fog | class TimeMultiplier : Fog | ||
{ | { | ||
displayName = "Time Multiplier"; | displayName = "Time Multiplier"; | ||
Line 296: | Line 321: | ||
// ... | // ... | ||
}; | }; | ||
delete Fog; // | delete Fog; // will not work since a child of Fog still exists. | ||
}; | }; | ||
}; | }; | ||
</syntaxhighlight> | </syntaxhighlight> | ||
{{GameCategory|arma1|Addon Configuration}} | {{GameCategory|arma1|Addon Configuration}} | ||
Line 306: | Line 332: | ||
{{GameCategory|arma3|Editing}} | {{GameCategory|arma3|Editing}} | ||
{{GameCategory|arma3|Tutorials}} | {{GameCategory|arma3|Tutorials}} | ||
{{GameCategory|tkoh|Editing}} |
Revision as of 14:44, 25 May 2023
Inheriting classes within the config is a concept that can easily go wrong. The purpose of this document is to clarify the cause of some common errors and contains instructions on how to resolve these mistakes.
Terms
- A config refers to a config.cpp or config.bin file that is loaded during the game start. It does not refer to a Model Config.
- The bin\config.bin is the father of all other configs. It forms the first state of the master config.bin and consists of the base game's root classes.
- The master config is built from the merging of all configs to the bin\config.bin, and it is what is seen through the Splendid Config Viewer.
This includes both the base game addons as well as user-made addons.
- Child configs are the configs that are merged into the master config.bin during game load. The order in which they are added is defined by the requiredAddons array within each CfgPatches (see Addon loading order below).
- CfgPatches is a necessary prerequisite for all child configs so that its addon name and required addons (if any) are known.
- Missions and campaigns are not configs, as the pbo they are in may or may not contain a config. This makes them mission pbos or mission addons respectively.
Basic Config Concepts
The config is a hierarchical structure based on classes that provide almost all information necessary for the game. Objects, their behavior, user interface elements, and even which functions to run on game start are defined through the config.
Classes can contain either child classes (see below) or properties. Properties work similarly to script variables, they have a name and are assigned a value. However, unlike script variables, properties can only have number values, string values, or (one or multidimensional) arrays that are made up of number or string values.
Parent and Child Classes
Due to the hierarchical nature of the config, classes can have parent, child and sibling classes.
class Isaac
{
class Esau
{
gender = "male";
firstborn = 1;
// ...
};
class Jacob
{
gender = "male";
firstborn = 0;
// ...
};
};
In this example, the classes "Esau" and "Jacob" are child classes of "Isaac". This makes them sibling classes.
In a more practical example, "arifle_MX_F" and "arifle_Katiba_F" are both child classes of CfgWeapons, and as a result, they too are sibling classes.
class CfgWeapons
{
class arifle_MX_F
{
// ...
};
class arifle_Katiba_F
{
// ...
};
};
Basic Inheritance
A class can inherit properties from sibling classes by defining it as a base class. All properties of the inherited base class will also exist within your class and, unless you overwrite them, they will have the same values as the base class. This is extremely useful for quickly writing sibling classes that share properties, or to build a class on top of another class that has the same properties with only minor value changes.
Taking the example from above we can create a common "ChildMaleBase" base class.
class Isaac
{
class ChildMaleBase
{
gender = "male";
};
class Esau : ChildMaleBase
{
firstborn = 1;
// ...
};
class Jacob : ChildMaleBase
{
firstborn = 0;
// ...
};
};
In this example, both "Esau" and "Jacob" will inherit the gender property from "ChildMaleBase".
Alternatively, we can use "Esau" as a base class for "Jacob" and overwrite the "firstborn" property:
class Isaac
{
class Esau
{
gender = "Male";
firstborn = 1;
// ...
};
class Jacob : Esau
{
firstborn = 0;
// ...
};
class Mary : Esau
{
gender = "Female";
// ...
};
};
In the above example, while "Mary" overwrites the gender property, the class would still inherit the firstborn value of 1 from sibling class "Esau".
Either method is stored, as written, in the master config, and either one achieves the same result for any other classes that access "Isaac".
External Base Classes
External base classes are essentially base classes that are first defined outside of your config, either by the base game or another addon. It is a simple concept to understand but not so simple to implement correctly.
When using an external base class the first thing you have to do is declare it in what is termed a "template" or "skeleton".
These do NOT affect those classes, they merely tell the compiler how they are constructed.
If the classes you declare are not yet discovered by the engine, it will build empty classes waiting to be filled later, based on what inheritance you have written.
As a result, it is important to get the inheritance (if any) right!
class externalBaseClass; // declare an external base class "skeleton"
class myClass : externalBaseClass // define that your class inherits from it
{
// ... // start using it
};
An example of both inheritance and the use of external base classes could be the addition of new weapon textures via "hiddenSelections".
class CfgWeapons
{
class arifle_MX_F; // declare an external base class "skeleton"
class arifle_MX_Black_F : arifle_MX_F // define that your class inherits from the external base class
{
hiddenSelections[] = { "camo1", "camo2" };
hiddenSelectionsTextures[] = { "\A3\Weapons_F_EPB\Rifles\MX_Black\Data\XMX_Base_Black_co.paa", "\A3\Weapons_F_EPB\Rifles\MX_Black\Data\XMX_short_Black_co.paa" };
// ...
};
class arifle_MX_khk_F : arifle_MX_Black_F
{
hiddenSelectionsTextures[] = { "\A3\Weapons_F_Exp\Rifles\MX\Data\XMX_Base_khk_co.paa", "\A3\Weapons_F_Exp\Rifles\MX\Data\XMX_Short_khk_co.paa" };
// ...
};
};
"CfgWeapons" serves as the parent class, while "arifle_MX_F", "arifle_MX_Black_F", and "arifle_MX_khk_F" serve as sibling classes. In this instance, the "arifle_MX_Black_F" is inheriting all of its values from the "arifle_MX_F" external base class skeleton, but it changes the value for the "hiddenSelections" and "hiddenSelectionsTextures" arrays. When defining "arifle_MX_khk_F" later, instead of changing the "hiddenSelections" array value again, it inherits the changed value from its sibling class, "arifle_MX_Black_F", and changes the necessary values in the "hiddenSelectionsTextures" array for the new textures to appear.
It is not necessary to declare the base class' inheritance tree (if any) if you have the correct inherited class' addon in the "requiredAddons" array.
External Base Child Classes
In the event that you want to access a child class of a base class, you must declare the child class as a part of the base class. In order to open the base class, however, you must declare what it inherits from as well. Opening the base class to declare the child class without defining what the base class inherits from will declare the base class as if it is the first time it is being defined.
class externalRootClass; // declare an external base class "skeleton"
class externalBaseClass : externalRootClass // define that the external parent base class inherits from the external base class
{
class externalChildClass; // define the external child base class "skeleton"
};
class myClass : externalBaseClass // define that your class inherits from the external parent base class
{
class myChildClass : externalChildClass // define that your child class inherits from the external child base class
{
// ... add or make changes
};
// ...
};
Implied Child Classes
Once an external child class is defined somewhere in the inheritance tree, you will not need to redefine it later, as it is implied by the game engine that that child class is the class you are looking for.
class A
{
class B
{
// whatever
};
};
class C : A
{
class D
{
// whatever
};
};
class E : C
{
class D : D // fairly standard
{
// change things
};
class B : B // fairly strange!
{
// change things
};
};
The reason why you can access, and use, "B" (which is not a direct child of "E") is due to inheritance. "B" does indeed become a child of "E"!
Note that "A", "B", "C", and "D" could be more simply shown as a skeleton tree defined as follows:
class A
{
class B;
};
class C : A
{
class D;
};
Addon Loading Order
When you are using external base classes it is extremely important to define what addons your addon depends on, or in other words, which external base classes need to be already loaded when your config gets loaded.
Defining which addons are required by your config is easily done through the "CfgPatches" class:
class CfgPatches
{
class MyAddon
{
// ...
requiredAddons[] = { "ExternalAddonA", "ExternalAddonB", ... };
// ...
};
};
Defining the required addons ensures that whatever changes you made, or whatever classes you defined based on external classes, are defined on top of the external addons' config. Without defining the required addons and load order, there is no guarantee that your config gets applied correctly.
Empty
The class Empty{};
syntax can be easily misunderstood for defining an external base class.
Using the empty syntax removes everything inside of the class (including embedded classes within the scope of the parent class it is found in, defining it as if it was being defined for the very first time.
It cannot be used for any class that has inheritance, because that is simply seen as a skeleton tree.
class MyAddon; // defines a skeleton class
class MyAddon{}; // removes everything inside the class
Delete
Within configs, the delete Classname;
keyword is available. It can be used to delete already existing classes.
class Intel
{
class AttributeCategories
{
class Date
{
class Attributes
{
delete Fog; // removes Fog attribute from Eden Editor
};
};
};
};
class Intel
{
class Attributes
{
class Fog;
class TimeMultiplier : Fog
{
displayName = "Time Multiplier";
tooltip = "Set the time multiplier";
// ...
};
delete Fog; // will not work since a child of Fog still exists.
};
};