263 lines
9.4 KiB
C#
Raw Normal View History

#region Using Statements
2011-10-31 05:36:24 +00:00
using System;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using System.Text;
#endregion // Using Statements
// This file is part of the ANX.Framework created by the
// "ANX.Framework developer group" and released under the Ms-PL license.
// For details see: http://anxframework.codeplex.com/license
2011-10-31 05:36:24 +00:00
namespace ANX.Framework.Graphics
{
public sealed class SpriteFont
{
#region Private Members
2011-10-31 05:36:24 +00:00
private Texture2D texture;
private List<Rectangle> glyphs;
private List<Rectangle> cropping;
private List<char> characterMap;
private int lineSpacing;
private float horizontalSpacing;
private List<Vector3> kerning;
private char? defaultCharacter;
private ReadOnlyCollection<Char> characters;
#endregion // Private Members
2011-10-31 05:36:24 +00:00
public ReadOnlyCollection<Char> Characters
{
get { return characters; }
}
public char? DefaultCharacter
{
get { return defaultCharacter; }
set
{
if (value.HasValue && !this.characterMap.Contains(value.Value))
{
throw new NotSupportedException("Character is not in used font");
}
this.defaultCharacter = value;
}
}
public int LineSpacing
{
get { return lineSpacing; }
set { lineSpacing = value; }
}
public float Spacing
{
get { return horizontalSpacing; }
set { horizontalSpacing = value; }
}
internal SpriteFont(
Texture2D texture, List<Rectangle> glyphs, List<Rectangle> cropping, List<char> charMap,
2011-10-31 05:36:24 +00:00
int lineSpacing, float horizontalSpacing, List<Vector3> kerning, char? defaultCharacter)
{
this.texture = texture;
this.glyphs = glyphs;
this.cropping = cropping;
this.characterMap = charMap;
this.lineSpacing = lineSpacing;
this.horizontalSpacing = horizontalSpacing;
this.kerning = kerning;
this.defaultCharacter = defaultCharacter;
this.characters = new ReadOnlyCollection<char>(characterMap);
}
public Vector2 MeasureString(string text)
{
if (text == null)
{
throw new ArgumentNullException("text");
}
return InternalMeasure(ref text);
2011-10-31 05:36:24 +00:00
}
public Vector2 MeasureString(StringBuilder text)
{
if (text == null)
{
throw new ArgumentNullException("text");
}
String cachedText = text.ToString();
return InternalMeasure(ref cachedText);
2011-10-31 05:36:24 +00:00
}
internal void DrawString(ref String text, SpriteBatch spriteBatch, Vector2 textPos, Color color, Vector2 scale, Vector2 origin, float rotation, float layerDepth, SpriteEffects effects)
2011-10-31 05:36:24 +00:00
{
Matrix transformation = Matrix.CreateRotationZ(rotation) * Matrix.CreateTranslation(-origin.X * scale.X, -origin.Y * scale.Y, 0f);
int horizontalFlipModifier = 1;
float width = 0f;
Vector2 topLeft = new Vector2();
bool firstCharacterInLine = true;
if ((effects & SpriteEffects.FlipHorizontally) == SpriteEffects.FlipHorizontally)
{
topLeft.X = width = this.InternalMeasure(ref text).X * scale.X;
horizontalFlipModifier = -1;
}
if ((effects & SpriteEffects.FlipVertically) == SpriteEffects.FlipVertically)
{
topLeft.Y = (this.InternalMeasure(ref text).Y - this.lineSpacing) * scale.Y;
}
2011-10-31 05:36:24 +00:00
for (int i = 0; i < text.Length; i++)
{
char currentCharacter = text[i];
switch (currentCharacter)
{
case '\r':
break;
case '\n':
firstCharacterInLine = true;
topLeft.X = width;
if ((effects & SpriteEffects.FlipVertically) == SpriteEffects.FlipVertically)
{
topLeft.Y -= this.lineSpacing * scale.Y;
}
else
{
topLeft.Y += this.lineSpacing * scale.Y;
}
break;
default:
{
int characterIndex = GetIndexForCharacter(currentCharacter);
Vector3 kerning = this.kerning[characterIndex];
Rectangle glyphRect = this.glyphs[characterIndex];
Rectangle croppingRect = this.cropping[characterIndex];
if (firstCharacterInLine)
{
kerning.X = Math.Max(kerning.X, 0f);
}
else
{
topLeft.X += (this.Spacing * scale.X) * horizontalFlipModifier;
}
topLeft.X += (kerning.X * scale.X) * horizontalFlipModifier;
if ((effects & SpriteEffects.FlipVertically) == SpriteEffects.FlipVertically)
{
croppingRect.Y = (this.lineSpacing - glyphRect.Height) - croppingRect.Y;
}
if ((effects & SpriteEffects.FlipHorizontally) == SpriteEffects.FlipHorizontally)
{
croppingRect.X -= croppingRect.Width;
}
Vector2 position = Vector2.Transform(topLeft + (new Vector2(croppingRect.X, croppingRect.Y) * scale), transformation);
spriteBatch.Draw(this.texture, position + textPos, glyphRect, color, rotation, Vector2.Zero, scale, effects, layerDepth);
firstCharacterInLine = false;
topLeft.X += ((kerning.Y + kerning.Z) * scale.X) * horizontalFlipModifier;
break;
}
}
}
}
private Vector2 InternalMeasure(ref String text)
{
if (text.Length < 1)
{
return Vector2.Zero;
}
2011-10-31 05:36:24 +00:00
Vector2 size = Vector2.Zero;
size.Y = this.lineSpacing;
float maxWidth = 0f;
int currentCharacter = 0;
float z = 0f;
bool firstCharacterInLine = true;
2011-10-31 05:36:24 +00:00
for (int i = 0; i < text.Length; i++)
{
char currentChar = text[i];
2011-10-31 05:36:24 +00:00
if (currentChar == '\r')
{
continue;
}
if (currentChar == '\n')
{
size.X += Math.Max(z, 0f);
z = 0f;
maxWidth = Math.Max(size.X, maxWidth);
size = Vector2.Zero;
size.Y = this.lineSpacing;
firstCharacterInLine = true;
currentCharacter++;
}
else
{
int currentCharIndex = this.GetIndexForCharacter(currentChar);
Vector3 kerning = this.kerning[currentCharIndex];
if (firstCharacterInLine)
{
kerning.X = Math.Max(kerning.X, 0f);
}
else
{
size.X += this.Spacing + z;
}
size.X += kerning.X + kerning.Y;
z = kerning.Z;
size.Y = Math.Max(size.Y, (float)this.cropping[currentCharIndex].Height);
firstCharacterInLine = false;
}
2011-10-31 05:36:24 +00:00
}
size.Y += currentCharacter * this.lineSpacing;
size.X = Math.Max(Math.Max(z, 0) + size.X, maxWidth);
return size;
2011-10-31 05:36:24 +00:00
}
private int GetIndexForCharacter(char character)
{
int currentIndex = 0;
int upperBound = this.characterMap.Count - 1;
char testChar;
int searchPos;
while (currentIndex <= upperBound)
{
searchPos = currentIndex + ((upperBound - currentIndex) >> 1);
testChar = this.characterMap[searchPos];
if (testChar == character)
{
return searchPos;
}
else if (testChar > character)
{
upperBound = searchPos - 1;
}
else
{
currentIndex = searchPos + 1;
}
}
if (this.defaultCharacter.HasValue)
{
return this.GetIndexForCharacter(this.defaultCharacter.Value);
}
throw new ArgumentException("character not found");
}
}
}