mirror of
https://port.numenaute.org/aleajactaest/khanat-opennel-code.git
synced 2024-12-14 13:54:49 +00:00
3732 lines
119 KiB
C++
3732 lines
119 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"
|
|
|
|
|
|
#include "action_handler.h"
|
|
#include "interface_manager.h"
|
|
#include "bot_chat_manager.h"
|
|
#include "../sheet_manager.h"
|
|
#include "skill_manager.h"
|
|
#include "dbctrl_sheet.h"
|
|
#include "interface_expr.h"
|
|
#include "group_container.h"
|
|
#include "group_editbox.h"
|
|
#include "group_quick_help.h"
|
|
#include "view_text_id.h"
|
|
#include "../user_entity.h"
|
|
#include "../entities.h"
|
|
#include "dbgroup_combo_box.h"
|
|
#include "dbview_bar.h"
|
|
#include "../debug_client.h"
|
|
#include "interface_3d_scene.h"
|
|
#include "character_3d.h"
|
|
#include "item_special_effect.h"
|
|
#include "item_consumable_effect.h"
|
|
#include "bonus_malus.h"
|
|
// Game share
|
|
#include "game_share/slot_types.h"
|
|
#include "game_share/item_family.h"
|
|
#include "game_share/skills.h"
|
|
#include "game_share/armor_types.h"
|
|
#include "game_share/weapon_types.h"
|
|
#include "game_share/damage_types.h"
|
|
#include "game_share/ecosystem.h"
|
|
#include "game_share/mp_category.h"
|
|
#include "game_share/item_origin.h"
|
|
#include "game_share/shield_types.h"
|
|
#include "game_share/trade_slot_type.h"
|
|
#include "../string_manager_client.h"
|
|
#include "game_share/sphrase_com.h"
|
|
#include "sbrick_manager.h"
|
|
#include "sphrase_manager.h"
|
|
#include "action_handler_help.h"
|
|
#include "game_share/mission_desc.h"
|
|
#include "game_share/inventories.h"
|
|
#include "game_share/visual_slot_manager.h"
|
|
#include "game_share/prerequisit_infos.h"
|
|
#include "game_share/resistance_type.h"
|
|
#include "../r2/editor.h"
|
|
#include "../init.h"
|
|
#include "../browse_faq.h"
|
|
|
|
extern CSheetManager SheetMngr;
|
|
|
|
extern NLMISC::CLog g_log;
|
|
|
|
using namespace std;
|
|
using namespace NLMISC;
|
|
using namespace STRING_MANAGER;
|
|
|
|
|
|
///////////////////////////////////
|
|
// STATIC FUNCTIONS DECLARATIONS //
|
|
///////////////////////////////////
|
|
static void setupCreatorName(CSheetHelpSetup &setup);
|
|
static void setHelpText(CSheetHelpSetup &setup, const ucstring &text);
|
|
static void setHelpTextID(CSheetHelpSetup &setup, sint32 id);
|
|
static void fillSabrinaPhraseListBrick(const CSPhraseCom &phrase, IListSheetBase *listBrick);
|
|
static void setupListBrickHeader(CSheetHelpSetup &setup);
|
|
static void hideListBrickHeader(CSheetHelpSetup &setup);
|
|
|
|
static void setupHelpPage(CInterfaceGroup *window, const string &url);
|
|
static void setupHelpTitle(CInterfaceGroup *group, const ucstring &title);
|
|
static void setHelpCtrlSheet(CSheetHelpSetup &setup, uint32 sheetId);
|
|
|
|
// Setup help for an item in a window (type is known)
|
|
static void setupItemHelp(CSheetHelpSetup &setup);
|
|
static void setupPactHelp(CSheetHelpSetup &setup);
|
|
static void setupSkillToTradeHelp(CSheetHelpSetup &setup);
|
|
static void setupSabrinaBrickHelp(CSheetHelpSetup &setup, bool auraDisabled= false);
|
|
void setupSabrinaPhraseHelp(CSheetHelpSetup &setup, const class CSPhraseCom &phrase, uint32 phraseSheetId);
|
|
static void setupMissionHelp(CSheetHelpSetup &setup);
|
|
|
|
// ***************************************************************************
|
|
#define INFO_LIST_ITEM "list_item"
|
|
#define INFO_LIST_BRICK "list_brick"
|
|
#define INFO_LIST_BRICK_HEADER "list_brick_header"
|
|
#define INFO_LIST_BRICK_REQUIREMENT "list_brick_requirement"
|
|
#define INFO_GROUP_MP_STAT "mp_stats"
|
|
#define INFO_GROUP_CHAR_3D "char3d"
|
|
|
|
|
|
// ***************************************************************************
|
|
std::deque<uint> CInterfaceHelp::_ActiveWindows;
|
|
std::vector<CInterfaceHelp::CInfoWindow> CInterfaceHelp::_InfoWindows;
|
|
bool CInterfaceHelp::_InfoWindowInit= false;
|
|
CInterfaceHelp::CFittedWeaponWeightObserver CInterfaceHelp::_FittedWeaponWeightObserver;
|
|
|
|
|
|
// ***************************************************************************
|
|
void CInterfaceHelp::CInfoWindow::infoReceived()
|
|
{
|
|
// refresh text
|
|
if(CtrlSheet && Window)
|
|
{
|
|
CSheetHelpSetup setup;
|
|
setup.setupDefaultIDs();
|
|
setup.HelpWindow = Window;
|
|
setup.SrcSheet = CtrlSheet;
|
|
refreshItemHelp(setup);
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInterfaceHelp::CInfoWindow::missionInfoReceived(const CPrerequisitInfos &infos)
|
|
{
|
|
// refresh text
|
|
if(CtrlSheet && Window)
|
|
{
|
|
CSheetHelpSetup setup;
|
|
setup.setupDefaultIDs();
|
|
setup.HelpWindow = Window;
|
|
setup.SrcSheet = CtrlSheet;
|
|
refreshMissionHelp(setup, infos);
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInterfaceHelp::initWindows()
|
|
{
|
|
if(_InfoWindowInit)
|
|
return;
|
|
|
|
_InfoWindowInit= true;
|
|
|
|
// Get the Max window allowed.
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
|
|
sint maxHelpWindow;
|
|
fromString(pIM->getDefine("MAX_HELP_WINDOW"), maxHelpWindow);
|
|
|
|
// Allow Max 256. More may be a script error...
|
|
clamp(maxHelpWindow, 0, 256);
|
|
|
|
for(sint i=0;i<maxHelpWindow;i++)
|
|
{
|
|
CInterfaceGroup *group= dynamic_cast<CInterfaceGroup*>(pIM->getElementFromId("ui:interface:sheet_help"+toString(i)));
|
|
// if the window exist, insert
|
|
if(group)
|
|
{
|
|
CInfoWindow newInfo;
|
|
newInfo.Window= group->getId();
|
|
newInfo.KeepButton= group->getId() + ":header_opened:keep";
|
|
_InfoWindows.push_back(newInfo);
|
|
}
|
|
else
|
|
break;
|
|
}
|
|
// add observers for the update of phrase help texts (depends of weight of equipped weapons)
|
|
for (uint i = 0; i < MAX_HANDINV_ENTRIES; ++i)
|
|
{
|
|
CCDBNodeLeaf *pNodeLeaf = pIM->getDbProp(std::string(LOCAL_INVENTORY) + ":HAND:" + toString(i), false);
|
|
if(pNodeLeaf)
|
|
{
|
|
ICDBNode::CTextId textId;
|
|
pNodeLeaf->addObserver(&_FittedWeaponWeightObserver, textId);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
// ***************************************************************************
|
|
void CInterfaceHelp::release()
|
|
{
|
|
if (_InfoWindowInit)
|
|
{
|
|
_InfoWindowInit = false;
|
|
_InfoWindows.clear();
|
|
}
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
// add observers for the update of phrase help texts (depends of weight of equipped weapons)
|
|
for (uint i = 0; i < MAX_HANDINV_ENTRIES; ++i)
|
|
{
|
|
CCDBNodeLeaf *pNodeLeaf = pIM->getDbProp(std::string(LOCAL_INVENTORY) + ":HAND:" + toString(i), false);
|
|
if(pNodeLeaf)
|
|
{
|
|
ICDBNode::CTextId textId;
|
|
pNodeLeaf->removeObserver(&_FittedWeaponWeightObserver, textId);
|
|
}
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInterfaceHelp::CFittedWeaponWeightObserver::update(ICDBNode* node)
|
|
{
|
|
CInterfaceHelp::updateWindowSPhraseTexts();
|
|
}
|
|
|
|
// ***************************************************************************
|
|
CInterfaceGroup *CInterfaceHelp::activateNextWindow(CDBCtrlSheet *elt, sint forceKeepWindow)
|
|
{
|
|
CInterfaceManager *pIM = CInterfaceManager::getInstance();
|
|
|
|
// update WindowList if possible
|
|
initWindows();
|
|
sint maxHelpWindow= (sint)_InfoWindows.size();
|
|
|
|
|
|
// Update Active window list
|
|
uint i;
|
|
for(i=0;i<_ActiveWindows.size();)
|
|
{
|
|
CInterfaceGroup *group= _InfoWindows[_ActiveWindows[i]].Window;
|
|
// if the window has been closed, remove it from list
|
|
if(!group->getActive())
|
|
{
|
|
_ActiveWindows.erase(_ActiveWindows.begin()+i);
|
|
}
|
|
else
|
|
i++;
|
|
}
|
|
|
|
// If an active window get the same object, abort, but make it top.
|
|
for(i=0;i<_ActiveWindows.size();i++)
|
|
{
|
|
CInterfaceGroup *group= _InfoWindows[_ActiveWindows[i]].Window;
|
|
CDBCtrlSheet *ctrlSrc= elt;
|
|
// get the ctrl sheet in this help window.
|
|
CDBCtrlSheet *ctrlDst= dynamic_cast<CDBCtrlSheet*>(group->getCtrl(":ctrl_slot"));
|
|
if(ctrlDst && ctrlSrc)
|
|
{
|
|
// if same Aspect
|
|
if( ctrlSrc->sameAspect(ctrlDst) )
|
|
{
|
|
bool ok= true;
|
|
|
|
// for items, must also test if they have the same itemSlotId, cause relies also on "ItemInfo system"
|
|
if(elt->getType() == CCtrlSheetInfo::SheetType_Item)
|
|
{
|
|
CDBCtrlSheet *realCtrlDst= _InfoWindows[_ActiveWindows[i]].CtrlSheet;
|
|
if(!realCtrlDst)
|
|
ok= false;
|
|
else if( getInventory().getItemSlotId(ctrlSrc) != getInventory().getItemSlotId(realCtrlDst) )
|
|
ok= false;
|
|
}
|
|
|
|
// if success to find same element
|
|
if(ok)
|
|
{
|
|
// then don't neet to open a new window, but make the older top.
|
|
pIM->setTopWindow(group);
|
|
return NULL;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// If some free window possible, search which to take
|
|
sint newIndexWindow= -1;
|
|
bool mustPlace= true;
|
|
bool mustAddToActiveWindows= true;
|
|
// if an active window is not in KeepMode, get it.
|
|
for(i=0;i<_ActiveWindows.size();i++)
|
|
{
|
|
// must also test forceKeep for special Action Help wich open Brick Help
|
|
if(!_InfoWindows[_ActiveWindows[i]].KeepMode && forceKeepWindow!=(sint)_ActiveWindows[i])
|
|
{
|
|
newIndexWindow= _ActiveWindows[i];
|
|
mustPlace= false;
|
|
mustAddToActiveWindows= false;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// If not found get new closed one.
|
|
if(newIndexWindow==-1)
|
|
{
|
|
if(_ActiveWindows.size() < (uint)maxHelpWindow)
|
|
{
|
|
// flag each active window
|
|
static std::vector<bool> windowActive;
|
|
windowActive.clear();
|
|
windowActive.resize(maxHelpWindow, false);
|
|
for(i=0;i<_ActiveWindows.size();i++)
|
|
{
|
|
windowActive[_ActiveWindows[i]]= true;
|
|
}
|
|
|
|
// Search the first window closed, and that is in KeepMode (if possible)
|
|
sint newIndexWindowKeep= -1;
|
|
for(i=0;i<(uint)maxHelpWindow;i++)
|
|
{
|
|
if(!windowActive[i])
|
|
{
|
|
// Keep the first window not active
|
|
if(newIndexWindow==-1)
|
|
newIndexWindow= i;
|
|
// Keep the first window not active, that is in KeepMode
|
|
if(_InfoWindows[i].KeepMode && newIndexWindowKeep==-1)
|
|
newIndexWindowKeep= i;
|
|
}
|
|
}
|
|
// Fail?
|
|
if( newIndexWindow==-1 )
|
|
return NULL;
|
|
// success to take a "keep" one?
|
|
if(newIndexWindowKeep!=-1)
|
|
newIndexWindow= newIndexWindowKeep;
|
|
}
|
|
else
|
|
{
|
|
// All the info window are opened (and should not be in KeepMode....), take the last recently opened.
|
|
newIndexWindow= _ActiveWindows.front();
|
|
// free space
|
|
_ActiveWindows.pop_front();
|
|
}
|
|
}
|
|
|
|
// get the next window
|
|
CInterfaceGroup *group= _InfoWindows[newIndexWindow].Window;
|
|
nlassert(group);
|
|
|
|
// activate it, set top, copy item watched
|
|
group->setActive(true);
|
|
pIM->setTopWindow(group);
|
|
_InfoWindows[newIndexWindow].CtrlSheet= elt;
|
|
// insert in list
|
|
if(mustAddToActiveWindows)
|
|
_ActiveWindows.push_back(newIndexWindow);
|
|
|
|
// If Item or mission, must add a callBack when itemInfo (or missionInfo) OK.
|
|
if (elt)
|
|
{
|
|
if( elt->getType() == CCtrlSheetInfo::SheetType_Item )
|
|
{
|
|
_InfoWindows[newIndexWindow].ItemSheet= elt->getSheetId();
|
|
_InfoWindows[newIndexWindow].ItemSlotId= getInventory().getItemSlotId(elt);
|
|
|
|
// Add the waiter only if really needed (not for raw materials)
|
|
const CItemSheet *itemSheet= elt->asItemSheet();
|
|
if(itemSheet && itemSheet->Family != ITEMFAMILY::RAW_MATERIAL )
|
|
getInventory().addItemInfoWaiter(&_InfoWindows[newIndexWindow]);
|
|
}
|
|
else if ( elt->getType() == CCtrlSheetInfo::SheetType_Mission )
|
|
{
|
|
_InfoWindows[newIndexWindow].MissionSlotId = (uint16)elt->getIndexInDB();
|
|
CBotChatManager::getInstance()->addMissionInfoWaiter(&_InfoWindows[newIndexWindow]);
|
|
}
|
|
}
|
|
|
|
// recompute at next pass
|
|
group->invalidateCoords();
|
|
|
|
// Hide some components.
|
|
// Hide the creator name
|
|
CSheetHelpSetup setup;
|
|
setup.setupDefaultIDs();
|
|
setup.SrcSheet = elt;
|
|
setup.HelpWindow = group;
|
|
setupCreatorName(setup);
|
|
|
|
// Hide elements by defaults
|
|
resetSheetHelp(setup);
|
|
|
|
return group;
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CInterfaceHelp::removeWaiterItemInfo(uint i)
|
|
{
|
|
if(i<_InfoWindows.size())
|
|
getInventory().removeItemInfoWaiter(&_InfoWindows[i]);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInterfaceHelp::removeWaiterMissionInfo(uint i)
|
|
{
|
|
if(i<_InfoWindows.size())
|
|
CBotChatManager::getInstance()->removeMissionInfoWaiter(&_InfoWindows[i]);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInterfaceHelp::changeKeepMode(uint i)
|
|
{
|
|
if(i<_InfoWindows.size())
|
|
{
|
|
_InfoWindows[i].KeepMode= !_InfoWindows[i].KeepMode;
|
|
bool state= _InfoWindows[i].KeepMode;
|
|
if(_InfoWindows[i].KeepButton)
|
|
_InfoWindows[i].KeepButton->setPushed(state);
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInterfaceHelp::setKeepMode(uint i, bool state)
|
|
{
|
|
if(i<_InfoWindows.size())
|
|
{
|
|
_InfoWindows[i].KeepMode= state;
|
|
if(_InfoWindows[i].KeepButton)
|
|
_InfoWindows[i].KeepButton->setPushed(state);
|
|
}
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CInterfaceHelp::closeAll()
|
|
{
|
|
// update WindowList if possible
|
|
initWindows();
|
|
sint maxHelpWindow= (sint)_InfoWindows.size();
|
|
|
|
_ActiveWindows.clear();
|
|
// For all windows
|
|
for(uint i=0;i<(uint)maxHelpWindow;i++)
|
|
{
|
|
_InfoWindows[i].Window->setActive(false);
|
|
}
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void CInterfaceHelp::resetWindowPos(sint y)
|
|
{
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
|
|
// update WindowList if possible
|
|
initWindows();
|
|
sint maxHelpWindow= (sint)_InfoWindows.size();
|
|
|
|
uint32 w, h;
|
|
pIM->getViewRenderer().getScreenSize(w,h);
|
|
|
|
// For all windows, reset pos
|
|
for(uint i=0;i<(uint)maxHelpWindow;i++)
|
|
{
|
|
_InfoWindows[i].Window->setX(i*_InfoWindows[i].Window->getW(false));
|
|
_InfoWindows[i].Window->setY((sint)h+y);
|
|
_InfoWindows[i].Window->invalidateCoords();
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInterfaceHelp::serialInfoWindows(NLMISC::IStream &f)
|
|
{
|
|
f.serialVersion(0);
|
|
|
|
vector<CInfoWindowSave> infoWindowSave;
|
|
if(f.isReading())
|
|
{
|
|
// Setup pos by default (for case where number of windows differ)
|
|
resetWindowPos(-100);
|
|
|
|
f.serialCont(infoWindowSave);
|
|
uint minSize= (uint)min(infoWindowSave.size(), _InfoWindows.size());
|
|
for(uint i=0;i<minSize;i++)
|
|
{
|
|
_InfoWindows[i].Window->setX(infoWindowSave[i].X);
|
|
_InfoWindows[i].Window->setY(infoWindowSave[i].Y);
|
|
_InfoWindows[i].Window->invalidateCoords();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
infoWindowSave.resize(_InfoWindows.size());
|
|
for(uint i=0;i<infoWindowSave.size();i++)
|
|
{
|
|
infoWindowSave[i].X= _InfoWindows[i].Window->getX();
|
|
infoWindowSave[i].Y= _InfoWindows[i].Window->getY();
|
|
}
|
|
f.serialCont(infoWindowSave);
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInterfaceHelp::debugOpenedInfoWindows()
|
|
{
|
|
for(uint i=0;i<_ActiveWindows.size();i++)
|
|
{
|
|
uint index= _ActiveWindows[i];
|
|
uint sheetId= 0;
|
|
if(_InfoWindows[index].CtrlSheet)
|
|
sheetId= _InfoWindows[index].CtrlSheet->getSheetId();
|
|
if(sheetId)
|
|
{
|
|
nlinfo("debugInfoWindow: %s", CSheetId(sheetId).toString().c_str() );
|
|
}
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CInterfaceHelp::updateWindowSPhraseTexts()
|
|
{
|
|
CSPhraseManager *pPM= CSPhraseManager::getInstance();
|
|
|
|
for(uint i=0;i<_ActiveWindows.size();i++)
|
|
{
|
|
uint index= _ActiveWindows[i];
|
|
CDBCtrlSheet *ctrl= _InfoWindows[index].CtrlSheet;
|
|
CInterfaceGroup *group= _InfoWindows[index].Window;
|
|
if(group && ctrl && (ctrl->isSPhraseId() || ctrl->isSPhrase()) )
|
|
{
|
|
CSheetHelpSetup setup;
|
|
setup.setupDefaultIDs();
|
|
setup.HelpWindow = group;
|
|
setup.SrcSheet = ctrl;
|
|
setup.DestSheet = dynamic_cast<CDBCtrlSheet*>(group->getCtrl("ctrl_slot"));
|
|
|
|
if(ctrl->isSPhraseId())
|
|
{
|
|
// reset up the complete window
|
|
setupSabrinaPhraseHelp(setup, pPM->getPhrase(ctrl->getSPhraseId()), 0);
|
|
}
|
|
else if(ctrl->isSPhrase())
|
|
{
|
|
CSPhraseCom phrase;
|
|
uint32 phraseSheetId= setup.SrcSheet->getSheetId();
|
|
pPM->buildPhraseFromSheet(phrase, phraseSheetId);
|
|
setupSabrinaPhraseHelp(setup, phrase, phraseSheetId);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
/** Close all the helps.
|
|
*/
|
|
class CHandlerCloseHelp : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase *pCaller, const std::string &sParams)
|
|
{
|
|
CInterfaceHelp::closeAll();
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerCloseHelp, "close_help");
|
|
|
|
|
|
// ***************************************************************************
|
|
/** Build the help window for an item and open it.
|
|
*/
|
|
class CHandlerOpenItemHelp : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase *pCaller, const std::string &sParams)
|
|
{
|
|
CDBCtrlSheet *cs = dynamic_cast<CDBCtrlSheet*>(pCaller);
|
|
if (cs != NULL && cs->getSheetId()!=0 )
|
|
{
|
|
// get the forceKeep param (for Building info)
|
|
sint forceKeepWindow= -1;
|
|
string forceKeepWindowStr= getParam(sParams, "force_keep");
|
|
if(!forceKeepWindowStr.empty())
|
|
fromString(forceKeepWindowStr, forceKeepWindow);
|
|
|
|
// open the next window
|
|
CInterfaceGroup *group = CInterfaceHelp::activateNextWindow(cs, forceKeepWindow);
|
|
if (!group) return;
|
|
CSheetHelpSetup setup;
|
|
setup.setupDefaultIDs();
|
|
setup.HelpWindow = group;
|
|
setup.SrcSheet = cs;
|
|
setup.DestSheet = dynamic_cast<CDBCtrlSheet*>(group->getElement(group->getId()+":content:ctrl_slot"));
|
|
setupItemHelp(setup);
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerOpenItemHelp, "open_item_help");
|
|
|
|
|
|
// ***************************************************************************
|
|
/** Build the help window for a pact and open it.
|
|
*/
|
|
class CHandlerOpenPactHelp : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase *pCaller, const std::string &sParams)
|
|
{
|
|
CDBCtrlSheet *cs = dynamic_cast<CDBCtrlSheet*>(pCaller);
|
|
if (cs != NULL && cs->getSheetId()!=0 )
|
|
{
|
|
CInterfaceGroup *group = CInterfaceHelp::activateNextWindow(cs);
|
|
if (!group) return;
|
|
CSheetHelpSetup setup;
|
|
setup.setupDefaultIDs();
|
|
setup.HelpWindow = group;
|
|
setup.SrcSheet = cs;
|
|
setup.DestSheet = dynamic_cast<CDBCtrlSheet*>(group->getElement(group->getId()+":content:ctrl_slot"));
|
|
setupPactHelp(setup);
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerOpenPactHelp, "open_pact_help");
|
|
|
|
|
|
// ***************************************************************************
|
|
/** Build the help window for a targeted entity to know what the title means
|
|
*/
|
|
class CHandlerOpenTitleHelp : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase *pCaller, const std::string &sParams)
|
|
{
|
|
// display web profile if necessary
|
|
if (getParam(sParams, "from") == "target")
|
|
{
|
|
// Require info on the target
|
|
CEntityCL *selection = EntitiesMngr.entity(UserEntity->selection());
|
|
if (selection == NULL) return;
|
|
//if(selection->isNPC())
|
|
{
|
|
ucstring name = selection->getEntityName();
|
|
if(name.empty())
|
|
{
|
|
// try to get the name from the string manager (for npc)
|
|
uint32 nDBid = selection->getNameId();
|
|
if (nDBid != 0)
|
|
{
|
|
STRING_MANAGER::CStringManagerClient *pSMC = STRING_MANAGER::CStringManagerClient::instance();
|
|
pSMC->getString (nDBid, name);
|
|
ucstring copyName = name;
|
|
name = CEntityCL::removeTitleAndShardFromName(name);
|
|
if (name.empty())
|
|
{
|
|
CCharacterCL *pChar = dynamic_cast<CCharacterCL*>(selection);
|
|
bool woman = false;
|
|
if (pChar != NULL)
|
|
woman = pChar->getGender() == GSGENDER::female;
|
|
|
|
// extract the replacement id
|
|
string strNewTitle = CEntityCL::getTitleFromName(copyName);
|
|
|
|
// retrieve the translated string
|
|
if (!strNewTitle.empty())
|
|
name = STRING_MANAGER::CStringManagerClient::getTitleLocalizedName(strNewTitle, woman);
|
|
else
|
|
name.clear();
|
|
}
|
|
}
|
|
}
|
|
if(!name.empty())
|
|
CInterfaceManager::getInstance()->runActionHandler("show_hide", pCaller, "profile|pname="+name.toUtf8()+"|ptype="+toString((int)selection->Type));
|
|
return;
|
|
}
|
|
}
|
|
|
|
CInterfaceGroup *group = CInterfaceHelp::activateNextWindow(NULL);
|
|
if (!group) return;
|
|
|
|
// prepare the help window
|
|
CSheetHelpSetup setup;
|
|
setup.setupDefaultIDs();
|
|
setup.HelpWindow = group;
|
|
setup.DestSheet = dynamic_cast<CDBCtrlSheet*>(group->getElement(group->getId()+":content:ctrl_slot"));
|
|
|
|
// Get name and title
|
|
// ------------------
|
|
ucstring name;
|
|
ucstring title;
|
|
bool reservedTitle = false;
|
|
string sFrom = getParam(sParams, "from");
|
|
if (sFrom == "target")
|
|
{
|
|
// Require info on the target
|
|
CEntityCL *selection = EntitiesMngr.entity(UserEntity->selection());
|
|
if (selection == NULL) return;
|
|
name = CEntityCL::removeTitleAndShardFromName(selection->getEntityName());
|
|
title = selection->getTitle();
|
|
reservedTitle = selection->hasReservedTitle();
|
|
}
|
|
else if (sFrom == "user")
|
|
{
|
|
// Require info on the local_player
|
|
name = CEntityCL::removeTitleAndShardFromName(UserEntity->getEntityName());
|
|
title = UserEntity->getTitle();
|
|
reservedTitle = UserEntity->hasReservedTitle();
|
|
}
|
|
else return;
|
|
|
|
// Get Title info (bricks and skills needed)
|
|
// -----------------------------------------
|
|
CUnblockTitlesSheet *pUTS = dynamic_cast<CUnblockTitlesSheet*>(SheetMngr.get(CSheetId("unblock.titles")));
|
|
if (pUTS == NULL)
|
|
{
|
|
nlwarning("cant find unblock.titles");
|
|
return;
|
|
}
|
|
|
|
// get title id number from the title id
|
|
uint titleIDnb;
|
|
for (titleIDnb = 0; titleIDnb < CHARACTER_TITLE::NB_CHARACTER_TITLE; ++titleIDnb)
|
|
{
|
|
bool women = UserEntity && UserEntity->getGender()==GSGENDER::female;
|
|
if (CStringManagerClient::getTitleLocalizedName(CHARACTER_TITLE::toString((CHARACTER_TITLE::ECharacterTitle)titleIDnb),women) == title)
|
|
break;
|
|
}
|
|
|
|
// Retrieve all infos about the title
|
|
const CUnblockTitlesSheet::STitleUnblock *pTU = NULL;
|
|
if (titleIDnb != CHARACTER_TITLE::NB_CHARACTER_TITLE)
|
|
pTU = &pUTS->TitlesUnblock[titleIDnb];
|
|
|
|
|
|
// Display all infos found
|
|
// -----------------------
|
|
ucstring titleText = CI18N::get("uihelpTitleFormat");
|
|
strFindReplace(titleText, "%name", name.toString());
|
|
|
|
// Display title
|
|
ucstring::size_type p1 = title.find('(');
|
|
if (p1 != ucstring::npos)
|
|
{
|
|
ucstring::size_type p2 = title.find(')', p1+1);
|
|
if (p2 != ucstring::npos)
|
|
title = title.substr(p1+1, p2-p1-1);
|
|
}
|
|
strFindReplace(titleText, "%title", title);
|
|
|
|
// Display all skills needed to obtain this title
|
|
ucstring sSkillsNeeded;
|
|
if (!title.empty() && pTU == NULL)
|
|
sSkillsNeeded = CI18N::get("uiTitleCantObtain");
|
|
|
|
if (pTU != NULL)
|
|
{
|
|
sSkillsNeeded = CI18N::get("uiTitleSkillHeader");
|
|
if (pTU->SkillsNeeded.size() == 0 || reservedTitle)
|
|
{
|
|
sSkillsNeeded += CI18N::get("uiTitleSkillNoNeed");
|
|
}
|
|
else
|
|
{
|
|
for (uint i = 0; i < pTU->SkillsNeeded.size(); ++i)
|
|
{
|
|
for (uint j = 0; j < pTU->SkillsNeeded[i].size(); ++j)
|
|
{
|
|
uint skillNb;
|
|
for (skillNb = 0; skillNb < SKILLS::NUM_SKILLS; ++skillNb)
|
|
if (pTU->SkillsNeeded[i][j] == SKILLS::toString((SKILLS::ESkills)skillNb))
|
|
break;
|
|
|
|
if (skillNb != SKILLS::NUM_SKILLS)
|
|
{
|
|
sSkillsNeeded += CStringManagerClient::getSkillLocalizedName((SKILLS::ESkills)skillNb);
|
|
sSkillsNeeded += " (" + toString(pTU->SkillsLevelNeeded[i][j]) + ")";
|
|
sSkillsNeeded += "\n";
|
|
}
|
|
else
|
|
{
|
|
nlwarning("cant find skill : %s", pTU->SkillsNeeded[i][j].c_str());
|
|
}
|
|
}
|
|
if( i != pTU->SkillsNeeded.size()-1 )
|
|
{
|
|
sSkillsNeeded += CI18N::get("uiTitleSkillOr");
|
|
sSkillsNeeded +="\n";
|
|
}
|
|
}
|
|
}
|
|
}
|
|
strFindReplace(titleText, "%skills", sSkillsNeeded);
|
|
|
|
// Display all bricks needed to obtain this title
|
|
ucstring sBricksNeeded;
|
|
if (pTU != NULL)
|
|
{
|
|
sBricksNeeded = CI18N::get("uiTitleBrickHeader");
|
|
if (pTU->BricksNeeded.size() == 0 || reservedTitle)
|
|
{
|
|
sBricksNeeded += CI18N::get("uiTitleBrickNoNeed");
|
|
}
|
|
else
|
|
{
|
|
// Get the list of "requirement bricks"
|
|
string listID = group->getId() + setup.PrefixForExtra + INFO_LIST_BRICK_REQUIREMENT;
|
|
IListSheetBase *listBrick= dynamic_cast<IListSheetBase*>(group->getElement(listID));
|
|
if(listBrick)
|
|
{
|
|
CSPhraseCom phrase;
|
|
phrase.Bricks = pTU->BricksNeeded;
|
|
fillSabrinaPhraseListBrick(phrase, listBrick);
|
|
}
|
|
}
|
|
}
|
|
strFindReplace(titleText, "%bricks", sBricksNeeded);
|
|
|
|
// setup the text
|
|
setHelpText(setup, titleText);
|
|
setupHelpTitle(setup.HelpWindow, CI18N::get("uihelpTitleInfo"));
|
|
|
|
// hide the ctrl sheet
|
|
setHelpCtrlSheet(setup, 0);
|
|
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerOpenTitleHelp, "open_title_help");
|
|
|
|
// ***************************************************************************
|
|
/** Build the help for a skill that is to trade
|
|
*/
|
|
class CHandlerOpenSkillToTradeHelp : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase *pCaller, const std::string &sParams)
|
|
{
|
|
CDBCtrlSheet *cs = dynamic_cast<CDBCtrlSheet*>(pCaller);
|
|
if (cs != NULL)
|
|
{
|
|
CInterfaceGroup *group = CInterfaceHelp::activateNextWindow(cs);
|
|
if (!group) return;
|
|
CSheetHelpSetup setup;
|
|
setup.setupDefaultIDs();
|
|
setup.HelpWindow = group;
|
|
setup.SrcSheet = cs;
|
|
setup.DestSheet = dynamic_cast<CDBCtrlSheet*>(group->getElement(group->getId()+":content:ctrl_slot"));
|
|
setupSkillToTradeHelp(setup);
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerOpenSkillToTradeHelp, "open_skill_to_trade_help");
|
|
|
|
// ***************************************************************************
|
|
/** Build the help window for a pact/item/brick and open it.
|
|
*/
|
|
class CHandlerOpenHelpAuto : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase *pCaller, const std::string &sParams)
|
|
{
|
|
CDBCtrlSheet *cs = dynamic_cast<CDBCtrlSheet*>(pCaller);
|
|
if (!cs)
|
|
{
|
|
nlwarning("<CHandlerOpenHelpAuto::execute> no caller sheet found.");
|
|
return;
|
|
}
|
|
if (cs->getSheetId()!=0 )
|
|
{
|
|
// create group to display help
|
|
CInterfaceGroup *group = CInterfaceHelp::activateNextWindow(cs);
|
|
if (!group) return;
|
|
// setup the item.
|
|
CSheetHelpSetup setup;
|
|
setup.setupDefaultIDs();
|
|
setup.HelpWindow = group;
|
|
setup.SrcSheet = cs;
|
|
setup.DestSheet = dynamic_cast<CDBCtrlSheet*>(group->getElement(group->getId()+":content:ctrl_slot"));
|
|
setupSheetHelp(setup);
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerOpenHelpAuto, "open_help_auto");
|
|
|
|
// ***************************************************************************
|
|
/** Browse an URL into a CGroupHTML
|
|
*/
|
|
class CHandlerBrowse : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase *pCaller, const std::string &sParams)
|
|
{
|
|
string container = getParam (sParams, "name");
|
|
CInterfaceElement *element = CInterfaceManager::getInstance()->getElementFromId(container);
|
|
CInterfaceGroup *elementGroup = dynamic_cast<CInterfaceGroup*>(element);
|
|
|
|
string urls = getParam (sParams, "url");
|
|
|
|
bool show = getParam (sParams, "show") != "0";
|
|
|
|
bool localizePage= getParam (sParams, "localize")=="1";
|
|
|
|
// Action Handler?
|
|
if (strncmp (urls.c_str(), "ah:", 3) == 0)
|
|
{
|
|
// Find next action handler
|
|
string::size_type start = 3;
|
|
string::size_type end = urls.find ("&&", start);
|
|
if (end == string::npos)
|
|
end = urls.size();
|
|
while (start < end)
|
|
{
|
|
// Extract the url
|
|
string url = urls.substr(start, end-start);
|
|
|
|
// Run an action handler
|
|
string::size_type index = url.find_first_of("&");
|
|
if (index == string::npos)
|
|
index = url.size();
|
|
string action = url.substr(0, index);
|
|
string params;
|
|
if (index<url.size())
|
|
params = url.substr(index+1, url.size()-index-1);
|
|
|
|
// Replace '&'
|
|
if(action=="command")
|
|
{
|
|
// Replace : by ' '
|
|
index = 0;
|
|
while ((index = params.find_first_of("&", index)) != string::npos)
|
|
{
|
|
params[index] = ' ';
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Replace : by '|'
|
|
index = 0;
|
|
while ((index = params.find_first_of("&", index)) != string::npos)
|
|
{
|
|
params[index] = '|';
|
|
}
|
|
}
|
|
|
|
// Replace %HH encoding with ASCII values (AFTER '&' replacing, to possibly have reals '&')
|
|
for(uint i=0;i<params.size();i++)
|
|
{
|
|
if(params[i]=='%' && i<params.size()-2)
|
|
{
|
|
if(isxdigit(params[i+1]) && isxdigit(params[i+2]))
|
|
{
|
|
// read value from heax decimal
|
|
uint8 val= 0;
|
|
params[i+1]= tolower(params[i+1]);
|
|
params[i+2]= tolower(params[i+2]);
|
|
if(isdigit(params[i+1])) val= params[i+1]-'0';
|
|
else val= 10+ params[i+1]-'a';
|
|
val*=16;
|
|
if(isdigit(params[i+2])) val+= params[i+2]-'0';
|
|
else val+= 10+ params[i+2]-'a';
|
|
|
|
// write
|
|
params[i]= val;
|
|
// erase heax value
|
|
params.erase(i+1, 2);
|
|
}
|
|
}
|
|
}
|
|
|
|
// go. NB: the action handler himself may translate params from utf8
|
|
CInterfaceManager::getInstance()->runActionHandler(action, elementGroup, params);
|
|
|
|
// Next name
|
|
start = end+2;
|
|
end = urls.find ("&&", start);
|
|
if (end == string::npos)
|
|
end = urls.size();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Get the group HTML
|
|
if (element)
|
|
{
|
|
// Group HTML ?
|
|
CGroupHTML *groupHtml = dynamic_cast<CGroupHTML*>(element);
|
|
if (groupHtml)
|
|
{
|
|
if (show)
|
|
{
|
|
// Look for a parent container
|
|
CInterfaceGroup *parent = groupHtml->getParent();
|
|
while (parent)
|
|
{
|
|
if (parent->getParent() && (parent->getParent()->getId() == "ui:interface"))
|
|
{
|
|
parent->setActive(true);
|
|
break;
|
|
}
|
|
|
|
parent = parent->getParent();
|
|
}
|
|
}
|
|
|
|
// localize if wanted
|
|
if(localizePage)
|
|
strFindReplace(urls, "_wk.", string("_")+ClientCfg.getHtmlLanguageCode()+"." );
|
|
|
|
// Browse the url
|
|
groupHtml->browse(urls.c_str());
|
|
// Set top of the page
|
|
CCtrlScroll *pScroll = groupHtml->getScrollBar();
|
|
if (pScroll != NULL)
|
|
pScroll->moveTrackY(10000);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerBrowse, "browse");
|
|
|
|
// ***************************************************************************
|
|
/** Browse Undo into a CGroupHTML
|
|
*/
|
|
class CHandlerBrowseUndo : public IActionHandler
|
|
{
|
|
public:
|
|
void execute (CCtrlBase *pCaller, const std::string &sParams)
|
|
{
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
string container = getParam (sParams, "name");
|
|
CGroupHTML *groupHtml = dynamic_cast<CGroupHTML*>(pIM->getElementFromId(container));
|
|
if (groupHtml)
|
|
{
|
|
groupHtml->browseUndo();
|
|
}
|
|
};
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerBrowseUndo, "browse_undo");
|
|
|
|
// ***************************************************************************
|
|
/** Browse Redo into a CGroupHTML
|
|
*/
|
|
class CHandlerBrowseRedo : public IActionHandler
|
|
{
|
|
public:
|
|
void execute (CCtrlBase *pCaller, const std::string &sParams)
|
|
{
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
string container = getParam (sParams, "name");
|
|
CGroupHTML *groupHtml = dynamic_cast<CGroupHTML*>(pIM->getElementFromId(container));
|
|
if (groupHtml)
|
|
{
|
|
groupHtml->browseRedo();
|
|
}
|
|
};
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerBrowseRedo, "browse_redo");
|
|
|
|
// ***************************************************************************
|
|
/** Browse Redo into a CGroupHTML
|
|
*/
|
|
class CHandlerBrowseRefresh : public IActionHandler
|
|
{
|
|
public:
|
|
void execute (CCtrlBase *pCaller, const std::string &sParams)
|
|
{
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
string container = getParam (sParams, "name");
|
|
CGroupHTML *groupHtml = dynamic_cast<CGroupHTML*>(pIM->getElementFromId(container));
|
|
if (groupHtml)
|
|
{
|
|
groupHtml->refresh();
|
|
}
|
|
};
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerBrowseRefresh, "browse_refresh");
|
|
|
|
|
|
|
|
// ***************************************************************************
|
|
/** Build the help window for a pact/item/brick and open it.
|
|
*/
|
|
class CHandlerHTMLSubmitForm : public IActionHandler
|
|
{
|
|
void execute (CCtrlBase *pCaller, const std::string &sParams)
|
|
{
|
|
string container = getParam (sParams, "name");
|
|
|
|
uint form;
|
|
fromString(getParam (sParams, "form"), form);
|
|
|
|
string submit_button = getParam (sParams, "submit_button");
|
|
|
|
CInterfaceElement *element = CInterfaceManager::getInstance()->getElementFromId(container);
|
|
{
|
|
// Group HTML ?
|
|
CGroupHTML *groupHtml = dynamic_cast<CGroupHTML*>(element);
|
|
if (groupHtml)
|
|
{
|
|
// Submit the form the url
|
|
groupHtml->submitForm (form, submit_button.c_str ());
|
|
}
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerHTMLSubmitForm, "html_submit_form");
|
|
|
|
//////////////////////////////////
|
|
// STATIC FUNCTIONS DEFINITIONS //
|
|
//////////////////////////////////
|
|
|
|
// ***************************************************************************
|
|
static void setupHelpPage(CInterfaceGroup *window, const string &url)
|
|
{
|
|
// must be a container
|
|
CGroupContainer *gc= dynamic_cast<CGroupContainer*>(window);
|
|
if(gc)
|
|
gc->setHelpPage(url);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void setHelpText(CSheetHelpSetup &setup, const ucstring &text)
|
|
{
|
|
ucstring copyStr= text;
|
|
// remove trailing \n
|
|
while(!copyStr.empty() && copyStr[copyStr.size()-1]=='\n')
|
|
{
|
|
copyStr.resize(copyStr.size()-1);
|
|
}
|
|
|
|
if (!setup.HelpWindow) return;
|
|
CViewText *viewText= dynamic_cast<CViewText *>(setup.HelpWindow->getView(setup.ViewText));
|
|
if(viewText)
|
|
{
|
|
viewText->setTextFormatTaged(copyStr);
|
|
}
|
|
CInterfaceGroup *viewTextGroup = setup.HelpWindow->getGroup(setup.ScrollTextGroup);
|
|
if (viewTextGroup) viewTextGroup->setActive(true);
|
|
if (!setup.ScrollTextIdGroup.empty())
|
|
{
|
|
viewTextGroup = setup.HelpWindow->getGroup(setup.ScrollTextIdGroup);
|
|
if (viewTextGroup) viewTextGroup->setActive(false);
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void setHelpCtrlSheet(CSheetHelpSetup &setup, uint32 sheetId)
|
|
{
|
|
if(setup.DestSheet)
|
|
setup.DestSheet->setSheetId (sheetId);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void setHelpTextID(CSheetHelpSetup &setup, sint32 id)
|
|
{
|
|
if (!setup.HelpWindow) return;
|
|
CViewTextID *viewTextID = dynamic_cast<CViewTextID *>(setup.HelpWindow->getView(setup.ViewTextID));
|
|
if(viewTextID)
|
|
{
|
|
viewTextID->setTextId(id);
|
|
}
|
|
if (!setup.ScrollTextGroup.empty())
|
|
{
|
|
CInterfaceGroup *viewTextGroup = setup.HelpWindow->getGroup(setup.ScrollTextGroup);
|
|
if (viewTextGroup) viewTextGroup->setActive(false);
|
|
}
|
|
CInterfaceGroup *viewTextGroup = setup.HelpWindow->getGroup(setup.ScrollTextIdGroup);
|
|
if (viewTextGroup) viewTextGroup->setActive(true);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
static void setupHelpTitle(CInterfaceGroup *group, const ucstring &title)
|
|
{
|
|
CGroupContainer *pGC= dynamic_cast<CGroupContainer*>(group);
|
|
if(!group)
|
|
return;
|
|
pGC->setUCTitle(title);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
static void setupSkillToTradeHelp(CSheetHelpSetup &setup)
|
|
{
|
|
if (!setup.HelpWindow) return;
|
|
|
|
// get the calling item
|
|
if (!setup.SrcSheet)
|
|
{
|
|
nlwarning("<CHandlerOpenBrickHelp::execute> no caller sheet found.");
|
|
return;
|
|
}
|
|
|
|
SKILLS::ESkills skill = setup.SrcSheet->getSkill();
|
|
if (skill >= SKILLS::unknown)
|
|
{
|
|
nlwarning("bad skill");
|
|
return;
|
|
}
|
|
|
|
|
|
if(setup.DestSheet)
|
|
{
|
|
setup.SrcSheet->copyAspect(setup.DestSheet);
|
|
}
|
|
|
|
ucstring skillText;
|
|
|
|
// Name in title
|
|
const ucstring title(CStringManagerClient::getSkillLocalizedName(skill));
|
|
setupHelpTitle(setup.HelpWindow, title);
|
|
|
|
// search all job that have minimum required level for that skill
|
|
// CInterfaceManager *im = CInterfaceManager::getInstance();
|
|
// for (uint career = 0; career < JOBS::NUM_CAREER_DB_SLOTS; ++career)
|
|
// {
|
|
// for (uint job = 0; job < 8; ++job)
|
|
// {
|
|
// std::string dbPath = toString("CHARACTER_INFO:CAREER%d:JOB%d:JOB_CAP", (int) career, (int) job);
|
|
// uint level = (uint) im->getDbProp(dbPath)->getValue32();
|
|
// if (level != 0) // has the player this job ?
|
|
// {
|
|
// // check if level in this job is enough to get the skills
|
|
// // TODO finish it
|
|
// /*
|
|
// JOBS::TJob job = JOBS::getJobFromDBIndex(career, job);
|
|
// if (job != JOBS::Unknown)
|
|
// {
|
|
// }*/
|
|
// }
|
|
// }
|
|
// }
|
|
|
|
// setup skill desc if available.
|
|
const ucstring desc(CStringManagerClient::getSkillLocalizedDescription(skill));
|
|
if( !desc.empty() )
|
|
{
|
|
skillText+= "\n" + desc;
|
|
}
|
|
|
|
setHelpText(setup, skillText);
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
static string toReadableFloat(float val)
|
|
{
|
|
sint iv= sint(val * 10);
|
|
if((iv%10)==0)
|
|
{
|
|
return toString(iv/10);
|
|
}
|
|
else
|
|
{
|
|
return toString("%.1f", val);
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
static string toPercentageText(float val)
|
|
{
|
|
sint iv= sint(val * 1000);
|
|
if((iv%10)==0)
|
|
{
|
|
return toString(iv/10);
|
|
}
|
|
else
|
|
{
|
|
val= float(iv)/10;
|
|
return toString("%.1f", val);
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void getItemDefenseText(CDBCtrlSheet *item, ucstring &itemText)
|
|
{
|
|
// retrieve the current itemInfo
|
|
const CClientItemInfo &itemInfo= getInventory().getItemInfo(getInventory().getItemSlotId(item) );
|
|
|
|
// Parry/Dodge
|
|
strFindReplace(itemText, "%dodge", toString(itemInfo.DodgeModifier) );
|
|
strFindReplace(itemText, "%parry", toString(itemInfo.ParryModifier) );
|
|
|
|
// Display All protections
|
|
strFindReplace( itemText, "%protect", toPercentageText(itemInfo.ProtectionFactor) );
|
|
strFindReplace( itemText, "%p_slash", toString(itemInfo.MaxSlashingProtection) );
|
|
strFindReplace( itemText, "%p_pierce", toString(itemInfo.MaxPiercingProtection) );
|
|
strFindReplace( itemText, "%p_blunt", toString(itemInfo.MaxBluntProtection) );
|
|
}
|
|
|
|
|
|
void getDamageText(CDBCtrlSheet *item, const CItemSheet*pIS, ucstring &itemText, bool displayAsMod)
|
|
{
|
|
// retrieve the current itemInfo
|
|
const CClientItemInfo &itemInfo= getInventory().getItemInfo(getInventory().getItemSlotId(item) );
|
|
|
|
string strMod;
|
|
if(displayAsMod)
|
|
strMod= itemInfo.CurrentDamage>=0?"+":"";
|
|
strFindReplace(itemText, "%dmg", strMod + toString(itemInfo.CurrentDamage));
|
|
if(displayAsMod)
|
|
strMod= itemInfo.MaxDamage>=0?"+":"";
|
|
strFindReplace(itemText, "%max_dmg", strMod + toString(itemInfo.MaxDamage));
|
|
|
|
if(pIS->Family!=ITEMFAMILY::AMMO)
|
|
{
|
|
// Display the Dodge/Parry Modifier
|
|
strFindReplace(itemText, "%dodge", toString(itemInfo.DodgeModifier) );
|
|
strFindReplace(itemText, "%parry", toString(itemInfo.ParryModifier) );
|
|
|
|
// Display the Adversary Dodge/Parry Modifier
|
|
strFindReplace(itemText, "%adv_dodge", toString(itemInfo.AdversaryDodgeModifier) );
|
|
strFindReplace(itemText, "%adv_parry", toString(itemInfo.AdversaryParryModifier) );
|
|
}
|
|
}
|
|
|
|
void getSpeedText(CDBCtrlSheet *item, ucstring &itemText, bool displayAsMod)
|
|
{
|
|
// retrieve the current itemInfo
|
|
const CClientItemInfo &itemInfo= getInventory().getItemInfo(getInventory().getItemSlotId(item) );
|
|
|
|
string strMod;
|
|
if(displayAsMod)
|
|
strMod= itemInfo.HitRate>=0?"+":"";
|
|
strFindReplace(itemText, "%speed", strMod + toReadableFloat(itemInfo.HitRate));
|
|
}
|
|
|
|
void getRangeText(CDBCtrlSheet *item, ucstring &itemText, bool displayAsMod)
|
|
{
|
|
// retrieve the current itemInfo
|
|
const CClientItemInfo &itemInfo= getInventory().getItemInfo(getInventory().getItemSlotId(item) );
|
|
|
|
string strMod;
|
|
if(displayAsMod)
|
|
strMod= itemInfo.Range>=0?"+":"";
|
|
strFindReplace(itemText, "%range", strMod + toReadableFloat(itemInfo.Range/1000.f));
|
|
}
|
|
|
|
void getHPAndSapLoadText(CDBCtrlSheet *item, const CItemSheet*pIS, ucstring &itemText)
|
|
{
|
|
// retrieve the current itemInfo
|
|
const CClientItemInfo &itemInfo= getInventory().getItemInfo(getInventory().getItemSlotId(item) );
|
|
|
|
if(pIS->Family != ITEMFAMILY::RAW_MATERIAL)
|
|
{
|
|
// must find first hpmax! (else replace error)
|
|
strFindReplace(itemText, "%hpmax", toString(itemInfo.HpMax) );
|
|
strFindReplace(itemText, "%hp", toString(itemInfo.Hp) );
|
|
// SapLoad
|
|
strFindReplace(itemText, "%sapmax", toString(itemInfo.SapLoadMax) );
|
|
strFindReplace(itemText, "%sap", toString(itemInfo.SapLoadCurrent) );
|
|
}
|
|
}
|
|
|
|
|
|
void getBuffText(CDBCtrlSheet *item, ucstring &itemText)
|
|
{
|
|
// retrieve the current itemInfo
|
|
const CClientItemInfo &itemInfo= getInventory().getItemInfo(getInventory().getItemSlotId(item) );
|
|
|
|
const string valIds[]={"Hp", "Sap", "Sta", "Focus"};
|
|
sint32 vals[]= {itemInfo.HpBuff, itemInfo.SapBuff, itemInfo.StaBuff, itemInfo.FocusBuff};
|
|
uint numVals= sizeof(vals) / sizeof(vals[0]);
|
|
ucstring bufInfo;
|
|
|
|
// For each buf, append a line if !=0
|
|
for(uint i=0;i<numVals;i++)
|
|
{
|
|
sint32 modifier= vals[i];
|
|
if(modifier!=0)
|
|
{
|
|
ucstring line= CI18N::get( "uihelpItem" + valIds[i] + (modifier>0?"Bonus":"Malus") );
|
|
strFindReplace(line, "%val", toString(modifier) );
|
|
bufInfo+= line;
|
|
}
|
|
}
|
|
|
|
// append a \n before
|
|
if(!bufInfo.empty())
|
|
bufInfo = "\n" + bufInfo + "\n";
|
|
|
|
// Display the buff info.
|
|
strFindReplace(itemText, "%buffs", bufInfo);
|
|
}
|
|
|
|
void getMagicProtection(CDBCtrlSheet *item, ucstring &itemText)
|
|
{
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
// retrieve the current itemInfo
|
|
const CClientItemInfo &itemInfo= getInventory().getItemInfo(getInventory().getItemSlotId(item) );
|
|
|
|
ucstring mProtInfo;
|
|
|
|
// Header (always here, because at least max absorb)
|
|
mProtInfo= CI18N::get("uihelpMagicProtectFormatHeader");
|
|
|
|
// For each protection
|
|
for(uint i=0;i<CClientItemInfo::MaxMagicProtectionByJewel;i++)
|
|
{
|
|
if(itemInfo.MagicProtection[i] != PROTECTION_TYPE::None)
|
|
{
|
|
// Protection info
|
|
ucstring str= CI18N::get("uihelpMagicProtectFormat");
|
|
strFindReplace(str, "%t", CI18N::get("pt"+PROTECTION_TYPE::toString(itemInfo.MagicProtection[i])) );
|
|
strFindReplace(str, "%v", toString(itemInfo.MagicProtectionFactor[i]) );
|
|
mProtInfo+= str;
|
|
}
|
|
}
|
|
|
|
// add Max damage absorbed
|
|
{
|
|
// Mul item quality by a constant
|
|
uint maxAbsorb= item->getQuality();
|
|
CCDBNodeLeaf *nodeFactor= pIM->getDbProp(pIM->getDefine("player_protect_absorbfactor"), false);
|
|
if(nodeFactor)
|
|
maxAbsorb= maxAbsorb*nodeFactor->getValue32()/100;
|
|
|
|
// Add to text
|
|
ucstring str= CI18N::get("uihelpMagicProtectMaxAbsorbFormat");
|
|
strFindReplace(str, "%v", toString(maxAbsorb) );
|
|
mProtInfo+= str;
|
|
}
|
|
|
|
// replace in item info
|
|
strFindReplace(itemText, "%magic_protection", mProtInfo);
|
|
}
|
|
|
|
void getMagicResistance(CDBCtrlSheet *item, ucstring &itemText)
|
|
{
|
|
// retrieve the current itemInfo
|
|
const CClientItemInfo &itemInfo= getInventory().getItemInfo(getInventory().getItemSlotId(item) );
|
|
|
|
ucstring mResistInfo;
|
|
|
|
// Header (always here, because at least max absorb)
|
|
mResistInfo= CI18N::get("uihelpMagicResistFormatHeader");
|
|
|
|
// For each resistance
|
|
uint32 resist[RESISTANCE_TYPE::NB_RESISTANCE_TYPE];
|
|
nlctassert(RESISTANCE_TYPE::NB_RESISTANCE_TYPE==5);
|
|
resist[RESISTANCE_TYPE::Desert]= itemInfo.DesertMagicResistance;
|
|
resist[RESISTANCE_TYPE::Forest]= itemInfo.ForestMagicResistance;
|
|
resist[RESISTANCE_TYPE::Lacustre]= itemInfo.LacustreMagicResistance;
|
|
resist[RESISTANCE_TYPE::Jungle]= itemInfo.JungleMagicResistance;
|
|
resist[RESISTANCE_TYPE::PrimaryRoot]= itemInfo.PrimaryRootMagicResistance;
|
|
for(uint i=0;i<RESISTANCE_TYPE::NB_RESISTANCE_TYPE;i++)
|
|
{
|
|
if(resist[i] != 0)
|
|
{
|
|
// Resist info
|
|
ucstring str= CI18N::get("uihelpMagicResistFormat");
|
|
strFindReplace(str, "%t", CI18N::get("rs"+RESISTANCE_TYPE::toString((RESISTANCE_TYPE::TResistanceType)i) ));
|
|
strFindReplace(str, "%v", toReadableFloat(float(resist[i])/100) );
|
|
mResistInfo+= str;
|
|
}
|
|
}
|
|
|
|
// replace in item info
|
|
strFindReplace(itemText, "%magic_resistance", mResistInfo);
|
|
}
|
|
|
|
void getActionMalus(CDBCtrlSheet *item, ucstring &itemText)
|
|
{
|
|
// retrieve the current itemInfo
|
|
const CClientItemInfo &itemInfo= getInventory().getItemInfo(getInventory().getItemSlotId(item) );
|
|
|
|
strFindReplace(itemText, "%actmalus", toPercentageText(itemInfo.WearEquipmentMalus) );
|
|
}
|
|
|
|
void getBulkText(CDBCtrlSheet *item, const CItemSheet*pIS, ucstring &itemText)
|
|
{
|
|
// Display direct value: because cannot know where this item will be drop!! (bag, mektoub etc...)
|
|
float slotBulkTotal= max((sint32)1, item->getQuantity()) * pIS->Bulk;
|
|
|
|
// If stackable and bulk not 0, display in form "1 (10)". where (10) is the total of quantity*bulk
|
|
if(pIS->Stackable>1 && pIS->Bulk>0)
|
|
strFindReplace(itemText, "%bulk", toString("%.2f (%.2f)", pIS->Bulk, slotBulkTotal) );
|
|
// else simple form
|
|
else
|
|
strFindReplace(itemText, "%bulk", toString("%.2f", slotBulkTotal) );
|
|
}
|
|
|
|
void getWeightText(CDBCtrlSheet *item, const CItemSheet*pIS, ucstring &itemText)
|
|
{
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
|
|
CCDBNodeLeaf *pWG = pIM->getDbProp(item->getSheet()+":WEIGHT",false);
|
|
if(pWG)
|
|
{
|
|
// must mul weight by quantity
|
|
sint32 slotWeight= pWG->getValue32();
|
|
sint32 slotWeightTotal= max((sint32)1, item->getQuantity()) * slotWeight;
|
|
|
|
// if stackable and weight not 0, display in form "1 (10)". where (10) is the total of quantity*weight
|
|
if(pIS->Stackable>1 && slotWeight>0)
|
|
{
|
|
string sws;
|
|
string swt;
|
|
if( (slotWeight%DB_WEIGHT_SCALE) == 0) sws= toString("%d", slotWeight/DB_WEIGHT_SCALE );
|
|
else sws= toString("%.2f", float(slotWeight)/DB_WEIGHT_SCALE);
|
|
if( (slotWeightTotal%DB_WEIGHT_SCALE) == 0) swt= toString("%d", slotWeightTotal/DB_WEIGHT_SCALE );
|
|
else swt= toString("%.2f", float(slotWeightTotal)/DB_WEIGHT_SCALE);
|
|
|
|
// combine
|
|
strFindReplace(itemText, "%weight", toString("%s (%s)", sws.c_str(), swt.c_str() ));
|
|
}
|
|
// else display in simple form
|
|
else
|
|
{
|
|
if( (slotWeightTotal%DB_WEIGHT_SCALE) == 0)
|
|
strFindReplace(itemText, "%weight", toString("%d", slotWeightTotal/DB_WEIGHT_SCALE ));
|
|
else
|
|
strFindReplace(itemText, "%weight", toString("%.2f", float(slotWeightTotal)/DB_WEIGHT_SCALE) );
|
|
}
|
|
}
|
|
else
|
|
strFindReplace(itemText, "%weight", "???" );
|
|
}
|
|
|
|
void getMagicBonus(CDBCtrlSheet *item, ucstring &itemText)
|
|
{
|
|
// retrieve the current itemInfo
|
|
const CClientItemInfo &itemInfo= getInventory().getItemInfo(getInventory().getItemSlotId(item) );
|
|
|
|
nlctassert(CClientItemInfo::NumMagicFactorType==4);
|
|
const string valIds[CClientItemInfo::NumMagicFactorType]={"OffElemental", "OffAffliction", "DefHeal", "DefAffliction"};
|
|
ucstring mbInfo;
|
|
|
|
// For each magic bonus, test first if equal
|
|
sint32 allCastSpeedFactor= sint(itemInfo.CastingSpeedFactor[0]*100);
|
|
sint32 allMagicPowerFactor= sint(itemInfo.MagicPowerFactor[0]*100);
|
|
bool equal= true;
|
|
for(uint i=1;i<CClientItemInfo::NumMagicFactorType;i++)
|
|
{
|
|
sint32 cs= sint(itemInfo.CastingSpeedFactor[i]*100);
|
|
sint32 mp= sint(itemInfo.MagicPowerFactor[i]*100);
|
|
if( cs!=allCastSpeedFactor ||
|
|
mp!=allMagicPowerFactor )
|
|
{
|
|
equal= false;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// if all are equal
|
|
if(equal)
|
|
{
|
|
// if 0, just display nothing
|
|
if(allCastSpeedFactor!=0 || allMagicPowerFactor!=0)
|
|
{
|
|
// else display "all"
|
|
ucstring line= CI18N::get( "uihelpItemMagicBonusAll");
|
|
strFindReplace(line, "%cs", toString("%+d", allCastSpeedFactor) );
|
|
strFindReplace(line, "%mp", toString("%+d", allMagicPowerFactor) );
|
|
mbInfo+= line;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// then display info separated for each
|
|
for(uint i=0;i<CClientItemInfo::NumMagicFactorType;i++)
|
|
{
|
|
sint32 cs= sint(itemInfo.CastingSpeedFactor[i]*100);
|
|
sint32 mp= sint(itemInfo.MagicPowerFactor[i]*100);
|
|
if(cs!=0 || mp!=0)
|
|
{
|
|
ucstring line= CI18N::get( string("uihelpItemMagicBonus") + valIds[i] );
|
|
strFindReplace(line, "%cs", toString("%+d", cs) );
|
|
strFindReplace(line, "%mp", toString("%+d", mp) );
|
|
mbInfo+= line;
|
|
}
|
|
}
|
|
}
|
|
|
|
// append a \n before
|
|
if(mbInfo.size())
|
|
{
|
|
// add spell level header
|
|
ucstring spellRuleFmt= CI18N::get("uihelpItemMagicBonusHeader");
|
|
strFindReplace(spellRuleFmt, "%mglvl", toString(item->getQuality()));
|
|
mbInfo= spellRuleFmt + mbInfo;
|
|
}
|
|
|
|
// Display the buff info.
|
|
strFindReplace(itemText, "%magic_bonus", mbInfo);
|
|
}
|
|
|
|
void getItemRequirementText(CDBCtrlSheet *item, const CItemSheet*pIS, ucstring &itemText)
|
|
{
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
bool requiredNeeded= false;
|
|
ucstring fmt, fmtc;
|
|
|
|
// retrieve the current itemInfo
|
|
const CClientItemInfo &itemInfo= getInventory().getItemInfo(getInventory().getItemSlotId(item) );
|
|
|
|
if( itemInfo.RequiredCharac != CHARACTERISTICS::Unknown && itemInfo.RequiredCharacLevel != 0 )
|
|
{
|
|
// Build the req string
|
|
if(pIM->isItemCaracRequirementMet(itemInfo.RequiredCharac, (sint32)itemInfo.RequiredCharacLevel))
|
|
fmtc = CI18N::get("uihelpItemCaracReqMetFmt");
|
|
else
|
|
fmtc = CI18N::get("uihelpItemCaracReqNotMetFmt");
|
|
strFindReplace(fmtc, "%d", toString((uint)itemInfo.RequiredCharacLevel));
|
|
strFindReplace(fmtc, "%s", CI18N::get(toString("uiCaracId%d", (uint)itemInfo.RequiredCharac)) );
|
|
|
|
//strFindReplace(itemText, "%caracreq", fmtc );
|
|
requiredNeeded = true;
|
|
}
|
|
else
|
|
{
|
|
//strFindReplace(itemText, "%caracreq", "" );
|
|
}
|
|
|
|
if( itemInfo.RequiredSkillLevel > 0 )
|
|
{
|
|
if( itemInfo.RequiredSkill != SKILLS::unknown )
|
|
{
|
|
if (CSkillManager::getInstance()->checkBaseSkillMetRequirement(itemInfo.RequiredSkill, itemInfo.RequiredSkillLevel))
|
|
fmt = CI18N::get("uihelpItemSkillReqMetFmt");
|
|
else
|
|
fmt = CI18N::get("uihelpItemSkillReqNotMetFmt");
|
|
|
|
strFindReplace(fmt, "%d", toString((uint)itemInfo.RequiredSkillLevel));
|
|
const ucstring skillName(STRING_MANAGER::CStringManagerClient::getSkillLocalizedName(itemInfo.RequiredSkill));
|
|
strFindReplace(fmt, "%s", skillName);
|
|
}
|
|
else
|
|
{
|
|
if (CSkillManager::getInstance()->checkBaseSkillMetRequirement(SKILLS::unknown, itemInfo.RequiredSkillLevel))
|
|
fmt = CI18N::get("uihelpItemAnySkillReqMetFmt");
|
|
else
|
|
fmt = CI18N::get("uihelpItemAnySkillReqNotMetFmt");
|
|
|
|
strFindReplace(fmt, "%d", toString((uint)itemInfo.RequiredSkillLevel));
|
|
}
|
|
|
|
strFindReplace(itemText, "%skillreq", fmt );
|
|
requiredNeeded = true;
|
|
}
|
|
else
|
|
{
|
|
strFindReplace(itemText, "%skillreq", "" );
|
|
}
|
|
|
|
if( itemInfo.RequiredSkillLevel2 > 0 )
|
|
{
|
|
if( itemInfo.RequiredSkill2 != SKILLS::unknown )
|
|
{
|
|
if (CSkillManager::getInstance()->checkBaseSkillMetRequirement(itemInfo.RequiredSkill2, itemInfo.RequiredSkillLevel2))
|
|
fmt = CI18N::get("uihelpItemSkillReqMetFmt");
|
|
else
|
|
fmt = CI18N::get("uihelpItemSkillReqNotMetFmt");
|
|
|
|
strFindReplace(fmt, "%d", toString((uint)itemInfo.RequiredSkillLevel2));
|
|
const ucstring skillName(STRING_MANAGER::CStringManagerClient::getSkillLocalizedName(itemInfo.RequiredSkill2));
|
|
strFindReplace(fmt, "%s", skillName);
|
|
}
|
|
else
|
|
{
|
|
if (CSkillManager::getInstance()->checkBaseSkillMetRequirement(SKILLS::unknown, itemInfo.RequiredSkillLevel2))
|
|
fmt = CI18N::get("uihelpItemAnySkillReqMetFmt");
|
|
else
|
|
fmt = CI18N::get("uihelpItemAnySkillReqNotMetFmt");
|
|
|
|
strFindReplace(fmt, "%d", toString((uint)itemInfo.RequiredSkillLevel2));
|
|
}
|
|
|
|
strFindReplace(itemText, "%skillreq2", fmt );
|
|
requiredNeeded = true;
|
|
}
|
|
else
|
|
{
|
|
strFindReplace(itemText, "%skillreq2", "" );
|
|
}
|
|
|
|
if( requiredNeeded )
|
|
strFindReplace(itemText, "%caracreq", fmtc );
|
|
else
|
|
strFindReplace(itemText, "%caracreq", CI18N::get("uihelpItemCaracReqNone") );
|
|
|
|
#if 0
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
|
|
// retrieve the current itemInfo
|
|
const CClientItemInfo &itemInfo= getInventory().getItemInfo(getInventory().getItemSlotId(item) );
|
|
|
|
CHARACTERISTICS::TCharacteristics caracType;
|
|
float caracValue= 0.f;
|
|
|
|
bool req = pIS->hasCharacRequirement(item->getQuality(), caracType, caracValue);
|
|
|
|
bool req2 = (itemInfo.RequiredCharac != CHARACTERISTICS::Unknown && itemInfo.MinRequiredCharacLevel > 0);
|
|
bool skillReq = (itemInfo.MinRequiredSkillLevel > 0 && itemInfo.RequiredSkill != SKILLS::unknown);
|
|
|
|
// check item specific req
|
|
if (req2)
|
|
{
|
|
if (req)
|
|
{
|
|
if (itemInfo.RequiredCharac == caracType && itemInfo.MinRequiredCharacLevel > caracValue)
|
|
{
|
|
caracValue = itemInfo.MinRequiredCharacLevel;
|
|
req2 = false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
caracValue = itemInfo.MinRequiredCharacLevel;
|
|
caracType = itemInfo.RequiredCharac;
|
|
req2 = false;
|
|
req = true;
|
|
}
|
|
}
|
|
|
|
if(req)
|
|
{
|
|
// Build the req string
|
|
ucstring fmt;
|
|
if(pIM->isItemCaracRequirementMet(caracType, (sint32)caracValue))
|
|
fmt= CI18N::get("uihelpItemCaracReqMetFmt");
|
|
else
|
|
fmt= CI18N::get("uihelpItemCaracReqNotMetFmt");
|
|
strFindReplace(fmt, "%d", toString((uint)caracValue));
|
|
strFindReplace(fmt, "%s", CI18N::get(toString("uiCaracId%d", (uint)caracType)) );
|
|
|
|
if (req2)
|
|
{
|
|
fmt += CI18N::get("uihelpItemCaracReqAnd");
|
|
if ( pIM->isItemCaracRequirementMet(itemInfo.RequiredCharac, (sint32)itemInfo.MinRequiredCharacLevel) )
|
|
fmt += CI18N::get("uihelpItemCaracReqMetFmt");
|
|
else
|
|
fmt += CI18N::get("uihelpItemCaracReqNotMetFmt");
|
|
strFindReplace(fmt, "%d", toString((uint)itemInfo.MinRequiredCharacLevel));
|
|
strFindReplace(fmt, "%s", CI18N::get(toString("uiCaracId%d", (uint)itemInfo.RequiredCharac)) );
|
|
}
|
|
|
|
strFindReplace(itemText, "%caracreq", fmt );
|
|
}
|
|
else if (!skillReq)
|
|
{
|
|
// empty
|
|
strFindReplace(itemText, "%caracreq", CI18N::get("uihelpItemCaracReqNone") );
|
|
}
|
|
else // skillReq and no charac req
|
|
{
|
|
// empty
|
|
strFindReplace(itemText, "%caracreq", "" );
|
|
}
|
|
|
|
if (skillReq)
|
|
{
|
|
// Build the req string
|
|
ucstring fmt;
|
|
if (req)
|
|
fmt = CI18N::get("uihelpItemCaracReqAnd");
|
|
|
|
if (CSkillManager::getInstance()->checkBaseSkillMetRequirement(itemInfo.RequiredSkill, itemInfo.MinRequiredSkillLevel))
|
|
fmt += CI18N::get("uihelpItemSkillReqMetFmt");
|
|
else
|
|
fmt += CI18N::get("uihelpItemSkillReqNotMetFmt");
|
|
strFindReplace(fmt, "%d", toString((uint)itemInfo.MinRequiredSkillLevel));
|
|
const ucstring skillName = STRING_MANAGER::CStringManagerClient::getSkillLocalizedName(itemInfo.RequiredSkill);
|
|
strFindReplace(fmt, "%s", skillName);
|
|
|
|
strFindReplace(itemText, "%skillreq", fmt );
|
|
}
|
|
else
|
|
{
|
|
strFindReplace(itemText, "%skillreq", "" );
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void getSkillModVsType(CDBCtrlSheet *item, const CItemSheet*pIS, ucstring &itemText)
|
|
{
|
|
// retrieve the current itemInfo
|
|
const CClientItemInfo &itemInfo= getInventory().getItemInfo(getInventory().getItemSlotId(item) );
|
|
|
|
ucstring sMod;
|
|
// check skill mod
|
|
if(!itemInfo.TypeSkillMods.empty())
|
|
{
|
|
for (uint i = 0 ; i < itemInfo.TypeSkillMods.size() ; ++i)
|
|
{
|
|
EGSPD::CClassificationType::TClassificationType type = itemInfo.TypeSkillMods[i].Type;
|
|
|
|
sMod= CI18N::get("uihelpSkillModVsType");
|
|
strFindReplace(sMod, "%mod", NLMISC::toString(itemInfo.TypeSkillMods[i].Modifier));
|
|
strFindReplace(sMod, "%type", CStringManagerClient::getClassificationTypeLocalizedName(type));
|
|
sMod += "\n\n";
|
|
|
|
// TODO : process ALL mod
|
|
break;
|
|
}
|
|
}
|
|
|
|
strFindReplace(itemText, "%skill_mod_vs_type", sMod);
|
|
}
|
|
|
|
void getArmorBonus(CDBCtrlSheet *item, ucstring &itemText, const CItemSheet*pIS)
|
|
{
|
|
ucstring armor_bonus("");
|
|
sint32 level;
|
|
|
|
if (pIS->Armor.ArmorType == ARMORTYPE::HEAVY)
|
|
level = item->getQuality();
|
|
else if (pIS->Armor.ArmorType == ARMORTYPE::MEDIUM)
|
|
level = item->getQuality() / 2;
|
|
|
|
if (pIS->Armor.ArmorType == ARMORTYPE::HEAVY || pIS->Armor.ArmorType == ARMORTYPE::MEDIUM)
|
|
armor_bonus = "@{FFFF}(+@{2F2F}" + toString(level) + " @{FFFF}" + CI18N::get("uiHP") + ")";
|
|
|
|
strFindReplace(itemText, "%armor_bonus", armor_bonus);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void getItemText (CDBCtrlSheet *item, ucstring &itemText, const CItemSheet*pIS)
|
|
{
|
|
if ((item == NULL) || (pIS == NULL))
|
|
return;
|
|
|
|
// *** Select the correct format according to item family.
|
|
switch(pIS->Family)
|
|
{
|
|
case ITEMFAMILY::ARMOR : itemText= CI18N::get("uihelpItemArmorFormat"); break;
|
|
case ITEMFAMILY::MELEE_WEAPON : itemText= CI18N::get("uihelpItemMeleeWeaponFormat"); break;
|
|
case ITEMFAMILY::RANGE_WEAPON : itemText= CI18N::get("uihelpItemRangeWeaponFormat"); break;
|
|
case ITEMFAMILY::AMMO : itemText= CI18N::get("uihelpItemAmmoFormat"); break;
|
|
case ITEMFAMILY::RAW_MATERIAL : itemText= CI18N::get("uihelpItemMPFormat"); break;
|
|
case ITEMFAMILY::SHIELD : itemText= CI18N::get("uihelpItemShieldFormat"); break;
|
|
case ITEMFAMILY::CRAFTING_TOOL : itemText= CI18N::get("uihelpItemCraftingToolFormat"); break;
|
|
case ITEMFAMILY::HARVEST_TOOL : itemText= CI18N::get("uihelpItemHarvestToolFormat"); break;
|
|
case ITEMFAMILY::TAMING_TOOL : itemText= CI18N::get("uihelpItemTamingToolFormat"); break;
|
|
case ITEMFAMILY::JEWELRY : itemText= CI18N::get("uihelpItemJewelFormat"); break;
|
|
case ITEMFAMILY::CRYSTALLIZED_SPELL : itemText= CI18N::get("uihelpItemCrystalSpell"); break;
|
|
case ITEMFAMILY::ITEM_SAP_RECHARGE : itemText= CI18N::get("uihelpItemChargeSpell"); break;
|
|
case ITEMFAMILY::PET_ANIMAL_TICKET : itemText= CI18N::get("uihelpItemAnimal"); break;
|
|
case ITEMFAMILY::TELEPORT : itemText= CI18N::get("uihelpItemTeleport"); break;
|
|
case ITEMFAMILY::COSMETIC : itemText= CI18N::get("uihelpItemCosmetic"); break;
|
|
case ITEMFAMILY::SCROLL : itemText= CI18N::get("uihelpItemScroll"); break;
|
|
case ITEMFAMILY::SCROLL_R2 : itemText = CI18N::get("uihelpItemScrollR2"); break;
|
|
case ITEMFAMILY::CONSUMABLE : itemText= CI18N::get("uihelpItemConsumableFormat"); break;
|
|
default: itemText= CI18N::get("uihelpItemDefaultFormat");
|
|
};
|
|
|
|
|
|
// *** Replace Common part
|
|
strFindReplace(itemText, "%origin", CI18N::get("io"+ITEM_ORIGIN::enumToString(pIS->ItemOrigin)) );
|
|
strFindReplace(itemText, "%quality", toString(item->getQuality()) );
|
|
strFindReplace(itemText, "%quantity", toString(item->getQuantity()) );
|
|
// display Weight
|
|
getWeightText(item, pIS, itemText);
|
|
// display Bulk
|
|
getBulkText(item, pIS, itemText);
|
|
// Get the SapLoad...
|
|
getHPAndSapLoadText(item, pIS, itemText);
|
|
// Get carac and skill Requirement
|
|
getItemRequirementText(item, pIS, itemText);
|
|
// Get Item effect
|
|
CItemSpecialEffectHelper::getInstance()->getItemSpecialEffectText(pIS, itemText);
|
|
|
|
// Description
|
|
const ucstring desc(CStringManagerClient::getItemLocalizedDescription(pIS->Id));
|
|
if(!desc.empty())
|
|
{
|
|
strFindReplace(itemText, "%desc", "@{FFF9}" + CI18N::get("uiMissionDesc") + "\n@{FFFF}" + desc + "\n" );
|
|
}
|
|
else
|
|
strFindReplace(itemText, "%desc", ucstring() );
|
|
|
|
// Custom text
|
|
const CClientItemInfo &itemInfo = getInventory().getItemInfo(getInventory().getItemSlotId(item) );
|
|
if (!itemInfo.CustomText.empty())
|
|
{
|
|
strFindReplace(itemText, "%custom_text", "\n@{FFFF}" + itemInfo.CustomText + "\n");
|
|
ucstring itemMFC = CI18N::get("uiItemTextMessageFromCrafter");
|
|
strFindReplace(itemText, "%mfc", itemMFC);
|
|
}
|
|
else
|
|
strFindReplace(itemText, "%custom_text", ucstring() );
|
|
|
|
if ( pIS->Family == ITEMFAMILY::COSMETIC )
|
|
{
|
|
EGSPD::CPeople::TPeople people = ITEM_ORIGIN::itemOriginStringToPeopleEnum( ITEM_ORIGIN::enumToString( pIS->ItemOrigin ) );
|
|
if ( UserEntity->getGender() != pIS->Cosmetic.Gender || UserEntity->people() != people )
|
|
strFindReplace(itemText, "%cansell", CI18N::get("uihelpItemCosmeticDontFit") );
|
|
else
|
|
strFindReplace(itemText, "%cansell", ucstring() );
|
|
}
|
|
else if(pIS->DropOrSell )
|
|
strFindReplace(itemText, "%cansell", ucstring() );
|
|
else
|
|
strFindReplace(itemText, "%cansell", CI18N::get("uihelpItemCantSell") );
|
|
|
|
// *** Replace special for each type
|
|
switch(pIS->Family)
|
|
{
|
|
case ITEMFAMILY::ARMOR :
|
|
{
|
|
strFindReplace(itemText, "%armor", CI18N::get("at"+ARMORTYPE::toString(pIS->Armor.ArmorType)));
|
|
// Armor bonus based on armor type
|
|
getArmorBonus(item, itemText, pIS);
|
|
// Protection
|
|
getItemDefenseText(item, itemText);
|
|
// Player buffs
|
|
getBuffText(item, itemText);
|
|
// action malus
|
|
getActionMalus(item, itemText);
|
|
}
|
|
break;
|
|
case ITEMFAMILY::MELEE_WEAPON :
|
|
{
|
|
strFindReplace(itemText, "%skill", CStringManagerClient::getSkillLocalizedName(pIS->MeleeWeapon.Skill) );
|
|
strFindReplace(itemText, "%cat", CI18N::get("wt"+WEAPONTYPE::toString(pIS->MeleeWeapon.WeaponType)) );
|
|
strFindReplace(itemText, "%dmtype", CI18N::get("dt"+DMGTYPE::toString(pIS->MeleeWeapon.DamageType)) );
|
|
strFindReplace(itemText, "%reach", toString(pIS->MeleeWeapon.MeleeRange) );
|
|
// Damage / Speed
|
|
getDamageText(item, pIS, itemText, false);
|
|
getSpeedText(item, itemText, false);
|
|
// Player buffs
|
|
getBuffText(item, itemText);
|
|
// action malus
|
|
getActionMalus(item, itemText);
|
|
// magical bonus
|
|
getMagicBonus(item, itemText);
|
|
// skill bonus against specific types
|
|
getSkillModVsType(item, pIS, itemText);
|
|
}
|
|
break;
|
|
case ITEMFAMILY::RANGE_WEAPON :
|
|
{
|
|
strFindReplace(itemText, "%skill", CStringManagerClient::getSkillLocalizedName(pIS->RangeWeapon.Skill) );
|
|
strFindReplace(itemText, "%cat", CI18N::get("wt"+WEAPONTYPE::toString(pIS->RangeWeapon.WeaponType)) );
|
|
// Damage / Speed / Range
|
|
getDamageText(item, pIS, itemText, true);
|
|
getSpeedText(item, itemText, false);
|
|
getRangeText(item, itemText, false);
|
|
// Player buffs
|
|
getBuffText(item, itemText);
|
|
// action malus
|
|
getActionMalus(item, itemText);
|
|
// magical bonus
|
|
getMagicBonus(item, itemText);
|
|
// skill bonus against specific types
|
|
getSkillModVsType(item, pIS, itemText);
|
|
}
|
|
break;
|
|
case ITEMFAMILY::AMMO :
|
|
{
|
|
// Localization
|
|
strFindReplace(itemText, "%skill", CStringManagerClient::getSkillLocalizedName(pIS->Ammo.Skill) ) ;
|
|
strFindReplace(itemText, "%dmtype", CI18N::get("dt"+DMGTYPE::toString(pIS->Ammo.DamageType)) );
|
|
// WARNING: here 999 is hardcoded because of the new global limit of stacks (999), Ammo.Magazine should not be used anymore
|
|
if (item->getUseQuantity())
|
|
strFindReplace(itemText, "%magazine", toString(item->getQuantity()) + " / " + toString(999 /*pIS->Ammo.Magazine*/) );
|
|
else
|
|
strFindReplace(itemText, "%magazine", toString(999 /*pIS->Ammo.Magazine*/) );
|
|
|
|
// Damage / Speed / Range
|
|
getDamageText(item, pIS, itemText, false);
|
|
getSpeedText(item, itemText, true);
|
|
getRangeText(item, itemText, true);
|
|
}
|
|
break;
|
|
case ITEMFAMILY::RAW_MATERIAL :
|
|
{
|
|
// Basics
|
|
strFindReplace(itemText, "%ecosystem", CI18N::get("ecosys"+ECOSYSTEM::toString(pIS->Mp.Ecosystem)) ) ;
|
|
strFindReplace(itemText, "%family", RM_FAMILY::toLocalString(pIS->Mp.Family) ) ;
|
|
strFindReplace(itemText, "%skill", CStringManagerClient::getSkillLocalizedName(pIS->Mp.HarvestSkill) ) ;
|
|
|
|
// MpColor
|
|
if( RM_COLOR::validColor(pIS->Mp.MpColor) )
|
|
strFindReplace(itemText, "%mpcolor", RM_COLOR::toLocalString(pIS->Mp.MpColor) ) ;
|
|
else
|
|
strFindReplace(itemText, "%mpcolor", "???" );
|
|
|
|
// Craft some part?
|
|
if(pIS->canBuildSomeItemPart())
|
|
{
|
|
ucstring fmt= CI18N::get("uihelpItemMPCraft");
|
|
ucstring ipList;
|
|
pIS->getItemPartListAsText(ipList);
|
|
strFindReplace(fmt, "%ip", ipList);
|
|
strFindReplace(itemText, "%craft", fmt);
|
|
}
|
|
// Craft Mp requirement?
|
|
else if(pIS->isUsedAsCraftRequirement())
|
|
strFindReplace(itemText, "%craft", CI18N::get("uihelpItemMPCraftRequirement"));
|
|
// No Craft at all
|
|
else
|
|
strFindReplace(itemText, "%craft", CI18N::get("uihelpItemMPNoCraft"));
|
|
}
|
|
break;
|
|
case ITEMFAMILY::SHIELD :
|
|
{
|
|
//strFindReplace(itemText, "%skill", CStringManagerClient::getSkillLocalizedName(SHIELDTYPE::shieldTypeToSkill(pIS->Shield.ShieldType)) );
|
|
strFindReplace(itemText, "%cat", CI18N::get("st"+SHIELDTYPE::toString(pIS->Shield.ShieldType)) );
|
|
// Protection
|
|
getItemDefenseText(item, itemText);
|
|
// Player buffs
|
|
getBuffText(item, itemText);
|
|
// action malus
|
|
getActionMalus(item, itemText);
|
|
}
|
|
break;
|
|
// Crafting Tool: the skill is not valid, since depends on what is built.
|
|
case ITEMFAMILY::CRAFTING_TOOL :
|
|
{
|
|
strFindReplace(itemText, "%tool", CI18N::get("tool"+TOOL_TYPE::toString(pIS->Tool.CraftingToolType)) );
|
|
// Player buffs
|
|
getBuffText(item, itemText);
|
|
}
|
|
break;
|
|
case ITEMFAMILY::HARVEST_TOOL :
|
|
{
|
|
strFindReplace(itemText, "%skill", CStringManagerClient::getSkillLocalizedName(pIS->Tool.Skill) );
|
|
// Player buffs
|
|
getBuffText(item, itemText);
|
|
}
|
|
break;
|
|
case ITEMFAMILY::TAMING_TOOL :
|
|
{
|
|
strFindReplace(itemText, "%skill", CStringManagerClient::getSkillLocalizedName(pIS->Tool.Skill) );
|
|
strFindReplace(itemText, "%cmdrange", toString(pIS->Tool.CommandRange) );
|
|
strFindReplace(itemText, "%maxpacker", toString(pIS->Tool.MaxDonkey) );
|
|
}
|
|
break;
|
|
case ITEMFAMILY::JEWELRY :
|
|
{
|
|
// Player buffs
|
|
getBuffText(item, itemText);
|
|
// Magic protection
|
|
getMagicProtection(item, itemText);
|
|
// Magic Resistances
|
|
getMagicResistance(item, itemText);
|
|
}
|
|
break;
|
|
case ITEMFAMILY::CONSUMABLE :
|
|
{
|
|
strFindReplace(itemText, "%consumption_time", toString(pIS->Consumable.ConsumptionTime));
|
|
strFindReplace(itemText, "%overdose_timer", toString(pIS->Consumable.OverdoseTimer/60));
|
|
// Get Item Consumable infos
|
|
CItemConsumableEffectHelper::getInstance()->getItemConsumableEffectText(pIS, itemText, item->getQuality());
|
|
}
|
|
break;
|
|
case ITEMFAMILY::SCROLL_R2:
|
|
{
|
|
strFindReplace(itemText, "%r2_description_text", toString(itemInfo.R2ItemDescription));
|
|
strFindReplace(itemText, "%r2_comment_text", toString(itemInfo.R2ItemComment));
|
|
}
|
|
break;
|
|
default:
|
|
{
|
|
strFindReplace(itemText, "%no_rent", pIS->IsItemNoRent ? CI18N::get("uihelpItemNoRent") : string(""));
|
|
strFindReplace(itemText, "%descnr", pIS->IsItemNoRent ? CI18N::get("uihelpItemNoRentDesc") : string(""));
|
|
}
|
|
break;
|
|
};
|
|
|
|
#ifdef NL_DEBUG
|
|
INVENTORIES::TInventory inventory = (INVENTORIES::TInventory)item->getInventoryIndex();
|
|
sint32 slot = item->getIndexInDB();
|
|
string debugText = NLMISC::toString("inventory: %s\nslot: %d\n", INVENTORIES::toString(inventory).c_str(), slot);
|
|
ucstring debugText2;
|
|
debugText2.fromUtf8(debugText);
|
|
itemText = debugText2 + itemText;
|
|
#endif
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
static void setupEnchantedItem(CSheetHelpSetup &setup, ucstring &itemText)
|
|
{
|
|
// if don't find the tag in the text (eg: if not usefull), no-op
|
|
static const ucstring enchantTag("%enchantment");
|
|
if( itemText.find(enchantTag) == ucstring::npos )
|
|
return;
|
|
|
|
// retrieve the current itemInfo
|
|
CDBCtrlSheet *ctrl= setup.SrcSheet;
|
|
CInterfaceGroup *group= setup.HelpWindow;
|
|
if(!ctrl || !group)
|
|
return;
|
|
const CClientItemInfo &itemInfo= getInventory().getItemInfo(getInventory().getItemSlotId(ctrl) );
|
|
IListSheetBase *listBrick= dynamic_cast<IListSheetBase*>(group->getElement(group->getId()+setup.PrefixForExtra+INFO_LIST_BRICK));
|
|
|
|
// if the item is enchanted
|
|
if( !itemInfo.Enchantment.empty())
|
|
{
|
|
CSPhraseManager *pPM= CSPhraseManager::getInstance();
|
|
|
|
// fill the enchantement info
|
|
ucstring enchantInfo;
|
|
const CItemSheet *pIS= ctrl->asItemSheet();
|
|
if(pIS && pIS->Family == ITEMFAMILY::CRYSTALLIZED_SPELL)
|
|
pPM->buildPhraseDesc(enchantInfo, itemInfo.Enchantment, 0, false, "uihelpPhraseCrystalSpellFormat");
|
|
else
|
|
pPM->buildPhraseDesc(enchantInfo, itemInfo.Enchantment, 0, false, "uihelpPhraseEnchantmentFormat");
|
|
|
|
// replace
|
|
strFindReplace(itemText, enchantTag, enchantInfo );
|
|
|
|
// if exist, setup text header
|
|
if(listBrick)
|
|
setupListBrickHeader(setup);
|
|
|
|
// fill the bricks
|
|
fillSabrinaPhraseListBrick(itemInfo.Enchantment, listBrick);
|
|
}
|
|
else
|
|
{
|
|
// must hide the listBrick
|
|
if(listBrick)
|
|
listBrick->setActive(false);
|
|
|
|
// hide the list brick header
|
|
hideListBrickHeader(setup);
|
|
|
|
// hide the text
|
|
strFindReplace(itemText, enchantTag, ucstring());
|
|
}
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
static void setupRawMaterialStats(CSheetHelpSetup &setup)
|
|
{
|
|
// retrieve the current itemInfo
|
|
CDBCtrlSheet *ctrl= setup.SrcSheet;
|
|
CInterfaceGroup *group= setup.HelpWindow;
|
|
if(!ctrl || !group)
|
|
return;
|
|
|
|
// get the group for raw material stat
|
|
CInterfaceGroup *groupMp= dynamic_cast<CInterfaceGroup*>(group->getElement(group->getId()+setup.PrefixForExtra+INFO_GROUP_MP_STAT));
|
|
if(!groupMp)
|
|
return;
|
|
|
|
// if the item is a Mp
|
|
const CItemSheet *pIS= ctrl->asItemSheet();
|
|
if(pIS && pIS->Family == ITEMFAMILY::RAW_MATERIAL)
|
|
{
|
|
// If the MP can craft some item part
|
|
if(pIS->canBuildSomeItemPart())
|
|
{
|
|
// activate the mp stat group
|
|
groupMp->setActive(true);
|
|
|
|
// Initialize the itempart selection combo box
|
|
CDBGroupComboBox *pCB= dynamic_cast<CDBGroupComboBox*>(groupMp->getElement(groupMp->getId()+":item_part_choice" ));
|
|
if( pCB )
|
|
{
|
|
pCB->resetTexts();
|
|
for(uint i=0;i<RM_FABER_TYPE::NUM_FABER_TYPE;i++)
|
|
{
|
|
RM_FABER_TYPE::TRMFType faberType= RM_FABER_TYPE::TRMFType(i);
|
|
|
|
if(pIS->canBuildItemPart(faberType))
|
|
{
|
|
pCB->addText(RM_FABER_TYPE::toLocalString(faberType));
|
|
}
|
|
}
|
|
|
|
// force reset, but try to keep the precedent selection
|
|
// (usefull to test same item-part from different MPs)
|
|
sint32 precSel= pCB->getSelection();
|
|
pCB->setSelection(1);
|
|
pCB->setSelection(0);
|
|
if(precSel>=0 && precSel<(sint32)pCB->getNumTexts())
|
|
{
|
|
pCB->setSelection(precSel);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// just hide it
|
|
groupMp->setActive(false);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// just hide it
|
|
groupMp->setActive(false);
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void resetSheetHelp(CSheetHelpSetup &setup)
|
|
{
|
|
CInterfaceGroup *group= setup.HelpWindow;
|
|
if(!group)
|
|
return;
|
|
|
|
// Hide the list of items by default
|
|
IListSheetBase *listItem= dynamic_cast<IListSheetBase*>(group->getElement(group->getId()+setup.PrefixForExtra+INFO_LIST_ITEM));
|
|
if(listItem)
|
|
{
|
|
listItem->setActive(false);
|
|
}
|
|
// Hide the list of brick by default
|
|
IListSheetBase *listBrick= dynamic_cast<IListSheetBase*>(group->getElement(group->getId()+setup.PrefixForExtra+INFO_LIST_BRICK));
|
|
if(listBrick)
|
|
{
|
|
listBrick->setActive(false);
|
|
}
|
|
// Hide the mpstats by default
|
|
CInterfaceGroup *groupMp= dynamic_cast<CInterfaceGroup*>(group->getElement(group->getId()+setup.PrefixForExtra+INFO_GROUP_MP_STAT));
|
|
if(groupMp)
|
|
groupMp->setActive(false);
|
|
// Hide the list of brick requirement by default
|
|
listBrick= dynamic_cast<IListSheetBase*>(group->getElement(group->getId()+setup.PrefixForExtra+INFO_LIST_BRICK_REQUIREMENT));
|
|
if(listBrick)
|
|
{
|
|
listBrick->setActive(false);
|
|
}
|
|
// Hide the listBrick header by default
|
|
CViewText *view= dynamic_cast<CViewText*>(group->getElement(group->getId()+setup.PrefixForExtra+INFO_LIST_BRICK_HEADER));
|
|
if(view)
|
|
{
|
|
view->setActive(false);
|
|
}
|
|
|
|
CInterfaceGroup *groupCosmetic = dynamic_cast<CInterfaceGroup*>(group->getElement(group->getId()+setup.PrefixForExtra+INFO_GROUP_CHAR_3D));
|
|
if(groupCosmetic)
|
|
groupCosmetic->setActive(false);
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
static void setupItemHelp(CSheetHelpSetup &setup)
|
|
{
|
|
if (!setup.HelpWindow) return;
|
|
|
|
// get the calling item
|
|
if (!setup.SrcSheet || setup.SrcSheet->getType()!=CCtrlSheetInfo::SheetType_Item )
|
|
{
|
|
nlwarning("<CHandlerOpenItemHelp::execute> no caller sheet found");
|
|
return;
|
|
}
|
|
|
|
// If the sheet is 0, don't open!
|
|
if(setup.SrcSheet->getSheetId()==0)
|
|
return;
|
|
|
|
if(setup.DestSheet)
|
|
{
|
|
setup.SrcSheet->copyAspect(setup.DestSheet);
|
|
}
|
|
|
|
// NB: for raw materials only, must do each once only, must not do it at refresh, cause combo reseted
|
|
setupRawMaterialStats(setup);
|
|
|
|
// update the item Help
|
|
refreshItemHelp(setup);
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void setupCosmetic(CSheetHelpSetup &setup, CItemSheet *pIS)
|
|
{
|
|
nlassert(pIS);
|
|
if(pIS->Family!=ITEMFAMILY::COSMETIC)
|
|
return;
|
|
|
|
EGSPD::CPeople::TPeople people = ITEM_ORIGIN::itemOriginStringToPeopleEnum( ITEM_ORIGIN::enumToString( pIS->ItemOrigin ) );
|
|
if ( !( UserEntity->getGender() != pIS->Cosmetic.Gender || UserEntity->people() != people ) )
|
|
{
|
|
CInterfaceGroup *groupCosmetic = dynamic_cast<CInterfaceGroup*>(setup.HelpWindow->getElement(setup.HelpWindow->getId()+setup.PrefixForExtra+INFO_GROUP_CHAR_3D));
|
|
if(groupCosmetic)
|
|
groupCosmetic->setActive(true);
|
|
// display the character head in the help window
|
|
CCharacterSummary cs;
|
|
SCharacter3DSetup::setupCharacterSummaryFromSERVERDB( cs );
|
|
// we dont want to display helmets
|
|
cs.VisualPropA.PropertySubData.HatModel = SCharacter3DSetup::getDB ( "SERVER:USER:HAIR_TYPE" );
|
|
cs.VisualPropA.PropertySubData.HatColor = SCharacter3DSetup::getDB ("SERVER:USER:HAIR_COLOR");
|
|
if ( pIS->ItemType == ITEM_TYPE::HAIR_MALE || pIS->ItemType == ITEM_TYPE::HAIR_FEMALE )
|
|
cs.VisualPropA.PropertySubData.HatModel = CVisualSlotManager::getInstance()->sheet2Index( CSheetId(setup.SrcSheet->getSheetId()), SLOTTYPE::HEAD_SLOT );
|
|
else if ( pIS->ItemType == ITEM_TYPE::TATOO_MALE || pIS->ItemType == ITEM_TYPE::TATOO_FEMALE )
|
|
cs.VisualPropC.PropertySubData.Tattoo = pIS->Cosmetic.VPValue;
|
|
else if ( pIS->ItemType == ITEM_TYPE::HAIRCOLOR_MALE || pIS->ItemType == ITEM_TYPE::HAIRCOLOR_FEMALE )
|
|
cs.VisualPropA.PropertySubData.HatColor = pIS->Cosmetic.VPValue;
|
|
else
|
|
nlwarning("<setupItemHelp> Invalid cosmetic item type '%s'",ITEM_TYPE::toString( pIS->ItemType ).c_str() );
|
|
|
|
SCharacter3DSetup::setupDBFromCharacterSummary("UI:TEMP:CHAR3D",cs );
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void refreshItemHelp(CSheetHelpSetup &setup)
|
|
{
|
|
// Setup creator name view
|
|
setupCreatorName(setup);
|
|
|
|
// **** setup the item Text info
|
|
ucstring itemText;
|
|
CEntitySheet *pES = SheetMngr.get ( CSheetId(setup.SrcSheet->getSheetId()) );
|
|
if ((pES != NULL) && (pES->type() == CEntitySheet::ITEM))
|
|
{
|
|
CItemSheet *pIS = (CItemSheet*)pES;
|
|
|
|
// ---- Common
|
|
ucstring title = setup.SrcSheet->getItemActualName();
|
|
setupHelpTitle(setup.HelpWindow, title );
|
|
getItemText (setup.SrcSheet, itemText, pIS);
|
|
|
|
// ---- Enchanted items only
|
|
setupEnchantedItem(setup, itemText);
|
|
|
|
// ---- Cosmetic only
|
|
setupCosmetic (setup, pIS);
|
|
}
|
|
|
|
// if this is a R2 plot item, Add comment and description
|
|
|
|
// BORIS : 06/09/2006 : removed because seams to build a double 'description' in the item info windows
|
|
// const CItemSheet *pIS= setup.SrcSheet->asItemSheet();
|
|
// if (pIS)
|
|
// {
|
|
// if (pIS->Family == ITEMFAMILY::SCROLL_R2)
|
|
// {
|
|
// const R2::TMissionItem *mi = R2::getEditor().getPlotItemInfos((uint32) setup.SrcSheet->getSheetId());
|
|
// if (mi)
|
|
// {
|
|
// itemText += CI18N::get("uiRingPlotItemDesc");
|
|
// itemText += mi->Description.empty() ? CI18N::get("uiRingPlotItemEmpty")
|
|
// : mi->Description;
|
|
// //itemText += ucstring("\n@{6F6F}") + CI18N::get("uiRingPlotItemComment") + ucstring("\n");
|
|
// /*
|
|
// itemText += mi->Comment.empty() ? CI18N::get("uiRingPlotItemEmpty")
|
|
// : (ucstring("\n") + mi->Comment);
|
|
// */
|
|
// }
|
|
// }
|
|
// }
|
|
|
|
|
|
// **** setup the text
|
|
setHelpText(setup, itemText);
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
static void setupPactHelp(CSheetHelpSetup &setup)
|
|
{
|
|
if (!setup.HelpWindow) return;
|
|
|
|
// get the calling item
|
|
if (!setup.SrcSheet)
|
|
{
|
|
nlwarning("<CHandlerOpenBrickHelp::execute> no caller sheet found.");
|
|
return;
|
|
}
|
|
|
|
// If the sheet is 0, don't open!
|
|
if(setup.SrcSheet->getSheetId()==0)
|
|
return;
|
|
|
|
const CPactSheet *pact = setup.SrcSheet->asPactSheet();
|
|
if (!pact)
|
|
{
|
|
nlwarning("<CHandlerOpenBrickHelp::execute> can't get pact.");
|
|
return;
|
|
}
|
|
|
|
// Level of the pact is in the quality.
|
|
sint32 pactLevel = setup.SrcSheet->getQuality();
|
|
|
|
if (pactLevel < 0 || pactLevel >= (sint32) pact->PactLose.size())
|
|
{
|
|
nlwarning("<CHandlerOpenBrickHelp::execute> bad level for pact.");
|
|
return;
|
|
}
|
|
|
|
if(setup.DestSheet)
|
|
{
|
|
setup.SrcSheet->copyAspect(setup.DestSheet);
|
|
}
|
|
|
|
|
|
const CPactSheet::SPact &pactLose = pact->PactLose[pactLevel];
|
|
// **** setup the brick Text info
|
|
ucstring pactText;
|
|
|
|
// TODO Localisation
|
|
setupHelpTitle(setup.HelpWindow, pactLose.Name);
|
|
|
|
pactText= CI18N::get("uihelpPactFormat");
|
|
strFindReplace(pactText, "%lvl", toString(pactLevel));
|
|
strFindReplace(pactText, "%hp", toString(pactLose.LoseHitPointsLevel));
|
|
strFindReplace(pactText, "%sta", toString(pactLose.LoseStaminaLevel));
|
|
strFindReplace(pactText, "%sap", toString(pactLose.LoseSapLevel));
|
|
strFindReplace(pactText, "%skill", toString(pactLose.LoseSkillsLevel));
|
|
|
|
// **** setup the text
|
|
setHelpText(setup, pactText);
|
|
return;
|
|
};
|
|
|
|
// ***************************************************************************
|
|
static void setupMissionHelp(CSheetHelpSetup &setup)
|
|
{
|
|
// get the calling item
|
|
if (!setup.SrcSheet)
|
|
{
|
|
nlwarning("<setupMissionHelp> no caller sheet found.");
|
|
return;
|
|
}
|
|
|
|
// setup the item.
|
|
// CDBCtrlSheet *ctrlMission= dynamic_cast<CDBCtrlSheet*>(setup.HelpWindow->getCtrl("ctrl_slot"));
|
|
if(setup.DestSheet)
|
|
{
|
|
setup.SrcSheet->copyAspect(setup.DestSheet);
|
|
}
|
|
|
|
// get detail text id from db
|
|
if (!setup.SrcSheet->getRootBranch()) return;
|
|
CCDBNodeLeaf *detailTextLeaf = dynamic_cast<CCDBNodeLeaf *>(setup.SrcSheet->getRootBranch()->getNode(ICDBNode::CTextId("DETAIL_TEXT")));
|
|
if (!detailTextLeaf) return;
|
|
|
|
// Change the title according to Mission Client Type
|
|
MISSION_DESC::TIconId iconId= (MISSION_DESC::TIconId)setup.SrcSheet->getSheetId();
|
|
MISSION_DESC::TClientMissionType mType= MISSION_DESC::getClientMissionType(iconId);
|
|
if(mType==MISSION_DESC::Mission)
|
|
setupHelpTitle(setup.HelpWindow, CI18N::get("uihelpMission"));
|
|
else if(mType==MISSION_DESC::ZCRequirement)
|
|
setupHelpTitle(setup.HelpWindow, CI18N::get("uihelpZCRequirement"));
|
|
else if(mType==MISSION_DESC::BuildingRequirement)
|
|
setupHelpTitle(setup.HelpWindow, CI18N::get("uihelpBuildingRequirement"));
|
|
else if(mType==MISSION_DESC::ZCCharge)
|
|
setupHelpTitle(setup.HelpWindow, CI18N::get("uihelpZCCharge"));
|
|
else if(mType==MISSION_DESC::Building)
|
|
setupHelpTitle(setup.HelpWindow, CI18N::get("uihelpBuilding"));
|
|
else if(mType==MISSION_DESC::RMBuy)
|
|
setupHelpTitle(setup.HelpWindow, CI18N::get("uihelpRMBuy"));
|
|
else if(mType==MISSION_DESC::RMUpgrade)
|
|
setupHelpTitle(setup.HelpWindow, CI18N::get("uihelpRMUpgrade"));
|
|
else
|
|
setupHelpTitle(setup.HelpWindow, CI18N::get("uihelpMission") );
|
|
|
|
|
|
// **** setup the text id
|
|
setHelpTextID(setup, detailTextLeaf->getValue32());
|
|
|
|
CViewTextID *viewTextTitleID = dynamic_cast<CViewTextID *>(setup.HelpWindow->getView("text_title_id"));
|
|
if (viewTextTitleID != NULL)
|
|
{
|
|
CCDBNodeLeaf *titleTextLeaf = dynamic_cast<CCDBNodeLeaf*>(setup.SrcSheet->getRootBranch()->getNode(ICDBNode::CTextId("TEXT")));
|
|
if (titleTextLeaf == NULL) return;
|
|
viewTextTitleID->setTextId(titleTextLeaf->getValue32());
|
|
}
|
|
|
|
};
|
|
|
|
// ***************************************************************************
|
|
void refreshMissionHelp(CSheetHelpSetup &setup, const CPrerequisitInfos &infos)
|
|
{
|
|
static NLMISC::CRGBA orange(250,150,0);
|
|
static NLMISC::CRGBA darkGreen(0,150,0);
|
|
|
|
if (infos.Prerequisits.size() > 15)
|
|
{
|
|
// blabla
|
|
}
|
|
|
|
bool conditionValidated = false;
|
|
|
|
// NB : firts prerequisit MUST be an 'and'
|
|
for (uint i = 0 ; i < infos.Prerequisits.size() ; )
|
|
{
|
|
nlassert(infos.Prerequisits[i].IsMandatory);
|
|
conditionValidated = infos.Prerequisits[i].Validated;
|
|
|
|
// check 'or' conditions, if any or (or enclosing 'and') is validated then global block is matched
|
|
uint orIndexMax;
|
|
for ( orIndexMax = i+1 ; orIndexMax < infos.Prerequisits.size() ; ++orIndexMax )
|
|
{
|
|
if (infos.Prerequisits[orIndexMax].IsMandatory)
|
|
break;
|
|
|
|
if (infos.Prerequisits[orIndexMax].Validated)
|
|
conditionValidated = true;
|
|
|
|
}
|
|
// fill text, choose color according to conditions and block
|
|
for (uint j = i ; j < orIndexMax ; ++j )
|
|
{
|
|
const std::string text = setup.HelpWindow->getId() + ":content:scroll_text_id:text_list:" + NLMISC::toString("text_%u",j+1);
|
|
CViewText *viewText = dynamic_cast<CViewText *>(setup.HelpWindow->getElement(text));
|
|
if (viewText)
|
|
{
|
|
viewText->setActive(true);
|
|
if (infos.Prerequisits[j].IsMandatory)
|
|
viewText->setHardText("uiMissionAnd");
|
|
else
|
|
viewText->setHardText("uiMissionOr");
|
|
}
|
|
|
|
const std::string textId = setup.HelpWindow->getId() + ":content:scroll_text_id:text_list:" + NLMISC::toString("text_id_prereq_%u",j+1);
|
|
|
|
CViewTextID *viewTextID = dynamic_cast<CViewTextID *>(setup.HelpWindow->getElement(textId));
|
|
if(viewTextID)
|
|
{
|
|
// not validated : change color to red
|
|
if (!infos.Prerequisits[j].Validated)
|
|
{
|
|
if (!conditionValidated)
|
|
viewTextID->setColor(orange);
|
|
else
|
|
viewTextID->setColor(CRGBA::White);
|
|
}
|
|
else
|
|
viewTextID->setColor(darkGreen);
|
|
|
|
viewTextID->setActive(true);
|
|
viewTextID->setTextId(infos.Prerequisits[j].Description);
|
|
}
|
|
}
|
|
|
|
// go to next 'and' statement (or end)
|
|
i = orIndexMax;
|
|
}
|
|
|
|
// inactivate other lines
|
|
for (uint i = (uint)infos.Prerequisits.size(); i < 15 ; ++i)
|
|
{
|
|
const std::string text = setup.HelpWindow->getId() + ":content:scroll_text_id:text_list:" + NLMISC::toString("text_%u",i+1);
|
|
CViewText *viewText = dynamic_cast<CViewText *>(setup.HelpWindow->getElement(text));
|
|
if (viewText)
|
|
viewText->setActive(false);
|
|
|
|
const std::string textId = setup.HelpWindow->getId() + ":content:scroll_text_id:text_list:" + NLMISC::toString("text_id_prereq_%u",i+1);
|
|
CViewTextID *viewTextID = dynamic_cast<CViewTextID *>(setup.HelpWindow->getElement(textId));
|
|
if(viewTextID)
|
|
viewTextID->setActive(false);
|
|
}
|
|
|
|
if (!setup.ScrollTextGroup.empty())
|
|
{
|
|
CInterfaceGroup *viewTextGroup = setup.HelpWindow->getGroup(setup.ScrollTextGroup);
|
|
if (viewTextGroup) viewTextGroup->setActive(false);
|
|
}
|
|
CInterfaceGroup *viewTextGroup = setup.HelpWindow->getGroup(setup.ScrollTextIdGroup);
|
|
if (viewTextGroup) viewTextGroup->setActive(true);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
class CPlayerShardNameRemover : public IOnReceiveTextId
|
|
{
|
|
virtual void onReceiveTextId(ucstring &str)
|
|
{
|
|
str= CEntityCL::removeShardFromName(str);
|
|
}
|
|
};
|
|
static CPlayerShardNameRemover PlayerShardNameRemover;
|
|
|
|
// ***************************************************************************
|
|
void setupCreatorName(CSheetHelpSetup &setup)
|
|
{
|
|
if (!setup.HelpWindow) return;
|
|
CViewTextID *vtid = dynamic_cast<CViewTextID*>(setup.HelpWindow->getView(setup.CreatorViewTextID));
|
|
CViewText *vthd = dynamic_cast<CViewText*>(setup.HelpWindow->getView("creator_header"));
|
|
if (vtid != NULL)
|
|
{
|
|
// if not an item, disable the view
|
|
if(!setup.SrcSheet || setup.SrcSheet->getType()!=CCtrlSheetInfo::SheetType_Item )
|
|
{
|
|
// important else a brick could display a creator name....
|
|
vtid->setActive(false);
|
|
if(vthd)
|
|
vthd->setActive(false);
|
|
}
|
|
else
|
|
{
|
|
// get the CreatorTextID
|
|
uint32 itemSlotId= getInventory().getItemSlotId(setup.SrcSheet);
|
|
uint32 creatorTextId= getInventory().getItemInfo(itemSlotId).CreatorName;
|
|
|
|
vtid->setActive(true);
|
|
vtid->setTextId(creatorTextId);
|
|
vtid->setOnReceiveTextId(&PlayerShardNameRemover);
|
|
if(vthd)
|
|
vthd->setActive(creatorTextId!=0);
|
|
}
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
// Outpost Building Help
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
|
|
|
|
// ***************************************************************************
|
|
void fillOutpostBuildingListItem(const std::vector<NLMISC::CSheetId> &mps, IListSheetBase *listItem, uint32 qualityLevel)
|
|
{
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
|
|
if(listItem)
|
|
{
|
|
listItem->setActive(true);
|
|
string branchBase= listItem->getDbBranchName();
|
|
// setup mps
|
|
uint i;
|
|
for(i=0;i<mps.size();i++)
|
|
{
|
|
CCDBNodeLeaf *node= pIM->getDbProp(toString("%s:%d:SHEET", branchBase.c_str(), i));
|
|
if(node)
|
|
node->setValue32(mps[i].asInt());
|
|
node= pIM->getDbProp(toString("%s:%d:QUALITY", branchBase.c_str(), i), false);
|
|
if(node)
|
|
node->setValue32(qualityLevel);
|
|
node= pIM->getDbProp(toString("%s:%d:PREREQUISIT_VALID", branchBase.c_str(), i), false);
|
|
if(node)
|
|
node->setValue32(1);
|
|
}
|
|
// Reset other to 0.
|
|
for(;;i++)
|
|
{
|
|
CCDBNodeLeaf *node= pIM->getDbProp(toString("%s:%d:SHEET", branchBase.c_str(), i), false);
|
|
if(node)
|
|
node->setValue32(0);
|
|
else
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void setupOutpostBuildingHelp(CSheetHelpSetup &setup)
|
|
{
|
|
// get the calling item
|
|
if (!setup.SrcSheet)
|
|
{
|
|
nlwarning("<setupOutpostBuildingHelp> no caller sheet found.");
|
|
return;
|
|
}
|
|
|
|
// setup the item.
|
|
if(setup.DestSheet)
|
|
{
|
|
setup.SrcSheet->copyAspect(setup.DestSheet);
|
|
}
|
|
|
|
const COutpostBuildingSheet *pOBS = setup.SrcSheet->asOutpostBuildingSheet();
|
|
if (pOBS == NULL)
|
|
{
|
|
nlwarning("<setupOutpostBuildingHelp> can't get outpost building sheet.");
|
|
return;
|
|
}
|
|
|
|
setupHelpTitle(setup.HelpWindow, CI18N::get("uihelpOutpostBuilding"));
|
|
|
|
ucstring sOBText;
|
|
|
|
sOBText = CI18N::get("uihelpOBFormat_"+COutpostBuildingSheet::toString(pOBS->OBType));
|
|
|
|
{
|
|
ucstring timeText;
|
|
timeText = toString(pOBS->CostTime/60) + CI18N::get("uiBotChatTimeMinute");
|
|
if ((pOBS->CostTime % 60) != 0)
|
|
timeText += toString(pOBS->CostTime%60) + CI18N::get("uiBotChatTimeSecond");
|
|
|
|
strFindReplace(sOBText, "%costtime", timeText);
|
|
}
|
|
|
|
strFindReplace(sOBText, "%costdapper", toString(pOBS->CostDapper));
|
|
|
|
// Set name of the building
|
|
strFindReplace(sOBText, "%name", STRING_MANAGER::CStringManagerClient::getOutpostBuildingLocalizedName(pOBS->Id));
|
|
|
|
// For driller, set lvl
|
|
strFindReplace(sOBText, "%lvl", toString(pOBS->MPLevelOfHighestExtractRate));
|
|
|
|
// **** setup the text
|
|
setHelpText(setup, sOBText);
|
|
|
|
// **** raw materials stats for driller to buy
|
|
if(pOBS->OBType==COutpostBuildingSheet::OB_Driller)
|
|
{
|
|
// Get the list of bricks container
|
|
IListSheetBase *listItem= dynamic_cast<IListSheetBase*>(setup.HelpWindow->getElement(setup.HelpWindow->getId()+setup.PrefixForExtra+INFO_LIST_ITEM));
|
|
|
|
// setup the bricks
|
|
fillOutpostBuildingListItem(pOBS->Mps, listItem, pOBS->MPLevelOfHighestExtractRate);
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
// SBrick / Phrase help
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
|
|
|
|
// ***************************************************************************
|
|
static bool getAuraDisabledState(CDBCtrlSheet *cs)
|
|
{
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
|
|
if(!cs)
|
|
return false;
|
|
|
|
// Get the DISABLED DBprop
|
|
string db= cs->getSheet() + ":DISABLED";
|
|
CCDBNodeLeaf *node= pIM->getDbProp(db, false);
|
|
return node && node->getValue32()!=0;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
static sint getBonusMalusSpecialTT(CDBCtrlSheet *cs)
|
|
{
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
|
|
if(!cs)
|
|
return 0;
|
|
|
|
// Get the SPECIAL_TOOLTIP DBprop
|
|
string db= cs->getSheet() + ":SPECIAL_TOOLTIP";
|
|
return pIM->getDbValue32 (db);
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
void getSabrinaBrickText(CSBrickSheet *pBR, ucstring &brickText)
|
|
{
|
|
if(!pBR)
|
|
return;
|
|
|
|
// *** get the formated text according to Brick type.
|
|
if( pBR->isFaber() && pBR->isMandatory() )
|
|
{
|
|
brickText= CI18N::get("uihelpBrickFaberFormat");
|
|
}
|
|
else
|
|
{
|
|
brickText= CI18N::get("uihelpBrickFormat");
|
|
}
|
|
|
|
// *** Basics
|
|
// Level
|
|
strFindReplace(brickText, "%lvl", toString(pBR->Level));
|
|
// Kill the whole text between %ks, if the skill is unknown
|
|
const ucstring killSkill("%ks");
|
|
if( pBR->getSkill()==SKILLS::unknown )
|
|
{
|
|
ucstring::size_type pos0= brickText.find(killSkill);
|
|
if(pos0 != ucstring::npos)
|
|
{
|
|
ucstring::size_type pos1= brickText.find(killSkill, pos0 + killSkill.size() );
|
|
if(pos1 != ucstring::npos)
|
|
brickText.replace(pos0, pos1+killSkill.size()-pos0, ucstring() );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// remove %ks tag
|
|
while(strFindReplace(brickText, "%ks", ""));
|
|
|
|
// Skill, or array of skill for combat
|
|
if(pBR->UsedSkills.size()==1)
|
|
strFindReplace(brickText, "%skill", CStringManagerClient::getSkillLocalizedName(pBR->getSkill()));
|
|
else
|
|
{
|
|
ucstring fullSkillText;
|
|
bool first= true;
|
|
for(uint i=0;i<pBR->UsedSkills.size();i++)
|
|
{
|
|
SKILLS::ESkills skill= pBR->UsedSkills[i];
|
|
if(skill!=SKILLS::unknown)
|
|
{
|
|
if(!first)
|
|
fullSkillText+= CI18N::get("uihelpBrickCombatSkillSeparator");
|
|
first= false;
|
|
fullSkillText+= CStringManagerClient::getSkillLocalizedName(skill);
|
|
}
|
|
}
|
|
|
|
strFindReplace(brickText, "%skill", fullSkillText);
|
|
}
|
|
}
|
|
|
|
// Cost
|
|
strFindReplace(brickText, "%cost", toString(pBR->SabrinaCost));
|
|
// Header Cost: cost or credit
|
|
if(pBR->SabrinaCost>=0)
|
|
strFindReplace(brickText, "%hcost", CI18N::get("uihelpSabrinaCost") );
|
|
else
|
|
strFindReplace(brickText, "%hcost", CI18N::get("uihelpSabrinaCredit") );
|
|
|
|
// Relative Cost
|
|
// Kill the whole text between %krc, if the relative cost is 0
|
|
if(pBR->SabrinaRelativeCost==0.f)
|
|
{
|
|
const ucstring killRC("%krc");
|
|
ucstring::size_type pos0= brickText.find(killRC);
|
|
if(pos0 != ucstring::npos)
|
|
{
|
|
ucstring::size_type pos1= brickText.find(killRC, pos0 + killRC.size() );
|
|
if(pos1 != ucstring::npos)
|
|
brickText.replace(pos0, pos1+killRC.size()-pos0, ucstring() );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// remove %krc tag
|
|
while(strFindReplace(brickText, "%krc", ""));
|
|
|
|
|
|
strFindReplace(brickText, "%relative_cost", toPercentageText(pBR->SabrinaRelativeCost)+string("%"));
|
|
// Header Cost: cost or credit
|
|
if(pBR->SabrinaRelativeCost>=0.f)
|
|
strFindReplace(brickText, "%hrel_cost", CI18N::get("uihelpSabrinaRelCost") );
|
|
else
|
|
strFindReplace(brickText, "%hrel_cost", CI18N::get("uihelpSabrinaRelCredit") );
|
|
}
|
|
|
|
// Description
|
|
strFindReplace(brickText, "%desc", CStringManagerClient::getSBrickLocalizedDescription(pBR->Id) );
|
|
|
|
// *** Faber
|
|
if( pBR->isFaber() && pBR->isMandatory() )
|
|
{
|
|
// Display the ToolType required.
|
|
strFindReplace(brickText, "%tool", CI18N::get("tool"+TOOL_TYPE::toString(pBR->FaberPlan.ToolType)));
|
|
|
|
// --- Display MP itempart information
|
|
if(pBR->FaberPlan.ItemPartMps.empty())
|
|
{
|
|
strFindReplace(brickText, "%mpinfo", CI18N::get("uihelpMpNone"));
|
|
}
|
|
else
|
|
{
|
|
ucstring mpInfo;
|
|
for(uint i=0;i<pBR->FaberPlan.ItemPartMps.size();i++)
|
|
{
|
|
CSBrickSheet::CFaberPlan::CItemPartMP &mpSlot= pBR->FaberPlan.ItemPartMps[i];
|
|
|
|
// Display the part this slot build.
|
|
mpInfo+= "@{T4}";
|
|
mpInfo+= RM_FABER_TYPE::toLocalString(mpSlot.FaberTypeFilter);
|
|
mpInfo+= "\n";
|
|
}
|
|
// replace in brickText
|
|
strFindReplace(brickText, "%mpinfo", mpInfo);
|
|
}
|
|
|
|
// --- Display MP formula information
|
|
if(pBR->FaberPlan.FormulaMps.empty())
|
|
{
|
|
strFindReplace(brickText, "%mpformula", CI18N::get("uihelpMpNone"));
|
|
}
|
|
else
|
|
{
|
|
ucstring mpInfo;
|
|
for(uint i=0;i<pBR->FaberPlan.FormulaMps.size();i++)
|
|
{
|
|
CSBrickSheet::CFaberPlan::CFormulaMP &mpSlot= pBR->FaberPlan.FormulaMps[i];
|
|
|
|
// Display the required item
|
|
mpInfo+= "@{T4}";
|
|
mpInfo+= STRING_MANAGER::CStringManagerClient::getItemLocalizedName(mpSlot.ItemRequired);
|
|
mpInfo+= "\n";
|
|
}
|
|
// replace in brickText
|
|
strFindReplace(brickText, "%mpformula", mpInfo);
|
|
}
|
|
}
|
|
|
|
// *** Magic
|
|
ucstring magicResistStr;
|
|
// Has Some Magic Resistance setuped?
|
|
if( pBR->isMagic() && pBR->MagicResistType!=RESISTANCE_TYPE::None)
|
|
{
|
|
magicResistStr= CI18N::get("uihelpBrickMagicResist");
|
|
strFindReplace(magicResistStr, "%t", CI18N::get("rs"+RESISTANCE_TYPE::toString(pBR->MagicResistType) ));
|
|
}
|
|
strFindReplace(brickText, "%magicresist", magicResistStr);
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
/*
|
|
* Used both by setupSabrinaPhraseHelp() and setupEnchantedItem()
|
|
*/
|
|
void fillSabrinaPhraseListBrick(const CSPhraseCom &phrase, IListSheetBase *listBrick)
|
|
{
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
CSBrickManager *pBM= CSBrickManager::getInstance();
|
|
|
|
if(listBrick)
|
|
{
|
|
listBrick->setActive(true);
|
|
string branchBase= listBrick->getDbBranchName();
|
|
// setup phrase bricks
|
|
uint i;
|
|
for(i=0;i<phrase.Bricks.size();i++)
|
|
{
|
|
CCDBNodeLeaf *node= pIM->getDbProp(toString("%s:%d:SHEET", branchBase.c_str(), i));
|
|
if(node)
|
|
node->setValue32(phrase.Bricks[i].asInt());
|
|
|
|
// For requirements bricks, update the LOCKED state
|
|
node= pIM->getDbProp(toString("%s:%d:LOCKED", branchBase.c_str(), i));
|
|
if(node)
|
|
{
|
|
if(pBM->isBrickKnown(phrase.Bricks[i]))
|
|
node->setValue32(0);
|
|
else
|
|
// 2 to redify it
|
|
node->setValue32(2);
|
|
}
|
|
}
|
|
// Reset other to 0.
|
|
for(;;i++)
|
|
{
|
|
CCDBNodeLeaf *node= pIM->getDbProp(toString("%s:%d:SHEET", branchBase.c_str(), i), false);
|
|
if(node)
|
|
node->setValue32(0);
|
|
else
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void hideListBrickHeader(CSheetHelpSetup &setup)
|
|
{
|
|
// CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
|
|
// get the header text
|
|
CViewText *view= dynamic_cast<CViewText*>(setup.HelpWindow->getElement(setup.HelpWindow->getId()+setup.PrefixForExtra+INFO_LIST_BRICK_HEADER));
|
|
if(view)
|
|
{
|
|
view->setActive(false);
|
|
}
|
|
}
|
|
|
|
void setupListBrickHeader(CSheetHelpSetup &setup)
|
|
{
|
|
// get the header text
|
|
CViewText *view= dynamic_cast<CViewText*>(setup.HelpWindow->getElement(setup.HelpWindow->getId()+setup.PrefixForExtra+INFO_LIST_BRICK_HEADER));
|
|
if(view)
|
|
{
|
|
view->setActive(true);
|
|
view->setTextFormatTaged(CI18N::get("uihelpPhraseHeaderBricks"));
|
|
}
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
/* phraseSheetId: not null if comes from a .sphrase Sheet, used to show progression info
|
|
*/
|
|
void setupSabrinaPhraseHelp(CSheetHelpSetup &setup, const CSPhraseCom &phrase, uint32 phraseSheetId)
|
|
{
|
|
CSPhraseManager *pPM = CSPhraseManager::getInstance();
|
|
|
|
if(!setup.SrcSheet || phrase.empty())
|
|
return;
|
|
|
|
if(!setup.HelpWindow)
|
|
return;
|
|
|
|
// setup the item.
|
|
if(setup.DestSheet)
|
|
{
|
|
setup.SrcSheet->copyAspect(setup.DestSheet);
|
|
}
|
|
|
|
// **** setup the phrase Text info
|
|
setupHelpTitle(setup.HelpWindow, phrase.Name);
|
|
|
|
// get the phraseText
|
|
ucstring phraseText;
|
|
// if required, add the .sphrase requirements.
|
|
// NB: don't add if from bot chat validation (unsusefull cause already filtered by server)
|
|
pPM->buildPhraseDesc(phraseText, phrase, phraseSheetId, !setup.FromBotChat);
|
|
|
|
|
|
// **** If interesting to do it, setup the brick list
|
|
if( pPM->allowListBrickInHelp(phrase) )
|
|
{
|
|
// Get the list of bricks container
|
|
IListSheetBase *listBrick= dynamic_cast<IListSheetBase*>(setup.HelpWindow->getElement(setup.HelpWindow->getId()+setup.PrefixForExtra+INFO_LIST_BRICK));
|
|
|
|
// if exist, setup text header
|
|
if(listBrick)
|
|
setupListBrickHeader(setup);
|
|
|
|
// setup the bricks
|
|
fillSabrinaPhraseListBrick(phrase, listBrick);
|
|
}
|
|
|
|
|
|
// **** For .sphrase only, setup the requirement bricks
|
|
if(phraseSheetId!=0)
|
|
{
|
|
IListSheetBase *listBrick= dynamic_cast<IListSheetBase*>(setup.HelpWindow->getElement(setup.HelpWindow->getId()+setup.PrefixForExtra+INFO_LIST_BRICK_REQUIREMENT));
|
|
if(listBrick)
|
|
{
|
|
CSPhraseCom dummyPhrase;
|
|
pPM->buildPhraseBrickRequirement(phraseSheetId, dummyPhrase.Bricks);
|
|
if(!dummyPhrase.Bricks.empty())
|
|
{
|
|
phraseText+= CI18N::get("uihelpPhraseBrickRequirement");
|
|
fillSabrinaPhraseListBrick(dummyPhrase, listBrick);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// **** setup the final text
|
|
setHelpText(setup, phraseText);
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
static void setupSabrinaBrickHelp(CSheetHelpSetup &setup, bool auraDisabled)
|
|
{
|
|
if (!setup.HelpWindow) return;
|
|
|
|
// get the calling item
|
|
if (!setup.SrcSheet || !setup.SrcSheet->isSBrick() )
|
|
{
|
|
nlwarning("<CHandlerOpenBrickHelp::execute> no caller sheet found");
|
|
return;
|
|
}
|
|
|
|
// If the sheet is 0, don't open!
|
|
if(setup.SrcSheet->getSheetId()==0)
|
|
return;
|
|
|
|
// setup the item.
|
|
if(setup.DestSheet)
|
|
{
|
|
setup.SrcSheet->copyAspect(setup.DestSheet);
|
|
}
|
|
|
|
|
|
// **** setup the brick Text info
|
|
ucstring brickText;
|
|
CSBrickManager *pBM= CSBrickManager::getInstance();
|
|
CSBrickSheet *pBR= pBM->getBrick(CSheetId(setup.SrcSheet->getSheetId()));
|
|
if(pBR)
|
|
{
|
|
const ucstring title(CStringManagerClient::getSBrickLocalizedName(pBR->Id));
|
|
setupHelpTitle(setup.HelpWindow, title);
|
|
|
|
// add brick info
|
|
getSabrinaBrickText(pBR, brickText);
|
|
|
|
// Append special Aura Info
|
|
if(auraDisabled)
|
|
{
|
|
brickText+= CI18N::get("uihelpAuraDisabled");
|
|
}
|
|
}
|
|
|
|
|
|
// **** setup the text
|
|
setHelpText(setup, brickText);
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void CSheetHelpSetup::setupDefaultIDs()
|
|
{
|
|
ViewText = "text";
|
|
ViewTextID = "text_id";
|
|
ScrollTextGroup = "scroll_text";
|
|
ScrollTextIdGroup = "scroll_text_id";
|
|
CreatorViewTextID = "creator";
|
|
PrefixForExtra= ":content:scroll_text:text_list:";
|
|
FromBotChat= false;
|
|
}
|
|
|
|
// ***************************************************************************
|
|
void setupSheetHelp(CSheetHelpSetup &setup)
|
|
{
|
|
if (!setup.SrcSheet) return;
|
|
switch(setup.SrcSheet->getType())
|
|
{
|
|
case CCtrlSheetInfo::SheetType_Skill: setupSkillToTradeHelp(setup); break;
|
|
case CCtrlSheetInfo::SheetType_Item: setupItemHelp(setup); break;
|
|
case CCtrlSheetInfo::SheetType_Pact: setupPactHelp(setup); break;
|
|
case CCtrlSheetInfo::SheetType_Mission: setupMissionHelp(setup); break;
|
|
case CCtrlSheetInfo::SheetType_SPhrase:
|
|
{
|
|
CSPhraseCom phrase;
|
|
CSPhraseManager *pPM= CSPhraseManager::getInstance();
|
|
uint32 phraseSheetId= setup.SrcSheet->getSheetId();
|
|
pPM->buildPhraseFromSheet(phrase, phraseSheetId);
|
|
setupSabrinaPhraseHelp(setup, phrase, phraseSheetId);
|
|
break;
|
|
}
|
|
case CCtrlSheetInfo::SheetType_OutpostBuilding: setupOutpostBuildingHelp(setup); break;
|
|
default:
|
|
nlwarning("<CHandlerOpenHelpAuto> Bad item type.");
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
class CHandlerOpenPhraseIdHelp : public IActionHandler
|
|
{
|
|
public:
|
|
virtual void execute (CCtrlBase *pCaller, const string &Params)
|
|
{
|
|
CDBCtrlSheet *cs = dynamic_cast<CDBCtrlSheet*>(pCaller);
|
|
if (cs != NULL && cs->getType()==CCtrlSheetInfo::SheetType_SPhraseId)
|
|
{
|
|
// Get the CSPhraseCom pointed.
|
|
sint32 id= cs->getSPhraseId();
|
|
if(id!=0)
|
|
{
|
|
CSPhraseManager *pPM= CSPhraseManager::getInstance();
|
|
CInterfaceGroup *group = CInterfaceHelp::activateNextWindow(cs);
|
|
if (!group) return;
|
|
CSheetHelpSetup setup;
|
|
setup.setupDefaultIDs();
|
|
setup.HelpWindow = group;
|
|
setup.SrcSheet = cs;
|
|
setup.DestSheet = dynamic_cast<CDBCtrlSheet*>(group->getCtrl("ctrl_slot"));
|
|
setupSabrinaPhraseHelp(setup, pPM->getPhrase(id), 0);
|
|
}
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerOpenPhraseIdHelp, "open_phraseid_help");
|
|
|
|
|
|
// ***************************************************************************
|
|
class CHandlerOpenSBrickHelp : public IActionHandler
|
|
{
|
|
public:
|
|
virtual void execute (CCtrlBase *pCaller, const string &Params)
|
|
{
|
|
CSBrickManager *pBM= CSBrickManager::getInstance();
|
|
CDBCtrlSheet *cs = dynamic_cast<CDBCtrlSheet*>(pCaller);
|
|
|
|
// No Info if bad control / no sheetid
|
|
if(!cs || cs->getSheetId()==0)
|
|
return;
|
|
CSheetId brickSheetId= CSheetId(cs->getSheetId());
|
|
// No Info for the special "Remove Me" brick
|
|
if(brickSheetId == pBM->getInterfaceRemoveBrick() )
|
|
return;
|
|
// No Info for special "XP catalyzer" or "PVP oupost" interface brick in bonus malus window
|
|
string brickName= brickSheetId.toString();
|
|
string xpCatBrickPrefix= "big_xpcat_";
|
|
string ringXpCatBrickPrefix= "big_ring_xpcat_";
|
|
string pvpOutpostBrickPrefix= "big_outpost_pvp_";
|
|
if(brickName.compare(0, xpCatBrickPrefix.size(), xpCatBrickPrefix)==0)
|
|
return;
|
|
if(brickName.compare(0, ringXpCatBrickPrefix.size(), ringXpCatBrickPrefix)==0)
|
|
return;
|
|
if(brickName.compare(0, pvpOutpostBrickPrefix.size(), pvpOutpostBrickPrefix)==0)
|
|
return;
|
|
|
|
// Else, Ok open the window
|
|
{
|
|
// get the forceKeep param (for Action info)
|
|
sint forceKeepWindow= -1;
|
|
string forceKeepWindowStr= getParam(Params, "force_keep");
|
|
if(!forceKeepWindowStr.empty())
|
|
fromString(forceKeepWindowStr, forceKeepWindow);
|
|
// get the Aura Disabled param
|
|
bool auraDisabled= false;
|
|
string auraDisabledStr= getParam(Params, "test_aura_disabled");
|
|
bool tmpAuraDisabled;
|
|
fromString(auraDisabledStr, tmpAuraDisabled);
|
|
if(tmpAuraDisabled)
|
|
auraDisabled= getAuraDisabledState(cs);
|
|
|
|
// open
|
|
CInterfaceGroup *group = CInterfaceHelp::activateNextWindow(cs, forceKeepWindow);
|
|
if (!group) return;
|
|
CSheetHelpSetup setup;
|
|
setup.setupDefaultIDs();
|
|
setup.HelpWindow = group;
|
|
setup.SrcSheet = cs;
|
|
setup.DestSheet = dynamic_cast<CDBCtrlSheet*>(group->getCtrl("ctrl_slot"));
|
|
setupSabrinaBrickHelp(setup, auraDisabled);
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerOpenSBrickHelp, "open_sbrick_help");
|
|
|
|
|
|
|
|
// ***************************************************************************
|
|
class CHandlerOnCloseHelp : public IActionHandler
|
|
{
|
|
public:
|
|
virtual void execute (CCtrlBase *pCaller, const string &Params)
|
|
{
|
|
// Remove the waiter for special ItemInfo
|
|
uint index;
|
|
fromString(Params, index);
|
|
CInterfaceHelp::removeWaiterItemInfo(index);
|
|
CInterfaceHelp::removeWaiterMissionInfo(index);
|
|
|
|
// unpuhsed the "Keep" button.
|
|
CInterfaceHelp::setKeepMode(index, false);
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerOnCloseHelp, "on_close_help");
|
|
|
|
|
|
// ***************************************************************************
|
|
class CHandlerHelpKeep : public IActionHandler
|
|
{
|
|
public:
|
|
virtual void execute (CCtrlBase *pCaller, const string &Params)
|
|
{
|
|
// flag
|
|
uint index;
|
|
fromString(Params, index);
|
|
CInterfaceHelp::changeKeepMode(index);
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerHelpKeep, "help_keep");
|
|
|
|
|
|
// ***************************************************************************
|
|
class CHandlerHelpResetPos : public IActionHandler
|
|
{
|
|
public:
|
|
virtual void execute(CCtrlBase *pCaller, const string &Params)
|
|
{
|
|
sint y;
|
|
fromString(getParam(Params, "y"), y);
|
|
// update WindowList if possible
|
|
CInterfaceHelp::resetWindowPos(y);
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerHelpResetPos, "help_reset_pos");
|
|
|
|
|
|
|
|
//-----------------------------------------------
|
|
// setConsoModSuccessTooltip
|
|
//-----------------------------------------------
|
|
void setConsoModSuccessTooltip( CDBCtrlSheet *cs )
|
|
{
|
|
if(!cs)
|
|
return;
|
|
|
|
CInterfaceManager * pIM = CInterfaceManager::getInstance();
|
|
|
|
CCDBNodeLeaf * nodeSM = NULL;
|
|
ucstring ustr;
|
|
if( CSheetId(cs->getSheetId()).toString() == "mod_melee_success.sbrick" )
|
|
{
|
|
ustr = CI18N::get("uittModMeleeSuccess");
|
|
nodeSM = pIM->getDbProp("SERVER:CHARACTER_INFO:SUCCESS_MODIFIER:MELEE", false);
|
|
}
|
|
else
|
|
if( CSheetId(cs->getSheetId()).toString() == "mod_range_success.sbrick" )
|
|
{
|
|
ustr = CI18N::get("uittModRangeSuccess");
|
|
nodeSM = pIM->getDbProp("SERVER:CHARACTER_INFO:SUCCESS_MODIFIER:RANGE", false);
|
|
}
|
|
else
|
|
if( CSheetId(cs->getSheetId()).toString() == "mod_craft_success.sbrick" )
|
|
{
|
|
ustr = CI18N::get("uittModCraftSuccess");
|
|
nodeSM = pIM->getDbProp("SERVER:CHARACTER_INFO:SUCCESS_MODIFIER:CRAFT", false);
|
|
}
|
|
else
|
|
if( CSheetId(cs->getSheetId()).toString() == "mod_defense_success.sbrick" )
|
|
{
|
|
ustr = CI18N::get("uittModDefenseSuccess");
|
|
nodeSM = pIM->getDbProp("SERVER:CHARACTER_INFO:SUCCESS_MODIFIER:DODGE", false);
|
|
}
|
|
else
|
|
if( CSheetId(cs->getSheetId()).toString() == "mod_dodge_success.sbrick" )
|
|
{
|
|
ustr = CI18N::get("uittModDodgeSuccess");
|
|
nodeSM = pIM->getDbProp("SERVER:CHARACTER_INFO:SUCCESS_MODIFIER:DODGE", false);
|
|
}
|
|
else
|
|
if( CSheetId(cs->getSheetId()).toString() == "mod_parry_success.sbrick" )
|
|
{
|
|
ustr = CI18N::get("uittModParrySuccess");
|
|
nodeSM = pIM->getDbProp("SERVER:CHARACTER_INFO:SUCCESS_MODIFIER:PARRY", false);
|
|
}
|
|
else
|
|
if( CSheetId(cs->getSheetId()).toString() == "mod_forage_success.sbrick" )
|
|
{
|
|
ustr = CI18N::get("uittModForageSuccess");
|
|
nodeSM = pIM->getDbProp("SERVER:CHARACTER_INFO:SUCCESS_MODIFIER:ECO:"+toString((uint8)ECOSYSTEM::common_ecosystem)+":FORAGE", false);
|
|
}
|
|
else
|
|
if( CSheetId(cs->getSheetId()).toString() == "mod_desert_forage_success.sbrick" )
|
|
{
|
|
ustr = CI18N::get("uittModDesertForageSuccess");
|
|
nodeSM = pIM->getDbProp("SERVER:CHARACTER_INFO:SUCCESS_MODIFIER:ECO:"+toString((uint8)ECOSYSTEM::desert)+":FORAGE", false);
|
|
}
|
|
else
|
|
if( CSheetId(cs->getSheetId()).toString() == "mod_forest_forage_success.sbrick" )
|
|
{
|
|
ustr = CI18N::get("uittModForestForageSuccess");
|
|
nodeSM = pIM->getDbProp("SERVER:CHARACTER_INFO:SUCCESS_MODIFIER:ECO:"+toString((uint8)ECOSYSTEM::forest)+":FORAGE", false);
|
|
}
|
|
else
|
|
if( CSheetId(cs->getSheetId()).toString() == "mod_lacustre_forage_success.sbrick" )
|
|
{
|
|
ustr = CI18N::get("uittModLacustreForageSuccess");
|
|
nodeSM = pIM->getDbProp("SERVER:CHARACTER_INFO:SUCCESS_MODIFIER:ECO:"+toString((uint8)ECOSYSTEM::lacustre)+":FORAGE", false);
|
|
}
|
|
else
|
|
if( CSheetId(cs->getSheetId()).toString() == "mod_jungle_forage_success.sbrick" )
|
|
{
|
|
ustr = CI18N::get("uittModJungleForageSuccess");
|
|
nodeSM = pIM->getDbProp("SERVER:CHARACTER_INFO:SUCCESS_MODIFIER:ECO:"+toString((uint8)ECOSYSTEM::jungle)+":FORAGE", false);
|
|
}
|
|
else
|
|
if( CSheetId(cs->getSheetId()).toString() == "mod_primary_root_forage_success.sbrick" )
|
|
{
|
|
ustr = CI18N::get("uittModPrimaryRootForageSuccess");
|
|
nodeSM = pIM->getDbProp("SERVER:CHARACTER_INFO:SUCCESS_MODIFIER:ECO:"+toString((uint8)ECOSYSTEM::primary_root)+":FORAGE", false);
|
|
}
|
|
|
|
if( nodeSM )
|
|
{
|
|
if( nodeSM->getValue32() < 0 )
|
|
strFindReplace(ustr, "%modifier", "@{E42F}-"+toString(nodeSM->getValue32())+"@{FFFF}");
|
|
else
|
|
strFindReplace(ustr, "%modifier", "@{0F0F}"+toString(nodeSM->getValue32())+"@{FFFF}");
|
|
|
|
// replace the context help that is required.
|
|
pIM->setContextHelpText(ustr);
|
|
}
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
class CHandlerAuraModifierTooltip : public IActionHandler
|
|
{
|
|
public:
|
|
virtual void execute(CCtrlBase *pCaller, const string &Params)
|
|
{
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
CDBCtrlSheet *cs= dynamic_cast<CDBCtrlSheet*>(pCaller);
|
|
if(!cs)
|
|
return;
|
|
|
|
// set value of consumable's tootltip
|
|
setConsoModSuccessTooltip(cs);
|
|
|
|
// special tooltip? (pvp outpost and xp catalyzer)
|
|
sint specialTTId= getBonusMalusSpecialTT(cs);
|
|
if(specialTTId==BONUS_MALUS::XpCatalyser)
|
|
pIM->setContextHelpText(CI18N::get("uittXpBonus"));
|
|
else if(specialTTId==BONUS_MALUS::OutpostPVPOn)
|
|
pIM->setContextHelpText(CI18N::get("uittPvpOutpostOn"));
|
|
else if(specialTTId==BONUS_MALUS::OutpostPVPOutOfZone)
|
|
pIM->setContextHelpText(CI18N::get("uittPvpOutpostOutOfZone"));
|
|
else if(specialTTId==BONUS_MALUS::OutpostPVPInRound)
|
|
pIM->setContextHelpText(CI18N::get("uittPvpOutpostInRound"));
|
|
else if(specialTTId==BONUS_MALUS::DeathPenalty)
|
|
{
|
|
CCDBNodeLeaf * node = pIM->getDbProp("SERVER:USER:DEATH_XP_MALUS", false);
|
|
if( node )
|
|
{
|
|
ucstring txt = CI18N::get("uittDeathPenalty");
|
|
strFindReplace(txt, "%dp", toString((100*node->getValue16())/254));
|
|
pIM->setContextHelpText(txt);
|
|
}
|
|
}
|
|
// if disabled.
|
|
else if( getAuraDisabledState(cs) )
|
|
{
|
|
// get the normal string, and append a short info.
|
|
ucstring str;
|
|
cs->getContextHelp(str);
|
|
|
|
str+= CI18N::get("uittAuraDisabled");
|
|
|
|
// and replace the context help that is required.
|
|
pIM->setContextHelpText(str);
|
|
}
|
|
// else keep the default one
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerAuraModifierTooltip, "aura_modifier_tooltip");
|
|
|
|
|
|
// ***************************************************************************
|
|
class CHandlerAnimalDeadPopupTooltip : public IActionHandler
|
|
{
|
|
public:
|
|
virtual void execute(CCtrlBase *pCaller, const string &Params)
|
|
{
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
// Find the mount's db leaf
|
|
CCDBNodeBranch *animalsNode = safe_cast<CCDBNodeBranch*>(pIM->getDB()->getNode( ICDBNode::CTextId( "SERVER:PACK_ANIMAL" ), false ));
|
|
BOMB_IF( ! animalsNode, "! animalsNode", return; );
|
|
sint32 minTimeRemaining = -1;
|
|
uint nbAnimals = (uint)animalsNode->getNbNodes();
|
|
for ( uint i=0; i!=nbAnimals; ++i )
|
|
{
|
|
CCDBNodeLeaf *statusNode = pIM->getDbProp(toString("SERVER:PACK_ANIMAL:BEAST%d", i) + ":STATUS", false);
|
|
if (statusNode && ANIMAL_STATUS::isDead((ANIMAL_STATUS::EAnimalStatus)statusNode->getValue32()) )
|
|
{
|
|
ICDBNode *beastNode = animalsNode->getNode( i );
|
|
// CCDBNodeLeaf *uidLeaf = safe_cast<CCDBNodeLeaf*>(beastNode->getNode( ICDBNode::CTextId( "UID" ) ));
|
|
CCDBNodeLeaf *despawnLeaf = safe_cast<CCDBNodeLeaf*>(beastNode->getNode( ICDBNode::CTextId( "DESPAWN" ) ));
|
|
if( minTimeRemaining == -1 )
|
|
{
|
|
minTimeRemaining = despawnLeaf->getValue32();
|
|
}
|
|
else
|
|
if( minTimeRemaining > despawnLeaf->getValue32() )
|
|
{
|
|
minTimeRemaining = despawnLeaf->getValue32();
|
|
}
|
|
}
|
|
}
|
|
|
|
ucstring str;
|
|
BOMB_IF( minTimeRemaining < 0, "at least one animal should be dead", return; );
|
|
|
|
str += CI18N::get("uittAnimalDeadPopupToolTip");
|
|
str += " : ";
|
|
str += toString(minTimeRemaining);
|
|
|
|
// replace the context help that is required.
|
|
pIM->setContextHelpText(str);
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerAnimalDeadPopupTooltip, "animal_dead_popup_tooltip");
|
|
|
|
|
|
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
// MILKO STUFF
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
|
|
#include "../motion/user_controls.h"
|
|
#include "../entities.h"
|
|
#include "people_interraction.h"
|
|
#include "../net_manager.h"
|
|
|
|
// ***************************************************************************
|
|
class CAHMilkoKick: public IActionHandler
|
|
{
|
|
public:
|
|
virtual void execute(CCtrlBase * /* pCaller */, const string &/* Params */)
|
|
{
|
|
CEntityCL *selection = EntitiesMngr.entity(UserEntity->selection());
|
|
if (selection != NULL)
|
|
{
|
|
sint n = PeopleInterraction.TeamList.getIndexFromName(selection->getEntityName());
|
|
if (n >= 0)
|
|
{
|
|
const string msgName = "TEAM:KICK";
|
|
CBitMemStream out;
|
|
if(GenericMsgHeaderMngr.pushNameToStream(msgName, out))
|
|
{
|
|
uint8 teamMember = (uint8)n;
|
|
out.serialEnum(teamMember);
|
|
NetMngr.push(out);
|
|
//nlinfo("impulseCallBack : %s %d sent", msgName.c_str(), teamMember);
|
|
}
|
|
else
|
|
nlwarning("unknown message named '%s'.", msgName.c_str());
|
|
}
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CAHMilkoKick, "milko_kick");
|
|
|
|
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
// RAW MATERIAL STATS
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
|
|
|
|
// ***************************************************************************
|
|
static void onMpChangeItemPart(CInterfaceGroup *wnd, uint32 itemSheetId, const std::string &statPrefixId)
|
|
{
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
|
|
uint i;
|
|
|
|
if(!wnd || !itemSheetId)
|
|
return;
|
|
|
|
// get the item sheet
|
|
const CItemSheet *pIS= dynamic_cast<const CItemSheet *>(SheetMngr.get(CSheetId(itemSheetId)));
|
|
if(!pIS || pIS->Family!=ITEMFAMILY::RAW_MATERIAL)
|
|
return;
|
|
|
|
// get the group for raw material stat
|
|
CInterfaceGroup *groupMp= dynamic_cast<CInterfaceGroup*>(wnd->getElement(wnd->getId()+statPrefixId+INFO_GROUP_MP_STAT));
|
|
if(!groupMp)
|
|
return;
|
|
|
|
// get the combo box
|
|
CDBGroupComboBox *pCB= dynamic_cast<CDBGroupComboBox*>(groupMp->getElement(groupMp->getId()+":item_part_choice" ));
|
|
if( !pCB )
|
|
return;
|
|
|
|
// get the selection
|
|
uint comboSelection= pCB->getSelection();
|
|
|
|
// get the related FaberType
|
|
RM_FABER_TYPE::TRMFType faberType= RM_FABER_TYPE::MPL;
|
|
uint bitCount= 0;
|
|
for(i=0;i<RM_FABER_TYPE::NUM_FABER_TYPE;i++)
|
|
{
|
|
if(pIS->Mp.ItemPartBF & (uint64)(1 << i))
|
|
{
|
|
if(bitCount==comboSelection)
|
|
{
|
|
faberType= RM_FABER_TYPE::TRMFType(i);
|
|
break;
|
|
}
|
|
bitCount++;
|
|
}
|
|
}
|
|
|
|
// get the item part
|
|
if(!pIS->canBuildItemPart(faberType))
|
|
return;
|
|
const CItemSheet::CMpItemPart &itemPart= pIS->getItemPart(faberType);
|
|
|
|
|
|
// **** Build Icon
|
|
// get the icon
|
|
CViewBitmap *viewBmp= dynamic_cast<CViewBitmap*>(groupMp->getElement(groupMp->getId()+":icon" ));
|
|
if(viewBmp)
|
|
{
|
|
// texture name in config.xml
|
|
viewBmp->setTexture(pIM->getDefine( RM_FABER_TYPE::toIconDefineString(faberType) ));
|
|
}
|
|
|
|
|
|
// **** Build text
|
|
// get the text
|
|
CViewText *viewText= dynamic_cast<CViewText*>(groupMp->getElement(groupMp->getId()+":text" ));
|
|
if(viewText)
|
|
{
|
|
ucstring mpCraft;
|
|
|
|
// add the Origin filter.
|
|
string originFilterKey= "iompf" + ITEM_ORIGIN::enumToString((ITEM_ORIGIN::EItemOrigin)itemPart.OriginFilter);
|
|
mpCraft+= CI18N::get(originFilterKey);
|
|
|
|
viewText->setText(mpCraft);
|
|
}
|
|
|
|
|
|
// **** Build Stat bars
|
|
// default: hide all
|
|
for(i=0;i<RM_FABER_STAT_TYPE::NumRMStatType;i++)
|
|
{
|
|
// get the stat group
|
|
CInterfaceGroup *groupStat= dynamic_cast<CInterfaceGroup*>(groupMp->getElement(groupMp->getId()+toString(":stat%d",i) ));
|
|
if(groupStat)
|
|
groupStat->setActive(false);
|
|
}
|
|
// enable only one that are relevant for this item part
|
|
uint groupIndex= 0;
|
|
for(i=0;i<RM_FABER_STAT_TYPE::NumRMStatType;i++)
|
|
{
|
|
RM_FABER_STAT_TYPE::TRMStatType statType= RM_FABER_STAT_TYPE::TRMStatType(i);
|
|
|
|
// if this stat is not relevant for this item part, don't display it!
|
|
if(!RM_FABER_STAT_TYPE::isStatRelevant(faberType, statType))
|
|
continue;
|
|
|
|
// get the next stat group
|
|
CInterfaceGroup *groupStat= dynamic_cast<CInterfaceGroup*>(groupMp->getElement(groupMp->getId()+toString(":stat%d",groupIndex) ));
|
|
if(groupStat)
|
|
{
|
|
groupStat->setActive(true);
|
|
// fill text and bar according to stat
|
|
CViewText *statTitle= dynamic_cast<CViewText*>(groupStat->getElement(groupStat->getId()+":text" ));
|
|
CDBViewBar *statValue= dynamic_cast<CDBViewBar*>(groupStat->getElement(groupStat->getId()+":bar" ));
|
|
if(statTitle)
|
|
statTitle->setText(RM_FABER_STAT_TYPE::toLocalString(statType));
|
|
if(statValue)
|
|
statValue->setValue(itemPart.Stats[i]);
|
|
}
|
|
|
|
groupIndex++;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
// ***************************************************************************
|
|
// MP Item Part selection in a info window
|
|
class CAHItemHelpMpChangeItemPart : public IActionHandler
|
|
{
|
|
public:
|
|
virtual void execute(CCtrlBase * /* pCaller */, const string &Params)
|
|
{
|
|
// get the info window associated
|
|
uint infoWindowIndex;
|
|
fromString(Params, infoWindowIndex);
|
|
if(infoWindowIndex>=CInterfaceHelp::_InfoWindows.size())
|
|
return;
|
|
CInterfaceHelp::CInfoWindow &infoWindow= CInterfaceHelp::_InfoWindows[infoWindowIndex];
|
|
|
|
// just to have the correct 'PrefixForExtra' string.
|
|
CSheetHelpSetup dummy;
|
|
dummy.setupDefaultIDs();
|
|
|
|
// common method for info and botchat
|
|
if(infoWindow.CtrlSheet)
|
|
{
|
|
onMpChangeItemPart(infoWindow.Window, infoWindow.CtrlSheet->getSheetId(), dummy.PrefixForExtra);
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CAHItemHelpMpChangeItemPart, "item_help_mp_change_item_part");
|
|
|
|
|
|
// ***************************************************************************
|
|
// same, but for BotChat dialog box
|
|
class CAHItemBotChatMpChangeItemPart : public IActionHandler
|
|
{
|
|
public:
|
|
virtual void execute(CCtrlBase * /* pCaller */, const string &Params)
|
|
{
|
|
CInterfaceManager *pIM= CInterfaceManager::getInstance();
|
|
|
|
// retreive direct params
|
|
string wndStr= getParam(Params, "wnd");
|
|
string dbitem= getParam(Params, "dbitem");
|
|
string prefix= getParam(Params, "prefix");
|
|
|
|
CInterfaceGroup *wnd= dynamic_cast<CInterfaceGroup*>(pIM->getElementFromId(wndStr));
|
|
CCDBNodeLeaf *node= pIM->getDbProp(dbitem);
|
|
|
|
// common method for info and botchat
|
|
if(wnd && node)
|
|
{
|
|
onMpChangeItemPart(wnd, node->getValue32(), prefix);
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CAHItemBotChatMpChangeItemPart, "item_botchat_mp_change_item_part");
|
|
|
|
|
|
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
// Stat report
|
|
// ***************************************************************************
|
|
// ***************************************************************************
|
|
|
|
|
|
// ***************************************************************************
|
|
|
|
void updateStatReport ()
|
|
{
|
|
// After 30 game minutes, send a stat report
|
|
const uint64 time4StatReport = 60*30*1000;
|
|
if ((ingameTime0 () <= time4StatReport) && (ingameTime1 () > time4StatReport))
|
|
{
|
|
CInterfaceManager *pIM = CInterfaceManager::getInstance();
|
|
pIM->runActionHandler ("proc", NULL, "proc_stat_report");
|
|
}
|
|
}
|
|
|
|
// ***************************************************************************
|
|
|
|
extern string getSystemInformation();
|
|
class CAHSendStatReport : public IActionHandler
|
|
{
|
|
public:
|
|
|
|
virtual void execute (CCtrlBase * /* pCaller */, const string &/* Params */)
|
|
{
|
|
CInterfaceManager *pIM = CInterfaceManager::getInstance();
|
|
pIM->displaySystemInfo(CI18N::get ("uiSendingStatReport"));
|
|
string s = getDebugInformation();
|
|
s += getSystemInformation();
|
|
|
|
string progname;
|
|
char name[1024] = "";
|
|
#ifdef NL_OS_WINDOWS
|
|
GetModuleFileName (NULL, name, 1023);
|
|
#else
|
|
// TODO for Linux
|
|
#endif
|
|
progname = CFile::getFilename(name);
|
|
progname += " ";
|
|
progname += "Statistic Report";
|
|
|
|
bool res = NLNET::sendEmail ("", "", "", progname, s, "");
|
|
if (res)
|
|
nlinfo ("Stat report sent");
|
|
else
|
|
nlwarning ("Can't send stat report");
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER (CAHSendStatReport, "send_stat_report");
|
|
|
|
// ***************************************************************************
|
|
class CHandlerMkInMode : public IActionHandler
|
|
{
|
|
public:
|
|
virtual void execute (CCtrlBase * /* pCaller */, const string &/* Params */)
|
|
{
|
|
CInterfaceManager *pIM = CInterfaceManager::getInstance();
|
|
|
|
CCDBNodeLeaf *pVal = pIM->getDbProp("UI:SAVE:MK_MODE", false);
|
|
if(pVal)
|
|
{
|
|
sint32 mode = pVal->getValue32() + 1;
|
|
if (mode > 5)
|
|
mode = 1;
|
|
pVal->setValue32(mode);
|
|
}
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerMkInMode, "mk_inc_mode");
|
|
|
|
// ***************************************************************************
|
|
class CHandlerBrowseFAQ : public IActionHandler
|
|
{
|
|
public:
|
|
virtual void execute (CCtrlBase * /* pCaller */, const string &/* Params */)
|
|
{
|
|
#ifdef NL_OS_WINDOWS
|
|
if (Driver)
|
|
{
|
|
HWND wnd = Driver->getDisplay();
|
|
ShowWindow(wnd, SW_MINIMIZE);
|
|
}
|
|
#endif
|
|
browseFAQ(ClientCfg.ConfigFile);
|
|
}
|
|
};
|
|
REGISTER_ACTION_HANDLER( CHandlerBrowseFAQ, "browse_faq");
|