1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-25 15:35:23 +00:00
OpenMW/apps/openmw/mwgui/dialogue.cpp

786 lines
31 KiB
C++
Raw Normal View History

2010-11-03 21:21:08 +01:00
#include "dialogue.hpp"
2015-01-10 02:50:43 +01:00
#include <MyGUI_LanguageManager.h>
#include <MyGUI_Window.h>
#include <MyGUI_ProgressBar.h>
2017-09-27 12:40:47 +02:00
#include <MyGUI_ScrollBar.h>
#include <MyGUI_Button.h>
2015-01-10 02:50:43 +01:00
2018-08-14 23:05:43 +04:00
#include <components/debug/debuglog.hpp>
#include <components/widgets/list.hpp>
2015-03-11 20:04:25 +01:00
#include <components/translation/translation.hpp>
2012-05-17 17:15:44 +02:00
#include "../mwbase/environment.hpp"
#include "../mwbase/windowmanager.hpp"
#include "../mwbase/mechanicsmanager.hpp"
#include "../mwbase/world.hpp"
2015-03-11 20:04:25 +01:00
#include "../mwbase/dialoguemanager.hpp"
2012-05-17 17:15:44 +02:00
2013-05-11 18:38:27 +02:00
#include "../mwworld/class.hpp"
#include "../mwworld/containerstore.hpp"
#include "../mwworld/esmstore.hpp"
2013-05-11 18:38:27 +02:00
2015-07-18 19:40:31 +02:00
#include "../mwmechanics/creaturestats.hpp"
2015-08-21 21:12:39 +12:00
#include "../mwmechanics/actorutil.hpp"
2015-07-18 19:40:31 +02:00
2013-05-04 14:15:47 +02:00
#include "bookpage.hpp"
2017-07-24 13:25:01 +02:00
#include "textcolours.hpp"
2012-05-17 17:15:44 +02:00
#include "journalbooks.hpp" // to_utf8_span
2012-11-09 20:18:38 +01:00
2013-04-17 18:56:48 -04:00
namespace MWGui
2012-11-09 20:18:38 +01:00
{
class ResponseCallback : public MWBase::DialogueManager::ResponseCallback
{
public:
ResponseCallback(DialogueWindow* win, bool needMargin=true)
: mWindow(win)
, mNeedMargin(needMargin)
{
}
void addResponse(const std::string& title, const std::string& text) override
{
mWindow->addResponse(title, text, mNeedMargin);
}
void updateTopics()
{
mWindow->updateTopics();
}
private:
DialogueWindow* mWindow;
bool mNeedMargin;
};
PersuasionDialog::PersuasionDialog(ResponseCallback* callback)
2013-04-17 18:56:48 -04:00
: WindowModal("openmw_persuasion_dialog.layout")
, mCallback(callback)
2012-11-09 20:18:38 +01:00
{
2013-04-17 18:56:48 -04:00
getWidget(mCancelButton, "CancelButton");
getWidget(mAdmireButton, "AdmireButton");
getWidget(mIntimidateButton, "IntimidateButton");
getWidget(mTauntButton, "TauntButton");
getWidget(mBribe10Button, "Bribe10Button");
getWidget(mBribe100Button, "Bribe100Button");
getWidget(mBribe1000Button, "Bribe1000Button");
getWidget(mGoldLabel, "GoldLabel");
mCancelButton->eventMouseButtonClick += MyGUI::newDelegate(this, &PersuasionDialog::onCancel);
mAdmireButton->eventMouseButtonClick += MyGUI::newDelegate(this, &PersuasionDialog::onPersuade);
mIntimidateButton->eventMouseButtonClick += MyGUI::newDelegate(this, &PersuasionDialog::onPersuade);
mTauntButton->eventMouseButtonClick += MyGUI::newDelegate(this, &PersuasionDialog::onPersuade);
mBribe10Button->eventMouseButtonClick += MyGUI::newDelegate(this, &PersuasionDialog::onPersuade);
mBribe100Button->eventMouseButtonClick += MyGUI::newDelegate(this, &PersuasionDialog::onPersuade);
mBribe1000Button->eventMouseButtonClick += MyGUI::newDelegate(this, &PersuasionDialog::onPersuade);
2012-11-09 20:18:38 +01:00
}
2013-04-17 18:56:48 -04:00
void PersuasionDialog::onCancel(MyGUI::Widget *sender)
2012-11-09 20:18:38 +01:00
{
2017-09-23 12:18:39 +02:00
setVisible(false);
2012-11-09 20:18:38 +01:00
}
2013-04-17 18:56:48 -04:00
void PersuasionDialog::onPersuade(MyGUI::Widget *sender)
{
MWBase::MechanicsManager::PersuasionType type;
if (sender == mAdmireButton) type = MWBase::MechanicsManager::PT_Admire;
else if (sender == mIntimidateButton) type = MWBase::MechanicsManager::PT_Intimidate;
else if (sender == mTauntButton) type = MWBase::MechanicsManager::PT_Taunt;
else if (sender == mBribe10Button)
type = MWBase::MechanicsManager::PT_Bribe10;
else if (sender == mBribe100Button)
type = MWBase::MechanicsManager::PT_Bribe100;
else /*if (sender == mBribe1000Button)*/
type = MWBase::MechanicsManager::PT_Bribe1000;
2012-11-10 00:29:36 +01:00
MWBase::Environment::get().getDialogueManager()->persuade(type, mCallback.get());
mCallback->updateTopics();
2012-11-09 20:18:38 +01:00
2013-04-17 18:56:48 -04:00
setVisible(false);
}
2012-11-09 20:18:38 +01:00
void PersuasionDialog::onOpen()
2013-04-17 18:56:48 -04:00
{
center();
2012-11-09 20:18:38 +01:00
2015-08-21 21:12:39 +12:00
MWWorld::Ptr player = MWMechanics::getPlayer();
int playerGold = player.getClass().getContainerStore(player).count(MWWorld::ContainerStore::sGoldId);
2012-11-09 20:18:38 +01:00
2013-04-17 18:56:48 -04:00
mBribe10Button->setEnabled (playerGold >= 10);
mBribe100Button->setEnabled (playerGold >= 100);
mBribe1000Button->setEnabled (playerGold >= 1000);
2012-11-09 20:18:38 +01:00
mGoldLabel->setCaptionWithReplacing("#{sGold}: " + MyGUI::utility::toString(playerGold));
WindowModal::onOpen();
}
MyGUI::Widget* PersuasionDialog::getDefaultKeyFocus()
{
return mAdmireButton;
2013-04-17 18:56:48 -04:00
}
2012-11-09 20:18:38 +01:00
2013-04-17 18:56:48 -04:00
// --------------------------------------------------------------------------------------------------
2010-11-03 21:21:08 +01:00
Response::Response(const std::string &text, const std::string &title, bool needMargin)
: mTitle(title), mNeedMargin(needMargin)
2013-05-04 14:15:47 +02:00
{
mText = text;
}
void Response::write(BookTypesetter::Ptr typesetter, KeywordSearchT* keywordSearch, std::map<std::string, Link*>& topicLinks) const
2013-05-04 14:15:47 +02:00
{
typesetter->sectionBreak(mNeedMargin ? 9 : 0);
2013-05-04 14:15:47 +02:00
if (mTitle != "")
{
2017-07-24 13:25:01 +02:00
const MyGUI::Colour& headerColour = MWBase::Environment::get().getWindowManager()->getTextColours().header;
BookTypesetter::Style* title = typesetter->createStyle("", headerColour, false);
2013-05-04 14:15:47 +02:00
typesetter->write(title, to_utf8_span(mTitle.c_str()));
typesetter->sectionBreak();
}
2013-05-04 14:15:47 +02:00
typedef std::pair<size_t, size_t> Range;
std::map<Range, intptr_t> hyperLinks;
// We need this copy for when @# hyperlinks are replaced
std::string text = mText;
size_t pos_end = std::string::npos;
2013-05-04 14:15:47 +02:00
for(;;)
{
2014-09-26 17:12:48 +02:00
size_t pos_begin = text.find('@');
2013-05-04 14:15:47 +02:00
if (pos_begin != std::string::npos)
pos_end = text.find('#', pos_begin);
2013-05-04 14:15:47 +02:00
if (pos_begin != std::string::npos && pos_end != std::string::npos)
{
std::string link = text.substr(pos_begin + 1, pos_end - pos_begin - 1);
2013-05-04 14:15:47 +02:00
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);
2013-05-04 14:15:47 +02:00
text.replace(pos_begin, pos_end+1-pos_begin, displayName);
2013-05-04 14:15:47 +02:00
if (topicLinks.find(Misc::StringUtils::lowerCase(topicName)) != topicLinks.end())
hyperLinks[std::make_pair(pos_begin, pos_begin+displayName.size())] = intptr_t(topicLinks[Misc::StringUtils::lowerCase(topicName)]);
2013-05-04 14:15:47 +02:00
}
2013-05-04 14:18:13 +02:00
else
break;
2013-05-04 14:15:47 +02:00
}
typesetter->addContent(to_utf8_span(text.c_str()));
2013-05-04 14:15:47 +02:00
if (hyperLinks.size() && MWBase::Environment::get().getWindowManager()->getTranslationDataStorage().hasTranslation())
2013-05-04 14:15:47 +02:00
{
2017-07-24 13:25:01 +02:00
const TextColours& textColours = MWBase::Environment::get().getWindowManager()->getTextColours();
BookTypesetter::Style* style = typesetter->createStyle("", textColours.normal, false);
size_t formatted = 0; // points to the first character that is not laid out yet
for (auto& hyperLink : hyperLinks)
{
intptr_t topicId = hyperLink.second;
2017-07-24 13:25:01 +02:00
BookTypesetter::Style* hotStyle = typesetter->createHotStyle (style, textColours.link,
textColours.linkOver, textColours.linkPressed,
topicId);
if (formatted < hyperLink.first.first)
typesetter->write(style, formatted, hyperLink.first.first);
typesetter->write(hotStyle, hyperLink.first.first, hyperLink.first.second);
formatted = hyperLink.first.second;
}
if (formatted < text.size())
typesetter->write(style, formatted, text.size());
2013-05-04 14:15:47 +02:00
}
else
2013-05-04 14:15:47 +02:00
{
std::vector<KeywordSearchT::Match> matches;
keywordSearch->highlightKeywords(text.begin(), text.end(), matches);
std::string::const_iterator i = text.begin ();
for (KeywordSearchT::Match& match : matches)
{
if (i != match.mBeg)
addTopicLink (typesetter, 0, i - text.begin (), match.mBeg - text.begin ());
2013-05-04 14:15:47 +02:00
addTopicLink (typesetter, match.mValue, match.mBeg - text.begin (), match.mEnd - text.begin ());
2013-05-04 14:15:47 +02:00
i = match.mEnd;
}
if (i != text.end ())
addTopicLink (typesetter, 0, i - text.begin (), text.size ());
}
2013-05-04 14:15:47 +02:00
}
void Response::addTopicLink(BookTypesetter::Ptr typesetter, intptr_t topicId, size_t begin, size_t end) const
2013-05-04 14:15:47 +02:00
{
2017-07-24 13:25:01 +02:00
const TextColours& textColours = MWBase::Environment::get().getWindowManager()->getTextColours();
BookTypesetter::Style* style = typesetter->createStyle("", textColours.normal, false);
2013-05-04 15:15:44 +02:00
2013-05-04 14:15:47 +02:00
if (topicId)
2017-07-24 13:25:01 +02:00
style = typesetter->createHotStyle (style, textColours.link, textColours.linkOver, textColours.linkPressed, topicId);
2013-05-04 14:15:47 +02:00
typesetter->write (style, begin, end);
}
2013-05-04 15:15:44 +02:00
Message::Message(const std::string& text)
{
mText = text;
}
void Message::write(BookTypesetter::Ptr typesetter, KeywordSearchT* keywordSearch, std::map<std::string, Link*>& topicLinks) const
2013-05-04 15:15:44 +02:00
{
2017-07-24 13:25:01 +02:00
const MyGUI::Colour& textColour = MWBase::Environment::get().getWindowManager()->getTextColours().notify;
BookTypesetter::Style* title = typesetter->createStyle("", textColour, false);
2013-05-04 15:15:44 +02:00
typesetter->sectionBreak(9);
typesetter->write(title, to_utf8_span(mText.c_str()));
}
2013-05-04 14:15:47 +02:00
// --------------------------------------------------------------------------------------------------
void Choice::activated()
{
MWBase::Environment::get().getWindowManager()->playSound("Menu Click");
eventChoiceActivated(mChoiceId);
2013-05-04 14:15:47 +02:00
}
void Topic::activated()
{
MWBase::Environment::get().getWindowManager()->playSound("Menu Click");
eventTopicActivated(mTopicId);
2013-05-04 14:15:47 +02:00
}
2013-05-04 15:15:44 +02:00
void Goodbye::activated()
{
MWBase::Environment::get().getWindowManager()->playSound("Menu Click");
eventActivated();
2013-05-04 15:15:44 +02:00
}
2013-05-04 14:15:47 +02:00
// --------------------------------------------------------------------------------------------------
2013-04-17 18:56:48 -04:00
DialogueWindow::DialogueWindow()
: WindowBase("openmw_dialogue_window.layout")
, mIsCompanion(false)
2013-05-04 15:15:44 +02:00
, mGoodbye(false)
, mPersuasionDialog(new ResponseCallback(this))
, mCallback(new ResponseCallback(this))
, mGreetingCallback(new ResponseCallback(this, false))
2013-04-17 18:56:48 -04:00
{
// Centre dialog
center();
2012-11-09 20:18:38 +01:00
2013-04-17 18:56:48 -04:00
mPersuasionDialog.setVisible(false);
2013-04-17 18:56:48 -04:00
//History view
getWidget(mHistory, "History");
2010-11-03 21:21:08 +01:00
2013-04-17 18:56:48 -04:00
//Topics list
getWidget(mTopicsList, "TopicsList");
mTopicsList->eventItemSelected += MyGUI::newDelegate(this, &DialogueWindow::onSelectListItem);
2010-11-03 21:21:08 +01:00
getWidget(mGoodbyeButton, "ByeButton");
mGoodbyeButton->eventMouseButtonClick += MyGUI::newDelegate(this, &DialogueWindow::onByeClicked);
2012-05-10 11:19:22 +02:00
2013-04-17 18:56:48 -04:00
getWidget(mDispositionBar, "Disposition");
getWidget(mDispositionText,"DispositionText");
2013-05-04 14:15:47 +02:00
getWidget(mScrollBar, "VScroll");
mScrollBar->eventScrollChangePosition += MyGUI::newDelegate(this, &DialogueWindow::onScrollbarMoved);
mHistory->eventMouseWheel += MyGUI::newDelegate(this, &DialogueWindow::onMouseWheel);
2017-05-06 15:12:06 +02:00
BookPage::ClickCallback callback = std::bind (&DialogueWindow::notifyLinkClicked, this, std::placeholders::_1);
2013-05-04 14:15:47 +02:00
mHistory->adviseLinkClicked(callback);
2010-11-03 21:21:08 +01:00
2014-09-13 04:07:40 +02:00
mMainWidget->castType<MyGUI::Window>()->eventWindowChangeCoord += MyGUI::newDelegate(this, &DialogueWindow::onWindowResize);
2013-04-17 18:56:48 -04:00
}
2010-11-03 21:21:08 +01:00
DialogueWindow::~DialogueWindow()
{
deleteLater();
for (Link* link : mLinks)
delete link;
for (auto link : mTopicLinks)
delete link.second;
for (auto history : mHistoryContents)
delete history;
}
void DialogueWindow::onTradeComplete()
{
addResponse("", MyGUI::LanguageManager::getInstance().replaceTags("#{sBarterDialog5}"));
}
2017-09-23 12:18:39 +02:00
bool DialogueWindow::exit()
{
if ((MWBase::Environment::get().getDialogueManager()->isInChoice()))
{
2017-09-23 12:18:39 +02:00
return false;
}
else
{
2017-09-25 19:58:34 +02:00
resetReference();
MWBase::Environment::get().getDialogueManager()->goodbyeSelected();
mTopicsList->scrollToTop();
2017-09-23 12:18:39 +02:00
return true;
}
}
2013-04-17 18:56:48 -04:00
void DialogueWindow::onWindowResize(MyGUI::Window* _sender)
{
2017-07-24 13:25:01 +02:00
// if the window has only been moved, not resized, we don't need to update
if (mCurrentWindowSize == _sender->getSize()) return;
2013-04-17 18:56:48 -04:00
mTopicsList->adjustSize();
2013-05-04 14:15:47 +02:00
updateHistory();
updateTopicFormat();
2017-07-24 13:25:01 +02:00
mCurrentWindowSize = _sender->getSize();
2013-04-17 18:56:48 -04:00
}
2010-11-03 21:21:08 +01:00
2013-04-17 18:56:48 -04:00
void DialogueWindow::onMouseWheel(MyGUI::Widget* _sender, int _rel)
{
2013-05-04 14:15:47 +02:00
if (!mScrollBar->getVisible())
return;
mScrollBar->setScrollPosition(std::min(static_cast<int>(mScrollBar->getScrollRange()-1),
std::max(0, static_cast<int>(mScrollBar->getScrollPosition() - _rel*0.3))));
onScrollbarMoved(mScrollBar, mScrollBar->getScrollPosition());
2013-04-17 18:56:48 -04:00
}
2013-04-17 18:56:48 -04:00
void DialogueWindow::onByeClicked(MyGUI::Widget* _sender)
2012-09-26 18:30:47 +02:00
{
2017-09-23 12:18:39 +02:00
if (exit())
{
2017-09-23 12:18:39 +02:00
MWBase::Environment::get().getWindowManager()->removeGuiMode(GM_Dialogue);
}
2012-09-26 18:30:47 +02:00
}
void DialogueWindow::onSelectListItem(const std::string& topic, int id)
2012-10-17 18:03:02 +02:00
{
MWBase::DialogueManager* dialogueManager = MWBase::Environment::get().getDialogueManager();
if (mGoodbye || dialogueManager->isInChoice())
2013-04-17 18:56:48 -04:00
return;
const MWWorld::Store<ESM::GameSetting> &gmst = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>();
2018-08-29 18:38:12 +03:00
const std::string sPersuasion = gmst.find("sPersuasion")->mValue.getString();
const std::string sCompanionShare = gmst.find("sCompanionShare")->mValue.getString();
const std::string sBarter = gmst.find("sBarter")->mValue.getString();
const std::string sSpells = gmst.find("sSpells")->mValue.getString();
const std::string sTravel = gmst.find("sTravel")->mValue.getString();
const std::string sSpellMakingMenuTitle = gmst.find("sSpellMakingMenuTitle")->mValue.getString();
const std::string sEnchanting = gmst.find("sEnchanting")->mValue.getString();
const std::string sServiceTrainingTitle = gmst.find("sServiceTrainingTitle")->mValue.getString();
const std::string sRepair = gmst.find("sRepair")->mValue.getString();
if (topic != sPersuasion && topic != sCompanionShare && topic != sBarter
&& topic != sSpells && topic != sTravel && topic != sSpellMakingMenuTitle
&& topic != sEnchanting && topic != sServiceTrainingTitle && topic != sRepair)
{
onTopicActivated(topic);
2017-09-28 17:00:07 +00:00
if (mGoodbyeButton->getEnabled())
MWBase::Environment::get().getWindowManager()->setKeyFocusWidget(mGoodbyeButton);
}
else if (topic == sPersuasion)
mPersuasionDialog.setVisible(true);
else if (topic == sCompanionShare)
MWBase::Environment::get().getWindowManager()->pushGuiMode(GM_Companion, mPtr);
else if (!dialogueManager->checkServiceRefused(mCallback.get()))
{
if (topic == sBarter && !dialogueManager->checkServiceRefused(mCallback.get(), MWBase::DialogueManager::Barter))
MWBase::Environment::get().getWindowManager()->pushGuiMode(GM_Barter, mPtr);
else if (topic == sSpells && !dialogueManager->checkServiceRefused(mCallback.get(), MWBase::DialogueManager::Spells))
MWBase::Environment::get().getWindowManager()->pushGuiMode(GM_SpellBuying, mPtr);
else if (topic == sTravel && !dialogueManager->checkServiceRefused(mCallback.get(), MWBase::DialogueManager::Travel))
MWBase::Environment::get().getWindowManager()->pushGuiMode(GM_Travel, mPtr);
else if (topic == sSpellMakingMenuTitle && !dialogueManager->checkServiceRefused(mCallback.get(), MWBase::DialogueManager::Spellmaking))
MWBase::Environment::get().getWindowManager()->pushGuiMode(GM_SpellCreation, mPtr);
else if (topic == sEnchanting && !dialogueManager->checkServiceRefused(mCallback.get(), MWBase::DialogueManager::Enchanting))
MWBase::Environment::get().getWindowManager()->pushGuiMode(GM_Enchanting, mPtr);
else if (topic == sServiceTrainingTitle && !dialogueManager->checkServiceRefused(mCallback.get(), MWBase::DialogueManager::Training))
MWBase::Environment::get().getWindowManager()->pushGuiMode(GM_Training, mPtr);
else if (topic == sRepair && !dialogueManager->checkServiceRefused(mCallback.get(), MWBase::DialogueManager::Repair))
MWBase::Environment::get().getWindowManager()->pushGuiMode(GM_MerchantRepair, mPtr);
}
else
updateTopics();
2012-10-17 18:03:02 +02:00
}
2010-11-03 21:21:08 +01:00
void DialogueWindow::setPtr(const MWWorld::Ptr& actor)
2013-04-17 18:56:48 -04:00
{
if (!actor.getClass().isActor())
{
2018-08-14 23:05:43 +04:00
Log(Debug::Warning) << "Warning: can not talk with non-actor object.";
return;
}
bool sameActor = (mPtr == actor);
if (!sameActor)
{
// The history is not reset here
mKeywords.clear();
mTopicsList->clear();
for (Link* link : mLinks)
mDeleteLater.push_back(link); // Links are not deleted right away to prevent issues with event handlers
mLinks.clear();
}
mPtr = actor;
mGoodbye = false;
mTopicsList->setEnabled(true);
if (!MWBase::Environment::get().getDialogueManager()->startDialogue(actor, mGreetingCallback.get()))
{
// No greetings found. The dialogue window should not be shown.
// If this is a companion, we must show the companion window directly (used by BM_bear_be_unique).
2017-10-03 22:07:56 +00:00
MWBase::Environment::get().getWindowManager()->removeGuiMode(MWGui::GM_Dialogue);
mPtr = MWWorld::Ptr();
if (isCompanion(actor))
MWBase::Environment::get().getWindowManager()->pushGuiMode(MWGui::GM_Companion, actor);
return;
}
MWBase::Environment::get().getWindowManager()->setKeyFocusWidget(mGoodbyeButton);
setTitle(mPtr.getClass().getName(mPtr));
updateTopics();
updateTopicsPane(); // force update for new services
updateDisposition();
restock();
}
void DialogueWindow::restock()
{
MWMechanics::CreatureStats &sellerStats = mPtr.getClass().getCreatureStats(mPtr);
2018-08-29 18:38:12 +03:00
float delay = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find("fBarterGoldResetDelay")->mValue.getFloat();
// Gold is restocked every 24h
if (MWBase::Environment::get().getWorld()->getTimeStamp() >= sellerStats.getLastRestockTime() + delay)
{
sellerStats.setGoldPool(mPtr.getClass().getBaseGold(mPtr));
sellerStats.setLastRestockTime(MWBase::Environment::get().getWorld()->getTimeStamp());
}
2013-04-17 18:56:48 -04:00
}
void DialogueWindow::deleteLater()
{
for (Link* link : mDeleteLater)
delete link;
mDeleteLater.clear();
}
void DialogueWindow::onClose()
{
if (MWBase::Environment::get().getWindowManager()->containsMode(GM_Dialogue))
return;
// Reset history
for (DialogueText* text : mHistoryContents)
delete text;
mHistoryContents.clear();
}
bool DialogueWindow::setKeywords(std::list<std::string> keyWords)
{
if (mKeywords == keyWords && isCompanion() == mIsCompanion)
return false;
mIsCompanion = isCompanion();
mKeywords = keyWords;
updateTopicsPane();
return true;
}
void DialogueWindow::updateTopicsPane()
2013-04-17 18:56:48 -04:00
{
mTopicsList->clear();
for (auto& linkPair : mTopicLinks)
mDeleteLater.push_back(linkPair.second);
2013-05-04 14:15:47 +02:00
mTopicLinks.clear();
mKeywordSearch.clear();
2017-09-25 19:52:20 +02:00
int services = mPtr.getClass().getServices(mPtr);
bool travel = (mPtr.getTypeName() == typeid(ESM::NPC).name() && !mPtr.get<ESM::NPC>()->mBase->getTransport().empty())
|| (mPtr.getTypeName() == typeid(ESM::Creature).name() && !mPtr.get<ESM::Creature>()->mBase->getTransport().empty());
2013-04-17 18:56:48 -04:00
const MWWorld::Store<ESM::GameSetting> &gmst =
MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>();
2013-04-17 18:56:48 -04:00
if (mPtr.getTypeName() == typeid(ESM::NPC).name())
2018-08-29 18:38:12 +03:00
mTopicsList->addItem(gmst.find("sPersuasion")->mValue.getString());
2012-11-09 20:18:38 +01:00
2017-09-25 19:52:20 +02:00
if (services & ESM::NPC::AllItems)
2018-08-29 18:38:12 +03:00
mTopicsList->addItem(gmst.find("sBarter")->mValue.getString());
2017-09-25 19:52:20 +02:00
if (services & ESM::NPC::Spells)
2018-08-29 18:38:12 +03:00
mTopicsList->addItem(gmst.find("sSpells")->mValue.getString());
2012-09-08 18:17:03 -04:00
2017-09-30 16:53:39 +02:00
if (travel)
2018-08-29 18:38:12 +03:00
mTopicsList->addItem(gmst.find("sTravel")->mValue.getString());
2012-09-26 18:30:47 +02:00
2017-09-25 19:52:20 +02:00
if (services & ESM::NPC::Spellmaking)
2018-08-29 18:38:12 +03:00
mTopicsList->addItem(gmst.find("sSpellmakingMenuTitle")->mValue.getString());
2017-09-25 19:52:20 +02:00
if (services & ESM::NPC::Enchanting)
2018-08-29 18:38:12 +03:00
mTopicsList->addItem(gmst.find("sEnchanting")->mValue.getString());
2012-09-23 00:36:20 +02:00
2017-09-25 19:52:20 +02:00
if (services & ESM::NPC::Training)
2018-08-29 18:38:12 +03:00
mTopicsList->addItem(gmst.find("sServiceTrainingTitle")->mValue.getString());
2012-10-17 18:03:02 +02:00
2017-09-25 19:52:20 +02:00
if (services & ESM::NPC::Repair)
2018-08-29 18:38:12 +03:00
mTopicsList->addItem(gmst.find("sRepair")->mValue.getString());
2013-03-22 14:13:10 +01:00
if (isCompanion())
2018-08-29 18:38:12 +03:00
mTopicsList->addItem(gmst.find("sCompanionShare")->mValue.getString());
2013-03-31 13:13:46 +02:00
2015-03-11 20:04:25 +01:00
if (mTopicsList->getItemCount() > 0)
2013-04-17 18:56:48 -04:00
mTopicsList->addSeparator();
2013-03-31 13:13:46 +02:00
for(const auto& keyword : mKeywords)
2013-04-17 18:56:48 -04:00
{
std::string topicId = Misc::StringUtils::lowerCase(keyword);
mTopicsList->addItem(keyword);
Topic* t = new Topic(keyword);
t->eventTopicActivated += MyGUI::newDelegate(this, &DialogueWindow::onTopicActivated);
mTopicLinks[topicId] = t;
2013-05-04 14:15:47 +02:00
mKeywordSearch.seed(topicId, intptr_t(t));
2013-04-17 18:56:48 -04:00
}
mTopicsList->adjustSize();
2013-05-04 14:15:47 +02:00
updateHistory();
// The topics list has been regenerated so topic formatting needs to be updated
updateTopicFormat();
}
2013-05-04 14:15:47 +02:00
void DialogueWindow::updateHistory(bool scrollbar)
2013-04-17 18:56:48 -04:00
{
2013-05-04 14:15:47 +02:00
if (!scrollbar && mScrollBar->getVisible())
{
mHistory->setSize(mHistory->getSize()+MyGUI::IntSize(mScrollBar->getWidth(),0));
mScrollBar->setVisible(false);
}
if (scrollbar && !mScrollBar->getVisible())
{
mHistory->setSize(mHistory->getSize()-MyGUI::IntSize(mScrollBar->getWidth(),0));
mScrollBar->setVisible(true);
}
BookTypesetter::Ptr typesetter = BookTypesetter::create (mHistory->getWidth(), std::numeric_limits<int>::max());
2013-05-04 14:15:47 +02:00
for (DialogueText* text : mHistoryContents)
text->write(typesetter, &mKeywordSearch, mTopicLinks);
2013-05-04 14:15:47 +02:00
BookTypesetter::Style* body = typesetter->createStyle("", MyGUI::Colour::White, false);
2013-05-04 14:15:47 +02:00
typesetter->sectionBreak(9);
2013-05-04 14:15:47 +02:00
// choices
2017-07-24 13:25:01 +02:00
const TextColours& textColours = MWBase::Environment::get().getWindowManager()->getTextColours();
mChoices = MWBase::Environment::get().getDialogueManager()->getChoices();
for (std::pair<std::string, int>& choice : mChoices)
2013-05-04 14:15:47 +02:00
{
Choice* link = new Choice(choice.second);
link->eventChoiceActivated += MyGUI::newDelegate(this, &DialogueWindow::onChoiceActivated);
2013-05-04 14:15:47 +02:00
mLinks.push_back(link);
typesetter->lineBreak();
2017-07-24 13:25:01 +02:00
BookTypesetter::Style* questionStyle = typesetter->createHotStyle(body, textColours.answer, textColours.answerOver,
textColours.answerPressed,
2013-05-04 14:15:47 +02:00
TypesetBook::InteractiveId(link));
typesetter->write(questionStyle, to_utf8_span(choice.first.c_str()));
2013-05-04 14:15:47 +02:00
}
mGoodbye = MWBase::Environment::get().getDialogueManager()->isGoodbye();
2013-05-04 15:15:44 +02:00
if (mGoodbye)
{
Goodbye* link = new Goodbye();
link->eventActivated += MyGUI::newDelegate(this, &DialogueWindow::onGoodbyeActivated);
mLinks.push_back(link);
2018-08-29 18:38:12 +03:00
std::string goodbye = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find("sGoodbye")->mValue.getString();
2017-07-24 13:25:01 +02:00
BookTypesetter::Style* questionStyle = typesetter->createHotStyle(body, textColours.answer, textColours.answerOver,
textColours.answerPressed,
TypesetBook::InteractiveId(link));
2013-05-04 15:15:44 +02:00
typesetter->lineBreak();
typesetter->write(questionStyle, to_utf8_span(goodbye.c_str()));
}
2013-05-04 14:15:47 +02:00
TypesetBook::Ptr book = typesetter->complete();
mHistory->showPage(book, 0);
size_t viewHeight = mHistory->getParent()->getHeight();
if (!scrollbar && book->getSize().second > viewHeight)
updateHistory(true);
else if (scrollbar)
{
mHistory->setSize(MyGUI::IntSize(mHistory->getWidth(), book->getSize().second));
size_t range = book->getSize().second - viewHeight;
mScrollBar->setScrollRange(range);
mScrollBar->setScrollPosition(range-1);
mScrollBar->setTrackSize(static_cast<int>(viewHeight / static_cast<float>(book->getSize().second) * mScrollBar->getLineSize()));
2013-05-04 14:15:47 +02:00
onScrollbarMoved(mScrollBar, range-1);
}
else
{
// no scrollbar
onScrollbarMoved(mScrollBar, 0);
}
bool goodbyeEnabled = !MWBase::Environment::get().getDialogueManager()->isInChoice() || mGoodbye;
bool goodbyeWasEnabled = mGoodbyeButton->getEnabled();
mGoodbyeButton->setEnabled(goodbyeEnabled);
if (goodbyeEnabled && !goodbyeWasEnabled)
MWBase::Environment::get().getWindowManager()->setKeyFocusWidget(mGoodbyeButton);
bool topicsEnabled = !MWBase::Environment::get().getDialogueManager()->isInChoice() && !mGoodbye;
mTopicsList->setEnabled(topicsEnabled);
2013-04-17 18:56:48 -04:00
}
2013-05-04 14:15:47 +02:00
void DialogueWindow::notifyLinkClicked (TypesetBook::InteractiveId link)
2012-04-17 23:47:50 +02:00
{
2013-05-04 14:15:47 +02:00
reinterpret_cast<Link*>(link)->activated();
2012-04-17 23:47:50 +02:00
}
void DialogueWindow::onTopicActivated(const std::string &topicId)
2013-05-04 14:15:47 +02:00
{
2018-07-06 19:38:36 +03:00
if (mGoodbye)
return;
MWBase::Environment::get().getDialogueManager()->keywordSelected(topicId, mCallback.get());
updateTopics();
2013-05-04 14:15:47 +02:00
}
void DialogueWindow::onChoiceActivated(int id)
2013-04-17 18:56:48 -04:00
{
if (mGoodbye)
{
onGoodbyeActivated();
return;
}
MWBase::Environment::get().getDialogueManager()->questionAnswered(id, mCallback.get());
updateTopics();
2013-05-04 14:15:47 +02:00
}
void DialogueWindow::onGoodbyeActivated()
2013-05-04 14:15:47 +02:00
{
MWBase::Environment::get().getDialogueManager()->goodbyeSelected();
MWBase::Environment::get().getWindowManager()->removeGuiMode(MWGui::GM_Dialogue);
resetReference();
}
void DialogueWindow::onScrollbarMoved(MyGUI::ScrollBar *sender, size_t pos)
{
mHistory->setPosition(0, static_cast<int>(pos) * -1);
2013-05-04 14:15:47 +02:00
}
void DialogueWindow::addResponse(const std::string &title, const std::string &text, bool needMargin)
2013-05-04 14:15:47 +02:00
{
mHistoryContents.push_back(new Response(text, title, needMargin));
2013-05-04 14:15:47 +02:00
updateHistory();
2013-04-17 18:56:48 -04:00
}
void DialogueWindow::addMessageBox(const std::string& text)
2013-04-17 18:56:48 -04:00
{
mHistoryContents.push_back(new Message(text));
2013-05-04 14:15:47 +02:00
updateHistory();
2013-04-17 18:56:48 -04:00
}
2017-09-25 21:29:06 +02:00
void DialogueWindow::updateDisposition()
2012-11-08 22:31:08 +01:00
{
bool dispositionVisible = false;
2017-09-25 21:29:06 +02:00
if (!mPtr.isEmpty() && mPtr.getClass().isNpc())
2013-04-17 18:56:48 -04:00
{
dispositionVisible = true;
2013-04-17 18:56:48 -04:00
mDispositionBar->setProgressRange(100);
mDispositionBar->setProgressPosition(MWBase::Environment::get().getMechanicsManager()->getDerivedDisposition(mPtr));
mDispositionText->setCaption(MyGUI::utility::toString(MWBase::Environment::get().getMechanicsManager()->getDerivedDisposition(mPtr))+std::string("/100"));
2013-04-17 18:56:48 -04:00
}
bool dispositionWasVisible = mDispositionBar->getVisible();
if (dispositionVisible && !dispositionWasVisible)
{
mDispositionBar->setVisible(true);
int offset = mDispositionBar->getHeight()+5;
mTopicsList->setCoord(mTopicsList->getCoord() + MyGUI::IntCoord(0,offset,0,-offset));
mTopicsList->adjustSize();
}
else if (!dispositionVisible && dispositionWasVisible)
{
mDispositionBar->setVisible(false);
int offset = mDispositionBar->getHeight()+5;
mTopicsList->setCoord(mTopicsList->getCoord() - MyGUI::IntCoord(0,offset,0,-offset));
mTopicsList->adjustSize();
}
2012-11-08 22:31:08 +01:00
}
2013-04-17 18:56:48 -04:00
void DialogueWindow::onReferenceUnavailable()
{
MWBase::Environment::get().getWindowManager()->removeGuiMode(GM_Dialogue);
}
2012-11-05 23:16:37 +01:00
void DialogueWindow::onFrame(float dt)
2012-11-05 23:16:37 +01:00
{
checkReferenceAvailable();
if (mPtr.isEmpty())
return;
2017-09-25 21:29:06 +02:00
updateDisposition();
deleteLater();
if (mChoices != MWBase::Environment::get().getDialogueManager()->getChoices()
|| mGoodbye != MWBase::Environment::get().getDialogueManager()->isGoodbye())
updateHistory();
}
void DialogueWindow::updateTopicFormat()
{
if (!Settings::Manager::getBool("color topic enable", "GUI"))
return;
std::string specialColour = Settings::Manager::getString("color topic specific", "GUI");
std::string oldColour = Settings::Manager::getString("color topic exhausted", "GUI");
for (const std::string& keyword : mKeywords)
{
int flag = MWBase::Environment::get().getDialogueManager()->getTopicFlag(keyword);
MyGUI::Button* button = mTopicsList->getItemWidget(keyword);
if (!specialColour.empty() && flag & MWBase::DialogueManager::TopicType::Specific)
button->getSubWidgetText()->setTextColour(MyGUI::Colour::parse(specialColour));
else if (!oldColour.empty() && flag & MWBase::DialogueManager::TopicType::Exhausted)
button->getSubWidgetText()->setTextColour(MyGUI::Colour::parse(oldColour));
}
}
void DialogueWindow::updateTopics()
{
// Topic formatting needs to be updated regardless of whether the topic list has changed
if (!setKeywords(MWBase::Environment::get().getDialogueManager()->getAvailableTopics()))
updateTopicFormat();
}
bool DialogueWindow::isCompanion()
{
2017-10-21 16:56:21 +04:00
return isCompanion(mPtr);
}
bool DialogueWindow::isCompanion(const MWWorld::Ptr& actor)
{
if (actor.isEmpty())
return false;
2017-10-21 16:56:21 +04:00
return !actor.getClass().getScript(actor).empty()
&& actor.getRefData().getLocals().getIntVar(actor.getClass().getScript(actor), "companion");
}
2012-11-05 23:16:37 +01:00
}