Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
All of FlatRedBall, including the editor and the core engine, is open source. If you are using FlatRedBall in your game, you may want to use source. Specifically you may want to:
Build the FlatRedBall Editor (aka Glue) from source to get the most up-to-date features and code generation
Link your game to source to improve debugging. For information on how to do this, see the Linking Games to FlatRedBall page.
Most users do not need to follow the steps outlined in this section. New users are encouraged to skip this and head over to the Tutorials section. If you would like to build FlatRedBall to do more advanced debugging, contribute to the project, or out of pure curiosity then feel free to continue through this section.
If you are interested in using FlatRedBall in a code-only project, see the Code-Only Projects page.
You first need to clone FlatRedBall and Gum (two separate repositories) whether you are going to run the FlatRedBall or link your game to source.
Currently all development is being done on the NetStandard branch:
If you plan on using FlatRedBall FNA, be sure that you include all submodules when pulling FlatRedBall. This may automatically happen if you are using some Git clients (such as GitHub Desktop), but you may need to perform additional configuration if using a client that doesn't pull submodules by default (such as Rider).
FlatRedBall uses Gum UI including at runtime. This is also open source and can be found on GitHub:
Both repositories should be cloned to the same folder to make linking easier.
FlatRedBall Editor (Glue) links a version of FlatRedBall which builds for multiple platforms including Android and iOS. You must have an installiation that supports these platforms even if you are only building the FlatRedBall Editor.
Be sure to install both workloads:
.NET Multi-platform App UI development
.NET desktop development
The easiest way to download and keep FlatRedBall source up to date is to use a Github client such as Github for Desktop. To download source using Github for Desktop:
Install Github for Desktop
Select File -> Clone Repository
Select the URL tab
Enter the FlatRedBall URL: https://github.com/vchelaru/FlatRedBall. Keep the default folder as FlatRedBall
Switch to the appropriate branch. The default branch is NetStandard
After FlatRedBall is cloned, you will have a local copy on your machine.
You will also need to download Gum source, as FlatRedBall games which link against source also reference Gum files. To do this (assuming Github for Desktop):
Select File -> Clone Repository
Select the URL tab
Enter the Gum source: https://github.com/vchelaru/Gum . Keeping the default folder is recommended, but if not, be sure to use the same root as the FlatRedBall clone above.
Verify that you have .NET 6 SDK installed https://dotnet.microsoft.com/en-us/download/dotnet/thank-you/sdk-6.0.309-windows-x64-installer
To build the FlatRedBall Editor (also referred to as Glue):
Download the FlatRedBall repository (either clone it or download the .zip file from Github) following the steps above
Download the Gum repository following the steps above. Make sure to download it to the same folder where you downloaded the FlatRedBall repository, and name the folder Gum. See below for an example. If you do not do this, you will have reference errors when opening the solution.
Open the file <FlatRedBall Root>\FRBDK\Glue\Glue with All.sln.
To rebuild Glue with all plugins, select Build -> Build Solution in Visual Studio. You must Build or Rebuild the first time you run FlatRedBall. If you make further changes to any plugins, you must either build the entire solution, or build the project that contains the plugin. Simply building the Glue project, or pressing F5 to build and run Glue, will not build all plugins.\
Set Glue as the Startup Project
Building the solution creates an .exe from which the FRB editor can be run. You can find this .exe at <your git repo location>\FlatRedBall\FRBDK\Glue\Glue\bin\Debug\GlueFormsCore.exe. Alternatively, you can start up the FRB editor from Visual Studio. Doing so in Debug Mode allows you to see error messages in case the FRB editor throws an exception.
It's important that Gum and FlatRedBall are checked out to the same folder. For example, you may have:
c:/MyDocuments/FlatRedBall
c:/MyDocuments/Gum
The Glue project references Gum projects, so the two project folders must be in the same root folder. To verify:
Open Glue with All.sln
Open the Solution Explorer
Expand the GumProjects folder
If the references are correct, your window should look similar to the following image:
If your references are broken, then you may see something similar to the following image:
Notice that the projects are marked as unloaded. To solve this make sure that both of the projects (FlatRedBall and Gum) are in the same root folder. If you unzipped the files, then they may need to be moved up one folder.
When opening the project, you may get a message which states:
For information on how to solve this problem, see the following link: http://www.c-sharpcorner.com/UploadFile/8a67c0/visual-studio-2015-feature-shared-project-part-2/
Click the Clone button
For Rider users, you will need to set Glue as the project to run in the Configuration dialog. \
https://visualstudio.microsoft.com/vs/community/ Although it is possible to make games without Visual Studio or Rider, doing so requires advanced knowledge of MSBuild. We recommend downloading and installing Visual Studio Community which is free.
At a minimum you need to install .NET desktop development.
https://www.microsoft.com/en-us/download/details.aspx?id=27598 Although this is not required to build and run FlatRedBall games, it is required to use Gum, which is the preferred FlatRedBall UI tool.
FlatRedBall projects are built with .NET 8 or newer. If you are using Visual Studio then you do not need to explicitly install .NET 8. If you are using a different IDE such as Visual Studio Code, then you need to install .NET SDK 8:
https://dotnet.microsoft.com/en-us/download/dotnet/8.0
You must also install the .NET 6 SDK even if you have .NET 8 installed since the FlatRedBall Editor relies on this version for loading projects. This may change in a future version of FlatRedBall:
https://dotnet.microsoft.com/en-us/download/dotnet/6.0
Newer Versions of Visual Studio (as of version 17.5.1) install .NET SDK 7.0 or newer which have a bug preventing projects from being loaded in the FlatRedBall Editor. Therefore, you need to manually install .NET 6 SDK.
This dependency is required to build shader files. If you are certain that you will not be using any custom shaders or post processing, then you can skip this installation. However, we recommend installing this to avoid confusing errors if you do end up using any shaders.
https://www.microsoft.com/en-us/download/details.aspx?id=40784
The most common approach to making FlatRedBall games is to use the FlatRedBall Editor. The Editor can be downloaded from a pre-built .zip file, or it can be built from source. New users should download the pre-built .zip file.
Download the latest zip file from https://files.flatredball.com/content/FrbXnaTemplates/DailyBuild/FRBDK.zip.
Alternatively, the FlatRedBall Editor (no additional tools) prebuilt can be downloaded from Github. This is not recommended for new users, but experienced users can replace the FlatRedball Glue folder with the contents from the built files: https://github.com/vchelaru/FlatRedBall/actions
Right-click the ZIP file and chose Properties
Check-mark the Unblock option and Click Apply
Close the Properties window
Unzip the file after downloading
Go to the folder where the .zip file unzipped to (by default called FRBDK)
Open the Run FlatRedBall.bat file (double click it)
If you see the Windows protected your PC dialog, click More info -> Run Anyway
The FlatRedBall Editor should appear.
If you're stuck on a problem or if you believe you've found a bug, there's lots of ways to get in touch with us.
(Optional) Unblock the ZIP file. This will prevent the windows protected your PC warning.
- the easiest way to get help, report a bug, or just say "hi" is to go to the FlatRedBall chat. We love helping new users, so hop over to the ❓| Help channel and ask us anything.
- If you have an issue or would like to start a discussion about a particular topic, you can do it on our github page.
- Of course, if you prefer to fix it yourself, want to see the details of what's been changing, or would like to browse the code, it's all available on Github.
To add the FRB source to your project:
Make sure you have already cloned the FlatRedBall and Gum repositories. We recommend using Github for Desktop and cloning to the default location so that your game and FlatRedBall are sibling folders in the same parent folder. For more information on cloning FlatRedBall, see the main Building FlatRedBall From Source page. For more information on cloning Gum, see this section on the main Building FlatRedBall From Source page.
Open your project in the FlatRedBall Editor
Select the Project -> Link Game to FRB Source menu item
The Add FRB Source tab appears, showing a text box for FlatRedBall and Gum root folders. If your current project is also a Git project which is cloned to the same folder as FlatRedBall and Gum, then the FRB Editor attempts to fill in the file paths.
If your paths are blank or incorrect, use the ... button to select the file paths for each repository. Select the root folder for where Gum and FRB repositories.
The FRB Root Folder is the folder where FlatRedBall is cloned locally. For example, if you use the default folders when cloning in Github Desktop, this would be <Documents Folder>\GitHub\FlatRedBall\
. This folder contains the root-most FlatRedBall files like LICENSE and README.doc
The Gum Root Folder is the folder where Gum is cloned locally. For example, if you use the default folders when cloning in Github Desktop, this would be <Documents Folder>\GitHub\Gum\
. This folder contains the root-most Gum files like Gum.sln
If you are planning to use Gum Skia, check the option
Click Link to Source
After your project is linked, the Add FRB Source tab will disappear
Your game project should now directly reference the FlatRedBall Source.
The FlatRedBall Editor attempts to link FlatRedBall source regardless of the location of your projects relative to source. We strongly recommend keeping your project in the same directory (such as C drive) as FlatRedBall source. Otherwise, projects will be linked using absolute paths which makes your project far less portable. By using absolute paths others who clone your project must have the same exact folder structure as you do or the project will not build and run.
Furthermore, Github for Desktop always clones projects to the same directory: C:\YouserUerName\Owner\Documents\GitHub\YourProjectName
We recommend keeping these defaults for all repositories - your game and FlatRedBall. By using default locations others can clone the engine and game without needing ot make modifications to their defaults.
If you would like to use the engine source in your game project:
Open your game project in Visual Studio
Expand the game project in the solution explorer
Expand the References item
Find the FlatRedBall entries. This is the reference to the prebuilt-dll. Note that these may be direct references or NuGet packages depending on which version of FlatRedBall you are using, so be sure to check under both Assemblies and Packages. Press the Delete key on all references as mentioned below:
FlatRedBall DesktopGL:
FlatRedBallDesktopGLNet6
FlatRedBall.Forms.DesktopGlNet6
GumCore.DesktopGlNet6
SkiaInGum
StateInterpolation.DesktopNet6
FlatRedBall FNA:
FlatRedBall.FNA
FlatRedBall.Forms.FNA
GumCore.FNA
StateInterpolation.FNA
Right-click on the solution
Select Add -> Existing Project...
Navigate to the location of the FlatRedBall .csproj file for your given platform. For example, for PC, add <FlatRedBall Root>\Engines\FlatRedBallXNA\FlatRedBall\FlatRedBallDesktopGL.csproj
Click Open to add the project to your game's solutionComment
Right-click on your game's References item and select Add Reference...Comment
Click the "Projects" categoryComment
Check the FlatRedBallDesktopGL project (or whichever FlatRedBall project used for the given platform)Comment
Repeat the process for the other librariesComment
Click OK
Build and run your project
If you're looking to get started without going through lengthy tutorials, this guide will get you running with the minimum number of steps.
Note that this guide assumes you are using the FlatRedBall Editor. If you are creating a code-only project, see the Code Only Projects page.
First you will need to make sure you have the prerequisites. Once you do, you need to download FlatRedBall. For detailed steps, see the Downloading FlatRedBall page.
To create a new project:
Click the Create New Project button
Leave the defaults and click the Create Project! button. The defaults select the most common project setup.
After the project loads, the Project Setup Wizard appears. Select which project type you would like such as Standard Platformer or Standard Top Down. These options are the quickest way to get a functional game running. If you would like to have an empty project, close the wizard by pressing the X button at the top right.
If you selected a project type, wait for the project creation to finish.
You now have a FlatRedBall Project. You can run the project through the FlatRedBall Editor, or in Visual Studio.
To run your Project in FlatRedBall, click the play button. Notice that this tells you which Screen will load when the project starts.
Once the project is running you will see it in its own window.
To run your project in Visual Studio:
Click the Visual Studio icon in FlatRedBall\
Wait for the project to load in Visual Studio
Press the Play button to build and run your project in Visual Studio. FlatRedBall projects are standard .NET projects, which means you can treat it as you would any other .NET project including debugging, NuGet packages, and full C# syntax support.
Now that you have FlatRedBall running on your machine, you can get started making games. Here are some next steps to consider:
Join the FlatRedBall Discord - come say "hi" and tell us about what you want to do with FlatRedBall. We have an active community who loves to help new users.
Check out the Beefball tutorials and other tutorials. These will get you up to speed on how to use the FlatRedBall Editor.
If you prefer to dive right in to the docs, check out the FlatRedBall Editor reference to see what it can do, or the FlatRedBall API Documentation for info about working with FlatRedBall in code.
Try things out! The best way to learn is by doing, so try making changes to the project and explore what is already there.
If you're looking for a jump-start on how to make games with FlatRedBall and the FlatRedBall Editor, you've come to the right place. The following tutorials walk you through how to make a game called Beefball. Beefball is a fast-paced physics based game similar to air-hockey. What are you waiting for, let's get started!
Completed Project: Beefball.zip
This tutorial covers how to create Beefball with the FlatRedBall Editor, which is the typical way to work with FlatRedBall. Of course, you can also use FlatRedBall without the editor. If you are interested in developing in a code only environment, you can download the BeefballCodeOnly project which is located in the FlatRedBall Samples folder:
You can still look through this documentation to see how the project might be structured conceptually as you browse the sample code.
By default FlatRedBall projects link pre-built binary files including prebuilt FNA libraries. You may want to link your FlatRedBall project to FNA source for improved debuggability or to contribute back to FNA.
This document includes the required steps for linking your game to FNA.
Before you link your game to FNA, you must first link your game to FRB source. If you do not do this, then the FlatRedBall.FNA nuget package will include FNA which results in the FNA libraries being included twice.
Follow the steps outlined here to link your game to source: Linking your Game to FlatRedBall Source.
Once you have finished these steps, your solution should include your game project and all FlatRedBall source projects.
By default FlatRedBall templates include FNA binaries which are marked as copy if newer. These can be found in your game's csproj at the root level:
These are marked as copy if newer:
If you would like to update these to the latest version, you can:
Download the latest daily files from https://github.com/FNA-XNA/fnalibs-dailies/actions
Unzip the folder for x64
Copy the files into your game project
The latest .dll files will now be copied to your bin folder whenever the game is built.
We're glad you came to check out FlatRedBall. Before jumping in, let's take a few moments to discover what FlatRedBall is all about. If you're new to game development or just new to FlatRedBall, this page will introduce you to common FlatRedBall (FRB) concepts.
The FlatRedBall game engine has been in active development since 2005 with the goal of making 2D game development as easy as possible. Even today FlatRedBall is continually being improved. Not only are we developing the best 2D game engine in the world, but we're actively using it to make our own games. Every FlatRedBall feature has been tested with a real-world game development team.
All FlatRedBall development centers around a program called the FlatRedBall Editor.
This program is where you'll start when making a new project, and you'll use it throughout development as your game grows. The editor is a visual tool, which means you won't be writing any code directly in the editor. However, the editor is a programmer-friendly environment. Everything you do in the editor results in generated code. This makes your game easy to debug and keeps your program as efficient as possible.
You still have the full power of C# and Visual Studio to write and debug code. Ultimately your project is a standard .NET project which builds and runs just like any other project. It's real .NET!
The editor can set up your project and keeps it structured using the best patterns, but gives you all of the freedom you'd expect from a code-based game engine.
Although FlatRedBall provides a powerful editor, if you prefer you can use FlatRedBall in code-only situations. Ultimately the FlatRedBall Editor does not generate any binary data - it is all code generated, which means you can do everything that the editor does yourself. You can even use the FlatRedBall Editor to look at the generated code to learn how to write your own code.
Screens define the flow of your game. Most games have a screen called GameScreen for all common parts of the main game play. For example, GameScreen may contain a list of enemies, a player, a list of collectable coins, and collision for the player and enemies to walk on. Anything that's specific to a particular level (such as the Tiled map file) is added to a new screen for that specific level. Games may also have screens which aren't created specifically for game play, such as settings, credit, and level select screens. To help visualize this, let's take a look at the screens that might be used in a game like Super Mario World.
Note that the three levels are grouped together because they use the GameScreen as their base screen. Of course, a full game like Super Mario World contains dozens of screens, but the picture above gives an example of how it could be organized.
Each level in FlatRedBall uses (at least) one Tiled Map File. Tiled is a powerful, commonly-used tool for creating level files using tileset images. The following image shows a typical platformer level in Tiled.
Notice that the level is composed of small squares which are called tiles. This approach to game development can be very efficient because it allows reuse of art to create large levels. The image above is created using tiles from a tileset sprite sheet which is a .png file with all of the art needed to create a level.
Once you have a tileset created, making new levels is easy.
Entities are objects in your game which can move (a player walking in a level), collide (a bullet hitting a wall), have animations (an enemy playing a walk animation), and have AI behavior (enemy ships flying towards a target). In other words, entities are objects in a game world which the player can interact with. There are two parts to creating an entity. The first is to define the entity. When creating an entity, you are answering the following questions:
What does it look like?
Does this entity have collision? In other words, do we care if this entity is touching any other entity or any solid part of the level? For example, coins in Super Mario World are entities which have collision so that the game can respond when Mario touches the coin.
Can the entity be controlled by the player?
Does the entity have any logic that has to be continually performed every frame? For example, an enemy in Mario must check whether it is going to walk off of a cliff. Some enemies turn around when reaching the edge of a platform.
Once you've defined an entity, you can place the entities in a screen. Each time you place an entity in a screen, you are creating an instance of that entity. A screen may have lots of instances of an entity. For example, consider the following image from Super Mario World.
Here we see eight instances of the Coin entity. Of course, the full level may have dozens or even hundreds of coins. The image above also has a few Enemy entity instances, and an instance of the Player (Mario).
FlatRedBall works well with the Gum UI tool. You can add user interfaces like menus and HUD (UI) to your game with no code. Experienced programmers will feel right at home working with Gum. It provides an event system similar to other common UI frameworks such as WPF, Winforms, and Xamarin Forms. Gum is a visual layout tool with a flexible, powerful layout engine. You can create pretty much anything in Gum!
You can use Gum to create interactive menu screens (as shown above) or in-game HUD.
FlatRedBall's Editor is also a live editor, enabling the creation of content in real time. The live editor comes out-of-the-box with powerful features, but it can be customized by adding code to your game. You can add error checking, real time visualizations, and custom logic in response to variable assignments. The editor runs your game, so your code runs while you are making changes.
FlatRedBall is one of the longest-running actively developed game engines available! We started development in 2005 and never stopped. It is fully open-source (MIT License) on GitHub and is updated almost daily with bug fixes and improvements.
If you need help you can post an issue or join our Discord channel. The FlatRedBall team is dedicated to making the best game engine for you and for ourselves. We are continually dogfooding (using it to make our own games). This means our improvements are not just to show off fancy visuals, but to help you complete your game as fast as possible.
If you're ready to start making games, head over to the downloads page to get the latest version of FlatRedBall. Once it's installed, start moving through the tutorials to start making your first game.
So far we have a very simple project with a PlayerBall (which appears as a Circle) which can be moved with a gamepad or the keyboard. This tutorial covers creating collision for the game. This tutorial introduces two new types.
ShapeCollection - an object which can contain any number of shapes. Adding multiple shapes to a ShapeCollection makes the creation of collision relationships very easy.
AxisAlignedRectangle - this is a rectangle shape which can be created in Glue just like We'll use multiple AxisAlignedRectangles to assemble our level, and we'll be organizing these in a list object.
First we'll create a ShapeCollection called Walls. This ShapeCollection contains all of the rectangles in our game. ShapeCollections are a specialized list which can contain multiple instances of Shape objects such as Circles and AxisAlignedRectangles. To create the Walls ShapeCollection:
Select GameScreen
Click the Quick Actions tab
Click Add Object to GameScreen
Select ShapeCollection as the type
Enter the name Walls
Click OK
Now that we have a ShapeCollection for our walls, we can add the individual walls. To add the first wall AxisAlignedRectangle:
Right-click on the Walls item.
Select Add Object
Select AxisAlignedRectangle as the type. Notice that available types are limited to shapes since we are adding to a ShapeCollection.
Enter the name Wall1 and click OK.
Now we can modify the properties of this wall. Select Wall1 and change the values as follows:
Y = 300
Width = 800
Height = 30
Doing so moves the rectangle to the top of the screen. If you run the game you should see the wall at the top of the screen.
Now that we've created a single wall, we can duplicate it by right-clicking on it and selecting the "Duplicate" command:
Create 5 duplicates, so that the Walls list holds a total of 6 AxisAlignedRectangle instances:
Next set the X, Y, Width, and Height variables for for the newly-created walls as follows: Wall2
X = 0
Y = -300
Width = 800
Height = 30
Wall 3
X = -400
Y = 200
Width = 30
Height = 200
Wall 4
X = 400
Y = 200
Width = 30
Height = 200
Wall 5
X = -400
Y = -200
Width = 30
Height = 200
Wall 6
X = 400
Y = -200
Width = 30
Height = 200
When finished, the game should have walls made of AxisAlignedRectangles with gaps on the left and right sides for goals:
Next we'll use collision relationships to define collision between the PlayerBall and Walls. We'll revisit collisions in later tutorials, so keep in mind that this is only the first step in setting up our game's collision. When we created our PlayerBall entity in an earlier tutorial, we marked that the entity should contain a Circle object. Doing so automatically marked the Entity as using the ICollidable interface. As a reminder, the following image shows the window we used to create the PlayerBall entity:
Since our PlayerBall is an ICollidable, that means it can collide with any other ICollidable and with any shapes - like the Walls. We need to tell our game that we want the PlayerList to collide against the PlayerWalls. The easiest way to do this is to create a Collision Relationship in Glue. To do this:
Drag+drop the PlayerBallList onto the Walls in the Glue explorer. This creates a new CollisionRelationship
Select the new CollisionRelationship (which is named PlayerBallVsWalls)
Select the Collision tab
Change the Collision Physics to Bounce Collision so that our PlayerBall bounces against the walls
Change the PlayerBall Mass to 0. This means that the PlayerBall behaves as if it has no mass, so it will not be able to move the walls. Be sure to press Enter or Tab to save the change.
If we run the game now, the PlayerBall collides with the walls.
We use CollisionRelationships here because they are very powerful and require no code. Just like everything else in FlatRedBall, collisions can also be managed in code. If you are interested in writing collision purely in code, see the CollideAgainstMove page.
The CollisionRelationship we created in the previous step set the Collision Physics to Bounce Collision, but if we move the PlayerBall to the wall, it doesn't bounce. While this may seem like bounce collision is broken at first glance, the real reason is because of the way our movement has been implemented. Our code in PlayerBall sets the velocity values every frame. When the PlayerBall collides against the wall, the bounce physics sets the velocity values to simulate a bounce; however, our code in PlayerBall overwrites this velocity immediately. We will make changes in a later tutorial to our movement code so we no longer overwrite the velocity every frame, and so the bounce movement works as expected.
Even though the game isn't really playable yet, we're definitely starting to see something come together. We have a movable object and game boundaries which have solid collision. The next tutorial duves deeper into the control of our PlayerBall Entity.
This tutorial implements more advanced controls for the PlayerBall. Most games with first-person control (that is control where you directly direct an Entity, as opposed to third person control like RTS games - not to be confused with first person view) have fairly complex control logic. We'll be implementing more advanced controls in our game, investigating the logic in detail along the way.
Our current implementation provides immediate control over the PlayerBall's velocity values:
In other words, if the input for moving right is held (whether that's a keyboard key or an Xbox360 analog stick), the PlayerBall immediately moves at maximum speed. Similarly, when the movement input is released, the PlayerBall immediately stops. As mentioned earlier, this logic prevents the collision relationship from making the ball bounce. We will modify our input code to gradually add to the speed of the PlayerBall when the input is held down. To do this, we'll change our code to set the PlayerBall's acceleration rather than velocity. Modify the MovementActivity in PlayerBall.cs as follows:
Notice that the code above still uses the MovementSpeed variable, which can be modified in Glue. This value can be increased to make movement more responsive. You may want to increase this value from 100 to a larger number such as 300. Now our ball can bounce against the walls, and it doesn't immediately speed up or slow down - it takes some time to gain speed.
Since we're no longer setting velocity values directly (acceleration values add and subtract to the current velocity), the ball continues to move even after releasing input. We'll address this in the next section.
We'll use the Drag property to slow the PlayerBall. Drag modifies velocity proportionally and in the opposite direction of current Velocity. In other words, drag slows an object regardless of its movement direction. The faster an object is moving, the more Drag reduces its velocity. Drag is a built-in variable on all Entities, so if you are creating a game which uses acceleration to move an object, you may want to also implement Drag. Drag is applied whether an object is accelerating or not. Drag slows an object in proportion to its absolute speed - the faster it moves the more Drag slows velocity. This results in a maximum speed, also known as terminal velocity. To add Drag to the PlayerBall Entity:
Select the PlayerBall Entity in Glue
Select the Variable tab
Click the Add New Variable button
Select the Expose an existing variable option
Click the Drag variable
Click OK
Change the Drag variable to 1
The addition of Drag has changed the way our ball moves:
The ball now has a maximum speed
Releasing all input results in the ball slowing down
The ball does not accelerate as quickly as it used to
We'll increase the MovementSpeed to make up for the addition of Drag on the PlayerBall's acceleration. Feel free to play with the MovementSpeed variable. A value around 350 should result in responsive movement.
The previous tutorial created a collision relationship between the PlayerBall and Walls. Now that we have bouncing implemented, we can revisit the PlayerListVsWalls collision relationship. Notice that the relationship provides an Elasticity value as shown in the following image:
This value controls how much velocity is preserved after a collision occurs. A value of 1 indicates that 100% of the velocity is preserved. A value less than 1 results in some of the velocity being absorbed. Feel free to play with this number to create a bounce elasticity that you like. If you want the balls to remain bouncy, you can keep it at 1. If you want the walls to absorb some of the PlayerBall velocity, try putting a (positive) value less than 1, such as 0.5.
Now we're getting somewhere! The game is starting to feel pretty solid. Next we'll add a Puck Entity which can be used to score goals.
So far we have a basic structure for our game: an Entity with graphical representation, a Screen, and an instance of our Entity in our Screen. This tutorial requires writing code.
Before implementing code to control your Entity you need to decide how to control an Entity. For this tutorial we will implement controls using gamepads as well as keyboard (in case you do not have a gamepad). We will use FlatRedBall's input interfaces so that our game code doesn't need to consider which input device is using after initial setup. Note that FlatRedBall uses the Xbox36GamePad name since FlatRedBall was originally written to work with XNA. Most controllers will work including wired and wireless XBox controllers, Switch Pro controllers, and Playstation DualShock (although usually the DualShock requires a wired connection).
First we need to define which controls are needed in our game. Our PlayerBall requires the following input:
Movement in the X and Y coordinates - also known as "2D input"
Input for executing a boost. Boosts temporarily increase the player's speed when executed for a short amount of time. The user does not need to hold the input down, simply pressing the button/key is enough.
We'll write the PlayerBall so that it works with any input device, whether that's Xbox 360 controller, Keyboard, or any other device. To add code to PlayerBall, double click PlayerBall.cs in Visual Studio. This is located in your project's Entities folder.
Modify the PlayerBall.cs file so it contains two input properties as follows:
Next we will add code to move the player ball. Notice that there are four methods in PlayerBall.cs:
CustomInitialize
CustomActivity
CustomDestroy
CustomLoadStaticContent
CustomActivity gets called every frame, so we can use it method to respond to controller input and modify our PlayerBall appropriately. Add the following code in the CustomActivity method in PlayerBall.cs:
At this point we have written fully-functional code for moving the ball according to its MovementInput; however, we haven't assigned the MovementInput yet. We'll do this next.
The MovementInput and BoostInput were intentionally created as public properties so that they could be assigned by the GameScreen. The GameScreen will contain logic for deciding which input device to use. To assign the input interfaces, open GameScreen.cs in the Screens folder and modify the CustomInitialize method as follows:
Notice that the object we are assigning code to (PlayerBall1) matches the name of the entity in the editor. FlatRedBall objects in the editor always have a matching name in code, as shown in the following image:
For more information on the Keyboard class, see the Keyboard page.
Clean code is very important. This is something we stress at FlatRedBall for all developers making any kind of game regardless of size. Therefore, this tutorial (and others in the future) discusses how code can be improved to be more flexible and maintainable. The code we wrote above has a number of problems:
The velocity (which was set to 10) is set right in the method where it's used. This velocity value is typically considered "data", while its application is considered "logic". The separation of data from logic is a fundamental concept in keeping game projects maintainable.
The game includes logic in the CustomActivity method. We encourage no logic, only method calls in the standard "Custom" methods.
FlatRedBall provides a number of ways to separate data from logic. The simplest of these is to create variables. To add a variable to the PlayerBall entity:
In the editor, click the PlayerBall Entity
Right the Variables tab
Click the Add New Variable button
Verify that float is the Type
Enter the name MovementSpeed and click the OK button
Verify Variables is selected and set MovementSpeed to 100. Deselect the text box or press ENTER to apply the value.
Finally, return to the movement code inside PlayerBall.cs and change the code to:
If we run the game now we can control the player with the W, A, S, and D keys:
The benefit of using the input interfaces (I2DInput and IPressableInput ) is that the input device being used can be set or changed without any code changes in the entity. For example, we can modify the GameScreen to optionally use an gamepad if connected, otherwise it falls back to using the keyboard. To add support for keyboard and gamepad controls, open GameScreen.cs and Modify CustomInitialize as shown in the following code:
For more information on Xbox360GamePad, see the Xbox360GamePad page.
Finally, we should clean up CustomActivity. In general, we encourage keeping the Custom methods free of logic so that it is clear what an Entity/Screen is doing when initialized and when destroyed without having to mentally translate code into concept. Open PlayerBall.cs and replace the CustomActivity with the following:
Then define MovementActivity in PlayerBall.cs file:
Similarly, we'll want to clean up the CustomInitialize method in GameScreen.cs:
We'll implement the AssignInput method in GameScreen.cs:
Now we have a PlayerBall Entity which is cleanly written, has speed which can be customized through the FlatRedBall Editor, and can be moved with the game pad or keyboard. The next tutorial covers defining collision in the GameScreen.
Now that we have a visible HUD, we need to add logic to have it update when players score. This tutorial adds code to the ScoreHud to update the displayed text according to public properties which are set by GameScreen whenever a goal is scored.
First we'll be adding properties to the Hud object for the scores of each of the players. We could do this purely in code, but the FRB Editor provides a convenient way to set the score values to integers. We will be combining two FRB Editor features for this:
Tunneling variables - Creating a variable in an entity which is tied to the variable of a contained object. In this case we'll be creating variables which are tied to each of the score Texts' DisplayText variable.
Built-in casting of the variable type - although DisplayText is a string, we'll tell FRB that we intend to use it as an integer.
To do this:
Expand ScoreHud's Objects in the FRB Editor
Drag+drop the Team1Score onto the Variables item
Select DisplayText as the variable
Set the Alternative Name to Score1
Set the Converted Type to int
Click OK
Repeat the steps above, but this time use Team2Score, and create a variable called "Score2".
Now that the ScoreHud can react to score changes, the ReactToNewScore can be modified to update the HUD. To do this, modify the ReactToNewScore method in GameScreen as follows:
If we run the game now we'll notice that scores start out at 0 and increments whenever a player scores a goal.
If you are seeing a conversion error similar to Cannot implicitly convert type 'int' to 'string' , then you may need to modify the variables for Score1 and Score2. See the steps above where the Variable Type is set to int. If you've already created the variables, you can set the variable conversion:
Select the Score1 variable
Select the Properties tab
Change OverridingPropertyType to int
Well, that was easy! Now we have scoring working. At this point we could call the game done. The next tutorial adds some extra finishing touches to the controls to make it more competitive and make the game play deeper.
Beefball is intended to be a competitive multiplayer game. So far we only have one PlayerBall instance, so let's add some more PlayerBall instances. Previously we added a list for our PlayerBall. We can now add a second PlayerBall with minimal changes to our project.
To add a new PlayerBall:
Expand the GameScreen's Objects folder
Select the PlayerBallList object
Select the Quick Actions tab
Click the Add a new PlayerBall to PlayerBall List. Alternatively, you can right-click on the PlayerBallList and select Add Object
Change the new PlayerBall's X value to 180
You should now see two PlayerBall instances under the PlayerBallList and in game. Also, since we created our collision relationships between the lists, the new PlayerBall can already collide against the walls and the Puck.
Now that we have two PlayerBall instances, we need to add a new collision relationship. This time, we will create a collision relationship between the PlayerBallList vs itself. To do this:
Select the PlayerBallList
Select the Collision tab
Find the PlayerBallList in the list of items
Click the Add button
Set Collision Physics to Bounce
If you run you game now, the two PlayerBall instances collide against each other. Also, if we added more players (a third or fourth player) those would also collide with each other automatically.
We'll assign input on PlayerBall2Instance with code similar to the input-assigning code for PlayerBallInstance. To do this, open GameScreen.cs and modify AssignInput as shown in the following code:
Now each PlayerBall uses a different Xbox360GamePad or set of keys.
Now that we have multiple PlayerBall instances, we have a game that is playable, but it's missing scoring and game rules. The next tutorial adds the ability to score goals.
Currently the game is playable, but scoring a goal results in the puck moving off-screen. We'll add logic and data to detect when a goal is scored by resetting the position of all objects and assigning points.
Conceptually detecting a goal is simple - whenever the Puck collides with a certain area then a goal has been scored. The first step is to define the goal area. First we'll create a Goal entity:
Select the Quick Actions tab
Click the Add Entity button
Name the entity Goal
Select the AxisAlignedRectangle option under Collisions
Click OK
The default rectangle size for a Goal is too small, so we should make it bigger:
Select the AxisAlignedRectangleInstance in the Goal entity
Click the Variables tab
Change Height to 200
Just like before, we'll also add a list of goals in our GameScreen:
Select the Goal entity
Select the Quick Actions tab
Click the Add Goal List to GameScreen button
Next we'll add two goal rectangles to the GameScreen. Instead of using the Quick Actions tab, we'll show an alternative method here by drag+dropping the Goal entity onto GameScreen. Do this twice, changing the name of the first to LeftGoal and the second to RightGoal:
Why do goal names matter? In an earlier tutorial we created the wall collision and named them sequentially (Wall1, Wall2, etc). However, the goals are slightly different - we will be checking which goal we collided with in code to determine whether the left or right team should earn a point. Setting clear names makes it easier to keep track of which goal is on each side.
Next, modify the goals values as follows: LeftGoal:
X = -410
RightGoal:
X = 410
If you tried playing the game, you may have noticed that the PlayerBall instances can leave the screen by passing through the goals. To fix this we can create another CollisionRelationship between the PlayerList and GoalList:
Expand the GameScreen's Objects folder
Drag+drop the PlayerBallList onto the GoalList object
Select the new PlayerBallVsGoal collision relationship
Click the Collision tab
Change the Collision Physics to Bounce Collision
Set the PlayerBall Mass to 0
If you run the game, you can no longer leave the play area with either PlayerBall.
Now that we have all of our data and object instances set up, we can write code to detect if a goal has occurred. First we'll need two variables to keep track of score. Add this code to GameScreen.cs at class scope (outside of any methods):
Now we'll create another collision relationship, but this time we won't use any physics. Instead, this will be an event only collision relationship. This means that when the relationship detects a collision, it raises an event (call a function) which allows us to perform custom logic such as increasing score and resetting the game. First we'll create a collision relationship:
Expand the GameScreen's Objects folder
Drag+drop the PuckList object onto the GoalList
Now we'll create an event, which is a function that is automatically called whenever the collision occurs:
Select the new PuckVsGoal collision relationship
Select the Collision tab
Scroll to the bottom and click the Add Event button
Accept the defaults and click OK
When an event is added, Glue automatically adds a new file to contain events called GameScreen.Events.cs. We can find the new event there. You need to expand the GameScreen.cs in Visual Studio's Solution Explorer to see this file.
We can check which Goal collided with the Puck in the code and perform different logic. We can Modify the OnPuckVsGoalCollided method and add a new ReactToNewScore method to the GameScreen.Event.cs file as shown in the following snippet:
Whenever a Puck collides with either goal, we call ReactToNewScore, which resets the player and puck positions and velocities back to their starting states. Notice that the code in GameScreen.Event.cs has access to the player1Score and player2Score variables. This is because the GameScreen is one class split into multiple files using the partial keyword. Therefore, anything you define in Glue or GameScreen.cs is available in GameScreen.Event.cs.
If you've made it this far, congratulations! You now have a fully-playable game...at least, to the point where you can play with friends. There's still a few more things we'll do to improve the design and add polish. The next tutorial covers adding a HUD to display player score.
This tutorial covers how to create an Entity, which is the FRB term for a "game object". Examples of entities include:
Game characters (like Mario)
Projectiles (like a bullet)
Power-up (like a health pickup)
Our first entity is called "PlayerBall".
To create an Entity:
Click the Add Entity in the Quick Actions tab...
...or right-click on the Entities folder and select Add Entity
Enter the name PlayerBall
Check the Circle checkbox under the Collisions category. This adds a circle object to the PlayerBall entity, which we'll use to test if it is touching the walls, goals, or other ball instances.
Notice that the ICollidable checkbox is checked - we'll cover this in a later tutorial. We'll leave it checked for now.
Notice that Create Factory is also checked. This option simplifies the creation of additional entities in code. We'll leave this checked as well.
Click OK
Our entity is now created with a Circle named CircleInstance under its Objects folder, as shown in the following image:
The previous section showed how to create an entity and add a Circle to the entity at the same time. Objects can be added after an entity is created as well. Note, the following steps are only shown for example, and do not need to be followed if you performed the previous steps. To add a Circle to an already-created entity:
Click the Add Object quick action...
...or right-click on Objects and select Add Object
Select the FlatRedBall Or Custom Type option
Select Circle in the list
Enter the name CircleInstance and click OK
When a new Entity or Screen is created, a number of files are created:
<EntityName>.glej or <ScreenName>.glsj
<EntityName>.cs
<EntityName>.Generated.cs
For example, the PlayerBall entity creates PlayerBall.glej, PlayerBall.cs, and PlayerBall.Generated.cs. These files can be viewed by right-clicking on the entity and selecting View in Explorer.
The PlayerBall.glej file is a JSON file which stores all of the information for the PlayerBall. This file does not need to be edited manually in most cases since the FlatRedBall Editor provides controls for making changes. Of course, the file can be edited by hand, and any changes are automatically reloaded by the FlatRedBall Editor if the file changes.
This file should be included in version control so that the project can be opened and re-generated on other computers.
The PlayerBall.cs is a code file which contains custom logic. By default this file contains only empty functions, but these can be filled in to customize initialization, every-frame logic, destruction, and content loading. This file is often referred to as the custom code file. In a typical game, a lot of your game's code is written in screen and entity custom code files.
PlayerBall.Generated.cs is a file which is generated by the FlatRedBall Editor mirroring the contents of the JSON (glej) file. Any changes to the JSON file, whether through the FlatRedBall Editor or by manually editing the contents of the JSON file, results in the PlayerBall.Generated.cs being re-generated.
The contents of this file are re-generated if any change is made to the Player entity, or if the FlatRedBall editor is re-opened. In other words, generated files are re-generated by the FRB Editor, so any changes made directly to the generated file should be considered temporary. In practice this file can be used to help debug, or to perform temporary tests, but final changes should either be made in the JSON file or in custom code.
Generated files can optionally be included in version control. By default FlatRedBall excludes generated files since they can be re-generated from the corresponding JSON files. Of course, if the project is cloned to a new computer then the project must be opened in the FlatRedBall Editor to re-create all generated files. If this is problematic (such as if the development team includes individuals who do not open the FlatRedBall Editor), then the generated code files can be added to version control. Keep in mind that doing so may result in a larger version history.
At this point our project has a PlayerBall Entity which is ready to be used in a game. Of course, we haven't yet created an instance of the newly-created Entity, so if you run your game you won't see it (yet). The next tutorial creates a Screen which contains our PlayerBall Entity.
The following tutorials provide an introduction to working with Platformers in the FlatRedBall Editor.
This tutorial is an introduction to making games with FlatRedBall. It covers using the FlatRedBall Editor and writing code in C#. The FlatRedBall Editor is a program which helps with the creation and organization of game projects. We'll be exploring its features by creating a game called Beefball - a multiplayer competitive game similar to air hockey. When finished our game will have two circles, each movable with either the keyboard or an gamepad, and a smaller circle which each player can use to earn points.
The first step in any game project is to open up the FlatRedBall Editor. If you've downloaded and unzipped the FRBDK.zip file, then you should already have this on your machine. Unzip the file, and double-click Run FlatRedBall.bat.
If you haven't yet downloaded the FRBDK.zip file, you can get it from the Download page.
Once you open the editor, you can create a new project. To create a new Project:
Select File -> New Project
Enter Beefball for the Project Name.
Leave Desktop GL as the platform. Our game targets this platform because it is easy to debug. Creating the project for a desktop platform is recommended even if the game is intended to run on non-desktop platforms (such as Android). Additional platforms can be added at any time.
Uncheck Open New Project Wizard. We'll make Beefball "from scratch".
(Optional) Change the location of the project. By default the project is created in Documents\FlatRedBallProjects.
Click the Create Project! button to create the project.
The latest FlatRedBall template is downloaded, so your project runs against the newest version available. Now that you've made a project, FlatRedBall remembers this and automatically open it for you next time it is started.
FlatRedBall Editor is a tool meant to work hand-in-hand with Visual Studio. It is not a replacement for Visual Studio, meaning you will be doing work in both Visual Studio and the editor. It is quite common to develop FlatRedBall games with both Visual Studio and the FRB Editor open at the same time.
FlatRedBall projects automatically create a Visual Studio project too. To open the project in Visual Studio, click the Visual Studio icon as show in the following image:
FlatRedBall uses the windows default file association for your .sln file. If you would like to change this association, you can right-click on the .sln file in Windows Explorer to change the default file association.
You can also open the project in visual studio by opening the .sln file. The project folder can be opened by clicking the folder icon in the task bar. This opens the location of the .csproj file, which is one folder below the .sln file. The following animation shows how to navigate to the solution:
When you double-click the .sln file you may see a window like this:
If so you should select the version of Visual Studio that is compatible with the type of project you are running. At the time of this writing, Visual Studio 2022 Community is the most common version to use with FlatRedBall. Once Visual Studio is open, you can run your project by pressing the "start" button, or by pressing F5.
Your game should run if all prerequisites have been properly installed. You should see a blank game
That was easy! So far you have a fully-functional game using FlatRedBall. The next tutorial covers making our first Entity.
This tutorial provides a set of steps for creating an entity with platformer behavior.
The FlatRedBall Editor provides a quick setup for creating a platformer project. To create a platformer project:
Launch the FlatRedBall Editor
Create a new project
Wait for the project to finish loading
Wait for the Wizard window to appear
Select the Platformer project option
Wait for the wizard to finish processing
Run the game from either Visual Studio or the FlatRedBall Editor
Although this tutorial is focused on creating a platformer entity, we will first add a GameScreen. Creating a GameScreen first makes it much easier to add an entity after. Most FlatRedBall projects have a GameScreen - it's a standard screen created by the wizard, so it's best to follow this naming convention in your own projects even if you aren't using the wizard.
Note that you may already have a GameScreen in your project. If so, you can skip this section. To add a GameScreen:
Select the Quick Actions
Click the Add Screen/Level button
Leave the default GameScreen name
Check both the Add SolidCollision ShapeCollection and Add CloudCollision ShapeCollection options
Click OK
We will return to the GameScreen in future tutorials, but having one created before we create entities will speed up the process.
The Player entity is our entity that will be controlled and have platformer physics. Regardless of the genre, most FlatRedBall games have a Player entity. It's a convention that your games should follow, just like having a GameScreen.
To create an entity with platformer behavior:
Select the Quick Actions tab in Glue
Click the Add Entity button
Enter Player as the name for the entity
Check the AxisAlignedRectangle checkbox - platformer entities perform collision against their environment
Verify that the ICollidable checkbox is checked - this enables collision which is necessary for platforming physics
Change the Input Movement Type to Platformer
Leave the Tiled options selected to automatically create a list for this new entity in GameScreen
Click OK
This will create a new platformer entity with a rich set of default functionality. We can verify that the entity is marked as a platformer by checking its Entity Input Movement tab to verify that it is marked as a platformer and that it has two movement types:
Ground
Air
Now that we have the Player entity set up with platfomer control values, we can add it to our GameScreen by drag+dropping the Player onto the GameScreen node. We should already have a PlayerList in our GameScreen so the newly-added object will be inside of that list after the drag+drop.
Note that we add a Player to the GameScreen before creating any Levels. We add the Player to the GameScreen so that we have a player in every Level (screens which inherit from GameScreen).
To create a CameraControllingEntity:
Select GameScreen
Select the Quick Actions tab
Click the Add Object to GameScreen button
Find and select Camera Controlling Entity
Click OK
We want to set up the newly created CameraControllingEntityInstance to track our player as it moves around in the map. To do this:
Select CameraControllingEntityInstance under GameScreen/Objects
Select the Variables tab on the right
Set the variable Targets to PlayerList
Set the variable Map to Map
Before we can run our game, we also need to add a Level. Typically, the GameScreen includes objects, files, and objects which are common to all screens (such as a Player), while each level includes objects and logic which are level-specific (such as the TMX).
To add a level:
Select the Quick Actions tab
Click the Add Screen/Level button
Leave all defaults and click OK
After clicking OK, another popup appears with options for the level tile map (TMX). Leave all defaults and click OK
Your project should now have a Screen named Level1. This is marked as the startup screen (it has the play icon and appears in the startup dropdown).
If we run our game now, we'll see the entity functional - at least, it seems to fall with gravity. In the next tutorial we'll add collision and controls using our entity.
FlatRedBall is a set of technology for creating games on a variety of platforms. FlatRedBall focuses primarily on creating 2D games, although it does provide 3D support and full access to the underlying XNA-like system (MonoGame, FNA, Kni) for creating full 3D games.
FlatRedBall has been used to create dozens of commercial games on a variety of platforms. FlatRedBall's ultimate goal is to boost your productivity as a game developer and to help you ship your game on your target platform.
The FlatRedBall Editor is a visual tool for creating and organizing screens, entities, and files.
Use the wizard to get your game up and running in seconds.
Tiled maps can be drag+dropped into the FlatRedBall Editor and loaded with no extra code. Use Tiled to define visuals, collision, and add entity instances.
FlatRedBall provides full integration with the Gum tool. Gum projects are added to your project and loaded by generated code, enabling the creation of UI from the very first click.
Run your game in edit mode to make changes in real time.
Make changes to files such as Tiled (.tmx), Textures (.png), and spreadsheet (.csv) files and see the changes update in realtime.
Native support for .aseprite files enables creating animated characters by drag+dropping files into the FlatRedBall Editor.
FlatRedBall Editor supports customizing accent colors and changing between light and dark mode.
Add built-in effects like Bloom and CRT with a few clicks, or create your own fully customizable shaders.
Creating a platformer has never been easier. Mark your entity as a Platformer or use the new project wizard to get production quality platformer physics and movement.
The flexible damage system simplifies and standardizes dealing damage and reacting to receiving damage.
Create node networks from Tiled maps or in code to navigate maps.
Use the AnimationEditor to define animations for your characters using individual files or sprite sheets. Preview animations, adjust timing, and add shapes for collision or defining key positions.
FlatRedBall.Forms is a flexible UI system similar to WPF and MAUI. It is fully integrated with Gum and has MVVM support.
Use SkiaSharp to render GPU-accelerated vector graphics in your game. Perform layout purely in code or use Gum.
Use all features in the Spine tool including defining skeletons, animations, and events.
Spine files can be dropped in to your project and loaded with no code. Play animations at runtime in response to input or game events.
Use built-in interpolation systems to move, resize, rotate, and color objects over time.
Physics engine handles collision between circles, rectangles, polygons, and lines.
Define collision relationships in UI or code to add physics, damage dealing, and custom events.
Set your initial game's resolution, whether it supports resizing, aspect ratio, and more.
The camera controlling entity can follow one or more targets, supports zooming, provides customizable tweening, and respects map bounds.
Turn difficult-to-find runtime errors into clear in-editor errors the moment a file is changed or removed.
MonoGame and FNA Song support included out of the box. For more flexibility song files can be loaded using NAudio.
Use CSV files or open office spreadsheet files to define your game data. FlatRedBall automatically generates the classes for loading your data and deserializes the files in geneated code.
FRB input supports mouse, keyboard, touchscreen, and a variety of gamepads (Xbox, Switch, PlayStation, and PC) including direct and abstract access through a variety of interfaces.
Add multiple languages to your game using a simple spreadsheet. Localization is automatically loaded and can be accessed through a LocalizationManager static class. Include multiple pages per string ID and add custom columns for your game's specific needs.
Target multiple platforms with the same code and content. FRB games can run on:
Windows
Browser
Mac
Linux
Android
iOS
Switch/Xbox using FNA AOT
PS4/PS5 using MonoGame AOT (coming soon)
FlatRedBall projects are regular .NET projects with full access to modern C# syntax, the full .NET library, and NuGet packages. FlatRedBall Projects are regular Visual Studio project, enabling you to use your favorite IDE and debugger.
FlatRedBall is fully open source using the MIT License. It is built using libraries which themselves are also open source. You can build your games without the worry of future inconveniences caused by license changes.
All data created by FlatRedBall is in text format. The FlatRedBall Editor saves .json files and generates pure C# code files. Never worry about binary file conflicts wiping out your work again.
FlatRedBall was built in the early days of XNA and has continued to grow, adding support for MonoGame and FNA. Your FlatRedBall game is a MonoGame/FNA game, and the full flexibility of these libraries is available to you.
FlatRedBall provides a syncrhonization context and multiple methods returning Tasks to simplify async programming.
Now that we have our PlayerBall movement working, we'll add a Puck Entity which the user can hit around. You'll find that the Puck is very similar to the PlayerBall.
To create a Puck Entity:
Click on the Quick Actions tab
Click the Add Entity button
Name the Entity Puck
Check the Circle check box under Collisions
Verify that ICollidable is checked (it should be checked automatically when Circle is checked)
Click OK
The Puck entity should appear in the FlatRedBall Editor.
Currently our Puck and PlayerBall both have Circle bodies, and by default the Circles have the same size and color. To differentiate the Puck from the PlayerBall:
Expand the Puck entity
Click on the CircleInstance object under the Puck Entity
Click the Variables tab
Find the Color variable
Change the value to Red using the drop-down
Change the Radius value to 6
Computer settings matter: If your computer is set up so the decimal separator is the comma ',' instead of the period '.' then you should enter values using the ',' character. Unlike C# code, Glue obeys your computer's language settings.
By default the FlatRedBall Editor adds lists of newly-created entities to the GameScreen. Therefore, you should already have a PuckList in your GameScreen.
If you unchecked the option, or if you would like to know how to manually add a PuckList to your GameScreen, the following section shows how to add a list. This is not necessary if you kept the defaults.
Click the Puck entity
Select the Quick Actions tab
Click the Add Puck List to GameScreen button
Select the Puck entity
Click the Add Puck Instance to GameScreen button
Now the GameScreen has a list and a single Puck.
If you run your game you'll notice that the PlayerBallInstance and PuckInstance are both at the center of the Screen. Let's reposition the PlayerBall1:
Select the PlayerBall1 object under your GameScreen
Change the X value to -180
Now that we have a Puck in our game, we need to create two collision relationships:
Puck vs Walls - this prevents the Puck from moving through the walls.
Puck vs PlayerBall - this allows the PlayerBall to "hit" the puck to try to score a goal. We haven't yet created the handling of goals, but this is the first step towards implementing that feature.
We create these two collision relationships just like the previous PlayerListVsWall collision relationship. To create a relationship between the PuckList and Wall:
Expand GameScreen's Objects folder in Glue
Drag+drop the PuckList onto the Walls object
Select the new PuckVsWalls collision relationship
Select the Collisions tab
Set the Collision Physics to Bounce
Change the Puck Mass to 0
Optionally adjust the Elasticity value
To create a relationship between the PuckList and PlayerBallList:
Expand GameScreen's Objects folder in Glue
Drag+drop the PlayerBallList onto the PuckList object
Select the new PlayerBallVsPuck collision relationship
Select the Collisions tab
Set the Collision Physics to Bounce
Change the Puck Mass to 0.3 - this makes it 30% the mass of the PlayerBall
Optionally adjust the Elasticity value
Notice that the mass variables for PlayerInstance vs. PuckInstance differ compared to wall collision. The PuckInstance is given a mass of .3 relative to a mass of 1 for the PlayerInstance, resulting in the PuckInstance behaving as if it has 30% of the mass of the PlayerInstance. If you run the game, you should be able to hit the Puck around the level.
Currently the Puck moves indefinitely after being hit. We'll assign the Drag value to the Puck just like we did to PlayerBall:
Select the Puck Entity in Glue
Select the Variables tab
Click the Add New Variable button
Select the Expose an existing variable option
Select the value Drag
Enter a value of 0.4 for Drag
Now the Puck slows down over time just like the PlayerBall.
If you are using the wizard as shown above, you can skip this tutorial and the next tutorial and move on to the . If you are interested in how to build a platformer in the editor "from scratch", keep reading.
The camera should now be able to track the player. For more information on automatic camera control, see the .
This tutorial series primarily focuses on creating a platformer entity. If you would like to control the camera manually, you can do so by modifying Camera.Main. More information about the Camera object can be found on the .
For more information on how to perform the above steps, you may want to review the tutorial which created the .
FlatRedBall makes platformer game creation easy. This section includes tutorials covering various platformer topics from beginner to expert.
Screens and Entities are two common FlatRedBall concepts. A Screen represents a container for game content and other Entities. Screens define the flow of your game. Often game developers create many screens up-front to help think through a game's structure. Here are some examples of Screens in a typical game:
Game play Screen (like the playing Screen in Pong). This is usually called "GameScreen"
Splash Screen (like a FlatRedBall logo displaying splash Screen)
Main menu Screen
As you work with Screens you will find that they are very similar to Entities. To create a Screen:
Click on the Screens folder and select the Add Screen quick action...
...or right-click on the Screens folder and select Add Screen
Uncheck Add Map LayeredTileMap option - Beefball doesn't use Tiled maps
Accept the other defaults by clicking OK.
Notice that FlatRedBall suggests the name GameScreen for your screen. Recommended practice is to always have the screen where your game takes place called GameScreen. If your game has multiple levels, each level would inherit from GameScreen. Since Beefball does not have multiple levels, we only create GameScreen.
By default, your GameScreen now has a PlayerBallList - this is a list which will contain all PlayerBall instances in your GameScreen.
If you unchecked the Add Lists for Entities option, or if you would like to know how to create lists manually, follow these steps:
Select the PlayerBall entity
Click the Quick Actions tab
Click the Add PlayerBall List to GameScreen button
The PlayerBallList object will contain all of the PlayerBalls we plan on adding later. Our game is a two-player game, so it will eventually contain two PlayerBall instances. The PlayerBallList object will be used to define collision relationships in a later tutorial. Collision relationships define which objects can collide with each other (such as players vs the walls) and what to do when they collide (such as performing bounce physics).
Once you have at least one Screen in your game (GameScreen), you can add Entity instances to that Screen. Entities can be added through the editor or through game code. The editor provides a number of ways to add an entity:
Drag+drop an entity on a screen to add an instance...
... or add an instance to the GameScreen by selecting PlayerBall and clicking the Add PlayerBall Instance to GameScreen quick action. Note that this option will only exist if you have a Screen called GameScreen...
... or add an object to a screen by right-clicking on the GameScreen's Objects folder:
Right-click on your GameScreen's Objects folder
Select Add Object
Select Entity as the object type
Select PlayerBall as the type. The name will automatically be changed to PlayerBallInstance
Click OK
Now that you have a PlayerBall instance in your GameScreen, you can run the game to see it. You can run the game through either the FlatRedBall Editor or Visual Studio.
To run the game through the editor, click the Play button in the toobar at the top
To run the game through Visual Studio, click the Visual Studio icon to open the game in Visual Studio and run it like any other desktop project
Your game should now be open in Visual Studio.
Once the game runs, you should see a circle (the PlayerBall1 instance) in your Screen.
Now that we have an object in our screen we can take a moment to understand how the coordinates in FlatRedBall work. By default, our entity exists at X=0 and Y=0. We can observe this by selecting the PlayerBall1 instance and looking at its Variables tab.
By default the center of the screen is at the origin (0,0), and objects are positioned by their center, so the PlayerBall appears at the center of the screen.
For more information on how to control the screen's resolution and world units, see the FlatRedBall Resolution section.
For more information on how to control the Camera to change the center of the screen, see the Camera code reference.
To recap we now have an Entity called PlayerBall which has a Circle. We've also created a GameScreen which contains an instance of our PlayerBall. If we run our game, it shows a white circle (our PlayerBall instance).
We're now ready to start adding some code to our project. The next tutorial covers controlling your Entity's movement.
Note - if you created your Platformer project using the wizard, feel free to skip this tutorial. This tutorial is only needed if you are manually creating your game.
The previous tutorial created a GameScreen which contains two TileShapeCollections:
SolidCollision
CloudCollision
By default, each is associated with a standard tile from the tileset included in our Level1Map.tmx. However, these collisions do not have any effect on our player since we haven't told the player to collide with them.
To set up collision between our PlayerList and SolidCollision:
Expand GameScreen Objects
Drag+drop the PlayerList onto the SolidCollision to create a relationship
Since our Player is marked as a Platformer entity, the FlatRedBall editor assumes that the PlayerVsSolidCollision relationship should use platformer physics. You can verify that this is the case by selecting the PlayerVsSolidCollision object and clicking on the Collision tab.
Now the player will collide with the level.
By default, the platformer entity already supports a default set of controls. To see this, select the Player entity, then select the Entity Input Movement tab.
By default, the platformer will be controllable with a plugged-in Xbox Gamepad. If no Gamepad is detected, then the entity can be controlled with WASD and Space.
Note - the animation above shows a game that is using the CameraControllingEntity to position the camera in the center of the map. If you are not using the CameraControllingEntity, then you can manually position the camera in your GameScreen by changing the Camera.Main.X
and Camera.Main.Y
variables in CustomInitialize
. Also, note that the Player is inside of the map. You can modify the player's starting X and Y values either in code or by selecting the Player1 object in GameScreen and changing its X and Y values.
If you want to override which input is used to move the player, you can change the controls in code. For example, to change the character to jump with the Enter key and to move with the arrow keys:
Go to GameScreen.cs
Modify the CustomInitialize function to contain the following input assignment code:
The code above added keyboard controls so that the Player can be moved horizontally with the Left and Right arrow keys and jumps using the Enter button.
Now our platformer character reacts to input and collides with the environment. We can change the environment by opening Level1Map at any time and painting new tiles. The next tutorial takes a deeper look at the control values.
This tutorial covers creating a HUD object which displays the scores of each player.
First we'll create an Entity to store all of our scoring information. We could place all of our HUD objects directly in the Screen, but this approach can result in a large number of objects in the GameScreen, making it difficult to maintain. We'll create a "Hud" entity to keep all HUD objects organized. To do this:
Select the Quick Actions tab
Click the Add Entity button
Enter the name ScoreHud
Click OK
For the ScoreHud we'll define the Text objects in Glue (just like we defined the body of the Puck in Glue). To do this:
Select ScoreHud
Select the Quick Actions tab
Click the Add Object to ScoreHud
Select Text as the type
Enter the name Team1Score
Click OK
Repeat the steps above to create another Text object called Team2Score
Repeat the steps above to create another Text object called Team1ScoreLabel
Repeat the steps above to create another Text object called Team2ScoreLabel
You should now have 4 Text objects:
Now we'll change the following variables on the Text objects in Glue. Select the following Text objects and set the variables as defined below: Team1Score
DisplayText = "99"
X = -150
Y = 270
Team2Score
DisplayText = "99"
X = 180
Y = 270
Team1ScoreLabel
DisplayText = "Team 1:"
X = -205
Y = 270
Team2ScoreLabel
DisplayText = "Team 2:"
X = 124
Y = 270
To add the ScoreHud to the GameScreen:
Select the ScoreHud
Select the Quick Actions tab
Click the Add ScoreHud Instance to GameScreen button
You should now see everything showing up correctly in your game
Now we have a score HUD that shows up when the game plays, but it doesn't react to scored goals. The next tutorial adds the necessary logic to have it react to scored goals.
This page covers all control values available in a platformer entity. It also provides ways to implement common functionality such as ice physics and under-water levels.
If you created your game using the wizard, then the Player entity already has a set of default values. If you manually created your Player entity, then it should also have a set of default control variables.
These values serve as a starting point for platformers - they can be tuned to provide a custom feel to platformer entities. The platformer control values can be viewed and edited by selecting the Player entity and clicking on the Entity Input Movement tab.
This is the maximum speed (maximum absolute horizontal velocity) that the character can move through input. Note that if using Immediate horizontal movement, then this is a hard value - not other forces can modify the character movement. For more information, see the next section. Increasing this value makes the character move more quickly, but doing so can make the game more difficult to control if the value is too large.
This value controls whether the character reaches maximum velocity immediately, or if it takes time for the character to speed up and slow down to the maximum velocity and back to standing still. Using Immediate increases the responsiveness of your game, and allows players to move very accurately. Examples of immediate-movement games include Mega Man and Castlevania.
The Speed Up/Down option results in the platformer entity accelerating to max speed and back down to a standstill over time, instead of immediately. This option creates more natural movement and requires more planning on the player's part (such as building up speed before a big jump). Examples of speed up/down movement include the Super Mario Bros. series and the Donkey Kong Country series.
The Speed Up Time value controls how many seconds are required for the platformer entity to reach max speed. This value is only available if using Speed Up/Down horizontal movement.
Increasing this value makes the character makes the platformer entity feel sluggish. Decreasing this value makes the platformer entity feel more responsive. A value of 0 is identical to using Immediate horizontal movement. A larger speed up time can also be used for different terrains and environments. For example, a larger value can make the ground feel more slippery (if the character is walking on ice). A larger value can also make the character seem heavier, or can be used to simulate under-water movement. A larger Speed Up Time can be used for air movement so that control is less precise when in the air.
The Slow Down Time value controls how many seconds it takes for the platformer entity to decelerate from full-speed to a standstill. A larger value can make the ground seem more slippery, especially when paired with a larger Speed Up Time. Usually Slow Down Time should not be larger than Speed Up Time. If the two values are similar, then this makes the ground feel slippery. The following table shows common combinations of Speed Up Time and Slow Down Time. The values High, Medium, and Low do not have fixed meaning, but a typical setup may include these values:
Low .15 seconds
Medium .4 seconds
High .8 seconds
Of course, you should modify values to achieve the desired movement for your specific game.
This value controls the velocity of the platformer entity at the moment when jumping off the ground, or when initiating a double-jump. Larger values allow the character to jump higher. This value is typically larger than Max Speed, but the exact value often requires multiple iterations to get the right feel.
A platformer entity's jump height is also impacted by Gravity, so both Jump Speed and Gravity may need to be modified together. A low jump speed can be used for double-jumps, or for swimming when under water. A large jump speed can be used for characters who can jump higher. The Jump Speed value on Air movement can control whether the character can perform a double jump. By default, this value is 0 which means that the character cannot double-jump. Setting a value greater than 0 means a character can double jump. This topic will be covered in more detail in the following tutorial.
If Hold to Jump Higher is checked, then the player will be able to hold the jump button to cause the platformer character to jump higher. This allows players to perform shorter hops when desired, and longer jumps to clear large obstacles. The larger this value, the longer the player can hold the button to jump higher.
Variable-height jumping can be implemented a number of different ways. The platformer entity implements variable height jumping by turning off gravity while the player is holding the button down. This approach has the following characteristics:
Jumps feel responsive and immediate - the platformer entity does not delay jumping while the player is holding the button down. Instead, jump height is determined by not applying acceleration immediately.
Jump motion may not appear totally fluid, especially in high-gravity games.
The following image shows two jump arcs. The first is the movement of the character when the button is held, the second is without:
Although it may be difficult to see, the entity moves in a straight line on the first part of the first jump, rather than moving in an arc. This is the result of gravity being turned off while the button is held. If we color the first part red, the linear movement is a little easier to see:
The Max Jump Hold Time value sets the maximum amount of time that the player can hold the jump button to extend the platformer entity's jump. A large value gives the player the option to hold the button to jump higher. A small value results in the max and min jump heights not differing by much. A very large value may result in the player appearing to "float" while jumping, so keep this in mind when setting large values (such as over 1 second). A large Max Jump Hold Time may be used with a small Max Jump Speed to create swimming controls.
This value controls whether the platformer entity can press the down arrow + jump to fall down through cloud collision. If this value is false, then cloud collisions can only be jumped up through, but the player cannot fall down through them.
This is the distance to fall when pressing down + jump on a cloud platform before testing cloud collision again. When falling through cloud collision, collision against clouds is temporarily disabled until the user has fallen far enough. Once that has happened, cloud collision is re-enabled. This value should be roughly the thickness of cloud collision objects plus the height of the player collision. For example, if the tile height is 16 and the player's collision height is 32, then the Cloud Platform Thickness value should be set to 48. If a TileShapeCollection's UpdateShapesForCloudCollision method is called, then only half of the shape will be used for collision, so only half of the tile height needs to be considered when determining the Cloud Platform Thickness.
Gravity controls how fast a character accelerates downward when falling. This value is assigned to the PositionedObject.YAcceleration property. Increasing this value makes the entity fall more quickly, while a smaller value makes the entity fall slower, or seem to float.
Max Falling Speed sets a limit to the platformer entity's y velocity. A smaller max falling speed results in the entity falling more slowly, even if Gravity is large. A smaller max falling speed can be used to slow a player's fall down with special abilities such as the raccoon tail in Super Mario Bros 3.
A smaller Max Falling Speed and low Gravity can be used to implement water physics such as the water levels in Donkey Kong Country.
At this point Beefball is a fully-playable game, but we'll be adding one final feature: dashing. This provides deeper gameplay that rewards skill and adds an element of anticipation and excitement.
A dash gives the player a burst of speed in the current direction that the player is pointing. The dash can be used to hit other players, shoot the puck, or dive in front of the puck to block a goal. To add a dash, modify CustomActivity in PlayerBall.cs as follows:
Next, implement DashActivity in PlayerBall.cs:
If you try to build your game, you'll notice that the variable DashSpeed is undefined. This is a variable that should be defined in the FRB Editor so that it can be tuned easily:
Click on PlayerBall
Select the Variables tab
Click the Add New Variable button
Verify that float is selected
Enter the name DashSpeed
Click OK
Enter a value of 600 for DashSpeed
Your game should build and dashing should be fully functional.
Currently the player can dash indefinitely. We'll modify the dash so it has a cool-down after it's used. We'll need a variable to keep track of when the dash was first used, and compare against that variable to check if the user can dash again. First, add the following variables to your PlayerBall.cs at class scope (make it a field):
Next, add a check for the last dash time and a set to lastDashTime in DashActivity in PlayerBall.cs. Your entire method should look like:
Just like before, we need to create a DashFrequency variable in the FRB Editor:
Click on PlayerBall
Select the Variables tab
Click the Add New Variable button
Verify that float is selected
Enter the name DashFrequency
Click OK
Enter a value of 2 for DashFrequency to indicate a 2 second frequency
Now the player can only dash once every two seconds.
Why did we create DashFrequency and DashSpeed variables in the FRB Editor, but lastTimeDashed in Visual Studio? You may have noticed that we defined some of our variables (like DashFrequency and DashSpeed) in the FRB Editor, but we defined lastTimeDashed in Visual Studio. When working in the FRB Editor it is important to distinguish between "data" and "logic". Variables considered data are created in the FRB Editor so that they can be easily modified. Game development is an iterative process, and even the most experienced game designers make changes to their game throughout development. Creating variables in the FRB Editor makes changing variables easy, and communicates to other developers that these variables should be tuned. The lastTimeDashed variable, on the other hand, exists solely to support the logic of limiting dashing. The actual value of lastTimeDashed changes many times as the game runs, and setting it through the FRB Editor either has no impact on the game or introduces an unintended bug of making dash not work (if the value is set to a large positive value).
Now that the player's dashing is limited, we need to add some visible indication of when another dash can occur. We'll do this by adding a second Circle to the PlayerBall which grows when the user has dashed, then disappear when the user can dash again. To do this:
Click on PlayerBall
Select the Quick Actions tab
Click Add Object to PlayerBall
Select Circle
Enter the name CooldownCircle
Next we'll use States (something we haven't used yet) to define what the PlayerBall should look like when the cooldown first begins and when the cooldown ends. States have the following benefits:
They let you "code against concept, not content". In other words, you can define a state and use it in code, and later make changes to the state without having to modify any code. This is desirable because when you set the PlayerBall to a "Tired" state, your code shouldn't depend on anything except for the presence of a Tired state. Code should simply set the state to Tired, while the logic of Tired should be handled elsewhere.
States can interpolate from one to the other. In this case, we'll have the "Tired" state set the CooldownCircle to have a small radius, and the "Rested" will have a large circle. The end result is the CooldownCircle "grows" as the player's dash is recovering.
All states must be categorized, so the first step is to create a new category:
Right-click on the States item under PlayerBall and select Add State Category
Name the category DashCategory and click OK
Now the PlayerBall has a category named DashCategory.
Next we'll add states to the category:
Right-click on DashCategory
Select Add State
Enter the name Tired and click OK
Repeat the above steps to create a "Rested" state as well
As mentioned above, we'll want the CooldownCircle ball to grow (increase its Radius) to give the player an indication of how much time is left in the cooldown. To do this, we'll need to tunnel in to the CooldownCircle's Radius property:
Drag+drop the CooldownCircle object onto the Variables item
Select Radius for the Variable
Click OK
Categories must be told which variables to modify. By default, categories do not modify any variables.
To add a variable to a state, drag+drop the variable onto the category:
In this case, the only variable is the CooldownCircleRadius. Next let's define the two states:
Change Tired CooldownCircleRadius to 0.
Change Rested CooldownCircleRadius to 16. This should match the default radius for Body.
Now that our states are defined, let's use them in code. The simplest way to use states is to assign an entity's current state variable. FlatRedBall also provides functions for interpolating between states, which is the process of gradually changing from one state to another. We will write code to set the state to "Tired", then interpolate to rested over the course of two seconds - of course we'll use DashFrequency rather than hard-coding the value 2. To use these newly-created states, go to PlayerBall.cs and modify the DashActivity function as follows: To use the states, add the following two lines of code inside the if-statement in DashActivity in PlayerBall.cs:
Lets change the color of the PlayerBall to help tell the two players apart. We will "tunnel" to the Body's Color so that it can be changed per player instance.
In FlatRedBall, expand the PlayerBall entity
Drag+drop the CircleInstance object onto the Variables folder
Select Color as the variable.
Click OK
Repeat for the CooldownCircle's Color
Now that the color variables is exposed, lets change one circle in GameScreen
In GameScreen, choose PlayerBallInstance2
Change both CircleInstanceColor and CooldownCircleColor custom variables to Cyan, or the color of your choosing.
Run the game!
You can now tell the difference between each player.
If you've read this far, then you have officially just finished your first FlatRedBall game. Way to go! At this point, you can continue reading other tutorials, tweak this game more, or start on your own brand new game. Good luck!
This walkthrough covers the concept of doors - objects which can move the player from one area on the map to another. Doors are often used to subdivide single levels, and are used in games like Super Mario World (pipes and doors) and Mega Man X (doors leading to bosses).
This walkthrough refers to the DoorsDemo as this demo and the demo.
This walkthrough covers a number of concepts for using doors:
The map is broken up into three sections which are all part of the same TMX file, but the Camera follows bounds defined by rectangles in the map
Doors are entities which are added to the Tiled map
The collision event between Player and Door instances is used to check if the player is pressing the up direction
When transitioning (the screen is fading in and out), the code uses async/await to wait for the fading animations before allowing the Player to move
The Level1Map file is a TMX which contains three sections. These three sections are drawn just like normal tile maps, but they also contain a layer called CameraBoundsLayer. This layer, which is an Object Layer, contains three rectangles. Object layers with shapes can be used any time a shape is needed for game logic. Other uses besides camera bounds include defining large triggers (such as the goal in a level) or areas to destroy objects which fall off of a level (such as enemies falling into pits).
These three rectangles will be used to mark the bounds of the Camera when the Player is in one of the three areas of the map. These bounds will apply directly to the Camera, so they must not extend beyond tile edges. Every object layer with shapes is automatically loaded into a ShapeCollection contained in the map. The name of the ShapeCollection matches the name of the layer, so it can be retrieved with a First linq call on the map's ShapeCollections object, as shown in GameScreen's CustomInitialize.
The boundsShapeCollection field is used to position the player in the UpdateBoundsForPosition method. This takes a Vector3 defining a point inside the rectangle. The method then looks through all rectangles to find which rectangle contains the argument position, and adjusts the CameraEntityInstance's Map object.
Normally the Map object references an entire MapDrawableBatch (the runtime for TMX files), but in this case we replace it with an AxisAlignedRectangle. In either case, the CameraControllingEntityInstance will respect the bounds of the Map object it is assigned. We can observe this behavior by walking to the edge of the map. Notice that the Camera doesn't move further to the right even though it hasn't reached the edge of the TMX file.
The demo defines a Door entity in Glue which contains a Sprite and AxisAlignedRectangle. The Sprite is used to display the visual, and the AxisAlignedRectangle defines the collision area used to detect if the player can enter the door. The Door instances are created in our map which contains four Door tiles. The standard door tile is used to place doors in the map.
Its Type is set to Door, which results in instances of that tile automatically being replaced with Door entities.
The map contains four doors. Each door in the map has another door which marks where the Player should appear when travelling through the door. These pairs are identified by their names. The demo uses the convention of setting the Name of each pair of doors with the same letter. Specifically, the doors "A 1" and "A 2" are paired together and the doors "B 1" and "B 2" are paired together. This convention could support up to 26 pairs of doors if using only upper case, and more if using lower case and numbers, so it will work for even larger levels.
When collision occurs between a Door and Player and the up direction is pressed, the demo obtains the otherDoor to determine where to place the player. This code is used in the OnPlayerListVsDoorListCollisionOccurred method.
The logic for moving between doors is ultimately driven by the PlayerListVsDoorList collision relationship in Glue. This collision relationship raises an event OnPlayerListVsDoorListCollisionOccurred in GameScreen.Event.cs.
We'll focus primarily on the collision and positioning logic in this section. Initially we check if the player has input enabled and whether the player is pressing up. Checking the InputEnabled property is important otherwise the Player could press Up multiple times during the transition animation and force the animations to play over and over, resulting in confusing behavior. We use the PressedUp property which returns whether the default up input was just pressed. This property is defined in Player.cs.
As mentioned earlier, once inside the if-statement, the demo searches for a matching door which has the same first letter in its name. The player is immediately moved to this position and UpdateBoundsForPosition is called. As shown earlier, this method updates the camera bounds to the rectangle which contains the other door.
The demo also immediately moves the camera to the new bounds through the ApplyTarget method. The ApplyTarget method can optionally lerp (move the Camera smoothly) or move immediately. In this case we move immediately so that the camera doesn't move across the level and through areas inbetween the map sections. Notice that we are moving between two doors when entering a door, but we could also have created a convention which results in players moving to a completely different level. In this case, we would have called MoveToScreen rather than moving the Player to a new position. Such a system would enable games with maps which have multiple areas but which also connect to other maps, such as a dungeon in Final Fantasy 4.
When the player collides with a door and the if-check passes, the player's input is disabled and the transition logic begins. The first part of this transition is the async prefix on the OnPlayerListVsDoorListCollisionOccured method. This enables the method to use the await keyword to delay logic. Without this, the method would need to use more complex code to enable and disable input as the fade animations are playing. The OnPlayerListVsDoorListCollisionOccurred method is effectively split up into three sections, separated by the await calls:
The first section (yellow) happens immediately when the player presses up - the player's input is disabled, the door shows its open graphic, and the fading animation begins playing. Once the animation finishes the second section (green) happens. This section moves the player immediately and adjusts the camera bounds, and opens the other door. This happens while the screen is completely covered by the black overlay, so the player cannot see the immediate movement happen. Once this movement happens, the black overlay fades away. Finally, once the fade happens, the third section (blue) happens. This closes the door and enables the player's movement.
This walkthrough has covered how to add doors to a map enabling the player to move between different sections in the same level.
This walkthrough covers the concepts of climbing ladders. When climbing a ladder, the platformer Player is able to move vertically by pressing up or down on the analog stick or d-pad. Ladders and vines are used to provide access to areas normally not reachable by jumping alone.
This walkthrough refers to LadderDemo as this demo and the demo.
This walkthrough covers a number of concepts for climbing ladders:
Defining ladder platformer values to control climbing speed
Defining ladders in the TMX file
Controlling whether currently climbing or not according to ladder collision and input
Limiting the climbing height
The Player entity defines a set of movement values for climbing called Climbing.
When this is set as the CurrentMovement, the player has direct control over vertical movement. When climbing up and down, the Climbing Speed is set as the player's Y velocity. As we will see later in the walkthrough, these values are explicitly set when the player presses Up to grab the ladder. Notice that the Player has a non-zero Max Speed under the Horizontal Movement section. This means that the player can move horizontally on the ladder. Some games like Super Mario world allow horizontal movement on ladders. Other games like Mega Man X only allow vertical movement on ladders. This game allows vertical movement, but changing the value to 0 results in no horizontal movement.
Ladders are placed in the TMX file as tiles. The following image shows just the GameplayLayer with ladders.
Notice that the ladder tiles define the maximum height that the player can climb.
The code for this is defined below, but we can add extra climb height by adding additional tiles to the map. Keep in mind the GameplayLayer tiles do not need to match the visual layer exactly.
These ladders tiles use the Ladder type.
This allows the creation of a LadderCollision TileShapeCollection.
Platformer Entities do not (currently) support automatic switching to climbing movement values. The demo includes custom code to enable switching to climbing. The main logic controlling the movement values is in the Player.cs CustomActivity method.
The CustomActivity method checks if the current movement can climb. If CurrentMovement.CanClimb is false, then the player is not climbing a ladder so we can do regular platformer logic for ducking and running. Otherwise, if the player is climbing, the game checks if the player is on ground (solid collision is colliding with the player from below) and if the user is pressing down. If so, we set the ground movement back to Ground so players can climb to the bottom and leave the climbing state.
The second half of CustomActivity code performs logic which switches between being on the ground, in the air, and on th eladder. Rather than only relying on a null check with LastCollisionLadderRectangle, the code also checks if the player's center point (X) is inside the bounds of the LastCollisionLadderRectangle. This prevents the player from moving too far off of the ladder horizontally before falling off. This code could be adjusted to allow the player to move more or less horizontally.
If the player is colliding with the ladder and presses up, then the player can grab a ladder. The player's movement on the horizontal axis is stopped and the player snaps its X position to the ladder's position. We also force the player to be on the ground and to use the Climbing movement values. If the player is not over the ladder but CurrentMovement.CanClimb is true, then the player has moved horizontally off of a ladder, so the player's movement is changed to air (falling). The LastCollisionLadderRectangle property is necessary rather than a bool value so that the player can snap to the ladder's X position. This is a regular property defined at the top of Player.cs.
This value is controlled by GameScreen. The ladder collision requires logic to be executed before collision occurs, so the PlayerListVsLadderCollision relationship does not automatically run.
Instead, all Players have their LastCollisionLadderRectangle explicitly set to null, then the PlayerListVsLadderCollision relationship is manually called in GameScreen CustomActivity.
Whenever a collision occurs, the LastCollisionLadderRectangle is set, as shown in GameScreen.Event.cs OnPlayerListVsLadderCollisionCollisionOccurred.
All of this results in the LastCollisionLadderRectangle storing a rectangle if the player collides with a ladder, and storing null if not. Note that this implementation will not work effectively if two ladders are placed next to each other.
When a player climbs down a ladder and collides with solid collision, IsOnGround will be set to true and the climbing movement values will be undone. The demo performs extra logic to prevent the player from climbing above the top of the ladder. Platformer entities like Player automatically have a TopOfLadderY property which can be assigned in custom code. The demo assigns this in the OnPlayerListVsLadderCollisionCollisionOccurred.
Whenever a collision occurs with ladder rectangles, the code moves up one tile at a time (using GridSize) until it finds the last tile. This is marked as the TopOfLadderY which prevents the player from climbing up indefinitely. The Player's position is defined as the bottom of the player, so in this case, the code limits the height that the player to the bottom of the ladder. This can be changed by modifying the last line of code in the code above. For example, we could let the player's feet reach the top of the ladder by changing the last line to:
This walkthrough has covered how to add ladder climbing to a platformer game.
This walkthrough covers concepts related to creating an end-of-level entity which moves the player to the next level and creating a checkpoint which lets the player start at a midpoint in a level after dying.
This walkthrough refers to CheckpointAndEndLevelDemo as the demo and this demo.
This walkthrough covers a number of concepts for checkpoints and end of level:
Creating instances of Checkpoint and EndOfLevel entities in Tiled on object layers to allow different values on each instance
Storing the current checkpoint name in a static variable so it persists across Screen instances
Spawning and re-spawning the player at checkpoint locations
Collision between the Player and objects controlling spawning (Checkpoint, EndOfLevel, and PitCollision)
This demo includes two levels: Level1 and Level2. Each level has its own TMX file: Level1Map.tmx and Level2Map.TMX. If your project used the platformer plugin then it should have these by default. If your game already has existing levels, you can follow along but you will work in your existing levels rather than Level1 and Level2.
The checkpoints and doors will be added to a Tiled object layer. You must have at least one object layer on each level which should include a checkpoint. For this video we'll use the name GameplayObjectLayer so that it is similar to the standard GameplayLayer.
We will create two entities: Checkpoint and EndOfLevel.
To add a Checkpoint entity:
Right-click on the Entities folder
Select Add Entity
Enter the name Checkpoint
Check the AxisAlignedRectangle option
Click OK
Repeat the process above to create an EndOfLevel entity
Next we'll add a new variable to EndOfLevel:
Expand EndOfLevel
Right-click on Variables
Select Add variable
Select the type string
Enter the name NextLevel
Click OK
Next we need to associate a particular tile with the entity. By doing this, FlatRedBall automatically instantiates the entities whenever it encounters a tile.
To do this:
Open your level in Tiled. Make sure you do not have other levels open as to avoid mixing tilesets
Select the TiledIcons tileset and click on the edit button
Select the tile that you would like to use as a checkpoint, such as the checkered flag
Set the Class to Checkpoint - be sure to match the name of your entity exactly\
Save your tileset
Now you can add instances of the Checkpoint tile to your level in the GameplayObjectLayer. You should provide a descriptive name of the Checkpoint, such as LevelStart. Note that checkpoints can exist at the beginning of a level - this is where the player may spawn when going from one level to another.
To do this:
Select the Checkpoint tile
Select the GameplayObjectLayer
Clck the Add Tile icon to go into tile placement mode
Click on the map to add the Checkpoint tile
All checkpoints must have names so that they can be referenced in code. For this project we assume that every level has at least one checkpoint with the name LevelStart. You can set this name on the newly-created Checkpoint by selecting it and setting its name in Tiled:
Next we'll declare which tile in our tileset should create an EndOfLevel instance. To do this, open up the TiledIcons tileset in edit mode again. Select the icon that looks like a door. You may notice that it already has a Class set, so you can change it from "Door" to "EndOfLevel". As mentioned above, the name must match your entity exactly.
Next, we can add a new property to the tile. This should match the name of our variable in the FRB Editor exactly. To do this:
Select the tile
Click the + icon at the bottom of the properties
Keep the type as string
Enter a property name of NextLevel
Click OK
The variable should appear on the tile.
Repeat the process above to add an EndOfLevel instance to your GameplayObjectLayer.
Once this instance has been placed, its variable can be changed. For example, we can set the variable to Level2.
We want our player to spawn at a checkpoint. We'll create a static variable called LastCheckpointName which indicates the starting Checkpoint.
Initially when the game starts the LastCheckpointName should be set to LevelStart so that the checkpoint that was previously created is used:
The following code shows the logic that does this:
The spawning checkpoint is used to set the player's position. Notice that the player is moved down 8 units after spawning - this accounts for the position of the spawn point being the center of the object, while the player's position is at the bottom (at the player's feet).
Collision between the Player and various objects controls the spawning behavior. As mentioned earlier, the LastCheckpointName variable controls which checkpoint is used to position the Player instance. The CustomInitialize method is called whenever a level is created (or recreated).
We will use the EndOfLevel instances to move the player from one level to the next, and to set the LastCheckpointName.
To do this:
Create a collision relationship between the PlayerList and EndOfLevelList in GameScreen
Add an event to the newly-created collision relationship
Open GameScreen.Event.cs
Add the following code to move to the next level and set the LastCheckpointName:
Notice that the code above assumes that the EndOfLevel instance has a valid NextLevel value. If the EndOfLevel NextLevel property is not set to a valid screen then this code will throw an exception. The code above resets the LastCheckpointName to LevelStart so that the Player spawns at the beginning of the level.
We can set the LastCheckpointName whenever the player collides with a checkpoint - it doesn't have to be only when the player collides with a door. To do this:
Create a collision relationship between PlayerList and CheckpointList
Add an event to the newly-created collision relationship
Open GameScreen.Event.cs
Add the following code:
The OnPlayerVsEndOfLevelCollided resets the LastCheckpointName whenever colliding with a door, so this checkpoint will apply whenever the screen changes. The OnPlayerListVsCheckpointListCollided sets the LastCheckpointName to the name of the collided checkpoint, but this will only apply when the screen is restarted. Typically, this would happen when the player dies.
Player death can be handled in a variety of ways, such as by collision with a TileShapeCollection, or even with a hotkey to test death. Regardless, the way to restart the screen is by calling this.RestartScreen().
For example, if you have a TileShapeCollection named PitCollision, a collision relationship can be created between the PlayerList and PitCollision to restart the screen. This relationship would have an event with the following code in GameScreen.Event.cs:
This code restarts the screen, which results in the entire screen being completely destroyed and recreated. Since CustomInitialize runs again, the Player will be re-positioned according to the LastCheckpointName.
The demo includes two types of checkpoints:
Visible checkpoints - when the player collides with these checkpoints, the checkpoint changes appearance and sets the LastCheckpointName property.
Invisible checkpoints - are used only to spawn the player. In the case of the demo, only at the beginning of the level.
Whether a checkpoint is visible or not is controlled by an exposed Visible property.
Please note that if you are adding the checkpoints to your own custom project, to have the Visible property available you will need to set the ImplementsIVisible in Checkpoint Properties to true and then create a variable via the Expose an existing variable and select Visible. Also, since FlatRedBall purely converts the Tiled objects in the objects layer to instances of a FlatRedBall Entity with the same class, to actually see the flag and the door in your game you will need to add a Sprite object to the Checkpoint and EndOfLevel entities and set them to appropriate images or animation chain files. This subject is explained in detail in following tutorials.
Only Visible Checkpoint instances are considered in the Player vs Checkpoint relationship event.
The Checkpoint is visually composed of two Sprites:
PoleSprite
FlagSprite
By default, the FlagSprite is invisible, but is turned on in the MarkAsChecked function. This provides visual confirmation to the user that the checkpoint has been triggered.
This walkthrough showed how the demo project creates checkpoints which can be used to restart the player mid-level and end of level objects (doors) which can move to the next level.
This tutorial covers how to read input to move a platformer entity. We will also be creating a level to test out our platformer entity. To create a level and collision, we will be using a Tiled level. For more information on working with Tiled, see the documentation.
The sample project can be downloaded from Github:
The sample project can be downloaded from GitHub:
Simple games may make use of automatically assigning movement values on collision as shown in the . In this document, platformer values are assigned through the FlatRedBall dropdowns on the Collision Relationship. While this is convenient, note that ladder movement is optionally assigned based on game logic rather than simple collision. Therefore, if your game uses ladders, you may need to move all assignment of movement values to code.
The sample project can be downloaded from GitHub:
Standard Ground Movement
Low or Medium
Low
Ice Ground Movement
High
High
Underwater Ground Movement
High
Medium
Air Movement
Medium
Medium
Heavy Character Ground Movement
High
Low
This tutorial explores how to add double jump to a platformer character. It covers how to add an extra jump (double jump), infinite jumping, or a limited number of jumps in the air.
Double jumping is a feature in many platforms which gives the player more control over player movement. Players can use double jumping to remain in the air for a longer period of time, to reach areas higher than possible with a single jump, and improve horizontal movement precision. Entities which support double jumping require two sets of values - the movement values before double jump and the movement values to apply after double jump. Platformer entities automatically receive a platformer values called Air which are the before double jump variables, so we need to create a new set of values. To do this:
Select your platformer entity (Player)
Expand the drop-down next to the Add Movement Type button
Select Default Air Control option
Change the name of the new values to AfterDoubleJump
Now that we have a set of values for after double jump, we need to tell our game to use these new values for double jump:
Select the platformer entity
Click the Variables tab
Change the After Double Jump value dropdown to the name of the new set of values we just created (AfterDoubleJump)
Finally we need to change the Jump Speed value on the Air movement values to be greater than zero. This is the velocity which will be applied when performing a double jump.
Now the platformer entity (Player) supports double jumping.
We can also support infinite double jumps by either setting the AfterDoubleJump Jump Speed value to greater than zero, or by setting the AfterDoubleJump variable to be Air. This results in the character being able to jump indefinitely which can be useful if implementing swimming or abilities like the flying racoon power-up in Super Mario Bros 3.
As shown above, the platformer entity can support a single double jump by making the AfterDoubleJump variables have a Jump Speed of 0. By increasing the Jump Speed to greater than 0, then the platformer entity can jump infinitely similar to flying or swimming. It is also possible to limit the number of jumps, but this requires custom code.
For this example we will use the two movement types from the previous sections: Air and AfterDoubleJump. To limit the number of jumps, make sure that AfterDoubleJump has a Jump Speed value of 0.
Next we will conditionally assign the Air value in code AirMovement
value in code according to the number of jumps the player has performed since touching the ground.
To do so, open your platformer entity's code file (Player.cs) and modify the code as shown in the following snippet:
Now our platformer entity can jump 5 times, as controlled by a variable in HandleJump.
The GroundCollidedAgainst property can be used to detect the type of terrain that a platformer entity is standing on. This property, along with the standard ItemsCollidedAgainst property, can be used to perform complex logic in response to collision after all collision for a frame has been resolved. The most common use of the GroundCollidedAgainst property is to assign movement values.
Typically a platformer game will have collision relationships between the Player and various TileShapeCollections. For example, the following screenshot shows collision relationships between the Player and a number of TileShapeCollections.
Whenever the player collides with one of these TileShapeCollections, the GroundCollidedAgainst property will be updated. Some games may have multiple TileShapeCollections represent one type of terrain (such as Ice and IceCloud), so all collision should be resolved before determining the terrain type. In such a situation, the Player entity can have properties for determining the ground type as shown in the following code snippet:
These properties enable the assignment of movement values, such as GroundMovement, in CustomActivity as shown in the following code snippet:
Similar logic could be performed to assign AirMovement and AfterDoubleJump. Consider that the code above requires that the GameScreen has its collision objects exposed publicly. To verify this is the case, select the object in the FlatRedBall Editor and check its HasPublicProperty value.
This set of tutorials covers how to add breakable blocks to your platformer game. Breakable blocks have been popularized in the Super Mario Bros franchise, but are also common in other games such as Castlevania, Metroid, and Mega Man.
Despite being introduced in an early Nintendo Entertainment System (and arcade) game, the feature of being able to break blocks requires advanced FlatRedBall collision functionality. We will be covering the following FlatRedBall topics:
Creating entities in Tiled
Using TileShapeCollections to adjust Entity RepositionDirections
Dynamic destruction of entities
Combining manual collision calls with CollisionRelationships
Manually calling CollisionRelationship
This walkthrough covers concepts related to creating moving platforms. Moving platforms can be used to transport a player vertically or horizontally, and can provide challenge and variety to a platformer level. When walking on a moving platform, the platformer Player is able to move faster and jump further.
This sample can be downloaded from GitHub: https://github.com/vchelaru/FlatRedBall/tree/NetStandard/Samples/Platformer/MovingPlatformDemo
This walkthrough refers to MovingPlatfomDemo as this demo and the demo.
This walkthrough covers a number of concepts related to moving platforms:
Moving platforms horizontally using acceleration and await calls
Performing platformer collision between the PlayerList and MovingPlatformList
The MovingPlatform entity is used in this demo. It moves itself by setting its XAcceleration value and changing this value on a timer controlled by async calls. This logic is started in MovingPlatform.cs and continues to loop until the MovingPlatform is destroyed.
Notice that the StartMoving function is an async method, and it uses the async functionality supported in FlatRedBall to simplify the looping logic for moving. The logic in StartMoving performs the following:
Set XAcceleration to 30 for 1 second - the platform starts stationary and ends moving 30 units/second to the right
Set XAcceleration to 0 for 2 seconds - the platform will be moving to the right at a constant speed
Set XAcceleration to -30 for 2 seconds - the platform will reverse direction
Set XAcceleration to 0 for 2 seconds - the platform will be moving to the left at a constant speed
Set XAcceleration t0 30 for 1 second - this will result in the platform standing still, but its acceleration will continue on the next loop
This code loops continually until the entity is destroyed. The entity sets this value to true in CustomDestroy which will result in the StartMoving loop eventually ending. Notice that this code uses acceleration values, which ultimately change the MovingPlatform velocity values. Moving platforms must use velocity to move as opposed to directly setting their position values to have an impact on the movement of the player.
The GameScreen contains a list of Players and a list of MovingPlatforms.
The PlayerList collides against the MovingPlatformList using Platformer Solid Collision physics
Note that usually platformer entities like Player collide against TileShapeCollections using Platformer Solid Collision, but this can also be used when colliding against another entity list.
This walkthrough showed how to move a moving platform using acceleration and how to collide the player entity against the moving platform using platformer physics.
This walkthrough looks at the FlatRedBall Multiplayer Platformer project and explains the important details of creating a (local) multiplayer game similar to games like Contra III.
The sample project can be downloaded from Github: https://github.com/vchelaru/FlatRedBall/tree/NetStandard/Samples/Platformer/MultiplayerPlatformerDemo
We will be referring to the MultiplayerPlatformerDemo as this demo and the demo throughout this walkthrough.
Local multiplayer games provide a variety of ways to join. Older games on the Super Nintendo assumed that if you selected two players, you would use the first and second controller without checking that the controllers were actually connected. More modern games, especially on the PC, check the connected status of controllers and allow players to join and leave. Joining and leaving can be performed on a dedicated page, or can be performed dynamically allowing players to join and leave mid-game. For simplicity, this demo only allows players to join and leave in a dedicated screen. Conceptually, the process for joining has the following steps:
A page displays UI to tell the user which controllers are connected. Players can connect controllers and press buttons to join.
The values for which players have joined must be stored somewhere which is accessible by both the GameScreen and the screen for joining/leaving. These values cannot be instance values on a screen.
The GameScreen must inspect these values and create Player instances (using a Factory) for every joined player. The correct gamepad must be assigned.
We will take a deep dive into these concepts throughout this walkthrough.
The demo includes a Screen called CharacterJoiningScreen. This screen does not inherit from the GameScreen - it is not a level. Rather, it is a screen which contains only Gum UI and logic.
It is marked as the startup screen to give players a chance to join/leave the game before starting Level1.
The CharacterJoiningScreen uses Gum and the MVVM pattern to make the UI update according to the join state stored in the ViewModel. The UI does not participate in the logic of the game - it only displays the values stored in the underlying view model objects. The state is stored in the CharacterJoiningScreen's ViewModel object.
The code modifies this ViewModel in a few different places.
The ViewModel object is instantiated and assigned as the BidingContext to the Gum screen in CustomInitialize. Notice that we instantiate the ViewModel and assign it initially, but thereafter we use the ViewModel property. Once the ViewModel is instantiated and assigned as the Gum UI BindingContext, we can initialize it based on the connected state of the controllers and whether the player had already joined.
Notice that if a controller is connected, we check the GameScreen.PlayerJoinStates to see if the player should be fully joined, or plugged in but not yet joined. This demo uses a static object in the GameScreen to store the joined status. Since the values are static, they are not reset when moving between screens. Larger games may store this information in a dedicated object such as a singleton which also stores profile information like inventory and experience points.
This screen needs to respond to Xbox360GamePads being connected and disconnected. The InputManager provides a ControllerConnectionEvent event which we can subscribe to in CustomInitialize:
The HandleControllerConnectionEvent method checks if the Xbox360GamePad was connected or disconnected and sets the ViewModel values accordingly.
As mentioned earlier, Gum is bound to the view model, so changing these values automatically updates the Gum visuals. We won't discuss this in much depth in this guide.
The demo checks each gamepad for button presses. The following buttons are considered:
A button - joins if the player isn't already joined
B button - unjoins if already joined
Start button - moves to Level1 if the Xbox360GamePad represents a player who has already joined
Buttons cannot be pushed on an Xbox360GamePad so we don't need to check the connected status when looping through the InputManager list.
As mentioned earlier, the values must be stored in variables which are not instance variables in any of the screens (either CharacterJoiningScreen or GameScreen). Instead, these values are stored as static values in the GameScreen.
These values are used to instantiate players in the GameScreen's CustomInitialize:
This for loop in CustomInitialize is solely responsible for creating Players. Notice that the GameScreen does not automatically create any Player instances through Glue.
Therefore, if the PlayerJoinStates are not assigned, then the game will begin without any Players. The CharacterJoiningScreen is responsible for assigning these values, and it does so right before moving into a level.
If using the Glue Wizard, then the PlayerList will automatically have a collision relationship set up between the PlayerList and SolidCollision. We recommend always creating collision relationships with lists rather than individual objects (such as Player1) so that moving to a multiplayer game is easy.
If your game includes more collision relationships, you will want to make sure that they always include the PlayerList.
Each player in the demo can be controlled by a separate Xbox360GamePad and is colored uniquely based on player index. Since FlatRedBall only supports four Xbox360GamePad instances, the demo is coded to handle up to four players. The game pad and index are assigned in the GameScreen's CustomInitialize method where each Player is assigned. The following snippet shows just the creation of the players inside the loop:
The SetIndex method notifies the player of their index. In a full game the player's index could be used for many reasons such as separate score display, game stats, and experience point awarding. This game is simpler so it only changes which AnimationChain is used, resulting in each player being drawn as a different color.
We also call InitializePlatformerInput which is a method provided automatically by the Player's generated code since it is marked as a Platformer character. This method assigns the input device which tells the Player which input device to read for platformer movement. It also results in the Player's CustomInitializeInputDevice method being called, where the run button is assigned.
The Player code includes an AnimationController which sets the current animation based on input and collision state. This code assigns names like CharacterWalkLeft and CharacterJumpRight, but it does not consider whether the Player is index 0, 1, 2, or 3. This code applies regardless of index because each of the animations have the same names. This approach is a common way to reduce code when displaying multiple player indexes or enemy types. Once a standard set of animations has been decided upon, the code can be written against this common set and it will work regardless of the .achx file used.
This walkthrough has shown how a typical FlatRedBall multiplayer game is created, including setting whether a player has joined, using this value to create players, and assigning input device and animations according to player index.
This walkthrough looks at the FlatRedBall DialogBoxDemo project and explains important details of implementing a dialog box in response to talking to NPCs. Platformers may include dialog boxes to display conversations between characters either as part of a cutscene (such as in Mega Man X) or in response to player input (such as in Castlevania 2).
This walkthrough will refer to the DialogBoxDemo as this demo and the demo.
This demo includes a number of important concepts which combine to create NPCs which the player can talk to using a talk button (X on the Xbox360 GamePad) to display unique dialog.
Dialog is dialog as defined in a CSV file. This file could also be used to support multiple languages, but this demo only includes English.
NPCs are placed in the Level1Map file through Tiled. NPCs can be moved and their dialog can change by making changes in Level1Map.
The DialogBox object from FlatRedBall Forms is used to display dialog. It is displayed in response to collision (player talk collision vs NPC) and button presses.
A typical game which supports NPC dialog may have hundreds or even thousands of pages of dialog. While it is possible to write the dialog directly into the C# code, maintaining this dialog can be very difficult. This is especially true if the game is being developed with a dedicated writing team, or if the game supports multiple languages. To address both of these considerations, FlatRedBall provides a standard way to store and access dialog. The demo includes a CSV file called LocalizationDatabase.csv which contains all dialog. This is added to Global Content Files so that the dialog can be accessed throughout the entire project, and it is marked as IsDatabaseForLocalizing.
Additional columns can be added for games which support multiple languages. Similarly, additional rows can be added to support more NPCs. We separate each page of text with a newline. The English text for T_Npc2 and T_Npc3 both have two pages. Even if the game includes a single language, it must be told whether to display this language or whether it should display the string IDs. This is done in the initialization for GameScreen. A game with more screens may set the CurrentLanguage property in the first screen which is shown (such as a splash screen) or even in Game1. Also, if the game supports multiple languages, this property would change if the user changes the current language in a settings page.
Notice that the first column (Id) is index 0, so English is index 1. The text contained in this file is accessed through the LocalizationManager.Translate method whenever we display dialog boxes. We will return to this code later in the walkthrough to examine how the DialogBox works, but for now we'll highlight the call to Translate in GameScreen.Event.cs ShowDialogBox.
The call to Translate passes in stringId which will be one of the IDs (T_Npc1, T_Npc2, or T_Npc3) depending on which NPC the player has talked to. For example, calling Translate with the string "T_Npc1" results in the string "Hi, I'm just hanging out over here." being returned. The returned string is then split according to the newline character ('\n') to create an IEnumerable<string> where each string is a separate page. The pages are passed to the currentDialogBox.ShowDialog call.
The demo includes an entity called Npc which represents a character in the game which the player can talk to. These NPCs are entities similar to the Player entity. Specifically, they are marked as platformer entities and collide with the level's solid collision.
It's worth noting that the Npc instances do not move in this game, so they could have been implemented as static entities with no collision. A full game may include NPCs which walk around a level, follow the player, or move in response to cinematic sequences. Therefore, they have been created as platformer entities so that they can be fully functional if a larger game calls for it. NPCs should not respond to input, so we mark the input device as None in Glue.
We must assign an input device to avoid NullReferenceExceptions from being thrown, so we do so in the Npc's CustomInitialize code by calling InitializePlatformerInput.
As shown in the code above, the demo also creates an AnimationController so that the Npc faces left or right in response to its DirectionFacing property. As we'll see later, we use this so the Npc faces the Player when dialog is shown. As mentioned earlier, each Npc includes its own dialog. This dialog is assigned in Tiled, but to support this we must create a variable in Glue so the Npc entity includes a DialogId variable.
This variable is defined in Glue, but set in Tiled.
Npc instances can be added to levels through Tiled. To do this, a Tile must have its Type set to Npc. We use one of the tiles in the standard tileset.
Any instance of this tile in Level1Map will result in an Npc instance at runtime.
Notice that the instances are on an object layer rather than a tile layer. This allows the setting of properties on each instance. For example, the selected tile in the screenshot above has a DialogId of T_Npc1. This must match one of the entries in the localization CSV file mentioned earlier. When the Level1Map is loaded, Npc instances are created automatically and added to their respective list (in this case NpcList) in GameScreen. The DialogId variable is also assigned automatically and it can be used to display dialog, as is done in GameScreen.Event.cs where the ShowDialogBox method is called.
For dialog to appear, two things must occur:
The Player must push the Talk button.
The Player must be close to the NPC. The Player entity includes an AxisAlignedRectangleInstance named TalkCollision specifically to test proximity.
If both occur, then dialog should be displayed.
The TalkInput is defined in the Player.cs file. The platformer generated code does not automatically add an input object for talking, so the demo adds it in custom code and assigns it in CustomInitializePlatformerInput.
The demo uses the same input for running and talking, but this could be any button. The TalkInput property must also be public so that we can inspect whether it has been pressed when colliding with the Npc.
As mentioned earlier, the Player includes an AxisAlignedRectangle named TalkCollision. This is excluded from the iCollidable collision so that it does not bump into walls or keep the player on a ledge.
A collision relationship specifically checking the TalkCollision vs the NpcList is included in the GameScreen.
This is handled in an event in GameScreen.Event.cs.
Notice that this collision relationship event may trigger every frame, but we only want to perform the dialog box logic if there isn't already a dialog box active, and if the user has just pressed the TalkInput. We could have also manually performed the CollisionRelationship logic in CustomActivity to reduce the number of collision checks, but games like this will typically have a small number of NPCs so the overhead of checking collision every frame is negligible. As mentioned earlier, we manually set the DirectionFacing on the Npc to turn the Npc towards the player. We also temporarily disable Player input so that players cannot move while the dialog is displayed. Notice that the event is an async method, and that we await the DialogBox. Code using the async/await pattern is not very common in FlatRedBall, but it can be very useful when displaying UI using FlatRedBall.Forms. Awaiting the ShowDialogBox method allows us to turn input off and on in a single method without continually checking if the DialogBox is displayed and without callbacks. The ShowDialogBox method displays the dialog box by calling ShowDialog. The ShowDialog method is responsible for the following whether dealing with the DialogBox type or any other FlatRedBall.Forms type:
Displaying new instances of the DialogBox - specifically adding the visuals to FlatRedBall and allowing the Cursor and Xbox360GamePad logic to be performed on the control
Displaying the pages of text as passed in to this method
Removing the DialogBox completely from the engine - both display and logic
If awaited, the ShowDialog call will not return until all pages have been shown. Since the game uses gamepads, we provide a custom AdvancePageInputPredicate which specifies how text is advanced. In this case, we progress a page of text whenever the A, X, B, or Y buttons are pressed, and only for the InputDevice for the player that talked to the NPC. This would matter if the game were to be expanded to support multiple players.
This walkthrough has covered how to add a dialog box to NPCs. Each NPC defines its own set of dialog using a CSV file which can also be used to support multiple languages.
We’ll begin this tutorial with an empty project. Mine will be called EnemyDamage.
The Glue Wizard can help us get a quick project set up. We will leave most of the options to their default, but we will change the following:
Change What kind of control will the player have? to Platformer
Uncheck Add Sprite to Player Entity unless you intend to add graphics to your Player entity. This tutorial will not cover how to add graphics to the Player entity.
Note that we will not be explicitly working with the Player entity in this game, but we still create one since most games need one eventually.
Change Number of levels to create to 1. We will only have one level in this game
Now our project is ready to go as a fully functional platformer. We can begin adding the Enemy entity in the next tutorial.
We'll begin this tutorial with an empty project. Mine will be called BreakingBlocksProject.
The New Project Wizard can help us get a quick project set up. We will leave most of the options to their default, but we will change the following:
Change What kind of control will the player have? to Platformer
Uncheck Add Sprite to Player Entity unless you intend to add graphics to your Player entity. This tutorial will not cover how to add graphics to the Player entity.
Change Number of levels to create to 1. We will only have one level in this game
Now our project is ready to go and we can start adding our level.
This set of tutorials covers how to create enemies which can receive damage from Bullets created by the Player. Many games include enemies which can take damage from the player such as Mega Man X and Super Metroid.
The following pages cover a number of topics related to dealing damage:
Dynamic entity creation and destruction (Bullets)
HP and damage amounts
Damage over time and area of effect
Friendly fire and teams
Now that we have a basic project (with a Player instance falling off the screen) we can create our map. Our map will have:
Solid collision to keep the player on screen
Graphics so our tutorial project looks like a real game
Tiles defining where to place our breakable Block instances
The Glue Wizard created a Map in our Level1 screen, which is useful if we were going to make our levels from scratch. Instead, we'll use a TMX file to speed things up. Download the following three files to the same folder (such as your downloads folder):
Our game already contains a file named Level1Map.tmx - this is the default name of the TMX added to our Level1 Screen. We can replace this file on disk with the downloaded file. We need to remember to also copy over the other two files. To do this:
In Glue, expand Level1 Files
Right-click on Level1Map.tmx
Select View in explorer to open the containing folder
Once open, drag+drop the three downloaded files into the Level1 content folder. If asked, replace the existing file.
Now our game will run and display the level, but our character still falls through the screen. We'll fix this next.
Our map already has visuals for a platformer game, but no tiles are marked as solid collision. We will add the standard tileset to our map and create a new layer which defines solid collision. To do this:
Double-click the new Level1Map.tmx - either in the file explorer or in Glue
Open the Content folder
Drag+drop the StandardTileset.tsx in the Content folder onto Tiled to access this tileset in Level1Map.tmx\
Add a new layer to the map called GameplayLayer\
Outline the solid collision areas in the level using the top-left brick tile to mark the SolidCollision tiles. Be sure to place these tiles on the GameplayLayer
Don't forget to save the TMX file after adding the collision. Since we used the StandardTileset.tmx file, our game automatically uses these tiles for the SolidCollision TileShapeCollection, and our player can walk around the level.
The GameplayLayer visibility can be toggled in Tiled. You may want this off at times to see the art of the game without the solid collision tiles blocking the visuals, or you may want it on to help diagnose issues.
Now that we have a functional level, we will create the Block entity.
Now that we have a simple platformer game with a Player moving around in a level with solid collision, we can add blocks to our game. We will be creating a new Block entity, instantiate blocks through our TileMap, and adjusting collision RepositionDirections.
Our Block entity will contain an AxisAlignedRectangle for collision and a Sprite for visual. Since the player will be able to walk on Block instances, we want to make sure the AxisAlignedRectangle size matches the tile size of our map (16x16). To create a Block entity:
Select the Quick Actions tab and click the Add Entity button
Enter the name Block
Check the AxisAlignedRectangle option
Check the Sprite option
Click OK
Select the newly-created AxisAlignedRectangle inside of Block
Click the Variables tab
Change the Width and Height to 16
FlatRedBall automatically creates a BlockList in our GameScreen, so we are finished creating our Block entity and can move on to instantiating it in our Tiled map.
To create Block instances in Tiled, we must decide which tile we want to use to represent the Block. When creating entities, we recommend using tiles in the standard tileset. This is the same tileset we used in the previous tutorial to add solid collision. To mark a tile as a Block tile:
Open Level1Map.tmx in Tiled
Select the TiledIcons tileset
Click the Edit button to make changes to the Tileset
Select the desired tile to mark as Block. I'll use the tile that looks like the solid collision with cracks
Change the Type to match the name of your Entity. In this case, it should be Block
Save the TSX file
Place blocks as desired in the GameplayLayer in Level1Map. Be sure to do this in the GameplayLayer, since mixing tilesets in a single layer will prevent your game from running.
Save the map file so the game can use the changes
Your game should now be creating Block instances for each Block tile placed in Level1Map.tmx. Since we haven't yet added graphics to the blocks, the blocks currently display only the white collision rectangles. Notice that each instance also has a small black square - this is the Sprite which we will be modifying next.
Our Block objects are currently displaying a black dot in the center of the collision rectangle for its graphics. This Sprite is not currently assigned a Texture yet. To fix this:
Expand the Block entity
Right-click on Files
Select Add File -> Existing File. We will be using the same file which is used for the graphics in our Level1Map.tmx - it has a graphic for a breakable block.
Search for FRBPlatformer.png and click OK
Drag+drop the FRBPlatformer file onto the SpriteInstance to set its Texture
Change the pixel coordinates to the following:
Left Texture Pixel = 0
Right Texture Pixel = 16
Top Texture Pixel = 144
Bottom Texture Pixel = 160
Currently our Player can jump through blocks rather than being able to hit and stand on them. We can fix this by creating a CollisionRelationship:
Expand the GameScreen Objects folder in Glue
Drag+drop the PlayerList onto the BlockList to create a new CollisionRelationship
Select the newly-created PlayerListVsBlockList and change the Collision Physics to Platformer Solid Collision
The Player can now collide with the blocks.
So far our game seems fairly functional, but it has a collision bug which can result in snagging - the unexpected stopping of the Player's velocity when moving across flat surfaces. This bug can be difficult to reproduce, so you may not have noticed it in the game yet, but it can definitely happen given our current setup. This problem is caused by the Block instances currently having AxisAlignedRectangles with RepositionDirections in all four directions. The following image provides a visualization of this problem:
The purple lines indicate possible RepositionDirections which can occur, and if these occur the Player will experience snagging. The topic of RepositionDirections is fairly extensive, and interested readers can see the following pages for more information:
We will be applying some of the concepts and code discussed in the tutorials above, but for the sake of keeping the tutorial shorter we will not take a deep dive into every topic here. To adjust the RepositionDirections of our Block instances, we will use a new TileShapeCollection which is created purely for this purpose. In other words, we'll make a new TileShapeCollection, but we won't create any collision relationships or fill it in Glue the way we normally do with other TileShapeCollections. First we'll create a TileShapeCollection to be used for adjusting the Block RepositionDirections:
Select the GameScreen in Glue
Select the Quick Actions tab and click Add Object to GameScreen
Select TileShapeCollection
Enter the name CombinedShapeCollection
Click OK
Now we can add the rectangles from our BlockList to the CombinedShapeCollection. To do this:
Open GameScreen.cs in Visual Studio
Modify CustomInitialize as shown in the following code snippet
Now our BlockList will have proper RepositionDirections. Remember the CombinedShapeCollection - we will be returning to this when we work on creating/destroying our Blocks in the next tutorial.
Our blocks are now fully functional as platforming collision. The next tutorial will add the ability to break the blocks when the player hits the block from below.
Currently our game is fully playable as a platformer, but it is missing the feature we've been working towards on this whole tutorial - the removal of blocks on collision. Games which remove blocks based on collision are common. Some games, such as Mega Man and Metroid, remove blocks in response t being shot by a weapon. Super Mario Bros removes blocks in response to collision with the player. Specifically, if Mario has collected a power-up which makes him grow, then hitting a block from below destroys it.
This tutorial will implement this type of destruction since it will give us the opportunity to perform more advanced collision logic.
Before we add logic to destroy Blocks, we will make some minor adjustments to the game to make it easier to control, and easier to see our player. First, we'll adjust the collision on the player.
Expand the Player Objects folder
Select the AxisAlignedRectangle
Change the Width to 16
Change the Height to 24
We'll also adjust the movement of the player so it's a little easier to control.
Click on the Player entity
Select the Entity Input Movement tab
Change the variables in this tab to the following values
Ground
Max Speed = 130
Jump Speed = 270
Air
Max Speed = 130
Next we'll zoom in the camera to make it easier to see the player.
Click the Camera icon in Glue
Change Resolution to 360x240
Change Scale to 300%
Now our character is easier to control and see.
To help us understand the code that we will be writing, let's first look at block breaking implementation concepts. For the player to break the blocks, a number of things must happen:
The player must collide with the blocks. This may seem like an obvious requirement, but it's worth noting since it we will be writing code in the PlayerListVsBlockList collision relationship.
The player must hit the block from below.
Only one block can be destroyed at a time. Modern Super Mario Bros. games do support destroying multiple blocks at the same time, but the older games only support one at a time. We will follow the old approach.
Once a block is destroyed, the surrounding block RepositionDirections must be adjusted to prevent snagging.
Of the four concepts listed above, the one which requires the most attention is the third.
To understand why breaking one block at a time adds complexity to our implementation, let's consider a few possible collision scenarios. The first scenario is the simplest - where the player collides with a single block as shown in the following image:
The player is represented by a blue rectangle and the block is represented by a red rectangle. In this case, the two overlap, and the player is hitting the block from below so the block should break. Keep in mind that the player may not be exactly below the block. The block should break even if the player is slightly offset, as shown in the following image:
In the image above, the block would still break. Of course things get a little more complicated when we add multiple blocks. For example, consider the following image:
Which block should break in this situation? The player overlaps more of the block on the right than the block on the left, so breaking the block on the right seems best. Unfortunately, when we perform collision between entities and players, the block on the right may not ever collide with the player. CollisionRelationships perform collisions between one player/block pair at a time, and which pair is checked first depends on the order of blocks in the BlockList, which can change depending on partitioning. In other words, when using CollisionRelationships, we cannot depend on collision being performed in a particular order. With this in mind, it's possible that the left-most tile collides with the player first, which would result in the player being shifted down, as shown in the following image:
If the left block collides first and the player is pushed down as a result of the collision, the right block will not perform collision with the player. The events raised for collision will only be raised for the left block. As explained above, the block which actually collides with the player is arbitrary - it could be the left or it could be the right. This means that we cannot rely purely on the Player vs Block relationship to decide which block to destroy - at least, not if we plan on controlling which block is destroyed in such a situation.
We can solve this problem by adding a second collision object to the Player to help us decide which block to destroy. Glue supports objects with multiple collision objects. Consider the following image, where the Player object has two collision shapes - a blue rectangle for the body and a green rectangle for determining which rectangle to destroy.
Of course, it is possible that a block collision may occur even without the green rectangle colliding, as is the case in a single-block collision where the player is offset from the block.
This means that we can not rely completely on either the body (blue) or sub collision (green) to decide which block to destroy. Instead, we need to use both to decide which to destroy. The logic should be as follows:
If the player collides with a block from below, then the player will destroy a block.
First, check all blocks to see if any block collides with the sub collision (green rectangle). If so, destroy that block.
Otherwise, if the sub collision (green rectangle) does not collide with any blocks, destroy the block that collided with the player body (blue)
This means that the sub collision is an optional collision. It should only be performed if the player body collides with the blocks from below. Therefore, we will be performing this collision manually in code rather than creating a collision relationship.
As mentioned above, Glue supports objects with multiple collision shapes. To add a new shape to the Player:
Select the Player in Glue
Select the Quick Actions tab and click the Add Object to Player button
Select the AxisAlignedRectangle object type
Enter the name BlockCollision
Click OK
Select the newly-created BlockCollision
Set the following values:
Y = 12
Width = 1
Height = 5
Our player now has a small rectangle at the top. However, this rectangle is currently considered as part of the entire player. This is most evident when hitting blocks from below. Notice that the new sub-collision performs solid collision.
We don't want this rectangle to perform solid collision - it should only be used in our code to check which block to break. To fix this, we can mark the BlockRectangle as being excluded from the default list of shapes used in collision. To do this:
Select the BlockCollision object under Player
Click the Properties tab
Set the IncludeInICollidable to false
Now BlockCollision will still be part of the Player object but will not be considered in any CollisionRelationships (by default).
Now that we have a BlockCollision rectangle, we can perform the logic explained above. We'll treat each of the four steps separately.
We can handle Player vs Block collision by adding an event to this collision relationship:
Expand GameScreen Objects folder in Glue
Expand the Collision Relationships item
Select PlayerListVsBlockList
Select the Collision tab
Click the Add Event button
Click OK to accept the defaults
Glue has now added an event to the GameScreen.Events.cs file which is called whenever the Player collides with a Block.
The OnPlayerListVsBlockListCollisionOccurred method is called whenever the player collides with a block from any side. This means that if the player is standing on a block, this function will be called every frame. We only want to destroy blocks if the player hits a block from below. Whenever a solid collision occurs, the objects being collided must be separated to prevent overlap. Since blocks cannot be moved when a collision occurs, the player must be moved. When hitting a block from below, the player must be moved downward (negative Y). We can check the player's AxisAlignedRectangleInstance.LastMoveCollisionReposition.Y value to see if the player was moved down due to the collision. If so, then the player hit the Block from below. Modify the OnPlayerListVsBlockListCollisionOccurred method as shown in the following code snippet:
Now we can perform our destroy logic. As mentioned above, we will first check if the BlockCollision collides with any blocks. If so, we will destroy the block. Otherwise we will destroy the block that the body collided with. To do this, modify the OnPlayerListVsBlockListCollisionOccurred method as shown in the following code snippet:
Now if we run the game, the Player can only destroy one Block at a time. If the Player collides with multiple Blocks, then the one which is directly above the Player will be destroyed. Of course, we there are some collision problems caused by the RepositionDirections not being adjusted properly. This is evident when trying to destroy the top row of blocks.
The RepositionDirections are initially set when all of the Blocks are created, and will work properly until one of the Blocks is removed. Once a Block is removed, surrounding Blocks must have their RepositionDirections updated. To do this, we will remove the Block's collision from the CombinedShapeCollection before destroying the Block. Doing so will result in all adjacent collisions updating their RepositionDirections. To do this, modify the OnPlayerListVsBlockListCollisionOccurred method as shown in the following code snippet:
Notice that the rectangle is removed before the Block is destroyed. Also, notice that the RemoveRectangle method is called in two spots since we destroy blocks in two spots in the code above. Now the RepositionDirections are updated properly and the player will be able to collide with the blocks properly even after blocks are destroyed.
If you've made it this far, congratulations! You have worked through a complex collision example in FlatRedBall. Now you should be able to add blocks freely in Tiled and the game will collide properly.
The sample project can be downloaded from Github:
For detailed information about how to create a localization database, see the . The LocalizationDatabase.csv file includes two columns. The leftmost column is the string ID - this is how code accesses the text in the CSV. The second column is the text for the string ID in English (as the column name indicates).
- the PNG containing the art for our game
- the tileset for the visuals in our game
- the map for our game which has the pre-made visuals
Return to Glue and click the Folder icon to open the project folder
For more information on texture pixels, see the page. Now our entity is using the breakable block graphic.
This set of tutorials covers how to create enemy movement in a platformer game. Enemies in many platformers move horizontally until hitting a wall or the edge of a platform, then turn around.
This tutorial creates an Enemy entity which uses Platformer physics, but is fully controlled by logic rather than input devices like the keyboard.
This tutorial creates an Enemy entity which is used in the remainder of the tutorials. This enemy is similar to the Player entity - it has collision and uses Platformer physics, but it does not use input from a keyboard or gamepad - instead its movement is controlled purely in code.
To create the Enemy entity:
Click the Quick Actions tab
Click the Add Entity button
Enter the name Enemy
Check AxisAlignedRectangle
Check Platformer for the Input Movement Type
Leave the rest of the defaults and click OK
You can optionally change the color of your Enemy if you would like by selecting the newly-created AxisAlignedRectangle and setting its color to Red.
Entities such as Enemy are usually added directly to levels such as Level1. Note that it is possible to add Entities in more ways including through Tiled and directly in code, but we will be adding an instance directly in the FlatRedBall Editor.
Note that the EnemyList object is defined in GameScreen, but it is also accessible in all levels, such as Level1.
The EnemyList must be accessible in both screens. It is used in GameScreen to create collision relationships, which should always be the same across all levels. It is used in each level to define which enemies should appear in a particular level. This setup is enabled by the EnemyList having its ExposedInDerived property set to true, which is set up by default from when we created our Enemy entity.
To add an enemy to Level 1, drag+drop the Enemy entity onto the Level1 Screen.
The newly created entity is automatically added to Level1's EnemyList.
Next the Enemy should be positioned so it is standing on the ground. You can position this through trial-and-error by running the game and adjusting the position of the entity, but the fastest way to accomplish this is in Live Edit mode. For information about setting up Live Edit, see the Live Edit page.
Now we have a fully-functional enemy, but it falls through the solid collision since we haven't yet set up an EnemyList vs SolidCollision relationship. To do this:
Expand GameScreen Objects folder
Drag+drop the EnemyList onto SolidCollision. Notice that we are doing this in the GameScreen rather than Level1 because we want all enemies to collide with the SolidCollision regardless of level. By doing this in the GameScreen, this new Collision Relationship will apply to all levels, including Level1.
Glue automatically sets the Collision Physics in the new Collision Relationship to Platformer Solid Collision since the Enemy entity is marked as a Platformer.
Now we have an Enemy entity and an instance of this Enemy in Level1. This instance collides with the game's SolidCollision and has full support for platformer physics. Currently both the Player and Enemy are controlled by the keyboard (or gamepad if one is plugged in). We will remove this input control from the Enemy and replace it with logic-based movement in the next tutorial.
This tutorial will implement dealing damage to an Enemy. Rather than immediately destroying the Enemy when it collides with a Bullet, we'll implement health points (HP) which can be reduced when an Enemy collides with a Bullet. Once an Enemy's HP is reduced to 0, it will be destroyed.
Typically enemies will have two values for HP:
MaxHP - the starting number of HP when an Enemy is first spawned
CurrentHP - the number of HP that the enemy currently has
The MaxHP is a designer variable - a variable which a game designer may change during the development of the game to adjust difficulty. We'll define this in Glue so it can be changed easily. By contrast, CurrentHP is a logic variable - a variable which is used in game logic and which should not be modified by a game designer. It will initially be assigned to MaxHP and will be reduced whenever the Player takes damage. To define a MaxHP variable:
Select the Enemy entity
Click on the Variables tab
Click the Add New Variable button
Select the int type
Enter the name MaxHP
Click OK
Set the new MaxHP variable to 6
Now we can add the CurrentHP value to the Enemy:
Open the project in Visual Studio
Open Enemy.cs
Add the CurrentHp variable and modify CustomInitialize as shown in the following code snippet
In a typical game, whenever an Enemy takes damage the game may perform many actions such as:
Flashing the Enemy or playing some visual effect
Playing a sound effect
Showing the amount of damage dealt (as a floating number)
Modifying how much health is shown in a health bar
Destroying the Enemy and showing a death effect if the Enemy has died
Keeping this logic in the Enemy.cs file can help keep code organized, so we'll be adding the following function to the Enemy.cs file:
Be sure to make the TakeDamage method public so it can be called from outside of the Enemy class.
Now we can create a collision relationship between the BulletList and EnemyList objects in the GameScreen:
Expand the GameScreen Objects folder
Drag+drop the BulletList object onto EnemyList\
Select the new BulletListVsEnemyList relationship
Click on the Collision tab
Click the Add Event button
Accept the defaults and click OK
Now we can fill in the event method. Open GameScreen.Event.cs and modify the BulletListVsEnemyListCollisionOccurred method as shown in the following snippet:
Now we can shoot at the enemy. After six shots, the enemy is destroyed.
Now we've created a game where we can destroy an enemy with 6 shots. A real game would have more than one enemy, but if we add additional enemies, our game will automatically collide bullets against each enemy, and each enemy will keep track of its own HP.
This tutorial covers how to add an Enemy to your project. To speed up the process we'll be importing an existing entity rather than building a new one ourselves. Once imported, we will modify the entity so it has the functionality we'll need for this tutorial - specifically adding the ability for the enemy to take damage.
To import the entity
Download the exported entity file from here: https://github.com/vchelaru/FlatRedBall/raw/NetStandard/Samples/Platformer/DealingDamage/Enemy.entz
In Glue, right-click on the Entities folder
Select Import Entity
Navigate to the location where you saved the Enemy file earlier and click OK
We now have a fully-functional enemy which has:
Collision shape named AxisAlignedRectangle
Sprite displaying a walking animation
Platformer values so that it can collide with solid collision
EnemyInput object which will keep the enemy from moving (does not use the gamepad or keyboard)
We will add a list to our GameScreen and an instance of our Enemy to Level1 so we can see the enemy in game. To add an enemy list to GameScreen:
Select the Enemy entity
Select the Quick Actions tab
Click the Add Enemy List to GameScreen button - this creates a list of enemies which we'll use to create collision relationships later
Click the Add Enemy Factory button - this allows us to create enemies in code and Tiled maps.
To add an enemy to your Level1:
Drag+drop the Enemy entity onto Level1\
Select Enemy1 and click on the Variables tab
Set X to 160
Set Y to -160
Now we have an enemy in the game, but it falls through the level. We can fix this by telling the enemies to collide against our GameScreen's SolidCollision object:
Expand GameScreen
Expand the Objects folder
Drag+drop the EnemyList onto the SolidCollision object
If we run the game now, the enemy will fall and land in the level next to the player.
This tutorial adds a Bullet entity which the player can shoot. The Bullet entity has the following characteristics:
It will be visually represented by a circle
It moves left or right depending on which way the Player is facing when shooting
It will be destroyed when colliding with the SolidCollision
It will be destroyed when colliding with the Enemy
To create a Bullet:
Click the Quick Actions tab
Click the Add Entity button
Enter the name Bullet
Click the Circle checkbox under Collisions
Leave all of the rest of the values default and click OK
When a Bullet is created, it will move either left or right. We need to control the speed of the bullet. We will create a variable which we'll use in our code later:
Select the Bullet entity
Click on the Variables tab
Click the Add New Variable button
Verify that float type is selected
Enter the name BulletSpeed
Click OK
Enter a value of 300 for BulletSpeed
We will also want to change the radius of the Bullet's CircleInstance:
Expand the Bullet Objects folder
Select CircleInstance
Click the Variables tab
Change Radius to 6
The Bullet creation logic will be added to the Player entity. We need to detect if the shoot button has been pressed. If so, we'll create a new bullet and have it move in the direction that the player is facing. To do this, open Player.cs in Visual Studio and modify the code as shown in the following snippet:
The first line of code in the Player class defines an IPressableInput. This is an object which can reference any pressable input hardware such as a keyboard key or an Xbox360GamePad button. We create this IPressableInput so that we can write code which will work regardless of input device. For more information on IPressableInput, see the IPressableInput page.
Whenever the input device is set on a platformer entity, the CustomInitializePlatformerInput method is called. Since our entity has custom input for shooting, we add the CustomInitializePlatformerInput where we assign shootingInput according to our InputDevice type. In this case we assign shooting to the right ALT key if using a keyboard and the X button if using an Xbox360GamePad. Any re-assignment of input should be done in CustomInitializePlatformerInput rather than CustomInitialize. This is because the order in which code is executed. When considering input, assignment, the following is performed:
CustomInitialize
InputDevice is assigned (can be assigned in generated code or custom code)
CustomInitializePlatformerInput which is called whenever the input device is assigned.
CustomInitialize always runs before an InputDevice is assigned. We want our shootingInput-assigning code to run after the InputDevice is assigned, so we should put it in CustomInitializePlatformerInput.
Finally, we check our shootingInput.WasJustPressed to see if the user just pushed the input. If so, we create a bullet and set its XVelocity according to the direction that the Player is facing. If we run our game now, we can shoot bullets in the direction we're facing.
Currently, our bullets can move through walls and enemies. First we'll add collision between our GameScreen BulletList and SolidCollision:
Expand the GameScreen Objects folder
Drag BulletList onto SolidCollision to create a new collision relationship
Select the new BulletListVsSolidCollision relationship
Click the Collision tab
Click the Add Event button
Click OK to accept the defaults
Now we can destroy the bullet whenever the event occurs:
Open the project in Visual Studio
Go to GameScreen.Event.cs
Find the BulletListVsSolidCollisionCollisionOccurred method
Modify the code as shown in the following snippet:
Now we can shoot bullets and they will get destroyed when they hit the wall.
We end this tutorial with the ability to shoot bullets and destroy them when they hit the wall. The next tutorial will implement the ability to damage and destroy enemies.
This tutorial shows how to replace the default keyboard (and gamepad) input logic with movement controlled only in code.
First we'll remove the Input Device from the Enemy entity:
Select the Enemy entity
Select the Enemy Input Movement tab
Check None under Input Device
Now the Enemy will not move in response to keyboard or gamepad input. Note that if we do not assign an InputDevice in code, the game crashes so we must do so before running the game.
Next we will create our own custom InputDevice which will control the movement of the enemy. The reason we are creating this is because Platformer entities expect to receive commands from an input device like a keyboard. We can create a class which simulates input commands without actually having physical input. To create this class:
Open the project in Visual Studio
Create a new class called EnemyInput. I will place mine in an Input folder
Modify the code so that the EnemyInput class inherits from FlatRedBall.Input.InputDeviceBase
This class provides many virtual methods which we can override to customize the behavior of our enemy. For example, we can override the method for GetHorizontalValue to control which way the Enemy is walking, as shown in the following code snippet:
To use EnemyInput, we can modify the CustomInitialize method in the Enemy.cs class to assign its InputDevice. The Enemy.cs file can be found in the Entities folder in the Visual Studio Solution Explorer.
To use EnemyInput as the Enemy's InputDevice, modify the Enemy CustomInitialize method as shown in the following code snippet:
Now if we run the game, the Enemy automatically walks to the right and ignores keyboard and gamepad input.
This tutorial does not include adding jumping to the enemy behavior. If your game needs jumping, you can add this by including a primary input override in the EnemyInput class. Of course, you would only want to press the jump button under some condition. The following code snippet shows how this could be accomplished:
Our EnemyInput object can be expanded to support any type of input - we just need to have the GetHorizontalValue function return a value between 0 and 1. Note that we are only using horizontal movement for this tutorial, but we could also have the Enemy jump by implementing the GetPrimaryActionPressed method, which controls whether the jump button is down.
For this tutorial we need access to a value to indicate whether the Enemy should move to the left or right. We will ignore values inbetween -1 (left) and +1 (right), but a full game may support enemies which may stand still or move at various speeds. We'll create a new enum value and expose a property in EnemyInput so that it can be controlled externally. To do this, modify the EnemyInput class as shown in the following code snippet:
Many platformer games include enemies which turn around when colliding with other objects such as walls, platform edges, and other enemies. We will implement wall collision here since it is the simplest scenario to control enemies turning around. First we will add an event whenever Enemies collide with SolidCollision:
Expand the GameScreen -> Objects -> Collision Relationships item in Glue
Select EnemyListVsSolidCollision
Select the Collision tab
Click the Add Event button\
Click OK to accept the defaults
Glue will add an event to GameScreen.Event.cs which we can modify to adjust the EnemyInput DesiredDirection, as shown in the following snippet.
The code above assigns the EnemyInput DesiredDirection according to the enemy's AxisAlignedRectangleInstance.LastMoveCollisionReposition. We'll take a deeper look at how this property works in the next section.
The collision relationship between Enemies and SolidCollision prevents Enemy instances from overlapping the SolidCollision TileShapeCollection. Whenever an Enemy overlaps one of the rectangles in the SolidCollision TileShapeCollection, the Enemy must be repositioned so it does not overlap the solid collision. For example, consider a falling Enemy (white). Initially the enemy may be falling but not overlapping any solid collision (red).
As the Enemy continues to fall, it eventually overlaps the solid collision.
When this happens, the Enemy must be repositioned up to resolve the overlapping collision.
In this case, the RepositionDirection would have a positive Y value. For example, it may be (0,1,0). Similarly, whenever an Enemy collides with a wall in the SolidCollision TileShapeCollection, the Enemy is repositioned horizontally, so the X value of the RepositionDirection is non-zero, as shown in the following diagram:
In the diagram above, the RepositionDirection has a negative X value (points to the left) so we know that the wall is to the right of the Enemy. If the RepositionDirection has a positive X value, then the wall is to the left of the Enemy.
Now that we have this logic in place, our Enemy will automatically walk until colliding with a wall, then it turns around and walks in the other direction.
The next tutorial will enable Enemies to turn around when reaching the end of a platform.
This set of tutorials covers how to change a Player's movement values in response to colliding with various ground types (such as ice). It also covers how to implement swimming, and how to transition between in and out of water movement. Multiple terrain types and water movement are common in many platformers such as Super Mario World.
This tutorial begins with an empty project and will create a Player which changes its movement values according to collision with different ShapeCollections.
The default Platformer project creates a game with a level (Level1). We will modify this level to create ice and water tiles.
We can open Level1Map.tmx to add additional tiles.
Level1Map.tmx should appear in the Tiled app. To make sure that no tiles from other maps are being used, be sure to close other TMX files in Tiled.
Default maps include a tileset named TiledIcons. Most of these icons have no built-in functionality, so they can be used to add custom behavior to your game. In this case we will use the ice and water tiles which are part of the tileset.
We can paint these tiles, along with more solid collision tiles, to create a test level with ice and water.
We have a level where the player can collide with the solid ground, but ice and water do not yet affect the Player's movement.
The next tutorial shows how to add collision and change the player's movement values on ice and in water.
This tutorial, like many others in this documentation, begins with an empty Glue project. I'll call mine PlatformerMovement.
Keep the Open New Project Wizard option checked.
Once the project finishes loading, Project Setup Wizard appears. Select the Standard Platformer option.
Now our project is ready to go. If we run the game we have a standard platformer project.
This tutorial shows how to add ice and water collision. We'll be setting up the TileShapeCollections for these two types of tiles, and creating collision relationships to control the interaction between the Player and these tiles.
As shown in the previous tutorial, the Player already collides with the solid collision. This is automatically added by the New Project Wizard, so we don't have to do any setup for ground collision. We'll add ice collision first. To do this:
Click GameScreen
Select the Quick Actions tab and click the Add Object to Game Screen button
Select TileShapeCollection
Enter the name IceCollision
Click OK
Select the newly-created IceCollision object
Click the TileShapeCollection Properties tab
Click the From Type option
Change the Source TMX File/Object to Map
Change the Type to IceCollision
We'll repeat the process above to create water collision:
Click GameScreen
Select the Quick Actions tab and click the Add Object to Game Screen button
Select TileShapeCollection
Enter the name WaterCollision
Click OK
Select the newly-created WaterCollision object
Click the TileShapeCollection Properties tab
Click the From Type option
Change the Source TMX File/Object to Map
Change the Type to Water
Now our game has two new collision relationships: IceCollision and WaterCollision. This means that when our game runs, collision shapes are created based on the water and ice tiles, but we haven't yet told the game how to handle collisions between the Player and these collision relationships. First, we'll set up collisions between the Player and IceCollision:
Drag+drop PlayerList onto IceCollision
The newly-created PlayerVsIceCollision is automatically set up to use Platformer Solid Collision physics, so we don't need to make any changes. We also need to create collision between PlayerList and WaterCollision, but this time we need to disable Platformer Solid Collision since the Player should be able to fall into the water. To do this, drag+drop PlayerList onto WaterCollision and select No Physics.
Now if we run our game we can collide with the ice tiles and fall through water.
Now our game has ice and water TileShapeCollections and collision relationships. You may have noticed that the ice currently acts identical to solid collision (bricks). The next tutorial will create new platformer variables for moving on ice and swimming in water, and will switch between them in response to collision.
As of the last tutorial our game has three tile shape collections:
SolidCollision - the bricks that make up most of the level
IceCollision - floating blocks of ice which should be slippery
WaterCollision - an area of the level where the player should be able to swim
This tutorial shows how to create new platformer values for ice and water, and how to change between them depending on collision type.
Before we change which platformer values are used based on collision, we first need to define the platformer values for the different movement types. As mentioned before, we will have the following platformer values:
Ground - already defined by default
Air - already defined by default
Ice - will apply when user collides with the ice TileShapeCollection
Water - will apply when the user collides with the water TileShapeCollection
To add a new movement value:
Click the Player entity
Click the Entity Input Movement tab
Click the Add Control Values button
Set the following values:
Movement Type = Ice
Max Speed = 160
Speed Up/Down = true (click radio button)
Speed Up Time = 1
Slow Down Time = 1
Jump Speed = 270
Hold to Jump Higher = true (click checkbox)
Max Jump Hold Time = .17
Repeat the process above to create a water movement. Click Add Control Values again, and set the following values:
Movement Type = Water
Max Speed = 120
Speed Up/Down = true (click radio button)
Speed Up Time = 1.3
Speed Down Time = 0.5
Jump Speed = 120
Hold to Jump Higher = true (click checkbox)
Max Jump Hold Time = 0.3
Gravity = 210
Max Falling Speed = 90
Now that we have Ice and Water movement defined, we can write code in the Player class to set the movement values according to what the player has collided with. To do this:
Open the project in Visual Studio
Open Player.cs
Add the following code to CustomActivity:
Now our Player will change movement values when moving on ice and solid ground.
This concludes the platformer movement value tutorials where we use multiple TileShapeCollections to change the movement values for the player between regular ground/air movement, ice, and water movement.
We will add this logic to our Enemies in this tutorial.
Currently Enemies have a single collision shape called AxisAlignedRectangleInstance.
This shape acts as the Enemy's body. It can be used to keep the enemy from moving through the ground, walls, and ceiling (the SolidCollision TileShapeCollection). Currently this shape does not provide enough information to detect ledges. From the point of view of the AxisAlignedRectangleInstance, this first situation...
... provides the same information as this second situation...
Both situations are identical in the collision event - they both result in collision and both have the same RepositionDirection. To detect if the Enemy is near the edge of a platform, we can use a new collision rectangle (displayed in green). If this green rectangle is not colliding with SolidCollision, then the player is near a ledge. For example, in the following diagram the Enemy is near the edge of a platform on the right side:
If the player is not near a ledge, then the green rectangle will collide with solid collision.
Since we want to detect ledges on both the left and right side, we need to have one new rectangle on each side.
When we define these rectangles, we need to remember a few things:
These rectangles should not be used for solid collision - they should be excluded from the ICollidable interface
These rectangles should be smaller than the size of our tiles (16x16) so they do not reach across gaps or down pits
These rectangles will be used to apply logic when not colliding with the SolidCollision. Normally we perform logic when a collision occurs, so we will need to write custom code to handle this situation.
As mentioned above, we'll need to create two AxisAlignedRectangle instances in our Enemy entity - one for left edge collision and one for right edge collision. To create the left edge collision rectangle:
Select the Enemy entity
Select the Quick Actions tab and click the Add Object to Enemy button
Select the type AxisAlignedRectangle
Enter the name LeftEdgeCollision
Click OK
With the LeftEdgeCollision object selected, click on the Variables tab
Change X to -12
Change Y to -12
Change Width to 8
Change Height to 8
To help visualize, change the Color to Green
Repeat the steps above to create RightEdgeCollision, but change the X value to a positive 12 so it sits on the right side of the Enemy.
Now our enemy has two extra rectangles which it can use for detecting edges, but these rectangles are being used for solid collision. To fix this:
Select the LeftEdgeCollision
Click on the Properties tab
Change IncludeInICollidable to False so that the rectangle is not used in default collision relationships (like EnemyListVsSolidCollision in GameScreen)
Repeat the steps above for the RightEdgeCollision and the Enemy will no longer use these two rectangles for solid collision.
Normally when performing logic related to collision, we do so inside a collision relationship event. Detecting an edge is different because we need to respond to a situation when there is no collision. Therefore, we will be writing code which happens every frame inside of our GameScreen CustomActivity. To detect if the enemy should turn around, modify the GameScreen.cs file so that its CustomActivity matches the following code snippet:
This code shows that we can check collisions directly in code at any time, rather than relying purely on CollisionRelationships. Notice that we only perform this logic if the Enemy is on the ground. Otherwise, enemies would alternate directions very quickly when falling. If using the default level, this logic won't do anything since there are no raised platforms. We can modify Level1Map to add a few platforms. Feel free to adjust it however you like, as long as you have platforms with edges which we can use to test this logic.
We should also adjust the Enemy1 starting position so it starts on a ledge instead of falling from the top of the level. Change the X and Y values on Enemy1 in Level1 so that it starts in the right spot. This may take a little trial-and-error. Once you have the Enemy on a platform, you may notice that it still runs off the edge. The reason for this is because the Enemy moves very quickly and it takes time to slow down when changing directions.
To do this:
Select the Enemy entity
Click the Entity Input Movement tab
Change the Max Speed to 100
Change the Slow Down Time to 0.03
These changes enable the enemy to turn around without falling off of the platform. It's important to note that with enough Max Speed or a large enough Slow Down Time will result in an enemy sliding off of an edge.
Now our enemy has functionality for turning around based on wall collision and edge detection. The two approaches shown here could also be used to turn around when Enemies collide with other Enemies (also using reposition values), but we'll leave this and other situations to individuals who are interested.
Some games include enemies which turn around when reaching a platform edge. For example, in Super Mario World, the red Koopa enemies (turtles) walk until they reach an edge, then turn around.
FlatRedBall provides a standardized, powerful, and flexible damage dealing system which can reduce the amount of code needed to add damage dealing to your game. This functionality is built around the IDamageable
and IDamageArea
interfaces. The following tutorials provide a step-by-step guide on how to add damage to your game.
This tutorial begins with a new platformer project. To create a new platformer project, see the Platformer Basics tutorial. In short, the easiest way to get an animated character is to use the new project wizard.
If you started a new project without using the wizard, you would need to create an entity with a sprite and assign a .achx file to the sprite. If you don't have a .achx file prepared already, use these below to give yourself a head start.
Once finished, your project will have a character which is animated and ready to go. The remainder of the tutorials in this series cover how to modify the animations on the player and how to add new animations in response to game logic.
This walkthrough covers the concept of wall sliding and wall jumping. When a player is in the air and pressing against a wall, the player's fall speed will slow the player's falling speed and enable jumping.
This tutorial assumes a project that has been created with the platformer wizard options, but this is not a requirement. Wall sliding and jumping can be added to any platformer project.
This walkthrough covers a number of main concepts for wall jumping
Detecting if the user is "pressing" against a wall when in the air
Defining sliding platformer values
Forcing the player to move "outward" on wall jumps
Playing animations when wall jumping
When the player is sliding against a wall, the player's movement is different than when the player is in the air. The most obvious changes are:
The player falls more slowly (max fall speed is reduced)
The player can jump off of the wall
We can create a new set of movement values by copying the existing Air movement values:
Once copied, change the following values:
Movement Type = WallSliding
Jump Speed = 250
Max Falling Speed = 150
Once the WallSliding values have been defined, we need to detect if the player has collided with a solid wall. We can do this by adding the following method to Player.cs which returns whether the player is wall sliding.
The code above assumes that any horizontal reposition on a collision allows wall jumping. You may want to further restrict wall jumping to certain collision types to prevent the player from wall jumping in some situations. For example, you may not want the player to wall jump when sliding against spikes.
To do this, you can check the LastFrameItemsCollidedAgainst or LastFrameObjectsCollidedAgainst to only allow wall jumping if colliding against certain objects. For example, your if statement may be modified as shown in the following code snippet:
The GetIfIsSlidingOnWall can then be used in the CustomActivity to decide whether the player's AirMovement should be WallSliding or Air, as shown in the following code:
Note that this code uses a property IsSlidingOnWall rather than a local variable. Although a property provides no additional benefits on this code, it will be used later when assigning animations.
With this code in place, the player can now slide down walls and jump when sliding.
The implementation above results in the Player being able to jump against walls indefinitely. Many games push the player outward on jumps. Some games such as Mega Man X push the player outward only slightly, allowing the player to climb a wall through wall jumps. Other games such as New Super Mario Bros U Deluxe push the player outward far enough that the player cannot climb up a single wall.
Whether a player can climb a wall depends on the outward XVelocity applied on a jump.
This code adds a handler to when the player jumps. The jump checks if the player is sliding on the wall and if so, pushes the player "outward". Note that the outwardVelocity is defined here in the HandleJumped method, but you may want to put this variable in the Player entity so that it can be tuned without changing code. Also, keep in mind that a smaller outwardVelocity value results in the player being pushed outward less. If the value is small enough then the player will be able to climb the wall through wall jumping. With the values used here, this value is roughly around 30.
We can add sliding animations by checking the IsSlidingOnWall variable either in code or in the Animations tab for the player. If you are already using the Animations UI, then handling the IsSlidingOnWall variable requires no code. To do so:
Select the Player Entity, and select the Entity Input Movement tab
Click on the Animation item
Copy the CharacterFall animation row
Select CharacterWallSlide animation on the new row
Enter IsSlidingOnWall for the Custom Condition on the new row
The player will now play the CharacterWallSlide animations if the IsSlidingOnWall property is set to true. Note that the .achx file contains CharacterWallSlideLeft and CharacterWallSlideRight, but the generated code selects the appropriate one based on which way the player is facing.
This set of tutorials covers how to write logic to control a platformer animation and to change platformer values according to input (such as a run button and ducking).
This tutorial will create a game similar to Super Mario World. The first begins with a completely empty project and covers the FlatRedBall platformer technology and best practices.
Platformer games (such as Super Mario World) include animated characters which play animations in response to input, physics, and collision with the environment. For example, a game like Super Mario World includes a number of animations:
Idle
Walk
Run
Jump
Duck
Look up
Skid (Trying to turn around when running)
Each of these animations is played in response to logic determining which should play. We can think of each of these animations as having different priority relative to other animations. For example, if the player is pushing the Up direction in Super Mario World, Mario looks upward. This animation has priority over the Idle animation, which plays if the Up is not held. Similarly, the Run animation is played if the player is moving faster than a certain speed, is on the ground, and is holding the run button. If this condition fails, then other animations (such as Walk) perform their logic. The logic to set animations can be performed through standard logic, but FlatRedBall offers a standard animation solution through the AnimationController class. This tutorial explores AnimationLayers and AnimationController which are used to update the Player animations.
By default our Player already has an AnimationController. It doesn't appear in the Player object in the FlatRedBall Editor, but it is generated automatically based on the default settings on the Player. AnimationControllers (both in code and in the FlatRedBall Editor) are containers for AnimationLayers. Each AnimationLayer typically maps to a single animation (or two animations - a left and right facing version). By default, our player has animations for idle, walk, fall (in air, moving downward), and jump (in air, moving upward). We can see these layers by selecting the Player, clicking on the Entity Input Movement tab, and selecting the Animation
Each animation layer plays if its conditions evaluate to true and if the animations below it evaluate to false. Another way to think about this is, entries further down can "overwrite" the current animation if they evaluate to true, with the bottom-most getting the highest priority if it evaluates to true.
Each animation layer defines which animation is played, conditions for playing, and animation speed to play. For example, the CharacterWalk animation plays only if the player has a greater velocity than 1 and if all other animations (CharacterFall and CharacterJump) fail to play.
If we change the Min X Velocity Absolute, we can modify the behavior of the player. For example, if we change the value to 140, the player will only play the walk animation if walking faster than 140 units per second.
While this specific example may not be a practical change to the game, it shows how the logic controls whether an animation is played, and how that logic can cascade up through the layers. The following conditions can be used to control whether an animation plays:
Min X Velocity Absolute - this is the minimum speed that the player must be moving to play the animation. For example, we set the minimum speed to 140 for the CharacterWalk animation, which means the character will only walk if moving faster than 140 units per second. Note that this is Absolute which means the character must be moving either greater than 140 pixels (to the right) or less than -140 pixels (to the left).
Max X Velocity Absolute - this is the maximum speed that the player can move to play the animation. For example, setting the maximum speed to 200 means that if the player is moving faster than 200 units (absolute), then the animation will not play.
Min Y Velocity - the minimum Y velocity (vertical movement) the player must have to play the animation. For example our player has CharacterJump animation which plays only if the velocity is greater than 0.
Max Y Velocity - the maximum Y Velocity (vertical movement) that the player can move to play the animation. If the player is moving faster than this value upward, then animation will no longer play.
Movement Type - this option restricts an animation to whether the player is on the ground, air, or either. For example, this can be used to restrict animations to only play when in the air, such as the CharacterFall and CharacterJump animations.
Movement Name - this option restricts an animation to only play when a particular set of Movement Values are active. This is a dropdown which contains all of the movement values defined in the Movement Values section. This is especially effective if you have logic which is already setting the Movement Values in code - such as allowing a user to run when the run button is held.
Custom Condition - This allows the entry of code (such as a bool variable name) to be checked for whether an animation should be played. See the section below for more information on this property.
The Custom Condition property is the most flexible solution for setting animations. Typically this is used in combination with a bool field or property in the Player object. For example, the Player may have an IsTired property which would return true if the Player's health is below 10% of the max.
This can be used by the animation layer by entering the IsTired variable in the Custom Condition text box.
This produces generated code which checked IsTired in code. The text entered in Custom Condition is directly copied over so you can add any text there which evaluates to true, although the recommended approach is to encapsulate any logic in a property (as shown above) which is then referenced.
The Player contains an .achx file (Animation Chain List XML) which defines all available animations for the Player.
This file can be opened in the Animation Editor to view all animations. To open this file, either:
Double-click the .achx file. The FlatRedBall Editor uses the Windows file association for .achx --or--
Go to the <unzipped FRBDK location>/FRBDK/Xna 4 Tools/AnimationEditor/AnimationEditor.exe
Notice that this .achx file happens to include animations which are not used by the Player - that's okay because not all animations must be used, and a single file can be shared by multiple entities. The FlatRedBall Editor scans this file and provides these animations in the dropdown.
Note that the .achx file contains "Left" and "Right" versions of character animations, but the FlatRedBall Editor only provides the names without Left/Right suffixes. This keeps the Animation Name dropdown smaller and makes it easier to read. This naming convention is encouraged for character animations, and if the Has Left and Right checkbox is checked, then only animations with Left and Right suffixes appear in the dropdown. If animations do not have left and right options, then this option can be unchecked. Then the full name of all animations show up in the dropdown.
The Animation Speed Assignment dropdown provides the animation speed assignment logic. By default, all animations use the ForceTo1 speed, which means the animation plays at the normal speed as defined in the .achx. This dropdown provides a number of options:
ForceTo1 - the Sprite's AnimationSpeed will be forcefully set to 1, so the animation will play at the same speed as defined in the .achx
NoAssignment - the Sprite's AnimationSpeed is left untouched. Previous assignments on the AnimationSpeed (either in custom code or through other animation layers) will remain on the Sprite.
BasedOnVelocityMultiplier - if values are specified, then the Absolute X Velocity Animation Speed Multiplier or Absolute Y Velocity Animation Speed Multiplier values are multiplied by the Player's runtime velocity to determine the speed. For example, if the X Velocity multiplier is set to .1, then the animation speed plays at full speed if the Player is moving at 10 units per second, and plays at 10 times the speed if the Player is moving at 100 Units per second.
BasedOnMaxSpeedRatioMultiplier - if these values are specified, the animation speed will be specified based on the speed of the Player relative to the max speed of the current movement value. For example, the Max Speed on the ground is set to 160 units on the Player. If the MaxSpeedXRatioMultiplier value is set to 1, then the animation plays at full speed when the player moves at max speed. If the MaxSpeedXRatioMultiplier is set to 2, then the animation plays at twice full speed when the player moves at max speed.
For example, the CharacterWalk animation can be set to use BasedOnMaxSpeedRatioMultiplier with a Max Speed X Ratio Mutiplier of 2. This results in the walk animation playing faster based on the movement speed of the player.
Animation layers can be added in the FlatRedBall Editor. New animations layers can be added through the Add button at the bottom of the animation list.
Also, existing animations can be copied with the copy button.
This guide covers all of the options available when defining animation layers. The next tutorial covers changing the platformer movement values.
The removal of IDamageArea entities can sometimes be handled automatically by collision relationshps, but sometimes this removal requires custom code. This tutorial discusses how IDamageArea removal can be customized.
The previous tutorials set up a BulletVsEnemy collision relationship which resulted in the removal of the Bullet when it collides with the enemy. To review, the removal of the bullet required the following:
A collision relationship between BulletList and EnemyList exists in GameScreen
The Enemy and Bullet have different team indexes
The BulletVsEnemy collision relationship has the Destroy Bullet on Damage option checked
Collision relationships with IDamageArea lists always display the Destroy option, even if the collision relationship is not against an IDamageable. This allows IDamageAreas such as Bullets to destroy themselves when colliding with non-IDamageArea objects such as solid collision. For example, a collision relationship between BulletList and SolidCollision can be added to the GameScreen to automatically destroy bullets. This collision relationship will have the Destroy Bullet on Damage option checked by default.
Bullets are now destroyed automatically when they collide with the wall.
Although the automatic removal is handy, we may want to control the removal of bullets. For example, we can modify our game so the bullets do not immediately disappear when hitting an enemy, but instead deal continuous damage over time. First, we'll modify the BulletVsEnemy collision relationship so bullets are not removed automatically by unchecking the Destroy Bullet on Damage option.
Now bullets will continually deal damage to the Enemy, one time per frame. This results in enemies dying very quickly.
As mentioned above, since the Bullet instance does not get destroyed immediately, it continues to live and it deals damage every frame. In this case the game runs at 60 frames per second, dealing 10 damage each frame. Therefore, enemies die after overlapping a bullet after 10 frames (1/6 of a second). We can change this behavior by changing the Bullet's default Seconds Between Damage. For example, if we change the value to 0.5, then bullets deal damage once every half second.
Each enemy automatically keeps track of the last time it took damage, so it will take damage only 2 times per second. To see this clearly, we'll also change the bullet speed in Player.cs so bullets move slower (50 instead of 200 pixels per second).
In this case the removal of our bullets is ultimately handled by collision with SolidCollision. Level 1 (by default) is surrounded by solid collision, so the bullets will eventually collide with a wall and be removed. Of course, this may not always be the case, or your game may require removal of bullets after travelling a certain distance, or after being alive for a certain amount of time. In this case, you can still add logic to the Bullet entity to destroy bullets based on any condition you need. For example, bullets could be destroyed after a few seconds using the following code in Player.cs.
The usage of automatic removal through collision relationships does not prevent you from also adding custom code for entity removal. You can mix these two approaches to achieve the desired entity removal.
This tutorial covers the different ways to remove an IDamageArea entity using automatic removal or manual removal. It also covers how to deal damage over time to IDamageables.
Most of the work needed to perform damage dealing can be done through the FlatRedBall Editor and generated code. Of course, damage dealing can also be managed in code for more flexibility.
This tutorial shows how to set your entities up so they can deal damage.
Damage dealing is done through two interfaces: IDamageable and IDamageArea. As the names suggest, IDamageable entities can take damage and IDamageArea entities can deal damage. Projects which use the standard damage system should mark entities as IDamageable or IDamageArea in the FlatRedBall Editor. Keep in mind that in some cases entities (such as enemies) can be both IDamageable and IDamageAreas.
If you have created a platformer or top-down project using the new project Wizard, then your Player entity already implements the IDamageable interface. You can verify this by selecting the Player entity and checking the Properties tab.
If your player does not already implement IDamageable, you can change this property in the Properties tab to true. Once this value is set, the Player will be ready to be used in the damage system.
Any entity can implement either IDamageable or IDamageArea interfaces - or both. Let's look at some examples of how to create IDamageable and IDamageArea entities.
To add a Bullet entity which implements IDamageArea:
Right-click on Entities
Select to add a new entity
Add appropriate collision (such as a Circle)
Check the option for IDamageArea
Keep the Team Index to 0 (Player Team) if the bullet is created by the Player. This can be set on a per-instance basis in code if your game has bullets that can damage enemies and players (such as a shoot-'em-up game).
Your Bullet entity is now marked as a IDamageArea.
The Team Index specified in the new Entity window defines the default team index. The default team index can be overridden in code, but specifying a team index enables the FlatRedBall Editor to generate collision relationships automatically. For example, consider a game which already has a Bullet defined which uses the Team Index of 0 as shown above. If a new Entity is created using Team Index 1, then collision relationships can automatically be created.
To create an Enemy entity:
Right-click on Entities
Select to add a new entity
Add appropriate collision (such as an AxisAlignedRectangle)
Check the option for IDamageable so the enemy can take damage from the bullet. Also, enemies may deal damage to the player so the enemy could also be marked as IDamageArea in a full game. To keep things simple we'll only check IDamageable for now.
Change the Team Index to 1 (Enemy Team)
Check the Add Opposing Team Index Collision Relationships to GameScreen option.
Notice that the new Entity window provides a preview of the collision relationships that are added to the GameScreen after creating the entity. This lets you check for unintended collision relationship creation. If the list of collision relationships matches what your game needs you can use the Add Opposing Team Index Collision Relationships to GameScreen option. Alternatively, if you would like to create your own collision relationships, you can uncheck this option.
In this case the game now has an IDamageArea entity (Bullet) and IDamageable entity (Enemy) on different Team Indexes. As shown in the new entity window, when the Enemy is added, the FRB creates collision relationships between the Enemy and Bullet.
In the example above, the EnemyVsBullet collision relationship was created automatically because:
Add Opposing Team Index Collision Relationships to GameScreen was checked when creating Enemy
The Enemy has a Team Index (1) different than the already-created Bullet Team Index (0)
The Enemy is IDamageable and the Bullet is IDamageArea
It is possible to manually create collision relationships between IDamageable and IDamageArea lists. For example, the default Bullet Team Index matches the Player Team Index, but your game may allow Enemy instances to shoot bullets too.
In this case we can still create a collision relationship between the PlayerList and BulletList. If the PlayerList is drag+dropped on the BulletList, a collision relationship is created with the damage-related checkboxes checked.
This guide shows how to create entities which are IDamageable and IDamageAreas. Once these entities are created, collision relationships are either automatically created when the entity is created, or they can be added manually with a drag+drop operation. The next guide will show the automatic damage-dealing logic provided by collision relationships and the Team Index variable functionality.
Now that we have our animations set up, we can work in our platformer values. These values control the way the Player entity moves in response input (such as max speed) and the physics which impact the Player's movement (such as gravity). Our Player entity automatically gets a set of default values which is why we are already able to walk and jump around the level. This tutorial will modify the default values and add additional platformer values for running and ducking.
In Glue, click the Camera icon
Change the resolution width to 256
Change the resolution height to 224
Change the TextureFilter to Point so that pixels draw without any blurring
Change the Scale so that the game is larger on your screen. A typical 1080 monitor can support the game at 400% Scale.
After making these changes the game should more closely resemble the resolution of the original Super Mario World.
Currently our game has two set of platformer movement values:
Ground
Air
Earlier we added running animations which play when the run button is held. We will be modifying our game so the Player entity can run faster when the run button is held. We will be creating two new set of platformer movement variables for running. We will also be creating a new type of movement for when the player is ducking. Therefore, we'll have three more sets of movement variables:
Running
RunningAir
Ducking
Click the Add Control Values button three times, and name the new control values as listed above.
You should now have five sets of values.
Next we'll modify the values to make the game feel a little more like Super Mario World. We'll be modifying these to get close - this is not intended to be an exact copy. Of course if you want to tune the values to make the game feel different, or if you want to mimic Super Mario World even more closely, feel free to change these values.
Max Speed = 100
Speed Up/Down
Speed Up Time = 0.25
Slow Down Time = 0.15
Jump Speed = 230
Hold to Jump Higher = true (checked)
Max Jump Hold Time = 0.2
Max Speed = 100
Speed Up/Down
Speed Up Time = 0.7
Slow Down Time = 0.7
Jump Speed = 0
Gravity = 900
Max Falling Speed = 260
Max Speed = 150
Speed Up/Down
Speed Up Time = 0.3
Slow Down Time = 0.2
Jump Speed = 250
Hold to Jump Higher = true (checked)
Max Jump Hold Time = 0.25
Max Speed = 150
Speed Up/Down
Speed Up Time = 0.7
Slow Down Time = 0.7
Jump Speed = 0
Gravity = 900
Max Falling Speed = 260
Max Speed = 0
Speed Up/Down
Custom deceleration above max speed = true (checked)
Custom deceleration value = 200
Jump Speed = 230
Hold to Jump Higher = true (checked)
Max Jump Hold Time = .2
Now that we have our platformer movement values created in Glue, we can assign them in code. To switch values, we can change the GroundMovement and AirMovement variables in CustomActivity. We will be looking at the VerticalInput (holding up/down) and the RunInput to decide whether the player should be using the default values, running values, or ducking. To switch between these platformer movement values, modify CustomActivity in Player.cs as shown in the following code snippet:
Now the Player will switch its values according to input, of course, the running animations are not currently being used. Use what was covered in the previous tutorial to see if you can modify the Player's animations to play the running and running jump animations as shown in the following animation.
The previous tutorial shows how to create new Entities which implement the IDamageable and IDamageArea interfaces. In some cases this results in the automatic creation of collision relationships, and other times manual collision relationships must be created (if the two opposing entity types have the same Team Index). This tutorial explores the Team Index property and its affect on the generated damage dealing code. This tutorial assumes a project which contains:
A Player entity implementing IDamageable with Team Index 0
A Bullet entity implementing IDamageArea with Team Index 0
An Enemy entity implementing IDamageable with Team Index 1
A collision relationship EnemyVsBullet
To test damage dealing, we need instances of entities which collide with one-another. Our game automatically contains a Player instance (this is created by the wizard) so we will use this instance to create bullets. To do this, add the following code to Player.cs CustomActivity:
Now the Player creates bullets when the Space key is pressed. Note that if you have a gamepad connected, the player automatically uses that as its input device. The space bar still shoots bullets, but movement will be controlled by the gamepad.
Next we'll create Enemy instances to shoot. To do this, add the following code to the GameScreen.cs CustomActivity:
Now we can click on the screen with the mouse to add enemies, and we can shoot these enemies with bullets.
You may have noticed that your enemies are destroyed by a single shot. We haven't set up the details for how damage should be dealt, so the default behavior is for bullets to deal damage to enemies every frame. This results in enemy health being drained very quickly, resulting in the enemies dying.
You may have also noticed that the bullets continue to move upward after destroying the enemy. We can control this behavior through an option on the collision relationships.
If we want the bullets to be destroyed after hitting the enemy, we can check the Destroy Bullet on Damage option on the EnemyVsBullet collision relationship.
This option provides a convenient way to handle a common situation - destroying the damage area (Bullet) on collision. Note that we can control whether the bullet should always be destroyed, or only if damage is dealt. Later tutorials show how to adjust the amount of damage dealt.
Also, keep in mind that these options are here for convenience but you are not required to use them. You can always leave them unchecked and handle everything manually in a collision relationship event.
With this option checked, the bullets are destroyed on collision, and each enemy takes multiple hits to be destroyed.
The EnemyVsBullet collision relationship also has the Deal Damage in Generated Code option checked. If this is checked, the following occurs on an Enemy vs Bullet collision:
The team index of the bullet and enemy are compared to see if they differ.
If so, the enemy performs damage over time checks to make sure it can take damage from the bullet. At this point our game does not have bullets which perform damage over time, but we will cover this in a later tutorial.
Damage calculations are performed by raising events which can modify how much damage the enemy receives. Currently we are not handling these events, but we will cover this in a future tutorial.
Health is subtracted from the enemy.
The enemy is killed (Destroy is called) if the enemy has less than or equal to 0 health
We can see that this logic works by shooting an enemy enough times to kill it - by default this is 10 shots.
The default variables used to deal damage and kill the enemy are defined in the Entity and Bullet entities. All entities created with the IDamageable interface default to have 100 health.
All Entities created with the IDamageArea interface default to dealing 10 damage.
If we change these values, we can change how many shots it takes to kill an enemy. For example, if the enemy is changed to have 20 health, it only takes 2 shots to kill each enemy.
As mentioned before, the collision relationship EnemyVsBullet results in damage dealing code because the Enemy and Bullet have different team indexes. Our bullets have a default Team Index of 0, so they will not deal damage to the Player even if a PlayerVsBullet collision relationship exists. To show this, we'll create a collision relationship by dragging PlayerList onto BulletList (you may have already done this in an earlier tutorial).
Even with this collision relationship, Bullets which are fired by the Player do not deal damage to the player - the generated code checks team index and both have a team index of 0 so no damage is dealt to the player.
We may want enemies to be able to shoot bullets at the player. These bullets should have the Enemy Team Index (a value of 1) which can be assigned in code. To do this, add the following code to Enemy.cs CustomActivity:
Also, you should check the option to destroy the bullet when it hits the player, but only if damage is dealt.
Now each Enemy shoots a bullet every 2 seconds which travels downward and which shares the same Team Index as the Enemy. These bullets now collide with the Player, deal damage to the Player, and ultimately kill the Player once the Player's health has dropped to 0.
When developing a full game, keep the following in mind:
Values like MaxHealth, TeamIndex, and DamageToDeal can all be set in the FlatRedBall Editor at the entity level. These should be used for useful defaults, but they can all be assigned in code.
Variables to control logic such as how frequently enemies shoot and bullet speed should be controlled through variables in the FlatRedBall Editor. These values were hardcoded above to keep the tutorial short, but do not reflect how full games should be structured.
This tutorial shows how the built-in logic in collision relationships results in automatic damage dealing and killing (destroying) of entities which can take damage. It also covered how to assign the Team Index variable in code to control whether a pair of entities result in damage dealing logic being applied. The next tutorial shows how to assign events to modify how much damage is dealt and to react to damage dealt visually.
Previous tutorials have discussed the automatic damage dealing logic provided by collision relationships. Full games often need to modify how much damage is dealt depending on a variety of factors such as element weaknesses. Similarly, games may need to respond to damage dealt such as by providing a visual indication of an entity taking damage.
This tutorial shows how to assign events on IDamageable entities to modify damage dealt and to visually indicate that damage has been dealt.
Entities which implement the IDamageable interface are required to have an event named ReactToDamageDealt. This is automatically generated by the FlatRedBall Editor, so the event is available in custom code. This can be used to add custom code to handle an entity receiving damage.
Keep in mind this event occurs after damage checks have been made. It will only be raised if the IDamageable entity has a different team index than the argument IDamageArea, and before the IDamageable is killed (if enough damage is dealt to kill the IDamageable). For example, we can make the enemy flash when it takes damage. First, we'll change the enemy's AxisAlignedRectangleInstance color to red in the FlatRedBall Editor.
Next, we can handle the event in code. In this case we'll use an async
call to set the color to white temporarily, then set it back to red.
Keep in mind thThe code above is written as a quick example of how to handle flashing an enemy to white whenever it takes damage. This code can result in the enemy remaining white if it takes multiple hits in the 0.1 second window, but it is provided here as a quick example to show how to handle damage received events.
Now the Enemy flashes white when taking damage. Note that to test this you may want to disable the enemy shooting logic added in the previous tutorial. Also, you may want to adjust the health of the enemy to take more hits.
The purpose of ReactToDamageDealt is to react to damage visually, or to perform other game logic outside of typical IDamageable logic. Therefore, ReactToDamageDealt should not:
Kill the damage receiver unless there are special circumstances
Deal more damage to the damage receiver
Destroy the IDamageArea (damage dealer) unless there are special circumstances
The ModifyDamageDealt event can be used to perform logic that can change how much damage an entity receives. For example, an enemy may be weak to a particular type of bullet such as an elemental bullet. For this example, we will add an IsFireBullet property to the Bullet, and an IsIceEnemy property to the Enemy. Note that a real game may handle these variants through entity inheritance, but we will add properties through code to keep the tutorial shorter. Add the following code to Bullet.cs:
Next, add the following code to Enemy.cs:
Now we'll modify Player.cs to create fire bullets when pressing left Alt key:
Next, we'll modify the code that creates Enemies in GameScreen.cs to create ice enemies on a right-click:
Now we can create fire enemies and ice bullets by right clicking the mouse and pressing the left Alt key, respectively.
Now that we have two different types of enemies and two different types of bullets, we can write logic in the Enemy's ModifyDamageDealt event in response to the elements:
Now ice enemies die after 5 hits instead of 10 since each bullet will do an effective 20 points of damage.
This tutorial discusses two events:
IDamageable.ReactToDamageReceived
IDamageable.ModifyDamageReceived
The IDamageable and IDamageArea entities provide additional events which can be used to handle damage modification, reaction to taking damage, and death. This is the full list of events provided by each interface:
IDamageable.ModifyDamageReceived - Allows an IDamageable to modify how much damage it takes
IDamageable.ReactToDamageReceived - Allows an IDamageable to respond to taking damage
IDamageable.Died - Allows an IDamageable to respond to dying, such as by playing a sound effect or creating particles
IDamageArea.ModifyDamageDealt - Allows an IDamageArea to modify how much damage it deals
IDamageArea.ReactToDamageDealt - Allows an IDamageArea to respond to dealing damage
IDamageArea.KilledDamageable - Allows an IDamageArea to respond to killing an IDamageable
IDamageArea.RemovedByCollision - Allows an IDamageArea to respond to being removed by collision
This tutorial has used both of the terms *delegate* and *event*, primarily using event to indicate that these are methods which are invoked in response to something happening in the game such as the enemy taking damage. Before moving on to a new topic we should cover some of the technical details of how these delegates work.
Technically, these are delegates and not events. In other words, the IDamageable and IDamageArea interfaces do not define these using the event
keyword. Therefore, these can be invoked externally, allowing the DamageableExtensionMethods.TakeDamage
methods to call these methods as necessary.
If you are familiar with event syntax in C#, then you are probably used to the +=
operator when assigning a handler. This approach works for the delegates defined on the IDamageable and IDamageArea interfaces, but we should take a moment to mention a subtlety related to the ModifyDamageDealt
and ModifyDamageReceived
delegates. These delegates use the Func
type, which means they return the modified damage value. If you use the +=
operator for these two delegates, you may end up dealing or receiving an unexpected amount of damage. The reason for this is because only the last handler added to these two delegates decides how much damage to deal. To understand why this might happen, let's look at an example.
Consider a game with an Enemy entity which has multiple variants (using inheritance). You may have an IsInvulnerable
property on the Enemy, resulting in the following code for handling damage:
You may also decide to have additional logic for an enemy variant which can hide in its shell. If the enemy is in its shell, it would receive half damage. The implementation may look like this:
In this situation, the base CustomInitialize
is called first, then the derived CustomInitialize
. This means that whenever the enemy receives damage, the base HandleDamageReceived
is called first, then the derived HandleDamageReceived
. Unfortunately, when HandleDamageReceived
is called on the derived, the initialDamage
is not modified by the first call. In other words, the initialDamage
will never be affected by the check for IsInvulnerable
because each method returns its value independent of the other. In effect this results in the TurtleEnemy never respecting its IsInvulnerable variable.
Therefore, it's best to never += on any of the Func
delegates - ModifyDamageDealt
and ModifyDamageReceived.
Instead only the =
operator should be used for clarity. In this case, we can resolve the problem by creating a virtual method for handling damage which derived classes can override, as shown in the following code:
Now the derived variant can override the method as shown in the following code:
This tutorial discusses how to handle events to modify damage and display visual feedback to the player when damage is dealt. The next tutorial covers the removal of IDamageAreas and how to implement damage over time.
Before we begin modifying the control values, we'll change the resolution of our game to match the original .
FlatRedBall.Forms is a library which provides a full set of UI functionality similar to other UI libraries like Windows Forms, Xamarin.Forms, and WPF.
FlatRedBall.Forms is intended to mimic the syntax of WPF while allowing you to customize the visuals and layout of your UI using the Gum UI tool. Properly using FlatRedBall.Forms requires at least a basic understanding of working with Gum. Developers who are not familiar with Gum should first visit the Gum tutorials, which can be found here: https://docs.flatredball.com/flatredball/gum/tutorials
This tutorial shows you how to access FlatRedBall.Forms controls in code to perform common logic like assigning event handlers and accessing properties.
The previous tutorial showed how to create a new project with a Button control. Most controls require no additional code to work - just drag+drop the control into a screen and it will have full functionality. For this tutorial we'll create an instance of the following controls:
Button (multiple versions exist, but ButtonStandard is the most common)
CheckBox
ListBox
TextBox
All controls are present in the Components/Controls folder.
To create these controls, drag drop them into your current screen, such as GameScreenGum or MainMenuGum. If you have a GameScreen, but would like to create a screen with only UI, you can add a new MainMenu screen to your FlatRedBall project.
You can drag+drop controls into your screen. The following screenshot shows a Gum screen with four controls:
Now we can access all of our controls from the Gum screen in code. Every Gum object can be accessed in its respective screen using the Forms
object. For example, to add a click event to the button:
Open the project in Visual Studio
Go to your screen's code file (GameScreen.cs or MainMenu.cs, for example)
Add the following code to the GameScreen:
This results in the button updating its text to indicate when it was last clicked:
Let's take a look at a few parts of the code. First, we should note that the code above is all compile-time protected. This means:
Any changes in the Gum project may result in a compile error, notifying you that something has to change in the code. For example, we are accessing the ButtonStandardInstance. If this object is removed or renamed, your code will not compile.
Visual Studio provides intellisense (code completion) to help you fill in the code
In the code above we accessed the button through the Forms.ButtonStandardInstance property. Note that ButtonStandardInstance is the same name as the object in Gum.
Notice that we are accessing the object through Forms allows us to interact with the Gum object casted as a FlatRedBall.Forms Button. Once we have access to the button we can interact with it in a standard way, such as by assigning a click event or by setting its Text property.
The CheckBox control can be used to allow the user to set true/false values. Just like with Button , before we'll get a reference to the CheckBox through the screen.
Clicking the CheckBox results in the value being printed to the screen.
The ListBox control is used to display options or a collection of current items to the user (such as active quests). The following code adds items to the list box whenever the user presses a key on the keyboard. Note that this code requires code in CustomActivity to read input from the keyboard.
Typing the A, B, or C characters on the keyboard results in items added to the list box:
The TextBox control provides free-form input support for users to enter string values like a character's name. Note that at the time of this writing the TextBox relies on the MonoGame key bindings which do not consider different keyboard configurations (such as languages other than English). The following code can be used to react to any changes in the TextBox Text property and print it out to the screen.
Note that the TextChanged event will be raised for each new character (including spaces) or whenever a character is deleted.
Most games include multiple groups of entities (which we'll refer to as variants) such as enemies, bullets, and pickups. Variants usually share common characteristics such as:
Reactions to collision, such as always dealing damage to the player
Creation logic, such as being created by the player when shooting or by an enemy spawner
Interfaces and code such as implementing ICollidable or being a top-down entity
Of course, variants can also differ. Common examples of how variants may differ include:
Visuals, such as different .achx files (Animation Chains)
Coefficients and variables such as movement speed or max HP
Logic, such as movement patterns or navigation behavior
Collision shapes and sizes
Conceptually variants can differ in any number of ways - the lists above are just common examples.
The current recommended approach for creating entity variants (as of the time of this writing in February 2024) is to use entity inheritance. Conceptually speaking you can achieve variants through States, loading custom data files, or any other approach. However, inheritance is very well supported in FlatRedBall. When using inheritance, the FRB Editor provides code generation which simplifies common operations including creating new instances, accessing derived type files, and reading coefficient values.
This and other Tutorials refer to the "Variants" (capitalized) as a class generated by the FRB. The term variant (lower-case) refers to the concept of creating multiple types of entities which differ as explained above.
This documentation discusses Entity variants primarily; however, the concepts and syntax discussed here also apply to Screens. In other words, the FRB Editor generates Variant classes in both Entity and Screen generated code, so variants can be used to access information about derived Screens such as Level screens.
FlatRedBall entities provide support for inheritance. By default, all entities inherit from the PositionedObject class, but inheritance can be changed to that entities can inherit from other entities. When working with entities with multiple variants, it's best to create a base entity such as Enemy, then create derived entities from the base.
For example, a game project may include an entity called Enemy which serves as the base for all other enemies.
Derived entities might include the different types of enemies that exist in the game. Note that the Enemy entity may never be directly instantiated in the game. It only serves to include all common code, files, and objects for all other enemies. This is similar to the GameScreen screen and the derived Level screens.
When creating a variant of the Enemy entity, the Enemy entity type should be selected as the base type.
It may be helpful to organize your derived variants in a subfolder. For example, the following enemy variants are all stored in an Enemies folder.
Inheritance can also be set on an Entity after it has been created by selecting the entity and changing its BaseEntity type in the Properties tab.
Notice that when a derived Entity is created, FlatRedBall does not create a list for that derived type. In most cases you will not need a separate list for each variant - you can use the list for the base type for collision and game logic. In this case, the GameScreen includes an EnemyList, but does not include lists for the specific enemy types.
As mentioned earlier, entity variants can differ in many ways. One of the most common is through the visuals. The most common way to vary the visuals for each type is to create a Sprite in the base entity, but to assign different animation chains on the derived types.
For example, the Enemy entity includes a Sprite instance, but the Sprite does not have its AnimationChains set.
Each derived entity needs to access the SpriteInstance defined in the base entity to modify its appearance. To do this, we can expose SpriteInstance in the base entity. This can be done by selecting the Sprite in the base entity and setting its ExposedInDerived property to true.
By setting this value to true, the Sprite is accessible in all of the derived entities.
Now we can assign the animation chain on each of the derived SpriteInstances. You can do this by selecting the Sprite and changing its AnimationChains dropdown.
Alternatively, you can drag+drop the .achx file onto the Sprite. This is a shortcut to setting the Animation Chains through the dropdown.
Regardless of which approach you use, be sure to also set the Current Chain Name to a valid animation.
If you do not have any animations in the Current Chain Name drop-down, you need to first add new animations to your .achx.
Typically, each AnimationChainListFile would have animations which are named the same. This allows code to set the animation for every enemy variant using the same logic. For example, you might have animations for Idle, Walk, Attack, and Die. These may also include different directions (such as Left, Right, Up, and Down).
Once you have created derived entities, you can add them to your levels. The easiest way is to use the Live Edit system to drag+drop them into your game.
Notice that every instance is added to EnemyList. FlatRedBall understands that the derived entities should be organized in the base entity list. This makes handling collisions much easier than if each type had its own list.
Entity instances can also be created in code by using their respective factories. When a derived entity's factory is used, the newly-created instance is automatically added to the base list. Derived entities end up in their base list whether created through the FRB Editor or in custom code.
The following code shows how to create an entity instance whenever the user clicks the cursor (mouse) in GameScreen.cs.
Games often need to refer variants in code or in the FRB Editor. For example, an EnemySpawner entity may need to define which Enemy variant to spawn. FlatRedBall provides a special type for all derived entities which can be used in the FRB Editor and code which is called a Variant.
For this example, consider an Entity called EnemySpawner. Each EnemySpawner instance may need to indicate which Enemy variant it should spawn. To support this, we can add a new variable of type EnemyVariant to the EnemySpawner.
In this case, the Entities.EnemyVariant appears as a possible variable type because the Enemy has at least one derived entity. If you create more entities which have derived types, then they will also appear in this list.
After adding this variable, instances of the EnemySpawner can be assigned an enemy type.
This variable can be used in code to create instances of your variant. For example, the following code could be added to EnemySpawner.cs to create its enemy variant whenever the space bar is pressed.
This code uses the space bar for the sake of simplicity, but a real implementation may spawn enemies based on a timer, player proximity, or other game logic.
Variants, such as enemies, may have different variables which may need to be accessed without the creation of an instance. For example, your game may include a bestiary of enemies that the player has encountered. This may be shown to the player using a Gum UI which displays information about the enemy such as its MaxHealth.
Variables such as MaxHealth can be defined per enemy type. For example, the following image shows a Skeleton which has 80 MaxHealth.
This can be accessed in a type-safe way in code using the EnemyVariant class as shown in the following snippet:
Similarly, the information is accessible through an EnemyVariant variable which might be assigned in the FRB Editor. For example, the EnemySpawner would also be able to access the MaxHealth of the enemy variant that it spawns.
EnemyVariant can also be accessed through their name. For example, the following code would get the MaxHealth of a Skeleton:
At times your game may need to access files from a derived entity. For example, if we consider the bestiary example from above, you may want to display an image of your enemy in Gum UI. To do this, you would need to access the AnimationChain from the entity. Variants provides access to file loading. For example, we can access a Skeleton's AnimationChainListFile using the following code:
Your game may need to serialize data about derived types. For example, the player may equip different weapon variants which should be saved in the profile. The Name property on a derived Variant can be used to uniquely identify the type as shown in the following code.
The Variant class includes a static list named All
which provides access to all Entity variants. The All list can be useful in a number of situations including:
Listing all Level screens (Screens inheriting from GameScreen) in a LevelSelect screen or in a debug menu
Displaying all enemies in a bestiary
Selecting a random entity for instantiation - either the entire set or subset
The following code shows how to fill a ListBox with available Screen names which could be used in level selection UI:
Variants are created automatically by FlatRedBall in the base entity's codegen. These entities are hard-coded and will be compiled into your game, but you are not limited to using just these variant types. Instead, you can create new variants in custom code.
For example, consider an entity type called Enemy. A new variant can be created in code using the following code:
This variant can be used to instantiate enemies with the assigned values as shown in the following code:
Since this new enemy type is not created in the FlatRedBall Editor, its C# type is Enemy (the base Entity type). Note that custom variants can only be created if the base class can be instantiated through a factory. That means that the base class must have the following:
Each derived entity would typically include its own AnimationChains. For simplicity, the file can be named the same in all cases (AnimationChainListFile). While this may seem less expressive compared to having different names for each file, keeping the name the same can make your code much simpler if you need to access the file. To learn more about creating .achx files, see the documentation.
Its must be set to true
It must not be abstract - it must not have any objects that are
Data binding is the process of controlling the appearance and behavior of UI through a separate object usually called a view model. The data binding pattern used by FlatRedBall.Forms is referred to as Model View ViewModel, abbreviated as MVVM. Developers familiar with data binding in WPF, Xamarin.Forms, or Knockout JS will find data binding in FlatRedBall.Forms familiar.
FlatRedBall.Forms is an easy-to-use UI library for games. This tutorial will walk you through setup of a game project including FlatRedBall.Forms.
First we'll create a Glue project. If you already have a Glue project you can skip this section. To create a new project:
Open Glue
Select File -> New Project
Enter your project name and click Create Project!
On the wizard, pick Standard Platformer, Standard Top Down, or Ui - any of those three will add FlatRedBall.Forms to your project.
Now we have a project that is ready to go. Note that if you already have an existing FlatRedBall project, you can continue on the next step.
If your project does not have Gum (such as if you skipped the wizard), you can it at any point. To do so:
Click the Gum icon in Glue to add a new Gum project
Note that if you already have a Gum project, this icon will open the file in Gum
Click the option to include forms
If you either created your project through the wizard, then all FlatRedBall Screens have an associated Gum screen. In other words, you can start adding FlatRedBall.Forms to any existing screen. You can verify that your FlatRedBall Screens have an associated Gum screen by checking the Files folder and finding a file with the .gusx extension. For example, the GameScreen has an associated Gum screen in the following image.
If you open Gum, it will have a GameScreenGum screen.
To add a Button to your GameScreen:
Expand the Components -> Controls folder
Drag+drop the ButtonStandard component onto the GameScreenGum or MainMenuGum (depending on which project type you created)
The Button should now appear in the Gum.
Running the game will also show a fully-functional button.
If you have used Gum before (or if you have read the Gum tutorials), you may be wondering about the difference between Gum and Forms. A deep dive into this topic would be too long for this tutorial, but we will briefly look compare Gum and Forms using the project we just created earlier in this tutorial. As mentioned earlier, FlatRedBall.Forms is a UI library which provides common UI logic. For example, notice that our button instance automatically reacts to a moue hover and click. We didn't have to write any code to tell it to change its appearance. Technically FlatRedBall.Forms is a collection of classes which can modify Gum objects in response to a variety of interactions such as mouse hover, keyboard actions, an clicks. Gum components may or may not be represented as Forms objects at runtime depending on their behaviors as defined in Gum. All Forms objects have a backing Gum object, which is stored in their Visual property. By contrast, not all Gum objects are wrapped by Forms objects.
To help understand the difference, let's take a look at the Gum button components. The default Gum project contains multiple Controls which begin with the word "Button".
If any of these components are added as instances to a screen, they will be represented by the Button forms type. We can see this by adding an instance of each to our Screen and running the game. For example, consider the following screen:
Each instance in this screen is a different component type: ButtonClose, ButtonConfirm, and so on. However, if we access these buttons at runtime through the Forms property, they are all of type Button.
We can even go to the definition of our Forms object and see that all Buttons are of the same type.
Not all Gum components are represented as Forms controls. For example, the default Gum project contains a component named Icon which can be used to display one of a collection of standard icons in games.
If we add an instance of our Icon to GameScreen, we can see that the Forms object does not contain a property for IconInstance.
Note that Visual Studio's auto complete does not provide a suggestion for IconInstance (but it does provide a suggestion for ButtonIconInstance, which is an instance of a Button).
We can still access the icon through the GumScreen property, which provides typed access to all Gum instances whether they are forms or not.
Ultimately, FlatRedBall decides whether a component should be in Forms or not depending on the Behaviors that the component implements. For example, all of our Button components implement the Button behavior.
The presence of this behavior is all that is needed to mark a component as a Forms object. You can browse other components such as TextBox and ListBox to see the behaviors they implement as well. If you would like to create custom components which should be treated as Forms objects, you can add these behaviors to your controls as well.
The previous tutorials showed how to deal damage to an enemy with a Bullet. The Bullet entity implemented only the IDamageArea interface (not the IDamageable), so it served purely as an entity which could deal damage but could not receive damage.
Games which implement melee damage can do so by adding additional shapes to the Player entity and by marking the Player as an IDamageArea (as well as IDamageable). This tutorial shows how to modify your project to allow the Player to deal damage to an Enemy using a melee attack collision.
Melee attacks increase the complexity of games because they result in entities that can both deal and receive damage. We need both the Player and Enemy to be able to deal damage.
First, mark the Player entity as both IDamageable and IDamageArea.
Repeat the same for the Enemy entity.
Next, we'll add a collision object to the Player for melee attacks. This can be any type of shape, but we'll use an AxisAlignedRectangle for this tutorial. Be sure to give the new shape a name that clearly explains its purpose, such as MeleeCollision.
Adjust the MeleeCollision so that it is positioned outside of the Player entity. You can use Live Edit to iterate quickly on this.
By default, this new shape is used in all relationships which use the Player's <Entire Object>, which is the default for all collisions. For example, the PlayerVsSolidCollision in GameScreen uses this option, so the newly-created MeleeCollision collides with the map's SolidCollision.
Typically, collision shapes which are added for a special purpose (such as only dealing damage to Enemies) should not be used for all collision relationships. We can fix this by setting the MeleeCollision's IncludeInICollidable to false.
Now the MeleeCollision will only be considered in collision relationships which explicitly select it in the Subcollision dropdown (which we'll do later in this tutorial).
Next, we will create collision relationships between players and enemies. Conceptually, we want the following behavior:
When the body of the enemy touches the body of the player, the player should take damage
When the sword of the player touches the body of the enemy, the enemy should take damage
Each entity type can deal damage to each other entity type, and the direction of damage dealing depends on which shapes have collided.
We need to have a collision relationship between players and enemies, so we can add this by drag+dropping the PlayerList onto the EnemyList in GameScreen.
Notice that the newly-created collision relationship has the option for dealing damage checked, and that both the player and enemy will receive damage.
This collision relationship should only deal damage to the player if the player collides with the enemy. This collision relationship should not deal any damage to the enemy.
We will handle the melee collision (Player dealing damage to Enemy) in a different collision relationship that has its Subcollision set to MeleeCollision. Since we want this collision relationship to deal damage one way, we need to handle dealing damage in code. First, uncheck the option to deal damage in the collision relationship, and click the button to add a new event.
By checking the Add Event button, the FRB editor adds an event in GameScreen.Event.cs which we can fill in with our collision logic. To deal damage from the enemy to the player, add the following code:
Now the player takes damage when colliding with the enemy.
Note that the player dies very quickly after touching the enemy. This happens because the player takes damage every frame (10 damage), so after about 10 frames (1/6th of a second) the player's health reaches 0.
This is typically not desirable in games, so we can use damage dealing frequency and invulnerability to solve this problem. Specifically, we have two variables which can be used to solve this problem:
Enemy SecondsBetweenDamage (how frequently the damage is dealt to the player)
Player InvulnerabilityTimeAfterDamage (how frequently the player can receive damage)
While these two may seem similar at first, which you use can have an impact on your game. Most older games (such as Super Mario World, Zelda: Link to the Past, Mega Man X, and Super Metroid) implemented invulnerability after being dealt damage. This means that if the player took damage from any damage source, the player would remain invulnerable for a period of time after the damage was dealt. This approach can give the player a moment to recover after taking damage, and can avoid situations where the player is "sandwiched" between multiple damage sources, resulting in health draining rapidly.
By contrast, some games (such as Mega Man X) do not have invulnerability time on regular (non-boss) enemies, allowing players to mash very quickly to deal large amounts of damage. Keep in mind that invulnerability periods also prevent the use of the damage system for gradual damage, such as poison which deals damage every frame.
For this tutorial we will implement invulnerability time for the Player and Enemy, but your game may ultimately require mixing these approaches. More complicated games may also implement their own damage suppression techniques without using the built-in invulnerability and attack frequency values.
To add invulnerability time to the player, select the Player and change the Invulnerability Time After Damage variable.
Now, the player can only receive damage one time every second. We can see this is the case because the player survives much longer (about 10 seconds) when overlapping the enemy.
Now that our Player can take damage, we can deal damage to enemies using a similar approach. At a high level the approach is:
Create a collision relationship for the Player vs Enemy, setting Subcollision to the Player's MeleeCollision
Remove the default (code generated) damage dealing logic on the collision relationship
Add an event
Implement damage dealing in the event
To create another collision relationship for player vs enemy:
Drag + drop PlayerList onto Enemy list in GameScreen again
Set the Player's Subcollision to MeleeCollision so only the MeleeCollision is considered
Uncheck the damage dealing check box
Click the Add Event button
Now we can modify the damage dealing code in GameScreen.Event.cs:
Notice this code is similar to the collision code used to deal damage to the Player.
You may also want to set the Enemy's Invulnerability Time After Damage to some non-zero value. Keep in mind that doing so will affect the invulnerability time of the enemy from all damage. If you worked through this tutorial by continuing work from previous tutorials, then this code will change the behavior of how enemies take damage.
Notice that the invulnerability time is lower for enemies than for players. You can tune this value to get the right feel for your game.
As implemented now, the player's melee collision deals damage continually to enemies so long as it continues to collide with the enemy when the enemy's invulnerability time expires. Most games with melee attacks require a button to be pushed to perform the attack. When the button is pushed, the attack is only active for a set amount of time, then the attack goes into cooldown.
We can implement this in code by keeping track of when attacks last occurred. Although this has no impact on collision, we can also change the visibility of the weapon. First we will define some variables in code in Player.cs:
Note that the variables AttackDamageDuration
and AttackCooldown
are defined in code. In a full game these variables should be defined in the FRB Editor, but we are defining them in code for the sake of brevity.
Next, we can perform attacks and modify the visibility of the melee shape in CustomActivity by adding the following code:
Please note the property DirectionFacing
and enum TopDownDirection will not be available if you are working in a custom, blank project created without using a platformer wizard. In that case you will need to write your own direction detection logic. As an example, assuming MovementActivity method is called in CustomActivity for every frame update and MovementInput has been instantiated in CustomInitialize in Player.cs, it could look like this:
Keep in mind the visibility logic exists only for the sake of visualizing when damage can be dealt - we cannot use visibility to control whether collision occurs because invisible shapes collide just like visible shapes. However, we can use the IsAttackActive
property in the collision event in GameScreen.Event.cs for dealing damage to the enemy, as shown in the following modified code:
By using the IsAttackActive property in the OnPlayerVsEnemyCollided, we could suppress the dealing of damage unless the player is actively attacking. This approach is effective in this simple case, but more complicated games may have multiple types of objects which can receive damage from a player. For example, the game may include destructible obstacles, or it may even support PvP. Therefore, we can use the ModifyDamageDealt event to prevent dealing damage against any type of object unless the attack is happening. We can do this by modifying the Player.cs file as shown in the following code:
Note that this method can be used to modify damage in other scenarios. For example, you may have different attacks (strong vs weak), or multiple attacks in a combo. You are free to define variables in your Player file to support attacks of any complexity. You can then process these variables in ModifyDamageDealt to vary the damage dealt to enemies.
To properly test this approach in our particular example, you might want to remove the IsAttackActive check from the OnPlayerMeleeCollisionVsEnemyCollided event handler to allow all the attacks through (remember the melee collisions are still happening whether the melee rectangle is visible or not) and add the following line to the beginning of HandleDamageReceived delegate handler in Enemy.cs (prevents false flashing when damage is supressed by the Player's damage modification in HandleModifyDamageDealt):
FlatRedBall Forms has full support for control with a mouse and keyboard as well as an Xbox360GamePad. This document discusses how to enable this behavior and details about its behavior.
FlatRedBall.Forms controls have built-in support for gamepad navigation. Once gamepad input is enabled for Forms objects, the user can move between controls and perform common UI operations. For example, consider the following layout using default Forms elements:
The layout above contains the following interactive Forms elements:
Button
CheckBox
ComboBox
ListBox
RadioButton
Slider
TextBox
To enable gamepad interaction with these elements:
Add the which gamepads can control focus and selection using the GuiManager.GamePadsForUiControl list as shown in the following snipet:
This code can be added anywhere, and gamepads can be added and removed to allow only certain gamepads to control the UI.
Select which control should begin with focus. For example, the code above sets the button as having initial focus: Forms.ButtonInstance.IsFocused = true;
Once one of the controls on a screen has focus, the gamepad can interact with it. For example, the following images and animations show interaction which can be performed with a gamepad including clicking a button, selecting items in a list box, and moving the slider. Once the ButtonInstance IsFocused is set to true, the button appears focused with a white rectangle.
Navigation between elements can be done by pressing up, down, left, or right on the analog stick or d-pad. The A button selects the control:
The tab order of the items matches the order of the items in Gum. The layout has the following tab order, as shown in Gum:
ListBoxes can have two focused states:
Top-level focus (the entire list box has focus)
Item focus - the items within the list box have focus
When the list box has top-level focus, tabbing between items selects the next sibling item. For example, the following image shows the ListBox on the top left of the screen having focus. Pressing "down" to tab to the next control results in the ComboBox (with the word "Impossible") gaining focus.
If a ListBox has item focus ( its DoListItemsHaveFocus
property is set to true), then tabbing up and down selects the next or previous items in the list, as shown in the following animation:
The DoListItemsHaveFocus property can be set to true in the following ways:
By explicitly setting the value in code, either directly or through data binding.
By giving the ListBox top-level focus, then pressing the confirm button on the gamepad. This is similar to pressing the confirm button on a ComboBox.
This two-layer selection mode is useful if the list box is part of a more complex screen and you would like the player to be able to navigate between controls. By contrast, if your screen contains a list box which must be selected before proceeding (such as a level select screen), then you may want to give the ListBox focus and also set its DoListItemsHaveFocus
to true on screen CustomInitialize.
Note that if an item is selected with the gamepad's confirm button, then the item appears selected and DoListItemsHaveFocus
is set to false, giving the list box top-level focus.
FlatRedBall.Forms includes a control called OnScreenKeyboard which can be used to enter text with a gamepad into a TextBox. For information on how to use this control, see the OnScreenKeyboard page.
Slider controls can be focused with the gamepad. When a Slider has focus, pressing left or right on the analog stick or d-pad moves the slider value by its SmallChange property. For more information, see the Slider page.
By default any control which implements IInputReceiver can be focused through tabbing. To skip a control, set its GamepadTabbingFocusBehavior to TabbingFocusBehavior.SkipOnTab. For example, the following code would mark a TextBox as not being selected through gamepad tabbing:
FlatRedBall Forms is a set of classes which are used to give UI controls automatic behavior. When using FlatRedBall Forms, your code has access to the FlatRedBall Forms object (such as Button) as well as the Visual for the Forms object (such as the component DefaultForms/Button). Since your code has access to both the Forms and Gum object, some confusion may arise about which object to interact with in code.
This guide discusses the relationship between the Forms and Gum objects, and provides some guidelines to determine whether to interact with one of the other.
The most important concept here is - some components in your project are Forms types, and some are not. Only Forms-implementing component instances appear in the Forms property, but all instances appear in the GumScreen property.
To understand how Forms and Gum objects interact, we will consider a simple example - a Screen with a single Button. The following image shows a default Button instance in a Gum screen:
In this example, the MenuScreenGum is loaded by the FlatRedBall screen MenuScreen.
We can access both the Gum and Forms objects in Visual Studio as shown in the following code. Keep in mind that every Screen with Gum and Forms objects will have GumScreen and Forms properties. These make it easy to access objects the same way no matter which Screen you are working on:
Notice that setting width and height modifies the same object at runtime - it appears as a square. Also, filling in the two click handlers would result in both handlers being called when the button is clicked.
You may be wondering - which object should I access in code? The answer is - usually it's best to use the Forms object. If you end up needing to make changes to the Gum object (such as to change its position or size), it's worth considering whether this change should actually be done in code. Most of the time these changes should be performed in the Gum tool.
FlatRedBall.Forms objects are wrappers around Gum objects. For example, in the example above we have a Forms.ButtonInstance. This object has a reference to the Gum object. The Forms object automates behavior so that the Gum object behaves like a UI element. For example, the Forms Button object automatically modifies the state of the button in response to cursor hovers and clicks. No custom code is necessary to achieve this behavior.
This behavior is convenient, but it is not particularly complex - at least not conceptually. The Forms object is responsible for detecting if the cursor is hovering over the button, or if the left mouse button is pressed. If so, the Forms object modifies the state of the button. These states are part of the default Button control, and can be inspected (and modified) in Gum.
The Forms Button logic is responsible for setting the button's state, and it will do so in response to any UI behavior. Therefore, manually setting the Gum object's state is not recommended - it will be over written by the Forms object as shown in the following code and animation:
Notice that the state is not immediately changed, but rather only when the mouse moves over the button. Once the mouse hovers over the button, its state changes immediatelz. After the mouse leaves the button, the state is reverted to Enabled. Setting the state through the Gum object can produce confusion. For example, a button may appear pushed or disabled if the state is manually assigned; however, the actual behavior of the button will not match a disabled state. Therefore, the button's state (specifically the CurrentButtonCategoryState) is controlled by the Forms Button object. The Forms Button object will modify the state according to standard button behavior, so custom code should not modify the Gum button's state.
The Forms property in FlatRedBall Screens contains properties for all Forms objects in the Gum screen, and only the Forms objects. Therefore, any object that is part of the Gum screen which is not a Forms type (such as Button or TextBox) does not appear in the Forms property.
For example, consider the following screen in Gum:
The left column consists of Forms controls: a Button, ListBox, and TextBox. The right column consists of instances of Gum objects which do not implement any Forms controls: a Text, ColoredRectangle, and Container.
In code, the Forms property only contains references to the Gum-implementing instances. Therefore, the following code is valid:
However, the following is not valid code because these instances are not Forms-implementing types:
To access the non-forms objects, you must use the GameScreen object. For example, the following code is valid:
Keep in mind that an instance in a Gum screen will always be a Gum object but only sometimes be a Forms object. This means that any instance that is a Forms object can also be accessed through GumScreen as shown in the following code:
As mentioned above, accessing Forms objects through GumScreen is generally not good practice.
FlatRedBall Forms contains many types of controls each with their own type of automatic behavior. The following list is an example (not a complete list) of properties which are controlled by Forms behaviors, and which should not be modified on the Gum object:
Button ButtonCategoryState controlled by cursor and Button.IsEnabled
RadioButton RadioButtonCategoryState controlled by RadioButton.IsChecked
ComboBox ListBoxInstance Visible controlled by ComboBox.IsDropDownOpen
ListBox and ScrollViewer VerticalScrollBarInstance Visible controlled by ScrollViewer.VerticalScrollBarVisibility and the number of items in the list
Slider ThumbInstance X controlled by Slider.Value, Slider.Minimum, and Slider.Maximum
TextBox.CaretInstance X and Y controlled by TextBox.CareIndex and TextBox.TextWrapping
As mentioned above, this is not a complete list. In general, if you plan on modifying a Forms object in code, you should first look to the Forms object to see if it provides a way to modify the object as desired. If your custom code modifies the Gum properties mentioned above, these changes will likely be overwritten by automatic Forms behavior.
FlatRedBall Forms also provides a way to standardize UI programming. When working against Gum objects, the types of the Gum objects depend on the project. For example, one project may use the default Button object for its Button, whereas another project may create a customized button called StyledButton which contains a different set of values. In both cases (assuming the behaviors have been set up correctly in Gum), both objects will produce a Forms Button, and programming against both will be identical. This makes it easier to learn the Forms syntax once and code the same everywhere.
The Forms objects standardize syntax, but they do not provide the full control that can be provided by access to the Gum objects. For example, consider a game which lets the player choose which save slot to load. Each save slot may be represented by a Button, and the Text on the button may display the name of the character. However, the game may also include additional information on each save slot, such as the level of the character, or the percentage completed in the game. These properties are not part of the Button class because they are custom to a particular game. In this case, the code may need to access the Gum object to modify the CharacterLevelText or PercentageCompletedText objects. Forms objects also provide a subset of the standard Gum layout variables. The following lists the current set of layout variables available to every Forms control:
X
Y
Width
Height
Games which need to modify the layout in code may benefit from access to the full set of Gum layout variables:
X
XUnits
XOrigin
Y
YUnits
YOrigin
Width
WidthUnits
Height
HeightUnits
To access the full set (units and origin values), the Gum object must be accessed.
The BindingContext property is used for all data binding. This tutorial introduces the BindingContext property and creates a simple example using BindingContext to control the UI.
For this tutorial we will create a Gum screen which contains the following:
Points Display
Health Bar
Button for awarding points
Button for taking damage
Button for healing
Normally the actions performed by the buttons listed above would occur through regular game logic (such as collision) but we will use buttons for the sake of simplicity. The specifics of the visuals do not matter, so your screen may look like the following:
A few details in the screen above are important:
The three buttons are FlatRedBall.Forms Button objects. In other words, the three buttons are created in Gum using components which implement the Button behavior. The most common of these is ButtonStandard.
The HealthBar (ColoredRectangle) is contained within a HealthContainer (Container) using the Percent Width Units
Each object in the screen is descriptively named. This will make data binding easier.
The screen is called GameScreenGum. This is the default Gum Screen name for a FlatRedBall Screen called GameScreen. If you have created a game with a GameScreen using the wizard, you will also have a GameScreenGum
The first step in binding to your UI is to create a ViewModel object. ViewModel objects contain all of the properties which you would want to control on your UI programmatically. For the screen above, we will control the following properties:
The string displayed by the PointsTextInstance
The string displayed by the HealthTextInstance
The width of the HealthBar
The ViewModel class should implement the INotifyPropertyChanged interface. Fortunately, FlatRedBall provides a convenient class for this. View models are plain C# classes that are created in Visual Studio. We recommend creating a ViewModels folder in your project, but you can organize them however you like. The following code is an example of what the ViewModel class might look like:
The base ViewModel class provides common functionality for MVVM implementation - specifically the Get and Set functions and the DependsOn attribute. Using the ViewModel base class is not required, and any INotifyPropertyChanged implementation will work.
The Get and Set functions provide a quick way to implement notification to the UI. When the Set function is called (such as when Score is assigned), internally the Score value will be stored, and any object watching for changes to the Score variable will be notified. Furthermore, any object watching properties which depend on Score will also be notified.
The DependsOn attribute creates a dependency relationship between the property which has the attribute (such as ScoreDisplay) and the property which it depends on (such as Score). Once this dependency is established, changing the Score property will also notify the UI that ScoreDisplay has changed. Notice that a single property can depend on multiple properties, as is the case of HealthDisplay depending on both CurrentHealth and MaxHealth.
The ViewModel establishes which properties can be assigned, and the dependency between properties. Once a ViewModel is created, it can be applied to a Gum object. Note that ViewModels can be applied to Gum objects (GraphicalUiElements) or Forms objects (such as Button). BindingContext assignments cascade - you only need to assign the BindingContext at the top level and all children will recursively receive the same BindingContext.
Typically the BindingContext is done at the root level. We recommend assigning the Forms object which is available if you are using Forms in your game. Once the BindingContext is assigned, each individual UI property needs to be bound to the corresponding ViewModel property. The following code shows how this type of binding would be done:
Note that the above code assigns the binding on Gum objects through GumScreen. Although we're not doing it in this tutorial, we could also call SetBinding on Forms objects. For example, if the ViewModel had a property which was to be used as the Text on one of the buttons, the following code could be used:
This topic is covered in more detail in the next tutorial.
The ViewModel object is defined at GameScreen class scope. This is required so that the game can make modifications to the ViewModel after the screen is initialized. FlatRedBall recommends that the ViewModel property is named ViewModel to standardize code across different pages and views.
Once the ViewModel is created, we assign the BindingContext. This assignment tells the GumScreen and everything inside of the GumScreen to use this as its BindingContext. This assignment will also assign the BindingContext on any Forms objects in your screen, so you only need to do this assignment on the GumScreen. Once the BindingContext is assigned, the code establishes individual bindings between UI properties and ViewModel properties. This binding creates an automatic connection between the ViewModel property to the UI property, so that any changes to the ViewModel will automatically update the UI (assuming the Get and Set functions and DependsOn are written correctly in the ViewModel). Notice that the code uses the nameof keyword in C#. The code above could also be written as shown in the following snippet:
While this is less verbose, it produces code which is easier to break. Using nameof provides compile-time checks against referenced properties.
Once the binding is set up, assigning properties on the ViewModel will automatically update the UI. For example, the values above initialize the score and health as shown in the following screenshot:
The ViewModel properties can be updated at any time. Doing so automatically updates the bound UI. For example, the following code can be used to modify the properties in response to button clicks:
Notice that the click events on the buttons do not directly access any UI elements - only the properties on the ViewModel. This makes it much easier to maintain the UI and to continue to add dependencies.
This section discusses how to add new element types to FlatRedBall.Forms. This section is for maintainers of FlatRedBall who would like to add new types to be used across all projects and to be included in new projects created with the wizard.
The high-level steps for creating a new Forms object are:
Define the class in FlatRedBall.Forms shared code project
Identify the required and optional Gum objects for your new control
Create a new Gum component in the embedded Gum project in the Gum plugin
Add a behavior for the new control
Implement the new behavior on the new component
Update FormsControlInfo
and GetIfIsCompleteFulfillment
to include your new behavior and control
Test the new control in a new project
You can verify that the Gum plugin code generation logic has picked up on the new behavior by checking a game's GumIdbExtensions
class. for the presence of your new control.
FlatRedBall.Forms is a logical wrapper over standard Gum objects. As such, the Gum objects can be modified in almost any way to fit the styling needs of your game. This document covers the default FlatRedBall.Forms controls and how to make modifications.
FlatRedBall.Forms provides flexibility in regards to Forms component structure, but some objects and states are required so that a Forms object can behave properly.
To identify the requirements you can look at the Forms behaviors in the Behaviors folder in your Gum project.
These behaviors are automatically added to your Gum project by the FlatRedBall Editor when you add a Gum project. If you used the FlatRedBall Wizard to create your project, these are also added automatically.
These behaviors indicate the requirements that a Forms control has. For example, ButtonBehavior requires that a ButtonBehavior-implementing Component must have a category called ButtonCategory, and that it must contain states with specific names.
If you create a Component which satisfies these behaviors, then your component can be used as a FlatRedBall.Forms button.
Some behaviors have additional requirements such as required objects. For example, the ComboBoxBehavior requires a category called ComboBoxCategory which contains a number of states, but it also requires two objects - a ListBoxInstance and a TextInstance.
If you modify any existing component which implements these behaviors, make sure you do not remove or change the name of any required categories, states, or instances.
Besides requirements from categories, some FlatRedBall.Forms can use optional instances if they exist. These are not required. For example, the Button component can optionally include an instance called TextInstance. Since this is an optional requirement, the Behavior does not include it as a requirement. However, you can find out more about optional requirements by looking at the API documentation for FlatRedBall.Forms.
In general, you will almost always be safe making the following modifications:
Changing variables such as position, size, or color
Modifying values assigned in states
Adding additional instances to a control, such as adding an icon Sprite to a ListBoxItem
Renaming components
The following modifications may introduce problems depending on if they are made on requirements:
Changing the name of states
Changing the name of instances
Changing the type of instances to incompatible types, such as changing an instance from Text to Sprite
Removing behaviors - you can do this if you would like the component to no longer be associated with FlatRedBall.Forms
Projects created through the FlatRedBall Wizard, or which have added Forms when creating a Gum project, should contain a default implementation of FlatRedBall.Forms controls in the Gum project. These can be found in the Components/Controls and Components/Elements folders. Note that as FlatRedBall.Forms is developed, new controls are created, so this list will change over time.
These forms controls are built to support a common, centralized style. You can make changes at the base level so that your styling changes apply to the entire project, or you can choose to modify individual components one-by-one without changing the global style.
To see how the styling is implemented, we'll look at one of the most common controls: ButtonStandard.
As mentioned earlier, a Gum component must implement a Forms behavior to be considered a Forms object. We can confirm that this is the case by checking the behaviors on ButtonStandard in the Behaviors tab.
This means that the ButtonStandard is required to have a ButtonCategory which contains states for displaying the button depending on whether it is enabled/disabled, highlighted, and pressed.
As mentioned above, we should not delete or rename these states, but we are free to make changes to the states to modify the styling so it matches our game.
Selecting a category in Gum shows the variables that are modified by all states in the category. In this case, the ButtonCategory modifies
Background.ColorCategoryState
FocusedIndicator.Visible
TextInstance.ColorCategoryState
If we want to change the state, we can make additional changes to the state by selecting it and modifying variables. For example, we may want to make the Text larger when the cursor is highlighting the button. To do this:
Select the TextInstance
Select the Highlighted state
Change FontScale from 1 to 1.5
The FontScale variable shoudl display with a white background instead of green to indicate that it is explicitly set on this state.
Note that the FontScale variable is now modified by all states in the category, not just the one that you set. All of the other states explicitly set the variable to the default of 1. You can verify this by clicking on the ButtonCategory.
Since these states are used at runtime automatically, we can run the game and see the font size change when the button is highlighted. Notice that the button text shrinks back to its normal size when the cursor moves off of the button. The text also shrinks when the button is pushed since we didn't modify the Pushed state.
In this case our modification to the button added a new variable that was previously unmodified by ButtonCategory. We can also change existing values. For example, when highlighted our Button's background uses the PrimaryLight ColorCategoryState.
In this case, the ColorCategoryState is defined on NineSlice, which is the base type for Background. Therefore, we should not modify the Red, Green, or Blue values on the Background object because they will be in conflict with the values assigned by this state.
If we would like to modify the Background color in the Highlight (or any other) state, we have a few options:
We can go to the NineSlice standard object and modify the PrimaryLight state. Be careful, making modifications here is concpetually similar to making modifications to a style sheet which is used across an entire project. If you would like to change this value globally, feel free to do so, but realize that making changes to the NineSlice styles can modify many components in your project.
Change the state value that is being used on the background. Although this is not recommended, you can change the background to use a different ColorCategoryState, such as changing the value to Success. This is not recommended because the ColorCategoryState states are named to indicate where they should be used throughout your project, and switching to different ColorCategoryStates may result in confusing the user by using the same color for different behaviors and states.
Add a new state to NineSlice for your specific case. This is useful if you have a case that isn't handled by the existing NineSlice category states.
Removing the usage of ColorCategoryStates from your ButtonCategory. The states in the NineSlice exist to make global styling easier, but they are not a requirement. If you would prefer to implement your own styling method, such as direct RGB values on the background, that's okay. Keep in mind that the Background NineSlice isn't even a requirement! You can remove it and replace it with something else like a Sprite if that works better for your game.
As mentioned above, using Gum to style your FlatRedBall.Forms controls provides a lot of flexibility. The existing structure exists to make changes easier to implement, but you are free to experiment and find your own preferred styling.
Binding can be performed on FlatRedBall Forms objects similar to binding on other UI frameworks such as WPF and Xamarin.Forms. This tutorial provides a binding example using Button, TextBox, and ListBox instances.
This tutorial uses a Glue screen which contains the following:
TextBoxInstance (TextBox)
AddButton (Button)
ListBoxInstance (ListBox)
RemoveButton (Button)
These instances are all default FlatRedBall.Forms objects which are available in all FlatRedBall projects created with the wizard.
Like the previous tutorial, this tutorial uses a ViewModel for binding. The ViewModel will need properties to bind to the following UI properties:
The string displayed in the text box.
Whether the Add button is enabled. It is enabled only if there is a string in the text box.
The list of objects in the list box.
The selected item in the list box.
Whether the Remove button is enabled. It is only enabled if there is a selection in the ListBox.
Not only will we be binding to FlatRedBall.Forms objects, but the binding will be two way binding. This means that the UI objects can make changes to the view model. For example, when the user clicks on an item in the list box, the SelectedItem is updated automatically.
The ViewModel includes a number of properties which will be controlled by the UI elements:
TextBoxText
SelectedItem
Notice that although these are controlled by the UI rather than code, they are written identical to other ViewModel properties. Also, they can be used as dependency properties for properties like IsAddButtonEnabled and IsRemoveButtonEnabled. In other words, the binding properties and dependency attributes will result in the add and remove buttons automatically enabling and disabling themselves according to properties that they depend on.
The ObservableCollection is used as the type for the ListBoxItems. This enables the ListBox to react to any items being added or removed. Whenever the ListBoxItem's Add or Remove methods are called, the ListBox will create ListBoxItems automatically. Notice that the ListBoxItems property has a private setter - there is no reason to make this public since the ObservableCollection will never change after it has been created.
The ViewModel can be created in the FlatRedBall screen class, just like in the previous tutorial. Binding to FlatRedBall.Forms object properties is done the same way as binding to Gum objects - by using the SetBinding method.
Like the previous tutorial, the add and remove buttons do not access any properties on the UI elements - only properties on the ViewModels are accessed.
Notice that the BindingContext is assigned on the Forms object. The Forms.BindingContext property is the same as the GumScreen.BindingContext property, so either can be assigned.
Often times games include custom reusable Forms components. These components may have dedicated ViewModel classes which can also be reused across multiple screens. This walkthrough shows how to create a reusable component, a reusable ViewModel, and how to assign the binding context appropriately.
For this view we will have a simple component which could be used for a game's settings. Note that if you actually need to create a view for settings, you should review the SettingsView control as a starting point. We'll be creating our own for this tutorial to keep it shorter.
Keep in mind the contents of this control are purely to serve as an example. The actual contents of your control could vary. For this example, consider the following control:
This control has two sliders and a checkbox, so our ViewModel should be built to support these controls. The following is an example ViewModel for this control:
To make the CustomSettingsView forms control reusable, we can set the binding on each of the controls in its custom code. To do this:
Open your project in Visual Studio
Open the CustomSettingsViewRuntime.cs file, which is located in the GumRuntimes folder. Note that if you added the control to a subfolder, you will find the runtime code in a subfolder as well.
Add the following code to CustomInitialize:
Note that this does not actually instantiate the CustomSettingsViewModel - it simply sets the binding so that the contained Forms components (sliders and checkbox) will be bound once the BindingContext is assigned. It is up to the parent container or screen to instantiate and assign the ViewModel.
Now our CustomSettingsViewRuntime is set up to bind its contained object's properties to a ViewModel's properties. Next we need to assign the view model.
This assignment can happen either directly by setting the BindingContext, or it can also be done by binding the BindingContext property. We'll cover both scenarios below and discuss why you might want to use one approach or another.
The simplest approach is to directly assign the BindingContext to an instance of the CustomSettingsViewModel.
For this example we'll consider a FlatRedball Screen called MainMenu which has an instance of the CustomSettingsView in its Gum screen. Even though we won't create it in this tutorial, the MainMenu may also have a corresponding MainMenuViewModel which would be assigned to Forms.BindingContext. We can override the BindingContext on the contained CustomSettingsViewInstance as shown in the following code:
This approach will correctly assign the BindingContext to the CustomSettingsView, but it does require explicit assignment of view models at the Screen level. The next section shows how to set up binding in a way that does not require multiple assignments at the screen level.
The BindingContext property is itself bindable. For simple situations this is not used very frequently, but if you have more advanced UI you may benefit from having binding context. For example, consider a situation where the CustomSettingsView is a child of another view, such as a PauseMenu.
In this situation, the PauseMenu may be part of the GameScreen and have its own binding context. Of course, it would be possible to access the internals of the PauseMenu in the GameScreen as shown in the following code:
This code begins to create a maintenance problem - if the Screen must explicitly assign all children BindingContext properties, then the children (PauseMenu) reusability goes down. Rather, we would want the PauseMenu to be responsible for assigning all of its children's binding context.
Therefore, we could modify our code to support this.
The first step is to modify the PauseMenuViewModel (the ViewModel of the parent component) to contain a CustomSettingsViewModel as shown in the following code:
The next step is to modify the PauseMenuRuntime's code to set the CustomSettingsView's BindingContext. As mentioned above, when CustomInitialize is called the view will not yet have a BindingContext assigned. Therefore, we cannot directly assign the CustomSettingViewInstance's BindingContext, but must bind to it:
Now, the GumScreen does not need to directly assign the CustomSettingsViewInstance's BindingContext. The code has become simpler:
By assigning the PauseMenu's BindingContext, the internal CustomSettingsViewInstance looks for a property on the PauseMenuViewModel with the name "CustomSettingsViewModel. Since the PauseMenuViewModel has this property, this will automatically be bound.
For this particular example it may seem like a pointless tradeoff - the binding context assignment was removed from the GameScreen, but additional code was added to the PauseMenuRuntime and PauseMenuViewModel. This extra code becomes more valuable if the PauseMenu (or any other container) were to be reused across multiple screens.
Notice that the code above has reduced the explicit assignments from three to two. We could further reduce this code by moving the PauseMenuViewMode inside the GameScreenViewModel and similarly binding the PauseMenu's BindingContext inside of the GameScreenGumRuntime. The amount of embedded ViewModel assignment you perform depends on the level of reusability your game needs.
The ListBox type includes the ability to customize individual items using templates. Two properties exist on the ListBox for customization:
VisualTemplate - this allows the usage of any Gum object as an item in a list box
FrameworkTemplate - this allows the usage of a FrameworkElement as an item in a list box. Note that the type of FrameworkElement must inherit from ListBoxItem
The following screenshot shows an example of a customized ListBoxItem. It contains a list box which displays levels with images and a check box.
Although the appearance of each item in the ListBox is different from the default appearance, the ListBox itself is still a standard ListBox with a custom VisualTemplate. VisualTemplates can be used for any type of modification to the ListBoxItem. Common examples include:
Adding additional Text instances for more information, such as the name and price of an item as separate Text instances
Adding images or icons to an object, such as a preview image in a level selection list box
Adding additional controls to a list box, such as a check box for multi-selection
Most of the time games only need a VisualTemplate and do not need to use a FrameworkTemplate. Therefore, this tutorial convers the usage of VisualTemplate which should cover almost all cases. FrameworkTemplate can be used for rare situations where the same visual is used for list box items which have different functionality. Most of the time this different behavior can be achieved with a VisualTemplate and data binding.
By default the ListBox creates one ListBoxItem for every instance in its Items property. The ListBoxItem is defined in the Gum project under the Controls folder as shown in the following screenshot:
The first step in replacing this ListBoxItem in our ListBox is to create a new component in Gum. The easiest way to do this is to copy/paste the existing ListBoxItem as a starting point. You can create a copy by using the CTRL+C, CTRL+V hotkeys or by right-clicking on the component and selecting the Duplicate option.
This component can be structured however you want. There are no requirements for what it must contain, how it must be named, or what states it must contain. By copying the existing ListBoxItem, we bring over the ability for the ListBox to be selected. Even this is optional, so if you do not want your ListBoxItem to respond to selections visually, you can remove the states or modify them to have no impact on appearance.
For this tutorial we will be creating a new ListBoxItem which has two Text instances:
InventoryNameTextInstance
InventoryCountTextInstance
Note that by default ListBoxItems attempt to assign a Text object by the name of TextInstance. By changing our Text names, our new ListBoxItem no longer supports default Text display. If you are creating a new ListBoxItem which is intended to be used as a default ListBoxItem throughout your entire game, you should consider keeping a Text instance with th ename TextInstance. In this case we are creating a ListBoxItem to be used in a specific case so this name isn't required.
To use the newly-created ListBoxItem, first you need a screen with a ListBox. For example, the following screenshot shows a single ListBox in the MainMenuGum screen.
Once this ListBox is added, the following code can be used to populate the ListBoxInstance, including using the new InventoryListBoxItem as the template:
Running the game shows something similar to the following screenshot:
The ListBox contains ten (10) instances of the InventoryListBoxItemRuntime Gum component - one for each integer added to listBox.Items. The VisualTemplate assignment tells the ListBox that each Item should result in a new InventoryListBoxItemRuntime being created.
The code above shows how to create instances of the InventoryListBoxItemRuntime component based on the contents of the ListBox.Items collection. This approach is good for testing how a custom ListBoxItem appears in your ListBox, but it won't work for a full game.
Typically each item in a ListBox needs more information than an integer. In the case of inventory, each item in the list box needs at least a name and count.
We'll create a new class which contains this information. We also want to create a class which will notify the ListBoxItem any time a property changes. To make this simpler, FlatRedBall provides a ViewModel
class which can be used as the base for your custom classes. By using ViewModel, and the Get/Set methods, any changes to properties are automatically broadcasted to UI. In other words, this enables changing a property on the ViewModel-inheriting class, and that updating the UI immediately.
The term "ViewModel" comes from the MVVM pattern which is a common way to display and manage data in C# front ends such as WPF, .NET MAUI, and Avalonia. Classes which inherit from ViewModel are often referred to generally as "view models", and should have the "ViewModel" or "VM" suffix for clarity.
We can create a new ViewModel for our new custom list box items as shown in the following code:
For more information on the specific syntax of ViewModels (such as Get/Set and DependsOn), see the BindingContext Property page.
Each InventoryItemViewModel will correspond to an individual InventoryListBoxItemRuntime. We also need to create a ViewModel for the entire screen. This ViewModel contains a list of InventoryItemViewModels in an ObservableCollection. Our ViewModel for the entire screen should match the name of the screen, so we will create a class called MainScreenViewModel.
Finally we can modify our MainMenu.cs class to use the MainScreenViewModel for its items, as shown in the following code:
If we run our game now, the ListBox shows four items (one for each InventoryItemViewModel instance). Notice that we do not directly add items to the ListBox.Items. Instead, we bind the listBox.Items to the viewModel.ListBoxItems. This results in the ListBox automatically keeping in sync with the ViewModel.
We can observe this behavior by adding code to the MainMenu's CustomActivith to add a new item whenever the space bar is pressed.
Note that removing items from the view model also results in removal of the matching item in the ListBox.
Next we'll update our binding so that the Text instances in our InventoryListBoxItem display the name and count properties from our InventoryItemViewModel. To do this, open InventoryListBoxItemRuntime.cs and modify the contents as shown in the following code.
Now if we run our application, each item displays the information from each InventoryItemView.
Next we'll keep track of the selected item. We can do this by creating a new property on the MainScreenViewModel which has
We can bind to the SelectedItem in MainMenu.CustomInitialize. Modify the code to add the "New" code shown here:
Just like all other properties, the SelectedObject/SelectedItem properties stay synced, so we can use the ViewModel's SelectedItem in code. For example, we can modify CustomActivity to increase inventory on the selected item when the Enter key is pressed:
If we run the code, select an item, and press Enter, the inventory count increases.
This tutorial shows how to bind a ListBox and ListBoxItems to view models. Interacting with the view models (adding or removing items, modifying properites on the view models) automatically update the view.
The Top Down settings for entities provides an easy way to implement top-down movement for a character. It can be extended to support any input device, multiple sets of movement variables, and animation.
Your game may require subscribing to various Forms for every instance of a particular type of Forms object. For example, you may want to play a sound effect whenever a Forms button is clicked. Subscribing to every instance of every Forms object can be difficult to maintain. Instead, we can create events inside the Gum runtime classes to handle behavior for every instance.
Every Forms object is backed by a Gum "runtime" object. For information about Forms vs Gum, see the Forms vs Gum tutorial. For example, the Button class has a number of types of Gum runtimes:
ButtonClose
ButtonConfirm
ButtonDeny
ButtonIcon
ButtonStandard
ButtonStandardIcon
ButtonTab
We can tell that this is the case by looking at the behaviors for each of these controls.
Note that while a full implementation may require handling each of the Runtime classes listed above, for simplicity this tutorial discusses only ButtonStandard.
Every runtime has a code file - even runtimes which correspond to built-in Forms types such as Button. For example, all of the Button Gum components have corresponding Runtime classes in Visual Studio.
Each runtime provides a CustomInitialize method which is similar to CustomInitialize on Screens and Components. The CustomInitialize is called once when a new instance of our component is created. This code runs regardless of how our Gum runtime is created. This includes if:
The component is added to a Gum screen in the Gum tool
The component is instantiated by calling its constructor in custom code. For example, calling new ButtonStandardRuntime();
The component is instantiated through its corresponding Forms control. For example, calling new Button();
Therefore, we can add code to CustomInitialize to perform custom logic on all of our Button instances.
To respond to whenever a Button is clicked, we can modify our ButtonStandardRuntime as shown in the following code:
This code results in printing "Button clicked" when any button (with a backing ButtonStandard Component) has been clicked.
Again, remember that the Button in Forms is a special type of control becuase it has multiple components handling its implementation, so you may need to add similar code to the other Runtime classes.
The CurrentMovement property controls the values used by the top down entity in response to input values. This value can be changed according to various conditions in your game such as:
Collision with different terrain (such as walking through mud)
Responding to power-ups (such as collecting a power-up which increases speed)
Responding to special moves or input which changes the character's movement variables (such as holding down a run button)
Movement values can be defined in Glue or code. If your game has a limited set of movement values, these can be defined in Glue. To define a movement value in Glue:
Select the entity
Click the Entity Input Movement tab
Verify that your entity is using the Top Down option for Input Movement Type
Click the Add Control Values button
Modify the newly-added movement values as necessary
The Top Down tab displays all movement values for the selected entity.
You can assign the current movement values in code through the CurrentMovement propety. For example, the following code assigns the movement to FastMovement or Default depending on the state of an Xbox360GamePad:
This tutorial provides information on how to change the input device used by the entity. By default the entity uses the first Xbox game pad if one is available, otherwise the entity uses the keyboard. Sometimes this functionality is not desired, such as if the game supports multiple players or AI-controlled entities.
The Top Down entity code interacts with the IInputDevice interface, which provides a standard set of values for controlling game objects. Both the Xbox360GamePad and Keyboard implement this interface, so if the default implementation for this interface meets your game's needs, you can use these objects as-is. Otherwise, your game can implement its own IInputDevice to change how the top down entity is controlled. First we'll look at how to customize the input to use different keyboard keys.
As mentioned earlier, our Player entity defaults to using an Xbox gamepad if one exists. If not, it uses the keyboard. Even though we won't customize this code, we can see its implementation by looking in Player.Generated.cs and searching for InitializeInput.
Ultimately, these calls make their way to calling CustomInitializeTopDownInput. Since this is a partial method, we need to add it ourselves to our Player if we want to customize the input. To do this:
Go to Player.cs
Add the following code to the Player.cs file
The player will now move around with the IJKL keys (rather than the default WASD). Note that this will also override input even if you have an Xbox gamepad plugged in. If this is not desired, we can change the input if it's using the keyboard. This is an example of how to keep the input device, but only change the movement code conditionally:
This code could contain further updates to change desired input based on other input devices such as Xbox360GamePad.
The code in the section above modifies which keys are used from the keyboard. It could be expanded to also handle other input devices like Xbox gamepads. However, instead of changing which keys are used when using the keyboard, you may want to change which input device is being used by your player. This is typically done in the GameScreen. This allows the GameScreen to initialize input, especially if your game allows the player to pick which input device to use. For example, to force the player to always use an Xbox gamepad, modify the following code in the GameScreen's CustomInitialize:
If your game followed the previous tutorials, then your GameScreen has a PlayerList with a single Player named Player1. In this setup, the GameScreen will always have at least 1 player, but can have more. A quick way to support multiplayer without any UI is to detect the number of Xbox gamepads plugged in. If more than 1 is plugged in, we can create additional players very easily. The following code will create one player per Xbox gamepad plugged in. Notice that we only create additional players if we have two or more gamepads since the first player is automatically created.
To see more than one player you must have at least two gamepads plugged in to your computer. Also, keep in mind that the players will initially overlap so you must move one to see both.
This tutorial sets up an entity with Top Down controls. It provides a default implementation which requires no code. Later tutorials show how to interact with this plugin using code.
The FRB Editor provides support for top-down entities through the Entity Input Movement tab. Any entity can be created as a Top Down entity; however, the most common setup is to have a Player entity which uses top down controls.
Empty projects can use the Project Setup Wizard to create a top down player entity. Existing games can add top down controls to new or existing entities with a few clicks. This tutorial shows you how to do both.
The simplest way to set your project up is to use the new project wizard. FlatRedBall automatically launches the wizard when creating a new project.
To create a top-down project, select the Standard Top Down button.
Now your game should be set up with a fully-functional top-down entity. You can verify this by clicking on the Player entity and then clicking on the Entity Input Movement tab. The Player should be marked as having Top-Down as its Input Movement Type.
This section will explain how to manually add a GameScreen and Top-Down entity. You do not need to follow this section if you have used the wizard as shown in the previous step.
Select the Quick Actions tab
Click the Add Screen button
Click OK to the default GameScreen name (all games should have a single GameScreen)
To add a Player entity:
Click the Add Entity button
Name the entity Player
Check:
Circle under Collisions
Top-Down under Input Movement Type
Leave the rest of the defaults and click OK
If you already have an entity created, you can make it a Top Down entity:
Select the entity
Click the Entity Input Movement tab
Click the Top-Down option
By default your GameScreen should have a list of Players (it was an option earlier when creating the Player entity). We recommend always creating a list of Players even if you intend to only have one player. This standard appears throughout FlatRedBall's documentation and can make moving from one project to another easier.
If you did not add a PlayerList earlier by keeping the Include lists in GameScreen, or if you created your GameScreen after your Player, you can manually add a PlayerList by following these steps:
Verify Player is selected
Click the Quick Actions tab
Click the Add Player List to GameScreen button
You will also need a Player instance in the list. To do this, drag+drop the Player onto the GameScreen and it will be added to the Player list.
Now that the entity is marked as a Top Down entity and now that we have an instance of the entity in the GameScreen, we can run the game and see the player move. By default the entity uses a gamepad if one is connected. Otherwise, the entity will use WASD keys on the keyboard.
Post Processing is a technique used to re-draw a portion or the entire screen, applying either a custom shader or resizing the portion to create a pixelated effect.
Post processing requires the use of RenderTargets - memory which can be the target of rendering which do not automatically draw to the screen.
A RenderTarget is a piece of video memory, similar to a Texture2D, which can be used as a temporary storage of rendered graphics before being processed and drawn to the screen. To understand how this works, first we'll consider a game which does not use render targets. This type of game may have visual objects which are added through the FlatRedBall Editor or code. These objects are added to the FlatRedBall Engine, which in turn draws the objects directly to a screen. The following diagram can help visualize this process:
Render targets act as "temporary" storage of graphics which can be used to further perform processing on the entire image at once. For example, a render target could be used to apply "bloom" - an effect which applies a glow to the brighter parts of an image.
The following diagram shows how such a render target might be used to apply bloom:
RenderTargets can be drawn to (usually by the FRB engine) and used when drawing to the screen in the same frame. Typically post processing does not add much overhead to a game, so games can use post processing without worrying about reducing frame rate or introducing additional frame lag.
Drawing to a RenderTarget can be done using one of the following methods:
The entire FlatRedBall draw call can be "wrapped" in a RenderTarget. In other words, every object added to FlatRedBall can be drawn to the screen.
Individual Layers in FlatRedBall can draw to a RenderTarget. This is useful if you only want to render a portion of your game to a render target - for example you may want to apply a blur effect to objects in-game, but not to the UI.
Similar to layers, entire Cameras can be used to draw to a render target. This technique is considered more advanced than the other two, but can be very effective if you want to control when a render target is updated. For example, you may want to only update a render target when something on-screen changes.
For simplicity this tutorial uses the first approach of rendering the entirety of FlatRedBall to a single layer.
We'll begin with a sample platformer project. Note that the contents of this project do not matter, so long as we have something on screen to use for testing.
As mentioned above, we will render the entirety of FlatRedBall to a RenderTarget, which then will be drawn to the screen. Since all of FlatRedBall will have been drawn to the RenderTarget, we must draw the RenderTarget to the screen using objects which are not part of FlatRedBall. We can use a SpriteBatch.
We can add post processing by adding the following code to our Game.
First, add the following using statement:
Next, add the following to the Game1 class at class scope:
Instantiate the RenderTarget2D and SpriteBatch in Game1's Initialize method after initializing FlatRedBall and after calling GeneratedInitialize:
Finally we can modify the Draw call as shown in the following code to perform drawing on our RenderTarget:
The entire Game1.cs class might look like this:
If we run our game, it looks the same as before. The difference is that we are now drawing our RenderTarget to screen using a SpriteBatch. This allows us to make modifications to the entire screen as we draw it - typically by using a shader.
We can draw the RenderTarget to the screen using any shader which works on SpriteBatch. For this example we'll use the AdaptableCrtEffect provided by mfigueirido: https://github.com/mfigueirido/AdaptableCrtEffect
Feel free to clone the entire repo to see how it works, or download the specific files:
Rock Blaster is a top-down Asteroids-like game. The following tutorials introduce a variety of advanced topics. These tutorials continue to advance the concepts and FlatRedBall functionality introduced by Beefball.
Note: RockBlaster contains useful information about how to use the FlatRedBall Editor (also referred to as Glue in these tutorials), but it is a tutorial which was written before many of the current FlatRedBall features were created. Therefore, some of the approaches in this tutorial may not represent the current recommended patterns.
Enemy pathfinding allows enemies to navigate through complex levels. This can be used to move to a target position (such as a patrol point) or follow an entity (such as enemies chasing a player). This tutorial shows how to create a pathfinding enemy. Specifically, it covers the following topics:
Creating a TileNodeNetwork which defines the walkable parts of a map
Creating an input device which is used to control the enemy so it follows its desired path
Creating line-of-sight pathing for more natural movement
First we'll create a TileNodeNetwork. This is defined in our GameScreen, but will use the Map for each level. For a more thorough walkthrough of creating a TileNodeNetwork, see the TileNodeNetwork page.
To create the TileNodeNetwork:
Select the GameScreen
Click the Add Object to GameScreen under Quick Actions, or right-click on GameScreen and select Add Object
Select the TileNodeNetwork type
Enter a name such as WalkingNodeNetwork
Click OK
Next we'll define a tile to use for pathfinding. To do this:
Open any of your levels in Tiled
Select the Tileset that you use for your GameplayLayer. By default this is called TiledIcons
Click the wrench to edit the Tiles
Select a tile that you would like to use for pathfinding
Change the Class for that tile to WalkableTile
Save your tileset (TSX)
Next, select your current level (such as Level1Map) and paint the WalkableTile onto the GameplayLayer. Note that if you would like to be able to walk on areas which already have other tiles, you can create a new layer specifically for Tiles.
Be sure to save both your level and the tileset.
Finally, we can indicate that the WalkableTile should be used to populate the nodes in the Tile:
Select the WalkingNodeNetwork Object under GameScreen
Click the TileNodeNetwork Properties
Select the From Type option
Select Map as the Source TMX File/Object
Select WalkableTile as the Type
Optionally, you may want to also set the Visible variable to true so the TileNodeNetwork shows up in your game.
The game should now show the TileNodeNetwork. Note that you can also make the GameplayLayer invisible in Tiled so that the TileNodeNetwork is easier to see in game.
Notice that the links between nodes are either vertical or horizontal. We will discuss diagonal movement later in the tutorial.
Now that we have a TileNodeNetwork defined, we can create an input device. Of course, we need to have an Enemy defined. For this tutorial I'll use a simple enemy with the following characteristics:
It is named EnemyBase - this naming convention is used so that the enemy could be used as a base Entity for derived variants. Larger games would likely need multiple enemy types so this sets up to expand easily.
The Enemy has a single Circle collision. More complex games may include multiple types of collision, but it's important to note which collidable object will be used as the solid collision - we'll use this for line-of-sight pathfinding later in this tutorial.
The enemy uses the Top-Down movement types, just like the Player.
The enemy's InputDevice is set to None. As indicated in the FRB Editor, this must be assigned in code or the game will crash.
Next we'll define the InputDevice. You may be familiar with the InputDevice as hardware input which can control the movement of the Player (such as the Keyboard or Xbox360GamePad). Although these are common implementations of the input device (IInputDevice interface), the concept of an InputDevice is something which can be read by an Entity (such as the Enemy) to determine how it should move.
In this case the InputDevice will not be tied to actual hardware. Instead, we will return input values which move the EnemyBase through the map by following the path obtained from the WalkingNodeNetwork.
Fortunately, the TopDownAiInput class is an InputDevice which automatically returns these movement values according to a desired path.
First, we will assign this InputDevice in the EnemyBase's CustomInitialize:
Notice that the topDownAiInput is defined at class scope - this lets us access it in other methods without needing to cast the InputDevice every time. Also, the IsPathVisible and PathColor properties can be used to control the appearance of the path that the EnemyBase is going to take to get to the player. We will enable this to show the path. Also, it may be worth turning off the WalkingNodeNetwork visibility so that the EnemyBase path can be seen more clearly.
Next, we'll create a method to set up the node network and target player. Add the following to EnemyBase.cs:
For this tutorial we assume a single player which never changes. A multiplayer game may require changing the target based on which player last hit the enemy, proximity, or whether the current target is dead.
Also, note that this method must be called by the GameScreen. We will add the code to call this method later in this tutorial.
Finally, we can set the path to follow the player in CustomActivity.
Notice that this code only updates the path every second. Updating the path is a fairly quick operation, but it can be expensive if the game includes hundreds of EnemyBase instances. We'll check once per second to prepare for a larger game.
Now that our EnemyBase has the code it needs to pathfind to the player, we can add code to the GameScreen to call InitializePathfinding.
EnemyBase instances can be created both before and after GameScreen.CustomInitialize is called, so we must handle both cases.
To handle EnemyBase instances created before CustomInitialize is called (if they have been added directly to a level in the FRB Editor or live edit), we can loop thorugh EnemyBaseList and call InitializePathfinding.
To handle EnemyBaseInstances created after CustomInitialize (if they are added using a factory or variant object such as through a spawner) we can subscribe to the factory method. Note that a game which has derived enemy variants must subscribe to all derived variant factories to handle creation.
Modify the GameScreen.cs to include the following code:
Finally, we can add the following code to GameScreen to create EnemyBase instances when clicking with the mouse.
We can now click to place EnemyBase instances. Each EnemyBase instance follows the player according to pathfinding along the WalkingNodeNetwork and the position of the Player.
Notice that the EnemyBase moves fairly quickly - the same speed as the Player. You may want to reduce the EnemyBase's movement speed to make it easier to test pathfinding.
So far we can create an EnemyBase by clicking in the game. The enemy base instance follows the player using the WalkingNodeNetwork. We have lots of options for customizing the way the enemy follows the player.
First, you may nave noticed that the enemy follows the player on a path by only moving horizontally or vertically. The reason for this is because our WalkingNodeNetwork is set up to only have four links per node: up, down, left, and right. we can include diagonal nodes to allow the enemy to also move diagonally.
Some games, such as the Legend of Zelda on the NES and the 2D Final Fantasy games do not allow the player, NPC, or enemies to move diagonally. If this limitation matches your game, then you can keep this default setup.
However, if your game requires diagonal movement, you can enable it in the TileNodeNetwork properties for the WalkingNodeNetwork.
Now our EnemyBase instance can travel diagonally.
You may have noticed that at times the EnemyBase backtracks - in other words at times the enemy seems to walk backwards rather than following its path. This can happen when the path is re-evaluated. The new path requires the enemy to move backwards slightly to move to its next node.
To help explain this problem we can visualize a simple path that an enemy may take. It's important to remember that each node in the TileNodeNetwork is always placed at the center of the tile. The following image shows an example path. The blue circles represent the node positions, the red circle represents the enemy, and the purple lines represent the path the Enemy takes.
Notice that the enemy moves directly towards the first node diagonally. This happens regardless of where the enemy is located - by default the very first segment in the path connects the enemy to the path. In this case the movement seems to be what we might expect, but in some situations the enemy may be positioned such that its first movement requires it to backtrack, as shown in the following image:
Notice that in this case the closest node to the enemy is to the right of the enemy. Therefore, the first movement segment has the enemy moving to the right before it resumes moving to the left.
We can solve this by removing the first segment of the path, but only if the path has more than one point. If there is exactly one point in the path, then we should not remove it since the enemy is near its final target.
We can modify our code as shown in the following EnemyBase.CustomActivity:
Note that while this approach does solve the backtracking problem, it can at times cause the enemy to move diagonally to the next node in a way which may result in getting stuck behind collision.
The reason we discuss this solution is not necessarily to suggest that this is the best solution for backtracking for production games, but rather to show that the TopDownAiInput object allows you to manipulate its Path at any time. You can perform more advanced logic to determine if backtracking will occur, and to remove, add, or modify existing nodes to correct for this problem according to your game's specific needs.
At this point our enemies can pathfind but they do not respect the level's solid collision. If we enable collision we will notice that this introduces a number of problems.
To enable collision between EnemyBase and the leve's SolidCollision:
Expand GameScreen's Objects folder
Drag+drop EnemyBaseList onto SolidCollision
Verify that the newly-created CollisionRelationship is using BounceCollision
Change the elasticity to 0 so that the EnemyBase doesn't actually bounce off the wall. By setting it to 0 the wall absorbs the EnemyBase's velocity.
If we run our game now we'll notice that the enemy gets stuck when trying to follow the player along any collision.
This is happening because the collision on our enemy is too big - it has a circle with a radius of 16. This prevents it from getting close enough to tiles to properly pathfind. We can fix some of the problems by changing the EnemyBase's CircleInstance radius to a smaller value. A value of 8 allows the EnemyBase to touch the center of a tile, so we will change the value to 8.
This change helps pathfinding, but it doesn't solve all of the problems. If your game is using diagonal movement, you may notice that the enemy slows down when colliding against corners. We can force the enemy to pathfind around a house. Notice the enemy slows down when moving around corners.
This is occurring because we have enabled 8-way pathfinding, and this includes corner pathfinding. We can set our node network visibility to true to see that the node networks cut corners, in other words diagonal paths connect nodes which result in the EnemyBase attempting to push against a corner collision.
We can disable corner cutting through an option in the TileNodeNetwork Properties tab:
Now our tile node network still supports diagonal movement, but it will not attempt to move diagonally around corners as shown in this screen shot:
So far we have made modifications to the EnemyBase entity and the WalkingNodeNetwork to address problems with collision, but one issue still remains - our enemy movement still feels unnatural. We can solve this problem by implementing line-of-sight pathfinding. This form of pathfinding initially uses the TileNodeNetwork to determine which path it should take, but it then eliminates nodes along the path if a more direct path is available. This allows an entity to move in any direction and it can make pathfinding feel more natural.
Of course, keep in mind that your game may intentionally have pathfinding strictly along a node network so whether you use line of sight pathfinding can be a stylistic choice.
To enable line of sight pathfinding, the TopDownAiInput must have two additional pieces of information:
The width of the collision. This is needed because line of sight pathfinding must check if a direct path exists to the target and also if following that direct path results in bumping against solid collision.
The solid collision to avoid
We can enable line of sight pathfinding by modifying the InitializePathfinding
method as shown in the following code:
Notice that we use CircleInstance.Radius*2
to determine the width of the entity. If using a different shape, or multiple shapes, your code needs to account for this. Also, keep in mind to also add any offsets if your shape is not centered on the X or Y. Finally, you may need to increase this value if your collision shapes can adjust during runtime - such as being modified by AnimationChains.
Since this method now requires passing in a TileShapeCollection, we need to modify GameScreen.PrepareEnemyPathfinding to pass in the SolidCollision as shown in the following code:
Now our enemy follows the player directly if there is a line of sight. If not, the enemy performs line of sight checks to skip nodes which are not necessary. Notice that when navigating tight spaces the enemy follows the node network, but when navigating open spaces the enemy moves more directly towards its target.
This tutorial has shown how to create an enemy that performs pathfinding towards a player. This pathfinding can be customized to address common problems, to enable or disable diagonal movement, and to use line of sight pathfinding for more natural movement.
The FRB Editor provides built-in support for animations. Simple games can implement animation with no code. More complex games can customize the default animations both in the FRB Editor and code.
This guide explores animations for top-down entities.
If your game was created using the New Project Wizard then you should already have a Player entity that is fully animated.
We can see the animations used by the player by expanding the Player's Objects folder, selecting the SpriteInstance, and looking at its Animation Chains variable. Notice that this variable is set to the .achx file in its Files folder.
The SpriteInstance and .achx file in the Player entity are just like any other Sprite and .achx file - they an be modified as desired. This tutorial won't cover the details of modifying a .achx file. For more information on working with Animation Chains, see the Animation Editor page.
If we run the game, the Player is animated - it displays idle and walk animations in each of the four directions.
We can view the setup for animations by selecting the Player Entity, clicking on Entity Input Movement, and selecting the Animation item.
This section controls which animations are displayed on the Player's sprite.
Each row defines one set of animations, and the condition for when those animations play. Animations at the bottom of the list play if their condition evalutes to true. In other words, Idle animations always play unless the Walk conditions are satisfied.
The next section covers each variable.
Animation Name is the variable that determines which animation to play if the conditions in a given row are fulfilled. The Animation Name property uses a dropdown which lets you pick animations from the referenced .achx.
Notice that the names in this dropdown do not match the names in the referenced .achx.
If the Is Direction Facing Appended checkbox is checked, then FRB automatically assumes that animations will have Up, Down, Left, and Right versions, where the direction is appended. Therefore, FRB strips off these directions and only displays the animation name (removing duplicates).
This means that if you wanted to add additional animations to the .achx, or if you wanted to create a brand new .achx, every animation should have 4 versions, one for each direction. For example, you might add:
AttackUp
AttackDown
AttackLeft
AttackRight
Notes have no impact at runtime. You can add any notes that might help in development.
These values restrict when an animation can play depending on the player's current speed relative to the ground. If these values are null, then here are no velocity-based restrictions on when an animation can play.
Min Velocity Absolute can set the minimum speed for an animation to play. For example, by default the Player has a minimum velocity set for when the Walk animation plays.
These values specify min and max bounds for when to play the animation. These rely on the input for the character which has an absolute value of 0 - 1.
The Animation Speed Assignment is used to optionally set the animation speed based on input or velocity.
ForceTo1 forces the animation speed to 1
NoAssignment removes all assignment of animation speed so it can be modified in custom code
BasedOnVelocityMultiplier multiplies the specified value by the current velocity. Usually the specified value is less than 1.0f.
BasedOnMaxSpeedRatioMultiplier multiplies the ratio of the entity's speed (current speed / max speed) by the specified value. Usually the specified value is 1.0f.
BasedOnInputMultiplier sets the animation speed based on movement input values which which are between 0 and 1.
Movement Name is used to limit an animation to a particular movement. For example, you may set the movement value to Running if the user is holding down a run button. In that case you may want to limit the movement to if the Running movement values are set.
The animation section of the top down Entity Input Movement tab is intended to contain all animations. Of course, games may need to assign animation using conditions that are not offered in this view. The Custom Condition text provides ultimate flexibility for assigning animations.
If this text box is not empty, then its contents are directly copied into generated code for an if-statement to decide whether the animation should play. The recommended approach is to create a single bool
variable in your code which can be used to determine if an animation should play.
For example, consider an animation that plays when the player is taking damage. Your entity might have the following code:
IsPlayingDamageAnimation can now be used as shown in the following image:
Multiple conditions for whether an animation plays can be combined. For example, an animation might only play if the Movement Name is set to OnIce and if Min Movement Input Absolute is set to 0.3. This results in the animation only playing if both values are true.
The GroundVelocity property is used to simulate the ground moving under the player. By default GroundVelocity is a 0-length vector which means the player is not standing on moving ground.
GroundVelocity can be set for the following reasons:
Simulating standing on moving ground such as a treadmill or moving walkway
Standing on a moving platform such as log floating on a river
Being pushed by the environment such as standing in blowing wind or flowing water
GroundVelocity is a Vector3 for convenience when applying velocity, but only the X and Y values should be assigned. Setting the Z value results in the entity's Z value changing which can affect ordering or even move the entity out of the camera's minimum and maximum Z values.
In other words, be sure to keep GroundVelocity.Z set to 0.
GroundVelocity can be set per instance. For example, the following code sets the GroundVelocity on Player1 when the Space bar is held down.
Top down entity movement is performed relative to GroundVelocity. If GroundVelocity is assigned to a non-zero vector, then all movement is performed relative to that new value. For example, if GroundVelocity is set to have an X value of 32, then the entity can move 32 pixels faster on the X axis when moving to the right, and 32 pixels slower when moving to the left.
Top-down games usually require entities and environment to sort in a way that suggests depth. This page covers the various forms of sorting in a top down game.
FlatRedBall sorts objects using cameras, layers, and positioning. Multiple cameras can be used to control draw order. Within a single camera multiple layers can be used to sort objects. This article assumes that all objects are on a single camera and on a single layer. Of course, if your game is using Gum for your game's HUD and UI then it is likely using multiple layers to keep Gum on top of your game. Note that the depth buffer can also be used to enforce sorting, but this article does not discuss this technique of sorting.
Entities and tile map layers use a combination of Z and Y values to control sorting.
Entities are drawn relative to tile map layers according to their Z values.
By default, all entities have a Z value of 0. This means that they will draw above layers with a Z value that is less than 0, and will draw below layers with a Z value that is less than 0.
By convention FlatRedBall loads Tiled maps so that the GameplayLayer is at Z=0. Each layer above and below the GameplayLayer is separated by 1 Z value. For example, consider a map with the following layers and their Z values:
AbovePlayer: Z=1
GameplayLayer: Z=0
GroundDecorations: Z=-1
Ground: Z=-2
In this map the AbovePlayer layer is positioned at Z = 1, so any tiles placed on this Layer draw on top of the Player.
If we want the player to overlap the pillar when below it, but be behind the pillar when above it, then the pillar must be broken up into two separate layers.
If we run the game we can see the player moving above and below the pillar.
Of course this requires adjusting the player's collision to prevent it from moving too far up when below, or too far down when above. Otherwise, the player may overlap both layers at the same time.
As mentioned above, the GameplayLayer is given a Z value of 0, equal to the default Z value for entities. Sorting between entities and the GameplayLayer should be considered arbitrary since it depends on the presence of other layers. FlatRedBall may choose to sort Tiled layers with a Z=0 above or below the player for performance reasons.
Most of the time this arbitrary sorting does not cause problems because the GameplayLayer is only made visible for debugging purposes, and is made invisible for final games. However, if you would like to force a particular sorting, you can shift your maps by a small amount in code.
For example, if only the GameplayLayer is visible, then entities will sort below the GameplayLayer.
We can shift our map slightly below the player by adding the following code to our GameScreen's CustomInitialize:
Now all entities sort above the GameplayLayer.
Keep in mind that this method of adjusting sorting is useful for keeping your entities above GameplayLayer, but it results in the player always above the GameplayLayer. If you would the player to sort above and below tiles according to its position, you must separate the tiles into different layers.
The Rock Blaster tutorials will reinforce some of the topics covered in the Beefball tutorials including:
Entity creation
Screen creation
Object creation
Variable creation
File creation and management
Collision
Scoring
Hud
Game Flow
The Rock Blaster tutorials will include the following topics as well:
Handling game graphics (PNGs)
Rotation
Dynamic Entity creation
Game data (score)
And much more!
So let's get started!
This project tutorial assumes a Top Down Standard starting point (using the wizard), but the steps here can be used to add an Enemy entity to any project. We assume that the game already has levels.
We'll be creating a new Enemy entity for this tutorial. To do this:
Click the Quick Actions tab
Click the Add Entity button
Enter the name Enemy
Check AxisAlignedRectangle
Check Top-Down for the Input Movement Type
Leave the rest of the defaults and click OK
We will also change the color of the enemy rectangle to tell it apart from the player:
Expand the Enemy Object folder
Select AxisAlignedRectangleInstance
Select the Variables tab
Change Width to 16
Change Height to 16
Change Color to Red
To add an enemy to Level1:
Expand the Screens folder, then expand Level1's Objects folder
Select the EnemyList
Click the Quick Actions tab
Click Add a new Enemy to Enemy List
Modify the X and Y values for the new enemy so it is inside of the level boundaries by changing X to 160 and Y to -160
So far our Enemy instance is an entity with no behavior - it simply stays in the same spot when the game runs. First, we'll mark it as using Top-Down movement:
Select the Enemy entity
Click the Entity Input Movement tab
Set Input Movement Type to Top-Down
Set Input Device to None (Can Assign in Code)
Now our Enemy has the behavior of top-down movement, but it is not using an input device for movement. Custom input can be set by defining an input device class which inherits from FlatRedBall.Input.InputDeviceBase. To do this:
Open the project in Visual Studio
Create a new class called EnemyInput. Mine is in an Input folder.
The EnemyInput class needs to inhert from the FlatRedBall.Input.InputDeviceBase class which provides virtual methods to control how the input device behaves. Although the InputDeviceBase class offers many virtual methods, the only two that the top-down movement logic uses are:
GetDefault2DInputX
GetDefault2DInputY
We can override these to return values between -1 and 1. In this case we'll return constant values to test the functionality, as shown in the following code snippet.
Next we need to assign the EnemyInput on the Enemy. To do this:
Open the Entities/Enemy.cs file in Visual Studio
Modify the CustomInitialize as shown in the following snippet:
Now we can run the game and see the enemy move down to the right.
This tutorial primarily shows how to create an EnemyInput class which can be used to control your enemy. So far the device returns constant values for GetDefault2DInputX and GetDefault2DInputY. A real game would use logic to determine what to return, such as the position of the Player, or how far along the enemy has moved along a patrol path.
At this point the rest of the implementation is up to you depending on your needs.
This tutorial begins with the creation of a new project in FlatRedBall. We'll make a new Desktop GL .NET 6 project.
To create a new project:
Select File -> New Project or click the New Project button in the Quick Actions tab if you do not have a project open already
Enter the name RockBlaster (no spaces) as your project name
Click Create Project!
Wait for your project to finish downloading and the Project Setup Wizard automatically opens.
Check the Custom (Advanced) option.
Select Start Wizard from Scratch.
We will change a few of the default options. Follow along with these images and make your options match.
Our game will not use Tiled Map files. CloudCollision is only used for platformer games.
Our player will not use default control types like Top-down or Platformer. We will be implementing our own custom controls, so select the None (controls will be added later) option. Our player will rotate, so Circle collision is preferred to Rectangle collision. Also, our player is not a platformer character, so uncheck Add Player vs. cloud collision.
Change Number of levels to create to 1. Uncheck the other options since our game does not have any Tiled Map files.
Leave UI options unchanged. We will use Gum to display game HUD.
Set the Game Resolution to 800x600 and the Game Scale% to 100. Uncheck all other Camera options. Our game will not have a Camera which moves.
Skip the option to Download/Import screens by clicking Next.
Skip the option to add additional objects by clicking Next.
Click Done. Wait a moment and your project will be all set up.
If we run our game now we will see our player which is a white circle.
That was easy! You now have a project that we will use in the following tutorials. Next we will set up the skeleton (the general structure) of our game.
Welcome to the Rock Blaster tutorials. Rock Blaster is a simple top-down game similar to the old Asteroids game. This set of tutorials will cover a variety of Glue topics. If you've been through the , this set of tutorials will build on that information and show more Glue functionality. If you haven't yet, we recommend going through the first - they cover a lot of the basics that we will use in this game.
The following tutorials will use a set of art from a classic PC game named which by , the original artist on the game. Thanks to Danc for supporting game development!
This tutorial covers how to create enemy movement in a top down game with direct control over the enemy's input device. Compared to the tutorial, this tutorial shows how to directly control the input device manually. This approach is considered low level and is not recommended for beginners. If your game includes enemies which can pathfind through a map then the tutorial is recommended.
So far we've done a lot of work on our project, but the only visual thing in our game is a white circle. This tutorial focuses on the Player - the Entity which is directly controlled by our input devices.
Before we start working on our Player entity, keep in mind that entities do not have to be fully-defined all at once. If you are familiar with the genre that you are creating a game for, or if you have very detailed plans about what you are creating, then you may be able to define your Entities very thoroughly the very first time you work with them. However, if you still have some questions in your head about how the game will play or how objects will behave (this is very common, so don't feel bad if you do) then you may implement some of your Entity before you play the game and possibly move on to a different area of the game.
This tutorial takes an iterative approach to defining Entities, as this is more common in real game development. This means we will define some of the Player Entity now, and we'll return to add more in later tutorials.
As mentioned in the introduction to this set of tutorials, this game will be built using art from Tyrian as provided by Dan Cook. For simplicity these tutorials provide a subset of the Tyrian art. Modifications have been made only to remove unused parts of various images, to add transparency, and to separate images into separate files for convenience. Otherwise all art is in its original state. If you finish this tutorial and would like to continue developing this game, visit https://lostgarden.com/2007/04/05/free-game-graphics-tyrian-ships-and-tiles/ for the full set of Tyrian art.
For the ship, save the following four files to any location on your computer - remember where you save them as you will need to be able to find them:
We will use four files to support up to four players at the same time. Note that we are using four separate images, one for each player. Games often combine images into a single file which is called a sprite sheet. FlatRedBall supports working with individual files and sprite sheets.
Next we will add these four ship files to the Player entity. To do this:
Expand the Player Entity in FRB
Drag+drop the four files onto the Files folder in your entity
You should now see the files as part of your Player Entity.
Where are my files located? If you saved the .png files in the project's Content folder, then FlatRedBall references the .png files from the same location. If you saved the files outside of this folder (such as the Desktop or Downloads folder), then FRB copies them into a folder specific to the Player Entity. If you ever need to find the files, you can right-click on the files in and select View in explorer.
When we created our project using the wizard, we kept the option selected to add a Sprite to the Player Entity. We can see that this exists by expanding the Objects folder:
Our Player contains multiple files, so we need to tell the Sprite which Texture (.png) to use. To do this:
Select the newly-created SpriteInstance under the Player entity
Select the Variables tab
Use the Texture drop-down to select MainShip1 (or whatever you named your PNG when you saved it earlier)
If we run the game now, we will see our ship displaying the MainShip1 Texture on its Sprite.
At this point our Player instance in GameScreen is using a Sprite defined in Glue. However, nothing is happening in our game so far. The next tutorial will add some simple behavior to our Player Entity.
So far we have created an simple project called Rock Blaster. Next we will create a skeleton for our game.
When we refer to a game skeleton, we mean an initial setup which contains empty or nearly-empty Screens and Entities. We are not referring to an actual skeleton, but rather the simplified structure that a skeleton implies.
Creating a skeleton is a great exercise because it can quickly get you to think about what your game will contain. You can add Screens and Entities as you think them up because there is no implementation required. As you become more experienced with making games (especially with FlatRedBall) you will find it easier to create skeletons.
The first step is purely conceptual. You can start creating a skeleton with any tool that you find comfortable. You may prefer to use a simple text or spreadsheet document, or perhaps you prefer to write a list on a piece of paper. The point of this step is to create a list of screens and entities which you expect to include in your game.
This game will be about flying a space ship, shooting rocks, and attempting to stay alive long enough to earn a high score. The Wizard has already taken care of creating our screens (GameScreen and Level1) so we don't need to list Screens as we think about our game skeleton. We'll create a list of Entities needed for our game:
Player (already created by the Wizard Wizard)
Rock
Bullet
Hud (will be made in Gum)
HealthBar (will be made in Gum)
EndGameUi (will be made in Gum)
If you don't think of all of the Screens and Entities that you'll need, that's okay! The purpose of this isn't to fully define the game, but rather to get you to think about the game from a development perspective - something which you may not do immediately when you think up a game idea. Working through the Screen and Entity list may help you realize things that you may need.
Now that we created a list of things that we need in our game, we will start by adding the entities.
Remember, you are modifying your Visual Studio Project. All of the work that you do in the FlatRedBall Editor results mainly in code being generated and your Visual Studio project (.csproj) being modified. Normally this "just works" - when you add things to FRB, they automatically show up in Visual Studio. However, keep in mind that any changes made in Visual Studio (such as adding new classes) will not be picked up by FlatRedBall until you save your project. It's best to get in the habit of saving your Visual Studio project after you add any new files.
For now, we'll work in the FRB Editor to add our entities. Let's start with the Rock entity:
Select the Quick Actions tab in FlatRedBall
Click Add Entity
Enter the name Rock
Check the Circle option. We check this option to identify that our Rock entity can collide with other objects (such as our Player) and that the collision shape is a circle.
Leave the other defaults checked
Click OK
To add the Bullet entity, repeat the steps above, but this time name the entity Bullet. Otherwise, all options should be the same including Circle collision.
Now you should have two new entities - Bullet and Rock.
You may have noticed that the window for creating new entities has two options checked
Create Factory
Include lists in GameScreen
Both of these options are checked by default because they are used in most games.
The Create Factory option results in FlatRedBall generating a Factory object. Factories are used to simplify the creation of new instances of an entity. They are responsible for adding new instances to their necessary lists, providing events for when new objects are created, and can improve the performance of your game through pooling and better sorting. We will be using Factories in later tutorials when we create bullet Bullet and Rock instances.
The Include lists in GameScreen option results in lists being added automatically to the GameScreen. If you are creating an entity which will collide with other entities, this is almost always handled through lists in GameScreen. We can verify that our GameScreen now has lists by expanding its Objects folder.
Now that we have created two new entities (Rock and Bullet), we have the basis for a game. We'll continue in the next tutorial by working on the Player entity.
At this point we have a ship which is visible on Screen, but it doesn't do anything. This tutorial adds behavior to our Player Entity so that it can move, turn, and shoot.
For this game, the Player will continuously move forward at a constant speed. The Player object will be turned left and right with the keyboard or Xbox gamepad. Before we begin writing any code we'll add two variables to Player: MovementSpeed and TurningSpeed. To do this:
Select the Player entity
Select the Variables tab
Click the Create a new variable button\
Leave the defaults Create a new variable option and float type
Enter the name MovementSpeed
Click OK
Repeat the steps above to also add a TurningSpeed variable.
Next let's give the variables some default values:
Enter a value of 100 for MovementSpeed. This is the number of pixels the Player will travel in one second.
Enter a value of 3.14 for TurningSpeed. This is the maximum number of radians the Player will rotate in one second.
To apply movement we will need to write some C# code. To do this:
Open the project in Visual Studio (or switch to Visual Studio if you already have it open)
Open Player.cs. This is in the Entities folder in the Solution Explorer.
Scroll to the CustomActivity method in Player.cs
Modify CustomActivity as shown in the following snippet:
If you now run the game you will see the ship move upward, then eventually move off-screen.
If you are unfamiliar with the RotationMatrix property, or with matrices in general then you may be wondering about the RotationMatrix.Up variable, and why we're using it. The RotationMatrix property contains information about how an Entity, Sprite, or any other PositionedObject is rotated. The Up property indicates you which way is "up" for the object given its current rotation. This value is in "object space" meaning that if the object rotates, then this value will rotate along with the object. This is especially convenient for this tutorial because this game will have the ships always moving forward. The code above will work regardless of which way the Player is rotated - something which we'll see in the coming sections.
The next step is to assign input logic so the Player can turn. We will add an object to our Player representing the input device. This could be a gampad, keyboard, or any other object. By using the I1DInput interface, we can write the code the same regardless of the actual hardware used to control the Player. Modify Player.cs as shown in the following snippet:
Note that all of the code we have written uses coefficients (MovementSpeed and TurningSpeed) defined in the FRB Editor. This means that you can modify these values in the Player Entity at any time if you want to tune how the game feels. For example, if you want the ship to turn faster, increase TurningSpeed to a larger value.
Also, keep in mind that the variables MovementSpeed and TurningSpeed can be modified both in the FRB Editor and also in code - so where should you make the change? Typically, these variables can be thought of as variables which a game designer might edit. Larger teams may include designers who are not as comfortable making changes in code. The FlatRedBall Editor provides a less-technical environment so that designers can make changes and see them in-game without venturing into complex C# code.
But you might be wondering if this is important in your case - after all you are likely just learning to use FlatRedBall, so you are a team of one. You are both the designer and programmer, so where should the changes be made? Even in this situation, the answer is the same - if you can make the changes in the FlatRedBall Editor, do so! This makes it easier to tune your game in one place instead of needing to hop around different code files.
Next we will add a file and Sprite to the Bullet entity. This process is essentially the same as when we added PNG files and a Sprite to our Player Entity so you may find these steps familiar. To add the PNG:
Expand the Bullet entity in Glue
Drag+drop Bullet1.png onto the Files folder in the Bullet
To add a Sprite to the Bullet entity:
Select the Bullet entity in Glue
Click the Quick Actions tab
Click the Add Object to Bullet button
Select the Sprite type
Click OK
Now we can set the Sprite's Texture:
Expand the Bullet Entity's Object folder
Select the newly-created SpriteInstance
Select the Variables tab
Set the Texture drop-down to Bullet1
The next step is to add firing bullets. We'll be using the BulletFactory which we created in an earlier tutorial to create a new bullet and automatically add it to the GameScreen's BulletList. For more information on factories, see the page on this topic. Next we will need to define a bullet speed. To do this:
Click the Bullet entity in Glue
Click the Variables tab
Click the Add New Variable button
Leave the defaults
Set the variable name to MovementSpeed
Set MovementSpeed to 300
Now we can use BulletFactory to create bullets when the player shoots. To do this:
Go to Player.cs in Visual Studio
Modify the Player.cs code so that it contains the following code:
Notice that we are using the RotationMatrix of the MainShip to adjust the initial positions of the bullets, as well as their velocity
If you run the game you should be able to fly, turn, and shoot.
Although we have a long way to go, this is a big milestone for Rock Blaster. You can now see how the game feels for the very first time. Since relevant coefficients are set in Glue, you can change the values to make the game feel differently. Now that we can shoot bullets we'll need something to shoot at. The next tutorial will add Rock entity instances to the GameScreen.
Download the following file to your computer: