From 80e1d9666d7c4cb6681cb2ceaac0242edf78b299 Mon Sep 17 00:00:00 2001 From: Xeight Date: Sun, 3 Nov 2024 22:53:21 +0500 Subject: [PATCH] c# rewrite --- .gitignore | 7 + 20241022214610_1.jpg | Bin 96601 -> 0 bytes PoppingBottles.sln | 16 + PoppingBottles/Config.cs | 7 + PoppingBottles/Mod.cs | 78 + PoppingBottles/PoppingBottles.csproj | 35 + PoppingBottles/manifest.json | 10 + README.md | 20 +- Scenes/Entities/Player/player-nyoomcompat.gd | 2343 ------------------ Scenes/Entities/Player/player.gd | 2343 ------------------ 10 files changed, 162 insertions(+), 4697 deletions(-) create mode 100644 .gitignore delete mode 100644 20241022214610_1.jpg create mode 100644 PoppingBottles.sln create mode 100644 PoppingBottles/Config.cs create mode 100644 PoppingBottles/Mod.cs create mode 100644 PoppingBottles/PoppingBottles.csproj create mode 100644 PoppingBottles/manifest.json delete mode 100644 Scenes/Entities/Player/player-nyoomcompat.gd delete mode 100644 Scenes/Entities/Player/player.gd 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 5e291a342fa2e76735d86f473438904a25c8f4dd..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 96601 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 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())