Scripting: Best Practices – Arma Reforger

From Bohemia Interactive Community
biki>Lou Montana
m (Some wiki formatting)
 
m (Text replacement - "\{\{GameCategory\|armaR\|Modding\|(Guidelines|Tutorials)\|([^=↵]*)\}\}" to "{{GameCategory|armaR|Modding|$2|$1}}")
 
(15 intermediate revisions by the same user not shown)
Line 1: Line 1:
{{TOC|side}}
{{TOC|side}}
== Getting started ==
== Getting Started ==


In the domain of development, any rule is a rule of thumb. If a rule states for example that it is better that a line of code doesn't go over 80 characters, it doesn't mean that any line '''''must not''''' go over 80 characters; sometimes, the situation needs it.
In the domain of development, any rule is a rule of thumb.
If a rule states for example that it is better that a line of code doesn't go over 80 characters, it doesn't mean that any line '''''must not''''' go over 80 characters; sometimes, the situation needs it.


If the code has a good structure, '''do not''' change it to enforce a single arbitrary rule. If many of them are not implemented/not respected, changes should be applied; again, this is according to one's judgement.
If the code has a good structure, '''do not''' change it to enforce a single arbitrary rule. If many of them are not implemented/not respected, changes should be applied; again, this is according to one's judgement.
Line 9: Line 10:




== Best practices ==
== Best Practices ==


=== Code format ===
{{Feature|informative|See {{Link|Arma Reforger:Scripting: Conventions|Scripting Conventions}} for all the conventions to date.}}
<!--
 
See Scripting Guidelines / Conventions for all the conventions to date.
=== Code Format ===
-->
 
* '''Reminder:''' chosen indentation for Enfusion is {{Wikipedia|Indentation_style#Allman_style|Allman style}}
* '''Reminder:''' chosen indentation for Enfusion is {{Link|https://en.wikipedia.org/wiki/Indentation_style#Allman_style|Allman style}}
* '''Reminder:''' indentation is done with '''tabulations'''
* '''Reminder:''' indentation is done with '''tabulations'''
* Use empty space. Line return, spaces before and after brackets, if this improves readability, use it: space is free
* Use empty space. Line return, spaces before and after brackets, if this improves readability, use it: space is free
Line 21: Line 22:
** it also hinders debugger's usage, e.g in the event of an inlined {{hl|if}}
** it also hinders debugger's usage, e.g in the event of an inlined {{hl|if}}


=== Variable format ===
=== Variable Format ===
<!--
See Scripting Guidelines / Conventions for all the conventions to date.
-->


* Name variables and functions properly: code must be readable by a human being, e.g variables like '''u''' instead of '''uniform''' should not exist.
* Name variables and functions properly: code must be readable by a human being, e.g variables like '''u''' instead of '''uniform''' should not exist.
** '''i''' is an accepted iteration variable name (e.g in {{hl|for}} loops).
** '''{{hl|i}}''' is an accepted iteration variable name (e.g in {{hl|for}} loops).
* Prefix any public content (classes, global methods, global variables) with a [[OFPEC tags|Creator Tag]] in order to prevent conflicts with other mods.
* Prefix any public content (classes, global methods, global variables) with a [[Scripting Tags|Creator Tag]] in order to prevent conflicts with other mods.
* Use the closest value type whenever possible; using {{hl|auto}} for a known variable type makes code less clear.
* Use the closest value type whenever possible; using {{hl|auto}} for a known variable type makes code more obscure and prevents autocompletion.


=== Code Structuration ===
=== Code Structuration ===
Line 36: Line 34:
A series of development principles to follow in order to ensure an easy code maintenance and lifetime.
A series of development principles to follow in order to ensure an easy code maintenance and lifetime.


{{Feature|informative|See {{Wikipedia|SOLID}}.}}
{{Feature|informative|See {{Link|https://en.wikipedia.org/wiki/SOLID}}.}}


==== DRY ====
==== DRY ====
'''D'''on't '''R'''epeat '''Y'''ourself. If within the same class, the same code or the same pattern is written in various places, write a protected method and use appropriate parameters.
'''D'''on't '''R'''epeat '''Y'''ourself. If within the same class, the same code or the same pattern is written in various places, write a protected method and use appropriate parameters.
{{Feature|informative|See {{Wikipedia|Don't repeat yourself}}.}}
{{Feature|informative|See {{Link|https://en.wikipedia.org/wiki/Don't_repeat_yourself}}.}}


==== Logical Simplifications ====
==== Logical Simplifications ====
Line 48: Line 46:


==== Examples ====
==== Examples ====
{| class="wikitable"
{| class="wikitable valign-top"
! Improvable
! Improvable
! Good
! Good
|- style="vertical-align: top"
 
| <syntaxhighlight lang="cpp">
|-
| <enforce>
auto number = 42;
auto number = 42;
Animal cutePet = new Dog();
Animal cutePet = new Dog();
</syntaxhighlight>
</enforce>
| <syntaxhighlight lang="cpp">
| <enforce>
int number = 42;
int number = 42;
Dog cutePet = new Dog();
Dog cutePet = new Dog();
</syntaxhighlight>
</enforce>
|- style="vertical-align: top"
 
| <syntaxhighlight lang="cpp">
|-
for (int i = 0; i < list.Count(); i++)
| <enforce>
int i = 0;
string result = "";
SCR_MyClass obj = null;
</enforce>
| <enforce>
int i; // default value = 0 - see {{Link|Arma Reforger:Scripting: Values#Integer|Values - integer}}
string result; // default value = "" (a string cannot be null)
SCR_MyClass obj; // default value = null
</enforce>
 
|-
| <enforce>
// a method call is more expensive than a bool check
if (obj.MustBeTreated() || m_bTreatAllObjects)
Print(obj);
 
if (obj.MustBeTreated() && m_bTreatAllObjects)
Print(obj);
</enforce>
| <enforce>
// cheap checks go first, expensive checks (method calls) go after
if (m_bTreatAllObjects || obj.MustBeTreated())
Print(obj);
 
if (m_bTreatAllObjects && obj.MustBeTreated())
Print(obj);
</enforce>
 
|-
| <enforce>
// many identical method calls
if (obj.MustBeTreated() && obj.GetObject())
Print("Result: " + obj.GetObject().m_sValue1 + " " + obj.GetObject().m_sValue2);
</enforce>
| <enforce>
// "bigger", non-repetitive code can be beneficial for performance and readability
if (obj.MustBeTreated())
{
SCR_Object subObj = obj.GetObject();
if (subObj)
Print("Result: " + subObj.m_sValue1 + " " + subObj.m_sValue2);
}
</enforce>
 
|-
| <enforce>
foreach (SCR_Object obj : list)
{
Method(obj); // one method call per iteration
}
 
void Method(SCR_Object obj)
{
if (!obj)
return;
 
Print(obj.m_sName + " has a value of " + obj.m_sValue);
}
</enforce>
| <enforce>
foreach (SCR_Object obj : list) // the least method calls, the better
{
if (!obj)
continue;
 
Print(obj.m_sName + " has a value of " + obj.m_sValue);
}
</enforce>
 
|-
| <enforce>
bool IsObjectAlive(SCR_Object obj)
{
if (!obj)
return false;
 
if (obj.m_Health > 0) // keep this structure for complex code
return true;
else
return false;
}
</enforce>
| <enforce>
bool IsObjectAlive(SCR_Object obj)
{
return obj && obj.m_Health > 0;
}
</enforce>
 
|-
| <enforce>
bool IsObjectValid(SCR_Object obj)
{
if (!obj)
return false;
 
return true;
}
</enforce>
| <enforce>
bool IsObjectValid(SCR_Object obj)
{
return obj != null; // for readability
}
</enforce>
 
|-
| <enforce>
for (int i; i < list.Count(); i++) // list.Count() is called on every iteration
{
// ...
}
</enforce>
| <enforce>
for (int i, count = list.Count(); i < count; i++) // only one list.Count() call
{
// ...
}
</enforce>
 
|-
| <enforce>
for (int i, count = list.Count(); i < count; i++)
{
if (list[i]) // first
Print(list[i]); // and second .Get(i) method calls
}
</enforce>
| <enforce>
foreach (SCR_Object obj : list) // foreach is faster for start-to-end iterating
{
if (obj)
Print(obj); // no additional method call
}
</enforce>
 
|-
| <enforce>
for (int i, count = list.Count(); i < count; i++)
{
PrintFormat("Object #%1 = %2", i, list[i]);
}
</enforce>
| <enforce>
foreach (int i, SCR_Object obj : list) // iteration index is available this way too
{
PrintFormat("Object #%1 = %2", i, obj);
}
</enforce>
 
|-
| <enforce>
// declaring an 'obj' every loop generates a pointer release each time
foreach (SCR_ParentObject parent : list)
{
SCR_Object obj = parent.m_Object;
if (obj)
Print(obj.m_sName);
}
</enforce>
| <enforce>
SCR_Object obj; // external declaration = only one release at the end of the scope
foreach (SCR_ParentObject parent : list)
{
{
obj = parent.m_Object;
if (obj)
Print(obj.m_sName);
}
}
</syntaxhighlight>
</enforce>
| <syntaxhighlight lang="cpp">
 
for (int i = 0, count = list.Count(); i < count; i++) // only one list.Count() call
|-
| <enforce>
array<SCR_Object> toRemove = {};
foreach (SCR_Object obj : bigArray)
{
{
if (obj.m_bShouldBeRemoved)
toRemove.Insert(obj);
}
}
</syntaxhighlight>
 
|- style="vertical-align: top"
foreach (SCR_Object obj : toRemove)
| <syntaxhighlight lang="cpp">
// declaring 'text' string in every for loop creates a pointer attribution for each loop
for (int i = 0, count = list.Count(); i < count; i++)
{
{
string text = "Value " + i + " is " + list[i];
bigArray.RemoveItem(obj); // or RemoveItemOrdered if order is important
Print(text);
}
}
</syntaxhighlight>
</enforce>
| <syntaxhighlight lang="cpp">
| <enforce>
string text; // external declaration = one pointer attribution
for (int i = bigArray.Count() - 1; i >= 0; i--) // reverse iterating
for (int i = 0, count = list.Count(); i < count; i++)
{
{
text = "Value " + i + " is " + list[i];
if (bigArray[i].m_bShouldBeRemoved)
Print(text);
bigArray.Remove(i); // or RemoveItemOrdered if order is important
}
}
</syntaxhighlight>
</enforce>
which can also be simplified to
 
<syntaxhighlight lang="cpp">
|-
foreach (int i, element : list)
| <enforce>
if (a)
{
{
PrintFormat("Value %1 is %2", i, element);
if (b)
{
if (c) // also known as Hadouken code
Method(true);
else
Method(false);
}
}
}
</syntaxhighlight>
</enforce>
|- style="vertical-align: top"
| <enforce>
| <syntaxhighlight lang="cpp">
if (a && b)
Method(c);
</enforce>
 
|-
| <enforce>
if (a)
if (a)
{
{
Method(a);
if (b)
if (b)
{
{
if (c)
Method(b);
if (c) // another Hadouken code, with complications
{
{
Method(true);
Method(c);
return 42;
}
}
else
else
{
{
Method(false);
return -1;
}
}
}
else
{
return -1;
}
}
}
}
</syntaxhighlight>
else
| <syntaxhighlight lang="cpp">
if (a && b)
{
{
Method(c);
return -1;
}
}
</syntaxhighlight>
</enforce>
|- style="vertical-align: top"
| <enforce>
| Boilerplate code:
if (!a)
<syntaxhighlight lang="cpp">
return -1; // this is called early return and helps funnel down the code
int i = 0;
 
Method(a);
if (!b)
return -1;
 
Method(b);
if (!c)
return -1;
 
Method(c);
return 42;
</enforce>
 
|-
| <enforce>
int i;
if (a)
if (a)
{
i++;
i++;
}
 
if (b)
if (b)
{
i++;
i++;
}
 
if (c)
if (c)
{
i++;
i++;
}
 
</syntaxhighlight>
// etc
| Simplified:
</enforce>
<syntaxhighlight lang="cpp">
| <enforce>
int i = 0;
int i;
array<bool> conditions = { a, b, c };
array<bool> conditions = { a, b, c, /* etc */ };
foreach (bool condition : conditions)
foreach (bool condition : conditions)
{
{
if (condition)
if (condition)
{
i++;
i++;
}
}
}
</syntaxhighlight>
</enforce>
or (if some other code is involved)
 
<syntaxhighlight lang="cpp">
|-
int i = 0;
| <enforce>
IncrementIfTrue(a, i);
IncrementIfTrue(b, i);
IncrementIfTrue(c, i);
void IncrementIfTrue(bool condition, out int value)
{
if (condition)
{
value++;
}
}
</syntaxhighlight>
|- style="vertical-align: top"
| <syntaxhighlight lang="cpp">
Initialise(player1, 1);
Initialise(player1, 1);
Initialise(player2, 2);
Initialise(player2, 2);
Line 170: Line 348:
Initialise(player5, 5);
Initialise(player5, 5);
Initialise(player6, 6);
Initialise(player6, 6);
</syntaxhighlight>
</enforce>
| <syntaxhighlight lang="cpp">
| <enforce>
array<IEntity> list = { player1, player2, player3, player4, player5, player6 };
array<IEntity> list = { player1, player2, player3, player4, player5, player6 };
foreach (int i, item : list)
foreach (int i, IEntity item : list)
{
{
Initialise(item, i + 1);
Initialise(item, i + 1);
}
}
</syntaxhighlight>
</enforce>
<enforce>
// or, better, one method call that initialises all of them
Initialise(list); // numbering is then done inside the method, if possible
Initialise(list, 1); // otherwise the starting number can be provided
</enforce>
 
|}
|}


==== Code Comments ====
==== Code Comments ====
Code comments are surprisingly '''not''' a must-have; code organisation combined to variable names should be enough to be read by a human, '''then''' comment can be used:
Code comments are surprisingly '''not''' a must-have for inside code; code organisation combined to variable names should be enough to be read by a human, '''then''' comment can be used:
** a comment should explain '''''why''''' the code is written this way
* a comment should explain '''''why''''' the code is written this way
** a comment should not tell '''''what''''' the code does; code should be self-explanatory
* a comment should not tell '''''what''''' the code does; code should be self-explanatory
** as a last resort in the event of a complex piece of code, a comment can be used to describe what the code actually does - or at least its intention
* as a last resort in the event of a complex piece of code, a comment can be used to describe what the code actually does - or at least its intention
*On the other hand, Documentation is more than welcome as it provides information from the outside without having to read the code. Enfusion uses [https://www.doxygen.nl/manual/docblocks.html Doxygen] format.
 
On the other hand, ''documentation'' is more than welcome as it provides information from the outside without having to read the code. Enfusion uses {{Link|Doxygen}}.


==== Files organisation ====
==== Files Organisation ====
{{Feature|informative|See [[Arma Reforger:Directory Structure|Directory Structure]] to know how/where to organise script files (Scripts\GameCode).}}
{{Feature|informative|See {{Link|Arma Reforger:Directory Structure}} to know how/where to organise script files (Scripts\GameCode).}}


* Have one class/enum per file
* Have one class/enum per file
** Small classes/enums can always be grouped together in the same file, provided they are part of the same system or only used there
** Small classes/enums can always be grouped together in the same file, provided they are part of the same system or only used there
* Use (sub-)directories to group related classes
* Use (sub-)directories to group related classes together




{{GameCategory|armaR|Modding|Tutorials|Scripting}}
{{GameCategory|armaR|Modding|Scripting|Guidelines}}

Latest revision as of 13:31, 26 February 2025

Getting Started

In the domain of development, any rule is a rule of thumb. If a rule states for example that it is better that a line of code doesn't go over 80 characters, it doesn't mean that any line must not go over 80 characters; sometimes, the situation needs it.

If the code has a good structure, do not change it to enforce a single arbitrary rule. If many of them are not implemented/not respected, changes should be applied; again, this is according to one's judgement.

With that being said, let's go!


Best Practices

See Scripting Conventions for all the conventions to date.

Code Format

  • Reminder: chosen indentation for Enfusion is Allman style
  • Reminder: indentation is done with tabulations
  • Use empty space. Line return, spaces before and after brackets, if this improves readability, use it: space is free
  • One-lining (putting everything in one statement) memory improvement is most of the time not worth the headache it gives when trying to read it: don't overuse it
    • it also hinders debugger's usage, e.g in the event of an inlined if

Variable Format

  • Name variables and functions properly: code must be readable by a human being, e.g variables like u instead of uniform should not exist.
    • i is an accepted iteration variable name (e.g in for loops).
  • Prefix any public content (classes, global methods, global variables) with a Creator Tag in order to prevent conflicts with other mods.
  • Use the closest value type whenever possible; using auto for a known variable type makes code more obscure and prevents autocompletion.

Code Structuration

SOLID

A series of development principles to follow in order to ensure an easy code maintenance and lifetime.

See SOLID.

DRY

Don't Repeat Yourself. If within the same class, the same code or the same pattern is written in various places, write a protected method and use appropriate parameters.

Logical Simplifications

If the code has too many repetitions, make a common method as stated above.

If the code has too many levels, it is time to split it and rethink it.

Examples

Improvable Good
auto number = 42; Animal cutePet = new Dog();
int number = 42; Dog cutePet = new Dog();
int i = 0; string result = ""; SCR_MyClass obj = null;
int i; // default value = 0 - see Values - integer string result; // default value = "" (a string cannot be null) SCR_MyClass obj; // default value = null
// a method call is more expensive than a bool check if (obj.MustBeTreated() || m_bTreatAllObjects) Print(obj); if (obj.MustBeTreated() && m_bTreatAllObjects) Print(obj);
// cheap checks go first, expensive checks (method calls) go after if (m_bTreatAllObjects || obj.MustBeTreated()) Print(obj); if (m_bTreatAllObjects && obj.MustBeTreated()) Print(obj);
// many identical method calls if (obj.MustBeTreated() && obj.GetObject()) Print("Result: " + obj.GetObject().m_sValue1 + " " + obj.GetObject().m_sValue2);
// "bigger", non-repetitive code can be beneficial for performance and readability if (obj.MustBeTreated()) { SCR_Object subObj = obj.GetObject(); if (subObj) Print("Result: " + subObj.m_sValue1 + " " + subObj.m_sValue2); }
foreach (SCR_Object obj : list) { Method(obj); // one method call per iteration } void Method(SCR_Object obj) { if (!obj) return; Print(obj.m_sName + " has a value of " + obj.m_sValue); }
foreach (SCR_Object obj : list) // the least method calls, the better { if (!obj) continue; Print(obj.m_sName + " has a value of " + obj.m_sValue); }
bool IsObjectAlive(SCR_Object obj) { if (!obj) return false; if (obj.m_Health > 0) // keep this structure for complex code return true; else return false; }
bool IsObjectAlive(SCR_Object obj) { return obj && obj.m_Health > 0; }
bool IsObjectValid(SCR_Object obj) { if (!obj) return false; return true; }
bool IsObjectValid(SCR_Object obj) { return obj != null; // for readability }
for (int i; i < list.Count(); i++) // list.Count() is called on every iteration { // ... }
for (int i, count = list.Count(); i < count; i++) // only one list.Count() call { // ... }
for (int i, count = list.Count(); i < count; i++) { if (list[i]) // first Print(list[i]); // and second .Get(i) method calls }
foreach (SCR_Object obj : list) // foreach is faster for start-to-end iterating { if (obj) Print(obj); // no additional method call }
for (int i, count = list.Count(); i < count; i++) { PrintFormat("Object #%1 = %2", i, list[i]); }
foreach (int i, SCR_Object obj : list) // iteration index is available this way too { PrintFormat("Object #%1 = %2", i, obj); }
// declaring an 'obj' every loop generates a pointer release each time foreach (SCR_ParentObject parent : list) { SCR_Object obj = parent.m_Object; if (obj) Print(obj.m_sName); }
SCR_Object obj; // external declaration = only one release at the end of the scope foreach (SCR_ParentObject parent : list) { obj = parent.m_Object; if (obj) Print(obj.m_sName); }
array<SCR_Object> toRemove = {}; foreach (SCR_Object obj : bigArray) { if (obj.m_bShouldBeRemoved) toRemove.Insert(obj); } foreach (SCR_Object obj : toRemove) { bigArray.RemoveItem(obj); // or RemoveItemOrdered if order is important }
for (int i = bigArray.Count() - 1; i >= 0; i--) // reverse iterating { if (bigArray[i].m_bShouldBeRemoved) bigArray.Remove(i); // or RemoveItemOrdered if order is important }
if (a) { if (b) { if (c) // also known as Hadouken code Method(true); else Method(false); } }
if (a && b) Method(c);
if (a) { Method(a); if (b) { Method(b); if (c) // another Hadouken code, with complications { Method(c); return 42; } else { return -1; } } else { return -1; } } else { return -1; }
if (!a) return -1; // this is called early return and helps funnel down the code Method(a); if (!b) return -1; Method(b); if (!c) return -1; Method(c); return 42;
int i; if (a) i++; if (b) i++; if (c) i++; // etc
int i; array<bool> conditions = { a, b, c, /* etc */ }; foreach (bool condition : conditions) { if (condition) i++; }
Initialise(player1, 1); Initialise(player2, 2); Initialise(player3, 3); Initialise(player4, 4); Initialise(player5, 5); Initialise(player6, 6);
array<IEntity> list = { player1, player2, player3, player4, player5, player6 }; foreach (int i, IEntity item : list) { Initialise(item, i + 1); }

// or, better, one method call that initialises all of them Initialise(list); // numbering is then done inside the method, if possible Initialise(list, 1); // otherwise the starting number can be provided

Code Comments

Code comments are surprisingly not a must-have for inside code; code organisation combined to variable names should be enough to be read by a human, then comment can be used:

  • a comment should explain why the code is written this way
  • a comment should not tell what the code does; code should be self-explanatory
  • as a last resort in the event of a complex piece of code, a comment can be used to describe what the code actually does - or at least its intention

On the other hand, documentation is more than welcome as it provides information from the outside without having to read the code. Enfusion uses Doxygen.

Files Organisation

See Directory Structure to know how/where to organise script files (Scripts\GameCode).
  • Have one class/enum per file
    • Small classes/enums can always be grouped together in the same file, provided they are part of the same system or only used there
  • Use (sub-)directories to group related classes together