Skip to content

Commit

Permalink
Merge branch 'main' into typeo
Browse files Browse the repository at this point in the history
  • Loading branch information
SimonDarksideJ authored Nov 13, 2024
2 parents 91253a4 + 69a8343 commit 180c22a
Show file tree
Hide file tree
Showing 16 changed files with 697 additions and 19 deletions.
12 changes: 11 additions & 1 deletion articles/getting_started/2_choosing_your_ide_visual_studio.md
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,16 @@ When installing Visual Studio, the following workloads are required depending on

To create new MonoGame projects from within Visual Studio 2022, you will need to install the **MonoGame Framework C# project templates** extension. The following steps demonstrate how to install the extension.

> [!WARNING]
> **Visual Studio Extension Issues**
>
> The extension that is installed by Visual Studio 2022 is currently outdated.
> Instead of using Visual Studio 2022 to install the templates, run the following command:
> ```sh
> dotnet new install MonoGame.Templates.CSharp
> ```
> After doing this, you should be able to launch Visual Studio 2022 and create a new project with the newly installed templates.
1. Launch Visual Studio 2022
2. Select **Continue without code**. This will launch Visual Studio without any project or solution opened.
Expand Down Expand Up @@ -86,4 +96,4 @@ To get you started with Visual Studio, here are the steps for setting up a new M
Next, get to know MonoGame's code structure and project layout:
- [Understanding the code](3_understanding_the_code.md)
- [Understanding the code](3_understanding_the_code.md)
8 changes: 5 additions & 3 deletions articles/getting_started/packaging_games.md
Original file line number Diff line number Diff line change
Expand Up @@ -174,7 +174,9 @@ However, you do need to currently add some additional settings to your `.csproj`

The `TrimmerRootAssembly` stops the trimmer removing code from these assemblies. This should allow the game to run without
any issues. However if you are using any Third Party or additional assemblies, you might need to add them to this list or fix your code to be `Aot` compliant.
It is recommended that you publish using AOT as it simplifies the app bundle.
It is recommended that you publish using AOT as it simplifies the app bundle.

You may see some trim and AOT analysis warnings related to MonoGame when using PublishAOT, even after adding `TrimmerRootAssembly` - these are normal and should not present any issue.

See [Trim self-contained deployments and executables](https://learn.microsoft.com/en-us/dotnet/core/deploying/trimming/trim-self-contained) for more information.

Expand Down Expand Up @@ -205,9 +207,9 @@ MonoGame templates for .NET projects explicitly set this to `false`.
Tiered compilation is **enabled by default**. To disable it, set the `TieredCompilation` property to `false` in your `.csproj`.
MonoGame templates for .NET projects explicitly set this to `false`.

### SingleFilePublish
### PublishSingleFile

SingleFilePublish packages your game into a single executable file with all dependencies and content integrated.
PublishSingleFile packages your game into a single executable file with all dependencies and content integrated.

While it sounds very convenient, be aware that it's not magical and is in fact a hidden self-extracting zip archive. As such, it may make app startup take **a lot** longer if your game is large, and may fail to launch on systems where user permissions don't allow extracting files (or if there is not enough storage space available).

Expand Down
4 changes: 2 additions & 2 deletions articles/getting_to_know/howto/graphics/HowTo_Tile_Sprites.md
Original file line number Diff line number Diff line change
Expand Up @@ -12,11 +12,11 @@ We will simply define an area ([Rectangle](xref:Microsoft.Xna.Framework.Rectangl

### End result

![The output of this tutorial](./images/HowTo_Tile_Sprites_Final.png)
![The output of this tutorial](images/HowTo_Tile_Sprites_Final.png)

## Requirements

![Sprite to Tile](./images/MonoGame.png)
![Sprite to Tile](images/MonoGame.png)

## Drawing a Tiled a Sprite

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,279 @@
---
title: Understanding different camera modes
description: Demonstrates Several camera views in a single script.
requireMSLicense: true
---

## Overview

This example is different from most as it is demonstrating several techniques to update a camera's view and position to achieve different camera modes or operations, including:

- A Fixed Camera.
- A Fixed tracking camera - follows the target.
- A first person camera.
- A third person camera - with spring.
- A top-down fixed camera view.
- A top-down view centered on a target.

### End result

These are the kinds of camera modes demonstrated in this sample:

|Fixed View|Fixed Tracking|First Person View|
|-|-|-|
|![Fixed](../images/HowTo_CameraModes_Fixed.gif)|![Fixed Tracking](../images/HowTo_CameraModes_Tracking.gif)|![First Person](../images/HowTo_CameraModes_FirstPerson.gif)|
|Third Person View|Top Down|Top Down tracking|
|![Third Person](../images/HowTo_CameraModes_ThirdPerson.gif)|![Top-Down](../images/HowTo_CameraModes_TopDownFixed.gif)|![Top Down tracking](../images/HowTo_CameraModes_TopDownTracking.gif)|

## Sample

As the sample is quite large and contains several pieces of content for the demonstration, the full source is available here:

### [Basic Camera Example Source](https://github.com/MonoGame/MonoGame.Samples/tree/3.8.2/Tutorials/BasicCameraExample)

## Understanding the different camera modes

To understand how these different camera modes work, the principle is fairly strait forward.

> [!TIP]
> Please review the [What is Camera](../../../whatis/graphics/WhatIs_Camera.md) guide for more detail on the concepts discussed here.
### World

The world that the camera and models are contained in usually static (at least for a single "world" or game level), for simplicity, this is usually kept at the center of the world which is `0,0,0` or `Matrix.Identity`.

All content is then Transformed / positioned relative to this central world point.

### Projection

The project of the view is also normally fixed for any single camera (although you might use different perspectives for different cameras, e.g. a map view). This is defined once in either a Perspective or Orthographic (with or without depth). This is then applied to the drawing methods to define the bounds of what is drawn to the screen.

> [!TIP]
> Elements drawn outside of the camera view / projection are culled or ignored by the graphics device. However, unless you control what data is sent, you are still paying for the performance to send to the graphics card and consider it for drawing. This is where techniques are used to limit the data sent to the graphics card and ONLY send models and textures that you know will be actually drawn and limiting what is discarded.
## Camera Views

The sample separates out the actual work in drawing each camera mode based on what is active at the time in the `Update` loop, as shown here:

```csharp
switch (currentCameraMode)
{
case CameraMode.Fixed:
UpdateFixedCamera();
break;

case CameraMode.Tracking:
UpdateTrackingCamera();
break;

case CameraMode.FirstPerson:
UpdateFirstPersonCamera();
break;

case CameraMode.ThirdPerson:
UpdateThirdPersonCamera((float)gameTime.ElapsedGameTime.TotalSeconds);
break;

case CameraMode.TopDownFixed:
UpdateTopDownFixedCamera();
break;

case CameraMode.TopDownCentred:
UpdateTopDownCenteredCamera();
break;
}
```

You would not normally do this, it is only done this way for the purposes of the sample. In reality you would only use the mode relevant to the current game view (although a lot of first person games do allow switching between First Person and Third Person), however, as you will see, you are only updating the way the View Matrix is generated along with the camera position, in 3D the content is always drawn the same regardless.

### The Fixed Camera view

This is the simplest way of rendering a camera, you just create a `View Matrix` using the Camera's current position and looking at the center of the world (or whatever you want it to look at). For example:

```csharp
void UpdateCameraView(Vector3 aCameraPosition, Vector3 aCameraTarget)
{
// Update the camera view matrix with the `Matrix.LookAt` method.
currentCameraView = Matrix.CreateLookAt(aCameraPosition, aCameraTarget, Vector3.Up);
}

// Set the position of the camera in world space, for the fixed camera view matrix.
private Vector3 cameraFixedPosition = new Vector3(0.0f, 1550.0f, 5000.0f);

void UpdateFixedCamera()
{
// Fixed view, the camera is always in the same position and looking the same way, no updates.
UpdateCameraView(cameraFixedPosition, Vector3.Zero);
}
```

### The Tracking Camera

Giving the camera something to look at is also quite simple, instead of looking at the center of the world (assuming the world is `Matrix.Identity`), we instead "Look at" the model itself, dymanically updated each frame, as follows:

```csharp
// Set the position of the camera in world space, for the fixed camera view matrix.
private Vector3 cameraFixedPosition = new Vector3(0.0f, 1550.0f, 5000.0f);

void UpdateTrackingCamera()
{
// Tracking view, the camera is always in the same position but changes the view matrix to "look" towards a target.
// Set up our world matrix, view matrix and projection matrix.
UpdateCameraView(cameraFixedPosition, modelPosition);
}
```

Run once per update, wherever in the world the target ends up, the camera will rotate to `LookAt` the target.

### The First Person Camera

There are several ways of achieving a first person view, some use the camera as the player and update the camera position, some (as in this example) take the player position and then transform the camera to it. Either way is fine (although in practice, if you want to use multiple camera views, it is better to keep them separate).

In this example, we apply all the same transformations to the camera we perform on the "player" model to achieve this effect:

```csharp
// 1st Person camera position relative to player model
private Vector3 cameraFirstPersonPosition = new Vector3(0.0f, 50.0f, 500.0f);

void UpdateFirstPersonCamera()
{
// First person view, the camera moves based on the Model's position (which is moved by input) and the view matrix is updated to always look "forward" from the model.
// Create a rotational matrix based on the current model's heading.
Matrix rotationMatrix = Matrix.CreateRotationY(modelRotation);

// Create a vector pointing the direction the camera is facing.
Vector3 transformedReference = Vector3.Transform(cameraFirstPersonPosition, rotationMatrix);

// Calculate the position the camera is looking from.
currentCameraPosition = transformedReference + modelPosition;

// Set up our world matrix, view matrix and projection matrix.
UpdateCameraView(currentCameraPosition, modelPosition);
}
```

You may want to look at other examples of First Person translation for cameras before deciding what works best for your game.

### Third Person Camera

Compared to the first person camera, the Third Person view is a much simpler approach, as the Model/Player and the Camera have to be kept separate. In most scenarios though, you have more considerations that the basic first person view, such as:

- Wall clipping
- Obscuring by other world objects
- Bad viewing angles

These are two complex to go into for now, so you may want to do some further reading.

In this sample we have included a simple physics based "Spring" system, so that the camera's view follows the player in a much smoother way (it also looks REALLY cool when the players ship "jumps" to a new location, try resetting the ships position after moving in the sample in this mode). The code is as follows:

```csharp
// 3rd Person camera position relative to player model
private Vector3 cameraThirdPersonPosition = new Vector3(0.0f, 1550.0f, 5000.0f);

void UpdateThirdPersonCamera(float aElapsed)
{
// First person view, the camera is offset behind and above the model and moves with it,the view matrix is updated to always look "forward" from the model.
// It also includes an optional spring physics system to smooth out the camera movement.
// Create a rotational matrix based on the current model's heading.
Matrix rotationMatrix = Matrix.CreateRotationY(modelRotation);

// Create a vector pointing the direction the camera is facing.
Vector3 transformedReference = Vector3.Transform(cameraThirdPersonPosition, rotationMatrix);

// If camera spring is enabled, update the position and rotation of the camera over several frames
if (cameraSpringEnabled)
{
// Calculate the position where we would like the camera to be looking from.
Vector3 desiredPosition = transformedReference + modelPosition;

// Calculate spring force
Vector3 stretch = currentCameraPosition - desiredPosition;
Vector3 force = -cameraStiffness * stretch - cameraDamping * cameraVelocity;

// Apply acceleration
Vector3 acceleration = force / cameraMass;
cameraVelocity += acceleration * aElapsed;

// Apply velocity
currentCameraPosition += cameraVelocity * aElapsed;
}
// If not, just move the camera without smoothing
else
{
// Calculate the position the camera is looking from.
currentCameraPosition = transformedReference + modelPosition;
}

// Set up our world matrix, view matrix and projection matrix.
UpdateCameraView(currentCameraPosition, modelPosition);
}
```

The main difference between the First Person and Third Person views is that the "Position" of the camera is offset behind and above the center of the player, this is then transformed and added to the players current position before constructing the view.

### Top Down Camera

The same as the [Fixed Camera](#the-fixed-camera-view) view, the camera simply has a position and looks at the center of the world, however, it is positioned far above the world so that it is looking down.

> [!TIP]
> In this sample it has been kept as a Perspective Camera and evaluates depth, however, if you are having a Map camera then it is more likely to use an orthographic projection instead.
```csharp
// Top Down camera position relative to player model
private Vector3 cameraTopDownPosition = new Vector3(0.0f, 25000.0f, 1.0f);

void UpdateTopDownFixedCamera()
{
// A Top-Down fixed view, the camera is always in the same position and looking down onto the game scene.
// Note, there are no boundaries to prevent the model from moving out of view.
// Set up our world matrix, view matrix and projection matrix.
UpdateCameraView(cameraTopDownPosition, Vector3.Zero);
}
```

### Top Down Tracked Camera

To finish off these camera modes, the top-down tracked view is more of a combination of a third person and top-down view, wherein the camera is way above the world and follows only the target on a 2D plane, like the third person view the camera is simply placed above the model/player and then transforms/updates its position based on the movement of the player.

```csharp
// Top Down camera position relative to player model
private Vector3 cameraTopDownPosition = new Vector3(0.0f, 25000.0f, 1.0f);

void UpdateTopDownCenteredCamera()
{
// A Top-Down view that moves according to two dimensional position of the model, looking down onto the model.
// Create a rotational matrix based on the current model's heading.
Matrix rotationMatrix = Matrix.CreateRotationY(modelRotation);

// Create a vector pointing the direction the camera is facing.
Vector3 transformedReference = Vector3.Transform(cameraTopDownPosition, rotationMatrix);

// Calculate the position the camera is looking from.
currentCameraPosition = transformedReference + modelPosition;

// Set up our world matrix, view matrix and projection matrix.
UpdateCameraView(currentCameraPosition, modelPosition);
}
```

## See Also

- [How to display Multiple Screens with Viewports](../HowTo_UseViewportForSplitscreenGaming.md)

### Concepts

- [What Is 3D Rendering?](../../../whatis/graphics/WhatIs_3DRendering.md)
- [What Is a Viewport?](../../../whatis/graphics/WhatIs_Viewport.md)
- [What Is a View Frustum?](../../../whatis/graphics/WhatIs_ViewFrustum.md)
- [What Is a Render Target?](../../../whatis/graphics/WhatIs_Render_Target.md)

### Reference

- [GraphicsDevice.Viewport](xref:Microsoft.Xna.Framework.Graphics.GraphicsDevice)
- [Matrix](xref:Microsoft.Xna.Framework.Matrix)
- [SpriteBatch](xref:Microsoft.Xna.Framework.Graphics.SpriteBatch)
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file modified articles/getting_to_know/howto/graphics/images/MonoGame.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading

0 comments on commit 180c22a

Please sign in to comment.