|
|
@ -11,68 +11,14 @@
|
|
|
|
#include <boost/lexical_cast.hpp>
|
|
|
|
#include <boost/lexical_cast.hpp>
|
|
|
|
#include <boost/range/adaptor/filtered.hpp>
|
|
|
|
#include <boost/range/adaptor/filtered.hpp>
|
|
|
|
#include <boost/range/algorithm/copy.hpp>
|
|
|
|
#include <boost/range/algorithm/copy.hpp>
|
|
|
|
#include <OgreUTFString.h>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#include <OgreUTFString.h>
|
|
|
|
#include <OgreResourceGroupManager.h>
|
|
|
|
#include <OgreResourceGroupManager.h>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#include <MyGUI_EditText.h>
|
|
|
|
|
|
|
|
|
|
|
|
namespace
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
{
|
|
|
|
int convertFromHex(std::string hex)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
int value = 0;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
case 'B':
|
|
|
|
|
|
|
|
case 'b':
|
|
|
|
|
|
|
|
value += 11 * (1 << (a * 4));
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
case 'C':
|
|
|
|
|
|
|
|
case 'c':
|
|
|
|
|
|
|
|
value += 12 * (1 << (a * 4));
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
case 'D':
|
|
|
|
|
|
|
|
case 'd':
|
|
|
|
|
|
|
|
value += 13 * (1 << (a * 4));
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
case 'E':
|
|
|
|
|
|
|
|
case 'e':
|
|
|
|
|
|
|
|
value += 14 * (1 << (a * 4));
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
case 'F':
|
|
|
|
|
|
|
|
case 'f':
|
|
|
|
|
|
|
|
value += 15 * (1 << (a * 4));
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
|
|
|
throw std::runtime_error("invalid character in hex number");
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return value;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Ogre::UTFString::unicode_char unicodeCharFromChar(char ch)
|
|
|
|
Ogre::UTFString::unicode_char unicodeCharFromChar(char ch)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
std::string s;
|
|
|
|
std::string s;
|
|
|
@ -80,21 +26,15 @@ namespace
|
|
|
|
Ogre::UTFString string(s);
|
|
|
|
Ogre::UTFString string(s);
|
|
|
|
return string.getChar(0);
|
|
|
|
return string.getChar(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool is_not_empty(const std::string& s) {
|
|
|
|
|
|
|
|
std::string temp = s;
|
|
|
|
|
|
|
|
boost::algorithm::trim(temp);
|
|
|
|
|
|
|
|
return !temp.empty();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
namespace MWGui
|
|
|
|
namespace MWGui
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
namespace Formatting
|
|
|
|
std::vector<std::string> BookTextParser::split(std::string utf8Text, const int width, const int height)
|
|
|
|
{
|
|
|
|
|
|
|
|
Paginator::Pages BookFormatter::markupToWidget(MyGUI::Widget * parent, std::string utf8Text, const int pageWidth, const int pageHeight)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
using Ogre::UTFString;
|
|
|
|
using Ogre::UTFString;
|
|
|
|
std::vector<std::string> result;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
MWScript::InterpreterContext interpreterContext(NULL, MWWorld::Ptr()); // empty arguments, because there is no locals or actor
|
|
|
|
MWScript::InterpreterContext interpreterContext(NULL, MWWorld::Ptr()); // empty arguments, because there is no locals or actor
|
|
|
|
utf8Text = Interpreter::fixDefinesBook(utf8Text, interpreterContext);
|
|
|
|
utf8Text = Interpreter::fixDefinesBook(utf8Text, interpreterContext);
|
|
|
@ -105,31 +45,37 @@ namespace MWGui
|
|
|
|
boost::algorithm::replace_all(utf8Text, "<P>", "\n\n");
|
|
|
|
boost::algorithm::replace_all(utf8Text, "<P>", "\n\n");
|
|
|
|
|
|
|
|
|
|
|
|
UTFString text(utf8Text);
|
|
|
|
UTFString text(utf8Text);
|
|
|
|
const int spacing = 48;
|
|
|
|
UTFString plainText;
|
|
|
|
|
|
|
|
|
|
|
|
const UTFString::unicode_char LEFT_ANGLE = unicodeCharFromChar('<');
|
|
|
|
const UTFString::unicode_char LEFT_ANGLE = unicodeCharFromChar('<');
|
|
|
|
const UTFString::unicode_char NEWLINE = unicodeCharFromChar('\n');
|
|
|
|
const UTFString::unicode_char NEWLINE = unicodeCharFromChar('\n');
|
|
|
|
const UTFString::unicode_char SPACE = unicodeCharFromChar(' ');
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
while (!text.empty())
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
// read in characters until we have exceeded the size, or run out of text
|
|
|
|
|
|
|
|
int currentWidth = 0;
|
|
|
|
|
|
|
|
int currentHeight = 0;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
size_t currentWordStart = 0;
|
|
|
|
Paginator pag(pageWidth, pageHeight);
|
|
|
|
size_t index = 0;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
while (parent->getChildCount())
|
|
|
|
{
|
|
|
|
{
|
|
|
|
std::string texToTrim = text.asUTF8();
|
|
|
|
MyGUI::Gui::getInstance().destroyWidget(parent->getChildAt(0));
|
|
|
|
boost::algorithm::trim( texToTrim );
|
|
|
|
|
|
|
|
text = UTFString(texToTrim);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
MyGUI::Widget * paper = parent->createWidget<MyGUI::Widget>("Widget", MyGUI::IntCoord(0, 0, pag.getPageWidth(), pag.getPageHeight()), MyGUI::Align::Left | MyGUI::Align::Top);
|
|
|
|
|
|
|
|
paper->setNeedMouseFocus(false);
|
|
|
|
|
|
|
|
|
|
|
|
while (currentHeight <= height - spacing && index < text.size())
|
|
|
|
bool ignoreNewlines = true;
|
|
|
|
|
|
|
|
for (size_t index = 0; index < text.size(); ++index)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
const UTFString::unicode_char ch = text.getChar(index);
|
|
|
|
const UTFString::unicode_char ch = text.getChar(index);
|
|
|
|
|
|
|
|
if (!plainText.empty() && (ch == LEFT_ANGLE || index == text.size() - 1))
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
// if there's a newline at the end of the box caption, remove it
|
|
|
|
|
|
|
|
if (plainText[plainText.size()-1] == NEWLINE)
|
|
|
|
|
|
|
|
plainText.erase(plainText.end()-1);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
TextElement elem(paper, pag, mTextStyle, plainText.asUTF8());
|
|
|
|
|
|
|
|
elem.paginate();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
plainText.clear();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (ch == LEFT_ANGLE)
|
|
|
|
if (ch == LEFT_ANGLE)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
const size_t tagStart = index + 1;
|
|
|
|
const size_t tagStart = index + 1;
|
|
|
@ -140,158 +86,46 @@ namespace MWGui
|
|
|
|
|
|
|
|
|
|
|
|
if (boost::algorithm::starts_with(tag, "IMG"))
|
|
|
|
if (boost::algorithm::starts_with(tag, "IMG"))
|
|
|
|
{
|
|
|
|
{
|
|
|
|
const int h = mHeight;
|
|
|
|
ImageElement elem(paper, pag, mTextStyle, tag);
|
|
|
|
parseImage(tag, false);
|
|
|
|
elem.paginate();
|
|
|
|
currentHeight += (mHeight - h);
|
|
|
|
|
|
|
|
currentWidth = 0;
|
|
|
|
ignoreNewlines = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (boost::algorithm::starts_with(tag, "FONT"))
|
|
|
|
else if (boost::algorithm::starts_with(tag, "FONT"))
|
|
|
|
{
|
|
|
|
{
|
|
|
|
parseFont(tag);
|
|
|
|
parseFont(tag);
|
|
|
|
if (currentWidth != 0) {
|
|
|
|
|
|
|
|
currentHeight += currentFontHeight();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
currentWidth = 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (boost::algorithm::starts_with(tag, "DIV"))
|
|
|
|
else if (boost::algorithm::starts_with(tag, "DIV"))
|
|
|
|
{
|
|
|
|
{
|
|
|
|
parseDiv(tag);
|
|
|
|
parseDiv(tag);
|
|
|
|
if (currentWidth != 0) {
|
|
|
|
|
|
|
|
currentHeight += currentFontHeight();
|
|
|
|
|
|
|
|
currentWidth = 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
index = tagEnd;
|
|
|
|
index = tagEnd;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (ch == NEWLINE)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
currentHeight += currentFontHeight();
|
|
|
|
|
|
|
|
currentWidth = 0;
|
|
|
|
|
|
|
|
currentWordStart = index;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (ch == SPACE)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
currentWidth += 3; // keep this in sync with the font's SpaceWidth property
|
|
|
|
|
|
|
|
currentWordStart = index;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
else
|
|
|
|
{
|
|
|
|
{
|
|
|
|
currentWidth += widthForCharGlyph(ch);
|
|
|
|
if (!ignoreNewlines || ch != NEWLINE)
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (currentWidth > width)
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
currentHeight += currentFontHeight();
|
|
|
|
plainText.push_back(ch);
|
|
|
|
currentWidth = 0;
|
|
|
|
ignoreNewlines = false;
|
|
|
|
// 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());
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
index += UTFString::_utf16_char_length(ch);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const size_t pageEnd = (currentHeight > height - spacing && currentWordStart != 0)
|
|
|
|
|
|
|
|
? currentWordStart : index;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
result.push_back(text.substr(0, pageEnd).asUTF8());
|
|
|
|
|
|
|
|
text.erase(0, pageEnd);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
std::vector<std::string> nonEmptyPages;
|
|
|
|
// insert last page
|
|
|
|
boost::copy(result | boost::adaptors::filtered(is_not_empty), std::back_inserter(nonEmptyPages));
|
|
|
|
pag << Paginator::Page(pag.getStartTop(), pag.getStartTop() + pag.getPageHeight());
|
|
|
|
return nonEmptyPages;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
float BookTextParser::widthForCharGlyph(unsigned unicodeChar) const
|
|
|
|
paper->setSize(paper->getWidth(), pag.getCurrentTop());
|
|
|
|
{
|
|
|
|
|
|
|
|
std::string fontName(mTextStyle.mFont == "Default" ? MyGUI::FontManager::getInstance().getDefaultFont() : mTextStyle.mFont);
|
|
|
|
|
|
|
|
return MyGUI::FontManager::getInstance().getByName(fontName)
|
|
|
|
|
|
|
|
->getGlyphInfo(unicodeChar)->width;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
float BookTextParser::currentFontHeight() const
|
|
|
|
return pag.getPages();
|
|
|
|
{
|
|
|
|
|
|
|
|
std::string fontName(mTextStyle.mFont == "Default" ? MyGUI::FontManager::getInstance().getDefaultFont() : mTextStyle.mFont);
|
|
|
|
|
|
|
|
return MyGUI::FontManager::getInstance().getByName(fontName)->getDefaultHeight();
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
MyGUI::IntSize BookTextParser::parsePage(std::string text, MyGUI::Widget* parent, const int width)
|
|
|
|
Paginator::Pages BookFormatter::markupToWidget(MyGUI::Widget * parent, std::string utf8Text)
|
|
|
|
{
|
|
|
|
|
|
|
|
MWScript::InterpreterContext interpreterContext(NULL, MWWorld::Ptr()); // empty arguments, because there is no locals or actor
|
|
|
|
|
|
|
|
text = Interpreter::fixDefinesBook(text, interpreterContext);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
mParent = parent;
|
|
|
|
|
|
|
|
mWidth = width;
|
|
|
|
|
|
|
|
mHeight = 0;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
assert(mParent);
|
|
|
|
|
|
|
|
while (mParent->getChildCount())
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
MyGUI::Gui::getInstance().destroyWidget(mParent->getChildAt(0));
|
|
|
|
return markupToWidget(parent, utf8Text, parent->getWidth(), parent->getHeight());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// remove trailing "
|
|
|
|
void BookFormatter::parseDiv(std::string tag)
|
|
|
|
if (text[text.size()-1] == '\"')
|
|
|
|
|
|
|
|
text.erase(text.size()-1);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
parseSubText(text);
|
|
|
|
|
|
|
|
return MyGUI::IntSize(mWidth, mHeight);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
MyGUI::IntSize BookTextParser::parseScroll(std::string text, MyGUI::Widget* parent, const int width)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
MWScript::InterpreterContext interpreterContext(NULL, MWWorld::Ptr()); // empty arguments, because there is no locals or actor
|
|
|
|
|
|
|
|
text = Interpreter::fixDefinesBook(text, interpreterContext);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
mParent = parent;
|
|
|
|
|
|
|
|
mWidth = width;
|
|
|
|
|
|
|
|
mHeight = 0;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
assert(mParent);
|
|
|
|
|
|
|
|
while (mParent->getChildCount())
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
MyGUI::Gui::getInstance().destroyWidget(mParent->getChildAt(0));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
boost::algorithm::replace_all(text, "<BR>", "\n");
|
|
|
|
|
|
|
|
boost::algorithm::replace_all(text, "<P>", "\n\n");
|
|
|
|
|
|
|
|
boost::algorithm::trim_left(text);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// remove trailing "
|
|
|
|
|
|
|
|
if (text[text.size()-1] == '\"')
|
|
|
|
|
|
|
|
text.erase(text.size()-1);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
parseSubText(text);
|
|
|
|
|
|
|
|
return MyGUI::IntSize(mWidth, mHeight);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void BookTextParser::parseImage(std::string tag, bool createWidget)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
int src_start = tag.find("SRC=")+5;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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()));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
std::string image = Misc::ResourceHelpers::correctBookartPath(tag.substr(src_start, tag.find('"', src_start)-src_start), width, height);
|
|
|
|
|
|
|
|
box->setImageTexture(image);
|
|
|
|
|
|
|
|
box->setProperty("NeedMouse", "false");
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
mWidth = std::max(mWidth, width);
|
|
|
|
|
|
|
|
mHeight += height;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void BookTextParser::parseDiv(std::string tag)
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (tag.find("ALIGN=") == std::string::npos)
|
|
|
|
if (tag.find("ALIGN=") == std::string::npos)
|
|
|
|
return;
|
|
|
|
return;
|
|
|
@ -304,17 +138,21 @@ namespace MWGui
|
|
|
|
mTextStyle.mTextAlign = MyGUI::Align::Left;
|
|
|
|
mTextStyle.mTextAlign = MyGUI::Align::Left;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void BookTextParser::parseFont(std::string tag)
|
|
|
|
void BookFormatter::parseFont(std::string tag)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (tag.find("COLOR=") != std::string::npos)
|
|
|
|
if (tag.find("COLOR=") != std::string::npos)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
int color_start = tag.find("COLOR=")+7;
|
|
|
|
int color_start = tag.find("COLOR=")+7;
|
|
|
|
std::string color = tag.substr(color_start, tag.find('"', color_start)-color_start);
|
|
|
|
|
|
|
|
|
|
|
|
int color;
|
|
|
|
|
|
|
|
std::stringstream ss;
|
|
|
|
|
|
|
|
ss << tag.substr(color_start, tag.find('"', color_start)-color_start);
|
|
|
|
|
|
|
|
ss >> std::hex >> color;
|
|
|
|
|
|
|
|
|
|
|
|
mTextStyle.mColour = MyGUI::Colour(
|
|
|
|
mTextStyle.mColour = MyGUI::Colour(
|
|
|
|
convertFromHex(color.substr(0, 2))/255.0,
|
|
|
|
(color>>16 & 0xFF) / 255.f,
|
|
|
|
convertFromHex(color.substr(2, 2))/255.0,
|
|
|
|
(color>>8 & 0xFF) / 255.f,
|
|
|
|
convertFromHex(color.substr(4, 2))/255.0);
|
|
|
|
(color & 0xFF) / 255.f);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (tag.find("FACE=") != std::string::npos)
|
|
|
|
if (tag.find("FACE=") != std::string::npos)
|
|
|
|
{
|
|
|
|
{
|
|
|
@ -330,73 +168,121 @@ namespace MWGui
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void BookTextParser::parseSubText(std::string text)
|
|
|
|
/* GraphicElement */
|
|
|
|
{
|
|
|
|
GraphicElement::GraphicElement(MyGUI::Widget * parent, Paginator & pag, const TextStyle & style)
|
|
|
|
if (text[0] == '<')
|
|
|
|
: mParent(parent), mPaginator(pag), mStyle(style)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
const size_t tagStart = 1;
|
|
|
|
|
|
|
|
const size_t tagEnd = text.find('>', tagStart);
|
|
|
|
|
|
|
|
if (tagEnd == std::string::npos)
|
|
|
|
|
|
|
|
throw std::runtime_error("BookTextParser Error: Tag is not terminated");
|
|
|
|
|
|
|
|
const std::string tag = text.substr(tagStart, tagEnd - tagStart);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (boost::algorithm::starts_with(tag, "IMG"))
|
|
|
|
|
|
|
|
parseImage(tag);
|
|
|
|
|
|
|
|
if (boost::algorithm::starts_with(tag, "FONT"))
|
|
|
|
|
|
|
|
parseFont(tag);
|
|
|
|
|
|
|
|
if (boost::algorithm::starts_with(tag, "DIV"))
|
|
|
|
|
|
|
|
parseDiv(tag);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
text.erase(0, tagEnd + 1);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
size_t tagStart = std::string::npos;
|
|
|
|
void GraphicElement::paginate()
|
|
|
|
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 != '>')
|
|
|
|
int newTop = mPaginator.getCurrentTop() + getHeight();
|
|
|
|
|
|
|
|
while (newTop-mPaginator.getStartTop() > mPaginator.getPageHeight())
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (i >= text.size())
|
|
|
|
int newStartTop = pageSplit();
|
|
|
|
throw std::runtime_error("BookTextParser Error: Tag is not terminated");
|
|
|
|
mPaginator << Paginator::Page(mPaginator.getStartTop(), newStartTop);
|
|
|
|
++i;
|
|
|
|
mPaginator.setStartTop(newStartTop);
|
|
|
|
c = text[i];
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
|
|
mPaginator.modifyCurrentTop(getHeight());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
|
|
|
|
|
|
|
|
int GraphicElement::pageSplit()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
tagStart = i;
|
|
|
|
return mPaginator.getStartTop() + mPaginator.getPageHeight();
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int GraphicElement::currentFontHeight() const
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
std::string fontName(mStyle.mFont == "Default" ? MyGUI::FontManager::getInstance().getDefaultFont() : mStyle.mFont);
|
|
|
|
|
|
|
|
return MyGUI::FontManager::getInstance().getByName(fontName)->getDefaultHeight();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
|
|
|
|
realText += c;
|
|
|
|
float GraphicElement::widthForCharGlyph(MyGUI::Char unicodeChar) const
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
std::string fontName(mStyle.mFont == "Default" ? MyGUI::FontManager::getInstance().getDefaultFont() : mStyle.mFont);
|
|
|
|
|
|
|
|
return MyGUI::FontManager::getInstance().getByName(fontName)
|
|
|
|
|
|
|
|
->getGlyphInfo(unicodeChar)->width;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
MyGUI::EditBox* box = mParent->createWidget<MyGUI::EditBox>("NormalText",
|
|
|
|
/* TextElement */
|
|
|
|
MyGUI::IntCoord(0, mHeight, mWidth, 24), MyGUI::Align::Left | MyGUI::Align::Top,
|
|
|
|
TextElement::TextElement(MyGUI::Widget * parent, Paginator & pag, const TextStyle & style, const std::string & text)
|
|
|
|
mParent->getName() + boost::lexical_cast<std::string>(mParent->getChildCount()));
|
|
|
|
: GraphicElement(parent, pag, style)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
MyGUI::EditBox* box = parent->createWidget<MyGUI::EditBox>("NormalText",
|
|
|
|
|
|
|
|
MyGUI::IntCoord(0, pag.getCurrentTop(), pag.getPageWidth(), 0), MyGUI::Align::Left | MyGUI::Align::Top,
|
|
|
|
|
|
|
|
parent->getName() + boost::lexical_cast<std::string>(parent->getChildCount()));
|
|
|
|
box->setProperty("Static", "true");
|
|
|
|
box->setProperty("Static", "true");
|
|
|
|
box->setProperty("MultiLine", "true");
|
|
|
|
box->setProperty("MultiLine", "true");
|
|
|
|
box->setProperty("WordWrap", "true");
|
|
|
|
box->setProperty("WordWrap", "true");
|
|
|
|
box->setProperty("NeedMouse", "false");
|
|
|
|
box->setProperty("NeedMouse", "false");
|
|
|
|
box->setMaxTextLength(realText.size());
|
|
|
|
box->setMaxTextLength(text.size());
|
|
|
|
box->setTextAlign(mTextStyle.mTextAlign);
|
|
|
|
box->setTextAlign(mStyle.mTextAlign);
|
|
|
|
box->setTextColour(mTextStyle.mColour);
|
|
|
|
box->setTextColour(mStyle.mColour);
|
|
|
|
box->setFontName(mTextStyle.mFont);
|
|
|
|
box->setFontName(mStyle.mFont);
|
|
|
|
box->setCaption(MyGUI::TextIterator::toTagsString(realText));
|
|
|
|
box->setCaption(MyGUI::TextIterator::toTagsString(text));
|
|
|
|
box->setSize(box->getSize().width, box->getTextSize().height);
|
|
|
|
box->setSize(box->getSize().width, box->getTextSize().height);
|
|
|
|
mHeight += box->getTextSize().height;
|
|
|
|
mEditBox = box;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int TextElement::getHeight()
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
return mEditBox->getTextSize().height;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (tagStart != std::string::npos)
|
|
|
|
int TextElement::pageSplit()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
parseSubText(text.substr(tagStart, text.size()));
|
|
|
|
// split lines
|
|
|
|
|
|
|
|
const int lineHeight = currentFontHeight();
|
|
|
|
|
|
|
|
unsigned int lastLine = (mPaginator.getStartTop() + mPaginator.getPageHeight() - mPaginator.getCurrentTop()) / lineHeight;
|
|
|
|
|
|
|
|
int ret = mPaginator.getCurrentTop() + lastLine * lineHeight;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// first empty lines that would go to the next page should be ignored
|
|
|
|
|
|
|
|
// unfortunately, getLineInfo method won't be available until 3.2.2
|
|
|
|
|
|
|
|
#if (MYGUI_VERSION >= MYGUI_DEFINE_VERSION(3, 2, 2))
|
|
|
|
|
|
|
|
const MyGUI::VectorLineInfo & lines = mEditBox->getSubWidgetText()->castType<MyGUI::EditText>()->getLineInfo();
|
|
|
|
|
|
|
|
for (unsigned int i = lastLine; i < lines.size(); ++i)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
if (lines[i].width == 0)
|
|
|
|
|
|
|
|
ret += lineHeight;
|
|
|
|
|
|
|
|
else
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* ImageElement */
|
|
|
|
|
|
|
|
ImageElement::ImageElement(MyGUI::Widget * parent, Paginator & pag, const TextStyle & style, const std::string & tag)
|
|
|
|
|
|
|
|
: GraphicElement(parent, pag, style),
|
|
|
|
|
|
|
|
mImageHeight(0)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
int src_start = tag.find("SRC=")+5;
|
|
|
|
|
|
|
|
std::string src = tag.substr(src_start, tag.find('"', src_start)-src_start);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
mImageHeight = boost::lexical_cast<int>(tag.substr(height_start, tag.find('"', height_start)-height_start));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
mImageBox = parent->createWidget<MyGUI::ImageBox> ("ImageBox",
|
|
|
|
|
|
|
|
MyGUI::IntCoord(0, pag.getCurrentTop(), width, mImageHeight), MyGUI::Align::Left | MyGUI::Align::Top,
|
|
|
|
|
|
|
|
parent->getName() + boost::lexical_cast<std::string>(parent->getChildCount()));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
std::string image = Misc::ResourceHelpers::correctBookartPath(src, width, mImageHeight);
|
|
|
|
|
|
|
|
mImageBox->setImageTexture(image);
|
|
|
|
|
|
|
|
mImageBox->setProperty("NeedMouse", "false");
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int ImageElement::getHeight()
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
return mImageHeight;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int ImageElement::pageSplit()
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
return mPaginator.getCurrentTop();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|