package sig.objects; import sig.RabiClone; import sig.engine.AnimatedObject; import sig.engine.Object; import sig.engine.Panel; import sig.engine.Sprite; import sig.map.CollisionType; import sig.map.Map; import sig.map.Tile; import sig.map.View; import sig.objects.actor.State; import java.awt.event.KeyEvent; public class Player extends AnimatedObject{ final double GRAVITY = 1300; final double NORMAL_FRICTION = 6400; final boolean LEFT = false; final boolean RIGHT = true; double y_acceleration = GRAVITY; double y_acceleration_limit = 100; double x_acceleration = 0; double x_acceleration_limit = 100; double x_velocity = 0; double x_velocity_limit = 164; double y_velocity = 5; double y_velocity_limit = 400; double horizontal_drag = 2000; double horizontal_friction = NORMAL_FRICTION; double horizontal_air_drag = 800; double horizontal_air_friction = 180; double jump_velocity = -300; int maxJumpCount=2; int jumpCount=maxJumpCount; State state = State.IDLE; final double viewBoundaryX=RabiClone.BASE_WIDTH/3; final double viewBoundaryY=RabiClone.BASE_HEIGHT/3; View lastCameraView = View.FIXED; boolean groundCollision = false; boolean spacebarReleased = true; boolean facing_direction = RIGHT; long spacebarPressed = System.currentTimeMillis(); int jumpHoldTime = 150; public Player(Panel panel) { super(Sprite.ERINA,5,panel); setX(RabiClone.BASE_WIDTH/2-getAnimatedSpr().getWidth()/2); setY(RabiClone.BASE_HEIGHT*(2/3d)-getAnimatedSpr().getHeight()/2); } @Override public void update(double updateMult) { super.update(updateMult); handleMovementPhysics(updateMult); handleCameraRoomMovement(); int right = (KeyHeld(KeyEvent.VK_RIGHT))||(KeyHeld(KeyEvent.VK_D))?1:0; int left = (KeyHeld(KeyEvent.VK_LEFT))||(KeyHeld(KeyEvent.VK_A))?1:0; switch(state){ case ATTACK: break; case FALLING: setAnimatedSpr(Sprite.ERINA_JUMP_FALL); break; case IDLE: if(KeyHeld(KeyEvent.VK_A)||KeyHeld(KeyEvent.VK_LEFT)){ setAnimatedSpr(Sprite.ERINA_WALK); } if(KeyHeld(KeyEvent.VK_D)||KeyHeld(KeyEvent.VK_RIGHT)){ setAnimatedSpr(Sprite.ERINA_WALK); } if(right-left==0){ setAnimatedSpr(Sprite.ERINA); } if(x_velocity!=0){ setAnimatedSpr(Sprite.ERINA_WALK); } else{ setAnimatedSpr(Sprite.ERINA); } break; case JUMP: setAnimatedSpr(Sprite.ERINA_JUMP_RISE); break; case SLIDE: break; case STAGGER: break; case UNCONTROLLABLE: break; default: break; } if (KeyHeld(KeyEvent.VK_SPACE)||KeyHeld(KeyEvent.VK_W)&&System.currentTimeMillis()-spacebarPressed0 && spacebarReleased && (key == KeyEvent.VK_SPACE || key == KeyEvent.VK_W)){ jumpCount=0; y_velocity = jump_velocity; spacebarReleased=false; spacebarPressed=System.currentTimeMillis(); } break; case SLIDE: break; case STAGGER: break; case UNCONTROLLABLE: break; default: break; } if (groundCollision) { if (spacebarReleased&&(key==KeyEvent.VK_SPACE||key==KeyEvent.VK_W)&&jumpCount>0) { state = State.JUMP; jumpCount--; y_velocity = jump_velocity; spacebarReleased=false; spacebarPressed=System.currentTimeMillis(); //System.out.println("Jump"); } } switch(key){ case KeyEvent.VK_LEFT: case KeyEvent.VK_A: facing_direction=LEFT; break; case KeyEvent.VK_RIGHT:case KeyEvent.VK_D: facing_direction=RIGHT; break; } } private void handleCameraRoomMovement() { int tileX = (int)(getX())/Tile.TILE_WIDTH; int tileY = (int)(getY())/Tile.TILE_HEIGHT; double newX=RabiClone.level_renderer.getX(),newY=RabiClone.level_renderer.getY(); //By default we use a fixed view. View currentView = RabiClone.CURRENT_MAP.getView(tileX, tileY); if (currentView!=lastCameraView) { lastCameraView=currentView; newX=(tileX/Tile.TILE_SCREEN_COUNT_X)*Map.MAP_WIDTH; newY=(tileY/Tile.TILE_SCREEN_COUNT_Y)*Map.MAP_HEIGHT; } else switch (currentView) { case LIGHT_FOLLOW: if (getX()-RabiClone.level_renderer.getX()RabiClone.BASE_WIDTH-viewBoundaryX) { newX=getX()-(RabiClone.BASE_WIDTH-viewBoundaryX); int newTileX = (int)(getX()+viewBoundaryX)/Tile.TILE_WIDTH; View newView = RabiClone.CURRENT_MAP.getView(newTileX, tileY); if (newView!=currentView) { newX=(tileX/Tile.TILE_SCREEN_COUNT_X)*Map.MAP_WIDTH; } } if (getY()-RabiClone.level_renderer.getY()RabiClone.BASE_HEIGHT-viewBoundaryY) { newY=getY()-(RabiClone.BASE_HEIGHT-viewBoundaryY); int newTileY = (int)(getY()+viewBoundaryY)/Tile.TILE_HEIGHT; View newView = RabiClone.CURRENT_MAP.getView(tileX, newTileY); if (newView!=currentView) { newY=(tileY/Tile.TILE_SCREEN_COUNT_Y)*Map.MAP_HEIGHT; } } break; case LIGHT_HORIZONTAL_FOLLOW: if (getX()-RabiClone.level_renderer.getX()RabiClone.BASE_WIDTH-viewBoundaryX) { newX=getX()-(RabiClone.BASE_WIDTH-viewBoundaryX); int newTileX = (int)(getX()+viewBoundaryX)/Tile.TILE_WIDTH; View newView = RabiClone.CURRENT_MAP.getView(newTileX, tileY); if (newView!=currentView) { newX=(tileX/Tile.TILE_SCREEN_COUNT_X)*Map.MAP_WIDTH; } } newY=(tileY/Tile.TILE_SCREEN_COUNT_Y)*Map.MAP_HEIGHT; break; case LIGHT_VERTICAL_FOLLOW: newX=(tileX/Tile.TILE_SCREEN_COUNT_X)*Map.MAP_WIDTH; if (getY()-RabiClone.level_renderer.getY()RabiClone.BASE_HEIGHT-viewBoundaryY) { newY=getY()-(RabiClone.BASE_HEIGHT-viewBoundaryY); int newTileY = (int)(getY()+viewBoundaryY)/Tile.TILE_HEIGHT; View newView = RabiClone.CURRENT_MAP.getView(tileX, newTileY); if (newView!=currentView) { newY=(tileY/Tile.TILE_SCREEN_COUNT_Y)*Map.MAP_HEIGHT; } } break; } RabiClone.level_renderer.setX(newX<0?0:newX); RabiClone.level_renderer.setY(newY<0?0:newY); } private void handleMovementPhysics(double updateMult) { int right = (KeyHeld(KeyEvent.VK_RIGHT))||(KeyHeld(KeyEvent.VK_D))?1:0; int left = (KeyHeld(KeyEvent.VK_LEFT))||(KeyHeld(KeyEvent.VK_A))?1:0; x_velocity = Math.abs(x_velocity+x_acceleration*updateMult)>x_velocity_limit ?Math.signum(x_velocity+x_acceleration*updateMult)*x_velocity_limit :x_velocity+x_acceleration*updateMult; y_velocity = Math.abs(y_velocity+y_acceleration*updateMult)>y_velocity_limit ?Math.signum(y_velocity+y_acceleration*updateMult)*y_velocity_limit :y_velocity+y_acceleration*updateMult; double displacement_y = y_velocity*updateMult; double displacement_x = x_velocity*updateMult; boolean sideCollision = false; for(int i=0;i<4;i++){ double check_distance_x = (displacement_x/4)*(i+1); Tile checked_tile_top_right = RabiClone.CURRENT_MAP.getTile((int)(getX()+getAnimatedSpr().getWidth()/2-4+check_distance_x)/Tile.TILE_WIDTH, (int)(getY()+getAnimatedSpr().getHeight()/2)/Tile.TILE_HEIGHT); Tile checked_tile_top_left = RabiClone.CURRENT_MAP.getTile((int)(getX()-getAnimatedSpr().getWidth()/2+4+check_distance_x)/Tile.TILE_WIDTH, (int)(getY()+getAnimatedSpr().getHeight()/2)/Tile.TILE_HEIGHT); if(checked_tile_top_right.getCollision()==CollisionType.BLOCK||checked_tile_top_left.getCollision()==CollisionType.BLOCK){ //System.out.println(checked_tile_top_right.getCollision()+"//"+checked_tile_top_left.getCollision()); if (checked_tile_top_right.getCollision()==CollisionType.BLOCK) { setX(((int)(getX()-getAnimatedSpr().getWidth()/2)/Tile.TILE_WIDTH)*Tile.TILE_WIDTH+Tile.TILE_WIDTH/2+3+check_distance_x); } else { setX(((int)(getX()+getAnimatedSpr().getWidth())/Tile.TILE_WIDTH)*Tile.TILE_WIDTH-Tile.TILE_WIDTH/2-3+check_distance_x); } x_acceleration = 0; x_velocity = 0; sideCollision=true; } } if (y_velocity==0) { Tile checked_tile_bottom_right = RabiClone.CURRENT_MAP.getTile((int)(getX()+getAnimatedSpr().getWidth()/2-4)/Tile.TILE_WIDTH, (int)(getY()+getAnimatedSpr().getHeight()/2+0.5)/Tile.TILE_HEIGHT); Tile checked_tile_bottom_left = RabiClone.CURRENT_MAP.getTile((int)(getX()-getAnimatedSpr().getWidth()/2+4)/Tile.TILE_WIDTH, (int)(getY()+getAnimatedSpr().getHeight()/2+0.5)/Tile.TILE_HEIGHT); if (!(checked_tile_bottom_right.getCollision()==CollisionType.BLOCK||checked_tile_bottom_left.getCollision()==CollisionType.BLOCK)) { groundCollision=false; } else { groundCollision=true; jumpCount=maxJumpCount; } } else { //System.out.println(x_velocity); //System.out.println(((int)(getX()+getAnimatedSpr().getWidth()/2+displacement_x)/Tile.TILE_WIDTH)+"//"+((int)(getY()+getAnimatedSpr().getHeight()/2)/Tile.TILE_HEIGHT)); boolean collisionOccured=false; for(int i=0;i<4;i++){ double check_distance_y = (displacement_y/4)*(i+1); Tile checked_tile_bottom_right = RabiClone.CURRENT_MAP.getTile((int)(getX()+getAnimatedSpr().getWidth()/2-4)/Tile.TILE_WIDTH, (int)(getY()+getAnimatedSpr().getHeight()/2+check_distance_y)/Tile.TILE_HEIGHT); Tile checked_tile_bottom_left = RabiClone.CURRENT_MAP.getTile((int)(getX()-getAnimatedSpr().getWidth()/2+4)/Tile.TILE_WIDTH, (int)(getY()+getAnimatedSpr().getHeight()/2+check_distance_y)/Tile.TILE_HEIGHT); //System.out.println((int)getX()/Tile.TILE_WIDTH); if(checked_tile_bottom_right.getCollision()==CollisionType.BLOCK||checked_tile_bottom_left.getCollision()==CollisionType.BLOCK){ setY((getY()-check_distance_y)); y_acceleration = 0; y_velocity = 0; groundCollision=true; collisionOccured=true; state = State.IDLE; } } if (!collisionOccured) { groundCollision=false; } } if (!groundCollision){ this.setY(this.getY()+displacement_y); y_acceleration = GRAVITY; if(y_velocity>0){ state = State.FALLING; } if (!sideCollision) { handleKeyboardMovement(updateMult, right-left, horizontal_air_friction, horizontal_air_drag); this.setX(this.getX()+displacement_x); } } else { if (!sideCollision) { handleKeyboardMovement(updateMult, right-left, horizontal_friction, horizontal_drag); this.setX(this.getX()+displacement_x); } } } private void handleKeyboardMovement(double updateMult, int movement, double friction, double drag) { if (movement!=0) { x_acceleration=drag*(movement); } else { if (x_velocity!=0) { x_velocity=x_velocity>0 ?x_velocity-friction*updateMult>0 ?x_velocity-friction*updateMult :0 :x_velocity+friction*updateMult<0 ?x_velocity+friction*updateMult :0; } x_acceleration=0; } } @Override public void draw(byte[] p) {} }