#region --- License --- /* Copyright (c) 2006, 2007 Stefanos Apostolopoulos * See license.txt for license info */ #endregion using System; using System.Collections.Generic; using System.Text; using System.Drawing.Text; using System.Drawing; using System.Drawing.Imaging; using System.Runtime.InteropServices; using System.Diagnostics; using OpenTK.Math; using OpenTK.Graphics.OpenGL; using OpenTK.Graphics.OpenGL.Enums; using OpenTK.Platform; namespace OpenTK.Fonts { using Graphics = System.Drawing.Graphics; using PixelFormat = OpenTK.Graphics.OpenGL.PixelFormat; public class TextureFont : IFont { Font font; Dictionary loaded_glyphs = new Dictionary(64); Bitmap bmp; Graphics gfx; // TODO: We need to be able to use multiple font sheets. static int texture; static TexturePacker pack; static int texture_width, texture_height; int[] data = new int[256]; // Used to upload glyph data to the OpenGL texture. object upload_lock = new object(); #region --- Constructor --- /// /// Constructs a new TextureFont object, using the specified System.Drawing.Font. /// /// The System.Drawing.Font to use. public TextureFont(Font font) { if (font == null) throw new ArgumentNullException("font", "Argument to TextureFont constructor cannot be null."); this.font = font; bmp = new Bitmap(font.Height * 2, font.Height * 2); gfx = Graphics.FromImage(bmp); // Adjust font rendering mode. Small sizes look blurry without gridfitting, so turn // that on. Increasing contrast also seems to help. if (font.Size <= 18.0f) { gfx.TextRenderingHint = TextRenderingHint.AntiAliasGridFit; gfx.TextContrast = 1; } else { gfx.TextRenderingHint = TextRenderingHint.AntiAlias; gfx.TextContrast = 0; } } #endregion #region private void PrepareTexturePacker() /// /// Calculates the optimal size for the font texture and TexturePacker, and creates both. /// 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(texture_width, texture_height); GL.GenTextures(1, out texture); GL.BindTexture(TextureTarget.Texture2D, texture); GL.TexParameter(TextureTarget.Texture2D, TextureParameterName.TextureMinFilter, (int)All.Linear); GL.TexParameter(TextureTarget.Texture2D, TextureParameterName.TextureMagFilter, (int)All.Linear); GL.TexImage2D(TextureTarget.Texture2D, 0, PixelInternalFormat.Alpha, texture_width, texture_height, 0, PixelFormat.Rgba, PixelType.UnsignedByte, IntPtr.Zero); } #endregion #region public void LoadGlyphs(string glyphs) /// /// Prepares the specified glyphs for rendering. /// /// The glyphs to prepare for rendering. public void LoadGlyphs(string glyphs) { Box2 rect = new Box2(); foreach (char c in glyphs) { if (!loaded_glyphs.ContainsKey(c)) LoadGlyph(c, out rect); } } #endregion #region public void LoadGlyph(char glyph) /// /// Prepares the specified glyph for rendering. /// /// The glyph to prepare for rendering. public void LoadGlyph(char glyph) { Box2 rect = new Box2(); if (!loaded_glyphs.ContainsKey(glyph)) LoadGlyph(glyph, out rect); } #endregion #region private void LoadGlyph(char c, out Box2 rectangle) /// /// Adds a glyph to the texture packer. /// /// The character of the glyph. /// An OpenTK.Math.Box2 that will hold the data for this glyph. private void LoadGlyph(char c, out Box2 rectangle) { if (pack == null) PrepareTexturePacker(); Glyph g = new Glyph(c, font); Rectangle rect = new Rectangle(); try { pack.Add(g, out rect); } catch (InvalidOperationException expt) { // TODO: The TexturePacker is full, create a new font sheet. Trace.WriteLine(expt); throw; } GL.BindTexture(TextureTarget.Texture2D, texture); gfx.Clear(System.Drawing.Color.Transparent); gfx.DrawString(g.Character.ToString(), g.Font, System.Drawing.Brushes.White, 0.0f, 0.0f); //BitmapData bitmap_data = bitmap.LockBits(new Rectangle(0, 0, rect.Width, rect.Height), ImageLockMode.ReadOnly, // System.Drawing.Imaging.PixelFormat.Format32bppArgb); //GL.TexSubImage2D(TextureTarget.Texture2D, 0, rect.Left, rect.Top, rect.Width, rect.Height, // OpenTK.Graphics.OpenGL.Enums.PixelFormat.Rgba, PixelType.UnsignedByte, bitmap_data.Scan0); //bitmap.UnlockBits(bitmap_data); BitmapData bitmap_data = bmp.LockBits(new Rectangle(0, 0, bmp.Width, bmp.Height), ImageLockMode.ReadOnly, System.Drawing.Imaging.PixelFormat.Format32bppArgb); int needed_size = rect.Width * rect.Height; if (data.Length < needed_size) Array.Resize(ref data, needed_size); Array.Clear(data, 0, needed_size); unsafe { int* bitmap_data_ptr = (int*)bitmap_data.Scan0; for (int y = 0; y < rect.Height; y++) { for (int x = 0; x < rect.Width; x++) { data[y * rect.Width + x] = *(bitmap_data_ptr + y * bmp.Width + x); } } fixed (int* data_ptr = data) GL.TexSubImage2D(TextureTarget.Texture2D, 0, rect.Left, rect.Top, rect.Width, rect.Height, PixelFormat.Rgba, PixelType.UnsignedByte, (IntPtr)data_ptr); } bmp.UnlockBits(bitmap_data); rectangle = new Box2( rect.Left / (float)texture_width, rect.Top / (float)texture_height, rect.Right / (float)texture_width, rect.Bottom / (float)texture_height); loaded_glyphs.Add(g.Character, rectangle); } #endregion #region public bool GlyphData(char glyph, out float width, out float height, out Box2 textureRectangle, out int texture) /// /// Returns the characteristics of a loaded glyph. /// /// The character corresponding to this glyph. /// The width of this glyph. /// The height of this glyph (line spacing). /// The bounding box of the texture data of this glyph. /// The handle to the texture that contains this glyph. /// True if the glyph has been loaded, false otherwise. /// public bool GlyphData(char glyph, out float width, out float height, out Box2 textureRectangle, out int texture) { 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 #region public float Height /// /// Gets a float indicating the default line spacing of this font. /// public float Height { get { return font.Height; } } #endregion #region public float Width /// /// Gets a float indicating the default line spacing of this font. /// public float Width { get { return font.SizeInPoints; } } #endregion #region internal int Texture /// /// Gets the handle to the texture were this font resides. /// internal int Texture { get { return TextureFont.texture; } } #endregion #region public void MeasureString(string str, out float width, out float height, bool accountForOverhangs) /// /// Measures the width of the specified string. /// /// The string to measure. /// The measured width. /// The measured height. /// If true, adds space to account for glyph overhangs. Set to true if you wish to measure a complete string. Set to false if you wish to perform layout on adjacent strings. public void MeasureString(string str, out float width, out float height, bool accountForOverhangs) { System.Drawing.StringFormat format = accountForOverhangs ? System.Drawing.StringFormat.GenericDefault : System.Drawing.StringFormat.GenericTypographic; format.FormatFlags |= StringFormatFlags.MeasureTrailingSpaces; System.Drawing.SizeF size = gfx.MeasureString(str, font, 16384, format); height = size.Height; width = size.Width; // width = 0; // height = 0; // int i = 0; // foreach (char c in str) // { // if (c != '\n' && c != '\r') // { // SizeF size = gfx.MeasureString(str.Substring(i, 1), font, 16384, System.Drawing.StringFormat.GenericTypographic); // width += size.Width == 0 ? font.SizeInPoints * 0.5f : size.Width; // if (height < size.Height) // height = size.Height; // } // ++i; // } } #endregion #region public void MeasureString(string str, out float width, out float height) /// /// Measures the width of the specified string. /// /// The string to measure. /// The measured width. /// The measured height. /// public void MeasureString(string str, out float width, out float height) { MeasureString(str, out width, out height, true); } #endregion #region --- IDisposable Members --- bool disposed; /// /// Releases all resources used by this OpenTK.Fonts.TextureFont. /// public void Dispose() { GC.SuppressFinalize(this); Dispose(true); } private void Dispose(bool manual) { if (!disposed) { pack = null; if (manual) { GL.DeleteTextures(1, ref texture); font.Dispose(); gfx.Dispose(); } disposed = true; } } ~TextureFont() { Dispose(false); } #endregion } }