mirror of
https://github.com/ComfyFactory/ComfyFactorio.git
synced 2025-01-22 03:38:48 +02:00
0d9be8e524
Changes: - Added protected run checkbox when creating new run proposal. This locks captain role when captain becomes afk or leaves the afk. - Increased crew run limit from 2 to 3. - Public crew will no longer have captain protection when captain goes afk. - Fixed an exploit where player could join private run bypassing password protection. - Fixed an issue where spectator could take a class. - Adjusted some GUI elements to look nicer.
538 lines
16 KiB
Lua
538 lines
16 KiB
Lua
-- This file is part of thesixthroc's Pirate Ship softmod, licensed under GPLv3 and stored at https://github.com/danielmartin0/ComfyFactorio-Pirates.
|
|
|
|
|
|
local Memory = require 'maps.pirates.memory'
|
|
local Common = require 'maps.pirates.common'
|
|
local CoreData = require 'maps.pirates.coredata'
|
|
-- local Utils = require 'maps.pirates.utils_local'
|
|
local Math = require 'maps.pirates.math'
|
|
local _inspect = require 'utils.inspect'.inspect
|
|
local Crew = require 'maps.pirates.crew'
|
|
-- local Progression = require 'maps.pirates.progression'
|
|
-- local Structures = require 'maps.pirates.structures.structures'
|
|
local Shop = require 'maps.pirates.shop.shop'
|
|
local Boats = require 'maps.pirates.structures.boats.boats'
|
|
local Surfaces = require 'maps.pirates.surfaces.surfaces'
|
|
|
|
|
|
local Public = {}
|
|
|
|
|
|
Public.bold_font_color = {255, 230, 192}
|
|
Public.default_font_color = {1, 1, 1}
|
|
Public.section_header_font_color = {r=0.40, g=0.80, b=0.60}
|
|
Public.subsection_header_font_color = {229, 255, 242}
|
|
Public.friendly_font_color = {240, 200, 255}
|
|
Public.sufficient_font_color = {66, 220, 124}
|
|
Public.insufficient_font_color = {1, 0.62, 0.19}
|
|
Public.achieved_font_color = {227, 250, 192}
|
|
|
|
Public.fuel_color_1 = {r=255, g=255, b=255}
|
|
Public.fuel_color_2 = {r=255, g=0, b=60}
|
|
|
|
Public.rage_font_color_1 = {r=1, g=1, b=1}
|
|
Public.rage_font_color_2 = {r=1, g=0.5, b=0.1}
|
|
Public.rage_font_color_3 = {r=1, g=0.1, b=0.05}
|
|
|
|
|
|
|
|
Public.default_window_positions = {
|
|
runs = {x = 10, y = 48},
|
|
crew = {x = 40, y = 48},
|
|
progress = {x = 250, y = 48},
|
|
fuel = {x = 468, y = 48},
|
|
minimap = {x = 10, y = 48},
|
|
color = {x = 160, y = 96},
|
|
}
|
|
|
|
-- Player who created the proposal is already an endorser
|
|
function Public.proposal_endorsers_required()
|
|
return Math.min(4, Math.ceil((#game.connected_players or 0)/5))
|
|
end
|
|
|
|
function Public.new_window(player, name)
|
|
|
|
local global_memory = Memory.get_global_memory()
|
|
local gui_memory = global_memory.player_gui_memories[player.index]
|
|
local flow
|
|
|
|
flow = player.gui.screen.add{
|
|
type = 'frame',
|
|
name = name .. '_piratewindow',
|
|
direction = 'vertical'
|
|
}
|
|
|
|
if gui_memory and gui_memory[name] and gui_memory[name].position then
|
|
flow.location = gui_memory[name].position
|
|
else
|
|
flow.location = Public.default_window_positions[name]
|
|
end
|
|
|
|
flow.style = 'map_details_frame'
|
|
flow.style.minimal_width = 210
|
|
flow.style.natural_width = 210
|
|
flow.style.maximal_width = 270
|
|
flow.style.minimal_height = 80
|
|
flow.style.natural_height = 80
|
|
flow.style.maximal_height = 760
|
|
flow.style.padding = 10
|
|
|
|
return flow
|
|
end
|
|
|
|
|
|
function Public.update_gui_memory(player, namespace, key, value)
|
|
local global_memory = Memory.get_global_memory()
|
|
if not global_memory.player_gui_memories[player.index] then
|
|
global_memory.player_gui_memories[player.index] = {}
|
|
end
|
|
local gui_memory = global_memory.player_gui_memories[player.index]
|
|
if not gui_memory[namespace] then
|
|
gui_memory[namespace] = {}
|
|
end
|
|
gui_memory[namespace][key] = value
|
|
end
|
|
|
|
|
|
function Public.flow_add_floating_sprite_button(flow1, button_name, width)
|
|
width = width or 40
|
|
local flow2, flow3
|
|
|
|
flow2 = flow1.add({
|
|
name = button_name .. '_frame',
|
|
type = 'frame',
|
|
})
|
|
flow2.style.height = 40
|
|
flow2.style.margin = 0
|
|
flow2.style.left_padding = -4
|
|
flow2.style.top_padding = -4
|
|
flow2.style.right_margin = -2
|
|
flow2.style.width = width
|
|
|
|
flow3 = flow2.add({
|
|
name = button_name,
|
|
type = 'sprite-button',
|
|
})
|
|
flow3.style.height = 40
|
|
flow3.style.width = width
|
|
-- flow3.style.padding = -4
|
|
|
|
return flow3
|
|
end
|
|
|
|
|
|
|
|
|
|
function Public.flow_add_floating_button(flow1, button_name)
|
|
local flow2, flow3
|
|
|
|
flow2 = flow1.add({
|
|
name = button_name .. '_flow_1',
|
|
type = 'flow',
|
|
direction = 'vertical',
|
|
})
|
|
flow2.style.height = 40
|
|
-- flow2.style.left_padding = 4
|
|
-- flow2.style.top_padding = 0
|
|
-- flow2.style.right_margin = -2
|
|
flow2.style.natural_width = 40
|
|
|
|
flow3 = flow2.add({
|
|
name = button_name,
|
|
type = 'button',
|
|
})
|
|
flow3.style = 'dark_rounded_button'
|
|
-- flow3.style.minimal_width = 60
|
|
-- flow3.style.natural_width = 60
|
|
flow3.style.minimal_height = 40
|
|
flow3.style.maximal_height = 40
|
|
flow3.style.left_padding = 10
|
|
flow3.style.right_padding = 4
|
|
flow3.style.top_padding = 3
|
|
-- flow3.style.padding = -4
|
|
flow3.style.natural_width = 40
|
|
flow3.style.horizontally_stretchable = true
|
|
|
|
flow3 = flow2.add({
|
|
name = button_name .. '_flow_2',
|
|
type = 'flow',
|
|
})
|
|
flow3.style.natural_width = 20
|
|
flow3.style.top_margin = -37
|
|
flow3.style.left_margin = 10
|
|
flow3.style.right_margin = 9
|
|
flow3.ignored_by_interaction=true
|
|
|
|
return flow3
|
|
end
|
|
|
|
|
|
|
|
function Public.flow_add_shop_item(flow, name)
|
|
local flow2, flow3
|
|
|
|
local shop_data_1 = Shop.Captains.main_shop_data_1
|
|
local shop_data_2 = Shop.Captains.main_shop_data_2
|
|
local trade_data = shop_data_1[name] or shop_data_2[name]
|
|
if not trade_data then return end
|
|
|
|
flow2 = flow.add({
|
|
name = name,
|
|
type = 'flow',
|
|
direction = 'horizontal',
|
|
})
|
|
flow2.style.top_margin = 3
|
|
flow2.style.horizontal_align = 'center'
|
|
flow2.style.vertical_align = 'center'
|
|
flow2.tooltip = trade_data.tooltip
|
|
|
|
|
|
for k, v in pairs(trade_data.what_you_get_sprite_buttons) do
|
|
flow3 = flow2.add({
|
|
type = 'sprite-button',
|
|
name = k,
|
|
sprite = k,
|
|
enabled = false,
|
|
})
|
|
flow3.style.minimal_height = 40
|
|
flow3.style.maximal_height = 40
|
|
if v == false then
|
|
flow3.number = nil
|
|
else
|
|
flow3.number = v
|
|
end
|
|
flow3.tooltip = trade_data.tooltip
|
|
end
|
|
|
|
flow3 = flow2.add({
|
|
type = 'label',
|
|
name = 'for',
|
|
caption = 'for'
|
|
})
|
|
flow3.style.font = 'default-large'
|
|
flow3.style.font_color = Public.default_font_color
|
|
flow3.tooltip = trade_data.tooltip
|
|
|
|
for k, _ in pairs(trade_data.base_cost) do
|
|
flow3 = flow2.add({
|
|
name = 'cost_' .. k,
|
|
type = 'sprite-button',
|
|
enabled = false,
|
|
})
|
|
flow3.style.minimal_height = 40
|
|
flow3.style.maximal_height = 40
|
|
flow3.tooltip = trade_data.tooltip
|
|
if k == 'fuel' then
|
|
flow3.sprite = 'item/coal'
|
|
elseif k == 'coins' then
|
|
flow3.sprite = 'item/coin'
|
|
elseif k == 'iron_plates' then
|
|
flow3.sprite = 'item/iron-plate'
|
|
elseif k == 'copper_plates' then
|
|
flow3.sprite = 'item/copper-plate'
|
|
end
|
|
end
|
|
|
|
|
|
flow3 = flow2.add({
|
|
name = 'spacing',
|
|
type = 'flow',
|
|
direction = 'horizontal',
|
|
})
|
|
flow3.style.horizontally_stretchable = true
|
|
|
|
flow3 = flow2.add({
|
|
type = 'sprite-button',
|
|
name = 'buy_button',
|
|
caption = 'Buy'
|
|
})
|
|
flow3.style.font = 'default-large'
|
|
flow3.style.font_color = Public.default_font_color
|
|
flow3.style.height = 32
|
|
flow3.style.width = 50
|
|
flow3.style.padding = 0
|
|
flow3.style.margin = 0
|
|
|
|
return flow2
|
|
end
|
|
|
|
|
|
function Public.flow_add_section(flow, name, caption)
|
|
local flow2, flow3
|
|
|
|
flow2 = flow.add({
|
|
name = name,
|
|
type = 'flow',
|
|
direction = 'vertical',
|
|
})
|
|
flow2.style.bottom_margin = 5
|
|
|
|
flow3 = flow2.add({
|
|
type = 'label',
|
|
name = 'header',
|
|
caption = caption
|
|
})
|
|
flow3.style.font = 'heading-2'
|
|
flow3.style.font_color = Public.section_header_font_color
|
|
flow3.style.maximal_width = 300
|
|
-- flow3.style.maximal_width = 220
|
|
-- flow3.style.single_line = false
|
|
|
|
flow3 = flow2.add({
|
|
name = 'body',
|
|
type = 'flow',
|
|
direction = 'vertical',
|
|
})
|
|
flow3.style.left_margin = 5
|
|
flow3.style.right_margin = 5
|
|
|
|
return flow3
|
|
end
|
|
|
|
|
|
function Public.flow_add_subpanel(flow, name)
|
|
local flow2
|
|
|
|
flow2 = flow.add({
|
|
name = name,
|
|
type = 'frame',
|
|
direction = 'vertical',
|
|
})
|
|
flow2.style = 'subpanel_frame'
|
|
flow2.style.natural_width = 100
|
|
flow2.style.top_padding = -2
|
|
flow2.style.top_margin = -8
|
|
|
|
return flow2
|
|
end
|
|
|
|
|
|
|
|
function Public.flow_add_close_button(flow, close_button_name)
|
|
local flow2, flow3, flow4
|
|
|
|
flow2 = flow.add({
|
|
name = 'close_button_flow',
|
|
type = 'flow',
|
|
direction = 'vertical',
|
|
})
|
|
|
|
flow3 = flow2.add{type="flow", name='hflow', direction="horizontal"}
|
|
flow3.style.vertical_align = 'center'
|
|
|
|
flow4 = flow3.add{type="flow", name='spacing', direction="horizontal"}
|
|
flow4.style.horizontally_stretchable = true
|
|
|
|
flow4 = flow3.add({
|
|
type = 'button',
|
|
name = close_button_name,
|
|
caption = {'pirates.gui_close_button'},
|
|
})
|
|
flow4.style = 'back_button'
|
|
flow4.style.minimal_width = 90
|
|
flow4.style.font = 'default-bold'
|
|
flow4.style.height = 28
|
|
flow4.style.horizontal_align = 'center'
|
|
|
|
return flow3
|
|
end
|
|
|
|
|
|
function Public.crew_overall_state_bools(player_index)
|
|
local global_memory = Memory.get_global_memory()
|
|
local memory = Memory.get_crew_memory()
|
|
|
|
--*** PLAYER STATUS ***--
|
|
|
|
local ret = {
|
|
adventuring = false,
|
|
spectating = false,
|
|
endorsing = false,
|
|
proposing = false,
|
|
sloops_full = false,
|
|
needs_more_capacity = false,
|
|
crew_count_capped = false,
|
|
needs_more_endorsers = false,
|
|
leaving = false,
|
|
proposal_can_launch = false,
|
|
}
|
|
if memory.crewstatus == Crew.enum.ADVENTURING then
|
|
for _, playerindex in pairs(memory.crewplayerindices) do
|
|
if player_index == playerindex then ret.adventuring = true end
|
|
end
|
|
for _, playerindex in pairs(memory.spectatorplayerindices) do
|
|
if player_index == playerindex then ret.spectating = true end
|
|
end
|
|
end
|
|
if memory.crewstatus == nil then
|
|
for _, crewid in pairs(global_memory.crew_active_ids) do
|
|
if global_memory.crew_memories[crewid].crewstatus == Crew.enum.LEAVING_INITIAL_DOCK then
|
|
for _, endorser_index in pairs(global_memory.crew_memories[crewid].original_proposal.endorserindices) do
|
|
if endorser_index == player_index then ret.leaving = true end
|
|
end
|
|
end
|
|
end
|
|
for _, proposal in pairs(global_memory.crewproposals) do
|
|
if #proposal.endorserindices > 0 and proposal.endorserindices[1] == player_index then
|
|
ret.proposing = true
|
|
if #global_memory.crew_active_ids >= 3 then
|
|
ret.sloops_full = true
|
|
elseif #global_memory.crew_active_ids >= global_memory.active_crews_cap then
|
|
ret.crew_count_capped = true
|
|
elseif global_memory.active_crews_cap > 1 and #global_memory.crew_active_ids == (global_memory.active_crews_cap - 1) and not ((global_memory.crew_memories[1] and global_memory.crew_memories[1].capacity >= Common.minimum_run_capacity_to_enforce_space_for) or (global_memory.crew_memories[2] and global_memory.crew_memories[2].capacity >= Common.minimum_run_capacity_to_enforce_space_for) or (global_memory.crew_memories[3] and global_memory.crew_memories[3].capacity >= Common.minimum_run_capacity_to_enforce_space_for)) and not (CoreData.capacity_options[proposal.capacity_option].value >= Common.minimum_run_capacity_to_enforce_space_for) then
|
|
ret.needs_more_capacity = true
|
|
elseif proposal.endorserindices and #global_memory.crew_active_ids > 0 and #proposal.endorserindices < Public.proposal_endorsers_required() then
|
|
ret.needs_more_endorsers = true
|
|
end
|
|
if (not (ret.sloops_full or ret.needs_more_capacity or ret.needs_more_endorsers or ret.crew_count_capped)) then
|
|
ret.proposal_can_launch = true
|
|
end
|
|
end
|
|
for _, i in pairs(proposal.endorserindices) do
|
|
if player_index == i then ret.endorsing = true end
|
|
end
|
|
end
|
|
end
|
|
|
|
return ret
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
function Public.player_and_crew_state_bools(player)
|
|
local memory = Memory.get_crew_memory()
|
|
|
|
-- if memory.player_and_crew_state_bools_memoized and memory.player_and_crew_state_bools_memoized.tick > game.tick - 30 then -- auto-memoize and only update every half-second
|
|
-- return memory.player_and_crew_state_bools_memoized.ret
|
|
-- else
|
|
local destination = Common.current_destination()
|
|
local dynamic_data = destination.dynamic_data --assumes this always exists
|
|
|
|
local in_crowsnest_bool, in_hold_bool, in_cabin_bool, onmap_bool, eta_bool, approaching_bool, retreating_bool, atsea_sailing_bool, landed_bool, quest_bool, silo_bool, charged_bool, launched_bool, captain_bool, atsea_loading_bool, atsea_waiting_bool, character_on_deck_bool, on_deck_standing_near_loco_bool, on_deck_standing_near_cabin_bool, on_deck_standing_near_crowsnest_bool, cost_bool, cost_includes_rocket_launch_bool, approaching_dock_bool, leaving_dock_bool, leave_anytime_bool
|
|
|
|
captain_bool = Common.is_captain(player)
|
|
|
|
in_crowsnest_bool = string.sub(player.surface.name, 9, 17) == 'Crowsnest'
|
|
in_hold_bool = string.sub(player.surface.name, 9, 12) == 'Hold'
|
|
in_cabin_bool = string.sub(player.surface.name, 9, 13) == 'Cabin'
|
|
|
|
onmap_bool = destination.surface_name and (player.surface.name == destination.surface_name or (
|
|
memory.boat and memory.boat.surface_name == destination.surface_name and (in_crowsnest_bool or in_hold_bool or in_cabin_bool)
|
|
))
|
|
|
|
if destination then
|
|
eta_bool = dynamic_data.time_remaining and dynamic_data.time_remaining > 0 and onmap_bool
|
|
approaching_bool = memory.boat and memory.boat.state == Boats.enum_state.APPROACHING and onmap_bool
|
|
retreating_bool = memory.boat and memory.boat.state == Boats.enum_state.RETREATING and onmap_bool
|
|
-- approaching_bool = memory.boat and memory.boat.state == Boats.enum_state.APPROACHING
|
|
atsea_sailing_bool = memory.boat and memory.boat.state == Boats.enum_state.ATSEA_SAILING
|
|
atsea_waiting_bool = memory.boat and memory.boat.state == Boats.enum_state.ATSEA_WAITING_TO_SAIL
|
|
landed_bool = memory.boat and memory.boat.state == Boats.enum_state.LANDED
|
|
quest_bool = (dynamic_data.quest_type ~= nil) and onmap_bool
|
|
charged_bool = dynamic_data.silocharged
|
|
silo_bool = dynamic_data.rocketsilos and onmap_bool and ((dynamic_data.rocketsilos[1] and dynamic_data.rocketsilos[1].valid) or charged_bool)
|
|
launched_bool = dynamic_data.rocketlaunched
|
|
|
|
cost_bool = destination.static_params.base_cost_to_undock and (not atsea_sailing_bool) and (not atsea_waiting_bool) and (not retreating_bool)
|
|
cost_includes_rocket_launch_bool = cost_bool and destination.static_params.base_cost_to_undock['launch_rocket']
|
|
|
|
leave_anytime_bool = (landed_bool and not (eta_bool or cost_bool))
|
|
end
|
|
|
|
if memory.boat then
|
|
atsea_loading_bool = memory.boat.state == Boats.enum_state.ATSEA_LOADING_MAP and memory.loadingticks
|
|
|
|
character_on_deck_bool = player.character and player.character.position and player.surface.name and player.surface.name == memory.boat.surface_name
|
|
|
|
if character_on_deck_bool then
|
|
local BoatData = Boats.get_scope(memory.boat).Data
|
|
|
|
on_deck_standing_near_loco_bool = Math.distance(player.character.position, Math.vector_sum(memory.boat.position, BoatData.loco_pos)) < 2.5
|
|
|
|
on_deck_standing_near_cabin_bool = Math.distance(player.character.position, Math.vector_sum(memory.boat.position, BoatData.cabin_car)) < 2.0
|
|
|
|
on_deck_standing_near_crowsnest_bool = Math.distance(player.character.position, Math.vector_sum(memory.boat.position, BoatData.crowsnest_center)) < 2.5
|
|
end
|
|
|
|
approaching_dock_bool = destination.type == Surfaces.enum.DOCK and memory.boat.state == Boats.enum_state.APPROACHING
|
|
leaving_dock_bool = destination.type == Surfaces.enum.DOCK and memory.boat.state == Boats.enum_state.LEAVING_DOCK
|
|
end
|
|
|
|
local ret = {
|
|
in_crowsnest_bool = in_crowsnest_bool,
|
|
in_hold_bool = in_hold_bool,
|
|
in_cabin_bool = in_cabin_bool,
|
|
-- onmap_bool = onmap_bool,
|
|
eta_bool = eta_bool,
|
|
approaching_bool = approaching_bool,
|
|
retreating_bool = retreating_bool,
|
|
atsea_sailing_bool = atsea_sailing_bool,
|
|
atsea_waiting_bool = atsea_waiting_bool,
|
|
-- landed_bool = landed_bool,
|
|
quest_bool = quest_bool,
|
|
silo_bool = silo_bool,
|
|
charged_bool = charged_bool,
|
|
launched_bool = launched_bool,
|
|
captain_bool = captain_bool,
|
|
atsea_loading_bool = atsea_loading_bool,
|
|
-- character_on_deck_bool = character_on_deck_bool,
|
|
on_deck_standing_near_loco_bool = on_deck_standing_near_loco_bool,
|
|
on_deck_standing_near_cabin_bool = on_deck_standing_near_cabin_bool,
|
|
on_deck_standing_near_crowsnest_bool = on_deck_standing_near_crowsnest_bool,
|
|
cost_bool = cost_bool,
|
|
cost_includes_rocket_launch_bool = cost_includes_rocket_launch_bool,
|
|
approaching_dock_bool = approaching_dock_bool,
|
|
leaving_dock_bool = leaving_dock_bool,
|
|
leave_anytime_bool = leave_anytime_bool
|
|
}
|
|
|
|
-- memory.player_and_crew_state_bools_memoized = {ret = ret, tick = game.tick}
|
|
|
|
return ret
|
|
-- end
|
|
end
|
|
|
|
|
|
|
|
function Public.update_listbox(listbox, table)
|
|
-- pass a table of strings of the form {'locale', unique_id, ...}
|
|
|
|
-- remove any that shouldn't be there
|
|
local marked_for_removal = {}
|
|
for index, item in pairs(listbox.items) do
|
|
local exists = false
|
|
for _, i in pairs(table) do
|
|
if tostring(i[2]) == item[2] then
|
|
exists = true
|
|
end
|
|
end
|
|
if exists == false then
|
|
marked_for_removal[#marked_for_removal + 1] = index
|
|
end
|
|
end
|
|
for i = #marked_for_removal, 1, -1 do
|
|
listbox.remove_item(marked_for_removal[i])
|
|
end
|
|
|
|
local indexalreadyat
|
|
for _, i in pairs(table) do
|
|
local contained = false
|
|
for index, item in pairs(listbox.items) do
|
|
if tostring(i[2]) == item[2] then
|
|
contained = true
|
|
indexalreadyat = index
|
|
end
|
|
end
|
|
|
|
if contained then
|
|
listbox.set_item(indexalreadyat, i)
|
|
else
|
|
listbox.add_item(i)
|
|
end
|
|
end
|
|
end
|
|
|
|
|
|
return Public |