2009-12-05 18:15:15 -05:00
// Copyright (C) 2002-2009 Nikolaus Gebhardt
// This file is part of the "Irrlicht Engine".
// For conditions of distribution and use, see copyright notice in irrlicht.h
# include "CGUIFont.h"
# include "IGUIEnvironment.h"
# include "IXMLReader.h"
# include "IReadFile.h"
# include "IVideoDriver.h"
# include "IGUISpriteBank.h"
# include <iostream>
# include "guiengine/engine.hpp"
# include "io/file_manager.hpp"
2010-05-30 15:11:39 -04:00
# include "utils/translation.hpp"
2009-12-05 18:15:15 -05:00
namespace irr
{
namespace gui
{
//! constructor
ScalableFont : : ScalableFont ( IGUIEnvironment * env , const io : : path & filename )
: Driver ( 0 ) , SpriteBank ( 0 ) , Environment ( env ) , WrongCharacter ( 0 ) ,
MaxHeight ( 0 ) , GlobalKerningWidth ( 0 ) , GlobalKerningHeight ( 0 )
{
2010-01-24 11:26:31 -05:00
m_fallback_font = NULL ;
m_fallback_font_scale = 1.0f ;
m_fallback_kerning_width = 0 ;
2010-03-06 21:03:39 -05:00
m_is_hollow_copy = false ;
2010-05-30 15:11:39 -04:00
m_rtl = translations - > isRTLLanguage ( ) ;
2010-01-24 11:48:13 -05:00
m_black_border = false ;
2009-12-05 18:15:15 -05:00
# ifdef _DEBUG
setDebugName ( " ScalableFont " ) ;
# endif
m_scale = 1.0f ;
2009-12-05 18:47:33 -05:00
m_shadow = false ;
2009-12-05 18:15:15 -05:00
if ( Environment )
{
// don't grab environment, to avoid circular references
Driver = Environment - > getVideoDriver ( ) ;
SpriteBank = Environment - > addEmptySpriteBank ( filename ) ;
if ( SpriteBank )
SpriteBank - > grab ( ) ;
}
if ( Driver )
Driver - > grab ( ) ;
setInvisibleCharacters ( L " " ) ;
load ( file_manager - > createXMLReader ( filename . c_str ( ) ) ) ;
2010-03-06 21:03:39 -05:00
assert ( Areas . size ( ) > 0 ) ;
2009-12-05 18:15:15 -05:00
}
//! destructor
ScalableFont : : ~ ScalableFont ( )
{
2010-03-06 21:03:39 -05:00
if ( ! m_is_hollow_copy )
{
if ( Driver ) Driver - > drop ( ) ;
if ( SpriteBank ) SpriteBank - > drop ( ) ;
}
2009-12-05 18:15:15 -05:00
}
2010-03-06 21:03:39 -05:00
2009-12-05 18:47:33 -05:00
void ScalableFont : : setShadow ( irr : : video : : SColor col )
{
m_shadow = true ;
m_shadow_color = col ;
}
2009-12-05 18:15:15 -05:00
//! loads a font file from xml
bool ScalableFont : : load ( io : : IXMLReader * xml )
{
if ( ! SpriteBank )
return false ;
while ( xml - > read ( ) )
{
if ( io : : EXN_ELEMENT = = xml - > getNodeType ( ) )
{
if ( core : : stringw ( L " Texture " ) = = xml - > getNodeName ( ) )
{
// add a texture
2010-01-23 19:26:41 -05:00
core : : stringc filename = xml - > getAttributeValue ( L " filename " ) ;
core : : stringc fn = file_manager - > getFontFile ( filename . c_str ( ) ) . c_str ( ) ;
2009-12-05 18:15:15 -05:00
u32 i = ( u32 ) xml - > getAttributeValueAsInt ( L " index " ) ;
2010-01-28 19:19:17 -05:00
float scale = xml - > getAttributeValueAsFloat ( L " scale " ) ;
if ( scale < 0.01f ) scale = 1.0f ; // FIXME: how do you check if some property exists in a cleaner way?
//std::cout << "scale = " << scale << std::endl;
2009-12-05 18:15:15 -05:00
core : : stringw alpha = xml - > getAttributeValue ( L " hasAlpha " ) ;
2009-12-22 14:01:03 -05:00
//std::cout << "---- Adding font texture " << fn.c_str() << "; alpha=" << alpha.c_str() << std::endl;
2009-12-05 18:15:15 -05:00
2010-01-23 14:55:19 -05:00
// make sure the sprite bank has enough textures in it
2009-12-05 18:15:15 -05:00
while ( i + 1 > SpriteBank - > getTextureCount ( ) )
2010-01-23 14:55:19 -05:00
{
SpriteBank - > addTexture ( NULL ) ;
}
2009-12-05 18:15:15 -05:00
2010-01-23 14:55:19 -05:00
TextureInfo info ;
info . m_file_name = fn ;
2010-05-30 15:11:39 -04:00
info . m_has_alpha = ( alpha = = core : : stringw ( " true " ) ) ;
2010-01-28 19:19:17 -05:00
info . m_scale = scale ;
2010-01-23 14:55:19 -05:00
2009-12-05 18:15:15 -05:00
// disable mipmaps+filtering
//bool mipmap = Driver->getTextureCreationFlag(video::ETCF_CREATE_MIP_MAPS);
//Driver->setTextureCreationFlag(video::ETCF_CREATE_MIP_MAPS, false);
2010-01-23 14:55:19 -05:00
m_texture_files [ i ] = info ;
/*
2009-12-05 18:15:15 -05:00
// load texture
SpriteBank - > setTexture ( i , Driver - > getTexture ( fn ) ) ;
// set previous mip-map+filter state
//Driver->setTextureCreationFlag(video::ETCF_CREATE_MIP_MAPS, mipmap);
// couldn't load texture, abort.
if ( ! SpriteBank - > getTexture ( i ) )
{
std : : cerr < < " !!!!! Unable to load all textures in the font, aborting " < < std : : endl ;
_IRR_IMPLEMENT_MANAGED_MARSHALLING_BUGFIX ;
return false ;
}
else
{
// colorkey texture rather than alpha channel?
if ( alpha = = core : : stringw ( " false " ) )
Driver - > makeColorKeyTexture ( SpriteBank - > getTexture ( i ) , core : : position2di ( 0 , 0 ) ) ;
}
2010-01-23 14:55:19 -05:00
*/
2009-12-05 18:15:15 -05:00
}
else if ( core : : stringw ( L " c " ) = = xml - > getNodeName ( ) )
{
// adding a character to this font
SFontArea a ;
SGUISpriteFrame f ;
SGUISprite s ;
core : : rect < s32 > rectangle ;
a . underhang = xml - > getAttributeValueAsInt ( L " u " ) ;
a . overhang = xml - > getAttributeValueAsInt ( L " o " ) ;
a . spriteno = SpriteBank - > getSprites ( ) . size ( ) ;
s32 texno = xml - > getAttributeValueAsInt ( L " i " ) ;
// parse rectangle
core : : stringc rectstr = xml - > getAttributeValue ( L " r " ) ;
wchar_t ch = xml - > getAttributeValue ( L " c " ) [ 0 ] ;
const c8 * c = rectstr . c_str ( ) ;
s32 val ;
val = 0 ;
while ( * c > = ' 0 ' & & * c < = ' 9 ' )
{
val * = 10 ;
val + = * c - ' 0 ' ;
c + + ;
}
rectangle . UpperLeftCorner . X = val ;
while ( * c = = L ' ' | | * c = = L ' , ' ) c + + ;
val = 0 ;
while ( * c > = ' 0 ' & & * c < = ' 9 ' )
{
val * = 10 ;
val + = * c - ' 0 ' ;
c + + ;
}
rectangle . UpperLeftCorner . Y = val ;
while ( * c = = L ' ' | | * c = = L ' , ' ) c + + ;
val = 0 ;
while ( * c > = ' 0 ' & & * c < = ' 9 ' )
{
val * = 10 ;
val + = * c - ' 0 ' ;
c + + ;
}
rectangle . LowerRightCorner . X = val ;
while ( * c = = L ' ' | | * c = = L ' , ' ) c + + ;
val = 0 ;
while ( * c > = ' 0 ' & & * c < = ' 9 ' )
{
val * = 10 ;
val + = * c - ' 0 ' ;
c + + ;
}
rectangle . LowerRightCorner . Y = val ;
2010-03-06 21:03:39 -05:00
CharacterMap [ ch ] = Areas . size ( ) ;
2010-05-30 15:11:39 -04:00
//std::cout << "Inserting character '" << (int)ch << "' with area " << Areas.size() << std::endl;
2009-12-05 18:15:15 -05:00
// make frame
f . rectNumber = SpriteBank - > getPositions ( ) . size ( ) ;
f . textureNumber = texno ;
// add frame to sprite
s . Frames . push_back ( f ) ;
s . frameTime = 0 ;
// add rectangle to sprite bank
SpriteBank - > getPositions ( ) . push_back ( rectangle ) ;
a . width = rectangle . getWidth ( ) ;
// add sprite to sprite bank
SpriteBank - > getSprites ( ) . push_back ( s ) ;
// add character to font
Areas . push_back ( a ) ;
}
}
}
// set bad character
2010-01-24 11:26:31 -05:00
WrongCharacter = getAreaFromCharacter ( L ' ' , NULL ) ;
2009-12-05 18:15:15 -05:00
setMaxHeight ( ) ;
return true ;
}
void ScalableFont : : setScale ( const float scale )
{
m_scale = scale ;
}
void ScalableFont : : setMaxHeight ( )
{
2010-01-28 19:19:17 -05:00
// FIXME: should consider per-texture scaling
2009-12-05 18:15:15 -05:00
MaxHeight = 0 ;
s32 t ;
core : : array < core : : rect < s32 > > & p = SpriteBank - > getPositions ( ) ;
for ( u32 i = 0 ; i < p . size ( ) ; + + i )
{
t = p [ i ] . getHeight ( ) ;
if ( t > MaxHeight )
MaxHeight = t ;
}
}
//! loads a font file, native file needed, for texture parsing
2010-05-30 15:11:39 -04:00
/*
2009-12-05 18:15:15 -05:00
bool ScalableFont : : load ( io : : IReadFile * file )
{
2010-01-23 14:55:19 -05:00
if ( ! Driver ) return false ;
2009-12-05 18:15:15 -05:00
2010-01-23 14:55:19 -05:00
return loadTexture ( Driver - > createImageFromFile ( file ) , file - > getFileName ( ) ) ;
2009-12-05 18:15:15 -05:00
}
2010-05-30 15:11:39 -04:00
*/
2009-12-05 18:15:15 -05:00
//! loads a font file, native file needed, for texture parsing
2010-05-30 15:11:39 -04:00
/*
2009-12-05 18:15:15 -05:00
bool ScalableFont : : load ( const io : : path & filename )
{
2010-01-23 14:55:19 -05:00
if ( ! Driver ) return false ;
2009-12-05 18:15:15 -05:00
2010-01-23 14:55:19 -05:00
return loadTexture ( Driver - > createImageFromFile ( filename ) , filename ) ;
2009-12-05 18:15:15 -05:00
}
2010-05-30 15:11:39 -04:00
*/
2009-12-05 18:15:15 -05:00
//! load & prepare font from ITexture
2010-05-30 15:11:39 -04:00
#if 0
2009-12-05 18:15:15 -05:00
bool ScalableFont : : loadTexture ( video : : IImage * image , const io : : path & name )
{
2010-01-23 14:55:19 -05:00
if ( ! image ) return false ;
2009-12-05 18:15:15 -05:00
s32 lowerRightPositions = 0 ;
video : : IImage * tmpImage = image ;
/*
bool deleteTmpImage = false ;
switch ( image - > getColorFormat ( ) )
{
case video : : ECF_R5G6B5 :
tmpImage = new video : : CImage ( video : : ECF_A1R5G5B5 , image ) ;
deleteTmpImage = true ;
break ;
case video : : ECF_A1R5G5B5 :
case video : : ECF_A8R8G8B8 :
break ;
case video : : ECF_R8G8B8 :
tmpImage = new video : : CImage ( video : : ECF_A8R8G8B8 , image ) ;
deleteTmpImage = true ;
break ;
} */
readPositions ( tmpImage , lowerRightPositions ) ;
2010-01-24 11:26:31 -05:00
WrongCharacter = getAreaFromCharacter ( L ' ' , NULL ) ;
2009-12-05 18:15:15 -05:00
// output warnings
if ( ! lowerRightPositions | | ! SpriteBank - > getSprites ( ) . size ( ) )
std : : cerr < < " Either no upper or lower corner pixels in the font file. If this font was made using the new font tool, please load the XML file instead. If not, the font may be corrupted. \n " ;
else
if ( lowerRightPositions ! = ( s32 ) SpriteBank - > getPositions ( ) . size ( ) )
std : : cerr < < " The amount of upper corner pixels and the lower corner pixels is not equal, font file may be corrupted. \n " ;
bool ret = ( ! SpriteBank - > getSprites ( ) . empty ( ) & & lowerRightPositions ) ;
if ( ret )
{
bool flag [ 2 ] ;
flag [ 0 ] = Driver - > getTextureCreationFlag ( video : : ETCF_ALLOW_NON_POWER_2 ) ;
flag [ 1 ] = Driver - > getTextureCreationFlag ( video : : ETCF_CREATE_MIP_MAPS ) ;
Driver - > setTextureCreationFlag ( video : : ETCF_ALLOW_NON_POWER_2 , true ) ;
2010-05-30 15:11:39 -04:00
Driver - > setTextureCreationFlag ( video : : ETCF_CREATE_MIP_MAPS , true ) ;
2009-12-05 18:15:15 -05:00
SpriteBank - > addTexture ( Driver - > addTexture ( name , tmpImage ) ) ;
Driver - > setTextureCreationFlag ( video : : ETCF_ALLOW_NON_POWER_2 , flag [ 0 ] ) ;
Driver - > setTextureCreationFlag ( video : : ETCF_CREATE_MIP_MAPS , flag [ 1 ] ) ;
}
//if (deleteTmpImage)
// tmpImage->drop();
image - > drop ( ) ;
setMaxHeight ( ) ;
return ret ;
}
2010-05-30 15:11:39 -04:00
# endif
2009-12-05 18:15:15 -05:00
2010-05-30 15:11:39 -04:00
/*
void ScalableFont : : readPositions ( video : : IImage * image , s32 & spriteID )
2009-12-05 18:15:15 -05:00
{
const core : : dimension2d < u32 > size = image - > getDimension ( ) ;
video : : SColor colorTopLeft = image - > getPixel ( 0 , 0 ) ;
colorTopLeft . setAlpha ( 255 ) ;
image - > setPixel ( 0 , 0 , colorTopLeft ) ;
video : : SColor colorLowerRight = image - > getPixel ( 1 , 0 ) ;
video : : SColor colorBackGround = image - > getPixel ( 2 , 0 ) ;
video : : SColor colorBackGroundTransparent = 0 ;
image - > setPixel ( 1 , 0 , colorBackGround ) ;
// start parsing
core : : position2d < s32 > pos ( 0 , 0 ) ;
for ( pos . Y = 0 ; pos . Y < ( s32 ) size . Height ; + + pos . Y )
{
for ( pos . X = 0 ; pos . X < ( s32 ) size . Width ; + + pos . X )
{
const video : : SColor c = image - > getPixel ( pos . X , pos . Y ) ;
if ( c = = colorTopLeft )
{
image - > setPixel ( pos . X , pos . Y , colorBackGroundTransparent ) ;
SpriteBank - > getPositions ( ) . push_back ( core : : rect < s32 > ( pos , pos ) ) ;
}
else
if ( c = = colorLowerRight )
{
// too many lower right points
2010-05-30 15:11:39 -04:00
if ( SpriteBank - > getPositions ( ) . size ( ) < = ( u32 ) spriteID )
2009-12-05 18:15:15 -05:00
{
2010-05-30 15:11:39 -04:00
spriteID = 0 ;
2009-12-05 18:15:15 -05:00
return ;
}
image - > setPixel ( pos . X , pos . Y , colorBackGroundTransparent ) ;
2010-05-30 15:11:39 -04:00
SpriteBank - > getPositions ( ) [ spriteID ] . LowerRightCorner = pos ;
2009-12-05 18:15:15 -05:00
// add frame to sprite bank
SGUISpriteFrame f ;
2010-05-30 15:11:39 -04:00
f . rectNumber = spriteID ;
2009-12-05 18:15:15 -05:00
f . textureNumber = 0 ;
SGUISprite s ;
s . Frames . push_back ( f ) ;
s . frameTime = 0 ;
SpriteBank - > getSprites ( ) . push_back ( s ) ;
// add character to font
SFontArea a ;
a . overhang = 0 ;
a . underhang = 0 ;
2010-05-30 15:11:39 -04:00
a . spriteno = spriteID ;
a . width = SpriteBank - > getPositions ( ) [ spriteID ] . getWidth ( ) ;
2009-12-05 18:15:15 -05:00
Areas . push_back ( a ) ;
// map letter to character
2010-05-30 15:11:39 -04:00
wchar_t ch = ( wchar_t ) ( spriteID + 32 ) ;
CharacterMap [ ch ] = spriteID ;
2009-12-05 18:15:15 -05:00
2010-05-30 15:11:39 -04:00
+ + spriteID ;
2009-12-05 18:15:15 -05:00
}
else
if ( c = = colorBackGround )
image - > setPixel ( pos . X , pos . Y , colorBackGroundTransparent ) ;
}
}
}
2010-05-30 15:11:39 -04:00
*/
2009-12-05 18:15:15 -05:00
//! set an Pixel Offset on Drawing ( scale position on width )
void ScalableFont : : setKerningWidth ( s32 kerning )
{
GlobalKerningWidth = kerning ;
}
//! set an Pixel Offset on Drawing ( scale position on width )
s32 ScalableFont : : getKerningWidth ( const wchar_t * thisLetter , const wchar_t * previousLetter ) const
{
s32 ret = GlobalKerningWidth ;
if ( thisLetter )
{
2010-01-24 11:26:31 -05:00
ret + = Areas [ getAreaFromCharacter ( * thisLetter , NULL ) ] . overhang ;
2009-12-05 18:15:15 -05:00
if ( previousLetter )
{
2010-01-24 11:26:31 -05:00
ret + = Areas [ getAreaFromCharacter ( * previousLetter , NULL ) ] . underhang ;
2009-12-05 18:15:15 -05:00
}
}
return ret ;
}
//! set an Pixel Offset on Drawing ( scale position on height )
void ScalableFont : : setKerningHeight ( s32 kerning )
{
GlobalKerningHeight = kerning ;
}
//! set an Pixel Offset on Drawing ( scale position on height )
s32 ScalableFont : : getKerningHeight ( ) const
{
return GlobalKerningHeight ;
}
//! returns the sprite number from a given character
u32 ScalableFont : : getSpriteNoFromChar ( const wchar_t * c ) const
{
2010-01-24 11:26:31 -05:00
return Areas [ getAreaFromCharacter ( * c , NULL ) ] . spriteno ;
2009-12-05 18:15:15 -05:00
}
2010-01-24 11:26:31 -05:00
s32 ScalableFont : : getAreaFromCharacter ( const wchar_t c , bool * fallback_font ) const
2009-12-05 18:15:15 -05:00
{
2010-03-06 21:03:39 -05:00
std : : map < wchar_t , s32 > : : const_iterator n = CharacterMap . find ( c ) ;
if ( n ! = CharacterMap . end ( ) )
2010-01-24 11:26:31 -05:00
{
if ( fallback_font ! = NULL ) * fallback_font = false ;
2010-03-06 21:03:39 -05:00
return ( * n ) . second ;
2010-01-24 11:26:31 -05:00
}
else if ( m_fallback_font ! = NULL & & fallback_font ! = NULL )
{
//std::wcout << L"Font does not have this character : <" << c << L">, trying fallback font" << std::endl;
* fallback_font = true ;
return m_fallback_font - > getAreaFromCharacter ( c , NULL ) ;
}
2009-12-05 18:15:15 -05:00
else
2010-01-24 11:26:31 -05:00
{
2010-05-30 15:11:39 -04:00
std : : cout < < " The font does not have this character : < " < < ( int ) c < < L " > " < < std : : endl ;
2010-01-24 11:26:31 -05:00
if ( fallback_font ! = NULL ) * fallback_font = false ;
2009-12-05 18:15:15 -05:00
return WrongCharacter ;
2010-01-24 11:26:31 -05:00
}
2009-12-05 18:15:15 -05:00
}
void ScalableFont : : setInvisibleCharacters ( const wchar_t * s )
{
Invisible = s ;
}
//! returns the dimension of text
core : : dimension2d < u32 > ScalableFont : : getDimension ( const wchar_t * text ) const
{
2010-03-06 21:03:39 -05:00
assert ( Areas . size ( ) > 0 ) ;
2009-12-05 18:15:15 -05:00
core : : dimension2d < u32 > dim ( 0 , 0 ) ;
2010-01-29 23:21:28 -05:00
core : : dimension2d < u32 > thisLine ( 0 , ( int ) ( MaxHeight * m_scale ) ) ;
2009-12-05 18:15:15 -05:00
for ( const wchar_t * p = text ; * p ; + + p )
{
bool lineBreak = false ;
if ( * p = = L ' \r ' ) // Windows breaks
{
lineBreak = true ;
if ( p [ 1 ] = = L ' \n ' ) // Windows breaks
+ + p ;
}
else if ( * p = = L ' \n ' ) // Unix breaks
{
lineBreak = true ;
}
if ( lineBreak )
{
dim . Height + = thisLine . Height ;
if ( dim . Width < thisLine . Width )
dim . Width = thisLine . Width ;
thisLine . Width = 0 ;
continue ;
}
2010-01-24 11:26:31 -05:00
bool fallback = false ;
const int areaID = getAreaFromCharacter ( * p , & fallback ) ;
2010-03-06 21:03:39 -05:00
assert ( areaID < ( int ) Areas . size ( ) ) ;
2010-01-24 11:26:31 -05:00
const SFontArea & area = ( fallback ? m_fallback_font - > Areas [ areaID ] : Areas [ areaID ] ) ;
2009-12-05 18:15:15 -05:00
2010-01-28 19:19:17 -05:00
//const TextureInfo& info = (*(m_texture_files.find(area.spriteno))).second;
//const float char_scale = info.m_scale;
2009-12-05 18:15:15 -05:00
thisLine . Width + = area . underhang ;
2010-01-24 11:26:31 -05:00
2010-01-28 19:19:17 -05:00
thisLine . Width + = getCharWidth ( area , fallback ) ;
2010-01-24 11:26:31 -05:00
2010-01-28 19:19:17 -05:00
//if (fallback) thisLine.Width += (area.width + area.overhang)*m_fallback_font_scale*char_scale*m_scale + m_fallback_kerning_width;
//else thisLine.Width += (area.width + area.overhang)*m_scale*char_scale + GlobalKerningWidth;
2009-12-05 18:15:15 -05:00
}
dim . Height + = thisLine . Height ;
2010-01-24 11:26:31 -05:00
if ( dim . Width < thisLine . Width ) dim . Width = thisLine . Width ;
2009-12-05 18:15:15 -05:00
// std::cout << "ScalableFont::getDimension returns : " << dim.Width << ", " << dim.Height << " --> ";
2010-01-28 19:19:17 -05:00
dim . Width = ( int ) ( dim . Width + 0.9f ) ; // round up
dim . Height = ( int ) ( dim . Height + 0.9f ) ;
2009-12-05 18:15:15 -05:00
//std::cout << dim.Width << ", " << dim.Height << std::endl;
return dim ;
}
//! draws some text and clips it to the specified rectangle if wanted
void ScalableFont : : draw ( const core : : stringw & text , const core : : rect < s32 > & position ,
video : : SColor color , bool hcenter , bool vcenter , const core : : rect < s32 > * clip )
{
2010-01-23 14:55:19 -05:00
if ( ! Driver ) return ;
2010-01-24 11:48:13 -05:00
2009-12-05 18:47:33 -05:00
if ( m_shadow )
{
m_shadow = false ; // avoid infinite recursion
core : : rect < s32 > shadowpos = position ;
shadowpos . LowerRightCorner . X + = 2 ;
shadowpos . LowerRightCorner . Y + = 2 ;
draw ( text , shadowpos , m_shadow_color , hcenter , vcenter , clip ) ;
m_shadow = true ; // set back
}
2009-12-05 18:15:15 -05:00
//color = video::SColor(255,255,255,255);
core : : dimension2d < s32 > textDimension ;
core : : position2d < s32 > offset = position . UpperLeftCorner ;
2010-05-30 15:11:39 -04:00
if ( m_rtl | | hcenter | | vcenter | | clip ) textDimension = getDimension ( text . c_str ( ) ) ;
2009-12-05 18:15:15 -05:00
if ( hcenter )
{
2010-01-23 14:55:19 -05:00
offset . X + = ( position . getWidth ( ) - textDimension . Width ) / 2 ;
2009-12-05 18:15:15 -05:00
}
2010-05-30 15:11:39 -04:00
else if ( m_rtl )
{
offset . X + = ( position . getWidth ( ) - textDimension . Width ) ;
}
2009-12-05 18:15:15 -05:00
if ( vcenter )
2010-01-23 14:55:19 -05:00
{
offset . Y + = ( position . getHeight ( ) - textDimension . Height ) / 2 ;
}
2009-12-05 18:15:15 -05:00
if ( clip )
{
core : : rect < s32 > clippedRect ( offset , textDimension ) ;
clippedRect . clipAgainst ( * clip ) ;
2010-01-23 14:55:19 -05:00
if ( ! clippedRect . isValid ( ) ) return ;
2009-12-05 18:15:15 -05:00
}
2010-01-30 14:32:19 -05:00
// ---- collect character locations
2010-01-24 11:26:31 -05:00
const unsigned int text_size = text . size ( ) ;
2010-01-30 14:32:19 -05:00
core : : array < s32 > indices ( text_size ) ;
2010-01-24 11:26:31 -05:00
core : : array < core : : position2di > offsets ( text_size ) ;
std : : vector < bool > fallback ( text_size ) ;
2010-01-30 14:32:19 -05:00
for ( u32 i = 0 ; i < text_size ; i + + )
2009-12-05 18:15:15 -05:00
{
wchar_t c = text [ i ] ;
2010-05-03 19:37:38 -04:00
//hack: one tab character is supported, it moves the cursor to the middle of the area
if ( c = = L ' \t ' )
{
offset . X = position . UpperLeftCorner . X + position . getWidth ( ) / 2 ;
continue ;
}
2009-12-05 18:15:15 -05:00
bool lineBreak = false ;
2010-01-23 14:55:19 -05:00
if ( c = = L ' \r ' ) // Windows breaks
2009-12-05 18:15:15 -05:00
{
lineBreak = true ;
2010-01-23 14:55:19 -05:00
if ( text [ i + 1 ] = = L ' \n ' ) c = text [ + + i ] ;
2009-12-05 18:15:15 -05:00
}
else if ( c = = L ' \n ' ) // Unix breaks
{
lineBreak = true ;
}
if ( lineBreak )
{
offset . Y + = MaxHeight ;
offset . X = position . UpperLeftCorner . X ;
if ( hcenter )
{
core : : dimension2d < u32 > lineDim = getDimension ( text . c_str ( ) ) ;
offset . X + = ( position . getWidth ( ) - lineDim . Width ) > > 1 ;
}
continue ;
}
2010-01-28 19:19:17 -05:00
2010-01-24 11:26:31 -05:00
bool fallback_font = false ;
const int areaID = getAreaFromCharacter ( c , & fallback_font ) ;
2009-12-05 18:15:15 -05:00
//std::cout << "Char " << c << " has area " << areaID << std::endl;
2010-01-24 11:26:31 -05:00
SFontArea & area = ( fallback_font ? m_fallback_font - > Areas [ areaID ] : Areas [ areaID ] ) ;
2010-01-28 19:19:17 -05:00
//float char_scale = m_texture_files[area.spriteno].m_scale;
2009-12-05 18:15:15 -05:00
offset . X + = area . underhang ;
if ( Invisible . findFirst ( c ) < 0 )
{
indices . push_back ( area . spriteno ) ;
offsets . push_back ( offset ) ;
2010-01-24 11:26:31 -05:00
fallback [ i ] = fallback_font ;
2009-12-05 18:15:15 -05:00
}
2010-01-30 14:32:19 -05:00
else
{
// invisible character. add something to the array anyway so that indices from
// the various arrays remain in sync
indices . push_back ( - 1 ) ;
offsets . push_back ( offset ) ;
fallback [ i ] = fallback_font ;
}
/*
if ( fallback_font )
{
std : : cout < < " Fallback : " ;
std : : cout < < " Char " < < ( unsigned ) ( ( c > > 24 ) & 0xFF ) < < " , " < < ( unsigned ) ( ( c > > 16 ) & 0xFF ) < < " , " < <
( unsigned ) ( ( c > > 8 ) & 0xFF ) < < " , " < < ( unsigned ) ( c & 0xFF ) < < std : : endl ;
std : : cout < < " w = " < < getCharWidth ( area , fallback [ i ] ) < < " , fallback[ " < < i < < " ]= " < < fallback [ i ] < < std : : endl ;
}
*/
// std::cout << "Char " << (char)((c >> 24) & 0xFF) << ", " << (char)((c >> 16) & 0xFF) << ", " <<
// (char)((c >> 8) & 0xFF) << ", " << (char)(c & 0xFF) << std::endl;
2010-01-24 11:26:31 -05:00
2010-01-28 19:19:17 -05:00
offset . X + = getCharWidth ( area , fallback [ i ] ) ;
}
2009-12-05 18:15:15 -05:00
//SpriteBank->draw2DSpriteBatch(indices, offsets, clip, color);
2010-01-30 14:32:19 -05:00
// ---- do the actual rendering
2009-12-05 18:15:15 -05:00
const int indiceAmount = indices . size ( ) ;
core : : array < SGUISprite > & sprites = SpriteBank - > getSprites ( ) ;
core : : array < core : : rect < s32 > > & positions = SpriteBank - > getPositions ( ) ;
2010-01-24 11:26:31 -05:00
core : : array < SGUISprite > * fallback_sprites = ( m_fallback_font ! = NULL ?
& m_fallback_font - > SpriteBank - > getSprites ( ) :
NULL ) ;
core : : array < core : : rect < s32 > > * fallback_positions = ( m_fallback_font ! = NULL ?
& m_fallback_font - > SpriteBank - > getPositions ( ) :
NULL ) ;
2010-01-30 14:32:19 -05:00
2009-12-05 18:15:15 -05:00
video : : IVideoDriver * driver = GUIEngine : : getDriver ( ) ;
const int spriteAmount = sprites . size ( ) ;
for ( int n = 0 ; n < indiceAmount ; n + + )
{
const int spriteID = indices [ n ] ;
2010-01-24 11:26:31 -05:00
if ( ! fallback [ n ] & & ( spriteID < 0 | | spriteID > = spriteAmount ) ) continue ;
2010-01-30 14:32:19 -05:00
if ( indices [ n ] = = - 1 ) continue ;
2009-12-05 18:15:15 -05:00
2010-01-24 11:26:31 -05:00
//assert(sprites[spriteID].Frames.size() > 0);
2009-12-05 18:15:15 -05:00
2010-01-24 11:26:31 -05:00
const int texID = ( fallback [ n ] ?
( * fallback_sprites ) [ spriteID ] . Frames [ 0 ] . textureNumber :
sprites [ spriteID ] . Frames [ 0 ] . textureNumber ) ;
core : : rect < s32 > source = ( fallback [ n ] ?
( * fallback_positions ) [ ( * fallback_sprites ) [ spriteID ] . Frames [ 0 ] . rectNumber ] :
positions [ sprites [ spriteID ] . Frames [ 0 ] . rectNumber ] ) ;
2009-12-05 18:15:15 -05:00
2010-01-28 19:19:17 -05:00
const TextureInfo & info = ( fallback [ n ] ?
( * ( m_fallback_font - > m_texture_files . find ( texID ) ) ) . second :
( * ( m_texture_files . find ( texID ) ) ) . second
) ;
float char_scale = info . m_scale ;
2010-01-27 07:11:24 -05:00
2010-01-28 19:19:17 -05:00
core : : dimension2d < s32 > size = source . getSize ( ) ;
float scale = ( fallback [ n ] ? m_scale * m_fallback_font_scale : m_scale ) ;
size . Width = ( int ) ( size . Width * scale * char_scale ) ;
size . Height = ( int ) ( size . Height * scale * char_scale ) ;
// align vertically if character is smaller
2010-01-29 23:21:28 -05:00
int y_shift = ( size . Height < MaxHeight * m_scale ? ( int ) ( ( MaxHeight * m_scale - size . Height ) / 2.0f ) : 0 ) ;
2010-01-28 19:19:17 -05:00
core : : rect < s32 > dest ( offsets [ n ] + core : : position2di ( 0 , y_shift ) , size ) ;
2009-12-05 18:15:15 -05:00
video : : SColor colors [ ] = { color , color , color , color } ;
2010-01-24 11:26:31 -05:00
video : : ITexture * texture = ( fallback [ n ] ?
m_fallback_font - > SpriteBank - > getTexture ( texID ) :
SpriteBank - > getTexture ( texID ) ) ;
/*
if ( fallback [ n ] )
{
2010-01-30 14:32:19 -05:00
std : : cout < < " USING fallback font " < < core : : stringc ( texture - > getName ( ) ) . c_str ( )
< < " ; source area is " < < source . UpperLeftCorner . X < < " , " < < source . UpperLeftCorner . Y
< < " , size " < < source . getWidth ( ) < < " , " < < source . getHeight ( ) < < " ; dest = "
< < offsets [ n ] . X < < " , " < < offsets [ n ] . Y < < std : : endl ;
}
*/
2009-12-05 18:15:15 -05:00
2010-01-23 14:55:19 -05:00
if ( texture = = NULL )
{
// perform lazy loading
2010-01-24 11:26:31 -05:00
if ( fallback [ n ] )
{
m_fallback_font - > lazyLoadTexture ( texID ) ;
texture = m_fallback_font - > SpriteBank - > getTexture ( texID ) ;
}
else
{
lazyLoadTexture ( texID ) ;
texture = SpriteBank - > getTexture ( texID ) ;
}
2010-01-23 20:30:33 -05:00
if ( texture = = NULL )
{
std : : cerr < < " WARNING: character not found in current font \n " ;
continue ; // no such character
}
2010-01-23 14:55:19 -05:00
}
2010-01-24 11:48:13 -05:00
if ( m_black_border )
2010-01-24 11:26:31 -05:00
{
// draw black border
2010-01-24 11:48:13 -05:00
video : : SColor black ( color . getAlpha ( ) , 0 , 0 , 0 ) ;
2010-01-24 11:26:31 -05:00
video : : SColor black_colors [ ] = { black , black , black , black } ;
for ( int x_delta = - 2 ; x_delta < = 2 ; x_delta + + )
{
for ( int y_delta = - 2 ; y_delta < = 2 ; y_delta + + )
{
if ( x_delta = = 0 | | y_delta = = 0 ) continue ;
driver - > draw2DImage ( texture ,
dest + core : : position2d < s32 > ( x_delta , y_delta ) ,
source ,
clip ,
black_colors , true ) ;
}
}
2010-01-24 11:48:13 -05:00
}
if ( fallback [ n ] )
{
2010-01-24 11:26:31 -05:00
// draw text over
2010-01-24 11:48:13 -05:00
static video : : SColor orange ( color . getAlpha ( ) , 255 , 100 , 0 ) ;
static video : : SColor yellow ( color . getAlpha ( ) , 255 , 220 , 15 ) ;
2010-01-24 11:26:31 -05:00
video : : SColor title_colors [ ] = { yellow , orange , orange , yellow } ;
driver - > draw2DImage ( texture ,
dest ,
source ,
clip ,
title_colors , true ) ;
}
else
2010-05-30 15:11:39 -04:00
{
2010-01-24 11:26:31 -05:00
driver - > draw2DImage ( texture ,
dest ,
source ,
clip ,
colors , true ) ;
2010-05-30 15:11:39 -04:00
# ifdef FONT_DEBUG
driver - > draw2DLine ( core : : position2d < s32 > ( dest . UpperLeftCorner . X , dest . UpperLeftCorner . Y ) ,
core : : position2d < s32 > ( dest . UpperLeftCorner . X , dest . LowerRightCorner . Y ) ,
video : : SColor ( 255 , 255 , 0 , 0 ) ) ;
driver - > draw2DLine ( core : : position2d < s32 > ( dest . LowerRightCorner . X , dest . LowerRightCorner . Y ) ,
core : : position2d < s32 > ( dest . LowerRightCorner . X , dest . UpperLeftCorner . Y ) ,
video : : SColor ( 255 , 255 , 0 , 0 ) ) ;
driver - > draw2DLine ( core : : position2d < s32 > ( dest . LowerRightCorner . X , dest . LowerRightCorner . Y ) ,
core : : position2d < s32 > ( dest . UpperLeftCorner . X , dest . LowerRightCorner . Y ) ,
video : : SColor ( 255 , 255 , 0 , 0 ) ) ;
driver - > draw2DLine ( core : : position2d < s32 > ( dest . UpperLeftCorner . X , dest . UpperLeftCorner . Y ) ,
core : : position2d < s32 > ( dest . LowerRightCorner . X , dest . UpperLeftCorner . Y ) ,
video : : SColor ( 255 , 255 , 0 , 0 ) ) ;
# endif
2010-01-24 11:26:31 -05:00
}
2009-12-05 18:15:15 -05:00
}
}
2010-01-23 14:55:19 -05:00
void ScalableFont : : lazyLoadTexture ( int texID )
{
2010-01-24 11:48:13 -05:00
Driver - > setTextureCreationFlag ( video : : ETCF_CREATE_MIP_MAPS , true ) ;
2010-01-23 14:55:19 -05:00
// load texture
2010-01-28 19:19:17 -05:00
assert ( m_texture_files [ texID ] . m_file_name . size ( ) > 0 ) ;
2010-01-23 14:55:19 -05:00
SpriteBank - > setTexture ( texID , Driver - > getTexture ( m_texture_files [ texID ] . m_file_name ) ) ;
// set previous mip-map+filter state
//Driver->setTextureCreationFlag(video::ETCF_CREATE_MIP_MAPS, mipmap);
// couldn't load texture, abort.
if ( ! SpriteBank - > getTexture ( texID ) )
{
std : : cerr < < " !!!!! Unable to load all textures in the font " < < std : : endl ;
_IRR_IMPLEMENT_MANAGED_MARSHALLING_BUGFIX ;
return ;
}
else
{
2010-01-28 19:19:17 -05:00
assert ( m_texture_files [ texID ] . m_file_name . size ( ) > 0 ) ;
2010-01-23 14:55:19 -05:00
// colorkey texture rather than alpha channel?
if ( ! m_texture_files [ texID ] . m_has_alpha )
{
Driver - > makeColorKeyTexture ( SpriteBank - > getTexture ( texID ) , core : : position2di ( 0 , 0 ) ) ;
}
}
}
2010-01-28 19:19:17 -05:00
int ScalableFont : : getCharWidth ( const SFontArea & area , const bool fallback ) const
{
core : : array < SGUISprite > & sprites = SpriteBank - > getSprites ( ) ;
core : : array < SGUISprite > * fallback_sprites = ( m_fallback_font ! = NULL ?
& m_fallback_font - > SpriteBank - > getSprites ( ) :
NULL ) ;
const int texID = ( fallback ?
( * fallback_sprites ) [ area . spriteno ] . Frames [ 0 ] . textureNumber :
sprites [ area . spriteno ] . Frames [ 0 ] . textureNumber ) ;
const TextureInfo & info = ( fallback ?
( * ( m_fallback_font - > m_texture_files . find ( texID ) ) ) . second :
( * ( m_texture_files . find ( texID ) ) ) . second
) ;
assert ( info . m_file_name . size ( ) > 0 ) ;
const float char_scale = info . m_scale ;
//std::cout << "area.spriteno=" << area.spriteno << ", char_scale=" << char_scale << std::endl;
if ( fallback ) return ( int ) ( ( ( area . width + area . overhang ) * m_fallback_font_scale + m_fallback_kerning_width ) * m_scale * char_scale ) ;
else return ( int ) ( ( area . width + area . overhang + GlobalKerningWidth ) * m_scale * char_scale ) ;
}
2009-12-05 18:15:15 -05:00
//! Calculates the index of the character in the text which is on a specific position.
s32 ScalableFont : : getCharacterFromPos ( const wchar_t * text , s32 pixel_x ) const
{
s32 x = 0 ;
s32 idx = 0 ;
while ( text [ idx ] )
{
2010-01-24 11:26:31 -05:00
const SFontArea & a = Areas [ getAreaFromCharacter ( text [ idx ] , NULL ) ] ;
2009-12-05 18:15:15 -05:00
x + = a . width + a . overhang + a . underhang + GlobalKerningWidth ;
if ( x > = pixel_x )
return idx ;
+ + idx ;
}
return - 1 ;
}
IGUISpriteBank * ScalableFont : : getSpriteBank ( ) const
{
return SpriteBank ;
}
} // end namespace gui
} // end namespace irr