This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
prpl:prpltutorial [2017/11/17 01:31] – Setup paragraph typos, starting note. GoodMorning | prpl:prpltutorial [2025/02/14 14:57] (current) – external edit 127.0.0.1 | ||
---|---|---|---|
Line 5: | Line 5: | ||
This is a full " | This is a full " | ||
- | I will try to show pictures where illustration could be welcomed, but will also just exaplin | + | I will try to show pictures where illustration could be welcomed, but will also just explain |
+ | |||
+ | After each section there is a little homework, that will hopefully help you try something on your own and get the grip on things. I recommend you actually do the homework yourself and not just read the solution, since doing the work yourself will give you more EXP :) | ||
Navigation: | Navigation: | ||
* [[.: | * [[.: | ||
* [[.: | * [[.: | ||
- | * [[.: | + | * [[.: |
+ | * [[.: | ||
+ | * [[.: | ||
+ | * [[.: | ||
+ | * [[.: | ||
- | < | ||
==== Making your first script ==== | ==== Making your first script ==== | ||
Line 47: | Line 52: | ||
=== Setting up the text editor === | === Setting up the text editor === | ||
- | This step is entirely optional, but can make it much easier to write scripts. Essentially, | + | This step is entirely optional, but can make it much easier to write scripts. Essentially, |
You may have seen a different text editor when editing the script, and almost definitely you didn't see the "Hello world!" | You may have seen a different text editor when editing the script, and almost definitely you didn't see the "Hello world!" | ||
- | To set it up like this, first download the Notepad++ editor from [[https:// | + | To set it up like this, first download the Notepad++ editor from [[https:// |
Next, let's set up the syntax highlighter, | Next, let's set up the syntax highlighter, | ||
+ | |||
+ | === Homework === | ||
+ | |||
+ | Change the script to write 2 messages instead of one: "Hello world!" | ||
+ | |||
+ | Solution: | ||
+ | < | ||
+ | <code prpl> | ||
+ | once | ||
+ | "Hello world!" | ||
+ | " | ||
+ | endonce | ||
+ | </ | ||
+ | </ | ||
< | < | ||
Line 70: | Line 89: | ||
* list: a list of several values. We will cover lists much later in this tutorial. | * list: a list of several values. We will cover lists much later in this tutorial. | ||
For example, the "Hello world!" | For example, the "Hello world!" | ||
+ | |||
+ | There is also an informal data type " | ||
Lastly, the //stack// is a place where you can store values. Imagine the stack as a bunch of values literally stacked on top of each other. New values are always added to the top, and commands always take values from the top as well. The stack can hold an arbitrary amount of values (Well, limited by RAM of course, but it could handle milions of values with ease). | Lastly, the //stack// is a place where you can store values. Imagine the stack as a bunch of values literally stacked on top of each other. New values are always added to the top, and commands always take values from the top as well. The stack can hold an arbitrary amount of values (Well, limited by RAM of course, but it could handle milions of values with ease). | ||
Line 98: | Line 119: | ||
- | 8 6 sub 5 mul #prints 10 | + | 8 6 sub 5 mul Trace #prints 10 |
8 | 8 | ||
Line 108: | Line 129: | ||
8 9 5 min max Trace #prints 8 | 8 9 5 min max Trace #prints 8 | ||
- | 8 6 | + | 8 9 |
| | | | | | | | | | | | | | ||
| | | | | | | | | | ||
Line 117: | Line 138: | ||
=== The command notation === | === The command notation === | ||
- | Since there is a lot of commands, an easy way to quickly | + | Since there is a lot of commands, an easy way to quickly |
- | For example, the " | + | For example, the " |
+ | |||
+ | === Homework === | ||
+ | |||
+ | Write a program that will sum numbers from 1 to 5. Do it in a way so that there are no more than 2 numbers on the stack at a time. | ||
+ | |||
+ | Solution: | ||
+ | < | ||
+ | <code prpl> | ||
+ | once | ||
+ | 1 2 add 3 add 4 add 5 add Trace | ||
+ | endonce | ||
+ | </ | ||
+ | </ | ||
- | < | ||
==== First usefull unit - a floodgate ==== | ==== First usefull unit - a floodgate ==== | ||
Line 162: | Line 195: | ||
You can barely see the land being created and destroyed because of the image, but hey, at least it's something. | You can barely see the land being created and destroyed because of the image, but hey, at least it's something. | ||
+ | |||
+ | === Homework === | ||
+ | |||
+ | Change the program to make a horizontal wall instead of vertical. Use '' | ||
+ | |||
+ | Solution: | ||
+ | < | ||
+ | <code prpl> | ||
+ | CurrentCoords swap 2 sub swap 1 SetLand | ||
+ | CurrentCoords swap 1 sub swap 1 SetLand | ||
+ | CurrentCoords 1 SetLand | ||
+ | CurrentCoords swap 1 add swap 1 SetLand | ||
+ | CurrentCoords swap 2 add swap 1 SetLand | ||
+ | |||
+ | 90 Delay #Sleep for 3 second | ||
+ | |||
+ | #destroy the land the land | ||
+ | CurrentCoords swap 2 sub swap 0 SetLand | ||
+ | CurrentCoords swap 1 sub swap 0 SetLand | ||
+ | CurrentCoords 0 SetLand | ||
+ | CurrentCoords swap 1 add swap 0 SetLand | ||
+ | CurrentCoords swap 2 add swap 0 SetLand | ||
+ | |||
+ | 90 Delay #Sleep for 3 second | ||
+ | |||
+ | # #sets land at (x-2, y) to 1 | ||
+ | CurrentCoords | ||
+ | | | | | | | | ||
+ | | | |y| |x| | ||
+ | | | |x| |y| | ||
+ | === === === | ||
+ | </ | ||
+ | </ | ||
+ | |||
+ | < | ||
+ | ==== Variables ==== | ||
+ | |||
+ | So far we have seen how to store values on the stack and do operations with them. However, sometimes we need to save a value aside and use it later, and leaving it on the stack is not always possible, since in the meantime we want to use other commands that also manipulate with the stack. In such a case, we use variables. | ||
+ | |||
+ | Variables are used for storing values. Unlike the stack, they work like a table of name-value pairs. Use -> | ||
+ | |||
+ | Example: | ||
+ | <code prpl> | ||
+ | 5 ->a | ||
+ | 7 ->b | ||
+ | <-a <-b add Trace #prints 12 | ||
+ | <-a 1 add ->a #increases a by 1 | ||
+ | <-a <-b add Trace #prints 13 | ||
+ | </ | ||
+ | |||
+ | Visual representation of what's going on: | ||
+ | |||
+ | {{: | ||
+ | |||
+ | Again, each variable can have only one value at a time, I just drew them multiple times to indicate the changed as the commands are being executed. Variables don't have to be only numbers, you can story any of the 4 types of value in a variable. Note that you cannot put space into the variable name, so separate the words in the name likeThisInstead. | ||
+ | |||
+ | Unlike the stack that is reset between executions when the script finishes, the variables are persistent enough to stay set across multiple executions (frames). However, all local variables are reset upon compilation. The variables are called local, because they are local to the instance of a script in a core. That means if the same script is attached multiple times to possibly multiple core, it has separate variables every time, with possibly different values. | ||
+ | |||
+ | === Other variable operation === | ||
+ | |||
+ | So far, we have seen how to read from and write to variables. Moreover, we can check if the variable exists at all: -? | ||
+ | |||
+ | <code prpl> | ||
+ | -> | ||
+ | < | ||
+ | -?varname # [ - bool ] Checks if a variable exists | ||
+ | --varname # [ - ] Deletes a variable | ||
+ | </ | ||
+ | |||
+ | All 4 operation also have a reference variant, where they take the name of the variable as an argument from the stack, instead of being written in the command itself. Just put ! instead of the variable name: | ||
+ | |||
+ | <code prpl> | ||
+ | ->! # [ anything string - ] Saves a value into a variable | ||
+ | <-! # [ string - anything ] Reads the value from a variable | ||
+ | -?! # [ string - bool ] Checks if a variable exists | ||
+ | --! # [ string - ] Deletes a variable | ||
+ | </ | ||
+ | |||
+ | They work exactly the same, so '' | ||
+ | |||
+ | === Global variables === | ||
+ | |||
+ | Instead of local variables for each script instance, there is also a single table of global variables. They are shared between all scripts, so when one script changes a global variable, another script will be able to read the change. Global variables are stored completely separately from local variables, so they don't affect each other in any way. Just add * to the end of the command to change it to work with global variables: | ||
+ | |||
+ | <code prpl> | ||
+ | -> | ||
+ | < | ||
+ | -?*varname # [ - bool ] Checks if a variable global exists | ||
+ | --*varname # [ - ] Deletes a global variable | ||
+ | ->!* # [ anything string - ] Saves a value into a global variable | ||
+ | <-!* # [ string - anything ] Reads the value from a global variable | ||
+ | -?!* # [ string - bool ] Checks if a global variable exists | ||
+ | --!* # [ string - ] Deletes a global variable | ||
+ | </ | ||
+ | |||
+ | Unlike local variables, global variables persist (are not deleted) even after a compilation. | ||
+ | |||
+ | === Homework === | ||
+ | |||
+ | Write a variable that will count the number of times the script has been executed. Use the '' | ||
+ | |||
+ | Solution: | ||
+ | < | ||
+ | <code prpl> | ||
+ | once | ||
+ | 0 -> | ||
+ | endonce | ||
+ | < | ||
+ | "Times executed: " < | ||
+ | </ | ||
+ | </ | ||
+ | |||
+ | ==== Flow control commands ==== | ||
+ | |||
+ | So far we have seen commands being executed one by one as we wrote them. But sometimes you need to execute a piece of code only when a certain condition is met, or you need to execute the same code sever times. For that, there are flow control commands that, well, allow you to control the flow of the command execution. | ||
+ | |||
+ | We have already the '' | ||
+ | |||
+ | <code prpl> | ||
+ | once | ||
+ | 0 -> | ||
+ | endonce | ||
+ | < | ||
+ | </ | ||
+ | |||
+ | === if - else - endif === | ||
+ | |||
+ | Next on the list is '' | ||
+ | |||
+ | <code prpl> | ||
+ | #ensure that variable x is at least zero | ||
+ | <-x 0 lt if | ||
+ | 0 ->x | ||
+ | endif | ||
+ | </ | ||
+ | |||
+ | '' | ||
+ | |||
+ | You can also insert '' | ||
+ | <code prpl> | ||
+ | <-energy 50 gte if | ||
+ | "We have enough energy, hooray!" | ||
+ | else | ||
+ | "Not enough energy :(" Trace | ||
+ | endif | ||
+ | </ | ||
+ | |||
+ | '' | ||
+ | * '' | ||
+ | * '' | ||
+ | * '' | ||
+ | * '' | ||
+ | * '' | ||
+ | * '' | ||
+ | * '' | ||
+ | * '' | ||
+ | |||
+ | Moreover, you can combine conditions using logical operators. For example, if you want to check for 2 conditions being true at once, just put them on stack and then call the '' | ||
+ | |||
+ | <code prpl> | ||
+ | <-energy 50 gte < | ||
+ | " | ||
+ | 0 ->energy | ||
+ | 0 -> | ||
+ | endif | ||
+ | </ | ||
+ | |||
+ | Note: we will get to how to accept energy from mines or how to fire actual shots later, but we need to get through the fundamentals first. List of all logical operators: | ||
+ | |||
+ | * '' | ||
+ | * '' | ||
+ | * '' | ||
+ | * '' | ||
+ | |||
+ | As always, check the [[https:// | ||
+ | |||
+ | === while - repeat - endwhile === | ||
+ | |||
+ | Next we take a look at loops - an easy way to execute the code multiple times. Let's start with a simple example: write numbers from 1 to 10. If I asked you to do that on paper, you could easily do it by starting with 1, then 2, then 3, and so on until you reach 10. Let's now write that in PRPL: | ||
+ | |||
+ | <code prpl> | ||
+ | 1 -> | ||
+ | while < | ||
+ | < | ||
+ | < | ||
+ | endwhile | ||
+ | </ | ||
+ | |||
+ | First we initialize the variable counter to 1. Next we use the '' | ||
+ | |||
+ | - Execute commands between '' | ||
+ | - If there is '' | ||
+ | - Execute commands between '' | ||
+ | - Go to 1) | ||
+ | - Terminate the loop execution by continuing after the '' | ||
+ | |||
+ | In other words, the " | ||
+ | |||
+ | === do - loop === | ||
+ | |||
+ | Since counting from one number to another is such a common usage of loops, there are 2 entirely different commands just for that: looping by counting a number from a value to another value. Just specify the upper and lower bound (in that order: first upper, then lower) and you are good to go: | ||
+ | |||
+ | <code prpl> | ||
+ | 11 1 do | ||
+ | I Trace | ||
+ | loop | ||
+ | </ | ||
+ | |||
+ | The '' | ||
+ | |||
+ | === Nesting loops into one another === | ||
+ | |||
+ | You can also put multiple loops into one another, both do-loop and while-repeat-endwhile can be nested. In such case, ' | ||
+ | |||
+ | <code prpl> | ||
+ | 11 1 do | ||
+ | 11 1 do | ||
+ | J " times " I " is " J I mul concat concat concat concat Trace | ||
+ | loop | ||
+ | loop | ||
+ | </ | ||
+ | |||
+ | '' | ||
+ | |||
+ | === Homework === | ||
+ | |||
+ | Add a variable '' | ||
+ | |||
+ | Solution: | ||
+ | < | ||
+ | <code prpl> | ||
+ | once | ||
+ | #easily change directions by commenting / uncommenting | ||
+ | " | ||
+ | #" | ||
+ | endonce | ||
+ | |||
+ | < | ||
+ | |||
+ | 3 -2 do | ||
+ | CurrentCoords I add 1 SetLand #add land | ||
+ | loop | ||
+ | 90 Delay #Sleep for 3 second | ||
+ | |||
+ | 3 -2 do | ||
+ | CurrentCoords I add 0 SetLand #remove land | ||
+ | loop | ||
+ | 90 Delay #Sleep for 3 second | ||
+ | |||
+ | endif | ||
+ | |||
+ | < | ||
+ | |||
+ | 3 -2 do | ||
+ | CurrentCoords swap I add swap 1 SetLand #add land | ||
+ | loop | ||
+ | 90 Delay #Sleep for 3 second | ||
+ | |||
+ | 3 -2 do | ||
+ | CurrentCoords swap I add swap 0 SetLand #remove land | ||
+ | loop | ||
+ | 90 Delay #Sleep for 3 second | ||
+ | |||
+ | endif | ||
+ | </ | ||
+ | </ | ||
+ | |||
+ | < | ||
+ | ==== Functions ==== | ||
+ | |||
+ | Apart from build-in commands like '' | ||
+ | |||
+ | <code prpl> | ||
+ | 5 @getMagicNumber add @addMagic Trace #prints 89 (5+42+42) | ||
+ | |||
+ | : | ||
+ | 42 | ||
+ | |||
+ | :addMagic # [ int - int ] | ||
+ | @getMagicNumber add | ||
+ | </ | ||
+ | |||
+ | Use '': | ||
+ | |||
+ | You can call functions withing other functions, and you can call the same function on multiple places (that' | ||
+ | |||
+ | Functions can read and write to the same local variables as the rest of the script, meaning that if you write into a variable in one functions, you can see the value in any other function in the same script. | ||
+ | |||
+ | === Homework === | ||
+ | |||
+ | Make a function that will take 2 numbers and return the bigger one multiplied by their difference. | ||
+ | |||
+ | Solution: | ||
+ | < | ||
+ | <code prpl> | ||
+ | :myFunction # [ number number - number ] | ||
+ | ->b ->a | ||
+ | | ||
+ | <-a <-b max ->bigger | ||
+ | <-a <-b sub abs -> | ||
+ | | ||
+ | <-bigger < | ||
+ | </ | ||
+ | </ | ||
+ | |||
+ | < | ||
+ | ==== Lists ==== | ||
+ | |||
+ | List is a data type that can contain other values. You can add, read, replace and remove values from a list. The 4 basic commands to get started with lists are: | ||
+ | |||
+ | * '' | ||
+ | * '' | ||
+ | * '' | ||
+ | * '' | ||
+ | |||
+ | See [[prpl: | ||
+ | |||
+ | <code prpl> | ||
+ | #create list | ||
+ | CreateList ->list | ||
+ | <-list " | ||
+ | <-list " | ||
+ | <-list " | ||
+ | |||
+ | #read it's values | ||
+ | <-list GetListCount 0 do | ||
+ | <-list I GetListElement Trace | ||
+ | # < | ||
+ | loop | ||
+ | </ | ||
+ | |||
+ | This will print " | ||
+ | |||
+ | Unlike the 3 other data types, lists are mutable. It means that if you have the same list in 2 variables and you change one, the other one will be changed as well - it is the same list, after all. Think about it like this: let's say that Alice and Bob have the same favourite playground. If you add a seesaw to Alice' | ||
+ | |||
+ | <code prpl> | ||
+ | CreateList -> | ||
+ | < | ||
+ | < | ||
+ | < | ||
+ | |||
+ | # set this playground as both Alice' | ||
+ | < | ||
+ | < | ||
+ | |||
+ | # add a seesaw to Alice' | ||
+ | < | ||
+ | |||
+ | # the seesaw will be in Bob's favourite playground as well | ||
+ | < | ||
+ | </ | ||
+ | |||
+ | I try to explain this in more detail in the [[prpl: | ||
+ | |||
+ | === Homework === | ||
+ | |||
+ | Create a list with some numbers and then find out how many of them are bigger than 10. | ||
+ | |||
+ | Solution: | ||
+ | < | ||
+ | <code prpl> | ||
+ | CreateList ->list | ||
+ | <-list 4 AppendToList | ||
+ | <-list 12 AppendToList | ||
+ | <-list 7 AppendToList | ||
+ | <-list 4 AppendToList | ||
+ | <-list 10 AppendToList | ||
+ | <-list 20 AppendToList | ||
+ | <-list -6 AppendToList | ||
+ | <-list 13 AppendToList | ||
+ | |||
+ | 0 ->count | ||
+ | <-list GetListCount 0 do | ||
+ | < | ||
+ | <-count 1 add ->count | ||
+ | endif | ||
+ | loop | ||
+ | <-count Trace | ||
+ | </ | ||
+ | </ | ||
+ | |||
+ |