LuaWML
Contents
The [lua] tag
This tag is a subtag of the [event] tag. It makes it possible to write actions with the Lua 5.1 language.
The tag supports only the code key, which is a string containing the Lua scripts. Since Lua makes usage of the quotes and the { and } symbols, it is certainly wise to enclose the script between stronger quotes, as they prevent the preprocessor from performing macro expansion and tokenization.
[lua]
    code = << wesnoth.message "Hello World!" >>
[/lua]
The Lua kernel can also be accessed from the command mode:
:lua local u = wesnoth.get_units({ id = "Konrad" })[1]; u.moves = 5
The [args] sub-tag can be used to pass a WML object to the script via its variadic local variable "...".
[lua]
    code = << local t = ...; wesnoth.message(tostring(t.text)) >>
    [args]
        text = _ "Hello world!"
    [/args]
[/lua]
Global environment
All the Lua scripts of a scenario share the same global environment (aka Lua state). For instance, a function defined in an event can be used in all the events that happen after it.
[event]
    name = preload
    first_time_only = no
    [lua]
        code = <<
            function narrator(t)
                -- Behave like the [message] tag.
                wesnoth.fire("message",
                  { speaker = "narrator", message = t.sentence })
            end
        >>
    [/lua]
[/event]
[event]
    name = turn 1
    [lua]
        code = << narrator(...) >>
        [args]
            sentence = _ "Hello world!"
        [/args]
    [/lua]
    [lua]
        code = << narrator(...) >>
        [args]
            sentence = _ "How are you today?"
        [/args]
    [/lua]
[/event]
In the example above, the redundant structure could be hidden behind macros. But it may be better to simply define a new WML tag.
[event]
    name = preload
    first_time_only = no
    [lua]
        code = <<
            -- The function is now local, since its name does not have to be
            -- visible outside this Lua scripts.
            local function handler(t)
                -- Behave like the [message] tag.
                wesnoth.fire("message",
                  { speaker = "narrator", message = t.sentence })
            end
            -- Create a new tag named [narrator].
            wesnoth.register_wml_action("narrator", handler)
        >>
    [/lua]
[/event]
[event]
    name = turn 1
    [narrator]
        sentence = _ "Hello world!"
    [/narrator]
    [narrator]
        sentence = _ "How are you today?"
    [/narrator]
[/event]
The global environment is not preserved over save/load cycles. Therefore, storing values in the global environment is generally a bad idea (unless it has been redirected to WML variables, see helper.set_wml_var_metatable). The only time assigning global variables (including function definitions) makes sense is during a preload event, as this event is always run. Therefore, helper functions defined at that time will be available to all the later scripts.
The global environment initially contains the following modules: basic (no name), string, table, and math. A wesnoth module is also available, it provides access to the C++ engine.
At the start of a script, the variadic local variable ... (three dots) is a proxy table representing WML data. This table is the content of the [args] sub-tag of the [lua] tag, if any.
Examples
The following WML event is taken from Wesnoth' tutorial. It will serve as an example to present how Lua scripts are embedded into Wesnoth. The event is fired whenever a unit from side 1 (that is, the hero controlled by the user) moves to a tile that is not the one set in the WML variable target_hex.
# General catch for them moving to the wrong place.
[event]
    name=moveto
    first_time_only=no
    [allow_undo][/allow_undo]
    [filter]
        side=1
    [/filter]
    [if]
        [variable]
            name=target_hex.is_set
            equals=yes
        [/variable]
        [then]
            [if]
                [variable]
                    name=x1
                    equals=$target_hex.x
                [/variable]
                [variable]
                    name=y1
                    equals=$target_hex.y
                [/variable]
                [then]
                [/then]
                [else]
                    [redraw][/redraw]
                    [message]
                        speaker=narrator
                        message=_ "*Oops!
You moved to the wrong place! After this message, you can press 'u' to undo, then try again." +
                        _ "
*Left click or press spacebar to continue..."
                    [/message]
                [/else]
            [/if]
        [/then]
    [/if]
[/event]
A Lua script that performs the same action is presented below.
[event]
    name=moveto
    first_time_only=no
    [allow_undo][/allow_undo]
    [filter]
        side=1
    [/filter]
    [lua]
        code = <<
            local event_data = wesnoth.current.event_context
            if target_hex.is_set and
               (event_data.x1 ~= target_hex.x or event_data.y1 ~= target_hex.y)
            then
                W.redraw()
                narrator_says(_ "*Oops!\nYou moved to the wrong place! After this message, you can press 'u' to undo, then try again.")
            end
        >>
    [/lua]
[/event]
Here is a more detailed explanation of the Lua code. Its first line
local event_data = wesnoth.current.event_context
puts the event data into the event_data local variable. Since it is a moveto event, the event_data table contains the destination of the unit in the x1 and y1 fields.
The next two lines then test
if target_hex.is_set and (event_data.x1 ~= target_hex.x or event_data.y1 ~= target_hex.y)
whether the variable target_hex matches the event parameters. Since target_hex is not a local variable, it is taken from the global environment (a table implicitly named _G, so it is actually _G.target_hex). The global environment is not persistent, so it cannot be used to store data. In order to make it useful, it was mapped to the storage of WML variables by the following preload event.
[event]
    name=preload
    first_time_only=no
    [lua]
        code = <<
            H = wesnoth.require "lua/helper.lua"
            -- skipping some other initializations
            -- ...
            H.set_wml_var_metatable(_G)
        >>
    [/lua]
[/event]
Without a prelude redirecting _G, the conditional would have been written
if wesnoth.get_variable("target_hex.is_set") and
   (event_data.x1 ~= wesnoth.get_variable("target_hex.x") or event_data.y1 ~= wesnoth.get_variable("target_hex.y")
The body of the conditional then performs the [redraw] action.
W.redraw()
Again, this short syntax is made possible by a line of the prelude that makes W a proxy for performing WML actions.
W = H.set_wml_action_metatable {}
Without this shortcut, the first statement would have been written
wesnoth.fire("redraw")
Finally the script displays a message by
narrator_says(_ "*Oops!\nYou moved to the wrong place! After this message, you can press 'u' to undo, then try again.")
The narrator_says function is defined in the prelude too, since the construct behind it occurs several times in the tutorial. In plain WML, macros would have been used instead. The definition of the function is
function narrator_says(m)
    W.message { speaker="narrator",
                message = m .. _ "\n*Left click or press spacebar to continue..." }
end
The function fires a [message] action and passes a WML object containing the usual two fields to it. The second field is initialized by concatenating the function argument with another string. Both strings are prefixed by the _ symbol to mark them as translatable. (Note that _ is just a unary function, not a keyword.) Again, this is made possible by a specific line of the prelude:
_ = wesnoth.textdomain "wesnoth-tutorial"
A longer translation of the tutorial is available at [1].
Interface to the engine and helper functions
Functionalities of the game engine are available through the functions of the wesnoth global table. Some of these functions return proxy tables. Writes to fields marked "read-only" are ignored. The __cfg fields return plain tables; in particular, writes do not modify the original object, and reads return the values from the time the dump was performed.
Some helper functions are provided by the lua/helper.lua library. They are stored inside a table that is returned when loading the library with wesnoth.require.
helper = wesnoth.require "lua/helper.lua"
WML variables
- wesnoth.get_variable
- wesnoth.set_variable
- helper.set_wml_var_metatable
- helper.get_child
- helper.child_range
- helper.get_variable_array
- helper.get_variable_proxy_array
- helper.set_variable_array
Events and WML actions
- wesnoth.fire
- wesnoth.register_wml_action
- wesnoth.wml_actions Template:DevFeature1.9
- wesnoth.game_events Template:DevFeature1.9
- wesnoth.fire_event
- wesnoth.eval_conditional
- wesnoth.tovconfig Template:DevFeature1.9
- helper.set_wml_action_metatable
- helper.wml_error
- helper.literal Template:DevFeature1.9
- helper.parsed Template:DevFeature1.9
- helper.shallow_literal Template:DevFeature1.9
- helper.shallow_parsed Template:DevFeature1.9
User interface
- wesnoth.message
- wesnoth.clear_messages Template:DevFeature1.9
- wesnoth.textdomain
- wesnoth.delay Template:DevFeature1.9
- wesnoth.float_label
- wesnoth.hilight_hex Template:DevFeature1.9
- wesnoth.select_hex Template:DevFeature1.9
- wesnoth.scroll_to_tile Template:DevFeature1.9
- wesnoth.play_sound Template:DevFeature1.9
- wesnoth.set_music Template:DevFeature1.9
- wesnoth.show_dialog Template:DevFeature1.9
- wesnoth.set_dialog_value Template:DevFeature1.9
- wesnoth.get_dialog_value Template:DevFeature1.9
- wesnoth.set_dialog_callback Template:DevFeature1.9
- wesnoth.set_dialog_canvas Template:DevFeature1.9
- helper.get_user_choice
Map and terrains
- wesnoth.get_map_size
- wesnoth.get_terrain
- wesnoth.set_terrain
- wesnoth.get_terrain_info
- wesnoth.get_selected_tile
- wesnoth.get_locations Template:DevFeature1.9
- wesnoth.match_location Template:DevFeature1.9
- wesnoth.add_tile_overlay Template:DevFeature1.9
- wesnoth.remove_tile_overlay Template:DevFeature1.9
- items.place_image Template:DevFeature1.9
- items.place_halo Template:DevFeature1.9
- items.remove Template:DevFeature1.9
Units
- wesnoth.get_units
- wesnoth.get_unit Template:DevFeature1.9
- wesnoth.match_unit Template:DevFeature1.9
- wesnoth.put_unit
- wesnoth.get_recall_units Template:DevFeature1.9
- wesnoth.put_recall_unit Template:DevFeature1.9
- wesnoth.create_unit
- wesnoth.copy_unit
- wesnoth.extract_unit Template:DevFeature1.9
- wesnoth.add_modification Template:DevFeature1.9
- wesnoth.unit_resistance
- wesnoth.unit_defense
- wesnoth.unit_movement_cost
- wesnoth.unit_ability Template:DevFeature1.9
- wesnoth.get_unit_type_ids
- wesnoth.get_unit_type
- wesnoth.unit_types Template:DevFeature1.9
- wesnoth.simulate_combat
Sides
- wesnoth.get_side
- wesnoth.sides Template:DevFeature1.9
- wesnoth.get_village_owner
- wesnoth.set_village_owner
- wesnoth.is_enemy Template:DevFeature1.9
- helper.all_teams
Pathfinder
- wesnoth.find_path
- wesnoth.find_vacant_tile
- wesnoth.find_reach
- helper.distance_between
- helper.adjacent_tiles Template:DevFeature1.9
Lua files
Location sets
- location_set.create Template:DevFeature1.9
- location_set.of_pairs Template:DevFeature1.9
- location_set.of_wml_var Template:DevFeature1.9
- location_set:empty Template:DevFeature1.9
- location_set:size Template:DevFeature1.9
- location_set:clear Template:DevFeature1.9
- location_set:get Template:DevFeature1.9
- location_set:insert Template:DevFeature1.9
- location_set:remove Template:DevFeature1.9
- location_set:of_pairs Template:DevFeature1.9
- location_set:of_wml_var Template:DevFeature1.9
- location_set:to_pairs Template:DevFeature1.9
- location_set:to_stable_pairs Template:DevFeature1.9
- location_set:to_wml_var Template:DevFeature1.9
- location_set:union Template:DevFeature1.9
- location_set:inter Template:DevFeature1.9
- location_set:iter Template:DevFeature1.9
- location_set:stable_iter Template:DevFeature1.9
- location_set:filter Template:DevFeature1.9
- location_set:union_merge Template:DevFeature1.9
- location_set:inter_merge Template:DevFeature1.9
Miscellaneous
- wesnoth.game_config
- wesnoth.current
- wesnoth.synchronize_choice Template:DevFeature1.9
- wesnoth.get_image_size Template:DevFeature1.9
- helper.set_wml_tag_metatable
- helper.modify_unit
- helper.move_unit_fake
- helper.rand Template:DevFeature1.9
Encoding WML objects into Lua tables
Function wesnoth.fire expects a table representing a WML object as its second argument (if needed). Function wesnoth.set_variable allows to modify whole WML objects, again by passing it a table. Function wesnoth.get_variable transforms a WML object into a table, if its second argument is not set to true. All these tables have the same format.
Scalar fields are transformed into WML attributes. For instance, the following Lua table
{
    a_bool = true,
    an_int = 42,
    a_float = 1.25,
    a_string = "scout",
    a_translation = _ "Hello World!"
}
is equivalent to the content of the following WML object
[dummy]
    a_bool = "yes"
    an_int = "42"
    a_float = "1.25"
    a_string = "scout"
    a_translation = _ "Hello World!"
[/dummy]
WML child objects are not stored as Lua named fields, since several of them can have the same tag. Moreover, their tags can conflict with the attribute keys. So child objects are stored as pairs string + table in the unnamed fields in definition order. This means that for every subtag appearing in the wml code there is an additional table "layer" in the corresponding WML table of the form {[1] = "tag_name", [2] = {}} which is equivalent to {"tag_name", {}}. [1] etc are the unnamed fields (as opposed to wml attributes). The table under [2] in this subtable then holds the wml attributes from inside the wml subtag. So every subtag other than the toplevel tag corresponds to two nested tables each. For instance, the following Lua table
{
    foo = 42,
    { "bar", { v = 1, w = 2 } },
    { "foo", { x = false } },
    { "bar", { y = "foo" } },
    { "foobar", { z = 5, { "barfoo", {} } } }
}
is equivalent to the content of the following WML object
[dummy]
    foo = 42
    [bar]
        v = 1
        w = 2
    [/bar]
    [foo]
        x = no
    [/foo]
    [bar]
        y = foo
    [bar]
    [foobar]
        z = 5
        [barfoo]
        [/barfoo]
    [/foobar]
[/dummy]
Both tables above are also equivalent to this WML table, where all unnamed fields are displayed:
{
    foo = 42,
    [1] = { [1] = "bar", [2] = { v = 1, w = 2 } },
    [2] = { [1] = "foo", [2] = { x = false } },
    [3] = { [1] = "bar", [2] = { y = "foo" } },
    [4] = { [1] = "foobar", [2] = { z = 5, [1] = { [1] = "barfoo", [2] = {} } } }
}
So assuming cfg contains the above WML object, the following accesses are possible:
a_int = cfg.foo        -- "dummy.foo", 42
a_string = cfg[3][2].y -- "dummy.bar[1].y", "foo"
a_table = cfg[4][2]    -- "dummy.foobar", { z = 5, { "barfoo", {} } }
Consider using the helper.get_child and helper.child_range to ease the access to subtags.
Functions registered by wesnoth.register_wml_action receive their data in a userdata object which has the exact same structure as above. It is read-only however. Accessing fields or children performs variable substitution on the fly. Its __parsed and __literal fields provide translations to plain tables (therefore writable). __literal returns the original text of the data (including dollar symbols in attributes and [insert_tag] children), while __parsed performs a variable substitution.
For instance, if you cannot stand any longer the fact that first_time_only is set to yes by default for the [event] tag, you can redefine it. But we have to be careful not to cause variable substitution, since the engine would perform a second variable substitution afterwards.
local old_event_handler
old_event_handler = register_wml_action("event",
    function(cfg)
        -- Get the plain text from the user.
        local new_cfg = cfg.__literal
        -- The expression below is equivalent to cfg.__parsed.first_time_only,
        -- only faster. It is needed, since the first_time_only attribute may
        -- reference variables.
        local first = cfg.first_time_only
        -- Modify the default behavior of first_time_only.
        if first == nil then first = false end
        new_cfg.first_time_only = first
        -- Call the engine handler.
        old_event_handler(new_cfg)
    end
)
Note that, since the object is a userdata and not a table, pairs and ipairs are unfortunately not usable on it. So scripts have to work at a lower level. For instance, the following function returns the first sub-tag with a given name and it works both on WML tables and WML userdata:
function get_child(cfg, name)
    for i = 1, #cfg do
        local v = cfg[i]
        if v[1] == name then return v[2] end
    end
end
Another approach for handling userdata and tables in the same way, would be to convert the former into the latter beforehand:
if getmetatable(cfg) == "wml object" then cfg = cfg.__parsed end
The WML userdata provides two other special fields: __shallow_parsed and __shallow_literal Template:DevFeature1.9. They return a table corresponding to the WML userdata with variable substitution performed on the attributes (or not). [insert_tag] tags have also been parsed, so the number of children is faithful. But contrarily to __parsed and __literal, the process is not recursive: all the children are still WML userdata and variable substitution can still happen for them. These shallow translators are meant as optimized versions of the deep ones, when only the toplevel attributes need to be writable.
Skeleton of a preload event
The following event is a skeleton for a prelude enabling Lua in your WML events. It creates a table H containing the functions from helper.lua and a table W that serves as a proxy for firing WML actions. It also sets up the global environment so that any access to an undefined global variable is redirected to the persistent WML storage.
[event]
    name=preload
    first_time_only=no
    [lua]
        code = <<
            H = wesnoth.require "lua/helper.lua"
            W = H.set_wml_action_metatable {}
            _ = wesnoth.textdomain "my-campaign"
            -- Define your global constants here.
            -- ...
            H.set_wml_var_metatable(_G)
            -- Define your global functions here.
            -- ...
        >>
    [/lua]
[/event]
It may be worth putting the whole Lua script above inside a separate file and having the preload event load it:
[event]
    name=preload
    first_time_only=no
    [lua]
        code = << wesnoth.dofile "~add-ons/Whatever/file.lua" >>
    [/lua]
[/event]
Remarks
The math.random function is not safe for replays and multiplayer games, since the random values will be different each time and on all the clients. Instead, the Lua code should rely on the [set_variable] tag to synchronize random values.
function random(min, max)
  if not max then min, max = 1, min end
  wesnoth.fire("set_variable", { name = "LUA_random", rand = string.format("%d..%d", min, max) })
  local res = wesnoth.get_variable "LUA_random"
  wesnoth.set_variable "LUA_random"
  return res
end