| 
									
										
										
										
											2021-05-22 10:06:21 +02:00
										 |  |  | {% from "macros.lua" import dict_to_lua %} | 
					
						
							|  |  |  | -- this file gets written automatically by the Archipelago Randomizer and is in its raw form a Jinja2 Template | 
					
						
							| 
									
										
										
										
											2021-04-07 01:55:53 +02:00
										 |  |  | require "lib" | 
					
						
							| 
									
										
										
										
											2021-04-17 21:06:53 -07:00
										 |  |  | require "util" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | FREE_SAMPLES = {{ free_samples }} | 
					
						
							| 
									
										
										
										
											2021-05-09 17:26:53 +02:00
										 |  |  | SLOT_NAME = "{{ slot_name }}" | 
					
						
							| 
									
										
										
										
											2021-05-16 00:21:00 +02:00
										 |  |  | SEED_NAME = "{{ seed_name }}" | 
					
						
							| 
									
										
										
										
											2021-06-25 01:31:48 +02:00
										 |  |  | FREE_SAMPLE_BLACKLIST = {{ dict_to_lua(free_sample_blacklist) }} | 
					
						
							| 
									
										
										
										
											2021-06-19 01:00:21 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | {% if not imported_blueprints -%} | 
					
						
							|  |  |  | function set_permissions() | 
					
						
							|  |  |  |     local group = game.permissions.get_group("Default") | 
					
						
							|  |  |  |     group.set_allows_action(defines.input_action.open_blueprint_library_gui, false) | 
					
						
							|  |  |  |     group.set_allows_action(defines.input_action.import_blueprint, false) | 
					
						
							|  |  |  |     group.set_allows_action(defines.input_action.import_blueprint_string, false) | 
					
						
							|  |  |  |     group.set_allows_action(defines.input_action.import_blueprints_filtered, false) | 
					
						
							|  |  |  | end | 
					
						
							|  |  |  | {%- endif %} | 
					
						
							| 
									
										
										
										
											2021-04-17 21:06:53 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | -- Initialize force data, either from it being created or already being part of the game when the mod was added. | 
					
						
							|  |  |  | function on_force_created(event) | 
					
						
							| 
									
										
										
										
											2021-04-29 04:34:47 +02:00
										 |  |  |     --event.force appears to be LuaForce.name, not LuaForce | 
					
						
							| 
									
										
										
										
											2021-04-17 21:06:53 -07:00
										 |  |  |     game.forces[event.force].research_queue_enabled = true | 
					
						
							|  |  |  |     local data = {} | 
					
						
							| 
									
										
										
										
											2021-05-09 18:13:17 +02:00
										 |  |  |     data['earned_samples'] = {{ dict_to_lua(starting_items) }} | 
					
						
							| 
									
										
										
										
											2021-04-29 04:34:47 +02:00
										 |  |  |     data["victory"] = 0 | 
					
						
							| 
									
										
										
										
											2021-04-17 21:06:53 -07:00
										 |  |  |     global.forcedata[event.force] = data | 
					
						
							|  |  |  | end | 
					
						
							|  |  |  | script.on_event(defines.events.on_force_created, on_force_created) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | -- Destroy force data.  This doesn't appear to be currently possible with the Factorio API, but here for completeness. | 
					
						
							|  |  |  | function on_force_destroyed(event) | 
					
						
							| 
									
										
										
										
											2021-04-29 04:34:47 +02:00
										 |  |  |     global.forcedata[event.force.name] = nil | 
					
						
							| 
									
										
										
										
											2021-04-17 21:06:53 -07:00
										 |  |  | end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | -- Initialize player data, either from them joining the game or them already being part of the game when the mod was | 
					
						
							|  |  |  | -- added.` | 
					
						
							|  |  |  | function on_player_created(event) | 
					
						
							| 
									
										
										
										
											2021-04-13 12:35:42 +02:00
										 |  |  |     local player = game.players[event.player_index] | 
					
						
							| 
									
										
										
										
											2021-04-17 21:06:53 -07:00
										 |  |  |     -- FIXME: This (probably) fires before any other mod has a chance to change the player's force | 
					
						
							|  |  |  |     -- For now, they will (probably) always be on the 'player' force when this event fires. | 
					
						
							|  |  |  |     local data = {} | 
					
						
							| 
									
										
										
										
											2021-05-09 20:39:42 +02:00
										 |  |  |     data['pending_samples'] = table.deepcopy(global.forcedata[player.force.name]['earned_samples']) | 
					
						
							| 
									
										
										
										
											2021-04-17 21:06:53 -07:00
										 |  |  |     global.playerdata[player.index] = data | 
					
						
							|  |  |  |     update_player(player.index)  -- Attempt to send pending free samples, if relevant. | 
					
						
							|  |  |  | end | 
					
						
							|  |  |  | script.on_event(defines.events.on_player_created, on_player_created) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function on_player_removed(event) | 
					
						
							|  |  |  |     global.playerdata[event.player_index] = nil | 
					
						
							|  |  |  | end | 
					
						
							|  |  |  | script.on_event(defines.events.on_player_removed, on_player_removed) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 04:34:47 +02:00
										 |  |  | function on_rocket_launched(event) | 
					
						
							|  |  |  |     global.forcedata[event.rocket.force.name]['victory'] = 1 | 
					
						
							|  |  |  |     dumpInfo(event.rocket.force) | 
					
						
							|  |  |  | end | 
					
						
							|  |  |  | script.on_event(defines.events.on_rocket_launched, on_rocket_launched) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-17 21:06:53 -07:00
										 |  |  | -- Updates a player, attempting to send them any pending samples (if relevant) | 
					
						
							|  |  |  | function update_player(index) | 
					
						
							|  |  |  |     local player = game.players[index] | 
					
						
							|  |  |  |     if not player or not player.valid then     -- Do nothing if we reference an invalid player somehow | 
					
						
							|  |  |  |         return | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |     local character = player.character or player.cutscene_character | 
					
						
							|  |  |  |     if not character or not character.valid then | 
					
						
							|  |  |  |         return | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |     local data = global.playerdata[index] | 
					
						
							|  |  |  |     local samples = data['pending_samples'] | 
					
						
							|  |  |  |     local sent | 
					
						
							|  |  |  |     --player.print(serpent.block(data['pending_samples'])) | 
					
						
							|  |  |  |     local stack = {} | 
					
						
							| 
									
										
										
										
											2021-06-19 01:00:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-17 21:06:53 -07:00
										 |  |  |     for name, count in pairs(samples) do | 
					
						
							|  |  |  |         stack.name = name | 
					
						
							|  |  |  |         stack.count = count | 
					
						
							|  |  |  |         if character.can_insert(stack) then | 
					
						
							|  |  |  |             sent = character.insert(stack) | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             sent = 0 | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |         if sent > 0 then | 
					
						
							|  |  |  |             player.print("Received " .. sent .. "x [item=" .. name .. "]") | 
					
						
							|  |  |  |             data.suppress_full_inventory_message = false | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |         if sent ~= count then               -- Couldn't full send. | 
					
						
							|  |  |  |             if not data.suppress_full_inventory_message then | 
					
						
							|  |  |  |                 player.print("Additional items will be sent when inventory space is available.", {r=1, g=1, b=0.25}) | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  |             data.suppress_full_inventory_message = true -- Avoid spamming them with repeated full inventory messages. | 
					
						
							|  |  |  |             samples[name] = count - sent    -- Buffer the remaining items | 
					
						
							|  |  |  |             break                           -- Stop trying to send other things | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             samples[name] = nil             -- Remove from the list | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2021-06-19 01:00:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-17 21:06:53 -07:00
										 |  |  | end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | -- Update players upon them connecting, since updates while they're offline are suppressed. | 
					
						
							|  |  |  | script.on_event(defines.events.on_player_joined_game, function(event) update_player(event.player_index) end) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function update_player_event(event) | 
					
						
							|  |  |  |     update_player(event.player_index) | 
					
						
							|  |  |  | end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-09 20:39:42 +02:00
										 |  |  | script.on_event(defines.events.on_player_main_inventory_changed, update_player_event) | 
					
						
							| 
									
										
										
										
											2021-04-17 21:06:53 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | function add_samples(force, name, count) | 
					
						
							|  |  |  |     local function add_to_table(t) | 
					
						
							|  |  |  |         t[name] = (t[name] or 0) + count | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |     -- Add to global table of earned samples for future new players | 
					
						
							|  |  |  |     add_to_table(global.forcedata[force.name]['earned_samples']) | 
					
						
							|  |  |  |     -- Add to existing players | 
					
						
							|  |  |  |     for _, player in pairs(force.players) do | 
					
						
							|  |  |  |         add_to_table(global.playerdata[player.index]['pending_samples']) | 
					
						
							|  |  |  |         update_player(player.index) | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | script.on_init(function() | 
					
						
							| 
									
										
										
										
											2021-06-19 01:00:21 +02:00
										 |  |  |     {% if not imported_blueprints %}set_permissions(){% endif %} | 
					
						
							| 
									
										
										
										
											2021-04-17 21:06:53 -07:00
										 |  |  |     global.forcedata = {} | 
					
						
							|  |  |  |     global.playerdata = {} | 
					
						
							|  |  |  |     -- Fire dummy events for all currently existing forces. | 
					
						
							|  |  |  |     local e = {} | 
					
						
							|  |  |  |     for name, _ in pairs(game.forces) do | 
					
						
							|  |  |  |         e.force = name | 
					
						
							|  |  |  |         on_force_created(e) | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |     e.force = nil | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     -- Fire dummy events for all currently existing players. | 
					
						
							|  |  |  |     for index, _ in pairs(game.players) do | 
					
						
							|  |  |  |         e.player_index = index | 
					
						
							|  |  |  |         on_player_created(e) | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2021-04-13 11:14:05 +02:00
										 |  |  | end) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | -- hook into researches done | 
					
						
							|  |  |  | script.on_event(defines.events.on_research_finished, function(event) | 
					
						
							| 
									
										
										
										
											2021-04-07 01:55:53 +02:00
										 |  |  |     local technology = event.research | 
					
						
							| 
									
										
										
										
											2021-05-29 20:02:36 +02:00
										 |  |  |     if technology.researched and string.find(technology.name, "ap%-") == 1 then | 
					
						
							|  |  |  |         dumpInfo(technology.force) --is sendable | 
					
						
							| 
									
										
										
										
											2021-06-25 01:31:48 +02:00
										 |  |  |     else | 
					
						
							|  |  |  |         if FREE_SAMPLES == 0 then | 
					
						
							|  |  |  |             return  -- Nothing else to do | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |         if not technology.effects then | 
					
						
							|  |  |  |             return  -- No technology effects, so nothing to do. | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |         for _, effect in pairs(technology.effects) do | 
					
						
							|  |  |  |             if effect.type == "unlock-recipe" then | 
					
						
							|  |  |  |                 local recipe = game.recipe_prototypes[effect.recipe] | 
					
						
							|  |  |  |                 for _, result in pairs(recipe.products) do | 
					
						
							|  |  |  |                     if result.type == "item" and result.amount then | 
					
						
							|  |  |  |                         local name = result.name | 
					
						
							|  |  |  |                         if FREE_SAMPLE_BLACKLIST[name] ~= 1 then | 
					
						
							|  |  |  |                             local count | 
					
						
							|  |  |  |                             if FREE_SAMPLES == 1 then | 
					
						
							|  |  |  |                                 count = result.amount | 
					
						
							|  |  |  |                             else | 
					
						
							|  |  |  |                                 count = get_any_stack_size(result.name) | 
					
						
							|  |  |  |                                 if FREE_SAMPLES == 2 then | 
					
						
							|  |  |  |                                     count = math.ceil(count / 2) | 
					
						
							|  |  |  |                                 end | 
					
						
							|  |  |  |                             end | 
					
						
							|  |  |  |                             add_samples(technology.force, name, count) | 
					
						
							| 
									
										
										
										
											2021-04-07 01:55:53 +02:00
										 |  |  |                         end | 
					
						
							|  |  |  |                     end | 
					
						
							|  |  |  |                 end | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | end) | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-13 12:35:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-02 20:52:06 +02:00
										 |  |  | function dumpInfo(force) | 
					
						
							|  |  |  |     log("Archipelago Bridge Data available for game tick ".. game.tick .. ".") -- notifies client | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-09 16:49:47 +02:00
										 |  |  | function chain_lookup(table, ...) | 
					
						
							|  |  |  |     for _, k in ipairs{...} do | 
					
						
							|  |  |  |         table = table[k] | 
					
						
							|  |  |  |         if not table then | 
					
						
							|  |  |  |             return nil | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |     return table | 
					
						
							|  |  |  | end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-02 20:52:06 +02:00
										 |  |  | -- add / commands | 
					
						
							|  |  |  | commands.add_command("ap-sync", "Used by the Archipelago client to get progress information", function(call) | 
					
						
							|  |  |  |     local force | 
					
						
							| 
									
										
										
										
											2021-05-09 16:49:47 +02:00
										 |  |  |     if call.player_index == nil then | 
					
						
							| 
									
										
										
										
											2021-07-02 20:52:06 +02:00
										 |  |  |         force = game.forces.player | 
					
						
							| 
									
										
										
										
											2021-05-09 16:49:47 +02:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2021-07-02 20:52:06 +02:00
										 |  |  |         force = game.players[call.player_index].force | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |     local research_done = {} | 
					
						
							|  |  |  |     local data_collection = { | 
					
						
							|  |  |  |         ["research_done"] = research_done, | 
					
						
							|  |  |  |         ["victory"] = chain_lookup(global, "forcedata", force.name, "victory"), | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for tech_name, tech in pairs(force.technologies) do | 
					
						
							|  |  |  |         if tech.researched and string.find(tech_name, "ap%-") == 1 then | 
					
						
							|  |  |  |             research_done[tech_name] = tech.researched | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2021-05-09 16:49:47 +02:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2021-07-02 20:52:06 +02:00
										 |  |  |     rcon.print(game.table_to_json({["slot_name"] = SLOT_NAME, ["seed_name"] = SEED_NAME, ["info"] = data_collection})) | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | end) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-27 14:59:16 +02:00
										 |  |  | commands.add_command("ap-print", "Used by the Archipelago client to print messages", function (call) | 
					
						
							|  |  |  |     game.print(call.parameter) | 
					
						
							|  |  |  | end) | 
					
						
							| 
									
										
										
										
											2021-07-02 20:52:06 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  | commands.add_command("ap-get-technology", "Grant a technology, used by the Archipelago Client.", function(call) | 
					
						
							| 
									
										
										
										
											2021-07-04 22:21:53 +02:00
										 |  |  |     if global.index_sync == nil then | 
					
						
							|  |  |  |         global.index_sync = {} | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  |     local tech | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |     local force = game.forces["player"] | 
					
						
							| 
									
										
										
										
											2021-07-04 15:25:56 +02:00
										 |  |  |     chunks = split(call.parameter, "\t") | 
					
						
							| 
									
										
										
										
											2021-04-13 11:14:05 +02:00
										 |  |  |     local tech_name = chunks[1] | 
					
						
							| 
									
										
										
										
											2021-07-04 15:25:56 +02:00
										 |  |  |     local index = chunks[2] | 
					
						
							|  |  |  |     local source = chunks[3] or "Archipelago" | 
					
						
							| 
									
										
										
										
											2021-07-04 22:21:53 +02:00
										 |  |  |     if progressive_technologies[tech_name] ~= nil then | 
					
						
							|  |  |  |         if global.index_sync[index] == nil then -- not yet received prog item | 
					
						
							|  |  |  |             global.index_sync[index] = tech_name | 
					
						
							|  |  |  |             local tech_stack = progressive_technologies[tech_name] | 
					
						
							|  |  |  |             for _, tech_name in ipairs(tech_stack) do | 
					
						
							|  |  |  |                 tech = force.technologies[tech_name] | 
					
						
							|  |  |  |                 if tech.researched ~= true then | 
					
						
							|  |  |  |                     game.print({"", "Received [technology=" .. tech.name .. "] from ", source}) | 
					
						
							|  |  |  |                     game.play_sound({path="utility/research_completed"}) | 
					
						
							|  |  |  |                     tech.researched = true | 
					
						
							|  |  |  |                     return | 
					
						
							|  |  |  |                 end | 
					
						
							|  |  |  |             end | 
					
						
							| 
									
										
										
										
											2021-07-04 15:25:56 +02:00
										 |  |  |         end | 
					
						
							| 
									
										
										
										
											2021-07-04 22:21:53 +02:00
										 |  |  |     elseif force.technologies[tech_name] ~= nil then | 
					
						
							|  |  |  |         tech = force.technologies[tech_name] | 
					
						
							|  |  |  |         if tech ~= nil then | 
					
						
							|  |  |  |             if global.index_sync[index] ~= nil and global.index_sync[index] ~= tech then | 
					
						
							|  |  |  |                 game.print("Warning: Desync Detected. Duplicate/Missing items may occur.") | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  |             global.index_sync[index] = tech | 
					
						
							|  |  |  |             if tech.researched ~= true then | 
					
						
							|  |  |  |                 game.print({"", "Received [technology=" .. tech.name .. "] from ", source}) | 
					
						
							|  |  |  |                 game.play_sound({path="utility/research_completed"}) | 
					
						
							|  |  |  |                 tech.researched = true | 
					
						
							|  |  |  |             end | 
					
						
							| 
									
										
										
										
											2021-04-01 11:40:58 +02:00
										 |  |  |         end | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         game.print("Unknown Technology " .. tech_name) | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2021-07-02 01:58:03 +02:00
										 |  |  | end) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-02 20:52:06 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-02 01:58:03 +02:00
										 |  |  | commands.add_command("ap-rcon-info", "Used by the Archipelago client to get information", function(call) | 
					
						
							|  |  |  |     rcon.print(game.table_to_json({["slot_name"] = SLOT_NAME, ["seed_name"] = SEED_NAME})) | 
					
						
							| 
									
										
										
										
											2021-07-04 22:21:53 +02:00
										 |  |  | end) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | -- data | 
					
						
							|  |  |  | progressive_technologies = {{ dict_to_lua(progressive_technology_table) }} |