2008-11-25 14:04:24 +00:00
|
|
|
#region License
|
|
|
|
//
|
|
|
|
// The Open Toolkit Library License
|
|
|
|
//
|
|
|
|
// Copyright (c) 2006 - 2008 the Open Toolkit library, except where noted.
|
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
// in the Software without restriction, including without limitation the rights to
|
|
|
|
// use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
|
|
|
// the Software, and to permit persons to whom the Software is furnished to do
|
|
|
|
// so, subject to the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be included in all
|
|
|
|
// copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
|
|
|
|
// OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
|
|
|
|
// HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
|
|
|
// WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
|
|
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
|
|
|
// OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
//
|
|
|
|
#endregion
|
|
|
|
|
2008-11-24 16:43:56 +00:00
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Text;
|
|
|
|
using System.Drawing;
|
|
|
|
|
2008-11-25 14:04:24 +00:00
|
|
|
using OpenTK.Math;
|
|
|
|
|
2008-11-24 16:43:56 +00:00
|
|
|
namespace OpenTK.Graphics.Text
|
|
|
|
{
|
|
|
|
class GL1TextOutputProvider : ITextOutputProvider
|
|
|
|
{
|
2008-11-25 14:04:24 +00:00
|
|
|
#region Fields
|
|
|
|
|
2008-11-25 16:45:24 +00:00
|
|
|
// Triangle lists, sorted by texture.
|
|
|
|
Dictionary<Texture2D, List<Vector2>> active_lists = new Dictionary<Texture2D, List<Vector2>>();
|
|
|
|
Queue<List<Vector2>> inactive_lists = new Queue<List<Vector2>>();
|
2008-11-25 14:04:24 +00:00
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
2008-11-24 16:43:56 +00:00
|
|
|
#region Constructors
|
|
|
|
|
2008-11-25 16:45:24 +00:00
|
|
|
public GL1TextOutputProvider()
|
|
|
|
{
|
|
|
|
inactive_lists.Enqueue(new List<Vector2>());
|
|
|
|
}
|
2008-11-24 16:43:56 +00:00
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
#region ITextOutputProvider Members
|
|
|
|
|
2008-11-26 21:49:05 +00:00
|
|
|
public void Print(TextBlock block, PointF location, Color color, IGlyphRasterizer rasterizer, GlyphCache cache)
|
2008-11-24 16:43:56 +00:00
|
|
|
{
|
2008-11-26 21:49:05 +00:00
|
|
|
using (TextExtents extents = rasterizer.MeasureText(block, location))
|
2008-11-26 16:34:50 +00:00
|
|
|
{
|
|
|
|
//GL.BindTexture(TextureTarget.Texture2D, 2);
|
2008-11-24 16:43:56 +00:00
|
|
|
|
2008-11-26 16:34:50 +00:00
|
|
|
//GL.Begin(BeginMode.Quads);
|
2008-11-24 16:43:56 +00:00
|
|
|
|
2008-11-26 16:34:50 +00:00
|
|
|
//GL.TexCoord2(0, 0);
|
|
|
|
//GL.Vertex2(0, 0);
|
|
|
|
//GL.TexCoord2(1, 0);
|
|
|
|
//GL.Vertex2(256, 0);
|
|
|
|
//GL.TexCoord2(1, 1);
|
|
|
|
//GL.Vertex2(256, 256);
|
|
|
|
//GL.TexCoord2(0, 1);
|
|
|
|
//GL.Vertex2(0, 256);
|
2008-11-24 16:43:56 +00:00
|
|
|
|
2008-11-26 16:34:50 +00:00
|
|
|
//GL.End();
|
2008-11-24 16:43:56 +00:00
|
|
|
|
2008-11-26 16:34:50 +00:00
|
|
|
//GL.Translate(0, 256, 0);
|
2008-11-24 16:43:56 +00:00
|
|
|
|
2008-11-26 16:34:50 +00:00
|
|
|
// Build layout
|
|
|
|
int current = 0;
|
|
|
|
foreach (Glyph glyph in block)
|
2008-11-25 16:45:24 +00:00
|
|
|
{
|
2008-11-26 16:34:50 +00:00
|
|
|
if (glyph.IsWhiteSpace)
|
|
|
|
{
|
|
|
|
current++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else if (!cache.Contains(glyph))
|
|
|
|
cache.Add(glyph);
|
|
|
|
|
|
|
|
CachedGlyphInfo info = cache[glyph];
|
|
|
|
RectangleF position = extents[current++];
|
|
|
|
|
|
|
|
// Use the real glyph width instead of the measured one (we want to achieve pixel perfect output).
|
|
|
|
position.Size = info.Rectangle.Size;
|
|
|
|
|
|
|
|
if (!active_lists.ContainsKey(info.Texture))
|
|
|
|
if (inactive_lists.Count > 0)
|
|
|
|
active_lists.Add(info.Texture, inactive_lists.Dequeue());
|
|
|
|
else
|
|
|
|
active_lists.Add(info.Texture, new List<Vector2>());
|
|
|
|
{
|
|
|
|
// Interleaved array: Vertex, TexCoord, Vertex, ...
|
|
|
|
active_lists[info.Texture].Add(new Vector2(info.RectangleNormalized.Left, info.RectangleNormalized.Top));
|
|
|
|
active_lists[info.Texture].Add(new Vector2(position.Left, position.Top));
|
|
|
|
active_lists[info.Texture].Add(new Vector2(info.RectangleNormalized.Left, info.RectangleNormalized.Bottom));
|
|
|
|
active_lists[info.Texture].Add(new Vector2(position.Left, position.Bottom));
|
|
|
|
active_lists[info.Texture].Add(new Vector2(info.RectangleNormalized.Right, info.RectangleNormalized.Bottom));
|
|
|
|
active_lists[info.Texture].Add(new Vector2(position.Right, position.Bottom));
|
|
|
|
|
|
|
|
active_lists[info.Texture].Add(new Vector2(info.RectangleNormalized.Right, info.RectangleNormalized.Bottom));
|
|
|
|
active_lists[info.Texture].Add(new Vector2(position.Right, position.Bottom));
|
|
|
|
active_lists[info.Texture].Add(new Vector2(info.RectangleNormalized.Right, info.RectangleNormalized.Top));
|
|
|
|
active_lists[info.Texture].Add(new Vector2(position.Right, position.Top));
|
|
|
|
active_lists[info.Texture].Add(new Vector2(info.RectangleNormalized.Left, info.RectangleNormalized.Top));
|
|
|
|
active_lists[info.Texture].Add(new Vector2(position.Left, position.Top));
|
|
|
|
}
|
2008-11-25 16:45:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Render
|
|
|
|
foreach (Texture2D key in active_lists.Keys)
|
|
|
|
{
|
|
|
|
List<Vector2> list = active_lists[key];
|
|
|
|
|
|
|
|
key.Bind();
|
|
|
|
GL.Begin(BeginMode.Triangles);
|
|
|
|
|
|
|
|
for (int i = 0; i < list.Count; i += 2)
|
|
|
|
{
|
|
|
|
GL.TexCoord2(list[i]);
|
|
|
|
GL.Vertex2(list[i + 1]);
|
|
|
|
}
|
|
|
|
|
|
|
|
GL.End();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Clean layout
|
|
|
|
foreach (List<Vector2> list in active_lists.Values)
|
|
|
|
{
|
|
|
|
list.Clear();
|
|
|
|
inactive_lists.Enqueue(list);
|
2008-11-24 16:43:56 +00:00
|
|
|
}
|
|
|
|
|
2008-11-25 16:45:24 +00:00
|
|
|
active_lists.Clear();
|
2008-11-24 16:43:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
}
|
|
|
|
}
|