diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..22e564d --- /dev/null +++ b/.gitignore @@ -0,0 +1,7 @@ +.idea/ +.vs/ +*.user +/local + +bin/ +obj/ diff --git a/20241022214610_1.jpg b/20241022214610_1.jpg deleted file mode 100644 index 5e291a3..0000000 Binary files a/20241022214610_1.jpg and /dev/null differ diff --git a/PoppingBottles.sln b/PoppingBottles.sln new file mode 100644 index 0000000..f8c7869 --- /dev/null +++ b/PoppingBottles.sln @@ -0,0 +1,16 @@ + +Microsoft Visual Studio Solution File, Format Version 12.00 +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "PoppingBottles", "PoppingBottles\PoppingBottles.csproj", "{1A273855-C2D0-40BB-B22C-CE17AA4D0A9B}" +EndProject +Global + GlobalSection(SolutionConfigurationPlatforms) = preSolution + Debug|Any CPU = Debug|Any CPU + Release|Any CPU = Release|Any CPU + EndGlobalSection + GlobalSection(ProjectConfigurationPlatforms) = postSolution + {1A273855-C2D0-40BB-B22C-CE17AA4D0A9B}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {1A273855-C2D0-40BB-B22C-CE17AA4D0A9B}.Debug|Any CPU.Build.0 = Debug|Any CPU + {1A273855-C2D0-40BB-B22C-CE17AA4D0A9B}.Release|Any CPU.ActiveCfg = Release|Any CPU + {1A273855-C2D0-40BB-B22C-CE17AA4D0A9B}.Release|Any CPU.Build.0 = Release|Any CPU + EndGlobalSection +EndGlobal diff --git a/PoppingBottles/Config.cs b/PoppingBottles/Config.cs new file mode 100644 index 0000000..d95d759 --- /dev/null +++ b/PoppingBottles/Config.cs @@ -0,0 +1,7 @@ +using System.Text.Json.Serialization; + +namespace PoppingBottles; + +public class Config { + [JsonInclude] public bool SomeSetting = true; +} diff --git a/PoppingBottles/Mod.cs b/PoppingBottles/Mod.cs new file mode 100644 index 0000000..b7026b3 --- /dev/null +++ b/PoppingBottles/Mod.cs @@ -0,0 +1,78 @@ +using GDWeave; +using GDWeave.Godot; +using GDWeave.Godot.Variants; +using GDWeave.Modding; + +namespace PoppingBottles; + +public class Mod : IMod { + public Config Config; + + public Mod(IModInterface modInterface) { + this.Config = modInterface.ReadConfig(); + modInterface.Logger.Information("Loaded PoppingBottles!"); + + modInterface.RegisterScriptMod(new Patch()); + } + + public void Dispose() { + // Cleanup anything you do here + } +} + +public class Patch : IScriptMod +{ + public bool ShouldRun(string path) => path == "res://Scenes/Entities/Player/player.gdc"; + + public IEnumerable Modify(string path, IEnumerable tokens) + { + var functionWaiter = new FunctionWaiter("_consume_item"); + + var growthWaiter = new MultiTokenWaiter([ + t => t is ConstantToken { Value: StringVariant { Value: "growth" } }, + t => t.Type is TokenType.Colon, + t => t is IdentifierToken { Name: "player_scale" }, + t => t.Type is TokenType.OpAssign, + t => t.Type is TokenType.BuiltInFunc, + t => t.Type is TokenType.ParenthesisOpen, + t => t is IdentifierToken { Name: "player_scale" }, + t => t.Type is TokenType.OpAdd, + t => t is ConstantToken { Value: RealVariant { Value: 0.1 } }, + t => t.Type is TokenType.Comma, + t => t is ConstantToken { Value: RealVariant { Value: 0.7 } }, + t => t.Type is TokenType.Comma, + t => t is ConstantToken { Value: RealVariant { Value: 1.3 } }, + ], true, true); + + var shrinkWaiter = new MultiTokenWaiter([ + t => t is ConstantToken { Value: StringVariant { Value: "shrink" } }, + t => t.Type is TokenType.Colon, + t => t is IdentifierToken { Name: "player_scale" }, + t => t.Type is TokenType.OpAssign, + t => t.Type is TokenType.BuiltInFunc, + t => t.Type is TokenType.ParenthesisOpen, + t => t is IdentifierToken { Name: "player_scale" }, + t => t.Type is TokenType.OpSub, + t => t is ConstantToken { Value: RealVariant { Value: 0.1 } }, + t => t.Type is TokenType.Comma, + t => t is ConstantToken { Value: RealVariant { Value: 0.7 } }, + t => t.Type is TokenType.Comma, + t => t is ConstantToken { Value: RealVariant { Value: 1.3 } }, + ], true, true); + + foreach (var token in tokens) + { + if (functionWaiter.Check(token)) + { + growthWaiter.SetReady(); + shrinkWaiter.SetReady(); + } + + if (growthWaiter.Check(token) || shrinkWaiter.Check(token)) + { + yield return new ConstantToken(new RealVariant(100.0)); + } + else yield return token; + } + } +} \ No newline at end of file diff --git a/PoppingBottles/PoppingBottles.csproj b/PoppingBottles/PoppingBottles.csproj new file mode 100644 index 0000000..6704e7e --- /dev/null +++ b/PoppingBottles/PoppingBottles.csproj @@ -0,0 +1,35 @@ + + + net8.0 + enable + enable + $(AssemblySearchPaths);$(GDWeavePath)/core + 1.0.0.0 + + + + + + + + + + + + + + true + true + + + + + + + diff --git a/PoppingBottles/manifest.json b/PoppingBottles/manifest.json new file mode 100644 index 0000000..26814af --- /dev/null +++ b/PoppingBottles/manifest.json @@ -0,0 +1,10 @@ +{ + "Id": "PoppingBottles", + "AssemblyPath": "PoppingBottles.dll", + "Metadata": { + "Name": "PoppingBottles", + "Author": "uncreativeCultist", + "Version": "1.0.0", + "Description": "Mod that uncaps the limit on size sodas." + } +} diff --git a/README.md b/README.md index a7bc2e9..8e30a25 100644 --- a/README.md +++ b/README.md @@ -1,16 +1,14 @@ -# PoppingBottles -PoppingBottles is a mod for WEBFISHING that modifies the player script uncap the limit on size sodas! Script created by serenekerosene on the WEBFISHING Community Discord. Code used with permission. -![oh dear god](https://git.lainiwakura.xyz/uncreativecultist/PoppingBottles/raw/branch/main/20241022214610_1.jpg) +# GDWeave.Sample -Downloads can be found on the [Releases tab](https://git.lainiwakura.xyz/uncreativecultist/PoppingBottles/releases) -## Notes -To use this mod with [Nyoom!!!](https://git.lainiwakura.xyz/uncreativecultist/NyoomMod), modify the manifest.json file to change `"PackPath": "poppinbottles.pck"` to `"PackPath": "poppinbottles-nyoom.pck"` +A sample for C# [GDWeave](https://github.com/NotNite/GDWeave) mods. -**This mod was created using hard-patching. It will cause conflicts with any mod that modifies the following files:** - - player.gd -## Acknowledgements +## Setup - - serenekerosene - Created script tht allows unlimited drinking. - - [GDWeave](https://github.com/NotNite/GDWeave) - PoppingBottles requires GDWeave +Clone/fork/whatever this repository. Pick an ID for your project (people like to do `Username.ProjectName`, but there is no enforced naming scheme). Rename the following: +- Solution name, project name, and project namespace to your project ID +- Various fields in the manifest.json to your project ID and name +## Building + +To build the project, you need to set the `GDWeavePath` environment variable to your game install's GDWeave directory (e.g. `G:\games\steam\steamapps\common\WEBFISHING\GDWeave`). This can also be done in Rider with `File | Settings | Build, Execution, Deployment | Toolset and Build | MSBuild global properties` or with a .user file in Visual Studio. diff --git a/Scenes/Entities/Player/player-nyoomcompat.gd b/Scenes/Entities/Player/player-nyoomcompat.gd deleted file mode 100644 index 4fb75c4..0000000 --- a/Scenes/Entities/Player/player-nyoomcompat.gd +++ /dev/null @@ -1,2343 +0,0 @@ -extends Actor - -enum STATES{DEFAULT, BUSY, OBTAIN, EMOTING, FREECAM, FISHING_CHARGE, FISHING_CAST, FISHING, FISHING_CANCEL, FISHING_STRUGGLE, SHOVEL_CAST, SHOVEL_STRUGGLE, SHOVEL_CANCEL, \ -NET_CAST, NET_STRUGGLE, NET_CANCEL, SHOWCASE, CONSUME_ITEM, METAL_DETECTOR, GUITAR, GAMBLING} - -const GRAVITY = 32.0 -const BAIT_DATA = { - "": {"catch": 0.0, "max_tier": 0, "quality": []}, - - "worms": {"catch": 0.06, "max_tier": 1, "quality": [1.0]}, - "cricket": {"catch": 0.06, "max_tier": 2, "quality": [1.0, 0.05]}, - "leech": {"catch": 0.06, "max_tier": 2, "quality": [1.0, 0.15, 0.05]}, - "minnow": {"catch": 0.06, "max_tier": 2, "quality": [1.0, 0.5, 0.25, 0.05]}, - "squid": {"catch": 0.06, "max_tier": 2, "quality": [1.0, 0.8, 0.45, 0.15, 0.05]}, - "nautilus": {"catch": 0.06, "max_tier": 2, "quality": [1.0, 0.98, 0.75, 0.55, 0.25, 0.05]}, -} -const PARTICLE_DATA = { - "dust_run": preload("res://Scenes/Particles/dust_running.tscn"), - "dust_land": preload("res://Scenes/Particles/dust_land.tscn"), - "splash": preload("res://Scenes/Particles/water_splash.tscn"), - "small_splash": preload("res://Scenes/Particles/small_splash.tscn"), - "music": preload("res://Scenes/Particles/music_particle.tscn"), - "kiss": preload("res://Scenes/Particles/kiss.tscn"), -} - -export (NodePath) var hand_sprite_node -export (NodePath) var hand_bone_node -export var NPC_body = false -export var NPC_cosmetics = {"species": "species_cat", "pattern": "pattern_none", "primary_color": "pcolor_white", "secondary_color": "scolor_tan", "hat": "hat_none", "undershirt": "shirt_none", "overshirt": "overshirt_none", "title": "title_rank_1", "bobber": "bobber_default", "eye": "eye_halfclosed", "nose": "nose_cat", "mouth": "mouth_default", "accessory": [], "tail": "tail_cat"} -export var NPC_name = "NPC Test" -export var NPC_title = "npc title here" - -var camera_zoom = 5.0 - -var direction = Vector3() -var gravity_vec = Vector3() -var dive_vec = Vector3() -var velocity = Vector3() -var player_scale = 1.0 -var player_scale_y = 1.0 - -var cam_orbit_node = null -var cam_follow_node = null -var cam_push = 0.0 -var cam_push_cur = 0.0 -var cam_return = 1.0 -var force_cam_look = false -var mouse_world_pos = Vector3() -var mouse_look = false - -var state = STATES.DEFAULT -var walk_speed = 3.2 -var slow_walk_speed = 1.0 -var sprint_speed = 6.88 -var sneak_speed = 1.3 -var jump_height = 7.5 -var jump_leap = 0.0 -var dive_distance = 15.0 -var accel = 64.0 -var sprinting = false -var slow_walking = false -var sneaking = false -var diving = false -var request_jump = false -var ignore_snap = 0 -var snapped = false -var sitting = false -var locked = false -var busy = false -var in_air = false -var gravity_disable = false -var hud -var int_text = "" -var interact_cooldown = 60 -var xp_buildup = 0 -var showcase_ref - -var held_item = PlayerData.FALLBACK_ITEM -var caught_item = PlayerData.FALLBACK_ITEM -var held_item_weight = 1.0 -var previous_item = 0 -var primary_hold_timer = 0 -var fish_zone_data = {} -var rod_cast_data = "" -var casted_bait = "" -var rod_cast_dist = 0.0 -var rod_depth = 0 -var failed_casts = 0 -var recent_reel = 0.0 -var item_scene = null -var bobber_hpos = Vector3() -var bobber_vpos = 0 -var bobber_control = true -var last_valid_pos = Vector3() -var retract_splash = false -var show_local = false -var bait_warn = 2 -var in_rain = false -var consume_on_state_change = - 1 -var item_cooldown = 0 - -var rod_damage = 1.0 -var rod_spd = 1.0 -var rod_chance = 0.0 - - - -var boost_timer = 0 -var boost_amt = 1.3 -var boost_mult = 1.0 -var catch_drink_timer = 0 -var catch_drink_boost = 1.0 -var catch_drink_reel = 1.0 -var catch_drink_xp = 1.0 -var catch_drink_gold_add = Vector2(0, 0) -var catch_drink_gold_percent = 0.0 -var catch_drink_tier = 0 -var drunk_timer = 0 -var drunk_tier = 0 -var drunk_wander_length = 0 -var drunk_wander_dir = 0 - -var death_counter = 0 -var metal_detect_flop = false -var metal_detect_alert_level = 0 -var metal_detect_alert_cd = 0 - -var emoting = false -var emote_full = false -var emote_locked = false -var emote_looping = false -var animation_timer = 0 -var animation_goal = 99 -var buffer_state = - 1 -var animation_data = { - "moving": false, - "sprinting": false, - "sneaking": false, - "emoting": false, - "emote_full": false, - "diving": false, - "sitting": false, - "alert": false, - "emote": "", - "arm_state": "", - "caught_item": {}, - "arm_value": 0.0, - "item_bend": 0.0, - "busy": false, - "land": 0.0, - "talking": 0.0, - "recent_reel": 0.0, - "bobber_position": Vector3(), - "bobber_visible": false, - "caught_fish": false, - "player_scale": 1.0, - "player_scale_y": 1.0, - "mushroom": false, - "run_mult": 1.0, - "walk_mult": 1.25, - "drunk_tier": 0, - "wagging": false, - "emote_timescale": 1.0, - "back_bend": 0.0, - "dive_scrape": false, - "reel_slow": false, - "reel_fast": false, - "state": state, -} - - - -var custom_held_item = "" - -var cosmetic_data = {} -var selected_build_object -var old_rot = Vector3() -var rot_diff = 0.0 - -var prop_ids = [] -var cam_move = false -var freecamming = false - -onready var cam_base = $cam_base -onready var cam_pivot = $cam_base / cam_pivot -onready var camera = $Camera -onready var camera_point = $cam_base / cam_pivot / SpringArm / camera_point -onready var cam_arm = $cam_base / cam_pivot / SpringArm -onready var body = $body -onready var body_mesh = $body / player_body / Armature / Skeleton / body_main -onready var rot_help = $rot_help -onready var interact_range = $interact_range -onready var anim_tree = $body / AnimationTree -onready var skeleton = $body / player_body / Armature / Skeleton -onready var title = $Viewport / player_label -onready var item_sprite = get_node(hand_sprite_node) -onready var hand_bone = get_node(hand_bone_node) -onready var sound_emit = $sound_emit -onready var face = $body / player_body / Armature / Skeleton / face / player_face -onready var tail = $body / player_body / Armature / Skeleton / tail / holder / tail -onready var freecam_anchor = $camera_freecam_anchor -onready var sound_manager = $sound_manager - -onready var fish_detect = $detection_zones / fishing_detect -onready var fishing_update = $detection_zones / fishing_update -onready var fishing_area = $detection_zones / fishing_detect / fishing_area -onready var fish_timer = $fish_catch_timer -onready var bobber_preview = $bobber_preview -onready var bobber = $bobber -onready var bobber_mesh = $bobber / bobber_mesh -onready var bobber_line = $bobber / line -onready var ripples = $bobber / ripples -onready var caught_fish = $bobber / caught_item - -onready var shovel_area = $detection_zones / shovel_detect -onready var net_area = $detection_zones / net_detect - -onready var safe_check = $safe_check - -onready var lvlparticle = $emotion_particles / lvl_particles -onready var lvlparticleb = $emotion_particles / lvl_particles2 - -signal _animation_finished -signal _primary_release -signal _state_change -signal _menu_closed - - - - - -func _ready(): - add_to_group("player") - rot_help.set_as_toplevel(true) - cam_base.set_as_toplevel(true) - camera.set_as_toplevel(true) - freecam_anchor.set_as_toplevel(true) - anim_tree.tree_root = anim_tree.tree_root.duplicate(true) - _update_cosmetics(cosmetic_data) - title.visible = not dead_actor - - if NPC_body: - camera.queue_free() - remove_from_group("player") - - yield (get_tree(), "idle_frame") - var new = NPC_cosmetics - Network._send_actor_action(actor_id, "_update_cosmetics", [new]) - _update_cosmetics(new) - - yield (get_tree(), "idle_frame") - title.visible = true - title.label = NPC_name - title.title = NPC_title - - -func _setup_controlled(): - if NPC_body: return - - add_to_group("controlled_player") - camera = $Camera - camera.current = true - hud = load("res://Scenes/HUD/playerhud.tscn").instance() - hud.player = self - hud.connect("_player_sit", self, "_toggle_sit") - hud.connect("_play_emote", self, "_play_emote") - hud.connect("_menu_entered", self, "_hud_menu_entered") - hud.connect("_message_sent", self, "_message_sent") - hud.connect("_freecam_toggle", self, "_toggle_freecam") - get_tree().get_root().add_child(hud) - - PlayerData.connect("_clear_all_props", self, "_clear_all_props") - PlayerData.connect("_place_prop", self, "_create_prop") - PlayerData.connect("_play_sfx", self, "_play_sfx") - PlayerData.connect("_play_guitar", self, "_strum_guitar") - PlayerData.connect("_hammer_guitar", self, "_hammer_string") - PlayerData.connect("_return_to_spawn", self, "_return_to_spawn") - PlayerData.connect("_punched", self, "_punched") - PlayerData.connect("_item_removal", self, "_item_removal") - var delayed_update = get_tree().create_timer(1.0).connect("timeout", self, "_change_cosmetics") - - PlayerData.connect("_wag_toggle", self, "_wag") - PlayerData.connect("_kiss", self, "_kiss") - Network.connect("_user_connected", self, "_refresh_cosmetics") - - -func _setup_not_controlled(): - camera = $Camera - camera.queue_free() - - $bobber_preview.visible = false - $local_range.visible = false - $detection_zones / metal_detect_consume.monitoring = false - $detection_zones / metal_detect_consume.monitorable = false - - - $detection_zones / prop_detect.monitoring = false - $detection_zones / metal_detect_far.monitoring = false - $detection_zones / metal_detect.monitoring = false - $detection_zones / metal_detect_close.monitoring = false - $detection_zones / metal_detect_veryclose.monitoring = false - $detection_zones / punch.monitoring = false - $interact_range.monitoring = false - $water_detect.monitoring = false - $raincloud_check.monitoring = false - - - - - -func _physics_process(delta): - _process_animation() - _process_sounds() - - if ( not in_zone != $CollisionShape.disabled): - $CollisionShape.disabled = not in_zone - -func _process(delta): - if controlled: $paint_node._paint_process(delta) - -func _controlled_process(delta): - _get_input() - _process_movement(delta) - _process_timers() - _interact_check() - _update_animation_data() - _camera_update() - - _freecam_input(delta) - - body.visible = camera_zoom > 0.5 or cam_orbit_node != null - current_zone = world.active_zone - current_zone_owner = world.active_zone_owner - - fish_detect.translation.z = - rod_cast_dist - bobber_preview.translation.z = - clamp(primary_hold_timer * 0.06, 1.5, 9.0) - if bobber_preview.is_colliding(): $"%bobber_prev_mesh".global_transform.origin = bobber_preview.get_collision_point() + Vector3(0, 0.05, 0) - bobber_preview.visible = state == STATES.FISHING_CHARGE - - if recent_reel > 0: recent_reel -= 1 - if interact_cooldown > 0: interact_cooldown -= 1 - - animation_data["reel_slow"] = recent_reel > 8 - animation_data["reel_fast"] = state == STATES.FISHING_STRUGGLE - - - if (packet_send_interval == - 1 or Engine.get_physics_frames() % packet_send_interval == 0): - Network._send_actor_animation_update(actor_id, animation_data) - - Network.MESSAGE_ORIGIN = global_transform.origin - $local_range.visible = show_local - - $paint_node.global_transform.origin = mouse_world_pos - $metaldetect_dot.modulate.a = lerp($metaldetect_dot.modulate.a, 0.0, 0.1) - - -func _camera_update(): - cam_push_cur = lerp(cam_push_cur, cam_push * rod_cast_dist, 0.2) - var push = global_transform.basis.z * cam_push_cur - camera_zoom = clamp(camera_zoom, 0.0, 20.0) - - var cam_zoom = camera_zoom - var cam_zoom_lerp = 0.4 - var cam_follow_point = true - var cam_follow_pos = Vector3() - var cam_follow_rot = Vector3() - var sit_add = Vector3(0, - 0.2, 0.0) if sitting else Vector3.ZERO - var cam_base_pos = global_transform.origin + push + sit_add - - var desired_fov = 50 - if sprinting and direction != Vector3.ZERO: - desired_fov += 2 - if boost_timer > 0: desired_fov += 2 * boost_amt - if animation_data["mushroom"]: desired_fov += 10 - - camera.fov = lerp(camera.fov, desired_fov, 0.2) - - if is_instance_valid(cam_orbit_node) and cam_orbit_node.is_visible_in_tree(): - cam_base_pos = cam_orbit_node.global_transform.origin - - if is_instance_valid(cam_follow_node) and cam_follow_node.is_visible_in_tree(): - cam_follow_point = false - cam_follow_pos = cam_follow_node.global_transform.origin - cam_follow_rot = cam_follow_node.global_rotation - - var cam_speed = 0.08 - if cam_follow_point: - cam_return = lerp(cam_return, 1.0, cam_speed * 0.5) - camera.global_transform.origin = lerp(camera.global_transform.origin, camera_point.global_transform.origin, cam_return) - camera.rotation.x = lerp_angle(camera.rotation.x, camera_point.global_rotation.x, cam_return) - camera.rotation.y = lerp_angle(camera.rotation.y, camera_point.global_rotation.y, cam_return) - camera.rotation.z = lerp_angle(camera.rotation.z, camera_point.global_rotation.z, cam_return) - else : - cam_return = 0.0 - camera.global_transform.origin = lerp(camera.global_transform.origin, cam_follow_pos, cam_speed) - camera.rotation.x = lerp_angle(camera.rotation.x, cam_follow_rot.x, cam_speed) - camera.rotation.y = lerp_angle(camera.rotation.y, cam_follow_rot.y, cam_speed) - camera.rotation.z = lerp_angle(camera.rotation.z, cam_follow_rot.z, cam_speed) - - cam_base.global_transform.origin = cam_base_pos - cam_arm.spring_length = lerp(cam_arm.spring_length, cam_zoom, cam_zoom_lerp) - -func _interact_check(): - if not controlled: return - var in_range = false - for area in interact_range.get_overlapping_areas(): - if area.is_in_group("interactable") and area.is_visible_in_tree(): - int_text = area.text - in_range = true - break - if hud: - hud.interact = in_range - hud.int_text = int_text - -func _hud_menu_entered(menu): - if menu == 0: - if not freecamming: - cam_orbit_node = null - cam_follow_node = null - force_cam_look = false - _exit_showcase() - emit_signal("_menu_closed") - -func _process_timers(): - if catch_drink_timer > 0: - catch_drink_timer -= 1 - if catch_drink_timer <= 0: - catch_drink_boost = 1.0 - catch_drink_reel = 1.0 - catch_drink_xp = 1.0 - catch_drink_gold_add = Vector2(0, 0) - catch_drink_gold_percent = 0.0 - - drunk_tier = 0 - if drunk_timer > 0: - drunk_timer -= 1 - if drunk_timer >= 19000: drunk_tier = 3 - elif drunk_timer >= 10000: drunk_tier = 2 - elif drunk_timer > 0: drunk_tier = 1 - else : drunk_tier = 0 - - if item_cooldown > 0: item_cooldown -= 1 - - - - - -func _get_input(): - direction = Vector3.ZERO - - if Input.is_action_just_released("primary_action"): _primary_action_release() - if Input.is_action_pressed("primary_action"): _primary_action_hold() - else : primary_hold_timer = 0 - - if busy: - Input.set_mouse_mode(Input.MOUSE_MODE_VISIBLE) - return - - if Input.is_action_pressed("secondary_action") or camera_zoom <= 0.0: - if Input.mouse_mode != Input.MOUSE_MODE_CAPTURED: - PlayerData.original_mouse_position = get_tree().get_nodes_in_group("world_viewport")[0].get_mouse_position() - Input.set_mouse_mode(Input.MOUSE_MODE_CAPTURED) - else : - if Input.mouse_mode != Input.MOUSE_MODE_VISIBLE: - Input.set_mouse_mode(Input.MOUSE_MODE_VISIBLE) - Input.warp_mouse_position(PlayerData.original_mouse_position) - - if Input.is_action_just_released("zoom_in"): camera_zoom -= 0.5 - if Input.is_action_just_released("zoom_out"): camera_zoom += 0.5 - - if Input.is_action_just_pressed("interact"): _interact() - - if Input.is_action_just_pressed("bind_1"): _equip_hotbar(0) - if Input.is_action_just_pressed("bind_2"): _equip_hotbar(1) - if Input.is_action_just_pressed("bind_3"): _equip_hotbar(2) - if Input.is_action_just_pressed("bind_4"): _equip_hotbar(3) - if Input.is_action_just_pressed("bind_5"): _equip_hotbar(4) - - if Input.is_action_just_pressed("bark"): - _bark() - if Input.is_action_just_pressed("kiss"): - _kiss() - - if locked: return - - - if is_instance_valid(camera): - var camera_cam = camera - var ray_length = 1000 - var mouse_pos = get_tree().get_nodes_in_group("world_viewport")[0].get_mouse_position() / Globals.pixelize_amount - var from = camera_cam.project_ray_origin(mouse_pos) - var to = from + camera_cam.project_ray_normal(mouse_pos) * ray_length - var space_state = get_world().get_direct_space_state() - var result = space_state.intersect_ray(from, to, []) - if result.has("position"): - mouse_world_pos = result["position"] - mouse_world_pos.y = global_transform.origin.y - - if freecamming: return - if Input.is_action_just_pressed("move_jump"): request_jump = true - - mouse_look = false - - if sitting: return - - if Input.is_action_pressed("move_forward"): direction -= cam_base.transform.basis.z - if Input.is_action_pressed("move_back"): direction += cam_base.transform.basis.z - if Input.is_action_pressed("move_right"): direction += cam_base.transform.basis.x - if Input.is_action_pressed("move_left"): direction -= cam_base.transform.basis.x - - if drunk_wander_length > 0: - direction += drunk_wander_dir - drunk_wander_length -= 1 - - mouse_look = Input.is_action_pressed("mouse_look") - - sprinting = not Input.is_action_pressed("move_sneak") and Input.is_action_pressed("move_sprint") - sneaking = Input.is_action_pressed("move_sneak") and not Input.is_action_pressed("move_sprint") - slow_walking = Input.is_action_pressed("move_walk") - - if emote_locked: - request_jump = false - direction = Vector3.ZERO - -func _input(event): - if not controlled: return - - var mouse_sens = OptionsMenu.mouse_sens - var invert = Vector2(1, 1) - if OptionsMenu.mouse_invert == 1: invert.x = - 1 - elif OptionsMenu.mouse_invert == 2: invert.y = - 1 - elif OptionsMenu.mouse_invert == 3: invert = Vector2( - 1, - 1) - - if event is InputEventMouseMotion and Input.get_mouse_mode() == Input.MOUSE_MODE_CAPTURED: - cam_base.rotation_degrees.y -= event.relative.x * mouse_sens * invert.x - cam_pivot.rotation_degrees.x -= event.relative.y * mouse_sens * invert.y - cam_pivot.rotation_degrees.x = clamp(cam_pivot.rotation_degrees.x, - 80, 80) - -func _unhandled_input(event): - if not controlled: return - - if event.is_action_pressed("primary_action"): _primary_action() - - - - -func _process_movement(delta): - var snap_vec = Vector3(0, - 0.4, 0) - var y_slow = 0.02 - - if is_on_floor() and in_air: - if gravity_vec.y <= - 12.0 and not diving: - diving = true - dive_vec = gravity_vec.length() * - transform.basis.z * 0.2 - player_scale_y = 1.0 - clamp(((abs(gravity_vec.y) / 24.0) * 6.0), 0.0, 0.6) - in_air = false - _sync_particle("dust_land", Vector3(0, - 1, 0)) - _sync_sfx("land") - animation_data["land"] = 0.3 - elif gravity_vec.y < - 6.0: - in_air = true - - player_scale_y = lerp(player_scale_y, 1.0, 0.35) - - if is_on_floor() and ignore_snap <= 0: - y_slow = 0.2 - gravity_vec = get_floor_normal() * - 1.0 - snapped = true - elif snapped and ignore_snap <= 0: - gravity_vec = Vector3.ZERO - snapped = false - else : - snap_vec = Vector3.ZERO - var grav = GRAVITY * Vector3.DOWN * delta - gravity_vec += grav - - if request_jump: - request_jump = false - snap_vec = Vector3.ZERO - if is_on_floor(): - _sync_sfx("jump") - snapped = false - diving = false - gravity_vec = Vector3(0, jump_height, 0) - if sprinting and jump_leap > 0.0: - gravity_vec += - transform.basis.z.normalized() * jump_leap - elif not diving: - _sync_sfx("dive_woosh") - snapped = false - diving = true - dive_vec = - transform.basis.z.normalized() * dive_distance - dive_vec.y = 0 - gravity_vec += Vector3(0, jump_height * 1, 0) - _toggle_sit(true) - - if ignore_snap > 0: - snap_vec = Vector3.ZERO - ignore_snap -= 1 - snapped = false - - boost_mult = 1.0 - if boost_timer > 0: - boost_timer -= 1 - boost_mult = boost_amt - - var _speed = walk_speed - if sprinting: _speed = sprint_speed * boost_mult - elif sneaking: _speed = sneak_speed - elif slow_walking: _speed = slow_walk_speed - var _accel = accel if is_on_floor() else accel * 0.35 - - var speed_mult = 1.25 - - - speed_mult = clamp(speed_mult - ((held_item_weight / 25.0) * 0.05), 0.15, 1.0) - - if diving: speed_mult = 1.0 - if gravity_disable: gravity_vec = Vector3.ZERO - - velocity = velocity.move_toward(direction.normalized() * _speed * speed_mult, delta * _accel) - move_and_slide_with_snap(velocity + gravity_vec + dive_vec, snap_vec, Vector3.UP) - - dive_vec = dive_vec.move_toward(Vector3.ZERO, delta * _accel * y_slow) - if not diving: dive_vec = Vector3.ZERO - - rot_help.global_transform.origin = global_transform.origin - - if direction != Vector3.ZERO: - var dir = direction - if diving and dive_vec != Vector3.ZERO: dir = dive_vec.normalized() - - rot_help.look_at(cam_base.global_transform.origin + dir, Vector3.UP) - rotation.y = lerp_angle(rotation.y, rot_help.rotation.y, 0.2) - elif force_cam_look: - rot_help.look_at(camera.global_transform.origin, Vector3.UP) - rotation.y = lerp_angle(rotation.y, rot_help.rotation.y, 0.2) - elif mouse_look and not busy: - rot_help.look_at(mouse_world_pos, Vector3.UP) - rot_diff = abs(rot_help.rotation.y - rotation.y) - rotation.y = lerp_angle(rotation.y, rot_help.rotation.y, 0.2) - - rot_diff = lerp(rot_diff, 0.0, 0.5) - - if diving and dive_vec.length() > 0.4 and is_on_floor(): - - animation_data["dive_scrape"] = true - - if Engine.get_idle_frames() % 4 == 0: - _sync_particle("dust_run", Vector3(0, - 0.8, 0)) - else : - - animation_data["dive_scrape"] = false - - if gravity_vec.y < - 250: - _kill() - - - - -func _enter_state(new_state): - if new_state == - 1: return - - if consume_on_state_change != - 1: - PlayerData._remove_item(consume_on_state_change, false) - consume_on_state_change = - 1 - - state = new_state - emit_signal("_state_change") - cam_push = 0.0 - animation_data["bobber_visible"] = false - - match state: - STATES.DEFAULT: - animation_data["item_bend"] = 0.0 - locked = false - STATES.BUSY: locked = false - STATES.OBTAIN: locked = true - STATES.SHOWCASE: - locked = true - _equip_item(PlayerData._find_item_code(showcase_ref), true, true) - - STATES.FISHING: - cam_push = - 0.3 - locked = true - animation_data["bobber_visible"] = true - PlayerData.emit_signal("_help_update", "hold to reel (SPRINT reels quicker)") - _enter_animation("rod_idle", true) - STATES.FISHING_CANCEL: - animation_data["item_bend"] = 0.1 - cam_push = - 0.3 - locked = true - rod_cast_dist = 0.0 - animation_data["bobber_visible"] = true - PlayerData.emit_signal("_item_equip", held_item.ref) - _bobber_retract() - _enter_animation("rod_retract", false, true, STATES.DEFAULT) - STATES.FISHING_CAST: - cam_push = - 0.3 - animation_data["bobber_visible"] = true - locked = true - STATES.FISHING_CHARGE: - cam_push = 0.0 - locked = false - STATES.FISHING_STRUGGLE: - animation_data["item_bend"] = - 0.7 - cam_push = - 0.3 - animation_data["bobber_visible"] = true - locked = true - _enter_animation("rod_struggle", true) - - STATES.GUITAR: locked = true - - STATES.SHOVEL_CAST: locked = true - STATES.SHOVEL_STRUGGLE: - locked = true - _shovel_check() - STATES.SHOVEL_CANCEL: - locked = true - _enter_animation("shovel_retract", false, true, STATES.DEFAULT) - - STATES.NET_CAST: locked = true - STATES.NET_STRUGGLE: - locked = true - _net_check() - STATES.NET_CANCEL: - locked = true - _enter_animation("net_retract", false, true, STATES.DEFAULT) - - STATES.CONSUME_ITEM: - locked = false - _equip_item({"ref": 0}, true, true) - _enter_state(STATES.DEFAULT) - -func _primary_action(): - match state: - STATES.DEFAULT: - _use_item() - STATES.OBTAIN: - _enter_animation("equip", false, true, 0, false, 1.5) - -func _primary_action_hold(): - primary_hold_timer += 1 - - match state: - STATES.DEFAULT: - return - STATES.FISHING: - _enter_animation("rod_reel", true) - rod_cast_dist -= 0.04 if Input.is_action_pressed("move_sprint") else 0.01 - bobber_control = true - recent_reel = 15 - if rod_cast_dist < 1.5: _enter_state(STATES.FISHING_CANCEL) - -func _primary_action_release(): - emit_signal("_primary_release") - - match state: - STATES.DEFAULT: - _release_item() - STATES.FISHING: - _enter_animation("rod_idle", true) - STATES.METAL_DETECTOR: - _enter_state(STATES.DEFAULT) - - - - - -func _interact(): - if not controlled or interact_cooldown > 0: return - for area in interact_range.get_overlapping_areas(): - if area.is_in_group("interactable") and area.is_visible_in_tree(): - area._activate(self) - interact_cooldown = 60 - return - -func _enter_zone(zone, entrance_id, zone_owner = - 1): - if not controlled: return - world._enter_zone(zone, zone_owner) - PlayerData.player_saved_zone = zone - PlayerData.player_saved_zone_owner = zone_owner - - print("Finding entrance w id: ", entrance_id, " and owner: ", zone_owner) - for entrance in get_tree().get_nodes_in_group("area_entrance"): - print(entrance.entrance_id, ": ", entrance.entrance_owner, " ?") - if entrance.entrance_id == entrance_id and entrance.entrance_owner == zone_owner: - global_transform.origin = entrance.global_transform.origin - last_valid_pos = global_transform.origin - return - - print("Fallback!") - global_transform.origin = world.map.spawn_position.global_transform.origin - world._enter_zone("main_zone", - 1) - PlayerData.player_saved_zone = "main_zone" - PlayerData.player_saved_zone_owner = - 1 - last_valid_pos = global_transform.origin - -func _obtain_item(ref, bonus_text = [], journal_check = true): - old_rot = rotation - showcase_ref = ref - _equip_item({"ref": - 1}, true, true) - _enter_animation("equip", false, false, STATES.SHOWCASE, false, 1.5) - _play_sfx("strum") - - - var data = PlayerData._find_item_code(ref) - var text = "You caught a " + PlayerData._get_item_name(ref) + "! [color=#d5aa73](It's " + str(data["size"]) + "cm!)[/color]\n\n" + str(Globals.item_data[data["id"]]["file"].catch_blurb) - - if journal_check: - var new = true - for type in PlayerData.journal_logs.keys(): - for entry in PlayerData.journal_logs[type].keys(): - if entry == data.id and PlayerData.journal_logs[type][entry].count > 1: - new = false - break - if new: - text = "Woah, a new creature! " + text - - hud.dialogue_text = [text] - if bonus_text != []: hud.dialogue_text.append_array(bonus_text) - hud._change_menu(6) - - force_cam_look = true - yield (get_tree().create_timer(0.1), "timeout") - cam_follow_node = $catch_cam_position - -func _level_up(): - yield (get_tree().create_timer(0.4), "timeout") - var bubble = title._create_level_bubble() - Network._send_actor_action(actor_id, "_sync_level_bubble", [], false) - GlobalAudio._play_sound("jingle_win") - - _play_emote("emote_cheer", "happy") - - $emotion_particles / lvl_particles.restart() - $emotion_particles / lvl_particles2.restart() - -func _exit_showcase(): - if state != STATES.SHOWCASE: return - _exit_animation() - _enter_state(STATES.DEFAULT) - _equip_item(PlayerData._find_item_code(previous_item), false) - get_tree().create_tween().tween_property(self, "rotation", old_rot, 0.3) - - if xp_buildup > 0: - PlayerData._add_xp(ceil(xp_buildup)) - xp_buildup = 0 - -func _equip_hotbar(slot): - if locked or not PlayerData.hotbar.keys().has(slot): return - var ref = PlayerData.hotbar[slot] - _equip_item(PlayerData._find_item_code(ref)) - -func _equip_item(item_data, skip_anim = false, forced = false, set_prev = true): - if set_prev and held_item["ref"] != 0: previous_item = held_item["ref"] - if (state != STATES.DEFAULT and not forced) or held_item["ref"] == item_data["ref"]: return - - if not item_data.keys().has("id") or not Globals.item_data.keys().has(item_data["id"]): - item_data = PlayerData.FALLBACK_ITEM - - PlayerData.emit_signal("_item_equip", item_data["ref"]) - - if not skip_anim: - _sync_sfx("equip") - _enter_state(STATES.BUSY) - _enter_animation("equip", false, false, STATES.DEFAULT, false, 1.5) - yield (self, "_animation_finished") - - - var held_data = item_data.duplicate() - - hud.show_bait = Globals.item_data[item_data["id"]]["file"].show_bait - - var data = Globals.item_data[item_data["id"]]["file"] - held_item_weight = item_data["size"] - if not data.uses_size: held_item_weight = 0.0 - - _update_held_item(held_data) - Network._send_actor_action(actor_id, "_update_held_item", [held_item], false) - -func _use_item(): - if held_item.empty(): return - var item_data = Globals.item_data[held_item["id"]]["file"] - if has_method(item_data.action) and item_data.action != "": - callv(item_data.action, item_data.action_params) - -func _release_item(): - if held_item.empty(): return - var item_data = Globals.item_data[held_item["id"]]["file"] - if has_method(item_data.release_action) and item_data.release_action != "": - call(item_data.release_action) - - - - - - - -func _cast_fishing_rod(): - rod_cast_data = PlayerData.LURE_DATA[PlayerData.lure_selected].effect_id - animation_data["caught_item"] = {} - bait_warn = 1 - - rod_damage = [1, 3, 10, 20, 35, 50][PlayerData.rod_power_level] - rod_spd = [0.0, 0.1, 0.24, 0.4, 0.7, 1.0][PlayerData.rod_speed_level] - rod_chance = [0.0, 0.02, 0.04, 0.06, 0.08, 0.1][PlayerData.rod_chance_level] - - _enter_state(STATES.FISHING_CHARGE) - _enter_animation("rod_wind", true, false, - 1, false) - yield (self, "_primary_release") - if state != STATES.FISHING_CHARGE: return - _enter_state(STATES.FISHING_CAST) - _sync_sfx("woosh", null, 1.0, 0.4) - var strength = clamp(primary_hold_timer * 0.06, 1.5, 9.0) - rod_cast_dist = strength - fish_detect.translation.z = - strength - - var is_valid_fishing_spot = false - if bobber_preview.is_colliding() and bobber_preview.get_collider().is_in_group("valid_water"): - is_valid_fishing_spot = true - - _bobber_cast(rod_cast_dist, bobber_preview.get_collision_point() + Vector3(0, 0.75, 0), is_valid_fishing_spot) - - rod_depth = 0 - - casted_bait = PlayerData.bait_selected - if casted_bait != "" and PlayerData.bait_inv[casted_bait] <= 0: casted_bait = "" - - _exit_animation() - if is_valid_fishing_spot: - animation_data["item_bend"] = 0.3 - retract_splash = true - _enter_animation("rod_cast", false, true, STATES.FISHING) - else : - animation_data["item_bend"] = 0.3 - retract_splash = false - _enter_animation("rod_cast", false, true, STATES.FISHING_CANCEL) - - yield (get_tree().create_timer(0.4), "timeout") - animation_data["item_bend"] = - 0.2 - -func _on_fish_catch_timer_timeout(): - if not controlled: return - - fish_timer.wait_time = rand_range(2.0, 3.0) - fish_timer.start() - - if state != STATES.FISHING: return - var fish_type = "ocean" - var junk_mult = 1.0 - - - fish_zone_data = {"id": - 1, "boost": 0.0} - for zone in fishing_area.get_overlapping_areas(): - if zone.is_in_group("fish_zone"): - fish_zone_data["id"] = zone.id - fish_zone_data["boost"] = zone.chance_boost - junk_mult = zone.junk_mult - if zone.fish_type != "": fish_type = zone.fish_type - - var fish_chance = 0.0 - var base_chance = BAIT_DATA[casted_bait]["catch"] - fish_chance = base_chance - fish_chance += (base_chance * failed_casts) - fish_chance += (base_chance * rod_chance) - fish_chance += fish_zone_data["boost"] * fish_chance - if recent_reel > 0: fish_chance *= 1.1 - if rod_cast_data == "attractive": fish_chance *= 1.3 - if in_rain: fish_chance *= 1.1 - - fish_chance *= catch_drink_boost - print("Fish chance w ", fish_chance, "w type ", fish_type) - - bait_warn -= 1 - if bait_warn <= 0 and fish_chance <= 0.0: - bait_warn = 8 - var text = "" - if casted_bait == "": - text = "[color=#ac0029]You've got no bait attached! You won't catch any fish like that...[/color]" - else : - text = "[color=#ac0029]Seems nothing is going to bite... perhaps your bait isn't for this water...[/color]" - - Network._update_chat(text) - - if randf() > fish_chance: - failed_casts += 0.05 - return - failed_casts = 0.0 - - - var bait_use_chance = 1.0 - if rod_cast_data == "efficient": bait_use_chance = 0.8 - - if randf() < bait_use_chance: PlayerData._use_bait(casted_bait) - else : PlayerData._send_notification("The Efficient Lure saved your bait!") - var max_tier = BAIT_DATA[casted_bait]["max_tier"] - - var double_bait = 0.0 - if ["large", "sparkling", "double"].has(rod_cast_data): double_bait = 0.25 - if randf() < double_bait: - PlayerData._use_bait(casted_bait) - PlayerData._send_notification("Your lure used extra bait...", 1) - - if rod_cast_data == "gold": - for i in 2: PlayerData._use_bait(casted_bait) - PlayerData._send_notification("Your golden lure used extra bait...", 1) - - var treasure_mult = 1.0 - if rod_cast_data == "magnet": treasure_mult = 2.0 - if rod_cast_data == "salty": fish_type = "ocean" - if rod_cast_data == "fresh": fish_type = "lake" - - var force_av_size = false - - if randf() < 0.05 * treasure_mult * junk_mult: - fish_type = "water_trash" - max_tier = 0 - force_av_size = true - - - - - - - - if in_rain and randf() < 0.08: - fish_type = "rain" - - - var rolls = [] - for i in 3: - var roll = Globals._roll_loot_table(fish_type, max_tier) - var s = Globals._roll_item_size(roll) - rolls.append([roll, s]) - - - var reroll_type = "none" - if rod_cast_data == "small": reroll_type = "small" - if rod_cast_data == "sparkling": reroll_type = "tier" - if rod_cast_data == "large": reroll_type = "large" - if rod_cast_data == "gold": reroll_type = "rare" - - var chosen = rolls[0] - for roll in rolls: - match reroll_type: - "none": - chosen = roll - - "small": - if roll[1] < chosen[1]: - chosen = roll - - "large": - if roll[1] > chosen[1]: - chosen = roll - - "tier": - var old_tier = Globals.item_data[chosen[0]]["file"].tier - var new_tier = Globals.item_data[roll[0]]["file"].tier - if new_tier > old_tier: - chosen = roll - - "rare": - var new_rare = Globals.item_data[roll[0]]["file"].rare - if new_rare: - chosen = roll - - var fish_roll = chosen[0] - var size = chosen[1] - - - var quality = PlayerData.ITEM_QUALITIES.NORMAL - var r = randf() - for q in PlayerData.ITEM_QUALITIES.size(): - if BAIT_DATA[casted_bait]["quality"].size() - 1 < q: - print("bait does not support rarity ", q) - break - if randf() < BAIT_DATA[casted_bait]["quality"][q]: - quality = q - print("-------------------------Rolled Quality: ", quality) - - if randf() < 0.02 * treasure_mult: - fish_roll = "treasure_chest" - size = 60.0 - quality = 0 - - var data = Globals.item_data[fish_roll]["file"] - var quality_data = PlayerData.QUALITY_DATA[quality] - - if force_av_size: size = data.average_size - - var diff_mult = clamp(size / data.average_size, 0.7, 1.8) - var difficulty = clamp((data.catch_difficulty * diff_mult * quality_data.diff) + quality_data.bdiff, 1.0, 250.0) - - var xp_mult = size / data.average_size - if xp_mult < 0.15: xp_mult = 1.25 + xp_mult - xp_mult = max(0.5, xp_mult) - var xp_add = ceil(data.obtain_xp * xp_mult * catch_drink_xp * quality_data.worth) - - print("Total Rolls: ", rolls) - print("Roll fish ", fish_roll, " with size ", size, " and diff Mult: ", diff_mult) - - if fish_zone_data["id"] != - 1: - _wipe_actor(fish_zone_data["id"]) - Network._send_actor_action(actor_id, "_wipe_actor", [fish_zone_data["id"]]) - - _enter_state(STATES.FISHING_STRUGGLE) - animation_data["alert"] = true - yield (get_tree().create_timer(1.0), "timeout") - - animation_data["alert"] = false - - var delay_time = 0 - while hud.using_chat: - yield (get_tree().create_timer(0.15), "timeout") - delay_time += 1 - - if delay_time > 200: - _enter_state(STATES.FISHING_CANCEL) - return - - hud._open_minigame("fishing3", {"fish": fish_roll, "rod_type": rod_cast_data, "reel_mult": catch_drink_reel, "quality": quality, "damage": rod_damage, "speed": rod_spd}, difficulty) - var success = yield (hud, "_minigame_finished") - print("SUCCESS: ", success) - if success: - animation_data["caught_item"] = {"id": fish_roll, "size": size} - _enter_state(STATES.FISHING_CANCEL) - yield (self, "_state_change") - - var ref - var catches = 1 - var bonus_text = [] - - if rod_cast_data == "double" and randf() < 0.15: - catches = 2 - bonus_text.append("Your Double Hook doubled the fish!") - - if PlayerData.rod_luck_level > 0 and randf() < 0.15: - bonus_text.append("How lucky! You found a bonus [color=#d57900]Coin Bag[/color] aswell!") - PlayerData._add_item("luck_moneybag", - 1, randi() % 15 + 15, PlayerData.rod_luck_level) - - var tags = [] - - for i in catches: - ref = PlayerData._add_item(fish_roll, - 1, size, quality, tags) - PlayerData._log_item(fish_roll, size, quality) - PlayerData._quest_progress("catch", fish_roll) - PlayerData._quest_progress("catch_type", data.loot_table) - PlayerData._quest_progress("catch_small", PlayerData._get_size_type(fish_roll, size)) - PlayerData._quest_progress("catch_big", PlayerData._get_size_type(fish_roll, size)) - if fish_roll == "treasure_chest": PlayerData._quest_progress("catch_treasure") - if in_rain: PlayerData._quest_progress("catch_rain") - if data.tier == 2: PlayerData._quest_progress("catch_hightier") - xp_buildup += xp_add - - PlayerData._catch_fish() - - _obtain_item(ref, bonus_text) - - if rod_cast_data == "lucky": - var worth = PlayerData._get_item_worth(ref) - var gold = max(1, ceil(worth * rand_range(0.01, 0.1))) - PlayerData.money += gold - PlayerData._send_notification("Your Lucky Lure got you $" + str(gold) + "!") - - if catch_drink_gold_add != Vector2(0, 0): - var gold = max(1, ceil(rand_range(catch_drink_gold_add.x, catch_drink_gold_add.y))) - PlayerData.money += gold - PlayerData._send_notification("Your Catcher's Luck got you $" + str(gold) + "!") - - if catch_drink_gold_percent > 0.0: - var worth = PlayerData._get_item_worth(ref) - var gold = max(1, ceil(rand_range(worth * 0.01, worth * catch_drink_gold_percent))) - PlayerData.money += gold - PlayerData._send_notification("Your Catcher's Luck Ultra got you $" + str(gold) + "!") - - - else : - _enter_state(STATES.FISHING_CANCEL) - -func _wipe_actor(id): - var actor = world._get_actor_by_id(id) - print("Wiping actor ", id, " : ", actor) - if is_instance_valid(actor): - actor._deinstantiate(true) - - -func _shovel(): - _enter_state(STATES.SHOVEL_CAST) - _enter_animation("shovel_dig", false, true, STATES.SHOVEL_STRUGGLE) - -func _shovel_check(): - var mound = false - - var area - for a in shovel_area.get_overlapping_areas(): - if a.is_in_group("mound"): - mound = true - area = a - - if is_instance_valid(area): - var id = area.owner.actor_id - _wipe_actor(id) - Network._send_actor_action(actor_id, "_wipe_actor", [id]) - - if mound: - _enter_animation("shovel_struggle", true, true) - hud._open_minigame("shoveling", {"damage": 5.0}) - var success = yield (hud, "_minigame_finished") - print("SUCCESS: ", success) - if success: - _enter_state(STATES.SHOVEL_CANCEL) - yield (self, "_state_change") - var ref = PlayerData._add_item("spider") - - - _obtain_item(ref) - else : - _enter_state(STATES.SHOVEL_CANCEL) - else : - _enter_state(STATES.SHOVEL_CANCEL) - - -func _cast_net(): - _enter_state(STATES.NET_CAST) - _enter_animation("net_use", false, true, STATES.NET_STRUGGLE) - -func _net_check(): - var bug = false - - var area - for a in net_area.get_overlapping_areas(): - if a.is_in_group("bug"): - bug = true - area = a - - var bug_id = "" - if is_instance_valid(area): - var id = area.owner.actor_id - bug_id = area.owner.bug_id - _wipe_actor(id) - Network._send_actor_action(actor_id, "_wipe_actor", [id]) - - if bug: - _enter_animation("net_struggle", true, true) - hud._open_minigame("shoveling", {"damage": 5.0}) - var success = yield (hud, "_minigame_finished") - print("SUCCESS: ", success) - if success: - _enter_state(STATES.NET_CANCEL) - yield (self, "_state_change") - var size = Globals._roll_item_size(bug_id) - var ref = PlayerData._add_item(bug_id, - 1, size) - - - _obtain_item(ref) - else : - _enter_state(STATES.NET_CANCEL) - else : - _enter_state(STATES.NET_CANCEL) - - - - - - - - - - - - - - -func _toggle_freecam(): - if busy: return - - if not freecamming: - PlayerData._send_notification("entering freecam mode") - freecamming = true - hud.freecamming = true - cam_orbit_node = freecam_anchor - freecam_anchor.global_transform.origin = global_transform.origin - else : - PlayerData._send_notification("exiting freecam mode") - freecamming = false - hud.freecamming = false - cam_orbit_node = null - -func _freecam_input(delta): - if not freecamming: return - - var speed = 4.0 - if Input.is_action_pressed("move_sprint"): - speed = 7.0 - if Input.is_action_pressed("move_sneak"): - speed = 1.0 - var max_dist = 15.0 - var build_dir = Vector3.ZERO - - if Input.is_action_pressed("move_forward"): build_dir -= cam_base.transform.basis.z - if Input.is_action_pressed("move_back"): build_dir += cam_base.transform.basis.z - if Input.is_action_pressed("move_right"): build_dir += cam_base.transform.basis.x - if Input.is_action_pressed("move_left"): build_dir -= cam_base.transform.basis.x - if Input.is_action_pressed("move_up"): build_dir += cam_base.transform.basis.y - if Input.is_action_pressed("move_down"): build_dir -= cam_base.transform.basis.y - - freecam_anchor.global_transform.origin += build_dir * speed * delta - - freecam_anchor.global_transform.origin.x = clamp(freecam_anchor.global_transform.origin.x, global_transform.origin.x - max_dist, global_transform.origin.x + max_dist) - freecam_anchor.global_transform.origin.y = clamp(freecam_anchor.global_transform.origin.y, global_transform.origin.y - max_dist, global_transform.origin.y + max_dist) - freecam_anchor.global_transform.origin.z = clamp(freecam_anchor.global_transform.origin.z, global_transform.origin.z - max_dist, global_transform.origin.z + max_dist) - - - - - -func _create_prop(ref, offset = Vector3(0, 1, 0)): - if not controlled: return - - - for prop in prop_ids: - if prop.ref == ref: - _wipe_actor(prop.id) - prop_ids.erase(prop) - PlayerData._send_notification("clearing prop", 1) - PlayerData.props_placed = prop_ids - PlayerData.emit_signal("_prop_update") - return - - - if $detection_zones / prop_detect.get_overlapping_bodies().size() > 0 or not is_on_floor() or not $detection_zones / prop_ray.is_colliding(): - PlayerData._send_notification("invalid prop placement", 1) - return - - - if prop_ids.size() > 4: - PlayerData._send_notification("prop limit reached", 1) - return - - - var item = PlayerData._find_item_code(ref) - var data = Globals.item_data[item["id"]]["file"] - var ver_offset = Vector3(0, 1.0, 0) * (1.0 - player_scale) - print(ver_offset) - var pos = global_transform.origin + (global_transform.basis.z * - 2.0) - offset + ver_offset - var rot = rotation + Vector3(0, deg2rad(180), 0) - var prop_code = data.prop_code - - - var blacklist = ["island_tiny", "island_med", "island_big"] - if current_zone_owner != - 1: - if blacklist.has(prop_code): - PlayerData._send_notification("this prop cannot be spawned here...", 1) - return - - - var new_id = Network._sync_create_actor(prop_code, pos, current_zone, - 1, Network.STEAM_ID, {"rotation": rot, "current_zone_owner": current_zone_owner}) - prop_ids.append({"id": new_id, "ref": ref, "prop_id": data.action}) - PlayerData.props_placed = prop_ids - PlayerData.emit_signal("_prop_update") - PlayerData._send_notification("prop placed", 0) - - _sync_particle("dust_land", pos, true) - _sync_sfx("poof", pos) - -func _clear_props(): - if prop_ids.size() <= 0: - PlayerData._send_notification("no props to undo", 1) - return - PlayerData._send_notification("undo prop", 0) - _wipe_actor(prop_ids[prop_ids.size() - 1][0]) - prop_ids.remove(prop_ids.size() - 1) - -func _clear_all_props(): - if prop_ids.size() <= 0: - PlayerData._send_notification("no props to clear", 1) - return - - PlayerData._send_notification("clearing all props", 0) - for prop in prop_ids: - _wipe_actor(prop.id) - prop_ids.clear() - PlayerData.props_placed = prop_ids - PlayerData.emit_signal("_prop_update") - -func _item_place_prop(): - if not controlled: return - _create_prop(held_item.ref) - - - - - -func _update_animation_data(): - if animation_data["emote"] != "": - var anim = $body / player_body / AnimationPlayer.get_animation(animation_data["emote"]) - animation_timer += 1 - animation_goal = floor((anim.length * 60) / animation_data["emote_timescale"]) - if animation_timer >= animation_goal and emoting and animation_goal > 0 and not emote_looping: - print("Finished Emoting ", animation_data["emote"], " bufferstate: ", buffer_state) - emit_signal("_animation_finished") - emoting = false - emote_full = false - emote_locked = false - animation_data["emote"] = "" - if buffer_state != - 1: _enter_state(buffer_state) - - animation_data["emoting"] = emoting - animation_data["emote_full"] = emoting and emote_full - animation_data["moving"] = direction != Vector3.ZERO or rot_diff > 0.05 - animation_data["sprinting"] = sprinting and direction != Vector3.ZERO - animation_data["sneaking"] = sneaking and direction != Vector3.ZERO - animation_data["diving"] = diving - animation_data["sitting"] = sitting - animation_data["busy"] = busy and state == STATES.DEFAULT and not emoting - animation_data["land"] = lerp(animation_data["land"], 0.0, 0.04) - animation_data["talking"] = lerp(animation_data["talking"], 0.0, 0.1) - animation_data["recent_reel"] = lerp(animation_data["recent_reel"], recent_reel, 0.2) - animation_data["caught_fish"] = state == STATES.FISHING_STRUGGLE - animation_data["player_scale"] = lerp(animation_data["player_scale"], player_scale, 0.06) - animation_data["player_scale_y"] = lerp(animation_data["player_scale_y"], player_scale_y, 0.06) - animation_data["run_mult"] = 1.15 * boost_mult - animation_data["walk_mult"] = 1.25 if not slow_walking else 0.7 - animation_data["drunk_tier"] = drunk_tier - animation_data["state"] = state - - $emotion_particles / mushroom_trail.emitting = animation_data["mushroom"] - $emotion_particles / drunk_particles.emitting = animation_data["drunk_tier"] > 1 - if is_on_floor(): animation_data["mushroom"] = false - - if not bobber_control: - animation_data["bobber_position"] = Vector3(bobber_hpos.x, bobber_vpos, bobber_hpos.z) - else : - var pos = global_transform.origin + ( - rod_cast_dist * global_transform.basis.z) - pos.y = bobber_vpos + sin(OS.get_ticks_msec() * 0.002) * 0.05 - animation_data["bobber_position"] = lerp(animation_data["bobber_position"], pos, 0.2) - -func _process_animation(): - var anim_lerp = 0.4 - - var emote_type = 0 - if animation_data["emoting"]: emote_type = 1 - if animation_data["emote_full"]: emote_type = 2 - - anim_tree.set("parameters/emote_full/blend_amount", lerp(anim_tree.get("parameters/emote_full/blend_amount"), float(emote_type == 2), anim_lerp)) - if anim_tree.get("parameters/emote_full/blend_amount") < 0.1 and emote_type != 2: anim_tree.set("parameters/emote_full/blend_amount", 0.0) - anim_tree.set("parameters/emote_half/blend_amount", lerp(anim_tree.get("parameters/emote_half/blend_amount"), float(emote_type == 1), anim_lerp)) - if anim_tree.get("parameters/emote_half/blend_amount") < 0.1 and emote_type != 1: anim_tree.set("parameters/emote_half/blend_amount", 0.0) - - anim_tree.set("parameters/arms/blend_amount", lerp(anim_tree.get("parameters/arms/blend_amount"), float( not animation_data["emoting"]), anim_lerp)) - anim_tree.set("parameters/movement/blend_amount", lerp(anim_tree.get("parameters/movement/blend_amount"), float(animation_data["moving"]), anim_lerp)) - anim_tree.set("parameters/run_blend/blend_amount", lerp(anim_tree.get("parameters/run_blend/blend_amount"), float(animation_data["sprinting"]) - float(animation_data["sneaking"]), anim_lerp)) - anim_tree.set("parameters/dive/blend_amount", lerp(anim_tree.get("parameters/dive/blend_amount"), float(animation_data["diving"]), anim_lerp)) - anim_tree.set("parameters/arm_blend/blend_position", float(animation_data["arm_value"])) - anim_tree.set("parameters/sitting/blend_amount", lerp(anim_tree.get("parameters/sitting/blend_amount"), float(animation_data["sitting"]), anim_lerp)) - anim_tree.set("parameters/thinking/blend_amount", lerp(anim_tree.get("parameters/thinking/blend_amount"), float(animation_data["busy"]), anim_lerp)) - anim_tree.set("parameters/land/blend_amount", lerp(anim_tree.get("parameters/land/blend_amount"), float(animation_data["land"]), anim_lerp)) - anim_tree.set("parameters/talking/blend_amount", lerp(anim_tree.get("parameters/talking/blend_amount"), float(animation_data["talking"]), anim_lerp)) - anim_tree.set("parameters/runmodif/scale", animation_data["run_mult"]) - anim_tree.set("parameters/walkmodif/scale", animation_data["walk_mult"]) - anim_tree.set("parameters/emote_time/scale", animation_data["emote_timescale"]) - anim_tree.set("parameters/emote_timeb/scale", animation_data["emote_timescale"]) - - face._show_blush(animation_data["drunk_tier"] > 1) - if animation_data["caught_item"] != caught_item: - caught_item = animation_data["caught_item"] - _update_caught_item(animation_data["caught_item"]) - - if item_scene: - item_scene.item_bend = lerp(item_scene.item_bend, animation_data["item_bend"], 0.4) - bobber_line.end_anchor = item_scene.get_node(item_scene.hotspot_node).global_transform.origin - bobber_line.y_drop = - 1.5 + ((animation_data["recent_reel"] / 15.0) * 0.5) - if animation_data["caught_fish"]: bobber_line.y_drop = 0 - if bobber.visible: bobber.global_transform.origin = animation_data["bobber_position"] + Vector3(0, (0.0 if not animation_data["caught_fish"] else - 0.3), 0) - bobber.visible = animation_data["bobber_visible"] - bobber_line.active = bobber.visible - - - if animation_data["player_scale"] != scale.y: - scale = animation_data["player_scale"] * Vector3.ONE - scale.y *= animation_data["player_scale_y"] - - bobber_line.line_scale = animation_data["player_scale"] - - ripples.visible = animation_data["state"] == STATES.FISHING - var rip_anim = "default" if animation_data["recent_reel"] <= 1.0 else "wake" - if ripples.animation != rip_anim: ripples.animation = rip_anim - - var root = anim_tree.tree_root - var node = root.get_node("emote_anim") - var node_b = root.get_node("emote_anim_b") - if node.animation != animation_data["emote"]: - anim_tree.set("parameters/emote_sync/seek_position", 0.0) - anim_tree.set("parameters/emote_half_sync/seek_position", 0.0) - node.set_animation(animation_data["emote"]) - node_b.set_animation(animation_data["emote"]) - - $"%head_alert".visible = animation_data["alert"] - - tail.sitting = animation_data["sitting"] - tail.diving = animation_data["diving"] - tail.motion = float(animation_data["moving"]) - tail.wagging = animation_data["wagging"] - - - if animation_data["back_bend"] != 0.0: - var pose = skeleton.get_bone_pose(1) - pose = pose.rotated(Vector3(1, 0, 0), animation_data["back_bend"]) - skeleton.set_bone_custom_pose(1, pose) - else : - skeleton.set_bone_custom_pose(1, Transform()) - -func _enter_animation(anim_name, loop = false, _locked = true, state_enter = - 1, full_anim = true, timescale = 1.0): - if anim_name == animation_data["emote"]: return - diving = false - - animation_data["emote_timescale"] = timescale - - - emote_full = full_anim - emote_locked = _locked - emote_looping = loop - buffer_state = state_enter if not emote_looping else - 1 - if emote_looping: _enter_state(state_enter) - animation_data["emote"] = anim_name - animation_timer = 0 - emoting = true - -func _exit_animation(): - emit_signal("_animation_finished") - emoting = false - emote_full = false - emote_locked = false - emote_looping = false - animation_data["emote"] = "" - -func _update_held_item(new): - if new.empty() or held_item == new: return - - held_item = new - var item_data = Globals.item_data[new["id"]]["file"] - var alt_scale_mult = 1.0 - - - item_sprite.texture = item_data.icon.duplicate() if (item_data.show_item and not item_data.show_scene) else null - - - - if is_instance_valid(item_scene): item_scene.queue_free() - item_scene = null - if item_data.show_scene and item_data.item_scene: - item_scene = item_data.item_scene.instance() - hand_bone.add_child(item_scene) - - var scale_mult = 0.07 - clamp(new["size"] * 0.01, 0.01, 0.061) - var item_scale = 1.0 if not item_data.uses_size else new["size"] * scale_mult - item_scale *= alt_scale_mult - item_sprite.scale = Vector3(item_scale, item_scale, item_scale) - - - var y = 0.0 - var offs = 0.0 - var back_bend = 0.0 - if item_data.uses_size: - y = clamp(item_scale * item_scale * 0.08, 0.0, 0.36) - offs = clamp(item_scale * item_scale, 0.0, 16.0) - back_bend = clamp(item_scale * item_scale, 0.0, 55.0) - - item_sprite.translation.y = y - item_sprite.offset.y = item_data.hold_offset + offs - - - $body / player_body / Armature / Skeleton / BoneAttachment / Spatial.rotation_degrees = Vector3(0.0, 0.0, back_bend * 0.7) - - item_sprite.translation.z = 0.0 - item_sprite.vel = 0.0 - item_sprite.mult = clamp(item_scale * 3.5, 0.0, 1.2) if item_data.category == "fish" and item_data.alive else 0.0 - - item_sprite.modulate = Color("#ffffff") - item_sprite.opacity = 1.0 - for child in item_sprite.get_children(): child.emitting = false - - if PlayerData.QUALITY_DATA.has(new["quality"]): - item_sprite.modulate = Color(PlayerData.QUALITY_DATA[new["quality"]]["mod"]) - item_sprite.opacity = PlayerData.QUALITY_DATA[new["quality"]]["op"] - if PlayerData.QUALITY_DATA[new["quality"]]["particle"] > - 1: item_sprite.get_child(PlayerData.QUALITY_DATA[new["quality"]]["particle"]).emitting = true - - animation_data["arm_value"] = item_data.arm_value - animation_data["back_bend"] = deg2rad( - back_bend) - -func _update_caught_item(new): - if new.empty(): - caught_fish.texture = null - return - caught_item = new - - var item_data = Globals.item_data[new["id"]]["file"] - caught_fish.texture = item_data.icon.duplicate() if item_data.show_item else null - - var item_scale = 1.0 if not item_data.uses_size else new["size"] * 0.01 - caught_fish.scale = Vector3(item_scale, item_scale, item_scale) - -func _bobber_cast(dist, end, splash): - bobber_control = false - var height = dist * 0.4 - height += max(0, global_transform.origin.y - end.y) - - end.y -= 1 - - bobber_hpos = global_transform.origin - bobber_vpos = global_transform.origin.y - yield (get_tree().create_timer(0.4), "timeout") - - var tween = get_tree().create_tween() - - tween.set_trans(1) - tween.set_ease(1) - tween.tween_property(self, "bobber_vpos", end.y + height, dist * 0.05) - tween.set_ease(0) - tween.tween_property(self, "bobber_vpos", end.y - (height * 0.2), dist * 0.05) - if splash: - tween.tween_callback(self, "_bobber_splash") - - - tween.set_trans(1) - tween.set_ease(1) - tween.tween_property(self, "bobber_vpos", end.y + (height * 0.1), (dist * 0.05)) - tween.set_ease(0) - tween.tween_property(self, "bobber_vpos", end.y - (height * 0.05), (dist * 0.1)) - - - tween.set_trans(1) - tween.set_ease(1) - tween.tween_property(self, "bobber_vpos", end.y, (dist * 0.1)) - - var htween = get_tree().create_tween() - htween.set_trans(4) - htween.set_ease(1) - htween.tween_property(self, "bobber_hpos", end + (global_transform.basis.z * 0.5), dist * 0.1) - htween.tween_property(self, "bobber_hpos", end + (global_transform.basis.z * 0.2), dist * 0.15) - htween.tween_property(self, "bobber_hpos", end, dist * 0.1) - - yield (htween, "finished") - bobber_control = true - -func _bobber_retract(): - bobber_control = false - bobber_hpos = bobber.global_transform.origin - bobber_vpos = bobber.global_transform.origin.y - - yield (get_tree().create_timer(0.4), "timeout") - - if retract_splash: _bobber_splash("splashb") - var tween = get_tree().create_tween() - tween.set_trans(1) - tween.set_ease(1) - tween.tween_property(self, "bobber_vpos", global_transform.origin.y + 2.0, 0.3) - tween.set_ease(0) - tween.tween_property(self, "bobber_vpos", global_transform.origin.y, 0.3) - - var htween = get_tree().create_tween() - htween.tween_property(self, "bobber_hpos", global_transform.origin, 0.6) - -func _bobber_splash(sfx = "splash"): - _sync_particle("splash", bobber.global_transform.origin, true) - _sync_sfx(sfx, Vector3(bobber_hpos.x, bobber_vpos, bobber_hpos.z)) - - - - - -func _toggle_sit(exit = false): - if not exit: sitting = not sitting - else : sitting = false - -func _play_emote(emote_id, emotion = ""): - if emote_id == "": return - - if emote_id == "sit": - _toggle_sit() - elif not emote_locked: - _enter_animation(emote_id, false, true) - if emotion != "": - _sync_face_emote(emotion) - - - - - -func _change_cosmetics(): - if cosmetic_data == PlayerData.cosmetics_equipped: return - var new = PlayerData.cosmetics_equipped.duplicate() - if not dead_actor: Network._send_actor_action(actor_id, "_update_cosmetics", [new]) - _update_cosmetics(new) - -func _update_cosmetics(data): - var valid = true - for key in PlayerData.FALLBACK_COSM.keys(): - if not data.keys().has(key): - print("missing key ", key) - valid = false - for key in data.keys(): - if not (data[key] is Array): - if not Globals._cosmetic_exists(data[key]): - print("cosm ", data[key], " does not exist") - valid = false - else : - for c in data[key]: - if not Globals._cosmetic_exists(c): - print("cosm ", data[key], " does not exist") - valid = false - - if not valid: data = PlayerData.FALLBACK_COSM.duplicate() - - cosmetic_data = data - - for child in skeleton.get_children(): - if child.is_in_group("cosmetic"): child.queue_free() - - if data.empty(): return - - title.label = str(Network._get_username_from_id(owner_id)) - title.title = Globals.cosmetic_data[data["title"]]["file"].title - - face._setup_face(data) - var species_id = Globals.cosmetic_data[data.species]["file"].cos_internal_id - var species = _create_cosmetic(data["species"], species_id) - _create_cosmetic(data["undershirt"], species_id) - _create_cosmetic(data["overshirt"], species_id) - _create_cosmetic(data["legs"], species_id) - _create_cosmetic(data["hat"], species_id) - for misc in data["accessory"]: _create_cosmetic(misc, species_id) - - - var pattern = Globals.cosmetic_data[data["pattern"]]["file"] - body_mesh.material_override.set_shader_param("texture_albedo", pattern.body_pattern[0]) - - var primary_color = Globals.cosmetic_data[data["primary_color"]]["file"] - var secondary_color = Globals.cosmetic_data[data["secondary_color"]]["file"] if pattern.body_pattern[0] else Globals.cosmetic_data[data["primary_color"]]["file"] - body_mesh.material_override.set_shader_param("albedo", primary_color.main_color) - body_mesh.material_override.set_shader_param("albedo_secondary", secondary_color.main_color) - - - var tail_data = Globals.cosmetic_data[data["tail"]]["file"] - tail._load_tail(tail_data.mesh, primary_color.main_color) - - if species: - species.material_override.set_shader_param("albedo", primary_color.main_color) - species.material_override.set_shader_param("albedo_secondary", secondary_color.main_color) - - match data["species"]: - "species_cat": species.material_override.set_shader_param("texture_albedo", pattern.body_pattern[1]) - "species_dog": species.material_override.set_shader_param("texture_albedo", pattern.body_pattern[2]) - - if not data.keys().has("bobber") or data["bobber"] == "": data["bobber"] = "bobber_default" - var bobber_file = Globals.cosmetic_data[data["bobber"]]["file"] - bobber_mesh.mesh = bobber_file.mesh - bobber_mesh.set_surface_material(0, bobber_file.material) - if bobber_file.secondary_material: bobber_mesh.set_surface_material(1, bobber_file.secondary_material) - if bobber_file.third_material: bobber_mesh.set_surface_material(2, bobber_file.third_material) - -func _create_cosmetic(id, species_id = 0): - if id == "": return - if not Globals.cosmetic_data.keys().has(id): - print("Failed finding cosmetic: ", id) - return - print("Creating Cosmetic: ", id) - var data = Globals.cosmetic_data[id]["file"] - - if data.scene_replace: - var bone_attach = BoneAttachment.new() - skeleton.add_child(bone_attach) - bone_attach.bone_name = "pelvis" - bone_attach.add_to_group("cosmetic") - - var cosm = data.scene_replace.instance() - bone_attach.add_child(cosm) - return cosm - - var cosm = preload("res://Scenes/Entities/Player/cosmetic_node.tscn").instance() - cosm.mesh = data.mesh - if data.species_alt_mesh.size() > 0: - cosm.mesh = data.species_alt_mesh[species_id] - - cosm.skin = data.mesh_skin - - if not data.material: - cosm.material_override = preload("res://Assets/Shaders/player_skins.tres").duplicate() - cosm.material_override.set_shader_param("albedo", data.main_color) - cosm.material_override.set_shader_param("albedo_secondary", data.main_color) - cosm.material_override.set_shader_param("texture_albedo", null) - else : - - cosm.material_override = null - cosm.set_surface_material(0, data.material) - if data.secondary_material: cosm.set_surface_material(1, data.secondary_material) - - cosm.skeleton = skeleton.get_path() - skeleton.add_child(cosm) - return cosm - - - - - - - - - - - - - - - - - - -func _on_step_timer_timeout(): - if not controlled: return - - - if randf() < 0.03 * drunk_tier and drunk_wander_length <= 0: - drunk_wander_length = randi() % 25 + 5 - drunk_wander_dir = Vector3( - rand_range( - 1, 1), - 0.0, - rand_range( - 1, 1)) - - if state == STATES.FISHING_STRUGGLE: - _sync_particle("small_splash", bobber.global_transform.origin, true) - - if recent_reel > 0 and state == STATES.FISHING: - fishing_update.translation.z = - rod_cast_dist - fishing_update.force_raycast_update() - if fishing_update.is_colliding() and not fishing_update.get_collider().is_in_group("valid_water"): - _enter_state(STATES.FISHING_CANCEL) - - if velocity != Vector3.ZERO: - var vol = 1.6 - if sprinting: vol = 2.8 - elif sneaking: vol = 0.3 - - if is_on_floor() and sprinting: _sync_particle("dust_run", Vector3(0, - 0.95, 0)) - - if is_on_floor() and safe_check.is_colliding() and safe_check.get_collision_normal() == Vector3(0, 1, 0) and not gravity_disable: - if death_counter > 0: death_counter -= 1 - last_valid_pos = global_transform.origin - - PlayerData.player_saved_position = global_transform.origin - - - - - -func _message_sent(text): - - var split = text.split(" ") - for s in split: - match s: - ":(", ":[", "D:", ";_;", ";~;", ":C", ":c": _sync_face_emote("sad") - ":)", ":D", ":]": _sync_face_emote("love") - "xD", "!!!", "<3", "xd", "LMAO", "LOL": _sync_face_emote("happy") - ">:(", "D:<", ">:[": _sync_face_emote("angry") - ":/", ":|": _sync_face_emote("flat") - ":3", ":3c", ">:3c", ">:3": _sync_face_emote("cat") - "O.o", "!?!?", "?!?!", ":o", ":O": _sync_face_emote("surprised") - - var bubble = title._create_speech_bubble(text, PlayerData.voice_speed) - bubble.connect("_letter_said", self, "_sync_talk") - Network._send_actor_action(actor_id, "_sync_create_bubble", [text], false) - -func _sync_talk(letter): - var blacklist = ["a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l", "m", "n", "o", "p", "q", "r", "s", "t", "u", "v", "w", "x", "y", "z"] - if not blacklist.has(letter.to_lower()): return - - animation_data["talking"] = 0.4 - _talk(letter, PlayerData.voice_pitch) - Network._send_actor_action(actor_id, "_talk", [letter, PlayerData.voice_pitch], false) - -func _talk(letter, pitch = 1.5): - if not in_zone or not visible: return - - face._talk() - sound_manager._construct_voice(letter, "NewVoice", pitch) - -func _sync_face_emote(emotion): - _face_emote(emotion) - Network._send_actor_action(actor_id, "_face_emote", [emotion], false) - -func _sync_create_bubble(text): - title._create_speech_bubble(text) - -func _sync_level_bubble(text): - title._create_level_bubble(text) - $emotion_particles / lvl_particles.restart() - $emotion_particles / lvl_particles2.restart() - -func _face_emote(emotion): - face._emote(emotion, 2.4) - - - - - -func _sync_particle(id, offset = Vector3.ZERO, global = false): - _play_particle(id, offset, global) - Network._send_actor_action(actor_id, "_play_particle", [id, offset, global], false) - -func _play_particle(id, offset, global): - if not PARTICLE_DATA.keys().has(id): return - var p = PARTICLE_DATA[id].instance() - add_child(p) - - if not global: - p.translation = offset - else : - p.set_as_toplevel(true) - p.global_transform.origin = offset - - p.emitting = true - yield (get_tree().create_timer(p.lifetime + 0.1), "timeout") - p.queue_free() - - - - - -func _on_water_detect_area_entered(area): - if gravity_disable: return - if area.is_in_group("water"): - _kill() - -func _kill(skip_anim = false): - if not controlled: return - - death_counter += 1 - if death_counter >= 10: - if not skip_anim: PlayerData._send_notification("too many deaths in a row! sending to spawn...", 1) - else : PlayerData._send_notification("returning to spawn", 1) - - world._enter_zone("main_zone", - 1) - PlayerData.player_saved_zone = "main_zone" - PlayerData.player_saved_zone_owner = - 1 - last_valid_pos = world.map.spawn_position.global_transform.origin - death_counter = 0 - - if state == STATES.FISHING or state == STATES.FISHING_CAST or state == STATES.FISHING_CAST: - _enter_state(STATES.FISHING_CANCEL) - - cam_push = 0.0 - gravity_disable = true - - if not skip_anim: - _enter_animation("drown", true, true) - _sync_sfx("drown") - _sync_particle("splash", global_transform.origin, true) - - SceneTransition._fake_scene_change() - yield (SceneTransition, "_finished") - global_transform.origin = last_valid_pos + Vector3(0, 0.5, 0) - yield (get_tree().create_timer(0.3), "timeout") - gravity_disable = false - _enter_state(0) - _exit_animation() - - - - - -func _consume_item(id): - if state != STATES.DEFAULT: return - _enter_state(STATES.EMOTING) - - consume_on_state_change = held_item["ref"] - _enter_animation("drink", false, true, STATES.CONSUME_ITEM, true) - - var sfx = "drink" - - match id: - "growth": player_scale = clamp(player_scale + 0.4, 0.7, 100) - "shrink": player_scale = clamp(player_scale - 0.1, 0.1, 100) - "revert": - player_scale = 1.0 - drunk_timer = max(drunk_timer - 9000, 0) - sfx = "drink_nocap" - "speed": - boost_timer = 18000 - boost_amt = 1.3 - sfx = "drink_nocap" - "speed_burst": - boost_timer = 900 - boost_amt = 4.0 - sfx = "drink_nocap" - "catch": - catch_drink_timer = 18000 - catch_drink_boost = 1.15 - catch_drink_reel = 1.25 - catch_drink_xp = 1.0 - catch_drink_tier = 1 - catch_drink_gold_add = Vector2(1, 10) - catch_drink_gold_percent = 0.0 - "catch_big": - catch_drink_timer = 18000 - catch_drink_boost = 1.3 - catch_drink_reel = 1.45 - catch_drink_xp = 1.0 - catch_drink_tier = 2 - catch_drink_gold_add = Vector2(10, 50) - catch_drink_gold_percent = 0.25 - "catch_deluxe": - catch_drink_timer = 18000 - catch_drink_boost = 1.3 - catch_drink_reel = 1.45 - catch_drink_xp = 1.25 - catch_drink_tier = 3 - catch_drink_gold_add = Vector2(0, 0) - catch_drink_gold_percent = 0.0 - "beer": - drunk_timer += 9000 - drunk_timer = clamp(drunk_timer, 0, 50000) - "beer_big": - drunk_timer += 42000 - drunk_timer = clamp(drunk_timer, 0, 50000) - sfx = "drink_nocap" - - _sync_sfx(sfx) - - - -func _open_chest(rare = false): - if state != STATES.DEFAULT: return - - consume_on_state_change = held_item["ref"] - _enter_state(STATES.CONSUME_ITEM) - - var cosm = PlayerData._get_unowned_cosmetic() - if cosm != null: PlayerData._unlock_cosmetic(cosm) - else : - PlayerData._send_notification("You found 100 dollars inside the chest!") - PlayerData.money += 100 - -func _open_ringbox(): - if state != STATES.DEFAULT: return - - consume_on_state_change = held_item["ref"] - _enter_state(STATES.CONSUME_ITEM) - - if not PlayerData.cosmetics_unlocked.has("accessory_ring"): PlayerData._unlock_cosmetic("accessory_ring") - else : - PlayerData._send_notification("You already have a ring unlocked... Obtained $1000 instead.") - PlayerData.money += 1000 - - - -func _scratch_off(type): - _enter_state(STATES.GUITAR) - hud._open_minigame("scratch_off", {"type": type}) - var _win = yield (hud, "_minigame_finished") - yield (get_tree().create_timer(0.15), "timeout") - _enter_state(STATES.DEFAULT) - - consume_on_state_change = held_item["ref"] - _enter_state(STATES.CONSUME_ITEM) - - if not _win: - _sync_face_emote("angry") - _sync_sfx("rip") - else : - _sync_face_emote("happy") - - - - - -func _mushroom_bounce(): - if not controlled: return - ignore_snap = 10 - var bounce_horz = 16.0 if direction != Vector3.ZERO else 0.0 - var bounce_vert = 32.0 - gravity_vec = Vector3.ZERO - gravity_vec.x += (direction.normalized() * bounce_horz).x - gravity_vec.z += (direction.normalized() * bounce_horz).z - gravity_vec.y += bounce_vert - animation_data["mushroom"] = true - - - - - -func _sync_sfx(id, position = null, pitch = 1.0, delay = 0.0): - if not controlled: return - if delay > 0.0: yield (get_tree().create_timer(delay), "timeout") - - _play_sfx(id, position, pitch) - Network._send_actor_action(actor_id, "_play_sfx", [id, position, pitch], false) - -func _play_sfx(id, position = null, pitch = 1.0): - if not in_zone or not visible: return - sound_manager._play_sound(id, position, pitch) - print("playing sfx ", id) - -func _process_sounds(): - - - - - if $sound_manager / dive_scrape.playing != animation_data["dive_scrape"]: $sound_manager / dive_scrape.playing = animation_data["dive_scrape"] and in_zone - if $sound_manager / reel_slow.stream_paused == animation_data["reel_slow"]: $sound_manager / reel_slow.stream_paused = not animation_data["reel_slow"] and in_zone - if $sound_manager / reel_fast.playing != animation_data["reel_fast"]: $sound_manager / reel_fast.playing = animation_data["reel_fast"] and in_zone - - - - - -func _on_rain_timer_timeout(): - if not controlled: return - - var rain = false - for child in $raincloud_check.get_overlapping_areas(): - if child.is_in_group("rain_cloud"): - rain = true - - if rain != in_rain: - in_rain = rain - PlayerData.emit_signal("_rain_toggle", in_rain) - - if not in_rain: Network.set_rich_presence("#default") - else : Network.set_rich_presence("#rain") - - - - - -func _wag(): - animation_data["wagging"] = not animation_data["wagging"] - -func _paint(size, color): - if not controlled: return - - var in_zone = false - for area in $paint_node / Area.get_overlapping_areas(): - if area.is_in_group("canvas"): in_zone = true - if not in_zone: PlayerData._send_notification("mouse must be on a drawing zone!", 1) - - $paint_node.size = size - $paint_node.color = color - $paint_node.drawing = true - -func _paint_stop(): - if not controlled: return - $paint_node.drawing = false - PlayerData.emit_signal("_chalk_send") - -func _metal_detect_begin(): - pass - -func _metaldetect_update(): - if not controlled or held_item.id == "": - return - - var idata = Globals.item_data[held_item["id"]]["file"] - if not idata.detect_item: - return - - var alert_level = 0 - for b in $detection_zones / metal_detect_far.get_overlapping_bodies(): - if b.is_in_group("metal_spawn"): - if abs(b.global_transform.origin.y - global_transform.origin.y) > 3.5: continue - alert_level = b.global_transform.origin.distance_to(global_transform.origin) - - metal_detect_alert_level = alert_level - - if alert_level > 0: _metal_detect_beep() - -func _metal_detect_beep(): - var new_cd = ceil(max(metal_detect_alert_level * 0.5, 1)) - - if abs(new_cd - metal_detect_alert_cd) > 4: metal_detect_alert_cd = 0 - - metal_detect_alert_cd -= 1 - if metal_detect_alert_cd > 0: return - - metal_detect_alert_cd = new_cd - - var pitch = max(6.0 - metal_detect_alert_level * 0.3, 0.5) - - $metaldetect_dot.modulate.a = 1.0 - if metal_detect_flop: _sync_sfx("md_beep_slowb", null, pitch) - else : _sync_sfx("md_beep_slow", null, pitch) - metal_detect_flop = not metal_detect_flop - -func _on_metal_detect_consume_body_entered(b): - if not controlled or held_item.id == "": return - - var idata = Globals.item_data[held_item["id"]]["file"] - if not idata.detect_item: return - - if b.is_in_group("metal_spawn"): - b._reveal() - -func _on_image_update_timeout(): - if not controlled or dead_actor: return - _update_held_item(held_item) - Network._send_actor_action(actor_id, "_update_held_item", [held_item], false) - -func _real_step(run = false): - if anim_tree.get("parameters/dive/blend_amount") > 0.2 or anim_tree.get("parameters/movement/blend_amount") < 0.8: return - if not is_on_floor(): return - var sfx = "step" if not run else "step_run" - if boost_amt > 1.0 and boost_timer > 1: sfx = "step_fastrun" - _sync_sfx(sfx) - - - -func _play_guitar(): - _enter_state(STATES.GUITAR) - hud._open_minigame("guitar") - var _win = yield (hud, "_minigame_finished") - yield (get_tree().create_timer(0.15), "timeout") - _enter_state(STATES.DEFAULT) - -func _strum_guitar(string, fret, volume): - _sync_strum(string, fret, volume) - Network._send_actor_action(actor_id, "_sync_strum", [string, fret, volume]) - - animation_data["land"] = clamp(animation_data["land"] + 0.06, 0.0, 0.3) - _sync_face_emote("strum") - if randf() < 0.05: - _sync_particle("music") - -func _sync_strum(string, fret, volume): - if not in_zone or not visible: return - $guitar_sounds.get_child(string)._play_fret(fret, false, volume) - -func _hammer_string(string, fret): - _sync_hammer(string, fret) - Network._send_actor_action(actor_id, "_sync_hammer", [string, fret]) - -func _sync_hammer(string, fret): - if not in_zone or not visible: return - $guitar_sounds.get_child(string)._hammer_fret(fret) - -func _bark(): - if not controlled: return - var bark_id = "bark_dog" - bark_id = { - "species_cat": ["bark_cat", "growl_cat", "whine_cat"], - "species_dog": ["bark_dog", "growl_dog", "whine_dog"], - }[PlayerData.cosmetics_equipped.species] - - var type = 0 - if Input.is_action_pressed("move_sneak"): - _sync_face_emote("growl") - type = 1 - elif Input.is_action_pressed("move_walk"): - _sync_face_emote("whine") - type = 2 - else : - _sync_face_emote("bark") - - bark_id = bark_id[type] - _sync_sfx(bark_id) - -func _kiss(): - if not controlled: return - - animation_data["land"] = animation_data["land"] + 0.2 - _sync_face_emote("kiss") - _sync_sfx("kiss") - _sync_particle("kiss") - -func _punch(type = 0): - if not controlled or item_cooldown > 0: return - item_cooldown = 30 - - animation_data["land"] = animation_data["land"] + 1.0 - for b in $detection_zones / punch.get_overlapping_bodies(): - if b.is_in_group("player") and b != self and not b.controlled: - Network._send_P2P_Packet({"type": "player_punch", "from": global_transform.origin, "player": owner_id, "punch_type": type}, str(b.owner_id), 2) - - _sync_face_emote("punch") - _sync_sfx("punch") - _sync_punch() - Network._send_actor_action(actor_id, "_sync_punch") - -func _sync_punch(): - $emotion_particles / punch_particles.restart() - $emotion_particles / punchb_particles.restart() - -func _punched(from, type): - if not controlled: return - - if OptionsMenu.punchable: return - - var dir = (global_transform.origin - from).normalized() - - ignore_snap = 10 - var bounce_horz = 4.0 - var bounce_vert = 8.0 - - match type: - 0: - bounce_horz = 4.0 - bounce_vert = 8.0 - 1: - bounce_horz = 12.0 - bounce_vert = 24.0 - - gravity_vec = Vector3.ZERO - gravity_vec.x += (dir.normalized() * bounce_horz).x - gravity_vec.z += (dir.normalized() * bounce_horz).z - gravity_vec.y += bounce_vert - - _sync_face_emote("angry") - -func _tambourine(): - _sync_sfx("tambourine") - - animation_data["land"] = clamp(animation_data["land"] + 0.14, 0.0, 0.3) - _sync_face_emote("strum") - if randf() < 0.25: - _sync_particle("music") - -func _on_cosmetic_refresh_timeout(): - if not controlled: return - _refresh_cosmetics() - -func _refresh_cosmetics(): - if not controlled: return - yield (get_tree().create_timer(1.0), "timeout") - var new = PlayerData.cosmetics_equipped.duplicate() - if not dead_actor: Network._send_actor_action(actor_id, "_update_cosmetics", [new]) - -func _return_to_spawn(): - if not controlled: return - death_counter = 11 - _kill(true) - -func _item_removal(ref): - print("Sold Item") - if held_item.keys().has("ref") and held_item["ref"] == ref: _equip_item(PlayerData.FALLBACK_ITEM.duplicate()) diff --git a/Scenes/Entities/Player/player.gd b/Scenes/Entities/Player/player.gd deleted file mode 100644 index dd82d68..0000000 --- a/Scenes/Entities/Player/player.gd +++ /dev/null @@ -1,2343 +0,0 @@ -extends Actor - -enum STATES{DEFAULT, BUSY, OBTAIN, EMOTING, FREECAM, FISHING_CHARGE, FISHING_CAST, FISHING, FISHING_CANCEL, FISHING_STRUGGLE, SHOVEL_CAST, SHOVEL_STRUGGLE, SHOVEL_CANCEL, \ -NET_CAST, NET_STRUGGLE, NET_CANCEL, SHOWCASE, CONSUME_ITEM, METAL_DETECTOR, GUITAR, GAMBLING} - -const GRAVITY = 32.0 -const BAIT_DATA = { - "": {"catch": 0.0, "max_tier": 0, "quality": []}, - - "worms": {"catch": 0.06, "max_tier": 1, "quality": [1.0]}, - "cricket": {"catch": 0.06, "max_tier": 2, "quality": [1.0, 0.05]}, - "leech": {"catch": 0.06, "max_tier": 2, "quality": [1.0, 0.15, 0.05]}, - "minnow": {"catch": 0.06, "max_tier": 2, "quality": [1.0, 0.5, 0.25, 0.05]}, - "squid": {"catch": 0.06, "max_tier": 2, "quality": [1.0, 0.8, 0.45, 0.15, 0.05]}, - "nautilus": {"catch": 0.06, "max_tier": 2, "quality": [1.0, 0.98, 0.75, 0.55, 0.25, 0.05]}, -} -const PARTICLE_DATA = { - "dust_run": preload("res://Scenes/Particles/dust_running.tscn"), - "dust_land": preload("res://Scenes/Particles/dust_land.tscn"), - "splash": preload("res://Scenes/Particles/water_splash.tscn"), - "small_splash": preload("res://Scenes/Particles/small_splash.tscn"), - "music": preload("res://Scenes/Particles/music_particle.tscn"), - "kiss": preload("res://Scenes/Particles/kiss.tscn"), -} - -export (NodePath) var hand_sprite_node -export (NodePath) var hand_bone_node -export var NPC_body = false -export var NPC_cosmetics = {"species": "species_cat", "pattern": "pattern_none", "primary_color": "pcolor_white", "secondary_color": "scolor_tan", "hat": "hat_none", "undershirt": "shirt_none", "overshirt": "overshirt_none", "title": "title_rank_1", "bobber": "bobber_default", "eye": "eye_halfclosed", "nose": "nose_cat", "mouth": "mouth_default", "accessory": [], "tail": "tail_cat"} -export var NPC_name = "NPC Test" -export var NPC_title = "npc title here" - -var camera_zoom = 5.0 - -var direction = Vector3() -var gravity_vec = Vector3() -var dive_vec = Vector3() -var velocity = Vector3() -var player_scale = 1.0 -var player_scale_y = 1.0 - -var cam_orbit_node = null -var cam_follow_node = null -var cam_push = 0.0 -var cam_push_cur = 0.0 -var cam_return = 1.0 -var force_cam_look = false -var mouse_world_pos = Vector3() -var mouse_look = false - -var state = STATES.DEFAULT -var walk_speed = 3.2 -var slow_walk_speed = 1.0 -var sprint_speed = 6.44 -var sneak_speed = 1.3 -var jump_height = 6.0 -var jump_leap = 0.0 -var dive_distance = 9.0 -var accel = 64.0 -var sprinting = false -var slow_walking = false -var sneaking = false -var diving = false -var request_jump = false -var ignore_snap = 0 -var snapped = false -var sitting = false -var locked = false -var busy = false -var in_air = false -var gravity_disable = false -var hud -var int_text = "" -var interact_cooldown = 60 -var xp_buildup = 0 -var showcase_ref - -var held_item = PlayerData.FALLBACK_ITEM -var caught_item = PlayerData.FALLBACK_ITEM -var held_item_weight = 1.0 -var previous_item = 0 -var primary_hold_timer = 0 -var fish_zone_data = {} -var rod_cast_data = "" -var casted_bait = "" -var rod_cast_dist = 0.0 -var rod_depth = 0 -var failed_casts = 0 -var recent_reel = 0.0 -var item_scene = null -var bobber_hpos = Vector3() -var bobber_vpos = 0 -var bobber_control = true -var last_valid_pos = Vector3() -var retract_splash = false -var show_local = false -var bait_warn = 2 -var in_rain = false -var consume_on_state_change = - 1 -var item_cooldown = 0 - -var rod_damage = 1.0 -var rod_spd = 1.0 -var rod_chance = 0.0 - - - -var boost_timer = 0 -var boost_amt = 1.3 -var boost_mult = 1.0 -var catch_drink_timer = 0 -var catch_drink_boost = 1.0 -var catch_drink_reel = 1.0 -var catch_drink_xp = 1.0 -var catch_drink_gold_add = Vector2(0, 0) -var catch_drink_gold_percent = 0.0 -var catch_drink_tier = 0 -var drunk_timer = 0 -var drunk_tier = 0 -var drunk_wander_length = 0 -var drunk_wander_dir = 0 - -var death_counter = 0 -var metal_detect_flop = false -var metal_detect_alert_level = 0 -var metal_detect_alert_cd = 0 - -var emoting = false -var emote_full = false -var emote_locked = false -var emote_looping = false -var animation_timer = 0 -var animation_goal = 99 -var buffer_state = - 1 -var animation_data = { - "moving": false, - "sprinting": false, - "sneaking": false, - "emoting": false, - "emote_full": false, - "diving": false, - "sitting": false, - "alert": false, - "emote": "", - "arm_state": "", - "caught_item": {}, - "arm_value": 0.0, - "item_bend": 0.0, - "busy": false, - "land": 0.0, - "talking": 0.0, - "recent_reel": 0.0, - "bobber_position": Vector3(), - "bobber_visible": false, - "caught_fish": false, - "player_scale": 1.0, - "player_scale_y": 1.0, - "mushroom": false, - "run_mult": 1.0, - "walk_mult": 1.25, - "drunk_tier": 0, - "wagging": false, - "emote_timescale": 1.0, - "back_bend": 0.0, - "dive_scrape": false, - "reel_slow": false, - "reel_fast": false, - "state": state, -} - - - -var custom_held_item = "" - -var cosmetic_data = {} -var selected_build_object -var old_rot = Vector3() -var rot_diff = 0.0 - -var prop_ids = [] -var cam_move = false -var freecamming = false - -onready var cam_base = $cam_base -onready var cam_pivot = $cam_base / cam_pivot -onready var camera = $Camera -onready var camera_point = $cam_base / cam_pivot / SpringArm / camera_point -onready var cam_arm = $cam_base / cam_pivot / SpringArm -onready var body = $body -onready var body_mesh = $body / player_body / Armature / Skeleton / body_main -onready var rot_help = $rot_help -onready var interact_range = $interact_range -onready var anim_tree = $body / AnimationTree -onready var skeleton = $body / player_body / Armature / Skeleton -onready var title = $Viewport / player_label -onready var item_sprite = get_node(hand_sprite_node) -onready var hand_bone = get_node(hand_bone_node) -onready var sound_emit = $sound_emit -onready var face = $body / player_body / Armature / Skeleton / face / player_face -onready var tail = $body / player_body / Armature / Skeleton / tail / holder / tail -onready var freecam_anchor = $camera_freecam_anchor -onready var sound_manager = $sound_manager - -onready var fish_detect = $detection_zones / fishing_detect -onready var fishing_update = $detection_zones / fishing_update -onready var fishing_area = $detection_zones / fishing_detect / fishing_area -onready var fish_timer = $fish_catch_timer -onready var bobber_preview = $bobber_preview -onready var bobber = $bobber -onready var bobber_mesh = $bobber / bobber_mesh -onready var bobber_line = $bobber / line -onready var ripples = $bobber / ripples -onready var caught_fish = $bobber / caught_item - -onready var shovel_area = $detection_zones / shovel_detect -onready var net_area = $detection_zones / net_detect - -onready var safe_check = $safe_check - -onready var lvlparticle = $emotion_particles / lvl_particles -onready var lvlparticleb = $emotion_particles / lvl_particles2 - -signal _animation_finished -signal _primary_release -signal _state_change -signal _menu_closed - - - - - -func _ready(): - add_to_group("player") - rot_help.set_as_toplevel(true) - cam_base.set_as_toplevel(true) - camera.set_as_toplevel(true) - freecam_anchor.set_as_toplevel(true) - anim_tree.tree_root = anim_tree.tree_root.duplicate(true) - _update_cosmetics(cosmetic_data) - title.visible = not dead_actor - - if NPC_body: - camera.queue_free() - remove_from_group("player") - - yield (get_tree(), "idle_frame") - var new = NPC_cosmetics - Network._send_actor_action(actor_id, "_update_cosmetics", [new]) - _update_cosmetics(new) - - yield (get_tree(), "idle_frame") - title.visible = true - title.label = NPC_name - title.title = NPC_title - - -func _setup_controlled(): - if NPC_body: return - - add_to_group("controlled_player") - camera = $Camera - camera.current = true - hud = load("res://Scenes/HUD/playerhud.tscn").instance() - hud.player = self - hud.connect("_player_sit", self, "_toggle_sit") - hud.connect("_play_emote", self, "_play_emote") - hud.connect("_menu_entered", self, "_hud_menu_entered") - hud.connect("_message_sent", self, "_message_sent") - hud.connect("_freecam_toggle", self, "_toggle_freecam") - get_tree().get_root().add_child(hud) - - PlayerData.connect("_clear_all_props", self, "_clear_all_props") - PlayerData.connect("_place_prop", self, "_create_prop") - PlayerData.connect("_play_sfx", self, "_play_sfx") - PlayerData.connect("_play_guitar", self, "_strum_guitar") - PlayerData.connect("_hammer_guitar", self, "_hammer_string") - PlayerData.connect("_return_to_spawn", self, "_return_to_spawn") - PlayerData.connect("_punched", self, "_punched") - PlayerData.connect("_item_removal", self, "_item_removal") - var delayed_update = get_tree().create_timer(1.0).connect("timeout", self, "_change_cosmetics") - - PlayerData.connect("_wag_toggle", self, "_wag") - PlayerData.connect("_kiss", self, "_kiss") - Network.connect("_user_connected", self, "_refresh_cosmetics") - - -func _setup_not_controlled(): - camera = $Camera - camera.queue_free() - - $bobber_preview.visible = false - $local_range.visible = false - $detection_zones / metal_detect_consume.monitoring = false - $detection_zones / metal_detect_consume.monitorable = false - - - $detection_zones / prop_detect.monitoring = false - $detection_zones / metal_detect_far.monitoring = false - $detection_zones / metal_detect.monitoring = false - $detection_zones / metal_detect_close.monitoring = false - $detection_zones / metal_detect_veryclose.monitoring = false - $detection_zones / punch.monitoring = false - $interact_range.monitoring = false - $water_detect.monitoring = false - $raincloud_check.monitoring = false - - - - - -func _physics_process(delta): - _process_animation() - _process_sounds() - - if ( not in_zone != $CollisionShape.disabled): - $CollisionShape.disabled = not in_zone - -func _process(delta): - if controlled: $paint_node._paint_process(delta) - -func _controlled_process(delta): - _get_input() - _process_movement(delta) - _process_timers() - _interact_check() - _update_animation_data() - _camera_update() - - _freecam_input(delta) - - body.visible = camera_zoom > 0.5 or cam_orbit_node != null - current_zone = world.active_zone - current_zone_owner = world.active_zone_owner - - fish_detect.translation.z = - rod_cast_dist - bobber_preview.translation.z = - clamp(primary_hold_timer * 0.06, 1.5, 9.0) - if bobber_preview.is_colliding(): $"%bobber_prev_mesh".global_transform.origin = bobber_preview.get_collision_point() + Vector3(0, 0.05, 0) - bobber_preview.visible = state == STATES.FISHING_CHARGE - - if recent_reel > 0: recent_reel -= 1 - if interact_cooldown > 0: interact_cooldown -= 1 - - animation_data["reel_slow"] = recent_reel > 8 - animation_data["reel_fast"] = state == STATES.FISHING_STRUGGLE - - - if (packet_send_interval == - 1 or Engine.get_physics_frames() % packet_send_interval == 0): - Network._send_actor_animation_update(actor_id, animation_data) - - Network.MESSAGE_ORIGIN = global_transform.origin - $local_range.visible = show_local - - $paint_node.global_transform.origin = mouse_world_pos - $metaldetect_dot.modulate.a = lerp($metaldetect_dot.modulate.a, 0.0, 0.1) - - -func _camera_update(): - cam_push_cur = lerp(cam_push_cur, cam_push * rod_cast_dist, 0.2) - var push = global_transform.basis.z * cam_push_cur - camera_zoom = clamp(camera_zoom, 0.0, 20.0) - - var cam_zoom = camera_zoom - var cam_zoom_lerp = 0.4 - var cam_follow_point = true - var cam_follow_pos = Vector3() - var cam_follow_rot = Vector3() - var sit_add = Vector3(0, - 0.2, 0.0) if sitting else Vector3.ZERO - var cam_base_pos = global_transform.origin + push + sit_add - - var desired_fov = 50 - if sprinting and direction != Vector3.ZERO: - desired_fov += 2 - if boost_timer > 0: desired_fov += 2 * boost_amt - if animation_data["mushroom"]: desired_fov += 10 - - camera.fov = lerp(camera.fov, desired_fov, 0.2) - - if is_instance_valid(cam_orbit_node) and cam_orbit_node.is_visible_in_tree(): - cam_base_pos = cam_orbit_node.global_transform.origin - - if is_instance_valid(cam_follow_node) and cam_follow_node.is_visible_in_tree(): - cam_follow_point = false - cam_follow_pos = cam_follow_node.global_transform.origin - cam_follow_rot = cam_follow_node.global_rotation - - var cam_speed = 0.08 - if cam_follow_point: - cam_return = lerp(cam_return, 1.0, cam_speed * 0.5) - camera.global_transform.origin = lerp(camera.global_transform.origin, camera_point.global_transform.origin, cam_return) - camera.rotation.x = lerp_angle(camera.rotation.x, camera_point.global_rotation.x, cam_return) - camera.rotation.y = lerp_angle(camera.rotation.y, camera_point.global_rotation.y, cam_return) - camera.rotation.z = lerp_angle(camera.rotation.z, camera_point.global_rotation.z, cam_return) - else : - cam_return = 0.0 - camera.global_transform.origin = lerp(camera.global_transform.origin, cam_follow_pos, cam_speed) - camera.rotation.x = lerp_angle(camera.rotation.x, cam_follow_rot.x, cam_speed) - camera.rotation.y = lerp_angle(camera.rotation.y, cam_follow_rot.y, cam_speed) - camera.rotation.z = lerp_angle(camera.rotation.z, cam_follow_rot.z, cam_speed) - - cam_base.global_transform.origin = cam_base_pos - cam_arm.spring_length = lerp(cam_arm.spring_length, cam_zoom, cam_zoom_lerp) - -func _interact_check(): - if not controlled: return - var in_range = false - for area in interact_range.get_overlapping_areas(): - if area.is_in_group("interactable") and area.is_visible_in_tree(): - int_text = area.text - in_range = true - break - if hud: - hud.interact = in_range - hud.int_text = int_text - -func _hud_menu_entered(menu): - if menu == 0: - if not freecamming: - cam_orbit_node = null - cam_follow_node = null - force_cam_look = false - _exit_showcase() - emit_signal("_menu_closed") - -func _process_timers(): - if catch_drink_timer > 0: - catch_drink_timer -= 1 - if catch_drink_timer <= 0: - catch_drink_boost = 1.0 - catch_drink_reel = 1.0 - catch_drink_xp = 1.0 - catch_drink_gold_add = Vector2(0, 0) - catch_drink_gold_percent = 0.0 - - drunk_tier = 0 - if drunk_timer > 0: - drunk_timer -= 1 - if drunk_timer >= 19000: drunk_tier = 3 - elif drunk_timer >= 10000: drunk_tier = 2 - elif drunk_timer > 0: drunk_tier = 1 - else : drunk_tier = 0 - - if item_cooldown > 0: item_cooldown -= 1 - - - - - -func _get_input(): - direction = Vector3.ZERO - - if Input.is_action_just_released("primary_action"): _primary_action_release() - if Input.is_action_pressed("primary_action"): _primary_action_hold() - else : primary_hold_timer = 0 - - if busy: - Input.set_mouse_mode(Input.MOUSE_MODE_VISIBLE) - return - - if Input.is_action_pressed("secondary_action") or camera_zoom <= 0.0: - if Input.mouse_mode != Input.MOUSE_MODE_CAPTURED: - PlayerData.original_mouse_position = get_tree().get_nodes_in_group("world_viewport")[0].get_mouse_position() - Input.set_mouse_mode(Input.MOUSE_MODE_CAPTURED) - else : - if Input.mouse_mode != Input.MOUSE_MODE_VISIBLE: - Input.set_mouse_mode(Input.MOUSE_MODE_VISIBLE) - Input.warp_mouse_position(PlayerData.original_mouse_position) - - if Input.is_action_just_released("zoom_in"): camera_zoom -= 0.5 - if Input.is_action_just_released("zoom_out"): camera_zoom += 0.5 - - if Input.is_action_just_pressed("interact"): _interact() - - if Input.is_action_just_pressed("bind_1"): _equip_hotbar(0) - if Input.is_action_just_pressed("bind_2"): _equip_hotbar(1) - if Input.is_action_just_pressed("bind_3"): _equip_hotbar(2) - if Input.is_action_just_pressed("bind_4"): _equip_hotbar(3) - if Input.is_action_just_pressed("bind_5"): _equip_hotbar(4) - - if Input.is_action_just_pressed("bark"): - _bark() - if Input.is_action_just_pressed("kiss"): - _kiss() - - if locked: return - - - if is_instance_valid(camera): - var camera_cam = camera - var ray_length = 1000 - var mouse_pos = get_tree().get_nodes_in_group("world_viewport")[0].get_mouse_position() / Globals.pixelize_amount - var from = camera_cam.project_ray_origin(mouse_pos) - var to = from + camera_cam.project_ray_normal(mouse_pos) * ray_length - var space_state = get_world().get_direct_space_state() - var result = space_state.intersect_ray(from, to, []) - if result.has("position"): - mouse_world_pos = result["position"] - mouse_world_pos.y = global_transform.origin.y - - if freecamming: return - if Input.is_action_just_pressed("move_jump"): request_jump = true - - mouse_look = false - - if sitting: return - - if Input.is_action_pressed("move_forward"): direction -= cam_base.transform.basis.z - if Input.is_action_pressed("move_back"): direction += cam_base.transform.basis.z - if Input.is_action_pressed("move_right"): direction += cam_base.transform.basis.x - if Input.is_action_pressed("move_left"): direction -= cam_base.transform.basis.x - - if drunk_wander_length > 0: - direction += drunk_wander_dir - drunk_wander_length -= 1 - - mouse_look = Input.is_action_pressed("mouse_look") - - sprinting = not Input.is_action_pressed("move_sneak") and Input.is_action_pressed("move_sprint") - sneaking = Input.is_action_pressed("move_sneak") and not Input.is_action_pressed("move_sprint") - slow_walking = Input.is_action_pressed("move_walk") - - if emote_locked: - request_jump = false - direction = Vector3.ZERO - -func _input(event): - if not controlled: return - - var mouse_sens = OptionsMenu.mouse_sens - var invert = Vector2(1, 1) - if OptionsMenu.mouse_invert == 1: invert.x = - 1 - elif OptionsMenu.mouse_invert == 2: invert.y = - 1 - elif OptionsMenu.mouse_invert == 3: invert = Vector2( - 1, - 1) - - if event is InputEventMouseMotion and Input.get_mouse_mode() == Input.MOUSE_MODE_CAPTURED: - cam_base.rotation_degrees.y -= event.relative.x * mouse_sens * invert.x - cam_pivot.rotation_degrees.x -= event.relative.y * mouse_sens * invert.y - cam_pivot.rotation_degrees.x = clamp(cam_pivot.rotation_degrees.x, - 80, 80) - -func _unhandled_input(event): - if not controlled: return - - if event.is_action_pressed("primary_action"): _primary_action() - - - - -func _process_movement(delta): - var snap_vec = Vector3(0, - 0.4, 0) - var y_slow = 0.02 - - if is_on_floor() and in_air: - if gravity_vec.y <= - 12.0 and not diving: - diving = true - dive_vec = gravity_vec.length() * - transform.basis.z * 0.2 - player_scale_y = 1.0 - clamp(((abs(gravity_vec.y) / 24.0) * 6.0), 0.0, 0.6) - in_air = false - _sync_particle("dust_land", Vector3(0, - 1, 0)) - _sync_sfx("land") - animation_data["land"] = 0.3 - elif gravity_vec.y < - 6.0: - in_air = true - - player_scale_y = lerp(player_scale_y, 1.0, 0.35) - - if is_on_floor() and ignore_snap <= 0: - y_slow = 0.2 - gravity_vec = get_floor_normal() * - 1.0 - snapped = true - elif snapped and ignore_snap <= 0: - gravity_vec = Vector3.ZERO - snapped = false - else : - snap_vec = Vector3.ZERO - var grav = GRAVITY * Vector3.DOWN * delta - gravity_vec += grav - - if request_jump: - request_jump = false - snap_vec = Vector3.ZERO - if is_on_floor(): - _sync_sfx("jump") - snapped = false - diving = false - gravity_vec = Vector3(0, jump_height, 0) - if sprinting and jump_leap > 0.0: - gravity_vec += - transform.basis.z.normalized() * jump_leap - elif not diving: - _sync_sfx("dive_woosh") - snapped = false - diving = true - dive_vec = - transform.basis.z.normalized() * dive_distance - dive_vec.y = 0 - gravity_vec += Vector3(0, jump_height * 0.5, 0) - _toggle_sit(true) - - if ignore_snap > 0: - snap_vec = Vector3.ZERO - ignore_snap -= 1 - snapped = false - - boost_mult = 1.0 - if boost_timer > 0: - boost_timer -= 1 - boost_mult = boost_amt - - var _speed = walk_speed - if sprinting: _speed = sprint_speed * boost_mult - elif sneaking: _speed = sneak_speed - elif slow_walking: _speed = slow_walk_speed - var _accel = accel if is_on_floor() else accel * 0.35 - - var speed_mult = 1.0 - - - speed_mult = clamp(speed_mult - ((held_item_weight / 25.0) * 0.05), 0.15, 1.0) - - if diving: speed_mult = 0.0 - if gravity_disable: gravity_vec = Vector3.ZERO - - velocity = velocity.move_toward(direction.normalized() * _speed * speed_mult, delta * _accel) - move_and_slide_with_snap(velocity + gravity_vec + dive_vec, snap_vec, Vector3.UP) - - dive_vec = dive_vec.move_toward(Vector3.ZERO, delta * _accel * y_slow) - if not diving: dive_vec = Vector3.ZERO - - rot_help.global_transform.origin = global_transform.origin - - if direction != Vector3.ZERO: - var dir = direction - if diving and dive_vec != Vector3.ZERO: dir = dive_vec.normalized() - - rot_help.look_at(cam_base.global_transform.origin + dir, Vector3.UP) - rotation.y = lerp_angle(rotation.y, rot_help.rotation.y, 0.2) - elif force_cam_look: - rot_help.look_at(camera.global_transform.origin, Vector3.UP) - rotation.y = lerp_angle(rotation.y, rot_help.rotation.y, 0.2) - elif mouse_look and not busy: - rot_help.look_at(mouse_world_pos, Vector3.UP) - rot_diff = abs(rot_help.rotation.y - rotation.y) - rotation.y = lerp_angle(rotation.y, rot_help.rotation.y, 0.2) - - rot_diff = lerp(rot_diff, 0.0, 0.5) - - if diving and dive_vec.length() > 0.4 and is_on_floor(): - - animation_data["dive_scrape"] = true - - if Engine.get_idle_frames() % 4 == 0: - _sync_particle("dust_run", Vector3(0, - 0.8, 0)) - else : - - animation_data["dive_scrape"] = false - - if gravity_vec.y < - 250: - _kill() - - - - -func _enter_state(new_state): - if new_state == - 1: return - - if consume_on_state_change != - 1: - PlayerData._remove_item(consume_on_state_change, false) - consume_on_state_change = - 1 - - state = new_state - emit_signal("_state_change") - cam_push = 0.0 - animation_data["bobber_visible"] = false - - match state: - STATES.DEFAULT: - animation_data["item_bend"] = 0.0 - locked = false - STATES.BUSY: locked = false - STATES.OBTAIN: locked = true - STATES.SHOWCASE: - locked = true - _equip_item(PlayerData._find_item_code(showcase_ref), true, true) - - STATES.FISHING: - cam_push = - 0.3 - locked = true - animation_data["bobber_visible"] = true - PlayerData.emit_signal("_help_update", "hold to reel (SPRINT reels quicker)") - _enter_animation("rod_idle", true) - STATES.FISHING_CANCEL: - animation_data["item_bend"] = 0.1 - cam_push = - 0.3 - locked = true - rod_cast_dist = 0.0 - animation_data["bobber_visible"] = true - PlayerData.emit_signal("_item_equip", held_item.ref) - _bobber_retract() - _enter_animation("rod_retract", false, true, STATES.DEFAULT) - STATES.FISHING_CAST: - cam_push = - 0.3 - animation_data["bobber_visible"] = true - locked = true - STATES.FISHING_CHARGE: - cam_push = 0.0 - locked = false - STATES.FISHING_STRUGGLE: - animation_data["item_bend"] = - 0.7 - cam_push = - 0.3 - animation_data["bobber_visible"] = true - locked = true - _enter_animation("rod_struggle", true) - - STATES.GUITAR: locked = true - - STATES.SHOVEL_CAST: locked = true - STATES.SHOVEL_STRUGGLE: - locked = true - _shovel_check() - STATES.SHOVEL_CANCEL: - locked = true - _enter_animation("shovel_retract", false, true, STATES.DEFAULT) - - STATES.NET_CAST: locked = true - STATES.NET_STRUGGLE: - locked = true - _net_check() - STATES.NET_CANCEL: - locked = true - _enter_animation("net_retract", false, true, STATES.DEFAULT) - - STATES.CONSUME_ITEM: - locked = false - _equip_item({"ref": 0}, true, true) - _enter_state(STATES.DEFAULT) - -func _primary_action(): - match state: - STATES.DEFAULT: - _use_item() - STATES.OBTAIN: - _enter_animation("equip", false, true, 0, false, 1.5) - -func _primary_action_hold(): - primary_hold_timer += 1 - - match state: - STATES.DEFAULT: - return - STATES.FISHING: - _enter_animation("rod_reel", true) - rod_cast_dist -= 0.04 if Input.is_action_pressed("move_sprint") else 0.01 - bobber_control = true - recent_reel = 15 - if rod_cast_dist < 1.5: _enter_state(STATES.FISHING_CANCEL) - -func _primary_action_release(): - emit_signal("_primary_release") - - match state: - STATES.DEFAULT: - _release_item() - STATES.FISHING: - _enter_animation("rod_idle", true) - STATES.METAL_DETECTOR: - _enter_state(STATES.DEFAULT) - - - - - -func _interact(): - if not controlled or interact_cooldown > 0: return - for area in interact_range.get_overlapping_areas(): - if area.is_in_group("interactable") and area.is_visible_in_tree(): - area._activate(self) - interact_cooldown = 60 - return - -func _enter_zone(zone, entrance_id, zone_owner = - 1): - if not controlled: return - world._enter_zone(zone, zone_owner) - PlayerData.player_saved_zone = zone - PlayerData.player_saved_zone_owner = zone_owner - - print("Finding entrance w id: ", entrance_id, " and owner: ", zone_owner) - for entrance in get_tree().get_nodes_in_group("area_entrance"): - print(entrance.entrance_id, ": ", entrance.entrance_owner, " ?") - if entrance.entrance_id == entrance_id and entrance.entrance_owner == zone_owner: - global_transform.origin = entrance.global_transform.origin - last_valid_pos = global_transform.origin - return - - print("Fallback!") - global_transform.origin = world.map.spawn_position.global_transform.origin - world._enter_zone("main_zone", - 1) - PlayerData.player_saved_zone = "main_zone" - PlayerData.player_saved_zone_owner = - 1 - last_valid_pos = global_transform.origin - -func _obtain_item(ref, bonus_text = [], journal_check = true): - old_rot = rotation - showcase_ref = ref - _equip_item({"ref": - 1}, true, true) - _enter_animation("equip", false, false, STATES.SHOWCASE, false, 1.5) - _play_sfx("strum") - - - var data = PlayerData._find_item_code(ref) - var text = "You caught a " + PlayerData._get_item_name(ref) + "! [color=#d5aa73](It's " + str(data["size"]) + "cm!)[/color]\n\n" + str(Globals.item_data[data["id"]]["file"].catch_blurb) - - if journal_check: - var new = true - for type in PlayerData.journal_logs.keys(): - for entry in PlayerData.journal_logs[type].keys(): - if entry == data.id and PlayerData.journal_logs[type][entry].count > 1: - new = false - break - if new: - text = "Woah, a new creature! " + text - - hud.dialogue_text = [text] - if bonus_text != []: hud.dialogue_text.append_array(bonus_text) - hud._change_menu(6) - - force_cam_look = true - yield (get_tree().create_timer(0.1), "timeout") - cam_follow_node = $catch_cam_position - -func _level_up(): - yield (get_tree().create_timer(0.4), "timeout") - var bubble = title._create_level_bubble() - Network._send_actor_action(actor_id, "_sync_level_bubble", [], false) - GlobalAudio._play_sound("jingle_win") - - _play_emote("emote_cheer", "happy") - - $emotion_particles / lvl_particles.restart() - $emotion_particles / lvl_particles2.restart() - -func _exit_showcase(): - if state != STATES.SHOWCASE: return - _exit_animation() - _enter_state(STATES.DEFAULT) - _equip_item(PlayerData._find_item_code(previous_item), false) - get_tree().create_tween().tween_property(self, "rotation", old_rot, 0.3) - - if xp_buildup > 0: - PlayerData._add_xp(ceil(xp_buildup)) - xp_buildup = 0 - -func _equip_hotbar(slot): - if locked or not PlayerData.hotbar.keys().has(slot): return - var ref = PlayerData.hotbar[slot] - _equip_item(PlayerData._find_item_code(ref)) - -func _equip_item(item_data, skip_anim = false, forced = false, set_prev = true): - if set_prev and held_item["ref"] != 0: previous_item = held_item["ref"] - if (state != STATES.DEFAULT and not forced) or held_item["ref"] == item_data["ref"]: return - - if not item_data.keys().has("id") or not Globals.item_data.keys().has(item_data["id"]): - item_data = PlayerData.FALLBACK_ITEM - - PlayerData.emit_signal("_item_equip", item_data["ref"]) - - if not skip_anim: - _sync_sfx("equip") - _enter_state(STATES.BUSY) - _enter_animation("equip", false, false, STATES.DEFAULT, false, 1.5) - yield (self, "_animation_finished") - - - var held_data = item_data.duplicate() - - hud.show_bait = Globals.item_data[item_data["id"]]["file"].show_bait - - var data = Globals.item_data[item_data["id"]]["file"] - held_item_weight = item_data["size"] - if not data.uses_size: held_item_weight = 0.0 - - _update_held_item(held_data) - Network._send_actor_action(actor_id, "_update_held_item", [held_item], false) - -func _use_item(): - if held_item.empty(): return - var item_data = Globals.item_data[held_item["id"]]["file"] - if has_method(item_data.action) and item_data.action != "": - callv(item_data.action, item_data.action_params) - -func _release_item(): - if held_item.empty(): return - var item_data = Globals.item_data[held_item["id"]]["file"] - if has_method(item_data.release_action) and item_data.release_action != "": - call(item_data.release_action) - - - - - - - -func _cast_fishing_rod(): - rod_cast_data = PlayerData.LURE_DATA[PlayerData.lure_selected].effect_id - animation_data["caught_item"] = {} - bait_warn = 1 - - rod_damage = [1, 3, 10, 20, 35, 50][PlayerData.rod_power_level] - rod_spd = [0.0, 0.1, 0.24, 0.4, 0.7, 1.0][PlayerData.rod_speed_level] - rod_chance = [0.0, 0.02, 0.04, 0.06, 0.08, 0.1][PlayerData.rod_chance_level] - - _enter_state(STATES.FISHING_CHARGE) - _enter_animation("rod_wind", true, false, - 1, false) - yield (self, "_primary_release") - if state != STATES.FISHING_CHARGE: return - _enter_state(STATES.FISHING_CAST) - _sync_sfx("woosh", null, 1.0, 0.4) - var strength = clamp(primary_hold_timer * 0.06, 1.5, 9.0) - rod_cast_dist = strength - fish_detect.translation.z = - strength - - var is_valid_fishing_spot = false - if bobber_preview.is_colliding() and bobber_preview.get_collider().is_in_group("valid_water"): - is_valid_fishing_spot = true - - _bobber_cast(rod_cast_dist, bobber_preview.get_collision_point() + Vector3(0, 0.75, 0), is_valid_fishing_spot) - - rod_depth = 0 - - casted_bait = PlayerData.bait_selected - if casted_bait != "" and PlayerData.bait_inv[casted_bait] <= 0: casted_bait = "" - - _exit_animation() - if is_valid_fishing_spot: - animation_data["item_bend"] = 0.3 - retract_splash = true - _enter_animation("rod_cast", false, true, STATES.FISHING) - else : - animation_data["item_bend"] = 0.3 - retract_splash = false - _enter_animation("rod_cast", false, true, STATES.FISHING_CANCEL) - - yield (get_tree().create_timer(0.4), "timeout") - animation_data["item_bend"] = - 0.2 - -func _on_fish_catch_timer_timeout(): - if not controlled: return - - fish_timer.wait_time = rand_range(2.0, 3.0) - fish_timer.start() - - if state != STATES.FISHING: return - var fish_type = "ocean" - var junk_mult = 1.0 - - - fish_zone_data = {"id": - 1, "boost": 0.0} - for zone in fishing_area.get_overlapping_areas(): - if zone.is_in_group("fish_zone"): - fish_zone_data["id"] = zone.id - fish_zone_data["boost"] = zone.chance_boost - junk_mult = zone.junk_mult - if zone.fish_type != "": fish_type = zone.fish_type - - var fish_chance = 0.0 - var base_chance = BAIT_DATA[casted_bait]["catch"] - fish_chance = base_chance - fish_chance += (base_chance * failed_casts) - fish_chance += (base_chance * rod_chance) - fish_chance += fish_zone_data["boost"] * fish_chance - if recent_reel > 0: fish_chance *= 1.1 - if rod_cast_data == "attractive": fish_chance *= 1.3 - if in_rain: fish_chance *= 1.1 - - fish_chance *= catch_drink_boost - print("Fish chance w ", fish_chance, "w type ", fish_type) - - bait_warn -= 1 - if bait_warn <= 0 and fish_chance <= 0.0: - bait_warn = 8 - var text = "" - if casted_bait == "": - text = "[color=#ac0029]You've got no bait attached! You won't catch any fish like that...[/color]" - else : - text = "[color=#ac0029]Seems nothing is going to bite... perhaps your bait isn't for this water...[/color]" - - Network._update_chat(text) - - if randf() > fish_chance: - failed_casts += 0.05 - return - failed_casts = 0.0 - - - var bait_use_chance = 1.0 - if rod_cast_data == "efficient": bait_use_chance = 0.8 - - if randf() < bait_use_chance: PlayerData._use_bait(casted_bait) - else : PlayerData._send_notification("The Efficient Lure saved your bait!") - var max_tier = BAIT_DATA[casted_bait]["max_tier"] - - var double_bait = 0.0 - if ["large", "sparkling", "double"].has(rod_cast_data): double_bait = 0.25 - if randf() < double_bait: - PlayerData._use_bait(casted_bait) - PlayerData._send_notification("Your lure used extra bait...", 1) - - if rod_cast_data == "gold": - for i in 2: PlayerData._use_bait(casted_bait) - PlayerData._send_notification("Your golden lure used extra bait...", 1) - - var treasure_mult = 1.0 - if rod_cast_data == "magnet": treasure_mult = 2.0 - if rod_cast_data == "salty": fish_type = "ocean" - if rod_cast_data == "fresh": fish_type = "lake" - - var force_av_size = false - - if randf() < 0.05 * treasure_mult * junk_mult: - fish_type = "water_trash" - max_tier = 0 - force_av_size = true - - - - - - - - if in_rain and randf() < 0.08: - fish_type = "rain" - - - var rolls = [] - for i in 3: - var roll = Globals._roll_loot_table(fish_type, max_tier) - var s = Globals._roll_item_size(roll) - rolls.append([roll, s]) - - - var reroll_type = "none" - if rod_cast_data == "small": reroll_type = "small" - if rod_cast_data == "sparkling": reroll_type = "tier" - if rod_cast_data == "large": reroll_type = "large" - if rod_cast_data == "gold": reroll_type = "rare" - - var chosen = rolls[0] - for roll in rolls: - match reroll_type: - "none": - chosen = roll - - "small": - if roll[1] < chosen[1]: - chosen = roll - - "large": - if roll[1] > chosen[1]: - chosen = roll - - "tier": - var old_tier = Globals.item_data[chosen[0]]["file"].tier - var new_tier = Globals.item_data[roll[0]]["file"].tier - if new_tier > old_tier: - chosen = roll - - "rare": - var new_rare = Globals.item_data[roll[0]]["file"].rare - if new_rare: - chosen = roll - - var fish_roll = chosen[0] - var size = chosen[1] - - - var quality = PlayerData.ITEM_QUALITIES.NORMAL - var r = randf() - for q in PlayerData.ITEM_QUALITIES.size(): - if BAIT_DATA[casted_bait]["quality"].size() - 1 < q: - print("bait does not support rarity ", q) - break - if randf() < BAIT_DATA[casted_bait]["quality"][q]: - quality = q - print("-------------------------Rolled Quality: ", quality) - - if randf() < 0.02 * treasure_mult: - fish_roll = "treasure_chest" - size = 60.0 - quality = 0 - - var data = Globals.item_data[fish_roll]["file"] - var quality_data = PlayerData.QUALITY_DATA[quality] - - if force_av_size: size = data.average_size - - var diff_mult = clamp(size / data.average_size, 0.7, 1.8) - var difficulty = clamp((data.catch_difficulty * diff_mult * quality_data.diff) + quality_data.bdiff, 1.0, 250.0) - - var xp_mult = size / data.average_size - if xp_mult < 0.15: xp_mult = 1.25 + xp_mult - xp_mult = max(0.5, xp_mult) - var xp_add = ceil(data.obtain_xp * xp_mult * catch_drink_xp * quality_data.worth) - - print("Total Rolls: ", rolls) - print("Roll fish ", fish_roll, " with size ", size, " and diff Mult: ", diff_mult) - - if fish_zone_data["id"] != - 1: - _wipe_actor(fish_zone_data["id"]) - Network._send_actor_action(actor_id, "_wipe_actor", [fish_zone_data["id"]]) - - _enter_state(STATES.FISHING_STRUGGLE) - animation_data["alert"] = true - yield (get_tree().create_timer(1.0), "timeout") - - animation_data["alert"] = false - - var delay_time = 0 - while hud.using_chat: - yield (get_tree().create_timer(0.15), "timeout") - delay_time += 1 - - if delay_time > 200: - _enter_state(STATES.FISHING_CANCEL) - return - - hud._open_minigame("fishing3", {"fish": fish_roll, "rod_type": rod_cast_data, "reel_mult": catch_drink_reel, "quality": quality, "damage": rod_damage, "speed": rod_spd}, difficulty) - var success = yield (hud, "_minigame_finished") - print("SUCCESS: ", success) - if success: - animation_data["caught_item"] = {"id": fish_roll, "size": size} - _enter_state(STATES.FISHING_CANCEL) - yield (self, "_state_change") - - var ref - var catches = 1 - var bonus_text = [] - - if rod_cast_data == "double" and randf() < 0.15: - catches = 2 - bonus_text.append("Your Double Hook doubled the fish!") - - if PlayerData.rod_luck_level > 0 and randf() < 0.15: - bonus_text.append("How lucky! You found a bonus [color=#d57900]Coin Bag[/color] aswell!") - PlayerData._add_item("luck_moneybag", - 1, randi() % 15 + 15, PlayerData.rod_luck_level) - - var tags = [] - - for i in catches: - ref = PlayerData._add_item(fish_roll, - 1, size, quality, tags) - PlayerData._log_item(fish_roll, size, quality) - PlayerData._quest_progress("catch", fish_roll) - PlayerData._quest_progress("catch_type", data.loot_table) - PlayerData._quest_progress("catch_small", PlayerData._get_size_type(fish_roll, size)) - PlayerData._quest_progress("catch_big", PlayerData._get_size_type(fish_roll, size)) - if fish_roll == "treasure_chest": PlayerData._quest_progress("catch_treasure") - if in_rain: PlayerData._quest_progress("catch_rain") - if data.tier == 2: PlayerData._quest_progress("catch_hightier") - xp_buildup += xp_add - - PlayerData._catch_fish() - - _obtain_item(ref, bonus_text) - - if rod_cast_data == "lucky": - var worth = PlayerData._get_item_worth(ref) - var gold = max(1, ceil(worth * rand_range(0.01, 0.1))) - PlayerData.money += gold - PlayerData._send_notification("Your Lucky Lure got you $" + str(gold) + "!") - - if catch_drink_gold_add != Vector2(0, 0): - var gold = max(1, ceil(rand_range(catch_drink_gold_add.x, catch_drink_gold_add.y))) - PlayerData.money += gold - PlayerData._send_notification("Your Catcher's Luck got you $" + str(gold) + "!") - - if catch_drink_gold_percent > 0.0: - var worth = PlayerData._get_item_worth(ref) - var gold = max(1, ceil(rand_range(worth * 0.01, worth * catch_drink_gold_percent))) - PlayerData.money += gold - PlayerData._send_notification("Your Catcher's Luck Ultra got you $" + str(gold) + "!") - - - else : - _enter_state(STATES.FISHING_CANCEL) - -func _wipe_actor(id): - var actor = world._get_actor_by_id(id) - print("Wiping actor ", id, " : ", actor) - if is_instance_valid(actor): - actor._deinstantiate(true) - - -func _shovel(): - _enter_state(STATES.SHOVEL_CAST) - _enter_animation("shovel_dig", false, true, STATES.SHOVEL_STRUGGLE) - -func _shovel_check(): - var mound = false - - var area - for a in shovel_area.get_overlapping_areas(): - if a.is_in_group("mound"): - mound = true - area = a - - if is_instance_valid(area): - var id = area.owner.actor_id - _wipe_actor(id) - Network._send_actor_action(actor_id, "_wipe_actor", [id]) - - if mound: - _enter_animation("shovel_struggle", true, true) - hud._open_minigame("shoveling", {"damage": 5.0}) - var success = yield (hud, "_minigame_finished") - print("SUCCESS: ", success) - if success: - _enter_state(STATES.SHOVEL_CANCEL) - yield (self, "_state_change") - var ref = PlayerData._add_item("spider") - - - _obtain_item(ref) - else : - _enter_state(STATES.SHOVEL_CANCEL) - else : - _enter_state(STATES.SHOVEL_CANCEL) - - -func _cast_net(): - _enter_state(STATES.NET_CAST) - _enter_animation("net_use", false, true, STATES.NET_STRUGGLE) - -func _net_check(): - var bug = false - - var area - for a in net_area.get_overlapping_areas(): - if a.is_in_group("bug"): - bug = true - area = a - - var bug_id = "" - if is_instance_valid(area): - var id = area.owner.actor_id - bug_id = area.owner.bug_id - _wipe_actor(id) - Network._send_actor_action(actor_id, "_wipe_actor", [id]) - - if bug: - _enter_animation("net_struggle", true, true) - hud._open_minigame("shoveling", {"damage": 5.0}) - var success = yield (hud, "_minigame_finished") - print("SUCCESS: ", success) - if success: - _enter_state(STATES.NET_CANCEL) - yield (self, "_state_change") - var size = Globals._roll_item_size(bug_id) - var ref = PlayerData._add_item(bug_id, - 1, size) - - - _obtain_item(ref) - else : - _enter_state(STATES.NET_CANCEL) - else : - _enter_state(STATES.NET_CANCEL) - - - - - - - - - - - - - - -func _toggle_freecam(): - if busy: return - - if not freecamming: - PlayerData._send_notification("entering freecam mode") - freecamming = true - hud.freecamming = true - cam_orbit_node = freecam_anchor - freecam_anchor.global_transform.origin = global_transform.origin - else : - PlayerData._send_notification("exiting freecam mode") - freecamming = false - hud.freecamming = false - cam_orbit_node = null - -func _freecam_input(delta): - if not freecamming: return - - var speed = 4.0 - if Input.is_action_pressed("move_sprint"): - speed = 7.0 - if Input.is_action_pressed("move_sneak"): - speed = 1.0 - var max_dist = 15.0 - var build_dir = Vector3.ZERO - - if Input.is_action_pressed("move_forward"): build_dir -= cam_base.transform.basis.z - if Input.is_action_pressed("move_back"): build_dir += cam_base.transform.basis.z - if Input.is_action_pressed("move_right"): build_dir += cam_base.transform.basis.x - if Input.is_action_pressed("move_left"): build_dir -= cam_base.transform.basis.x - if Input.is_action_pressed("move_up"): build_dir += cam_base.transform.basis.y - if Input.is_action_pressed("move_down"): build_dir -= cam_base.transform.basis.y - - freecam_anchor.global_transform.origin += build_dir * speed * delta - - freecam_anchor.global_transform.origin.x = clamp(freecam_anchor.global_transform.origin.x, global_transform.origin.x - max_dist, global_transform.origin.x + max_dist) - freecam_anchor.global_transform.origin.y = clamp(freecam_anchor.global_transform.origin.y, global_transform.origin.y - max_dist, global_transform.origin.y + max_dist) - freecam_anchor.global_transform.origin.z = clamp(freecam_anchor.global_transform.origin.z, global_transform.origin.z - max_dist, global_transform.origin.z + max_dist) - - - - - -func _create_prop(ref, offset = Vector3(0, 1, 0)): - if not controlled: return - - - for prop in prop_ids: - if prop.ref == ref: - _wipe_actor(prop.id) - prop_ids.erase(prop) - PlayerData._send_notification("clearing prop", 1) - PlayerData.props_placed = prop_ids - PlayerData.emit_signal("_prop_update") - return - - - if $detection_zones / prop_detect.get_overlapping_bodies().size() > 0 or not is_on_floor() or not $detection_zones / prop_ray.is_colliding(): - PlayerData._send_notification("invalid prop placement", 1) - return - - - if prop_ids.size() > 4: - PlayerData._send_notification("prop limit reached", 1) - return - - - var item = PlayerData._find_item_code(ref) - var data = Globals.item_data[item["id"]]["file"] - var ver_offset = Vector3(0, 1.0, 0) * (1.0 - player_scale) - print(ver_offset) - var pos = global_transform.origin + (global_transform.basis.z * - 2.0) - offset + ver_offset - var rot = rotation + Vector3(0, deg2rad(180), 0) - var prop_code = data.prop_code - - - var blacklist = ["island_tiny", "island_med", "island_big"] - if current_zone_owner != - 1: - if blacklist.has(prop_code): - PlayerData._send_notification("this prop cannot be spawned here...", 1) - return - - - var new_id = Network._sync_create_actor(prop_code, pos, current_zone, - 1, Network.STEAM_ID, {"rotation": rot, "current_zone_owner": current_zone_owner}) - prop_ids.append({"id": new_id, "ref": ref, "prop_id": data.action}) - PlayerData.props_placed = prop_ids - PlayerData.emit_signal("_prop_update") - PlayerData._send_notification("prop placed", 0) - - _sync_particle("dust_land", pos, true) - _sync_sfx("poof", pos) - -func _clear_props(): - if prop_ids.size() <= 0: - PlayerData._send_notification("no props to undo", 1) - return - PlayerData._send_notification("undo prop", 0) - _wipe_actor(prop_ids[prop_ids.size() - 1][0]) - prop_ids.remove(prop_ids.size() - 1) - -func _clear_all_props(): - if prop_ids.size() <= 0: - PlayerData._send_notification("no props to clear", 1) - return - - PlayerData._send_notification("clearing all props", 0) - for prop in prop_ids: - _wipe_actor(prop.id) - prop_ids.clear() - PlayerData.props_placed = prop_ids - PlayerData.emit_signal("_prop_update") - -func _item_place_prop(): - if not controlled: return - _create_prop(held_item.ref) - - - - - -func _update_animation_data(): - if animation_data["emote"] != "": - var anim = $body / player_body / AnimationPlayer.get_animation(animation_data["emote"]) - animation_timer += 1 - animation_goal = floor((anim.length * 60) / animation_data["emote_timescale"]) - if animation_timer >= animation_goal and emoting and animation_goal > 0 and not emote_looping: - print("Finished Emoting ", animation_data["emote"], " bufferstate: ", buffer_state) - emit_signal("_animation_finished") - emoting = false - emote_full = false - emote_locked = false - animation_data["emote"] = "" - if buffer_state != - 1: _enter_state(buffer_state) - - animation_data["emoting"] = emoting - animation_data["emote_full"] = emoting and emote_full - animation_data["moving"] = direction != Vector3.ZERO or rot_diff > 0.05 - animation_data["sprinting"] = sprinting and direction != Vector3.ZERO - animation_data["sneaking"] = sneaking and direction != Vector3.ZERO - animation_data["diving"] = diving - animation_data["sitting"] = sitting - animation_data["busy"] = busy and state == STATES.DEFAULT and not emoting - animation_data["land"] = lerp(animation_data["land"], 0.0, 0.04) - animation_data["talking"] = lerp(animation_data["talking"], 0.0, 0.1) - animation_data["recent_reel"] = lerp(animation_data["recent_reel"], recent_reel, 0.2) - animation_data["caught_fish"] = state == STATES.FISHING_STRUGGLE - animation_data["player_scale"] = lerp(animation_data["player_scale"], player_scale, 0.06) - animation_data["player_scale_y"] = lerp(animation_data["player_scale_y"], player_scale_y, 0.06) - animation_data["run_mult"] = 1.15 * boost_mult - animation_data["walk_mult"] = 1.25 if not slow_walking else 0.7 - animation_data["drunk_tier"] = drunk_tier - animation_data["state"] = state - - $emotion_particles / mushroom_trail.emitting = animation_data["mushroom"] - $emotion_particles / drunk_particles.emitting = animation_data["drunk_tier"] > 1 - if is_on_floor(): animation_data["mushroom"] = false - - if not bobber_control: - animation_data["bobber_position"] = Vector3(bobber_hpos.x, bobber_vpos, bobber_hpos.z) - else : - var pos = global_transform.origin + ( - rod_cast_dist * global_transform.basis.z) - pos.y = bobber_vpos + sin(OS.get_ticks_msec() * 0.002) * 0.05 - animation_data["bobber_position"] = lerp(animation_data["bobber_position"], pos, 0.2) - -func _process_animation(): - var anim_lerp = 0.4 - - var emote_type = 0 - if animation_data["emoting"]: emote_type = 1 - if animation_data["emote_full"]: emote_type = 2 - - anim_tree.set("parameters/emote_full/blend_amount", lerp(anim_tree.get("parameters/emote_full/blend_amount"), float(emote_type == 2), anim_lerp)) - if anim_tree.get("parameters/emote_full/blend_amount") < 0.1 and emote_type != 2: anim_tree.set("parameters/emote_full/blend_amount", 0.0) - anim_tree.set("parameters/emote_half/blend_amount", lerp(anim_tree.get("parameters/emote_half/blend_amount"), float(emote_type == 1), anim_lerp)) - if anim_tree.get("parameters/emote_half/blend_amount") < 0.1 and emote_type != 1: anim_tree.set("parameters/emote_half/blend_amount", 0.0) - - anim_tree.set("parameters/arms/blend_amount", lerp(anim_tree.get("parameters/arms/blend_amount"), float( not animation_data["emoting"]), anim_lerp)) - anim_tree.set("parameters/movement/blend_amount", lerp(anim_tree.get("parameters/movement/blend_amount"), float(animation_data["moving"]), anim_lerp)) - anim_tree.set("parameters/run_blend/blend_amount", lerp(anim_tree.get("parameters/run_blend/blend_amount"), float(animation_data["sprinting"]) - float(animation_data["sneaking"]), anim_lerp)) - anim_tree.set("parameters/dive/blend_amount", lerp(anim_tree.get("parameters/dive/blend_amount"), float(animation_data["diving"]), anim_lerp)) - anim_tree.set("parameters/arm_blend/blend_position", float(animation_data["arm_value"])) - anim_tree.set("parameters/sitting/blend_amount", lerp(anim_tree.get("parameters/sitting/blend_amount"), float(animation_data["sitting"]), anim_lerp)) - anim_tree.set("parameters/thinking/blend_amount", lerp(anim_tree.get("parameters/thinking/blend_amount"), float(animation_data["busy"]), anim_lerp)) - anim_tree.set("parameters/land/blend_amount", lerp(anim_tree.get("parameters/land/blend_amount"), float(animation_data["land"]), anim_lerp)) - anim_tree.set("parameters/talking/blend_amount", lerp(anim_tree.get("parameters/talking/blend_amount"), float(animation_data["talking"]), anim_lerp)) - anim_tree.set("parameters/runmodif/scale", animation_data["run_mult"]) - anim_tree.set("parameters/walkmodif/scale", animation_data["walk_mult"]) - anim_tree.set("parameters/emote_time/scale", animation_data["emote_timescale"]) - anim_tree.set("parameters/emote_timeb/scale", animation_data["emote_timescale"]) - - face._show_blush(animation_data["drunk_tier"] > 1) - if animation_data["caught_item"] != caught_item: - caught_item = animation_data["caught_item"] - _update_caught_item(animation_data["caught_item"]) - - if item_scene: - item_scene.item_bend = lerp(item_scene.item_bend, animation_data["item_bend"], 0.4) - bobber_line.end_anchor = item_scene.get_node(item_scene.hotspot_node).global_transform.origin - bobber_line.y_drop = - 1.5 + ((animation_data["recent_reel"] / 15.0) * 0.5) - if animation_data["caught_fish"]: bobber_line.y_drop = 0 - if bobber.visible: bobber.global_transform.origin = animation_data["bobber_position"] + Vector3(0, (0.0 if not animation_data["caught_fish"] else - 0.3), 0) - bobber.visible = animation_data["bobber_visible"] - bobber_line.active = bobber.visible - - - if animation_data["player_scale"] != scale.y: - scale = animation_data["player_scale"] * Vector3.ONE - scale.y *= animation_data["player_scale_y"] - - bobber_line.line_scale = animation_data["player_scale"] - - ripples.visible = animation_data["state"] == STATES.FISHING - var rip_anim = "default" if animation_data["recent_reel"] <= 1.0 else "wake" - if ripples.animation != rip_anim: ripples.animation = rip_anim - - var root = anim_tree.tree_root - var node = root.get_node("emote_anim") - var node_b = root.get_node("emote_anim_b") - if node.animation != animation_data["emote"]: - anim_tree.set("parameters/emote_sync/seek_position", 0.0) - anim_tree.set("parameters/emote_half_sync/seek_position", 0.0) - node.set_animation(animation_data["emote"]) - node_b.set_animation(animation_data["emote"]) - - $"%head_alert".visible = animation_data["alert"] - - tail.sitting = animation_data["sitting"] - tail.diving = animation_data["diving"] - tail.motion = float(animation_data["moving"]) - tail.wagging = animation_data["wagging"] - - - if animation_data["back_bend"] != 0.0: - var pose = skeleton.get_bone_pose(1) - pose = pose.rotated(Vector3(1, 0, 0), animation_data["back_bend"]) - skeleton.set_bone_custom_pose(1, pose) - else : - skeleton.set_bone_custom_pose(1, Transform()) - -func _enter_animation(anim_name, loop = false, _locked = true, state_enter = - 1, full_anim = true, timescale = 1.0): - if anim_name == animation_data["emote"]: return - diving = false - - animation_data["emote_timescale"] = timescale - - - emote_full = full_anim - emote_locked = _locked - emote_looping = loop - buffer_state = state_enter if not emote_looping else - 1 - if emote_looping: _enter_state(state_enter) - animation_data["emote"] = anim_name - animation_timer = 0 - emoting = true - -func _exit_animation(): - emit_signal("_animation_finished") - emoting = false - emote_full = false - emote_locked = false - emote_looping = false - animation_data["emote"] = "" - -func _update_held_item(new): - if new.empty() or held_item == new: return - - held_item = new - var item_data = Globals.item_data[new["id"]]["file"] - var alt_scale_mult = 1.0 - - - item_sprite.texture = item_data.icon.duplicate() if (item_data.show_item and not item_data.show_scene) else null - - - - if is_instance_valid(item_scene): item_scene.queue_free() - item_scene = null - if item_data.show_scene and item_data.item_scene: - item_scene = item_data.item_scene.instance() - hand_bone.add_child(item_scene) - - var scale_mult = 0.07 - clamp(new["size"] * 0.01, 0.01, 0.061) - var item_scale = 1.0 if not item_data.uses_size else new["size"] * scale_mult - item_scale *= alt_scale_mult - item_sprite.scale = Vector3(item_scale, item_scale, item_scale) - - - var y = 0.0 - var offs = 0.0 - var back_bend = 0.0 - if item_data.uses_size: - y = clamp(item_scale * item_scale * 0.08, 0.0, 0.36) - offs = clamp(item_scale * item_scale, 0.0, 16.0) - back_bend = clamp(item_scale * item_scale, 0.0, 55.0) - - item_sprite.translation.y = y - item_sprite.offset.y = item_data.hold_offset + offs - - - $body / player_body / Armature / Skeleton / BoneAttachment / Spatial.rotation_degrees = Vector3(0.0, 0.0, back_bend * 0.7) - - item_sprite.translation.z = 0.0 - item_sprite.vel = 0.0 - item_sprite.mult = clamp(item_scale * 3.5, 0.0, 1.2) if item_data.category == "fish" and item_data.alive else 0.0 - - item_sprite.modulate = Color("#ffffff") - item_sprite.opacity = 1.0 - for child in item_sprite.get_children(): child.emitting = false - - if PlayerData.QUALITY_DATA.has(new["quality"]): - item_sprite.modulate = Color(PlayerData.QUALITY_DATA[new["quality"]]["mod"]) - item_sprite.opacity = PlayerData.QUALITY_DATA[new["quality"]]["op"] - if PlayerData.QUALITY_DATA[new["quality"]]["particle"] > - 1: item_sprite.get_child(PlayerData.QUALITY_DATA[new["quality"]]["particle"]).emitting = true - - animation_data["arm_value"] = item_data.arm_value - animation_data["back_bend"] = deg2rad( - back_bend) - -func _update_caught_item(new): - if new.empty(): - caught_fish.texture = null - return - caught_item = new - - var item_data = Globals.item_data[new["id"]]["file"] - caught_fish.texture = item_data.icon.duplicate() if item_data.show_item else null - - var item_scale = 1.0 if not item_data.uses_size else new["size"] * 0.01 - caught_fish.scale = Vector3(item_scale, item_scale, item_scale) - -func _bobber_cast(dist, end, splash): - bobber_control = false - var height = dist * 0.4 - height += max(0, global_transform.origin.y - end.y) - - end.y -= 1 - - bobber_hpos = global_transform.origin - bobber_vpos = global_transform.origin.y - yield (get_tree().create_timer(0.4), "timeout") - - var tween = get_tree().create_tween() - - tween.set_trans(1) - tween.set_ease(1) - tween.tween_property(self, "bobber_vpos", end.y + height, dist * 0.05) - tween.set_ease(0) - tween.tween_property(self, "bobber_vpos", end.y - (height * 0.2), dist * 0.05) - if splash: - tween.tween_callback(self, "_bobber_splash") - - - tween.set_trans(1) - tween.set_ease(1) - tween.tween_property(self, "bobber_vpos", end.y + (height * 0.1), (dist * 0.05)) - tween.set_ease(0) - tween.tween_property(self, "bobber_vpos", end.y - (height * 0.05), (dist * 0.1)) - - - tween.set_trans(1) - tween.set_ease(1) - tween.tween_property(self, "bobber_vpos", end.y, (dist * 0.1)) - - var htween = get_tree().create_tween() - htween.set_trans(4) - htween.set_ease(1) - htween.tween_property(self, "bobber_hpos", end + (global_transform.basis.z * 0.5), dist * 0.1) - htween.tween_property(self, "bobber_hpos", end + (global_transform.basis.z * 0.2), dist * 0.15) - htween.tween_property(self, "bobber_hpos", end, dist * 0.1) - - yield (htween, "finished") - bobber_control = true - -func _bobber_retract(): - bobber_control = false - bobber_hpos = bobber.global_transform.origin - bobber_vpos = bobber.global_transform.origin.y - - yield (get_tree().create_timer(0.4), "timeout") - - if retract_splash: _bobber_splash("splashb") - var tween = get_tree().create_tween() - tween.set_trans(1) - tween.set_ease(1) - tween.tween_property(self, "bobber_vpos", global_transform.origin.y + 2.0, 0.3) - tween.set_ease(0) - tween.tween_property(self, "bobber_vpos", global_transform.origin.y, 0.3) - - var htween = get_tree().create_tween() - htween.tween_property(self, "bobber_hpos", global_transform.origin, 0.6) - -func _bobber_splash(sfx = "splash"): - _sync_particle("splash", bobber.global_transform.origin, true) - _sync_sfx(sfx, Vector3(bobber_hpos.x, bobber_vpos, bobber_hpos.z)) - - - - - -func _toggle_sit(exit = false): - if not exit: sitting = not sitting - else : sitting = false - -func _play_emote(emote_id, emotion = ""): - if emote_id == "": return - - if emote_id == "sit": - _toggle_sit() - elif not emote_locked: - _enter_animation(emote_id, false, true) - if emotion != "": - _sync_face_emote(emotion) - - - - - -func _change_cosmetics(): - if cosmetic_data == PlayerData.cosmetics_equipped: return - var new = PlayerData.cosmetics_equipped.duplicate() - if not dead_actor: Network._send_actor_action(actor_id, "_update_cosmetics", [new]) - _update_cosmetics(new) - -func _update_cosmetics(data): - var valid = true - for key in PlayerData.FALLBACK_COSM.keys(): - if not data.keys().has(key): - print("missing key ", key) - valid = false - for key in data.keys(): - if not (data[key] is Array): - if not Globals._cosmetic_exists(data[key]): - print("cosm ", data[key], " does not exist") - valid = false - else : - for c in data[key]: - if not Globals._cosmetic_exists(c): - print("cosm ", data[key], " does not exist") - valid = false - - if not valid: data = PlayerData.FALLBACK_COSM.duplicate() - - cosmetic_data = data - - for child in skeleton.get_children(): - if child.is_in_group("cosmetic"): child.queue_free() - - if data.empty(): return - - title.label = str(Network._get_username_from_id(owner_id)) - title.title = Globals.cosmetic_data[data["title"]]["file"].title - - face._setup_face(data) - var species_id = Globals.cosmetic_data[data.species]["file"].cos_internal_id - var species = _create_cosmetic(data["species"], species_id) - _create_cosmetic(data["undershirt"], species_id) - _create_cosmetic(data["overshirt"], species_id) - _create_cosmetic(data["legs"], species_id) - _create_cosmetic(data["hat"], species_id) - for misc in data["accessory"]: _create_cosmetic(misc, species_id) - - - var pattern = Globals.cosmetic_data[data["pattern"]]["file"] - body_mesh.material_override.set_shader_param("texture_albedo", pattern.body_pattern[0]) - - var primary_color = Globals.cosmetic_data[data["primary_color"]]["file"] - var secondary_color = Globals.cosmetic_data[data["secondary_color"]]["file"] if pattern.body_pattern[0] else Globals.cosmetic_data[data["primary_color"]]["file"] - body_mesh.material_override.set_shader_param("albedo", primary_color.main_color) - body_mesh.material_override.set_shader_param("albedo_secondary", secondary_color.main_color) - - - var tail_data = Globals.cosmetic_data[data["tail"]]["file"] - tail._load_tail(tail_data.mesh, primary_color.main_color) - - if species: - species.material_override.set_shader_param("albedo", primary_color.main_color) - species.material_override.set_shader_param("albedo_secondary", secondary_color.main_color) - - match data["species"]: - "species_cat": species.material_override.set_shader_param("texture_albedo", pattern.body_pattern[1]) - "species_dog": species.material_override.set_shader_param("texture_albedo", pattern.body_pattern[2]) - - if not data.keys().has("bobber") or data["bobber"] == "": data["bobber"] = "bobber_default" - var bobber_file = Globals.cosmetic_data[data["bobber"]]["file"] - bobber_mesh.mesh = bobber_file.mesh - bobber_mesh.set_surface_material(0, bobber_file.material) - if bobber_file.secondary_material: bobber_mesh.set_surface_material(1, bobber_file.secondary_material) - if bobber_file.third_material: bobber_mesh.set_surface_material(2, bobber_file.third_material) - -func _create_cosmetic(id, species_id = 0): - if id == "": return - if not Globals.cosmetic_data.keys().has(id): - print("Failed finding cosmetic: ", id) - return - print("Creating Cosmetic: ", id) - var data = Globals.cosmetic_data[id]["file"] - - if data.scene_replace: - var bone_attach = BoneAttachment.new() - skeleton.add_child(bone_attach) - bone_attach.bone_name = "pelvis" - bone_attach.add_to_group("cosmetic") - - var cosm = data.scene_replace.instance() - bone_attach.add_child(cosm) - return cosm - - var cosm = preload("res://Scenes/Entities/Player/cosmetic_node.tscn").instance() - cosm.mesh = data.mesh - if data.species_alt_mesh.size() > 0: - cosm.mesh = data.species_alt_mesh[species_id] - - cosm.skin = data.mesh_skin - - if not data.material: - cosm.material_override = preload("res://Assets/Shaders/player_skins.tres").duplicate() - cosm.material_override.set_shader_param("albedo", data.main_color) - cosm.material_override.set_shader_param("albedo_secondary", data.main_color) - cosm.material_override.set_shader_param("texture_albedo", null) - else : - - cosm.material_override = null - cosm.set_surface_material(0, data.material) - if data.secondary_material: cosm.set_surface_material(1, data.secondary_material) - - cosm.skeleton = skeleton.get_path() - skeleton.add_child(cosm) - return cosm - - - - - - - - - - - - - - - - - - -func _on_step_timer_timeout(): - if not controlled: return - - - if randf() < 0.03 * drunk_tier and drunk_wander_length <= 0: - drunk_wander_length = randi() % 25 + 5 - drunk_wander_dir = Vector3( - rand_range( - 1, 1), - 0.0, - rand_range( - 1, 1)) - - if state == STATES.FISHING_STRUGGLE: - _sync_particle("small_splash", bobber.global_transform.origin, true) - - if recent_reel > 0 and state == STATES.FISHING: - fishing_update.translation.z = - rod_cast_dist - fishing_update.force_raycast_update() - if fishing_update.is_colliding() and not fishing_update.get_collider().is_in_group("valid_water"): - _enter_state(STATES.FISHING_CANCEL) - - if velocity != Vector3.ZERO: - var vol = 1.6 - if sprinting: vol = 2.8 - elif sneaking: vol = 0.3 - - if is_on_floor() and sprinting: _sync_particle("dust_run", Vector3(0, - 0.95, 0)) - - if is_on_floor() and safe_check.is_colliding() and safe_check.get_collision_normal() == Vector3(0, 1, 0) and not gravity_disable: - if death_counter > 0: death_counter -= 1 - last_valid_pos = global_transform.origin - - PlayerData.player_saved_position = global_transform.origin - - - - - -func _message_sent(text): - - var split = text.split(" ") - for s in split: - match s: - ":(", ":[", "D:", ";_;", ";~;", ":C", ":c": _sync_face_emote("sad") - ":)", ":D", ":]": _sync_face_emote("love") - "xD", "!!!", "<3", "xd", "LMAO", "LOL": _sync_face_emote("happy") - ">:(", "D:<", ">:[": _sync_face_emote("angry") - ":/", ":|": _sync_face_emote("flat") - ":3", ":3c", ">:3c", ">:3": _sync_face_emote("cat") - "O.o", "!?!?", "?!?!", ":o", ":O": _sync_face_emote("surprised") - - var bubble = title._create_speech_bubble(text, PlayerData.voice_speed) - bubble.connect("_letter_said", self, "_sync_talk") - Network._send_actor_action(actor_id, "_sync_create_bubble", [text], false) - -func _sync_talk(letter): - var blacklist = ["a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l", "m", "n", "o", "p", "q", "r", "s", "t", "u", "v", "w", "x", "y", "z"] - if not blacklist.has(letter.to_lower()): return - - animation_data["talking"] = 0.4 - _talk(letter, PlayerData.voice_pitch) - Network._send_actor_action(actor_id, "_talk", [letter, PlayerData.voice_pitch], false) - -func _talk(letter, pitch = 1.5): - if not in_zone or not visible: return - - face._talk() - sound_manager._construct_voice(letter, "NewVoice", pitch) - -func _sync_face_emote(emotion): - _face_emote(emotion) - Network._send_actor_action(actor_id, "_face_emote", [emotion], false) - -func _sync_create_bubble(text): - title._create_speech_bubble(text) - -func _sync_level_bubble(text): - title._create_level_bubble(text) - $emotion_particles / lvl_particles.restart() - $emotion_particles / lvl_particles2.restart() - -func _face_emote(emotion): - face._emote(emotion, 2.4) - - - - - -func _sync_particle(id, offset = Vector3.ZERO, global = false): - _play_particle(id, offset, global) - Network._send_actor_action(actor_id, "_play_particle", [id, offset, global], false) - -func _play_particle(id, offset, global): - if not PARTICLE_DATA.keys().has(id): return - var p = PARTICLE_DATA[id].instance() - add_child(p) - - if not global: - p.translation = offset - else : - p.set_as_toplevel(true) - p.global_transform.origin = offset - - p.emitting = true - yield (get_tree().create_timer(p.lifetime + 0.1), "timeout") - p.queue_free() - - - - - -func _on_water_detect_area_entered(area): - if gravity_disable: return - if area.is_in_group("water"): - _kill() - -func _kill(skip_anim = false): - if not controlled: return - - death_counter += 1 - if death_counter >= 10: - if not skip_anim: PlayerData._send_notification("too many deaths in a row! sending to spawn...", 1) - else : PlayerData._send_notification("returning to spawn", 1) - - world._enter_zone("main_zone", - 1) - PlayerData.player_saved_zone = "main_zone" - PlayerData.player_saved_zone_owner = - 1 - last_valid_pos = world.map.spawn_position.global_transform.origin - death_counter = 0 - - if state == STATES.FISHING or state == STATES.FISHING_CAST or state == STATES.FISHING_CAST: - _enter_state(STATES.FISHING_CANCEL) - - cam_push = 0.0 - gravity_disable = true - - if not skip_anim: - _enter_animation("drown", true, true) - _sync_sfx("drown") - _sync_particle("splash", global_transform.origin, true) - - SceneTransition._fake_scene_change() - yield (SceneTransition, "_finished") - global_transform.origin = last_valid_pos + Vector3(0, 0.5, 0) - yield (get_tree().create_timer(0.3), "timeout") - gravity_disable = false - _enter_state(0) - _exit_animation() - - - - - -func _consume_item(id): - if state != STATES.DEFAULT: return - _enter_state(STATES.EMOTING) - - consume_on_state_change = held_item["ref"] - _enter_animation("drink", false, true, STATES.CONSUME_ITEM, true) - - var sfx = "drink" - - match id: - "growth": player_scale = clamp(player_scale + 0.4, 0.1, 100) - "shrink": player_scale = clamp(player_scale - 0.1, 0.1, 100) - "revert": - player_scale = 1.0 - drunk_timer = max(drunk_timer - 9000, 0) - sfx = "drink_nocap" - "speed": - boost_timer = 18000 - boost_amt = 1.3 - sfx = "drink_nocap" - "speed_burst": - boost_timer = 900 - boost_amt = 4.0 - sfx = "drink_nocap" - "catch": - catch_drink_timer = 18000 - catch_drink_boost = 1.15 - catch_drink_reel = 1.25 - catch_drink_xp = 1.0 - catch_drink_tier = 1 - catch_drink_gold_add = Vector2(1, 10) - catch_drink_gold_percent = 0.0 - "catch_big": - catch_drink_timer = 18000 - catch_drink_boost = 1.3 - catch_drink_reel = 1.45 - catch_drink_xp = 1.0 - catch_drink_tier = 2 - catch_drink_gold_add = Vector2(10, 50) - catch_drink_gold_percent = 0.25 - "catch_deluxe": - catch_drink_timer = 18000 - catch_drink_boost = 1.3 - catch_drink_reel = 1.45 - catch_drink_xp = 1.25 - catch_drink_tier = 3 - catch_drink_gold_add = Vector2(0, 0) - catch_drink_gold_percent = 0.0 - "beer": - drunk_timer += 9000 - drunk_timer = clamp(drunk_timer, 0, 50000) - "beer_big": - drunk_timer += 42000 - drunk_timer = clamp(drunk_timer, 0, 50000) - sfx = "drink_nocap" - - _sync_sfx(sfx) - - - -func _open_chest(rare = false): - if state != STATES.DEFAULT: return - - consume_on_state_change = held_item["ref"] - _enter_state(STATES.CONSUME_ITEM) - - var cosm = PlayerData._get_unowned_cosmetic() - if cosm != null: PlayerData._unlock_cosmetic(cosm) - else : - PlayerData._send_notification("You found 100 dollars inside the chest!") - PlayerData.money += 100 - -func _open_ringbox(): - if state != STATES.DEFAULT: return - - consume_on_state_change = held_item["ref"] - _enter_state(STATES.CONSUME_ITEM) - - if not PlayerData.cosmetics_unlocked.has("accessory_ring"): PlayerData._unlock_cosmetic("accessory_ring") - else : - PlayerData._send_notification("You already have a ring unlocked... Obtained $1000 instead.") - PlayerData.money += 1000 - - - -func _scratch_off(type): - _enter_state(STATES.GUITAR) - hud._open_minigame("scratch_off", {"type": type}) - var _win = yield (hud, "_minigame_finished") - yield (get_tree().create_timer(0.15), "timeout") - _enter_state(STATES.DEFAULT) - - consume_on_state_change = held_item["ref"] - _enter_state(STATES.CONSUME_ITEM) - - if not _win: - _sync_face_emote("angry") - _sync_sfx("rip") - else : - _sync_face_emote("happy") - - - - - -func _mushroom_bounce(): - if not controlled: return - ignore_snap = 10 - var bounce_horz = 16.0 if direction != Vector3.ZERO else 0.0 - var bounce_vert = 32.0 - gravity_vec = Vector3.ZERO - gravity_vec.x += (direction.normalized() * bounce_horz).x - gravity_vec.z += (direction.normalized() * bounce_horz).z - gravity_vec.y += bounce_vert - animation_data["mushroom"] = true - - - - - -func _sync_sfx(id, position = null, pitch = 1.0, delay = 0.0): - if not controlled: return - if delay > 0.0: yield (get_tree().create_timer(delay), "timeout") - - _play_sfx(id, position, pitch) - Network._send_actor_action(actor_id, "_play_sfx", [id, position, pitch], false) - -func _play_sfx(id, position = null, pitch = 1.0): - if not in_zone or not visible: return - sound_manager._play_sound(id, position, pitch) - print("playing sfx ", id) - -func _process_sounds(): - - - - - if $sound_manager / dive_scrape.playing != animation_data["dive_scrape"]: $sound_manager / dive_scrape.playing = animation_data["dive_scrape"] and in_zone - if $sound_manager / reel_slow.stream_paused == animation_data["reel_slow"]: $sound_manager / reel_slow.stream_paused = not animation_data["reel_slow"] and in_zone - if $sound_manager / reel_fast.playing != animation_data["reel_fast"]: $sound_manager / reel_fast.playing = animation_data["reel_fast"] and in_zone - - - - - -func _on_rain_timer_timeout(): - if not controlled: return - - var rain = false - for child in $raincloud_check.get_overlapping_areas(): - if child.is_in_group("rain_cloud"): - rain = true - - if rain != in_rain: - in_rain = rain - PlayerData.emit_signal("_rain_toggle", in_rain) - - if not in_rain: Network.set_rich_presence("#default") - else : Network.set_rich_presence("#rain") - - - - - -func _wag(): - animation_data["wagging"] = not animation_data["wagging"] - -func _paint(size, color): - if not controlled: return - - var in_zone = false - for area in $paint_node / Area.get_overlapping_areas(): - if area.is_in_group("canvas"): in_zone = true - if not in_zone: PlayerData._send_notification("mouse must be on a drawing zone!", 1) - - $paint_node.size = size - $paint_node.color = color - $paint_node.drawing = true - -func _paint_stop(): - if not controlled: return - $paint_node.drawing = false - PlayerData.emit_signal("_chalk_send") - -func _metal_detect_begin(): - pass - -func _metaldetect_update(): - if not controlled or held_item.id == "": - return - - var idata = Globals.item_data[held_item["id"]]["file"] - if not idata.detect_item: - return - - var alert_level = 0 - for b in $detection_zones / metal_detect_far.get_overlapping_bodies(): - if b.is_in_group("metal_spawn"): - if abs(b.global_transform.origin.y - global_transform.origin.y) > 3.5: continue - alert_level = b.global_transform.origin.distance_to(global_transform.origin) - - metal_detect_alert_level = alert_level - - if alert_level > 0: _metal_detect_beep() - -func _metal_detect_beep(): - var new_cd = ceil(max(metal_detect_alert_level * 0.5, 1)) - - if abs(new_cd - metal_detect_alert_cd) > 4: metal_detect_alert_cd = 0 - - metal_detect_alert_cd -= 1 - if metal_detect_alert_cd > 0: return - - metal_detect_alert_cd = new_cd - - var pitch = max(6.0 - metal_detect_alert_level * 0.3, 0.5) - - $metaldetect_dot.modulate.a = 1.0 - if metal_detect_flop: _sync_sfx("md_beep_slowb", null, pitch) - else : _sync_sfx("md_beep_slow", null, pitch) - metal_detect_flop = not metal_detect_flop - -func _on_metal_detect_consume_body_entered(b): - if not controlled or held_item.id == "": return - - var idata = Globals.item_data[held_item["id"]]["file"] - if not idata.detect_item: return - - if b.is_in_group("metal_spawn"): - b._reveal() - -func _on_image_update_timeout(): - if not controlled or dead_actor: return - _update_held_item(held_item) - Network._send_actor_action(actor_id, "_update_held_item", [held_item], false) - -func _real_step(run = false): - if anim_tree.get("parameters/dive/blend_amount") > 0.2 or anim_tree.get("parameters/movement/blend_amount") < 0.8: return - if not is_on_floor(): return - var sfx = "step" if not run else "step_run" - if boost_amt > 1.0 and boost_timer > 1: sfx = "step_fastrun" - _sync_sfx(sfx) - - - -func _play_guitar(): - _enter_state(STATES.GUITAR) - hud._open_minigame("guitar") - var _win = yield (hud, "_minigame_finished") - yield (get_tree().create_timer(0.15), "timeout") - _enter_state(STATES.DEFAULT) - -func _strum_guitar(string, fret, volume): - _sync_strum(string, fret, volume) - Network._send_actor_action(actor_id, "_sync_strum", [string, fret, volume]) - - animation_data["land"] = clamp(animation_data["land"] + 0.06, 0.0, 0.3) - _sync_face_emote("strum") - if randf() < 0.05: - _sync_particle("music") - -func _sync_strum(string, fret, volume): - if not in_zone or not visible: return - $guitar_sounds.get_child(string)._play_fret(fret, false, volume) - -func _hammer_string(string, fret): - _sync_hammer(string, fret) - Network._send_actor_action(actor_id, "_sync_hammer", [string, fret]) - -func _sync_hammer(string, fret): - if not in_zone or not visible: return - $guitar_sounds.get_child(string)._hammer_fret(fret) - -func _bark(): - if not controlled: return - var bark_id = "bark_dog" - bark_id = { - "species_cat": ["bark_cat", "growl_cat", "whine_cat"], - "species_dog": ["bark_dog", "growl_dog", "whine_dog"], - }[PlayerData.cosmetics_equipped.species] - - var type = 0 - if Input.is_action_pressed("move_sneak"): - _sync_face_emote("growl") - type = 1 - elif Input.is_action_pressed("move_walk"): - _sync_face_emote("whine") - type = 2 - else : - _sync_face_emote("bark") - - bark_id = bark_id[type] - _sync_sfx(bark_id) - -func _kiss(): - if not controlled: return - - animation_data["land"] = animation_data["land"] + 0.2 - _sync_face_emote("kiss") - _sync_sfx("kiss") - _sync_particle("kiss") - -func _punch(type = 0): - if not controlled or item_cooldown > 0: return - item_cooldown = 30 - - animation_data["land"] = animation_data["land"] + 1.0 - for b in $detection_zones / punch.get_overlapping_bodies(): - if b.is_in_group("player") and b != self and not b.controlled: - Network._send_P2P_Packet({"type": "player_punch", "from": global_transform.origin, "player": owner_id, "punch_type": type}, str(b.owner_id), 2) - - _sync_face_emote("punch") - _sync_sfx("punch") - _sync_punch() - Network._send_actor_action(actor_id, "_sync_punch") - -func _sync_punch(): - $emotion_particles / punch_particles.restart() - $emotion_particles / punchb_particles.restart() - -func _punched(from, type): - if not controlled: return - - if OptionsMenu.punchable: return - - var dir = (global_transform.origin - from).normalized() - - ignore_snap = 10 - var bounce_horz = 4.0 - var bounce_vert = 8.0 - - match type: - 0: - bounce_horz = 4.0 - bounce_vert = 8.0 - 1: - bounce_horz = 12.0 - bounce_vert = 24.0 - - gravity_vec = Vector3.ZERO - gravity_vec.x += (dir.normalized() * bounce_horz).x - gravity_vec.z += (dir.normalized() * bounce_horz).z - gravity_vec.y += bounce_vert - - _sync_face_emote("angry") - -func _tambourine(): - _sync_sfx("tambourine") - - animation_data["land"] = clamp(animation_data["land"] + 0.14, 0.0, 0.3) - _sync_face_emote("strum") - if randf() < 0.25: - _sync_particle("music") - -func _on_cosmetic_refresh_timeout(): - if not controlled: return - _refresh_cosmetics() - -func _refresh_cosmetics(): - if not controlled: return - yield (get_tree().create_timer(1.0), "timeout") - var new = PlayerData.cosmetics_equipped.duplicate() - if not dead_actor: Network._send_actor_action(actor_id, "_update_cosmetics", [new]) - -func _return_to_spawn(): - if not controlled: return - death_counter = 11 - _kill(true) - -func _item_removal(ref): - print("Sold Item") - if held_item.keys().has("ref") and held_item["ref"] == ref: _equip_item(PlayerData.FALLBACK_ITEM.duplicate())