Divergent/mods/World Spawns Rework/gamedata/scripts/game_setup.script

1090 lines
31 KiB
Plaintext
Raw Normal View History

--[[
Heavily modified by Utjan
Kits are seperated from the normal spawn system and has it's own more customisable system
Pulls min/max kit spawns for each map from barry_kit_spawns.ltx
]]--
--[[
- Created by tdef
- Updated by Tronex
- Randomized world items on new game
- Released blacklisted objects on new game
- Created: 2018/10/27
- 2019/31/3 script now read from config to set up
- 2019/4/25 objects to release are now handled by another config
- 2019/5/20 improved the way suffled consumables uses are set
used ini:
items\settings\dynamic_item_spawn.ltx
plugins\new_game_setup.ltx
set enable_debug to true, for debugging and map markers
--]]
-- these vehicles are supposed to shoot at you but call of misery broke them so they don't
-- also you can board them, turn them on and drive around so should remove them?
local ini_dyn
local enable_debug = false
local inited = false
local sfind = string.find
local world_itm_info = {} -- [name] = {}
local world_itm_off = {} -- [name] = true
local world_itm_num = {} -- [name] = num
local world_itm_on = {} -- [id] = name
local ini_kits -- barry_kit_spawns.ltx
local world_kit_off = {} -- Kit spawn points with no kit
local world_kit_on = {} -- Kit spawn points with a kit
local world_upg_kit_on = {} -- Kit spawn points with an upgrade kit
local min_kit_dist -- Minimum distance between kit spawns
local dist_exclude_table = {} -- Table used to temporarily store spawn points excluded by the minimum distance mechanic
local level_exclude_table = {} -- Used to exclude certain maps from having kits respawned unto
local itm_list = {}
local limited_uses = {}
function get_itm_type(name)
if sfind(name,"kolbasa")
or sfind(name,"conserva")
or sfind(name,"bread")
then
return "food"
end
if sfind(name,"energy")
or sfind(name,"vodka")
or sfind(name,"drink")
then
return "drink"
end
if sfind(name,"drug")
or sfind(name,"antirad")
or sfind(name,"bandage")
or sfind(name,"medkit")
then
return "medical"
end
--if sfind(name,"repair") then
--return "tool"
--end
if sfind(name,"ammo") then
return "ammo"
end
if sfind(name,"misc")
then
return "misc"
end
return "NA"
end
function print_debug(...)
if enable_debug then
printf(...)
end
end
local marker_by_type = {
["kit"] = "item_kit",
["medical"] = "item_medical",
["food"] = "item_food",
["drink"] = "item_drink",
["ammo"] = "item_ammo",
["misc"] = "item_misc",
}
function add_marker(name, section, id, typ)
if enable_debug then
local spot = marker_by_type[typ] or marker_by_type["misc"]
level.map_add_object_spot_ser(id, spot, "Name: " .. name .. " \\nType: " .. typ .. " \\nSection: " .. section)
end
end
function remove_marker(id, typ)
if enable_debug then
local spot = marker_by_type[typ] or marker_by_type["misc"]
if (level.map_has_object_spot(id, spot) ~= 0) then
level.map_remove_object_spot(id, spot)
end
end
end
function init_settings()
if (inited) then return end
inited = true
ini_dyn = ini_file("items\\settings\\dynamic_item_spawn.ltx")
ini_kits = ini_file("items\\settings\\barry_kit_spawns.ltx")
local n,m = 0,0
local result, id, value = "","",""
local name, info = "","",""
-- Gather items list
n = ini_dyn:line_count("categories") or 0
for i=0,n-1 do
result, id, value = ini_dyn:r_line_ex("categories",i,"","")
itm_list[id] = {}
m = ini_dyn:line_count(id) or 0
for ii=0,m-1 do
result, name, info = ini_dyn:r_line_ex(id,ii,"","")
if name and info then
for j=1,tonumber(info) do
local size = #itm_list[id] + 1
itm_list[id][size] = name
print_debug("- Game Setup | itm_list[%s][%s] = %s", id, size, name)
end
end
end
end
-- Gather recorded items pos
n = ini_dyn:line_count("levels") or 0
for i=0,n-1 do
result, id, value = ini_dyn:r_line_ex("levels",i,"","")
world_kit_off[id] = {} -- make table of levels
if not world_kit_on[id] then
world_kit_on[id] = {} -- make table of levels if it's not loaded from save
end
if not world_upg_kit_on[id] then
world_upg_kit_on[id] = {}
end
m = ini_dyn:line_count(id) or 0
for ii=0,m-1 do
result, name, info = ini_dyn:r_line_ex(id,ii,"","")
if name and info then
local t = str_explode(info,",")
if (#t == 6) and (t[1] ~= "NA") then
world_itm_info[name] = {
typ = t[1],
x = tonumber(t[2]),
y = tonumber(t[3]),
z = tonumber(t[4]),
lvl_id = tonumber(t[5]),
gm_id = tonumber(t[6]),
}
if sfind(t[1], "kit") then -- if item is kit
world_kit_off[id][name] = true -- add kit to world kit list
else
world_itm_off[name] = true
end
end
end
end
end
-- Gather uses
n = ini_dyn:line_count("possible_uses") or 0
for i=0,n-1 do
result, id, value = ini_dyn:r_line_ex("possible_uses",i,"","")
if id and value then
local t = str_explode(value,",")
limited_uses[id] = { tonumber(t[1]) or 1 , tonumber(t[2]) or 1 }
end
end
-- Gather respawn exclude list
n = ini_kits:line_count("kit_respawn_exclude") or 0
for i=0,n-1 do
result, id, value = ini_kits:r_line_ex("kit_respawn_exclude",i,"","")
level_exclude_table[id] = true
end
for id,name in pairs(world_itm_on) do
world_itm_off[name] = nil
end
for level, kit_table in pairs(world_kit_on) do
for spawn_name, v in pairs(kit_table) do
world_kit_off[level][spawn_name] = nil
end
end
for level, kit_table in pairs(world_upg_kit_on) do
for spawn_name, v in pairs(kit_table) do
world_kit_off[level][spawn_name] = nil
end
end
print_debug("- Game Setup | world_itm_info: %s - world_itm_on: %s - world_itm_off: %s", size_table(world_itm_info), size_table(world_itm_on), size_table(world_itm_off))
--print_debug("- Game Setup KITS | world_kit_on: %s - world_kit_off: %s", size_table(world_kit_on), size_table(world_kit_off))
end
function try_spawn_world_item(ignore)
-- Get spawn place name
local _name
if ignore then
if not (world_itm_off and size_table(world_itm_off) > 0) then
print_debug("! Game Setup | world_itm_off is nil or empty")
return
end
_name = random_key_table(world_itm_off)
else
local lvl_short = txr_routes.get_map(level.name())
local t = {}
-- Gather validated item places to spawn at
for name,_ in pairs(world_itm_off) do
if (not sfind(name,lvl_short)) then
t[#t+1] = name
end
end
_name = (#t > 0) and t[math.random(#t)]
if _name then printf("Respawning non-kit item: " .. _name) end
end
-- Return if not available place has been found
if (not _name) then
print_debug("! Game Setup | can't find available item place", _name)
return
end
-- Return if place already has spawned item
for id,name in pairs(world_itm_on) do
if (name == _name) then
print_debug("! Game Setup | place {%s} is already occupied", _name)
return
end
end
-- Get info
local info = world_itm_info[_name]
if (not info) then
print_debug("! Game Setup | no info is found for {%s}", _name)
return
end
-- Get section
local itm_type = info.typ and itm_list[info.typ]
local section = itm_type and itm_type[math.random(#itm_type)]
if (not section) then
print_debug("! Game Setup | couldn't get section [%s] for type (%s)", section, info.typ)
return
end
if (not ini_sys:section_exist(section)) then
print_debug("! Game Setup | section [%s] doesn't exist", section)
return
end
-- Info check
if not (info.x and info.y and info.z and info.lvl_id and info.gm_id and true) then
print_debug("! Game Setup | item {%s} has wrong or incomplete info", _name)
return
end
-- Spawn and adjust uses/condition/ammo size
if IsItem("ammo",section) then
local pos = vector():set(info.x, info.y, info.z)
local se_obj = alife_create_item(section, {pos, info.lvl_id, info.gm_id})
if se_obj then
add_marker(_name, section, se_obj.id, info.typ)
world_itm_on[se_obj.id] = _name
world_itm_off[_name] = nil
local box_size = ini_sys:r_u32(section, "box_size")
world_itm_num[_name] = math.random( math.ceil(box_size * 0.25) , math.ceil(box_size * 0.75) )
print_debug("/ Game Setup | created ammo [%s](%s) - place: %s - size = %s", section, se_obj.id, _name, world_itm_num[_name])
else
print_debug("! Game Setup | ammo [%s] couldn't be created", section)
end
else
local pos = vector():set(info.x, info.y, info.z)
local se_obj = alife_create_item(section, {pos, info.lvl_id, info.gm_id})
if se_obj then
add_marker(_name, section, se_obj.id, info.typ)
world_itm_on[se_obj.id] = _name
world_itm_off[_name] = nil
-- Multi-use
if limited_uses[section] then
world_itm_num[_name] = math.random(limited_uses[section][1], limited_uses[section][2])
print_debug("/ Game Setup | created multiuse item [%s](%s) - place: %s - uses = %s", section, se_obj.id, _name, world_itm_num[_name])
else
local is_using_con = utils_item.is_degradable(nil, section)
if is_using_con then
-- Parts
if IsItem("part",section) then
world_itm_num[_name] = random_choice(0.5,0.75,1)
print_debug("/ Game Setup | created degraded item [%s](%s) - place: %s - con = %s", section, se_obj.id, _name, world_itm_num[_name])
-- Degradable items
else
world_itm_num[_name] = (math.random(30,70)/100)
print_debug("/ Game Setup | created degraded item [%s](%s) - place: %s - con = %s", section, se_obj.id, _name, world_itm_num[_name])
end
else
print_debug("/ Game Setup | created item [%s](%s)", section, se_obj.id)
end
end
else
print_debug("! Game Setup | item [%s] couldn't be created", section)
end
end
end
function try_spawn_world_item_kit(level_name, ignore)
-- Get spawn place name
local _name
if ignore then
if not (world_kit_off[level_name] and size_table(world_kit_off[level_name]) > 0) then
print_debug("! Game Setup | world_itm_off is nil or empty")
return
end
_name = random_key_table(world_kit_off[level_name])
else
_name = find_kit_respawn_name(level_name)
end
-- Return if not available place has been found
if (not _name) then
print_debug("! Game Setup | can't find available item place", _name)
return
end
-- Return if place already has spawned item
if world_kit_on[level_name] then
for id,name in pairs(world_kit_on[level_name]) do
if (name == _name) then
print_debug("! Game Setup | place {%s} is already occupied", _name)
return
end
end
end
-- Get info
local info = world_itm_info[_name]
if (not info) then
print_debug("! Game Setup | no info is found for {%s}", _name)
return
end
-- Get section
local itm_type = info.typ and itm_list[info.typ]
local section = itm_type and itm_type[math.random(#itm_type)]
if (not section) then
print_debug("! Game Setup | couldn't get section [%s] for type (%s)", section, info.typ)
return
end
if (not ini_sys:section_exist(section)) then
print_debug("! Game Setup | section [%s] doesn't exist", section)
return
end
-- Info check
if not (info.x and info.y and info.z and info.lvl_id and info.gm_id and true) then
print_debug("! Game Setup | item {%s} has wrong or incomplete info", _name)
return
end
-- Check if trying to spawn too close to other kit
local pos = vector():set(info.x, info.y, info.z)
if ignore then
for id, kit_name in pairs(world_kit_on[level_name]) do
local o_info = world_itm_info[kit_name]
local o_pos = vector():set(o_info.x, o_info.y, o_info.z)
if pos:distance_to(o_pos) <= min_kit_dist then
dist_exclude_table[level_name][_name] = true
world_kit_off[level_name][_name] = nil
return "too close"
end
end
end
-- Spawn and adjust uses
local se_obj = alife_create_item(section, {pos, info.lvl_id, info.gm_id})
if se_obj then
add_marker(_name, section, se_obj.id, info.typ)
world_kit_on[level_name][se_obj.id] = _name
world_kit_off[level_name][_name] = nil
-- Multi-use
if limited_uses[section] then
world_itm_num[_name] = math.random(limited_uses[section][1], limited_uses[section][2])
print_debug("/ Game Setup | created multiuse item [%s](%s) - place: %s - uses = %s", section, se_obj.id, _name, world_itm_num[_name])
else
print_debug("/ Game Setup | created item [%s](%s) - place: %s", section, se_obj.id, _name)
end
else
print_debug("! Game Setup | item [%s] couldn't be created", section)
end
end
function try_spawn_world_item_kit_upg(level_name, ignore)
-- Get spawn place name
local _name
if ignore then
if not (world_kit_off[level_name] and size_table(world_kit_off[level_name]) > 0) then
print_debug("! Game Setup | world_itm_off is nil or empty")
return
end
_name = random_key_table(world_kit_off[level_name])
else
_name = find_kit_respawn_name(level_name)
end
-- Return if not available place has been found
if (not _name) then
print_debug("! Game Setup | can't find available item place", _name)
return
end
-- Return if place already has spawned item
if world_upg_kit_on[level_name] then
for id,name in pairs(world_upg_kit_on[level_name]) do
if (name == _name) then
print_debug("! Game Setup | place {%s} is already occupied", _name)
return
end
end
end
-- Get info
local info = world_itm_info[_name]
if (not info) then
print_debug("! Game Setup | no info is found for {%s}", _name)
return
end
-- Get section
local itm_type = itm_list["kit_upgrade"]
if _G["kit_binder"] then
itm_type = itm_list["kit_upg_streamlined"]
end
local section = itm_type and itm_type[math.random(#itm_type)]
if (not section) then
print_debug("! Game Setup | couldn't get section [%s] for type (%s)", section, info.typ)
return
end
if (not ini_sys:section_exist(section)) then
print_debug("! Game Setup | section [%s] doesn't exist", section)
return
end
-- Info check
if not (info.x and info.y and info.z and info.lvl_id and info.gm_id and true) then
print_debug("! Game Setup | item {%s} has wrong or incomplete info", _name)
return
end
-- Check if trying to spawn too close to other kit
local pos = vector():set(info.x, info.y, info.z)
if ignore then
for id, kit_name in pairs(world_kit_on[level_name]) do
local o_info = world_itm_info[kit_name]
local o_pos = vector():set(o_info.x, o_info.y, o_info.z)
if pos:distance_to(o_pos) <= min_kit_dist then
dist_exclude_table[level_name][_name] = true
world_kit_off[level_name][_name] = nil
return "too close"
end
end
for id, kit_name in pairs(world_upg_kit_on[level_name]) do
local o_info = world_itm_info[kit_name]
local o_pos = vector():set(o_info.x, o_info.y, o_info.z)
if pos:distance_to(o_pos) <= min_kit_dist then
dist_exclude_table[level_name][_name] = true
world_kit_off[level_name][_name] = nil
return "too close"
end
end
end
-- Spawn and adjust uses
local se_obj = alife_create_item(section, {pos, info.lvl_id, info.gm_id})
if se_obj then
add_marker(_name, section, se_obj.id, info.typ)
world_upg_kit_on[level_name][se_obj.id] = _name
world_kit_off[level_name][_name] = nil
local is_using_con = utils_item.is_degradable(nil, section)
-- Multi-use
if limited_uses[section] and not is_using_con then
world_itm_num[_name] = math.random(limited_uses[section][1], limited_uses[section][2])
print_debug("/ Game Setup | created multiuse item [%s](%s) - place: %s - uses = %s", section, se_obj.id, _name, world_itm_num[_name])
else
if is_using_con and limited_uses[section] then
-- Degradable items
world_itm_num[_name] = math.random(limited_uses[section][1], limited_uses[section][2])/100
print_debug("/ Game Setup | created degraded item [%s](%s) - place: %s - con = %s", section, se_obj.id, _name, world_itm_num[_name])
else
print_debug("/ Game Setup | created item [%s](%s) - place: %s", section, se_obj.id, _name)
end
end
else
print_debug("! Game Setup | item [%s] couldn't be created", section)
end
end
function spawn_guaranteed_kit(level_name, _name, section)
if (not _name) then
print_debug("Name not found")
return
end
-- Return if place already has spawned item
if world_kit_on[level_name] then
for id,name in pairs(world_kit_on[level_name]) do
if (name == _name) then
print_debug("Guaranteed spawn occupied")
return
end
end
end
-- Get info
local info = world_itm_info[_name]
if (not info) then
print_debug("Info not found for .. " .. _name)
return
end
if (not section) then
print_debug("Section not found for " .. _name)
return
end
if (not ini_sys:section_exist(section)) then
print_debug("Section " .. section .. " not in ini for " .. _name)
return
end
-- Info check
if not (info.x and info.y and info.z and info.lvl_id and info.gm_id and true) then
print_debug("! Game Setup | item {%s} has wrong or incomplete info", _name)
return
end
-- Spawn and adjust uses/condition/ammo size
local pos = vector():set(info.x, info.y, info.z)
local se_obj = alife_create_item(section, {pos, info.lvl_id, info.gm_id})
if se_obj then
add_marker(_name, section, se_obj.id, info.typ)
world_kit_on[level_name][se_obj.id] = _name
world_kit_off[level_name][_name] = nil
--[[
-- Multi-use
if limited_uses[section] then
world_itm_num[_name] = math.random(limited_uses[section][1], limited_uses[section][2])
--printf("Num: " .. world_itm_num[_name])
print_debug("/ Game Setup | created multiuse item [%s](%s) - place: %s - uses = %s", section, se_obj.id, _name, world_itm_num[_name])
end
]]--
else
print_debug("! Game Setup | item [%s] couldn't be created", section)
end
end
function reset()
for level, kits in pairs(world_upg_kit_on) do
for id, spawn_name in pairs(kits) do
alife_release_id(id)
end
end
for level, kits in pairs(world_kit_on) do
for id, spawn_name in pairs(kits) do
alife_release_id(id)
end
end
for id, spawn_name in pairs(world_itm_on) do
alife_release_id(id)
end
world_upg_kit_on = {}
world_itm_num = {}
world_kit_on = {}
world_itm_on = {}
world_kit_off = {}
world_itm_off = {}
inited = false
init_settings()
create_world_itms()
--create_guaranteed_itms()
create_world_kits()
create_world_kits_upg()
news_manager.send_tip(db.actor, string.format("Kit Spawn Rework: Kit spawn locations have been reset!"), 0, nil, 10000)
end
function find_kit_respawn_name(level_name)
local t = {}
-- Gather validated item places to spawn at
for level, kits in pairs(world_kit_off) do
if (size_table(kits) > 0) and not(level_name == level or level_exclude_table[level]) then
t[level] = {}
for name, _ in pairs(kits) do
local key = size_table(t[level]) + 1
t[level][key] = name
end
end
end
_rand_level = random_key_table(t)
_rand_kit = random_key_table(t[_rand_level])
return t[_rand_level][_rand_kit]
end
function create_world_itms()
local multi = game_difficulties.get_eco_factor("random_items") or 0.5
-- ZCP compatibility code
if smr_amain_mcm and smr_amain_mcm.get_config("smr_enabled") then
multi = smr_loot_mcm.get_config("random_items")
end
-- ZCP END
multi = (multi < 1) and multi or 1
local num = math.ceil(size_table(world_itm_off) * multi)
for i=1,num do
try_spawn_world_item(true)
end
end
function create_guaranteed_itms()
local guaranteed_tags = {}
local n = ini_kits:line_count("tags_guaranteed") or 0
for i=0,n-1 do
result, tag, item_section = ini_kits:r_line_ex("tags_guaranteed",i,"","")
guaranteed_tags[tag] = item_section
end
for level_name, kit_table in pairs(world_kit_off) do
for kit, v in pairs(kit_table) do
for tag, section in pairs(guaranteed_tags) do
if sfind(kit, tag) then
spawn_guaranteed_kit(level_name, kit, section)
end
end
end
end
end
function create_world_kits()
min_kit_dist = tonumber(ini_kits:r_string_ex("settings","min_kit_distance"))
local min_kit_level_list = {}
local max_kit_level_list = {}
local n = ini_kits:line_count("kit_spawn_counts") or 0
for i=0,n-1 do
result, level_name, value = ini_kits:r_line_ex("kit_spawn_counts",i,"","")
local t = str_explode(value,",")
local min_value = worldspawns_rework_mcm.get_config(level_name .. "_min") or t[1]
local max_value = worldspawns_rework_mcm.get_config(level_name .. "_max") or t[2]
min_kit_level_list[level_name] = math.min(min_value, max_value)
max_kit_level_list[level_name] = math.max(min_value, max_value)
end
for level_name, v in pairs(world_kit_off) do
local kit_spawn_count = math.min(math.random(min_kit_level_list[level_name], max_kit_level_list[level_name]), size_table(world_kit_off[level_name]))
dist_exclude_table[level_name] = {}
for i=1, kit_spawn_count do
local result = try_spawn_world_item_kit(level_name, true)
while result and result == "too close" and size_table(world_kit_off[level_name]) > 0 do
result = try_spawn_world_item_kit(level_name, true)
end
if is_empty(world_kit_off[level_name]) then
break
end
end
for kit_name, v in pairs(dist_exclude_table[level_name]) do
world_kit_off[level_name][kit_name] = true
end
dist_exclude_table[level_name] = {}
end
end
function create_world_kits_upg()
local min_upg_kit_level_list = {}
local max_upg_kit_level_list = {}
local n = ini_kits:line_count("upg_kit_spawn_counts") or 0
for i=0,n-1 do
result, level_name, value = ini_kits:r_line_ex("upg_kit_spawn_counts",i,"","")
local t = str_explode(value,",")
local min_value = worldspawns_rework_mcm.get_config(level_name .. "_upg_min") or t[1]
local max_value = worldspawns_rework_mcm.get_config(level_name .. "_upg_max") or t[2]
min_upg_kit_level_list[level_name] = math.min(min_value, max_value)
max_upg_kit_level_list[level_name] = math.max(min_value, max_value)
end
for level_name, v in pairs(world_kit_off) do
local kit_spawn_count = math.min(math.random(min_upg_kit_level_list[level_name], max_upg_kit_level_list[level_name]), size_table(world_kit_off[level_name]))
dist_exclude_table[level_name] = {}
for i=1, kit_spawn_count do
local result = try_spawn_world_item_kit_upg(level_name, true)
while result and result == "too close" and size_table(world_kit_off[level_name]) > 0 do
result = try_spawn_world_item_kit_upg(level_name, true)
end
if is_empty(world_kit_off[level_name]) then
break
end
end
for kit_name, v in pairs(dist_exclude_table[level_name]) do
world_kit_off[level_name][kit_name] = true
end
dist_exclude_table[level_name] = {}
end
end
function is_world_item(id)
if id and world_itm_on[id] then
--print_debug("! Game Setup | is_world_item[%s]", id)
return true
elseif id then
for level_name, v in pairs(world_kit_on) do
if world_kit_on[level_name][id] then
return true
end
end
for level_name, v in pairs(world_upg_kit_on) do
if world_upg_kit_on[level_name][id] then
return true
end
end
end
--print_debug("/ Game Setup | is_world_item[%s]", id)
return false
end
-- TODO IN 1.6 OR WHENEVER WE CAN EDIT ALL.SPAWN
-- remove these 2 objects because vetham is making new office for medic and they get in the way
function bar_medic_remove_stuff()
if not alife_storage_manager.get_state().duty_medic_fix then
alife_storage_manager.get_state().duty_medic_fix = true
for i=1,65534 do
local se = alife():object(i)
if se and (se:name() == 'bar_physic_object_mlr_0002' or se:name() == 'bar_physic_object_mlr_0003') then
alife():release(se)
end
end
end
end
-- TODO IN 1.6 OR WHENEVER WE CAN EDIT ALL.SPAWN
-- remove these 4 objects because they're stuck in the train and physics impulse makes them jitter around at 5 fps
function darkscape_remove_physics_objects()
if not alife_storage_manager.get_state().darkscape_phys_fix then
alife_storage_manager.get_state().darkscape_phys_fix = true
for i=1,65534 do
local se = alife():object(i)
if se and (se:name() == 'ds_physic_destroyable_object_0046' or se:name() == 'ds_physic_object_0009' or se:name() == 'ds_physic_object_0010' or se:name() == 'ds_physic_object_0002') then
alife():release(se)
end
end
end
end
-- TODO IN 1.6 OR WHENEVER WE CAN EDIT ALL.SPAWN
-- delete the chair and move smart cover in this new position
-- OR
-- make the chair part of level geometry and delete the object
-- OR
-- find a way to make that specific object not react to physics
function freedom_medic_fix()
if not alife_storage_manager.get_state().freedom_medic_fix then
alife_storage_manager.get_state().freedom_medic_fix = true
for i=1,65534 do
local se = alife():object(i)
if se then
if se:name() == 'mil_physic_object_0048' then
alife():release(se)
elseif se:name() == 'sc_freedom_medic_mlr' then
alife():teleport_object(i, 2165, 315401, vector():set(27.681089401245, -6.9381303787231, 17.38550567627))
end
end
end
end
end
-------------------------------
-- CALLBACKS
-------------------------------
function actor_on_first_update()
init_settings()
freedom_medic_fix()
bar_medic_remove_stuff()
darkscape_remove_physics_objects()
if alife_storage_manager.get_state().item_removal_done or IsTestMode() then
UnregisterScriptCallback("actor_on_first_update",actor_on_first_update)
return
end
alife_storage_manager.get_state().item_removal_done = true
print_debug("- Game Setup | create dynamic items")
local ini_setup = ini_file("plugins\\new_game_setup.ltx")
local enabled = true --ini_dyn:r_bool_ex("settings","enabled") or false
if (not enabled) then
return
end
-- Release static items and mines
local sim = alife()
local boxes = {}
for i=1, 65534 do
local se_obj = sim:object(i)
if se_obj then
local name = se_obj:name()
local cls = se_obj:clsid()
if cls == clsid.inventory_box_s then
--print_debug('%s_%s is a box', i, name)
boxes[i] = true
elseif ini_dyn:line_exist("replace_items",name) then
--print_debug('releasing %s', name)
--sim:release(se_obj, true)
alife_release(se_obj)
end
if ini_setup:line_exist("remove_objects",name) then
print_debug('/ Game Setup | Releasing object (%s)', name)
-- Clear inventory boxes from their manager
if (cls == clsid.inventory_box_s) then
treasure_manager.release_stash_by_id(se_obj.id)
end
safe_release_manager.release(se_obj)
end
end
end
-- Clear stashes
for i=1, 65534 do
local se_obj = sim:object(i)
if se_obj then
local name = se_obj:name()
if boxes[se_obj.parent_id] and (not sfind(name, 'mlr_strelok_item')) then
print_debug('/ Game Setup | Releasing {%s} from box', name)
--sim:release(se_obj, true)
alife_release(se_obj)
end
end
end
-- Setup items
create_world_itms()
-- Find all kit spawns with a tag for guaranteed spawn and spawn them
--create_guaranteed_itms()
-- Spawn kits
create_world_kits()
-- Upgrade kits spawning
create_world_kits_upg()
print_debug("- Game Setup | world_itm_info: %s - world_itm_on: %s - world_itm_off: %s", size_table(world_itm_info), size_table(world_itm_on), size_table(world_itm_off))
end
local tg_stkr = 0
local function actor_on_update()
if time_global() < tg_stkr then
return
end
-- No need to process if actor is outside cordon / visited more levels / not a loner / Warfare is active
if (level.name() ~= "l01_escape")
or IsWarfare()
or (game_statistics.get_statistic_count("level_changes") > 1)
or (get_actor_true_community() ~= "stalker")
then
UnregisterScriptCallback("actor_on_update",actor_on_update)
return
end
-- Remove common military or mutant squads
local on_act_lvl = simulation_objects.is_on_the_actor_level
for id,v in pairs( SIMBOARD.squads ) do
local squad = alife_object(id)
if squad and squad.common and (squad.player_id == "army") and on_act_lvl(squad) then
squad:remove_squad()
break
end
end
tg_stkr = time_global() + 10000
end
function actor_on_item_take(obj)
local id = obj:id()
local level_name = level.name()
local name
if id and level_name and world_kit_on[level_name] and world_kit_on[level_name][id] then
name = world_kit_on[level_name][id]
elseif id and level_name and world_upg_kit_on[level_name] and world_upg_kit_on[level_name][id] then
name = world_upg_kit_on[level_name][id]
end
if name then
printf("Found kit; name: %s - level: %s - section: %s - id: %s", name, level_name, obj:section(), id)
else
name = world_itm_on[id]
end
if name then
local section = obj:section()
local info = world_itm_info[name]
if (not info) then
print_debug("! Game Setup | can't get info for {%s}", name)
end
-- Spawn a new world item
local result
if world_itm_on[id] then
try_spawn_world_item()
elseif world_kit_on[level_name][id] then
try_spawn_world_item_kit(level_name)
elseif world_upg_kit_on[level_name][id] then
try_spawn_world_item_kit_upg(level_name)
end
-- Switch state
if world_itm_on[id] then
world_itm_on[id] = nil
world_itm_off[name] = true
elseif world_kit_on[level_name][id] then
world_kit_on[level_name][id] = nil
world_kit_off[level_name][name] = true
elseif world_upg_kit_on[level_name][id] then
world_upg_kit_on[level_name][id] = nil
world_kit_off[level_name][name] = true
end
-- Read info
local num = world_itm_num[name]
if num then
--printf("Got num")
--printf("section: " .. section)
-- Ammo
if IsItem("ammo",section) then
obj:ammo_set_count(num)
print_debug("- Game Setup | taken world ammo [%s](%s) is set to %s ammo - info name: %s", section, id, num, name)
world_itm_num[name] = nil
-- Multi-use
elseif limited_uses[section] and not utils_item.is_degradable(nil, section) then
alife_process_item( section, id , {uses = num} )
print_debug("- Game Setup | taken world consumable [%s](%s) is set to %s uses - info name: %s", section, id, num, name)
world_itm_num[name] = nil
-- Condition
elseif utils_item.is_degradable(nil, section) then
alife_process_item( section, id , {cond = num} )
print_debug("- Game Setup | taken world degraded item [%s](%s) is set to %s condition - info name: %s", section, id, num, name)
world_itm_num[name] = nil
end
-- Normal
else
print_debug("- Game Setup | taken world item [%s](%s) - info name: %s", section, id, uses, name)
end
-- Send message
itms_manager.send_itm_msg(section)
if info then
remove_marker(id, info.typ)
end
else
print_debug("! Game Setup | can't get name for id {%s} level %s", id, level_name)
end
-- Ammo aggregation (it's important to start ammo aggregation after sorting taken world ammo size first, to prevent issues)
if IsAmmo(obj) then
item_weapon.ammo_aggregation(obj)
end
end
local get_config = worldspawns_rework_mcm.get_config
function on_option_change()
if get_config("reset_kits") and level.present() and db.actor then
reset()
ui_mcm.set("kit_rework/reset_kits", false)
end
end
local function save_state(m_data)
m_data.world_itm_on = world_itm_on
m_data.world_kit_on = world_kit_on
m_data.world_itm_num = world_itm_num
m_data.world_upg_kit_on = world_upg_kit_on
print_debug("# SAVING: world_itm_on [%s] - world_itm_num [%s]", size_table(world_itm_on), size_table(world_itm_num))
end
local function load_state(m_data)
world_itm_on = m_data.world_itm_on or {}
world_itm_num = m_data.world_itm_num or {}
world_kit_on = m_data.world_kit_on or {}
world_upg_kit_on = m_data.world_upg_kit_on or {}
print_debug("# LOADING: world_itm_on [%s] - world_itm_num [%s]", size_table(world_itm_on), size_table(world_itm_num))
end
function on_game_start()
RegisterScriptCallback("actor_on_first_update",actor_on_first_update)
RegisterScriptCallback("actor_on_update",actor_on_update)
RegisterScriptCallback("actor_on_item_take",actor_on_item_take)
RegisterScriptCallback("save_state",save_state)
RegisterScriptCallback("load_state",load_state)
RegisterScriptCallback("on_option_change", on_option_change)
end