449 lines
20 KiB
CoffeeScript
449 lines
20 KiB
CoffeeScript
|
|
# 00000000 000 0000000 000 000 00000000 00000000
|
|
# 000 000 000 000 000 000 000 000 000 000
|
|
# 00000000 000 000000000 00000 0000000 0000000
|
|
# 000 000 000 000 000 000 000 000
|
|
# 000 0000000 000 000 000 00000000 000 000
|
|
{
|
|
clamp
|
|
} = require '/Users/kodi/s/ko/js/tools/tools'
|
|
log = require '/Users/kodi/s/ko/js/tools/log'
|
|
Bot = require './bot'
|
|
Action = require './action'
|
|
Timer = require './timer'
|
|
Vector = require './lib/vector'
|
|
Quaternion = require './lib/quaternion'
|
|
Perspective = require './perspective'
|
|
|
|
class Player extends Bot
|
|
|
|
constructor: ->
|
|
|
|
super
|
|
@name = 'player'
|
|
@key =
|
|
forward: 'w'
|
|
backward: 's'
|
|
left: 'a'
|
|
right: 'd'
|
|
lookUp: 'up'
|
|
lookDown: 'down'
|
|
shoot: 'enter'
|
|
jump: 'space'
|
|
view: 'c'
|
|
push: 'shift'
|
|
|
|
@look_action = null
|
|
@look_angle = 0.0
|
|
@new_dir_sgn = 1.0
|
|
@rotate = 0
|
|
|
|
@recorder = null
|
|
@playback = null
|
|
|
|
@addAction new Action @, Action.LOOK_UP, "look up", 220
|
|
@addAction new Action @, Action.LOOK_DOWN, "look down", 220
|
|
@addAction new Action @, Action.LOOK_RESET, "look reset", 60
|
|
|
|
@addEventWithName "landed"
|
|
|
|
@projection = new Perspective 90.0
|
|
@projection.updateViewport()
|
|
|
|
bulletHitSound: -> 'BULLET_HIT_PLAYER'
|
|
|
|
# 00000000 00000000 0000000 000 00000000 0000000 000000000 000 0000000 000 000
|
|
# 000 000 000 000 000 000 000 000 000 000 000 000 000 0000 000
|
|
# 00000000 0000000 000 000 000 0000000 000 000 000 000 000 000 0 000
|
|
# 000 000 000 000 000 000 000 000 000 000 000 000 000 000 0000
|
|
# 000 000 000 0000000 0000000 00000000 0000000 000 000 0000000 000 000
|
|
|
|
getInsideProjection: () ->
|
|
|
|
# smooth camera movement a little bit
|
|
posDelta = world.getSpeed() / 10.0
|
|
playerDir = @getCurrentDir()
|
|
playerUp = @current_orientation.rotate(new Vector(0,1,0)).normal()
|
|
camPos = @current_position.clone()
|
|
if @look_angle < 0
|
|
camPos.add playerUp.mul -2*@look_angle/90
|
|
@projection.setPosition @projection.getPosition().mul(1.0-posDelta).plus camPos.mul posDelta
|
|
|
|
if @look_angle # player is looking up or down
|
|
@projection.setXVector playerDir.cross(playerUp).normal()
|
|
rot = Quaternion.rotationAroundVector @look_angle, @projection.getXVector()
|
|
@projection.setYVector rot.rotate playerUp
|
|
@projection.setZVector rot.rotate playerDir #.neg()
|
|
else
|
|
# smooth camera rotation a little bit
|
|
lookDelta = (2.0 - @projection.getZVector().dot playerDir) * world.getSpeed() / 50.0
|
|
newLookVector = @projection.getZVector().mul(1.0 - lookDelta).plus playerDir.mul lookDelta
|
|
newLookVector.normalize()
|
|
@projection.setXVector playerUp.cross(newLookVector).normal()
|
|
@projection.setYVector playerUp
|
|
@projection.setZVector newLookVector
|
|
|
|
@projection
|
|
|
|
# 0000000 00000000 000 000 000 000 000 0000000
|
|
# 000 000 000 000 000 000 0000 000 000 000
|
|
# 0000000 0000000 000000000 000 000 0 000 000 000
|
|
# 000 000 000 000 000 000 000 0000 000 000
|
|
# 0000000 00000000 000 000 000 000 000 0000000
|
|
|
|
getBehindProjection: () ->
|
|
|
|
playerDir = @getCurrentDir()
|
|
playerUp = @current_orientation.rotate(new Vector(0,1,0)).normal()
|
|
|
|
# find a valid camera position
|
|
botToCamera = playerUp.minus playerDir.mul 2
|
|
min_f = botToCamera.length()
|
|
botToCamera.normalize()
|
|
|
|
min_f = Math.min world.getWallDistanceForRay(@current_position, botToCamera), min_f
|
|
camPos = @current_position.plus botToCamera.mul Math.max(min_f, 0.72) * (1-Math.abs(@look_angle)/90)
|
|
if @look_angle < 0
|
|
camPos.add playerUp.mul -2*@look_angle/90
|
|
|
|
camPos = world.getInsideWallPosWithDelta camPos, 0.2
|
|
|
|
# smooth camera movement a little bit
|
|
posDelta = 0.2
|
|
@projection.setPosition @projection.getPosition().mul(1.0 - posDelta).plus camPos.mul posDelta
|
|
|
|
if @look_angle
|
|
# log "look_angle #{@look_angle}"
|
|
@projection.setXVector playerDir.cross(playerUp).normal()
|
|
rot = Quaternion.rotationAroundVector @look_angle, @projection.getXVector()
|
|
@projection.setYVector rot.rotate playerUp
|
|
@projection.setZVector rot.rotate playerDir #.neg()
|
|
else
|
|
# smooth camera rotation a little bit
|
|
lookDelta = 0.3
|
|
newLookVector = @projection.getZVector().mul(1.0 - lookDelta).plus (playerDir.minus(@getCurrentUp().mul(0.2)).normal()).mul lookDelta
|
|
newLookVector.normalize()
|
|
|
|
@projection.setZVector newLookVector
|
|
@projection.setXVector playerUp.cross(newLookVector).normal()
|
|
@projection.setYVector newLookVector.cross(@projection.getXVector()).normal()
|
|
|
|
# log 'Player.getBehindProjection', @projection.getPosition()
|
|
@projection
|
|
|
|
# 00000000 0000000 000 000 0000000 000 000
|
|
# 000 000 000 000 000 000 000 000 0 000
|
|
# 000000 000 000 000 000 000 000 000000000
|
|
# 000 000 000 000 000 000 000 000 000
|
|
# 000 0000000 0000000 0000000 0000000 00 00
|
|
|
|
getFollowProjection: () ->
|
|
|
|
camPos = @projection.getPosition()
|
|
desiredDistance = 2.0 # desired distance from camera to bot
|
|
|
|
playerPos = @current_position
|
|
playerDir = @getCurrentDir()
|
|
playerUp = @getCurrentUp()
|
|
playerLeft = @getCurrentLeft()
|
|
|
|
# first, adjust distance from camera to bot
|
|
|
|
botToCamera = camPos.minus playerPos # vector from bot to current pos
|
|
cameraBotDistance = botToCamera.length() # distance from camera to bot
|
|
|
|
if cameraBotDistance >= desiredDistance
|
|
difference = cameraBotDistance - desiredDistance
|
|
delta = difference*difference/400.0 # weight for following speed
|
|
camPos = camPos.mul(1.0-delta).plus playerPos.mul delta
|
|
else
|
|
difference = desiredDistance - cameraBotDistance
|
|
delta = difference/20.0 # weight for negative following speed
|
|
camPos = camPos.mul(1.0-delta).plus (playerPos.plus botToCamera.normal().mul desiredDistance).mul delta
|
|
|
|
# second, rotate around bot
|
|
|
|
botToCamera = camPos.minus playerPos
|
|
botToCameraNormal = botToCamera.normal()
|
|
|
|
# rotate camera vertically
|
|
verticalAngle = Vector.RAD2DEG Math.acos(clamp(-1.0, 1.0, botToCameraNormal.dot playerUp))
|
|
if verticalAngle > 45
|
|
# log "verticalAngle #{verticalAngle}"
|
|
rotQuat = Quaternion.rotationAroundVector(verticalAngle/400.0, botToCameraNormal.cross(playerUp))
|
|
botToCamera = rotQuat.rotate botToCamera
|
|
botToCameraNormal = botToCamera.normal()
|
|
camPos = playerPos.plus botToCamera
|
|
|
|
rotFactor = 1.0
|
|
|
|
wall_distance = world.getWallDistanceForPos camPos
|
|
if wall_distance < 0.5 # try avoid piercing walls
|
|
if wall_distance < 0.2
|
|
camPos = world.getInsideWallPosWithDelta camPos, 0.2
|
|
botToCamera = camPos.minus playerPos
|
|
botToCameraNormal = botToCamera.normal()
|
|
rotFactor = 0.5 / (wall_distance-0.2)
|
|
|
|
# try view bot from behind
|
|
# calculate horizontal angle between bot orientation and vector to camera
|
|
mappedToXZ = (botToCamera.minus playerUp.mul(botToCamera.dot playerUp)).normal()
|
|
horizontalAngle = Vector.RAD2DEG Math.acos(clamp(-1.0, 1.0, -playerDir.dot mappedToXZ))
|
|
if botToCameraNormal.dot(playerLeft) < 0
|
|
horizontalAngle = -horizontalAngle
|
|
rotQuat = Quaternion.rotationAroundVector horizontalAngle/(rotFactor*400.0), playerUp
|
|
camPos = playerPos.plus rotQuat.rotate botToCamera
|
|
|
|
botToCamera = camPos.minus playerPos
|
|
botToCameraNormal = botToCamera.normal()
|
|
|
|
@projection.setPosition camPos # finally, set the position
|
|
|
|
# slowly adjust look direction by interpolating current and desired directions
|
|
lookDelta = @projection.getZVector().dot botToCameraNormal
|
|
lookDelta *= lookDelta / 30.0
|
|
newLookVector = @projection.getZVector().mul(1.0-lookDelta).plus botToCameraNormal.neg().mul(lookDelta)
|
|
newLookVector.normalize()
|
|
|
|
# slowly adjust up vector by interpolating current and desired up vectors
|
|
upDelta = 1.5-@projection.getYVector().dot playerUp
|
|
upDelta *= upDelta / 100.0
|
|
newUpVector = @projection.getYVector().mul(1.0-upDelta).plus playerUp.mul(upDelta)
|
|
newUpVector.normalize()
|
|
|
|
newLeftVector = newUpVector.cross newLookVector
|
|
|
|
# finished interpolations, update camera matrix
|
|
@projection.setXVector newLeftVector
|
|
@projection.setYVector newUpVector
|
|
@projection.setZVector newLookVector
|
|
@projection
|
|
|
|
# 0000000 0000000 000000000 000 0000000 000 000
|
|
# 000 000 000 000 000 000 000 0000 000
|
|
# 000000000 000 000 000 000 000 000 0 000
|
|
# 000 000 000 000 000 000 000 000 0000
|
|
# 000 000 0000000 000 000 0000000 000 000
|
|
|
|
initAction: (action) ->
|
|
# log "initAction #{action.id} #{action.name}"
|
|
switch action.id
|
|
when Action.CLIMB_DOWN
|
|
# @addMoves 1
|
|
world.playSound 'BOT_CLIMB'
|
|
when Action.FORWARD
|
|
# @addMoves 1
|
|
world.playSound 'BOT_MOVE'
|
|
when Action.TURN_LEFT, Action.TURN_RIGHT
|
|
world.playSound 'BOT_TURN'
|
|
when Action.JUMP
|
|
# @addMoves actionId == Action.JUMP and 1 or 2
|
|
world.playSound 'BOT_JUMP'
|
|
|
|
super action
|
|
|
|
# 00000000 00000000 00000000 00000000 0000000 00000000 00 00
|
|
# 000 000 000 000 000 000 000 000 000 000 000 000
|
|
# 00000000 0000000 0000000 000000 000 000 0000000 000000000
|
|
# 000 000 000 000 000 000 000 000 000 000 0 000
|
|
# 000 00000000 000 000 000 0000000 000 000 000 000
|
|
|
|
performAction: (action) ->
|
|
relTime = action.getRelativeTime()
|
|
switch action.id
|
|
when Action.NOOP then return
|
|
when Action.LOOK_UP then @look_angle = relTime * 90.0
|
|
when Action.LOOK_DOWN then @look_angle = relTime * -90.0
|
|
when Action.LOOK_RESET
|
|
if @look_angle > 0 then @look_angle = Math.min @look_angle, (1.0-relTime) * 90.0
|
|
else @look_angle = Math.max @look_angle, (1.0-relTime) * -90.0
|
|
else
|
|
super action
|
|
|
|
# 00000000 000 000 000 000 0000000 000 000
|
|
# 000 000 0000 000 000 000 000 000
|
|
# 000000 000 000 0 000 000 0000000 000000000
|
|
# 000 000 000 0000 000 000 000 000
|
|
# 000 000 000 000 000 0000000 000 000
|
|
|
|
finishAction: (action) ->
|
|
|
|
if action.id == Action.LOOK_RESET
|
|
@look_action = null
|
|
@look_angle = 0.0
|
|
else
|
|
if action.id == @move_action?.id # move finished, update direction
|
|
@dir_sgn = @new_dir_sgn
|
|
|
|
if action.id != Action.LOOK_UP and action.id != Action.LOOK_DOWN
|
|
super action
|
|
|
|
if action.id == Action.TURN_LEFT or action.id == Action.TURN_RIGHT
|
|
if @rotate
|
|
@rotate_action = @getActionWithId @rotate
|
|
@rotate_action.reset()
|
|
Timer.addAction @rotate_action
|
|
|
|
die: () ->
|
|
super
|
|
# Controller.displayText "game over"
|
|
world.playSound 'BOT_DEATH'
|
|
world.setCameraMode world.CAMERA_FOLLOW
|
|
|
|
# 000 000 00000000 000 000
|
|
# 000 000 000 000 000
|
|
# 0000000 0000000 00000
|
|
# 000 000 000 000
|
|
# 000 000 00000000 000
|
|
|
|
modKeyComboEventDown: (mod, key, combo, event) ->
|
|
|
|
# log "player.modKeyComboEventDown mod:#{mod} key:#{key} combo:#{combo}"
|
|
|
|
switch key
|
|
when @key.forward, @key.backward
|
|
@push = mod == @key.push
|
|
@move = true # try to move as long as the key is not released
|
|
if not @move_action?
|
|
@new_dir_sgn = @dir_sgn = (key == @key.backward) and -1 or 1
|
|
@moveBot() # perform new move action (depending on environment)
|
|
else
|
|
if @move_action.id == Action.JUMP and @move_action.getRelativeTime() < 1
|
|
if world.isUnoccupiedPos(@position.plus(@getUp()).plus(@getDir())) and
|
|
world.isUnoccupiedPos(@position.plus(@getDir())) # forward and above forward also empty
|
|
action = @getActionWithId Action.JUMP_FORWARD
|
|
action.takeOver @move_action
|
|
Timer.removeAction @move_action
|
|
@move_action = action
|
|
Timer.addAction @move_action
|
|
@new_dir_sgn = (key == @key.backward) and -1 or 1
|
|
return true
|
|
|
|
when @key.left, @key.right
|
|
@rotate = (key == @key.left) and Action.TURN_LEFT or Action.TURN_RIGHT
|
|
if not @rotate_action? and not @spiked # player is not performing a rotation and unspiked
|
|
@rotate_action = @getActionWithId @rotate
|
|
Timer.addAction @rotate_action
|
|
return true
|
|
|
|
when @key.jump
|
|
@jump = true # switch to jump mode until jump_key released
|
|
@jump_once = true
|
|
if not @move_action?
|
|
@moveBot() # perform jump action (depending on environment)
|
|
@jump_once = false
|
|
else
|
|
# log 'jump:moving'
|
|
if @move_action.id == Action.FORWARD and @move_action.getRelativeTime() < 0.6 or
|
|
@move_action.id == Action.CLIMB_DOWN and @move_action.getRelativeTime() < 0.4
|
|
# abort current move and jump instead
|
|
# log 'jump:move or climb down'
|
|
if world.isUnoccupiedPos @position.plus @getUp()
|
|
# log 'jump:can do'
|
|
if world.isUnoccupiedPos @position.plus @getUp().plus @getDir()
|
|
action = @getActionWithId Action.JUMP_FORWARD
|
|
else
|
|
action = @getActionWithId Action.JUMP
|
|
action.takeOver @move_action
|
|
Timer.removeAction @move_action
|
|
@move_action = action
|
|
@jump_once = false
|
|
Timer.addAction @move_action
|
|
else if @move_action.id in [Action.JUMP, Action.JUMP_FORWARD]
|
|
@jump_once = false
|
|
return true
|
|
|
|
when @key.push
|
|
@push = true
|
|
return true
|
|
|
|
when @key.shoot
|
|
if not @shoot
|
|
@shoot = true
|
|
Timer.addAction @getActionWithId Action.SHOOT
|
|
return true
|
|
|
|
when @key.lookUp, @key.lookDown
|
|
if not @look_action
|
|
@look_action = @getActionWithId (key == @key.lookUp) and Action.LOOK_UP or Action.LOOK_DOWN
|
|
@look_action.reset()
|
|
Timer.addAction @look_action
|
|
return true
|
|
|
|
when @key.view
|
|
world.changeCameraMode()
|
|
return true
|
|
|
|
false
|
|
|
|
# 00000000 00000000 000 00000000 0000000 0000000 00000000
|
|
# 000 000 000 000 000 000 000 000 000
|
|
# 0000000 0000000 000 0000000 000000000 0000000 0000000
|
|
# 000 000 000 000 000 000 000 000 000
|
|
# 000 000 00000000 0000000 00000000 000 000 0000000 00000000
|
|
|
|
modKeyComboEventUp: (mod, key, combo, event) ->
|
|
@push = false if @key.push == 'shift'
|
|
# log "player.modKeyComboEventUp mod:#{mod} key:#{key} combo:#{combo}"
|
|
switch key
|
|
when @key.shoot
|
|
Timer.removeAction @getActionWithId Action.SHOOT
|
|
@shoot = false
|
|
return true
|
|
|
|
when @key.forward, @key.backward
|
|
@move = false
|
|
return true
|
|
|
|
when @key.jump
|
|
@jump = false
|
|
return true
|
|
|
|
when @key.left, @key.right
|
|
@rotate = 0
|
|
return true
|
|
|
|
when @key.push
|
|
@push = false
|
|
return true
|
|
|
|
when @key.lookDown, @key.lookUp
|
|
if @look_action and @look_action.id != Action.LOOK_RESET
|
|
Timer.removeAction @look_action
|
|
@look_action = @getActionWithId Action.LOOK_RESET
|
|
Timer.addAction @look_action
|
|
return true
|
|
|
|
when @key.view
|
|
return true
|
|
|
|
false
|
|
|
|
# 0000000 000 0000000 00000000 000 0000000 000 000
|
|
# 000 000 000 000 000 000 000 000 000 000 000
|
|
# 000 000 000 0000000 00000000 000 000000000 00000
|
|
# 000 000 000 000 000 000 000 000 000
|
|
# 0000000 000 0000000 000 0000000 000 000 000
|
|
|
|
step: (step) -> super step
|
|
|
|
getBodyColor: () ->
|
|
if world.getCameraMode() == world.CAMERA_BEHIND
|
|
# static bodyColor
|
|
bodyColor = colors[KikiPlayer_base_color]
|
|
bodyColor.setAlpha(kMin(0.7, (@projection.getPosition()-@current_position).length()-0.4))
|
|
return bodyColor
|
|
|
|
return colors[KikiPlayer_base_color]
|
|
|
|
getTireColor: () ->
|
|
if world.getCameraMode() == world.CAMERA_BEHIND
|
|
# static tireColor
|
|
tireColor = colors[KikiPlayer_tire_color]
|
|
tireColor.setAlpha(kMin(1.0, (@projection.getPosition()-@current_position).length()-0.4))
|
|
return tireColor
|
|
|
|
return colors[KikiPlayer_tire_color]
|
|
|
|
module.exports = Player
|