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

387 lines
13 KiB
C++
Raw Normal View History

#include "journalviewmodel.hpp"
2013-05-04 12:15:47 +00:00
#include <map>
#include <sstream>
#include <MyGUI_LanguageManager.h>
2015-01-31 22:27:34 +00:00
#include <components/translation/translation.hpp>
#include "../mwbase/world.hpp"
#include "../mwbase/journal.hpp"
#include "../mwbase/environment.hpp"
2013-05-06 13:14:39 +00:00
#include "../mwbase/windowmanager.hpp"
#include "../mwdialogue/keywordsearch.hpp"
2013-05-06 14:04:28 +00:00
namespace MWGui {
2013-05-06 14:04:28 +00:00
struct JournalViewModelImpl;
2013-05-06 14:04:28 +00:00
struct JournalViewModelImpl : JournalViewModel
{
typedef MWDialogue::KeywordSearch <std::string, intptr_t> KeywordSearchT;
mutable bool mKeywordSearchLoaded;
2013-05-04 12:15:47 +00:00
mutable KeywordSearchT mKeywordSearch;
JournalViewModelImpl ()
{
mKeywordSearchLoaded = false;
}
virtual ~JournalViewModelImpl ()
{
}
/// \todo replace this nasty BS
static Utf8Span toUtf8Span (std::string const & str)
{
if (str.size () == 0)
return Utf8Span (Utf8Point (NULL), Utf8Point (NULL));
Utf8Point point = reinterpret_cast <Utf8Point> (str.c_str ());
return Utf8Span (point, point + str.size ());
}
void load ()
{
}
void unload ()
{
mKeywordSearch.clear ();
mKeywordSearchLoaded = false;
}
void ensureKeyWordSearchLoaded () const
{
if (!mKeywordSearchLoaded)
{
2013-02-09 18:44:20 +00:00
MWBase::Journal * journal = MWBase::Environment::get().getJournal();
2013-02-09 18:44:20 +00:00
for(MWBase::Journal::TTopicIter i = journal->topicBegin(); i != journal->topicEnd (); ++i)
mKeywordSearch.seed (i->first, intptr_t (&i->second));
mKeywordSearchLoaded = true;
}
}
bool isEmpty () const
{
2013-02-09 18:44:20 +00:00
MWBase::Journal * journal = MWBase::Environment::get().getJournal();
return journal->begin () == journal->end ();
}
2013-05-06 14:05:56 +00:00
template <typename t_iterator, typename Interface>
struct BaseEntry : Interface
{
2013-02-09 18:44:20 +00:00
typedef t_iterator iterator_t;
iterator_t itr;
JournalViewModelImpl const * mModel;
BaseEntry (JournalViewModelImpl const * model, iterator_t itr) :
itr (itr), mModel (model), loaded (false)
{}
virtual ~BaseEntry () {}
mutable bool loaded;
mutable std::string utf8text;
2013-05-06 13:14:39 +00:00
typedef std::pair<size_t, size_t> Range;
// hyperlinks in @link# notation
mutable std::map<Range, intptr_t> mHyperLinks;
virtual std::string getText () const = 0;
void ensureLoaded () const
{
if (!loaded)
{
2013-05-06 13:14:39 +00:00
mModel->ensureKeyWordSearchLoaded ();
utf8text = getText ();
2013-05-06 13:14:39 +00:00
2014-09-26 15:12:48 +00:00
size_t pos_end = 0;
2013-05-06 13:14:39 +00:00
for(;;)
{
2014-09-26 15:12:48 +00:00
size_t pos_begin = utf8text.find('@');
2013-05-06 13:14:39 +00:00
if (pos_begin != std::string::npos)
pos_end = utf8text.find('#', pos_begin);
if (pos_begin != std::string::npos && pos_end != std::string::npos)
{
std::string link = utf8text.substr(pos_begin + 1, pos_end - pos_begin - 1);
const char specialPseudoAsteriskCharacter = 127;
std::replace(link.begin(), link.end(), specialPseudoAsteriskCharacter, '*');
std::string topicName = MWBase::Environment::get().getWindowManager()->
getTranslationDataStorage().topicStandardForm(link);
std::string displayName = link;
while (displayName[displayName.size()-1] == '*')
displayName.erase(displayName.size()-1, 1);
utf8text.replace(pos_begin, pos_end+1-pos_begin, displayName);
intptr_t value;
if (mModel->mKeywordSearch.containsKeyword(topicName, value))
mHyperLinks[std::make_pair(pos_begin, pos_begin+displayName.size())] = value;
}
else
break;
}
loaded = true;
}
}
Utf8Span body () const
{
ensureLoaded ();
return toUtf8Span (utf8text);
}
2017-05-06 22:04:46 +00:00
void visitSpans (std::function < void (TopicId, size_t, size_t)> visitor) const
{
ensureLoaded ();
mModel->ensureKeyWordSearchLoaded ();
2013-05-06 13:14:39 +00:00
if (mHyperLinks.size() && MWBase::Environment::get().getWindowManager()->getTranslationDataStorage().hasTranslation())
{
size_t formatted = 0; // points to the first character that is not laid out yet
for (std::map<Range, intptr_t>::const_iterator it = mHyperLinks.begin(); it != mHyperLinks.end(); ++it)
{
intptr_t topicId = it->second;
if (formatted < it->first.first)
visitor (0, formatted, it->first.first);
visitor (topicId, it->first.first, it->first.second);
formatted = it->first.second;
}
if (formatted < utf8text.size())
visitor (0, formatted, utf8text.size());
}
else
{
std::vector<KeywordSearchT::Match> matches;
mModel->mKeywordSearch.highlightKeywords(utf8text.begin(), utf8text.end(), matches);
std::string::const_iterator i = utf8text.begin ();
for (std::vector<KeywordSearchT::Match>::const_iterator it = matches.begin(); it != matches.end(); ++it)
2013-05-06 13:14:39 +00:00
{
const KeywordSearchT::Match& match = *it;
2013-05-06 13:14:39 +00:00
if (i != match.mBeg)
visitor (0, i - utf8text.begin (), match.mBeg - utf8text.begin ());
2013-05-06 13:14:39 +00:00
visitor (match.mValue, match.mBeg - utf8text.begin (), match.mEnd - utf8text.begin ());
2013-05-06 13:14:39 +00:00
i = match.mEnd;
}
2013-05-06 13:14:39 +00:00
if (i != utf8text.end ())
visitor (0, i - utf8text.begin (), utf8text.size ());
}
}
};
2017-05-06 22:04:46 +00:00
void visitQuestNames (bool active_only, std::function <void (const std::string&, bool)> visitor) const
{
2013-02-09 18:44:20 +00:00
MWBase::Journal * journal = MWBase::Environment::get ().getJournal ();
std::set<std::string> visitedQuests;
// Note that for purposes of the journal GUI, quests are identified by the name, not the ID, so several
// different quest IDs can end up in the same quest log. A quest log should be considered finished
// when any quest ID in that log is finished.
2013-02-09 18:44:20 +00:00
for (MWBase::Journal::TQuestIter i = journal->questBegin (); i != journal->questEnd (); ++i)
{
const MWDialogue::Quest& quest = i->second;
bool isFinished = false;
for (MWBase::Journal::TQuestIter j = journal->questBegin (); j != journal->questEnd (); ++j)
{
if (quest.getName() == j->second.getName() && j->second.isFinished())
isFinished = true;
}
if (active_only && isFinished)
continue;
// Unfortunately Morrowind.esm has no quest names, since the quest book was added with tribunal.
// Note that even with Tribunal, some quests still don't have quest names. I'm assuming those are not supposed
// to appear in the quest book.
if (!quest.getName().empty())
{
// Don't list the same quest name twice
if (visitedQuests.find(quest.getName()) != visitedQuests.end())
continue;
visitor (quest.getName(), isFinished);
visitedQuests.insert(quest.getName());
}
}
}
template <typename iterator_t>
struct JournalEntryImpl : BaseEntry <iterator_t, JournalEntry>
{
using BaseEntry <iterator_t, JournalEntry>::itr;
mutable std::string timestamp_buffer;
2013-05-04 12:15:47 +00:00
JournalEntryImpl (JournalViewModelImpl const * model, iterator_t itr) :
BaseEntry <iterator_t, JournalEntry> (model, itr)
{}
std::string getText () const
{
return itr->getText();
}
Utf8Span timestamp () const
{
if (timestamp_buffer.empty ())
{
std::string dayStr = MyGUI::LanguageManager::getInstance().replaceTags("#{sDay}");
std::ostringstream os;
os
<< itr->mDayOfMonth << ' '
<< MWBase::Environment::get().getWorld()->getMonthName (itr->mMonth)
<< " (" << dayStr << " " << (itr->mDay) << ')';
timestamp_buffer = os.str ();
}
return toUtf8Span (timestamp_buffer);
}
};
2017-05-06 22:04:46 +00:00
void visitJournalEntries (const std::string& questName, std::function <void (JournalEntry const &)> visitor) const
{
2013-02-09 18:44:20 +00:00
MWBase::Journal * journal = MWBase::Environment::get().getJournal();
if (!questName.empty())
{
std::vector<MWDialogue::Quest const*> quests;
for (MWBase::Journal::TQuestIter questIt = journal->questBegin(); questIt != journal->questEnd(); ++questIt)
{
if (Misc::StringUtils::ciEqual(questIt->second.getName(), questName))
quests.push_back(&questIt->second);
}
2013-02-09 18:44:20 +00:00
for(MWBase::Journal::TEntryIter i = journal->begin(); i != journal->end (); ++i)
{
for (std::vector<MWDialogue::Quest const*>::iterator questIt = quests.begin(); questIt != quests.end(); ++questIt)
{
MWDialogue::Quest const* quest = *questIt;
for (MWDialogue::Topic::TEntryIter j = quest->begin (); j != quest->end (); ++j)
{
if (i->mInfoId == j->mInfoId)
visitor (JournalEntryImpl <MWBase::Journal::TEntryIter> (this, i));
}
}
}
}
else
{
2013-02-09 18:44:20 +00:00
for(MWBase::Journal::TEntryIter i = journal->begin(); i != journal->end (); ++i)
visitor (JournalEntryImpl <MWBase::Journal::TEntryIter> (this, i));
}
}
2017-05-06 22:04:46 +00:00
void visitTopicName (TopicId topicId, std::function <void (Utf8Span)> visitor) const
{
MWDialogue::Topic const & topic = * reinterpret_cast <MWDialogue::Topic const *> (topicId);
visitor (toUtf8Span (topic.getName()));
}
2017-08-04 16:21:13 +00:00
void visitTopicNamesStartingWith (int index, std::function < void (const std::string&) > visitor) const
{
2013-02-09 18:44:20 +00:00
MWBase::Journal * journal = MWBase::Environment::get().getJournal();
2013-02-09 18:44:20 +00:00
for (MWBase::Journal::TTopicIter i = journal->topicBegin (); i != journal->topicEnd (); ++i)
{
2017-08-04 16:21:13 +00:00
unsigned char byte1 = i->first[0];
// First, check for two-byte UTF-8 symbols, e.g. Cyrillic ones
// TODO: check which language journal index is using
if ((byte1 == 0xd0 || byte1 == 0xd1) && i->first.length() >= 2)
{
unsigned char byte2 = i->first[1];
2017-08-04 16:21:13 +00:00
std::pair<unsigned char, unsigned char> symbol = Misc::StringUtils::toLower(byte1, byte2);
2017-08-04 16:21:13 +00:00
// CYRILLIC LETTER A - CYRILLIC LETTER PE
// index from 1 to 16
if (symbol.first == 0xd0 && symbol.second >= (0xaf + index) && symbol.second < (0xbf + index) && symbol.second == (0xaf + index))
{
visitor (i->second.getName());
continue;
}
2017-08-04 16:21:13 +00:00
// CYRILLIC LETTERL R - CYRILLIC LETTER YA
// index from 17 to 32
if (symbol.first == 0xd1 && symbol.second >= (0x6f + index) && symbol.second < (0x7f + index) && symbol.second == (0x6f + index))
{
visitor (i->second.getName());
continue;
}
}
else
{
// Otherwise check for regular Latin symbols, 0x61 = 'a'
if (i->first [0] != 0x60 + index)
continue;
visitor (i->second.getName());
}
}
}
struct TopicEntryImpl : BaseEntry <MWDialogue::Topic::TEntryIter, TopicEntry>
{
MWDialogue::Topic const & mTopic;
TopicEntryImpl (JournalViewModelImpl const * model, MWDialogue::Topic const & topic, iterator_t itr) :
BaseEntry (model, itr), mTopic (topic)
2013-02-09 18:44:20 +00:00
{}
2013-02-09 18:44:20 +00:00
std::string getText () const
{
return itr->getText();
2013-02-09 18:44:20 +00:00
}
Utf8Span source () const
2013-02-09 18:44:20 +00:00
{
return toUtf8Span (itr->mActorName);
2013-02-09 18:44:20 +00:00
}
2013-02-09 18:44:20 +00:00
};
2017-05-06 22:04:46 +00:00
void visitTopicEntries (TopicId topicId, std::function <void (TopicEntry const &)> visitor) const
2013-02-09 18:44:20 +00:00
{
typedef MWDialogue::Topic::TEntryIter iterator_t;
MWDialogue::Topic const & topic = * reinterpret_cast <MWDialogue::Topic const *> (topicId);
for (iterator_t i = topic.begin (); i != topic.end (); ++i)
visitor (TopicEntryImpl (this, topic, i));
}
};
JournalViewModel::Ptr JournalViewModel::create ()
{
return std::make_shared <JournalViewModelImpl> ();
}
2013-05-06 14:04:28 +00:00
}