mirror of
https://github.com/Refactorio/RedMew.git
synced 2025-01-18 03:21:47 +02:00
commit
9225b154df
18
config.lua
18
config.lua
@ -339,6 +339,24 @@ global.config = {
|
||||
-- gradually informs players of features such as chat, toasts, etc.
|
||||
player_onboarding = {
|
||||
enabled = true
|
||||
},
|
||||
-- allows for large-scale biter attacks
|
||||
biter_attacks = {
|
||||
enabled = true,
|
||||
-- whether or not to send attacks on timed intervals (against a random player)
|
||||
timed_attacks = {
|
||||
enabled = true,
|
||||
-- frequency of automatic attacks (in seconds)
|
||||
attack_frequency = 40 * 60, -- 40 minutes
|
||||
-- difficulty of automatic attacks (1-easy, 3-normal, 10-hard, 40-brutal)
|
||||
attack_difficulty = 3
|
||||
},
|
||||
-- whether or not to send attacks on rocket launches
|
||||
launch_attacks = {
|
||||
enabled = true,
|
||||
-- whether to only attack on the first launch
|
||||
first_launch_only = true
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -89,6 +89,9 @@ end
|
||||
if config.player_onboarding.enabled then
|
||||
require 'features.player_onboarding'
|
||||
end
|
||||
if config.biter_attacks then
|
||||
require 'map_gen.shared.biter_attacks'
|
||||
end
|
||||
|
||||
-- GUIs
|
||||
-- The order determines the order they appear from left to right.
|
||||
|
@ -104,3 +104,8 @@ mention_not_found_singular=__1__ Player not found: __2__
|
||||
[player_onboarding]
|
||||
teach_toast=This is a feature in RedMew called a toast!\nThey're little pieces of information that pop up.\nYou can dismiss it by waiting or by clicking on it.
|
||||
teach_chat=To chat with other players, press the __CONTROL__toggle-console__ key on your keyboard.\nThe default key on English keyboards is the grave (`) and is below the ESC key.\nThis key can be changed in __1__ -> __2__ -> __3__.
|
||||
|
||||
[biter_attacks]
|
||||
first_rocket_launch_attack=The ground rumbles as the first rocket is launched.. But there's more.. The rocket is out of sight but the ground continues to tremble. An attack is coming, bigger than any previous.
|
||||
rocket_launch_attack=The locals launch another attack as our next rocket takes off.
|
||||
biter_command_success=An attack against __1__ has been ordered.
|
||||
|
395
map_gen/shared/biter_attacks.lua
Normal file
395
map_gen/shared/biter_attacks.lua
Normal file
@ -0,0 +1,395 @@
|
||||
--[[
|
||||
This module allows for massive biter attacks without inflicting lag spikes by scanning large areas at once.
|
||||
There are options in the config to regulate the following:
|
||||
Attacks sent against random players on timed intervals.
|
||||
Attacks sent on the launch of rockets.
|
||||
|
||||
There is a command available in admin_commands.
|
||||
There is also a public function allowing the sending of attacks by other modules.
|
||||
Public.launch_attack takes a table of arguments and will send an attack against a specific entity or position.
|
||||
]]
|
||||
-- Dependencies
|
||||
local Global = require 'utils.global'
|
||||
local Task = require 'utils.task'
|
||||
local Token = require 'utils.token'
|
||||
local table = require 'utils.table'
|
||||
local Event = require 'utils.event'
|
||||
local Game = require 'utils.game'
|
||||
local Command = require 'utils.command'
|
||||
local RS = require 'map_gen.shared.redmew_surface'
|
||||
local Ranks = require 'resources.ranks'
|
||||
local Color = require 'resources.color_presets'
|
||||
|
||||
local config = global.config.biter_attacks -- The local copy of config should only be used during the control stage
|
||||
|
||||
-- Localized functions
|
||||
local random = math.random
|
||||
local insert = table.insert
|
||||
local ceil = math.ceil
|
||||
|
||||
-- Constants
|
||||
local defaults = {
|
||||
total_scan_radius = 5000,
|
||||
individual_scan_radius = 500 -- a 500 radius scan is < 0.5ms on avg
|
||||
}
|
||||
|
||||
-- Local vars
|
||||
local timed_attack_token
|
||||
local setup_scans_token
|
||||
local biter_scan_token
|
||||
local Public = {}
|
||||
|
||||
-- Global tokens
|
||||
local Attack_data = {
|
||||
attack_lockout = nil,
|
||||
enemy_unit_group = nil,
|
||||
scan_index = 1,
|
||||
biter_count = 0,
|
||||
attack_pos = nil,
|
||||
force_name = nil
|
||||
}
|
||||
|
||||
Global.register(
|
||||
{
|
||||
Attack_data = Attack_data
|
||||
},
|
||||
function(tbl)
|
||||
Attack_data = tbl.Attack_data
|
||||
end
|
||||
)
|
||||
|
||||
-- Local functions
|
||||
|
||||
--- Cleans the primitive data for a new attack
|
||||
local function init_data(surface, scan_center)
|
||||
if Attack_data.enemy_unit_group then
|
||||
Attack_data.enemy_unit_group.destroy()
|
||||
Attack_data.enemy_unit_group = nil
|
||||
end
|
||||
|
||||
if not surface or not surface.valid then
|
||||
return
|
||||
end
|
||||
|
||||
Attack_data.enemy_unit_group = surface.create_unit_group {position = scan_center}
|
||||
Attack_data.scan_index = 1
|
||||
Attack_data.biter_count = 0
|
||||
end
|
||||
|
||||
--- Calculates the number of biters to send for timed attacks according to the difficulty selected
|
||||
-- @return <number>
|
||||
local function calculate_biters()
|
||||
local multiplier = global.config.biter_attacks.timed_attacks.attack_difficulty
|
||||
return ceil((game.forces.enemy.evolution_factor * 100 * multiplier))
|
||||
end
|
||||
|
||||
--- Take a large scan radius and break it into smaller pieces
|
||||
-- Spirals from the middle outward (right first then clockwise)
|
||||
-- @param data <table> contains:
|
||||
-- scan_center <table> Position
|
||||
-- total_scan_radius <number> radius of total scan desired
|
||||
-- individual_scan_radius <number> radius of individual scans
|
||||
-- @return <table> array of Positions (centers of scans)
|
||||
local function split_scan_radius(data)
|
||||
local center, scan_size = data.scan_center, data.individual_scan_radius
|
||||
local scan_centers = {}
|
||||
local scan_diameter = scan_size * 2
|
||||
local num_scan_rows = math.ceil(data.total_scan_radius / scan_size) -- number of scans per row/column
|
||||
local total_scans = num_scan_rows ^ 2
|
||||
local x_offset = center.x or center[1]
|
||||
local y_offset = center.y or center[2]
|
||||
local dx, x, y = 0, 0, 0
|
||||
local dy = -1
|
||||
local half_rows = num_scan_rows / 2
|
||||
|
||||
for i = 1, total_scans do
|
||||
if (-half_rows <= x and x <= half_rows) and (-half_rows < y and y <= half_rows) then
|
||||
scan_centers[i] = {(x * scan_diameter) + x_offset, (y * scan_diameter) + y_offset}
|
||||
end
|
||||
if x == y or (x < 0 and x == -y) or (x > 0 and x == 1 - y) then
|
||||
dx, dy = -dy, dx
|
||||
end
|
||||
x, y = x + dx, y + dy
|
||||
end
|
||||
|
||||
return scan_centers
|
||||
end
|
||||
|
||||
--- Sets up a queue of scans
|
||||
-- @param data <table> for specifics see data param for Public.launch_attack
|
||||
local function setup_scans(data)
|
||||
-- If an attack is already being setup, try again in a minute
|
||||
if Attack_data.attack_lockout then
|
||||
Task.set_timeout(60, setup_scans_token, data)
|
||||
return
|
||||
else
|
||||
Attack_data.attack_lockout = true
|
||||
end
|
||||
-- Initialize our data for this attack
|
||||
init_data(data.surface, data.scan_center)
|
||||
Attack_data.attack_pos = data.attack_pos or data.scan_center
|
||||
Attack_data.force_name = data.force -- allowed to be nil
|
||||
|
||||
-- Split the large scan into parts
|
||||
local scan_centers =
|
||||
split_scan_radius(
|
||||
{
|
||||
scan_center = data.scan_center,
|
||||
total_scan_radius = data.total_scan_radius or defaults.total_scan_radius,
|
||||
individual_scan_radius = data.individual_scan_radius or defaults.individual_scan_radius
|
||||
}
|
||||
)
|
||||
|
||||
-- Queue the scans
|
||||
Task.queue_task(
|
||||
biter_scan_token,
|
||||
{
|
||||
scans = scan_centers,
|
||||
surface = data.surface,
|
||||
scan_center = data.scan_center,
|
||||
biters_to_send = data.biters_to_send or calculate_biters(),
|
||||
radius = data.individual_scan_radius or defaults.individual_scan_radius,
|
||||
target_ent = data.target_ent -- allowed to be nil
|
||||
},
|
||||
#scan_centers
|
||||
)
|
||||
end
|
||||
|
||||
--- Sends attacks against players on launches
|
||||
local function rocket_launched(event)
|
||||
local entity = event.rocket_silo
|
||||
|
||||
if not entity or not entity.valid or not entity.force == 'player' then
|
||||
return
|
||||
end
|
||||
|
||||
local count = game.forces.player.rockets_launched
|
||||
local data = {
|
||||
surface = entity.surface,
|
||||
scan_center = entity.position,
|
||||
attack_pos = entity.position,
|
||||
biters_to_send = 1000,
|
||||
total_scan_radius = 10000,
|
||||
force = 'player'
|
||||
}
|
||||
|
||||
if not global.config.biter_attacks.launch_attacks.first_launch_only and count > 1 then
|
||||
--send attack of 1k
|
||||
setup_scans(data)
|
||||
game.print({'biter_attacks.rocket_launch_attack'})
|
||||
elseif count == 1 then
|
||||
-- send every living biter
|
||||
data.biters_to_send = math.huge
|
||||
setup_scans(data)
|
||||
game.print({'biter_attacks.first_rocket_launch_attack'})
|
||||
end
|
||||
end
|
||||
|
||||
-- Tokens
|
||||
setup_scans_token = Token.register(setup_scans)
|
||||
|
||||
--- Issues attack orders to the enemy unit group
|
||||
-- @param data <table> contains attack_pos (a Position), target_ent (a LuaEntity)
|
||||
local function set_attack_command(data)
|
||||
local command_table = {
|
||||
type = defines.command.compound,
|
||||
structure_type = defines.compound_command.return_last,
|
||||
commands = {
|
||||
{
|
||||
type = defines.command.attack_area,
|
||||
destination = data.attack_pos,
|
||||
radius = 150,
|
||||
distraction = defines.distraction.by_anything
|
||||
},
|
||||
{
|
||||
type = defines.command.attack_area,
|
||||
destination = {0, 0},
|
||||
radius = 1500,
|
||||
distraction = defines.distraction.by_anything
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
local target_ent = data.target_ent
|
||||
if target_ent and target_ent.valid then
|
||||
insert(
|
||||
command_table.commands,
|
||||
1,
|
||||
{
|
||||
type = defines.command.attack,
|
||||
target = target_ent,
|
||||
distraction = defines.distraction.by_damage
|
||||
}
|
||||
)
|
||||
end
|
||||
|
||||
Attack_data.enemy_unit_group.set_command(command_table)
|
||||
Debug.print({message = 'attack sent', num_sent = #Attack_data.enemy_unit_group.members})
|
||||
Attack_data.attack_lockout = nil
|
||||
end
|
||||
|
||||
--- Scans a segment of map and enters the biters into the unit group
|
||||
-- @param data <table> contains surface (a LuaSurface), scans (a table of scan centers)
|
||||
-- radius (number), scan_center (a Position), target_ent (a LuaEntity), force (string)
|
||||
biter_scan_token =
|
||||
Token.register(
|
||||
function(data)
|
||||
-- Localized data not passed through to next run
|
||||
local scan_index = Attack_data.scan_index
|
||||
local biter_count = Attack_data.biter_count
|
||||
local add_member = Attack_data.enemy_unit_group.add_member
|
||||
|
||||
-- Localize function
|
||||
local biters_to_send = data.biters_to_send
|
||||
|
||||
-- Scan the area and enter biters into the unit group
|
||||
local ents = data.surface.find_enemy_units(data.scans[scan_index], data.radius, Attack_data.force_name or 'player')
|
||||
for i = 1, #ents do
|
||||
biter_count = biter_count + 1
|
||||
add_member(ents[i])
|
||||
if biter_count >= biters_to_send then
|
||||
Debug.print({message = 'attack ordered', biter_count = biter_count, biters_to_send = biters_to_send})
|
||||
Attack_data.biter_count = biter_count
|
||||
set_attack_command({target_ent = data.target_ent, attack_pos = data.scan_center})
|
||||
return false
|
||||
end
|
||||
end
|
||||
|
||||
Attack_data.biter_count = biter_count
|
||||
if scan_index == #data.scans then
|
||||
Debug.print({message = 'attack ordered', biter_count = biter_count, biters_to_send = biters_to_send})
|
||||
set_attack_command({target_ent = data.target_ent, attack_pos = data.scan_center})
|
||||
return false
|
||||
end
|
||||
|
||||
Attack_data.scan_index = scan_index + 1
|
||||
return true
|
||||
end
|
||||
)
|
||||
|
||||
--- Sets up the parameters for an auto attack on a random player
|
||||
timed_attack_token =
|
||||
Token.register(
|
||||
function()
|
||||
local surface
|
||||
local scan_center
|
||||
local target_ent
|
||||
-- Pick a random online player
|
||||
local connected_players = game.connected_players
|
||||
local player = connected_players[random(#connected_players)]
|
||||
if player and player.valid then
|
||||
surface = player.surface
|
||||
scan_center = player.position
|
||||
|
||||
local character = player.character
|
||||
if character and character.valid then
|
||||
target_ent = character
|
||||
end
|
||||
else
|
||||
surface = RS.get_surface()
|
||||
scan_center = game.forces.player.get_spawn_position(surface)
|
||||
end
|
||||
|
||||
local data = {
|
||||
surface = surface,
|
||||
scan_center = scan_center,
|
||||
attack_pos = scan_center,
|
||||
target_ent = target_ent,
|
||||
total_scan_radius = defaults.total_scan_radius,
|
||||
individual_scan_radius = defaults.individual_scan_radius
|
||||
}
|
||||
setup_scans(data)
|
||||
Task.set_timeout(global.config.biter_attacks.timed_attacks.attack_frequency, timed_attack_token, {})
|
||||
end
|
||||
)
|
||||
|
||||
-- Public functions
|
||||
|
||||
--- Launches a biter attack
|
||||
-- @param data <table> contains:
|
||||
-- surface <LuaSurface>
|
||||
-- scan_center <table> Position center location of total scan radius
|
||||
-- attack_pos <table> (optional, defaults to using scan_center) Position for biters to attack
|
||||
-- biters_to_send <number> (optional, defaults to calling calculate_biters) the maximum number of biters to send as an attack
|
||||
-- target_ent <LuaEntity> (optional) the entity for attacks to target, if given, takes priority over attack_pos
|
||||
-- total_scan_radius <number> (optional) the maximum radius to scan for biters
|
||||
-- individual_scan_radius <number> (optional) radius of the individual scans
|
||||
-- force <string> (optional, default = 'player') the force to send an attack against
|
||||
function Public.launch_attack(data)
|
||||
setup_scans(
|
||||
{
|
||||
surface = data.surface,
|
||||
scan_center = data.scan_center,
|
||||
attack_pos = data.attack_pos,
|
||||
biters_to_send = data.biters_to_send,
|
||||
target_ent = data.target_ent,
|
||||
total_scan_radius = data.total_scan_radius or defaults.total_scan_radius,
|
||||
individual_scan_radius = data.individual_scan_radius or defaults.individual_scan_radius,
|
||||
force = data.force
|
||||
}
|
||||
)
|
||||
end
|
||||
|
||||
-- Events
|
||||
|
||||
if config.launch_attacks.enabled then
|
||||
Event.add(defines.events.on_rocket_launched, rocket_launched)
|
||||
end
|
||||
|
||||
if config.timed_attacks.enabled then
|
||||
Event.on_init(
|
||||
function()
|
||||
Task.set_timeout(global.config.biter_attacks.timed_attacks.attack_frequency, timed_attack_token, {})
|
||||
end
|
||||
)
|
||||
end
|
||||
|
||||
-- Commands
|
||||
|
||||
--- Launches a biter attack
|
||||
local function biter_attack(args)
|
||||
local target_name = args.player
|
||||
local target = game.players[target_name]
|
||||
if not target or not target.valid then
|
||||
Game.player_print({'common.fail_no_target', target_name}, Color.fail)
|
||||
return
|
||||
end
|
||||
|
||||
local biters_to_send = tonumber(args.quantity)
|
||||
if not biters_to_send then
|
||||
Game.player_print('Not a number', Color.white)
|
||||
return
|
||||
end
|
||||
|
||||
local target_pos = target.position
|
||||
local surface = target.surface
|
||||
local spawn_loc = target.force.get_spawn_position(surface)
|
||||
local character = target.character
|
||||
if not character or not character.valid then
|
||||
character = nil
|
||||
end
|
||||
|
||||
local data = {
|
||||
surface = surface,
|
||||
scan_center = target_pos,
|
||||
attack_pos = spawn_loc,
|
||||
biters_to_send = biters_to_send,
|
||||
target_ent = character
|
||||
}
|
||||
|
||||
Public.launch_attack(data)
|
||||
Game.player_print('Attack ordered', Color.success)
|
||||
end
|
||||
|
||||
Command.add(
|
||||
'biter-attack',
|
||||
{
|
||||
description = 'Orders the provided number of biters to attack the provided player ',
|
||||
arguments = {'player', 'quantity'},
|
||||
required_rank = Ranks.admin,
|
||||
allowed_by_server = true
|
||||
},
|
||||
biter_attack
|
||||
)
|
||||
|
||||
return Public
|
Loading…
x
Reference in New Issue
Block a user