// Workshop 4 part 1 import sdljava.*; import sdljava.video.*; import sdljava.event.*; // This is a general purpose vector class class c_vector { // Constructor that takes another vector as input public c_vector( c_vector value ) { x = value.x; y = value.y; } // Constructor that takes two floats as input public c_vector( float a, float b ) { x = a; y = b; } // Constructor that takes one float as input public c_vector( float value ) { x = y = value; } // Constructor that takes no input public c_vector() { x = y = 0.0f; } // Normalises the vector public void normalise() { /*************************************** ***************************************/ // Retrieve the magnitude float len = get_length(); if( len > 0.0f ) { // Normalise the vector by dividing the components by this magnitude x /= len; y /= len; } else // This should stop divide by zero errors x = y = 0.0f; /*************************************** ***************************************/ } // Returns the length of the vector public float get_length() { /*************************************** ***************************************/ // Return the magnitude (length) using Pythagoras' theorem return (float) Math.sqrt( (x*x) + (y*y) ); /*************************************** ***************************************/ } // Our vector's components public float x, y; } // A ball class class ball { // Stores the ball image protected static SDLSurface image; // Initialises the ball class public static void init() throws SDLException { /*************************************** ***************************************/ // Load the ball image image = SDLVideo.loadBMP("ball.bmp"); /*************************************** ***************************************/ } // The ball constructor, places the ball at the given coordinates, moving in the given direction, and of the given radius public ball( c_vector pos, float direction_angle, float input_radius ) { // Set initial values max_speed = 500.0f; position = new c_vector(pos); radius = input_radius; // In this class we store the speed in a vector as its x and y speed components /*************************************** ***************************************/ // Need to set the speed of the ball using the direction angle and the max_speed: float x_component = max_speed * (float) Math.sin(direction_angle); float y_component = -max_speed * (float) Math.cos(direction_angle); speed = new c_vector(x_component,y_component); /*************************************** ***************************************/ } // Makes the ball bounce in the horizontal (i.e. off the left or right) public void bounce_horizontal() { // Need to reverse the horizontal direction speed.x = -speed.x; } // Makes the ball bounce in the vertical (i.e. off the floor / ceiling) public void bounce_vertical() { // Need to reverse the vertical direction speed.y = -speed.y; } // Updates the ball's position public void update( float deltaT_s ) { // Update our position position.x += deltaT_s * speed.x; position.y += deltaT_s * speed.y; // Make it bounce off the walls: if( position.x < 0.0f+radius ) { // Left wall bounce_horizontal(); position.x = 0.0f+radius; } else if( position.x > 640.0f-radius ) { // Right wall bounce_horizontal(); position.x = 640.0f-radius; } if( position.y < 0.0f+radius ) { // Top wall bounce_vertical(); position.y = 0.0f+radius; } else if( position.y > 480.0f-radius ) { // Bottom wall bounce_vertical(); position.y = 480.0f-radius; } } // Draws the ball public void draw(SDLSurface screen) throws SDLException { SDLRect rectangle = new SDLRect((int)position.x-(int)radius,(int)position.y-(int)radius,(int)(2*radius),(int)(2*radius)); //screen.fillRect(rectangle,screen.mapRGB(255,0,0)); // To draw the ball's image, we need to use this function: // It does a blit operation from the image surface to the screen surface, using the rectangle's position image.blitSurface(screen,rectangle); } // For retrieving the ball's position public c_vector get_position() { // We don't want anyone else to be able to modify the position, we we will make a copy to send return new c_vector(position); } // Retrieves the radius of the ball public float get_radius() { return radius; } // The maximum speed at which it can travel protected float max_speed; // The current speed of the ball protected c_vector speed; // The current position protected c_vector position; // The radius of our ball protected float radius; } // A paddle class (a moveable rectangle that the ball can bounce off of) class paddle { // Need to set the 'playing field' dimensions of where we want this paddle to be allowed to move. // This could be useful if we want two players, to keep them to their own court. public paddle( c_vector pos, int window_width, int window_height ) { aim = new c_vector(pos); position = new c_vector(pos); direction = new c_vector(0.0f); max_speed = 500.0f; width = 30; height = 10; field_width = window_width; field_height = window_height; // To demonstrate an alternative method of storing speed, // in this class we store the speed as a scalar speed value (max_speed) // and the direction in a normalised vector. // By storing it this way, we have separated the actual speed from the direction. } // Called when we need to generate the ball public ball release( float direction_angle, float radius ) { // Create a new ball. // As position, we give it our position with an offset due to our height and the radius of the ball ball our_ball = new ball(new c_vector(position.x,position.y-(height/2.0f)-radius),direction_angle, radius); return our_ball; } // Gives the paddle a coordinate to move towards public void set_position_aim( c_vector aim_input_pos ) { // Our input vector becomes our new aim vector aim = aim_input_pos; // Work out the new direction: // This will be the direction from our current position to the input position direction.x = aim_input_pos.x - position.x; direction.y = aim_input_pos.y - position.y; // We are storing the speed separately (as max_speed) // so normalise the direction vector direction.normalise(); } // This function checks whether a ball collides with the paddle, and makes it bounce off public void check_ball_collision( ball input_ball ) { // Retrieve information about the ball c_vector ball_pos = input_ball.get_position(); float ball_radius = input_ball.get_radius(); // We are going to perform a rectangle - rectangle collision test. // This is not actually correct, as the ball is really a circle, but it is quick and easy, and good enough for our uses. // Plus, because the ball is moving quite quickly, the user will probably never even notice! // To do this, we will perform four tests. // First, test the left edge of the paddle against the right edge of the ball // Then test the right edge of the paddle against the left edge of the ball if( position.x - (width/2) < ball_pos.x + ball_radius && position.x + (width/2) > ball_pos.x - ball_radius ) { // If these tests pass, it means that the ball is at least partially vertically aligned with the paddle // We now need to see if it is also horizontally aligned. // If it is, we will have a collision // First, test the top edge of the paddle against the bottom edge of the ball // Then test the bottom edge of the paddle against the top edge of the ball if( position.y - (height/2) < ball_pos.y + ball_radius && position.y + (height/2) > ball_pos.y - ball_radius ) { // The ball is partially aligned with the paddle both horizontally and vertically. // This means that the two rectangles actually intersect - i.e. there has been a collision. // Now we need to "resolve" it (decide what to do). // For the purposes of this demo, we're not going to do anything fancy or even correct. // We know that the paddle is wider than it is long, so statistically the ball is more likely to be hitting the long side. // Therefore, we'll just assume that it is hitting the paddle in this way and tell the ball to perform an vertical bounce: input_ball.bounce_vertical(); } } } // Updates the paddle's position public void update( float deltaT_s ) { // Update our position if( position.x < aim.x - 2 || position.x > aim.x + 2) position.x += deltaT_s * direction.x * max_speed; if( position.y < aim.y - 2 || position.y > aim.y + 2 ) position.y += deltaT_s * direction.y * max_speed; // Stop the paddle if it tries to go off the playing field if( position.x - (width/2) < 0.0f ) position.x = width/2; else if( position.x + (width/2) > field_width ) position.x = field_width - (width/2); if( position.y - (height/2) < 0.0f ) position.y = height/2; else if( position.y + (height/2) > field_height ) position.y = field_height - (height/2); } // Draws the paddle public void draw(SDLSurface screen) throws SDLException { // Draw the paddle SDLRect rectangle = new SDLRect((int)position.x-(width/2),(int)position.y-(height/2),30,10); screen.fillRect(rectangle,screen.mapRGB(255,255,255)); } // The maximum speed at which it can travel protected float max_speed; // The normalised direction vector that the paddle is travelling in protected c_vector direction; // The current position protected c_vector position; // The position that the paddle is heading towards protected c_vector aim; // Dimensions of the paddle protected int width, height; // Dimensions of our 'playing field' (i.e. the window size) protected int field_width, field_height; } class s04_p01 { // Our SDL surface for drawing on static SDLSurface screen; // The speed of our object static ball our_ball; static paddle our_paddle; // We will call this function whenever we want to exit the program. // It uninitialises SDL and forces a program exit. public static void exit(int value) { SDLMain.quit(); System.out.println("Exiting.."); System.exit(value); } // In this function, we retrieve and handle events. // The function returns true if the user wishes to exit the program. public static boolean handle_events() { try { // SDL stores events in a queue. // To look at events we have two options: // 1. We can wait for events, in which case we call a function that will not return until there is an event. // 2. We can have a quick look at the event queue, using a function that will retrieve an event if there is one, returning null if there isn't. // // In an interactive game, it's not much use if we get stuck waiting for events, as we won't be able to draw or update the scene. // Therefore, we want to use the 'quick look' option. // This is called polling for events. // The waiting technique is known as blocking. // Both methods remove the oldest event from the queue and give it to us to decide what to do with it. // Poll for events SDLEvent event = SDLEvent.pollEvent(); // If there was an event.. if( event != null ) // What we do depends on the type switch( event.getType() ) { case SDLEvent.SDL_QUIT: // This event type is generated when the user clicks on the 'x' to close the window. // We want to exit our handle_events function, returning true to indicate that the user wishes to quit. return true; case SDLEvent.SDL_KEYDOWN: // The user has pressed a key break; case SDLEvent.SDL_KEYUP: // The user has released a key break; case SDLEvent.SDL_MOUSEBUTTONDOWN: // The user has pressed a mouse button /*************************************** ***************************************/ // Set release angle to be random - don't forget it's in radians our_ball = our_paddle.release((float)(Math.random()*2.0f - 1.0f)*(3.141592653589793238462643f),5.0f); /*************************************** ***************************************/ break; case SDLEvent.SDL_MOUSEBUTTONUP: // The user has released a mouse button break; case SDLEvent.SDL_MOUSEMOTION: // The mouse has moved // This event is generated whenever the mouse moves in the window // (it gets called a lot!) // Retrieve the position of the mouse: float x = (float) ((SDLMouseMotionEvent) event).getX(); float y = (float) ((SDLMouseMotionEvent) event).getY(); // We want the paddle to move towards this position: our_paddle.set_position_aim(new c_vector(x,y)); break; } } catch(SDLException e) { System.err.println("Error while polling event : " + SDLMain.getError()); exit(1); } return false; } // In this function, we update our scene public static void update( float deltaT_s ) { // Update our paddle our_paddle.update(deltaT_s); // If we have a ball.. if( our_ball != null ) { // Update it our_ball.update(deltaT_s); // And check to see if it collides with our paddle our_paddle.check_ball_collision(our_ball); } } // In this function, we draw our scene // Upon encountering an error, this function throws an SDLException, which is passed on from SDL function calls public static void draw() throws sdljava.SDLException { // First, we clear the screen screen.fillRect(screen.mapRGB(0,0,0)); our_paddle.draw(screen); if( our_ball != null ) our_ball.draw(screen); screen.flip(); } public static void main(String[] args) { // Create a paddle: our_paddle = new paddle( new c_vector(320.0f,240.0f), 640, 480 ); // Create an SDL object SDLMain sdl = new SDLMain(); // SDL uses exceptions when it encounters problems try { // Initialise SDL to use its video (graphics) capabilities sdl.init(SDLMain.SDL_INIT_VIDEO); // SDL uses what it calls a 'surface' as a window. screen = SDLVideo.setVideoMode(640, 480, 0, 0 ); // Initialise our objects ball.init(); // Timing is one of the most crucial aspects of an interactive game-like program. // Since the graphics on screen can take a variable amount of time to draw, // we cannot rely on a constant frame rate. Besides, if we run our program on a // faster or slower machine, it will be different again. // We must therefore calculate the amount of time that has elapsed between each // iteration of our game loop, and use this when we update things. long previous_time = 0, current_time = 0, deltaT = 0; float deltaT_s = 0.0f; current_time = System.currentTimeMillis(); previous_time = current_time; // This will store whether or not the user has asked to quit boolean quit = false; // This is our main "game loop", and we will run it // while we've not been asked to quit... while( quit == false ) { // When the user presses a key, or uses the mouse, that input is stored in SDL as an 'event'. // As an interactive program, we need to look at these events and decide what to do about them. // Our function here also returns true if we want to quit: quit = handle_events(); // In an interactive program, be it a game or whatever, we usually have a variety of things to update each frame. // These could be character animations, physics simulations, explosions, whatever we want... update(deltaT_s); // Finally, we need to draw our scene draw(); // Update our current time current_time = System.currentTimeMillis(); // Work out the frame rate for that frame deltaT_s = (float)(current_time-previous_time)/1000.0f; // For the next frame, our current time will the the previous one previous_time = current_time; } exit(0); } catch(SDLException e) // Catch SDL problems { // We shall handle any problems by printing a stack trace: e.printStackTrace(); // and an explanation if that wasn't enough System.err.println("SDL encountered a problem"); exit(1); } } }