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 = 890; final double NORMAL_FRICTION = 6400; 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 = 600; double horizontal_air_friction = 180; double jump_velocity = -200; 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; 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(); switch(state){ case ATTACK: break; case FALLING: break; case IDLE: break; case JUMP: break; case SLIDE: break; case STAGGER: break; case UNCONTROLLABLE: break; default: break; } System.out.println(state); } @Override protected void KeyReleased(int key) { if (key==KeyEvent.VK_SPACE) { spacebarReleased=true; } } @Override protected void KeyPressed(int key) { if (groundCollision) { if (spacebarReleased&&key==KeyEvent.VK_SPACE&&jumpCount>0) { state = State.JUMP; jumpCount--; y_velocity = jump_velocity; spacebarReleased=false; //System.out.println("Jump"); } } else if ((state == State.JUMP||state==State.FALLING) && jumpCount>0 && spacebarReleased && key == KeyEvent.VK_SPACE){ jumpCount=0; y_velocity = jump_velocity; spacebarReleased=false; } } 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)?1:0; int left = KeyHeld(KeyEvent.VK_LEFT)?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++){ if (groundCollision) { break; } 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)); if (y_velocity>0) { jumpCount=maxJumpCount; } 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) {} }