2024-04-29 22:14:22 +03:00
dofile_once ( " mods/quant.ew/NoitaPatcher/load.lua " )
local np = require ( " noitapatcher " )
dofile_once ( " data/scripts/lib/utilities.lua " )
np.InstallShootProjectileFiredCallbacks ( )
np.EnableGameSimulatePausing ( false )
np.InstallDamageDetailsPatch ( )
np.SilenceLogs ( " Warning - streaming didn \' t find any chunks it could stream away... \n " )
local player_fns = dofile_once ( " mods/quant.ew/files/src/player_fns.lua " )
local net = dofile_once ( " mods/quant.ew/files/src/net.lua " )
2024-05-10 22:17:45 +03:00
local util = dofile_once ( " mods/quant.ew/files/src/util.lua " )
2024-04-29 22:14:22 +03:00
local ctx = dofile_once ( " mods/quant.ew/files/src/ctx.lua " )
2024-05-03 23:38:40 +03:00
local pretty = dofile_once ( " mods/quant.ew/files/lib/pretty_print.lua " )
2024-05-07 23:46:15 +03:00
local perk_fns = dofile_once ( " mods/quant.ew/files/src/perk_fns.lua " )
2024-05-08 20:33:41 +03:00
local enemy_sync = dofile_once ( " mods/quant.ew/files/src/enemy_sync.lua " )
2024-05-10 18:47:01 +03:00
local world_sync = dofile_once ( " mods/quant.ew/files/src/world_sync.lua " )
2024-05-13 13:07:28 +03:00
local item_sync = dofile_once ( " mods/quant.ew/files/src/item_sync.lua " )
2024-04-29 22:14:22 +03:00
2024-05-12 15:31:08 +03:00
ModLuaFileAppend ( " data/scripts/gun/gun.lua " , " mods/quant.ew/files/append/gun.lua " )
ModLuaFileAppend ( " data/scripts/gun/gun_actions.lua " , " mods/quant.ew/files/append/action_fix.lua " )
local my_player = nil
2024-05-12 16:23:28 +03:00
function OnProjectileFired ( shooter_id , projectile_id , initial_rng , position_x , position_y , target_x , target_y , send_message ,
2024-05-02 20:24:27 +03:00
unknown1 , multicast_index , unknown3 )
2024-05-12 15:31:08 +03:00
if not EntityHasTag ( shooter_id , " player_unit " ) and not EntityHasTag ( shooter_id , " ew_client " ) then
return -- Not fired by player, we don't care about it (for now?)
end
local projectileComponent = EntityGetFirstComponentIncludingDisabled ( projectile_id , " ProjectileComponent " )
local entity_that_shot = ComponentGetValue2 ( projectileComponent , " mEntityThatShot " )
2024-05-12 16:23:28 +03:00
local shooter_player_data = player_fns.get_player_data_by_local_entity_id ( shooter_id )
local rng = 0
-- Was shot locally
if shooter_id == my_player.entity then
-- If it was an initial shot by host
if ( entity_that_shot == 0 and multicast_index ~= - 1 and unknown3 == 0 ) then
rng = initial_rng
table.insert ( shooter_player_data.projectile_rng_init , rng )
else
rng = shooter_player_data.projectile_seed_chain [ entity_that_shot ] + 25
end
else
if ( entity_that_shot == 0 and multicast_index ~= - 1 and unknown3 == 0 ) then
2024-05-12 18:26:54 +03:00
if # shooter_player_data.projectile_rng_init > 0 then
2024-05-12 16:23:28 +03:00
rng = table.remove ( shooter_player_data.projectile_rng_init , 1 )
else
-- Shouldn't happen
GamePrint ( " No values in projectile_rng_init " )
rng = 0
end
else
rng = shooter_player_data.projectile_seed_chain [ entity_that_shot ] + 25
end
end
shooter_player_data.projectile_seed_chain [ projectile_id ] = rng
-- GamePrint("on fired "..projectile_id.." "..entity_that_shot.." "..shooter_id.." "..rng)
2024-05-12 15:31:08 +03:00
np.SetProjectileSpreadRNG ( rng )
2024-05-02 20:24:27 +03:00
end
function OnProjectileFiredPost ( shooter_id , projectile_id , rng , position_x , position_y , target_x , target_y , send_message ,
unknown1 , multicast_index , unknown3 )
end
function OnPausedChanged ( paused , is_wand_pickup )
local players = EntityGetWithTag ( " player_unit " ) or { }
if ( players [ 1 ] ) then
np.RegisterPlayerEntityId ( players [ 1 ] )
local inventory_gui = EntityGetFirstComponentIncludingDisabled ( players [ 1 ] , " InventoryGuiComponent " )
local controls_component = EntityGetFirstComponentIncludingDisabled ( players [ 1 ] , " ControlsComponent " )
if ( paused ) then
--EntitySetComponentIsEnabled(players[1], inventory_gui, false)
np.EnableInventoryGuiUpdate ( false )
np.EnablePlayerItemPickUpper ( false )
ComponentSetValue2 ( controls_component , " enabled " , false )
else
--EntitySetComponentIsEnabled(players[1], inventory_gui, true)
np.EnableInventoryGuiUpdate ( true )
np.EnablePlayerItemPickUpper ( true )
ComponentSetValue2 ( controls_component , " enabled " , true )
end
end
end
2024-04-29 22:14:22 +03:00
-- all functions below are optional and can be left out
--[[
function OnModPreInit ( )
print ( " Mod - OnModPreInit() " ) -- First this is called for all mods
end
function OnModInit ( )
print ( " Mod - OnModInit() " ) -- After that this is called for all mods
end
function OnModPostInit ( )
print ( " Mod - OnModPostInit() " ) -- Then this is called for all mods
end
] ] --
2024-05-12 15:31:08 +03:00
2024-04-29 22:14:22 +03:00
function OnWorldInitialized ( ) -- This is called once the game world is initialized. Doesn't ensure any world chunks actually exist. Use OnPlayerSpawned to ensure the chunks around player have been loaded or created.
2024-05-02 20:24:27 +03:00
--GamePrint( "OnWorldInitialized() " .. tostring(GameGetFrameNum()) )
2024-04-29 22:14:22 +03:00
end
2024-05-12 15:31:08 +03:00
2024-04-29 22:14:22 +03:00
function OnPlayerSpawned ( player_entity ) -- This runs when player entity has been created
GamePrint ( " OnPlayerSpawned() - Player entity id: " .. tostring ( player_entity ) )
2024-05-07 23:46:15 +03:00
my_player = player_fns.make_playerdata_for ( player_entity , ctx.my_id )
2024-05-02 20:24:27 +03:00
GamePrint ( " My peer_id: " .. ctx.my_id )
ctx.players [ ctx.my_id ] = my_player
2024-05-12 16:23:28 +03:00
ctx.player_data_by_local_entity [ player_entity ] = my_player
2024-05-02 20:24:27 +03:00
ctx.ready = true
2024-05-07 23:46:15 +03:00
np.SetPauseState ( 4 )
np.SetPauseState ( 0 )
2024-05-10 23:43:32 +03:00
if ctx.is_host then
EntityAddTag ( player_entity , " ew_host " )
else
2024-05-10 22:17:45 +03:00
EntityAddComponent2 ( player_entity , " LuaComponent " , { script_damage_about_to_be_received = " mods/quant.ew/files/cbs/immortal.lua " } )
end
2024-05-12 15:31:08 +03:00
EntityAddComponent2 ( player_entity , " LuaComponent " , { script_wand_fired = " mods/quant.ew/files/cbs/count_times_wand_fired.lua " } )
2024-05-07 23:46:15 +03:00
dofile_once ( " data/scripts/perks/perk.lua " )
local x , y = EntityGetFirstHitboxCenter ( player_entity )
2024-05-08 20:33:41 +03:00
perk_spawn ( x , y , " LASER_AIM " , true )
perk_spawn ( x - 50 , y , " GLASS_CANNON " , true )
2024-05-13 13:07:28 +03:00
perk_spawn ( x - 25 , y , " EDIT_WANDS_EVERYWHERE " , true )
2024-04-29 22:14:22 +03:00
end
2024-05-12 15:31:08 +03:00
local function on_world_pre_update_inner ( )
2024-04-29 22:14:22 +03:00
if my_player == nil then return end
2024-05-10 22:17:45 +03:00
2024-05-12 15:31:08 +03:00
GlobalsSetValue ( " ew_player_rng " , tostring ( GameGetFrameNum ( ) ) )
2024-04-29 22:14:22 +03:00
2024-05-02 20:24:27 +03:00
net.update ( )
2024-05-11 18:06:48 +03:00
if ctx.is_host and not EntityGetIsAlive ( my_player.entity ) then
if not ctx.run_ended then
GamePrint ( " Notifying of run end " )
net.proxy_notify_game_over ( )
ctx.run_ended = true
end
end
if not ctx.is_host then
local hp , _ = util.get_ent_health ( my_player.entity )
if hp == 0 then
EntityInflictDamage ( my_player.entity , 10000000 , " DAMAGE_CURSE " , " Out of shared health " , " NONE " , 0 , 0 , GameGetWorldStateEntity ( ) )
end
end
2024-05-13 13:07:28 +03:00
if ctx.is_host then
item_sync.host_upload_items ( my_player )
end
2024-05-11 18:06:48 +03:00
-- Player sync
2024-05-08 20:33:41 +03:00
if GameGetFrameNum ( ) % 1 == 0 then
2024-05-02 20:24:27 +03:00
local input_data = player_fns.serialize_inputs ( my_player )
local pos_data = player_fns.serialize_position ( my_player )
2024-05-05 23:49:10 +03:00
local current_slot = player_fns.get_current_slot ( my_player )
net.send_player_update ( input_data , pos_data , current_slot )
2024-04-29 22:14:22 +03:00
end
2024-05-11 18:06:48 +03:00
-- Enemy sync
2024-05-08 20:33:41 +03:00
if GameGetFrameNum ( ) % 2 == 1 then
if ctx.is_host then
net.send_enemy_data ( enemy_sync.host_upload_entities ( ) )
else
enemy_sync.client_cleanup ( )
end
end
2024-05-11 18:06:48 +03:00
-- World sync
2024-05-10 18:47:01 +03:00
if ctx.is_host then
local world_data = world_sync.host_upload ( )
if world_data ~= nil then
net.send_world_data ( world_data )
end
end
2024-05-11 18:06:48 +03:00
-- Health and air sync
2024-05-10 22:17:45 +03:00
if ctx.is_host and GameGetFrameNum ( ) % 4 == 3 then
local player_info = { }
2024-05-10 23:43:32 +03:00
local hp , max_hp = util.get_ent_health ( my_player.entity )
2024-05-10 22:17:45 +03:00
for id , player_data in pairs ( ctx.players ) do
local entity = player_data.entity
2024-05-10 23:43:32 +03:00
local air , max_air = util.get_ent_air ( entity )
player_info [ id ] = { hp , max_hp , air , max_air }
2024-05-10 22:17:45 +03:00
end
net.send_host_player_info ( player_info )
end
2024-05-11 18:06:48 +03:00
-- Inventory and perk sync
2024-05-03 23:38:40 +03:00
if GameGetFrameNum ( ) % 120 == 0 then
local inventory_state = player_fns.serialize_items ( my_player )
2024-05-10 22:17:45 +03:00
if inventory_state ~= nil then
net.send_player_inventory ( inventory_state )
end
2024-05-07 23:46:15 +03:00
local perk_data = perk_fns.get_my_perks ( )
2024-05-10 22:17:45 +03:00
if perk_data ~= nil then
net.send_player_perks ( perk_data )
end
2024-05-03 23:38:40 +03:00
end
2024-04-29 22:14:22 +03:00
end
2024-05-12 15:31:08 +03:00
function OnWorldPreUpdate ( ) -- This is called every time the game is about to start updating the world
util.tpcall ( on_world_pre_update_inner )
end
local function on_world_post_update_inner ( )
local times_wand_fired = tonumber ( GlobalsGetValue ( " ew_wand_fired " , " 0 " ) )
GlobalsSetValue ( " ew_wand_fired " , " 0 " )
if times_wand_fired > 0 then
local special_seed = tonumber ( GlobalsGetValue ( " ew_player_rng " , " 0 " ) )
local fire_data = player_fns.make_fire_data ( special_seed , my_player )
if fire_data ~= nil then
net.send_fire ( fire_data )
end
end
end
function OnWorldPostUpdate ( ) -- This is called every time the game has finished updating the world
util.tpcall ( on_world_post_update_inner )
end
2024-04-29 22:14:22 +03:00
function register_localizations ( translation_file , clear_count )
clear_count = clear_count or 0
local loc_content = ModTextFileGetContent ( " data/translations/common.csv " ) -- Gets the original translations of the game
local append_content = ModTextFileGetContent ( translation_file ) -- Gets my own translations file
-- Split the append_content into lines
local lines = { }
for line in append_content : gmatch ( " [^ \n ]+ " ) do
table.insert ( lines , line )
end
-- Remove the first clear_count lines
for i = 1 , clear_count do
table.remove ( lines , 1 )
end
-- Reconstruct append_content after removing clear_count lines
local new_append_content = table.concat ( lines , " \n " )
-- if loc_content does not end with a new line, add one
if not loc_content : match ( " \n $ " ) then
loc_content = loc_content .. " \n "
end
-- Concatenate loc_content and new_append_content without extra newline character
local new_content = loc_content .. new_append_content .. " \n "
-- Set the new content to the file
ModTextFileSetContent ( " data/translations/common.csv " , new_content )
end
function OnModPreInit ( )
register_localizations ( " mods/quant.ew/translations.csv " , 1 )
2024-05-02 20:24:27 +03:00
ctx.init ( )
2024-04-30 23:49:51 +03:00
net.init ( )
2024-04-29 22:14:22 +03:00
end
function OnModPostInit ( )
end
print ( " entangled_worlds init ok " )