Firing Drills – Arma 3

From Bohemia Interactive Community
Jump to navigation Jump to search
m (Added target configuration)
m (Most basic information added)
Line 6: Line 6:
See also: [[Arma 3 Field Manual - Firing Drills]] (for players)
See also: [[Arma 3 Field Manual - Firing Drills]] (for players)
{{wip}}
{{wip}}


= Terminology =
= Terminology =
Line 50: Line 51:
# For each CP (including start and finish), insert a trigger with default properties named f.e. {{Inline code|BIS_FD_CP1}} {{cc|Referenced later}}
# For each CP (including start and finish), insert a trigger with default properties named f.e. {{Inline code|BIS_FD_CP1}} {{cc|Referenced later}}
## Re-size the trigger to meet your CP activation expectations.
## Re-size the trigger to meet your CP activation expectations.
## For each optional shooting box within the CP, insert a trigger with default properties named f.e. {{Inline code|BIS_FD_CP1_Box}} {{cc|Referenced later}}
# Insert a ''Target - Oval (Ground)'' prop named {{Inline code|BIS_FD_ruleSelector1}} {{cc|Referenced later}}
# Insert a ''Target - Oval (Ground)'' prop named {{Inline code|BIS_FD_ruleSelector1}} {{cc|Referenced later}}
## Using a similar workflow as above, manually change its class name to ''Land_Target_Oval_NoPop_F'' (all targets need their hidden ''NoPop'' variant).
## Using a similar workflow as above, manually change its class name to ''Land_Target_Oval_NoPop_F'' (all targets need their hidden ''NoPop'' variant).
Line 72: Line 74:


== Randomized Targets ==
== Randomized Targets ==


== Common Additions ==
== Common Additions ==
Line 150: Line 151:
magazinesCompetitor[] = {"30Rnd_9x21_Mag", "16Rnd_9x21_Mag"}; // Starting magazines for the competitor
magazinesCompetitor[] = {"30Rnd_9x21_Mag", "16Rnd_9x21_Mag"}; // Starting magazines for the competitor
magazinesCompetitorCount = 6; // The starting magazine count per weapon for the competitor
magazinesCompetitorCount = 6; // The starting magazine count per weapon for the competitor
competitorStart = "BIS_rangeOfficer1A"; // Marker reference to a non-default (re)starting position for the competitor (used for the special intro in CoF: Green)
timesMedals[] = { 60, 45, 30 }; // Bronze, silver, gold medal times (be sure to follow this order or results may glitch)
timesMedals[] = { 60, 45, 30 }; // Bronze, silver, gold medal times (be sure to follow this order or results may glitch)
Line 191: Line 193:
object = "BIS_FD_CP2";
object = "BIS_FD_CP2";


// Normally targets (positions) are only randomized once on starting a drill, not when restarting it in the same session.
// Normally targets (positions) are only randomized once on starting a drill, not when restarting it in the same session
// This changes that (undefined uses 0).
// This changes that (undefined uses 0)
resetRandomizeTargets = 1;
resetRandomizeTargets = 1;
disableFeedback = 1; // Disables the default Range Officer radio feedback for this CP (undefined uses 0)
disableFeedback = 1; // Disables the default Range Officer radio feedback for this CP (undefined uses 0)
Line 199: Line 201:
onDeactivate = "CP2_onDeactivate.sqf"; // Script executed when this CP is deactivated (passed [CP object])
onDeactivate = "CP2_onDeactivate.sqf"; // Script executed when this CP is deactivated (passed [CP object])
onClear = "CP2_onClear.sqf"; // Script executed when this CP is cleared (passed [CP object])
onClear = "CP2_onClear.sqf"; // Script executed when this CP is cleared (passed [CP object])
weapons[] = {"hgun_Rook40_F"}; // Makes this the required weapon to use at this CP when strict / competitive rules are used
weapons[] = {"hgun_Rook40_F"}; // Makes this the required weapon(s) to use at this CP when strict / competitive rules are used
stances[] = {"crouch"}; // Makes this the required stance(s) to use at this CP when strict / competitive rules are used (prone / crouch / stand)
shootingBoxes[] = {"BIS_FD_CP2_Box"}; // Trigger reference to one or more shooting boxes within the CP when strict / competitive rules are used
binocularCompetitor = "Binocular"; // Optionally defining a non-default binocular class (this is the default)
nvgCompetitor = ""; // Optionally defining a non-default NVG class (this is the default so no NVG)


// Configuring all targets associated with this CP
// Configuring all targets associated with this CP
Line 245: Line 251:
3
3
};
};
hitCondition = "targets_hitCondition.sqf"; // Special script executed when the target hit, which can determine if the hit is legal or not (return true to count as hit, false otherwise)
hitCondition = "targets_hitCondition.sqf"; // Special compiled script called when the target is hit, which can determine if the hit is legal or not (return true to count as hit, false otherwise)
};
class T4
{
object = "BIS_FD_target2_4";
type = "PopUp_Acc2";
upTime = 10;
triggerChance = 0.5; // Chance to trigger this target at all (0 - 1, so 50% in this case)
};
class T5
{
object = "BIS_FD_target2_5";
type = "PopUp";
 
// Chance for this trigger to even spawn at all (0 - 1, so 80% in this case)
// Be careful not to create impossible situations, where this target is a trigger for another target!
chance = 0.8;
};
};
class NoShoot1  
class NoShoot1  
Line 390: Line 412:
= Targets =
= Targets =


{{Informative|These are the vanilla target type profiles you can reference in your drill's CfgFiringDrills class}}
{{Informative|These are the vanilla target type profiles you can reference in your drill's ''CfgFiringDrills'' class}}
TODO: special Eden Editor properties for best results
* '''PopUp''': classic human-shaped pop-up targets (all target picture variants)
* '''PopUp''': classic human-shaped pop-up targets (all target picture variants)
* '''PopUp_Acc1''': the same targets, but with the accuracy zones
* '''PopUp_Acc1''': the same targets, but with the accuracy zones
Line 397: Line 420:
* '''Swivel''': swivel variants of the pop-up targets
* '''Swivel''': swivel variants of the pop-up targets
* '''PopUp_Oval''': simple round metal plate pop-up targets
* '''PopUp_Oval''': simple round metal plate pop-up targets
* '''Duel''': dueling target (this one effectively requires setting up ''hitCondition'' and other scripts to work well)
* '''Duel''': dueling target (this one effectively requires setting up ''hitCondition'' and other event scripts to work well)
* '''Skeet''': clay skeets
* '''Skeet''': clay skeets
* '''Metal_Pole_Skeet''': metal poles with a clay skeet mounted to them
* '''Metal_Pole_Skeet''': metal poles with a clay skeet mounted to them
Line 409: Line 432:




= Run-Time Overrides =
= Run-Time Tools =


* TODO: Competitor weapons / magazines
{{Informative|Some things are exposed and possible to change or use run-time, f.e. to make special unlocks or events happen.}}
* ''BIS_FD_weaponsCompetitor'': an array of weapon classes provided to the competitor when the drill (re)starts
* ''BIS_FD_magazinesCompetitor'': an array of magazine classes provided to the competitor when the drill (re)starts
* ''BIS_FD_magazinesCompetitorCount'': the amount of magazines per type provided to the competitor when the drill (re)starts - default is 5
* ''BIS_FD_targetHitCode'': code which can be called with dummy parameters to simulate a fake target hit for special events (expects ''hitPart'' Event Handler parameters)
* ''BIS_FD_hasReset'': code which returns true when the drill has reset, ended, or is ending
* ''BIS_FD_phase'': 0 - drill pre-init / 1 - drill selecting rules / 2 - drill started / 3 - drill terminated / -1 - drill restarting




= Colors =
= Colors =


TODO
{{Important|While you can define any color you want for a drill, only several are fully supported. These colors are used for: object recoloring, HUD, frangible target hit VFX, map markers, etc. If you want to use another ''color'', consider selecting the nearest matching ''colorName''.}}
* '''"red"''': multi-gun
** {{Inline code|__EVAL(177/255), __EVAL(51/255), __EVAL(57/255), 1}}
* '''"green"''': introduction / tutorial
** {{Inline code|__EVAL(173/255), __EVAL(191/255), __EVAL(131/255), 1}}
* '''"orange"''': long-range
** {{Inline code|__EVAL(240/255), __EVAL(130/255), __EVAL(49/255), 1}}
* '''"blue"''': Close Quarters Combat
** {{Inline code|__EVAL(103/255), __EVAL(139/255), __EVAL(155/255), 1}}
* '''"purple"''': specialist marksman
** {{Inline code|__EVAL(176/255), __EVAL(64/255), __EVAL(167/255), 1}}
* '''"gray"''': CTRG
** {{Inline code|__EVAL(90/255), __EVAL(89/255), __EVAL(90/255), 1}}
* '''"white"''': not used in vanilla




Line 429: Line 471:
* For random / moving target position references, use small colored helper spheres in Eden Editor. Just remember to hide them all in an init script for example: {{Inline code|{hideObject _x} forEach (allMissionObjects "Sign_Sphere10cm_F");}}
* For random / moving target position references, use small colored helper spheres in Eden Editor. Just remember to hide them all in an init script for example: {{Inline code|{hideObject _x} forEach (allMissionObjects "Sign_Sphere10cm_F");}}
* In special cases you may wish for a more complex target trigger. Do this by placing a dummy trigger object (f.e. a Game Logic) with named reference. Configure this reference as the target's ''trigger''. When you want to trigger this target, execute in any script environment: {{Inline code|YourTriggerObject setVariable ["state", 3];}}
* In special cases you may wish for a more complex target trigger. Do this by placing a dummy trigger object (f.e. a Game Logic) with named reference. Configure this reference as the target's ''trigger''. When you want to trigger this target, execute in any script environment: {{Inline code|YourTriggerObject setVariable ["state", 3];}}
* There are several useful variables stored in the scenario and CP / target objects (but in most cases manipulating them directly will break the system; read them only):
** ''BIS_FD_CPs'': array of CP object references
** CP
*** ''"CP"'': CP index number
*** ''"activated"'': true when activated
*** ''"clear"'': true when cleared
** Target (most configuration parameters are also stored as similarly named variables, f.e. ''"optional"'')
*** ''"state"'': 0 - spawned / 1 - will trigger / 2 - triggered / 3 - hit / 4 - missed
*** ''"targets"'': array of all CP target object references
* In order to make targets that are not positioned on the ground work well, you will need to execute on them: {{Inline code|this setVariable ["BIS_DynO_ASL", true]}}
* While designing a drill, you may wish to use function ''BIS_fnc_moduleFDAutoTraining'', which let's you quickly go to a specific CP.
** {{Inline code|[5, true] spawn BIS_fnc_moduleFDAutoTraining}} {{cc| Skip instantly to CP with index 5 (using false will teleport between CPs until reaching the specified one)}}
* Vanilla no-shoot targets are using the ''Simple Target'' (''Target_F'') with init: {{Inline code|this setObjectTexture [0, '']; this setObjectTexture [1, '#(argb,8,8,3)color(0.4,0.0,0.5,1.0,co)'];}}
* TODO: medal time / bonus balancing
* TODO: medal time / bonus balancing
* TODO: useful variables stored in CP / target objects
* TODO: vanilla no-shoot targets




[[Category:Arma 3: Editing|Firing Drills]]
[[Category:Arma 3: Editing|Firing Drills]]

Revision as of 16:33, 22 October 2020

Template:SideTOC Template:Cfg ref

This guide will describe how to design and implement a custom Firing Drill Challenge as available in the vanilla game.

See also: Arma 3 Field Manual - Firing Drills (for players) Template:wip


Terminology

  • Firing Drill (FD) - a live-fire exercise in which a competitor is graded for tactically moving through a course while engaging a set of targets
  • Competitor - the actor performing the drill
  • Range Officer (RO) - the actor observing the drill and providing feedback / guidance
  • Course of Fire (CoF) - the physical space and layout in which the drill is performed
  • Checkpoint (CP) - a navigation point within a course which a competitor must follow in a fixed sequence (if so defined as a rule - see below)
    • The first CP is the start.
    • The last CP is the finish.
    • A strict CP contains shooting boxes (marked areas) from which to fire.
  • Target - an artificial object shot to complete a drill or to receive time bonuses
    • Appearing targets appear and disappear based on activation triggers (see below).
    • Accuracy targets contain zones corresponding to extra time bonuses.
    • Moving targets travel on rails between positions.
    • No-shoot targets are not allowed be hit (adding a time penalty).
    • Bonus targets are optional (rewarding a time bonus).
  • Rule - these help to determine the completion and scoring conditions for a drill
    • Training rules - a rule-set allowing familiarization of the CoF (targets at CPs do not need to be shot to move on)
    • Competitive rules - a rule-set used for Achievements / Leaderboards (targets at CPs need to be shot to move on)
  • Time - the time from the start it takes to complete a drill (raw)
    • Time bonuses / penalties are substracted / added to arrive at the final time.
    • The fastest / lowest time is the best and wins.
    • There are bronze, silver and gold medal times to beat.
    • There may be one special time to beat (typically faster than gold).


Setup

At time of writing it is not possible to add new Firing Drills as independent (Workshop) scenarios; they need to be added as 'mods' with configuration.
  1. Insert a player character named BIS_FD_Competitor // Fixed system name
  2. Insert a NPC Range Master character of type B_RangeMaster_F named BIS_rangeOfficer // Fixed system name
  3. Insert a Firing Drill module
    1. It's not exposed in Eden Editor, so place any other module. We may change this after this documentation is completed.
    2. Save the scenario.
    3. Close Eden Editor to avoid auto-saves.
    4. Open the scenario SQM file in a text editor.
    5. Replace the module's class by ModuleFiringDrill_F.
    6. Save the SQM.
    7. Re-open the scenario in Eden Editor.
  4. Insert an ellipse trigger with default properties named f.e. BIS_FD_COF // Referenced later
    1. Re-size the trigger to encompass your entire CoF and a little extra buffer area (you'll likely re-size it later as you progress).
  5. For each CP (including start and finish), insert a trigger with default properties named f.e. BIS_FD_CP1 // Referenced later
    1. Re-size the trigger to meet your CP activation expectations.
    2. For each optional shooting box within the CP, insert a trigger with default properties named f.e. BIS_FD_CP1_Box // Referenced later
  6. Insert a Target - Oval (Ground) prop named BIS_FD_ruleSelector1 // Referenced later
    1. Using a similar workflow as above, manually change its class name to Land_Target_Oval_NoPop_F (all targets need their hidden NoPop variant).
    2. Position this target in comfortable shooting range / direction from the start CP (a few meters at most, oriented to be clearly visible and safe).
    3. In the target's Texture #0 field, enter: A3\modules_f_beta\data\FiringDrills\training_gray_ca // Other colors have corresponding textures - see below
  7. Copy and paste the first rule target and name it BIS_FD_ruleSelector2 // Referenced later
    1. Position it near the first rule target, typically right from it.
    2. In the target's Texture #0 field, enter: A3\modules_f_beta\data\FiringDrills\competitive_gray_ca // Other colors have corresponding textures - see below
  8. Copy and paste a rule target and name it BIS_FD_restartSelector1 // Referenced later
    1. Position it near the finish CP (a few meters from it, oriented to be clearly visible and safe).
    2. In the target's Texture #0 field, enter: A3\modules_f_beta\data\FiringDrills\restart_gray_ca // Other colors have corresponding textures - see below
  9. Copy and paste a rule target and name it BIS_FD_quitSelector1 // Referenced later
    1. Position it near the restart target, typically right from it.
    2. In the target's Texture #0 field, enter: A3\modules_f_beta\data\FiringDrills\quit_ca // Fixed texture no matter the drill color
  10. For each target at each CP, generally do the following (more specifics detailed below):
    1. Insert the desired target object named f.e. BIS_FD_target2_1 // Referenced later (apply a logical naming pattern, such as numbering based on CP index and target index)
    2. Using a similar workflow as above, manually replace its class name with the corresponding NoPop variant (see the look-up table below).
  11. Set up drill properties via description.ext (see below).
  12. Configure your drill via CfgFiringDrills and CfgMissions (see below).

Moving Targets

Randomized Targets

Common Additions

  • CP marking objects
    • Recolored objects
  • Start whiteboard
  • Ammunition containers
  • Decorative props / sponsoring


Configuration

description.ext

#include "\A3\Missions_F_Beta\Challenges\description.hpp" // This will set up various system default settings, such as custom debriefings

onLoadName = "CoF: My First Drill"; // Name of your drill (normally the same as defined in CfgMissions)
onLoadMission = "Can you beat my splendid drill?!"; // Overview text of your drill (normally the same as defined in CfgMissions)
loadScreen = "myfirstfiringdrill_overview_CO.paa"; // Overview picture of your drill (normally the same as defined in CfgMissions)
briefingName = "CoF: My First Drill"; // Same name as above
overviewPicture = "myfirstfiringdrill_overview_CO.paa"; // Same picture as above
overviewText = "Can you beat my splendid drill?!"; // Same text as above
author = "Range Officer Bob"; // You!

doneKeys[] = {"MyFirstFiringDrill_done"}; // Registers having completed (achieved competitive gold or special) your drill

CfgFiringDrills

Drill

class MyFirstFiringDrill // This class name must correspond to the missionName (also CfgMissions class)
{
	displayName = "CoF: My First Drill"; // Vanilla drills use a format like this, but it's not enforced
	// This color is used in many places
	// It can be any color, but for best results pick a fully supported color (see section below)
	color[] = { __EVAL(177/255), __EVAL(51/255), __EVAL(57/255), 1 };
	colorName = "red"; // This version of the color is more restrictive and can only use specific supported colors (see section below)
	objectCOF = "BIS_FD_COF"; // The CoF trigger referenced in the setup guide
	// statistic = ""; // Steam stats are not supported for user-generated drills (used for Achievements)
	// leaderboard = ""; // Steam Leaderboards are not supported for user-generated drills
	onReset = "reset.sqf"; // This script is executed each time the drill resets (restarts)
	
	// List any number of objects that you want to be re-colored to the drill color above (f.e. road cones, small flags, etc.)
	recolor[] =
	{
		"BIS_FD_recolor1", 
		"BIS_FD_recolor2"
	};

	// Set up objects that should have special decal textures applied (this was more useful before Eden Editor - see vanilla decals below)
	// In vanilla drills these are often graffiti on wooden shoot house walls
	decals[] =
	{
		{ "BIS_FD_decal1", "A3\Missions_F_Beta\data\img\decals\decal_sidearm1_ca", 5 } // Object reference, texture path, hidden selection index
	};

	// Object references to the 4 special targets for drill mechanics, as defined in the setup guide
	ruleSelectors[] = 
	{
		"BIS_FD_ruleSelector1", 
		"BIS_FD_ruleSelector2"
	};
	restartSelectors[] = { "BIS_FD_restartSelector1" };
	quitSelectors[] = { "BIS_FD_quitSelector1" };
		
	containersWeapons[] = { "BIS_weaponsFD1" }; // All containers that should be filled with the drill weapons
	containersMagazines[] = { "BIS_ammoFD1_1", "BIS_ammoFD1_2", "BIS_ammoFD1_3", "BIS_ammoFD1_4" }; // All containers that should be filled with the drill ammunition
	containersMagazinesCount[] = { 10, 10 }; // The count of magazines of each type in the containers
	weapons[] = { "SMG_02_ACO_F", "hgun_Rook40_F" }; // The legal weapons for this drill
	magazines[] = { "30Rnd_9x21_Mag", "16Rnd_9x21_Mag" }; // The legal magazines for this drill
	weaponsCompetitor[] = { "SMG_02_ACO_F", "hgun_Rook40_F" }; // Starting weapons for the competitor
	magazinesCompetitor[] = {"30Rnd_9x21_Mag", "16Rnd_9x21_Mag"}; // Starting magazines for the competitor
	magazinesCompetitorCount = 6; // The starting magazine count per weapon for the competitor
	competitorStart = "BIS_rangeOfficer1A"; // Marker reference to a non-default (re)starting position for the competitor (used for the special intro in CoF: Green)
		
	timesMedals[] = { 60, 45, 30 }; // Bronze, silver, gold medal times (be sure to follow this order or results may glitch)
		
	timeSpecial = 15; // Optional special time that is faster than gold (when defining this, you do need to also provide the data below)
	nameSpecial = "Nemisis"; // Optional special time label for the HUD
	colorSpecial = "#ffa500"; // Optional special time color for the HUD
	iconSpecial = "\A3\Ui_f\data\GUI\Cfg\Ranks\colonel_gs"; // Optional special time icon for the HUD
	
	// Rule-set parameters for competitive and training
	// All vanilla drills use these same settings
	// It is possible to tweak these values, but that will likely create confusion for players!
	class RuleSets 
	{
		// CP mode: 1 - CPs on but targets don't need shooting / 2 - CPs on and all non-optional targets need shooting
		// Target randomization: rule not used - ignored
		// Weapons used: 1 - any weapon can be used everywhere / 2 - CPs may define a required weapon (or incur a time penalty)
		// Shooting boxes: 1 - shooting boxes in CPs are ignored / 2 - shooting boxes in CPs are forced (shooting outside of them incurs a time penalty)
		// Stances used: 1 - any stance can be used everywhere / 2 - CPs may define a required stance (or incur a time penalty)
		// Medals: 0 - medal times are not tracked / 1 - medal times are tracked and feedback is given 
		competitive[] = { 2, 0, 2, 2, 2, 1 };
		training[] = { 1, 2, 1, 1, 1, 0 };
	};
	
	// Legacy class used before Eden Editor to load precise object compositions (should not be needed anymore)
	// class DynOs 
	// {
	//		script = "dyno_myfirstfiringdrill.sqf"; // Valid DynO mapper output script
	//		positionAnchor[] = {500, 500}; // DynO mapper anchor position in the world
	// }; 

	// All of the drill's CPs
	class CheckPoints 
	{
		class CP1 // Start
		{
			object = "BIS_FD_CP1"; // The CP trigger referenced in the setup guide
		};
		class CP2 // CP #1
		{
			object = "BIS_FD_CP2";

			// Normally targets (positions) are only randomized once on starting a drill, not when restarting it in the same session
			// This changes that (undefined uses 0)
			resetRandomizeTargets = 1;
			disableFeedback = 1; // Disables the default Range Officer radio feedback for this CP (undefined uses 0)
			height = 5; // Only use this ASL forced height for CPs above ground, such as on top of a movement obstacle or inside a building on a raised floor
			onActivate = "CP2_onActivate.sqf"; // Script executed when this CP is activated (passed [CP object])
			onDeactivate = "CP2_onDeactivate.sqf"; // Script executed when this CP is deactivated (passed [CP object])
			onClear = "CP2_onClear.sqf"; // Script executed when this CP is cleared (passed [CP object])
			weapons[] = {"hgun_Rook40_F"}; // Makes this the required weapon(s) to use at this CP when strict / competitive rules are used
			stances[] = {"crouch"}; // Makes this the required stance(s) to use at this CP when strict / competitive rules are used (prone / crouch / stand)
			shootingBoxes[] = {"BIS_FD_CP2_Box"}; // Trigger reference to one or more shooting boxes within the CP when strict / competitive rules are used
			binocularCompetitor = "Binocular"; // Optionally defining a non-default binocular class (this is the default)
			nvgCompetitor = ""; // Optionally defining a non-default NVG class (this is the default so no NVG)

			// Configuring all targets associated with this CP
			class Targets 
			{
				class T1 
				{
					object = "BIS_FD_target2_1"; // Object reference for the target placed as in the setup guide
					type = "PopUp_Acc1"; // Target type profile as defined in the central CfgFiringDrills (see vanilla targets below)
					upTime = 5; // How long in seconds does the target stay active before counting as missed (if not hit)?
				};
				class T2: T1 
				{
					object = "BIS_FD_target2_2";
					trigger = "BIS_FD_target2_1"; // Reference to another target which must be hit for this one to activate
					triggerTime = 2; // How long in seconds after triggering this target does it actually activate?

					// Marker or object references to positions and directions used (in pairs) to randomize the target 
					// Normally it would happen once when starting, but for this CP every time you restart
					positions[] = 
					{
						"BIS_FD_target2_2", 
						"BIS_FD_target2_2_Pos2"
					};
					directions[] = 
					{
						"BIS_FD_target2_2", 
						"BIS_FD_target2_2_Pos2"
					};
				};
				class T3 
				{
					object = "BIS_FD_target2_3";
					type = "PopUp";
					trigger[] = {"BIS_FD_target2_1", "BIS_FD_target2_2"}; // Triggers can also be multiple other targets to have hit
					
					// Marker or object position references to form a chain of movement waypoints for this target
					// These are matched to the times in seconds for each segment of movement
					positionsMove[] = 
					{
						"BIS_FD_target8_1_WP1"
					};
					timesMove[] = 
					{
						3
					};
					hitCondition = "targets_hitCondition.sqf"; // Special compiled script called when the target is hit, which can determine if the hit is legal or not (return true to count as hit, false otherwise)
				};
				class T4 
				{
					object = "BIS_FD_target2_4";
					type = "PopUp_Acc2";
					upTime = 10;
					triggerChance = 0.5; // Chance to trigger this target at all (0 - 1, so 50% in this case)
				};
				class T5 
				{
					object = "BIS_FD_target2_5";
					type = "PopUp";

					// Chance for this trigger to even spawn at all (0 - 1, so 80% in this case)
					// Be careful not to create impossible situations, where this target is a trigger for another target!
					chance = 0.8;
				};
				class NoShoot1 
				{
					object = "BIS_FD_noShoot2_1";
					noShoot = 1; // Makes this target illegal to hit (incurring a time penalty if hit)
				};
				class Bonus1
				{
					object = "BIS_FD_bonus2_1";
					optional = 1; // Makes this an optional target that does not need shooting even on strict / competitive rules
					noRecolor = 1; // By default most optional targets get recolored to the drill color, but here you can disable that (undefined uses 0)
					bonus = 5; // Defines the bonus points received for hitting this target (translates to -5 seconds)
					type = "Balloon"; // Some vanilla target type profiles are well-suited as optional, such as this one
					onActivate = "targets_onActivate.sqf"; // Script executed when this target activates (passed [target object])
					onDeactivate = "targets_onDeactivate.sqf"; // Script executed when this target deactivates (passed [target object])
					onHit = "targets_onHit.sqf"; // Script executed on hitting this target (passed [target, projectile velocity array, hit selections array, shooter object])
					onReset = "targets_onReset.sqf"; // Script executed when this target resets (passed [target object])
				};
			};
		};
		class CP3 // Finish
		{
			object = "BIS_FD_CP3";
		};
	};
};

System

The system-level configuration could be adjusted by a mod, but consider if that is not going to confuse players who may be trained on vanilla drills.
class CfgFiringDrills
{
	// Below point bonuses and penalties are multiplied by this factor to determine time bonuses and penalties
	pointTimeMultiplier = 0.1;

	bonusWeapon = 1; // Bonus for using the correct weapon at a CP (never awarded - disabled)
	bonusShootingBox = 3; // Bonus for shooting from shooting boxes at a CP (never awarded - disabled)
	bonusStance = 1.5; // Bonus for using the correct stance at a CP (never awarded - disabled)
	penaltyWeapon = 3; // Penalty for engaging using an incorrect weapon at a CP
	penaltyShootingBox = 5; //Penalty for engaging from outside a shooting box at a CP
	penaltyStance = 2; // Penalty for engaging from the incorrect stance at a CP
	penaltyNoShoot = 50; // Penalty for hitting a no-shoot target (so this is 5 seconds)
	penaltyMissed = 25; // Penalty for failing to hit an activated target

	// HUD icons for the medal times
	iconsMedals[] =
	{
		"A3\modules_f_beta\data\FiringDrills\medal_bronze_ca", // Bronze
		"A3\modules_f_beta\data\FiringDrills\medal_silver_ca", // Silver
		"A3\modules_f_beta\data\FiringDrills\medal_gold_ca" // Gold
	};
	// HUD colors for the medal times
	colorsMedals[] =
	{
		"#A0522D", // Bronze
		"#C0C0C0", // Silver
		"#FFD700" // Gold
	};

	// The vanilla configuration defines several common target type profiles
	// This example configures a theoretical new one
	class Targets 
	{
		class MyFirstTarget 
		{
			selectionsHit[] = {"target"}; // All named selections on the model that count as valid hit
			onReset = "myFirstTarget_onReset.sqf"; // Optional script executed when the target is reset
			onActivate = "myFirstTarget_onActivate.sqf"; // Optional script executed when the target is activated
			onDeactivate = "myFirstTarget_onDeactivate.sqf"; // Optional script executed when the target is deactivated
			onHit = "myFirstTarget_onHit.sqf"; // Optional script executed when the target is hit while activated
			downTime = 0.5; // An extra delay in seconds before a valid target hit is registered (i.e. onHit is executed - undefined uses 0)
			noRecolor = 1; // This target is never re-colored by the drill color (mainly applies to optional targets - undefined uses 0)

			// Certain targets over 200 meters away from the competitor will open a Picture-in-Picture camera feed to show hits
			cameraPos[] = {0, -1.5, 0.5}; // Relative camera position from the target center
			cameraTarget[] = {0, 0, 1}; // Relative camera target from the target center
			cameraFOV = 0.4; // Camera Field of View setting (zoom)

			// Accuracy (bonus) hit zones
			class Zones
			{
				class Top
				{
					center[] = {0, 0, 0.5}; //Relative center position of the zone from the target center
					maxDistances[] = //Radii in meters of accuracy rings within this zone
					{
						0.15,
						0.25
					};
					bonuses[] = //Bonuses associated to the rings defined above
					{
						6,
						3
					};
				};
			};

			// In some cases the same target model has several texture variants with different accuracy zones
			// These can be overridden within the same target profile by creating a sub-class of the same class as the specific target itself
			// In that sub-class, define only the relevant class Zones
			// class MyTargetType 
			// {
			// 	class Zones { ... };
			// };
		};
	};
};

CfgMissions

class CfgMissions
{
	// This will expose the drill in the SINGLEPLAYER > CHALLENGES > Firing Drills menu
	class Challenges
	{
		class Firing_Drills
		{
			class MyFirstFiringDrill // This class should correspond to the class in CfgFiringDrills above
			{
				directory = "myfirstfiringdrill.stratis"; // Full path to your drill's scenario folder
				briefingName = "CoF: My First Drill"; // Name of your drill (normally the same as defined in CfgFiringDrills)
				overviewText = "Can you beat my splendid drill?!"; // Overview text of your drill
				overviewPicture = "myfirstfiringdrill_overview_CO.paa"; // Overview picture of your drill
				overviewScript = "\A3\Modules_F_Beta\FiringDrills\scripts\overviewScript.sqf"; // Standard system script to handle the pause menu properly - don't change
				author = "Range Officer Bob"; // You!
			};

			// Alternatively you can list your drill in one of the vanilla color categories: Red, Blue, Orange, Purple
			// class Blue 
			// {
			// 		Define your drill here instead for example
			// };
		};
	};
};


Targets

These are the vanilla target type profiles you can reference in your drill's CfgFiringDrills class

TODO: special Eden Editor properties for best results

  • PopUp: classic human-shaped pop-up targets (all target picture variants)
  • PopUp_Acc1: the same targets, but with the accuracy zones
  • PopUp_Zom_Acc1: zombie-variant of the target with accuracy zones
  • PopUp_Acc2: the same targets, but with the bull's eye paper
  • Swivel: swivel variants of the pop-up targets
  • PopUp_Oval: simple round metal plate pop-up targets
  • Duel: dueling target (this one effectively requires setting up hitCondition and other event scripts to work well)
  • Skeet: clay skeets
  • Metal_Pole_Skeet: metal poles with a clay skeet mounted to them
  • Steel_Plate: the simplest of standing steel plates
  • Balloon: air and water balloons (often used as optional bonus)
  • Orange: fruit (often used as optional bonus)
  • Pumpkin: fruit (often used as optional bonus)
  • Aftershave: "Arma" Eau de Combat (often used as optional bonus)
  • VR_Entity: special-purpose in CoF: Gray
  • VR_Skeet: special-purpose in CoF: Gray


Run-Time Tools

Some things are exposed and possible to change or use run-time, f.e. to make special unlocks or events happen.
  • BIS_FD_weaponsCompetitor: an array of weapon classes provided to the competitor when the drill (re)starts
  • BIS_FD_magazinesCompetitor: an array of magazine classes provided to the competitor when the drill (re)starts
  • BIS_FD_magazinesCompetitorCount: the amount of magazines per type provided to the competitor when the drill (re)starts - default is 5
  • BIS_FD_targetHitCode: code which can be called with dummy parameters to simulate a fake target hit for special events (expects hitPart Event Handler parameters)
  • BIS_FD_hasReset: code which returns true when the drill has reset, ended, or is ending
  • BIS_FD_phase: 0 - drill pre-init / 1 - drill selecting rules / 2 - drill started / 3 - drill terminated / -1 - drill restarting


Colors

While you can define any color you want for a drill, only several are fully supported. These colors are used for: object recoloring, HUD, frangible target hit VFX, map markers, etc. If you want to use another color, consider selecting the nearest matching colorName.
  • "red": multi-gun
    • __EVAL(177/255), __EVAL(51/255), __EVAL(57/255), 1
  • "green": introduction / tutorial
    • __EVAL(173/255), __EVAL(191/255), __EVAL(131/255), 1
  • "orange": long-range
    • __EVAL(240/255), __EVAL(130/255), __EVAL(49/255), 1
  • "blue": Close Quarters Combat
    • __EVAL(103/255), __EVAL(139/255), __EVAL(155/255), 1
  • "purple": specialist marksman
    • __EVAL(176/255), __EVAL(64/255), __EVAL(167/255), 1
  • "gray": CTRG
    • __EVAL(90/255), __EVAL(89/255), __EVAL(90/255), 1
  • "white": not used in vanilla


Decals

TODO


Tips

  • Vanilla CoFs are designed to be somewhat realistic and safe as a non-lethal competitive shooting training / sport. Try to imagine and implement range safety concerns in your target and prop placement. For example, don't place targets in a line of sight of the start, Range Officer, or other spectators. Generally try to solidly block off areas behind targets where possible.
  • For random / moving target position references, use small colored helper spheres in Eden Editor. Just remember to hide them all in an init script for example: {hideObject _x} forEach (allMissionObjects "Sign_Sphere10cm_F");
  • In special cases you may wish for a more complex target trigger. Do this by placing a dummy trigger object (f.e. a Game Logic) with named reference. Configure this reference as the target's trigger. When you want to trigger this target, execute in any script environment: YourTriggerObject setVariable ["state", 3];
  • There are several useful variables stored in the scenario and CP / target objects (but in most cases manipulating them directly will break the system; read them only):
    • BIS_FD_CPs: array of CP object references
    • CP
      • "CP": CP index number
      • "activated": true when activated
      • "clear": true when cleared
    • Target (most configuration parameters are also stored as similarly named variables, f.e. "optional")
      • "state": 0 - spawned / 1 - will trigger / 2 - triggered / 3 - hit / 4 - missed
      • "targets": array of all CP target object references
  • In order to make targets that are not positioned on the ground work well, you will need to execute on them: this setVariable ["BIS_DynO_ASL", true]
  • While designing a drill, you may wish to use function BIS_fnc_moduleFDAutoTraining, which let's you quickly go to a specific CP.
    • [5, true] spawn BIS_fnc_moduleFDAutoTraining // Skip instantly to CP with index 5 (using false will teleport between CPs until reaching the specified one)
  • Vanilla no-shoot targets are using the Simple Target (Target_F) with init: this setObjectTexture [0, ]; this setObjectTexture [1, '#(argb,8,8,3)color(0.4,0.0,0.5,1.0,co)'];
  • TODO: medal time / bonus balancing