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.
- Official Lua book
- Great Lua Guide
- Lua Users Wiki
- Learn Lua in 15 minutes
- Learn Lua in Y minutes
- Lua on Exercism
- Lua style guide
If you prefer learning by example, you should visit:
- LOVR Slack - has a nice Showcase channel
- My experiments - see how we (barely) make it work
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:
Making your own is a good exercise to understand better Lua tables and metatables
Some resources for going deeper on the topic are:
- Lua Guide section on metatables
- Lua Object-Oriented Programming
- Lua User Wiki about OOP
- Lua Book on
_index
metatable
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.