local event = require 'utils.event' local key_item = "computer" local blacklisted_tiles = {"out-of-map", "water", "deepwater", "water-green", "lab-white", "lab-dark-1"} local teleporter_names = {"Stuedrik", "Wrirrirb", "Cekoht", "Deokels", "Gnaecl", "Yffolf", "Xuohsywae", "Flublodoe", "Dicyhnea", "Ruovyk", "Truecuhz", "Vaux'ers", "Gyttux", "Flys", "Qlammed", "Gynneo", "Xraeqoeht", "Phuashlk", "Cuahnennu", "Kneizigh", "Zruex'iz", "Stux'ar", "Zrihq", "Opsyms", "Ogigh", "Nek'oke", "Knoebhybeo", "Kluetryrceo", "Chahz", "Xralarb", "Wrib", "Breipuhz", "Nueglahloe", "Wuammea", "Iblameo", "Wuansyrfua", "Flohlilue", "Vev'unnae", "Deivrym", "Atahz", "Zrux'ysk", "Kyq'yks", "Gnyf'ilm", "Knaegnom", "Gnaelphiss", "Kmaek'irba", "Zruffira", "Kicremme", "Tuot'az", "Ouprard", "Tyv'im", "Get'yks", "Essyh", "Vliln", "Glucutha", "Teoblux", "Feohshowtha", "Dedrapt", "Isom", "Xoxxywth", "Qrokmeg", "Uzzuhz", "Achumea", "Caelhume", "Diewylfi", "Deak'yrbie", "Bepsilp", "Uogeptue", "Gouq'oht", "Strauyb", "Evvyks", "Riux", "Ielfahs", "Myls", "Dael'eth", "Tluymnyrbu", "Qluephaulpie", "Bruetheltua"} local charged_accumulators_required = 8 function get_power_status(teleporter_index, drain_power) local surface = game.surfaces[global.teleporters[teleporter_index].surface] local a = { left_top = {x = global.teleporters[teleporter_index].position.x - 5, y = global.teleporters[teleporter_index].position.y - 5}, right_bottom = {x = global.teleporters[teleporter_index].position.x + 6, y = global.teleporters[teleporter_index].position.y + 6} } local power_cells = surface.find_entities_filtered({area = a, name = "accumulator"}) if not power_cells[1] then return "No energy source found - Operation not possible" end if #power_cells < charged_accumulators_required then return "Low Energy - More energy sources needed" end local charged_cells = {} for _, cell in pairs(power_cells) do if cell.energy >= 5000000 then table.insert(charged_cells, cell) end if #charged_cells == charged_accumulators_required then break end end if #charged_cells < charged_accumulators_required then return "Low Energy - Not enough accumulator charge" end if drain_power == true then for _, cell in pairs(charged_cells) do cell.energy = 0 end end return true end local function gui_spawn_new_teleporter(player) if player.gui.left["spawn_new_teleporter_button"] then player.gui.left["spawn_new_teleporter_button"].destroy() end local b = player.gui.left.add({ type = "button", name = "spawn_new_teleporter_button", caption = "Deploy Teleporter"}) b.style.minimal_height = 38 b.style.minimal_width = 38 b.style.top_padding = 6 b.style.left_padding = 12 b.style.right_padding = 12 b.style.bottom_padding = 6 b.style.font = "default-listbox" b.style.font_color = {r = 0.35, g = 0.5, b = 1} end local function gui_teleporter(player, visited_teleporter_index) if player.gui.left["gui_teleporter"] then player.gui.left["gui_teleporter"].destroy() end local frame = player.gui.left.add({ type = "frame", name = "gui_teleporter", direction = "vertical"}) local t = frame.add({type = "table", column_count = 2, name = "teleporter_heading"}) local l = t.add({type = "label", caption = " "}) l.style.font_color = {r = 0.35, g = 0.5, b = 1} l.style.font = "heading-1" local l = t.add({type = "label", caption = global.teleporters[visited_teleporter_index].name, name = visited_teleporter_index}) l.style.font_color = {r = 0.77, g = 0.77, b = 0.77} l.style.font = "default-bold" l.style.top_padding = 4 local frame2 = frame.add({ type = "frame", direction = "vertical"}) frame2.style.maximal_height = 400 frame2.style.top_padding = 8 frame2.style.font = "default-bold" frame2.style.font_color = {r = 0.88, g = 0.22, b = 0.22} if #global.teleporters < 2 then frame2.caption = "No connected teleporters found." frame2.style.top_padding = 14 frame2.style.bottom_padding = 0 return end local power_status = get_power_status(visited_teleporter_index) if power_status ~= true then frame2.caption = power_status frame2.style.top_padding = 14 frame2.style.bottom_padding = 0 return end local scroll_pane = frame2.add({ type = "scroll-pane", direction = "vertical", horizontal_scroll_policy = "never", vertical_scroll_policy = "auto"}) for x = #global.teleporters, 1, -1 do local surface = game.surfaces[global.teleporters[x].surface] local tile = surface.get_tile(global.teleporters[x].position) if x ~= visited_teleporter_index and tile.name == "lab-white" then local t = scroll_pane.add({ type = "table", column_count = 2}) local b = t.add({type = "button", caption = "> " .. global.teleporters[x].name .. " <", name = "teleporter_" .. x}) b.style.minimal_width = 250 b.style.font_color = {r = 0.35, g = 0.5, b = 1} b.style.font = "default-listbox" b.style.top_padding = 7 b.style.bottom_padding = 7 local tt = t.add({ type = "table", column_count = 2}) local l = tt.add({type = "label", caption = global.teleporters[x].surface .. ": "}) l.style.font_color = {r = 0.22, g = 0.88, b = 0.44} l.style.font = "default-bold" l.style.minimal_width = 65 l.style.top_padding = 0 l.style.bottom_padding = 0 l.style.left_padding = 8 local l = tt.add({type = "label", caption = "X: " .. tostring(global.teleporters[x].position.x) .. " Y: " .. tostring(global.teleporters[x].position.y)}) l.style.font = "default" l.style.font_color = {r = 0.77, g = 0.77, b = 0.77} l.style.minimal_width = 100 l.style.top_padding = 0 l.style.bottom_padding = 0 local l = tt.add({type = "label", caption = "Distance: "}) l.style.font_color = {r = 0.22, g = 0.88, b = 0.44} l.style.font = "default-bold" l.style.minimal_width = 65 l.style.top_padding = 0 l.style.bottom_padding = 0 l.style.left_padding = 8 local l = tt.add({type = "label", caption = tostring(math.ceil(math.sqrt((global.teleporters[x].position.x - player.position.x)^2 + (global.teleporters[x].position.y - player.position.y)^2), 0)) .. " Units"}) l.style.font = "default" l.style.font_color = {r = 0.77, g = 0.77, b = 0.77} l.style.minimal_width = 100 l.style.top_padding = 0 l.style.bottom_padding = 0 if #global.teleporters > 2 and x ~= 1 then local l = scroll_pane.add({ type = "label", caption = "-----------------------------------------------------------------"}) l.style.font_color = {r = 0.77, g = 0.77, b = 0.77} l.style.font = "default" l.style.top_padding = 0 l.style.bottom_padding = 0 end end end end local function spawn_teleporter(player) if not global.teleporters then global.teleporters = {} end local surface = player.surface local pos = {x = math.floor(player.position.x, 0), y = math.floor(player.position.y, 0)} local a = { left_top = {x = pos.x - 3, y = pos.y - 3}, right_bottom = {x = pos.x + 3, y = pos.y + 3} } local c = surface.count_tiles_filtered{area = a, name = blacklisted_tiles, limit = 1} if c == 0 then local i = player.get_main_inventory() local removed_item_count = i.remove({name = key_item, count = 1}) if removed_item_count ~= 1 then return end local str = teleporter_names[math.random(1, #teleporter_names)] local str2 = str while str == str2 do str2 = teleporter_names[math.random(1, #teleporter_names)] end table.insert(global.teleporters, {position = {x = pos.x, y = pos.y}, name = str .. " " .. str2, surface = surface.name}) local tiles = { {name = "lab-white", position = pos}, {name = "lab-dark-1", position = {pos.x - 1, pos.y - 1}}, {name = "lab-dark-1", position = {pos.x, pos.y - 1}}, {name = "lab-dark-1", position = {pos.x + 1, pos.y - 1}}, {name = "lab-dark-1", position = {pos.x + 1, pos.y}}, {name = "lab-dark-1", position = {pos.x + 1, pos.y + 1}}, {name = "lab-dark-1", position = {pos.x, pos.y + 1}}, {name = "lab-dark-1", position = {pos.x - 1, pos.y + 1}}, {name = "lab-dark-1", position = {pos.x - 1, pos.y}}, } surface.set_tiles(tiles, true) game.print(player.name .. " has deployed a Teleporter!", {r = 0.35, g = 0.5, b = 1}) end end local function check_inventory_for_key_item(player_index) local player = game.players[player_index] if player.get_item_count(key_item) > 0 then gui_spawn_new_teleporter(player) else if player.gui.left["spawn_new_teleporter_button"] then player.gui.left["spawn_new_teleporter_button"].destroy() end end end local function on_player_changed_position(event) if not global.teleporters then return end local player = game.players[event.player_index] if player.character.driving == true then return end --if game.tick % 2 == 1 then return end local a = { left_top = {x = player.position.x - 1, y = player.position.y - 1}, right_bottom = {x = player.position.x + 1, y = player.position.y + 1} } local tile = player.surface.find_tiles_filtered{area = a, name = "lab-white", limit = 1} if not tile[1] then if player.gui.left["gui_teleporter"] then player.gui.left["gui_teleporter"].destroy() end return end for x, teleporter in pairs(global.teleporters) do if teleporter.position.x == tile[1].position.x and teleporter.position.y == tile[1].position.y then gui_teleporter(player, x) break end end end local function on_player_main_inventory_changed(event) check_inventory_for_key_item(event.player_index) end local function on_player_dropped_item(event) check_inventory_for_key_item(event.player_index) end local function on_gui_click(event) if not event then return end if not event.element then return end if not event.element.valid then return end local player = game.players[event.element.player_index] local name = event.element.name if name == "spawn_new_teleporter_button" then spawn_teleporter(player) end if string.sub(name, 1, 10) ~= "teleporter" then return end local index = tonumber(string.sub(name, 12)) local visited_teleporter_index = tonumber(player.gui.left["gui_teleporter"]["teleporter_heading"].children[2].name) local status = get_power_status(visited_teleporter_index, true) if status == true then local surface = game.surfaces[global.teleporters[index].surface] for _, p in pairs(game.connected_players) do p.play_sound{path="utility/armor_insert", volume_modifier=1, position = global.teleporters[visited_teleporter_index].position} p.play_sound{path="utility/armor_insert", volume_modifier=1, position = global.teleporters[index].position} end surface.create_entity({name = "water-splash", position = player.position}) surface.create_entity({name = "blood-explosion-big", position = player.position}) local p = surface.find_non_colliding_position("character",global.teleporters[index].position, 2,0.5) if p then player.teleport(p, global.teleporters[index].surface) else player.teleport(global.teleporters[index].position, global.teleporters[index].surface) end end end event.add(defines.events.on_player_changed_position, on_player_changed_position) event.add(defines.events.on_player_dropped_item, on_player_dropped_item) event.add(defines.events.on_player_main_inventory_changed, on_player_main_inventory_changed) event.add(defines.events.on_gui_click, on_gui_click)