Scripting: Difference between revisions

From EDukeWiki
Jump to navigation Jump to search
No edit summary
 
(134 intermediate revisions by 6 users not shown)
Line 1: Line 1:
{{EDuke32 Intro}}
{{EDuke32 Intro}}
<div class="floatleft">__NOTOC__</div> <!-- changed to NOTOC to get rid of it temporarily -->
<div class="floatright">__TOC__</div>
==About This Guide==
==About This Guide==
===Intro===
===Intro===
This guide will get you started with the basic aspects of EDuke32's commands -- those which set it apart from vanilla DN3D.  This guide makes no assumptions regarding the programming experience of the reader, other than that the reader comprehends and understands the basic CON system as was seen in Duke Nukem 3D 1.3D/1.5.
This guide will get you started with the basic aspects of EDuke32's commands -- those which set it apart from vanilla DN3D.  This guide assumes both that the reader comprehends and understands the basic CON system as was seen in Duke Nukem 3D 1.3D/1.5, and that the reader is familiar with a few basic programming concepts.


===CON Basics===
===CON Basics===
As stated, this guide assume that you are familiar with the original CON code from Duke3D.  If you are not already familiar with the [[:Category:Duke3D_1.3/1.5_commands|default commands]], the authors of this guide recommend the following guide to the basics, imported into the EDukeWiki for your convenience:
*full command list
**[[Full command list|alphabetized]]
**[[:Category:All commands|categorized]]
If you are not already familiar with the default commands, the authors of this guide recommend the following guide, imported into the EDukeWiki for your convenience:
* [[Confaq42|Con FAQ 4.2 by Jonah Bishop (note: some sections have been shuffled around a bit by TX)]]  
* [[Confaq42|Con FAQ 4.2 by Jonah Bishop (note: some sections have been shuffled around a bit by TX)]]  
** [http://rtcm.thecomitatus.com/knowledge-base/downloads-rtcm/duke3d-faqs-con/confaq42.zip Download stand-alone zipped version (original version from 1999 in .txt format)]


Another good source of basic information is the following FAQ:
Other good sources of basic information are the following FAQs:
* [http://rtcm.thecomitatus.com/knowledge-base/downloads-rtcm/duke3d-faqs-con/H_CONFAQ.zip Con Editing Information v1.3 by James Ferry]
* [http://dukertcm.com/knowledge-base/downloads-rtcm/duke3d-faqs-con/H_CONFAQ.zip Con Editing Information v1.3 by James Ferry]
* [http://dukertcm.com/knowledge-base/downloads-rtcm/duke3d-faqs-con/conedit.zip Con Editing FAQ v2.0 by Joris B. Weimar]


==Gamevars==
==CON Mechanics==


===Overview===
===Gamevars===
[[Gamevar|Gamevars]] were introduced in [[WW2GI]] and remain the most important aspect of the new commandset.  Gamevars work like variables in programming languages, allowing the user to store, manipulate, and compare data.  Prior to gamevars, the only usable alternatives were manipulations of inventory item counters and move commands.


===Types===
====Overview====
There are three basic types of gamevar, each type storing a signed 32-bit fixed-point integer.  The three basic types are as follows:
 
[[Gamevar]]s are the heart and soul of modern EDuke32 CON scripting.  Gamevars are signed 32-bit fixed-point integer variables, a ubiquitous staple of programming languages. Gamevars are used to store, manipulate, and compare various numerical informationPrior to gamevars' introduction in [[WWII GI]], the only usable alternatives were manipulations of inventory item counters and move commands.
 
====Scope====
There are three scopes of gamevars.
* '''Global variable''': If a variable is declared as global, there is only one copy of it which may be accessed by any actor in the game.
* '''Global variable''': If a variable is declared as global, there is only one copy of it which may be accessed by any actor in the game.


For example:
For example:


   gamevar shots_fired 0 0 // gamevar declarations must not occur inside states, events, or actor code
   gamevar shots_fired 0 0 // gamevar declarations occur outside of states, events, and actor code


This line tells the game that there is a variable named "shots_fired" which is initialized to 0 ( the first 0 ) and is global ( the second 0 ).
This line tells the game to create a variable named "shots_fired" which is initialized to 0.  But wait -- what's the second 0 for?  If you're asking yourself this question, go back and read the actual page on [[gamevar]]s.  Now that you know that gamevars have flags which must be declared along with the var's name and value, let's continue.
You could then find the places in the GAME.CON code where the command "shoot SHOTSPARK1" occurs, and in each instance add the line "addvar shots_fired 1", so that in each case it looks like this:
 
So, we've created a gamevar.  Now what?  For this example, we'll find the places in the GAME.CON code where the command "shoot SHOTSPARK1" occurs, and in each instance add the line "addvar shots_fired 1", so that in each case it looks like this:


   shoot SHOTSPARK1
   shoot SHOTSPARK1
   addvar shots_fired 1 // increments the global var
   addvar shots_fired 1 // increments the counter


This will make the shots_fired var increment each time an enemy fires a SHOTSPARK1 projectile.
This will make the shots_fired var increment each time an enemy fires a SHOTSPARK1 projectile.
Line 36: Line 43:
* '''Per-player variable''': If a variable is declared as per-player, there is one copy of it for each player in the game, meaning that the variable is set independently for each player.  If a player performs an action that triggers a per-player variable change within an event or the APLAYER actor code, it will only change for the player that initiated that action.  If an actor changes a per-player variable, it will change for the closest player to the actor what changed it.  You can use the [[setplayervar]] command to set a per-player var for a particular player who is not the closest one, but this is only necessary in multiplayer games (and even then it is seldom necessary).  In single player games, there is little practical difference between a global gamevar and a per-player variable.
* '''Per-player variable''': If a variable is declared as per-player, there is one copy of it for each player in the game, meaning that the variable is set independently for each player.  If a player performs an action that triggers a per-player variable change within an event or the APLAYER actor code, it will only change for the player that initiated that action.  If an actor changes a per-player variable, it will change for the closest player to the actor what changed it.  You can use the [[setplayervar]] command to set a per-player var for a particular player who is not the closest one, but this is only necessary in multiplayer games (and even then it is seldom necessary).  In single player games, there is little practical difference between a global gamevar and a per-player variable.


* '''Per-actor variable''': If a variable is declared as per-actor, then there is separate copy of it for each sprite in the game.  That means that every monster, projectile, player, decorative sprite -- pretty much anything in the game that isn't a wall or sector -- has its own copy of the variable.  This is useful if you want to have actors other than the player store information for more than one tick.  For example, suppose you want each monster to keep track of how many bullets it has fired:
* '''Per-actor variable''': If a variable is declared as per-actor, then there is separate copy of it for each sprite in the game.  That means that every monster, projectile, player, decorative sprite -- pretty much anything in the game that isn't a wall or sector -- has its own copy of the variable.  This is useful if you want to have actors other than the player store information for more than one [[tic]].  For example, suppose you want each monster to keep track of how many bullets it has fired:


   gamevar shots_fired 0 2
   gamevar shots_fired 0 2
Line 43: Line 50:
With the gamevar declared, you could then add code to the monsters to increment the var when firing shots (just like in the example above for global gamevars).  Since the gamevar is per-actor, the same code would now increment the variable separately for each monster.   
With the gamevar declared, you could then add code to the monsters to increment the var when firing shots (just like in the example above for global gamevars).  Since the gamevar is per-actor, the same code would now increment the variable separately for each monster.   


There are also some [[pre-defined gamevars]] and special [[constantly updated gamevars]] which should only be used in certain ways.
'''There are also some [[pre-defined gamevars]] and special [[constantly updated gamevars]] which should only be used in certain ways.'''


===Use and manipulation===
====Use and manipulation====


Custom variables, or "gamevars" are declared with the [[gamevar]] command.
Custom variables, or "gamevars" are declared with the [[gamevar]] command.


The syntax for defining a gamevar is:
The syntax for defining a gamevar is:
::gamevar <varname> <value> <flags>
:gamevar <varname> <value> <flags>


You may use upper or lowercase letters for <varname>.  EDuke32 is case-sensitive, so a var named "EXAMPLE1" is different from "example1."  Variable names should not begin with a digit.  
You may use upper or lowercase letters for <varname>.  EDuke32 is case-sensitive, so a var named "EXAMPLE1" is different from "example1."  Variable names should not begin with a digit.  
Line 58: Line 65:
The <flags> may be set to either 0 (global), 1 (per-player), or 2 (per-actor).
The <flags> may be set to either 0 (global), 1 (per-player), or 2 (per-actor).


Manipulation of gamevars is accomplished by using a variety of commands that range in functionality from simple mathematical functions to grabbing internal values for manipulation in the CONs.  Here are a few of the more common variable manipulation primitives:
*[[ifvar conditions]]
*[[gamevar operators]]


*[[setvar]], [[setvarvar]]
===Members of game structures===
*[[addvar]], [[addvarvar]]
*[[subvar]], [[subvarvar]]
*[[mulvar]], [[mulvarvar]]
*[[divvar]], [[divvarvar]]
*[[randvar]], [[randvarvar]]


And some of the more rarely used ones:
*[[structure access]]
*[[full list of structure members]]
EDuke32, like prior EDuke iterations, gives you access to game structures--basic elements of the game such as the player, sprites, walls, and sectors--and their predefined property variables, or members.


*[[andvar]], [[andvarvar]]
The most common of these structures is that of the veritable '''actor'''. Objects in the game world, such as monsters, missiles, bits of debris, and so on, are referred to as "sprites" (because they are typically rendered as 2D pictures).  Sprites that execute code are usually referred to as "actors". Every sprite has its own data structure containing all of the member elements in the list linked to above.  Various original CON commands, such as "cstat", "spritepal", "sizeat", etc. work by changing values in the sprite structure of the actor executing the command.  These structure members can be modified directly in an EDuke32 script by using the [[setactor]] command.  Commands like setactor enable us to modify every member of the structure, even those that could not be accessed with the original scripting system. See [[getactor]] and [[setactor]] to see how these commands are used.
*[[sin]], [[cos]]
*[[orvar]], [[orvarvar]]
*[[shiftvarl]], [[shiftvarr]]
*[[xorvar]], [[xorvarvar]]


To test the values of gamevars, the following commands are used:
The members of the '''player''' structure are controlled using the [[getplayer]] and [[setplayer]] commands.  The player structure members deal with properties unique to the player which are not shared with other actors.  These include variables associated to the players weapons, inventory items, special animations, jumping counters, etc.  See the complete list of [[members of the player structure]] for details.  [[Make Nightvision light rooms|This example]] shows how to get a member of the player structure into a variable, check its value, then alter a member of the player structure accordingly.


*[[ifvarand]]
The '''tsprite''' structure is an interesting one. It operates on sprites within the context of just the renderer. This means that you can alter the properties of what is displayed without altering the sprite itself. The possibilities are endless with this functionality. It is also useful for intercepting and redirecting hardcoded behavior.
*[[ifvarvarand]]
*[[ifvare]]
*[[ifvarg]]
*[[ifvarl]]
*[[ifvarn]]
*[[ifvarvare]]
*[[ifvarvarg]]
*[[ifvarvarl]]
*[[ifvarvarn]]


A complete list of primitives is available [[Full_command_list|here]].  The importance of manipulating gamevars will become clear when we get to talking about getting and setting the members of the various structures in the game.  All commands which accept read-only gamevars will also accept constants in place of them.
The '''input''' structure holds data about what game-defined control functions are being operated.


==Members of game structures==
Check out the pages linked above for a full index of the various structures available in scripting.


===Overview===
===Events===
EDuke32, like prior EDuke iterations, gives you access to game structures--basic elements of the game such as the player, sprites, walls, and sectors--and their predefined property variables, or members.  There are many different structures which can be accessed, and a list of the common ones can be found [[full list of structure members|here]].


===Player===
*[[EDuke32 event list|events list]]
The [[members of the player structure]] are controlled using the [[getplayer]] and [[setplayer]] commands.  The player structure members deal with properties unique to the player which are not shared with other actors.  These include variables associated to the players weapons, inventory items, special animations, jumping counters, etc.  See the complete list of [[members of the player structure]] for details.  [[Make_Nightvision_light_rooms|This example]] shows how to get a member of the player structure into a variable, check its value, then alter a member of the player structure accordingly.


===Actor / Sprite===
EDuke32 provides both an object-oriented and a procedural interface to the game's internal workings.  As you already know, the object-oriented part of Duke is the actor system -- in contrast to that, we'll be talking about the event-oriented portion in this section.  As the name suggests, an [[event]] is a block of code that is triggered when a certain event in the game happens.  Events in the game are triggered internally whenever a certain point in the code is reached.
[[Members of the sprite and hittype structures]]
====Hittype====
====Spriteext====
===Sector===
[[Members of the sector structure]]
===Wall===
[[Members of the wall structure]]
===Game configuration===
[[Members of the userdef structure]]
===Player input===
[[Members of the input structure]]
==Events==


===Overview===
Events are a key component in the manipulation of the game.  Using events, we can do a variety of things such as intercept keypresses, draw to the screen (more on this later), redefine what a player does when certain actions are executed, et cetera.
EDuke32 provides both an object-oriented and an event-oriented interface to the game's internal workingsAs you already know, the object-oriented part of Duke is the actor system -- in contrast to that, we'll be talking about the event-oriented portion in this section.  As the name suggests, an [[events|event]] is a block of code that is triggered when a certain event in the game happens.  Events in the game are triggered internally whenever a certain point in the code is reached.


Events are a key component in the manipulation of the gameUsing events, we can do a variety of things such as intercept keypresses, draw to the screen (more on this later), redefine what a player does when certain actions are executed, et cetera.
Using an event is very easy. Similar to the [[actor]] or [[state]] primitives, [[onevent]] is the primitive used to start the definition of a block of code to be associated with one of the [[EDuke32 event list|events]]Furthering the similarity, onevent must be terminated by the equivalent of an [[enda]] or an [[ends]], an [[endevent]].


===Event use===
The following example of event code will cause the player to have a full first aid kit every time the inventory is reset (e.g. when the player starts a game):
Using an event is very easy &mdash; similar to the [[actor]] or [[state]] primitives, [[onevent]] is the primitive used to start the definition of a block of code to be associated with one of the [[EDuke32 event list|events]]. Furthering the similarity, onevent must be terminated by the equivalent of an [[enda]] or an [[ends]], an [[endevent]].


====A note on synchronization====
onevent EVENT_RESETINVENTORY
Events introduce the ability for CON code to be run in a manner that isn't synchronized. For example, display events may run a different number of times on each individual system in multiplayer. If you want your mod to function properly in multiplayer, it is very important to note whether an event is synchronized or not.  Typically, you'll just want to avoid doing anything that changes the game state (including [[ifrnd]] and [[randvar]]) in display code.
    setplayer[THISACTOR].firstaid_amount 100
endevent


==Drawing to the screen==
===Drawing to the screen===
The ability to properly draw to the screen is a huge improvement over the hackish abominations involving quotes and unused characters that we were forced to endure when using 1.3d and 1.5.  There are several different drawing commands available, ranging in functionality from drawing graphics to the screen to printing text stored in quotes (see the section on string manipulation for more information).
The ability to properly draw to the screen is a huge improvement over the hackish abominations involving quotes and unused characters that we were forced to endure when using 1.3D and 1.5.  There are several different drawing commands available, ranging in functionality from drawing graphics to the screen to printing text stored in quotes (see the section on string manipulation for more information).


===Drawing commands===
*[[rotatesprite]]
*[[rotatesprite]]
*[[gametext]]
*[[screentext]]
*[[minitext]]
*[[digitalnumber]]
*[[myos]] (deprecated)
*[[myospal]] (deprecated)
*[[myosx]] (deprecated)
*[[myospalx]] (deprecated)


==Custom projectiles==
===Custom projectiles===
===Overview===
EDuke32 adds the ability to create new projectiles in addition to the default types. A few basic types exist (shorthand: melee, hitscan (bullets), and visible projectiles like RPGs) which can be used to create a nearly infinite variety of weapon types and even some entirely new effects, such as new types of debris, gibs, clouds of billowing smoke, et cetera.
Eduke32 adds the ability to create new projectiles in addition to the default types. A few basic types exist (shorthand: melee, bullet, and slow projectile) which can be used to create a nearly infinite variety of weapon types and even new effects (example: additional debris types).


===Defining custom projectiles===
Projectiles are defined via the [[defineprojectile]] command. The syntax for this command is:
Projectiles are defined via the [[defineprojectile]] command. The syntax for this command is:
  defineprojectile <tilenum> <function> <value>
  defineprojectile <tilenum> <function> <value>


Before using this command list of defines should be created:
These values can be modified with the [[getprojectile]], [[getthisprojectile]] and [[setprojectile]], [[setthisprojectile]] commands.
define PROJ_WORKSLIKE 1
 
define PROJ_SPAWNS 2
===String manipulation===
define PROJ_SXREPEAT 3
 
define PROJ_SYREPEAT 4
[[:Category:String manipulation|Strings]] (lines of text) are called "quotes" in Duke Nukem 3D. String manipulation consists of some simple copying and concatenation commands, as well as the ability to print the text to the screen and to the log file.
define PROJ_SOUND 5
 
define PROJ_ISOUND 6
==Scripting Guidelines==
define PROJ_VEL 7
 
define PROJ_EXTRA 8
When scripting, there are guidelines that you should follow in order to keep your code clean, concise, correct, and optimized.
define PROJ_DECAL 9
define PROJ_TRAIL 10
define PROJ_TXREPEAT 11
define PROJ_TYREPEAT 12
define PROJ_TOFFSET 13
define PROJ_TNUM 14
define PROJ_DROP 15
define PROJ_CSTAT 16
define PROJ_CLIPDIST 17
define PROJ_SHADE 18
define PROJ_XREPEAT 19
define PROJ_YREPEAT 20
define PROJ_PAL 21
define PROJ_EXTRA_RAND 22
define PROJ_HITRADIUS 23
define PROJ_VEL_MULT 24
define PROJ_OFFSET 25
define PROJ_BOUNCES 26
define PROJ_BSOUND 27
define PROJ_RANGE 28


These defines are used to fill the <function> block of defineprojectile.
===Design===


===Manipulating custom projectiles mid-game===
*If you are making a mini-mod, consider creating a [[CON mutator]].


==String manipulation==
===Formatting===
===Quote redefinition mid-game===
*Always comment your code! It will help you and others later, and it will also help keep the code straightforward.
redefinequote <quotenumber> <string> // sets the quote number <quotenumber>'s text to <string>
*Indentation is key. Every level of code ("if" and "[[switch]]" statements, loops, and [[:Category:All commands#Block Delimiting|delimited blocks]]) should be indented with four spaces. TAB characters should be avoided.
===Copying and concatenation===
*[[Label]]s, [[gamevar]]s, [[action]]s, [[move]]s, and [[AI]]s are case-sensitive and cannot start with numbers. Do not use the same names as CON commands.
qstrcpy <quotenumber1> <quotenumber2> // sets the text of <quotenumber1> to the text of <quotenumber2>
*<code>if<something> [[nullop]] [[else]] (...)</code> is a common construction used when an if condition lacks a NOT variant.
qstrcat <quotenumber1> <quotenumber2> // adds the text of <quotenumber2> to the text of <quotenumber1>
*Only use '''{''' braces '''}''' for if conditions when you intend to execute more than one command if the condition is true. If you only have one command, do not use braces. Braces tell the CON compiler specifically to branch, which slows down your code a tiny fraction. Another if condition counts as one command.
**The exception is that if you have an [[else]] following multiple non-braced if conditions where it may be ambiguous, you need to add braces to ensure the else is used correctly.
*Because [[event]]s can be defined multiple times in an additive fashion, keep separate instances in your own organization scheme based on the '''function''' of your events. Writing a con file named '''events.con''' is HIGHLY discouraged.


===Dynamic quotes===
===Methodology===
*Not all "temp" variables are created equal. [[Gamevar]]s used only to perform a mathematical calculations and not to store values afterwards should be declared GLOBAL to save memory. Remember, the game VM that runs CON code will always be single-threaded by nature. If you are using a [[gamevar]] to store a value to be preserved between [[tic]]s, simply to save it for use later, or for a construct like a counter, it should be declared PERACTOR.


==References==
==Links==
*[[Full command list]]
*full command list
*[[EDuke32 event list]]
**[[Full command list|alphabetized]]
*[[Full list of structure members]]
**[[:Category:All commands|categorized]]
*[[gamevar]]s
*[[pre-defined gamevars]]
*[[constantly updated gamevars]]
*[[full list of structure members|structure members]]
*[[EDuke32 event list|event list]]
*[[:Category:Screen drawing commands|screen drawing commands]]
*[[defineprojectile|projectile information]]
*[[Tutorials]]


*[[:Category:All commands|Categorized documentation]]
[[Category:Scripting documentation]]

Latest revision as of 11:58, 31 January 2014

EDuke32 Scripting

About This Guide

Intro

This guide will get you started with the basic aspects of EDuke32's commands -- those which set it apart from vanilla DN3D. This guide assumes both that the reader comprehends and understands the basic CON system as was seen in Duke Nukem 3D 1.3D/1.5, and that the reader is familiar with a few basic programming concepts.

CON Basics

If you are not already familiar with the default commands, the authors of this guide recommend the following guide, imported into the EDukeWiki for your convenience:

Other good sources of basic information are the following FAQs:

CON Mechanics

Gamevars

Overview

Gamevars are the heart and soul of modern EDuke32 CON scripting. Gamevars are signed 32-bit fixed-point integer variables, a ubiquitous staple of programming languages. Gamevars are used to store, manipulate, and compare various numerical information. Prior to gamevars' introduction in WWII GI, the only usable alternatives were manipulations of inventory item counters and move commands.

Scope

There are three scopes of gamevars.

  • Global variable: If a variable is declared as global, there is only one copy of it which may be accessed by any actor in the game.

For example:

  gamevar shots_fired 0 0 // gamevar declarations occur outside of states, events, and actor code

This line tells the game to create a variable named "shots_fired" which is initialized to 0. But wait -- what's the second 0 for? If you're asking yourself this question, go back and read the actual page on gamevars. Now that you know that gamevars have flags which must be declared along with the var's name and value, let's continue.

So, we've created a gamevar. Now what? For this example, we'll find the places in the GAME.CON code where the command "shoot SHOTSPARK1" occurs, and in each instance add the line "addvar shots_fired 1", so that in each case it looks like this:

  shoot SHOTSPARK1
  addvar shots_fired 1 // increments the counter

This will make the shots_fired var increment each time an enemy fires a SHOTSPARK1 projectile.

  • Per-player variable: If a variable is declared as per-player, there is one copy of it for each player in the game, meaning that the variable is set independently for each player. If a player performs an action that triggers a per-player variable change within an event or the APLAYER actor code, it will only change for the player that initiated that action. If an actor changes a per-player variable, it will change for the closest player to the actor what changed it. You can use the setplayervar command to set a per-player var for a particular player who is not the closest one, but this is only necessary in multiplayer games (and even then it is seldom necessary). In single player games, there is little practical difference between a global gamevar and a per-player variable.
  • Per-actor variable: If a variable is declared as per-actor, then there is separate copy of it for each sprite in the game. That means that every monster, projectile, player, decorative sprite -- pretty much anything in the game that isn't a wall or sector -- has its own copy of the variable. This is useful if you want to have actors other than the player store information for more than one tic. For example, suppose you want each monster to keep track of how many bullets it has fired:
  gamevar shots_fired 0 2

This line tells the game that there is a variable named "shots_fired" which is initialized to 0 ( the 0 ) and is per-actor( the 2 ). With the gamevar declared, you could then add code to the monsters to increment the var when firing shots (just like in the example above for global gamevars). Since the gamevar is per-actor, the same code would now increment the variable separately for each monster.

There are also some pre-defined gamevars and special constantly updated gamevars which should only be used in certain ways.

Use and manipulation

Custom variables, or "gamevars" are declared with the gamevar command.

The syntax for defining a gamevar is:

gamevar <varname> <value> <flags>

You may use upper or lowercase letters for <varname>. EDuke32 is case-sensitive, so a var named "EXAMPLE1" is different from "example1." Variable names should not begin with a digit.

The <value> of the gamevar may be positive or negative and must be a whole number.

The <flags> may be set to either 0 (global), 1 (per-player), or 2 (per-actor).

Members of game structures

EDuke32, like prior EDuke iterations, gives you access to game structures--basic elements of the game such as the player, sprites, walls, and sectors--and their predefined property variables, or members.

The most common of these structures is that of the veritable actor. Objects in the game world, such as monsters, missiles, bits of debris, and so on, are referred to as "sprites" (because they are typically rendered as 2D pictures). Sprites that execute code are usually referred to as "actors". Every sprite has its own data structure containing all of the member elements in the list linked to above. Various original CON commands, such as "cstat", "spritepal", "sizeat", etc. work by changing values in the sprite structure of the actor executing the command. These structure members can be modified directly in an EDuke32 script by using the setactor command. Commands like setactor enable us to modify every member of the structure, even those that could not be accessed with the original scripting system. See getactor and setactor to see how these commands are used.

The members of the player structure are controlled using the getplayer and setplayer commands. The player structure members deal with properties unique to the player which are not shared with other actors. These include variables associated to the players weapons, inventory items, special animations, jumping counters, etc. See the complete list of members of the player structure for details. This example shows how to get a member of the player structure into a variable, check its value, then alter a member of the player structure accordingly.

The tsprite structure is an interesting one. It operates on sprites within the context of just the renderer. This means that you can alter the properties of what is displayed without altering the sprite itself. The possibilities are endless with this functionality. It is also useful for intercepting and redirecting hardcoded behavior.

The input structure holds data about what game-defined control functions are being operated.

Check out the pages linked above for a full index of the various structures available in scripting.

Events

EDuke32 provides both an object-oriented and a procedural interface to the game's internal workings. As you already know, the object-oriented part of Duke is the actor system -- in contrast to that, we'll be talking about the event-oriented portion in this section. As the name suggests, an event is a block of code that is triggered when a certain event in the game happens. Events in the game are triggered internally whenever a certain point in the code is reached.

Events are a key component in the manipulation of the game. Using events, we can do a variety of things such as intercept keypresses, draw to the screen (more on this later), redefine what a player does when certain actions are executed, et cetera.

Using an event is very easy. Similar to the actor or state primitives, onevent is the primitive used to start the definition of a block of code to be associated with one of the events. Furthering the similarity, onevent must be terminated by the equivalent of an enda or an ends, an endevent.

The following example of event code will cause the player to have a full first aid kit every time the inventory is reset (e.g. when the player starts a game):

onevent EVENT_RESETINVENTORY
    setplayer[THISACTOR].firstaid_amount 100
endevent

Drawing to the screen

The ability to properly draw to the screen is a huge improvement over the hackish abominations involving quotes and unused characters that we were forced to endure when using 1.3D and 1.5. There are several different drawing commands available, ranging in functionality from drawing graphics to the screen to printing text stored in quotes (see the section on string manipulation for more information).

Custom projectiles

EDuke32 adds the ability to create new projectiles in addition to the default types. A few basic types exist (shorthand: melee, hitscan (bullets), and visible projectiles like RPGs) which can be used to create a nearly infinite variety of weapon types and even some entirely new effects, such as new types of debris, gibs, clouds of billowing smoke, et cetera.

Projectiles are defined via the defineprojectile command. The syntax for this command is:

defineprojectile <tilenum> <function> <value>

These values can be modified with the getprojectile, getthisprojectile and setprojectile, setthisprojectile commands.

String manipulation

Strings (lines of text) are called "quotes" in Duke Nukem 3D. String manipulation consists of some simple copying and concatenation commands, as well as the ability to print the text to the screen and to the log file.

Scripting Guidelines

When scripting, there are guidelines that you should follow in order to keep your code clean, concise, correct, and optimized.

Design

  • If you are making a mini-mod, consider creating a CON mutator.

Formatting

  • Always comment your code! It will help you and others later, and it will also help keep the code straightforward.
  • Indentation is key. Every level of code ("if" and "switch" statements, loops, and delimited blocks) should be indented with four spaces. TAB characters should be avoided.
  • Labels, gamevars, actions, moves, and AIs are case-sensitive and cannot start with numbers. Do not use the same names as CON commands.
  • if<something> nullop else (...) is a common construction used when an if condition lacks a NOT variant.
  • Only use { braces } for if conditions when you intend to execute more than one command if the condition is true. If you only have one command, do not use braces. Braces tell the CON compiler specifically to branch, which slows down your code a tiny fraction. Another if condition counts as one command.
    • The exception is that if you have an else following multiple non-braced if conditions where it may be ambiguous, you need to add braces to ensure the else is used correctly.
  • Because events can be defined multiple times in an additive fashion, keep separate instances in your own organization scheme based on the function of your events. Writing a con file named events.con is HIGHLY discouraged.

Methodology

  • Not all "temp" variables are created equal. Gamevars used only to perform a mathematical calculations and not to store values afterwards should be declared GLOBAL to save memory. Remember, the game VM that runs CON code will always be single-threaded by nature. If you are using a gamevar to store a value to be preserved between tics, simply to save it for use later, or for a construct like a counter, it should be declared PERACTOR.