2014-12-30 21:50:46 +03:00
|
|
|
---
|
2018-09-14 17:00:44 +03:00
|
|
|
title: Lua Scripting
|
2014-12-30 21:50:46 +03:00
|
|
|
layout: default
|
2018-07-15 21:36:35 +03:00
|
|
|
root: ../..
|
2018-07-15 17:28:10 +03:00
|
|
|
idx: 1.2
|
|
|
|
description: A basic introduction to Lua, including a guide on global/local scope.
|
2018-07-15 21:13:16 +03:00
|
|
|
redirect_from: /en/chapters/lua.html
|
2014-12-30 21:50:46 +03:00
|
|
|
---
|
|
|
|
|
2015-02-22 13:28:37 +03:00
|
|
|
## Introduction
|
2014-12-30 21:50:46 +03:00
|
|
|
|
|
|
|
In this chapter we will talk about scripting in Lua, the tools required,
|
|
|
|
and go over some techniques which you will probably find useful.
|
|
|
|
|
2018-09-14 17:00:44 +03:00
|
|
|
* [Code Editors](#code-editors)
|
|
|
|
* [Integrated Programming Environments](#integrated-programming-environments)
|
|
|
|
* [Coding in Lua](#coding-in-lua)
|
|
|
|
* [Program Flow](#program-flow)
|
|
|
|
* [Variable Types](#variable-types)
|
|
|
|
* [Arithmetic Operators](#arithmetic-operators)
|
|
|
|
* [Selection](#selection)
|
|
|
|
* [Logical Operators](#logical-operators)
|
|
|
|
* [Programming](#programming)
|
|
|
|
* [Local and Global Scope](#local-and-global-scope)
|
|
|
|
* [Including other Lua Scripts](#including-other-lua-scripts)
|
|
|
|
|
|
|
|
## Code Editors
|
|
|
|
|
|
|
|
A code editor with code highlighting is sufficient for writing scripts in Lua.
|
|
|
|
Code highlighting gives different colours to different words and characters
|
2015-02-22 13:15:37 +03:00
|
|
|
depending on what they mean. This allows you to spot mistakes.
|
2014-12-30 21:50:46 +03:00
|
|
|
|
2018-09-19 14:04:51 +03:00
|
|
|
```lua
|
2014-12-30 21:50:46 +03:00
|
|
|
function ctf.post(team,msg)
|
2017-08-26 21:01:51 +03:00
|
|
|
if not ctf.team(team) then
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
if not ctf.team(team).log then
|
|
|
|
ctf.team(team).log = {}
|
|
|
|
end
|
2014-12-30 21:50:46 +03:00
|
|
|
|
2017-08-26 21:01:51 +03:00
|
|
|
table.insert(ctf.team(team).log,1,msg)
|
|
|
|
ctf.save()
|
2014-12-30 21:50:46 +03:00
|
|
|
|
2017-08-26 21:01:51 +03:00
|
|
|
return true
|
2014-12-30 21:50:46 +03:00
|
|
|
end
|
2018-09-19 14:04:51 +03:00
|
|
|
```
|
2014-12-30 21:50:46 +03:00
|
|
|
|
2018-10-06 18:56:42 +03:00
|
|
|
For example, keywords in the above snippet are highlighted such as if, then, end, and return.
|
2014-12-30 21:50:46 +03:00
|
|
|
table.insert is a function which comes with Lua by default.
|
|
|
|
|
2018-09-14 17:00:44 +03:00
|
|
|
Here is a list of common editors well suited for Lua.
|
2014-12-31 20:23:15 +03:00
|
|
|
Other editors are available, of course.
|
|
|
|
|
2018-09-14 17:00:44 +03:00
|
|
|
* Windows: [Notepad++](http://notepad-plus-plus.org/), [Atom](http://atom.io/), [VS Code](https://code.visualstudio.com/)
|
|
|
|
* Linux: Kate, Gedit, [Atom](http://atom.io/), [VS Code](https://code.visualstudio.com/)
|
|
|
|
* OSX: [Atom](http://atom.io/), [VS Code](https://code.visualstudio.com/)
|
2014-12-31 20:23:15 +03:00
|
|
|
|
2014-12-30 21:50:46 +03:00
|
|
|
### Integrated Programming Environments
|
|
|
|
|
|
|
|
IDEs allow you to debug code like a native application.
|
|
|
|
These are harder to set up than just a text editor.
|
|
|
|
|
|
|
|
One such IDE is Eclipse with the Koneki Lua plugin:
|
|
|
|
|
|
|
|
* Install Eclipse + Koneki.
|
|
|
|
* Create a new Lua project from existing source (specify Minetest's base directory).
|
2018-10-06 18:56:42 +03:00
|
|
|
* Follow instructions from Koneki wiki on how to do "Attach to remote Application" debugging (just a few steps).
|
2014-12-30 21:50:46 +03:00
|
|
|
* It is suggested to add those lines from wiki at beginning of builtin.lua.
|
|
|
|
* Start the debugger (set "Break on first line" in debugger configuration to see if it is working).
|
|
|
|
* Start Minetest.
|
|
|
|
* Enter the game to startup Lua.
|
|
|
|
|
2015-02-22 14:43:58 +03:00
|
|
|
## Coding in Lua
|
|
|
|
|
2018-09-14 17:00:44 +03:00
|
|
|
### Program Flow
|
|
|
|
|
2015-02-22 14:43:58 +03:00
|
|
|
Programs are a series of commands that run one after another.
|
|
|
|
We call these commands "statements."
|
2018-09-14 17:00:44 +03:00
|
|
|
Program flow is how these statements are executed.
|
|
|
|
Different types of flow allow you to skip or jump over sets of commands.
|
|
|
|
There are three main types of flow:
|
2015-02-22 14:43:58 +03:00
|
|
|
|
|
|
|
* Sequence: Just run one statement after another, no skipping.
|
2018-09-14 17:00:44 +03:00
|
|
|
* Selection: Skip over sequences depending on conditions.
|
2015-02-22 14:43:58 +03:00
|
|
|
* Iteration: Repeating, looping. Keep running the same
|
|
|
|
statements until a condition is met.
|
|
|
|
|
|
|
|
So, what do statements in Lua look like?
|
|
|
|
|
2018-09-19 14:04:51 +03:00
|
|
|
```lua
|
2015-02-22 14:43:58 +03:00
|
|
|
local a = 2 -- Set 'a' to 2
|
|
|
|
local b = 2 -- Set 'b' to 2
|
|
|
|
local result = a + b -- Set 'result' to a + b, which is 4
|
|
|
|
a = a + 10
|
|
|
|
print("Sum is "..result)
|
2018-09-19 14:04:51 +03:00
|
|
|
```
|
2015-02-22 14:43:58 +03:00
|
|
|
|
2018-09-14 17:00:44 +03:00
|
|
|
Whoa, what happened there?
|
|
|
|
|
|
|
|
a, b, and result are *variables*. Local variables are declared
|
|
|
|
by using the local keyword, and then given an initial value.
|
|
|
|
Local will be discussed in a bit, as it's part of a very important concept called
|
|
|
|
*scope*.
|
2015-02-22 14:43:58 +03:00
|
|
|
|
2018-09-14 17:00:44 +03:00
|
|
|
The `=` means *assignment*, so `result = a + b` means set "result" to a + b.
|
|
|
|
Variable names can be longer than one character unlike in mathematics, as seen with the "result" variable.
|
2018-10-06 18:56:42 +03:00
|
|
|
It's also worth noting that Lua is *case-sensitive*; A is a different variable to a.
|
2015-02-22 14:43:58 +03:00
|
|
|
|
|
|
|
### Variable Types
|
|
|
|
|
2018-09-14 17:00:44 +03:00
|
|
|
A variable will be only one of the following types and can change type after an
|
|
|
|
assignment.
|
|
|
|
It's good practice to make sure a variable is only ever nil or a single non-nil type.
|
|
|
|
|
2018-09-27 17:27:07 +03:00
|
|
|
| Type | Description | Example |
|
|
|
|
|----------|---------------------------------|----------------|
|
2018-09-14 17:00:44 +03:00
|
|
|
| Nil | Not initialised. The variable is empty, it has no value | `local A`, `D = nil` |
|
2018-09-27 17:27:07 +03:00
|
|
|
| Number | A whole or decimal number. | `local A = 4` |
|
|
|
|
| String | A piece of text | `local D = "one two three"` |
|
|
|
|
| Boolean | True or False | `local is_true = false`, `local E = (1 == 1)` |
|
|
|
|
| Table | Lists | Explained below |
|
2018-09-14 17:00:44 +03:00
|
|
|
| Function | Can run. May require inputs and may return a value | `local result = func(1, 2, 3)` |
|
2015-02-22 14:43:58 +03:00
|
|
|
|
|
|
|
### Arithmetic Operators
|
|
|
|
|
2018-09-14 17:00:44 +03:00
|
|
|
Not an exhaustive list. Doesn't contain every possible operator.
|
|
|
|
|
2015-02-22 14:43:58 +03:00
|
|
|
| Symbol | Purpose | Example |
|
|
|
|
|--------|----------------|---------------------------|
|
|
|
|
| A + B | Addition | 2 + 2 = 4 |
|
|
|
|
| A - B | Subtraction | 2 - 10 = -8 |
|
|
|
|
| A * B | Multiplication | 2 * 2 = 4 |
|
|
|
|
| A / B | Division | 100 / 50 = 2 |
|
|
|
|
| A ^ B | Powers | 2 ^ 2 = 2<sup>2</sup> = 4 |
|
|
|
|
| A .. B | Join strings | "foo" .. "bar" = "foobar" |
|
|
|
|
|
|
|
|
### Selection
|
|
|
|
|
2018-10-07 17:22:56 +03:00
|
|
|
The most basic selection is the if statement. It looks like this:
|
2015-02-22 14:43:58 +03:00
|
|
|
|
2018-09-19 14:04:51 +03:00
|
|
|
```lua
|
2015-02-22 14:43:58 +03:00
|
|
|
local random_number = math.random(1, 100) -- Between 1 and 100.
|
|
|
|
if random_number > 50 then
|
2017-08-26 21:01:51 +03:00
|
|
|
print("Woohoo!")
|
2015-02-22 14:43:58 +03:00
|
|
|
else
|
2017-08-26 21:01:51 +03:00
|
|
|
print("No!")
|
2015-02-22 14:43:58 +03:00
|
|
|
end
|
2018-09-19 14:04:51 +03:00
|
|
|
```
|
2015-02-22 14:43:58 +03:00
|
|
|
|
|
|
|
That example generates a random number between 1 and 100. It then prints
|
2015-04-13 21:58:06 +03:00
|
|
|
"Woohoo!" if that number is bigger than 50, otherwise it prints "No!".
|
2015-02-22 14:43:58 +03:00
|
|
|
What else can you get apart from '>'?
|
|
|
|
|
|
|
|
### Logical Operators
|
|
|
|
|
|
|
|
| Symbol | Purpose | Example |
|
|
|
|
|---------|--------------------------------------|-------------------------------------------------------------|
|
|
|
|
| A == B | Equals | 1 == 1 (true), 1 == 2 (false) |
|
|
|
|
| A ~= B | Doesn't equal | 1 ~= 1 (false), 1 ~= 2 (true) |
|
|
|
|
| A > B | Greater than | 5 > 2 (true), 1 > 2 (false), 1 > 1 (false) |
|
|
|
|
| A < B | Less than | 1 < 3 (true), 3 < 1 (false), 1 < 1 (false) |
|
|
|
|
| A >= B | Greater than or equals | 5 >= 5 (true), 5 >= 3 (true), 5 >= 6 (false) |
|
|
|
|
| A <= B | Less than or equals | 3 <= 6 (true), 3 <= 3 (true) |
|
|
|
|
| A and B | And (both must be correct) | (2 > 1) and (1 == 1) (true), (2 > 3) and (1 == 1) (false) |
|
|
|
|
| A or B | either or. One or both must be true. | (2 > 1) or (1 == 2) (true), (2 > 4) or (1 == 3) (false) |
|
|
|
|
| not A | not true | not (1 == 2) (true), not (1 == 1) (false) |
|
|
|
|
|
|
|
|
That doesn't contain every possible operator, and you can combine operators like this:
|
|
|
|
|
2018-09-19 14:04:51 +03:00
|
|
|
```lua
|
2015-02-22 14:43:58 +03:00
|
|
|
if not A and B then
|
2017-08-26 21:01:51 +03:00
|
|
|
print("Yay!")
|
2015-02-22 14:43:58 +03:00
|
|
|
end
|
2018-09-19 14:04:51 +03:00
|
|
|
```
|
2015-02-22 14:43:58 +03:00
|
|
|
|
|
|
|
Which prints "Yay!" if A is false and B is true.
|
|
|
|
|
2018-10-06 18:56:42 +03:00
|
|
|
Logical and arithmetic operators work exactly the same;
|
|
|
|
they both accept inputs and return a value which can be stored.
|
2015-02-22 14:43:58 +03:00
|
|
|
|
2018-09-19 14:04:51 +03:00
|
|
|
```lua
|
2015-02-22 14:43:58 +03:00
|
|
|
local A = 5
|
|
|
|
local is_equal = (A == 5)
|
|
|
|
if is_equal then
|
2017-08-26 21:01:51 +03:00
|
|
|
print("Is equal!")
|
2015-02-22 14:43:58 +03:00
|
|
|
end
|
2018-09-19 14:04:51 +03:00
|
|
|
```
|
2015-02-22 14:43:58 +03:00
|
|
|
|
|
|
|
## Programming
|
|
|
|
|
2018-10-06 18:56:42 +03:00
|
|
|
Programming is the action of taking a problem, such as sorting a list
|
2015-02-22 14:43:58 +03:00
|
|
|
of items, and then turning it into steps that a computer can understand.
|
|
|
|
|
2015-11-08 18:57:40 +03:00
|
|
|
Teaching you the logical process of programming is beyond the scope of this book;
|
|
|
|
however, the following websites are quite useful in developing this:
|
2015-02-22 14:43:58 +03:00
|
|
|
|
2018-09-14 17:00:44 +03:00
|
|
|
* [Codecademy](http://www.codecademy.com/) is one of the best resources for
|
|
|
|
learning to 'code', it provides an interactive tutorial experience.
|
|
|
|
* [Scratch](https://scratch.mit.edu) is a good resource when starting from
|
|
|
|
absolute basics, learning the problem solving techniques required to program.\\
|
2018-10-06 18:56:42 +03:00
|
|
|
Scratch is **designed to teach children** how to program, and isn't a serious
|
2018-09-14 17:00:44 +03:00
|
|
|
programming language.
|
2015-02-22 14:43:58 +03:00
|
|
|
|
2018-09-14 17:00:44 +03:00
|
|
|
## Local and Global Scope
|
2014-12-30 21:50:46 +03:00
|
|
|
|
2014-12-30 22:24:41 +03:00
|
|
|
Whether a variable is local or global determines where it can be written to or read to.
|
|
|
|
A local variable is only accessible from where it is defined. Here are some examples:
|
|
|
|
|
2018-09-19 14:04:51 +03:00
|
|
|
```lua
|
2014-12-30 22:24:41 +03:00
|
|
|
-- Accessible from within this script file
|
|
|
|
local one = 1
|
|
|
|
|
|
|
|
function myfunc()
|
2017-08-26 21:01:51 +03:00
|
|
|
-- Accessible from within this function
|
|
|
|
local two = one + one
|
2014-12-30 22:24:41 +03:00
|
|
|
|
2017-08-26 21:01:51 +03:00
|
|
|
if two == one then
|
|
|
|
-- Accessible from within this if statement
|
|
|
|
local three = one + two
|
|
|
|
end
|
2014-12-30 22:24:41 +03:00
|
|
|
end
|
2018-09-19 14:04:51 +03:00
|
|
|
```
|
2014-12-30 22:24:41 +03:00
|
|
|
|
|
|
|
Whereas global variables can be accessed from anywhere in the script file, and from any other mod.
|
|
|
|
|
2018-09-19 14:04:51 +03:00
|
|
|
```lua
|
2014-12-30 22:24:41 +03:00
|
|
|
my_global_variable = "blah"
|
|
|
|
|
|
|
|
function one()
|
2017-08-26 21:01:51 +03:00
|
|
|
my_global_variable = "three"
|
2014-12-30 22:24:41 +03:00
|
|
|
end
|
|
|
|
|
2015-02-24 12:53:25 +03:00
|
|
|
print(my_global_variable) -- Output: "blah"
|
|
|
|
one()
|
|
|
|
print(my_global_variable) -- Output: "three"
|
2018-09-19 14:04:51 +03:00
|
|
|
```
|
2014-12-30 22:24:41 +03:00
|
|
|
|
|
|
|
|
|
|
|
### Locals should be used as much as possible
|
|
|
|
|
|
|
|
Lua is global by default (unlike most other programming languages).
|
|
|
|
Local variables must be identified as such.
|
2014-12-30 21:50:46 +03:00
|
|
|
|
2018-09-19 14:04:51 +03:00
|
|
|
```lua
|
2014-12-30 21:50:46 +03:00
|
|
|
function one()
|
2017-08-26 21:01:51 +03:00
|
|
|
foo = "bar"
|
2014-12-30 21:50:46 +03:00
|
|
|
end
|
|
|
|
|
|
|
|
function two()
|
2017-08-26 21:01:51 +03:00
|
|
|
print(dump(foo)) -- Output: "bar"
|
2014-12-30 21:50:46 +03:00
|
|
|
end
|
2015-02-24 12:53:25 +03:00
|
|
|
|
|
|
|
one()
|
|
|
|
two()
|
2018-09-19 14:04:51 +03:00
|
|
|
```
|
2014-12-30 21:50:46 +03:00
|
|
|
|
2015-02-24 12:53:25 +03:00
|
|
|
dump() is a function that can turn any variable into a string so the programmer can
|
|
|
|
see what it is. The foo variable will be printed as "bar", including the quotes
|
|
|
|
which show it is a string.
|
|
|
|
|
2018-09-14 17:00:44 +03:00
|
|
|
This is sloppy coding, and Minetest will in fact warn about this:
|
2014-12-30 22:24:41 +03:00
|
|
|
|
2018-09-24 19:16:00 +03:00
|
|
|
Assignment to undeclared global 'foo' inside function at init.lua:2
|
2014-12-30 22:24:41 +03:00
|
|
|
|
2014-12-30 21:50:46 +03:00
|
|
|
To correct this, use "local":
|
|
|
|
|
2018-09-19 14:04:51 +03:00
|
|
|
```lua
|
2014-12-30 21:50:46 +03:00
|
|
|
function one()
|
2017-08-26 21:01:51 +03:00
|
|
|
local foo = "bar"
|
2014-12-30 21:50:46 +03:00
|
|
|
end
|
|
|
|
|
|
|
|
function two()
|
2017-08-26 21:01:51 +03:00
|
|
|
print(dump(foo)) -- Output: nil
|
2014-12-30 21:50:46 +03:00
|
|
|
end
|
2015-02-24 12:53:25 +03:00
|
|
|
|
|
|
|
one()
|
|
|
|
two()
|
2018-09-19 14:04:51 +03:00
|
|
|
```
|
2014-12-30 21:50:46 +03:00
|
|
|
|
2018-09-14 17:00:44 +03:00
|
|
|
Remember that nil means **not initialised**.
|
|
|
|
The variable hasn't been assigned a value yet,
|
|
|
|
doesn't exist, or has been uninitialised (ie: set to nil).
|
2015-02-24 12:53:25 +03:00
|
|
|
|
2018-09-14 17:00:44 +03:00
|
|
|
The same goes for functions. Functions are variables of a special type, and
|
|
|
|
should be made local, as other mods could have functions of the same name.
|
2014-12-30 21:50:46 +03:00
|
|
|
|
2018-09-19 14:04:51 +03:00
|
|
|
```lua
|
2014-12-30 21:50:46 +03:00
|
|
|
local function foo(bar)
|
2017-08-26 21:01:51 +03:00
|
|
|
return bar * 2
|
2014-12-30 21:50:46 +03:00
|
|
|
end
|
2018-09-19 14:04:51 +03:00
|
|
|
```
|
2014-12-30 21:50:46 +03:00
|
|
|
|
2018-09-14 17:00:44 +03:00
|
|
|
API tables should be used to allow other mods to call the functions, like so:
|
2014-12-30 21:50:46 +03:00
|
|
|
|
2018-09-19 14:04:51 +03:00
|
|
|
```lua
|
2014-12-30 21:50:46 +03:00
|
|
|
mymod = {}
|
|
|
|
|
|
|
|
function mymod.foo(bar)
|
2017-08-26 21:01:51 +03:00
|
|
|
return "foo" .. bar
|
2014-12-30 21:50:46 +03:00
|
|
|
end
|
|
|
|
|
|
|
|
-- In another mod, or script:
|
|
|
|
mymod.foo("foobar")
|
2018-09-19 14:04:51 +03:00
|
|
|
```
|
2014-12-30 21:50:46 +03:00
|
|
|
|
2015-02-22 13:28:37 +03:00
|
|
|
## Including other Lua Scripts
|
2014-12-30 21:50:46 +03:00
|
|
|
|
2018-09-14 17:00:44 +03:00
|
|
|
The recommended way to include other Lua scripts in a mod is to use *dofile*.
|
2014-12-30 21:50:46 +03:00
|
|
|
|
2018-09-19 14:04:51 +03:00
|
|
|
```lua
|
2014-12-30 21:50:46 +03:00
|
|
|
dofile(minetest.get_modpath("modname") .. "/script.lua")
|
2018-09-19 14:04:51 +03:00
|
|
|
```
|
2014-12-30 21:50:46 +03:00
|
|
|
|
|
|
|
"local" variables declared outside of any functions in a script file will be local to that script.
|
2018-09-14 17:00:44 +03:00
|
|
|
A script can return a value, which is useful for sharing private locals:
|
|
|
|
|
2018-09-19 14:04:51 +03:00
|
|
|
```lua
|
2018-09-14 17:00:44 +03:00
|
|
|
-- script.lua
|
|
|
|
return "Hello world!"
|
|
|
|
|
|
|
|
-- init.lua
|
|
|
|
local ret = dofile(minetest.get_modpath("modname") .. "/script.lua")
|
|
|
|
print(ret) -- Hello world!
|
2018-09-19 14:04:51 +03:00
|
|
|
```
|
2014-12-30 21:50:46 +03:00
|
|
|
|
2018-09-14 17:00:44 +03:00
|
|
|
Later chapters will discuss how to split up the code of a mod in a lot of detail.
|
|
|
|
However, the simplistic approach for now is to have different files for different
|
|
|
|
types of things - nodes.lua, crafts.lua, craftitems.lua, etc.
|