1
0
mirror of https://github.com/ComfyFactory/ComfyFactorio.git synced 2025-01-06 00:23:49 +02:00
ComfyFactorio/modules/autofill.lua
Gerkiz 0552cb3da9 soft_modded autofill
Works either globally or per player
2020-09-04 17:04:46 +02:00

554 lines
14 KiB
Lua

local Color = require 'utils.color_presets'
local Event = require 'utils.event'
local Global = require 'utils.global'
local this = {
refill_turrets = {},
refill_chests = {index = 1, placed = 0},
full_turrets = {},
valid_chest = {
['iron-chest'] = {valid = true, limit = 4}
},
valid_turrets = {
['gun-turret'] = true,
['artillery-turret'] = true
},
valid_ammo = {
['firearm-magazine'] = {valid = true, priority = 1},
['piercing-rounds-magazine'] = {valid = true, priority = 2},
['uranium-rounds-magazine'] = {valid = true, priority = 3}
},
message_limit = {},
player_settings = {},
globally_enabled = true
}
Global.register(
this,
function(t)
this = t
end
)
local Public = {}
local insert = table.insert
local autofill_amount = 10
local valid_chest = this.valid_chest
local valid_turrets = this.valid_turrets
local valid_ammo = this.valid_ammo
local function validate_entity(entity)
if not entity then
return false
end
if not entity.valid then
return false
end
return true
end
local function fast_remove(tbl, index)
local count = #tbl
if index > count then
return
elseif index < count then
tbl[index] = tbl[count]
end
tbl[count] = nil
end
local function get_player_data(player, remove_user_data)
if this.globally_enabled then
return
end
if remove_user_data then
if this.player_settings[player.index] then
this.player_settings[player.index] = nil
end
end
if not this.player_settings[player.index] then
this.player_settings[player.index] = {
placed = 0,
chests = {},
turrets = {},
enabled = true
}
end
return this.player_settings[player.index]
end
local function contains(table, entity)
if not table then
return
end
for k, turret in pairs(table) do
if not validate_entity(turret) then
return false
end
if not validate_entity(entity) then
return false
end
if turret.unit_number == entity.unit_number then
return true
end
end
end
local function get_highest(table)
local highest = -math.huge
local item
local count
for k, v in pairs(table) do
if (k and valid_ammo[k] and valid_ammo[k].priority > highest) then
item = k
count = v
end
end
if not item or not count then
return false
end
return item, count
end
local function get_valid_chest()
if this.globally_enabled then
local refill_chests = this.refill_chests
if not next(refill_chests) then
return
end
local chest
local chests = {}
for i = 1, #refill_chests do
chest = refill_chests[i]
if chest then
chests[#chests + 1] = chest
end
if not chest.valid then
fast_remove(refill_chests, i)
refill_chests.placed = refill_chests.placed - 1
if refill_chests.placed <= 0 then
refill_chests.placed = 0
end
return false
end
end
return chests
end
local player_data = this.player_settings
if not next(player_data) then
return
end
local chest
for i = 1, #player_data do
local player = game.get_player(i)
if player and player.valid then
local p_data = get_player_data(player)
local chests = p_data.chests
if not next(chests) then
return
end
for t = 1, #chests do
chest = chests[t]
if chest then
return chest
elseif not chest.valid then
fast_remove(chests, i)
p_data.placed = p_data.placed - 1
if p_data.placed <= 0 then
p_data.placed = 0
end
return
end
end
end
end
end
local function get_ammo(entity_turret)
local turret = entity_turret.get_inventory(defines.inventory.turret_ammo)
local contents = turret.get_contents()
local c = 0
for item, count in pairs(contents) do
if valid_ammo[item] and valid_ammo[item].valid and count >= 1 then
c = count
return item, c
end
end
return false, c
end
local function get_items(chest)
local contents = chest.get_contents()
local item, count = get_highest(contents)
if valid_ammo[item] and valid_ammo[item].valid and count >= 1 then
return item, count
end
end
local function remove_ammo(chest, entity_turret)
local turret = entity_turret.get_inventory(defines.inventory.turret_ammo)
local current_ammo
if not chest or not chest.valid then
return
end
local contents = turret.get_contents()
for item, count in pairs(contents) do
if count >= 1 then
local t = {name = item, count = count}
if chest.can_insert(t) then
local c = chest.insert(t)
current_ammo = item
turret.remove({name = item, count = c})
return current_ammo
end
end
end
end
local function refill(entity_turret, entity_chest)
local turret = entity_turret
for _, chests in next, entity_chest do
local chest = chests.get_inventory(defines.inventory.chest)
local item, count = get_items(chest)
local turret_inv = turret.get_inventory(defines.inventory.turret_ammo)
if valid_ammo[item] and valid_ammo[item].valid and count >= 1 then
local ammo_name, ammo_count = get_ammo(turret)
if ammo_name and valid_ammo[ammo_name].priority < valid_ammo[item].priority then
remove_ammo(chest, turret)
end
if ammo_count and ammo_count >= 10 then
goto continue
end
local t = {name = item, count = 1}
local c = turret_inv.insert(t)
if (c > 0) then
chest.remove({name = item, count = c})
end
::continue::
end
end
end
local function do_refill_turrets()
local chest = get_valid_chest()
if not chest then
return
end
if this.globally_enabled then
local turrets = this.refill_turrets
if not next(turrets) then
return
end
for i = 1, #turrets do
local turret = turrets[i]
if not turret then
return
end
if not turret.valid then
fast_remove(turrets, i)
return
end
refill(turret, chest)
end
return
end
local player_data = this.player_settings
if not next(player_data) then
return
end
for i = 1, #player_data do
local player = game.get_player(i)
if player and player.valid then
local turrets = player_data[i].turrets
if not next(turrets) then
return
end
for t = 1, #turrets do
local turret = turrets[t]
if not turret then
return
end
if not turret.valid then
fast_remove(turrets, t)
return
end
refill(turret, chest)
end
end
end
end
local function show_text(msg, pos, color, surface)
if color == nil then
surface.create_entity({name = 'flying-text', position = pos, text = msg})
else
surface.create_entity({name = 'flying-text', position = pos, text = msg, color = color})
end
end
local function move_items(source, destination, stack)
if (source.get_item_count(stack.name) == 0) then
return -1
end
if (not destination.can_insert(stack)) then
return -2
end
local itemsRemoved = source.remove(stack)
stack.count = itemsRemoved
return destination.insert(stack)
end
local function move_multiple(source, destination, stack, amount)
local ret = 0
for _, itemName in pairs(stack) do
ret = move_items(source, destination, {name = itemName, count = amount})
if (ret > 0) then
return ret
end
end
return ret
end
local function auto_insert_into_turret(player, turret)
local inventory = player.get_main_inventory()
if (inventory == nil) then
return
end
local ret =
move_multiple(
inventory,
turret,
{'artillery-shell', 'uranium-rounds-magazine', 'piercing-rounds-magazine', 'firearm-magazine'},
autofill_amount
)
if (ret > 1) then
show_text('[Autofill] Inserted ' .. ret .. '!', turret.position, Color.info, player.surface)
elseif (ret == -1) then
show_text('[Autofill] Out of ammo!', turret.position, Color.red, player.surface)
elseif (ret == -2) then
show_text('[Autofill] Autofill ERROR! - Report this bug!', turret.position, Color.red, player.surface)
end
end
local function auto_insert_into_vehicle(player, vehicle)
local inventory = player.get_main_inventory()
if (inventory == nil) then
return
end
if ((vehicle.name == 'car') or (vehicle.name == 'tank') or (vehicle.name == 'locomotive')) then
move_multiple(inventory, vehicle, {'nuclear-fuel', 'rocket-fuel', 'solid-fuel', 'coal', 'wood'}, 50)
end
if ((vehicle.name == 'car') or (vehicle.name == 'tank')) then
move_multiple(
inventory,
vehicle,
{'uranium-rounds-magazine', 'piercing-rounds-magazine', 'firearm-magazine'},
autofill_amount
)
end
if (vehicle.name == 'tank') then
move_multiple(
inventory,
vehicle,
{'explosive-uranium-cannon-shell', 'uranium-cannon-shell', 'explosive-cannon-shell', 'cannon-shell'},
autofill_amount
)
end
end
local function on_entity_built(event)
local player = game.players[event.player_index]
if not (player and player.valid) then
return
end
local ce = event.created_entity
if not (ce and ce.valid) then
return
end
if (valid_chest[ce.name] and valid_chest[ce.name].valid) then
if this.globally_enabled then
if (this.refill_chests.placed < valid_chest[ce.name].limit) then
if this.message_limit[player.index] then
this.message_limit[player.index] = nil
end
Public.add_chest_to_refill_callback(nil, ce)
else
if not this.message_limit[player.index] then
this.message_limit[player.index] = true
player.print('[Autofill] Chest limit reached.', Color.warning)
end
end
else
local player_data = get_player_data(player)
if (player_data.placed < valid_chest[ce.name].limit) then
if this.message_limit[player.index] then
this.message_limit[player.index] = nil
end
Public.add_chest_to_refill_callback(player, ce)
else
if not this.message_limit[player.index] then
this.message_limit[player.index] = true
player.print('[Autofill] Chest limit reached.', Color.warning)
end
end
end
end
if (valid_turrets[ce.name]) then
if this.globally_enabled then
Public.refill_turret_callback(nil, ce)
else
Public.refill_turret_callback(player, ce)
end
auto_insert_into_turret(player, ce)
end
if ((ce.name == 'car') or (ce.name == 'tank') or (ce.name == 'locomotive')) then
auto_insert_into_vehicle(player, ce)
end
end
local function on_pre_player_mined_item(event)
local player = game.get_player(event.player_index)
if not validate_entity(player) then
return
end
local entity = event.entity
if not validate_entity(entity) then
return
end
if not valid_turrets[entity.name] then
return
end
if this.globally_enabled then
local chest = get_valid_chest()
if not chest then
return
end
local refill_turrets = this.refill_turrets
local full_turrets = this.full_turrets
if contains(refill_turrets, entity) or contains(full_turrets, entity) then
remove_ammo(chest, entity)
end
return
end
local chest = get_valid_chest(player)
if not chest then
return
end
local refill_turrets = this.refill_turrets
local full_turrets = this.full_turrets
if contains(refill_turrets, entity) or contains(full_turrets, entity) then
remove_ammo(chest, entity)
end
end
local function on_tick()
do_refill_turrets()
end
Public.refill_turret_callback = function(player, turret)
if this.globally_enabled then
local refill_turrets = this.refill_turrets
if turret and turret.valid then
refill_turrets[#refill_turrets + 1] = turret
end
return
end
if turret and turret.valid then
local player_data = get_player_data(player)
local turrets = player_data.turrets
insert(turrets, turret)
end
end
Public.add_chest_to_refill_callback = function(player, entity)
if entity and entity.valid then
if this.globally_enabled then
local refill_chests = this.refill_chests
refill_chests[#refill_chests + 1] = entity
refill_chests.placed = refill_chests.placed + 1
else
local player_data = get_player_data(player)
local chest_placed = player_data.placed
local chests = player_data.chests
insert(chests, entity)
player_data.placed = chest_placed + 1
end
rendering.draw_text {
text = '',
surface = entity.surface,
target = entity,
target_offset = {0, -0.5},
scale = 1.5,
color = {r = 0, g = 0.6, b = 1},
alignment = 'center'
}
end
end
Event.add(defines.events.on_built_entity, on_entity_built)
Event.add(defines.events.on_pre_player_mined_item, on_pre_player_mined_item)
Event.on_nth_tick(25, on_tick)
return Public