Divergent/mods/Dynamic Anomalies Overhaul/gamedata/scripts/drx_da_main.script

3499 lines
116 KiB
Plaintext
Raw Normal View History

-- Dynamic Anomaly Generator by DoctorX
-- Revisited by demonized, 2022
-- Creating anomalies at the start of level after emission/psi-storm and removing anomalies after emission/psi-storm instead of just disabling them, allowing for truly dynamic generation
-- Anomalies behaviour:
-- enable/disable with randomized periods, duration and cooldowns for every single anomaly
-- behaviour if actor is near an anomaly
-- behaviour on hit
-- Spawning artefacts in new anomaly zones
--=============================================================
--
-- Dynamic Anomaly Generator (drx_da_main.script)
-- CoC 1.5b r4 - DoctorX Dynamic Anomalies 2.1
--
-- - Generates randomly placed anomalies on specified smart terrains
-- - Setting file: configs\drx\drx_da_config.ltx
--
-- Created by: DoctorX
-- Last revised: April 02, 2018
--
--=============================================================
-- Imports
local clamp = clamp
local time_global = time_global
local get_start_time = level.get_start_time
local get_game_time = game.get_game_time
local level_vertex_id = level.vertex_id
local level_vertex_position = level.vertex_position
local abs = math.abs
local ceil = math.ceil
local cos = math.cos
local floor = math.floor
local max = math.max
local min = math.min
local random = math.random
local sin = math.sin
local sqrt = math.sqrt
local CreateTimeEvent = demonized_time_events.CreateTimeEvent
local RemoveTimeEvent = demonized_time_events.RemoveTimeEvent
local process_queue = demonized_concurrent_queues.process_queue
local remove_queue = demonized_concurrent_queues.remove_queue
local add_speed = speed.add_speed
local remove_speed = speed.remove_speed
-- MCM
-- Load the defaults
local function load_defaults()
local t = {}
local op = drx_da_main_mcm.op
for i, v in ipairs(op.gr) do
if v.def ~= nil then
t[v.id] = v.def
end
end
return t
end
local settings = load_defaults()
local function load_settings()
settings = load_defaults()
if ui_mcm then
for k, v in pairs(settings) do
settings[k] = ui_mcm.get("drx_da/" .. k)
end
end
end
-- UTILS
--Recursive print of tables similar to PHP print_r function
local print_r = print_r or function(t)
local print_r_cache={}
local function sub_print_r(t,indent)
if (print_r_cache[tostring(t)]) then
printf(indent.."*"..tostring(t))
else
print_r_cache[tostring(t)]=true
if (type(t)=="table") then
for pos,val in pairs(t) do
if (type(val)=="table") then
printf(indent.."["..pos.."] => "..tostring(t).." {")
sub_print_r(val,indent..string.rep(" ",string.len(pos)+8))
printf(indent..string.rep(" ",string.len(pos)+6).."}")
else
printf(indent.."["..pos.."] => "..tostring(val))
end
end
else
printf(indent..tostring(t))
end
end
end
sub_print_r(t," ")
end
--Protected function call to prevent crashes to desktop
--Prints error in console if occured, otherwise proceed normally
--Use for test only, slower than usual
local try = try or function(func, ...)
local status, error_or_result = pcall(func, ...)
if not status then
printf(error_or_result)
return false, status, error_or_result
else
return error_or_result, status
end
end
-- Shuffle table, Fisher-Yates shuffle with preserving original table
local function shuffle(t)
local s = {}
for i = 1, #t do s[i] = t[i] end
for i = #t, 2, -1 do
local j = random(i)
s[i], s[j] = s[j], s[i]
end
return s
end
-- Get time elapsed from the start of the game in IRL seconds
local time_factor = 6
local function get_time_elapsed()
--trace(time_factor)
return floor(get_game_time():diffSec(get_start_time()) / time_factor * 10) * 0.1
end
local function round(amount)
return floor(amount + 0.5)
end
local og_printf = printf
local function printf(str, ...)
if settings.debug_mode then
og_printf("DRX DA: " .. str, ...)
end
end
--EMA smoothing for changing values, frame independent
local default_smoothing = 11.5
local smoothed_values = {}
local function ema(key, value, def, steps, delta)
local steps = steps or default_smoothing
local delta = delta or steps
local smoothing_alpha = 2.0 / (steps + 1)
smoothed_values[key] = smoothed_values[key] and smoothed_values[key] + min(smoothing_alpha * (delta / steps), 1) * (value - smoothed_values[key]) or def or value
--printf("EMA fired, key %s, target %s, current %s, going %s", key, value, smoothed_values[key], (value > smoothed_values[key] and "up" or "down"))
return smoothed_values[key]
end
local get_safe_sound_object = xr_sound.get_safe_sound_object
local function play_sound_on_actor(snd, volume, frequency, obj)
if not snd then
printf("snd is nil")
return
end
local actor = db.actor
local snd = get_safe_sound_object(snd)
if snd then
if obj and obj:id() ~= AC_ID then
snd:play_at_pos(obj, obj:position(), 0, sound_object.s3d)
else
snd:play(actor, 0, sound_object.s2d)
end
snd.volume = volume or 1
snd.frequency = frequency or 1
return snd
end
end
-- Table of current active effects, contains tables of these {
-- timer = time_elapsed + timer in seconds, how long effect will be applied
-- effect = function(), the function of the effect
-- effect_function - string, dump of effect function to store in m_data
-- effect_args - array, args to effect_function
-- on_end = function(), the function on the expiration of effect
-- on_end_function - string, dump of on_end function to store in m_data
-- on_end_args - array, args to on_end_function
-- key - string, custom key to set in timed_effects table, otherwise will be used first available one
-- not_save - boolean, do not save in mdata }
-- this is the complex function intended to have persistence between saves, moving to different maps and so on, use if needed
-- no upvalues are allowed in the functions, best to reference globals by _G. lookup
-- The precision for both cooldown and timed effects is 0.1 or 100ms, making more precise timer or timed effect is pointless
local time_elapsed = 0
timed_effects = {}
function add_timed_effect(timer, effect_function, effect_args, on_end_function, on_end_args, key, not_save)
printf("current_time %s, adding effect %s", time_elapsed, time_elapsed + (timer or 0))
local dump = string.dump
local load = loadstring
local unpack = unpack
local table_insert = table.insert
local effect_args = effect_args or {}
local on_end_args = on_end_args or {}
local effect = {
timer = time_elapsed + (timer or 0),
effect = effect_function and function()
effect_function(unpack(effect_args))
end,
effect_function = effect_function and dump(effect_function),
effect_args = effect_args,
on_end = on_end_function and function()
on_end_function(unpack(on_end_args))
end,
on_end_function = on_end_function and dump(on_end_function),
on_end_args = on_end_args,
save = not not_save
}
if key then
timed_effects[key] = effect
else
table_insert(timed_effects, effect)
end
end
-- This is the simpler version of the function above if you do not care about persistence and saving states
function add_simple_timed_effect(timer, effect_function, on_end_function, key, overwrite_mode)
-- printf("current_time %s, adding effect %s", time_elapsed, time_elapsed + (timer or 0))
if key and timed_effects[key] then
if overwrite_mode == false or overwrite_mode == 0 then
-- printf("can't add effect %s, already exists", k)
return
elseif overwrite_mode == 1 then
timed_effects[key].timer = time_elapsed + (timer or 0)
return
end
end
local dump = string.dump
local load = loadstring
local unpack = unpack
local table_insert = table.insert
local effect = {
timer = time_elapsed + (timer or 0),
effect = effect_function,
on_end = on_end_function
}
if effect.on_end then
-- printf("effect has on end function %s", effect.on_end)
end
if key then
timed_effects[key] = effect
else
table_insert(timed_effects, effect)
end
end
function remove_timed_effect(key, on_end)
if not timed_effects[key] then return end
printf("removing effect, key %s", key)
if on_end and timed_effects[key].on_end then
printf("removing effect, firing on end, key %s", key)
timed_effects[key].on_end()
end
timed_effects[key] = nil
end
-- Processing the effects
-- Whatever lowest time is set for effect, it will be processed at least once on process cycle
local function process_timed_effects()
local pairs = pairs
local printf = printf
for key, props in pairs(timed_effects) do
if props.effect then
props.effect()
end
-- printf("effect %s, timer %s, current_time %s", key, props.timer, time_elapsed)
if props.timer < time_elapsed then
printf("removing effect, effect timer %s, current_time %s", props.timer, time_elapsed)
if props.on_end then
props.on_end()
end
timed_effects[key] = nil
end
end
end
-- Callbacks
callbacks = {}
function register_callback(callback, callback_function, on_end_function, key)
if key and callbacks[key] then
UnregisterScriptCallback(callbacks[key].callback, callbacks[key].func)
end
local t = {
callback = callback,
func = callback_function,
on_end = on_end_function
}
local key = key or (#callbacks + 1)
callbacks[key] = t
printf("registering callback %s, key %s", callback, key)
RegisterScriptCallback(callbacks[key].callback, callbacks[key].func)
return key
end
function unregister_callback(key)
if not callbacks[key] then return end
printf("unregistering callback %s, %s", key, callbacks[key].callback)
UnregisterScriptCallback(callbacks[key].callback, callbacks[key].func)
if callbacks[key].on_end then
callbacks[key].on_end()
end
callbacks[key] = nil
end
function unregister_callbacks()
for i, props in pairs(callbacks) do
unregister_callback(i)
end
end
-- Get psy table to manipulate psy health values
local actor_psy_table = {}
function get_actor_psy_table()
if is_not_empty(actor_psy_table) then return end
local m_data = alife_storage_manager.get_state()
arszi_psy.save_state(m_data)
actor_psy_table = m_data.psy_table
end
function set_psy_health(amount)
if actor_psy_table.actor_psy_health then
actor_psy_table.actor_psy_health = amount <= 1 and amount or 1
end
end
function change_psy_health(amount)
if actor_psy_table.actor_psy_health then
set_psy_health(actor_psy_table.actor_psy_health + amount)
end
end
-- DRX DA
-- Location of the settings file:
local ini = ini_file("drx\\drx_da_config.ltx")
-- Table of levels that will have reduced chance to spawn anomalies
reduced_chance_levels = {
k00_marsh = true,
l03u_agr_underground = true,
l04_darkvalley = true,
l04u_labx18 = true,
l05_bar = true,
l10_radar = true,
jupiter_underground = true,
jupiter = true,
l11_pripyat = true,
pripyat = true,
zaton = true,
}
anomaly_radii = {
zone_field_radioactive = {min = 5, max = 8},
zone_field_radioactive_average = {min = 5, max = 8},
zone_field_radioactive_strong = {min = 5, max = 8},
zone_field_radioactive_weak = {min = 5, max = 8},
zone_radioactive = {min = 4, max = 6},
zone_radioactive_average = {min = 4, max = 6},
zone_radioactive_strong = {min = 4, max = 6},
zone_radioactive_weak = {min = 4, max = 6},
zone_mine_acid = {min = 2, max = 3},
zone_mine_acidic_weak = {min = 2, max = 3},
zone_mine_acidic_average = {min = 2, max = 3},
zone_mine_acidic_strong = {min = 2, max = 3},
zone_mine_blast = {min = 2, max = 3},
zone_mine_darkness = {min = 2, max = 3},
zone_mine_electra = {min = 2, max = 3},
zone_mine_electric_weak = {min = 2, max = 3},
zone_mine_electric_average = {min = 2, max = 3},
zone_mine_electric_strong = {min = 2, max = 3},
zone_mine_flash = {min = 3, max = 4},
zone_mine_ghost = {min = 2, max = 3},
zone_mine_gold = {min = 2, max = 3},
zone_mine_gravitational_weak = {min = 2, max = 3},
zone_mine_gravitational_average = {min = 3, max = 5},
zone_mine_gravitational_strong = {min = 4, max = 6},
zone_mine_green_dragon = {min = 3, max = 4},
zone_mine_mefistotel = {min = 3, max = 4},
zone_mine_net = {min = 2, max = 3},
zone_mine_point = {min = 2, max = 3},
zone_mine_radar = {min = 2, max = 3},
zone_mine_sphere = {min = 4, max = 5},
zone_mine_springboard = {min = 4, max = 6},
zone_mine_thermal_weak = {min = 1, max = 2},
zone_mine_thermal_average = {min = 1, max = 2},
zone_mine_thermal_strong = {min = 1, max = 2},
zone_mine_vapour = {min = 1, max = 2},
zone_mine_vortex = {min = 3, max = 5},
zone_mine_zharka = {min = 1, max = 2},
}
updated_anomaly_levels = {}
last_surge_time = 0
function init_anomaly_table_on_level(level_name)
local level_name = level_name or level.name()
if not updated_anomaly_levels[level_name] then updated_anomaly_levels[level_name] = {} end
if not updated_anomaly_levels[level_name].cleaned_old_anomalies then updated_anomaly_levels[level_name].cleaned_old_anomalies = false end
if not updated_anomaly_levels[level_name].anomalies then updated_anomaly_levels[level_name].anomalies = {} end
if not updated_anomaly_levels[level_name].anomalies_properties then updated_anomaly_levels[level_name].anomalies_properties = {} end
if not updated_anomaly_levels[level_name].anomalies_by_smart then updated_anomaly_levels[level_name].anomalies_by_smart = {} end
if not updated_anomaly_levels[level_name].smart_by_anomalies then updated_anomaly_levels[level_name].smart_by_anomalies = {} end
if not updated_anomaly_levels[level_name].anomaly_types_by_smart then updated_anomaly_levels[level_name].anomaly_types_by_smart = {} end
if not updated_anomaly_levels[level_name].available_smarts_reduced then updated_anomaly_levels[level_name].available_smarts_reduced = {} end
if not updated_anomaly_levels[level_name].artefacts then updated_anomaly_levels[level_name].artefacts = {} end
if not updated_anomaly_levels[level_name].time then updated_anomaly_levels[level_name].time = -1 end
if not updated_anomaly_levels[level_name].disabled then updated_anomaly_levels[level_name].disabled = false end
end
function init_anomaly_table_global(current_level)
init_anomaly_table_on_level(current_level)
for k, v in pairs(updated_anomaly_levels) do
init_anomaly_table_on_level(k)
end
end
local obj_restrictions = {}
local in_restrictions = {}
local smart_restrictions = {}
function clean_restriction_tables()
empty_table(obj_restrictions)
empty_table(in_restrictions)
empty_table(smart_restrictions)
end
function get_obj_restrictions(clean)
if clean then
clean_restriction_tables()
end
if is_not_empty(obj_restrictions) then return obj_restrictions end
local alife = alife()
local alife_level_name = alife.level_name
local alife_object = alife.object
local gg = game_graph()
local gg_vertex = gg.vertex
local level_name = level.name()
local get_monster_data = utils_stpk.get_monster_data
local get_stalker_data = utils_stpk.get_stalker_data
local function get_nearest_smart_id(se_obj)
local dist
local min_dist
local nearest
local nearest_name
for name,smart in pairs( SIMBOARD.smarts_by_names ) do
local dist = smart.position:distance_to(se_obj.position)
if (not min_dist) then
min_dist = dist
nearest = smart
nearest_name = name
elseif (dist < min_dist) then
min_dist = dist
nearest = smart
nearest_name = name
end
end
if (nearest) then
if (simulation_objects.is_on_the_same_level(nearest, se_obj)) then
return nearest.id, nearest_name
end
end
end
local restrictions = {
["dynamic_out_restrictions"] = true,
["dynamic_in_restrictions"] = true,
["base_out_restrictors"] = true,
["base_in_restrictors"] = true,
}
for i = 1, 65534 do
local se_obj = alife_object(alife, i)
if se_obj then
local cls = se_obj:clsid()
if IsMonster(_, cls) then
local se_obj_level = alife_level_name(alife, gg_vertex(gg, se_obj.m_game_vertex_id):level_id())
if true or se_obj_level == level_name then
local monster_data = get_monster_data(se_obj)
if monster_data then
-- printf(".")
-- printf("monster_data for %s, %s, level %s", se_obj:section_name(), se_obj.id, se_obj_level)
for k, v in spairs(restrictions) do
-- printf("[%s] => %s", k, v)
if monster_data[k] and type(monster_data[k]) == "table" then
-- printf("..")
for k1, v1 in spairs(monster_data[k]) do
if not obj_restrictions[se_obj.id] then obj_restrictions[se_obj.id] = {} end
if not obj_restrictions[se_obj.id][k] then obj_restrictions[se_obj.id][k] = {} end
obj_restrictions[se_obj.id][k][v1] = true
in_restrictions[v1] = true
local nearest_smart_id, nearest_smart_name = get_nearest_smart_id(se_obj)
if nearest_smart_name then
-- printf("%s", nearest_smart_name)
smart_restrictions[nearest_smart_name] = v1
end
-- printf("[%s] => %s", k1, v1)
end
end
end
end
end
end
end
end
-- print_r(smart_restrictions)
-- print_r(obj_restrictions)
return obj_restrictions
end
function remove_all_restrictions(level_name)
local alife_release_id = alife_release_id
local gg = game_graph()
local gg_vertex = gg.vertex
local invert_table = invert_table
local is_not_empty = is_not_empty
local IsMonster = IsMonster
local load_var = load_var
local pairs = pairs
local printf = printf
local sim = alife()
local sim_level_name = sim.level_name
local sim_object = sim.object
local sim_release = sim.release
local sim_remove_in_restriction = sim.remove_in_restriction
local sim_remove_out_restriction = sim.remove_out_restriction
local spairs = spairs
local strformat = strformat
local type = type
local globally = not level_name
local get_monster_data = utils_stpk.get_monster_data
local get_stalker_data = utils_stpk.get_stalker_data
local restrictions = {
["dynamic_out_restrictions"] = true,
["dynamic_in_restrictions"] = true,
-- ["base_out_restrictors"] = true,
-- ["base_in_restrictors"] = true,
}
local anomalies_ids = {}
for i = 1, 65534 do
local se_obj = sim_object(sim, i)
if se_obj then
local cls = se_obj:clsid()
if IsMonster(_, cls) or IsStalker(_, cls) then
local se_obj_level = sim_level_name(sim, gg_vertex(gg, se_obj.m_game_vertex_id):level_id())
if globally or se_obj_level == level_name then
if not anomalies_ids[se_obj_level] then
anomalies_ids[se_obj_level] = updated_anomaly_levels[se_obj_level] and invert_table(updated_anomaly_levels[se_obj_level].anomalies) or {}
end
local monster_data = IsMonster(_, cls) and get_monster_data(se_obj) or get_stalker_data(se_obj)
if monster_data then
for k, v in pairs(restrictions) do
if monster_data[k] and type(monster_data[k]) == "table" then
for k1, v1 in pairs(monster_data[k]) do
if anomalies_ids[se_obj_level][v1] then
printf("removed restriction %s for level %s", se_obj:name(), se_obj_level)
sim_remove_in_restriction(sim, se_obj, v1)
sim_remove_out_restriction(sim, se_obj, v1)
end
end
end
end
end
end
end
end
end
-- for i = 1, 65534 do
-- local obj = level.object_by_id(i)
-- if obj and obj ~= 0 and IsMonster(obj) then
-- printf("removing restrictions for %s, %s", obj:section(), i)
-- obj:remove_all_restrictions()
-- end
-- end
end
function get_anomaly_obj(id, level_name)
local level_name = level_name or level.name()
local obj = alife():object(id)
if not obj or obj == 0 or obj.id == 0 then
-- printf("Error, anomaly game object not found by id %s, level %s", id, level_name)
return
end
if not IsAnomaly(_, obj:clsid()) then
-- printf("Error, object is not an anomaly, %s, %s, on level %s", obj:section_name(), obj.id, level_name)
return
end
local sim = alife()
local sim_level_name = sim.level_name
local gg = game_graph()
local gg_vertex = gg.vertex
local obj_level = sim_level_name(sim, gg_vertex(gg, obj.m_game_vertex_id):level_id())
if obj_level ~= level_name then
-- printf("Error, anomaly game object found by id %s but on different level %s, requested level %s", id, obj_level, level_name)
return
end
return obj
end
function get_anomaly_smart(id, level_name)
local obj = get_anomaly_obj(id, level_name)
if obj then
return updated_anomaly_levels[level_name].smart_by_anomalies[id]
end
end
function get_anomalies_by_smart(level_name)
local level_name = level_name or level.name()
if not updated_anomaly_levels[level_name].anomalies_by_smart or is_empty(updated_anomaly_levels[level_name].anomalies_by_smart) then
local sim = alife()
local gg = game_graph()
for _, id in pairs(updated_anomaly_levels[level_name].anomalies) do
local se_obj = alife_object(id)
if se_obj then
for smart_name, smart in pairs(SIMBOARD.smarts_by_names) do
if smart and smart.m_game_vertex_id == se_obj.m_game_vertex_id then
printf("adding anomaly %s to smart %s", se_obj:section_name(), smart_name)
if not updated_anomaly_levels[level_name].anomalies_by_smart then
updated_anomaly_levels[level_name].anomalies_by_smart = {}
end
if not updated_anomaly_levels[level_name].smart_by_anomalies then
updated_anomaly_levels[level_name].smart_by_anomalies = {}
end
if not updated_anomaly_levels[level_name].anomalies_by_smart[smart_name] then
updated_anomaly_levels[level_name].anomalies_by_smart[smart_name] = {}
end
if not updated_anomaly_levels[level_name].anomaly_types_by_smart[smart_name] then
updated_anomaly_levels[level_name].anomaly_types_by_smart[smart_name] = ""
end
updated_anomaly_levels[level_name].anomalies_by_smart[smart_name][id] = true
updated_anomaly_levels[level_name].smart_by_anomalies[id] = smart_name
end
end
end
end
end
end
function disable_anomaly_obj(id, level_name)
local obj = get_anomaly_obj(id, level_name)
if obj then
local g_obj = level.object_by_id(id)
if g_obj and g_obj ~= 0 and g_obj:id() ~= 0 then
printf("disabling anomaly %s, %s, level %s", g_obj:section(), g_obj:id(), level_name)
g_obj:disable_anomaly()
end
end
end
function disable_anomalies_on_level(level_name)
local level_name = level_name or level.name()
for _, id in pairs(updated_anomaly_levels[level_name].anomalies) do
disable_anomaly_obj(id, level_name)
end
end
function enable_anomaly_obj(id, level_name)
local obj = get_anomaly_obj(id, level_name)
if obj then
local g_obj = level.object_by_id(id)
if g_obj and g_obj ~= 0 and g_obj:id() ~= 0 then
printf("enabling anomaly %s, %s, level %s", g_obj:section(), g_obj:id(), level_name)
g_obj:enable_anomaly()
end
end
end
function enable_anomalies_on_level(level_name)
local level_name = level_name or level.name()
for _, id in pairs(updated_anomaly_levels[level_name].anomalies) do
enable_anomaly_obj(id, level_name)
end
end
function remove_anomaly_obj(id, level_name)
local obj = get_anomaly_obj(id, level_name)
if obj then
if not IsAnomaly(_, obj:clsid()) then
printf("Error, object is not an anomaly, %s, %s, on level %s", obj:section_name(), obj.id, level_name)
return
end
printf("removing anomaly object %s, %s, on level %s", obj:section_name(), obj.id, level_name)
-- obj:disable_anomaly()
local db_tables = {
db.actor_inside_zones,
db.anim_obj_by_name,
db.anomaly_by_name,
db.bridge_by_name,
db.camp_storage,
db.campfire_by_name,
db.campfire_table_by_smart_names,
db.dynamic_ltx,
db.heli,
db.heli_enemies,
db.info_restr,
db.level_doors,
db.no_weap_zones,
db.offline_objects,
db.script_ids,
db.signal_light,
db.smart_terrain_by_id,
db.spawned_vertex_by_id,
db.storage,
db.story_by_id,
db.story_object,
db.used_level_vertex_ids,
db.zone_by_name,
}
for i = 1, #db_tables do
if is_not_empty(db_tables[i]) then
db_tables[i][obj.id] = nil
db_tables[i][obj:name()] = nil
end
end
bind_anomaly_field.fields_by_names[obj:name()] = nil
local g_obj = level.object_by_id(id)
if g_obj and g_obj ~= 0 and g_obj:id() ~= 0 then
g_obj:destroy_object()
else
alife_record(obj, false)
alife():release(obj, true)
end
end
end
-- Delete old anomalies persisting from old code
function clean_old_dynamic_anomalies_on_level(level_name)
if not updated_anomaly_levels[level_name].cleaned_old_anomalies then
local load_var = load_var
local pairs = pairs
local sim = alife()
local sim_level_name = sim.level_name
local gg = game_graph()
local gg_vertex = gg.vertex
local strformat = strformat
local fully_cleaned = true
for smart_name, v in pairs(SIMBOARD.smarts_by_names) do
local smart_level = sim_level_name(sim, gg_vertex(gg, v.m_game_vertex_id):level_id())
if smart_level == level_name then
for j = 1, 1000 do
local anom_id = load_var(db.actor, strformat("drx_da_anom_id_%s_%s", smart_name, j), nil)
if anom_id then
if not in_restrictions[anom_id] then
remove_anomaly_obj(anom_id, level_name)
else
printf("Error, can't remove old anomaly %s, level %s, in restriction", anom_id, level_name)
fully_cleaned = false
end
end
end
end
end
updated_anomaly_levels[level_name].cleaned_old_anomalies = fully_cleaned
else
printf("old anomalies already cleaned on level %s", level_name)
end
end
-- Clean dynamic anomalies on level, normally after surge on level change or forcefully
function clean_dynamic_anomalies_on_level_func(level_name)
local t = updated_anomaly_levels[level_name]
if not t then
printf("Error, updated_anomaly_levels table not found for %s", level_name)
return
end
local t = t.anomalies
for i, id in pairs(t) do
if t[i] and not in_restrictions[t[i]] then
remove_anomaly_obj(t[i], level_name)
for k, v in pairs(updated_anomaly_levels[level_name].anomalies_by_smart) do
v[t[i]] = nil
updated_anomaly_levels[level_name].anomaly_types_by_smart[k] = nil
if is_empty(v) then
updated_anomaly_levels[level_name].anomalies_by_smart[k] = nil
updated_anomaly_levels[level_name].available_smarts_reduced[k] = nil
end
end
updated_anomaly_levels[level_name].smart_by_anomalies[t[i]] = nil
updated_anomaly_levels[level_name].anomalies_properties[t[i]] = nil
t[i] = nil
else
printf("Error, can't remove anomaly %s, level %s, in restriction", t[i], level_name)
end
end
clean_old_dynamic_anomalies_on_level(level_name)
printf("Anomalies cleaned on level %s", level_name)
end
function clean_dynamic_anomalies_on_level(level_name, dont_clean_restrictions)
get_anomalies_by_smart(level_name)
remove_all_restrictions(level_name)
get_obj_restrictions(not dont_clean_restrictions)
-- disable_anomalies_on_level(level_name)
-- remove_restrictions(level_name)
clean_dynamic_anomalies_on_level_func(level_name)
-- enable_anomalies_on_level(level_name)
-- clean_restriction_tables()
end
function clean_dynamic_anomalies_global()
remove_all_restrictions()
get_obj_restrictions(true)
unregister_anomalies_behaviour()
local alife_release_id = alife_release_id
local gg = game_graph()
local gg_vertex = gg.vertex
local level_name = level.name()
local load_var = load_var
local pairs = pairs
local printf = printf
local sim = alife()
local sim_level_name = sim.level_name
local sim_object = sim.object
local sim_release = sim.release
local alife_record = alife_record
local strformat = strformat
for k, v in pairs(updated_anomaly_levels) do
get_anomalies_by_smart(k)
clean_artefacts_on_level(k)
if k == level_name then
disable_anomalies_on_level(level_name)
v.disabled = true
-- clean_dynamic_anomalies_on_level_func(level_name)
else
for k1, v1 in pairs(v.anomalies) do
if not in_restrictions[v1] then
local se_obj = sim_object(sim, v1)
if se_obj then
if IsAnomaly(_, se_obj:clsid()) then
printf("Deleting anomaly %s, %s globally, level %s", se_obj:section_name(), v1, k)
alife_record(se_obj ,false)
alife():release(se_obj, true)
else
printf("Error, object is not an anomaly, %s, %s, on level %s", se_obj:section_name(), v1, k)
end
end
for k2, v2 in pairs(v.anomalies_by_smart) do
v2[v1] = nil
v.anomaly_types_by_smart[k2] = nil
if is_empty(v2) then
v.anomalies_by_smart[k2] = nil
v.available_smarts_reduced[k2] = nil
end
end
v.smart_by_anomalies[v1] = nil
v.anomalies_properties[v1] = nil
v.anomalies[k1] = nil
else
printf("can't delete anomaly %s globally, level %s, in restriction", v1, k)
end
end
-- Old anomalies
if not v.cleaned_old_anomalies then
local fully_cleaned = true
for smart_name, v in pairs(SIMBOARD.smarts_by_names) do
local smart_level = sim_level_name(sim, gg_vertex(gg, v.m_game_vertex_id):level_id())
if smart_level == k then
for j = 1, 1000 do
local anom_id = load_var(db.actor, strformat("drx_da_anom_id_%s_%s", smart_name, j), nil)
if anom_id then
if not in_restrictions[anom_id] then
local o = alife_object(anom_id)
if o then
alife_record(o ,false)
alife():release(o, true)
end
else
printf("Error, can't remove old anomaly %s, level %s, in restriction", anom_id, k)
fully_cleaned = false
end
end
end
end
end
v.cleaned_old_anomalies = fully_cleaned
else
printf("old anomalies already cleaned on level %s", k)
end
end
end
printf("Cleaned dynamic anomalies globally")
if settings.save_after_cleanup then
CreateTimeEvent("drx_da_save_after_cleanup", 0, 0.1, function()
exec_console_cmd("save " .. (user_name() or "") .. " - DAO tempsave")
return true
end)
end
end
function drx_da_spawn_anomaly_on_smart(level_file, smart_name, anomaly_type, level_name, position_data)
-- Get the smart terrain:
local smart = SIMBOARD.smarts_by_names[smart_name]
if not smart then
printf("Error: Unable to create dynamic anomaly field for %s, the specified smart location does not exist", smart_name)
return false
end
-- Select a location for the current anomaly:
local pos = drx_da_generate_position(smart_name, anomaly_type, position_data)
if pos then
-- Get the new level vertex id for the generated position:
local lvid = level_vertex_id(pos)
-- Spawn the anomaly:
local anom_id = drx_da_spawn_anomaly(anomaly_type, pos, lvid, smart.m_game_vertex_id, level_file)
-- Return the anomaly id:
if anom_id then
printf("Dynamic anomaly field %s spawned at %s, level %s", anomaly_type, smart_name, level_name)
return anom_id
end
else
printf("Error: failed to generate position")
end
end
function get_level_data(level_name)
local level_file_name = "hazardous_anomalies\\regions\\" .. level_name .. ".ltx"
local level_file = ini_file(level_file_name)
if not level_file then
printf("ltx file not found: %s", level_file_name)
return false
end
-- Get the percent chance for anomalies to spawn:
local spawn_percent = level_file:r_float_ex("spawn_properties", "spawn_percent") or 0
if not spawn_percent or spawn_percent <= 0 then
printf("Dynamic anomalies not spawned, spawn chance is 0")
return false
end
-- Determine the maximum amount of anomalies spawned in each anomaly field:
local anomaly_max_number = level_file:r_float_ex("spawn_properties", "anomaly_max_number") or 0
if not anomaly_max_number or anomaly_max_number < 1 then
printf("Dynamic anomalies not spawned, max anomaly count is 0")
return false
end
-- Determine the maximum amount of anomalies active in each anomaly field:
local anomaly_max_active = level_file:r_float_ex("spawn_properties", "anomaly_max_active") or 0
if not anomaly_max_active or anomaly_max_active < 1 then
printf("Dynamic anomalies not spawned, max active count is 0")
return false
end
return {
level_file = level_file,
spawn_percent = spawn_percent,
anomaly_max_number = anomaly_max_number,
anomaly_max_active = anomaly_max_active,
}
end
function generate_random_anomaly_properties()
return {
time_active = random(6500, 15000),
time_cooldown = random(2200, 3800),
active = true,
}
end
function drx_da_spawn_anomalies_on_level(level_name)
local level_data = get_level_data(level_name)
if not level_data then
printf("Error, unable to get data for %s", level_name)
return
end
local level_file = level_data.level_file
local spawn_percent = level_data.spawn_percent
local anomaly_max_number = level_data.anomaly_max_number * settings.anomaly_amount_modifier
local anomaly_max_active = level_data.anomaly_max_active
local pairs = pairs
local collect_section = utils_data.collect_section
local size_table = size_table
local invert_table = invert_table
local is_not_empty = is_not_empty
local is_empty = is_empty
local table_remove = table.remove
-- Build a list of available smart terrains:
local smart_list = collect_section(level_file, "available_smarts")
if is_not_empty(smart_list) then
if reduced_chance_levels[level_name] then
local t = {}
for k, v in pairs(smart_list) do
if random(100) <= 50 then
t[#t + 1] = v
end
end
smart_list = t
end
end
-- Build a list of available smart terrains with reduced amount:
local smart_list_reduced = collect_section(level_file, "available_smarts_reduced")
if is_not_empty(smart_list_reduced) then
smart_list_reduced = invert_table(smart_list_reduced)
for k, v in pairs(smart_list_reduced) do
smart_list[#smart_list + 1] = k
end
end
-- Build a list of available anomalies:
local anomaly_list = collect_section(level_file, "anomaly_types")
if settings.disable_new_anomalies then
local t = {}
for _, v in pairs(anomaly_list) do
if not drx_da_main_mcm.new_anomalies_sections[v] then
t[#t + 1] = v
else
printf("disable all new anomalies, found section %s", v)
end
end
anomaly_list = t
else
local t = {}
for k, v in pairs(anomaly_list) do
if drx_da_main_mcm.new_anomalies_sections[v] then
if drx_da_main_mcm.is_enabled_anomaly(v) then
t[#t + 1] = v
else
printf("anomaly %s is not enabled", v)
end
else
t[#t + 1] = v
end
end
anomaly_list = t
end
-- Combine radiation fields to one type
local rad_fields = {}
for i = #anomaly_list, 1, -1 do
if anomalies_radiation_fields[anomaly_list[i]] then
rad_fields[#rad_fields + 1] = anomaly_list[i]
table_remove(anomaly_list, i)
end
end
if is_not_empty(rad_fields) then
for i = 1, math.ceil(#rad_fields / 2) do
if is_empty(rad_fields) then break end
anomaly_list[#anomaly_list + 1] = table_remove(rad_fields, math.random(#rad_fields))
end
end
-- Build anomalies table
if is_not_empty(smart_list) then
local anomalies = updated_anomaly_levels[level_name].anomalies or {}
local anomalies_by_smart = updated_anomaly_levels[level_name].anomalies_by_smart or {}
local smart_by_anomalies = updated_anomaly_levels[level_name].smart_by_anomalies or {}
local anomaly_types_by_smart = updated_anomaly_levels[level_name].anomaly_types_by_smart or {}
local anomalies_properties = updated_anomaly_levels[level_name].anomalies_properties or {}
local available_smarts_reduced = updated_anomaly_levels[level_name].available_smarts_reduced or {}
for i, smart_name in pairs(smart_list) do
if (true or not smart_restrictions[smart_name]) and random() <= settings.anomaly_zone_spawn_chance then
-- Choose an anomaly type to spawn:
if anomaly_list and #anomaly_list >= 1 then
local anomaly_type = anomaly_list[random(#anomaly_list)]
-- if anomaly_type == "zone_radiation_field" then
-- anomaly_type = rad_fields[math.random(#rad_fields)]
-- end
printf("picked anomaly type %s", anomaly_type)
if not anomalies_by_smart[smart_name] then anomalies_by_smart[smart_name] = {} end
local j = size_table(anomalies_by_smart[smart_name])
-- Store position data of generated anomalies
local position_data = kd_tree.buildTreeVectors()
if j > 0 then
for k, v in pairs(anomalies_by_smart[smart_name]) do
local se_obj = get_anomaly_obj(k)
if se_obj then
local pos = se_obj.position
if pos then
position_data:insertAndRebuild({x = pos.x, y = pos.y, z = pos.z})
end
end
end
end
while j < (smart_list_reduced[smart_name] and random(3) or anomaly_max_number) do
if random() <= (smart_list_reduced[smart_name] and 0.5 or spawn_percent) then
local anom_id = drx_da_spawn_anomaly_on_smart(level_file, smart_name, anomaly_type, level_name, position_data)
if anom_id then
anomalies[#anomalies + 1] = anom_id
anomalies_by_smart[smart_name][anom_id] = true
smart_by_anomalies[anom_id] = smart_name
anomaly_types_by_smart[smart_name] = anomaly_type
available_smarts_reduced[smart_name] = smart_list_reduced[smart_name]
anomalies_properties[anom_id] = generate_random_anomaly_properties()
end
end
j = j + 1
end
else
printf("No dynamic anomaly types specified for level %s", level_name)
end
else
printf("no anomalies spawn on smart %s, level %s in restriction", smart_name, level_name)
end
end
return #anomalies > 0
end
end
-- \\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\
-- ////////////////////////////////////////////////////////////////////////////////////////////////
--
-- drx_da_spawn_anomaly function
--
-- ------------------------------------------------------------------------------------------------
--
-- Description:
-- - Spawns an anomaly at the specified location
--
-- Usage:
-- drx_da_spawn_anomaly( anom_type, pos, lvid, gvid )
--
-- Parameters:
-- anom_type (type: string, anomaly type name)
-- - Type of anomaly to spawn
-- pos (type: vector)
-- - Positional data for the anomaly
-- lvid (type: int, level vertex id)
-- - Level vertex id
-- gvid (type: int, game vertex id)
-- - Game vertex id
--
-- Return value (type: object id):
-- Returns the id of the spawned anomaly
-- Returns nil on failure
--
-- ------------------------------------------------------------------------------------------------
-- Created by DoctorX
-- for DoctorX Dynamic Anomalies 2.0
-- Last modified March 02, 2018
-- ------------------------------------------------------------------------------------------------
-- Spawn a single anomaly:
function drx_da_spawn_anomaly(anom_type, pos, lvid, gvid, level_file)
local function abort_creation(se_obj_id, anom_type)
CreateTimeEvent("drx_da_abort_creation" .. se_obj_id, "drx_da_abort_creation" .. se_obj_id, 0.2, function()
local obj = alife():object(se_obj_id)
if obj then
printf("Error, anomaly %s failed to spawn correctly, releasing", anom_type)
alife_record(obj ,false)
alife():release(obj, true)
end
return true
end)
end
local min_radius = (level_file:r_float_ex("radius_properties", "min_radius") or 2)
local max_radius = (level_file:r_float_ex("radius_properties", "max_radius") or 3)
-- Spawn the anomaly:
local se_obj = alife():create(anom_type, pos, lvid, gvid)
if (not se_obj) then
printf("Error: Unable to spawn dynamic anomaly")
return
end
-- Set anomaly properties:
local data = utils_stpk.get_anom_zone_data(se_obj)
if (not data) then
printf("Error: Unable to set dynamic anomaly properties")
abort_creation(se_obj.id, anom_type)
return
end
data.shapes[1] = {}
data.shapes[1].shtype = 0
data.shapes[1].offset = vector():set(0, 0, 0) -- Leave for compatibility with CoC 1.4.22, delete later
data.shapes[1].center = vector():set(0, 0, 0)
data.shapes[1].radius = anomaly_radii[anom_type] and random(anomaly_radii[anom_type].min, anomaly_radii[anom_type].max) or random(min_radius, max_radius)
utils_stpk.set_anom_zone_data(data, se_obj)
-- Return the anomaly id:
return se_obj.id, se_obj
end
-- \\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\
-- ////////////////////////////////////////////////////////////////////////////////////////////////
--
-- drx_da_generate_position function
--
-- ------------------------------------------------------------------------------------------------
--
-- Description:
-- - Generates a random position vector on the ground within a smart terrain location
--
-- Usage:
-- drx_da_generate_position( smart_name )
--
-- Parameters:
-- smart_name (type: string, smart terrain name)
-- - Name of the smart terrain
--
-- Ini requirements:
-- drx\drx_da_config.ltx
-- [location_offset]
-- max_offset_x (type: float, meters)
-- - Magnitude of maximum offset from smart terrain center in x (north-south) direction
-- max_offset_y (type: float, meters)
-- - Magnitude of maximum offset from smart terrain center in y (up-down) direction
-- max_offset_z (type: float, meters)
-- - Magnitude of maximum offset from smart terrain center in z (east-west) direction
-- max_tries (type: int)
-- - Maximum number of iterations to try generating a spawn position before aborting
--
-- Return value (type: vector):
-- Returns the generated positional data
-- Returns nil on failure
--
-- ------------------------------------------------------------------------------------------------
-- Created by DoctorX, (modification of method suggested by Alundaio)
-- for DoctorX Dynamic Anomalies 2.0
-- Last modified January 31, 2018
-- ------------------------------------------------------------------------------------------------
-- Table of small levels to adjust spawn extra to underground levels
small_levels = {
y04_pole = true,
l11_hospital = true,
}
-- Table of anomalies to ignore distance check
anomalies_distance_check_ignore = {
zone_field_radioactive = true,
zone_field_radioactive_above_average = true,
zone_field_radioactive_average = true,
zone_field_radioactive_below_average = true,
zone_field_radioactive_lethal = true,
zone_field_radioactive_strong = true,
zone_field_radioactive_very_weak = true,
zone_field_radioactive_weak = true,
zone_radioactive = true,
zone_radioactive_above_average = true,
zone_radioactive_average = true,
zone_radioactive_below_average = true,
zone_radioactive_lethal = true,
zone_radioactive_strong = true,
zone_radioactive_very_weak = true,
zone_radioactive_weak = true,
}
-- Table of radiation fields
anomalies_radiation_fields = {
zone_field_radioactive = true,
zone_field_radioactive_above_average = true,
zone_field_radioactive_average = true,
zone_field_radioactive_below_average = true,
zone_field_radioactive_lethal = true,
zone_field_radioactive_strong = true,
zone_field_radioactive_very_weak = true,
zone_field_radioactive_weak = true,
zone_radioactive = true,
zone_radioactive_above_average = true,
zone_radioactive_average = true,
zone_radioactive_below_average = true,
zone_radioactive_lethal = true,
zone_radioactive_strong = true,
zone_radioactive_very_weak = true,
zone_radioactive_weak = true,
}
-- Generate positional data:
function drx_da_generate_position(smart_name, anomaly_type, position_data)
-- Get the smart terrain:
local smart = SIMBOARD.smarts_by_names[smart_name]
if (not smart) then
printf("Error: Unable to generate positional data, specified smart location does not exist")
return
end
-- Get maximum offset values:
local max_offset_x = settings.anomaly_zone_anomalies_distance_max or ini:r_float_ex("location_offset", "max_offset_x") or 40
local max_offset_y = ini:r_float_ex("location_offset", "max_offset_y") or 0
local max_offset_z = settings.anomaly_zone_anomalies_distance_max or ini:r_float_ex("location_offset", "max_offset_z") or 40
local num_tries = (ini:r_float_ex("location_offset", "max_tries") or 64)
-- Reduce offset by 2 times for underground
if level_weathers.bLevelUnderground or small_levels[level.name()] then
max_offset_x = floor(max_offset_x * 0.5)
max_offset_y = floor(max_offset_y * 0.5)
max_offset_z = floor(max_offset_z * 0.5)
end
-- Try to generate valid positional data on the ground:
local pos = vector():set(0, 0, 0)
local valid_lvid = false
while ((valid_lvid ~= true) and (num_tries > 0)) do
-- Randomly offset anomaly x-position from center of smart terrain:
local offset_x = max_offset_x * random()
if (random() <= 0.5) then
offset_x = -(offset_x)
end
local pos_x = (smart.position.x + offset_x)
-- Randomly offset anomaly y-position from center of smart terrain:
local offset_y = (max_offset_y * random())
if (random() <= 0.5) then
offset_y = -(offset_y)
end
local pos_y = (smart.position.y + offset_y)
-- Randomly offset anomaly z-position from center of smart terrain:
local offset_z = max_offset_z * random()
if (random() <= 0.5) then
offset_z = -(offset_z)
end
local pos_z = (smart.position.z + offset_z)
-- Set anomaly position at location vertex and check if valid:
pos = vector():set(pos_x, pos_y, pos_z)
local lvid = level_vertex_id(pos)
if (lvid < 4294967295) then
pos = level_vertex_position(lvid)
-- Don't check distance for certain anomalies
if anomaly_type and anomalies_distance_check_ignore[anomaly_type] then
valid_lvid = true
else
-- If position data exists and distance of generated position is more than anomaly radius - valid
try(function()
if anomaly_type and anomaly_radii[anomaly_type] and position_data and position_data.root then
local nearest = position_data:nearest(pos)
if nearest and nearest[1] and nearest[1][2] then
local distance = sqrt(nearest[1][2]) - anomaly_radii[anomaly_type].max * 2
if distance >= settings.anomaly_zone_anomalies_distance_min then
printf("Anomaly type %s, Position data valid, distance %s, saving %s, %s, %s", anomaly_type, distance, pos_x, pos_y, pos_z)
position_data:insertAndRebuild({x = pos_x, y = pos_y, z = pos_z})
valid_lvid = true
else
printf("Anomaly type %s, Position data invalid, too close, distance %s, %s, %s, %s", anomaly_type, distance, pos_x, pos_y, pos_z)
valid_lvid = false
end
else
printf("Anomaly type %s, Can't check position data %s, %s, %s", anomaly_type, pos_x, pos_y, pos_z)
position_data:insertAndRebuild({x = pos_x, y = pos_y, z = pos_z})
valid_lvid = true
end
else
if position_data then
printf("Anomaly type %s, Position data provided, saving %s, %s, %s", anomaly_type, pos_x, pos_y, pos_z)
position_data:insertAndRebuild({x = pos_x, y = pos_y, z = pos_z})
end
valid_lvid = true
end
end)
end
end
-- Decrement the number of tries left:
num_tries = (num_tries - 1)
if ((num_tries <= 0) and (valid_lvid ~= true)) then
printf("Error: Unable to generate valid lvid pos, aborting")
return
end
end
-- Return the position vector:
return pos
end
function clean_artefacts_on_level(level_name)
local level_name = level_name or level.name()
init_anomaly_table_on_level(level_name)
local artefacts = updated_anomaly_levels[level_name].artefacts
if is_not_empty(artefacts) then
for k, v in pairs(artefacts) do
local o = alife_object(k)
if o then safe_release_manager.release(o) end
printf("releasing artefact %s, sec %s, level_name %s", k, v, level_name)
artefacts[k] = nil
end
end
end
-- Spawn single artefact on smart
function spawn_artefact_on_smart(level_file, smart_name, picked_artefact, level_name)
local level_name = level_name or level.name()
if not picked_artefact then
printf("Error: Unable to create artefact %s, is nil", picked_artefact)
return false
end
-- Get the smart terrain:
local smart = SIMBOARD.smarts_by_names[smart_name]
if not smart then
printf("Error: Unable to create artefact for %s, the specified smart location does not exist", smart_name)
return false
end
-- Select a location for the current artefact:
local pos = drx_da_generate_position(smart_name)
if pos then
-- Correct y position so the artefact wouldnt fall OOB
pos = vector():set(pos.x, (level_weathers.bLevelUnderground or small_levels[level.name()]) and pos.y + 1 or pos.y + 7, pos.z)
-- Get the new level vertex id for the generated position:
local lvid = level_vertex_id(pos)
-- Spawn the artefact:
local artefact = alife_create(picked_artefact, pos, lvid, smart.m_game_vertex_id)
-- Return the anomaly id:
if artefact then
printf("Artefact %s, id %s spawned at %s, level %s", picked_artefact, artefact.id, smart_name, level_name)
return artefact.id
end
else
printf("Error: failed to generate position")
end
end
-- Spawn artefacts on level
function spawn_artefacts_on_level(level_name)
local level_name = level_name or level.name()
local level_data = get_level_data(level_name)
if not level_data then
printf("Error, unable to get data for %s", level_name)
return
end
local level_file = level_data.level_file
-- Build a list of available smart terrains:
local smart_list = {}
for k, v in pairs(updated_anomaly_levels[level_name].anomaly_types_by_smart) do
if not updated_anomaly_levels[level_name].available_smarts_reduced[k] then
smart_list[#smart_list + 1] = k
end
end
smart_list = invert_table(smart_list)
local pairs = pairs
local collect_section = utils_data.collect_section
local size_table = size_table
local allowed_artefacts = drx_da_main_artefacts.allowed_artefacts
local allowed_artefacts_flipped = invert_table(allowed_artefacts)
local anomaly_type_to_artefacts = drx_da_main_artefacts.anomaly_type_to_artefacts
local artefacts_map_tiers = drx_da_main_artefacts.artefacts_map_tiers[level_name] and shuffle(drx_da_main_artefacts.artefacts_map_tiers[level_name])
local artefacts_map_chances = drx_da_main_artefacts.artefacts_map_chances and drx_da_main_artefacts.artefacts_map_chances[level_name]
-- Build anomalies table
if is_not_empty(smart_list) then
printf("%s has smarts with anomalies, try to spawn artefacts", level_name)
local artefacts = updated_anomaly_levels[level_name].artefacts or {}
local anomalies_by_smart = updated_anomaly_levels[level_name].anomalies_by_smart
for smart_name, _ in pairs(anomalies_by_smart) do
printf("checking smart %s for spawning artefacts", smart_name)
if is_not_empty(anomalies_by_smart[smart_name]) and smart_list[smart_name] then
printf("try to spawn artefacts on smart %s", smart_name)
for i = 1, settings.max_artefacts_per_zone do
-- Increased chance by 2 times for underground levels
local dice_roll = random(100)
local chance = artefacts_map_chances or ceil(settings.artefacts_spawn_chance * ((level_weathers.bLevelUnderground or small_levels[level_name]) and 2 or 1))
printf("artefacts dice roll %s, chance %s, spawn %s", dice_roll, chance, dice_roll <= chance)
if dice_roll <= chance then
-- Choose an artefact to spawn:
local anomaly_type = updated_anomaly_levels[level_name].anomaly_types_by_smart[smart_name]
local picked_artefact = (function()
local res
if artefacts_map_tiers and random(100) > settings.random_artefact_spawn_chance then
local tries = 40
while tries > 0 and (not res or not allowed_artefacts_flipped[res]) do
if anomaly_type_to_artefacts[anomaly_type] then
local t = {}
for k, v in pairs(artefacts_map_tiers) do
if anomaly_type_to_artefacts[anomaly_type][v] then
t[#t + 1] = v
end
end
printf("picking artefact by level %s, anomaly zone %s has defined arty list", level_name, anomaly_type)
res = t[random(#t)]
else
printf("picking artefact by level %s", level_name)
res = artefacts_map_tiers[random(#artefacts_map_tiers)]
end
if not allowed_artefacts_flipped[res] then
printf("artefact is not allowed to spawn, repicking")
end
tries = tries - 1
end
else
printf("picking random artefacts")
res = allowed_artefacts[random(#allowed_artefacts)]
end
if not res then
printf("failed to pick artefact by level, pick random from allowed, level_name %s, anomaly_type %s, has artefacts_map_tiers %s, has anomaly_type_to_artefacts %s", level_name, anomaly_type, artefacts_map_tiers ~= nil, anomaly_type_to_artefacts[anomaly_type] ~= nil)
res = allowed_artefacts[random(#allowed_artefacts)]
end
return res
end)()
-- Artefact Variationizer compatibility
if artefact_variationizer then
local av = artefact_variationizer
picked_artefact = av.get_artefact_base(picked_artefact)
if av.valid_artys[picked_artefact] then
local variationizer_tier = av.artefact_chances[random(#av.artefact_chances)]
local arty = av.artefact_by_variationizer_tier[picked_artefact][variationizer_tier]
picked_artefact = arty[random(#arty)]
end
end
printf("picked artefact to spawn %s, anomaly_type %s, smart %s level %s", picked_artefact, anomaly_type, smart_name, level_name)
local artefact_id = spawn_artefact_on_smart(level_file, smart_name, picked_artefact, level_name)
if artefact_id then
artefacts[artefact_id] = picked_artefact
else
printf("error, unabled to spawn artefact %s, anomaly_type %s, smart %s level %s", picked_artefact, anomaly_type, smart_name, level_name)
end
end
end
end
end
return size_table(artefacts) > 0
else
printf("%s has no smarts with anomalies, dont spawn artefacts", level_name)
end
end
-- Update dynamic anomalies:
function drx_da_update_dynamic_anomalies(force)
-- Verify db.actor is available:
if not db.actor then
printf("Error: Cannot update anomalies, db.actor not available")
return false
end
-- Get surge manager:
local surgeman = surge_manager.get_surge_manager()
if not surgeman then
printf("Error: Cannot update anomalies, surge manager not available")
return false
end
local level_name = level.name()
init_anomaly_table_on_level(level_name)
if last_surge_time > updated_anomaly_levels[level_name].time or force then
clean_dynamic_anomalies_on_level(level_name)
clean_artefacts_on_level(level_name)
CreateTimeEvent("drx_da_spawn_anomalies_on_level", "drx_da_spawn_anomalies_on_level", 0.3, function()
local anomalies = drx_da_spawn_anomalies_on_level(level_name)
if not anomalies then
printf("Error, failed to spawn anomalies")
build_anomalies_pos_tree()
return true
end
printf("updated anomalies on level %s", level_name)
updated_anomaly_levels[level_name].time = last_surge_time
updated_anomaly_levels[level_name].disabled = false
build_anomalies_pos_tree()
if settings.enable_anomalies_behaviour or level_weathers.bLevelUnderground then register_anomalies_behaviour() end
spawn_artefacts_on_level(level_name)
return true
end)
else
printf("anomalies not updated on level %s, no emission happened, last_surge_time %s, level update time %s", level_name, last_surge_time, updated_anomaly_levels[level_name].time)
if settings.enable_anomalies_behaviour or level_weathers.bLevelUnderground then register_anomalies_behaviour() end
end
return true
end
-- Scripts to run when the game loads:
local tg = 0
local tg_interval = 5000
function drx_da_actor_on_update_callback()
local t = time_global()
if t < tg then return end
tg = t + tg_interval
load_settings()
get_actor_psy_table()
init_anomaly_table_global(level.name())
printf("saved level %s, current level %s", alife_storage_manager.get_state().drx_da_previous_level, level.name())
if level.name() == alife_storage_manager.get_state().drx_da_previous_level then
local level_name = level.name()
if level_name and updated_anomaly_levels[level_name] and updated_anomaly_levels[level_name].disabled then
if last_surge_time > updated_anomaly_levels[level_name].time then
clean_dynamic_anomalies_on_level(level_name)
updated_anomaly_levels[level_name].disabled = false
end
end
printf("on the same level, only register behaviour")
if settings.enable_anomalies_behaviour or level_weathers.bLevelUnderground then register_anomalies_behaviour() end
unregister_drx_da()
build_anomalies_pos_tree()
RegisterScriptCallback("actor_on_update", actor_on_update)
return
else
local level_name = alife_storage_manager.get_state().drx_da_previous_level
if level_name and updated_anomaly_levels[level_name] and updated_anomaly_levels[level_name].disabled then
if last_surge_time > updated_anomaly_levels[level_name].time then
clean_dynamic_anomalies_on_level(level_name)
updated_anomaly_levels[level_name].disabled = false
end
end
end
get_anomalies_by_smart(level.name())
-- Update dynamic anomalies:
local updated = drx_da_update_dynamic_anomalies()
unregister_drx_da()
build_anomalies_pos_tree()
RegisterScriptCallback("actor_on_update", actor_on_update)
end
function unregister_drx_da()
printf("anomalies updated, unregistering")
UnregisterScriptCallback("actor_on_update", drx_da_actor_on_update_callback)
end
function unregister_anomalies_behaviour()
remove_queue("drx_da_anomalies_behaviour")
end
-- Sections to ignore on/off switch behaviour
anomalies_do_not_register_behaviour = {
zone_mine_ghost = true
}
function register_anomalies_behaviour()
local level_name = level.name()
if is_empty(updated_anomaly_levels[level_name]) or is_empty(updated_anomaly_levels[level_name].anomalies) then
printf("anomalies behaviour, anomalies not found for level %s", level_name)
return
end
if is_empty(updated_anomaly_levels[level_name].anomalies_properties) or
size_table(updated_anomaly_levels[level_name].anomalies_properties) ~= size_table(updated_anomaly_levels[level_name].anomalies)
then
for k, v in pairs(updated_anomaly_levels[level_name].anomalies) do
updated_anomaly_levels[level_name].anomalies_properties[v] = generate_random_anomaly_properties()
end
end
printf("anomalies behaviour, turning on behaviour for level %s", level_name)
for k, v in pairs(updated_anomaly_levels[level_name].anomalies_properties) do
v.update_time = time_global()
end
local get_object = level.object_by_id
local time_global = time_global
process_queue("drx_da_anomalies_behaviour", updated_anomaly_levels[level_name].anomalies_properties, function(id, props, i)
local t = time_global()
-- if is_empty(props) then
-- printf("Error, anomaly behaviour not found for %s, level %s", id, level_name)
-- return true
-- end
local obj = get_object(id)
if not obj then return end
-- Remove from queue if its in do_not_register_behaviour table
if anomalies_do_not_register_behaviour[obj:section()] then return true end
if props.active then
if t - props.update_time > props.time_active then
-- printf("anomaly disabled, id %s, i %s, t %s, u %s", id, i, t, props.update_time)
props.update_time = t
props.active = false
obj:disable_anomaly()
end
else
if t - props.update_time > props.time_cooldown then
-- printf("anomaly enabled, id %s, i %s, t %s, u %s", id, i, t, props.update_time)
props.update_time = t
props.active = true
obj:enable_anomaly()
end
end
end, nil, 5)
end
anomalies_pos_tree = nil
detectable_anomalies_pos_tree = nil
detectable_anomalies_ids = {}
anomalies_obj_to_pos = {}
anomalies_sec_to_obj = {}
common_sec = {
zone_mine_electric = "zone_mine_electric",
zone_mine_electric_weak = "zone_mine_electric",
zone_mine_electric_average = "zone_mine_electric",
zone_mine_electric_strong = "zone_mine_electric",
zone_mine_static = "zone_mine_electric",
zone_mine_static_weak = "zone_mine_electric",
zone_mine_static_average = "zone_mine_electric",
zone_mine_static_strong = "zone_mine_electric",
zone_witches_galantine = "zone_mine_electric",
zone_witches_galantine_weak = "zone_mine_electric",
zone_witches_galantine_average = "zone_mine_electric",
zone_witches_galantine_strong = "zone_mine_electric",
}
function build_anomalies_pos_tree()
local alife_release_id = alife_release_id
local gg = game_graph()
local gg_vertex = gg.vertex
local level_name = level.name()
local load_var = load_var
local pairs = pairs
local printf = printf
local sim = alife()
local sim_level_name = sim.level_name
local sim_object = sim.object
local sim_release = sim.release
local alife_record = alife_record
local strformat = strformat
local level_name = level.name()
local objects = {}
local obj_to_pos = {}
local sec_to_obj = {}
for i = 1, 65534 do
local obj = get_anomaly_obj(i, level_name)
if obj then
table.insert(objects, obj)
obj_to_pos[i] = {
id = i,
section = obj:section_name(),
position = obj.position
}
local sec = obj:section_name()
sec = common_sec[sec] or sec
if not sec_to_obj[sec] then sec_to_obj[sec] = {} end
table.insert(sec_to_obj[sec], {
id = i,
section = obj:section_name(),
position = obj.position
})
end
end
-- try(function()
-- anomalies_pos_tree = kd_tree.buildTreeSeObjects(objects)
-- anomalies_obj_to_pos = anomalies_pos_tree and obj_to_pos
-- end)
empty_table(detectable_anomalies_ids)
local t = {}
for k, v in pairs(sec_to_obj) do
if sec_to_obj[k] then
local ids = {}
for k1, v1 in pairs(sec_to_obj[k]) do
ids[#ids + 1] = v1.id
if not anomaly_detector_ignore[v1.section] then
detectable_anomalies_ids[v1.id] = v1.position
t[#t + 1] = v1.id
end
end
anomalies_sec_to_obj[k] = kd_tree.buildTreeSeObjectIds(ids)
end
end
detectable_anomalies_pos_tree = kd_tree.buildTreeSeObjectIds(t)
end
-- Rays
local function ray_main(pos1, pos2, args)
local pos1 = vector():set(pos1.x or pos1[1], pos1.y or pos1[2], pos1.z or pos1[3])
local pos2 = vector():set(pos2.x or pos2[1], pos2.y or pos2[2], pos2.z or pos2[3])
local args = args or {}
local pick = ray_pick()
pick:set_position(pos1)
pick:set_direction(pos2:sub(pos1):normalize())
pick:set_flags(args.flags or 2)
pick:set_range(args.range or 200)
if args.ignore_object then
pick:set_ignore_object(args.ignore_object)
end
pick:query()
return pick
end
anomalies_vars = {
-- Quick lookup of sine values by degrees
sin_lut = (function()
local t = {}
for i = 0, 360 do
t[i] = sin(i * 0.0174533)
end
return t
end)(),
-- Quick lookup of cosine values by degrees
cos_lut = (function()
local t = {}
for i = 0, 360 do
t[i] = cos(i * 0.0174533)
end
return t
end)(),
-- Current anomaly factors
factors = {},
add_factor = function(self, anomaly, actor, distance_to_sqr, radius_sqr, section, factor)
self.factors[anomaly.object:name()] = {
section = section or anomaly.object:section(),
factor = factor or 1 - distance_to_sqr / radius_sqr
}
end,
remove_factor = function(self, anomaly, actor, distance_to_sqr, radius_sqr)
self.factors[anomaly.object:name()] = nil
end,
find_max_factor = function(self, anomaly, actor, distance_to_sqr, radius_sqr, condition)
if is_empty(self.factors) then
return 1 - distance_to_sqr / radius_sqr
end
local factor = 0
if condition == nil or condition == true then
condition = function() return true end
elseif condition == false then
condition = function() return false end
end
for k, v in pairs(self.factors) do
if v.factor > factor and condition(v) then
factor = v.factor
end
end
return factor
end,
find_factor_sum = function(self, anomaly, actor, distance_to_sqr, radius_sqr, condition)
if is_empty(self.factors) then
return 1 - distance_to_sqr / radius_sqr
end
local factor = 0
if condition == nil or condition == true then
condition = function() return true end
elseif condition == false then
condition = function() return false end
end
for k, v in pairs(self.factors) do
if condition(v) then
factor = factor + v.factor
end
end
return factor
end,
zone_mine_gravitational_weak_tg = 0,
zone_mine_electric_tg = 0,
zone_mine_electric_pp_effectors = {
{code = 98324, file = "electra_mine.ppe", factor = 0.5},
{code = 98325, file = "electra.ppe", factor = 1},
},
zone_mine_electric_factor_function = function(factors)
return factors.section == "zone_mine_electric"
end
}
-- Defined radius of anomalies behaviour
anomalies_near_actor_radii = {
zone_mine_darkness = anomaly_radii.zone_mine_darkness.max * 3.5,
zone_mine_gold = anomaly_radii.zone_mine_gold.max * 3,
zone_mine_sphere = anomaly_radii.zone_mine_sphere.max * 2,
zone_mine_green_dragon = anomaly_radii.zone_mine_green_dragon.max * 2,
zone_mine_ghost = anomaly_radii.zone_mine_ghost.max * 5,
}
-- Special anomalies behaviour if the actor is near an anomaly
anomalies_near_actor_functions = {
-- Darkness, spawns poltergeist behind the actor
zone_mine_darkness = function(anomaly, actor, distance_to_sqr, radius_sqr)
if not anomaly.spawn_time then anomaly.spawn_time = 0 end
if time_elapsed < anomaly.spawn_time then return end
local spawn_cooldown = 300
local spawn_max_amount = 2
local spawn_table = {
"m_poltergeist_normal_tele",
"m_poltergeist_normal_flame",
}
printf("trying to spawn poltergeist")
if random() * 100 <= 0.5 then
anomaly.spawn_time = time_elapsed + spawn_cooldown
for i = 1, ceil(random() ^ 2 * spawn_max_amount) do
local actor_position = actor:position()
local spawn_position = vector():set(actor_position.x - random_float(5, 7), actor_position.y, actor_position.z - random_float(5, 7))
local spawn_section = spawn_table[random(#spawn_table)]
alife_create(spawn_section, spawn_position, level_vertex_id(spawn_position), alife():actor().m_game_vertex_id)
end
end
end,
-- Green Dragon, lower speed by 50% near it
zone_mine_green_dragon = function(anomaly, actor, distance_to_sqr, radius_sqr)
local key ="zone_mine_green_dragon_speed"
local steps_in = 70
local steps_out = 22
add_simple_timed_effect(1, function()
add_speed(key, ema(key, 0.5, 1, steps_in), false, true)
RemoveTimeEvent(key, key)
end, function()
CreateTimeEvent(key, key, 0, function()
add_speed(key, ema(key, 1, 1, steps_out, device().time_delta * 0.1), false, true)
if smoothed_values[key] >= 0.98 then
remove_speed(key)
return true
end
end)
end, key, 1)
end,
-- Radar, negative psy aura
zone_mine_radar = function(anomaly, actor, distance_to_sqr, radius_sqr)
local h = hit()
h.power = level_environment.is_actor_immune() and 0 or (1 - distance_to_sqr / radius_sqr) * 0.004
if h.power > 0 then
h.type = hit.telepatic
h.impulse = 0
h.direction = VEC_Z
h.draftsman = actor
actor:hit(h)
-- Artefacts protection
local hit_additional = 0
actor:iterate_belt( function(owner, obj)
local sec = obj:section()
local cond = obj:condition()
local immunities_sec = SYS_GetParam(0, sec, "hit_absorbation_sect", sec)
local prot = SYS_GetParam(2, immunities_sec, "telepatic_immunity", 0) * cond
-- Optional modifier for viability
prot = prot * 10
hit_additional = hit_additional + prot
end)
-- Final modifier
local hit_modifier = hit_additional >= 0 and 1 + hit_additional or 1 / (1 - hit_additional)
local actor_hit_power = h.power / hit_modifier * 0.375
change_psy_health(-actor_hit_power)
end
end,
-- Sphere, add 50-100% bullet damage resist
zone_mine_sphere = function(anomaly, actor, distance_to_sqr, radius_sqr)
local key = "zone_mine_sphere"
if not callbacks[key] then
register_callback("actor_on_before_hit", function(s_hit, bone_id, flags)
if s_hit.type == hit.fire_wound then
s_hit.power = s_hit.power * random_float(0, 0.5)
play_sound_on_actor("eugenium_anomaly\\sphere\\sphere_blowout", 0.9, random_float(0.95, 1.05))
local gibs = particles_object("artefact\\artefact_gravi")
if gibs and not gibs:playing() then
gibs:play_at_pos(actor:position())
end
end
end, nil, key)
end
add_simple_timed_effect(1, nil, function()
unregister_callback(key)
end, key, 1)
end,
-- Springboard, camera effect dependant on distance
zone_mine_gravitational_weak = function(anomaly, actor, distance_to_sqr, radius_sqr, power_modifier)
local key = "zone_mine_gravitational_weak"
local tg = time_global()
if tg < anomalies_vars.zone_mine_gravitational_weak_tg and timed_effects[key] then return end
anomalies_vars.zone_mine_gravitational_weak_tg = tg + 1000
local power_modifier = (power_modifier or 2) * settings.gravitational_shake_modifier
local earthquake_cam_eff = 98323
if power_modifier > 0 then
local power = (1.03 - distance_to_sqr / radius_sqr) * power_modifier
level.add_cam_effector("camera_effects\\earthquake_40.anm", earthquake_cam_eff, false, "", 0, false, power)
end
add_simple_timed_effect(1, nil, function()
level.remove_cam_effector(earthquake_cam_eff)
end, key, 1)
end,
zone_mine_gravitational_average = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_gravitational_weak(anomaly, actor, distance_to_sqr, radius_sqr, 1)
end,
zone_mine_gravitational_strong = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_gravitational_weak(anomaly, actor, distance_to_sqr, radius_sqr, 1)
end,
zone_mine_vortex = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_gravitational_weak(anomaly, actor, distance_to_sqr, radius_sqr, 1)
end,
zone_mine_springboard = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_gravitational_weak(anomaly, actor, distance_to_sqr, radius_sqr, 1)
end,
zone_mine_blast = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_gravitational_weak(anomaly, actor, distance_to_sqr, radius_sqr)
end,
zone_gravi_zone = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_gravitational_weak(anomaly, actor, distance_to_sqr, radius_sqr)
end,
zone_mosquito_bald = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_gravitational_weak(anomaly, actor, distance_to_sqr, radius_sqr)
end,
zone_mosquito_bald_weak = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_gravitational_weak(anomaly, actor, distance_to_sqr, radius_sqr)
end,
zone_mosquito_bald_weak_noart = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_gravitational_weak(anomaly, actor, distance_to_sqr, radius_sqr)
end,
zone_mosquito_bald_average = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_gravitational_weak(anomaly, actor, distance_to_sqr, radius_sqr)
end,
zone_mosquito_bald_strong = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_gravitational_weak(anomaly, actor, distance_to_sqr, radius_sqr)
end,
zone_mosquito_bald_strong_noart = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_gravitational_weak(anomaly, actor, distance_to_sqr, radius_sqr)
end,
-- Electra, static field
zone_mine_electric = function(anomaly, actor, distance_to_sqr, radius_sqr, effector_modifier)
local hit_type_shock = HitTypeID["Shock"]
local hit_power = level_environment.is_actor_immune() and 0 or (1 - distance_to_sqr / radius_sqr) ^ 2 * settings.electric_field_modifier
local distance_to = distance_to_sqr ^ 0.5
local reduction_coeff = 1
local anom_pos = anomaly.object:position()
-- anom_pos = level_vertex_id(anom_pos)
-- anom_pos = level_vertex_position(anom_pos)
local anom_pos_adjusted = vector():set(anom_pos.x, anom_pos.y + 0.3, anom_pos.z)
local actor_pos = utils_obj.safe_bone_pos(actor, "bip01_spine")
-- local actor_pos = actor:position()
-- actor_pos = vector():set(actor_pos.x, actor_pos.y + 0.5, actor_pos.z)
local anom_to_actor_ray = ray_main(anom_pos_adjusted, actor_pos, {range = distance_to})
local actor_to_anom_ray = ray_main(actor_pos, anom_pos, {range = distance_to})
if anom_to_actor_ray and actor_to_anom_ray then
local anom_to_actor_distance = anom_to_actor_ray:get_distance()
local actor_to_anom_distance = actor_to_anom_ray:get_distance()
local obstacle_distance = 0
if anom_to_actor_distance > 0 then
obstacle_distance = distance_to - anom_to_actor_distance - actor_to_anom_distance
reduction_coeff = clamp((1 - obstacle_distance / radius_sqr ^ 0.5) ^ 3.5, 0, 1)
end
-- _G.printf("reduction coeff due to obstacle - %s", reduction_coeff)
-- _G.printf("anom_to_actor_distance - %s", anom_to_actor_distance)
-- _G.printf("actor_to_anom_distance - %s", actor_to_anom_distance)
-- _G.printf("obstacle_distance - %s", obstacle_distance)
-- _G.printf("combined distance - %s", anom_to_actor_distance + obstacle_distance + actor_to_anom_distance)
-- _G.printf("input distance - %s", distance_to)
end
hit_power = hit_power * reduction_coeff
if hit_power > 0 then
local hit_additional = 0
-- Outfit protection
local outfit = actor:item_in_slot(7)
if outfit then
local c_obj = outfit:cast_CustomOutfit()
local prot = c_obj and c_obj:GetDefHitTypeProtection(hit_type_shock) or 0
-- Optional modifier for less viability
prot = prot * 1
hit_additional = hit_additional + prot
end
-- Helmet protection
local helm = actor:item_in_slot(12)
if helm then
local c_obj = helm:cast_Helmet()
local prot = c_obj and c_obj:GetDefHitTypeProtection(hit_type_shock) or 0
-- Optional modifier for less viability
prot = prot * 1
hit_additional = hit_additional + prot
end
-- Artefacts protection
local artefacts_protection = 0
actor:iterate_belt( function(owner, obj)
local sec = obj:section()
local cond = obj:condition()
local immunities_sec = SYS_GetParam(0, sec, "hit_absorbation_sect", sec)
local prot = SYS_GetParam(2, immunities_sec, "shock_immunity", 0) * cond
-- Optional modifier for viability
prot = prot * 5
artefacts_protection = artefacts_protection + prot
hit_additional = hit_additional + prot
end)
-- Final modifier
local hit_modifier = hit_additional >= 0 and 1 + hit_additional or 1 / (1 - hit_additional)
local actor_hit_power = hit_power / hit_modifier * 0.0015
-- printf("hit %s", actor_hit_power)
actor:change_health(-actor_hit_power)
-- Affect condition of items
if outfit then
local obj = outfit
local sec = obj:section()
local cond = obj:condition()
if cond > 0.01 then
local immunities_sec = SYS_GetParam(0, sec, "immunities_sect", sec)
local shock_immunity = SYS_GetParam(2, immunities_sec, "shock_immunity", 0) * cond
shock_immunity = shock_immunity + artefacts_protection
local hit_modifier = shock_immunity >= 0 and 1 + shock_immunity or 1 / (1 - shock_immunity)
obj:set_condition(cond - hit_power / hit_modifier * 0.00015)
end
end
if helm then
local obj = helm
local sec = obj:section()
local cond = obj:condition()
if cond > 0.01 then
local immunities_sec = SYS_GetParam(0, sec, "immunities_sect", sec)
local shock_immunity = SYS_GetParam(2, immunities_sec, "shock_immunity", 0) * cond
shock_immunity = shock_immunity + artefacts_protection
local hit_modifier = shock_immunity >= 0 and 1 + shock_immunity or 1 / (1 - shock_immunity)
obj:set_condition(cond - hit_power / hit_modifier * 0.00015)
end
end
end
anomalies_vars:add_factor(anomaly, actor, distance_to_sqr, radius_sqr, "zone_mine_electric", (1 - distance_to_sqr / radius_sqr) ^ 2 * reduction_coeff)
local mine_factor = clamp(anomalies_vars:find_factor_sum(anomaly, actor, distance_to_sqr, radius_sqr, anomalies_vars.zone_mine_electric_factor_function), 0, 1)
-- PPE effector
local effector_modifier = effector_modifier or 0.66
local effector_power = (mine_factor + 0.05) * effector_modifier
local pps = anomalies_vars.zone_mine_electric_pp_effectors
for i = 1, #pps do
local key = "zone_mine_electric" .. pps[i].code
if not timed_effects[key] then
level.add_pp_effector(pps[i].file, pps[i].code, true)
end
level.set_pp_effector_factor(pps[i].code, effector_power * pps[i].factor)
add_simple_timed_effect(0.2, nil, function()
level.remove_pp_effector(pps[i].code)
end, key, 1)
end
-- PDA glitching, set value for binder, patch binder, see below
pda_glitch_value = clamp(mine_factor ^ 0.5, 0, 1)
add_simple_timed_effect(0.2, nil, function()
pda_glitch_value = nil
end, "zone_mine_electric_pda_glitch", 1)
end,
zone_mine_electric_weak = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_electric(anomaly, actor, distance_to_sqr, radius_sqr)
end,
zone_mine_electric_average = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_electric(anomaly, actor, distance_to_sqr, radius_sqr)
end,
zone_mine_electric_strong = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_electric(anomaly, actor, distance_to_sqr, radius_sqr)
end,
zone_mine_static = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_electric(anomaly, actor, distance_to_sqr, radius_sqr)
end,
zone_mine_static_weak = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_electric(anomaly, actor, distance_to_sqr, radius_sqr)
end,
zone_mine_static_average = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_electric(anomaly, actor, distance_to_sqr, radius_sqr)
end,
zone_mine_static_strong = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_electric(anomaly, actor, distance_to_sqr, radius_sqr)
end,
zone_witches_galantine = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_electric(anomaly, actor, distance_to_sqr, radius_sqr)
end,
zone_witches_galantine_weak = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_electric(anomaly, actor, distance_to_sqr, radius_sqr)
end,
zone_witches_galantine_average = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_electric(anomaly, actor, distance_to_sqr, radius_sqr)
end,
zone_witches_galantine_strong = function(anomaly, actor, distance_to_sqr, radius_sqr)
anomalies_near_actor_functions.zone_mine_electric(anomaly, actor, distance_to_sqr, radius_sqr)
end,
-- Ball Lightning, charges batteries
zone_mine_point = function(anomaly, actor, distance_to_sqr, radius_sqr)
for i = 8, 10 do
local item = actor:item_in_slot(i)
if item then
local cond = item:condition()
if cond > 0.01 then
item:set_condition(cond + 0.0006 * (1 - distance_to_sqr / radius_sqr) ^ 2)
end
end
end
anomalies_vars:add_factor(anomaly, actor, distance_to_sqr, radius_sqr, "zone_mine_electric", (1 - distance_to_sqr / radius_sqr) ^ 2)
local mine_factor = clamp(anomalies_vars:find_factor_sum(anomaly, actor, distance_to_sqr, radius_sqr, anomalies_vars.zone_mine_electric_factor_function), 0, 1)
-- PDA glitching, set value for binder, patch binder, see below
pda_glitch_value = clamp(mine_factor ^ 0.5, 0, 1)
add_simple_timed_effect(0.2, nil, function()
pda_glitch_value = nil
end, "zone_mine_electric_pda_glitch", 1)
end,
-- Liquid Gold, drains stamina and thirst
zone_mine_gold = function(anomaly, actor, distance_to_sqr, radius_sqr)
local change_modifier = 1 - distance_to_sqr / radius_sqr
if not anomaly.trigger_threshold then anomaly.trigger_threshold = 0 end
anomaly.trigger_threshold = anomaly.trigger_threshold + change_modifier * 0.5
if anomaly.trigger_threshold >= 1 then
thirst_sleep_changer.change_thirst_small(0.01)
anomaly.trigger_threshold = 0
end
actor:change_power(-change_modifier * 0.002)
end,
}
-- Special behaviour for anomalies after spawn
anomalies_spawn_functions = {
-- Ghost, wandering path along its start position
zone_mine_ghost = function(anomaly)
if anomaly.spawn_position then
local spawn_position = anomaly.spawn_position
local obj = anomaly.object
local obj_set_anomaly_position = obj.set_anomaly_position
local vector = vector()
local vector_set = vector.set
local key = obj:name()
local sin_lut = anomalies_vars.sin_lut
local cos_lut = anomalies_vars.cos_lut
local i = random_choice(0, 45, 90, 135, 180)
local i_step = random_choice(1, 2, 3)
local j = 0
local j_step = 1
if i_step == 1 then
j_step = random_choice(2, 3)
elseif i_step == 2 then
j_step = random_choice(1, 3)
elseif i_step == 3 then
j_step = random_choice(2, 4)
end
local full_circle = 360
local function generate_distance()
return random() * 2.5 + 0.5
end
local x_modifier = generate_distance() * random_choice(1, -1)
local z_modifier = generate_distance() * random_choice(1, -1)
local function wrap(i)
return i >= full_circle and i % full_circle or i
end
local max_offset_y = 0.5
local max_offset_y_low = spawn_position.y - max_offset_y
local max_offset_y_high = spawn_position.y + max_offset_y
local prev_pos_x = spawn_position.x
local prev_pos_y = spawn_position.y
local prev_pos_z = spawn_position.z
register_callback("actor_on_update", function()
if i >= full_circle then
i = i % full_circle
end
if j >= full_circle then
j = j % full_circle
end
local offset_x = spawn_position.x + sin_lut[i] * x_modifier
local offset_z = spawn_position.z + sin_lut[j] * z_modifier
local offset_y = spawn_position.y
-- Adjust to height by checking lvid
local lvid = level_vertex_id(vector_set(vector, offset_x, offset_y, offset_z))
if (lvid < 4294967295) then
local pos_y = level_vertex_position(lvid).y
if abs(pos_y - prev_pos_y) < max_offset_y then
offset_y = pos_y
end
-- offset_y = clamp(pos_y, max_offset_y_low, max_offset_y_high)
end
obj_set_anomaly_position(obj, vector_set(vector, offset_x, offset_y, offset_z))
prev_pos_x = offset_x
prev_pos_y = offset_y
prev_pos_z = offset_z
local delta = min(50, device().time_delta) / 22 -- Frame independent movement, less than 20 fps will be slower due to prevent sudden pops
i = floor(i + i_step * delta)
j = floor(j + j_step * delta)
end, nil, key)
end
end,
}
-- Special behaviour for anomalies after destroy
anomalies_destroy_functions = {
-- Ghost, destroy callback for wandering
zone_mine_ghost = function(anomaly)
local key = anomaly.object:name()
unregister_callback(key)
end,
}
anomaly_detector_ignore = {
zone_field_radioactive = true,
zone_field_radioactive_above_average = true,
zone_field_radioactive_average = true,
zone_field_radioactive_below_average = true,
zone_field_radioactive_lethal = true,
zone_field_radioactive_strong = true,
zone_field_radioactive_very_weak = true,
zone_field_radioactive_weak = true,
zone_radioactive = true,
zone_radioactive_above_average = true,
zone_radioactive_average = true,
zone_radioactive_below_average = true,
zone_radioactive_lethal = true,
zone_radioactive_strong = true,
zone_radioactive_very_weak = true,
zone_radioactive_weak = true,
zone_field_acidic = true,
zone_field_acidic_average = true,
zone_field_acidic_strong = true,
zone_field_acidic_weak = true,
zone_field_psychic = true,
zone_field_psychic_average = true,
zone_field_psychic_strong = true,
zone_field_psychic_weak = true,
zone_field_thermal = true,
zone_field_thermal_average = true,
zone_field_thermal_strong = true,
zone_field_thermal_weak = true,
zone_mine_field = true,
zone_mine_field_soc = true,
campfire = true,
campfire_base = true,
campfire_base_noshadow = true,
zone_base = true,
zone_base_noshadow = true,
zone_burning_fuzz = true,
zone_burning_fuzz1 = true,
zone_burning_fuzz_weak = true,
zone_burning_fuzz_average = true,
zone_burning_fuzz_strong = true,
zone_buzz = true,
zone_buzz_weak = true,
zone_buzz_average = true,
zone_buzz_strong = true,
zone_emi = true,
zone_liana = true,
zone_student = true,
zone_teleport = true,
zone_zhar = true,
-- MP items
mp_af_electra_flash = true,
mp_zone_witches_galantine = true,
mp_af_cta_green = true,
mp_af_cta_blue = true,
mp_medkit = true,
mp_medkit_scientic = true,
mp_medkit_army = true,
mp_energy_drink = true,
mp_bandage = true,
mp_antirad = true,
mp_drug_coagulant = true,
mp_drug_radioprotector = true,
mp_medkit_old = true,
mp_antirad_old = true,
mp_detector_advanced = true,
mp_device_torch = true,
mp_players_rukzak = true,
mp_wood_stolb_fixed = true,
mp_wood_stolb_fixed_immunities = true,
mp_explosive_fuelcan = true,
mp_explosive_tank = true,
mp_explosive_barrel = true,
}
function find_nearest_anomaly(force_enabled)
local actor = db.actor
local actor_pos = actor:position()
local actor_pos_distance = actor_pos.distance_to_sqr
local distance = math.huge
local anom_id
local anom_pos
local anom_sec
local pairs = pairs
local level_object_by_id = level.object_by_id
local level_name = level.name()
for id, _ in pairs(detectable_anomalies_ids) do
local obj = level_object_by_id(id)
if obj then
local enabled = true
if not force_enabled then
local props = updated_anomaly_levels[level_name].anomalies_properties[id]
if props and props.active ~= nil then
enabled = props.active
end
end
if enabled then
local pos = obj:position()
local d = actor_pos_distance(actor_pos, pos)
if d < distance then
distance = d
anom_id = id
anom_pos = pos
anom_sec = obj:section()
end
end
end
end
return distance, anom_id, anom_sec, anom_pos
end
detector_functions = {
-- Play sound
play_anomaly_sound = function(self, anomaly, actor, distance_to_sqr, radius_sqr)
if not anomaly.sound_threshold then anomaly.sound_threshold = 0 end
anomaly.sound_threshold = anomaly.sound_threshold + (1 - distance_to_sqr / radius_sqr) ^ 2
if anomaly.sound_threshold > 1 then
play_sound_on_actor("detectors\\da-2_beep1")
anomaly.sound_threshold = 0
end
end,
-- Store anomaly factors to find biggest one for correct display of anomaly_detector bars
detector_anomaly_factors = {},
detector_anomaly_add_factor = function(self, anomaly, actor, distance_to_sqr, radius_sqr, custom_callback)
self.detector_anomaly_factors[anomaly.object:name()] = 1 - distance_to_sqr / radius_sqr
end,
detector_anomaly_remove_factor = function(self, anomaly, actor, distance_to_sqr, radius_sqr, custom_callback)
self.detector_anomaly_factors[anomaly.object:name()] = nil
end,
detector_anomaly_find_max_factor = function(self, anomaly, actor, distance_to_sqr, radius_sqr, custom_callback)
if is_empty(self.detector_anomaly_factors) then
return 1 - distance_to_sqr / radius_sqr
end
local factor = 0
for k, v in pairs(self.detector_anomaly_factors) do
if v > factor then
factor = v
end
end
return factor
end,
detector_anomaly = function(self, anomaly, actor, distance_to_sqr, radius_sqr, custom_callback)
if not custom_callback then return end
if actor:active_detector() then
local sec = anomaly.section
self:detector_anomaly_add_factor(anomaly, actor, distance_to_sqr, radius_sqr, custom_callback)
local factor = self:detector_anomaly_find_max_factor(anomaly, actor, distance_to_sqr, radius_sqr, custom_callback)
local ui = tasks_measure.get_UI()
if ui and ui.step <= 0 then
for i = 1, ui.step_tot do
ui.m_seg[i]:InitTextureEx(i <= min(ui.step_tot, ceil(factor * 10)) and "green_seg_ano" or "green_black_ano", "hud\\p3d")
end
end
-- Play sound when detector is active
if game_difficulties.get_game_factor("notify_anomaly")
and (drx_da_main_mcm.new_anomalies_sections[sec] or drx_da_main_mcm.variations_anomalies_sections[sec])
or (not game_difficulties.get_game_factor("notify_anomaly") and actor:active_detector())
then
self:play_anomaly_sound(anomaly, actor, distance_to_sqr, radius_sqr)
end
else
local ui = tasks_measure.get_UI()
if ui and ui.step <= 0 then
for i = 1, ui.step_tot do
ui.m_seg[i]:InitTextureEx("green_black_ano", "hud\\p3d")
end
end
end
end,
detector_scientific = function(self, anomaly, actor, distance_to_sqr, radius_sqr, custom_callback)
if custom_callback then return end
local sec = anomaly.section
if drx_da_main_mcm.new_anomalies_sections[sec] or drx_da_main_mcm.variations_anomalies_sections[sec] then
self:play_anomaly_sound(anomaly, actor, distance_to_sqr, radius_sqr)
end
end
}
function notify_anomaly(anomaly, actor, distance_to_sqr, radius_sqr, custom_callback)
local detector = actor:item_in_slot(9)
local sec = detector and detector:section()
if sec and detector_functions[sec] then
detector_functions[sec](detector_functions, anomaly, actor, distance_to_sqr, radius_sqr, custom_callback)
elseif game_difficulties.get_game_factor("notify_anomaly") and (drx_da_main_mcm.new_anomalies_sections[anomaly.section] or drx_da_main_mcm.variations_anomalies_sections[anomaly.section]) and not custom_callback then
detector_functions:play_anomaly_sound(anomaly, actor, distance_to_sqr, radius_sqr)
end
end
bind_anomaly_field_spawn = bind_anomaly_field.anomaly_field_binder.net_spawn
bind_anomaly_field.anomaly_field_binder.net_spawn = function(self, se_abstract)
if not bind_anomaly_field_spawn(self, se_abstract) then
return false
end
self.spawn_position = self.object:position()
self.section = self.object:section()
if not anomaly_detector_ignore[self.section] then
detectable_anomalies_ids[self.object:id()] = self.spawn_position
end
local section = self.section
if anomalies_spawn_functions[section] then
anomalies_spawn_functions[section](self)
end
-- Update callback similar to binder update, but works even if anomaly is disabled
self.on_update_timer_default = 100
self.on_update_timer_max = 7500
self.on_update_timer = self.on_update_timer_default
self.on_update_time = time_global() + random(self.on_update_timer_default, self.on_update_timer_default * 10)
self.update_key = self.object:name() .. "_update"
local anom_zone_data = utils_stpk.get_anom_zone_data(alife_object(self.object:id()))
if anom_zone_data and anom_zone_data.shapes[1] and anom_zone_data.shapes[1].radius then
self.radius_sqr = anomalies_near_actor_radii[section] or ((anom_zone_data.shapes[1].radius + 1) * 2)
self.radius_sqr = self.radius_sqr * self.radius_sqr
-- printf("getting anom radius from net packet, %s", self.radius_sqr)
else
self.radius_sqr = get_anomaly_behaviour_radius(section)
end
self.radius = sqrt(self.radius_sqr)
self.behaviour_radius_sqr = get_anomaly_behaviour_radius(section)
local level_name = level.name()
register_callback("actor_on_update", function()
local tg = time_global()
if tg < self.on_update_time then return end
-- Get behaviour radius and check if actor inside it, then apply effect
local actor = db.actor
local radius_sqr = self.radius_sqr
local distance_to_sqr = self.object:position():distance_to_sqr(actor:position())
if distance_to_sqr <= radius_sqr then
-- Beep near anoms even if disabled for anomaly_detector
if not (
anomaly_detector_ignore[self.section]
or (
level_name
and updated_anomaly_levels[level_name]
and updated_anomaly_levels[level_name].disabled
and updated_anomaly_levels[level_name].anomalies_properties
and updated_anomaly_levels[level_name].anomalies_properties[self.object:id()]
)
) then
local og_dyn_anomalies = bind_anomaly_field.dyn_anomalies
if not og_dyn_anomalies then
notify_anomaly(self, actor, distance_to_sqr, radius_sqr, true)
else
local level_name = level.name()
if og_dyn_anomalies[level_name] and og_dyn_anomalies[level_name][self.object:id()] ~= nil then
if og_dyn_anomalies[level_name][self.object:id()] then
notify_anomaly(self, actor, distance_to_sqr, radius_sqr, true)
end
else
notify_anomaly(self, actor, distance_to_sqr, radius_sqr, true)
end
end
end
self.on_update_timer = self.on_update_timer_default
else
detector_functions.detector_anomaly_remove_factor(detector_functions, self, actor, distance_to_sqr, radius_sqr, true)
self.on_update_timer = clamp(self.on_update_timer_default * (distance_to_sqr / radius_sqr * 0.5), self.on_update_timer_default, self.on_update_timer_max)
-- _G.printf("%s, on_update_timer %s", self.object:name(), self.on_update_timer)
end
self.on_update_time = tg + self.on_update_timer
end, nil, self.update_key)
return true
end
bind_anomaly_field_destroy = bind_anomaly_field.anomaly_field_binder.net_destroy
bind_anomaly_field.anomaly_field_binder.net_destroy = function(self)
local section = self.section
detectable_anomalies_ids[self.object:id()] = nil
anomalies_vars.remove_factor(anomalies_vars, self, actor, distance_to_sqr, radius_sqr)
detector_functions.detector_anomaly_remove_factor(detector_functions, self, actor, distance_to_sqr, radius_sqr, true)
if anomalies_destroy_functions[section] then
anomalies_destroy_functions[section](self)
end
unregister_callback(self.update_key)
bind_anomaly_field_destroy(self)
end
-- Behaviour radius in priorities
-- 1. anomalies_near_actor_radii[section] -- specially defined behaviour radius
-- 2. anomaly_radii[section].max -- defined hit radius of anomaly
-- 3. max_radius -- max hit radius defined in level config
-- 4. 3 -- default radius
-- 5. p.2,3,4 or 5 is then added 1 and multiplied by 2
function get_anomaly_behaviour_radius(section)
local radius = anomalies_near_actor_radii[section] or (((anomaly_radii[section] and anomaly_radii[section].max or max_radius or 3) + 1) * 2)
local radius_sqr = radius * radius
return radius_sqr
end
-- Special behaviour is actor near an anomaly
bind_anomaly_field_update = bind_anomaly_field.anomaly_field_binder.update
bind_anomaly_field.anomaly_field_binder.update = function(self, delta)
bind_anomaly_field_update(self, delta)
if not self.object then return end
local section = self.section
if anomalies_near_actor_functions[section] or (additional_articles_to_category.encyclopedia_anomalies[section] and not opened_articles.encyclopedia_anomalies[additional_articles_to_category.encyclopedia_anomalies[section]]) then
-- Get behaviour radius and check if actor inside it, then apply effect
local actor = db.actor
local radius_sqr = self.radius_sqr
local distance_to_sqr = self.object:position():distance_to_sqr(actor:position())
if distance_to_sqr <= radius_sqr then
-- Open anomaly article
open_anomaly_article(section)
-- Beep near anoms if option enabled or have Svarog or Anomaly Detector
if not anomaly_detector_ignore[section] then
notify_anomaly(self, actor, distance_to_sqr, radius_sqr, false)
end
-- Behaviour near actor
if anomalies_near_actor_functions[section] then
anomalies_near_actor_functions[section](self, actor, distance_to_sqr, radius_sqr)
end
-- printf("actor near anomaly %s, firing effect, delta %s", section, delta)
else
anomalies_vars.remove_factor(anomalies_vars, self, actor, distance_to_sqr, radius_sqr)
end
end
if npc_on_near_anomalies_functions[section] then
if not self.iterate_nearest_func then
self.iterate_nearest_func = function(obj)
if obj
and (IsStalker(obj) or IsMonster(obj))
and (obj.alive and obj:alive())
and obj:id() ~= AC_ID
and obj:position():distance_to_sqr(self.object:position()) <= self.radius_sqr
then
npc_on_near_anomalies_functions[section](self, obj, db.actor)
end
end
end
level.iterate_nearest(self.object:position(), self.radius, self.iterate_nearest_func)
end
end
-- Apply glitches and flickers to active items near electrical anomalies
-- See above how value is set
pda_glitch_value = nil
process_glitch = item_device.device_binder.process_glitch
item_device.device_binder.process_glitch = function(self, id, section, condition)
process_glitch(self, id, section, condition)
if pda_glitch_value then
self.object:set_psy_factor(pda_glitch_value)
end
end
process_flicker = item_device.device_binder.process_flicker
item_device.device_binder.process_flicker = function(self, force)
process_flicker(self, pda_glitch_value and pda_glitch_value > 0.4 or force)
end
process_torch = item_device.device_binder.process_torch
item_device.device_binder.process_torch = function(self, id, section, condition)
process_torch(self, id, section, condition)
-- Beef's NVG integration
if z_beefs_nvgs then
if self.N_V then
z_beefs_nvgs.nvg_glitch(clamp(pda_glitch_value or 0, 0, 0.9))
else
z_beefs_nvgs.nvg_glitch(0)
end
end
end
local actor_on_update_time = 0
local actor_on_update_timer = 100
function actor_on_update()
local tg = time_global()
if tg < actor_on_update_time then return end
actor_on_update_time = tg + actor_on_update_timer
time_elapsed = get_time_elapsed()
process_timed_effects()
end
function actor_on_interaction(typ, obj, name)
-- check if emission happened globally
if typ == "anomalies" and (name == "emission_end" or name == "psi_storm_end") then
local level_name = level.name()
init_anomaly_table_on_level(level_name)
-- 50/50 chance to remove anomalies globally or just update artefacts
if random(100) <= 50 then
CreateTimeEvent("clean_dynamic_anomalies_global", "clean_dynamic_anomalies_global", 0.5, function()
last_surge_time = get_time_elapsed()
printf("surge happened globally at %s", last_surge_time)
printf("update on level %s after emission", level.name())
clean_dynamic_anomalies_global()
build_anomalies_pos_tree()
return true
end)
else
printf("surge happened globally at %s", get_time_elapsed())
printf("update artefacts on level %s after emission", level.name())
clean_artefacts_on_level(level.name())
spawn_artefacts_on_level(level.name())
build_anomalies_pos_tree()
end
end
end
function actor_on_item_take(obj)
local level_name = level.name()
local id = obj:id()
if updated_anomaly_levels[level_name] and updated_anomaly_levels[level_name].artefacts and updated_anomaly_levels[level_name].artefacts[id] then
printf("taken created artefact %s, id %s, level_name %s", updated_anomaly_levels[level_name].artefacts[id], id, level_name)
updated_anomaly_levels[level_name].artefacts[id] = nil
end
end
function npc_on_item_take(npc, obj)
actor_on_item_take(obj)
end
-- Anomalies special hit behaviour
anomalies_hit_functions = {
-- Flash special hit behaviour - Time travel
zone_mine_flash = function(s_hit, bone_id, flags, actor)
printf("change_time")
local health = actor.health
local change_hours = random(3, 8)
local change_minutes = random(1, 59)
level.change_game_time(0, change_hours, change_minutes)
level_weathers.get_weather_manager():select_weather(true)
surge_manager.get_surge_manager().time_forwarded = true
psi_storm_manager.get_psi_storm_manager().time_forwarded = true
s_hit.power = 0.001
local fatigue_light_state = true
if fatigue_light then
fatigue_light_state = fatigue_light.enabled
fatigue_light.enabled = false
end
CreateTimeEvent("zone_mine_flash", "zone_mine_flash", 0.1, function()
local new_health = actor.health
actor:set_health_ex(health)
actor:change_health(-random_float(0.01, 0.04))
-- Change thirst and sleep, params are from vanilla actor_status_sleep/thirst scripts
local sleep_params = {
step = 27,
check_after_sec = 300,
}
local thirst_params = {
step = 30,
check_after_sec = 300
}
local change_sleep_amount = (change_hours * 3600 + change_minutes * 60) / sleep_params.check_after_sec * sleep_params.step * 0.01
local change_thirst_amount = (change_hours * 3600 + change_minutes * 60) / thirst_params.check_after_sec * thirst_params.step * 0.01
thirst_sleep_changer.change_sleep(round(change_sleep_amount))
thirst_sleep_changer.change_thirst(round(change_thirst_amount))
if fatigue_light then
fatigue_light.enabled = fatigue_light_state
end
return true
end)
end,
-- Radiation field: drain batteries of active items
-- zone_field_radioactive = function(s_hit, bone_id, flags, actor)
-- if s_hit.power <= 0 then return end
-- for i = 8, 10 do
-- local item = actor:item_in_slot(i)
-- if item then
-- local cond = item:condition()
-- if cond > 0.01 then
-- item:set_condition(cond - 0.03 * s_hit.power)
-- end
-- end
-- end
-- end,
-- zone_field_radioactive_weak = function(s_hit, bone_id, flags, actor)
-- anomalies_hit_functions.zone_field_radioactive(s_hit, bone_id, flags, actor)
-- end,
-- zone_field_radioactive_average = function(s_hit, bone_id, flags, actor)
-- anomalies_hit_functions.zone_field_radioactive(s_hit, bone_id, flags, actor)
-- end,
-- zone_field_radioactive_strong = function(s_hit, bone_id, flags, actor)
-- anomalies_hit_functions.zone_field_radioactive(s_hit, bone_id, flags, actor)
-- end,
-- zone_radioactive = function(s_hit, bone_id, flags, actor)
-- anomalies_hit_functions.zone_field_radioactive(s_hit, bone_id, flags, actor)
-- end,
-- zone_radioactive_weak = function(s_hit, bone_id, flags, actor)
-- anomalies_hit_functions.zone_field_radioactive(s_hit, bone_id, flags, actor)
-- end,
-- zone_radioactive_average = function(s_hit, bone_id, flags, actor)
-- anomalies_hit_functions.zone_field_radioactive(s_hit, bone_id, flags, actor)
-- end,
-- zone_radioactive_strong = function(s_hit, bone_id, flags, actor)
-- anomalies_hit_functions.zone_field_radioactive(s_hit, bone_id, flags, actor)
-- end,
}
-- Anomalies special hit behaviour on monster
anomalies_monster_hit_functions = {
-- Flash special hit behaviour - Time travel
zone_mine_flash = function(monster, s_hit, bone_id, flags, actor)
printf("flash got hit %s", monster:section())
s_hit.power = 0
flags.ret_value = false
safe_release_manager.release(alife_object(monster:id()))
end,
-- Reduce power for Rebounder
-- zone_mine_sphere = function(monster, s_hit, bone_id, flags, actor)
-- local sec = monster:section()
-- printf("rebounder got hit %s", sec)
-- s_hit.power = 0
-- flags.ret_value = false
-- local immunities_sec = SYS_GetParam(0, sec, "protections_sect", sec)
-- local hit_power = SYS_GetParam(2, "zone_mine_sphere", "max_start_power", 1) * SYS_GetParam(2, immunities_sec, "hit_fraction_monster", 1)
-- monster:change_health(-hit_power)
-- end,
}
-- Also used for monsters, filter for npcs or monsters specifically in functions bodies
npc_on_near_anomalies_functions = {
-- Rebounder effect for NPCs
zone_mine_sphere = function(anomaly, npc, actor)
local key = "zone_mine_sphere_" .. npc:name()
if not callbacks[key] then
register_callback(IsStalker(npc) and "npc_on_before_hit" or "monster_on_before_hit", function(npc_hit, s_hit, bone_id, flags)
if s_hit.type == hit.fire_wound and npc_hit:name() == npc:name() then
s_hit.power = s_hit.power * random_float(0, 0.5)
play_sound_on_actor("eugenium_anomaly\\sphere\\sphere_blowout", 1, random_float(0.95, 1.05), anomaly.object)
local gibs = particles_object("artefact\\artefact_gravi")
if gibs and not gibs:playing() then
gibs:play_at_pos(npc_hit:position())
end
end
end, nil, key)
end
add_simple_timed_effect(0.7, nil, function()
unregister_callback(key)
end, key, 1)
end,
}
function actor_on_before_hit(s_hit, bone_id, flags)
if not s_hit.draftsman then return end
local sec = s_hit.draftsman:section()
if anomalies_hit_functions[sec] then
anomalies_hit_functions[sec](s_hit, bone_id, flags, db.actor)
end
end
function monster_on_before_hit(monster, s_hit, bone_id, flags)
if not s_hit.draftsman then return end
local sec = s_hit.draftsman:section()
if anomalies_monster_hit_functions[sec] then
anomalies_monster_hit_functions[sec](monster, s_hit, bone_id, flags, db.actor)
end
end
function load_state(m_data)
local dump = string.dump
local load = loadstring
local unpack = unpack
local t = m_data.drx_da_effects or {}
for key, effect in pairs(t) do
if t[key].effect_function then
local effect = load(t[key].effect_function)
t[key].effect = function()
effect(unpack(t[key].effect_args))
end
end
if t[key].on_end_function then
local on_end = load(t[key].on_end_function)
t[key].on_end = function()
on_end(unpack(t[key].on_end_args))
end
end
end
timed_effects = t
last_surge_time = m_data.drx_da_last_surge_time or 0
updated_anomaly_levels = m_data.drx_da_updated_anomaly_levels or {}
if m_data.drx_da_opened_articles then opened_articles = m_data.drx_da_opened_articles end
RegisterScriptCallback("actor_on_update", drx_da_actor_on_update_callback)
end
function actor_on_first_update()
load_state(alife_storage_manager.get_state())
end
function save_anomalies(m_data)
local t = {}
for key, effect in pairs(timed_effects) do
if effect.save then
t[key] = {}
copy_table(t[key], effect)
t[key].effect = nil
t[key].on_end = nil
end
end
m_data.drx_da_effects = t
m_data.drx_da_updated_anomaly_levels = updated_anomaly_levels
m_data.drx_da_last_surge_time = last_surge_time
m_data.drx_da_opened_articles = opened_articles
end
function save_level(m_data)
m_data.drx_da_previous_level = level.name()
end
function on_before_level_changing()
printf("saving previous_level")
alife_storage_manager.get_state().drx_da_previous_level = level.name()
UnregisterScriptCallback("save_state", save_level)
end
local function on_option_change()
load_settings()
if settings.delete_dynamic_anomalies then
unregister_anomalies_behaviour()
force_clean_dynamic_anomalies_global()
build_anomalies_pos_tree()
if ui_mcm then
ui_mcm.set("drx_da/delete_dynamic_anomalies", false)
end
return
end
unregister_anomalies_behaviour()
if updated_anomaly_levels[level.name()] and not updated_anomaly_levels[level.name()].disabled then
if settings.enable_anomalies_behaviour or level_weathers.bLevelUnderground then
register_anomalies_behaviour()
else
enable_anomalies_on_level()
end
end
end
-- Register callback scripts:
function on_game_start()
RegisterScriptCallback("actor_on_before_hit", actor_on_before_hit)
RegisterScriptCallback("monster_on_before_hit", monster_on_before_hit)
RegisterScriptCallback("actor_on_item_take", actor_on_item_take)
RegisterScriptCallback("npc_on_item_take", npc_on_item_take)
RegisterScriptCallback("actor_on_interaction", actor_on_interaction)
RegisterScriptCallback("on_before_level_changing", on_before_level_changing)
RegisterScriptCallback("save_state", save_anomalies)
RegisterScriptCallback("save_state", save_level)
-- RegisterScriptCallback("load_state", load_state)
RegisterScriptCallback("actor_on_first_update", actor_on_first_update)
RegisterScriptCallback("on_game_load", load_settings)
RegisterScriptCallback("on_option_change", on_option_change)
end
-- Patches
-- Encyclopedia articles about new anomalies
-- Table contains category and entries in game section => article string id format
additional_articles_to_category = {
encyclopedia_anomalies = {
zone_mine_radar = "encyclopedia_anomalies_radar",
zone_mine_darkness = "encyclopedia_anomalies_darkness",
zone_mine_flash = "encyclopedia_anomalies_flash",
zone_mine_ghost = "encyclopedia_anomalies_ghost",
zone_mine_gold = "encyclopedia_anomalies_gold",
zone_mine_green_dragon = "encyclopedia_anomalies_green_dragon",
zone_mine_mefistotel = "encyclopedia_anomalies_mefistotel",
zone_mine_net = "encyclopedia_anomalies_net",
zone_mine_point = "encyclopedia_anomalies_point",
zone_mine_sphere = "encyclopedia_anomalies_sphere",
}
}
-- Table contains opened articles after interacting with anomalies
opened_articles = {
encyclopedia_anomalies = {}
}
-- Open anomaly article
function open_anomaly_article(section)
-- If there is no article to begin with - return
if not additional_articles_to_category.encyclopedia_anomalies[section] then
printf("article not found for %s", section)
return
end
-- If already opened - return
if opened_articles.encyclopedia_anomalies[additional_articles_to_category.encyclopedia_anomalies[section]] then
printf("article already opened for %s", section)
return
end
-- Return if the player didn't spend 3 hours in the zone
if not (has_alife_info("actor_spent_2_hrs_in_zone") or utils_obj.is_time_spent_in_zone(0,0,2)) then
printf("youre too stupid for article boy")
return
end
-- Add statistics and notify
game_statistics.increment_statistic("articles")
actor_menu.set_notification(nil, "ui_inGame2_notify_article", 20, "device\\pda\\pda_guide_2")
-- Add to already opened articles
opened_articles.encyclopedia_anomalies[additional_articles_to_category.encyclopedia_anomalies[section]] = true
-- Instance of the pda_ui object.
local pda_ui = ui_pda_encyclopedia_tab.get_ui()
pda_ui:InitCategories()
pda_ui:SelectCategory("encyclopedia_anomalies")
pda_ui:SelectArticle(additional_articles_to_category.encyclopedia_anomalies[section])
if pda_ui.article_list and pda_ui.article_list.GetSelectedIndex then
pda_ui.article_list:SetSelectedIndex(pda_ui.article_list:GetSelectedIndex() + 1)
end
if zz_Encyclopedia_messages_restored or zz_encyclopedia_messages_restored or encyclopedia_messages_restored then
-- Article and category texts.
local message = game.translate_string("pda_encyclopedia_notify")
--printf("Monkeyzz " .. categories[index_c].section)
local text_c = game.translate_string("encyclopedia_anomalies")
local text_a = game.translate_string(additional_articles_to_category.encyclopedia_anomalies[section])
-- Other information.
local header = game.translate_string("st_tip")
local texture = news_manager.tips_icons["guide_unlock"] or "ui_inGame2_Poslednie_razrabotki"
db.actor:give_game_news(header, strformat(message, text_c, text_a), texture, 0, 5000, 0)
end
end
-- Patch article list
InitArticles = ui_pda_encyclopedia_tab.pda_encyclopedia_tab.InitArticles
ui_pda_encyclopedia_tab.pda_encyclopedia_tab.InitArticles = function(self, section_c)
InitArticles(self, section_c)
-- If not existing list - return
if is_empty(opened_articles[section_c]) then return end
local n = self.article_list:GetSize()
for article, _ in pairs(opened_articles[section_c]) do
local clr = ui_pda_encyclopedia_tab.UpdateColor(article)
local item = ui_pda_encyclopedia_tab.pda_encyclopedia_entry(article, n, clr)
self.article_list:AddExistingItem(item)
n = n + 1
end
end
-- Extra Utils
function reset_timers()
last_surge_time = 0
for k, v in pairs(updated_anomaly_levels) do
v.time = -1
printf("time resetted for %s", k)
end
printf("last_surge_time is 0")
end
function print_anomalies()
return print_r(updated_anomaly_levels)
end
function debug_spawn_anomaly(sec)
local gvid = db.actor:game_vertex_id()
local r = level.get_target_dist and level.get_target_dist() or 3
local pos = vector():set(db.actor:position())
pos:add(device().cam_dir:mul(r))
pos = vector():set(pos.x,db.actor:position().y,pos.z)
local lvid = level.vertex_id(pos)
local level_name = level.name()
local level_data = get_level_data(level_name)
if not level_data then
printf("Error, unable to get data for %s", level_name)
return
end
local level_file = level_data.level_file
drx_da_spawn_anomaly(sec, pos, lvid, gvid, level_file)
build_anomalies_pos_tree()
end
-- Buggy consecutive update leading to softlocks and crashes
-- Refresh anomalies on level after surge, in concurrent fashion
function update_dynamic_anomalies_on_level_after_surge(level_name)
local level_name = level_name or level.name()
-- Clean all anomalies in one action first
clean_dynamic_anomalies_global()
local function generate_anomalies()
local level_data = get_level_data(level_name)
if not level_data then
printf("Error, unable to get data for %s", level_name)
return
end
local level_file = level_data.level_file
local spawn_percent = level_data.spawn_percent
local anomaly_max_number = level_data.anomaly_max_number
local anomaly_max_active = level_data.anomaly_max_active
-- Build a list of available smart terrains:
local smart_list = utils_data.collect_section(level_file, "available_smarts")
local pairs = pairs
local collect_section = utils_data.collect_section
local size_table = size_table
local to_generate = {}
-- Build anomalies table
if is_not_empty(smart_list) then
local anomalies = updated_anomaly_levels[level_name].anomalies or {}
local anomalies_by_smart = updated_anomaly_levels[level_name].anomalies_by_smart or {}
local smart_by_anomalies = updated_anomaly_levels[level_name].smart_by_anomalies or {}
for i, smart_name in pairs(smart_list) do
if true or not smart_restrictions[smart_name] then
-- Choose an anomaly type to spawn:
local anomaly_list = collect_section(level_file, "anomaly_types")
if anomaly_list and #anomaly_list >= 1 then
local anomaly_type = anomaly_list[random(#anomaly_list)]
printf("picked anomaly type %s", anomaly_type)
if not anomalies_by_smart[smart_name] then anomalies_by_smart[smart_name] = {} end
local j = size_table(anomalies_by_smart[smart_name])
while j < anomaly_max_number do
if random() <= spawn_percent then
to_generate[#to_generate + 1] = {
anomaly_type = anomaly_type,
level_file = level_file,
smart_name = smart_name,
level_name = level_name
}
end
j = j + 1
end
else
printf("No dynamic anomaly types specified for level %s", level_name)
end
else
printf("no anomalies spawn on smart %s, level %s in restriction", smart_name, level_name)
end
end
end
return to_generate
end
local anomalies = generate_anomalies()
if is_not_empty(anomalies) then
local tg_interval = 100
local tg = time_global() + 500
local anomalies_ids = {}
process_queue("drx_da_spawn_anomalies_on_level", anomalies, function(k, anomaly)
local t = time_global()
if t < tg then return end
tg = t + tg_interval
local anom_id = drx_da_spawn_anomaly_on_smart(anomaly.level_file, anomaly.smart_name, anomaly.anomaly_type, level_name)
if anom_id then
updated_anomaly_levels[level_name].anomalies[#updated_anomaly_levels[level_name].anomalies + 1] = anom_id
updated_anomaly_levels[level_name].anomalies_by_smart[anomaly.smart_name][anom_id] = true
updated_anomaly_levels[level_name].anomaly_types_by_smart[anomaly.smart_name] = anomaly.anomaly_type
updated_anomaly_levels[level_name].smart_by_anomalies[anom_id] = anomaly.smart_name
end
return true
end, function()
printf("anomalies updated after surge on level %s", level_name)
end)
end
end
-- Removes all anomalies forcefully
function force_clean_dynamic_anomalies_global()
remove_all_restrictions()
get_obj_restrictions(true)
unregister_anomalies_behaviour()
local alife_release_id = alife_release_id
local gg = game_graph()
local gg_vertex = gg.vertex
local level_name = level.name()
local load_var = load_var
local pairs = pairs
local printf = printf
local sim = alife()
local sim_level_name = sim.level_name
local sim_object = sim.object
local sim_release = sim.release
local alife_record = alife_record
local strformat = strformat
for k, v in pairs(updated_anomaly_levels) do
get_anomalies_by_smart(k)
clean_artefacts_on_level(k)
if false then
-- disable_anomalies_on_level(level_name)
-- v.disabled = true
-- clean_dynamic_anomalies_on_level_func(level_name)
else
for k1, v1 in pairs(v.anomalies) do
if true then
local se_obj = sim_object(sim, v1)
if se_obj then
if IsAnomaly(_, se_obj:clsid()) then
printf("Deleting anomaly %s, %s globally, level %s", se_obj:section_name(), v1, k)
alife_record(se_obj ,false)
alife():release(se_obj, true)
else
printf("Error, object is not an anomaly, %s, %s, on level %s", se_obj:section_name(), v1, k)
end
end
for k2, v2 in pairs(v.anomalies_by_smart) do
v2[v1] = nil
v.anomaly_types_by_smart[k2] = nil
if is_empty(v2) then
v.anomalies_by_smart[k2] = nil
v.available_smarts_reduced[k2] = nil
end
end
v.smart_by_anomalies[v1] = nil
v.anomalies_properties[v1] = nil
v.anomalies[k1] = nil
else
printf("can't delete anomaly %s globally, level %s, in restriction", v1, k)
end
end
-- Old anomalies
if true then
local fully_cleaned = true
for smart_name, v in pairs(SIMBOARD.smarts_by_names) do
local smart_level = sim_level_name(sim, gg_vertex(gg, v.m_game_vertex_id):level_id())
if smart_level == k then
for j = 1, 1000 do
local anom_id = load_var(db.actor, strformat("drx_da_anom_id_%s_%s", smart_name, j), nil)
if anom_id then
if true then
local o = alife_object(anom_id)
if o then
alife_record(o ,false)
alife():release(o, true)
end
else
printf("Error, can't remove old anomaly %s, level %s, in restriction", anom_id, k)
fully_cleaned = false
end
end
end
end
end
v.cleaned_old_anomalies = fully_cleaned
else
printf("old anomalies already cleaned on level %s", k)
end
end
end
-- Clean from new sections
local new_sections = {
zone_mine_darkness = true,
zone_mine_flash = true,
zone_mine_ghost = true,
zone_mine_gold = true,
zone_mine_green_dragon = true,
zone_mine_mefistotel = true,
zone_mine_net = true,
zone_mine_point = true,
zone_mine_radar = true,
zone_mine_sphere = true,
zone_unknown = true,
zone_mine_acid = true,
zone_mine_electra = true,
zone_mine_springboard = true,
zone_mine_vortex = true,
zone_mine_blast = true,
zone_mine_zharka = true,
zone_mine_vapour = true,
}
for i = 1, 65534 do
local se_obj = sim_object(sim, i)
if se_obj and new_sections[se_obj:section_name()] then
sim_release(sim, se_obj, true)
end
end
printf("Cleaned dynamic anomalies globally")
if settings.save_after_cleanup then
CreateTimeEvent("drx_da_save_after_cleanup", 0, 0.1, function()
exec_console_cmd("save " .. (user_name() or "") .. " - DAO tempsave")
return true
end)
end
end