Skip to main content Link Menu Expand (external link) Document Search Copy Copied

Lua

The LOVR engine uses the Lua scripting language as the main method to build programs. Almost everything is done in Lua, from importing models to defining objects, properties, functions, players and so on. Major exclusions are shaders, done in OpenGL and lower level functions, possible via shared C libraries.

Lua is a high level, dynamic language. If you already have some basic experience with programming, Lua should be easy to pick up, with automatic type inference, no complex types, small number of reserved keywords and high flexibility. You’ll likely have to build your own tools and methods to do a lot of non-trivial tasks, and you can also rely on a lot of available libraries built in Lua.

The Lua version used in the engine is LuaJIT, so for more advanced thing and some more complex libraries, keep an eye out.

To get started


These are some good resources, both for complete novices and more experienced programmers.

If you prefer learning by example, you should visit:

Libraries and Object-Oriented Programming


LOVR doesn’t have to run on a single main file; while this is the core of the program, it can also call other files, via Lua’s require.

As the LOVR Docs show, you can call other files via this function and import their functionality, allowing you to create libraries and use them between projects and sharing them with others.

-- library.lua is sitting next to our main.lua here
local lib = require('library')

function lovr.load()
  lib.doStuff()
end

The LOVR docs have a list for some link with LOVR or developed with LOVR in mind directly.

One of the main features this allows us to do is extend the core libraries of Lua, which are very small compared to languages like Python. This includes

Object-Oriented Programming


No, Lua does not natively use OOP. It can be implemented easily using Lua’s tables There are two ways:

  1. use a library like 30log or classic
  2. make your own Objects and Classes

Making your own is a good exercise to understand better Lua tables and metatables

Some resources for going deeper on the topic are:

Or you can use these examples from My Notes:

Objects

Tables are the fundamental associative arrays of Lua. These can be used to build dictionaries, lists, vectors, arrays and objects.

Simply define a function as

obj={}
function obj.method(self)
    -- do stuff
end

And you have a valid object.

Classes

Here we need the metatables. These are advanced tables that can define more complex properties such as interaction through operators like + or ==, but also things like length, calling the table like a function, what to do when a certain value is called and what to do when a new value is defined.

The core question is defining standard keys and values for standard functions, those of the class, so each instance can call the same functions but with their values. This function is covered by the __index metatable, which defines where else to go to check for unknown indices.

This is achieved by defining the functions for the class on an empty or minimal table, then at initialization of the instance, we create a new table with the needed values, associate the instance with the class via the __index metatable and returning the instance. Each instance will access the methods of the class unless overridden and any usage of self will be in reference to the instance, not the class

Class = {}  -- empty table used by class

function Class.new(self, val1, val_2) --define generting method
    local instance = {
        key_1 = val_1,
        key_2 = val_2
    } -- crate table for instance and fill with datas
    setmetatable(instance, { __index = Class }) --associate the instance with the class object and inherit the methods and properties
    return instance -- return the instance, not the class
end

Plugins


Beyond Lua libraries, we can extend LOVR’s capabilities with Plugins.

These are similarly accessed using require, but under the hood they call on compiled libraries, such as .dll or .so. The compiled libraries can be accessed using the predefined Lua method for calling libraries, detailed more in the Official LOVR Docs or in the Lua Wiki This method requires the library to be built explicitly for Lua integration.

Alternatively, you can use the FFI Lua system, a common set of functions that allow functions from one language to call functions from a library, without requiring editing the code of the library. This method is pretty complex and the LOVR documentation does not really acknowledge it, but some users have successfully used it to implement libraries not built for Lua.

To get a grasp on FFI, you can read this SO post

Making plugin work is much easier on PCVR, where finding libraries, compiling and accessing them is easy and well documented. LOVR also allows the require call to access libraries from any folder. This is not the case for Mobile VR, where you need to find and compile valid libraries for the Android OS, and these need to be added to the lib/arm64-v8a of the APK. This can in theory be achieved using software like APK Tool and then Resigning the APK, but it’s a much less straightforward process and requires much more experience with Android OS and the APK system. It also means that plugins are not shared via shared projects, but require you to share the entire APK or library.

Plugins are extremely powerful, allowing access to things like network capabilities, high performance C functions or even emulate the PSX system

If you’re interested in network access, come see what we have on Telegram.