2002-11-26 16:01:28 +00:00
|
|
|
// Font.cc
|
2004-09-11 23:01:34 +00:00
|
|
|
// Copyright (c) 2002-2004 Henrik Kinnunen (fluxgen at users.sourceforge.net)
|
2002-11-26 16:01:28 +00:00
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
|
2004-11-19 11:37:27 +00:00
|
|
|
//$Id$
|
2002-11-26 16:01:28 +00:00
|
|
|
|
|
|
|
|
2004-08-10 18:08:37 +00:00
|
|
|
#include "StringUtil.hh"
|
2002-11-26 16:01:28 +00:00
|
|
|
#include "Font.hh"
|
|
|
|
#include "FontImp.hh"
|
2004-06-07 11:46:05 +00:00
|
|
|
#include "I18n.hh"
|
2002-11-26 16:01:28 +00:00
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif // HAVE_CONFIG_H
|
|
|
|
|
|
|
|
// for antialias
|
|
|
|
#ifdef USE_XFT
|
|
|
|
#include "XftFontImp.hh"
|
|
|
|
#endif // USE_XFT
|
|
|
|
|
|
|
|
// for multibyte support
|
|
|
|
#ifdef USE_XMB
|
|
|
|
#include "XmbFontImp.hh"
|
|
|
|
#endif //USE_XMB
|
|
|
|
|
|
|
|
// standard font system
|
|
|
|
#include "XFontImp.hh"
|
|
|
|
|
2003-11-28 23:27:29 +00:00
|
|
|
#include "GContext.hh"
|
2002-11-26 16:01:28 +00:00
|
|
|
//use gnu extensions
|
|
|
|
#ifndef _GNU_SOURCE
|
|
|
|
#define _GNU_SOURCE
|
|
|
|
#endif //_GNU_SOURCE
|
|
|
|
|
|
|
|
#ifndef __USE_GNU
|
|
|
|
#define __USE_GNU
|
|
|
|
#endif //__USE_GNU
|
|
|
|
|
|
|
|
#include <iostream>
|
2004-08-31 15:26:40 +00:00
|
|
|
#ifdef HAVE_CSTRING
|
|
|
|
#include <cstring>
|
|
|
|
#else
|
|
|
|
#include <string.h>
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_CSTDLIB
|
|
|
|
#include <cstdlib>
|
|
|
|
#else
|
|
|
|
#include <stdlib.h>
|
|
|
|
#endif
|
2004-08-10 18:08:37 +00:00
|
|
|
#include <list>
|
2003-01-05 23:00:19 +00:00
|
|
|
#include <typeinfo>
|
2004-08-18 16:30:33 +00:00
|
|
|
#include <langinfo.h>
|
2004-08-10 18:08:37 +00:00
|
|
|
|
|
|
|
#ifdef HAVE_SSTREAM
|
|
|
|
#include <sstream>
|
|
|
|
#define FB_istringstream istringstream
|
|
|
|
#elif HAVE_STRSTREAM
|
|
|
|
#include <strstream>
|
|
|
|
#define FB_istringstream istrstream
|
|
|
|
#else
|
|
|
|
#error "You dont have sstream or strstream headers!"
|
|
|
|
#endif // HAVE_STRSTREAM
|
|
|
|
|
2004-08-31 15:26:40 +00:00
|
|
|
#ifdef HAVE_CSTDLIB
|
|
|
|
#include <cstdlib>
|
|
|
|
#else
|
|
|
|
#include <stdlib.h>
|
|
|
|
#endif
|
2004-08-10 11:22:10 +00:00
|
|
|
|
2002-11-26 16:01:28 +00:00
|
|
|
using namespace std;
|
|
|
|
|
2004-08-10 18:08:37 +00:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2002-11-26 16:01:28 +00:00
|
|
|
#ifdef HAVE_SETLOCALE
|
|
|
|
#include <locale.h>
|
|
|
|
#endif //HAVE_SETLOCALE
|
|
|
|
|
2004-08-31 20:27:08 +00:00
|
|
|
#ifdef HAVE_ICONV
|
2004-08-10 11:22:10 +00:00
|
|
|
/**
|
|
|
|
Recodes the text from one encoding to another
|
|
|
|
assuming cd is correct
|
|
|
|
@param cd the iconv type
|
|
|
|
@param msg text to be converted
|
|
|
|
@param len number of chars to convert
|
|
|
|
@return the recoded string, or 0 on failure
|
|
|
|
*/
|
2004-08-10 18:08:37 +00:00
|
|
|
char* recode(iconv_t cd,
|
2004-08-31 20:27:08 +00:00
|
|
|
const char *msg, size_t size) {
|
2004-08-10 11:22:10 +00:00
|
|
|
|
|
|
|
// If empty message, yes this can happen, return
|
2004-08-25 10:03:09 +00:00
|
|
|
if(strlen(msg) == 0 || size == 0)
|
2004-08-10 11:22:10 +00:00
|
|
|
return 0;
|
|
|
|
|
2004-08-25 10:03:09 +00:00
|
|
|
if(strlen(msg) < size)
|
|
|
|
size = strlen(msg);
|
|
|
|
|
|
|
|
size_t inbytesleft = size;
|
2004-08-10 11:22:10 +00:00
|
|
|
size_t outbytesleft = 4*inbytesleft;
|
|
|
|
char *new_msg = new char[outbytesleft];
|
|
|
|
char *new_msg_ptr = new_msg;
|
|
|
|
char *msg_ptr = strdup(msg);
|
2004-08-18 16:30:33 +00:00
|
|
|
char *orig_msg_ptr = msg_ptr; // msg_ptr modified in iconv call
|
2004-08-31 21:24:05 +00:00
|
|
|
size_t result = (size_t)(-1);
|
2004-08-10 11:22:10 +00:00
|
|
|
|
2004-08-31 21:24:05 +00:00
|
|
|
#ifdef HAVE_CONST_ICONV
|
2004-09-01 00:09:03 +00:00
|
|
|
result = iconv(cd, (const char**)(&msg_ptr), &inbytesleft, &new_msg, &outbytesleft);
|
2004-08-31 21:24:05 +00:00
|
|
|
#else
|
|
|
|
result = iconv(cd, &msg_ptr, &inbytesleft, &new_msg, &outbytesleft);
|
|
|
|
#endif // HAVE_CONST_ICONV
|
|
|
|
|
|
|
|
if (result == (size_t)(-1)) {
|
2004-08-10 11:22:10 +00:00
|
|
|
// iconv can fail for three reasons
|
|
|
|
// 1) Invalid multibyte sequence is encountered in the input
|
|
|
|
// 2) An incomplete multibyte sequence
|
|
|
|
// 3) The output buffer has no more room for the next converted character.
|
|
|
|
// So we the delete new message and return original message
|
2004-11-19 12:00:20 +00:00
|
|
|
delete[] new_msg_ptr;
|
2004-08-18 16:30:33 +00:00
|
|
|
free(orig_msg_ptr);
|
2004-08-10 11:22:10 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2004-08-18 16:30:33 +00:00
|
|
|
free(orig_msg_ptr);
|
2004-08-10 11:22:10 +00:00
|
|
|
|
2004-08-18 16:30:33 +00:00
|
|
|
*new_msg = '\0';
|
2004-08-10 11:22:10 +00:00
|
|
|
|
|
|
|
if(inbytesleft != 0) {
|
2004-11-19 12:00:20 +00:00
|
|
|
delete[] new_msg_ptr;
|
2004-08-10 11:22:10 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2004-08-18 16:30:33 +00:00
|
|
|
|
|
|
|
return new_msg_ptr;
|
2004-08-10 11:22:10 +00:00
|
|
|
}
|
2004-08-31 20:27:08 +00:00
|
|
|
#else
|
2004-08-10 11:22:10 +00:00
|
|
|
|
2004-08-31 21:24:05 +00:00
|
|
|
char *recode(int cd,
|
2004-08-31 20:27:08 +00:00
|
|
|
const char *msg, size_t size) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif // HAVE_ICONV
|
2004-08-10 18:08:37 +00:00
|
|
|
|
|
|
|
int extract_halo_options(const std::string& opts, std::string& color) {
|
|
|
|
std::list< std::string > tokens;
|
|
|
|
size_t sep= opts.find_first_of(':');
|
|
|
|
|
|
|
|
if ( sep == std::string::npos )
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
FbTk::StringUtil::stringtok(tokens, opts.substr(sep + 1, opts.length()), ";");
|
|
|
|
tokens.unique();
|
|
|
|
std::list< std::string >::const_iterator token;
|
|
|
|
|
|
|
|
for ( token= tokens.begin(); token != tokens.end(); token++ ) {
|
|
|
|
if ( (*token).find("color=", 0) != std::string::npos ) {
|
|
|
|
size_t s= (*token).find_first_of('=');
|
|
|
|
std::string c= (*token).substr(s + 1, (*token).length());
|
|
|
|
if ( !c.empty() )
|
|
|
|
std::swap(color, c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int extract_shadow_options(const std::string& opts,
|
|
|
|
std::string& color,
|
|
|
|
int& offx, int& offy) {
|
|
|
|
|
|
|
|
std::list< std::string > tokens;
|
|
|
|
size_t sep= opts.find_first_of(':');
|
|
|
|
|
|
|
|
if ( sep == std::string::npos )
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
FbTk::StringUtil::stringtok(tokens, opts.substr(sep + 1, opts.length()), ";");
|
|
|
|
tokens.unique();
|
|
|
|
std::list< std::string >::const_iterator token;
|
|
|
|
|
|
|
|
for ( token= tokens.begin(); token != tokens.end(); token++ ) {
|
|
|
|
if ( (*token).find("color=", 0) != std::string::npos ) {
|
|
|
|
size_t s= (*token).find_first_of('=');
|
|
|
|
std::string c= (*token).substr(s + 1, (*token).length());
|
|
|
|
if ( !c.empty() )
|
|
|
|
std::swap(color, c);
|
|
|
|
}
|
|
|
|
else if ( (*token).find("offsetx=", 0) != std::string::npos ) {
|
|
|
|
size_t s= (*token).find_first_of('=');
|
|
|
|
FB_istringstream o((*token).substr(s + 1, (*token).length()));
|
|
|
|
if ( !o.eof() ) {
|
|
|
|
o >> offx;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if ( (*token).find("offsety=", 0) != std::string::npos ) {
|
|
|
|
size_t s= (*token).find_first_of('=');
|
|
|
|
FB_istringstream o((*token).substr(s + 1, (*token).length()));
|
|
|
|
if ( !o.eof() ) {
|
|
|
|
o >> offy;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}; // end nameless namespace
|
|
|
|
|
|
|
|
|
|
|
|
|
2002-11-26 16:01:28 +00:00
|
|
|
namespace FbTk {
|
|
|
|
|
|
|
|
bool Font::m_multibyte = false;
|
|
|
|
bool Font::m_utf8mode = false;
|
|
|
|
|
2004-08-18 16:30:33 +00:00
|
|
|
// some initialisation for using fonts
|
|
|
|
void fontInit() {
|
2004-11-19 12:00:20 +00:00
|
|
|
setlocale(LC_CTYPE, "");
|
2004-08-18 16:30:33 +00:00
|
|
|
}
|
|
|
|
|
2002-11-26 16:01:28 +00:00
|
|
|
Font::Font(const char *name, bool antialias):
|
2002-12-01 13:42:15 +00:00
|
|
|
m_fontimp(0),
|
2004-08-10 18:08:37 +00:00
|
|
|
m_antialias(false), m_rotated(false),
|
|
|
|
m_shadow(false), m_shadow_color("#000000"),
|
|
|
|
m_shadow_offx(1), m_shadow_offy(1),
|
2004-11-19 12:00:20 +00:00
|
|
|
m_halo(false), m_halo_color("#ffffff"),
|
2004-08-31 21:24:05 +00:00
|
|
|
#ifdef HAVE_ICONV
|
2004-11-19 12:00:20 +00:00
|
|
|
m_iconv((iconv_t)(-1))
|
2004-08-31 21:24:05 +00:00
|
|
|
#else
|
2004-11-19 12:00:20 +00:00
|
|
|
m_iconv(-1)
|
2004-08-31 21:24:05 +00:00
|
|
|
#endif // HAVE_ICONV
|
2004-11-19 12:00:20 +00:00
|
|
|
{
|
2004-08-10 11:22:10 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
// MB_CUR_MAX returns the size of a char in the current locale
|
|
|
|
if (MB_CUR_MAX > 1) // more than one byte, then we're multibyte
|
|
|
|
m_multibyte = true;
|
|
|
|
|
|
|
|
// check for utf-8 mode
|
2004-08-31 23:07:58 +00:00
|
|
|
#ifdef CODESET
|
2004-08-18 16:30:33 +00:00
|
|
|
char *locale_codeset = nl_langinfo(CODESET);
|
2004-08-31 23:07:58 +00:00
|
|
|
#else // openbsd doesnt have this (yet?)
|
|
|
|
char *locale_codeset = 0;
|
|
|
|
#endif // CODESET
|
2004-08-14 09:33:09 +00:00
|
|
|
|
2004-08-31 23:07:58 +00:00
|
|
|
if (locale_codeset && strcmp("UTF-8", locale_codeset) == 0) {
|
2004-08-18 16:30:33 +00:00
|
|
|
m_utf8mode = true;
|
2004-08-31 23:07:58 +00:00
|
|
|
} else if (locale_codeset != 0) {
|
2004-08-18 16:30:33 +00:00
|
|
|
// if locale isn't UTF-8 we try to
|
|
|
|
// create a iconv pointer so we can
|
|
|
|
// convert non utf-8 strings to utf-8
|
2004-08-14 09:33:09 +00:00
|
|
|
|
2004-08-10 11:22:10 +00:00
|
|
|
#ifdef DEBUG
|
2004-08-18 16:30:33 +00:00
|
|
|
cerr<<"FbTk::Font: check UTF-8 convert for codeset = "<<locale_codeset<<endl;
|
2004-08-10 11:22:10 +00:00
|
|
|
#endif // DEBUG
|
2004-08-31 21:24:05 +00:00
|
|
|
|
|
|
|
#ifdef HAVE_ICONV
|
2004-08-18 16:30:33 +00:00
|
|
|
m_iconv = iconv_open("UTF-8", locale_codeset);
|
2004-08-10 11:22:10 +00:00
|
|
|
if(m_iconv == (iconv_t)(-1)) {
|
2004-08-18 16:30:33 +00:00
|
|
|
cerr<<"FbTk::Font: code error: from "<<locale_codeset<<" to: UTF-8"<<endl;
|
2004-08-10 11:22:10 +00:00
|
|
|
// if we failed with iconv then we can't convert
|
|
|
|
// the strings to utf-8, so we disable utf8 mode
|
|
|
|
m_utf8mode = false;
|
|
|
|
} else {
|
|
|
|
// success, we can now enable utf8mode
|
|
|
|
// and if antialias is on later we can recode
|
|
|
|
// the non utf-8 string to utf-8 and use utf-8
|
|
|
|
// drawing functions
|
|
|
|
m_utf8mode = true;
|
|
|
|
}
|
2004-08-31 21:24:05 +00:00
|
|
|
#endif // HAVE_ICONV
|
2004-08-31 21:47:56 +00:00
|
|
|
}
|
2002-12-01 13:42:15 +00:00
|
|
|
|
2004-08-10 11:22:10 +00:00
|
|
|
#ifdef DEBUG
|
|
|
|
cerr<<"FbTk::Font m_iconv = "<<(int)m_iconv<<endl;
|
|
|
|
#endif // DEBUG
|
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
// create the right font implementation
|
|
|
|
// antialias is prio 1
|
2002-11-26 16:01:28 +00:00
|
|
|
#ifdef USE_XFT
|
2002-12-01 13:42:15 +00:00
|
|
|
if (antialias) {
|
|
|
|
m_fontimp.reset(new XftFontImp(0, m_utf8mode));
|
|
|
|
}
|
2002-11-26 16:01:28 +00:00
|
|
|
#endif //USE_XFT
|
2002-12-01 13:42:15 +00:00
|
|
|
// if we didn't create a Xft font then create basic font
|
|
|
|
if (m_fontimp.get() == 0) {
|
2002-11-26 16:01:28 +00:00
|
|
|
#ifdef USE_XMB
|
2002-12-01 13:42:15 +00:00
|
|
|
if (m_multibyte || m_utf8mode)
|
|
|
|
m_fontimp.reset(new XmbFontImp(0, m_utf8mode));
|
|
|
|
else // basic font implementation
|
2002-11-26 16:01:28 +00:00
|
|
|
#endif // USE_XMB
|
2002-12-01 13:42:15 +00:00
|
|
|
m_fontimp.reset(new XFontImp());
|
|
|
|
}
|
2004-08-10 11:22:10 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
if (name != 0) {
|
|
|
|
load(name);
|
|
|
|
}
|
2002-11-26 16:01:28 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
Font::~Font() {
|
2004-08-31 21:47:56 +00:00
|
|
|
#ifdef HAVE_ICONV
|
2004-08-10 11:22:10 +00:00
|
|
|
if (m_iconv != (iconv_t)(-1))
|
|
|
|
iconv_close(m_iconv);
|
2004-08-31 21:47:56 +00:00
|
|
|
#endif // HAVE_ICONV
|
2002-11-26 16:01:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Font::setAntialias(bool flag) {
|
2002-12-01 13:42:15 +00:00
|
|
|
bool loaded = m_fontimp->loaded();
|
2002-11-26 16:01:28 +00:00
|
|
|
#ifdef USE_XFT
|
2002-12-01 13:42:15 +00:00
|
|
|
if (flag && !isAntialias() && !m_rotated) {
|
|
|
|
m_fontimp.reset(new XftFontImp(m_fontstr.c_str(), m_utf8mode));
|
|
|
|
} else if (!flag && isAntialias())
|
2002-11-26 16:01:28 +00:00
|
|
|
#endif // USE_XFT
|
|
|
|
{
|
|
|
|
#ifdef USE_XMB
|
2002-12-01 13:42:15 +00:00
|
|
|
if (m_multibyte || m_utf8mode)
|
|
|
|
m_fontimp.reset(new XmbFontImp(m_fontstr.c_str(), m_utf8mode));
|
|
|
|
else
|
2002-11-26 16:01:28 +00:00
|
|
|
#endif // USE_XMB
|
2002-12-01 13:42:15 +00:00
|
|
|
m_fontimp.reset(new XFontImp(m_fontstr.c_str()));
|
2002-11-26 16:01:28 +00:00
|
|
|
}
|
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
if (m_fontimp->loaded() != loaded) { // if the new font failed to load, fall back to 'fixed'
|
2004-06-07 11:46:05 +00:00
|
|
|
if (!m_fontimp->load("fixed")) {// if that failes too, output warning
|
|
|
|
_FB_USES_NLS;
|
|
|
|
cerr<<_FBTKTEXT(Error, CantFallbackFont, "Warning: can't load fallback font", "Attempt to load the last-resort default font failed")<<" 'fixed'."<<endl;
|
2004-11-19 12:00:20 +00:00
|
|
|
}
|
2002-12-01 13:42:15 +00:00
|
|
|
}
|
2004-11-19 12:00:20 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
m_antialias = flag;
|
2002-11-26 16:01:28 +00:00
|
|
|
}
|
|
|
|
|
2003-11-28 23:27:29 +00:00
|
|
|
bool Font::load(const std::string &name) {
|
|
|
|
if (name.size() == 0)
|
2002-12-01 13:42:15 +00:00
|
|
|
return false;
|
2004-09-11 23:01:34 +00:00
|
|
|
// default values for font options
|
2003-11-28 23:27:29 +00:00
|
|
|
m_shadow = false;
|
2004-08-10 18:08:37 +00:00
|
|
|
m_halo = false;
|
|
|
|
|
|
|
|
// everything after ':' is a fontoption
|
|
|
|
// -> extract 'halo' and 'shadow' and
|
|
|
|
// load remaining fname
|
|
|
|
size_t sep= name.find_first_of(':');
|
|
|
|
|
|
|
|
if ( sep != std::string::npos ) {
|
2004-09-03 14:17:47 +00:00
|
|
|
|
|
|
|
std::list< std::string > tokens;
|
|
|
|
std::string fname;
|
|
|
|
|
2004-10-31 23:04:30 +00:00
|
|
|
fname= std::string(name.c_str(), sep);
|
|
|
|
|
|
|
|
FbTk::StringUtil::stringtok(tokens, name.substr(sep + 1), ",");
|
2004-09-03 14:17:47 +00:00
|
|
|
|
|
|
|
tokens.unique();
|
|
|
|
bool firstone= true;
|
|
|
|
std::list< std::string >::const_iterator token;
|
|
|
|
|
|
|
|
// check tokens and extract extraoptions for halo and shadow
|
|
|
|
for( token= tokens.begin(); token != tokens.end(); token++ ) {
|
|
|
|
if ( (*token).find("halo",0) != std::string::npos ) {
|
|
|
|
m_halo= true;
|
|
|
|
extract_halo_options(*token, m_halo_color);
|
|
|
|
}
|
|
|
|
else if ( (*token).find("shadow", 0) != std::string::npos ) {
|
|
|
|
m_shadow= true;
|
|
|
|
extract_shadow_options(*token, m_shadow_color, m_shadow_offx, m_shadow_offy);
|
2004-09-11 23:01:34 +00:00
|
|
|
}
|
2004-09-03 14:17:47 +00:00
|
|
|
else {
|
|
|
|
if ( !firstone )
|
2004-09-11 23:01:34 +00:00
|
|
|
fname+= ", ";
|
2004-09-03 14:17:47 +00:00
|
|
|
else
|
2004-09-11 23:01:34 +00:00
|
|
|
firstone= false;
|
2004-10-31 23:04:30 +00:00
|
|
|
fname= fname + ":" + *token;
|
2003-12-01 19:57:01 +00:00
|
|
|
}
|
|
|
|
}
|
2003-11-28 23:27:29 +00:00
|
|
|
|
2004-09-03 14:17:47 +00:00
|
|
|
m_fontstr = fname;
|
|
|
|
} else
|
|
|
|
m_fontstr = name;
|
|
|
|
|
|
|
|
return m_fontimp->load(m_fontstr.c_str());
|
2002-11-26 16:01:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int Font::textWidth(const char * const text, unsigned int size) const {
|
2004-08-31 21:47:56 +00:00
|
|
|
#ifdef HAVE_ICONV
|
2004-11-19 12:00:20 +00:00
|
|
|
if (m_fontimp->utf8() && m_iconv != (iconv_t)(-1)) {
|
2004-08-10 11:22:10 +00:00
|
|
|
char* rtext = recode(m_iconv, text, size);
|
|
|
|
if (rtext != 0)
|
|
|
|
size = strlen(rtext);
|
|
|
|
unsigned int r = m_fontimp->textWidth(rtext ? rtext : text, size);
|
|
|
|
if (rtext != 0)
|
2004-11-19 12:00:20 +00:00
|
|
|
delete[] rtext;
|
2004-08-10 11:22:10 +00:00
|
|
|
return r;
|
|
|
|
}
|
2004-08-31 21:47:56 +00:00
|
|
|
#endif // HAVE_ICONV
|
2002-12-01 13:42:15 +00:00
|
|
|
return m_fontimp->textWidth(text, size);
|
2002-11-26 16:01:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int Font::height() const {
|
2002-12-01 13:42:15 +00:00
|
|
|
return m_fontimp->height();
|
2002-11-26 16:01:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int Font::ascent() const {
|
2002-12-01 13:42:15 +00:00
|
|
|
return m_fontimp->ascent();
|
2002-11-26 16:01:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int Font::descent() const {
|
2002-12-01 13:42:15 +00:00
|
|
|
return m_fontimp->descent();
|
2002-11-26 16:01:28 +00:00
|
|
|
}
|
2004-08-10 11:22:10 +00:00
|
|
|
|
2004-09-11 23:01:34 +00:00
|
|
|
void Font::drawText(const FbDrawable &w, int screen, GC gc,
|
2003-11-28 23:27:29 +00:00
|
|
|
const char *text, size_t len, int x, int y,
|
2002-12-08 19:12:07 +00:00
|
|
|
bool rotate) const {
|
2002-12-01 13:42:15 +00:00
|
|
|
if (text == 0 || len == 0)
|
|
|
|
return;
|
2003-11-28 23:27:29 +00:00
|
|
|
|
2004-08-10 11:22:10 +00:00
|
|
|
char* rtext = 0;
|
|
|
|
|
2003-11-28 23:27:29 +00:00
|
|
|
// so we don't end up in a loop with m_shadow
|
|
|
|
static bool first_run = true;
|
2004-08-31 21:47:56 +00:00
|
|
|
|
|
|
|
#ifdef HAVE_ICONV
|
2004-11-19 12:00:20 +00:00
|
|
|
if (m_fontimp->utf8() && m_iconv != (iconv_t)(-1) && first_run) {
|
2004-08-10 11:22:10 +00:00
|
|
|
rtext = recode(m_iconv, text, len);
|
|
|
|
if (rtext != 0) {
|
|
|
|
len = strlen(rtext);
|
|
|
|
// ok, we can't use utf8 mode since the string is invalid
|
|
|
|
}
|
|
|
|
}
|
2004-08-31 21:47:56 +00:00
|
|
|
#endif // HAVE_ICONV
|
2004-08-10 11:22:10 +00:00
|
|
|
|
|
|
|
const char *real_text = rtext ? rtext : text;
|
|
|
|
|
2004-08-10 18:08:37 +00:00
|
|
|
// draw "effects" first
|
|
|
|
if (first_run) {
|
|
|
|
if (m_shadow) {
|
2004-09-11 23:01:34 +00:00
|
|
|
FbTk::GContext shadow_gc(w);
|
2004-08-10 18:08:37 +00:00
|
|
|
shadow_gc.setForeground(FbTk::Color(m_shadow_color.c_str(), screen));
|
|
|
|
first_run = false;
|
|
|
|
drawText(w, screen, shadow_gc.gc(), real_text, len,
|
|
|
|
x + m_shadow_offx, y + m_shadow_offy, rotate);
|
2004-09-11 23:01:34 +00:00
|
|
|
first_run = true;
|
2004-08-10 18:08:37 +00:00
|
|
|
} else if (m_halo) {
|
|
|
|
FbTk::GContext halo_gc(w);
|
|
|
|
halo_gc.setForeground(FbTk::Color(m_halo_color.c_str(), screen));
|
|
|
|
first_run = false;
|
|
|
|
drawText(w, screen, halo_gc.gc(), real_text, len, x + 1, y + 1, rotate);
|
|
|
|
drawText(w, screen, halo_gc.gc(), real_text, len, x - 1, y + 1, rotate);
|
|
|
|
drawText(w, screen, halo_gc.gc(), real_text, len, x - 1, y - 1, rotate);
|
|
|
|
drawText(w, screen, halo_gc.gc(), real_text, len, x + 1, y - 1, rotate);
|
|
|
|
first_run = true;
|
|
|
|
}
|
2003-11-28 23:27:29 +00:00
|
|
|
}
|
|
|
|
|
2002-12-08 19:12:07 +00:00
|
|
|
if (!rotate && isRotated()) {
|
|
|
|
// if this was called with request to not rotated the text
|
|
|
|
// we just forward it to the implementation that handles rotation
|
|
|
|
// currently just XFontImp
|
|
|
|
// Using dynamic_cast just temporarly until there's a better solution
|
|
|
|
// to put in FontImp
|
|
|
|
try {
|
|
|
|
XFontImp *font = dynamic_cast<XFontImp *>(m_fontimp.get());
|
|
|
|
font->setRotate(false); // disable rotation temporarly
|
2003-11-28 23:27:29 +00:00
|
|
|
|
2004-08-10 11:22:10 +00:00
|
|
|
font->drawText(w, screen, gc, real_text, len, x, y);
|
2002-12-08 19:12:07 +00:00
|
|
|
font->setRotate(true); // enable rotation
|
|
|
|
} catch (std::bad_cast &bc) {
|
|
|
|
// draw normal...
|
2004-08-10 11:22:10 +00:00
|
|
|
m_fontimp->drawText(w, screen, gc, real_text, len, x, y);
|
2002-12-08 19:12:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} else
|
2004-08-10 11:22:10 +00:00
|
|
|
m_fontimp->drawText(w, screen, gc, real_text, len, x, y);
|
2003-11-28 23:27:29 +00:00
|
|
|
|
2004-08-10 11:22:10 +00:00
|
|
|
if (rtext != 0)
|
2004-11-19 12:00:20 +00:00
|
|
|
delete[] rtext;
|
2003-11-28 23:27:29 +00:00
|
|
|
|
2002-11-26 16:01:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Font::rotate(float angle) {
|
|
|
|
#ifdef USE_XFT
|
2002-12-01 13:42:15 +00:00
|
|
|
// if we are rotated and we are changing to horiz text
|
|
|
|
// and we were antialiased before we rotated then change to XftFontImp
|
|
|
|
if (isRotated() && angle == 0 && isAntialias())
|
|
|
|
m_fontimp.reset(new XftFontImp(m_fontstr.c_str(), m_utf8mode));
|
2002-11-26 16:01:28 +00:00
|
|
|
#endif // USE_XFT
|
2002-12-01 13:42:15 +00:00
|
|
|
// change to a font imp that handles rotated fonts (i.e just XFontImp at the moment)
|
|
|
|
// if we're going to rotate this font
|
|
|
|
if (angle != 0 && isAntialias() && !isRotated()) {
|
|
|
|
m_fontimp.reset(new XFontImp(m_fontstr.c_str()));
|
2002-12-08 19:12:07 +00:00
|
|
|
if (!m_fontimp->loaded()) // if it failed to load font, try default font fixed
|
|
|
|
m_fontimp->load("fixed");
|
2002-12-01 13:42:15 +00:00
|
|
|
}
|
2002-11-26 16:01:28 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
//Note: only XFontImp implements FontImp::rotate
|
|
|
|
m_fontimp->rotate(angle);
|
2002-11-26 16:01:28 +00:00
|
|
|
|
2002-12-01 13:42:15 +00:00
|
|
|
m_rotated = (angle == 0 ? false : true);
|
2002-12-08 19:12:07 +00:00
|
|
|
m_angle = angle;
|
2002-11-26 16:01:28 +00:00
|
|
|
}
|
|
|
|
|
2004-08-10 11:22:10 +00:00
|
|
|
|
2002-11-26 16:01:28 +00:00
|
|
|
};
|
2004-08-10 11:22:10 +00:00
|
|
|
|