1
0
Fork 1
mirror of https://github.com/TES3MP/openmw-tes3mp.git synced 2025-01-19 23:23:52 +00:00
openmw-tes3mp/apps/openmw/mwgui/formatting.cpp

365 lines
12 KiB
C++
Raw Normal View History

#include "formatting.hpp"
#include <components/interpreter/defines.hpp>
#include "../mwscript/interpretercontext.hpp"
#include <boost/algorithm/string/replace.hpp>
#include <boost/algorithm/string/predicate.hpp>
#include <boost/lexical_cast.hpp>
#include <OgreUTFString.h>
using namespace MWGui;
2012-05-09 00:11:44 +00:00
namespace
{
int convertFromHex(std::string hex)
{
int value = 0;
2012-05-10 09:03:27 +00:00
2012-05-09 00:11:44 +00:00
int a = 0;
int b = hex.length() - 1;
for (; b >= 0; a++, b--)
{
if (hex[b] >= '0' && hex[b] <= '9')
{
value += (hex[b] - '0') * (1 << (a * 4));
}
else
{
switch (hex[b])
{
case 'A':
case 'a':
value += 10 * (1 << (a * 4));
break;
2012-05-10 09:03:27 +00:00
2012-05-09 00:11:44 +00:00
case 'B':
case 'b':
value += 11 * (1 << (a * 4));
break;
2012-05-10 09:03:27 +00:00
2012-05-09 00:11:44 +00:00
case 'C':
case 'c':
value += 12 * (1 << (a * 4));
break;
2012-05-10 09:03:27 +00:00
2012-05-09 00:11:44 +00:00
case 'D':
case 'd':
value += 13 * (1 << (a * 4));
break;
2012-05-10 09:03:27 +00:00
2012-05-09 00:11:44 +00:00
case 'E':
case 'e':
value += 14 * (1 << (a * 4));
break;
2012-05-10 09:03:27 +00:00
2012-05-09 00:11:44 +00:00
case 'F':
case 'f':
value += 15 * (1 << (a * 4));
break;
2012-05-10 09:03:27 +00:00
2012-05-09 00:11:44 +00:00
default:
throw std::runtime_error("invalid character in hex number");
break;
}
}
}
2012-05-10 09:03:27 +00:00
2012-05-09 00:11:44 +00:00
return value;
}
Ogre::UTFString::unicode_char unicodeCharFromChar(char ch)
{
std::string s;
s += ch;
Ogre::UTFString string(s);
return string.getChar(0);
}
2012-05-09 00:11:44 +00:00
}
std::vector<std::string> BookTextParser::split(std::string utf8Text, const int width, const int height)
2012-05-10 09:03:27 +00:00
{
using Ogre::UTFString;
2012-05-10 09:03:27 +00:00
std::vector<std::string> result;
MWScript::InterpreterContext interpreterContext(NULL, MWWorld::Ptr()); // empty arguments, because there is no locals or actor
utf8Text = Interpreter::fixDefinesBook(utf8Text, interpreterContext);
boost::algorithm::replace_all(utf8Text, "\n", "");
boost::algorithm::replace_all(utf8Text, "<BR>", "\n");
boost::algorithm::replace_all(utf8Text, "<P>", "\n\n");
2012-05-10 09:03:27 +00:00
UTFString text(utf8Text);
2012-05-10 09:03:27 +00:00
const int spacing = 48;
const UTFString::unicode_char LEFT_ANGLE = unicodeCharFromChar('<');
const UTFString::unicode_char NEWLINE = unicodeCharFromChar('\n');
const UTFString::unicode_char SPACE = unicodeCharFromChar(' ');
while (!text.empty())
2012-05-10 09:03:27 +00:00
{
// read in characters until we have exceeded the size, or run out of text
int currentWidth = 0;
int currentHeight = 0;
size_t currentWordStart = 0;
size_t index = 0;
while (currentHeight <= height - spacing && index < text.size())
2012-05-10 09:03:27 +00:00
{
const UTFString::unicode_char ch = text.getChar(index);
if (ch == LEFT_ANGLE)
2012-05-10 09:03:27 +00:00
{
const size_t tagStart = index + 1;
const size_t tagEnd = text.find('>', tagStart);
if (tagEnd == UTFString::npos)
2012-05-10 09:03:27 +00:00
throw std::runtime_error("BookTextParser Error: Tag is not terminated");
const std::string tag = text.substr(tagStart, tagEnd - tagStart).asUTF8();
2012-05-10 09:03:27 +00:00
if (boost::algorithm::starts_with(tag, "IMG"))
2012-05-10 09:03:27 +00:00
{
const int h = mHeight;
parseImage(tag, false);
currentHeight += (mHeight - h);
2012-05-10 09:03:27 +00:00
currentWidth = 0;
}
else if (boost::algorithm::starts_with(tag, "FONT"))
2012-05-10 09:03:27 +00:00
{
parseFont(tag);
if (currentWidth != 0) {
currentHeight += currentFontHeight();
currentWidth = 0;
}
2012-05-10 09:03:27 +00:00
currentWidth = 0;
}
else if (boost::algorithm::starts_with(tag, "DIV"))
2012-05-10 09:03:27 +00:00
{
parseDiv(tag);
if (currentWidth != 0) {
currentHeight += currentFontHeight();
currentWidth = 0;
}
2012-05-10 09:03:27 +00:00
}
index = tagEnd;
2012-05-10 09:03:27 +00:00
}
else if (ch == NEWLINE)
2012-05-10 09:03:27 +00:00
{
currentHeight += currentFontHeight();
2012-05-10 09:03:27 +00:00
currentWidth = 0;
currentWordStart = index;
2012-05-10 09:03:27 +00:00
}
else if (ch == SPACE)
2012-05-10 09:03:27 +00:00
{
currentWidth += 3; // keep this in sync with the font's SpaceWidth property
currentWordStart = index;
2012-05-10 09:03:27 +00:00
}
else
{
currentWidth += widthForCharGlyph(ch);
2012-05-10 09:03:27 +00:00
}
if (currentWidth > width)
{
currentHeight += currentFontHeight();
2012-05-10 09:03:27 +00:00
currentWidth = 0;
// add size of the current word
UTFString word = text.substr(currentWordStart, index - currentWordStart);
for (UTFString::const_iterator it = word.begin(), end = word.end(); it != end; ++it)
currentWidth += widthForCharGlyph(it.getCharacter());
2012-05-10 09:03:27 +00:00
}
index += UTFString::_utf16_char_length(ch);
2012-05-10 09:03:27 +00:00
}
const size_t pageEnd = (currentHeight > height - spacing && currentWordStart != 0)
? currentWordStart : index;
2012-05-10 09:03:27 +00:00
result.push_back(text.substr(0, pageEnd).asUTF8());
text.erase(0, pageEnd);
2012-05-10 09:03:27 +00:00
}
return result;
}
float BookTextParser::widthForCharGlyph(unsigned unicodeChar) const
{
std::string fontName(mTextStyle.mFont == "Default" ? "EB Garamond" : mTextStyle.mFont);
return MyGUI::FontManager::getInstance().getByName(fontName)
->getGlyphInfo(unicodeChar)->width;
}
float BookTextParser::currentFontHeight() const
{
std::string fontName(mTextStyle.mFont == "Default" ? "EB Garamond" : mTextStyle.mFont);
return MyGUI::FontManager::getInstance().getByName(fontName)->getDefaultHeight();
}
MyGUI::IntSize BookTextParser::parse(std::string text, MyGUI::Widget* parent, const int width)
{
MWScript::InterpreterContext interpreterContext(NULL, MWWorld::Ptr()); // empty arguments, because there is no locals or actor
2012-12-26 18:07:56 +00:00
text = Interpreter::fixDefinesBook(text, interpreterContext);
mParent = parent;
mWidth = width;
2012-05-08 23:34:32 +00:00
mHeight = 0;
assert(mParent);
while (mParent->getChildCount())
{
MyGUI::Gui::getInstance().destroyWidget(mParent->getChildAt(0));
}
boost::algorithm::replace_all(text, "\n", "");
boost::algorithm::replace_all(text, "<BR>", "\n");
2012-05-09 00:11:44 +00:00
boost::algorithm::replace_all(text, "<P>", "\n\n");
// remove leading newlines
// while (text[0] == '\n')
// text.erase(0);
2012-05-09 00:11:44 +00:00
// remove trailing "
if (text[text.size()-1] == '\"')
text.erase(text.size()-1);
2012-05-08 23:34:32 +00:00
parseSubText(text);
return MyGUI::IntSize(mWidth, mHeight);
}
2012-05-10 09:03:27 +00:00
void BookTextParser::parseImage(std::string tag, bool createWidget)
{
2012-05-08 23:34:32 +00:00
int src_start = tag.find("SRC=")+5;
std::string image = tag.substr(src_start, tag.find('"', src_start)-src_start);
// fix texture extension to .dds
if (image.size() > 4)
{
image[image.size()-3] = 'd';
image[image.size()-2] = 'd';
image[image.size()-1] = 's';
}
int width_start = tag.find("WIDTH=")+7;
int width = boost::lexical_cast<int>(tag.substr(width_start, tag.find('"', width_start)-width_start));
int height_start = tag.find("HEIGHT=")+8;
int height = boost::lexical_cast<int>(tag.substr(height_start, tag.find('"', height_start)-height_start));
2012-05-10 09:03:27 +00:00
if (createWidget)
{
MyGUI::ImageBox* box = mParent->createWidget<MyGUI::ImageBox> ("ImageBox",
MyGUI::IntCoord(0, mHeight, width, height), MyGUI::Align::Left | MyGUI::Align::Top,
mParent->getName() + boost::lexical_cast<std::string>(mParent->getChildCount()));
box->setImageTexture("bookart\\" + image);
box->setProperty("NeedMouse", "false");
}
2012-05-08 23:34:32 +00:00
mWidth = std::max(mWidth, width);
mHeight += height;
}
2012-05-09 00:11:44 +00:00
void BookTextParser::parseDiv(std::string tag)
{
if (tag.find("ALIGN=") == std::string::npos)
return;
int align_start = tag.find("ALIGN=")+7;
std::string align = tag.substr(align_start, tag.find('"', align_start)-align_start);
if (align == "CENTER")
mTextStyle.mTextAlign = MyGUI::Align::HCenter;
else if (align == "LEFT")
mTextStyle.mTextAlign = MyGUI::Align::Left;
}
void BookTextParser::parseFont(std::string tag)
{
if (tag.find("COLOR=") != std::string::npos)
{
int color_start = tag.find("COLOR=")+7;
std::string color = tag.substr(color_start, tag.find('"', color_start)-color_start);
mTextStyle.mColour = MyGUI::Colour(
convertFromHex(color.substr(0, 2))/255.0,
convertFromHex(color.substr(2, 2))/255.0,
convertFromHex(color.substr(4, 2))/255.0);
}
if (tag.find("FACE=") != std::string::npos)
{
int face_start = tag.find("FACE=")+6;
std::string face = tag.substr(face_start, tag.find('"', face_start)-face_start);
if (face != "Magic Cards")
mTextStyle.mFont = face;
}
if (tag.find("SIZE=") != std::string::npos)
{
/// \todo
}
}
2012-05-08 23:34:32 +00:00
void BookTextParser::parseSubText(std::string text)
{
if (text[0] == '<')
{
const size_t tagStart = 1;
const size_t tagEnd = text.find('>', tagStart);
if (tagEnd == std::string::npos)
2012-05-08 23:34:32 +00:00
throw std::runtime_error("BookTextParser Error: Tag is not terminated");
const std::string tag = text.substr(tagStart, tagEnd - tagStart);
2012-05-08 23:34:32 +00:00
if (boost::algorithm::starts_with(tag, "IMG"))
parseImage(tag);
if (boost::algorithm::starts_with(tag, "FONT"))
parseFont(tag);
if (boost::algorithm::starts_with(tag, "DOV"))
parseDiv(tag);
2012-05-08 23:34:32 +00:00
text.erase(0, tagEnd + 1);
2012-05-08 23:34:32 +00:00
}
size_t tagStart = std::string::npos;
std::string realText; // real text, without tags
for (size_t i = 0; i<text.size(); ++i)
{
char c = text[i];
if (c == '<')
{
if ((i + 1 < text.size()) && text[i+1] == '/') // ignore closing tags
{
while (c != '>')
{
if (i >= text.size())
throw std::runtime_error("BookTextParser Error: Tag is not terminated");
++i;
c = text[i];
}
continue;
}
else
{
tagStart = i;
break;
}
}
else
realText += c;
}
2012-05-08 23:34:32 +00:00
MyGUI::EditBox* box = mParent->createWidget<MyGUI::EditBox>("NormalText",
MyGUI::IntCoord(0, mHeight, mWidth, 24), MyGUI::Align::Left | MyGUI::Align::Top,
mParent->getName() + boost::lexical_cast<std::string>(mParent->getChildCount()));
box->setProperty("Static", "true");
box->setProperty("MultiLine", "true");
box->setProperty("WordWrap", "true");
box->setProperty("NeedMouse", "false");
box->setMaxTextLength(realText.size());
box->setTextAlign(mTextStyle.mTextAlign);
box->setTextColour(mTextStyle.mColour);
box->setFontName(mTextStyle.mFont);
box->setCaption(realText);
box->setSize(box->getSize().width, box->getTextSize().height);
mHeight += box->getTextSize().height;
if (tagStart != std::string::npos)
{
parseSubText(text.substr(tagStart, text.size()));
}
}