A while ago, I was expanding a game framework of my own in my spare time, and I found a design problem in the process of implementing it. This game framework is based onMonoGameimplementation, in MonoGame all texture rendering (Texture Rendering) is done via theSpriteBatch
class to accomplish this. For example, if you want to display an imageTexture somewhere on the screen, you would add a class to theGame
of a subclass of the classDraw
method, use the following code to draw the image:
protected override void Draw(GameTime gameTime)
{
// ...
(imageTexture, new Vector2(x, y), );
// ...
}
Then if you want to display a string in a certain font somewhere on the screen, similarly call theSpriteBatch
(used form a nominal expression)DrawString
method to accomplish this:
protected override void Draw(GameTime gameTime)
{
// ...
(spriteFont, "Hello World", new Vector2(x, y), );
// ...
}
For the time being, you can leave the two code inspriteBatch
How the object is initialized, andDraw
cap (a poem)DrawString
What the individual parameters of the two methods mean is, for the purposes of this discussion, only of interest to thespriteFont
MonoGame uses a technique called Content Pipeline to compile various resources (sounds, music, fonts, textures, etc.) into thexnb
file, after which it is passed through theContentManager
class that reads these resources into memory and creates the corresponding objects.SpriteFont
is one of the resource (font) objects in theGame
(used form a nominal expression)Load
method, you can specify thexnb
The filename is derived from theContentManager
Get font information:
private SpriteFont? spriteFont;
protected override void LoadContent()
{
// ...
spriteFont = <SpriteFont>("fonts\\arial"); // Load from fonts\\
// ...
}
OK, that's all there is to know related to MonoGame. Next, let's get into the specifics. Since I'm doing a game development framework, in order to make it easier to display strings on the screen (in the current scene, to be exact), I've wrapped up aLabel
class, which is shown roughly as follows:
public class Label : VisibleComponent
{
private readonly SpriteFont _spriteFont;
public Label(string text, SpriteFont spriteFont, Vector2 pos, Color color)
{
Text = text;
_spriteFont = spriteFont;
Position = pos;
TextColor = color;
}
public string Text { get; set; }
public Vector2 Position { get; set; }
public Color TextColor { get; set; }
protected override void ExecuteDraw(GameTime gameTime, SpriteBatch spriteBatch)
=> (_spriteFont, Text, Position, TextColor);
}
There is nothing inherently wrong with this realization, but on reflection it is easy to see that theSpriteFont
The font information is read from the Content Pipeline, and the font information contains not only the font name, but also the font size (font size), and in the Pipeline compilation time has been determined, so if the game wants to use the same font with different font sizes to display different strings, you need to load multiple SpriteFont, which is not only troublesome, but also resource-consuming, and not flexible. This is not only troublesome but also resource-consuming and not very flexible.
After some searching, I found an open source font rendering library:FontStashSharpIt has a MonoGame extension that can dynamically load font objects based on different font sizes (called "Dynamic Sprite Fonts").DynamicSpriteFont
"), then use MonoGame's nativeSpriteBatch
Displays the string in the scene in the specified dynamic font, for example:
private readonly FontSystem _fontSystem = new();
private DynamicSpriteFont? _menuFont;
public override void Load(ContentManager contentManager)
{
// Fonts
_fontSystem.AddFont(("res/"));
_menuFont = _fontSystem.GetFont(30);
}
public override void Draw(GameTime gameTime, SpriteBatch spriteBatch)
{
(_menuFont, "Hello World", new Vector2(100, 100), );
}
upperDraw
method, which still uses themethod to display the string, the difference being that this
DrawString
The first parameter accepted by the method isDynamicSpriteFont
object, thisDynamicSpriteFont
The object is provided by a third-party library, FontStashSharp, and it's not a type found in the standard MonoGame, so there are two possibilities here:
-
DynamicSpriteFont
It's in MonoGame.SpriteFont
subclass of - FontStashSharp uses a C# extension method for the
SpriteBatch
type has been extended so that theDrawString
The method can be accessed using theDynamicSpriteFont
to draw text
If it's the first possibility, then the problem is simple, basically, you can develop this game framework by yourself without any modification, for example, in the creation of theLabel
instances, the second argument to the constructor directly sets theDynamicSpriteFont
object is passed in just fine. But unfortunately, this falls into the second category here, which is the FontStashSharpDynamicSpriteFont
together withSpriteFont
There is no inheritance relationship between
Now to summarize, the current state of affairs is:
-
DynamicSpriteFont
is notSpriteFont
subclass of - Both offer similar capabilities: both are capable of being
SpriteBatch
are used to draw text, both are able to calculate the width and height of the drawing area based on a given text string (both provide aMeasureString
(Methods) - I would like to be able to use both in my game frame
SpriteFont
cap (a poem)DynamicSpriteFont
That is, I want Label to be compatible with both theSpriteFont
cap (a poem)DynamicSpriteFont
text-drawing capabilities
Obviously, it's possible to use GoF95'sAdapter modeto solve the current problem to fulfill the conditions of 3 above. For this purpose, it is possible to define aIFontAdapter
interface, and then based on theSpriteFont
respond in singingDynamicSpriteFont
to provide two different adapter implementations, and finally, let the types in the framework (such as theLabel
) is dependent onIFontAdapter
interface is sufficient, the UML class diagram is roughly as follows:
DynamicSpriteFontAdapter
is implemented in a separate package (Assembly in C#), the purpose of this is to prevent the project from having a direct dependency on FontStashSharp, because as the core component of the whole game framework, it will be referenced by different game bodies or other components that don't need to depend on FontStashSharp.
In addition, it is also possible to use C#'s extension method feature to make theSpriteBatch
This can be done based on theIFontAdapter
Perform text drawing:
public static class SpriteBatchExtensions
{
public static void DrawString(
this SpriteBatch spriteBatch,
IFontAdapter fontAdapter,
string text) => (spriteBatch, text);
}
Other related codes are similar as below:
public interface IFontAdapter
{
void DrawString(SpriteBatch spriteBatch, string text);
Vector2 MeasureString(string text);
}
public sealed class SpriteFontAdapter(SpriteFont spriteFont) : IFontAdapter
{
public Vector2 MeasureString(string text) => (text);
public void DrawString(SpriteBatch spriteBatch, string text)
=> (spriteFont, text);
}
public sealed class FontStashSharpAdapter(DynamicSpriteFont spriteFont) : IFontAdapter
{
public void DrawString(SpriteBatch spriteBatch, string text)
=> (spriteFont, text);
public Vector2 MeasureString(string text) => (text);
}
public class Label(string text, IFontAdapter fontAdapter) : VisibleComponent
{
// Other members ignore
public string Text { get; set; } = text;
protected override void ExecuteDraw(GameTime gameTime, SpriteBatch spriteBatch)
=> (fontAdapter, Text);
}
To summarize:This article discusses the application of the Adapter pattern of GoF95 design patterns in real projects through the analysis of a real case, showing how object-oriented design patterns can be used to solve real problems.The introduction of the Adapter pattern also produces some boundary effects, such as in this case FontStashSharp'sDynamicSpriteFont
In fact, it can also provide more rich functionality, but the use of the Adapter pattern, so that these features can not be fully used by the homemade game framework (because the interface is unified, and the standard SpriteFont does not provide these features), an effective solution is to extend theIAdapter
interface's responsibilities, and then use theempty object modelto complement the functionality of an adapter is not supported in the features, but this approach will be in the framework design, so that certain types of hierarchical design to become specialization, that is, in order to cater to an external framework to do the abstraction, so that the design has become less pure, so, still need to be based on the needs of the actual project to determine the design of the way.