/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* 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 3 of the License, or
* (at your option) any later version.
*
* 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, see .
*
*/
#ifndef ULTIMA8_GFX_FONTS_FONT_H
#define ULTIMA8_GFX_FONTS_FONT_H
#include "common/rect.h"
#include "ultima/shared/std/containers.h"
#include "ultima/shared/std/string.h"
#include "ultima/ultima8/misc/encoding.h"
namespace Ultima {
namespace Ultima8 {
class RenderedText;
struct PositionedText {
Std::string _text;
Common::Rect32 _dims;
Std::string::size_type _cursor;
};
class Font {
public:
Font();
virtual ~Font();
enum TextAlign {
TEXT_LEFT,
TEXT_CENTER,
TEXT_RIGHT
};
//! get the height of the font
virtual int getHeight() = 0;
//! get the baseline of the font (relative from the top)
virtual int getBaseline() = 0;
//! get the baselineskip of the font (distance between two baselines)
virtual int getBaselineSkip() = 0;
//! get the dimensions of a string (not containing any newlines)
//! \param text The string
//! \param width Returns the width
//! \param height Returns the height
virtual void getStringSize(const Std::string &text,
int32 &width, int32 &height) = 0;
//! render a string
//! \param text The text
//! \param remaining Returns index of the first character not printed
//! \param width The width of the target rectangle, or 0 for unlimited
//! \param height The height of the target rectangle, or 0 for unlimited
//! \param align Alignment of the text (left, right, center)
//! \param u8specials If true, interpret the special characters U8 uses
//! \param pagebreaks If true (and u8specials too), stop at U8 pagebreaks
//! \return the rendered text in a RenderedText object
virtual RenderedText *renderText(const Std::string &text,
unsigned int &remaining, int32 width = 0, int32 height = 0,
TextAlign align = TEXT_LEFT, bool u8specials = false,
bool pagebreaks = false,
Std::string::size_type cursor = Std::string::npos) = 0;
//! get the dimensions of a rendered string
//! \param text The text
//! \param resultwidth Returns the resulting width
//! \param resultheight Returns the resulting height
//! \param remaining Returns index of the first character not printed
//! \param width The width of the target rectangle, or 0 for unlimited
//! \param height The height of the target rectangle, or 0 for unlimited
//! \param align Alignment of the text (left, right, center)
//! \param u8specials If true, interpret the special characters U8 uses
//! \param pagebreaks If true (and u8specials too), stop at U8 pagebreaks
virtual void getTextSize(const Std::string &text,
int32 &resultwidth, int32 &resultheight, unsigned int &remaining,
int32 width = 0, int32 height = 0, TextAlign align = TEXT_LEFT,
bool u8specials = false, bool pagebreaks = false);
void setHighRes(bool hr) {
_highRes = hr;
}
bool isHighRes() const {
return _highRes;
}
protected:
bool _highRes;
struct Traits {
static bool isSpace(Std::string::const_iterator &i, bool u8specials) {
char c = *i;
return (c == ' ' || c == '\t' || c == '\n' || c == '\r' ||
(u8specials && (c == '%' || c == '~' || c == '*' || c == '^')));
}
static bool isTab(Std::string::const_iterator &i, bool u8specials) {
char c = *i;
return (c == '\t' ||
(u8specials && (c == '\t' || c == '%')));
}
static bool isBreak(Std::string::const_iterator &i, bool u8specials) {
char c = *i;
return (c == '\n' ||
(u8specials && (c == '\n' || c == '~' || c == '*')));
}
static bool isPageBreak(Std::string::const_iterator &i, bool u8specials) {
char c = *i;
return (u8specials && c == '*');
}
static bool canBreakAfter(Std::string::const_iterator &i);
static void advance(Std::string::const_iterator &i) {
++i;
}
static Std::string::size_type length(const Std::string &t) {
return t.size();
}
static uint32 unicode(Std::string::const_iterator &i) {
return encoding[static_cast(*i++)];
}
};
struct SJISTraits : public Traits {
static bool canBreakAfter(Std::string::const_iterator &i);
static void advance(Std::string::const_iterator &i) {
// FIXME: this can advance past the end of a malformed string
uint8 c = *i;
i++;
if (c >= 0x80) i++;
}
static Std::string::size_type length(const Std::string &t) {
Std::string::size_type l = 0;
Std::string::const_iterator iter = t.begin();
while (iter != t.end()) {
advance(iter);
l++;
}
return l;
}
static uint32 unicode(Std::string::const_iterator &i) {
uint16 s = static_cast(*i);
i++;
if (s >= 0x80) {
uint16 t = static_cast(*i++);
s |= (t << 8);
}
return shiftjis_to_unicode(s);
}
};
};
template
Std::list typesetText(Font *font,
const Std::string &text, unsigned int &remaining,
int32 width, int32 height, Font::TextAlign align,
bool u8specials, bool pagebreaks,
int32 &resultwidth, int32 &resultheight,
Std::string::size_type cursor = Std::string::npos);
} // End of namespace Ultima8
} // End of namespace Ultima
#endif