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

296 lines
9.0 KiB
C++
Raw Normal View History

2010-11-03 21:21:08 +01:00
#include "dialogue.hpp"
#include <iostream>
#include <iterator>
#include <boost/algorithm/string.hpp>
#include <boost/lexical_cast.hpp>
2012-05-17 17:15:44 +02:00
#include <components/esm_store/store.hpp>
#include "../mwbase/environment.hpp"
#include "../mwbase/dialoguemanager.hpp"
#include "../mwbase/world.hpp"
#include "../mwbase/windowmanager.hpp"
2012-05-17 17:15:44 +02:00
#include "dialogue_history.hpp"
#include "widgets.hpp"
#include "list.hpp"
#include "tradewindow.hpp"
#include "spellbuyingwindow.hpp"
2012-05-17 21:15:48 +02:00
#include "inventorywindow.hpp"
2012-05-17 17:15:44 +02:00
2010-11-03 21:21:08 +01:00
using namespace MWGui;
using namespace Widgets;
/**
*Copied from the internet.
*/
2012-09-23 00:36:20 +02:00
namespace {
std::string lower_string(const std::string& str)
{
2012-02-10 16:21:04 +01:00
std::string lowerCase;
std::transform (str.begin(), str.end(), std::back_inserter (lowerCase),
(int(*)(int)) std::tolower);
return lowerCase;
}
std::string::size_type find_str_ci(const std::string& str, const std::string& substr,size_t pos)
{
return lower_string(str).find(lower_string(substr),pos);
}
2012-09-23 00:36:20 +02:00
}
DialogueWindow::DialogueWindow(MWBase::WindowManager& parWindowManager)
: WindowBase("openmw_dialogue_window.layout", parWindowManager)
, mEnabled(true)
2012-09-23 00:36:20 +02:00
, mServices(0)
2010-11-03 21:21:08 +01:00
{
// Centre dialog
center();
2010-11-03 21:21:08 +01:00
//History view
getWidget(mHistory, "History");
mHistory->setOverflowToTheLeft(true);
mHistory->setMaxTextLength(1000000);
MyGUI::Widget* eventbox;
//An EditBox cannot receive mouse click events, so we use an
//invisible widget on top of the editbox to receive them
getWidget(eventbox, "EventBox");
eventbox->eventMouseButtonClick += MyGUI::newDelegate(this, &DialogueWindow::onHistoryClicked);
eventbox->eventMouseWheel += MyGUI::newDelegate(this, &DialogueWindow::onMouseWheel);
2012-03-19 19:21:08 +01:00
//Topics list
getWidget(mTopicsList, "TopicsList");
mTopicsList->eventItemSelected += MyGUI::newDelegate(this, &DialogueWindow::onSelectTopic);
2010-11-03 21:21:08 +01:00
MyGUI::ButtonPtr byeButton;
getWidget(byeButton, "ByeButton");
byeButton->eventMouseButtonClick += MyGUI::newDelegate(this, &DialogueWindow::onByeClicked);
2010-11-03 21:21:08 +01:00
getWidget(mDispositionBar, "Disposition");
getWidget(mDispositionText,"DispositionText");
2012-05-10 11:19:22 +02:00
static_cast<MyGUI::Window*>(mMainWidget)->eventWindowChangeCoord += MyGUI::newDelegate(this, &DialogueWindow::onWindowResize);
2010-11-03 21:21:08 +01:00
}
void DialogueWindow::onHistoryClicked(MyGUI::Widget* _sender)
{
MyGUI::ISubWidgetText* t = mHistory->getClient()->getSubWidgetText();
2010-11-03 21:21:08 +01:00
if(t == nullptr)
return;
const MyGUI::IntPoint& lastPressed = MyGUI::InputManager::getInstance().getLastPressedPosition(MyGUI::MouseButton::Left);
2010-11-03 21:21:08 +01:00
size_t cursorPosition = t->getCursorPosition(lastPressed);
MyGUI::UString color = mHistory->getColorAtPos(cursorPosition);
if (!mEnabled && color == "#572D21")
MWBase::Environment::get().getDialogueManager()->goodbyeSelected();
if(color != "#B29154")
2010-11-03 21:21:08 +01:00
{
MyGUI::UString key = mHistory->getColorTextAt(cursorPosition);
if(color == "#686EBA") MWBase::Environment::get().getDialogueManager()->keywordSelected(lower_string(key));
if(color == "#572D21") MWBase::Environment::get().getDialogueManager()->questionAnswered(lower_string(key));
2010-11-03 21:21:08 +01:00
}
}
2012-05-10 11:19:22 +02:00
void DialogueWindow::onWindowResize(MyGUI::Window* _sender)
{
mTopicsList->adjustSize();
2012-05-10 11:19:22 +02:00
}
void DialogueWindow::onMouseWheel(MyGUI::Widget* _sender, int _rel)
{
if (mHistory->getVScrollPosition() - _rel*0.3 < 0)
mHistory->setVScrollPosition(0);
else
mHistory->setVScrollPosition(mHistory->getVScrollPosition() - _rel*0.3);
}
2010-11-03 21:21:08 +01:00
void DialogueWindow::onByeClicked(MyGUI::Widget* _sender)
{
MWBase::Environment::get().getDialogueManager()->goodbyeSelected();
2010-11-03 21:21:08 +01:00
}
void DialogueWindow::onSelectTopic(std::string topic)
2010-11-03 21:21:08 +01:00
{
if (!mEnabled) return;
2012-09-22 21:35:57 +02:00
if (topic == MWBase::Environment::get().getWorld()->getStore().gameSettings.find("sBarter")->getString())
2012-05-17 17:15:44 +02:00
{
/// \todo check if the player is allowed to trade with this actor (e.g. faction rank high enough)?
2012-05-23 12:23:35 +02:00
mWindowManager.pushGuiMode(GM_Barter);
mWindowManager.getTradeWindow()->startTrade(mPtr);
2012-05-17 17:15:44 +02:00
}
2012-09-22 21:35:57 +02:00
else if (topic == MWBase::Environment::get().getWorld()->getStore().gameSettings.find("sSpells")->getString())
2012-09-08 18:17:03 -04:00
{
mWindowManager.pushGuiMode(GM_SpellBuying);
mWindowManager.getSpellBuyingWindow()->startSpellBuying(mPtr);
2012-09-08 18:17:03 -04:00
}
else if (topic == MWBase::Environment::get().getWorld()->getStore().gameSettings.find("sSpellMakingMenuTitle")->getString())
2012-09-23 00:36:20 +02:00
{
mWindowManager.pushGuiMode(GM_SpellCreation);
mWindowManager.startSpellMaking (mPtr);
}
else if (topic == MWBase::Environment::get().getWorld()->getStore().gameSettings.find("sEnchanting")->getString())
{
mWindowManager.pushGuiMode(GM_Enchanting);
mWindowManager.startEnchanting (mPtr);
}
2012-05-17 17:15:44 +02:00
else
MWBase::Environment::get().getDialogueManager()->keywordSelected(lower_string(topic));
2010-11-03 21:21:08 +01:00
}
2012-05-17 17:15:44 +02:00
void DialogueWindow::startDialogue(MWWorld::Ptr actor, std::string npcName)
{
mEnabled = true;
mPtr = actor;
mTopicsList->setEnabled(true);
setTitle(npcName);
2012-05-23 12:23:35 +02:00
mTopicsList->clear();
mHistory->eraseText(0,mHistory->getTextLength());
2012-05-23 12:23:35 +02:00
updateOptions();
}
2012-03-19 18:30:52 +01:00
void DialogueWindow::setKeywords(std::list<std::string> keyWords)
{
mTopicsList->clear();
2012-09-23 00:36:20 +02:00
bool anyService = mServices > 0;
2012-09-23 00:36:20 +02:00
if (mServices & Service_Trade)
2012-09-22 21:35:57 +02:00
mTopicsList->addItem(MWBase::Environment::get().getWorld()->getStore().gameSettings.find("sBarter")->getString());
2012-09-23 00:36:20 +02:00
if (mServices & Service_BuySpells)
2012-09-22 21:35:57 +02:00
mTopicsList->addItem(MWBase::Environment::get().getWorld()->getStore().gameSettings.find("sSpells")->getString());
2012-09-08 18:17:03 -04:00
2012-09-23 00:36:20 +02:00
if (mServices & Service_CreateSpells)
mTopicsList->addItem(MWBase::Environment::get().getWorld()->getStore().gameSettings.find("sSpellmakingMenuTitle")->getString());
if (mServices & Service_Enchant)
mTopicsList->addItem(MWBase::Environment::get().getWorld()->getStore().gameSettings.find("sEnchanting")->getString());
2012-09-23 00:36:20 +02:00
if (anyService)
mTopicsList->addSeparator();
2012-06-06 20:29:30 +02:00
for(std::list<std::string>::iterator it = keyWords.begin(); it != keyWords.end(); ++it)
{
mTopicsList->addItem(*it);
}
mTopicsList->adjustSize();
}
void DialogueWindow::removeKeyword(std::string keyWord)
{
if(mTopicsList->hasItem(keyWord))
{
mTopicsList->removeItem(keyWord);
}
mTopicsList->adjustSize();
}
void addColorInString(std::string& str, const std::string& keyword,std::string color1, std::string color2)
{
size_t pos = 0;
while((pos = find_str_ci(str,keyword, pos)) != std::string::npos)
{
2012-02-15 13:23:59 +01:00
if(pos==0)
{
str.insert(pos,color1);
pos += color1.length();
pos += keyword.length();
str.insert(pos,color2);
pos+= color2.length();
}
2012-03-19 19:21:08 +01:00
else
2012-02-15 13:23:59 +01:00
{
2012-03-17 16:56:22 +01:00
if(str.substr(pos -1,1) == " ")
{
str.insert(pos,color1);
pos += color1.length();
pos += keyword.length();
str.insert(pos,color2);
pos+= color2.length();
}
else
{
pos += keyword.length();
}
2012-02-15 13:23:59 +01:00
}
}
}
std::string DialogueWindow::parseText(std::string text)
{
bool separatorReached = false; // only parse topics that are below the separator (this prevents actions like "Barter" that are not topics from getting blue-colored)
for(unsigned int i = 0;i<mTopicsList->getItemCount();i++)
{
std::string keyWord = mTopicsList->getItemNameAt(i);
if (separatorReached && keyWord != "")
addColorInString(text,keyWord,"#686EBA","#B29154");
else
separatorReached = true;
}
return text;
}
void DialogueWindow::addText(std::string text)
{
mHistory->addDialogText("#B29154"+parseText(text)+"#B29154");
}
void DialogueWindow::addTitle(std::string text)
{
2012-04-17 23:47:50 +02:00
// This is called from the dialogue manager, so text is
// case-smashed - thus we have to retrieve the correct case
// of the text through the topic list.
for (size_t i=0; i<mTopicsList->getItemCount(); ++i)
2012-04-17 23:47:50 +02:00
{
std::string item = mTopicsList->getItemNameAt(i);
2012-04-17 23:47:50 +02:00
if (lower_string(item) == text)
text = item;
}
mHistory->addDialogHeading(text);
}
void DialogueWindow::askQuestion(std::string question)
{
mHistory->addDialogText("#572D21"+question+"#B29154"+" ");
}
2010-11-03 21:21:08 +01:00
void DialogueWindow::updateOptions()
{
//Clear the list of topics
mTopicsList->clear();
mHistory->eraseText(0, mHistory->getTextLength());
mDispositionBar->setProgressRange(100);
mDispositionBar->setProgressPosition(40);
mDispositionText->eraseText(0, mDispositionText->getTextLength());
mDispositionText->addText("#B29154"+std::string("40/100")+"#B29154");
2010-11-03 21:21:08 +01:00
}
void DialogueWindow::goodbye()
{
2012-09-22 21:35:57 +02:00
mHistory->addDialogText("\n#572D21" + MWBase::Environment::get().getWorld()->getStore().gameSettings.find("sGoodbye")->getString());
mTopicsList->setEnabled(false);
mEnabled = false;
}
void DialogueWindow::onReferenceUnavailable()
{
mWindowManager.removeGuiMode(GM_Dialogue);
}