A team game with an emphasis on movement (with no shooting), inspired by Overwatch and Zineth
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

280 lines
8.4 KiB

  1. # Original: https://raw.githubusercontent.com/Calinou/fps-test/master/scripts/player.gd
  2. extends RigidBody
  3. var view_sensitivity = 0.25
  4. # Walking speed and jumping height are defined later.
  5. var walk_speed = 0.8 # Actually acceleration; m/s/s
  6. var jump_speed = 5 # m/s
  7. var air_accel = .1 # m/s/s
  8. var floor_friction = 1-0.08
  9. var air_friction = 1-0.03
  10. var player_info # Set by lobby
  11. var walk_speed_build = 0.006 # `walk_speed` per `switch_charge`
  12. var air_speed_build = 0.006 # `air_accel` per `switch_chare`
  13. var switch_charge = 0
  14. var switch_charge_cap = 200 # While switching is always at 100, things like speed boost might go higher!
  15. var movement_charge = 0.15 # In percent per meter (except when heroes change that)
  16. const fall_height = -50
  17. var debug_node
  18. var recording
  19. slave var slave_transform = Basis()
  20. slave var slave_lin_v = Vector3()
  21. slave var slave_ang_v = Vector3()
  22. var tp_camera = "TPCamera"
  23. var master_only = "MasterOnly"
  24. var master_player
  25. var ai_instanced = false
  26. func _ready():
  27. set_process_input(true)
  28. debug_node = get_node("/root/Level/Debug")
  29. if is_network_master():
  30. get_node(tp_camera).set_enabled(true)
  31. spawn()
  32. if "is_ai" in player_info and player_info.is_ai and not ai_instanced:
  33. add_child(preload("res://scenes/ai.tscn").instance())
  34. ai_instanced = true
  35. else:
  36. get_node("PlayerName").set_text(player_info.username)
  37. # Remove HUD
  38. remove_child(get_node(master_only))
  39. func spawn():
  40. if "record" in player_info:
  41. write_recording() # Write each spawn as a separate recording
  42. var placement = Vector3()
  43. var x_varies = 5
  44. var z_varies = 5
  45. # No Z, because that's the left-right question
  46. if player_info.is_right_team:
  47. placement = get_node("/root/Level/RightSpawn").get_translation()
  48. else:
  49. placement = get_node("/root/Level/LeftSpawn").get_translation()
  50. # So we don't all spawn on top of each other
  51. placement.x += rand_range(0, x_varies)
  52. placement.z += rand_range(0, z_varies)
  53. recording = { "time": 0, "states": [], "events": [], "spawn": Vector3() }
  54. recording.spawn = var2str(placement)
  55. recording.switch_charge = var2str(switch_charge)
  56. set_transform(Basis())
  57. set_translation(placement)
  58. set_linear_velocity(Vector3())
  59. func event_to_obj(event):
  60. var d = {}
  61. if event is InputEventMouseMotion:
  62. d.relative = {}
  63. d.relative.x = event.relative.x
  64. d.relative.y = event.relative.y
  65. d.type = "motion"
  66. if event is InputEventKey:
  67. d.scancode = event.scancode
  68. d.pressed = event.pressed
  69. d.echo = event.echo
  70. d.type = "key"
  71. if event is InputEventMouseButton:
  72. d.button_index = event.button_index
  73. d.pressed = event.pressed
  74. d.type = "mb"
  75. return d
  76. func _input(event):
  77. if is_network_master():
  78. if Input.is_action_just_pressed("switch_hero"):
  79. switch_hero_interface()
  80. # Quit the game:
  81. if Input.is_action_pressed("quit"):
  82. quit()
  83. if "record" in player_info:
  84. recording.events.append([recording.time, event_to_obj(event)])
  85. func begin():
  86. master_player = get_node("/root/Level/Players/%d" % get_tree().get_network_unique_id())
  87. # Set color to blue (teammate) or red (enemy)
  88. var color
  89. if master_player.player_info.is_right_team == player_info.is_right_team:
  90. color = Color("#073a98") # Blue for friendly
  91. else:
  92. color = Color("#62071a") # Red for enemy
  93. var mat = SpatialMaterial.new()
  94. mat.albedo_color = color
  95. get_node("Yaw/MainMesh").set_surface_material(0, mat)
  96. get_node("Yaw/Pitch/RotatedHead").set_surface_material(0, mat)
  97. func toggle_mouse_capture():
  98. if (Input.get_mouse_mode() == Input.MOUSE_MODE_CAPTURED):
  99. Input.set_mouse_mode(Input.MOUSE_MODE_VISIBLE)
  100. view_sensitivity = 0
  101. else:
  102. Input.set_mouse_mode(Input.MOUSE_MODE_CAPTURED)
  103. view_sensitivity = 0.25
  104. # Update visual yaw + pitch components to match camera
  105. func set_rotation():
  106. get_node("Yaw").set_rotation(Vector3(0, deg2rad(get_node(tp_camera).cam_yaw), 0))
  107. get_node("Yaw/Pitch").set_rotation(Vector3(deg2rad(-get_node(tp_camera).cam_pitch), 0, 0))
  108. func record_status(status):
  109. if "record" in player_info:
  110. for i in range(status.size()):
  111. status[i] = var2str(status[i])
  112. recording.states.append([recording.time, status])
  113. func _integrate_forces(state):
  114. if is_network_master():
  115. control_player(state)
  116. var status = get_status()
  117. rpc_unreliable("set_status", status)
  118. record_status(status)
  119. set_rotation()
  120. slave func set_status(s):
  121. set_transform(s[0])
  122. set_linear_velocity(s[1])
  123. set_angular_velocity(s[2])
  124. get_node(tp_camera).cam_yaw = s[3]
  125. get_node(tp_camera).cam_pitch = s[4]
  126. func get_status():
  127. return [
  128. get_transform(),
  129. get_linear_velocity(),
  130. get_angular_velocity(),
  131. get_node(tp_camera).cam_yaw,
  132. get_node(tp_camera).cam_pitch,
  133. ]
  134. func control_player(state):
  135. var aim = get_node("Yaw").get_global_transform().basis
  136. var direction = Vector3()
  137. if Input.is_action_pressed("move_forwards"):
  138. direction -= aim[2]
  139. if Input.is_action_pressed("move_backwards"):
  140. direction += aim[2]
  141. if Input.is_action_pressed("move_left"):
  142. direction -= aim[0]
  143. if Input.is_action_pressed("move_right"):
  144. direction += aim[0]
  145. direction = direction.normalized()
  146. var ray = get_node("Ray")
  147. if get_colliding_bodies(): # We can navigate normally, we have a surface
  148. var up = state.get_total_gravity().normalized()
  149. var normal = ray.get_collision_normal()
  150. var floor_velocity = Vector3()
  151. var object = ray.get_collider()
  152. var accel = (1 + switch_charge * walk_speed_build) * walk_speed
  153. state.apply_impulse(Vector3(), direction * accel * get_mass())
  154. var lin_v = state.get_linear_velocity()
  155. lin_v.x *= floor_friction
  156. lin_v.z *= floor_friction
  157. state.set_linear_velocity(lin_v)
  158. if Input.is_action_just_pressed("jump"):
  159. # This may be kinda expensive but we only check while pressing jump so it's ok
  160. # Detect jumpable
  161. var jump_dot = 0.8 # If normal.dot(up) > jump_dot, we can jump
  162. var jumpable = false
  163. for i in range(state.get_contact_count()):
  164. var n = state.get_contact_local_normal(0)
  165. if n.dot(Vector3(0,1,0)) > jump_dot:
  166. jumpable = true
  167. if jumpable:
  168. state.apply_impulse(Vector3(), normal * jump_speed * get_mass())
  169. else:
  170. var accel = (1 + switch_charge * air_speed_build) * air_accel
  171. state.apply_impulse(Vector3(), direction * accel * get_mass())
  172. var lin_v = state.get_linear_velocity()
  173. lin_v.x *= air_friction
  174. lin_v.z *= air_friction
  175. state.set_linear_velocity(lin_v)
  176. state.integrate_forces()
  177. func _process(delta):
  178. # All player code not caused by input, and not causing movement
  179. if is_network_master():
  180. var vel = get_linear_velocity()
  181. switch_charge += movement_charge * vel.length() * delta
  182. var switch_node = get_node("MasterOnly/SwitchCharge")
  183. switch_node.set_text("%.f%%" % switch_charge)
  184. if switch_charge >= 100:
  185. # Let switch_charge keep building, because we use it for walk_speed and things
  186. switch_node.set_text("100%% (%.f)\nQ - Switch hero" % switch_charge)
  187. if switch_charge > switch_charge_cap:
  188. # There is however a cap
  189. switch_charge = switch_charge_cap
  190. if get_translation().y < fall_height:
  191. spawn()
  192. if "record" in player_info:
  193. recording.time += delta
  194. func switch_hero_interface():
  195. if switch_charge >= 100:
  196. # Interface needs the mouse!
  197. toggle_mouse_capture()
  198. # Pause so if we have walls and such nothing funny happens
  199. get_tree().set_pause(true)
  200. var interface = preload("res://scenes/HeroSelect.tscn").instance()
  201. add_child(interface)
  202. interface.get_node("Confirm").connect("pressed", self, "switch_hero_master")
  203. func switch_hero_master():
  204. rpc("switch_hero", get_node("HeroSelect/Hero").get_selected_id())
  205. # Remove the mouse and enable looking again
  206. toggle_mouse_capture()
  207. get_tree().set_pause(false)
  208. sync func switch_hero(hero):
  209. var new_hero = load("res://scenes/heroes/%d.tscn" % hero).instance()
  210. var net_id = int(get_name())
  211. set_name("%d-delete" % net_id) # Can't have duplicate names
  212. new_hero.set_name("%d" % net_id)
  213. new_hero.set_network_master(net_id)
  214. new_hero.player_info = player_info
  215. get_node("/root/Level/Players").call_deferred("add_child", new_hero)
  216. # We must wait until after _ready is called, so that we don't end up at spawn
  217. new_hero.call_deferred("set_status", get_status())
  218. queue_free()
  219. func _exit_scene():
  220. Input.set_mouse_mode(Input.MOUSE_MODE_VISIBLE)
  221. func _exit_tree():
  222. if "record" in player_info:
  223. write_recording()
  224. # Functions
  225. # =========
  226. func write_recording():
  227. if recording and recording.events.size() > 0:
  228. var save = File.new()
  229. var fname = "res://recordings/%d-%d-%d.rec" % [player_info.level, player_info.hero, randi() % 10000]
  230. save.open(fname, File.WRITE)
  231. save.store_line(to_json(recording))
  232. save.close()
  233. # Quits the game:
  234. func quit():
  235. get_tree().quit()