require 'core/appkit/lua/class' require 'core/appkit/lua/app' local SimpleProject = require 'core/appkit/lua/simple_project' local ComponentManager = require 'core/appkit/lua/component_manager' local UnitController = require 'core/appkit/lua/unit_controller' local UnitLink = require 'core/appkit/lua/unit_link' local Util = require 'core/appkit/lua/util' local CameraWrapper = require 'core/appkit/lua/camera_wrapper' require 'script/lua/input_mapper' local PlayerUtil = require 'script/lua/util' local PlayerHud = require 'script/lua/player_hud' -- on touch devices we will override the input_mapper in Appkit to use the Hud Controls if scaleform and Util.use_touch() then function Appkit.input_mapper:get_motion_input() return PlayerHud:get_motion_input() end end Project.Player = Appkit.class(Project.Player) local Player = Project.Player -- cache off for readability and speed -- cache off for readability and speed local Vector3 = stingray.Vector3 local Vector3Box = stingray.Vector3Box local Quaternion = stingray.Quaternion local QuaternionBox = stingray.QuaternionBox local Matrix4x4 = stingray.Matrix4x4 local Matrix4x4Box = stingray.Matrix4x4Box local Math = stingray.Math local Unit = stingray.Unit local Camera = stingray.Camera local World = stingray.World local Actor = stingray.Actor local Mover = stingray.Mover local PhysicsWorld = stingray.PhysicsWorld local Wwise = stingray.Wwise local WwiseWorld = stingray.WwiseWorld -- cache off move directions for readability local NORTH = 2 local SOUTH = 4 local EAST = 3 local WEST = 1 local NORTH_EAST = 2.5 local NORTH_WEST = 1.5 local SOUTH_EAST = 3.5 local SOUTH_WEST = 0.5 local free_cam_move_speed = Vector3Box(3.8, 3.8, 3.8) local free_cam_sprint_speed = Vector3Box(9, 9, 8) local free_cam_yaw_speed = 0.085 local free_cam_pitch_speed = 0.075 local character_walk_speed = Vector3Box(1.67, 1.67, 1.67) local character_walk_speed_in_air = Vector3Box(2.5, 2.5, 2.5) local character_run_speed = Vector3Box(6.13, 6.13, 6.13) local character_run_speed_in_air = Vector3Box(5.5, 5.5, 5.5) local character_sprint_speed = Vector3Box(10, 10, 10) local character_sprint_speed_in_air = Vector3Box(8, 8, 8) local character_jump_strength = 15 local character_max_slope_angle = math.rad(90) -- 1st person camera parameters local camera_positioning_1p = Vector3Box(-0.08, 0.0, 1.50) -- best for method 2 local character_cam_yaw_speed_1p = 0.3 local character_cam_pitch_speed_1p = 0.1 local camera_pitch_min_1p = -85 local camera_pitch_max_1p = 75 local fov_1p = math.rad(60) -- FOV of 90 for 1st person (standard) local fov_1p_aim = math.rad(35) local player_1p_aim_rate = 8.0 local head_bob_amp_1p_walk = 0.1 -- magnitude of overall bob while walking local head_bob_freq_walk = 3.8 -- frequency of bobbing per walking distance local head_bob_trav_mult_walk = 0.2 -- how much side to side bob relative to up-down bob, higher values bob side to side more local head_bob_amp_1p_sprint = 0.1 -- magnitude of overall bob while sprinting local head_bob_trav_mult_sprint = 0.2 -- how much side to side bob relative to up-down bob, higher values bob side to side more local head_bob_freq_sprint = 2.0 -- frequency of bobbing per walking distance -- 3rd person camera parameters local camera_stand_target = Vector3Box(0.3, 0, 1.3) local camera_crouch_target = Vector3Box(0.3, 0, 1.3) local camera_default_yaw = math.rad(-90) local camera_default_pitch = math.rad(85) local camera_pitch_min = math.rad(15) local camera_pitch_max = math.rad(105) local camera_yaw_speed = math.rad(10) local camera_pitch_speed = math.rad(6) local camera_pitch_aim_speed = math.rad(3) local character_walk_speed_3p = 1.67 local character_walk_speed_in_air_3p = 2.5 local character_run_speed_3p = 6.13 local character_run_speed_in_air_3p = 5.5 local character_sprint_speed_3p = 10 local character_sprint_speed_in_air_3p = 8 local character_cam_yaw_speed_3p = 0.1 local fov_3p = math.rad(45) local fov_3p_aim = 0.5 local character_rotation_blending_speed = 12 local camera_offset_blending_speed = 10 local camera_center_blending_speed = 10 local camera_fov_blending_speed = 10 local character_cam_offset_length_3p = 5 local character_cam_offset_length_3p_aim = 3.5 local wwise_world = nil ------------------------ Unit Controller ------------------------ local function compute_rotation(self, input, dt) local q_original = Unit.local_rotation(self.unit, 1) local m_original = Matrix4x4.from_quaternion(q_original) local aim_modifier = self.fov_modifier or 1 -- always need dt since unit_controller computes rotation by (new_rotation = previous_rotation + rotation_speed * dt) local q_identity = Quaternion.identity() local q_yaw = Quaternion(Vector3(0, 0, 1), -Vector3.x(input.pan) * self.yaw_speed * aim_modifier * dt) local q_pitch = Quaternion(Matrix4x4.x(m_original), -Vector3.y(input.pan) * self.pitch_speed * aim_modifier * dt) local q_frame = Quaternion.multiply(q_yaw, q_pitch) local q_new = Quaternion.multiply(q_frame, q_original) if self.pitch_min or self.pitch_max then local xyz = {Quaternion.to_euler_angles_xyz(q_new)} if self.pitch_min ~= nil and xyz[1] < self.pitch_min then xyz[1] = self.pitch_min elseif self.pitch_max ~= nil and xyz[1] > self.pitch_max then xyz[1] = self.pitch_max end q_new = Quaternion.from_euler_angles_xyz(xyz[1], xyz[2], xyz[3]) end return q_new end local function compute_translation(self, input, q, dt) local input_move = input.move local pose = Matrix4x4.from_quaternion(q) local pos = Unit.local_position(self.unit, 1) local mover = self.mover local frame_vel if mover then local active_gravity = self.active_gravity:unbox() local velocity = self.velocity:unbox() local gravity = active_gravity * dt self.velocity:store(velocity + gravity) frame_vel = velocity * dt + gravity else frame_vel = Vector3(0,0,0) end local frame_move = (Vector3.multiply_elements(input_move, self.translation_speed:unbox()) * dt) + frame_vel local move = Matrix4x4.transform(pose, frame_move) if mover then Mover.move(mover, move, dt) if Mover.standing_frames(mover) > 0 then local velocity = self.velocity:unbox() velocity.z = 0 self.velocity:store(velocity) end return Mover.position(mover) else return Unit.local_position(self.unit, 1) + move end end function UnitController:update(dt) if not self.is_enabled then return end local input = self.input_mapper:get_motion_input() local q = compute_rotation(self, input, dt) local p = compute_translation(self, input, q, dt) local unit = self.unit Unit.set_local_rotation(unit, 1, q) Unit.set_local_position(unit, 1, p) end ------------------------ Unit Controller ------------------------ ------------------------ SFX ------------------------ local function play_spawn_sound() if Wwise then wwise_world = wwise_world or Wwise.wwise_world(SimpleProject.world) WwiseWorld.trigger_event(wwise_world, "sfx_spawn_sound") end end local function play_ambient_sound() if Wwise then wwise_world = wwise_world or Wwise.wwise_world(SimpleProject.world) WwiseWorld.trigger_event(wwise_world, "sfx_amb_warehouse") end end local function play_fire_sound() if Wwise then wwise_world = wwise_world or Wwise.wwise_world(SimpleProject.world) WwiseWorld.trigger_event(wwise_world, "sfx_ballgun_fire") end end ------------------------ SFX ------------------------ ------------------------ Helper Function ------------------------ -- return a random unit vector3 -- local function random_unit_vector3() -- return Vector3.normalize(Vector3(Math.random() - 0.5, Math.random() - 0.5, Math.random() - 0.5)) -- end -- return a new quaternion containing only euler z-axis rotation local function get_rotation_z(rotation) local _, _, euler_z = Quaternion.to_euler_angles_xyz(rotation) return Quaternion(Vector3.up(), math.rad(euler_z)) end -- return if character is in air local function is_character_in_air(land_character) local unit_controller = UnitController.manager:get(land_character) return unit_controller and unit_controller.velocity.z ~= 0 end -- set parameters from unit data local function set_unit_script_data_values(character_unit) local walk_speed = Unit.get_data(character_unit, "walk speed") local run_speed = Unit.get_data(character_unit, "run speed") local sprint_speed = Unit.get_data(character_unit, "sprint speed") local air_speed = Unit.get_data(character_unit, "air control") local jump_strength = Unit.get_data(character_unit, "jump strength") if walk_speed then character_walk_speed:store(walk_speed, walk_speed, walk_speed) character_walk_speed_3p = walk_speed else print('Unit walk speed not set. Using default value.') end if run_speed then character_run_speed:store(run_speed, run_speed, run_speed) character_run_speed_3p = run_speed else print('Unit run speed not set. Using default value.') end if sprint_speed then character_sprint_speed:store(sprint_speed, sprint_speed, sprint_speed) character_sprint_speed_3p = sprint_speed else print('Unit sprint speed not set. Using default value.') end if air_speed then character_walk_speed_in_air:store(air_speed, air_speed, air_speed) character_walk_speed_in_air_3p = air_speed else print('Unit air control not set. Using default value.') end if jump_strength then character_jump_strength = jump_strength else print('Unit jump strength not set. Using default value.') end end local function compute_move_direction(motion_direction) if(motion_direction == nil) then print("Error compute_move_direction received nil motion_direction") return nil end local norm_motion_dir = Vector3.normalize(motion_direction) local x_angle = math.acos(Vector3.dot(norm_motion_dir, Vector3.right())) local y_angle = math.acos(Vector3.dot(norm_motion_dir, Vector3.forward())) local thresh = math.rad(15) local move_direction if y_angle <= thresh then move_direction = NORTH -- forward elseif y_angle >= math.pi - thresh then move_direction = SOUTH -- back elseif x_angle <= thresh then move_direction = EAST -- right elseif x_angle>= math.pi - thresh then move_direction = WEST -- left elseif x_angle > thresh and x_angle < math.pi/2 and y_angle > thresh and y_angle < math.pi/2 then move_direction = NORTH_EAST -- forward, right elseif x_angle > math.pi/2 and x_angle < math.pi - thresh and y_angle > thresh and y_angle < math.pi/2 then move_direction = NORTH_WEST -- forward, left elseif x_angle > thresh and x_angle < math.pi/2 and y_angle > math.pi/2 and y_angle < math.pi-thresh then move_direction = SOUTH_EAST -- back, right elseif x_angle > math.pi/2 and x_angle < math.pi - thresh and y_angle > math.pi/2 and y_angle < math.pi - thresh then move_direction = SOUTH_WEST -- back, left else print("Error in compute_move_direction:\nmotion_direction", motion_direction, "\nx_angle", x_angle, "\ny_angle", y_angle) end return move_direction end local function blend_to_move_direction(cur, targ, dt) local full_range = 4 local halfrange = full_range * 0.5 local delta_direction = targ - cur local abs_dir = math.abs(delta_direction) local rotation_direction if delta_direction >= 0 and abs_dir <= halfrange then rotation_direction = 1 elseif delta_direction >= 0 and abs_dir > halfrange then rotation_direction = -1 elseif delta_direction < 0 and abs_dir <= halfrange then rotation_direction = -1 elseif delta_direction < 0 and abs_dir > halfrange then rotation_direction = 1 end local rotation_rate = 3 -- blend speed local abs_frame_rotation_amt = rotation_rate * dt local frame_rotation_amt = abs_frame_rotation_amt * rotation_direction local new_val = cur + frame_rotation_amt if new_val > full_range then new_val = new_val - full_range elseif new_val < 0 then new_val = new_val + full_range end if (math.abs(new_val - targ) <= abs_frame_rotation_amt) or (full_range - math.abs(new_val - targ) <= abs_frame_rotation_amt) then new_val = targ end return new_val end ------------------------ Helper Function ------------------------ ------------------------ Spawn Function ------------------------ local function spawn_freecam_player(self, level, view_position, view_rotation) view_position = view_position or Vector3(0, 0, 2) view_rotation = view_rotation or Quaternion.identity() local player_camera = self.player_camera player_camera.unit = SimpleProject.camera_unit -- camera local camera_wrapper = CameraWrapper(player_camera, player_camera.unit, 1) camera_wrapper:set_local_position(view_position) camera_wrapper:set_local_rotation(view_rotation) camera_wrapper:enable() -- add camera input movement, starts enabled local unit_controller = UnitController(player_camera, player_camera.unit, Appkit.input_mapper) unit_controller:set_move_speed(free_cam_move_speed:unbox()) unit_controller:set_yaw_speed(free_cam_yaw_speed) unit_controller:set_pitch_speed(free_cam_pitch_speed) -- give free cam ability to attached to character for walking mode, starts disabled local unit_link = UnitLink(player_camera, level, player_camera.unit, 1, nil, 1, false) self.is_freecam_mode = true end local function spawn_player_character(self, pose) local land_character = self.land_character land_character.unit = World.spawn_unit(SimpleProject.world, "content/models/character/skm_chr_genrig_3p", pose) -- set up parent of link without actually linking for 3p controller local unit_link = UnitLink.manager:get(self.player_camera) unit_link:set_parent(land_character.unit, 1) -- initialize unit controller for character local controller = UnitController(land_character, land_character.unit, Appkit.input_mapper) controller:set_gravity_enabled(true) controller:set_mover("body") Mover.set_max_slope_angle(controller.mover, character_max_slope_angle) -- set the movement parameters according to the units script data values set_unit_script_data_values(land_character.unit) if Unit.has_visibility_group(land_character.unit, "first person only meshes") then Unit.set_visibility(land_character.unit, "first person only meshes", false) end end local function spawn_player_weapon(self, character_unit) local player_weapon = self.player_weapon player_weapon.unit = World.spawn_unit(SimpleProject.world, "content/models/weapon/foam_gun") -- link weapon to character local unit_link = UnitLink(player_weapon, SimpleProject.level, player_weapon.unit, 1, nil, 1, false) local node_index = Unit.node(character_unit, "s_genrig_RightHandAttach") unit_link:set_parent(character_unit, node_index) unit_link:link() -- set orientation local rotation = Quaternion.from_euler_angles_xyz(180, 0, 180) Unit.set_local_rotation(player_weapon.unit, 1, rotation) end ------------------------ Spawn Function ------------------------ ------------------------ Despawn Function ------------------------ local function cleanup_projectiles(self) for _, unit in pairs(self.spawn_projs) do if unit then ComponentManager.remove_components(unit) World.destroy_unit(SimpleProject.world, unit) end end self.spawn_projs = {} end local function despawn_weapon(self) local player_weapon = self.player_weapon if player_weapon.unit then ComponentManager.remove_components(player_weapon) World.destroy_unit(SimpleProject.world, player_weapon.unit) player_weapon.unit = nil player_weapon = {} end end local function despawn_character_1p(self) if self.unit_1p then World.destroy_unit(SimpleProject.world, self.unit_1p) self.unit_1p = nil end end local function despawn_character(self) local land_character = self.land_character if land_character.unit then ComponentManager.remove_components(land_character) World.destroy_unit(SimpleProject.world, land_character.unit) land_character.unit = nil land_character = {} end end local function despawn_freecam(self) local player_camera = self.player_camera ComponentManager.remove_components(player_camera) player_camera.unit = nil player_camera = {} end ------------------------ Despawn Function ------------------------ ------------------------ 3P Controller Function ------------------------ local function set_character_walking_3p(self) local character_controller = UnitController.manager:get(self.land_character) character_controller:set_move_speed(Vector3(0, 0, 0)) character_controller:set_yaw_speed(0) -- hide crosshair PlayerHud:set_crosshair_visibility(false) end local function set_character_aiming_3p(self) local character_controller = UnitController.manager:get(self.land_character) character_controller:set_move_speed(character_run_speed:unbox()) character_controller:set_yaw_speed(character_cam_yaw_speed_3p) -- show crosshair PlayerHud:set_crosshair_visibility(true) end local function update_camera_offset(self, yaw, pitch, is_blending) local player_camera = self.player_camera if yaw == nil or pitch == nil then local camera_offset = player_camera.offset:unbox() local current_offset_length = Vector3.length(camera_offset) yaw = yaw or math.atan2(camera_offset.y, camera_offset.x) pitch = pitch or math.acos(camera_offset.z / current_offset_length) end local sin_yaw = math.sin(yaw) local cos_yaw = math.cos(yaw) local sin_pitch = math.sin(pitch) local cos_pitch = math.cos(pitch) local x = player_camera.offset_length * cos_yaw * sin_pitch local y = player_camera.offset_length * sin_yaw * sin_pitch local z = player_camera.offset_length * cos_pitch if is_blending then self.camera_blend_offset_target = Vector3Box(x, y, z) else self.player_camera.offset = Vector3Box(x, y, z) end end local function center_camera(self, is_blending) local player_camera = self.player_camera local land_unit = self.land_character.unit local land_rotation = Unit.world_rotation(land_unit, 1) local euler_x, euler_y, euler_z = Quaternion.to_euler_angles_xyz(land_rotation) local yaw = camera_default_yaw + math.rad(euler_z) local pitch = camera_default_pitch update_camera_offset(self, yaw, pitch, is_blending) end ------------------------ 3P Controller Function ------------------------ ------------------------ 1P Controller Function ------------------------ local function set_1p_stand_position(self) -- assuming camera linked to character and unit_1p mesh linked to camera local camera_unit = self.player_camera.unit -- set the camera to the character unit head position local camera_offset = camera_positioning_1p:unbox() local camera_position = Unit.local_position(camera_unit, 1) local camera_blend_position = Vector3.lerp(camera_offset, camera_position, 0.5) Unit.set_local_position(camera_unit, 1, camera_blend_position) -- set the unit_1p mesh to character unit position local mesh_1p_position = -camera_positioning_1p:unbox() Unit.set_local_position(self.unit_1p, 1, mesh_1p_position) end local function get_aim_modifier(self) local camera_wrapper = CameraWrapper.manager:get(self.player_camera) local fov = Camera.vertical_fov(camera_wrapper.camera) local modifier = 1 if self.is_firstperson then if self.is_aim then modifier = fov / fov_1p end else if self.is_aim then modifier = fov / fov_3p end end return modifier end local function update_head_bob(self, dt, is_running) local cam_unit = self.player_camera.unit local land_unit = self.land_character.unit local cam_offset = camera_positioning_1p:unbox() local input = Appkit.input_mapper:get_motion_input() local speed = Vector3.length(input.move) if speed < 0.01 then Unit.set_local_position(cam_unit, 1, Vector3.lerp(cam_offset, Unit.local_position(cam_unit, 1), 0.5)) -- smooth it with lerp return end local move_direction = compute_move_direction(input.move) if started_moving then self.current_direction = move_direction else self.current_direction = blend_to_move_direction(self.current_direction, move_direction, dt) end local move_speed = 0 local bob_frequency = 0 local bob_trans_mult = 0 local bob_amp = 0 if is_running then move_speed = character_sprint_speed.x bob_frequency = head_bob_freq_sprint bob_trans_mult = head_bob_trav_mult_sprint bob_amp = head_bob_amp_1p_sprint else move_speed = character_walk_speed.x bob_frequency = head_bob_freq_walk bob_trans_mult = head_bob_trav_mult_walk bob_amp = head_bob_amp_1p_walk end if self.current_direction == 1 or self.current_direction == 3 then -- purely strafing to left or right, or not moving return else -- walking somewhat forward or somewhat backward self.walk_distance = self.walk_distance + speed * move_speed * dt -- assumes isotropic speed end local x_head = bob_trans_mult * bob_amp * math.sin(3 / 2 * bob_frequency * self.walk_distance) + cam_offset.x local y_head = cam_offset.y local z_head = bob_amp * math.abs(math.sin(bob_frequency * self.walk_distance)) + cam_offset.z local cameraLocalPos = Vector3(x_head, y_head, z_head) Unit.set_local_position(cam_unit, 1, Vector3.lerp(cameraLocalPos, Unit.local_position(cam_unit, 1), 0.5)) -- smooth it with lerp end ------------------------ 1P Controller Function ------------------------ ------------------------ Enable Function ------------------------ local function enable_third_person_mode(self) local land_character = self.land_character local player_camera = self.player_camera -- turn on the visiblity of the land unit Unit.set_unit_visibility(land_character.unit, true) despawn_character_1p(self) despawn_weapon(self) spawn_player_weapon(self, land_character.unit) -- set 3p FOV local camera = Unit.camera(player_camera.unit, "camera") Camera.set_vertical_fov(camera, fov_3p) Camera.set_near_range(camera, 0.1) -- default is 0.1 -- attach camera to character player_camera.offset_length = character_cam_offset_length_3p set_character_walking_3p(self) local unit_link = UnitLink.manager:get(player_camera) if unit_link.is_linked then unit_link:unlink() end player_camera.target_offset = Vector3Box(camera_stand_target:unbox()) center_camera(self, false) local freecam_controller = UnitController.manager:get(player_camera) freecam_controller:set_pitch_speed(0) local character_controller = UnitController.manager:get(land_character) character_controller:set_move_speed(Vector3(0, 0, 0)) character_controller:set_yaw_speed(0) character_controller:set_pitch_speed(0) self.is_firstperson = false end local function enable_first_person_mode(self) local land_unit = self.land_character.unit local player_camera = self.player_camera -- attach camera to character local unit_link = UnitLink.manager:get(player_camera) unit_link:link() -- reset character pose Unit.set_local_pose(land_unit, 1, Matrix4x4.identity()) Unit.set_local_pose(player_camera.unit, 1, Matrix4x4.identity()) -- offset camera to 1p view local camera_wrapper = CameraWrapper.manager:get(player_camera) local pose_offset = Matrix4x4.identity() Matrix4x4.set_translation(pose_offset, camera_positioning_1p:unbox()) camera_wrapper:set_local_pose(pose_offset) -- set 1p FOV and near/far distances local camera = Unit.camera(player_camera.unit, "camera") Camera.set_vertical_fov(camera, fov_1p) Camera.set_near_range(camera, 0.01) -- default is 0.1 -- turn off existing char mesh and make new one linked as child to camera, does not use the aim animation to rotate with view -- remove the current weapon despawn_weapon(self) -- turn off the visiblity of the nominal char unit Unit.set_unit_visibility(land_unit, false) -- completely constrain the character controller to only control translation local character_controller = UnitController.manager:get(self.land_character) character_controller:set_yaw_speed(character_cam_yaw_speed_1p) character_controller:set_pitch_speed(0) -- freecam controller will control all viewport rotations local freecam_controller = UnitController.manager:get(player_camera) freecam_controller.pitch_min = camera_pitch_min_1p freecam_controller.pitch_max = camera_pitch_max_1p freecam_controller:set_pitch_speed(character_cam_pitch_speed_1p) freecam_controller:set_yaw_speed(0) -- spawn a new unit that only shows arms and have it rotate passively with camera rotation via UnitLink local unit_1p = World.spawn_unit(SimpleProject.world, "content/models/character/skm_chr_genrig_3p", Unit.local_pose(player_camera.unit, 1)) Unit.set_animation_state_machine(unit_1p, "content/animations/1p_character/skm_chr_genrig_1p") self.unit_1p = unit_1p if Unit.has_visibility_group(unit_1p, "third person only meshes") then Unit.set_visibility(unit_1p, "third person only meshes", false) end -- link the first person mesh as a child to the camera unit UnitLink(unit_1p, SimpleProject.level, unit_1p, 1, player_camera.unit, 1) -- will automatically link true -- offset 1p mesh so that the head lines up with the camera (opposite of camera offset above) Unit.set_local_position(unit_1p, 1, -camera_positioning_1p:unbox()) set_1p_stand_position(self) -- spawn character weapon for new 1p mesh spawn_player_weapon(self, unit_1p) -- show crosshair PlayerHud:set_crosshair_visibility(true) self.is_firstperson = true end local function enable_walk_mode(self) local player_camera = self.player_camera -- save off freecam camera position local camera_wrapper = CameraWrapper.manager:get(player_camera) self.saved_freecam_pose:store(camera_wrapper:local_pose()) -- disable movement and yaw on camera local freecam_controller = UnitController.manager:get(player_camera) freecam_controller:set_move_speed(Vector3(0, 0, 0)) freecam_controller:set_yaw_speed(0) -- spawn character local player_start_pose = Appkit.PlayerUtil.get_player_start_pose(SimpleProject.world) or Matrix4x4.from_translation(Vector3(0, 0, 5)) spawn_player_character(self, player_start_pose) self.is_freecam_mode = false enable_third_person_mode(self) end local function enable_free_mode(self) local player_camera = self.player_camera -- detach camera from character local unit_link = UnitLink.manager:get(player_camera) if unit_link.is_linked then unit_link:unlink() end despawn_weapon(self) despawn_character(self) despawn_character_1p(self) -- enable camera movement and yaw local camera_controller = UnitController.manager:get(player_camera) camera_controller:set_move_speed(free_cam_move_speed:unbox()) camera_controller:set_yaw_speed(free_cam_yaw_speed) camera_controller:set_pitch_speed(free_cam_pitch_speed) camera_controller.pitch_min = nil camera_controller.pitch_max = nil -- set freecam to saved off camera position local camera_wrapper = CameraWrapper.manager:get(player_camera) camera_wrapper:set_local_pose(self.saved_freecam_pose:unbox()) -- hide crosshair PlayerHud:set_crosshair_visibility(false) self.is_freecam_mode = true self.is_firstperson = false end ------------------------ Enable Function ------------------------ ------------------------ Check Function ------------------------ local function check_camera_mode(self) if Appkit.input_mapper:if_toggle_camera() or PlayerHud:check_toggle_camera() then if self.is_freecam_mode then enable_walk_mode(self) else enable_free_mode(self) end end end local function check_first_person_mode(self) if Appkit.input_mapper:if_switch_perspective() then if self.is_firstperson then enable_third_person_mode(self) else enable_first_person_mode(self) end end end local function check_sprint(self) self.is_sprinting = Appkit.input_mapper:if_run() or PlayerHud:check_sprint() if not self.is_freecam_mode then local input = Appkit.input_mapper:get_motion_input() if self.is_firstperson then self.is_sprinting = self.is_sprinting and not self.is_aim and (input.move.y > 0.9 and math.abs(input.move.x) < 0.5) else self.is_sprinting = self.is_sprinting and not self.is_aim and Vector3.length(input.move) > 0.9 end end end local function check_moving(self) local input = Appkit.input_mapper:get_motion_input() local speed = Vector3.length(input.move) if self.is_moving == false and speed > 0 then self.started_moving = true else self.started_moving = false end self.is_moving = speed > 0 end local function check_move_speed(self) --TODO: refactor so this only does sprint check. Get speed in another function if self.is_freecam_mode then local unit_controller = UnitController.manager:get(self.player_camera) if self.is_sprinting then unit_controller:set_move_speed(free_cam_sprint_speed:unbox()) else unit_controller:set_move_speed(free_cam_move_speed:unbox()) end else local land_character = self.land_character local unit_controller = UnitController.manager:get(land_character) if self.is_firstperson then if self.is_sprinting then if not is_character_in_air(land_character) then unit_controller:set_move_speed(character_sprint_speed:unbox()) self.jumped_while_sprinting = false elseif self.jumped_while_sprinting then unit_controller:set_move_speed(character_sprint_speed_in_air:unbox()) else unit_controller:set_move_speed(character_walk_speed_in_air:unbox()) end else if not is_character_in_air(land_character) then unit_controller:set_move_speed(character_run_speed:unbox()) self.jumped_while_sprinting = false elseif self.jumped_while_sprinting then unit_controller:set_move_speed(character_sprint_speed_in_air:unbox()) else unit_controller:set_move_speed(character_run_speed_in_air:unbox()) end end else if self.is_sprinting then if not is_character_in_air(land_character) then self.character_speed_3p = character_sprint_speed_3p self.jumped_while_sprinting = false elseif self.jumped_while_sprinting then self.character_speed_3p = character_sprint_speed_in_air_3p else self.character_speed_3p = character_run_speed_in_air_3p end else if not is_character_in_air(land_character) then if not self.is_crouching then local input = Appkit.input_mapper:get_motion_input() self.character_speed_3p = character_walk_speed_3p + Vector3.length(input.move) * (character_run_speed_3p - character_walk_speed_3p) else self.character_speed_3p = character_walk_speed_3p end self.jumped_while_sprinting = false elseif self.jumped_while_sprinting then self.character_speed_3p = character_sprint_speed_in_air_3p else self.character_speed_3p = character_run_speed_in_air_3p end end end end end local function check_aim(self, dt) local aim = Appkit.input_mapper:if_aim() or PlayerHud:check_aim() if self.is_aim ~= aim then if self.is_firstperson then if aim then self.camera_blend_fov_target = fov_1p_aim else self.camera_blend_fov_target = fov_1p end else local player_camera = self.player_camera if aim then self.camera_blend_fov_target = fov_3p_aim -- blend camera offset local camera_offset = player_camera.offset:unbox() local current_offset_length = Vector3.length(camera_offset) local current_yaw = math.atan2(camera_offset.y, camera_offset.x) local current_pitch = math.acos(camera_offset.z / current_offset_length) player_camera.offset_length = character_cam_offset_length_3p_aim update_camera_offset(self, current_yaw, current_pitch, true) -- blend character rotation local camera_wrapper = CameraWrapper.manager:get(player_camera) local camera_rotation = camera_wrapper:world_rotation() local camera_rotation_z = get_rotation_z(camera_rotation) self.character_blend_target = QuaternionBox(camera_rotation_z) set_character_aiming_3p(self) else self.camera_blend_fov_target = fov_3p player_camera.offset_length = character_cam_offset_length_3p set_character_walking_3p(self) end end end self.is_aim = aim local controller = UnitController.manager:get(self.land_character) controller.fov_modifier = get_aim_modifier(self) end local function check_projectile_fire(self) -- retrun if 3p non-aim mode if not self.is_firstperson and not self.is_aim then return end -- check input local is_firing = Appkit.input_mapper:if_fire() or PlayerHud:check_fire() if not is_firing then return end -- return if firing too fast local world = SimpleProject.world local current_time = world:time() local time_difference = current_time - self.last_fire_time if time_difference < 0.075 then return end -- the projectile spawn position is based off a fire node in the weapon unit, which inextricably links the projectile spawning to the weapon animation local player_weapon = self.player_weapon local weapon_unit = player_weapon.unit local fire_node_id = Unit.node(weapon_unit, "weapon_fire_node") local fire_node_position = Unit.world_position(weapon_unit, fire_node_id) local player_camera = self.player_camera local camera_wrapper = CameraWrapper.manager:get(player_camera) local source_pose = camera_wrapper:world_pose() local source_forward = Matrix4x4.forward(source_pose) -- here we'd then use our offset position. right now it's still using the old fire node position, though local proj_pose = Matrix4x4.from_translation(fire_node_position + source_forward * 0.1) local proj_unit = World.spawn_unit(world, "content/models/props/foam_projectile", proj_pose) if proj_unit then local physics_world = World.physics_world(world) local source_position = camera_wrapper:world_position() local hit_found, hit_position, hit_distance, hit_normal, hit_actor = PhysicsWorld.raycast(physics_world, source_position, source_forward, "collision_filter", "aim") -- get target position local minimum_distance = Vector3.distance(fire_node_position, source_position) local target_position = source_position + source_forward * 20 if hit_found and hit_distance > minimum_distance then target_position = hit_position end -- compute impulse local proj_forward = Vector3.normalize(target_position - fire_node_position) * 750 proj_forward = proj_forward + Vector3(0, 0, 10) -- add tiny bit of up to combat gravity vs cursor -- add noise to impulse local noise_vector = PlayerUtil.random_unit_vector3() local noise_scalar = 1 / time_difference if self.is_aim == false then noise_scalar = noise_scalar * 2 end proj_forward = proj_forward + noise_vector * noise_scalar -- apply impulse local proj_act = Unit.actor(proj_unit, "foam_projectile") Actor.add_impulse(proj_act, proj_forward) play_fire_sound() local spawn_projs = self.spawn_projs spawn_projs[current_time] = proj_unit self.last_fire_time = current_time else print("did not spawn projectile") end end local function check_crouch(self) -- return if character is in air, could be taken out if there is an additional animation for crouch in air if is_character_in_air(self.land_character) then return end local is_crouching = Appkit.input_mapper:if_crouch() or PlayerHud:check_crouch() if self.is_firstperson == false and self.is_crouching ~= is_crouching then if is_crouching then self.camera_blend_center_target = camera_crouch_target else self.camera_blend_center_target = camera_stand_target end end self.is_crouching = is_crouching end local function check_jump(self) local is_jumping = Appkit.input_mapper:if_jump() or PlayerHud:check_jump() if is_jumping then local land_character = self.land_character if not is_character_in_air(land_character) then local controller = UnitController.manager:get(land_character) controller:add_impulse(Vector3(0, 0, character_jump_strength)) if self.is_sprinting then self.jumped_while_sprinting = true end end end end -- for 3rd person controller local function check_move_character(self, dt) -- return if aim mode if self.is_aim then return end local input = Appkit.input_mapper:get_motion_input() -- return if no move input if Vector3.equal(input.move, Vector3.zero()) then return end local player_camera = self.player_camera local camera_wrapper = CameraWrapper.manager:get(player_camera) local camera_rotation = camera_wrapper:world_rotation() local camera_rotation_z = get_rotation_z(camera_rotation) local move_direction = Quaternion.rotate(camera_rotation_z, Vector3.normalize(input.move)) local target_rotation = Quaternion.look(move_direction) -- set target rotation to blend self.character_blend_target = QuaternionBox(target_rotation) -- translate character local land_mover = Unit.mover(self.land_character.unit) -- walk_speed + input local move_offset = move_direction * self.character_speed_3p * dt Mover.move(land_mover, move_offset, dt) -- update camera offset local camera_offset = player_camera.offset:unbox() - move_offset Vector3Box.store(player_camera.offset, camera_offset) end -- for 3rd person controller local function check_center_camera(self) -- return if aim mode if self.is_aim then return end -- TODO: support touch if Appkit.input_mapper:if_center_camera() then center_camera(self, true) end end -- for 3rd person controller local function check_rotate_camera(self, dt) local input = Appkit.input_mapper:get_motion_input() -- return if no pan input if Vector3.equal(input.pan, Vector3.zero()) then return end local camera_offset = self.player_camera.offset:unbox() local current_offset_length = Vector3.length(camera_offset) local current_yaw = math.atan2(camera_offset.y, camera_offset.x) local current_pitch = math.acos(camera_offset.z / current_offset_length) local yaw_speed = camera_yaw_speed local pitch_speed if self.is_aim then pitch_speed = camera_pitch_aim_speed else pitch_speed = camera_pitch_speed end local desired_yaw = current_yaw - input.pan.x * yaw_speed * dt local desired_pitch = current_pitch - input.pan.y * pitch_speed * dt -- boundry check if desired_pitch < camera_pitch_min then desired_pitch = camera_pitch_min elseif desired_pitch > camera_pitch_max then desired_pitch = camera_pitch_max end if self.is_aim then local land_unit = self.land_character.unit local land_rotation = Unit.world_rotation(land_unit, 1) local euler_x, euler_y, euler_z = Quaternion.to_euler_angles_xyz(land_rotation) desired_yaw = math.rad(euler_z - 90) end update_camera_offset(self, desired_yaw, desired_pitch, false) end local function check_exit_level(self) if Appkit.input_mapper:if_exit_level() or PlayerHud:exit_level() then SimpleProject.change_level(Project.level_names.menu) end end ------------------------ Check Function ------------------------ ------------------------ Update Function ------------------------ local function update_character_blend(self, dt) if self.character_blend_target == nil then return end local land_unit = self.land_character.unit local source = Unit.local_rotation(land_unit, 1) local target = self.character_blend_target:unbox() local ratio = dt * character_rotation_blending_speed if ratio > 1 then ratio = 1 end local blend = Quaternion.lerp(source, target, ratio) -- if blend is close to target enough if Quaternion.dot(blend, target) > 0.99 then blend = target end Unit.set_local_rotation(land_unit, 1, blend) -- blending is done if Quaternion.equal(blend, target) then self.character_blend_target = nil end end local function update_camera_blend(self, dt) local player_camera = self.player_camera local camera_wrapper = CameraWrapper.manager:get(player_camera) -- blend camera offset if self.camera_blend_offset_target then local source = player_camera.offset:unbox() local target = self.camera_blend_offset_target:unbox() local ratio = dt * camera_offset_blending_speed if ratio > 1 then ratio = 1 end local blend = Vector3.lerp(source, target, ratio) -- if blend is close to target enough if Vector3.distance_squared(blend, target) < 0.01 then blend = target end Vector3Box.store(player_camera.offset, blend) -- blending is done if Vector3.equal(blend, target) then self.camera_blend_offset_target = nil end end -- blend camera target offset (standing or crouching) if self.camera_blend_center_target then local source = player_camera.target_offset:unbox() local target = self.camera_blend_center_target:unbox() local ratio = dt * camera_center_blending_speed if ratio > 1 then ratio = 1 end local blend = Vector3.lerp(source, target, ratio) -- if blend is close to target enough if Vector3.distance_squared(blend, target) < 0.01 then blend = target end Vector3Box.store(player_camera.target_offset, blend) -- blending is done if Vector3.equal(blend, target) then self.camera_blend_center_target = nil end end -- blend camera fov if self.camera_blend_fov_target then local source = Camera.vertical_fov(camera_wrapper.camera) local target = self.camera_blend_fov_target local ratio = dt * camera_fov_blending_speed if ratio > 1 then ratio = 1 end local blend = (1 - ratio) * source + ratio * target -- if blend is close to target enough if math.abs(blend - target) < 0.01 then blend = target end Camera.set_vertical_fov(camera_wrapper.camera, blend) -- blending is done if blend == target then self.camera_blend_fov_target = nil end end end local function set_aim_target_position(self) local character_unit = self.land_character.unit local aim_pose if self.is_aim or self.is_firstperson then local player_camera = self.player_camera local camera_wrapper = CameraWrapper.manager:get(player_camera) aim_pose = camera_wrapper:world_pose() else aim_pose = Unit.world_pose(character_unit, 1) end local target = Matrix4x4.translation(aim_pose) + Matrix4x4.forward(aim_pose) * 1000 -- this number controls how many meters in the front of the camera the focal point is Unit.animation_set_constraint_target(character_unit, 1, target) end local function animate_character_3p(self, dt) -- movement animation local input = Appkit.input_mapper:get_motion_input() local speed = Vector3.length(input.move) local land_character = self.land_character local land_unit = land_character.unit if self.is_aim then Unit.animation_event(land_unit, "aim_up") else Unit.animation_event(land_unit, "aim_down") end if self.is_moving then -- play jump animation while moving if is_character_in_air(land_character) then Unit.animation_event(land_unit, "jump") -- setting sprint animation and move direction elseif self.is_sprinting then Unit.animation_event(land_unit, "sprint") -- setting crouch animation and move direction elseif self.is_crouching then Unit.animation_event(land_unit, "crouch_walk") -- setting walk animation and move direction else Unit.animation_event(land_unit, "stand_walk") end local move_direction = compute_move_direction(input.move) if self.started_moving then self.current_direction = move_direction else self.current_direction = blend_to_move_direction(self.current_direction, move_direction, dt) end local move_dir_var = Unit.animation_find_variable(land_unit, "move_direction") local move_speed_var = Unit.animation_find_variable(land_unit, "move_speed") if self.is_aim then Unit.animation_set_variable(land_unit, move_dir_var, self.current_direction) Unit.animation_set_variable(land_unit, move_speed_var, speed) -- only walking when aiming else -- normally we just rotate the character to move -- so animation dir is just forward (NORTH) Unit.animation_set_variable(land_unit, move_dir_var, 2) Unit.animation_set_variable(land_unit, move_speed_var, speed) end --play jump animation not moving elseif is_character_in_air(land_character) then Unit.animation_event(land_unit, "jump") move_direction = self.current_direction --setting crouch idle animation elseif self.is_crouching then Unit.animation_event(land_unit, "crouch_idle") --setting stand idle animation else Unit.animation_event(land_unit, "stand_idle") end self.previous_player_altitude = Unit.world_position(land_unit, 1).z -- update character aiming set_aim_target_position(self) if Unit.has_node(land_unit, "s_genrig_RightHand") and Unit.has_node(land_unit, "s_genrig_AimNode") then local hand_index = Unit.node(land_unit, "s_genrig_RightHand") local aim_index = Unit.node(land_unit, "s_genrig_AimNode") local hand_aim_offset = Unit.world_position(land_unit, hand_index) - Unit.world_position(land_unit, 1) hand_aim_offset = hand_aim_offset + Vector3(0, 100, 0) Unit.set_local_position(land_unit, aim_index, hand_aim_offset) end end local function animate_character_1p(self, dt) local land_character = self.land_character local land_unit = self.unit_1p -- movement animation if self.is_aim then Unit.animation_event(land_unit, "aim_up") else Unit.animation_event(land_unit, "aim_down") end if land_character and self.is_moving and not is_character_in_air(land_character) then update_head_bob(self, dt, self.is_sprinting) elseif land_character and not self.is_moving then self.walk_distance = 0.0 set_1p_stand_position(self) end -- animate standard (3p) character that is hidden for sounds and physics interaction animate_character_3p(self, dt) end local function animate_camera(self, dt) local player_camera = self.player_camera local camera_wrapper = CameraWrapper.manager:get(player_camera) local camera_offset = player_camera.offset:unbox() local land_unit = self.land_character.unit local land_position = Unit.world_position(land_unit, 1) local land_rotation = Unit.world_rotation(land_unit, 1) local target_offset = player_camera.target_offset:unbox() local desired_target_offset = Quaternion.rotate(land_rotation, target_offset) local desired_camera_offset = Vector3.normalize(camera_offset) * player_camera.offset_length local desired_translation = land_position + desired_target_offset + desired_camera_offset local desired_rotation = Quaternion.look(-desired_camera_offset) local camera_pose = Matrix4x4.from_quaternion_position(desired_rotation, desired_translation) camera_wrapper:set_local_pose(camera_pose) Vector3Box.store(player_camera.offset, desired_camera_offset) end local function update_projectiles(self) local world = SimpleProject.world for time, unit in pairs(self.spawn_projs) do local proj_act = Unit.actor(unit, "foam_projectile") local dt = world:time() - time if dt > 5 and unit then table.remove(self.spawn_projs, time) ComponentManager.remove_components(unit) World.destroy_unit(world, unit) self.spawn_projs[time] = nil end end end ------------------------ Update and Animate Function ------------------------ ------------------------ Player Function ------------------------ function Player:init() -- The Basic Project gameplay design is for the player character to spawn, but the -- camera is initially set to freecam and the player character will be stationary. self.player_camera = {} self.land_character = {} self.player_weapon = {} self.spawn_projs = {} self.is_freecam_mode = false self.is_firstperson = false self.saved_freecam_pose = Matrix4x4Box() self.is_aim = false self.is_moving = false self.started_moving = false self.is_crouching = false self.is_sprinting = false self.current_direction = 0 self.last_fire_time = 0 self.previous_player_altitude = 0 self.character_speed_3p = 0 self.walk_distance = 0 self.jumped_while_sprinting = false self.character_blend_target = nil self.camera_blend_offset_target = nil self.camera_blend_center_target = nil self.camera_blend_fov_target = nil end -- Main Player Spawn function -- player starts in Free Cam mode, and can toggle to spawning a character and back to Free Cam function Player.spawn_player(level, view_position, view_rotation) if not level then print("ERROR: No current level - cannot spawn") return end local player = Player() PlayerHud:init_hud() spawn_freecam_player(player, level, view_position, view_rotation) enable_walk_mode(player) play_spawn_sound() play_ambient_sound() Appkit.manage_level_object(level, Player, player) end function Player:update(dt) PlayerHud:update(dt) check_camera_mode(self) check_moving(self) check_sprint(self) check_move_speed(self) -- non free cam mode if self.is_freecam_mode == false then check_first_person_mode(self) check_aim(self, dt) check_projectile_fire(self) check_crouch(self) check_jump(self) -- 3p character and camera controller if self.is_firstperson == false then check_move_character(self, dt) check_rotate_camera(self, dt) check_center_camera(self) update_character_blend(self, dt) end update_camera_blend(self, dt) if self.is_firstperson then animate_character_1p(self, dt) else animate_character_3p(self, dt) animate_camera(self, dt) end end update_projectiles(self) check_exit_level(self) end function Player:shutdown() PlayerHud:shutdown() cleanup_projectiles(self) despawn_weapon(self) despawn_character_1p(self) despawn_character(self) despawn_freecam(self) end ------------------------ Player Function ------------------------ return Player