437 lines
12 KiB
C#
437 lines
12 KiB
C#
#region Copyright & License Information
|
|
/*
|
|
* Copyright (c) The OpenRA Developers and Contributors
|
|
* This file is part of OpenRA, which is free software. It is made
|
|
* available to you under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation, either version 3 of
|
|
* the License, or (at your option) any later version. For more
|
|
* information, see COPYING.
|
|
*/
|
|
#endregion
|
|
|
|
using System;
|
|
using OpenRA.Primitives;
|
|
using OpenRA.Support;
|
|
|
|
namespace OpenRA.Graphics
|
|
{
|
|
public sealed class SpriteFont : IDisposable
|
|
{
|
|
public int TopOffset { get; }
|
|
readonly int size;
|
|
readonly SheetBuilder builder;
|
|
readonly IFont font;
|
|
readonly Cache<char, GlyphInfo> glyphs;
|
|
readonly Cache<(char C, int Radius), Sprite> contrastGlyphs;
|
|
readonly Cache<int, float[]> dilationElements;
|
|
|
|
float deviceScale;
|
|
|
|
public SpriteFont(string name, byte[] data, int size, int ascender, float scale, SheetBuilder builder)
|
|
{
|
|
if (builder.Type != SheetType.BGRA)
|
|
throw new ArgumentException("The sheet builder must create BGRA sheets.", nameof(builder));
|
|
|
|
deviceScale = scale;
|
|
this.size = size;
|
|
this.builder = builder;
|
|
|
|
font = Game.Renderer.CreateFont(data);
|
|
glyphs = new Cache<char, GlyphInfo>(CreateGlyph);
|
|
contrastGlyphs = new Cache<(char, int), Sprite>(CreateContrastGlyph);
|
|
dilationElements = new Cache<int, float[]>(CreateCircularWeightMap);
|
|
|
|
// Pre-cache small font sizes so glyphs are immediately available when we need them
|
|
if (size <= 24)
|
|
using (new PerfTimer($"Precache {name} {size}px"))
|
|
for (var n = (char)0x20; n < (char)0x7f; n++)
|
|
if (glyphs[n] == null)
|
|
throw new InvalidOperationException();
|
|
|
|
TopOffset = size - ascender;
|
|
}
|
|
|
|
public void SetScale(float scale)
|
|
{
|
|
deviceScale = scale;
|
|
glyphs.Clear();
|
|
contrastGlyphs.Clear();
|
|
}
|
|
|
|
void DrawTextContrast(string text, float2 location, Color contrastColor, int contrastOffset)
|
|
{
|
|
// Offset from the baseline position to the top-left of the glyph for rendering
|
|
location += new float2(0, size);
|
|
|
|
// Calculate positions in screen pixel coordinates
|
|
var screenContrast = (int)(contrastOffset * deviceScale);
|
|
var screen = new int2((int)(location.X * deviceScale + 0.5f), (int)(location.Y * deviceScale + 0.5f));
|
|
var contrastVector = new float2(screenContrast, screenContrast);
|
|
var tint = new float3(contrastColor.R / 255f, contrastColor.G / 255f, contrastColor.B / 255f);
|
|
foreach (var s in text)
|
|
{
|
|
if (s == '\n')
|
|
{
|
|
location += new float2(0, size);
|
|
screen = new int2((int)(location.X * deviceScale + 0.5f), (int)(location.Y * deviceScale + 0.5f));
|
|
continue;
|
|
}
|
|
|
|
var g = glyphs[s];
|
|
|
|
// Convert screen coordinates back to UI coordinates for drawing
|
|
if (g.Sprite != null)
|
|
{
|
|
var contrastSprite = contrastGlyphs[(s, screenContrast)];
|
|
Game.Renderer.RgbaSpriteRenderer.DrawSprite(contrastSprite,
|
|
(screen + g.Offset - contrastVector) / deviceScale,
|
|
1f / deviceScale,
|
|
tint, 1f);
|
|
}
|
|
|
|
screen += new int2((int)(g.Advance + 0.5f), 0);
|
|
}
|
|
}
|
|
|
|
public void DrawText(string text, float2 location, Color c)
|
|
{
|
|
// Offset from the baseline position to the top-left of the glyph for rendering
|
|
location += new float2(0, size);
|
|
|
|
// Calculate positions in screen pixel coordinates
|
|
var screen = new int2((int)(location.X * deviceScale + 0.5f), (int)(location.Y * deviceScale + 0.5f));
|
|
var tint = new float3(c.R / 255f, c.G / 255f, c.B / 255f);
|
|
foreach (var s in text)
|
|
{
|
|
if (s == '\n')
|
|
{
|
|
location += new float2(0, size);
|
|
screen = new int2((int)(location.X * deviceScale + 0.5f), (int)(location.Y * deviceScale + 0.5f));
|
|
continue;
|
|
}
|
|
|
|
var g = glyphs[s];
|
|
|
|
// Convert screen coordinates back to UI coordinates for drawing
|
|
if (g.Sprite != null)
|
|
Game.Renderer.RgbaSpriteRenderer.DrawSprite(g.Sprite,
|
|
(screen + g.Offset).ToFloat2() / deviceScale,
|
|
1f / deviceScale,
|
|
tint, 1f);
|
|
|
|
screen += new int2((int)(g.Advance + 0.5f), 0);
|
|
}
|
|
}
|
|
|
|
float2 Rotate(float2 v, float sina, float cosa, float2 offset)
|
|
{
|
|
return new float2(
|
|
v.X * cosa - v.Y * sina + offset.X,
|
|
v.X * sina + v.Y * cosa + offset.Y);
|
|
}
|
|
|
|
public void DrawText(string text, float2 location, Color c, float angle)
|
|
{
|
|
// Offset from the baseline position to the top-left of the glyph for rendering
|
|
// All positions are calculated in UI coordinates
|
|
var offset = new float2(0, size);
|
|
var cosa = (float)Math.Cos(-angle);
|
|
var sina = (float)Math.Sin(-angle);
|
|
var tint = new float3(c.R / 255f, c.G / 255f, c.B / 255f);
|
|
|
|
var p = offset;
|
|
foreach (var s in text)
|
|
{
|
|
if (s == '\n')
|
|
{
|
|
offset += new float2(0, size);
|
|
p = offset;
|
|
continue;
|
|
}
|
|
|
|
var g = glyphs[s];
|
|
if (g.Sprite != null)
|
|
{
|
|
var tl = new float2(
|
|
p.X + g.Offset.X / deviceScale,
|
|
p.Y + g.Offset.Y / deviceScale);
|
|
var br = tl + g.Sprite.Size.XY / deviceScale;
|
|
var tr = new float2(br.X, tl.Y);
|
|
var bl = new float2(tl.X, br.Y);
|
|
|
|
var ra = Rotate(tl, sina, cosa, location);
|
|
var rb = Rotate(tr, sina, cosa, location);
|
|
var rc = Rotate(br, sina, cosa, location);
|
|
var rd = Rotate(bl, sina, cosa, location);
|
|
|
|
// Offset rotated glyph to align the top-left corner with the screen pixel grid
|
|
var screenOffset = new float2((int)(ra.X * deviceScale + 0.5f), (int)(ra.Y * deviceScale + 0.5f)) / deviceScale - ra;
|
|
Game.Renderer.RgbaSpriteRenderer.DrawSprite(g.Sprite,
|
|
ra + screenOffset,
|
|
rb + screenOffset,
|
|
rc + screenOffset,
|
|
rd + screenOffset,
|
|
tint, 1f);
|
|
}
|
|
|
|
p += new float2(g.Advance / deviceScale, 0);
|
|
}
|
|
}
|
|
|
|
public void DrawTextWithContrast(string text, float2 location, Color fg, Color bg, int offset)
|
|
{
|
|
if (offset > 0)
|
|
DrawTextContrast(text, location, bg, offset);
|
|
|
|
DrawText(text, location, fg);
|
|
}
|
|
|
|
public void DrawTextWithContrast(string text, float2 location, Color fg, Color bgDark, Color bgLight, int offset)
|
|
{
|
|
DrawTextWithContrast(text, location, fg, GetContrastColor(fg, bgDark, bgLight), offset);
|
|
}
|
|
|
|
public void DrawTextWithShadow(string text, float2 location, Color fg, Color bg, int offset)
|
|
{
|
|
if (offset != 0)
|
|
{
|
|
// Shadow offsets are rounded to an integer number of screen pixels.
|
|
// This makes sure the shadow will be positioned consistently everywhere on the screen.
|
|
var screenOffset = (int)(offset * deviceScale) / deviceScale;
|
|
DrawText(text, location + new float2(screenOffset, screenOffset), bg);
|
|
}
|
|
|
|
DrawText(text, location, fg);
|
|
}
|
|
|
|
public void DrawTextWithShadow(string text, float2 location, Color fg, Color bgDark, Color bgLight, int offset)
|
|
{
|
|
DrawTextWithShadow(text, location, fg, GetContrastColor(fg, bgDark, bgLight), offset);
|
|
}
|
|
|
|
public void DrawTextWithShadow(string text, float2 location, Color fg, Color bg, int offset, float angle)
|
|
{
|
|
if (offset != 0)
|
|
{
|
|
// Shadow offsets are rounded to an integer number of screen pixels.
|
|
// This makes sure the shadow will be positioned consistently everywhere on the screen.
|
|
var screenOffset = (int)(offset * deviceScale) / deviceScale;
|
|
DrawText(text, location + new float2(screenOffset, screenOffset), bg, angle);
|
|
}
|
|
|
|
DrawText(text, location, fg, angle);
|
|
}
|
|
|
|
public void DrawTextWithShadow(string text, float2 location, Color fg, Color bgDark, Color bgLight, int offset, float angle)
|
|
{
|
|
DrawTextWithShadow(text, location, fg, GetContrastColor(fg, bgDark, bgLight), offset, angle);
|
|
}
|
|
|
|
public int2 Measure(string text)
|
|
{
|
|
if (string.IsNullOrEmpty(text))
|
|
return new int2(0, size);
|
|
|
|
var lines = text.SplitLines('\n');
|
|
|
|
var maxWidth = 0f;
|
|
var rows = 0;
|
|
foreach (var line in lines)
|
|
{
|
|
rows++;
|
|
maxWidth = Math.Max(maxWidth, LineWidth(line));
|
|
}
|
|
|
|
return new int2((int)Math.Ceiling(maxWidth), rows * size);
|
|
}
|
|
|
|
float LineWidth(ReadOnlySpan<char> line)
|
|
{
|
|
var result = 0f;
|
|
foreach (var c in line)
|
|
result += glyphs[c].Advance;
|
|
|
|
return result / deviceScale;
|
|
}
|
|
|
|
GlyphInfo CreateGlyph(char c)
|
|
{
|
|
var glyph = font.CreateGlyph(c, size, deviceScale);
|
|
if (glyph.Data == null)
|
|
{
|
|
return new GlyphInfo
|
|
{
|
|
Sprite = null,
|
|
Advance = 0,
|
|
Offset = int2.Zero
|
|
};
|
|
}
|
|
|
|
var s = builder.Allocate(glyph.Size);
|
|
var g = new GlyphInfo
|
|
{
|
|
Sprite = s,
|
|
Advance = glyph.Advance,
|
|
Offset = glyph.Offset
|
|
};
|
|
|
|
var dest = s.Sheet.GetData();
|
|
var destStride = s.Sheet.Size.Width * 4;
|
|
|
|
for (var j = 0; j < s.Size.Y; j++)
|
|
{
|
|
for (var i = 0; i < s.Size.X; i++)
|
|
{
|
|
var p = glyph.Data[j * glyph.Size.Width + i];
|
|
if (p != 0)
|
|
{
|
|
var q = destStride * (j + s.Bounds.Top) + 4 * (i + s.Bounds.Left);
|
|
dest[q] = p;
|
|
dest[q + 1] = p;
|
|
dest[q + 2] = p;
|
|
dest[q + 3] = p;
|
|
}
|
|
}
|
|
}
|
|
|
|
s.Sheet.CommitBufferedData();
|
|
|
|
return g;
|
|
}
|
|
|
|
float[] CreateCircularWeightMap(int r)
|
|
{
|
|
// Create circular weight maps that are used by CreateContrastGlyph for
|
|
// both the structuring element and to weight the resulting pixel value.
|
|
// The output is a 2 * r + 1 square array giving the pixel intersection
|
|
// with a circle of radius (r + 0.5).
|
|
//
|
|
// Example output for r=1:
|
|
// 0.60 1.00 0.60
|
|
// 1.00 1.00 1.00
|
|
// 0.60 1.00 0.60
|
|
//
|
|
// Example output for r=3:
|
|
// 0.00 0.44 0.80 1.00 0.80 0.44 0.00
|
|
// 0.44 1.00 1.00 1.00 1.00 1.00 0.44
|
|
// 0.80 1.00 1.00 1.00 1.00 1.00 0.80
|
|
// 1.00 1.00 1.00 1.00 1.00 1.00 1.00
|
|
// 0.80 1.00 1.00 1.00 1.00 1.00 0.80
|
|
// 0.44 1.00 1.00 1.00 1.00 1.00 0.44
|
|
// 0.00 0.44 0.80 1.00 0.80 0.44 0.00
|
|
var stride = 2 * r + 1;
|
|
var elem = new float[stride * stride];
|
|
|
|
for (var j = 0; j <= 2 * r; j++)
|
|
{
|
|
for (var i = 0; i <= 2 * r; i++)
|
|
{
|
|
var di = i - r;
|
|
var dj = j - r;
|
|
|
|
// No intersection with circle
|
|
if (di * di + dj * dj > (r + 1) * (r + 1))
|
|
continue;
|
|
|
|
// Fully contained within circle
|
|
if (di * di + dj * dj < (r - 1) * (r - 1))
|
|
{
|
|
elem[j * stride + i] = 1;
|
|
continue;
|
|
}
|
|
|
|
// Approximate sub-pixel intersection using a 5x5 grid
|
|
for (var jj = 0; jj < 5; jj++)
|
|
{
|
|
for (var ii = 0; ii < 5; ii++)
|
|
{
|
|
var si = di - (float)Math.Sign(di) * ii / 5;
|
|
var sj = dj - (float)Math.Sign(dj) * jj / 5;
|
|
if (si * si + sj * sj <= r * r)
|
|
elem[j * stride + i] += 0.04f;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return elem;
|
|
}
|
|
|
|
Sprite CreateContrastGlyph((char C, int Radius) c)
|
|
{
|
|
var glyph = glyphs[c.C];
|
|
var r = c.Radius;
|
|
|
|
var s = builder.Allocate(new Size(glyph.Sprite.Bounds.Width + 2 * r, glyph.Sprite.Bounds.Height + 2 * r));
|
|
var dest = s.Sheet.GetData();
|
|
var destStride = s.Sheet.Size.Width * 4;
|
|
|
|
var glyphData = glyph.Sprite.Sheet.GetData();
|
|
var glyphStride = glyph.Sprite.Sheet.Size.Width * 4;
|
|
var glyphBounds = glyph.Sprite.Bounds;
|
|
|
|
var elem = dilationElements[r];
|
|
var elemStride = 2 * r + 1;
|
|
|
|
// Expand the glyph by applying the greyscale dilation operator to the source glyph's alpha channel
|
|
for (var j = 0; j < s.Size.Y; j++)
|
|
{
|
|
for (var i = 0; i < s.Size.X; i++)
|
|
{
|
|
// Apply the weight map to the source glyph and find the largest weighted alpha
|
|
var first = true;
|
|
var alpha = (byte)0;
|
|
for (var wj = 0; wj <= 2 * r; wj++)
|
|
{
|
|
for (var wi = 0; wi <= 2 * r; wi++)
|
|
{
|
|
// Ignore pixels that are outside the source glyph bounds
|
|
var ii = i + wi - 2 * r;
|
|
var jj = j + wj - 2 * r;
|
|
if (ii < 0 || ii >= glyphBounds.Width || jj < 0 || jj >= glyphBounds.Height)
|
|
continue;
|
|
|
|
// Weighted alpha for this pixel
|
|
var weighted = (byte)(elem[wj * elemStride + wi] * glyphData[glyphStride * (jj + glyphBounds.Top) + 4 * (ii + glyphBounds.Left) + 3]);
|
|
if (first || weighted > alpha)
|
|
{
|
|
alpha = weighted;
|
|
first = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (alpha > 0)
|
|
{
|
|
var q = destStride * (j + s.Bounds.Top) + 4 * (i + s.Bounds.Left);
|
|
dest[q] = alpha;
|
|
dest[q + 1] = alpha;
|
|
dest[q + 2] = alpha;
|
|
dest[q + 3] = alpha;
|
|
}
|
|
}
|
|
}
|
|
|
|
s.Sheet.CommitBufferedData();
|
|
return s;
|
|
}
|
|
|
|
static Color GetContrastColor(Color fgColor, Color bgDark, Color bgLight)
|
|
{
|
|
return fgColor == Color.White || fgColor.GetBrightness() > 0.33 ? bgDark : bgLight;
|
|
}
|
|
|
|
public void Dispose()
|
|
{
|
|
font.Dispose();
|
|
}
|
|
}
|
|
|
|
class GlyphInfo
|
|
{
|
|
public float Advance;
|
|
public int2 Offset;
|
|
public Sprite Sprite;
|
|
}
|
|
}
|