3366 lines
108 KiB
C++
3366 lines
108 KiB
C++
// Ryzom - MMORPG Framework <http://dev.ryzom.com/projects/ryzom/>
|
|
// 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 <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
#include "stdpch.h"
|
|
// client
|
|
#include "../string_manager_client.h"
|
|
#include "people_interraction.h"
|
|
#include "nel/gui/interface_expr.h"
|
|
#include "interface_manager.h"
|
|
#include "nel/gui/action_handler.h"
|
|
#include "action_handler_misc.h"
|
|
#include "chat_window.h"
|
|
#include "../entity_animation_manager.h"
|
|
#include "nel/gui/group_editbox.h"
|
|
#include "nel/gui/group_menu.h"
|
|
#include "../client_chat_manager.h"
|
|
#include "../string_manager_client.h"
|
|
#include "nel/gui/interface_expr.h"
|
|
#include "nel/gui/ctrl_button.h"
|
|
#include "nel/gui/ctrl_text_button.h"
|
|
#include "filtered_chat_summary.h"
|
|
#include "input_handler_manager.h"
|
|
#include "../user_entity.h"
|
|
#include "../entities.h"
|
|
#include "../net_manager.h"
|
|
#include "../connection.h"
|
|
#include "nel/gui/group_tab.h"
|
|
#include "guild_manager.h"
|
|
// Game share
|
|
#include "game_share/entity_types.h"
|
|
// NeL
|
|
#include <nel/misc/command.h>
|
|
#include <nel/misc/rgba.h>
|
|
#include <nel/misc/i18n.h>
|
|
|
|
#include <string>
|
|
|
|
#include "../misc.h"
|
|
|
|
using namespace NLMISC;
|
|
using namespace std;
|
|
|
|
|
|
|
|
////////////
|
|
// EXTERN //
|
|
////////////
|
|
|
|
extern CEntityAnimationManager *EAM;
|
|
extern CClientChatManager ChatMngr;
|
|
extern NLMISC::CLog g_log;
|
|
|
|
/////////////
|
|
// GLOBALS //
|
|
/////////////
|
|
|
|
CPeopleInterraction PeopleInterraction;
|
|
|
|
|
|
static const string MAIN_CHAT_SOURCE_MENU = "ui:interface:main_chat_source_menu";
|
|
static const string USER_CHAT_SOURCE_MENU = "ui:interface:user_chat_source_menu";
|
|
static const string STD_CHAT_SOURCE_MENU = "ui:interface:std_chat_source_menu";
|
|
static const string NEW_PARTY_CHAT_WINDOW = "ui:interface:create_new_party_chat";
|
|
|
|
NLMISC::CRefPtr<CChatWindow> ChatWindowForFilter;
|
|
|
|
static const sint PARTY_CHAT_SPAWN_DELTA = 20; // to avoid that all party chat appear at the same position, a random value is added
|
|
|
|
//////////////////////////////////
|
|
// STATIC FUNCTIONs DECLARATION //
|
|
//////////////////////////////////
|
|
|
|
/** Display an error msg in the system info window, and also in the last window that triggered the command (so that the user is sure to see it)
|
|
*/
|
|
static void displayVisibleSystemMsg(const ucstring &msg, const string &cat = "CHK");
|
|
|
|
|
|
//////////////////////////////
|
|
// HANDLER FOR CHAT WINDOWS //
|
|
//////////////////////////////
|
|
|
|
// handler to manage user entry in party chat windows
|
|
|
|
struct CPartyChatEntryHandler : public IChatWindowListener
|
|
{
|
|
virtual void msgEntered(const ucstring &msg, CChatWindow *chatWindow)
|
|
{
|
|
if (ClientCfg.Local)
|
|
{
|
|
chatWindow->displayMessage(msg, CRGBA::White, CChatGroup::player, 0);
|
|
}
|
|
else
|
|
{
|
|
// TODO GAMEDEV : manage entry in the party chat
|
|
}
|
|
}
|
|
};
|
|
|
|
|
|
|
|
// handler to manage user entry in 'around me' window
|
|
struct CAroundMeEntryHandler : public IChatWindowListener
|
|
{
|
|
virtual void msgEntered(const ucstring &msg, CChatWindow *chatWindow)
|
|
{
|
|
if (ClientCfg.Local)
|
|
{
|
|
chatWindow->displayMessage(msg, CRGBA::White, CChatGroup::arround, 0);
|
|
}
|
|
else
|
|
{
|
|
// process msg as usual
|
|
ChatMngr.setChatMode(CChatGroup::arround);
|
|
ChatMngr.chat(msg);
|
|
}
|
|
}
|
|
};
|
|
|
|
// handler to manage user entry in 'region' window
|
|
struct CRegionEntryHandler : public IChatWindowListener
|
|
{
|
|
virtual void msgEntered(const ucstring &msg, CChatWindow *chatWindow)
|
|
{
|
|
if (ClientCfg.Local)
|
|
{
|
|
chatWindow->displayMessage(msg, CRGBA::White, CChatGroup::region, 0);
|
|
}
|
|
else
|
|
{
|
|
// process msg as usual
|
|
ChatMngr.setChatMode(CChatGroup::region);
|
|
ChatMngr.chat(msg);
|
|
}
|
|
}
|
|
};
|
|
|
|
// handler to manage user entry in 'universe' window
|
|
struct CUniverseEntryHandler : public IChatWindowListener
|
|
{
|
|
virtual void msgEntered(const ucstring &msg, CChatWindow *chatWindow)
|
|
{
|
|
if (ClientCfg.Local)
|
|
{
|
|
chatWindow->displayMessage(msg, CRGBA::White, CChatGroup::universe, 0);
|
|
}
|
|
else
|
|
{
|
|
// process msg as usual
|
|
ChatMngr.setChatMode(CChatGroup::universe);
|
|
ChatMngr.chat(msg);
|
|
}
|
|
}
|
|
};
|
|
|
|
// handler to manage user entry in 'guild chat' window
|
|
struct CGuildChatEntryHandler : public IChatWindowListener
|
|
{
|
|
virtual void msgEntered(const ucstring &msg, CChatWindow *chatWindow)
|
|
{
|
|
if (ClientCfg.Local)
|
|
{
|
|
chatWindow->displayMessage(msg, CRGBA::White, CChatGroup::guild, 0);
|
|
}
|
|
else
|
|
{
|
|
ChatMngr.setChatMode(CChatGroup::guild);
|
|
ChatMngr.chat(msg);
|
|
}
|
|
}
|
|
};
|
|
|
|
// handler to manage user entry in 'team chat' window
|
|
struct CTeamChatEntryHandler : public IChatWindowListener
|
|
{
|
|
virtual void msgEntered(const ucstring &msg, CChatWindow *chatWindow)
|
|
{
|
|
if (ClientCfg.Local)
|
|
{
|
|
chatWindow->displayMessage(msg, CRGBA::White, CChatGroup::team, 0);
|
|
}
|
|
else
|
|
{
|
|
ChatMngr.setChatMode(CChatGroup::team);
|
|
ChatMngr.chat(msg, true);
|
|
}
|
|
}
|
|
};
|
|
|
|
// handler to manage user entry in a 'talk with friend' window
|
|
struct CFriendTalkEntryHandler : public IChatWindowListener
|
|
{
|
|
virtual void msgEntered(const ucstring &msg, CChatWindow *chatWindow)
|
|
{
|
|
if (ClientCfg.Local)
|
|
{
|
|
chatWindow->displayMessage(msg, CRGBA::White, CChatGroup::player, 0);
|
|
}
|
|
else
|
|
{
|
|
|
|
// TODO GAMEDEV : send msg to other player
|
|
}
|
|
}
|
|
};
|
|
|
|
// handler to manage user entry in a debug console window
|
|
struct CDebugConsoleEntryHandler : public IChatWindowListener
|
|
{
|
|
virtual void msgEntered(const ucstring &msg, CChatWindow * /* chatWindow */)
|
|
{
|
|
std::string str = msg.toString();
|
|
NLMISC::ICommand::execute( str, g_log );
|
|
}
|
|
};
|
|
|
|
// handler to manager user entry in a Yubo Chat
|
|
struct CYuboChatEntryHandler : public IChatWindowListener
|
|
{
|
|
virtual void msgEntered(const ucstring &msg, CChatWindow * /* chatWindow */)
|
|
{
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
pIM->sendStringToYuboChat(msg);
|
|
}
|
|
};
|
|
|
|
// handler to manager user entry in a Dynamic Chat
|
|
struct CDynamicChatEntryHandler : public IChatWindowListener
|
|
{
|
|
public:
|
|
uint DbIndex;
|
|
CDynamicChatEntryHandler()
|
|
{
|
|
DbIndex= 0;
|
|
}
|
|
|
|
virtual void msgEntered(const ucstring &msg, CChatWindow *chatWindow)
|
|
{
|
|
if (ClientCfg.Local)
|
|
{
|
|
chatWindow->displayMessage(msg, CRGBA::White, CChatGroup::dyn_chat, DbIndex);
|
|
}
|
|
else
|
|
{
|
|
ChatMngr.setChatMode(CChatGroup::dyn_chat, ChatMngr.getDynamicChannelIdFromDbIndex(DbIndex));
|
|
ChatMngr.chat(msg);
|
|
}
|
|
}
|
|
};
|
|
|
|
|
|
// handler instances
|
|
static CPartyChatEntryHandler PartyChatEntryHandler;
|
|
static CAroundMeEntryHandler AroundMeEntryHandler;
|
|
static CRegionEntryHandler RegionEntryHandler;
|
|
static CUniverseEntryHandler UniverseEntryHandler;
|
|
static CGuildChatEntryHandler GuildChatEntryHandler;
|
|
static CTeamChatEntryHandler TeamChatEntryHandler;
|
|
static CFriendTalkEntryHandler FriendTalkEntryHandler;
|
|
static CDebugConsoleEntryHandler DebugConsoleEntryHandler;
|
|
static CYuboChatEntryHandler YuboChatEntryHandler;
|
|
static CDynamicChatEntryHandler DynamicChatEntryHandler[CChatGroup::MaxDynChanPerPlayer];
|
|
|
|
|
|
//////////////////////
|
|
// MEMBER FUNCTIONS //
|
|
//////////////////////
|
|
|
|
//===========================================================================================================
|
|
void CChatStdInput::registerListeningWindow(CChatWindow *cw)
|
|
{
|
|
if (!cw) return;
|
|
Guild.addListeningWindow(cw);
|
|
Team.addListeningWindow(cw);
|
|
Tell.addListeningWindow(cw);
|
|
AroundMe.addListeningWindow(cw);
|
|
Region.addListeningWindow(cw);
|
|
SystemInfo.addListeningWindow(cw);
|
|
Universe.addListeningWindow(cw);
|
|
}
|
|
|
|
|
|
//===========================================================================================================
|
|
CPeopleInterraction::CPeopleInterraction() : Region(NULL),
|
|
Universe(NULL),
|
|
TeamChat(NULL),
|
|
GuildChat(NULL),
|
|
SystemInfo(NULL),
|
|
TellWindow(NULL),
|
|
DebugInfo(NULL),
|
|
YuboChat(NULL),
|
|
CurrPartyChatID(0)
|
|
{
|
|
for(uint i=0;i<CChatGroup::MaxDynChanPerPlayer;i++)
|
|
{
|
|
DynamicChat[i]= NULL;
|
|
}
|
|
}
|
|
|
|
|
|
//===========================================================================================================
|
|
void CPeopleInterraction::release()
|
|
{
|
|
ChatInput.Tell.removeListeningPeopleList(&FriendList);
|
|
ChatInput.Tell.removeListeningPeopleList(&TeamList);
|
|
ChatInput.Team.removeListeningPeopleList(&TeamList);
|
|
|
|
CChatWindowManager &cwm = getChatWndMgr();
|
|
|
|
AroundMe.release();
|
|
if (Region) cwm.removeChatWindow(Region);
|
|
if (Universe) cwm.removeChatWindow(Universe);
|
|
if (TeamChat) cwm.removeChatWindow(TeamChat);
|
|
if (GuildChat) cwm.removeChatWindow(GuildChat);
|
|
if (SystemInfo) cwm.removeChatWindow(SystemInfo);
|
|
TheUserChat.release();
|
|
if (DebugInfo) cwm.removeChatWindow(DebugInfo);
|
|
if (YuboChat) cwm.removeChatWindow(YuboChat);
|
|
// if (TellWindow) cwm.removeChatWindow(TellWindow);
|
|
TeamList.reset();
|
|
FriendList.reset();
|
|
IgnoreList.reset();
|
|
|
|
Region = NULL;
|
|
Universe = NULL;
|
|
TeamChat = NULL;
|
|
GuildChat = NULL;
|
|
SystemInfo = NULL;
|
|
TellWindow = NULL;
|
|
DebugInfo = NULL;
|
|
YuboChat = NULL;
|
|
// TellWindow = NULL;
|
|
|
|
for(uint i=0;i<CChatGroup::MaxDynChanPerPlayer;i++)
|
|
{
|
|
if(DynamicChat[i]) cwm.removeChatWindow(DynamicChat[i]);
|
|
DynamicChat[i]= NULL;
|
|
}
|
|
|
|
removeAllPartyChat();
|
|
|
|
ChatGroup.release();
|
|
|
|
for(uint k = 0; k < MaxNumUserChats; ++k)
|
|
{
|
|
UserChat[k].release();
|
|
}
|
|
uint numCW = cwm.getNumChatWindow();
|
|
if (numCW != 0)
|
|
{
|
|
nlwarning("%d chat windows have not been deleted", (int) numCW);
|
|
uint numCW = cwm.getNumChatWindow();
|
|
for(uint k = 0; k < numCW; ++k)
|
|
{
|
|
nlwarning("Window %d : %s", (int) k, (cwm.getChatWindowByIndex(k)->getTitle().toString()).c_str());
|
|
}
|
|
}
|
|
}
|
|
|
|
//===========================================================================================================
|
|
void CPeopleInterraction::removeAllPartyChat()
|
|
{
|
|
CChatWindowManager &cwm = getChatWndMgr();
|
|
for(std::vector<CPartyChatInfo>::iterator it = PartyChats.begin(); it != PartyChats.end(); ++it)
|
|
{
|
|
if (it->Window) cwm.removeChatWindow(it->Window);
|
|
it->Window = NULL;
|
|
}
|
|
PartyChats.clear();
|
|
// remove filtered chats
|
|
//cwm.removeChatWindow(MainChat.Window);
|
|
//MainChat.Window = NULL;
|
|
cwm.removeChatWindow(ChatGroup.Window);
|
|
ChatGroup.Window = NULL;
|
|
for(uint k = 0; k < MaxNumUserChats; ++k)
|
|
{
|
|
if (UserChat[k].Window)
|
|
{
|
|
cwm.removeChatWindow(UserChat[k].Window);
|
|
UserChat[k].Window = NULL;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
//===========================================================================================================
|
|
bool CPeopleInterraction::isUserChat(CChatWindow *cw) const
|
|
{
|
|
// if (cw == MainChat.Window) return true;
|
|
if (cw == ChatGroup.Window) return true;
|
|
for(uint k = 0; k < MaxNumUserChats; ++k)
|
|
{
|
|
if (UserChat[k].Window == cw) return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
//===========================================================================================================
|
|
void CPeopleInterraction::init()
|
|
{
|
|
// create chat windows
|
|
// todo : internationnalization
|
|
createAroundMeWindow();
|
|
createRegionWindow();
|
|
createUniverseWindow();
|
|
createTeamChat();
|
|
createGuildChat();
|
|
createSystemInfo();
|
|
createTheUserChat();
|
|
createYuboChat();
|
|
createDynamicChats();
|
|
|
|
createDebugInfo();
|
|
//createTellWindow();
|
|
//
|
|
createTeamList();
|
|
createFriendList();
|
|
createIgnoreList();
|
|
|
|
// main chat should be created after other windows, because it relies on them to receive its messages
|
|
createChatGroup();
|
|
|
|
// init the standard inputs
|
|
initStdInputs();
|
|
}
|
|
|
|
|
|
//===========================================================================================================
|
|
void CPeopleInterraction::initAfterLoad()
|
|
{
|
|
/* activate the USER chat per default.
|
|
Important: we must do it after ChatGroup.Window var init, DB color init etc...
|
|
because the latest are used in chat_group_filter ActionHandler
|
|
*/
|
|
CChatGroupWindow *pCGW= PeopleInterraction.getChatGroupWindow();
|
|
if(pCGW)
|
|
pCGW->setTabIndex(5);
|
|
}
|
|
|
|
|
|
//===========================================================================================================
|
|
void CPeopleInterraction::initStdInputs()
|
|
{
|
|
uint i;
|
|
|
|
ChatInput.AroundMe.addListeningWindow (ChatGroup.Window);
|
|
ChatInput.Region.addListeningWindow (ChatGroup.Window);
|
|
ChatInput.Team.addListeningWindow (ChatGroup.Window);
|
|
ChatInput.Guild.addListeningWindow (ChatGroup.Window);
|
|
ChatInput.Tell.addListeningWindow (ChatGroup.Window);
|
|
ChatInput.SystemInfo.addListeningWindow (ChatGroup.Window);
|
|
ChatInput.YuboChat.addListeningWindow (ChatGroup.Window);
|
|
ChatInput.Universe.addListeningWindow (ChatGroup.Window);
|
|
|
|
if (AroundMe.Window)
|
|
ChatInput.AroundMe.addListeningWindow(AroundMe.Window);
|
|
|
|
if (Region)
|
|
ChatInput.Region.addListeningWindow(Region);
|
|
|
|
if (Universe)
|
|
ChatInput.Universe.addListeningWindow(Universe);
|
|
|
|
if (TeamChat)
|
|
ChatInput.Team.addListeningWindow(TeamChat);
|
|
|
|
if (GuildChat)
|
|
ChatInput.Guild.addListeningWindow(GuildChat);
|
|
|
|
if (SystemInfo)
|
|
ChatInput.SystemInfo.addListeningWindow(SystemInfo);
|
|
|
|
if (DebugInfo)
|
|
ChatInput.DebugInfo.addListeningWindow(DebugInfo);
|
|
|
|
if (YuboChat)
|
|
ChatInput.YuboChat.addListeningWindow(YuboChat);
|
|
|
|
if (TheUserChat.Window)
|
|
{
|
|
ChatInput.AroundMe.addListeningWindow(TheUserChat.Window);
|
|
ChatInput.Region.addListeningWindow(TheUserChat.Window);
|
|
ChatInput.Team.addListeningWindow(TheUserChat.Window);
|
|
ChatInput.Guild.addListeningWindow(TheUserChat.Window);
|
|
ChatInput.Universe.addListeningWindow (TheUserChat.Window);
|
|
// Don't add the system info by default
|
|
// Dynamic chats
|
|
for(i = 0; i < CChatGroup::MaxDynChanPerPlayer; i++)
|
|
{
|
|
ChatInput.DynamicChat[i].addListeningWindow(TheUserChat.Window);
|
|
}
|
|
}
|
|
|
|
ChatInput.Tell.addListeningPeopleList(&FriendList);
|
|
ChatInput.Tell.addListeningPeopleList(&TeamList);
|
|
ChatInput.Team.addListeningPeopleList(&TeamList);
|
|
|
|
// Dynamic chats
|
|
for(i=0;i<CChatGroup::MaxDynChanPerPlayer;i++)
|
|
{
|
|
ChatInput.DynamicChat[i].addListeningWindow(ChatGroup.Window);
|
|
if(DynamicChat[i])
|
|
ChatInput.DynamicChat[i].addListeningWindow(DynamicChat[i]);
|
|
}
|
|
}
|
|
|
|
//===========================================================================================================
|
|
void CPeopleInterraction::createTeamList()
|
|
{
|
|
// temp to test people list before it is connected to the server
|
|
CPeopleListDesc peopleListDesc;
|
|
//peopleListDesc.FatherContainer = "ui:interface:communication";
|
|
peopleListDesc.PeopleListTitle = "uiTeamTitle";
|
|
peopleListDesc.Id = "team_list";
|
|
peopleListDesc.ContactType = CPeopleListDesc::Team /* CPeopleListDesc::Contact*/;
|
|
peopleListDesc.BaseContainerTemplateName = "people_list_container";
|
|
peopleListDesc.Localize = true;
|
|
peopleListDesc.Savable = true;
|
|
peopleListDesc.AHOnActive = "proc";
|
|
peopleListDesc.AHOnActiveParams = "team_list_proc_active";
|
|
peopleListDesc.AHOnDeactive = "proc";
|
|
peopleListDesc.AHOnDeactiveParams = "team_list_proc_deactive";
|
|
peopleListDesc.HeaderColor = "UI:SAVE:WIN:COLORS:MEM";
|
|
//
|
|
TeamList.create(peopleListDesc, NULL); // &chatDesc // create the team list with a chat box in it
|
|
TeamList.setMenu("ui:interface:sort_menu");
|
|
// Special case for team : each entry is connected to the database so we create all team member at once
|
|
for(uint k = 0; k < MaxNumPeopleInTeam; ++k)
|
|
{
|
|
TeamList.addPeople(NLMISC::toString(k), k);
|
|
}
|
|
|
|
TeamList.setPeopleMenu("ui:interface:team_member_menu");
|
|
TeamList.setMenu("ui:interface:team_chat_member_menu");
|
|
|
|
// Open/Close team list when at least one / no team mate
|
|
// NB: use an intermediate temp var, to avoid show of the window each time a new team member enters
|
|
string sExpr = "@UI:VARIABLES:IS_TEAM_PRESENT";
|
|
string sAction = "set";
|
|
string sCond = "";
|
|
string sParams = "target_property=ui:interface:team_list:active|value=@UI:VARIABLES:IS_TEAM_PRESENT";
|
|
|
|
if (TeamChat)
|
|
{
|
|
CInterfaceLink *il = new CInterfaceLink;
|
|
vector<CInterfaceLink::CTargetInfo> targets;
|
|
vector<CInterfaceLink::CCDBTargetInfo> cdbTargets;
|
|
il->init(targets, cdbTargets, sExpr, sAction, sParams, sCond, TeamChat->getContainer());
|
|
}
|
|
}
|
|
|
|
//===========================================================================================================
|
|
void CPeopleInterraction::createFriendList()
|
|
{
|
|
// create friend list
|
|
CPeopleListDesc peopleListDesc;
|
|
peopleListDesc.FatherContainer = "ui:interface:contact_list";
|
|
peopleListDesc.PeopleListTitle = "uiFriendList";
|
|
peopleListDesc.Id = "friend_list";
|
|
peopleListDesc.BaseContainerTemplateName= "people_list_container_with_add_edit_box";
|
|
peopleListDesc.ContactType = CPeopleListDesc::Contact;
|
|
peopleListDesc.Localize = true;
|
|
//
|
|
FriendList.create(peopleListDesc);
|
|
FriendList.setPeopleMenuEx("ui:interface:friend_list_menu_offline_unblocked",
|
|
"ui:interface:friend_list_menu_online_unblocked",
|
|
"ui:interface:friend_list_menu_online_abroad_unblocked",
|
|
"ui:interface:friend_list_menu_offline_blocked",
|
|
"ui:interface:friend_list_menu_online_blocked",
|
|
"ui:interface:friend_list_menu_online_abroad_blocked"
|
|
);
|
|
FriendList.setMenu("ui:interface:sort_menu");
|
|
}
|
|
|
|
|
|
//===========================================================================================================
|
|
void CPeopleInterraction::createIgnoreList()
|
|
{
|
|
// create ignore list
|
|
CPeopleListDesc peopleListDesc;
|
|
peopleListDesc.FatherContainer = "ui:interface:contact_list";
|
|
peopleListDesc.PeopleListTitle = "uiIgnoreList";
|
|
peopleListDesc.Id = "ignore_list";
|
|
peopleListDesc.BaseContainerTemplateName = "people_list_container_with_add_edit_box";
|
|
peopleListDesc.ContactType = CPeopleListDesc::Ignore;
|
|
peopleListDesc.Localize = true;
|
|
//
|
|
IgnoreList.create(peopleListDesc);
|
|
//
|
|
IgnoreList.setPeopleMenu("ui:interface:ignore_list_menu");
|
|
IgnoreList.setMenu("ui:interface:sort_menu");
|
|
}
|
|
|
|
//===========================================================================================================
|
|
void CPeopleInterraction::createSystemInfo()
|
|
{
|
|
CChatWindowDesc chatDesc;
|
|
chatDesc.FatherContainer = "ui:interface";
|
|
chatDesc.Title = "uiSystemInfoTitle";
|
|
chatDesc.Listener = NULL;
|
|
chatDesc.Savable = true;
|
|
chatDesc.Localize = true;
|
|
chatDesc.ChatTemplate ="system_info_id";
|
|
chatDesc.Id = "system_info";
|
|
chatDesc.AHOnCloseButton = "proc";
|
|
chatDesc.AHOnCloseButtonParams = "sysinfo_chat_proc_close";
|
|
|
|
SystemInfo = getChatWndMgr().createChatWindow(chatDesc);
|
|
if (!SystemInfo) return;
|
|
SystemInfo->setMenu("ui:interface:base_chat_box_menu");
|
|
}
|
|
|
|
//===========================================================================================================
|
|
void CPeopleInterraction::createDebugInfo()
|
|
{
|
|
// can only be used by devs and CSR or in local mode
|
|
#if FINAL_VERSION
|
|
if( ClientCfg.Local || hasPrivilegeDEV() || hasPrivilegeSGM() )
|
|
#endif
|
|
{
|
|
CChatWindowDesc chatDesc;
|
|
chatDesc.FatherContainer = "ui:interface";
|
|
chatDesc.Title = "uiDebugConsole";
|
|
chatDesc.Listener = NULL;
|
|
chatDesc.Savable = true;
|
|
chatDesc.Localize = true;
|
|
chatDesc.Listener = &DebugConsoleEntryHandler;
|
|
chatDesc.ChatTemplate ="clearable_chat_id";
|
|
chatDesc.Id = "debug_info";
|
|
|
|
DebugInfo = getChatWndMgr().createChatWindow(chatDesc);
|
|
if (!DebugInfo) return;
|
|
DebugInfo->setMenu("ui:interface:base_chat_box_menu");
|
|
}
|
|
}
|
|
|
|
//===========================================================================================================
|
|
void CPeopleInterraction::createAroundMeWindow()
|
|
{
|
|
CChatWindowDesc chatDesc;
|
|
chatDesc.FatherContainer = "ui:interface";
|
|
chatDesc.Title = "uiAroundMeTitle";
|
|
chatDesc.Listener = NULL;
|
|
chatDesc.Localize = true;
|
|
chatDesc.Savable = true;
|
|
//chatDesc.ChatTemplate = "around_me_id";
|
|
chatDesc.Id = "around_me";
|
|
chatDesc.AHOnCloseButton = "proc";
|
|
chatDesc.AHOnCloseButtonParams = "around_me_chat_proc_close";
|
|
|
|
AroundMe.Window = getChatWndMgr().createChatWindow(chatDesc);
|
|
if (!AroundMe.Window) return;
|
|
AroundMe.Window->setMenu(STD_CHAT_SOURCE_MENU);
|
|
// Configure filter for the main chat. By default, it listen to everything (no party chats : none have been created yet)
|
|
AroundMe.Filter.setTargetGroup(CChatGroup::say);
|
|
// associate filter with chat window
|
|
AroundMe.Filter.setChat(AroundMe.Window);
|
|
}
|
|
|
|
|
|
//===========================================================================================================
|
|
void CPeopleInterraction::createRegionWindow()
|
|
{
|
|
// create region window
|
|
CChatWindowDesc chatDesc;
|
|
chatDesc.FatherContainer = "ui:interface";
|
|
chatDesc.Title = "uiRegionTitle";
|
|
chatDesc.Listener = &RegionEntryHandler;
|
|
chatDesc.Localize = true;
|
|
chatDesc.Savable = true;
|
|
chatDesc.Id = "region_chat";
|
|
chatDesc.AHOnCloseButton = "proc";
|
|
chatDesc.AHOnCloseButtonParams = "region_chat_proc_close";
|
|
chatDesc.HeaderColor = "UI:SAVE:WIN:COLORS:MEM";
|
|
|
|
Region = getChatWndMgr().createChatWindow(chatDesc);
|
|
if (!Region) return;
|
|
Region->setMenu(STD_CHAT_SOURCE_MENU);
|
|
}
|
|
|
|
|
|
//===========================================================================================================
|
|
void CPeopleInterraction::createUniverseWindow()
|
|
{
|
|
// create universe window
|
|
CChatWindowDesc chatDesc;
|
|
chatDesc.FatherContainer = "ui:interface";
|
|
chatDesc.Title = "uiUniverseTitle";
|
|
chatDesc.Listener = &UniverseEntryHandler;
|
|
chatDesc.Localize = true;
|
|
chatDesc.Savable = true;
|
|
chatDesc.Id = "universe_chat";
|
|
chatDesc.AHOnCloseButton = "proc";
|
|
chatDesc.AHOnCloseButtonParams = "universe_chat_proc_close";
|
|
chatDesc.HeaderColor = "UI:SAVE:WIN:COLORS:MEM";
|
|
|
|
Universe = getChatWndMgr().createChatWindow(chatDesc);
|
|
if (!Universe) return;
|
|
Universe->setMenu(STD_CHAT_SOURCE_MENU);
|
|
}
|
|
|
|
|
|
//===========================================================================================================
|
|
void CPeopleInterraction::createTellWindow()
|
|
{
|
|
/*CChatWindowDesc chatDesc;
|
|
chatDesc.FatherContainer = "ui:interface";
|
|
chatDesc.Title = "uiTellWindow";
|
|
chatDesc.Listener = NULL;
|
|
chatDesc.Savable = true;
|
|
chatDesc.Localize = true;
|
|
chatDesc.Id = "tell";
|
|
chatDesc.ChatTemplate ="chat_no_eb_id";
|
|
chatDesc.AHOnActive = "set";
|
|
chatDesc.AHOnActiveParams = "dblink=UI:SAVE:ISDETACHED:TELL|value=1";
|
|
chatDesc.AHOnDeactive = "set";
|
|
chatDesc.AHOnDeactiveParams = "dblink=UI:SAVE:ISDETACHED:TELL|value=0";
|
|
|
|
TellWindow = getChatWndMgr().createChatWindow(chatDesc);
|
|
if (!TellWindow) return;
|
|
TellWindow->setMenu("ui:interface:base_chat_box_menu"); */
|
|
}
|
|
|
|
//===========================================================================================================
|
|
void CPeopleInterraction::createTeamChat()
|
|
{
|
|
// create team chat (inserted in team people list)
|
|
CChatWindowDesc chatDesc;
|
|
chatDesc.FatherContainer = "ui:interface";
|
|
chatDesc.Title = "uiTeamChatTitle";
|
|
chatDesc.Listener = &TeamChatEntryHandler;
|
|
chatDesc.Localize = true;
|
|
chatDesc.Savable = true;
|
|
chatDesc.Id = "team_chat";
|
|
chatDesc.AHOnCloseButton = "proc";
|
|
chatDesc.AHOnCloseButtonParams = "team_chat_proc_close";
|
|
chatDesc.HeaderColor = "UI:SAVE:WIN:COLORS:MEM";
|
|
|
|
TeamChat = getChatWndMgr().createChatWindow(chatDesc);
|
|
if (!TeamChat) return;
|
|
TeamChat->setMenu(STD_CHAT_SOURCE_MENU);
|
|
}
|
|
|
|
//===========================================================================================================
|
|
void CPeopleInterraction::createGuildChat()
|
|
{
|
|
CChatWindowDesc chatDesc;
|
|
chatDesc.FatherContainer = "ui:interface";
|
|
chatDesc.Title = "uiGuildChat";
|
|
chatDesc.Listener = &GuildChatEntryHandler;
|
|
chatDesc.Localize = true;
|
|
chatDesc.Savable = true;
|
|
chatDesc.Id = "guild_chat";
|
|
chatDesc.AHOnCloseButton = "proc";
|
|
chatDesc.AHOnCloseButtonParams = "guild_chat_proc_close";
|
|
chatDesc.HeaderColor = "UI:SAVE:WIN:COLORS:MEM";
|
|
|
|
GuildChat = getChatWndMgr().createChatWindow(chatDesc);
|
|
if (!GuildChat) return;
|
|
GuildChat->setMenu(STD_CHAT_SOURCE_MENU);
|
|
}
|
|
|
|
//===========================================================================================================
|
|
void CPeopleInterraction::createYuboChat()
|
|
{
|
|
CChatWindowDesc chatDesc;
|
|
chatDesc.FatherContainer = "ui:interface";
|
|
chatDesc.Title = "uiYuboChat";
|
|
chatDesc.Listener = &YuboChatEntryHandler;
|
|
chatDesc.Localize = true;
|
|
chatDesc.Savable = true;
|
|
chatDesc.Id = "yubo_chat";
|
|
chatDesc.AHOnCloseButton = "proc";
|
|
chatDesc.AHOnCloseButtonParams = "yubo_chat_proc_close";
|
|
chatDesc.HeaderColor = "UI:SAVE:WIN:COLORS:MEM";
|
|
|
|
YuboChat = getChatWndMgr().createChatWindow(chatDesc);
|
|
if (!YuboChat) return;
|
|
YuboChat->setMenu(STD_CHAT_SOURCE_MENU);
|
|
}
|
|
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::createDynamicChats()
|
|
{
|
|
for(uint i=0;i<CChatGroup::MaxDynChanPerPlayer;i++)
|
|
{
|
|
CChatWindowDesc chatDesc;
|
|
chatDesc.FatherContainer = "ui:interface";
|
|
chatDesc.Title = toString("dummyDynChatTitle%d",i); // title not used, but still important because create() test uniqueness
|
|
DynamicChatEntryHandler[i].DbIndex= i;
|
|
chatDesc.Listener = &DynamicChatEntryHandler[i];
|
|
chatDesc.Localize = false;
|
|
chatDesc.Savable = true;
|
|
chatDesc.ChatTemplate ="dynamic_chat_id";
|
|
chatDesc.ChatTemplateParams.push_back(make_pair(string("dyn_chat_nb"),toString(i)));
|
|
chatDesc.Id = string("dynamic_chat") + toString(i);
|
|
// no active proc because active state is driven by database
|
|
chatDesc.AHOnCloseButton = "proc";
|
|
chatDesc.AHOnCloseButtonParams = string("dynamic_chat_proc_close|") + toString(i);
|
|
chatDesc.HeaderColor = "UI:SAVE:WIN:COLORS:MEM";
|
|
|
|
DynamicChat[i] = getChatWndMgr().createChatWindow(chatDesc);
|
|
if (!DynamicChat[i]) continue;
|
|
DynamicChat[i]->setMenu(STD_CHAT_SOURCE_MENU);
|
|
}
|
|
}
|
|
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::createTheUserChat()
|
|
{
|
|
CChatWindowDesc chatDesc;
|
|
chatDesc.FatherContainer = "ui:interface";
|
|
chatDesc.Title = "uiUserChat";
|
|
chatDesc.Listener = NULL;
|
|
chatDesc.Localize = true;
|
|
chatDesc.Savable = true;
|
|
chatDesc.Id = "user_chat";
|
|
chatDesc.ChatTemplate = "filtered_chat_id";
|
|
chatDesc.AHOnActive = "user_chat_active";
|
|
chatDesc.AHOnActiveParams = "";
|
|
chatDesc.AHOnCloseButton = "set";
|
|
chatDesc.AHOnCloseButtonParams = "dblink=UI:SAVE:ISDETACHED:USER_CHAT|value=0";
|
|
|
|
TheUserChat.Window = getChatWndMgr().createChatWindow(chatDesc);
|
|
if (!TheUserChat.Window) return;
|
|
TheUserChat.Window->getContainer()->setup();
|
|
// Configure filter for the new chat (by default, listen to everything but party chats)
|
|
TheUserChat.Filter.setTargetGroup(CChatGroup::say);
|
|
// assoviate filter with chat window
|
|
TheUserChat.Filter.setChat(TheUserChat.Window);
|
|
}
|
|
|
|
|
|
class CHandlerUserChatActive : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
CChatWindow *pCGW = dynamic_cast<CChatWindow*>(PeopleInterraction.TheUserChat.Window);
|
|
if (pCGW == NULL) return;
|
|
CCtrlTextButton *pUserBut = dynamic_cast<CCtrlTextButton*>(pCGW->getContainer()->getCtrl("content:target_button"));
|
|
CInterfaceGroup *pEditBox = dynamic_cast<CInterfaceGroup*>(pCGW->getContainer()->getGroup("content:ebw"));
|
|
|
|
if(!pUserBut)
|
|
return;
|
|
|
|
CChatGroup::TGroupType m = PeopleInterraction.TheUserChat.Filter.getTargetGroup();
|
|
switch(m)
|
|
{
|
|
default:
|
|
case CChatGroup::arround:
|
|
case CChatGroup::say: pUserBut->setHardText("uiFilterAround"); break;
|
|
case CChatGroup::region: pUserBut->setHardText("uiFilterRegion"); break;
|
|
case CChatGroup::universe: pUserBut->setHardText("uiFilterUniverse"); break;
|
|
case CChatGroup::team: pUserBut->setHardText("uiFilterTeam"); break;
|
|
case CChatGroup::guild: pUserBut->setHardText("uiFilterGuild"); break;
|
|
}
|
|
pUserBut->getParent()->updateCoords();
|
|
pUserBut->updateCoords();
|
|
|
|
if (pEditBox != NULL) pEditBox->setW(-pUserBut->getWReal()-4);
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER(CHandlerUserChatActive, "user_chat_active");
|
|
|
|
//===========================================================================================================
|
|
void CPeopleInterraction::createChatGroup()
|
|
{
|
|
CChatWindowDesc chatDesc;
|
|
chatDesc.FatherContainer = "ui:interface";
|
|
chatDesc.Listener = NULL;
|
|
chatDesc.Title= ""; // NB: the chatgroup is the only one that can be not named (because of uniqueness title test)
|
|
chatDesc.Localize = true;
|
|
chatDesc.Savable = true;
|
|
chatDesc.ChatTemplate = "main_chat_group";
|
|
chatDesc.Id = "main_chat";
|
|
chatDesc.AHOnActive = "proc";
|
|
chatDesc.AHOnActiveParams = "main_chat_group_active";
|
|
chatDesc.AHOnDeactive = "proc";
|
|
chatDesc.AHOnDeactiveParams = "main_chat_group_deactive";
|
|
|
|
ChatGroup.Window = getChatWndMgr().createChatGroupWindow(chatDesc);
|
|
if (!ChatGroup.Window) return;
|
|
// Configure filter for the main chat. By default, it listen to everything (no party chats : none have been created yet)
|
|
ChatGroup.Filter.setTargetGroup(CChatGroup::say);
|
|
// associate filter with chat window
|
|
ChatGroup.Filter.setChat(ChatGroup.Window);
|
|
}
|
|
|
|
class CHandlerChatGroupFilter : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase *pCaller, const std::string &sParams)
|
|
{
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
|
|
const string dynChatId="dyn_chat";
|
|
|
|
bool writeRight= true;
|
|
bool isDynChat= false;
|
|
uint32 dynChatDbIndex= 0;
|
|
|
|
// Entry output
|
|
CChatTargetFilter &rCTF = PeopleInterraction.ChatGroup.Filter;
|
|
if (sParams == "around") rCTF.setTargetGroup(CChatGroup::say);
|
|
else if (sParams == "region") rCTF.setTargetGroup(CChatGroup::region);
|
|
else if (sParams == "universe") rCTF.setTargetGroup(CChatGroup::universe);
|
|
else if (sParams == "team") rCTF.setTargetGroup(CChatGroup::team);
|
|
else if (sParams == "guild") rCTF.setTargetGroup(CChatGroup::guild);
|
|
else if (sParams == "sysinfo")
|
|
{
|
|
rCTF.setTargetGroup(CChatGroup::system);
|
|
writeRight= false;
|
|
}
|
|
else if (sParams == "yubo_chat") rCTF.setTargetGroup(CChatGroup::yubo_chat);
|
|
else if (sParams.compare(0, dynChatId.size(), dynChatId)==0)
|
|
{
|
|
// get the number of this tab
|
|
isDynChat= true;
|
|
fromString(sParams.substr(dynChatId.size()), dynChatDbIndex);
|
|
dynChatDbIndex= min(dynChatDbIndex, (uint32)(CChatGroup::MaxDynChanPerPlayer-1));
|
|
rCTF.setTargetGroup(CChatGroup::dyn_chat, dynChatDbIndex);
|
|
}
|
|
|
|
// inform DB for write right.
|
|
NLGUI::CDBManager::getInstance()->getDbProp("UI:VARIABLES:MAIN_CHAT:WRITE_RIGHT")->setValue32(writeRight);
|
|
NLGUI::CDBManager::getInstance()->getDbProp("UI:VARIABLES:MAIN_CHAT:IS_DYN_CHAT")->setValue32(isDynChat);
|
|
NLGUI::CDBManager::getInstance()->getDbProp("UI:VARIABLES:MAIN_CHAT:INDEX_DYN_CHAT")->setValue32(dynChatDbIndex);
|
|
|
|
|
|
// Update Chat Group Window from user chat button
|
|
|
|
CChatGroupWindow *pCGW = PeopleInterraction.getChatGroupWindow();
|
|
if (!pCGW) return;
|
|
CCtrlTextButton *pUserBut = dynamic_cast<CCtrlTextButton*>(pCGW->getContainer()->getCtrl("content:but_user"));
|
|
CCtrlTextButton *pEmoteBut = dynamic_cast<CCtrlTextButton*>(pCGW->getContainer()->getCtrl("content:but_emote"));
|
|
CInterfaceGroup *pEditBox = dynamic_cast<CInterfaceGroup*>(pCGW->getContainer()->getGroup("content:ebw"));
|
|
CInterfaceGroup *pTextList = dynamic_cast<CInterfaceGroup*>(pCGW->getContainer()->getGroup("content:cb"));
|
|
|
|
// Target button choose the right filter
|
|
|
|
|
|
// Special case of the user defined chat
|
|
if (sParams == "user")
|
|
{
|
|
if (pUserBut != NULL)
|
|
{
|
|
CChatGroup::TGroupType m = PeopleInterraction.TheUserChat.Filter.getTargetGroup();
|
|
switch(m)
|
|
{
|
|
default:
|
|
case CChatGroup::arround:
|
|
case CChatGroup::say: pUserBut->setHardText("uiFilterAround"); break;
|
|
case CChatGroup::region: pUserBut->setHardText("uiFilterRegion"); break;
|
|
case CChatGroup::team: pUserBut->setHardText("uiFilterTeam"); break;
|
|
case CChatGroup::guild: pUserBut->setHardText("uiFilterGuild"); break;
|
|
case CChatGroup::universe: pUserBut->setHardText("uiFilterUniverse"); break;
|
|
case CChatGroup::dyn_chat:
|
|
uint32 index = PeopleInterraction.TheUserChat.Filter.getTargetDynamicChannelDbIndex();
|
|
uint32 textId = NLGUI::CDBManager::getInstance()->getDbProp("SERVER:DYN_CHAT:CHANNEL"+toString(index)+":NAME")->getValue32();
|
|
ucstring title;
|
|
STRING_MANAGER::CStringManagerClient::instance()->getDynString(textId, title);
|
|
if (title.empty())
|
|
{
|
|
// Dyn channel not available yet, so set to around
|
|
PeopleInterraction.TheUserChat.Filter.setTargetGroup(CChatGroup::arround);
|
|
pUserBut->setHardText("uiFilterAround");
|
|
}
|
|
else
|
|
{
|
|
pUserBut->setHardText(title.toUtf8());
|
|
}
|
|
break;
|
|
// NB: user chat cannot have yubo_chat target
|
|
}
|
|
|
|
pUserBut->setActive(true);
|
|
pUserBut->getParent()->updateCoords();
|
|
pUserBut->updateCoords();
|
|
|
|
pEmoteBut->setActive (true);
|
|
pEmoteBut->updateCoords ();
|
|
|
|
if (pEditBox != NULL)
|
|
{
|
|
pEditBox->setW(-pUserBut->getWReal()-pEmoteBut->getWReal()-8);
|
|
pEditBox->setX(pUserBut->getWReal()+4);
|
|
}
|
|
|
|
|
|
}
|
|
rCTF.setTargetGroup(PeopleInterraction.TheUserChat.Filter.getTargetGroup(), PeopleInterraction.TheUserChat.Filter.getTargetDynamicChannelDbIndex());
|
|
}
|
|
else
|
|
{
|
|
if (pUserBut != NULL) pUserBut->setActive(false);
|
|
|
|
if (pEmoteBut)
|
|
{
|
|
pEmoteBut->setActive (true);
|
|
pEmoteBut->updateCoords ();
|
|
}
|
|
|
|
if (pEditBox != NULL)
|
|
{
|
|
if(pEmoteBut)
|
|
pEditBox->setW(-pEmoteBut->getWReal()-4);
|
|
else
|
|
pEditBox->setW(0);
|
|
pEditBox->setX(0);
|
|
}
|
|
if (pTextList != NULL) pTextList->setX(0);
|
|
}
|
|
|
|
// if called from a tab button => force the tab ctrl button to have standard color
|
|
CCtrlTabButton *pTabButton= dynamic_cast<CCtrlTabButton*>(pCaller);
|
|
if(pTabButton)
|
|
{
|
|
CRGBA stdColor= CRGBA::stringToRGBA(CWidgetManager::getInstance()->getParser()->getDefine("chat_group_tab_color_normal").c_str());
|
|
pTabButton->setTextColorNormal(stdColor);
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER(CHandlerChatGroupFilter, "chat_group_filter");
|
|
|
|
|
|
//===========================================================================================================
|
|
class CHandlerChatGroupUpdatePrompt : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
// re set the target group will automatically reset the prompt and prompt color
|
|
CChatTargetFilter &rCTF = PeopleInterraction.ChatGroup.Filter;
|
|
rCTF.setTargetGroup(rCTF.getTargetGroup(), rCTF.getTargetDynamicChannelDbIndex());
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER(CHandlerChatGroupUpdatePrompt, "chat_group_update_prompt");
|
|
|
|
|
|
//===========================================================================================================
|
|
CPeopleList *CPeopleInterraction::getPeopleListFromContainerID(const std::string &id)
|
|
{
|
|
if (TeamList.getContainerID() == id) return &TeamList;
|
|
else if (FriendList.getContainerID() == id) return &FriendList;
|
|
else if (IgnoreList.getContainerID() == id) return &IgnoreList;
|
|
return NULL;
|
|
}
|
|
|
|
|
|
//===========================================================================================================
|
|
bool CPeopleInterraction::getPeopleFromContainerID(const std::string &id, CPeopleList *&peopleList, uint &destIndex)
|
|
{
|
|
// get people index
|
|
// the name has the form "ui:interface:container_name_people_index"
|
|
typedef std::string::size_type TCharPos;
|
|
TCharPos index = id.find_last_of("_");
|
|
if (index == std::string::npos || index == 0) return false;
|
|
TCharPos nextIndex = id.rfind(":", index);
|
|
if (nextIndex == std::string::npos) return false;
|
|
std::string containerId = id.substr(nextIndex + 1, index - nextIndex - 1);
|
|
// search a container with the good name
|
|
CPeopleList *pl = getPeopleListFromContainerID(containerId);
|
|
if (!pl) return false;
|
|
//
|
|
sint peopleIndex = pl->getIndexFromContainerID(id);
|
|
if (peopleIndex == -1) return false;
|
|
//
|
|
destIndex = (uint) peopleIndex;
|
|
peopleList = pl;
|
|
|
|
return true;
|
|
}
|
|
|
|
//===========================================================================================================
|
|
bool CPeopleInterraction::getPeopleFromCurrentMenu(CPeopleList *&peopleList, uint &index)
|
|
{
|
|
CInterfaceManager *im = CInterfaceManager::getInstance();
|
|
// the group that launched the modal window (the menu) must be the header of the group container that represent a people entry
|
|
CInterfaceGroup *header = dynamic_cast<CInterfaceGroup *>(CWidgetManager::getInstance()->getCtrlLaunchingModal());
|
|
if (!header) return false;
|
|
// get the parent container
|
|
CGroupContainer *gc = dynamic_cast<CGroupContainer *>(header->getParent());
|
|
if (!gc) return false;
|
|
return getPeopleFromContainerID(gc->getId(), peopleList, index);
|
|
}
|
|
|
|
//===========================================================================================================
|
|
CPeopleList *CPeopleInterraction::getPeopleListFromCurrentMenu()
|
|
{
|
|
CInterfaceManager *im = CInterfaceManager::getInstance();
|
|
// the group that launched the modal window (the menu) must be the header of the group container that represent a people entry
|
|
CInterfaceGroup *header = dynamic_cast<CInterfaceGroup *>(CWidgetManager::getInstance()->getCtrlLaunchingModal());
|
|
if (!header) return NULL;
|
|
// get the parent container
|
|
CGroupContainer *gc = dynamic_cast<CGroupContainer *>(header->getParent());
|
|
if (!gc) return NULL;
|
|
std::string::size_type pos = gc->getId().find_last_of(":");
|
|
if (pos == std::string::npos) return NULL;
|
|
return getPeopleListFromContainerID(gc->getId().substr(pos + 1));
|
|
}
|
|
|
|
//===========================================================================================================
|
|
CFilteredChat *CPeopleInterraction::getFilteredChatFromChatWindow(CChatWindow *cw)
|
|
{
|
|
//if (cw == MainChat.Window) return &MainChat;
|
|
if (cw == ChatGroup.Window) return &ChatGroup;
|
|
if (cw == AroundMe.Window) return &AroundMe;
|
|
if (cw == TheUserChat.Window) return &TheUserChat;
|
|
for(uint k = 0; k < MaxNumUserChats; ++k)
|
|
{
|
|
if (UserChat[k].Window == cw) return &UserChat[k];
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
//===========================================================================================================
|
|
void CPeopleInterraction::askAddContact(const ucstring &contactName, CPeopleList *pl)
|
|
{
|
|
if (pl == NULL)
|
|
return;
|
|
|
|
if ((pl != &IgnoreList) && (pl != &FriendList))
|
|
{
|
|
nlwarning("<askAddContact> For now, only support friend list & ignore list");
|
|
return;
|
|
}
|
|
|
|
// check that name isn't already in people list
|
|
if (pl->getIndexFromName(contactName) != -1)
|
|
{
|
|
// people already in list, can't add twice
|
|
CInterfaceManager::getInstance()->displaySystemInfo(CI18N::get("uiContactAlreadyInList"));
|
|
return;
|
|
}
|
|
|
|
// add into server (NB: will be added by the server response later)
|
|
const std::string sMsg = "TEAM:CONTACT_ADD";
|
|
CBitMemStream out;
|
|
if(GenericMsgHeaderMngr.pushNameToStream(sMsg, out))
|
|
{
|
|
uint8 list = 0;
|
|
|
|
if (pl == &IgnoreList)
|
|
list = 1;
|
|
|
|
if (pl == &FriendList)
|
|
list = 0;
|
|
|
|
ucstring temp = contactName;
|
|
out.serial(temp);
|
|
out.serial(list);
|
|
NetMngr.push(out);
|
|
//nlinfo("impulseCallBack : %s %s %d sent", sMsg.c_str(), contactName.toString().c_str(), list);
|
|
}
|
|
else
|
|
nlwarning("impulseCallBack : unknown message name : '%s'.", sMsg.c_str());
|
|
|
|
// NB: no client prediction, will be added by server later
|
|
|
|
// Fake Local simulation
|
|
if (ClientCfg.Local)
|
|
{
|
|
sint index = pl->addPeople(contactName);
|
|
pl->setOnline(index, ccs_online);
|
|
updateAllFreeTellerHeaders();
|
|
}
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::askMoveContact(uint peopleIndexInSrc, CPeopleList *plSRC, CPeopleList *plDST)
|
|
{
|
|
if ((plSRC == NULL) || (plDST == NULL)) return;
|
|
if ((plSRC != &IgnoreList) && (plSRC != &FriendList))
|
|
{
|
|
nlwarning("<askMoveContact> For now, only support friend list & ignore list");
|
|
return;
|
|
}
|
|
if ((plDST != &IgnoreList) && (plDST != &FriendList))
|
|
{
|
|
nlwarning("<askMoveContact> For now, only support friend list & ignore list");
|
|
return;
|
|
}
|
|
if ( plDST == plSRC )
|
|
{
|
|
// move from list to same => no op
|
|
return;
|
|
}
|
|
|
|
// check that index is already in people list
|
|
if (peopleIndexInSrc >= plSRC->getNumPeople()) return;
|
|
|
|
|
|
// Send message to server
|
|
uint32 contactId= plSRC->getContactId(peopleIndexInSrc);
|
|
uint8 nListSRC;
|
|
|
|
if (plSRC == &FriendList)
|
|
nListSRC = 0;
|
|
|
|
if (plSRC == &IgnoreList)
|
|
nListSRC = 1;
|
|
|
|
const std::string sMsg = "TEAM:CONTACT_MOVE";
|
|
CBitMemStream out;
|
|
if(GenericMsgHeaderMngr.pushNameToStream(sMsg, out))
|
|
{
|
|
out.serial(contactId);
|
|
out.serial(nListSRC);
|
|
NetMngr.push(out);
|
|
//nlinfo("impulseCallBack : %s %d %d sent", sMsg.c_str(), contactId, nListSRC);
|
|
}
|
|
else
|
|
nlwarning("impulseCallBack : unknown message name : '%s'.", sMsg.c_str());
|
|
|
|
// NB: no client prediction, will be added by server later
|
|
|
|
// Fake Local simulation
|
|
if (ClientCfg.Local)
|
|
{
|
|
ucstring peopleName= plSRC->getName(peopleIndexInSrc);
|
|
plSRC->removePeople(peopleIndexInSrc);
|
|
sint dstIndex = plDST->addPeople(peopleName);
|
|
plDST->setOnline(dstIndex, ccs_online);
|
|
if (getChatGroupWindow())
|
|
{
|
|
getChatGroupWindow()->updateAllFreeTellerHeaders();
|
|
}
|
|
updateAllFreeTellerHeaders();
|
|
}
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::askRemoveContact(uint peopleIndex, CPeopleList *pl)
|
|
{
|
|
if (pl == NULL) return;
|
|
if ((pl != &IgnoreList) && (pl != &FriendList))
|
|
{
|
|
nlwarning("<askRemoveContact> For now, only support friend pl & ignore pl");
|
|
return;
|
|
}
|
|
if (peopleIndex >= pl->getNumPeople())
|
|
{
|
|
nlwarning("<askRemoveContact> bad index given");
|
|
return;
|
|
}
|
|
|
|
// send server message
|
|
const std::string sMsg = "TEAM:CONTACT_DEL";
|
|
CBitMemStream out;
|
|
if(GenericMsgHeaderMngr.pushNameToStream(sMsg, out))
|
|
{
|
|
uint32 contactId = pl->getContactId(peopleIndex);
|
|
out.serial(contactId);
|
|
uint8 nList = 0;
|
|
if (pl == &PeopleInterraction.FriendList)
|
|
nList = 0;
|
|
if (pl == &PeopleInterraction.IgnoreList)
|
|
nList = 1;
|
|
out.serial(nList);
|
|
NetMngr.push(out);
|
|
//nlinfo("impulseCallBack : %s %d %d sent", sMsg.c_str(), contactId, nList);
|
|
}
|
|
else
|
|
nlwarning("impulseCallBack : unknown message name : '%s'.", sMsg.c_str());
|
|
|
|
// NB: no client prediction, let the server delete the contact by message.
|
|
|
|
// Fake Local simulation
|
|
if (ClientCfg.Local)
|
|
{
|
|
pl->removePeople(peopleIndex);
|
|
updateAllFreeTellerHeaders();
|
|
}
|
|
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::initContactLists( const std::vector<uint32> &vFriendListName,
|
|
const std::vector<TCharConnectionState> &vFriendListOnline,
|
|
const std::vector<ucstring> &vIgnoreListName )
|
|
|
|
{
|
|
// clear the current lists if any
|
|
FriendList.removeAllPeoples();
|
|
IgnoreList.removeAllPeoples();
|
|
|
|
// build the contact ids, like server did
|
|
uint32 contactIdPool= 0;
|
|
for (uint i = 0; i < vFriendListName.size(); ++i)
|
|
addContactInList(contactIdPool++, vFriendListName[i], vFriendListOnline[i], 0);
|
|
for (uint i = 0; i < vIgnoreListName.size(); ++i)
|
|
addContactInList(contactIdPool++, vIgnoreListName[i], ccs_offline, 1);
|
|
updateAllFreeTellerHeaders();
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::addContactInList(uint32 contactId, const ucstring &nameIn, TCharConnectionState online, uint8 nList)
|
|
{
|
|
// select correct people list
|
|
CPeopleList &pl= nList==0?FriendList:IgnoreList;
|
|
|
|
// remove the shard name if possible
|
|
ucstring name= CEntityCL::removeShardFromName(nameIn);
|
|
|
|
// add the contact to this list
|
|
sint index = pl.getIndexFromName(name);
|
|
// try to create if not found
|
|
if (index == -1)
|
|
index = pl.addPeople(name);
|
|
|
|
if (index != -1)
|
|
{
|
|
pl.setOnline(index, online);
|
|
pl.setContactId(index, contactId);
|
|
}
|
|
|
|
CInterfaceManager* pIM= CInterfaceManager::getInstance();
|
|
CPeopleList::TSortOrder order = (CPeopleList::TSortOrder)(NLGUI::CDBManager::getInstance()->getDbProp("UI:SAVE:CONTACT_LIST:SORT_ORDER")->getValue32());
|
|
FriendList.sortEx(order);
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::addContactInList(uint32 contactId, uint32 nameID, TCharConnectionState online, uint8 nList)
|
|
{
|
|
ucstring name;
|
|
STRING_MANAGER::CStringManagerClient *pSMC = STRING_MANAGER::CStringManagerClient::instance();
|
|
if (pSMC->getString(nameID, name))
|
|
{
|
|
addContactInList(contactId, name, online, nList);
|
|
// update free teller header
|
|
updateAllFreeTellerHeaders();
|
|
}
|
|
else
|
|
{
|
|
SWaitingContact w;
|
|
w.ContactId= contactId;
|
|
w.NameId = nameID;
|
|
w.List = nList; // Friend list == 0 // Ignore list == 1
|
|
w.Online = online;
|
|
WaitingContacts.push_back(w);
|
|
|
|
CInterfaceManager* pIM= CInterfaceManager::getInstance();
|
|
CPeopleList::TSortOrder order = (CPeopleList::TSortOrder)(NLGUI::CDBManager::getInstance()->getDbProp("UI:SAVE:CONTACT_LIST:SORT_ORDER")->getValue32());
|
|
FriendList.sortEx(order);
|
|
}
|
|
}
|
|
|
|
//=================================================================================================================
|
|
bool CPeopleInterraction::isContactInList(const ucstring &nameIn, uint8 nList) const
|
|
{
|
|
// select correct people list
|
|
const CPeopleList &pl= nList==0?FriendList:IgnoreList;
|
|
// remove the shard name if possible
|
|
ucstring name= CEntityCL::removeShardFromName(nameIn);
|
|
return pl.getIndexFromName(name) != -1;
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::updateAllFreeTellerHeaders()
|
|
{
|
|
CChatGroupWindow *gcw = getChatGroupWindow();
|
|
if (gcw)
|
|
{
|
|
gcw->updateAllFreeTellerHeaders();
|
|
}
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::removeAllFreeTellers()
|
|
{
|
|
CChatGroupWindow *gcw = getChatGroupWindow();
|
|
if (gcw)
|
|
{
|
|
gcw->removeAllFreeTellers();
|
|
}
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::updateWaitingContacts()
|
|
{
|
|
bool touched = false;
|
|
for (uint32 i = 0; i < WaitingContacts.size();)
|
|
{
|
|
SWaitingContact &w = WaitingContacts[i];
|
|
ucstring name;
|
|
STRING_MANAGER::CStringManagerClient *pSMC = STRING_MANAGER::CStringManagerClient::instance();
|
|
if (pSMC->getString(w.NameId, name))
|
|
{
|
|
addContactInList(w.ContactId, name, w.Online, w.List);
|
|
WaitingContacts.erase(WaitingContacts.begin()+i);
|
|
touched = true;
|
|
}
|
|
else
|
|
{
|
|
++i;
|
|
}
|
|
}
|
|
if (touched)
|
|
{
|
|
updateAllFreeTellerHeaders();
|
|
}
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::updateContactInList(uint32 contactId, TCharConnectionState online, uint nList)
|
|
{
|
|
if (nList == 0)
|
|
{
|
|
sint index = FriendList.getIndexFromContactId(contactId);
|
|
if (index != -1)
|
|
{
|
|
// Only do work if online status has changed
|
|
if (FriendList.getOnline(index) != online)
|
|
{
|
|
CCDBNodeLeaf* node = NLGUI::CDBManager::getInstance()->getDbProp("UI:SAVE:CHAT:SHOW_ONLINE_OFFLINE_NOTIFICATIONS_CB", false);
|
|
if (node && node->getValueBool())
|
|
{
|
|
// Only show the message if this player is not in my guild (because then the guild manager will show a message)
|
|
std::vector<SGuildMember> GuildMembers = CGuildManager::getInstance()->getGuildMembers();
|
|
bool bOnlyFriend = true;
|
|
ucstring name = toLower(FriendList.getName(index));
|
|
for (uint i = 0; i < GuildMembers.size(); ++i)
|
|
{
|
|
if (toLower(GuildMembers[i].Name) == name)
|
|
{
|
|
bOnlyFriend = false;
|
|
break;
|
|
}
|
|
}
|
|
|
|
TCharConnectionState prevState = FriendList.getOnline(index);
|
|
bool showMsg = bOnlyFriend && (prevState == ccs_offline || online == ccs_offline);
|
|
|
|
// Player is not in my guild, and the status change is from offline to online/abroad online or vice versa.
|
|
if (showMsg)
|
|
{
|
|
ucstring msg = (online != ccs_offline) ? CI18N::get("uiPlayerOnline") : CI18N::get("uiPlayerOffline");
|
|
strFindReplace(msg, "%s", FriendList.getName(index));
|
|
string cat = getStringCategory(msg, msg);
|
|
map<string, CClientConfig::SSysInfoParam>::const_iterator it;
|
|
NLMISC::CRGBA col = CRGBA::Yellow;
|
|
it = ClientCfg.SystemInfoParams.find(toLower(cat));
|
|
if (it != ClientCfg.SystemInfoParams.end())
|
|
{
|
|
col = it->second.Color;
|
|
}
|
|
bool dummy;
|
|
PeopleInterraction.ChatInput.AroundMe.displayMessage(msg, col, 2, &dummy);
|
|
}
|
|
}
|
|
|
|
FriendList.setOnline(index, online);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
sint index = IgnoreList.getIndexFromContactId(contactId);
|
|
if (index != -1)
|
|
IgnoreList.setOnline(index, online);
|
|
}
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::removeContactFromList(uint32 contactId, uint8 nList)
|
|
{
|
|
if (nList == 0)
|
|
{
|
|
sint index = FriendList.getIndexFromContactId(contactId);
|
|
if (index != -1)
|
|
FriendList.removePeople(index);
|
|
}
|
|
else
|
|
{
|
|
sint index = IgnoreList.getIndexFromContactId(contactId);
|
|
if (index != -1)
|
|
IgnoreList.removePeople(index);
|
|
}
|
|
updateAllFreeTellerHeaders();
|
|
}
|
|
|
|
//=================================================================================================================
|
|
bool CPeopleInterraction::testValidPartyChatName(const ucstring &title)
|
|
{
|
|
if (title.empty()) return false;
|
|
// shouldn't begin like 'user chat 1-5'
|
|
ucstring userChatStr = CI18N::get("uiUserChat");
|
|
if (title.substr(0, userChatStr.length()) == userChatStr) return false;
|
|
for(uint k = 0; k < PartyChats.size(); ++k) // there shouldn't be that much party chat simultaneously so a linear search is ok
|
|
{
|
|
if (PartyChats[k].Window->getTitle() == title) return false;
|
|
}
|
|
// check for other chat window names (local only ?)
|
|
if (SystemInfo && title == SystemInfo->getTitle()) return false;
|
|
if (AroundMe.Window && title == AroundMe.Window->getTitle()) return false;
|
|
if (GuildChat && title == GuildChat->getTitle()) return false;
|
|
if (TeamChat && title == TeamChat->getTitle()) return false;
|
|
sint index;
|
|
index = FriendList.getIndexFromName(title);
|
|
if (index != -1) return false;
|
|
index = IgnoreList.getIndexFromName(title);
|
|
if (index != -1) return false;
|
|
// TODO_GAMEDEV server test for the name (not only local), & modify callers of this function
|
|
// The party chat should NOT have the name of a player
|
|
// A player name is NOT valid if it is the same that a party chat name
|
|
return true;
|
|
}
|
|
|
|
//=================================================================================================================
|
|
bool CPeopleInterraction::removePartyChat(CChatWindow *window)
|
|
{
|
|
if (!window) return false;
|
|
std::vector<CPartyChatInfo>::iterator it;
|
|
for(it = PartyChats.begin(); it != PartyChats.end(); ++it)
|
|
{
|
|
if (it->Window == window) break;
|
|
}
|
|
if (it != PeopleInterraction.PartyChats.end())
|
|
{
|
|
PeopleInterraction.PartyChats.erase(it);
|
|
getChatWndMgr().removeChatWindow(window->getTitle());
|
|
// TODO GAMEDEV : send msg to server to tell that the player has deleted this party chat.
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::assignPartyChatMenu(CChatWindow *partyChat)
|
|
{
|
|
if (!partyChat) return;
|
|
// TODO GAMEDEV : fill the 2 following boolean
|
|
bool isTeamLeader = true;
|
|
bool isGuildLeader = true;
|
|
|
|
if (isTeamLeader && isGuildLeader)
|
|
{
|
|
partyChat->setMenu("ui:interface:team_and_guild_chief_party_chat_menu");
|
|
}
|
|
else if (isTeamLeader)
|
|
{
|
|
partyChat->setMenu("ui:interface:team_chief_party_chat_menu");
|
|
}
|
|
else if (isGuildLeader)
|
|
{
|
|
partyChat->setMenu("ui:interface:guild_chief_party_chat_menu");
|
|
}
|
|
}
|
|
|
|
//=================================================================================================================
|
|
bool CPeopleInterraction::createNewPartyChat(const ucstring &title)
|
|
{
|
|
// now there are no party chat windows, party chat phrases must be filtered from the main chat
|
|
|
|
// create a new party chat and set the focus on it
|
|
CChatWindowDesc chatDesc;
|
|
//chatDesc.FatherContainer = "ui:interface:communication";
|
|
chatDesc.FatherContainer = "ui:interface:contact_list";
|
|
chatDesc.Title = title;
|
|
chatDesc.Title = title;
|
|
chatDesc.Listener = &PartyChatEntryHandler;
|
|
chatDesc.Localize = false;
|
|
|
|
// CChatWindow *newPartyChat = getChatWndMgr().createChatWindow(chatDesc);
|
|
CChatWindow *newPartyChat = NULL;
|
|
|
|
//if (newPartyChat)
|
|
{
|
|
// popup the container
|
|
/*
|
|
newPartyChat->getContainer()->setup();
|
|
newPartyChat->getContainer()->setOpen(true);
|
|
newPartyChat->getContainer()->popupCurrentPos();
|
|
newPartyChat->getContainer()->updateCoords();
|
|
newPartyChat->getContainer()->center();
|
|
newPartyChat->getContainer()->setX(newPartyChat->getContainer()->getX() + (sint32) (rand() % PARTY_CHAT_SPAWN_DELTA));
|
|
newPartyChat->getContainer()->setY(newPartyChat->getContainer()->getY() + (sint32) (rand() % PARTY_CHAT_SPAWN_DELTA));
|
|
newPartyChat->getContainer()->enableBlink(2);
|
|
*/
|
|
|
|
CPartyChatInfo pci;
|
|
pci.Window = newPartyChat;
|
|
pci.ID = PeopleInterraction.CurrPartyChatID ++;
|
|
pci.Filter = new CChatInputFilter;
|
|
pci.Filter->addListeningWindow(pci.Window);
|
|
//CPeopleInterraction::assignPartyChatMenu(newPartyChat);
|
|
PartyChats.push_back(pci);
|
|
//newPartyChat->setKeyboardFocus();
|
|
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::buildFilteredChatSummary(const CFilteredChat &src, CFilteredChatSummary &fcs)
|
|
{
|
|
// fill src infos
|
|
fcs.SrcGuild = ChatInput.Guild.isListeningWindow(src.Window);
|
|
fcs.SrcAroundMe = ChatInput.AroundMe.isListeningWindow(src.Window);
|
|
fcs.SrcSystemInfo = ChatInput.SystemInfo.isListeningWindow(src.Window);
|
|
fcs.SrcTeam = ChatInput.Team.isListeningWindow(src.Window);
|
|
fcs.SrcTell = ChatInput.Tell.isListeningWindow(src.Window);
|
|
fcs.SrcRegion = ChatInput.Region.isListeningWindow(src.Window);
|
|
fcs.SrcUniverse = ChatInput.Universe.isListeningWindow(src.Window);
|
|
|
|
// fill target infos
|
|
if (src.Filter.getTargetPartyChat() != NULL || !src.Filter.getTargetPlayer().empty())
|
|
{
|
|
fcs.Target = CChatGroup::say;
|
|
}
|
|
else
|
|
{
|
|
fcs.Target = src.Filter.getTargetGroup();
|
|
}
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::buildFilteredDynChatSummary(const CFilteredChat &src, CFilteredDynChatSummary &fcs)
|
|
{
|
|
for (uint8 i = 0; i < CChatGroup::MaxDynChanPerPlayer; i++)
|
|
{
|
|
fcs.SrcDynChat[i] = ChatInput.DynamicChat[i].isListeningWindow(src.Window);
|
|
}
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::saveFilteredChat(NLMISC::IStream &f, const CFilteredChat &src)
|
|
{
|
|
bool present;
|
|
if (src.Window == NULL)
|
|
{
|
|
present = false;
|
|
f.serial(present);
|
|
}
|
|
else
|
|
{
|
|
present = true;
|
|
f.serial(present);
|
|
CFilteredChatSummary fcs;
|
|
buildFilteredChatSummary(src, fcs);
|
|
f.serial(fcs);
|
|
}
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::saveFilteredDynChat(NLMISC::IStream &f, const CFilteredChat &src)
|
|
{
|
|
bool present;
|
|
if (src.Window == NULL)
|
|
{
|
|
present = false;
|
|
f.serial(present);
|
|
}
|
|
else
|
|
{
|
|
present = true;
|
|
f.serial(present);
|
|
CFilteredDynChatSummary fcs;
|
|
buildFilteredDynChatSummary(src, fcs);
|
|
f.serial(fcs);
|
|
}
|
|
}
|
|
|
|
//=================================================================================================================
|
|
CChatGroupWindow *CPeopleInterraction::getChatGroupWindow() const
|
|
{
|
|
return dynamic_cast<CChatGroupWindow*>(ChatGroup.Window);
|
|
}
|
|
|
|
#define USER_CHATS_INFO_VERSION 2
|
|
#define USER_DYN_CHATS_INFO_VERSION 1
|
|
|
|
//=================================================================================================================
|
|
bool CPeopleInterraction::saveUserChatsInfos(NLMISC::IStream &f)
|
|
{
|
|
nlassert(!f.isReading());
|
|
try
|
|
{
|
|
sint ver= f.serialVersion(USER_CHATS_INFO_VERSION);
|
|
f.serialCheck(NELID("TAHC"));
|
|
//saveFilteredChat(f, MainChat);
|
|
saveFilteredChat(f, ChatGroup);
|
|
for(uint k = 0; k < MaxNumUserChats; ++k)
|
|
{
|
|
saveFilteredChat(f, UserChat[k]);
|
|
}
|
|
f.serialCheck(NELID("TAHC"));
|
|
if (ver>=1)
|
|
{
|
|
CChatGroupWindow *pCGW = PeopleInterraction.getChatGroupWindow();
|
|
sint32 index = pCGW ? pCGW->getTabIndex() : 0;
|
|
f.serial(index);
|
|
saveFilteredChat(f, TheUserChat);
|
|
}
|
|
// Save the free tellers only if they belongs to friend list to avoid the 'only growing' situation
|
|
if (ver>=2)
|
|
{
|
|
CChatGroupWindow *pCGW = PeopleInterraction.getChatGroupWindow();
|
|
pCGW->saveFreeTeller(f);
|
|
}
|
|
}
|
|
catch(const NLMISC::EStream &e)
|
|
{
|
|
nlwarning("Error while saving user chat infos : %s", e.what());
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
//=================================================================================================================
|
|
bool CPeopleInterraction::saveUserDynChatsInfos(NLMISC::IStream &f)
|
|
{
|
|
nlassert(!f.isReading());
|
|
try
|
|
{
|
|
sint ver = f.serialVersion(USER_DYN_CHATS_INFO_VERSION);
|
|
f.serialCheck(NELID("OMGY"));
|
|
if (ver >= 1)
|
|
{
|
|
saveFilteredDynChat(f, TheUserChat);
|
|
}
|
|
}
|
|
catch(const NLMISC::EStream &e)
|
|
{
|
|
nlwarning("Error while saving user dyn chat infos : %s", e.what());
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
//=================================================================================================================
|
|
bool CPeopleInterraction::loadUserChatsInfos(NLMISC::IStream &f)
|
|
{
|
|
removeAllUserChats();
|
|
nlassert(f.isReading());
|
|
try
|
|
{
|
|
bool present;
|
|
sint ver = f.serialVersion(USER_CHATS_INFO_VERSION);
|
|
f.serialCheck(NELID("TAHC"));
|
|
f.serial(present);
|
|
if (!present)
|
|
{
|
|
nlwarning("Bad data in user chats infos");
|
|
return false;
|
|
}
|
|
CFilteredChatSummary fcs;
|
|
f.serial(fcs);
|
|
//setupUserChatFromSummary(fcs, MainChat);
|
|
setupUserChatFromSummary(fcs, ChatGroup);
|
|
|
|
for(uint k = 0; k < MaxNumUserChats; ++k)
|
|
{
|
|
f.serial(present);
|
|
if (present)
|
|
{
|
|
createUserChat(k);
|
|
f.serial(fcs);
|
|
setupUserChatFromSummary(fcs, UserChat[k]);
|
|
}
|
|
}
|
|
f.serialCheck(NELID("TAHC"));
|
|
if (ver>=1)
|
|
{
|
|
// CChatGroupWindow *pCGW = PeopleInterraction.getChatGroupWindow();
|
|
sint32 index;
|
|
f.serial(index);
|
|
/* Yoyo: decide to always start with the default channel (user) activated
|
|
because complex (at this time, the buttons are not all active, must wait guild loading, UI:SAVE loading etc...)
|
|
Hence this doesn't work for anything but User and Sysinfo (if it is activated....)
|
|
NB: must still load the index for file format reason
|
|
//if (pCGW) pCGW->setTabIndex(index);
|
|
*/
|
|
f.serial(present);
|
|
if (present)
|
|
{
|
|
f.serial(fcs);
|
|
setupUserChatFromSummary(fcs, TheUserChat);
|
|
}
|
|
}
|
|
// Load the free tellers
|
|
if (ver>=2)
|
|
{
|
|
CChatGroupWindow *pCGW = PeopleInterraction.getChatGroupWindow();
|
|
if (pCGW) pCGW->loadFreeTeller(f);
|
|
}
|
|
}
|
|
catch(const NLMISC::EStream &e)
|
|
{
|
|
nlwarning("Error while loading user chat infos : %s", e.what());
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
//=================================================================================================================
|
|
bool CPeopleInterraction::loadUserDynChatsInfos(NLMISC::IStream &f)
|
|
{
|
|
nlassert(f.isReading());
|
|
try
|
|
{
|
|
bool present;
|
|
sint ver = f.serialVersion(USER_DYN_CHATS_INFO_VERSION);
|
|
f.serialCheck(NELID("OMGY"));
|
|
f.serial(present);
|
|
if (!present)
|
|
{
|
|
nlwarning("Bad data in user dyn chats infos");
|
|
return false;
|
|
}
|
|
CFilteredDynChatSummary fcs;
|
|
if (ver >= 1)
|
|
{
|
|
f.serial(fcs);
|
|
setupUserDynChatFromSummary(fcs, TheUserChat);
|
|
}
|
|
}
|
|
catch(const NLMISC::EStream &e)
|
|
{
|
|
nlwarning("Error while loading user dyn chat infos : %s", e.what());
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::setupUserChatFromSummary(const CFilteredChatSummary &summary, CFilteredChat &dest)
|
|
{
|
|
// User Dest. Do not allow Universe Warning, because do not want a warning open at load (moreover, the UNIVERSE tab should not be activated)
|
|
dest.Filter.setTargetGroup(summary.Target, 0, false);
|
|
// src
|
|
ChatInput.AroundMe.setWindowState(dest.Window, summary.SrcAroundMe);
|
|
ChatInput.Guild.setWindowState(dest.Window, summary.SrcGuild);
|
|
ChatInput.SystemInfo.setWindowState(dest.Window, summary.SrcSystemInfo);
|
|
ChatInput.Team.setWindowState(dest.Window, summary.SrcTeam);
|
|
ChatInput.Tell.setWindowState(dest.Window, summary.SrcTell);
|
|
ChatInput.Region.setWindowState(dest.Window, summary.SrcRegion);
|
|
ChatInput.Universe.setWindowState(dest.Window, summary.SrcUniverse);
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::setupUserDynChatFromSummary(const CFilteredDynChatSummary &summary, CFilteredChat &dest)
|
|
{
|
|
// src
|
|
for (uint8 i = 0; i < CChatGroup::MaxDynChanPerPlayer; i++)
|
|
{
|
|
ChatInput.DynamicChat[i].setWindowState(dest.Window, summary.SrcDynChat[i]);
|
|
}
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::removeAllUserChats()
|
|
{
|
|
for(uint k = 0; k < MaxNumUserChats; ++k)
|
|
{
|
|
if (UserChat[k].Window)
|
|
{
|
|
getChatWndMgr().removeChatWindow(UserChat[k].Window);
|
|
UserChat[k].Filter.reset();
|
|
UserChat[k].Window = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::createUserChat(uint index)
|
|
{
|
|
if (index >= MaxNumUserChats)
|
|
{
|
|
nlwarning("Bad index");
|
|
return;
|
|
}
|
|
CChatWindowDesc chatDesc;
|
|
ucstring userChatStr = CI18N::get("uiUserChat");
|
|
userChatStr += ucchar(' ') + ucstring(toString(index + 1));
|
|
//chatDesc.FatherContainer = "ui:interface:communication";
|
|
chatDesc.FatherContainer = "ui:interface:contact_list";
|
|
chatDesc.Title = userChatStr;
|
|
chatDesc.Listener = NULL;
|
|
chatDesc.Localize = false;
|
|
chatDesc.Savable = true;
|
|
chatDesc.ChatTemplate = "filtered_chat_id";
|
|
UserChat[index].Window = getChatWndMgr().createChatWindow(chatDesc);
|
|
if (!UserChat[index].Window) return;
|
|
UserChat[index].Window->getContainer()->setup();
|
|
// Configure filter for the new chat (by default, listen to everything but party chats)
|
|
UserChat[index].Filter.setTargetGroup(CChatGroup::say);
|
|
// assoviate filter with chat window
|
|
UserChat[index].Filter.setChat(UserChat[index].Window);
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::refreshActiveUserChats()
|
|
{
|
|
for(uint k = 0; k < MaxNumUserChats; ++k)
|
|
{
|
|
if (UserChat[k].Window)
|
|
{
|
|
UserChat[k].Window->getContainer()->setActive(true);
|
|
}
|
|
}
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::talkInDynamicChannel(uint32 channelNb,ucstring sentence)
|
|
{
|
|
if(channelNb<CChatGroup::MaxDynChanPerPlayer)
|
|
{
|
|
DynamicChatEntryHandler[channelNb].msgEntered(sentence,DynamicChat[channelNb]);
|
|
}
|
|
}
|
|
|
|
//=================================================================================================================
|
|
void CPeopleInterraction::displayTellInMainChat(const ucstring &playerName)
|
|
{
|
|
//CChatWindow *chat = PeopleInterraction.MainChat.Window;
|
|
CChatWindow *chat = PeopleInterraction.ChatGroup.Window;
|
|
if (!chat) return;
|
|
chat->getContainer()->setActive (true);
|
|
// make the container blink
|
|
chat->getContainer()->enableBlink(2);
|
|
// TODO : center the view on the newly created container ?
|
|
// display a new command '/name' in the chat. The player must enter a new unique name for the party chat.
|
|
chat->setCommand("tell " + playerName + " ", false);
|
|
chat->setKeyboardFocus();
|
|
}
|
|
|
|
/////////////////////////////////////
|
|
// ACTION HANDLERS FOR PEOPLE LIST //
|
|
/////////////////////////////////////
|
|
|
|
//=================================================================================================================
|
|
// Target a member of the team
|
|
// See also CAHTargetTeammateShortcut
|
|
class CHandlerTeamTarget : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &sParams)
|
|
{
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
|
|
// retrieve the index of the people
|
|
uint peopleIndex= 0;
|
|
bool ok= false;
|
|
// If comes from the button, get direct index
|
|
if( !sParams.empty() )
|
|
{
|
|
fromString(sParams, peopleIndex);
|
|
ok= true;
|
|
}
|
|
// else comes from a menu.
|
|
else
|
|
{
|
|
CPeopleList *list;
|
|
if (PeopleInterraction.getPeopleFromCurrentMenu(list, peopleIndex))
|
|
{
|
|
if (list == &PeopleInterraction.TeamList) // check for good list
|
|
ok= true;
|
|
}
|
|
}
|
|
|
|
// If success to get the team index
|
|
if(ok)
|
|
{
|
|
// Get the team name id.
|
|
CLFECOMMON::TClientDataSetIndex entityId= CLFECOMMON::INVALID_CLIENT_DATASET_INDEX;
|
|
CCDBNodeLeaf *prop = NLGUI::CDBManager::getInstance()->getDbProp(toString(TEAM_DB_PATH ":%d:UID", peopleIndex), false);
|
|
if (prop)
|
|
entityId= prop->getValue32();
|
|
|
|
if(entityId != CLFECOMMON::INVALID_CLIENT_DATASET_INDEX)
|
|
{
|
|
// get the entity by its received name
|
|
CEntityCL *entity= EntitiesMngr.getEntityByCompressedIndex(entityId);
|
|
if(entity)
|
|
// Select this entity.
|
|
UserEntity->selection(entity->slot());
|
|
else
|
|
{
|
|
// the entity is not in vision, can't select it
|
|
pIM->displaySystemInfo(CI18N::get("uiTeamSelectNotInVision"), "CHK");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerTeamTarget, "team_target" );
|
|
|
|
//=================================================================================================================
|
|
// Dismiss a member from the team
|
|
class CHandlerDismissMember : public IActionHandler
|
|
{
|
|
public:
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
// retrieve the index of the people
|
|
CPeopleList *list;
|
|
uint peopleIndex;
|
|
if (PeopleInterraction.getPeopleFromCurrentMenu(list, peopleIndex))
|
|
{
|
|
if (list == &PeopleInterraction.TeamList) // check for good list
|
|
{
|
|
const string msgName = "TEAM:KICK";
|
|
CBitMemStream out;
|
|
if(GenericMsgHeaderMngr.pushNameToStream(msgName, out))
|
|
{
|
|
uint8 teamMember = (uint8) peopleIndex;
|
|
out.serial(teamMember);
|
|
NetMngr.push(out);
|
|
//nlinfo("impulseCallBack : %s %d sent", msgName.c_str(), teamMember);
|
|
}
|
|
else
|
|
nlwarning("command 'dismiss_member': unknown message named '%s'.", msgName.c_str());
|
|
}
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerDismissMember, "dismiss_member");
|
|
|
|
//=================================================================================================================
|
|
// Set the leader of the team
|
|
class CHandlerSetTeamLeader : public IActionHandler
|
|
{
|
|
public:
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
// retrieve the index of the people
|
|
CPeopleList *list;
|
|
uint peopleIndex;
|
|
if (PeopleInterraction.getPeopleFromCurrentMenu(list, peopleIndex))
|
|
{
|
|
if (list == &PeopleInterraction.TeamList) // check for good list
|
|
{
|
|
/*
|
|
const string msgName = "TEAM:SET_LEADER";
|
|
CBitMemStream out;
|
|
if(GenericMsgHeaderMngr.pushNameToStream(msgName, out))
|
|
{
|
|
uint8 teamMember = (uint8)(peopleIndex);
|
|
out.serial(teamMember);
|
|
NetMngr.push(out);
|
|
//nlinfo("impulseCallBack : %s %d sent", msgName.c_str(), teamMember);
|
|
}
|
|
else
|
|
nlwarning("command 'set_leader': unknown message named '%s'.", msgName.c_str());
|
|
*/
|
|
NLMISC::ICommand::execute("a setTeamLeader " + toString(peopleIndex), g_log);
|
|
}
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerSetTeamLeader, "set_team_leader");
|
|
|
|
//=================================================================================================================
|
|
// Set a successor for the team
|
|
class CHandlerSetSuccessor : public IActionHandler
|
|
{
|
|
public:
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
// retrieve the index of the people
|
|
CPeopleList *list;
|
|
uint peopleIndex;
|
|
if (PeopleInterraction.getPeopleFromCurrentMenu(list, peopleIndex))
|
|
{
|
|
if (list == &PeopleInterraction.TeamList) // check for good list
|
|
{
|
|
if (ClientCfg.Local)
|
|
{
|
|
NLGUI::CDBManager::getInstance()->getDbProp(TEAM_DB_PATH ":SUCCESSOR_INDEX")->setValue32(peopleIndex);
|
|
}
|
|
else
|
|
{
|
|
const string msgName = "TEAM:SET_SUCCESSOR";
|
|
CBitMemStream out;
|
|
if(GenericMsgHeaderMngr.pushNameToStream(msgName, out))
|
|
{
|
|
uint8 teamMember = (uint8) peopleIndex;
|
|
out.serial(teamMember);
|
|
NetMngr.push(out);
|
|
//nlinfo("impulseCallBack : %s %d sent", msgName.c_str(), teamMember);
|
|
}
|
|
else
|
|
nlwarning("command 'set_successor': unknown message named '%s'.", msgName.c_str());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerSetSuccessor, "set_successor");
|
|
|
|
|
|
//=================================================================================================================
|
|
// player or leader quit the team
|
|
class CHandlerQuitTeam : public IActionHandler
|
|
{
|
|
public:
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
// Create the message for the server to execute a phrase.
|
|
const string msgName = "TEAM:LEAVE";
|
|
CBitMemStream out;
|
|
if(GenericMsgHeaderMngr.pushNameToStream(msgName, out))
|
|
{
|
|
NetMngr.push(out);
|
|
//nlinfo("impulseCallBack : %s sent", msgName.c_str());
|
|
}
|
|
else
|
|
nlwarning("CHandlerContextQuitTeam::execute: unknown message name : '%s'.", msgName.c_str());
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerQuitTeam, "quit_team");
|
|
|
|
//=================================================================================================================
|
|
// The leader enable / disbale seeds sharing
|
|
class CHandlerShareSeeds : public IActionHandler
|
|
{
|
|
public:
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
// TODO_GAMEDEV : enable disable seeds sharing
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerShareSeeds, "share_seeds");
|
|
|
|
//////////////////
|
|
// CONTACT LIST //
|
|
//////////////////
|
|
|
|
//=================================================================================================================
|
|
// Remove a contact from a list
|
|
class CHandlerRemoveContact : public IActionHandler
|
|
{
|
|
public:
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
// retrieve the index of the people
|
|
CPeopleList *list;
|
|
uint peopleIndex;
|
|
if (PeopleInterraction.getPeopleFromCurrentMenu(list, peopleIndex))
|
|
{
|
|
PeopleInterraction.askRemoveContact(peopleIndex, list);
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerRemoveContact, "remove_contact");
|
|
|
|
//=================================================================================================================
|
|
// Invoke the 'tell' command on a contact from its menu
|
|
// The tell command is displayed in the 'around me' window
|
|
class CHandlerMenuTellContact : public IActionHandler
|
|
{
|
|
public:
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
// retrieve the index of the people
|
|
CPeopleList *list;
|
|
uint peopleIndex;
|
|
if (PeopleInterraction.getPeopleFromCurrentMenu(list, peopleIndex))
|
|
{
|
|
CPeopleInterraction::displayTellInMainChat(list->getName(peopleIndex));
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerMenuTellContact, "menu_tell_contact");
|
|
|
|
|
|
//=================================================================================================================
|
|
// Invoke the 'tell' command on a contact from a left click
|
|
class CHandlerTellContact : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase *pCaller, const std::string &/* sParams */)
|
|
{
|
|
if (!pCaller) return;
|
|
CInterfaceGroup *ig = pCaller->getParent();
|
|
if (!ig) return;
|
|
CGroupContainer *gc = static_cast< CGroupContainer* >( ig->getEnclosingContainer() );
|
|
if (!gc) return;
|
|
CPeopleList *list;
|
|
uint peopleIndex;
|
|
if (PeopleInterraction.getPeopleFromContainerID(gc->getId(), list, peopleIndex))
|
|
{
|
|
CPeopleInterraction::displayTellInMainChat(list->getName(peopleIndex));
|
|
}
|
|
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerTellContact, "tell_contact");
|
|
|
|
|
|
//=================================================================================================================
|
|
std::string LastFatherAddContactId;
|
|
// Add a contact to the list, first step
|
|
class CHandlerAddContactBegin : public IActionHandler
|
|
{
|
|
public:
|
|
void execute (CCtrlBase *pCaller, const std::string &sParams)
|
|
{
|
|
/** This msg may have been triggered from valid button or from the edit box itself, so retrieve
|
|
* the edit box from the enclosing group
|
|
*/
|
|
// Get enclosing container to know in which people list we are
|
|
if (pCaller)
|
|
{
|
|
// Get header_open
|
|
CInterfaceGroup *group = pCaller->getParent();
|
|
if (group)
|
|
{
|
|
// Get container
|
|
group = group->getParent();
|
|
if (group)
|
|
{
|
|
LastFatherAddContactId = group->getId();
|
|
if (!LastFatherAddContactId.empty())
|
|
{
|
|
CInterfaceManager *pIM = CInterfaceManager::getInstance();
|
|
string groupName= getParam(sParams, "group");
|
|
CInterfaceGroup *gc = dynamic_cast<CInterfaceGroup *>(CWidgetManager::getInstance()->getElementFromId(groupName));
|
|
if (gc)
|
|
{
|
|
CGroupEditBox *geb = dynamic_cast<CGroupEditBox *>(gc->getGroup("add_contact_eb:eb"));
|
|
geb->setInputString(ucstring(""));
|
|
}
|
|
CAHManager::getInstance()->runActionHandler("enter_modal", pCaller, sParams);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerAddContactBegin, "add_contact_begin");
|
|
|
|
|
|
//=================================================================================================================
|
|
// Add a contact to the list
|
|
class CHandlerAddContact : public IActionHandler
|
|
{
|
|
public:
|
|
void execute (CCtrlBase *pCaller, const std::string &/* sParams */)
|
|
{
|
|
CInterfaceManager *pIM = CInterfaceManager::getInstance();
|
|
|
|
/** This msg may have been triggered from valid button or from the edit box itself, so retrieve
|
|
* the edit box from the enclosing group
|
|
*/
|
|
// Get enclosing container to know in which people list we are
|
|
if (!LastFatherAddContactId.empty() && pCaller)
|
|
{
|
|
CInterfaceGroup *fatherGC = pCaller->getParent();
|
|
if (fatherGC)
|
|
{
|
|
// Look for the root parent
|
|
for(;;)
|
|
{
|
|
CInterfaceGroup *parent = fatherGC->getParent();
|
|
if (!parent || (parent->getId()=="ui:interface"))
|
|
break;
|
|
fatherGC = parent;
|
|
}
|
|
|
|
// Get the modal edit box
|
|
CGroupEditBox *geb = dynamic_cast<CGroupEditBox *>(fatherGC->getGroup("add_contact_eb:eb"));
|
|
if (geb && !geb->getInputString().empty())
|
|
{
|
|
std::string::size_type lastIndex = LastFatherAddContactId.rfind(":");
|
|
if (lastIndex != std::string::npos)
|
|
{
|
|
// Get the people list with the preselected container ID
|
|
CPeopleList *peopleList = PeopleInterraction.getPeopleListFromContainerID(LastFatherAddContactId.substr(lastIndex+1));
|
|
if (peopleList)
|
|
{
|
|
// don't add if it is the player name
|
|
if (!ClientCfg.Local && (UserEntity->getEntityName() == geb->getInputString()))
|
|
{
|
|
displayVisibleSystemMsg(CI18N::get("uiCantAddYourSelfInContactList"));
|
|
}
|
|
else
|
|
{
|
|
PeopleInterraction.askAddContact(geb->getInputString(), peopleList);
|
|
geb->setInputString(ucstring(""));
|
|
}
|
|
}
|
|
}
|
|
geb->setInputString(ucstring(""));
|
|
}
|
|
}
|
|
}
|
|
CAHManager::getInstance()->runActionHandler("leave_modal", pCaller, "");
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerAddContact, "add_contact");
|
|
|
|
|
|
//=================================================================================================================
|
|
class CHandlerMoveContact : public IActionHandler
|
|
{
|
|
public:
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &sParams)
|
|
{
|
|
// retrieve the index of the people
|
|
CPeopleList *srcList;
|
|
uint peopleIndex;
|
|
if (PeopleInterraction.getPeopleFromCurrentMenu(srcList, peopleIndex))
|
|
{
|
|
// get the destination list
|
|
CPeopleList *destList;
|
|
int listIndex;
|
|
if (!fromString(getParam(sParams, "list"), listIndex))
|
|
{
|
|
nlwarning("Bad list index");
|
|
return;
|
|
}
|
|
switch(listIndex)
|
|
{
|
|
case 0:
|
|
destList = &PeopleInterraction.IgnoreList;
|
|
break;
|
|
case 1:
|
|
destList = &PeopleInterraction.FriendList;
|
|
break;
|
|
default: nlwarning("Bad list index"); return;
|
|
}
|
|
|
|
PeopleInterraction.askMoveContact(peopleIndex, srcList, destList);
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerMoveContact, "move_contact");
|
|
|
|
|
|
//=================================================================================================================
|
|
class CHandlerSortContacts : public IActionHandler
|
|
{
|
|
public:
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
CInterfaceManager* pIM= CInterfaceManager::getInstance();
|
|
nlinfo("Load Order : %d", NLGUI::CDBManager::getInstance()->getDbProp("UI:SAVE:CONTACT_LIST:SORT_ORDER")->getValue32());
|
|
CPeopleList::TSortOrder order = (CPeopleList::TSortOrder)(NLGUI::CDBManager::getInstance()->getDbProp("UI:SAVE:CONTACT_LIST:SORT_ORDER")->getValue32());
|
|
|
|
order = (CPeopleList::TSortOrder)(order + 1);
|
|
if (order == CPeopleList::END_SORT_ORDER)
|
|
{
|
|
order = CPeopleList::START_SORT_ORDER;
|
|
}
|
|
|
|
nlinfo("Save Order : %d", order);
|
|
NLGUI::CDBManager::getInstance()->getDbProp("UI:SAVE:CONTACT_LIST:SORT_ORDER")->setValue32((sint32)order);
|
|
CPeopleList *pl = PeopleInterraction.getPeopleListFromCurrentMenu();
|
|
if (pl)
|
|
pl->sortEx(order);
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerSortContacts, "sort_contacts");
|
|
|
|
|
|
//=================================================================================================================
|
|
// Directly chat with a friend (launch a container chat)
|
|
class CHandlerContactDirectChat : public IActionHandler
|
|
{
|
|
public:
|
|
void execute (CCtrlBase *pCaller, const std::string &/* sParams */)
|
|
{
|
|
if (pCaller == NULL)
|
|
return;
|
|
|
|
CInterfaceGroup *fatherGC = pCaller->getParent();
|
|
if (fatherGC == NULL)
|
|
return;
|
|
fatherGC = fatherGC->getParent();
|
|
if (fatherGC == NULL)
|
|
return;
|
|
string str = fatherGC->getId().substr(0,fatherGC->getId().rfind('_'));
|
|
str = str.substr(str.rfind(':')+1, str.size());
|
|
CPeopleList *peopleList = PeopleInterraction.getPeopleListFromContainerID(str);
|
|
if (peopleList == NULL)
|
|
return;
|
|
|
|
sint index = peopleList->getIndexFromContainerID(fatherGC->getId());
|
|
if (index == -1)
|
|
return;
|
|
|
|
peopleList->openCloseChat(index, true);
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerContactDirectChat, "contact_direct_chat");
|
|
|
|
|
|
////////////////
|
|
// PARTY CHAT //
|
|
////////////////
|
|
|
|
//=================================================================================================================
|
|
/** Menu to create a new party chat
|
|
*/
|
|
class CHandlerNewPartyChat : public IActionHandler
|
|
{
|
|
public:
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
nlwarning("Deactivated for now!");
|
|
return;
|
|
CInterfaceManager *im = CInterfaceManager::getInstance();
|
|
CGroupContainer *gc = dynamic_cast<CGroupContainer *>(CWidgetManager::getInstance()->getElementFromId(NEW_PARTY_CHAT_WINDOW));
|
|
if (!gc) return;
|
|
CWidgetManager::getInstance()->setTopWindow(gc);
|
|
// Set the keyboard focus
|
|
CGroupEditBox *eb = dynamic_cast<CGroupEditBox *>(gc->getGroup("eb"));
|
|
if (eb)
|
|
{
|
|
CWidgetManager::getInstance()->setCaptureKeyboard(eb);
|
|
eb->setInputString(ucstring(""));
|
|
}
|
|
//
|
|
if (gc->getActive())
|
|
{
|
|
gc->enableBlink(1);
|
|
return;
|
|
}
|
|
gc->setActive(true);
|
|
gc->updateCoords();
|
|
gc->center();
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerNewPartyChat, "new_party_chat");
|
|
|
|
//=================================================================================================================
|
|
/** The name of a party chat has been validated
|
|
*/
|
|
class CHandlerValidatePartyChatName : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
CInterfaceManager *im = CInterfaceManager::getInstance();
|
|
CGroupContainer *gc = dynamic_cast<CGroupContainer *>(CWidgetManager::getInstance()->getElementFromId(NEW_PARTY_CHAT_WINDOW));
|
|
if (!gc) return;
|
|
CGroupEditBox *eb = dynamic_cast<CGroupEditBox *>(gc->getGroup("eb"));
|
|
if (!eb) return;
|
|
ucstring title = eb->getInputString();
|
|
|
|
// TODO GAMEDEV : create (or join ?) a new channel. Each channel (party chat) should have a unique name in the game
|
|
// moreover, it should not have the name of another available chat window (for example, it shouldn't be named 'Around Me')
|
|
// It shouldn't have the name of an existing player, either..
|
|
// Maybe this last test can be done in local only
|
|
|
|
if (!PeopleInterraction.testValidPartyChatName(title))
|
|
{
|
|
displayVisibleSystemMsg(title + ucstring(" : ") + CI18N::get("uiInvalidPartyChatName"));
|
|
return;
|
|
}
|
|
|
|
// create the party chat
|
|
PeopleInterraction.createNewPartyChat(title);
|
|
return;
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER(CHandlerValidatePartyChatName, "validate_party_chat_name");
|
|
|
|
|
|
//=================================================================================================================
|
|
/** Menu to create a new party chat
|
|
*/
|
|
|
|
|
|
//=================================================================================================================
|
|
/** Menu to remove a currenlty created party chat
|
|
*/
|
|
class CHandlerRemovePartyChat : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
CChatWindow *chat = getChatWndMgr().getChatWindowFromCaller(CWidgetManager::getInstance()->getCtrlLaunchingModal());
|
|
if (chat) PeopleInterraction.removePartyChat(chat);
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerRemovePartyChat, "remove_party_chat");
|
|
|
|
//=================================================================================================================
|
|
/** TEMP : just create an 'invite' command in the 'around me' edit box
|
|
*/
|
|
class CHandlerPartyChatInvite : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
CChatWindow *am = PeopleInterraction.AroundMe.Window;
|
|
if (!am) return;
|
|
CCtrlBase *cb = am->getContainer();
|
|
while (cb)
|
|
{
|
|
cb->forceOpen();
|
|
cb = cb->getParent();
|
|
}
|
|
// make the container blink
|
|
am->getContainer()->enableBlink(2);
|
|
// TODO : center the view on the newly created container ?
|
|
// display a new command '/name' in the chat. The player must enter a new unique name for the party chat.
|
|
am->setCommand("invite ", false);
|
|
am->setKeyboardFocus();
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerPartyChatInvite, "party_chat_invite" );
|
|
|
|
|
|
//=================================================================================================================
|
|
/** Add all members of the team to the party chat
|
|
*/
|
|
class CHandlerAddAllTeamMembersToPartyChat : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
// CChatWindow *chat = getChatWndMgr().getChatWindowFromCaller(CWidgetManager::getInstance()->getCtrlLaunchingModal());
|
|
// TODO GAMEDEV : add all team members
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerAddAllTeamMembersToPartyChat, "add_all_team_members");
|
|
|
|
//=================================================================================================================
|
|
/** Remove all members of the team to the party chat
|
|
*/
|
|
class CHandlerRemoveAllTeamMembersToPartyChat : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
// CChatWindow *chat = getChatWndMgr().getChatWindowFromCaller(CWidgetManager::getInstance()->getCtrlLaunchingModal());
|
|
// TODO GAMEDEV : remove all team members
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerRemoveAllTeamMembersToPartyChat, "remove_all_team_members");
|
|
|
|
//=================================================================================================================
|
|
/** Add all members of the guild to the party chat
|
|
*/
|
|
class CHandlerAddAllGuildMembersToPartyChat : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
// CChatWindow *chat = getChatWndMgr().getChatWindowFromCaller(CWidgetManager::getInstance()->getCtrlLaunchingModal());
|
|
// TODO GAMEDEV : add all guild members
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerAddAllGuildMembersToPartyChat, "add_all_guild_members");
|
|
|
|
//=================================================================================================================
|
|
/** Remove all members of the team to the party chat
|
|
*/
|
|
class CHandlerRemoveAllGuildMembersToPartyChat : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
// CChatWindow *chat = getChatWndMgr().getChatWindowFromCaller(CWidgetManager::getInstance()->getCtrlLaunchingModal());
|
|
// TODO_GAMEDEV : remove all guild members
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerRemoveAllGuildMembersToPartyChat, "remove_all_guild_members");
|
|
|
|
/////////////////////////////////////////
|
|
// ACTION HANDLERS FOR MAIN/USER CHATS //
|
|
/////////////////////////////////////////
|
|
|
|
//=================================================================================================================
|
|
/** Select the target on a filtered chat window
|
|
* This create a menu with the standard window (team, around me ...) + the party chat windows
|
|
*/
|
|
class CHandlerSelectChatTarget : public IActionHandler
|
|
{
|
|
public:
|
|
void execute (CCtrlBase *pCaller, const std::string &sParams)
|
|
{
|
|
CChatWindow *cw = getChatWndMgr().getChatWindowFromCaller(pCaller);
|
|
if (!cw) return;
|
|
ChatWindowForFilter = cw;
|
|
CInterfaceManager *im = CInterfaceManager::getInstance();
|
|
//
|
|
std::string menuName = getParam(sParams, "menu");
|
|
std::string strPartyChats = getParam(sParams, "party_chats");
|
|
bool partyChats = true;
|
|
if (!strPartyChats.empty())
|
|
{
|
|
partyChats = nlstricmp("true", strPartyChats.c_str()) == 0;
|
|
}
|
|
// get the menu
|
|
CGroupMenu *menu = dynamic_cast<CGroupMenu *>(CWidgetManager::getInstance()->getElementFromId(menuName));
|
|
if (!menu) return;
|
|
// remove all party chat from the previous list
|
|
uint lastTargetSelectedIndex = 0;
|
|
for(uint k = 0; k < menu->getNumLine();)
|
|
{
|
|
if (nlstricmp("chat_target_selected", menu->getActionHandler(k)) == 0)
|
|
{
|
|
lastTargetSelectedIndex = k;
|
|
int dummy;
|
|
if (fromString(menu->getActionHandlerParam(k), dummy))
|
|
{
|
|
// this is a party chat, removes the entry
|
|
menu->deleteLine(k);
|
|
-- lastTargetSelectedIndex;
|
|
}
|
|
else
|
|
{
|
|
++k;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
++k;
|
|
}
|
|
}
|
|
|
|
CPeopleInterraction &pl = PeopleInterraction;
|
|
// add names of the party chats
|
|
uint insertionIndex = lastTargetSelectedIndex + 1; // insert after standard options
|
|
if (partyChats)
|
|
{
|
|
for(uint l = 0; l < pl.PartyChats.size(); ++l)
|
|
{
|
|
menu->addLineAtIndex(insertionIndex, pl.PartyChats[l].Window->getTitle(), "chat_target_selected", toString(pl.PartyChats[l].ID));
|
|
++ insertionIndex;
|
|
}
|
|
}
|
|
|
|
// Case of user chat in grouped chat window
|
|
if ((cw == PeopleInterraction.ChatGroup.Window) || (cw = PeopleInterraction.TheUserChat.Window))
|
|
{
|
|
CInterfaceManager *pIM = CInterfaceManager::getInstance();
|
|
cw = PeopleInterraction.TheUserChat.Window;
|
|
// CChatStdInput &ci = PeopleInterraction.ChatInput;
|
|
CGroupMenu *pMenu = dynamic_cast<CGroupMenu*>(CWidgetManager::getInstance()->getElementFromId("ui:interface:user_chat_target_menu"));
|
|
CViewTextMenu *pMenuAround = dynamic_cast<CViewTextMenu*>(pMenu->getElement("ui:interface:user_chat_target_menu:around"));
|
|
CViewTextMenu *pMenuRegion = dynamic_cast<CViewTextMenu*>(pMenu->getElement("ui:interface:user_chat_target_menu:region"));
|
|
CViewTextMenu *pMenuUniverse = dynamic_cast<CViewTextMenu*>(pMenu->getElement("ui:interface:user_chat_target_menu:universe"));
|
|
CViewTextMenu *pMenuTeam = dynamic_cast<CViewTextMenu*>(pMenu->getElement("ui:interface:user_chat_target_menu:team"));
|
|
CViewTextMenu *pMenuGuild = dynamic_cast<CViewTextMenu*>(pMenu->getElement("ui:interface:user_chat_target_menu:guild"));
|
|
const bool teamActive = NLGUI::CDBManager::getInstance()->getDbProp("SERVER:GROUP:0:PRESENT")->getValueBool();
|
|
const bool guildActive = NLGUI::CDBManager::getInstance()->getDbProp("SERVER:GUILD:NAME")->getValueBool();
|
|
if (pMenuAround) pMenuAround->setGrayed (false);
|
|
if (pMenuRegion) pMenuRegion->setGrayed (false);
|
|
if (pMenuUniverse) pMenuUniverse->setGrayed (false);
|
|
if (pMenuTeam) pMenuTeam->setGrayed (!teamActive);
|
|
if (pMenuGuild) pMenuGuild->setGrayed (!guildActive);
|
|
|
|
// Remove existing dynamic chats
|
|
while (pMenu->getNumLine() > 5)
|
|
{
|
|
pMenu->deleteLine(pMenu->getNumLine()-1);
|
|
}
|
|
|
|
// Add dynamic chats
|
|
uint insertion_index = 0;
|
|
for (uint i = 0; i < CChatGroup::MaxDynChanPerPlayer; i++)
|
|
{
|
|
string s = toString(i);
|
|
uint32 textId = ChatMngr.getDynamicChannelNameFromDbIndex(i);
|
|
bool active = (textId != 0);
|
|
if (active)
|
|
{
|
|
uint32 canWrite = NLGUI::CDBManager::getInstance()->getDbProp("SERVER:DYN_CHAT:CHANNEL"+s+":WRITE_RIGHT")->getValue32();
|
|
if (canWrite != 0)
|
|
{
|
|
ucstring title;
|
|
STRING_MANAGER::CStringManagerClient::instance()->getDynString(textId, title);
|
|
pMenu->addLineAtIndex(5 + insertion_index, title+" @{T8}/"+s, "chat_target_selected", "dyn"+s, "dyn"+s);
|
|
insertion_index++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// activate the menu
|
|
CWidgetManager::getInstance()->enableModalWindow(pCaller, menuName);
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerSelectChatTarget, "select_chat_target");
|
|
|
|
//=================================================================================================================
|
|
/** A target has been selected for a filtered chat
|
|
*/
|
|
class CHandlerChatTargetSelected : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &sParams)
|
|
{
|
|
// for now, manage a single filtered chat window
|
|
CChatWindow *cw = ChatWindowForFilter;
|
|
if (!cw) return;
|
|
ChatWindowForFilter = NULL;
|
|
CFilteredChat *fc = PeopleInterraction.getFilteredChatFromChatWindow(cw);
|
|
if (!fc) return;
|
|
CChatTargetFilter &cf = fc->Filter;
|
|
// Team
|
|
if (nlstricmp(sParams, "team") == 0)
|
|
{
|
|
cf.setTargetGroup(CChatGroup::team);
|
|
}
|
|
// Guild
|
|
else if (nlstricmp(sParams, "guild") == 0)
|
|
{
|
|
cf.setTargetGroup(CChatGroup::guild);
|
|
}
|
|
// Say
|
|
else if (nlstricmp(sParams, "say") == 0)
|
|
{
|
|
cf.setTargetGroup(CChatGroup::say);
|
|
}
|
|
// Shout
|
|
else if (nlstricmp(sParams, "shout") == 0)
|
|
{
|
|
cf.setTargetGroup(CChatGroup::shout);
|
|
}
|
|
// Region
|
|
else if (nlstricmp(sParams, "region") == 0)
|
|
{
|
|
cf.setTargetGroup(CChatGroup::region);
|
|
}
|
|
// Universe
|
|
else if (nlstricmp(sParams, "universe") == 0)
|
|
{
|
|
cf.setTargetGroup(CChatGroup::universe);
|
|
}
|
|
else
|
|
{
|
|
for (uint i = 0; i < CChatGroup::MaxDynChanPerPlayer; i++)
|
|
{
|
|
if (nlstricmp(sParams, "dyn"+toString("%d", i)) == 0)
|
|
{
|
|
cf.setTargetGroup(CChatGroup::dyn_chat, i);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Case of user chat in grouped chat window
|
|
if (cw == PeopleInterraction.ChatGroup.Window)
|
|
{
|
|
PeopleInterraction.TheUserChat.Filter.setTargetGroup(cf.getTargetGroup(), cf.getTargetDynamicChannelDbIndex());
|
|
CAHManager::getInstance()->runActionHandler("chat_group_filter", NULL, "user");
|
|
}
|
|
if (cw == PeopleInterraction.TheUserChat.Window)
|
|
{
|
|
PeopleInterraction.TheUserChat.Filter.setTargetGroup(cf.getTargetGroup(), cf.getTargetDynamicChannelDbIndex());
|
|
CAHManager::getInstance()->runActionHandler("user_chat_active", NULL, "");
|
|
}
|
|
|
|
// The target should be a party chat
|
|
int partyChatID;
|
|
if (fromString(sParams, partyChatID))
|
|
{
|
|
// search party chat in the list
|
|
std::vector<CPartyChatInfo> &partyChats = PeopleInterraction.PartyChats;
|
|
for(uint k = 0; k < partyChats.size(); ++k)
|
|
{
|
|
if (partyChats[k].ID == (uint) partyChatID)
|
|
{
|
|
cf.setTargetPartyChat(partyChats[k].Window);
|
|
return;
|
|
}
|
|
}
|
|
// The party chat has been deleted while the menu was displayed it seems.. -> no-op
|
|
return;
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerChatTargetSelected, "chat_target_selected");
|
|
|
|
|
|
//=================================================================================================================
|
|
/** If no more in team, leave team chat mode
|
|
*/
|
|
class CHandlerLeaveTeamChat : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
if( PeopleInterraction.TheUserChat.Filter.getTargetGroup() == CChatGroup::team )
|
|
{
|
|
CInterfaceManager *im = CInterfaceManager::getInstance();
|
|
if( im )
|
|
{
|
|
if( !NLGUI::CDBManager::getInstance()->getDbProp("UI:VARIABLES:IS_TEAM_PRESENT")->getValueBool() )
|
|
{
|
|
ChatMngr.updateChatModeAndButton(CChatGroup::say);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerLeaveTeamChat, "leave_team_chat");
|
|
|
|
|
|
|
|
|
|
|
|
/** Create checkbox for a menu.
|
|
*/
|
|
static CInterfaceGroup *createMenuCheckBox(const std::string &onclickL, const std::string ¶msL, bool checked)
|
|
{
|
|
pair<string, string> params [2];
|
|
params[0].first = "onclick_l";
|
|
params[0].second = onclickL;
|
|
params[1].first = "params_l";
|
|
params[1].second = paramsL;
|
|
|
|
CInterfaceManager *im = CInterfaceManager::getInstance();
|
|
CInterfaceGroup *ig = CWidgetManager::getInstance()->getParser()->createGroupInstance("menu_checkbox", "", params, sizeof(params) / sizeof(params[0]));
|
|
if (!ig) return NULL;
|
|
CCtrlBaseButton *cb = dynamic_cast<CCtrlBaseButton *>(ig->getCtrl("b"));
|
|
if (!cb) return NULL;
|
|
cb->setPushed(checked);
|
|
return ig;
|
|
}
|
|
|
|
|
|
|
|
//=================================================================================================================
|
|
/** Display a menu to select the source on a filtered chat
|
|
*/
|
|
class CHandlerSelectChatSource : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase *pCaller, const std::string &/* sParams */)
|
|
{
|
|
static const char *FILTER_TOGGLE = "chat_source_selected";
|
|
CPeopleInterraction &pi = PeopleInterraction;
|
|
CChatWindow *cw = getChatWndMgr().getChatWindowFromCaller(pCaller);
|
|
if (!cw) return;
|
|
ChatWindowForFilter = cw;
|
|
CInterfaceManager *im = CInterfaceManager::getInstance();
|
|
|
|
|
|
// *** get the main_chat or user_chat menu
|
|
CGroupMenu *menu= NULL;
|
|
bool addUserChatEntries= false;
|
|
// If the current window is the chat group
|
|
if (cw == pi.ChatGroup.Window)
|
|
{
|
|
// select main chat menu
|
|
menu = dynamic_cast<CGroupMenu *>(CWidgetManager::getInstance()->getElementFromId(MAIN_CHAT_SOURCE_MENU));
|
|
|
|
// Remove all unused dynamic channels and set the names
|
|
for (uint i = 0; i < CChatGroup::MaxDynChanPerPlayer; i++)
|
|
{
|
|
string s = toString(i);
|
|
CViewTextMenu *pVTM = dynamic_cast<CViewTextMenu *>(CWidgetManager::getInstance()->getElementFromId(MAIN_CHAT_SOURCE_MENU+":tab:dyn"+s));
|
|
if (pVTM)
|
|
{
|
|
uint32 textId = ChatMngr.getDynamicChannelNameFromDbIndex(i);
|
|
bool active = (textId != 0);
|
|
pVTM->setActive(active);
|
|
if (active)
|
|
{
|
|
ucstring title;
|
|
STRING_MANAGER::CStringManagerClient::instance()->getDynString(textId, title);
|
|
pVTM->setText("["+s+"] " + title);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Menu with Filters
|
|
CChatGroupWindow *pWin = pi.getChatGroupWindow();
|
|
if (pWin->getTabIndex() == 5) // (5 == user) -> complete menu
|
|
{
|
|
// get the real user chat setup
|
|
cw = pi.TheUserChat.Window;
|
|
addUserChatEntries= true;
|
|
}
|
|
else
|
|
{
|
|
// Don't add user chat since the selected TAB is not the user chat
|
|
addUserChatEntries= false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Menu with Filters
|
|
if (cw == pi.TheUserChat.Window)
|
|
{
|
|
// select user chat menu
|
|
menu = dynamic_cast<CGroupMenu *>(CWidgetManager::getInstance()->getElementFromId(USER_CHAT_SOURCE_MENU));
|
|
addUserChatEntries= true;
|
|
}
|
|
// Simple menu
|
|
else
|
|
{
|
|
// This is neither the ChatGroup, nor the UserChat. Should not be here.
|
|
// Just open the STD chat menu, and quit
|
|
NLGUI::CDBManager::getInstance()->getDbProp("UI:VARIABLES:GC_POPUP")->setValue64(cw->getContainer()->isPopuped() || cw->getContainer()->getLayerSetup() == 0 ? 1 : 0);
|
|
NLGUI::CDBManager::getInstance()->getDbProp("UI:VARIABLES:GC_HAS_HELP")->setValue64(!cw->getContainer()->getHelpPage().empty());
|
|
CWidgetManager::getInstance()->enableModalWindow(pCaller, STD_CHAT_SOURCE_MENU);
|
|
return;
|
|
}
|
|
}
|
|
if (!menu) return;
|
|
|
|
|
|
// *** remove any previous entries
|
|
for(uint k = 0; k < menu->getNumLine();)
|
|
{
|
|
if (nlstricmp(FILTER_TOGGLE, menu->getActionHandler(k)) == 0)
|
|
{
|
|
menu->deleteLine(k);
|
|
}
|
|
else
|
|
{
|
|
++k;
|
|
}
|
|
}
|
|
|
|
|
|
// *** create new entries
|
|
if(addUserChatEntries)
|
|
{
|
|
uint insertionIndex = 0;
|
|
// AROUND ME
|
|
menu->addLineAtIndex(insertionIndex, CI18N::get("uiAroundMe"), FILTER_TOGGLE, "am");
|
|
// add a checkbox
|
|
menu->setUserGroupLeft(insertionIndex, createMenuCheckBox(FILTER_TOGGLE, "am", pi.ChatInput.AroundMe.isListeningWindow(cw)));
|
|
++ insertionIndex;
|
|
|
|
// REGION
|
|
menu->addLineAtIndex(insertionIndex, CI18N::get("uiREGION"), FILTER_TOGGLE, "region");
|
|
// add a checkbox
|
|
menu->setUserGroupLeft(insertionIndex, createMenuCheckBox(FILTER_TOGGLE, "region", pi.ChatInput.Region.isListeningWindow(cw)));
|
|
++ insertionIndex;
|
|
|
|
// UNIVERSE
|
|
menu->addLineAtIndex(insertionIndex, CI18N::get("uiUNIVERSE"), FILTER_TOGGLE, "universe");
|
|
// add a checkbox
|
|
menu->setUserGroupLeft(insertionIndex, createMenuCheckBox(FILTER_TOGGLE, "universe", pi.ChatInput.Universe.isListeningWindow(cw)));
|
|
++ insertionIndex;
|
|
|
|
// TEAM
|
|
menu->addLineAtIndex(insertionIndex, CI18N::get("uiTeam"), FILTER_TOGGLE, "team");
|
|
// add a checkbox
|
|
menu->setUserGroupLeft(insertionIndex, createMenuCheckBox(FILTER_TOGGLE, "team", pi.ChatInput.Team.isListeningWindow(cw)));
|
|
++insertionIndex;
|
|
|
|
// GUILD
|
|
menu->addLineAtIndex(insertionIndex, CI18N::get("uimGuild"), FILTER_TOGGLE, "guild");
|
|
// add a checkbox
|
|
menu->setUserGroupLeft(insertionIndex, createMenuCheckBox(FILTER_TOGGLE, "guild", pi.ChatInput.Guild.isListeningWindow(cw)));
|
|
++ insertionIndex;
|
|
|
|
|
|
// TELL
|
|
//menu->addLineAtIndex(insertionIndex, CI18N::get("uiTell"), FILTER_TOGGLE, "tell");
|
|
// add a checkbox
|
|
//menu->setUserGroupLeft(insertionIndex, createMenuCheckBox(FILTER_TOGGLE, "tell", pi.ChatInput.Tell.isListeningWindow(cw)));
|
|
//++ insertionIndex;
|
|
|
|
// SYSTEM INFOS
|
|
menu->addLineAtIndex(insertionIndex, CI18N::get("uiSystemInfo"), FILTER_TOGGLE, "si");
|
|
// add a checkbox
|
|
menu->setUserGroupLeft(insertionIndex, createMenuCheckBox(FILTER_TOGGLE, "si", pi.ChatInput.SystemInfo.isListeningWindow(cw)));
|
|
++insertionIndex;
|
|
|
|
// add party chats
|
|
std::vector<CPartyChatInfo> &pc = pi.PartyChats;
|
|
for(uint l = 0; l < pc.size(); ++l)
|
|
{
|
|
if (pc[l].Filter != NULL)
|
|
{
|
|
menu->addLineAtIndex(insertionIndex, pc[l].Window->getTitle(), FILTER_TOGGLE, toString(pc[l].ID));
|
|
menu->setUserGroupLeft(insertionIndex, createMenuCheckBox(FILTER_TOGGLE, toString(pc[l].ID), pc[l].Filter->isListeningWindow(cw)));
|
|
++ insertionIndex;
|
|
}
|
|
}
|
|
|
|
// Add all existing dynamic channels and set the names
|
|
for (uint8 i = 0; i < CChatGroup::MaxDynChanPerPlayer; i++)
|
|
{
|
|
string s = toString(i);
|
|
uint32 textId = ChatMngr.getDynamicChannelNameFromDbIndex(i);
|
|
bool active = (textId != 0);
|
|
if (active)
|
|
{
|
|
ucstring title;
|
|
STRING_MANAGER::CStringManagerClient::instance()->getDynString(textId, title);
|
|
menu->addLineAtIndex(insertionIndex, "["+s+"] " + title, FILTER_TOGGLE, "dyn"+s);
|
|
menu->setUserGroupLeft(insertionIndex, createMenuCheckBox(FILTER_TOGGLE, "dyn"+s, pi.ChatInput.DynamicChat[i].isListeningWindow(cw)));
|
|
++insertionIndex;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|
|
// *** active the menu
|
|
NLGUI::CDBManager::getInstance()->getDbProp("UI:VARIABLES:GC_POPUP")->setValue64(cw->getContainer()->isPopuped() || cw->getContainer()->getLayerSetup() == 0 ? 1 : 0);
|
|
NLGUI::CDBManager::getInstance()->getDbProp("UI:VARIABLES:GC_HAS_HELP")->setValue64(!cw->getContainer()->getHelpPage().empty());
|
|
CWidgetManager::getInstance()->enableModalWindow(pCaller, menu);
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER(CHandlerSelectChatSource, "select_chat_source");
|
|
|
|
|
|
|
|
//=================================================================================================================
|
|
/** A new source has been selected / unselected from a filtered chat
|
|
*/
|
|
class CHandlerChatSourceSelected : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &sParams)
|
|
{
|
|
int partyChatID;
|
|
|
|
CChatWindow *cw = ChatWindowForFilter;
|
|
if (!cw) return;
|
|
|
|
CChatStdInput &ci = PeopleInterraction.ChatInput;
|
|
|
|
|
|
if (cw == PeopleInterraction.ChatGroup.Window)
|
|
{
|
|
CChatGroupWindow *pWin = PeopleInterraction.getChatGroupWindow();
|
|
if (pWin->getTabIndex() != 5) // (5 == user)
|
|
return; // Nothing to select except if user chat
|
|
|
|
cw = PeopleInterraction.TheUserChat.Window;
|
|
}
|
|
|
|
|
|
/*CCtrlBaseButton *button = dynamic_cast<CCtrlBaseButton *>(pCaller);
|
|
if (button)
|
|
{
|
|
button->setPushed(!button->getPushed());
|
|
}*/
|
|
// GUILD
|
|
if (nlstricmp(sParams, "guild") == 0)
|
|
{
|
|
if (ci.Guild.isListeningWindow(cw)) ci.Guild.removeListeningWindow(cw);
|
|
else ci.Guild.addListeningWindow(cw);
|
|
}
|
|
else
|
|
// TEAM
|
|
if (nlstricmp(sParams, "team") == 0)
|
|
{
|
|
if (ci.Team.isListeningWindow(cw)) ci.Team.removeListeningWindow(cw);
|
|
else ci.Team.addListeningWindow(cw);
|
|
}
|
|
else
|
|
// AROUND ME
|
|
if (nlstricmp(sParams, "am") == 0)
|
|
{
|
|
if (ci.AroundMe.isListeningWindow(cw)) ci.AroundMe.removeListeningWindow(cw);
|
|
else ci.AroundMe.addListeningWindow(cw);
|
|
}
|
|
else
|
|
// REGION
|
|
if (nlstricmp(sParams, "region") == 0)
|
|
{
|
|
if (ci.Region.isListeningWindow(cw)) ci.Region.removeListeningWindow(cw);
|
|
else ci.Region.addListeningWindow(cw);
|
|
}
|
|
else
|
|
// UNIVERSE
|
|
if (nlstricmp(sParams, "universe") == 0)
|
|
{
|
|
if (ci.Universe.isListeningWindow(cw)) ci.Universe.removeListeningWindow(cw);
|
|
else ci.Universe.addListeningWindow(cw);
|
|
}
|
|
else
|
|
// TELL
|
|
if (nlstricmp(sParams, "tell") == 0)
|
|
{
|
|
if (ci.Tell.isListeningWindow(cw)) ci.Tell.removeListeningWindow(cw);
|
|
else ci.Tell.addListeningWindow(cw);
|
|
}
|
|
else
|
|
// SYSTEM INFOS
|
|
if (nlstricmp(sParams, "si") == 0)
|
|
{
|
|
if (ci.SystemInfo.isListeningWindow(cw)) ci.SystemInfo.removeListeningWindow(cw);
|
|
else ci.SystemInfo.addListeningWindow(cw);
|
|
}
|
|
else
|
|
// PARTY CHAT
|
|
if (fromString(sParams, partyChatID))
|
|
{
|
|
std::vector<CPartyChatInfo> &partyChats = PeopleInterraction.PartyChats;
|
|
for(uint k = 0; k < partyChats.size(); ++k)
|
|
{
|
|
if (partyChats[k].ID == (uint) partyChatID)
|
|
{
|
|
if (partyChats[k].Filter != NULL)
|
|
{
|
|
if (partyChats[k].Filter->isListeningWindow(cw)) partyChats[k].Filter->removeListeningWindow(partyChats[k].Window);
|
|
else partyChats[k].Filter->addListeningWindow(cw);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if (nlstricmp(sParams.substr(0, 3), "dyn") == 0)
|
|
{
|
|
uint8 i = 0;
|
|
fromString(sParams.substr(3), i);
|
|
if (ci.DynamicChat[i].isListeningWindow(cw)) ci.DynamicChat[i].removeListeningWindow(cw);
|
|
else ci.DynamicChat[i].addListeningWindow(cw);
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerChatSourceSelected, "chat_source_selected");
|
|
|
|
|
|
// show / hide the edit/box of a chatbox
|
|
class CHandlerToggleChatEBVis : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
CCtrlBase *clm = CWidgetManager::getInstance()->getCtrlLaunchingModal();
|
|
if (!clm) return;
|
|
CInterfaceGroup *ig = clm->getParent();
|
|
do
|
|
{
|
|
if (ig->isGroupContainer()) break;
|
|
ig = ig->getParent();
|
|
}
|
|
while(ig);
|
|
if (!ig) return;
|
|
CGroupContainer *gc = static_cast<CGroupContainer *>(ig);
|
|
CInterfaceGroup *eb = gc->getGroup("ebw");
|
|
if (eb)
|
|
{
|
|
eb->setActive(!eb->getActive());
|
|
}
|
|
CCtrlBase *tb = gc->getCtrl("target_button");
|
|
if (tb)
|
|
{
|
|
tb->setActive(!tb->getActive());
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerToggleChatEBVis, "toggle_chat_eb_vis");
|
|
|
|
// create a new user chat
|
|
class CHandlerNewUserChat : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase * /* pCaller */, const std::string &/* sParams */)
|
|
{
|
|
CPeopleInterraction &pi = PeopleInterraction;
|
|
for(uint k = 0; k < MaxNumUserChats; ++k)
|
|
{
|
|
if (pi.UserChat[k].Window == NULL) // not used ?
|
|
{
|
|
pi.createUserChat(k);
|
|
// add to std listeners
|
|
pi.ChatInput.registerListeningWindow(pi.UserChat[k].Window);
|
|
CGroupContainer *gc = pi.UserChat[k].Window->getContainer();
|
|
gc->setOpen(true);
|
|
gc->popupCurrentPos();
|
|
gc->updateCoords();
|
|
gc->center();
|
|
// change pos by a random amount
|
|
gc->setX(gc->getX() + rand() % 20 - 10);
|
|
gc->setY(gc->getY() + rand() % 20 - 10);
|
|
gc->invalidateCoords();
|
|
gc->enableBlink(2);
|
|
pi.UserChat[k].Window->setKeyboardFocus();
|
|
return;
|
|
}
|
|
}
|
|
nlwarning("Too much user chats created");
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER(CHandlerNewUserChat, "new_user_chat");
|
|
|
|
class CHandlerRemoveUserChat : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase *pCaller, const std::string &/* sParams */)
|
|
{
|
|
CPeopleInterraction &pi = PeopleInterraction;
|
|
CChatWindow *cw = getChatWndMgr().getChatWindowFromCaller(pCaller);
|
|
if (!cw) return;
|
|
CFilteredChat *fc = pi.getFilteredChatFromChatWindow(cw);
|
|
if (!fc) return;
|
|
getChatWndMgr().removeChatWindow(fc->Window);
|
|
fc->Filter.reset();
|
|
fc->Window = NULL;
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER(CHandlerRemoveUserChat, "remove_user_chat");
|
|
|
|
////////////////////////////////////////////
|
|
// COMMAND RELATED TO PEOPLE INTERRACTION //
|
|
////////////////////////////////////////////
|
|
|
|
|
|
//-----------------------------------------------
|
|
// 'ignore'
|
|
//-----------------------------------------------
|
|
NLMISC_COMMAND(ignore, "add or remove a player from the ignore list", "<player name>")
|
|
{
|
|
// Check parameters.
|
|
if(args.size() < 1)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
// NB: playernames cannot have special characters
|
|
ucstring playerName = ucstring(args[0]);
|
|
|
|
// add to the ignore list
|
|
PeopleInterraction.askAddContact(playerName, &PeopleInterraction.IgnoreList);
|
|
|
|
return true;
|
|
} // ignore //
|
|
|
|
/*
|
|
****
|
|
Yoyo: Party chat is not ended: DON'T LET THOSE COMMANDS AVAILABLE!
|
|
they made the client crash (cf createNewPartyChat)...
|
|
****
|
|
|
|
// create a new party chat with the given name
|
|
NLMISC_COMMAND(party_chat, "Create a new party chat", "<party_chat_name>")
|
|
{
|
|
if (args.size() != 1)
|
|
{
|
|
displayVisibleSystemMsg(CI18N::get("uiPartyChatCmd"));
|
|
return true;
|
|
}
|
|
CPeopleInterraction &pi = PeopleInterraction;
|
|
ucstring title = args[0];
|
|
|
|
if (!pi.testValidPartyChatName(title))
|
|
{
|
|
displayVisibleSystemMsg(CI18N::get("uiInvalidPartyChatName"));
|
|
return true;
|
|
}
|
|
|
|
PeopleInterraction.createNewPartyChat(title);
|
|
return true;
|
|
}
|
|
|
|
// Remove the party chat with the given name
|
|
NLMISC_COMMAND(remove_party_chat, "Remove a party chat", "<party_chat_name>")
|
|
{
|
|
if (args.size() != 1)
|
|
{
|
|
displayVisibleSystemMsg(CI18N::get("uiRemovePartyChatCmd"));
|
|
return true;
|
|
}
|
|
ucstring title = ucstring(args[0]);
|
|
CChatWindow *chat = getChatWndMgr().getChatWindow(title);
|
|
if (!chat)
|
|
{
|
|
displayVisibleSystemMsg(title + ucstring(" : ") + CI18N::get("uiBadPartyChatName"));
|
|
return true;
|
|
}
|
|
if (!PeopleInterraction.removePartyChat(chat))
|
|
{
|
|
displayVisibleSystemMsg(title + ucstring(" : ") + CI18N::get("uiCantRemovePartyChat"));
|
|
return true;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
|
|
// Join a party chat whose name is known
|
|
NLMISC_COMMAND(add_to_party_chat, "Join the given party chat", "<party_chat_name>")
|
|
{
|
|
if (args.size() != 1)
|
|
{
|
|
displayVisibleSystemMsg(CI18N::get("uiAddPartyChatCmd"));
|
|
return true;
|
|
}
|
|
// TODO GAMEDEV : join the party chat
|
|
return true;
|
|
}
|
|
|
|
// Invite someone in a party chat
|
|
NLMISC_COMMAND(invite, "Invite someone to a party chat", "<people_name> <party_chat_name>")
|
|
{
|
|
if (args.size() != 2)
|
|
{
|
|
displayVisibleSystemMsg(CI18N::get("uiInviteCmd"));
|
|
return true;
|
|
}
|
|
// TODO GAMEDEV : Send invite message to the server
|
|
// Check that the inviter has created the chat ?
|
|
// The people being invited should receive a popup to announce that he is being invited
|
|
return true;
|
|
}
|
|
|
|
*/
|
|
|
|
// ***************************************************************************
|
|
// chatLog
|
|
// Arg : none
|
|
// log all current chats in the file log_playername.txt saved in save directory
|
|
// ***************************************************************************
|
|
NLMISC_COMMAND(chatLog, "", "")
|
|
{
|
|
CInterfaceManager *pIM = CInterfaceManager::getInstance();
|
|
|
|
if(args.size() != 0)
|
|
return false;
|
|
|
|
if (pIM->getLogState())
|
|
pIM->displaySystemInfo(CI18N::get("uiLogTurnedOff"));
|
|
|
|
pIM->setLogState(!pIM->getLogState());
|
|
|
|
if (pIM->getLogState())
|
|
pIM->displaySystemInfo(CI18N::get("uiLogTurnedOn"));
|
|
|
|
CCDBNodeLeaf *node = NLGUI::CDBManager::getInstance()->getDbProp("UI:SAVE:CHATLOG_STATE", false);
|
|
if (node)
|
|
{
|
|
node->setValue32(pIM->getLogState() ? 1 : 0);
|
|
}
|
|
|
|
return true;
|
|
};
|
|
|
|
|
|
/////////////////////////
|
|
// INTERFACE FUNCTIONS //
|
|
/////////////////////////
|
|
static DECLARE_INTERFACE_USER_FCT(getNumUserChatLeft)
|
|
{
|
|
CPeopleInterraction &pi = PeopleInterraction;
|
|
uint left = 0;
|
|
for(uint k = 0; k < MaxNumUserChats; ++k)
|
|
{
|
|
if (pi.UserChat[k].Window == NULL) ++ left;
|
|
}
|
|
result.setInteger(left);
|
|
return true;
|
|
}
|
|
REGISTER_INTERFACE_USER_FCT("getNumUserChatLeft", getNumUserChatLeft)
|
|
|
|
|
|
//////////////////////////////////////
|
|
// STATIC FUNCTIONS IMPLEMENTATIONS //
|
|
//////////////////////////////////////
|
|
|
|
static void displayVisibleSystemMsg(const ucstring &msg, const string &cat)
|
|
{
|
|
CInterfaceManager *im = CInterfaceManager::getInstance();
|
|
im->displaySystemInfo(msg, cat);
|
|
if (CChatWindow::getChatWindowLaunchingCommand())
|
|
{
|
|
CChatWindow::getChatWindowLaunchingCommand()->displayMessage(msg, im->getSystemInfoColor(cat), CChatGroup::system, 0, 2);
|
|
}
|
|
}
|
|
|
|
#if !FINAL_VERSION
|
|
NLMISC_COMMAND(testSI, "tmp", "tmp")
|
|
{
|
|
PeopleInterraction.ChatInput.DebugInfo.displayMessage(ucstring("test"), CRGBA::Red);
|
|
return true;
|
|
}
|
|
#endif
|