// Ryzom - MMORPG Framework
// Copyright (C) 2010 Winch Gate Property Limited
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as
// published by the Free Software Foundation, either version 3 of the
// License, or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see .
// client
#include "stdpch.h"
#include "game_share/shard_names.h"
#include "../r2/editor.h"
#include "chat_window.h"
#include "chat_text_manager.h"
#include "../user_entity.h"
#include "people_interraction.h"
#include "../connection.h"
//
#include "group_container.h"
#include "group_editbox.h"
#include "group_tab.h"
#include "interface_manager.h"
#include "action_handler.h"
#include "../client_chat_manager.h"
//
#include "../session_browser_impl.h"
#include "../r2/editor.h"
#include "../r2/dmc/client_edition_module.h"
using namespace NLMISC;
using namespace std;
////////////
// STATIC //
////////////
CChatWindow *CChatWindow::_ChatWindowLaunchingCommand = NULL;
////////////
// EXTERN //
////////////
extern NLMISC::CLog g_log;
extern CClientChatManager ChatMngr;
/////////////////////
// CChatWindowDesc //
/////////////////////
CChatWindowDesc::CChatWindowDesc() : InsertPosition(-1),
ParentBlink(false),
Savable(false),
Localize(false),
Listener(NULL)
{
}
/////////////////
// CChatWindow //
/////////////////
//=================================================================================
CChatWindow::CChatWindow() : _Listener(NULL), _Chat(NULL), _EB(NULL), _ParentBlink(false)
{
}
//=================================================================================
bool CChatWindow::create(const CChatWindowDesc &desc, const std::string &chatId)
{
deleteContainer();
CInterfaceManager *im = CInterfaceManager::getInstance();
// get the father container
CGroupContainer *fatherContainer = NULL;
if (!desc.FatherContainer.empty())
{
if (desc.FatherContainer != "ui:interface" )
{
fatherContainer = dynamic_cast(im->getElementFromId(desc.FatherContainer));
if (!fatherContainer)
{
nlwarning(" Can't get father group, or bad type");
return false;
}
}
}
// get the good template
std::string chatTemplate;
if (desc.ChatTemplate.empty())
{
// no chat template provided : use default
chatTemplate = "chat_id";
}
else
{
chatTemplate = desc.ChatTemplate;
}
// build params
CChatWindowDesc::TTemplateParams params;
params.push_back(make_pair(string("id"), chatId));
params.insert(params.end(), desc.ChatTemplateParams.begin(), desc.ChatTemplateParams.end());
// create a chat container from the template
CInterfaceGroup *chatGroup = im->createGroupInstance(chatTemplate, "ui:interface", params);
if (chatGroup)
{
_Chat = dynamic_cast(chatGroup);
if (!_Chat)
{
nlwarning(" Bad type for chat group");
delete chatGroup;
return false;
}
_Chat->setLocalize (desc.Localize);
if (desc.Localize)
_Chat->setTitle(desc.Title.toString());
else
_Chat->setUCTitle(desc.Title);
_Chat->setSavable(desc.Savable);
// groups like system info don't have edit box.
_EB = dynamic_cast(_Chat->getGroup("eb"));
if (_EB)
{
_EB->setAHOnEnter("chat_box_entry");
}
CInterfaceGroup *pRoot = dynamic_cast(im->getElementFromId("ui:interface"));
if (fatherContainer)
{
fatherContainer->attachContainer(_Chat, desc.InsertPosition);
}
// If root container
if (desc.FatherContainer == "ui:interface")
{
im->addWindowToMasterGroup("ui:interface", _Chat);
_Chat->setParent(pRoot);
_Chat->setMovable(true);
_Chat->setActive(false);
_Chat->setOpen(true);
}
_ParentBlink = desc.ParentBlink;
_Listener = desc.Listener;
pRoot->addGroup (_Chat);
return true;
}
else
{
return false;
}
}
//=================================================================================
bool CChatWindow::isVisible() const
{
if (!_Chat) return false;
if (_Chat->isOpen())
{
CInterfaceGroup *ig = _Chat;
do
{
if (ig->isGroupContainer())
{
if (!static_cast(ig)->isOpen()) break;
}
if (!ig->getActive()) break;
ig = ig->getParent();
}
while(ig);
return ig == NULL; // all parent windows must be open & visible
}
else
{
return false;
}
}
//=================================================================================
void CChatWindow::displayMessage(const ucstring &msg, NLMISC::CRGBA col, CChatGroup::TGroupType /* gt */, uint32 /* dynamicChatDbIndex */, uint numBlinks /* = 0*/, bool *windowVisible /*= NULL*/)
{
if (!_Chat)
{
nlwarning(" There's no global chat");
return;
}
CGroupList *gl;
CChatTextManager &ctm = getChatTextMngr();
gl = dynamic_cast(_Chat->getGroup("cb:text_list"));
if (gl) gl->addChild(ctm.createMsgText(msg, col));
// if the group is closed, make it blink
if (!_Chat->isOpen())
{
if (numBlinks) _Chat->enableBlink(numBlinks);
}
if (_ParentBlink)
{
CGroupContainer *father = dynamic_cast(_Chat->getParent());
if (father && !father->isOpen())
{
father->enableBlink(numBlinks);
}
}
if (windowVisible != NULL)
{
*windowVisible = isVisible();
}
/*for(std::vector::iterator it = _Observers.begin(); it != _Observers.end(); ++it)
{
(*it)->displayMessage(this, msg, col, numBlinks);
}*/
}
//=================================================================================
void CChatWindow::setMenu(const std::string &menuName)
{
if (!_Chat) return;
if (_Chat->getHeaderOpened())
{
_Chat->getHeaderOpened()->setRightClickHandler("active_menu");
_Chat->getHeaderOpened()->setRightClickHandlerParams("menu=" + menuName);
}
if (_Chat->getHeaderClosed())
{
_Chat->getHeaderClosed()->setRightClickHandler("active_menu");
_Chat->getHeaderClosed()->setRightClickHandlerParams("menu=" + menuName);
}
}
//=================================================================================
void CChatWindow::setPrompt(const ucstring &prompt)
{
if (!_Chat) return;
CGroupEditBox *eb = dynamic_cast(_Chat->getGroup("eb"));
if (!eb) return;
eb->setPrompt(prompt);
}
void CChatWindow::setPromptColor(NLMISC::CRGBA col)
{
if (!_Chat)
return;
CGroupEditBox *eb = dynamic_cast(_Chat->getGroup("eb"));
if (!eb)
return;
eb->setColor(col);
}
//=================================================================================
void CChatWindow::deleteContainer()
{
if (!_Chat) return;
CGroupContainer *proprietaryContainer = _Chat->getProprietaryContainer();
if (proprietaryContainer)
{
if (_Chat->isPopuped())
{
_Chat->popin(-1, false); // popin & detach
}
else
{
proprietaryContainer->detachContainer(_Chat); // just detach
}
CInterfaceManager *im = CInterfaceManager::getInstance();
CInterfaceGroup *pRoot = dynamic_cast(im->getElementFromId("ui:interface"));
pRoot->delGroup (_Chat);
}
else
{
CInterfaceManager::getInstance()->unMakeWindow(_Chat);
if (_Chat->getParent())
{
_Chat->getParent()->delGroup(_Chat);
}
}
// Removes from parent group
_Chat = NULL;
}
//=================================================================================
bool CChatWindow::rename(const ucstring &newName, bool newNameLocalize)
{
return getChatWndMgr().rename(getTitle(), newName, newNameLocalize);
}
//=================================================================================
void CChatWindow::setKeyboardFocus()
{
if (!_EB || !_Chat) return;
CInterfaceManager::getInstance()->setCaptureKeyboard(_EB);
if (!_Chat->isOpenable() || _Chat->isOpenWhenPopup())
{
if (_Chat->isPopable() && !_Chat->isPopuped())
{
_Chat->popup();
if (_Chat->getPopupW() != -1) // restore previous popup position if there's one
{
_Chat->setX(_Chat->getPopupX());
_Chat->setY(_Chat->getPopupY());
_Chat->setW(_Chat->getPopupW());
// must resize the children to get correct height
//_Chat->setChildrenH(_Chat->getPopupChildrenH());
}
}
}
}
//=================================================================================
void CChatWindow::enableBlink(uint numBlinks)
{
if (!_Chat) return;
_Chat->enableBlink(numBlinks);
}
//=================================================================================
void CChatWindow::setCommand(const std::string &command, bool execute)
{
if (!_EB) return;
_EB->setCommand(ucstring(command), execute);
}
void CChatWindow::setCommand(const ucstring &command,bool execute)
{
if (!_EB) return;
_EB->setCommand(command, execute);
}
//=================================================================================
void CChatWindow::setEntry(const ucstring &entry)
{
if (!_EB) return;
_EB->setInputString(entry);
}
//=================================================================================
ucstring CChatWindow::getTitle() const
{
if (!_Chat)
{
return ucstring("");
}
else
{
return _Chat->getUCTitle();
}
}
//=================================================================================
void CChatWindow::addObserver(IObserver *obs)
{
if (!obs)
{
nlwarning("NULL observer is invalid");
return;
}
if (isObserver(obs))
{
nlwarning("Observer added twice");
return;
}
_Observers.push_back(obs);
}
//=================================================================================
void CChatWindow::removeObserver(IObserver *obs)
{
std::vector::iterator it = std::find(_Observers.begin(), _Observers.end(), obs);
if (it == _Observers.end())
{
nlwarning("Observer doesn't belong to this chatbox");
return;
}
_Observers.erase(it);
}
//=================================================================================
bool CChatWindow::isObserver(const IObserver *obs) const
{
std::vector::const_iterator it = std::find(_Observers.begin(), _Observers.end(), obs);
return it != _Observers.end();
}
//=================================================================================
CChatWindow::~CChatWindow()
{
for(std::vector::iterator it = _Observers.begin(); it != _Observers.end(); ++it)
{
(*it)->chatWindowRemoved(this);
}
if (this == _ChatWindowLaunchingCommand)
{
_ChatWindowLaunchingCommand = NULL;
}
}
//=================================================================================
void CChatWindow::setAHOnActive(const std::string &n)
{
if (_Chat) _Chat->setOnActiveHandler(n);
}
//=================================================================================
void CChatWindow::setAHOnActiveParams(const std::string &n)
{
if (_Chat) _Chat->setOnActiveParams(n);
}
//=================================================================================
void CChatWindow::setAHOnDeactive(const std::string &n)
{
if (_Chat) _Chat->setOnDeactiveHandler(n);
}
//=================================================================================
void CChatWindow::setAHOnDeactiveParams(const std::string &n)
{
if (_Chat) _Chat->setOnDeactiveParams(n);
}
//=================================================================================
void CChatWindow::setAHOnCloseButton(const std::string &n)
{
if (_Chat) _Chat->setOnCloseButtonHandler(n);
}
//=================================================================================
void CChatWindow::setAHOnCloseButtonParams(const std::string &n)
{
if (_Chat) _Chat->setOnCloseButtonParams(n);
}
//=================================================================================
void CChatWindow::setHeaderColor(const std::string &n)
{
if (_Chat) _Chat->setHeaderColor(n);
}
//=================================================================================
void CChatWindow::displayLocalPlayerTell(const ucstring &msg, uint numBlinks /*= 0*/)
{
ucstring finalMsg;
CInterfaceProperty prop;
prop.readRGBA("UI:SAVE:CHAT:COLORS:SPEAKER"," ");
encodeColorTag(prop.getRGBA(), finalMsg, false);
ucstring cur_time;
CCDBNodeLeaf *pNL = CInterfaceManager::getInstance()->getDbProp("UI:SAVE:CHAT:SHOW_TIMES_IN_CHAT_CB", false);
if (pNL && pNL->getValueBool())
{
cur_time = CInterfaceManager::getTimestampHuman();
}
ucstring csr;
if (CHARACTER_TITLE::isCsrTitle(UserEntity->getTitleRaw())) csr += ucstring("(CSR) ");
finalMsg += cur_time + csr + CI18N::get("youTell") + ": ";
prop.readRGBA("UI:SAVE:CHAT:COLORS:TELL"," ");
encodeColorTag(prop.getRGBA(), finalMsg, true);
finalMsg += msg;
displayMessage(finalMsg, prop.getRGBA(), CChatGroup::tell, 0, numBlinks);
}
void CChatWindow::encodeColorTag(const NLMISC::CRGBA &color, ucstring &text, bool append)
{
// WARNING : The lookup table MUST contains 17 element (with the last doubled)
// because we add 7 to the 8 bit color before shifting to right in order to match color
// more accurately.
// Have 17 entry remove the need for a %16 for each color component.
// By the way, this comment is more longer to type than to add the %16...
//
static ucchar ConvTable[] = {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F', 'F'};
ucstring str;
if (append)
{
str.reserve(7 + str.size());
str = text;
}
else
str.reserve(7);
str += "@{";
str += ConvTable[(uint(color.R)+7)>>4];
str += ConvTable[(uint(color.G)+7)>>4];
str += ConvTable[(uint(color.B)+7)>>4];
str += ConvTable[(uint(color.A)+7)>>4];
str += '}';
text.swap(str);
}
//=================================================================================
void CChatWindow::clearMessages(CChatGroup::TGroupType /* gt */, uint32 /* dynamicChatDbIndex */)
{
// if not correctly init, abort
if(!_Chat)
return;
// get the group list
CGroupList *gl = dynamic_cast(_Chat->getGroup("cb:text_list"));
if (gl) gl->deleteAllChildren();
}
//////////////////////
// CChatGroupWindow //
//////////////////////
void CChatGroupWindow::displayMessage(const ucstring &msg, NLMISC::CRGBA col, CChatGroup::TGroupType gt, uint32 dynamicChatDbIndex, uint numBlinks, bool *windowVisible)
{
if (!_Chat)
{
nlwarning(" There's no global chat");
return;
}
CChatTextManager &ctm = getChatTextMngr();
if (_Chat->getHeaderOpened()==NULL)
return;
// *** Display the message in the correct tab window
// get the gl and tab according to filter
CGroupList *gl;
CCtrlTabButton *tab;
getAssociatedSubWindow(gt, dynamicChatDbIndex, gl, tab);
// on a new message, change the Tab color
CInterfaceManager *pIM= CInterfaceManager::getInstance();
CRGBA newMsgColor= stringToRGBA(pIM->getDefine("chat_group_tab_color_newmsg").c_str());
if (gl != NULL)
{
gl->addChild(ctm.createMsgText(msg, col));
if (!gl->getParent()->getActive())
if (tab != NULL)
tab->setTextColorNormal(newMsgColor);
}
// *** Display the message in the UserChat (special case)
{
tab = dynamic_cast(_Chat->getCtrl("header_opened:channel_select:tab5"));
gl = NULL;
CGroupList *gl2 = dynamic_cast(_Chat->getGroup("content:cb:user:text_list"));
CChatWindow *cw = PeopleInterraction.TheUserChat.Window;
CChatStdInput &ci = PeopleInterraction.ChatInput;
switch(gt)
{
default:
case CChatGroup::arround:
case CChatGroup::say: if (ci.AroundMe.isListeningWindow(cw)) gl = gl2; break;
case CChatGroup::region: if (ci.Region.isListeningWindow(cw)) gl = gl2; break;
case CChatGroup::team: if (ci.Team.isListeningWindow(cw)) gl = gl2; break;
case CChatGroup::guild: if (ci.Guild.isListeningWindow(cw)) gl = gl2; break;
case CChatGroup::system: if (ci.SystemInfo.isListeningWindow(cw)) gl = gl2; break;
case CChatGroup::universe: if (ci.Universe.isListeningWindow(cw)) gl = gl2; break;
// NB: the yubo chat and dyn_chat cannot be in a user chat
case CChatGroup::yubo_chat: gl = NULL; break;
case CChatGroup::dyn_chat: gl = NULL; break;
}
if (gl != NULL)
{
gl->addChild(ctm.createMsgText(msg, col));
if (!gl->getParent()->getActive())
if (tab != NULL)
tab->setTextColorNormal(newMsgColor);
}
}
// *** Blink and visibility event
// if the group is closed, make it blink
if (!_Chat->isOpen())
{
if (numBlinks) _Chat->enableBlink(numBlinks);
}
if (_ParentBlink)
{
CGroupContainer *father = dynamic_cast(_Chat->getParent());
if (father && !father->isOpen())
{
father->enableBlink(numBlinks);
}
}
if (windowVisible != NULL)
{
*windowVisible = isVisible();
}
}
//=================================================================================
void CChatGroupWindow::displayTellMessage(const ucstring &msg, NLMISC::CRGBA col, const ucstring &sender)
{
// If we are here with a tell message this is because the teller doesn't belong to any people list
CGroupContainer *gcChat = createFreeTeller(sender);
if (gcChat == NULL)
{
nlwarning(" cannot open chat.");
return;
}
gcChat->requireAttention();
CInterfaceManager::getInstance()->setTopWindow(gcChat);
// add the text to this window
CGroupList *gl = dynamic_cast(gcChat->getGroup("text_list"));
if (gl == NULL)
{
nlwarning(" can't get text_list.");
return;
}
gl->addChild(getChatTextMngr().createMsgText(msg, col));
}
//=================================================================================
sint32 CChatGroupWindow::getTabIndex()
{
CGroupTab *pTab = dynamic_cast(_Chat->getGroup("header_opened:channel_select"));
if (pTab != NULL)
return pTab->getSelection();
else
return -1;
}
//=================================================================================
void CChatGroupWindow::setTabIndex(sint32 n)
{
CGroupTab *pTab = dynamic_cast(_Chat->getGroup("header_opened:channel_select"));
if (pTab != NULL)
{
pTab->select(n);
// if the current button is hidden, select default not hid
pTab->selectDefaultIfCurrentHid();
}
}
//=================================================================================
const string CChatGroupWindow::getValidUiStringId(const string &stringId)
{
string validStringId;
for (uint32 i=0; i < stringId.length(); i++)
{
if ((stringId[i] < 'a') || (stringId[i] > 'z'))
validStringId += '_';
else
validStringId += stringId[i];
}
return validStringId;
}
//=================================================================================
CGroupContainer *CChatGroupWindow::createFreeTeller(const ucstring &winNameIn, const string &winColor)
{
// must parse the entity name, and eventually make it Full with shard name (eg: 'ani.yoyo' becomes 'yoyo(Aniro)')
string winNameFull= CShardNames::getInstance().makeFullNameFromRelative(PlayerSelectedMainland, winNameIn.toString());
// remove shard name if necessary
ucstring winName= CEntityCL::removeShardFromName(winNameFull);
// get the color
string sWinColor = winColor;
if (sWinColor.empty())
sWinColor = "UI:SAVE:WIN:COLORS:COM";
// Look if the free teller do not already exists
uint32 i;
string sWinName = winName.toString();
sWinName = toLower(sWinName);
for (i = 0; i < _FreeTellers.size(); ++i)
{
CGroupContainer *pGC = _FreeTellers[i];
if (toLower(pGC->getUCTitle().toString()) == sWinName)
break;
}
// Create container if not present
if (i == _FreeTellers.size())
{
// Corresponding Chat not created -> create and open
vector > properties;
properties.push_back(make_pair(string("posparent"), string("parent")));
properties.push_back(make_pair(string("id"), "free_chat_" + getValidUiStringId(sWinName)));
properties.push_back(make_pair(string("title"), std::string("")));
properties.push_back(make_pair(string("header_color"), sWinColor));
std::string templateName = "contact_chat_friend";
CInterfaceManager *pIM = CInterfaceManager::getInstance();
CInterfaceGroup *pIG = pIM->createGroupInstance(templateName, "ui:interface", properties);
if (!pIG) return NULL;
CGroupContainer *pGC = dynamic_cast(pIG);
if (!pGC)
{
delete pIG;
nlwarning(" group is not a container.(%s)", winName.toString().c_str());
return NULL;
}
// set title from the name
pGC->setUCTitle(winName);
//
pGC->setSavable(true);
pGC->setEscapable(true);
CInterfaceGroup *pRoot = dynamic_cast(pIM->getElementFromId("ui:interface"));
pRoot->addGroup (pGC);
pGC->setParent(pRoot); // must be done before makeWindow
pIM->makeWindow(pGC);
pGC->open();
pGC->updateCoords();
pGC->center();
_FreeTellers.push_back(pGC);
CInterfaceElement *result = pRoot->findFromShortId(string("invite"));
if(result)
{
CCtrlBaseButton * inviteButton = dynamic_cast(result);
if(inviteButton)
{
bool isDM = false;
R2::CEditor::TMode mode = R2::getEditor().getMode();
if (mode != R2::CEditor::NotInitialized )
{
isDM = R2::getEditor().getDMC().getEditionModule().isSessionOwner()
&& uint32(R2::CEditor::AnimationModeLoading) <= uint32(mode)
&& uint32(mode) <= uint32(R2::CEditor::AnimationModeGoingToPlay);
}
inviteButton->setActive(isDM);
}
}
// Create the free teller in all the desktops
uint8 nMode = pIM->getMode();
pGC->setActive(false);
for (uint8 m = 0; m < MAX_NUM_MODES; ++m)
{
if (m != nMode)
pIM->updateGroupContainerImage(*pGC, m);
}
// the group is only active on the current desktop
pGC->setActive(true);
}
if (!winColor.empty())
_FreeTellers[i]->setHeaderColor(winColor);
updateFreeTellerHeader(*_FreeTellers[i]);
return _FreeTellers[i];
}
//=================================================================================
void CChatGroupWindow::updateAllFreeTellerHeaders()
{
for(uint k = 0; k < _FreeTellers.size(); ++k)
{
if (_FreeTellers[k])
{
updateFreeTellerHeader(*_FreeTellers[k]);
}
}
}
//=================================================================================
void CChatGroupWindow::updateFreeTellerHeader(CGroupContainer &ft)
{
ucstring name = ft.getUCTitle();
CCtrlBaseButton *newFriendBut = dynamic_cast(ft.getCtrl("new_friend"));
CCtrlBaseButton *ignoreBut = dynamic_cast(ft.getCtrl("ignore"));
CCtrlBaseButton *inviteBut = dynamic_cast(ft.getCtrl("invite"));
if (newFriendBut)
{
newFriendBut->setFrozen(PeopleInterraction.isContactInList(name, 0));
}
if (ignoreBut)
{
ignoreBut->setFrozen(PeopleInterraction.isContactInList(name, 1));
}
if (inviteBut)
{
bool isDM = false;
R2::CEditor::TMode mode = R2::getEditor().getMode();
if (mode != R2::CEditor::NotInitialized )
{
isDM = R2::getEditor().getDMC().getEditionModule().isSessionOwner()
&& uint32(R2::CEditor::AnimationModeLoading) <= uint32(mode)
&& uint32(mode) <= uint32(R2::CEditor::AnimationModeGoingToPlay);
}
inviteBut->setActive(isDM);
if (isDM)
{
inviteBut->setFrozen(false); // TODO Boris : true if player is already invited in anim + do
// PeopleInterraction.updateAllFreeTellerHeaders() when list is updated
}
}
}
//=================================================================================
void CChatGroupWindow::setActiveFreeTeller(const ucstring &winName, bool bActive)
{
CGroupContainer *pGC = createFreeTeller(winName);
if (pGC != NULL)
pGC->setActive(bActive);
}
//=================================================================================
ucstring CChatGroupWindow::getFreeTellerName(const std::string &containerID)
{
uint32 i;
for (i = 0; i < _FreeTellers.size(); ++i)
{
CGroupContainer *pGC = _FreeTellers[i];
if (pGC->getId() == containerID)
break;
}
if (i == _FreeTellers.size())
return ucstring("");
return _FreeTellers[i]->getUCTitle();
}
//=================================================================================
bool CChatGroupWindow::removeFreeTeller(const std::string &containerID)
{
uint32 i;
for (i = 0; i < _FreeTellers.size(); ++i)
{
CGroupContainer *pGC = _FreeTellers[i];
if (pGC->getId() == containerID)
break;
}
if (i == _FreeTellers.size())
return false;
CInterfaceManager *pIM = CInterfaceManager::getInstance();
// Create the free teller in all the desktops images
for (uint m = 0; m < MAX_NUM_MODES; ++m)
{
pIM->removeGroupContainerImage(_FreeTellers[i]->getId(), m);
}
CInterfaceGroup *pRoot = dynamic_cast(pIM->getElementFromId("ui:interface"));
pIM->unMakeWindow(_FreeTellers[i]);
pRoot->delGroup (_FreeTellers[i]);
_FreeTellers[i] = NULL;
_FreeTellers.erase(_FreeTellers.begin()+i);
return true;
}
//=================================================================================
void CChatGroupWindow::removeAllFreeTellers()
{
while (!_FreeTellers.empty())
{
if (_FreeTellers[0])
{
removeFreeTeller(_FreeTellers[0]->getId());
}
else
{
_FreeTellers.erase(_FreeTellers.begin());
}
}
}
//=================================================================================
void CChatGroupWindow::saveFreeTeller(NLMISC::IStream &f)
{
f.serialVersion(2);
// Save the free teller only if it is present in the friend list to avoid the only-growing situation
// because free tellers are never deleted in game if we save/load all the free tellers, we just create more
// and more container.
uint32 i, nNbFreeTellerSaved = 0;
for (i = 0; i < _FreeTellers.size(); ++i)
if (PeopleInterraction.FriendList.getIndexFromName(_FreeTellers[i]->getUCTitle()) != -1)
nNbFreeTellerSaved++;
f.serial(nNbFreeTellerSaved);
for (i = 0; i < _FreeTellers.size(); ++i)
{
CGroupContainer *pGC = _FreeTellers[i];
if (PeopleInterraction.FriendList.getIndexFromName(pGC->getUCTitle()) != -1)
{
ucstring sTitle = pGC->getUCTitle();
f.serial(sTitle);
}
}
}
//=================================================================================
void CChatGroupWindow::loadFreeTeller(NLMISC::IStream &f)
{
sint ver = f.serialVersion(2);
if (ver == 1)
{
// Old serialized FreeTellIdCounter (deprecated with v2).
uint32 tmp;
f.serial(tmp);
}
uint32 i, nNbFreeTellerSaved = 0;
f.serial(nNbFreeTellerSaved);
for (i = 0; i < nNbFreeTellerSaved; ++i)
{
if (ver == 1)
{
// Old serialized sID (deprecated with v2).
string sID;
f.serial(sID);
}
ucstring sTitle;
f.serial(sTitle);
CGroupContainer *pGC = createFreeTeller(sTitle, "");
// With version 1 all tells are active because windows information have "title based" ids and no "sID based".
if ((ver == 1) && (pGC != NULL))
pGC->setActive(false);
}
}
//=================================================================================
void CChatGroupWindow::clearMessages(CChatGroup::TGroupType gt, uint32 dynamicChatDbIndex)
{
// if not correctly init, abort
if(!_Chat)
return;
// get the gl and tab according to filter
CGroupList *gl;
CCtrlTabButton *tab;
getAssociatedSubWindow(gt, dynamicChatDbIndex, gl, tab);
// delete all text lines
if(gl!=NULL)
{
gl->deleteAllChildren();
}
}
//=================================================================================
void CChatGroupWindow::getAssociatedSubWindow(CChatGroup::TGroupType gt, uint32 dynamicChatDbIndex, CGroupList *&gl, CCtrlTabButton *&tab)
{
nlassert(_Chat);
gl= NULL;
tab= NULL;
switch(gt)
{
default:
case CChatGroup::say:
gl = dynamic_cast(_Chat->getGroup("content:cb:around:text_list"));
tab = dynamic_cast(_Chat->getCtrl("header_opened:channel_select:tab0"));
break;
case CChatGroup::team:
gl = dynamic_cast(_Chat->getGroup("content:cb:team:text_list"));
tab = dynamic_cast(_Chat->getCtrl("header_opened:channel_select:tab2"));
break;
case CChatGroup::guild:
gl = dynamic_cast(_Chat->getGroup("content:cb:guild:text_list"));
tab = dynamic_cast(_Chat->getCtrl("header_opened:channel_select:tab3"));
break;
case CChatGroup::system:
gl = dynamic_cast(_Chat->getGroup("content:cb:sysinfo:text_list"));
tab = dynamic_cast(_Chat->getCtrl("header_opened:channel_select:tab4"));
break;
case CChatGroup::region:
gl = dynamic_cast(_Chat->getGroup("content:cb:region:text_list"));
tab = dynamic_cast(_Chat->getCtrl("header_opened:channel_select:tab1"));
break;
case CChatGroup::yubo_chat:
gl = dynamic_cast(_Chat->getGroup("content:cb:yubo_chat:text_list"));
tab = dynamic_cast(_Chat->getCtrl("header_opened:channel_select:tab6"));
break;
case CChatGroup::dyn_chat:
{
// use dynamicChatDbIndex to get the wanted tab button/group
gl = dynamic_cast(_Chat->getGroup(toString("content:cb:dyn_chat%d:text_list", dynamicChatDbIndex)));
tab = dynamic_cast(_Chat->getCtrl(toString("header_opened:channel_select:tab_array0_%d",dynamicChatDbIndex)));
}
break;
case CChatGroup::universe:
gl = dynamic_cast(_Chat->getGroup("content:cb:universe:text_list"));
tab = dynamic_cast(_Chat->getCtrl("header_opened:channel_select:tab7"));
break;
}
}
////////////////////////
// CChatWindowManager //
////////////////////////
//=================================================================================
CChatWindowManager::CChatWindowManager() : _WindowID(0)
{
}
//=================================================================================
CChatWindowManager::~CChatWindowManager()
{
for(TChatWindowMap::iterator it = _ChatWindowMap.begin(); it != _ChatWindowMap.end(); ++it)
{
delete it->second;
}
}
//=================================================================================
CChatWindow *CChatWindowManager::createChatWindow(const CChatWindowDesc &desc)
{
if (getChatWindow(desc.Title))
{
return NULL; // duplicate name encountered
}
CChatWindow *w;
w = new CChatWindow;
string zeId;
if (desc.Id.empty())
zeId = NLMISC::toString("chat_window_id_%d", (int) _WindowID);
else
zeId = desc.Id;
if (w->create(desc, zeId))
{
if (desc.Id.empty())
_WindowID++;
if (desc.Localize)
_ChatWindowMap[CI18N::get(desc.Title.toString())] = w;
else
_ChatWindowMap[desc.Title] = w;
w->setAHOnActive(desc.AHOnActive);
w->setAHOnActiveParams(desc.AHOnActiveParams);
w->setAHOnDeactive(desc.AHOnDeactive);
w->setAHOnDeactiveParams(desc.AHOnDeactiveParams);
w->setAHOnCloseButton(desc.AHOnCloseButton);
w->setAHOnCloseButtonParams(desc.AHOnCloseButtonParams);
if (!desc.HeaderColor.empty())
w->setHeaderColor(desc.HeaderColor);
return w;
}
else
{
return NULL;
}
}
//=================================================================================
CChatWindow *CChatWindowManager::createChatGroupWindow(const CChatWindowDesc &desc)
{
if (getChatWindow(desc.Title))
{
return NULL; // duplicate name encountered
}
CChatGroupWindow *w;
w = new CChatGroupWindow;
string zeId;
if (desc.Id.empty())
zeId = NLMISC::toString("chat_window_id_%d", (int) _WindowID);
else
zeId = desc.Id;
if (w->create(desc, zeId))
{
if (desc.Id.empty())
_WindowID++;
if (desc.Localize)
_ChatWindowMap[CI18N::get(desc.Title.toString())] = w;
else
_ChatWindowMap[desc.Title] = w;
w->setAHOnActive(desc.AHOnActive);
w->setAHOnActiveParams(desc.AHOnActiveParams);
w->setAHOnDeactive(desc.AHOnDeactive);
w->setAHOnDeactiveParams(desc.AHOnDeactiveParams);
w->setAHOnCloseButton(desc.AHOnCloseButton);
w->setAHOnCloseButtonParams(desc.AHOnCloseButtonParams);
if (!desc.HeaderColor.empty())
w->setHeaderColor(desc.HeaderColor);
return w;
}
else
{
return NULL;
}
}
//=================================================================================
CChatWindow *CChatWindowManager::getChatWindow(const ucstring &title)
{
TChatWindowMap::iterator it = _ChatWindowMap.find(title);
if (it == _ChatWindowMap.end())
return NULL;
else
{
nlassert(it->second != NULL);
return it->second;
}
}
//=================================================================================
void CChatWindowManager::removeChatWindow(const ucstring &title)
{
TChatWindowMap::iterator it = _ChatWindowMap.find(title);
if (it == _ChatWindowMap.end())
{
nlwarning("unknwown window %s", title.toString().c_str());
return;
}
it->second->deleteContainer();
delete it->second;
_ChatWindowMap.erase(it);
}
//=================================================================================
CChatWindowManager &CChatWindowManager::getInstance()
{
static CChatWindowManager instance;
return instance;
}
//=================================================================================
CChatWindow *CChatWindowManager::getChatWindowFromCaller(CCtrlBase *caller)
{
// retrieve pointer on the CChatWindow instance associated with the ui
// find first enclosing group container
CGroupContainer *father = NULL;
while (caller)
{
father = dynamic_cast(caller);
if (father) break;
caller = caller->getParent();
}
if (!father) return NULL;
return getChatWindow(father->getUCTitle());
}
//=================================================================================
bool CChatWindowManager::rename(const ucstring &oldName, const ucstring &newName, bool newNameLocalize)
{
// if (oldName == newName) return true;
CChatWindow *newWin = getChatWindow(newName);
if (newWin != NULL) return false; // target window exists
TChatWindowMap::iterator it = _ChatWindowMap.find(oldName);
if (it == _ChatWindowMap.end()) return false;
if (newNameLocalize)
{
_ChatWindowMap[CI18N::get(newName.toString())] = it->second;
it->second->getContainer()->setLocalize(true);
it->second->getContainer()->setTitle(newName.toString());
}
else
{
_ChatWindowMap[newName] = it->second;
it->second->getContainer()->setLocalize(false);
it->second->getContainer()->setUCTitle(newName);
}
_ChatWindowMap.erase(it);
return true;
}
//=================================================================================
CGroupEditBox *CChatWindow::getEditBox() const
{
if (!_Chat) return NULL;
return dynamic_cast(_Chat->getGroup("eb"));
}
//=================================================================================
void CChatWindowManager::removeChatWindow(CChatWindow *cw)
{
if (!cw) return;
removeChatWindow(cw->getTitle());
}
//=================================================================================
CChatWindow *CChatWindowManager::getChatWindowByIndex(uint index)
{
TChatWindowMap::iterator it = _ChatWindowMap.begin();
while (index--) { ++it; }
return it->second;
}
/////////////////////
// ACTION HANDLERS //
/////////////////////
// ***************************************************************************************
class CHandlerChatBoxEntry : public IActionHandler
{
public:
void execute (CCtrlBase *pCaller, const std::string &/* sParams */)
{
CGroupEditBox *pEB = dynamic_cast(pCaller);
if (pEB == NULL) return;
ucstring text = pEB->getInputString();
// If the line is empty, do nothing
if(text.size() == 0)
return;
CChatWindow *chat = getChatWndMgr().getChatWindowFromCaller(pCaller);
if (!chat)
{
nlwarning("No chat box associated with %s", pEB->getId().c_str());
return;
}
// Parse any tokens in the text
if ( ! CInterfaceManager::parseTokens(text))
{
pEB->setInputString (string(""));
return;
}
// if, it s a command, execute it and don't send the command to the server
if(text[0] == '/')
{
CChatWindow::_ChatWindowLaunchingCommand = chat;
string str = text.toUtf8();
string cmdWithArgs = str.substr(1);
/* In the chat context, only ' ' is a possible separator */
string cmd = cmdWithArgs.substr(0, cmdWithArgs.find(' '));
if ( NLMISC::ICommand::exists( cmd ) )
{
NLMISC::ICommand::execute( cmdWithArgs, g_log );
}
else
{
CInterfaceManager *im = CInterfaceManager::getInstance();
im->displaySystemInfo (ucstring(cmd+": ")+CI18N::get ("uiCommandNotExists"));
}
}
else
{
if (chat->getListener())
{
chat->getListener()->msgEntered(text, chat);
}
}
// Clear input string
pEB->setInputString (string(""));
CGroupContainer *gc = pEB->getEnclosingContainer();
if (gc)
{
// Restore position of enclosing container if it hasn't been moved/scaled/poped by the user
if (!gc->getTouchFlag(true))
{
gc->restorePosition();
}
}
}
};
REGISTER_ACTION_HANDLER(CHandlerChatBoxEntry, "chat_box_entry");
static ucstring getFreeTellerName(CInterfaceElement *pCaller)
{
if (!pCaller) return ucstring();
CChatGroupWindow *cgw = PeopleInterraction.getChatGroupWindow();
if (!cgw) return ucstring();
CGroupContainer *freeTeller = pCaller->getParentContainer();
if (!freeTeller) return ucstring();
return cgw->getFreeTellerName(freeTeller->getId());
}
// ***************************************************************************************
class CHandlerAddTellerToFriendList : public IActionHandler
{
public:
void execute (CCtrlBase *pCaller, const std::string &/* sParams */)
{
ucstring playerName = ::getFreeTellerName(pCaller);
if (!playerName.empty())
{
sint playerIndex = PeopleInterraction.IgnoreList.getIndexFromName(playerName);
// if already in friend list, ask to move rather than add
if (playerIndex != -1)
{
PeopleInterraction.askMoveContact(playerIndex, &PeopleInterraction.IgnoreList, &PeopleInterraction.FriendList);
}
else
{
PeopleInterraction.askAddContact(playerName, &PeopleInterraction.FriendList);
}
}
}
};
REGISTER_ACTION_HANDLER(CHandlerAddTellerToFriendList, "add_teller_to_friend_list");
// ***************************************************************************************
class CHandlerAddTellerToIgnoreList : public IActionHandler
{
public:
void execute (CCtrlBase *pCaller, const std::string &sParams)
{
CInterfaceManager *im = CInterfaceManager::getInstance();
std::string callerId = getParam(sParams, "id");
CInterfaceElement *prevCaller = im->getElementFromId(callerId);
ucstring playerName = ::getFreeTellerName(prevCaller);
if (!playerName.empty())
{
// if already in friend list, ask to move rather than add
sint playerIndex = PeopleInterraction.FriendList.getIndexFromName(playerName);
if (playerIndex != -1)
{
PeopleInterraction.askMoveContact(playerIndex, &PeopleInterraction.FriendList, &PeopleInterraction.IgnoreList);
}
else
{
PeopleInterraction.askAddContact(playerName, &PeopleInterraction.IgnoreList);
}
if (pCaller)
{
CGroupContainer *win = prevCaller->getParentContainer();
if (win)
{
win->setActive(false);
}
}
}
}
};
REGISTER_ACTION_HANDLER(CHandlerAddTellerToIgnoreList, "add_teller_to_ignore_list");
// ***************************************************************************************
class CHandlerInviteToRingSession : public IActionHandler
{
public:
void execute (CCtrlBase *pCaller, const std::string &/* sParams */)
{
ucstring playerName = ::getFreeTellerName(pCaller);
if (!playerName.empty())
{
// ask the SBS to invite the character in the session
CSessionBrowserImpl::getInstance().inviteCharacterByName(CSessionBrowserImpl::getInstance().getCharId(), playerName.toUtf8());
// additionaly, send a tell to signal the player he has been invited to a ring session
ChatMngr.tell(playerName.toUtf8(), CI18N::get("uiRingInviteNotification"));
//
CInterfaceManager *im = CInterfaceManager::getInstance();
im->displaySystemInfo(ucstring("@{6F6F}") + playerName +ucstring(" @{FFFF}") + CI18N::get("uiRingInvitationSent"), "BC");
// force a refresh of the ui
im->executeLuaScript("CharTracking:forceRefresh()");
}
}
};
REGISTER_ACTION_HANDLER(CHandlerInviteToRingSession, "invite_to_ring_session");