Helpful stuff for making games with lua, especially with löve.
Get your projects off the ground faster! batteries
fills out lua's sparse standard library, and provides implementations of many common algorithms and data structures useful for games.
General purpose and special case, extensively documented in-line, and around a hundred kilobytes uncompressed - including the license and this readme - so you get quite a lot per byte! Of course, feel free to trim it down for your use case as required (see below).
Examples are in another repo to avoid cluttering the repo history and your project/filesystem when used as a submodule.
They have short, straightforward usage examples of much of the provided functionality.
Documentation is provided as comments alongside the code, because that is the most resilient place for it. Even auto-generated docs often end up out of date and the annotations bloat the code.
Use find-in-all from your editor, or just browse through the code. The module overview below is the only non-code documentation - it is a jumping off point, not a reference.
batteries
works straight out of the repo with no separate build step. The license file required for use is included.
- Put the files in their own directory, somewhere your project can access them.
require
the basebatteries
directory - the one withinit.lua
in it.- Don't forget to use dot syntax on the path!
- With a normal
require
setup (ie stock LÖVE or lua),init.lua
will pull in all the submodules. - Batteries uses the (very common) init.lua convention. If your installation doesn't already have init.lua support (eg plain 5.1 on windows), add
package.path = package.path .. ";./?/init.lua"
before the require line. You can also modify yourLUA_PATH
environment variable.
- (optionally)
export
everything to the global environment.
--everything as globals
require("path.to.batteries"):export()
-- OR --
--self-contained
local batteries = require("path.to.batteries")
See below for a discussion of the pros and cons of export
.
batteries
aims to be approachable to almost everyone, but I do expect you to get your hands dirty. I'm very open to collaboration, and happy to talk through issues or shortcomings in good faith.
Pull requests are welcome for anything - positive changes will be merged optimistically, and I'm happy to work with you to get anything sensible ready for inclusion.
If you have something "big" to contribute please get in touch before starting work so we can make sure it fits. I'm quite open minded!
If you've had a good look for the answer but something remains unclear, raise an issue and I'll address it. If you haven't had a good look for the answer, checking the source always helps!
If you'd prefer to talk with me about batteries
in real time, I'm often available on the love2d discord.
Lua Core Extensions:
Extensions to existing lua core modules to provide missing features.
mathx
- Mathematical extensions. Aliasmath
.tablex
- Table handling extensions. Aliastable
.stringx
- String handling extensions. Aliasstring
.
General Utility:
General utility data structures and algorithms to speed you along your way.
class
- OOP with inheritance and interfaces in a single function.functional
- Functional programming facilities.map
,reduce
,any
,match
,minmax
,mean
...sequence
- An oo wrapper on sequential tables, so you can dot:insert(i, v)
instead oftable.insert(t, i, v)
. Also supports method chaining for thefunctional
interface above, which can save a lot of needless typing!set
- A set type supporting a full suite of set operations with fast membership testing andipairs
-style iteration.sort
- Provides a stable merge+insertion sorting algorithm that is also, as a bonus, often faster thantable.sort
under luajit. Also exposesinsertion_sort
if needed. Aliasstable_sort
.state_machine
- Finite state machine implementation with state transitions and all the rest. Useful for game states, AI, cutscenes...timer
- a "countdown" style timer with progress and completion callbacks.pubsub
- a self-contained publish/subscribe message bus. Immediate mode rather than queued, local rather than networked, but if you were expecting mqtt in 60 lines I don't know what to tell you. Scales pretty well nonetheless.pretty
- pretty printing tables for debug inspection.
Geometry:
Modules to help work with spatial concepts.
intersect
- 2d intersection routines, a bit sparse at the moment.vec2
- 2d vectors with method chaining, and garbage saving modifying operations. A bit of a mouthful at times, but you get used to it. (there's an issue discussing future solutions).vec3
- 3d vectors as above.
Special Interest:
These modules are probably only useful to some folks in some circumstances, or are under-polished for one reason or another.
async
- Asynchronous/"Background" task management.colour
- Colour conversion routines. Aliascolor
.manual_gc
- Get GC out of your update/draw calls. Useful when trying to get accurate profiling information; moves "randomness" of GC. Requires you to think a bit about your garbage budgets though.measure
- Benchmarking helpers - measure the time or memory taken to run some code.unique_mapping
- Generate a unique mapping from arbitrary lua values to numeric keys - essentially making up a consistent ordering for unordered data. Niche, but can be used to optimise draw batches for example, as you can't sort on textures without it.make_pooled
- add pooling/recycling capability to a class
Any aliases are provided at both the batteries
module level, and globally when exported.
Endless, of course :)
colour
- Bidirectional hsv/hsl/etc conversion would fit nicely here.- Geometry:
vec3
- Needs more fleshing out for serious use, and a refactor to fit the same naming patterns asvec2
.matrix
- A geometry focussed matrix module would made 3d work a lot nicer. Possibly justmat4
.intersect
- More routines, more optimisation :)
- Network:
- Various helpers for networked systems, game focus of course.
rpc
- Remote procedure call system on top ofenet
orsocket
or both.delta
- Detect and sync changes to objects.
- Broadphase:
- Spatial simplification systems for different needs. Probably AABB or point insertion of data.
bucket_grid
- Dumb 2d bucket broadphase.sweep_and_prune
- Popular for bullet hell games.quadtree
/octree
- Everyone's favourite ;)
- UI
- Maybe adopt partner in here, or something evolved from it.
- Image
- Maybe adopt chromatic in here, or something evolved from it.
You are strongly encouraged to use the library in a "fire and forget" manner through require("batteries"):export()
(or whatever appropriate module path), which will modify builtin lua modules (such as table
and math
), and expose all the other modules directly as globals for your convenience.
This eases consumption across your project - you don't have to require modules everywhere, or remember if say, table.remove_value
is built in to lua or not, or get used to accessing the builtin table functions through batteries.table
or tablex
.
While this will likely sit badly with anyone who's had "no globals! ever!" hammered into them, I believe that for batteries
(and many foundational libraries) it makes sense to just import once at boot. You're going to be pulling it in almost everywhere anyway; why bother making yourself jump through more hoops?
You can, of course, use the separate modules on their own, either requiring individual modules explicitly, or a single require for all of batteries
and use through something like batteries.functional.map
. This more involved approach will let you be more clear about your dependencies, if you care deeply about that - at the cost of more setup work needing to re-require batteries everywhere you use it, or expose it as a global in the first place.
I'd strongly recommend that if you find yourself frustrated with the above, stop and think why/if you really want to avoid globals for something intended to be commonly used across your entire codebase! Are you explicitly require
ing math
and table
everywhere you use it too? Are you just as ideologically opposed to require
being a global?
You may wish to reconsider, and save yourself typing batteries
a few hundred times :)
batteries
is fairly easily used as a git submodule - this is how I use it in my own projects, because updating is as quick and easy as a git pull
, and it's easy to roll back changes if needed, and to contribute changes back upstream.
A static install is harder to update, but easier to trim down if you only need some of the functionality provided. It can also never mysteriously break when updating, which might be appealing to those who just cant stop themselves using the latest and greatest.
Many of the modules "just work" on their own, if you just want to grab something specific.
Some of them depend on class
, which can be included alongside pretty easily.
There are some other inter-dependencies in the larger modules, which should be straightforward to detect and figure out the best course of action (either include the dependency or strip out dependent functionality), if you want to make a stripped-down version for your specific use case.
Currently (july 2021) the lib is 40kb or so compressed, including this readme, so do think carefully whether you really need to worry about it!
Currently, the library is operated in a rolling-release manner - the head of the master branch is intended for public consumption. While this is kept as stable as practical, breaking API changes do happen, and more are planned!
For this reason, you should try to check the commit history for what has changed rather than blindly updating. If you let me know that you're using it actively, I'm generally happy to let you know when something breaking is on its way to master
as well.
If there is a large enough user base in the future to make a versioning scheme + non-repo changelog make sense, I will accomodate.
I personally prefer it, but I accept that it's a matter of taste and puts some people off.
I've implemented experimental automatic API conversion (UpperCamelCase for types, lowerCamelCase for methods) that you can opt in to by calling :camelCase()
before :export()
, let me know if you use it and encounter any issues.
zlib, see here