2007-11-02 00:17:57 +01:00
|
|
|
|
#region --- License ---
|
|
|
|
|
/* Copyright (c) 2006, 2007 Stefanos Apostolopoulos
|
|
|
|
|
* See license.txt for license info
|
|
|
|
|
*/
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
using System;
|
2007-10-20 12:31:59 +02:00
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Text;
|
2007-11-06 14:30:46 +01:00
|
|
|
|
//using System.Drawing;
|
2007-10-20 12:31:59 +02:00
|
|
|
|
using System.Drawing.Text;
|
|
|
|
|
|
|
|
|
|
using OpenTK.Math;
|
|
|
|
|
using OpenTK.OpenGL;
|
2007-11-04 16:31:09 +01:00
|
|
|
|
using OpenTK.OpenGL.Enums;
|
2007-10-20 12:31:59 +02:00
|
|
|
|
using System.Drawing.Imaging;
|
|
|
|
|
using OpenTK.Platform;
|
|
|
|
|
|
|
|
|
|
namespace OpenTK.Fonts
|
|
|
|
|
{
|
2007-11-02 00:17:57 +01:00
|
|
|
|
public class TextureFont : IFont
|
2007-10-20 12:31:59 +02:00
|
|
|
|
{
|
2007-11-06 14:30:46 +01:00
|
|
|
|
System.Drawing.Font font;
|
|
|
|
|
Dictionary<char, Box2> loaded_glyphs = new Dictionary<char, Box2>(64);
|
2007-11-02 00:17:57 +01:00
|
|
|
|
|
2007-11-06 14:30:46 +01:00
|
|
|
|
System.Drawing.Graphics gfx = System.Drawing.Graphics.FromImage(new System.Drawing.Bitmap(1, 1));
|
2007-10-20 12:31:59 +02:00
|
|
|
|
static int texture;
|
|
|
|
|
static TexturePacker<Glyph> pack;
|
|
|
|
|
static int texture_width, texture_height;
|
2007-10-26 17:55:24 +02:00
|
|
|
|
float[] viewport = new float[6];
|
2007-10-20 12:31:59 +02:00
|
|
|
|
|
2007-11-02 00:17:57 +01:00
|
|
|
|
#region --- Constructor ---
|
|
|
|
|
|
2007-10-26 17:55:24 +02:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Constructs a new TextureFont object, using the specified System.Drawing.Font.
|
|
|
|
|
/// </summary>
|
2007-11-02 00:17:57 +01:00
|
|
|
|
/// <param name="font">The System.Drawing.Font to use.</param>
|
2007-11-06 14:30:46 +01:00
|
|
|
|
public TextureFont(System.Drawing.Font font)
|
2007-10-20 12:31:59 +02:00
|
|
|
|
{
|
|
|
|
|
if (font == null)
|
|
|
|
|
throw new ArgumentNullException("font", "Argument to TextureFont constructor cannot be null.");
|
|
|
|
|
|
|
|
|
|
this.font = font;
|
|
|
|
|
}
|
|
|
|
|
|
2007-11-02 00:17:57 +01:00
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
#region private void PrepareTexturePacker()
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Calculates the optimal size for the font texture and TexturePacker, and creates both.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private void PrepareTexturePacker()
|
|
|
|
|
{
|
|
|
|
|
// Calculate the size of the texture packer. We want a power-of-two size
|
|
|
|
|
// that is less than 1024 (supported in Geforce256-era cards), but large
|
|
|
|
|
// enough to hold at least 256 (16*16) font glyphs.
|
|
|
|
|
// TODO: Find the actual card limits, maybe?
|
|
|
|
|
int size = (int)(font.Size * 16);
|
|
|
|
|
size = (int)System.Math.Pow(2.0, System.Math.Ceiling(System.Math.Log((double)size, 2.0)));
|
|
|
|
|
if (size > 1024)
|
|
|
|
|
size = 1024;
|
|
|
|
|
|
|
|
|
|
texture_width = size;
|
|
|
|
|
texture_height = size;
|
|
|
|
|
pack = new TexturePacker<Glyph>(texture_width, texture_height);
|
|
|
|
|
|
|
|
|
|
GL.GenTextures(1, out texture);
|
2007-11-04 16:31:09 +01:00
|
|
|
|
GL.BindTexture(TextureTarget.Texture2d, texture);
|
|
|
|
|
GL.TexParameter(TextureTarget.Texture2d, TextureParameterName.TextureMinFilter, (int)All.Linear);
|
|
|
|
|
GL.TexParameter(TextureTarget.Texture2d, TextureParameterName.TextureMagFilter, (int)All.Linear);
|
2007-11-02 00:17:57 +01:00
|
|
|
|
|
2007-11-04 16:31:09 +01:00
|
|
|
|
GL.TexImage2D(TextureTarget.Texture2d, 0, PixelInternalFormat.Alpha, texture_width, texture_height, 0,
|
|
|
|
|
OpenTK.OpenGL.Enums.PixelFormat.Rgba, PixelType.UnsignedByte, IntPtr.Zero);
|
2007-11-02 00:17:57 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
#region public void LoadGlyphs(string glyphs)
|
|
|
|
|
|
2007-10-20 12:31:59 +02:00
|
|
|
|
/// <summary>
|
2007-10-26 17:55:24 +02:00
|
|
|
|
/// Prepares the specified glyphs for rendering.
|
2007-10-20 12:31:59 +02:00
|
|
|
|
/// </summary>
|
2007-10-26 17:55:24 +02:00
|
|
|
|
/// <param name="glyphs">The glyphs to prepare for rendering.</param>
|
2007-10-20 12:31:59 +02:00
|
|
|
|
public void LoadGlyphs(string glyphs)
|
|
|
|
|
{
|
2007-11-06 14:30:46 +01:00
|
|
|
|
Box2 rect = new Box2();
|
2007-10-20 12:31:59 +02:00
|
|
|
|
foreach (char c in glyphs)
|
|
|
|
|
{
|
|
|
|
|
if (!loaded_glyphs.ContainsKey(c))
|
2007-11-06 14:30:46 +01:00
|
|
|
|
LoadGlyph(c, out rect);
|
2007-10-20 12:31:59 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2007-11-02 00:17:57 +01:00
|
|
|
|
#endregion
|
|
|
|
|
|
2007-11-06 14:30:46 +01:00
|
|
|
|
#region private void LoadGlyph(char c, out Box2 rectangle)
|
2007-11-02 00:17:57 +01:00
|
|
|
|
|
2007-11-06 14:30:46 +01:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Adds a glyph to the texture packer.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="c">The character of the glyph.</param>
|
|
|
|
|
/// <param name="rectangle">A RectangleF that will hold the data for this glyph.</param>
|
|
|
|
|
private void LoadGlyph(char c, out Box2 rectangle)
|
2007-10-20 12:31:59 +02:00
|
|
|
|
{
|
2007-10-26 17:55:24 +02:00
|
|
|
|
if (pack == null)
|
|
|
|
|
PrepareTexturePacker();
|
|
|
|
|
|
2007-10-20 12:31:59 +02:00
|
|
|
|
Glyph g = new Glyph(c, font);
|
2007-11-06 14:30:46 +01:00
|
|
|
|
System.Drawing.Rectangle rect = pack.Add(g);
|
2007-10-20 12:31:59 +02:00
|
|
|
|
|
2007-11-06 14:30:46 +01:00
|
|
|
|
using (System.Drawing.Bitmap bmp = new System.Drawing.Bitmap(
|
|
|
|
|
rect.Width, rect.Height, System.Drawing.Imaging.PixelFormat.Format32bppArgb))
|
|
|
|
|
using (System.Drawing.Graphics gfx = System.Drawing.Graphics.FromImage(bmp))
|
2007-10-20 12:31:59 +02:00
|
|
|
|
{
|
2007-11-06 14:30:46 +01:00
|
|
|
|
// Upload texture data.
|
2007-11-04 16:31:09 +01:00
|
|
|
|
GL.BindTexture(TextureTarget.Texture2d, texture);
|
2007-10-26 17:55:24 +02:00
|
|
|
|
|
2007-11-02 00:17:57 +01:00
|
|
|
|
//gfx.TextRenderingHint = System.Drawing.Text.TextRenderingHint.ClearTypeGridFit;
|
2007-10-26 17:55:24 +02:00
|
|
|
|
//gfx.TextRenderingHint = System.Drawing.Text.TextRenderingHint.AntiAliasGridFit;
|
2007-11-02 00:17:57 +01:00
|
|
|
|
gfx.TextRenderingHint = System.Drawing.Text.TextRenderingHint.AntiAlias;
|
2007-11-06 14:30:46 +01:00
|
|
|
|
gfx.TextContrast = 0;
|
|
|
|
|
gfx.Clear(System.Drawing.Color.Transparent);
|
|
|
|
|
gfx.DrawString(g.Character.ToString(), g.Font, System.Drawing.Brushes.White, 0.0f, 0.0f);
|
2007-10-20 12:31:59 +02:00
|
|
|
|
|
2007-11-06 14:30:46 +01:00
|
|
|
|
BitmapData bmp_data = bmp.LockBits(new System.Drawing.Rectangle(0, 0, rect.Width, rect.Height), ImageLockMode.ReadOnly,
|
2007-11-04 16:31:09 +01:00
|
|
|
|
System.Drawing.Imaging.PixelFormat.Format32bppArgb);
|
|
|
|
|
GL.TexSubImage2D(TextureTarget.Texture2d, 0, rect.Left, rect.Top, rect.Width, rect.Height,
|
|
|
|
|
OpenTK.OpenGL.Enums.PixelFormat.Rgba, PixelType.UnsignedByte, bmp_data.Scan0);
|
2007-11-02 00:17:57 +01:00
|
|
|
|
bmp.UnlockBits(bmp_data);
|
|
|
|
|
|
2007-11-06 14:30:46 +01:00
|
|
|
|
rectangle = new Box2(
|
2007-11-02 00:17:57 +01:00
|
|
|
|
rect.Left / (float)texture_width,
|
|
|
|
|
rect.Top / (float)texture_height,
|
|
|
|
|
rect.Right / (float)texture_width,
|
|
|
|
|
rect.Bottom / (float)texture_height);
|
2007-10-20 12:31:59 +02:00
|
|
|
|
|
2007-11-06 14:30:46 +01:00
|
|
|
|
loaded_glyphs.Add(g.Character, rectangle);
|
2007-10-20 12:31:59 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2007-11-02 00:17:57 +01:00
|
|
|
|
#endregion
|
|
|
|
|
|
2007-11-06 14:30:46 +01:00
|
|
|
|
#region public bool GlyphData(char glyph, out float width, out float height, out Box2 textureRectangle, out int texture)
|
2007-11-02 00:17:57 +01:00
|
|
|
|
|
2007-11-06 14:30:46 +01:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Returns the characteristics of a loaded glyph.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="glyph">The character corresponding to this glyph.</param>
|
|
|
|
|
/// <param name="width">The width of this glyph.</param>
|
|
|
|
|
/// <param name="height">The height of this glyph (line spacing).</param>
|
|
|
|
|
/// <param name="textureRectangle">The bounding box of the texture data of this glyph.</param>
|
|
|
|
|
/// <param name="texture">The handle to the texture that contains this glyph.</param>
|
|
|
|
|
/// <returns>True if the glyph has been loaded, false otherwise.</returns>
|
|
|
|
|
/// <seealso cref="LoadGlyphs"/>
|
|
|
|
|
public bool GlyphData(char glyph, out float width, out float height, out Box2 textureRectangle, out int texture)
|
2007-11-02 00:17:57 +01:00
|
|
|
|
{
|
|
|
|
|
if (loaded_glyphs.TryGetValue(glyph, out textureRectangle))
|
|
|
|
|
{
|
|
|
|
|
width = textureRectangle.Width * texture_width;
|
|
|
|
|
height = textureRectangle.Height * texture_height;
|
|
|
|
|
texture = TextureFont.texture;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
width = height = texture = 0;
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
2007-11-06 14:30:46 +01:00
|
|
|
|
#region public float Height
|
2007-11-02 00:17:57 +01:00
|
|
|
|
|
2007-10-26 17:55:24 +02:00
|
|
|
|
/// <summary>
|
2007-11-06 14:30:46 +01:00
|
|
|
|
/// Gets a float indicating the default line spacing of this font.
|
2007-10-26 17:55:24 +02:00
|
|
|
|
/// </summary>
|
2007-11-06 14:30:46 +01:00
|
|
|
|
public float Height
|
2007-10-26 17:55:24 +02:00
|
|
|
|
{
|
2007-11-06 14:30:46 +01:00
|
|
|
|
get { return font.Height; }
|
2007-10-20 12:31:59 +02:00
|
|
|
|
}
|
|
|
|
|
|
2007-11-02 00:17:57 +01:00
|
|
|
|
#endregion
|
|
|
|
|
|
2007-11-06 14:30:46 +01:00
|
|
|
|
#region internal int Texture
|
2007-10-20 12:31:59 +02:00
|
|
|
|
|
2007-10-26 17:55:24 +02:00
|
|
|
|
/// <summary>
|
2007-11-06 14:30:46 +01:00
|
|
|
|
/// Gets the handle to the texture were this font resides.
|
2007-10-26 17:55:24 +02:00
|
|
|
|
/// </summary>
|
2007-11-06 14:30:46 +01:00
|
|
|
|
internal int Texture
|
2007-10-26 17:55:24 +02:00
|
|
|
|
{
|
2007-11-06 14:30:46 +01:00
|
|
|
|
get { return TextureFont.texture; }
|
2007-10-26 17:55:24 +02:00
|
|
|
|
}
|
|
|
|
|
|
2007-11-02 00:17:57 +01:00
|
|
|
|
#endregion
|
|
|
|
|
|
2007-11-06 14:30:46 +01:00
|
|
|
|
#region public void MeasureString(string str, out float width, out float height)
|
2007-11-02 00:17:57 +01:00
|
|
|
|
|
2007-10-26 17:55:24 +02:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Measures the width of the specified string.
|
|
|
|
|
/// </summary>
|
2007-11-06 14:30:46 +01:00
|
|
|
|
/// <param name="str">The string to measure.</param>
|
|
|
|
|
/// <param name="width">The measured width.</param>
|
|
|
|
|
/// <param name="height">The measured height.</param>
|
|
|
|
|
public void MeasureString(string str, out float width, out float height)
|
2007-10-26 17:55:24 +02:00
|
|
|
|
{
|
2007-11-06 14:30:46 +01:00
|
|
|
|
System.Drawing.SizeF size = gfx.MeasureString(str, font, 16384, System.Drawing.StringFormat.GenericTypographic);
|
2007-11-02 00:17:57 +01:00
|
|
|
|
if (size.Width == 0)
|
2007-11-06 14:30:46 +01:00
|
|
|
|
width = font.SizeInPoints * 0.5f;
|
|
|
|
|
else
|
|
|
|
|
width = size.Width;
|
|
|
|
|
height = size.Height;
|
2007-10-26 17:55:24 +02:00
|
|
|
|
}
|
2007-10-20 12:31:59 +02:00
|
|
|
|
|
2007-11-02 00:17:57 +01:00
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
#region --- IDisposable Members ---
|
2007-10-20 12:31:59 +02:00
|
|
|
|
|
2007-10-26 17:55:24 +02:00
|
|
|
|
bool disposed;
|
2007-10-20 12:31:59 +02:00
|
|
|
|
|
2007-10-26 17:55:24 +02:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Releases all resources used by this OpenTK.Fonts.TextureFont.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public void Dispose()
|
|
|
|
|
{
|
|
|
|
|
GC.SuppressFinalize(this);
|
|
|
|
|
Dispose(true);
|
2007-10-20 12:31:59 +02:00
|
|
|
|
}
|
|
|
|
|
|
2007-10-26 17:55:24 +02:00
|
|
|
|
private void Dispose(bool manual)
|
2007-10-20 12:31:59 +02:00
|
|
|
|
{
|
2007-10-26 17:55:24 +02:00
|
|
|
|
if (!disposed)
|
|
|
|
|
{
|
|
|
|
|
pack = null;
|
|
|
|
|
if (manual)
|
|
|
|
|
{
|
|
|
|
|
GL.DeleteTextures(1, ref texture);
|
|
|
|
|
font.Dispose();
|
|
|
|
|
gfx.Dispose();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
disposed = true;
|
|
|
|
|
}
|
2007-10-20 12:31:59 +02:00
|
|
|
|
}
|
|
|
|
|
|
2007-10-26 17:55:24 +02:00
|
|
|
|
~TextureFont()
|
2007-10-20 12:31:59 +02:00
|
|
|
|
{
|
2007-10-26 17:55:24 +02:00
|
|
|
|
Dispose(false);
|
2007-10-20 12:31:59 +02:00
|
|
|
|
}
|
2007-10-26 17:55:24 +02:00
|
|
|
|
|
|
|
|
|
#endregion
|
2007-10-20 12:31:59 +02:00
|
|
|
|
}
|
|
|
|
|
}
|