This example demonstrates multiplayer 3D gaming in a split-screen environment.
Two aliens will be rendered and controlled in two separate viewports. Each player has her own viewport and is given control of one alien’s spaceship, which moves with her camera as she travels. Figure 28-1 shows a split screen for two players. Each player can control her view and position inside the viewport and ultimately travel within the world independently of the other player.
A multiplayer racing game or first-person shooter game uses the same code founda- tion, so converting the logic to suit a different type of 3D multiplayer game is a simple task. Converting this logic to handle more than two players is also straightforward.
When you run this code on the Xbox 360, you will be able to handle two players, each with her own controller. When the code is run on the PC, you can handle either two controllers, or one controller and a mouse/keyboard combination. If you run this code on the PC with only a mouse and keyboard, you will be able to control one of the viewports, but the other viewport will be disabled until a controller is con- nected.
M I C R O S O F T X N A G A M E S T U D I O C R E A T O R ’ S G U I D E
494
495
This example begins with either the MGHWinBaseCode or MGH360BaseCode project, which can be found in the BaseCode folder in the download available from this book’s website.
To enable a two-player game, and to identify each player, you declare the NUMPLAYERS,ALIEN0, andALIEN1definitions at the top of the game class:
const int NUMPLAYERS = 2;
const int ALIEN0 = 0; const int ALIEN1 = 1;
To give each player control to move through the 3D environment, and to allow them to view it independently, you declare an array with two separate instances for the camera. Use this revision to replace the existing camera object declaration:
private Camera[] cam = new Camera[NUMPLAYERS];
C H A P T E R 2 8
MultiplayerGaming
F I G U R E 2 8 - 1
Two viewports for a two-player game. Each player controls her view of the world and can travel independently.
When you’re initializing each camera, the starting position and view position for each person needs to be different. Otherwise, with just a default position and view, when the game begins, the players would all be positioned in the same place, one on top of the other. To set the players up at opposite ends of the world, and to have them looking at their opponent, each instance of the camera is initialized with parameters to set the position and view.
An override to the camera constructor allows you to set the position and view of the camera when it is initialized for each player:
public Camera(Vector3 startPosition, Vector3 startView){
position = startPosition;
view = startView;
up = new Vector3(0.0f, 1.0f, 0.0f);
}
To initialize the camera for the players, you pass their individual starting positions and views to the camera constructor. This is done from theInitialize()method (in the game class) at the program start:
Vector3 position, view;
position = new Vector3( 0.5f, 0.9f, BOUNDARY - 0.5f);
view = new Vector3( 0.5f, 0.7f, BOUNDARY - 1.0f);
cam[0] = new Camera(position, view);
position = new Vector3(-0.5f, 0.9f,-BOUNDARY + 0.5f);
view = new Vector3(-0.5f, 0.7f,-BOUNDARY + 1.0f);
cam[1] = new Camera(position, view);
As mentioned earlier in this chapter, because both viewport heights are half the ac- tual window height, the aspect-ratio parameter in the Projection matrix must be ad- justed. The aspect ratio for the projection becomes (width/(height/2)). To apply this to the Projection matrix, after initializing each camera, replace the call to initialize the Projection matrix in theInitialize()method:
for (int player = 0; player < 2; player++)
cam[player].SetProjection(Window.ClientBounds.Width, Window.ClientBounds.Height/2);
Now that you have properly set up the projection matrix for a multiplayer envi- ronment, you will need to comment out the original call statement to initialize the projection matrix. You can find this call statement inside the InitializeBaseCode()method of the game class:
M I C R O S O F T X N A G A M E S T U D I O C R E A T O R ’ S G U I D E
496
497
C H A P T E R 2 8
MultiplayerGaming
// cam.setProjection(Window.ClientBounds.Width, // Window.ClientBounds.Height);
A routine is needed in the game class to determine how many game controllers are connected so it can assign control to both players accordingly. The TotalControllersUsed()method needed for this example only considers a sit- uation where up to two controllers are used:
int TotalControllersUsed(){
GamePadState gp0 = GamePad.GetState(PlayerIndex.One);
GamePadState gp1 = GamePad.GetState(PlayerIndex.Two);
if (gp0.IsConnected && gp1.IsConnected) return 2;
else if (gp0.IsConnected) return 1;
return 0;
}
TheChangeView(),Move(),Strafe(), andDrawGround()methods inside the game class need to be modified so they can be used for each player. Since each player is a viewport owner, these method headers must be adjusted to accept the player number, as follows:
Vector2 ChangeView(GameTime gameTime, int player) float Move(int player)
float Strafe(int player) void DrawGround(int player)
TheChangeView(),Move(), andStrafe()methods are called once for each viewport owner. These methods must then select the correct input device to allow each player to control their view and position.
For this example, by default, aGamePadStateobject is set for the first player us- ing the PlayerIndex.One parameter value regardless of whether a controller is connected or not. If zero or one controllers are connected on a PC, the mouse is desig- nated for the first player to control their viewport. If two game controllers are con- nected on the PC, theGamePadStateobject is set for the second player using the PlayerIndex.Twoparameter.
The following code must be added in the ChangeView(), Move(), and Strafe()methods after theGamePadStateobject,gp, is declared:
bool useMouse = false;
int totalControllers = TotalControllersUsed();
// when fewer than two controllers connected use mouse for 1st player if (totalControllers <2 && player == 0)
useMouse = true;
// when 2 controllers connected use 2nd controller for 2nd player else if (totalControllers == 2 && player == 1)
gp = GamePad.GetState(PlayerIndex.Two);
Also, to ensure that code for the game pad is executed on the PC when either one or two controllers are connected, inside the ChangeView(), Move(), and Strafe()methods, replace:
if(gp.IsConnected == true) with
if(!useMouse)
The code that sets the WorldViewProjection matrix insideDrawGround()must also be adjusted according to the player’s view. This adjustment sets the WorldViewProjection matrix so it is drawn according to each viewport owner’s view:
textureEffectWVP.SetValue(world * cam[player].viewMatrix
* cam[player].projectionMatrix);
Because each player has a separate instance of the camera, each camera must be updated separately; this gives the players the ability to travel independently in the game world. To enable this, insideUpdate(), replace the code that handles the cam- era’s time tracking, forward movement, backward movement, and strafing with this revision to update these values for each player:
for (int player = 0; player < NUMPLAYERS; player++){
// update timer in camera to regulate camera speed cam[player].SetFrameInterval(gameTime);
// adjust camera position and view cam[player].Move(Move(player));
cam[player].Strafe(Strafe(player));
cam[player].SetView(ChangeView(gameTime, player));
}
When the viewports are being drawn (while your code is run on the Xbox 360), it is possible that the full window may not be visible—it may fall outside the title-safe region. As discussed previously in examples where 2D graphics are used, on some
M I C R O S O F T X N A G A M E S T U D I O C R E A T O R ’ S G U I D E
498
499
C H A P T E R 2 8
MultiplayerGaming
televisions, this nonvisible range may be as high as 20 percent. To adjust for this pos- sibility, the starting top-left pixel that is used as the viewport should allow for this po- tential difference. When viewports are used, you are going to want to account for this possibility so that your 3D graphics do not appear to be off center if truncation oc- curs. To fix this, add this version of theTitleSafeRegion()method to obtain the bounding margins for the top viewport. These margins in turn will be used to deter- mine the starting top-left pixel for each viewport in this demonstration:
Rectangle TitleSafeRegion(){
int windowWidth = Window.ClientBounds.Width;
int windowHeight = Window.ClientBounds.Height;
#if Xbox
// some televisions only show 80% of the window
Vector2 start = new Vector2(); // starting pixel X & Y const float UNSAFEAREA = 0.2f; // 80% not visible on
// Xbox 360 start.X = windowWidth * UNSAFEAREA/2.0f;
start.Y = windowHeight * UNSAFEAREA/2.0f;
// ensure viewport drawn in safe region on all sides return new Rectangle(
(int)start.X, (int)start.Y,
(int)((1.0f-UNSAFEAREA)* windowWidth), (int)((1.0f-UNSAFEAREA)* windowHeight/2));
#endif
// PC show the entire region
return new Rectangle(0, 0, windowWidth, windowHeight/2);
}
The next method needed in your game class is the CurrentViewport()method to set your viewport. In a multiplayer game, theDraw()method must trigger rendering of the entire scene for each viewport. Before drawing each viewport, you must set the top-left pixel where the viewport begins, the height and width properties for each viewport, and the clip minimum and maximum. If the clip minimum and maximum values are not set between 0 and 1, your 3D models will not render properly:
Viewport CurrentViewport(int player){
Viewport viewport = new Viewport();
Rectangle safeRegion = TitleSafeRegion();
Vector2 startPixel = new Vector2((float)safeRegion.Left, (float)safeRegion.Top);
// get starting top left pixel for viewport
if (player == 1) // 2nd player - bottom startPixel.Y += (float)safeRegion.Height;
// assign viewport properties
viewport.X = (int)startPixel.X; // top left pixel X viewport.Y = (int)startPixel.Y; // top left pixel Y viewport.Width = safeRegion.Width; // pixel width viewport.Height = safeRegion.Height; // pixel height
viewport.MinDepth = 0.0f; // depth is between viewport.MaxDepth = 1.0f; // 0 & 1 so models
// appear properly return viewport;
}
When multiple viewports are used, each one is rendered separately. In effect, the same scene is drawn more than once. With careful planning, the same methods can be used for drawing your primitive objects and models. Note also that when drawing with multiple viewports, the viewport is set first before the viewport is cleared. The drawing is performed afterward.
Replace the existingDraw()method with this revision to draw all objects in each viewport according to the view and perspective of each player:
protected override void Draw(GameTime gameTime){
for (int player = 0; player < NUMPLAYERS; player++) { // set the viewport before clearing screen
graphics.GraphicsDevice.Viewport = CurrentViewport(player);
graphics.GraphicsDevice.Clear(Color.CornflowerBlue);
// draw objects DrawGround(player);
}
base.Draw(gameTime);
}
If you ran the project now, you would see two viewports. Remember that this code can serve as a base for any multiplayer game.
To make this demonstration more interesting, two aliens will be added. Each player will be given control of an alien, which will be used as a third-person charac- ter. The alien will move with the camera. This not only allows each player to control her own spaceship, but also enables her to view the movements of her opponent in her own viewport.
M I C R O S O F T X N A G A M E S T U D I O C R E A T O R ’ S G U I D E
500
501
For this example, you can use the alien models in the Models folder in the book’s download. To do this, obtain the alien0.fbx, alien1.fbx, and spaceA.bmp files from the Models folder. Create a Models folder in your project and reference the .fbx files from the Solution Explorer.
To load these models and to control their transformations, declarations for the model objects and their bone-transformation matrices are required at the top of the game class:
Model alien0Model; Model alien1Model;
Matrix[] alien0Matrix; Matrix[] alien1Matrix;
The code used to load these two models and their accompanying transformation matrices is contained in the InitializeAliens()method. To initialize the models, add this method to your game class:
void InitializeAliens(){
alien0Model = Content.Load<Model>("Models\\alien0");
alien0Matrix = new Matrix[alien0Model.Bones.Count];
alien0Model.CopyAbsoluteBoneTransformsTo(alien0Matrix);
alien1Model = Content.Load<Model>("Models\\alien1");
alien1Matrix = new Matrix[alien1Model.Bones.Count];
alien1Model.CopyAbsoluteBoneTransformsTo(alien1Matrix);
}
To load the aliens when the program begins, add the call statement InitializeAliens()to theInitialize()method:
InitializeAliens();
For this two-player game, one alien and its spaceship are controlled by each player. Each alien’s spaceship moves with the player’s camera. To rotate the alien about the Y axis—so it always points in the direction it is traveling—use the follow- ing method to calculate the angle of direction based on the camera’sLookdirection:
float RotationAngle(Vector3 view, Vector3 position){
Vector3 look = view - position;
return (float)Math.Atan2((double)look.X, (double)look.Z);
}
To save on code, the same method is used to draw both aliens. When these items are rendered in a viewport, this method is called once for each model. This process is
C H A P T E R 2 8
MultiplayerGaming
repeated for each player. For this example,alien0’s position and angle of orienta- tion is based on the first player’s camera.Alien1’s position and orientation is based on the second player’s camera. The view is adjusted for each player. The rest of the routine is identical to the routines you have already used in this book for drawing models.
void DrawAliens(int player, Model model, int modelNum){
foreach (ModelMesh mesh in model.Meshes){
// 1: declare matrices
Matrix world, scale, rotationY, translation, translationOrbit;
// 2: initialize matrices
scale = Matrix.CreateScale(0.5f, 0.5f, 0.5f);
translation = Matrix.CreateTranslation(Vector3.Zero);
rotationY = Matrix.CreateRotationY(MathHelper.Pi);
translationOrbit = Matrix.CreateTranslation(0.0f, 0.0f, 1.0f);
translation = Matrix.CreateTranslation( // one alien cam[modelNum].position.X, // is located cam[modelNum].position.Y - 0.6f, // at each cam[modelNum].position.Z); // camera float angleY = RotationAngle(cam[modelNum].view,
cam[modelNum].position);
rotationY = Matrix.CreateRotationY(angleY);
// 3: build cumulative world matrix using I.S.R.O.T. sequence // identity, scale, rotate, orbit(translate & rotate), translate world = scale * translationOrbit * rotationY * translation;
foreach (BasicEffect effect in mesh.Effects){
// 4: pass wvp to shader
effect.View = cam[player].viewMatrix;
switch (modelNum){
case ALIEN0:
effect.World =
alien0Matrix[mesh.ParentBone.Index] * world; break;
case ALIEN1:
effect.World =
alien1Matrix[mesh.ParentBone.Index] * world; break;
}
effect.Projection = cam[player].projectionMatrix;
// 4b: set lighting
effect.EnableDefaultLighting();
M I C R O S O F T X N A G A M E S T U D I O C R E A T O R ’ S G U I D E
502
503
}
// 5: draw object mesh.Draw();
} }
To draw each model, add these call statements to the end of theDraw()method inside the for-loop that triggers drawing for each player’s viewport:
DrawAliens(player, alien0Model, ALIEN0);
DrawAliens(player, alien1Model, ALIEN1);
When you run this version of the code, each of the two players can control the movement of her alien separately. In addition, she can view her world and travel in- dependently of the other player.
Being able to shift the view up and down might be useful for a first-person shooter game, but it doesn’t look right for this setup. InsideChangeView(), you can pre- vent the camera from bobbing up and down by modifying the return statement to set the Y view to zero:
return new Vector2(change.X, 0.0f);
When you run the code now, each player will be able to control her view of the world. This example was kept simple, but you can apply this logic for different types of games, such as first-person shooter, racing, role-playing, or adventure games.