kiki/coffee/player.coffee
monsterkodi d4c28ee1d6 levels
2016-08-23 18:35:44 +02:00

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