Avendar:Prog Section 3

Control Structures

"We offered them ORDER!" 
   -- Khan, Star Trek: Episode 24: Space Seed
As we have seen, control structures allow us to alter the normally linear order
in which program commands are executed.

If Checks

The most common type of control structure is the if check, which tests to see if
a certain aspect of the game world is true or false. An if check alters the flow of
the program so that if it is true, a block of commands following the check will be
executed. If it is false, these commands will not be executed, and execution of the
program will skip past them. Every if statement appears before a block of commands,
and must always have a matching endif statement, to mark the end of this block of
commands. The syntax of an if check is generally:
if check([<argument1>]) <logical operator> <argument2>
[block of commands]
endif
The check is the actual if check. The check is really a function, which tests an
aspect of something in the game world. This can be something as simple as whether
or not a target is a pc, or as complex as a person’s percentage in a particular
skill or spell. The argument 1 is the pc, mobile, object, number, or other entity
which is being checked. Generally, the target is a variable. It can, however, be a
number, or the literal name of the pc or npc in question. A check can also have a
blank argument, in the case of checks which do not require a target. The logical
operator expresses a logical relationship between the argument1 and the argument2
of the if check. The operator can either be numerical (if either side resolves to
a number), or a string operator (if either side resolves to a string.)
The following are numerical operators:
   == equality 
   != negation of equality (i.e., "is not equal to") 
   < less than 
   <= less than or equal to 
   > greater than 
   >= greater than or equal to 
   & Bitwise and 
   | Bitwise or 
The following are string operators:
   == Exact equality 
   != Strings do not match 
   / String contains the given text 
   !/ String does not contain the given text 
As can be seen, not every logical operator works for every if check. In
general, if the result of a check’s evaluation is not numerical, only the first
two operators can be used (as these are the only ones which do not require a
number to appear to either side.)
The argument2 is an argument which is compared to the result of the check. Generally
speaking, this can be a variable that represents a pc, mobile, object, a number, or
some other form of data. So, putting this all together, let’s consider a few examples
of some if checks.
   greet_prog 100
   if ispc($n)
   say Hello, $N. 
   endif
   say $N just entered the room.
This is a simple program, that will greet a pc if they enter the room, then let everyone
know the name of anything that enters a room, even if it’s not a pc. So, in this prog,
we can see that:
- The check is ‘ispc’
- The argument1 is ‘$n’
- There is no logical operator.
- There is no argument2.
- The if block is one line: "say Hello, $N".
- The endif comes right after that.
Remember, the only required thing for an if check is the if, the check, and the endif.
Now, let’s suppose Jolinn, and his (Charmed) Ethron Love NPC were to enter a room, and
this prog is on a hen. (Yes, a hen.) Someone standing in the room would see:
The hen says, ‘Hello, Jolinn’
The hen says, ‘Jolinn just entered the room.’
The hen says, ‘The hawt ethron love slave has just entered the room.’
As we can see, since the ethron love slave is not a pc, the commands in the if block
are not executed.
Now, what have if checks really allowed us to do? Basically, we can now say, "If a
condition is TRUE, then execute a specified list of commands." But, what if we want to
say, "If a condition is TRUE, then execute a specified set of commands. However, if it’s
FALSE, execute a different set of commands"? To help us with this scenario, we have the
else command. This command allows you to specify alternate set of commands in a body of
an if check, to be executed in the event that the check does not return a true result.
To see how else works, let’s return to our example.
   greet_prog 100
   if ispc($n)
   say Hello, $N. 
   else
   say I would say hello, $N, but you’re just a mobile.
   endif
   say $N just entered the room.
In this case, if Jolinn and his ethron entered the room, they would see:
The hen says, ‘Hello, Jolinn.’
The hen says, ‘Jolinn just entered the room.’
The hen says, ‘I would say hello, a hawt ethron love slave, but you’re just a mobile.’
The hen says, ‘A hawt ethron love slave just entered the room.’
As we can see, the commands after the else statement are executed for the ethron, since
the ethron is NOT a pc, and therefore the else block has its triggered.
Now, suppose that we have an effect which we want to occur under multiple conditions. An
example might be: A mob who will speak either to an aelin OR a nefortu when given a sum
of a gold. Now, we can easily do this by having two (or more) if check blocks in a row,
but this requires us to unnecessarily duplicate commands. To this end, we introduce the
or statement, which allows us to have an if-block that executes on any of multiple conditions
being satisfied.
Consider the following example:
   speech_prog p Hello
   if race($n) == aelin
   or race($n) == nefortu
   say Why hello to you too, $N.
   else
   say Hmph. Whatever, wingless person! You are dead to me!
   endif
So, if:
An aelin said "Hello", they would see: ‘Why hello to you too, aelin.’
A nefortu said "Hello", they would see ‘Why hello to you too, nefortu.’
An ethron said "Hello", they would see ‘Hmph. Whatever, wingless person! You are dead to me!’
So, the or check allows multiple possible conditions to trigger the same if block.
Now, what if we want to execute a block of commands when two (or more) simultaneous
conditions hold true at once. For example, suppose we want a mob that attacks a target
if it both a pc AND an ethron. Or, suppose we have a sword that requires you be good
AND lawful to wield. We can even imagine situations where we might want three, four,
five, or even an arbitrary number of conditions to hold true.
To support this functionality, progs support the "nesting" of if check. "Nesting" means
that you can put one if check (including its body, else statement, endif) inside of another
if check. So, in essence, we would have:
   if check1(var)
       if check2(var)
       command
       command
       command
       else (optional)
       command
       endif
   else (optional)
   endif
Essentially, the if block of check2 is a subset of the if block of check1. The commands
in check2’s block are only executed if both check1 and check2 are true. To illustrate how
this works, let’s consider a concrete example. Suppose that this prog appeared on a hen:
   prog add speech_prog p Hello
   if ispc($n)
     if race($n) == ethron
     say Why hello there, $N, you beautiful green thing, you!
     else
     say Why hello there, $N.
     endif
   else
   say You’re nothing more than a drone, $N!
   endif
Let’s suppose we have Jolinn, who is a human, Aeolis, who is an ethron, and a hot ethron
love slave, who is an npc who is an ethron.
If...
Jolinn says, "Hello", he would see:
The hen says, "Why hello there, Jolinn."
Aeolis says, "Hello", he would see:
The hen says, "Why hello there, Aeolis, you beautiful green thing, you!"
A hawt ethron love slave says, "Hello", she would see:
The hen says, "You’re nothing more than a drone, hawt ethron love slave!"
As we can see, "nesting" results in a block of commands which is executed only if BOTH
checks are true. This is perfectly equivalent to the logical AND, "If A AND B are true,
then do C".


Loops

Often times, we may wish to repeat a command a certain number of times in a program.
Now, clearly, we could do this just by including that number of copies of the command,
but this solution leaves certain things to be desired. Firstly, it is inefficient
(what if we want to execute a command a hundred times?). Secondly, and most importantly,
it does not allow us to execute a command a variable number of times.
Loop Syntax:
   Loop <low number> to <high number>
   command_1
   command_2
   .
   .
   command_n
   endloop
This will execute the commands in the loop (1-n), a number of times equal to high
number - low number. In general, low number and high number are both numbers, or variables
whose values resolve to being numbers. Just like if checks, you can "nest" loops, by placing
one loop block inside of another one.
In a loop, you can reference the "iteration", or number the loop is on, by the $v variable,
adjoined to a number. The number represents the "depth" of the loop you are currently in.
So, by default, $v1 references the outermost loop’s current iteration. $v2 references the
next innermost loop iteration, and so on, until $vn, where n is the innermost loop iteration.
This can be a little confusing, so let’s give a concrete example. Suppose that our favorite
counting hen has a prog that reads:
   prog add speech_prog p Loopy
   loop 1 to 3
     say The outermost loop is on number $v1.
     loop 1 to 2
       say The second loop check is on number $v2.
       loop 7 to 8
         say The innermost loop check is on number $v3.
       endloop
     endloop
   endloop
   say The loop now, is ended.
So, if you were to say "Loopy" to the hen, you would see the following:
   The hen says, ‘The outermost loop is on number 1.’ 
   The hen says, ‘The second loop check is on number 1.’ 
   The hen says, ‘The innermost loop check is on number 7.’ 
   The hen says, ‘The innermost loop check is on number 8.’ 
   The hen says, ‘The second loop check is on number 2.’ 
   The hen says, ‘The innermost loop check is on number 7.’
   The hen says, ‘The innermost loop check is on number 8.’
   The hen says, ‘The outermost loop is on number 2.’
   The hen says, ‘The second loop check is on number 1.’
   The hen says, ‘The innermost loop check is on number 7.’
   The hen says, ‘The innermost loop check is on number 8.’
   The hen says, ‘The second loop check is on number 2.’
   The hen says, ‘The innermost loop check is on number 7.’
   The hen says, ‘The innermost loop check is on number 8.’
   The hen says, ‘The outermost loop is on number 3.’
   The hen says, ‘The second loop check is on number 1.’
   The hen says, ‘The innermost loop check is on number 7.’
   The hen says, ‘The innermost loop check is on number 8.’
   The hen says, ‘The second loop check is on number 2.’
   The hen says, ‘The innermost loop check is on number 7.’
   The hen says, ‘The innermost loop check is on number 8.’
   The hen says, ‘The loop now, is ended.’
As we can see, loops are interpreted from the "outside in". Begin with the outermost
loop, then execute its contents. If those contents contain a loop statement, execute
that loop. Continue until the innermost loop is reached.
And, as we can see, $v1, $v2, and $v3 are variables which refer to iteration of the
corresponding loop.


Break

Finally, it is sometimes necessary to terminate execution of a prog entirely. In this
instance, the break command accomplishes this effect. The moment this command is executed,
the prog will stop. This is particularly useful for terminating out of certain branches
of if checks.