2011-04-14 11:11:17 +02:00
|
|
|
//
|
|
|
|
// This program is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU General Public License as
|
|
|
|
// published by the Free Software Foundation; either version 2 of the
|
|
|
|
// License, or (at your option) any later version.
|
2016-12-04 18:36:53 +00:00
|
|
|
//
|
2011-04-14 11:11:17 +02:00
|
|
|
// This program is distributed in the hope that it will be useful, but
|
|
|
|
// WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
// General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with this program; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
//
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
# include <config.h>
|
|
|
|
#endif
|
|
|
|
|
2011-05-27 20:53:16 +02:00
|
|
|
#ifdef HAVE_WINDOWS_H
|
|
|
|
#include <windows.h>
|
|
|
|
#endif
|
|
|
|
|
2016-12-04 18:36:53 +00:00
|
|
|
#include <math.h>
|
2011-11-19 20:25:51 +00:00
|
|
|
|
2016-12-04 18:36:53 +00:00
|
|
|
#include "FGGLApplication.hxx"
|
2011-04-14 11:11:17 +02:00
|
|
|
#include "ApplicationProperties.hxx"
|
|
|
|
#include "FGFontCache.hxx"
|
2011-11-19 20:25:51 +00:00
|
|
|
|
2011-04-14 11:11:17 +02:00
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
// FGFontCache class.
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2016-12-04 18:36:53 +00:00
|
|
|
const unsigned short FGFontCache::First_Printable_Char = 32;
|
|
|
|
const unsigned short FGFontCache::Last_Printable_Char = 127;
|
|
|
|
|
|
|
|
FGFontCache::FGFontCache () :
|
|
|
|
m_Face_Map (),
|
|
|
|
m_Current_Face_Ptr (NULL),
|
|
|
|
m_Pos_Map (),
|
|
|
|
m_Current_Pos (0) {
|
|
|
|
if (FT_Init_FreeType (&m_Ft)) {
|
|
|
|
SG_LOG (SG_COCKPIT, SG_ALERT, "Could not init freetype library");
|
|
|
|
}
|
|
|
|
for (unsigned int Index = 0; Index < Texture_Size * Texture_Size; ++Index) {
|
|
|
|
m_Texture[Index] = char (0);
|
|
|
|
}
|
|
|
|
glGenTextures (1, &m_Glyph_Texture);
|
2011-04-14 11:11:17 +02:00
|
|
|
}
|
|
|
|
|
2016-12-04 18:36:53 +00:00
|
|
|
FGFontCache::~FGFontCache () {
|
|
|
|
for (Face_Map_Type::iterator It = m_Face_Map.begin (); It != m_Face_Map.end (); ++It) {
|
|
|
|
delete (It->second);
|
|
|
|
}
|
|
|
|
m_Face_Map.clear ();
|
|
|
|
glDeleteTextures (1, &m_Glyph_Texture);
|
2011-04-14 11:11:17 +02:00
|
|
|
}
|
|
|
|
|
2016-12-04 18:36:53 +00:00
|
|
|
bool
|
|
|
|
FGFontCache::Set_Font (const string& Font_Name,
|
|
|
|
const float Size,
|
|
|
|
GLuint &Glyph_Texture) {
|
|
|
|
if (m_Face_Map.find (Font_Name) != m_Face_Map.end ()) {
|
|
|
|
m_Current_Face_Ptr = m_Face_Map[Font_Name];
|
|
|
|
} else {
|
|
|
|
FT_Face * const Face_Ptr (new FT_Face);
|
|
|
|
if (FT_New_Face (m_Ft, Font_Name.c_str (), 0, Face_Ptr)) {
|
|
|
|
SG_LOG (SG_COCKPIT, SG_ALERT, "Could not open font : " + Font_Name);
|
|
|
|
return false;
|
2011-04-14 11:11:17 +02:00
|
|
|
}
|
2016-12-04 18:36:53 +00:00
|
|
|
m_Face_Map.insert (pair <string, FT_Face *> (Font_Name, Face_Ptr));
|
|
|
|
m_Current_Face_Ptr = Face_Ptr;
|
|
|
|
}
|
|
|
|
if (m_Current_Face_Ptr != NULL) {
|
|
|
|
FT_Set_Pixel_Sizes (*m_Current_Face_Ptr, 0, Size);
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
const string Key_Str (Font_Name + "_" + Get_Size (Size));
|
|
|
|
if (m_Pos_Map.find (Key_Str) != m_Pos_Map.end ()) {
|
|
|
|
m_Current_Pos = m_Pos_Map[Key_Str];
|
|
|
|
} else {
|
|
|
|
m_Current_Pos = m_Pos_Map.size ();
|
|
|
|
for (unsigned short ASCII_Code = First_Printable_Char;
|
|
|
|
ASCII_Code < Last_Printable_Char;
|
|
|
|
++ASCII_Code) {
|
|
|
|
if (FT_Load_Char (*m_Current_Face_Ptr, char (ASCII_Code), FT_LOAD_RENDER)) {
|
|
|
|
SG_LOG (SG_COCKPIT, SG_ALERT, "Could not load character : " << char (ASCII_Code));
|
|
|
|
} else {
|
|
|
|
unsigned int Line;
|
|
|
|
unsigned int Column;
|
|
|
|
const FT_GlyphSlot Glyph ((*m_Current_Face_Ptr)->glyph);
|
|
|
|
for (unsigned int Row = 0; Row < Glyph->bitmap.rows; ++Row) {
|
|
|
|
for (unsigned int Width = 0; Width < Glyph->bitmap.width; ++Width) {
|
|
|
|
Get_Pos (ASCII_Code, Row, Width, Line, Column);
|
|
|
|
m_Texture[Line * Texture_Size + Column] =
|
|
|
|
Glyph->bitmap.buffer[Row * Glyph->bitmap.width + Width];
|
|
|
|
}
|
2011-04-14 11:11:17 +02:00
|
|
|
}
|
2016-12-04 18:36:53 +00:00
|
|
|
}
|
2011-04-14 11:11:17 +02:00
|
|
|
}
|
2016-12-04 18:36:53 +00:00
|
|
|
glBindTexture (GL_TEXTURE_2D, m_Glyph_Texture);
|
|
|
|
/* We require 1 byte alignment when uploading texture data */
|
|
|
|
glPixelStorei (GL_UNPACK_ALIGNMENT, 1);
|
|
|
|
/* Clamping to edges is important to prevent artifacts when scaling */
|
|
|
|
glTexParameteri (GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
|
|
|
|
glTexParameteri (GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
|
|
|
|
/* Linear filtering usually looks best for text */
|
|
|
|
glTexParameteri (GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
|
|
|
|
glTexParameteri (GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
|
|
|
|
glTexImage2D (GL_TEXTURE_2D,
|
|
|
|
0,
|
|
|
|
GL_ALPHA,
|
|
|
|
Texture_Size,
|
|
|
|
Texture_Size,
|
|
|
|
0,
|
|
|
|
GL_ALPHA,
|
|
|
|
GL_UNSIGNED_BYTE,
|
|
|
|
m_Texture);
|
|
|
|
m_Pos_Map.insert (pair <string, unsigned int> (Key_Str, m_Current_Pos));
|
|
|
|
}
|
|
|
|
Glyph_Texture = m_Glyph_Texture;
|
|
|
|
return true;
|
2011-04-14 11:11:17 +02:00
|
|
|
}
|
|
|
|
|
2016-12-04 18:36:53 +00:00
|
|
|
bool
|
|
|
|
FGFontCache::Get_Char (const char Char,
|
|
|
|
int &X, int &Y,
|
|
|
|
int &Left, int &Bottom,
|
|
|
|
int &W, int &H,
|
|
|
|
double &X1, double &Y1,
|
|
|
|
double &X2, double &Y2) const {
|
|
|
|
if (m_Current_Face_Ptr != NULL) {
|
|
|
|
if (FT_Load_Char (*m_Current_Face_Ptr, Char, FT_LOAD_RENDER)) {
|
|
|
|
SG_LOG (SG_COCKPIT, SG_ALERT, "Could not load character : " + Char);
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
const FT_GlyphSlot Glyph ((*m_Current_Face_Ptr)->glyph);
|
|
|
|
Left = Glyph->bitmap_left;
|
|
|
|
W = Glyph->bitmap.width;
|
|
|
|
H = Glyph->bitmap.rows;
|
|
|
|
Bottom = Glyph->bitmap_top - H;
|
|
|
|
X += (Glyph->advance.x >> 6);
|
|
|
|
Y += (Glyph->advance.y >> 6);
|
|
|
|
Get_Relative_Pos (int (Char), 0, 0, X1, Y1);
|
|
|
|
Get_Relative_Pos (int (Char), Glyph->bitmap.rows, Glyph->bitmap.width, X2, Y2);
|
|
|
|
return true;
|
2011-04-14 11:11:17 +02:00
|
|
|
}
|
2016-12-04 18:36:53 +00:00
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
2011-04-14 11:11:17 +02:00
|
|
|
}
|
|
|
|
|
2016-12-04 18:36:53 +00:00
|
|
|
void
|
|
|
|
FGFontCache::Get_Pos (const unsigned short ASCII_Code,
|
|
|
|
const unsigned short Row,
|
|
|
|
const unsigned short Width,
|
|
|
|
unsigned int &Line,
|
|
|
|
unsigned int &Column) const {
|
|
|
|
const unsigned short Font_Size (32);
|
|
|
|
Line = (((((ASCII_Code - First_Printable_Char) * Font_Size) / Texture_Size) + 3 * m_Current_Pos) * Font_Size) + Row;
|
|
|
|
Column = (((ASCII_Code - First_Printable_Char) * Font_Size) % Texture_Size) + Width;
|
2011-04-14 11:11:17 +02:00
|
|
|
}
|
|
|
|
|
2016-12-04 18:36:53 +00:00
|
|
|
void
|
|
|
|
FGFontCache::Get_Relative_Pos (const unsigned short ASCII_Code,
|
|
|
|
const unsigned short Row,
|
|
|
|
const unsigned short Width,
|
|
|
|
double &X,
|
|
|
|
double &Y) const {
|
|
|
|
unsigned int Line;
|
|
|
|
unsigned int Column;
|
|
|
|
Get_Pos (ASCII_Code, Row, Width, Line, Column);
|
|
|
|
X = double (Column) / double (Texture_Size);
|
|
|
|
Y = double (Line) / double (Texture_Size);
|
|
|
|
}
|
2011-04-14 11:11:17 +02:00
|
|
|
|
2016-12-04 18:36:53 +00:00
|
|
|
string
|
|
|
|
FGFontCache::Get_Size (const float Size) {
|
|
|
|
const int Half_Size (int (round (2.0 * Size)));
|
|
|
|
const int Int_Part (Half_Size / 2);
|
|
|
|
const int Dec_Part ((Half_Size % 2) ? 5 : 0);
|
|
|
|
stringstream Result_SS;
|
|
|
|
Result_SS << Int_Part << "." << Dec_Part;
|
|
|
|
return Result_SS.str ();
|
|
|
|
}
|