Once again, because I still am not, I'll say right now that I am in no way an expert on either lua or Factorio so there may be (read: very likely are) mistakes, Please let me know if/when you find them so I can edit this and not mislead people.
Also the current Factorio version is 0.7.5 (this guide was originally for 0.3). If you are reading this after a new update has been released and this has not been updated BEWARE OF POSSIBLE CHANGES. Also for those of you that need just a bit of help a very good way to learn lua is to look at the Programming in Lua book (first edition (lua 5.0) can be read online here. It is possible to get the current (3rd edition lua 5.2) here). These wiki pages are also very very useful Lua/Objects - for properties and methods of entities and Lua/Events - for the names of events and the variables they have.
For those who need something a bit more hands on than simply looking at existing code, there also is a step by step mod on the wiki, located here that you can see how a mod is made and easily download (or recreate) and play with it. Now that that is out of the way, here it is:
First, I'd like to say again that I am not an expert on this . Second I use Notepad++ for modding (and Adobe Photoshop CS6 for the graphics). I also happen to be using the Windows 7 64 bit OS (on a laptop, not that it matters much). As such any locations I give as to where to find files may, or may not, be accurate. In relation to my second statement, Notepad++ is only for windows, it has not been ported to Linux or Mac (though it is open source, GPL license, written in c++) do a Google search and I'm sure you will find alternatives however.
Next I will say there are two things, once you start with lua, that are really really helpful.
First the wiki (go there, read it, especially the Modding and Scripting section lol): https://forums.factorio.com/wiki/inde ... =Main_Page
Second Factorio uses lua files to add content (as well as images). The game files are found (on my computer) in C:\Program Files\Factorio\data (On mac I BELIEVE it is ~/Library/Application\ Support/factorio/data, on linux systems I BELIEVE it is ~/.factorio/data). Within the data folder you have the base and core folders.
Base contains the data that come with Factorio (inserters, furnaces, the player entity, etc.)
Core contains the rest of the data that comes with the game that is not mod specific (ie. will typically be the same regardless of what mods or campaign you play, fonts, lua code (such as freeplay.lua for new games, and the story code the devs use with their campaigns), sounds, and graphics
The campaign files are found under data\base\campaigns, there you have demo and beta, within both of those there are the levels folders (the individual missions) inside of each level is a control.lua. These are the lua files that control the campaigns. It can be very useful to look through these (especially when you are first learning) to see how the devs did something The blueprint.dat file is the actual map that is loaded when someone plays the campaign, it contains the location of everything from the tiles (ie grass/hills/water) to creepers/spawners to the player and everything else. the locale folder contains the translation from code variables to what is seen in-game (so you don't see 'Unknown Key:"entity.biter-spawner-mk2"' but instead "Killer Biter Spawner") as well as story messages and languages (to change the name of the basic-inserter to (de) Knickarmroboter).
Mods go inside the mods folder (on Windows either %appdata%\Factorio\mods or the install directory if you are using the zip. If it's in %appdata% there is a shortcut in the install directory instead of a folder). In the mods folder there is a modlist.json file, this is ONLY to disable the mods without removing them completely. Factorio will automatically find mods placed in the mods folder and add them to the modlist.json file on its own (though that can be disabled). However, you can do this ingame as well (usually )
Beyond that info, use the forums (read what people have said and look at how they wrote their mods) and experiment, and not necessarily in that order (you will practically ALWAYS learn more from figuring things out on your own, but it will almost always take more time to do so ). If you get completely stuck (which can happen from a simple change that wasn't documented, or weird errors, lol) go ahead and make a post on the forums, I'd 99% guarantee that someone will help you
On to making a mod!
--First off, (straight from wiki) "Every mod is required to provide certain meta information. Let's say the mod name is foo-mod. Then the mod meta information are stored in the mods/foo-mod/info.json file. The file is supposed to contain a map with following fields: name*, author*, version*, title*, homepage, description. The fields with an asterisk are required. The name field is required to have a same value as the name of the directory for now. The version field is in the format "x.y.z" where x is the major version y is the middle version and z is the minor version" (z added in 0.4). Mod dependencies were added with update 0.4. The basic dependency (though if missing entirely this will be default) is "dependencies": ["base"],
An example would be:
Code: Select all
{
"name": "RemoteInterfacesTest",
"version": "0.1",
"title": "Testing Mod for Remote Interfaces",
"author": "FreeER",
"description": "Simple mod to test remote interfaces between mods."
"dependencies": ["base >= 0.4.1", "scenario-pack", "? bar = 0.3"],
}
This info file is the one I used for testing with remote interfaces between mods It's not an actual mod that does anything so no you won't find it in the forums mod section lol. Now the next file that will (almost always) be included with mods is data.lua. Small mods can easily get away with just these two files and a control.lua if they want in-game code (which is explained below). Only the largest mods really need to layout files in the manner that the devs have done with the base 'mod', Though I like the layout because A) it is easy to tell exactly where everything is if B) if someone want to look through it C) I'm used to it from pre 0.3.x and D) It's simply more orderly in my opinion.
Now to start with the prototypes guide, I'll first say that an 'entity' is anything that you see in the world, except for the ground itself and the items that you drop on the ground. Anything that you see in your inventory is an 'item'. The ground itself (just for reference) is called a 'tile'. I'll show you a few examples of 'prototypes' which is the data behind every object in the game (including the player). We'll start with my standard (the stone furnace), because once you understand it you can create a faster furnace and who doesn't want that
Code: Select all
{
type = "item",
name = "stone-furnace",
icon = "__base__/graphics/icons/stone-furnace.png",
flags = {"goes-to-quickbar"},
group = "production",
order = "h-b-a",
place_result = "stone-furnace",
stack_size = 64
},
Now the really important part (especially if you copy and paste and then change the relevant info lol) place_result = "entity_name". THIS is where you put the name of the entity that we'll go over in a minute, if you are creating an item (like iron plates or alien artifacts) that players will not be placing on the ground you actually skip this line. and last is the stack_size, fairly obvious this is the maximum size that one slot in an inventory can hold.
Now onto the recipe code :
Code: Select all
{
type = "recipe",
name = "stone-furnace",
ingredients = {{"stone", 5}},
result = "stone-furnace"
},
And finally the part some of you have been waiting for! The entity description!
Code: Select all
{
type = "furnace",
name = "stone-furnace",
icon = "__base__/graphics/icons/stone-furnace.png",
flags = {"placeable-neutral", "placeable-player", "player-creation"},
minable = {mining_time = 1, result = "stone-furnace"},
max_health = 150,
corpse = "medium-remnants",
resistances =
{
{
type = "fire",
percent = 80
},
{
type = "explosion",
percent = 30
}
},
collision_box = {{-0.7, -0.7}, {0.7, 0.7}},
selection_box = {{-0.8, -1}, {0.8, 1}},
smelting_categories = {"smelting"},
result_inventory_size = 1,
smelting_energy_consumption = 3,
smelting_speed = 0.5,
source_inventory_size = 1,
energy_source =
{
type = "burner",
effectivity = 1,
fuel_inventory_size = 1,
emissions = 0.01,
smoke =
{
{
name = "smoke",
deviation = {0.1, 0.1},
frequency = 0.5,
position = {0, -2.3}
}
}
},
drawing_scale = 1,
on_animation =
{
filename = "__base__/graphics/entity/stone-furnace/stone-furnace.png",
priority = "extra-high",
frame_width = 81,
frame_height = 64,
frame_count = 1,
shift = {0.5, 0.05 }
},
off_animation =
{
filename = "__base__/graphics/entity/stone-furnace/stone-furnace.png",
priority = "extra-high",
frame_width = 81,
frame_height = 64,
frame_count = 1,
shift = {0.5, 0.05 }
},
fire_animation =
{
filename = "__base__/graphics/entity/stone-furnace/stone-furnace-fire.png",
priority = "extra-high",
frame_width = 23,
frame_height = 27,
frame_count = 12,
shift = { 0.078125, 0.5234375}
},
fast_replaceable_group = "furnace"
},
It should be fairly obvious what they mean, but the possible flags are: "pushable", "placeable-player", "placeable-off-grid", "placeable-neutral", and "placeable-enemy". Pushable means it can be moved pushed by the car, and placeable-player/enemy/neutral means it can placed (belong to) either 'force'. placeable-off-grid refers to the fact that most entities are placed on an invisible 'grid' within the world, entities with the placeable-off-grid flag do not have to line up with this grid (like trees, players, biters, and land-mines).
minable is of course the time it takes to mine (I believe with a iron-pickaxe) and the item you get when you mine it. Of course if it is not minable (ie, player, biter, biter corpse, etc.) you skip the minable line. max_health is the health of the entity when placed down or healing.
The corpse is the 'corpse' type entity that gets spawned when the entity dies. I do not really understand the resistances (I have not played with them) but I assume that 80% means it resists 80% of any fire damage (or only takes 20% of fire damage, however you'd like to look at it)
The collision box and selection box are important because the selection is the blue outline that shows up when you mouse over an entity and the collision box is what actually prevents you from running through something. Of course if you want players to be able to walk through an entity, leave the collision box off, however unless you do not want players to be able to pick an entity up you need a selection box (in that case don't use minable lol) I believe the numbers correspond in some manner to the size of the graphic (I belive 32 pixels are 1 tile) but honestly I just play with these numbers until it looks right
smelting_categories is semi-important If you simply want it to work as a regular furnace leave it as smelting, but you can create your own categories with
Code: Select all
type = "recipe-category",
name = "smelting"
The energy source type can be burner, meaning it uses coal and has the values you see above (again fairly self-explanatory but play with the numbers if you like), or type = "electric", with input_priority = (secondary or primary or tertiary) and buffer_capacity and input_flow_limit and resting_consumption_ratio
The smoke is fairly easy to understand, play with it if you do not understand just by looking at it Drawing scale is, I'm fairly certain, the scale at which the entity graphic is drawn (say you created a large image so you could work with it easier, rather than resaving it smaller you can use the drawing_scale).
Then you give it the path/location to the pictures and or animations, in the case of the furnace there are three animations on/off_animation and fire_animation.
fast_replaceable_froup is the entity 'type' that can be replaced by the furnace (or can replace the furnace) without having to mine it first. And that's it, it may seem like alot if you are brand new, but once you have played with it it is really pretty easy to understand. The rest of the prototypes are really similar and any differences are obvious simply from the name, like attack_cooldown from unit entities, and spawning_radius for the unit-spawner.
Now, if you were to open factorio\data\base\scenarios\freeplay\freeplay.lua you would be looking at the lua code that is run whenever you start a new game. Two things before we jump into the Lua code. In lua 'if' statements (sometimes called if-then statements) MUST include 'then' and MUST be ended with 'end' . Second, if you see '--' anything on the same line after '--' is a comment and is ignored by lua, this '--[[multiline-comment--]]' is also a comment but everything in between the '--[[' and the '--]]' is ignored, regardless of how many lines are in between. The basic syntax of an if statement is:
Code: Select all
if something == somethingElse then --[[if something is equal to somethingElse then (note: you can also use ~= as not equal, > as greater than, < as less than, >= as greater than or equal to, and <= as less than or equal to)--]]
--place the code you want executed when something is equal to somethingElse here
end
Now then onto a bit of code:
When you open freeplay.lua you will see
Code: Select all
require "defines"
Code: Select all
normalattacksentevent = game.generateeventname()
landingattacksentevent = game.generateeventname()
This is left over from how the devs let other modders find out if the player is currently being attacked by creepers, either during a normal creeper wave or from the 'landing wave' (sent when the player places the rocket defense down and must then defend it). The rest of this is part of the next bit of code that starts with remote.addinterface, but I'll actually be skipping because it is slightly more complicated than it looks and it is intended for your mod to interact with other mods (this bit of code specifically for interacting with the freeplay.lua code). We WILL come back to this later, but for now let us move on to some code we can use to actually create our own mod before we worry too much about adding interfaces with other mods.
We'll be skipping down to line 39 (if you are using an editor that allows you to see line number it will be easy to find, other wise here is the code, you'll just have to scroll down a bit until you see it)
Code: Select all
gt = game.gettext
game.oninit(function()
local character = game.player.character
character.insert{name="iron-plate", count=8}
character.insert{name="pistol", count=1}
character.insert{name="basic-bullet-magazine", count=10}
initattackdata()
game.showmessagedialog(gt("msg-intro"))
end)
To continue the variable discussion there are 4 (maybe 5) 'datatypes' in lua. You have your typical numbers (which can be integers with a -/+ sign and can be floating like 1.246578, or just your standard number 5), then you have strings (which are any bit of text "this is a string", "as is this"), what can be useful and confusing about lua is that it can sometimes change numbers into strings and strings into numbers depending on how a variable is being used. If you have a string of "513" and want to add 32 to it lua would have no problem doing so because it can recognize that the string is made entirely of numbers. Now I said there were 4 types of data, the other two are booleans (either 'true' or 'false') and tables or arrays (just like you would expect a table to do, it can contain other datatypes within itself, including other tables). Tables are created using tableName = {key=value} or tableName = {value1, value2, etc.} (where key is usually a short recognizable name for what the value is of, if you do not specify a key then it is automatically assigned a number in consecutive order 1,2,3,etc.). Tables are referenced by their name and their data like so: tableName.secondTable["key"] (which would give you the value of key from secondTable, that is within tableName).
All other variables are assigned using variableName=value/data. There is one other 'type' which is 'nil', any variable that has not been defined (ie. does not exist) has a value of nil. To delete a variable all you have to do is set it equal to nil like: variableName = nil (more info on variables can be found here: http://www.troubleshooters.com/codecorn ... iables.htm)
Now If you are still following along after that long (and probably somewhat rambling) paragraph about variables datatypes and scopes, you can understand that
Code: Select all
local character = game.player.character
Now what does game.player.character.insert{name="iron-plate", count=8} do? Well insert is a function that (like it's name says) inserts what ever you tell it to into an inventory. In this case it is associated with game.player.character (which is the player you see running around on the screen) and thus it inserts the count (8) of the item name ("iron-plate") into the player's inventory (note that the name of the item MUST be within the "" because the function expects a string, without the quotes it assumes that what you gave it was a variable name). Since it is within the onit block this code is ran (as I said at the beginning of that long paragraph above) when the new world first loads. Thus what happens is that when you start a new world you (almost immediately) are given 8 iron plates, a pistol, and 10 magazines of basic ammo. The next thing that happens is that the game runs the function initattackdata (which if you read the function above, sets the first attack wave to be five enemies and to start in 216000 ticks or 3600 seconds or 60 minutes) and then the game pops up a message telling you to 'prepare the planet for colonizing forces' and 'protect [yourself] from the natives'.
The next bit of coding may help explain if statements a bit more so lets go over it:
Code: Select all
-- for backwards compatibility
game.onload(function()
if glob.attackdata == nil then
initattackdata()
if glob.attackcount ~= nil then glob.attackdata.attackcount = glob.attackcount end
if glob.untilnextattacknormal ~= nil then glob.attackdata.untilnextattack = glob.untilnextattacknormal end
end
end)
slpwnd wrote:The saving/loading mechanism is meant for plain lua structures and factorio data. Serializing functions (though possible) can cause issues. Well plain functions should be fine, however closures will most probably cause issues. Just a sidenote the library we use for serialization is called serpent. It is actually required by default for every script run in factorio. It can also be used for pretty printing the variables. Something like:For details have a look at: https://github.com/pkulchenko/serpentCode: Select all
game.player.print(serpent.block(variabletoprint))
Now to the code, this time we are using onload, which as you might guess from the name occurs whenever the data is loaded (ie from a previous save). What the code does is as soon as it is loaded it checks to see if attackdata exists (if a variable is equal to nil then it has not been defined and thus does not exist) if it does not then it runs initattackdata() and then checks to see if attackcount is NOT equal to nil (glob.attackdat.attackcount I THINK is the pre 0.3.x location of the number of enemies that attack in the next wave. Thus if the old location is being used we need to change it to what the new value is) the same occurs for untilnextattacknormal. After that both the original if block and the onload event function is closed/ended. This is an important piece of code (especially for when you update your mod) because IF/WHEN you change the location of one/several of your variables you must expect that someone will update and expect to continue from their previous save (unless you explicitly say on your page that the new version is incompatible with previous versions of your mod and you will probably still[\b] get a couple who didn't read your page lol).
Finally we get down to the code block starting with
Code: Select all
game.onevent(defines.events.ontick, function(event)
That is the very basics of lua coding for factorio. To really learn how to code then you will need to watch the wiki for updated information (especially the events) and simply TRY IT OUT. Another good way is to look at the code that other people have written to see if you can A) understand it and B) see why they did it that way and if how they did it is better than you would have done it. Also watch the forums for people who ask for help and see if you can figure out how why their code does not work or if there is a better way to do it (just playing around with code will help you get better at using it)
One last thing to be aware of, in-game you can type lua commands by pressing the ~ (the key next to the 1 on a standard us keyboard). You can use it to set variables and then use those variables to execute a longer command several times. HOWEVER, you will need to set up a mod interface before you can use any of your mod functions, because the console uses the code described in freeplay.lua (like creating a simple function to give you all the items in your mod for testing purposes).
I'll explain mod interfaces shortly but first:
Some common variables/commands to know (there are probably more, I'll edit this when people yell at me or i remember them)
game.getplayer().print("some text") or game.player.print("test")--Will print text to the screen (very useful when debugging your code, and if you were creating a mission for someone)
game.showmessagedialog("text") --Will pop a message up so that the user has to hit TAB to acknowledge it, use this if the info you are telling the player is something you do not what them to ignore/miss
game.player.character.position.x and game.player.character.position.y --the x and y positions of the player. You probably want to first check if game.player.character then --code end, since you can now play as a 'ghost'/'god'.
game.findentities{topLeftX, topLeftY, bottomRightX, bottomRightY} --gives a table named entities containing all the data from all of the entities found, small example below, to destroy an entity find it and then do entities[key].destroy() Note that you may find the older version where it literally states topleft = {x , y}, bottomright = {x, y}. This has been changed in 0.3.x and it now causes random issues, swapping of the variables, use the NEW version shown on this line.
Useful code aka "cheater chest":
Code: Select all
game.onevent(defines.events.onbuiltentity, function(event)
if event.createdentity.name == "wooden-chest" then --if the created entity is a wooden chest
entities = game.findentities{event.createdentity.position.x - .1, event.createdentity.position.y - .1, event.createdentity.position.x + .1, event.createdentity.position.y + .1}
for fieldName, _ in pairs(entities) do
chest=entities[fieldName].getinventory(defines.inventory.chest) end --[[a better method would be chest = event.createdentity.getinventory(defines.inventory.chest) BUT I left the findentities and the for loop as examples for when you need to find the position of something near the player but you don't have it's exact position, simply change the event.createdentity.position.x to game.player.position.x and game.player.position.y--]]
if chest.isvalid and chest.caninsert(name="iron-plate", count="200"} then --if the chest exists (to prevent accessing after it has been mined/destoryed) and if there is room for 200 iron-plates
chest.insert({name="iron-plate", count="200"}) --insert 200 iron-plates
game.showmessagedialog("Gratis") --tell the player it's free
game.player.character.getinventory(defines.inventory.playerguns)[1].clear() --clear the player's first gun slot
game.player.character.getinventory(defines.inventory.playerguns)[1].insert({name="rifle-gun", count="1"}) --give submachine gun
game.player.character.getinventory(defines.inventory.playerammo)[1].clear()
game.player.character.getinventory(defines.inventory.playerammo)[1].insert({name="piercing-bullet-magazine", count="100"}) --give good ammo
game.player.character.insert({name="piercing-bullet-magazine", count="200"}) -- give more good ammo
game.player.character.insert({name="coal", count="200"}) --give coal
game.player.character.insert({name="car", count="1"})--give a car
end --end if chest is valid
end --end if entity=wooden-chest
end--end onbuiltentity event
Code: Select all
for i=1, v do --i is a variable local to the for loop. the syntax of a for loop is for i to v count by x (notice that there is no x in the code, if you do not add the count by (x in this example) variable it is assumed to be 1)
count=count+1 --increase count by one
game.getplayer().print(count) --note we are printing the value of the variable count, not the text string count. This is why "" are important when you want to print something to the screen
end
Did you notice how the chest had several properties and functions, like valid and position and insert() and perhaps player.print() and player.character? How do you know what entities have what properties and methods/functions? Well you go to the wiki here
Now, remember that code WAY up at the top? The one that began with
Code: Select all
remote.addinterface("freeplay",
Code: Select all
remote.addinterface("freeplay",
{
setattackdata = function(data)
glob.attackdata = data
end,
getattackdata = function()
return glob.attackdata
end,
getnormalattacksentevent = function()
return normalattacksentevent
end,
getlandingattacksentevent = function()
return landingattacksentevent
end
})
Code: Select all
if remote.interfaces.freeplay and remote.interfaces.freeplay.getattackdata then your code
Code: Select all
remote.addinterface("yourMod",
{
gimme = function(name, count) --remote.call("yourMod", "gimme", "gun-turret", 10)
game.player.character.insert{name=name, count=count}
end,
}
Code: Select all
cheat = function() --remote.call("RemoteInterfacesTest", "cheat")
game.player.character.insert{name="car", count=10}
game.player.character.insert{name="iron-ore", count=10}
end,
version 0.3 edited on 5/21 (thanks to the developers for the great game (and especially to slpwnd and kovarex for helping me understand modding, and thus, hopefully, helping you))
version 0.4 edited to 0.7.5 on 11/11