1
0
mirror of https://github.com/ComfyFactory/ComfyFactorio.git synced 2025-01-24 03:47:58 +02:00
ComfyFactorio/maps/pirates/progression.lua

553 lines
18 KiB
Lua
Raw Normal View History

2021-10-13 09:21:53 +01:00
local Public = {}
local Memory = require 'maps.pirates.memory'
local Math = require 'maps.pirates.math'
local Balance = require 'maps.pirates.balance'
local Common = require 'maps.pirates.common'
local CoreData = require 'maps.pirates.coredata'
local Utils = require 'maps.pirates.utils_local'
local inspect = require 'utils.inspect'.inspect
local Structures = require 'maps.pirates.structures.structures'
local Boats = require 'maps.pirates.structures.boats.boats'
local Surfaces = require 'maps.pirates.surfaces.surfaces'
local Crowsnest = require 'maps.pirates.surfaces.crowsnest'
local Server = require 'utils.server'
local Dock = require 'maps.pirates.surfaces.dock'
local Islands = require 'maps.pirates.surfaces.islands.islands'
local Sea = require 'maps.pirates.surfaces.sea.sea'
local Crew = require 'maps.pirates.crew'
local Roles = require 'maps.pirates.roles.roles'
local Parrot = require 'maps.pirates.parrot'
local Quest = require 'maps.pirates.quest'
local Shop = require 'maps.pirates.shop.shop'
local Overworld = require 'maps.pirates.overworld'
local Hold = require 'maps.pirates.surfaces.hold'
local Cabin = require 'maps.pirates.surfaces.cabin'
local Upgrades = require 'maps.pirates.boat_upgrades'
local Task = require 'utils.task'
local Token = require 'utils.token'
local ShopMini = require 'maps.pirates.shop.minimarket'
2022-02-24 19:39:03 +00:00
function Public.fuel_depletion_rate()
local memory = Memory.get_crew_memory()
local state = memory.boat.state
2022-02-27 16:42:25 +00:00
if state == Boats.enum_state.ATSEA_SAILING or state == Boats.enum_state.APPROACHING then
2022-02-24 19:39:03 +00:00
return Balance.fuel_depletion_rate_sailing()
2022-02-27 16:42:25 +00:00
elseif state == Boats.enum_state.LEAVING_DOCK then
return Balance.fuel_depletion_rate_sailing() * 2
2022-02-26 13:55:36 +00:00
elseif state == Boats.enum_state.RETREATING then
return Balance.fuel_depletion_rate_sailing() / 10
2022-02-24 19:39:03 +00:00
elseif state == Boats.enum_state.LANDED then
return Balance.fuel_depletion_rate_static()
else
return 0
end
end
2021-10-13 09:21:53 +01:00
function Public.set_off_from_starting_dock()
local memory = Memory.get_crew_memory()
if memory.game_lost then return end
memory.crewstatus = Crew.enum.LEAVING_INITIAL_DOCK
memory.boat.state = Boats.enum_state.LEAVING_DOCK
Boats.place_boat(memory.boat, CoreData.moving_boat_floor, false, false)
Common.current_destination().type = Surfaces.enum.LOBBY
memory.mapbeingloadeddestination_index = 1 -- whatever the index of the first island is
memory.loadingticks = 0
local surface = game.surfaces[CoreData.lobby_surface_name]
local p = Utils.psum{memory.boat.position, Boats.get_scope(memory.boat).Data.crewname_rendering_position}
memory.boat.rendering_crewname_text = rendering.draw_text{
text = memory.name,
-- render_layer = '125', --does nothing
surface = surface,
target = p,
color = CoreData.colors.renderingtext_yellow,
scale = 8,
font = 'default-game',
alignment = 'left'
}
end
function Public.go_from_starting_dock_to_first_destination()
local memory = Memory.get_crew_memory()
local boat = memory.boat
local crew_members = Crew.choose_crew_members()
local crew_members_count = #memory.crewplayerindices
if crew_members_count > 0 then
memory.crewstatus = Crew.enum.ADVENTURING
local message = '[' .. memory.name .. '] Crew members: '
local b = false
for _, index in pairs(memory.crewplayerindices) do
if game.players[index] and game.players[index].name then
if b == true then
message = message .. ', '
else b = true end
message = message .. game.players[index].name
end
end
message = message .. '.'
Server.to_discord_embed_raw(CoreData.comfy_emojis.pogkot .. message)
Roles.assign_captain_based_on_priorities()
for _, player in pairs(crew_members) do
Crew.player_abandon_endorsements(player)
for item, amount in pairs(Balance.starting_items_player) do
player.insert({name = item, count = amount})
end
end
boat.stored_resources = {}
Shop.initialise_main_shop()
Hold.create_hold_surface(1)
Cabin.create_cabin_surface()
local items = Balance.starting_items_crew_upstairs()
Boats.place_random_obstacle_boxes(boat, 6, items, 0)
-- go:
Public.progress_to_destination(1) --index of first destination
boat.EEI_stage = 1
Boats.update_EEIs(boat)
if Common.difficulty() == 1 then
Boats.upgrade_chests(boat, 'iron-chest')
Hold.upgrade_chests(1, 'iron-chest')
Crowsnest.upgrade_chests('iron-chest')
elseif Common.difficulty() > 1 then
Boats.upgrade_chests(boat, 'steel-chest')
Hold.upgrade_chests(1, 'steel-chest')
Crowsnest.upgrade_chests('steel-chest')
end
memory.age = 0
memory.real_age = 0
else
Boats.destroy_boat(boat)
Crew.disband_crew()
end
end
local place_dock_jetty_and_boats = Token.register(
function(data)
local memory = Memory.get_crew_memory()
if memory.game_lost then return end
Surfaces.Dock.place_dock_jetty_and_boats()
local destination = Common.current_destination()
ShopMini.create_minimarket(game.surfaces[destination.surface_name], Surfaces.Dock.Data.market_position)
end
)
function Public.progress_to_destination(destination_index)
local memory = Memory.get_crew_memory()
local global_memory = Memory.get_global_memory()
if memory.game_lost then return end
local boat = memory.boat
local oldsurface = game.surfaces[boat.surface_name]
local old_type = Surfaces.SurfacesCommon.decode_surface_name(oldsurface.name).type
local destination_data = memory.destinations[destination_index]
local static_params = destination_data.static_params
local type = destination_data.type
local subtype = destination_data.subtype
local newsurface_name = Surfaces.SurfacesCommon.encode_surface_name(memory.id, destination_index, type, subtype)
local newsurface = game.surfaces[newsurface_name]
local initial_boatspeed, starting_boatposition
if type == Surfaces.enum.ISLAND then --moved from overworld generation, so that it updates properly
local covered1_requirement = Balance.covered1_entry_price()
destination_data.dynamic_data.covered1_requirement = covered1_requirement
end
if type == Surfaces.enum.DOCK then
local BoatData = Boats.get_scope(boat).Data
starting_boatposition = Utils.snap_coordinates_for_rails({x = Dock.Data.playerboat_starting_xcoord, y = Dock.Data.top_boat_bottom - BoatData.height/2})
-- starting_boatposition = {x = -destination_data.static_params.width/2 + BoatData.width + 10, y = Dock.Data.top_boat_bottom - BoatData.height/2}
Common.current_destination().dynamic_data.time_remaining = 180
-- memory.mainshop_availability_bools.sell_iron = true
memory.mainshop_availability_bools.buy_iron = true
memory.mainshop_availability_bools.buy_copper = true
-- memory.mainshop_availability_bools.buy_fast_loader = true
-- memory.mainshop_availability_bools.sell_copper = true
local boat_for_sale_type = Common.current_destination().static_params.boat_for_sale_type
if boat_for_sale_type then
if boat_for_sale_type == Boats.enum.CUTTER then
memory.mainshop_availability_bools.new_boat_cutter = true
elseif boat_for_sale_type == Boats.enum.CUTTER_WITH_HOLD then
memory.mainshop_availability_bools.new_boat_cutter_with_hold = true
elseif boat_for_sale_type == Boats.enum.SLOOP_WITH_HOLD then
memory.mainshop_availability_bools.new_boat_sloop_with_hold = true
end
end
local upgrade_for_sale = Common.current_destination().static_params.upgrade_for_sale
if upgrade_for_sale then
for _, u in pairs(Upgrades.List) do
if upgrade_for_sale == u then
memory.mainshop_availability_bools[u] = true
end
end
end
2022-03-01 21:59:48 +00:00
memory.mainshop_availability_bools.repair_cannons = true
2021-10-13 09:21:53 +01:00
-- Delay.add(Delay.enum.PLACE_DOCK_JETTY_AND_BOATS)
Task.set_timeout_in_ticks(2, place_dock_jetty_and_boats, {})
else
starting_boatposition = {x = static_params.boat_starting_xposition, y = static_params.boat_starting_yposition or 0}
end
-- if oldsurface.name == CoreData.lobby_surface_name then
-- initial_boatspeed = 3
-- else
-- initial_boatspeed = 1.5
-- end
initial_boatspeed = 1.4
boat.speed = initial_boatspeed
boat.state = destination_data.init_boat_state
boat.dockedposition = nil
memory.enemyboats = {}
local old_water = 'deepwater'
if old_type == Surfaces.enum.LOBBY or old_type == Surfaces.enum.DOCK then old_water = 'water' end
Boats.teleport_boat(boat, newsurface_name, starting_boatposition, CoreData.moving_boat_floor, old_water)
if old_type == Surfaces.enum.LOBBY then
Crowsnest.draw_extra_bits()
end
Crowsnest.paint_around_destination(destination_index, CoreData.overworld_presence_tile)
if memory.loadingticks then memory.loadingticks = -120 end
if old_type == Surfaces.enum.SEA then
game.delete_surface(oldsurface)
end
memory.destinationsvisited_indices[#memory.destinationsvisited_indices + 1] = destination_index
memory.currentdestination_index = destination_index
local destination = Common.current_destination()
destination.dynamic_data.timer = 0
destination.dynamic_data.timeratlandingtime = nil
memory.extra_time_at_sea = 0
if old_type == Surfaces.enum.SEA or old_type == Surfaces.enum.CHANNEL or old_type == Surfaces.enum.DOCK then
-- move over anyone who was left behind, such as dead and spectating players
for _, player in pairs(game.connected_players) do
if type == Surfaces.enum.ISLAND and player.controller_type == defines.controllers.spectator then
if player.surface == oldsurface then --avoid moving players in hold etc
-- put them at a nice viewing position:
player.teleport({x = memory.spawnpoint.x + 120, y = memory.spawnpoint.y}, newsurface)
end
elseif player.surface == oldsurface then
player.teleport(memory.spawnpoint, newsurface)
end
end
end
Surfaces.destination_on_arrival(Common.current_destination())
end
function Public.check_for_end_of_boat_movement(boat)
local memory = Memory.get_crew_memory()
local destination = Common.current_destination()
local collided = Boats.collision_infront(boat)
local approaching_island = boat.state == Boats.enum_state.APPROACHING and destination.type == Surfaces.enum.ISLAND
local retreating_island = boat.state == Boats.enum_state.RETREATING and destination.type == Surfaces.enum.ISLAND
local approaching_dock = destination.type == Surfaces.enum.DOCK and boat.state == Boats.enum_state.APPROACHING
local leaving_dock = destination.type == Surfaces.enum.DOCK and boat.state == Boats.enum_state.LEAVING_DOCK
--=== Collision
if approaching_island and collided then
boat.state = Boats.enum_state.LANDED
boat.speed = 0
boat.dockedposition = boat.position
game.surfaces[boat.surface_name].play_sound{path = "utility/axe_fighting"}
game.surfaces[boat.surface_name].play_sound{path = "utility/axe_fighting"}
end
--=== Enemy
if boat.force_name == memory.enemy_force_name then
if approaching_island then
if collided then
boat.landing_time = destination.dynamic_data.timer
Boats.place_landingtrack(boat, CoreData.enemy_landing_tile)
if boat.unit_group and boat.unit_group.ref and boat.unit_group.ref.valid then boat.unit_group.ref.set_command({
type = defines.command.attack_area,
destination = ({memory.boat.position.x - 32, memory.boat.position.y} or {0,0}),
radius = 32,
distraction = defines.distraction.by_enemy
}) end
elseif boat.spawner and boat.spawner.valid and boat.spawner.destructible then
local boat2 = Utils.deepcopy(boat)
boat2.position = {x = boat.position.x + 5, y = boat.position.y}
if Boats.collision_infront(boat2) then
boat.spawner.destructible = false
end
end
end
--=== Friendly
elseif boat.force_name == memory.force_name then
if approaching_island and collided then
Surfaces.destination_on_crewboat_hits_shore(destination)
return true
elseif retreating_island and boat.position.x < ((boat.dockedposition.x or 999) - Boats.get_scope(boat).Data.width - 2 * Boats.get_scope(boat).Data.rightmost_gate_position - 8) then
Public.go_from_currentdestination_to_sea()
return true
elseif approaching_dock and boat.position.x + Boats.get_scope(boat).Data.rightmost_gate_position >= Dock.Data.rightmostgate_stopping_xposition then
boat.state = Boats.enum_state.DOCKED
boat.speed = 0
boat.dockedposition = boat.position
destination.dynamic_data.timeratlandingtime = destination.dynamic_data.timer
Boats.place_boat(boat, CoreData.static_boat_floor, false, false)
return true
elseif leaving_dock and boat.position.x >= game.surfaces[boat.surface_name].map_gen_settings.width/2 - 2 then
memory.mainshop_availability_bools.new_boat_cutter = false
memory.mainshop_availability_bools.new_boat_cutter_with_hold = false
memory.mainshop_availability_bools.new_boat_sloop_with_hold = false
-- memory.mainshop_availability_bools.sell_iron = false
memory.mainshop_availability_bools.buy_iron = false
memory.mainshop_availability_bools.buy_copper = false
-- memory.mainshop_availability_bools.buy_fast_loader = false
-- memory.mainshop_availability_bools.sell_copper = false
2022-03-01 21:59:48 +00:00
memory.mainshop_availability_bools.repair_cannons = false
2021-10-13 09:21:53 +01:00
memory.mainshop_availability_bools.extra_hold = false
memory.mainshop_availability_bools.upgrade_power = false
memory.mainshop_availability_bools.unlock_merchants = false
memory.mainshop_availability_bools.rockets_for_sale = false
Public.go_from_currentdestination_to_sea()
return true
--=== Fallthrough right-hand side
elseif destination.type == Surfaces.enum.ISLAND and boat.position.x >= game.surfaces[boat.surface_name].map_gen_settings.width/2 - 10 then
Public.go_from_currentdestination_to_sea()
return true
end
end
return false
end
function Public.try_retreat_from_island() -- Assumes the cost can be paid
local memory = Memory.get_crew_memory()
if memory.game_lost then return end
local destination = Common.current_destination()
local captain_index = memory.playerindex_captain
local captain = game.players[captain_index]
2021-10-14 10:23:34 +01:00
if captain and Common.validate_player(captain) and destination.dynamic_data.timeratlandingtime and destination.dynamic_data.timer < destination.dynamic_data.timeratlandingtime + 10 then
2022-02-27 16:42:25 +00:00
Common.notify_player_error(captain, 'Can\'t depart in the first 10 seconds.')
2021-10-13 09:21:53 +01:00
else
local cost = destination.static_params.cost_to_leave
-- if cost and (not destination.dynamic_data.rocketlaunched) then
-- local gold = memory.gold
-- local captain_index = memory.playerindex_captain
-- if not (gold and captain_index) then return end
-- local captain = game.players[captain_index]
-- if not Common.validate_player_and_character(captain) then return end
-- local captain_inv = captain.get_inventory(defines.inventory.character_main)
-- if not captain_inv then return end
-- local can_buy = true
-- if cost.name == 'gold' then
-- can_buy = gold >= cost.count
-- else
-- can_buy = captain_inv.get_item_count(cost.name) >= cost.count
-- end
-- if can_buy then
-- if cost.name == 'gold' then
-- memory.gold = memory.gold - cost.count
-- else
-- captain_inv.remove{name=cost.name, count=cost.count}
-- end
-- Public.retreat_from_island()
-- else
-- Common.notify_player(captain, 'Can\'t afford it.')
-- end
-- else
-- Public.retreat_from_island()
-- end
if cost then
Common.spend_stored_resources(cost)
end
Public.retreat_from_island()
end
end
function Public.retreat_from_island()
local memory = Memory.get_crew_memory()
local boat = memory.boat
if boat.state and boat.state == Boats.enum_state.RETREATING then return end
boat.state = Boats.enum_state.RETREATING
2022-03-03 00:19:20 +00:00
boat.speed = 1.25
2021-10-13 09:21:53 +01:00
Boats.place_boat(boat, CoreData.moving_boat_floor, false, false)
local force = game.forces[memory.force_name]
if not (force and force.valid) then return end
Common.notify_force_light(force,'[font=heading-1]Boat undocked.[/font]')
if memory.overworldx == 40*9 then Parrot.parrot_kraken_warning() end
end
function Public.undock_from_dock()
local memory = Memory.get_crew_memory()
local boat = memory.boat
local destination = Common.current_destination()
boat.state = Boats.enum_state.LEAVING_DOCK
destination.dynamic_data.time_remaining = -1
Boats.place_boat(boat, CoreData.moving_boat_floor, false, false)
memory.mainshop_availability_bools.new_boat_cutter = false
memory.mainshop_availability_bools.new_boat_cutter_with_hold = false
memory.mainshop_availability_bools.new_boat_sloop_with_hold = false
2022-02-24 19:39:03 +00:00
Crew.summon_crew()
2021-10-13 09:21:53 +01:00
local force = game.forces[memory.force_name]
if not (force and force.valid) then return end
Common.notify_force_light(force,'Leaving the dock.')
end
function Public.go_from_currentdestination_to_sea()
local memory = Memory.get_crew_memory()
local destination = Common.current_destination()
if memory.game_lost then return end
local oldsurface = game.surfaces[destination.surface_name]
Sea.ensure_sea_surface()
local seaname = memory.sea_name
local boat = memory.boat
local new_boatposition = Utils.snap_coordinates_for_rails({x = Boats.get_scope(memory.boat).Data.width / 2, y = 0})
Boats.teleport_boat(boat, seaname, new_boatposition, CoreData.static_boat_floor, 'water')
memory.boat.state = Boats.enum_state.ATSEA_SAILING
memory.boat.speed = 0
memory.boat.position = new_boatposition
memory.boat.surface_name = seaname
memory.loadingticks = nil
memory.mapbeingloadeddestination_index = nil
local d = destination.iconized_map_width + Crowsnest.platformwidth
Crowsnest.paint_around_destination(destination.destination_index, 'deepwater')
Overworld.try_overworld_move_v2{x = d, y = 0}
local players_marooned_count = 0
for _, player in pairs(game.connected_players) do
if (player.surface == oldsurface and player.character and player.character.valid) then
players_marooned_count = players_marooned_count + 1
end
end
if players_marooned_count == 0 then
Surfaces.clean_up(destination)
end
end
return Public