From 28ef4d97da898eb9708922e5947c0ec10fd04984 Mon Sep 17 00:00:00 2001 From: kpp Date: Thu, 2 May 2013 07:35:25 +0400 Subject: [PATCH] Fixes Bug #734 'Book empty line problem' Hides buttons 'Next' and 'Prev' at the last and at the first page --- apps/openmw/mwgui/bookwindow.cpp | 17 ++++++++- apps/openmw/mwgui/formatting.cpp | 58 +++++++++++++++++++++++++----- apps/openmw/mwgui/formatting.hpp | 11 +++++- apps/openmw/mwgui/scrollwindow.cpp | 2 +- 4 files changed, 77 insertions(+), 11 deletions(-) diff --git a/apps/openmw/mwgui/bookwindow.cpp b/apps/openmw/mwgui/bookwindow.cpp index 6d44e9d2c..12e5b466b 100644 --- a/apps/openmw/mwgui/bookwindow.cpp +++ b/apps/openmw/mwgui/bookwindow.cpp @@ -76,7 +76,7 @@ namespace MWGui parent = mRightPage; MyGUI::Widget* pageWidget = parent->createWidgetReal("", MyGUI::FloatCoord(0.0,0.0,1.0,1.0), MyGUI::Align::Default, "BookPage" + boost::lexical_cast(i)); - parser.parse(*it, pageWidget, mLeftPage->getSize().width); + parser.parsePage(*it, pageWidget, mLeftPage->getSize().width); mPages.push_back(pageWidget); ++i; } @@ -157,6 +157,21 @@ namespace MWGui } ++i; } + + //If it is the last page, hide the button "Next Page" + if ( (mCurrentPage+1)*2 == mPages.size() + || (mCurrentPage+1)*2 == mPages.size() + 1) + { + mNextPageButton->setVisible(false); + } else { + mNextPageButton->setVisible(true); + } + //If it is the fist page, hide the button "Prev Page" + if (mCurrentPage == 0) { + mPrevPageButton->setVisible(false); + } else { + mPrevPageButton->setVisible(true); + } } } diff --git a/apps/openmw/mwgui/formatting.cpp b/apps/openmw/mwgui/formatting.cpp index 2d72f2174..6f1e16ff1 100644 --- a/apps/openmw/mwgui/formatting.cpp +++ b/apps/openmw/mwgui/formatting.cpp @@ -6,7 +6,10 @@ #include #include +#include #include +#include +#include #include namespace @@ -74,6 +77,12 @@ namespace Ogre::UTFString string(s); 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 @@ -88,6 +97,7 @@ namespace MWGui utf8Text = Interpreter::fixDefinesBook(utf8Text, interpreterContext); boost::algorithm::replace_all(utf8Text, "\n", ""); + boost::algorithm::replace_all(utf8Text, "\r", ""); boost::algorithm::replace_all(utf8Text, "
", "\n"); boost::algorithm::replace_all(utf8Text, "

", "\n\n"); @@ -106,6 +116,14 @@ namespace MWGui size_t currentWordStart = 0; size_t index = 0; + + { + std::string texToTrim = text.asUTF8(); + boost::algorithm::trim( texToTrim ); + text = UTFString(texToTrim); + } + + while (currentHeight <= height - spacing && index < text.size()) { const UTFString::unicode_char ch = text.getChar(index); @@ -176,8 +194,10 @@ namespace MWGui result.push_back(text.substr(0, pageEnd).asUTF8()); text.erase(0, pageEnd); } - - return result; + + std::vector nonEmptyPages; + boost::copy(result | boost::adaptors::filtered(is_not_empty), std::back_inserter(nonEmptyPages)); + return nonEmptyPages; } float BookTextParser::widthForCharGlyph(unsigned unicodeChar) const @@ -193,7 +213,30 @@ namespace MWGui return MyGUI::FontManager::getInstance().getByName(fontName)->getDefaultHeight(); } - MyGUI::IntSize BookTextParser::parse(std::string text, MyGUI::Widget* parent, const int width) + MyGUI::IntSize BookTextParser::parsePage(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)); + } + + // remove trailing " + 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); @@ -209,12 +252,10 @@ namespace MWGui } boost::algorithm::replace_all(text, "\n", ""); + boost::algorithm::replace_all(text, "\r", ""); boost::algorithm::replace_all(text, "
", "\n"); boost::algorithm::replace_all(text, "

", "\n\n"); - - // remove leading newlines - // while (text[0] == '\n') - // text.erase(0); + boost::algorithm::trim_left(text); // remove trailing " if (text[text.size()-1] == '\"') @@ -223,6 +264,7 @@ namespace MWGui parseSubText(text); return MyGUI::IntSize(mWidth, mHeight); } + void BookTextParser::parseImage(std::string tag, bool createWidget) { @@ -309,7 +351,7 @@ namespace MWGui parseImage(tag); if (boost::algorithm::starts_with(tag, "FONT")) parseFont(tag); - if (boost::algorithm::starts_with(tag, "DOV")) + if (boost::algorithm::starts_with(tag, "DIV")) parseDiv(tag); text.erase(0, tagEnd + 1); diff --git a/apps/openmw/mwgui/formatting.hpp b/apps/openmw/mwgui/formatting.hpp index ab1ee3af4..a32d98fe5 100644 --- a/apps/openmw/mwgui/formatting.hpp +++ b/apps/openmw/mwgui/formatting.hpp @@ -32,7 +32,16 @@ namespace MWGui * @param maximum width * @return size of the created widgets */ - MyGUI::IntSize parse(std::string text, MyGUI::Widget* parent, const int width); + MyGUI::IntSize parsePage(std::string text, MyGUI::Widget* parent, const int width); + + /** + * Parse markup as MyGUI widgets + * @param markup to parse + * @param parent for the created widgets + * @param maximum width + * @return size of the created widgets + */ + MyGUI::IntSize parseScroll(std::string text, MyGUI::Widget* parent, const int width); /** * Split the specified text into pieces that fit in the area specified by width and height parameters diff --git a/apps/openmw/mwgui/scrollwindow.cpp b/apps/openmw/mwgui/scrollwindow.cpp index 95bf17635..b44c21c6e 100644 --- a/apps/openmw/mwgui/scrollwindow.cpp +++ b/apps/openmw/mwgui/scrollwindow.cpp @@ -39,7 +39,7 @@ namespace MWGui MWWorld::LiveCellRef *ref = mScroll.get(); BookTextParser parser; - MyGUI::IntSize size = parser.parse(ref->mBase->mText, mTextView, 390); + MyGUI::IntSize size = parser.parseScroll(ref->mBase->mText, mTextView, 390); if (size.height > mTextView->getSize().height) mTextView->setCanvasSize(MyGUI::IntSize(410, size.height));