271 lines
5.8 KiB
C++
271 lines
5.8 KiB
C++
/*
|
|
** font.cpp
|
|
**
|
|
** This file is part of mkxp.
|
|
**
|
|
** Copyright (C) 2013 Jonas Kulla <Nyocurio@gmail.com>
|
|
**
|
|
** mkxp 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.
|
|
**
|
|
** mkxp 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 mkxp. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "font.h"
|
|
|
|
#include "sharedstate.h"
|
|
#include "filesystem.h"
|
|
#include "exception.h"
|
|
#include "boost-hash.h"
|
|
#include "util.h"
|
|
|
|
#include <string>
|
|
#include <utility>
|
|
|
|
#include "../liberation.ttf.xxd"
|
|
|
|
#include <SDL_ttf.h>
|
|
|
|
#define BUNDLED_FONT liberation
|
|
|
|
#define BUNDLED_FONT_D(f) assets_## f ##_ttf
|
|
#define BUNDLED_FONT_L(f) assets_## f ##_ttf_len
|
|
|
|
// Go fuck yourself CPP
|
|
#define BNDL_F_D(f) BUNDLED_FONT_D(f)
|
|
#define BNDL_F_L(f) BUNDLED_FONT_L(f)
|
|
|
|
typedef std::pair<std::string, int> FontKey;
|
|
|
|
static void strToLower(std::string &str)
|
|
{
|
|
for (size_t i = 0; i < str.size(); ++i)
|
|
str[i] = tolower(str[i]);
|
|
}
|
|
|
|
struct FontPoolPrivate
|
|
{
|
|
BoostHash<FontKey, TTF_Font*> hash;
|
|
};
|
|
|
|
FontPool::FontPool()
|
|
{
|
|
p = new FontPoolPrivate;
|
|
}
|
|
|
|
FontPool::~FontPool()
|
|
{
|
|
BoostHash<FontKey, TTF_Font*>::const_iterator iter;
|
|
for (iter = p->hash.cbegin(); iter != p->hash.cend(); ++iter)
|
|
TTF_CloseFont(iter->second);
|
|
|
|
delete p;
|
|
}
|
|
|
|
static SDL_RWops *openBundledFont()
|
|
{
|
|
return SDL_RWFromConstMem(BNDL_F_D(BUNDLED_FONT), BNDL_F_L(BUNDLED_FONT));
|
|
}
|
|
|
|
_TTF_Font *FontPool::request(const char *filename,
|
|
int size)
|
|
{
|
|
// FIXME Find out how font path resolution is done in VX/Ace
|
|
std::string nameKey(filename);
|
|
strToLower(nameKey);
|
|
strReplace(nameKey, ' ', '_');
|
|
|
|
bool useBundled = false;
|
|
std::string path = std::string("Fonts/") + nameKey;
|
|
if (!shState->fileSystem().exists(path.c_str(), FileSystem::Font))
|
|
{
|
|
/* Use the same name key for the bundled font
|
|
* even when it resulted from multiple different
|
|
* font name requests. The space at the front is
|
|
* to prevent collisions (spaces are normally
|
|
* replaced with '_' */
|
|
useBundled = true;
|
|
nameKey = " bundled";
|
|
}
|
|
|
|
FontKey key(nameKey, size);
|
|
|
|
TTF_Font *font = p->hash.value(key, 0);
|
|
|
|
if (font)
|
|
return font;
|
|
|
|
/* Not in hash, open */
|
|
SDL_RWops *ops;
|
|
|
|
if (useBundled)
|
|
{
|
|
ops = openBundledFont();
|
|
}
|
|
else
|
|
{
|
|
ops = SDL_AllocRW();
|
|
shState->fileSystem().openRead(*ops, path.c_str(), FileSystem::Font, true);
|
|
}
|
|
|
|
// FIXME 0.9 is guesswork at this point
|
|
font = TTF_OpenFontRW(ops, 1, (float) size * .90);
|
|
|
|
if (!font)
|
|
throw Exception(Exception::SDLError, "SDL: %s", SDL_GetError());
|
|
|
|
p->hash.insert(key, font);
|
|
|
|
return font;
|
|
}
|
|
|
|
|
|
struct FontPrivate
|
|
{
|
|
std::string name;
|
|
int size;
|
|
bool bold;
|
|
bool italic;
|
|
Color *color;
|
|
|
|
Color colorTmp;
|
|
|
|
static std::string defaultName;
|
|
static int defaultSize;
|
|
static bool defaultBold;
|
|
static bool defaultItalic;
|
|
static Color *defaultColor;
|
|
|
|
static Color defaultColorTmp;
|
|
|
|
/* The actual font is opened as late as possible
|
|
* (when it is queried by a Bitmap), prior it is
|
|
* set to null */
|
|
TTF_Font *sdlFont;
|
|
|
|
FontPrivate(const char *name = 0,
|
|
int size = 0)
|
|
: name(name ? std::string(name) : defaultName),
|
|
size(size ? size : defaultSize),
|
|
bold(defaultBold),
|
|
italic(defaultItalic),
|
|
color(&colorTmp),
|
|
colorTmp(*defaultColor),
|
|
sdlFont(0)
|
|
{}
|
|
|
|
FontPrivate(const FontPrivate &other)
|
|
: name(other.name),
|
|
size(other.size),
|
|
bold(other.bold),
|
|
italic(other.italic),
|
|
color(&colorTmp),
|
|
colorTmp(*other.color),
|
|
sdlFont(other.sdlFont)
|
|
{}
|
|
};
|
|
|
|
std::string FontPrivate::defaultName = "MS PGothic";
|
|
int FontPrivate::defaultSize = 22;
|
|
bool FontPrivate::defaultBold = false;
|
|
bool FontPrivate::defaultItalic = false;
|
|
Color *FontPrivate::defaultColor = &FontPrivate::defaultColorTmp;
|
|
|
|
Color FontPrivate::defaultColorTmp(255, 255, 255, 255);
|
|
|
|
bool Font::doesExist(const char *name)
|
|
{
|
|
std::string path = std::string("Fonts/") + std::string(name);
|
|
|
|
return shState->fileSystem().exists(path.c_str(), FileSystem::Font);
|
|
}
|
|
|
|
Font::Font(const char *name,
|
|
int size)
|
|
{
|
|
p = new FontPrivate(name, size);
|
|
}
|
|
|
|
Font::Font(const Font &other)
|
|
{
|
|
p = new FontPrivate(*other.p);
|
|
}
|
|
|
|
Font::~Font()
|
|
{
|
|
delete p;
|
|
}
|
|
|
|
const char *Font::getName() const
|
|
{
|
|
return p->name.c_str();
|
|
}
|
|
|
|
void Font::setName(const char *value)
|
|
{
|
|
if (p->name == value)
|
|
return;
|
|
|
|
p->name = value;
|
|
p->sdlFont = 0;
|
|
}
|
|
|
|
void Font::setSize(int value)
|
|
{
|
|
if (p->size == value)
|
|
return;
|
|
|
|
p->size = value;
|
|
p->sdlFont = 0;
|
|
}
|
|
|
|
#undef CHK_DISP
|
|
#define CHK_DISP
|
|
|
|
DEF_ATTR_RD_SIMPLE(Font, Size, int, p->size)
|
|
DEF_ATTR_SIMPLE(Font, Bold, bool, p->bold)
|
|
DEF_ATTR_SIMPLE(Font, Italic, bool, p->italic)
|
|
DEF_ATTR_SIMPLE(Font, Color, Color*, p->color)
|
|
|
|
DEF_ATTR_SIMPLE_STATIC(Font, DefaultSize, int, FontPrivate::defaultSize)
|
|
DEF_ATTR_SIMPLE_STATIC(Font, DefaultBold, bool, FontPrivate::defaultBold)
|
|
DEF_ATTR_SIMPLE_STATIC(Font, DefaultItalic, bool, FontPrivate::defaultItalic)
|
|
DEF_ATTR_SIMPLE_STATIC(Font, DefaultColor, Color*, FontPrivate::defaultColor)
|
|
|
|
const char *Font::getDefaultName()
|
|
{
|
|
return FontPrivate::defaultName.c_str();
|
|
}
|
|
|
|
void Font::setDefaultName(const char *value)
|
|
{
|
|
FontPrivate::defaultName = value;
|
|
}
|
|
|
|
_TTF_Font *Font::getSdlFont()
|
|
{
|
|
if (!p->sdlFont)
|
|
p->sdlFont = shState->fontPool().request(p->name.c_str(),
|
|
p->size);
|
|
|
|
int style = TTF_STYLE_NORMAL;
|
|
|
|
if (p->bold)
|
|
style |= TTF_STYLE_BOLD;
|
|
|
|
if (p->italic)
|
|
style |= TTF_STYLE_ITALIC;
|
|
|
|
TTF_SetFontStyle(p->sdlFont, style);
|
|
|
|
return p->sdlFont;
|
|
}
|