# 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