khanat-code-old/code/ryzom/client/src/r2/editor.cpp

7841 lines
241 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 <sstream>
#include "nel/misc/path.h"
#include "nel/misc/i18n.h"
#include "nel/misc/sheet_id.h"
#include "nel/misc/polygon.h"
#include "nel/misc/time_nl.h"
//
#include "nel/3d/u_driver.h"
#include "nel/3d/u_text_context.h"
//
#include "nel/ligo/ligo_config.h"
#include "nel/net/unified_network.h"
#include "nel/net/module_manager.h"
#include "nel/pacs/u_global_retriever.h"
//
#include "r2_lua.h"
#include "game_share/r2_share_itf.h"
#include "game_share/r2_messages.h"
#include "game_share/scenario_entry_points.h"
//
#include "dmc/dmc.h" // client interface to the dynamic scenario service
#include "dmc/client_edition_module.h"
#include "dmc/property_accessor.h"
#include "dmc/com_lua_module.h"
#include "game_share/dms.h"
//
#include "../client_sheets/item_sheet.h"
#include "editor.h"
//
#include "../interface_v3/lua_helper.h"
#include "../interface_v3/group_tree.h"
#include "../interface_v3/interface_manager.h"
#include "../contextual_cursor.h"
#include "../cursor_functions.h"
#include "../entities.h"
#include "../events_listener.h"
#include "../interface_v3/group_list.h"
#include "../interface_v3/event_descriptor.h"
#include "../interface_v3/group_tree.h"
#include "../client_cfg.h"
#include "../interface_v3/lua_ihm.h"
#include "../interface_v3/lua_object.h"
#include "../global.h"
#include "../connection.h"
#include "../main_loop.h"
#include "../interface_v3/people_interraction.h"
#include "../time_client.h"
#include "../pacs_client.h"
#include "../interface_v3/lua_ihm.h"
#include "../actions.h"
#include "../actions_client.h"
#include "object_factory_client.h"
#include "../weather.h"
#include "../light_cycle_manager.h"
#include "../dummy_progress.h"
#include "../continent_manager.h"
#include "../world_database_manager.h"
#include "../init_main_loop.h"
#include "../net_manager.h"
#include "../interface_v3/input_handler_manager.h"
#include "../connection.h"
#include "../init_main_loop.h"
#include "../interface_v3/group_editbox.h"
#include "../landscape_poly_drawer.h"
#include "../input.h"
#include "../motion/user_controls.h"
#include "../game_context_menu.h"
#include "../interface_v3/macrocmd_manager.h"
//
#include "../player_r2_cl.h"
#include "palette_node.h"
#include "tool_create_entity.h"
#include "tool_select_move.h"
#include "tool_new_vertex.h"
#include "displayer_visual_entity.h"
#include "displayer_visual_group.h"
#include "displayer_visual_shape.h"
#include "displayer_visual_activity_sequence.h"
#include "displayer_lua.h"
#include "verbose_clock.h"
#include "r2_config.h"
#include "entity_sorter.h"
//
#include "tool_draw_prim.h"
#include "tool_select_move.h"
#include "tool_select_rotate.h"
#include "tool_choose_pos_lua.h"
//
#include "../sheet_manager.h"
#include "../session_browser_impl.h"
#include "../far_tp.h"
using namespace NLMISC;
using namespace NLNET;
using namespace NL3D;
extern CEventsListener EventsListener;
extern CLog g_log;
extern CGameContextMenu GameContextMenu;
extern void badXMLParseMessageBox();
R2::TUserRole UserRoleInSession;
#define OPERATOR_EQUAL(X,Y) X==Y
//#define OPERATOR_EQUAL(X, Y) ::operator==(X,Y)
namespace R2
{
class CEditorCheck
{
public:
CEditorCheck() { check(); }
~CEditorCheck() { check(); }
void check()
{
if (!EditorCreated) return; // avoid infinite loop
if (getEditor().getMode() != CEditor::NotInitialized)
{
nlassert(getEditor().getEnv().isValid());
}
}
static bool EditorCreated;
};
bool CEditorCheck::EditorCreated = false;
#ifdef NL_DEBUG
#define CHECK_EDITOR CEditorCheck __ec;
#else
#define CHECK_EDITOR (void) 0;
#endif
const char *DEFAULT_CURSOR = "curs_default.tga";
bool ResetWanted = false;
bool ReloadUIFlag = true; // by default, CEditor loads its own UI
bool ResetScenarioWanted = false;
bool ReloadScenarioWanted = false;
bool ConnectionWanted = false;
std::string CEditor::_ScenarioToLoadWhenEntreringIntoAnimation="";
bool CEditor::_IsStartingScenario=false;
// *********************************************************************************************************
/** this class forward modification events from the network so that the editor can
* update its state to match the current state of the map
*/
class CDynamicMapClientEventForwarder : public CDynamicMapClient
{
public:
CDynamicMapClientEventForwarder(const std::string &eid, NLNET::IModuleSocket * clientGateway, lua_State *luaState);
virtual void nodeErased(const std::string& instanceId, const std::string& attrName, sint32 position);
virtual void nodeSet(const std::string& instanceId, const std::string& attrName, CObject* value);
virtual void nodeInserted(const std::string& instanceId, const std::string& attrName, sint32 position,
const std::string& key, CObject* value);
virtual void nodeMoved(const std::string& instanceId, const std::string& attrName, sint32 position,
const std::string& destInstanceId, const std::string& destAttrName, sint32 destPosition);
virtual void scenarioUpdated(CObject* highLevel, bool willTP, uint32 initialActIndex);
virtual void onAnimationModeConnected(const CClientMessageAdventureUserConnection& connected);
virtual void onEditionModeConnected( uint32 userSlotId, uint32 adventureId, CObject* highLevel, const std::string& versionName, bool willTP, uint32 initialActIndex);
virtual void onEditionModeDisconnected();
virtual void onResetEditionMode();
virtual void onTestModeConnected();
virtual void onTestModeDisconnected(TSessionId sessionId, uint32 lasAct, TScenarioSessionType sessionType);
};
CDynamicMapClientEventForwarder::CDynamicMapClientEventForwarder(const std::string &eid, NLNET::IModuleSocket * clientGateway, lua_State *luaState) :
CDynamicMapClient(eid, clientGateway, luaState)
{
}
void CDynamicMapClientEventForwarder::nodeErased(const std::string& instanceId, const std::string& attrName, sint32 position)
{
//H_AUTO(R2_CDynamicMapClientEventForwarder_nodeErased)
if (!getEditor().getMode() == CEditor::EditionMode) return;
getEditor().nodeErased(instanceId, attrName, position);
}
void CDynamicMapClientEventForwarder::nodeSet(const std::string& instanceId, const std::string& attrName, CObject* value)
{
//H_AUTO(R2_CDynamicMapClientEventForwarder_nodeSet)
if (!getEditor().getMode() == CEditor::EditionMode) return;
getEditor().nodeSet(instanceId, attrName, value);
}
void CDynamicMapClientEventForwarder::nodeInserted(const std::string& instanceId, const std::string& attrName, sint32 position,
const std::string& key, CObject* value)
{
//H_AUTO(R2_CDynamicMapClientEventForwarder_nodeInserted)
if (!getEditor().getMode() == CEditor::EditionMode) return;
getEditor().nodeInserted(instanceId, attrName, position, key, value);
}
void CDynamicMapClientEventForwarder::nodeMoved(
const std::string& instanceId, const std::string& attrName, sint32 position,
const std::string& destInstanceId, const std::string& destAttrName, sint32 destPosition)
{
//H_AUTO(R2_CDynamicMapClientEventForwarder_nodeMoved)
if (!getEditor().getMode() == CEditor::EditionMode) return;
getEditor().nodeMoved(instanceId, attrName, position, destInstanceId, destAttrName, destPosition);
}
void CDynamicMapClientEventForwarder::scenarioUpdated(CObject* highLevel, bool willTP, uint32 initialActIndex)
{
//H_AUTO(R2_CDynamicMapClientEventForwarder_scenarioUpdated)
if (!getEditor().getMode() == CEditor::EditionMode) return;
getEditor().scenarioUpdated(highLevel, willTP, initialActIndex);
}
void CDynamicMapClientEventForwarder::onEditionModeConnected( uint32 userSlotId, uint32 adventureId, CObject* highLevel, const std::string& versionName, bool willTP, uint32 initialActIndex )
{
//H_AUTO(R2_CDynamicMapClientEventForwarder_onEditionModeConnected)
getEditor().onEditionModeConnected(userSlotId, adventureId, highLevel, versionName, willTP, initialActIndex);
}
void CDynamicMapClientEventForwarder::onEditionModeDisconnected()
{
//H_AUTO(R2_CDynamicMapClientEventForwarder_onEditionModeDisconnected)
getEditor().onEditionModeDisconnected();
}
void CDynamicMapClientEventForwarder::onResetEditionMode()
{
//H_AUTO(R2_CDynamicMapClientEventForwarder_onResetEditionMode)
getEditor().onResetEditionMode();
}
void CDynamicMapClientEventForwarder::onTestModeConnected()
{
//H_AUTO(R2_CDynamicMapClientEventForwarder_onTestModeConnected)
getEditor().onTestModeConnected();
}
void CDynamicMapClientEventForwarder::onTestModeDisconnected(TSessionId sessionId, uint32 lastAct, TScenarioSessionType sessionType)
{
//H_AUTO(R2_CDynamicMapClientEventForwarder_onTestModeDisconnected)
getEditor().onTestModeDisconnected(sessionId, lastAct, sessionType);
}
void CDynamicMapClientEventForwarder::onAnimationModeConnected(const CClientMessageAdventureUserConnection& connected)
{
//H_AUTO(R2_CDynamicMapClientEventForwarder_onAnimationModeConnected)
getEditor().onAnimationModeConnected(connected);
}
// *********************************************************************************************************
CEditor::CEditor()
{
//H_AUTO(R2_CEditor_CEditor)
_SelectedInstance = NULL;
_FocusedInstance = NULL;
_Scenario = NULL;
_EnteredInSetSelectedInstance = false;
_DecalRefTime = 0;
_Initialized = false;
//vianney?? _LastUIModificationDate = 0;
_InstanceObserverHandleCounter = 0;
// module are initialized one time and never reconnected
_DMC = NULL;
_DMS = NULL;
_Mode = NotInitialized;
_AccessMode = AccessModeUnknown;
_SerializeUIConfig = true;
_LastAutoSaveTime = NLMISC::CTime::getLocalTime();// wait ClientCfg.R2EDAutoSaveWait
setForceDesktopReset(false);
CEditorCheck::EditorCreated = true;
_ClearingContent = false;
_Season = UnknownSeason;
_IsWaitingTPForSeasonChange = true;
_UpdatingScenario = false;
_WillTP = false;
_FixedLighting = false;
//
_ScenarioReceivedFlag = false;
_TPReceivedFlag = false;
_WaitScenarioScreenWanted = false;
_WaitScenarioScreenActive = false;
_NewScenario = NULL;
_NewScenarioInitialAct = 0;
_EditionModeDisconnectedFlag = true;
_PostponeScenarioUpdated = false;
_EntitySorter = NULL;
_MaxVisibleEntityExceededFlag = false;
}
// *********************************************************************************************************
void CEditor::setForceDesktopReset(bool force)
{
//H_AUTO(R2_CEditor_setForceDesktopReset)
CHECK_EDITOR
std::fill(_ForceDesktopReset, _ForceDesktopReset + sizeofarray(_ForceDesktopReset), force);
}
// *********************************************************************************************************
void CEditor::autoConfigInit(bool serverIsRingSession)
{
//H_AUTO(R2_CEditor_autoConfigInit)
// R2ED enabled ?
if (ClientCfg.R2EDEnabled)
{
// Editor and Animator (DM) modes
initModules(true);
TMode initialMode;
TAccessMode initialAccessMode;
switch (UserRoleInSession.getValue())
{
case R2::TUserRole::ur_editor:
initialMode = R2::CEditor::EditionMode;
initialAccessMode = R2::CEditor::AccessEditor;
break;
case R2::TUserRole::ur_animator:
initialMode = R2::CEditor::AnimationModeDm;
initialAccessMode = R2::CEditor::AccessDM;
break;
case R2::TUserRole::ur_outland_owner:
initialMode = R2::CEditor::AnimationModeDm;
initialAccessMode = R2::CEditor::AccessOutlandOwner;
break;
case R2::TUserRole::ur_player:
initialMode = R2::CEditor::AnimationModePlay;
// NOTE: I'm not sure if 'editor' is the right mode here - but I can't find anything better for now
// If I try 'AccessModeUnknown' the code asserts later on.
initialAccessMode = R2::CEditor::AccessEditor;
break;
default:
nlwarning( "Unexpected user role %u while editor enabled", (uint)(UserRoleInSession.getValue()) );
initialMode = R2::CEditor::EditionMode;
initialAccessMode = R2::CEditor::AccessEditor;
}
init(initialMode, initialAccessMode);
}
else
{
// Normal player and Ring player modes
ActionsContext.setContext("game");
if (serverIsRingSession)
initModules(false);
if (UserRoleInSession.getValue() != R2::TUserRole::ur_player)
nlwarning( "Unexpected user role %u while editor disabled", (uint)(UserRoleInSession.getValue()) );
}
}
// *********************************************************************************************************
void CEditor::autoConfigRelease(bool serverIsRingSession)
{
//H_AUTO(R2_CEditor_autoConfigRelease)
if (ClientCfg.R2EDEnabled)
{
R2::getEditor().release();
R2::getEditor().releaseModules();
}
else if (serverIsRingSession)
R2::getEditor().releaseModules();
}
// *********************************************************************************************************
void CEditor::initModules(bool connectDMC)
{
//H_AUTO(R2_CEditor_initModules)
CHECK_EDITOR
IModuleManager &mm = IModuleManager::getInstance();
IModule *gateway = mm.createModule("StandardGateway", "clientGw", "");
nlassert(gateway != NULL);
NLNET::IModuleSocket *socketClientGw = mm.getModuleSocket("clientGw");
nlassert(socketClientGw != NULL);
// connect the client gateway to the server
CCommandRegistry::getInstance().execute("clientGw.transportAdd FEClient fec", *InfoLog);
if (!ClientCfg.Local)
{
CCommandRegistry::getInstance().execute("clientGw.transportCmd fec(open)", *InfoLog);
}
if (ClientCfg.Local)
{
_DMS = new CDynamicMapService(ClientCfg.ConfigFile, socketClientGw);
_DMS->init();
}
else
{
_DMS = 0;
}
if (connectDMC)
_DMC = new CDynamicMapClientEventForwarder("Client0",socketClientGw, getLua().getStatePointer());
else
_DMC = new CDynamicMapClient("Client0",socketClientGw, getLua().getStatePointer());
_DMC->setInstantFeedBackFlag(true);
}
// *********************************************************************************************************
void CEditor::releaseModules()
{
//H_AUTO(R2_CEditor_releaseModules)
CHECK_EDITOR
delete _DMC; // must have a virtual destructor
if (ClientCfg.Local)
delete _DMS;
_DMC = NULL;
_DMS = NULL;
IModuleManager &mm = IModuleManager::getInstance();
NLNET::IModule* clientGw = mm.getLocalModule("clientGw");
if (clientGw)
{
mm.deleteModule(clientGw);
}
_Initialized = false;
}
// *********************************************************************************************************
void CEditor::inGameSelection(const CLFECOMMON::TCLEntityId &/* slot */)
{
//H_AUTO(R2_CEditor_inGameSelection)
switch(_Mode)
{
case DMMode:
case AnimationModeDm:
// reset current selection options
getEditor().callEnvMethod("updateAnimBarActions", 0);
getLua().push("");
getEditor().callEnvMethod("dssTarget", 1);
break;
default:
// no-op
break;
}
}
// *********************************************************************************************************
CEditor::~CEditor()
{
release();
}
// ***************************************************************
void CEditor::requestSetLocalNode(const std::string& instanceId, const std::string& attrName, const CObject *value)
{
//H_AUTO(R2_CEditor_requestSetLocalNode)
CHECK_EDITOR
CObject* obj = NULL;
{
CObject *src = _DMC->find(instanceId);
if (!src)
{
nlwarning("Can't find object with id %s", instanceId.c_str());
return;
}
if (!attrName.empty())
{
CObject *subObj = src->getAttr(attrName);
if (!subObj)
{
CObject * valueBase = (CObject *) getObject(src, attrName); // from Base
if (!valueBase)
{
nlwarning("Can't find attribute %s inside object (or is base) with InstanceId = %s", attrName.c_str(), instanceId.c_str());
return;
}
CObject *valueClone = valueBase->clone();
valueClone->setParent(0);
src->setObject(attrName, valueClone);
subObj = src->getAttr(attrName);
}
if (!subObj)
{
nlwarning("Can't find attribute %s inside object with InstanceId = %s", attrName.c_str(), instanceId.c_str());
return;
}
obj = subObj;
}
}
getDMC().getPropertyAccessor().shadowValue(obj, value->clone());
// trigger handlers
onAttrModified(obj);
}
// *****************************************
void CEditor::requestCommitLocalNode(const std::string& instanceId, const std::string& attrName)
{
//H_AUTO(R2_CEditor_requestCommitLocalNode)
CHECK_EDITOR
CObject *obj = _DMC->find(instanceId, attrName);
if (!obj)
{
nlwarning("(while calling requestSetLocalNode");
return;
}
//nlwarning("**** current unshadowed value");
//obj->dump();
/*
// when requestSetNode is called, the old value can't be retrieved because it will
// be erased by 'commitValue', so backup old value
CObject *oldValueBackup = obj->clone();
// send net request
getDMC().getPropertyAccessor().commitValue(obj);
getDMC().requestSetNode(instanceId, attrName, obj);
*/
CObject *localValue = getDMC().getPropertyAccessor().getShadowingValue(obj);
if (!localValue)
{
nlwarning("value is not shadowed : ");
obj->dump();
return;
}
CObject *localValueBackup = localValue->clone();
getDMC().getPropertyAccessor().rollbackValue(obj);
getDMC().requestSetNode(instanceId, attrName, localValueBackup);
}
// ***************************************************************
void CEditor::requestRollbackLocalNode(const std::string& instanceId, const std::string& attrName)
{
//H_AUTO(R2_CEditor_requestRollbackLocalNode)
CHECK_EDITOR
CObject *obj = _DMC->find(instanceId, attrName);
if (!obj)
{
nlwarning("(while calling requestSetLocalNode");
return;
}
getDMC().getPropertyAccessor().rollbackValue(obj);
// force the displayer to update their state from the shadowed value in place of the local value
onAttrModified(obj);
}
// ***************************************************************
CInterfaceManager &CEditor::getUI()
{
//H_AUTO(R2_CEditor_getUI)
CHECK_EDITOR
CInterfaceManager *im = CInterfaceManager::getInstance();
nlassert(im);
return *im;
}
// *********************************************************************************************************
CLuaState &CEditor::getLua()
{
//H_AUTO(R2_CEditor_getLua)
CHECK_EDITOR
CLuaState *ls = getUI().getLuaState();
nlassert(ls);
return *ls;
}
// *********************************************************************************************************
CLuaObject &CEditor::getEnv()
{
//H_AUTO(R2_CEditor_getEnv)
nlassert(_Env.isValid());
return _Env;
}
// *********************************************************************************************************
CLuaObject CEditor::getConfig()
{
//H_AUTO(R2_CEditor_getConfig)
CHECK_EDITOR
return getEnv()["Config"];
}
// *********************************************************************************************************
void CEditor::clearDebugWindow()
{
//H_AUTO(R2_CEditor_clearDebugWindow)
CHECK_EDITOR
getUI().flushDebugWindow();
CGroupList *gl = dynamic_cast<CGroupList *>(getUI().getElementFromId("ui:interface:debug_info:content:cb:text_list"));
if (gl)
{
gl->deleteAllChildren();
}
}
// *********************************************************************************************************
// callback to reload the editor when one of the config files changed
void CEditor::reloadEditorCallback(const std::string &filename)
{
//H_AUTO(R2_CEditor_reloadEditorCallback)
CHECK_EDITOR
ResetWanted = true;
if (nlstricmp(CFile::getExtension(filename), "xml") == 0 ||
nlstricmp(CFile::getExtension(filename), "uxt") == 0)
{
ReloadUIFlag = true;
}
}
// *********************************************************************************************************
void CEditor::setFixedLighting(bool enabled)
{
//H_AUTO(R2_CEditor_setFixedLighting)
if (_FixedLighting == enabled) return;
_FixedLighting = enabled;
LightCycleManager.touch();
}
// *********************************************************************************************************
int CEditor::luaGetSelectedInstanceId(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaGetSelectedInstanceId)
CHECK_EDITOR
if (!getEditor().getSelectedInstance())
{
ls.pushNil();
}
else
{
ls.push(toString(getEditor().getSelectedInstance()->getId()));
}
return 1;
}
void CEditor::setStartingAnimationFilename(const std::string& filename)
{
//H_AUTO(R2_CEditor_setStartingAnimationFilename)
_ScenarioToLoadWhenEntreringIntoAnimation = filename;
}
// *********************************************************************************************************
int CEditor::luaGetStartingAnimationFilename(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaGetStartingAnimationFilename)
ls.push( _ScenarioToLoadWhenEntreringIntoAnimation);
return 1;
}
// *********************************************************************************************************
int CEditor::luaSetSelectedInstanceId(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaSetSelectedInstanceId)
CHECK_EDITOR
CLuaIHM::checkArgCount(ls, "setSelectedInstance (method)", 2);
CLuaIHM::checkArgType(ls, "setSelectedInstance", 2, LUA_TSTRING);
if (strcmp(ls.toString(2), "") == 0)
{
getEditor().setSelectedInstance(NULL);
return 0;
}
CInstance *inst = getEditor().getInstanceFromId(ls.toString(2));
if (!inst)
{
nlwarning("setSelectedInstance : Instance with Id %s not found", ls.toString(2));
return 0;
}
if (!inst->getSelectableFromRoot())
{
nlwarning("Instance with id %s or one of its ancestor is not selectable", ls.toString(2));
return 0;
}
getEditor().setSelectedInstance(inst);
return 0;
}
// *********************************************************************************************************
int CEditor::luaSetCurrentTool(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaSetCurrentTool)
CHECK_EDITOR
CLuaIHM::check(ls, ls.getTop() == 2 || ls.getTop() == 3, "luaSetCurrentTool (method)");
CLuaIHM::checkArgType(ls, "setCurrentTool", 2, LUA_TSTRING);
if (ls.strlen(2) == 0)
{
getEditor().setCurrentTool(NULL);
}
else
{
CTool *tool = createObjectFromClassName<CTool>(ls.toString(2));
if (tool)
{
if (ls.getTop() == 3)
{
CLuaObject initParams(ls);
tool->init(initParams);
}
getEditor().setCurrentTool(tool);
}
}
return 0;
}
// *********************************************************************************************************
int CEditor::luaGetCurrentTool(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaGetCurrentTool)
CHECK_EDITOR
CLuaIHM::checkArgCount(ls, "getCurrentTool", 1); // 1 because of the method call
CLuaIHM::pushReflectableOnStack(ls, getEditor().getCurrentTool());
return 1;
}
// *********************************************************************************************************
int CEditor::luaGetSelectedInstance(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaGetSelectedInstance)
CHECK_EDITOR
if (!getEditor().getSelectedInstance())
{
ls.pushNil();
}
else
{
getEditor().projectInLua(getEditor().getSelectedInstance()->getObjectTable());
}
return 1;
}
// *********************************************************************************************************
int CEditor::luaGetInstanceFromId(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaGetInstanceFromId)
CHECK_EDITOR
const char *funcName = "getInstanceFromId";
CLuaIHM::checkArgCount(ls, funcName, 2);
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TSTRING);
CInstance *inst = getEditor().getInstanceFromId(ls.toString(2));
if (inst == NULL)
{
ls.pushNil();
return 1;
}
else
{
getEditor().projectInLua(inst->getObjectTable());
return 1;
}
return 0;
}
// ********************************************************************************************************
int CEditor::luaGetVisualPropertiesFromInstanceId(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaGetVisualPropertiesFromInstanceId)
const char *funcName = "getVisualPropertiesFromInstanceId";
CLuaIHM::checkArgCount(ls, funcName, 1);
CLuaIHM::checkArgType(ls, funcName, 1, LUA_TSTRING); // instance sheetClient
CInstance *inst = getEditor().getInstanceFromId(ls.toString(1));
if (!inst) { return 0; }
CObject *object = inst->getObjectTable();
if (!object) { return 0; }
SPropVisualA vA;
SPropVisualB vB;
SPropVisualC vC;
bool ok = getEditor().getVisualPropertiesFromObject(object, vA, vB, vC);
if (!ok)
{
return 0; //no param
}
uint64 uVPA = vA.get();
uint64 uVPB = vB.get();
uint64 uVPC = vC.get();
std::string strVPABC = NLMISC::toString( "VPA:%016.16"NL_I64"x\nVPB:%016.16"NL_I64"x\nVPC:%016.16"NL_I64"x", uVPA, uVPB, uVPC );
ls.push(strVPABC);
return 1; //3 string
}
// *********************************************************************************************************
int CEditor::luaDisplayContextMenu(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaDisplayContextMenu)
CHECK_EDITOR
CLuaIHM::checkArgCount(ls, "displayContextMenu", 1); // this is a method
getEditor().displayContextMenu();
return 0;
}
// *********************************************************************************************************
int CEditor::luaConnectAsCreator(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaConnectAsCreator)
CHECK_EDITOR
CLuaIHM::checkArgCount(ls, "connectAsCreator", 1); // this is a method
//getEditor().connectAsCreator();
return 0;
}
// *********************************************************************************************************
int CEditor::luaDofile(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaDofile)
CHECK_EDITOR
CLuaIHM::checkArgCount(ls, "doFile", 1);
CLuaIHM::checkArgType(ls, "doFile", 1, LUA_TSTRING);
getEditor().doLuaScript(ls.toString(-1), ls.toString(-1));
return 0;
}
// *********************************************************************************************************
int CEditor::luaTryFile(CLuaState &/* ls */)
{
//H_AUTO(R2_CEditor_luaTryFile)
/* CHECK_EDITOR
CLuaIHM::checkArgCount(ls, "doFile", 1);
CLuaIHM::checkArgType(ls, "doFile", 1, LUA_TSTRING);
getEditor().doLuaScript(ls.toString(-1), ls.toString(-1));*/
return 0;
}
// *********************************************************************************************************
int CEditor::luaSetEntityCustomSelectBox(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaSetEntityCustomSelectBox)
CHECK_EDITOR
CLuaIHM::checkArgCount(ls, "setEntityCustomSelectBox", 3); // 3 because of method call
CLuaIHM::checkArgType(ls, "setEntityCustomSelectBox", 2, LUA_TSTRING);
CLuaIHM::checkArgType(ls, "setEntityCustomSelectBox", 3, LUA_TTABLE);
std::string sheetName = ls.toString(2);
CEntityCustomSelectBox selectBox;
CLuaObject obj(ls);
selectBox.fromTable(obj);
TEntityCustomSelectBoxMap &boxMap = getEditor().getEntityCustomSelectBoxMap();
boxMap[sheetName] = selectBox;
/*
static volatile bool dumpMap = false;
if (dumpMap)
{
for (TEntityCustomSelectBoxMap::const_iterator watchIt = boxMap.begin(); watchIt != boxMap.end(); ++ watchIt)
{
nlwarning(watchIt->first.c_str());
}
}
*/
return 0;
}
// *********************************************************************************************************
int CEditor::luaGetEntityCustomSelectBox(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaGetEntityCustomSelectBox)
CHECK_EDITOR
CLuaIHM::checkArgCount(ls, "getEntityCustomSelectBox", 2); // 3 because of method call
CLuaIHM::checkArgType(ls, "getEntityCustomSelectBox", 2, LUA_TSTRING);
std::string sheetName = ls.toString(2);
TEntityCustomSelectBoxMap boxMap = getEditor().getEntityCustomSelectBoxMap();
if (boxMap.count(sheetName))
{
ls.newTable();
CLuaObject result(ls);
boxMap[sheetName].toTable(result);
result.push();
}
else
{
ls.pushNil();
}
return 1;
}
// *********************************************************************************************************
int CEditor::luaChoosePos(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaChoosePos)
CHECK_EDITOR
const char *funcName = "choosePos";
CLuaIHM::checkArgMin(ls, funcName, 5);
CLuaIHM::checkArgMax(ls, funcName, 10);
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TSTRING);
CLuaIHM::checkArgType(ls, funcName, 5, LUA_TSTRING);
std::string toolName = ls.toString(5);
std::string cursValid = "curs_create.tga";
std::string cursInvalid = "curs_stop.tga";
if (ls.getTop() >= 6)
{
CLuaIHM::checkArgType(ls, funcName, 6, LUA_TSTRING);
cursValid = ls.toString(6);
}
if (ls.getTop() >= 7)
{
CLuaIHM::checkArgType(ls, funcName, 7, LUA_TSTRING);
cursInvalid = ls.toString(7);
}
// method param 1 -> sheet id
// method param 2 -> ok function
// method param 3 -> cancel function
ls.pushValue(3);
CLuaObject validFunc(ls); // pop valid function
ls.pushValue(4);
CLuaObject cancelFunc(ls); // pop cancel function
//
sint ghostSlot;
if (ls.strlen(2) != 0)
{
CSheetId sheetId(ls.toString(2));
if (sheetId == CSheetId::Unknown)
{
nlwarning("Can't get sheet %s", ls.toString(2));
return 0;
}
ghostSlot = 1; // TMP TMP
if (!createEntity(ghostSlot, sheetId, CVector::Null, 0.f))
{
nlwarning("Cannot create entity for sheet %s", ls.toString(2));
return 0;
}
}
else
{
ghostSlot = -1; // no representation in scene
}
// additionnal polys displayed by the choose pos
std::vector<NLMISC::CPolygon2D> polys;
CPrimLook validLook;
CPrimLook invalidLook;
if (ls.getTop() >= 8)
{
CLuaStackRestorer lsr(&ls, ls.getTop());
CLuaIHM::checkArgType(ls, funcName, 8, LUA_TTABLE);
ls.pushValue(8);
CLuaObject poly;
poly.pop(ls);
ENUM_LUA_TABLE(poly, it)
{
NLMISC::CPolygon2D newPoly;
it.nextValue().push();
CLuaIHM::getPoly2DOnStack(ls, -1, newPoly);
ls.pop();
polys.push_back(newPoly);
}
}
if (ls.getTop() >= 9)
{
ls.pushValue(9);
CLuaObject look;
look.pop(ls);
validLook.init(look);
}
if (ls.getTop() >= 10)
{
ls.pushValue(10);
CLuaObject look;
look.pop(ls);
invalidLook.init(look);
}
//
getEditor().setCurrentTool(new CToolChoosePosLua(ghostSlot, validFunc, cancelFunc, toolName, cursValid, cursInvalid, polys, validLook, invalidLook));
return 0;
}
// *********************************************************************************************************
int CEditor::luaSnapPosToGround(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaSnapPosToGround)
CHECK_EDITOR
const char *funcName = "snapPosToGround";
CLuaIHM::checkArgCount(ls, funcName, 3);
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TNUMBER);
CLuaIHM::checkArgType(ls, funcName, 3, LUA_TNUMBER);
//
/*NLPACS::UGlobalPosition gpos = GR->retrievePosition(CVector((float) ls.toNumber(2), (float) ls.toNumber(3), 2000.f));
if (gpos.InstanceId != -1)
{
CVector snappedPos = GR->getGlobalPosition(gpos);
ls.push((double) snappedPos.x);
ls.push((double) snappedPos.y);
ls.push((double) snappedPos.z);
return 3;
}
else
{
// default z to 0
ls.push(0.f);
return 3;
}*/
// new algo : to avoid to snapping to a cliff, start from the approximate (valid) height found in the height map
CVector inter;
if (CTool::computeWorldMapIntersection((float) ls.toNumber(2), (float) ls.toNumber(3), inter) != CTool::NoIntersection)
{
ls.push((double) inter.x);
ls.push((double) inter.y);
ls.push((double) inter.z);
return 3;
}
else
{
// default z to 0
ls.push(0.f);
return 3;
}
}
// *********************************************************************************************************
int CEditor::luaGetUserEntityPosition(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaGetUserEntityPosition)
CHECK_EDITOR
const char *funcName = "getUserEntityPosition";
CLuaIHM::checkArgCount(ls, funcName, 1);
ls.push((double) UserEntity->pos().x);
ls.push((double) UserEntity->pos().y);
ls.push((double) UserEntity->pos().z);
return 3;
}
// *********************************************************************************************************
int CEditor::luaGetUserEntityFront(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaGetUserEntityFront)
CHECK_EDITOR
const char *funcName = "getUserEntityPosition";
CLuaIHM::checkArgCount(ls, funcName, 1);
ls.push((double) UserEntity->front().x);
ls.push((double) UserEntity->front().y);
return 2;
}
// *********************************************************************************************************
int CEditor::luaRequestSetLocalNode(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaRequestSetLocalNode)
CHECK_EDITOR
const char *funcName = "requestSetLocalNode";
CLuaIHM::checkArgCount(ls, funcName, 3);
CLuaIHM::checkArgType(ls, funcName, 1, LUA_TSTRING);
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TSTRING);
CObject *object = CComLuaModule::getObjectFromLua(ls.getStatePointer());
if (!object)
{
CLuaIHM::fails(ls, "%s : can't read object from parameter 3", funcName);
}
getEditor().requestSetLocalNode(ls.toString(1), ls.toString(2), object);
delete object;
return 0;
}
// *********************************************************************************************************
int CEditor::luaRequestCommitLocalNode(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaRequestCommitLocalNode)
CHECK_EDITOR
const char *funcName = "requestCommitLocalNode";
CLuaIHM::checkArgCount(ls, funcName, 2);
CLuaIHM::checkArgType(ls, funcName, 1, LUA_TSTRING);
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TSTRING);
getEditor().requestCommitLocalNode(ls.toString(1), ls.toString(2));
return 0;
}
// *********************************************************************************************************
int CEditor::luaRequestRollbackLocalNode(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaRequestRollbackLocalNode)
CHECK_EDITOR
const char *funcName = "requestRollbackLocalNode";
CLuaIHM::checkArgCount(ls, funcName, 2);
CLuaIHM::checkArgType(ls, funcName, 1, LUA_TSTRING);
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TSTRING);
getEditor().requestRollbackLocalNode(ls.toString(1), ls.toString(2));
return 0;
}
// *********************************************************************************************************
int CEditor::luaSetCurrentActFromId(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaSetCurrentActFromId)
CHECK_EDITOR
const char *funcName = "setCurrentActFromId";
CLuaIHM::checkArgCount(ls, funcName, 2); // this is a method
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TSTRING);
CInstance *act = getEditor().getInstanceFromId(ls.toString(2));
if (getEditor()._ClearingContent) { return 0; }
if (!act)
{
nlwarning("%s : act with id %s not found", funcName, ls.toString(1));
return 0;
}
if (!act->isKindOf("Act"))
{
nlwarning("%s : instance with id %s is not an act", funcName, ls.toString(1));
return 0;
}
getEditor().setCurrentAct(act);
return 0;
}
// *********************************************************************************************************
int CEditor::luaGetCurrentAct(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaGetCurrentAct)
CHECK_EDITOR
const char *funcName = "getCurrentAct";
CLuaIHM::checkArgCount(ls, funcName, 1); // this is a method
if (getEditor().getCurrentAct())
{
getEditor().projectInLua(getEditor().getCurrentAct()->getObjectTable());
}
else
{
ls.pushNil();
}
return 1;
}
// *********************************************************************************************************
int CEditor::luaGenInstanceName(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaGenInstanceName)
CHECK_EDITOR
const char *funcName = "genInstanceName";
CLuaIHM::checkArgCount(ls, funcName, 2); // this is a method
CLuaIHM::checkArgTypeUCString(ls, funcName, 2); // name
ucstring baseName;
nlverify(CLuaIHM::getUCStringOnStack(ls, 2, baseName));
CLuaIHM::push(ls, getEditor().genInstanceName(baseName));
return 1;
}
// *********************************************************************************************************
int CEditor::luaIsPostFixedByNumber(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaIsPostFixedByNumber)
CHECK_EDITOR
const char *funcName = "isPostFixedByNumber";
CLuaIHM::checkArgCount(ls, funcName, 2); // this is a method
CLuaIHM::checkArgTypeUCString(ls, funcName, 2); // name
ucstring baseName;
nlverify(CLuaIHM::getUCStringOnStack(ls, 2, baseName));
ls.push(getEditor().isPostFixedByNumber(baseName));
return 1;
}
// *********************************************************************************************************
int CEditor::luaIsClearingContent(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaIsPostFixedByNumber)
CHECK_EDITOR
const char *funcName = "isClearingContent";
CLuaIHM::checkArgCount(ls, funcName, 1); // this is a method
ls.push(getEditor().isClearingContent());
return 1;
}
// *********************************************************************************************************
int CEditor::luaSetCookie(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaSetCookie)
CHECK_EDITOR
const char *funcName = "setCookie";
CLuaIHM::checkArgCount(ls, funcName, 4); // this is a method (self + 3 params)
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TSTRING); // instance id
CLuaIHM::checkArgType(ls, funcName, 3, LUA_TSTRING); // key
// last parameters may have any type
CLuaObject value(ls); // pop it
getEditor().setCookie(ls.toString(2), ls.toString(3), value);
return 0;
}
// *********************************************************************************************************
int CEditor::luaIsCreature(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaIsCreature)
CHECK_EDITOR
const char *funcName = "isCreature";
CLuaIHM::checkArgCount(ls, funcName, 1);
CLuaIHM::checkArgType(ls, funcName, 1, LUA_TSTRING); // instance sheetClient
CInstance *inst = getEditor().getInstanceFromId(ls.toString(1));
if (!inst) return 0;
const CCharacterSheet *sheet = dynamic_cast<const CCharacterSheet *>(SheetMngr.get(NLMISC::CSheetId(inst->getSheet())));
if (!sheet) return 0;
bool isNPC = sheet->Race < EGSPD::CPeople::Creature;
// neither NPC, not Kami, nor Karavan
bool creature = !(isNPC || (!isNPC && (sheet->Race==EGSPD::CPeople::Kami || sheet->Race==EGSPD::CPeople::Karavan)));
ls.push(creature);
return 1;
}
// *********************************************************************************************************
int CEditor::luaSetEditorSeason(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaSetEditorSeason)
CHECK_EDITOR
const char *funcName = "setEditorSeason";
if (getEditor().getMode() != CEditor::EditionMode
&& getEditor().getMode() != CEditor::AnimationModeLoading )
{
CLuaIHM::fails(ls, "%s : cannot set weather value outside of edit mode", funcName);
}
CLuaIHM::checkArgCount(ls, funcName, 2);
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TSTRING); // this is a method (self + 1 param)
if (strcmp(ls.toString(2), "Automatic") == 0) getEditor()._Season = Automatic;
else if (strcmp(ls.toString(2), "Spring") == 0) getEditor()._Season = Spring;
else if (strcmp(ls.toString(2), "Summer") == 0) getEditor()._Season = Summer;
else if (strcmp(ls.toString(2), "Autumn") == 0) getEditor()._Season = Autumn;
else if (strcmp(ls.toString(2), "Winter") == 0) getEditor()._Season = Winter;
else CLuaIHM::fails(ls, "%s, invalid season name : %s", ls.toString(2));
// If season is changed while scenario is being received, we do not have received the tp command yet,
// so signal the weather system that the season should not be changed for now
if ((getEditor()._UpdatingScenario && getEditor()._WillTP)
|| getEditor().getMode() != CEditor::AnimationModeLoading )
{
getEditor()._IsWaitingTPForSeasonChange = true;
}
else
{
getEditor()._IsWaitingTPForSeasonChange = false;
}
return 0;
}
// *********************************************************************************************************
int CEditor::luaSetFixedLighting(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaSetFixedLighting)
const char *funcName = "setEditorSeason";
CLuaIHM::checkArgCount(ls, funcName, 2);
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TBOOLEAN); // this is a method (self + 1 param)
getEditor().setFixedLighting(ls.toBoolean(2));
return 0;
}
// *********************************************************************************************************
int CEditor::luaGetFixedLighting(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaGetFixedLighting)
const char *funcName = "getEditorSeason";
CLuaIHM::checkArgCount(ls, funcName, 1); // this is a method
ls.push(getEditor().getFixedLighting());
return 1;
}
// *********************************************************************************************************
int CEditor::luaSetPlotItemInfos(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaSetPlotItemInfos)
const char *funcName = "setPlotItemInfos";
CLuaIHM::checkArgCount(ls, funcName, 5); // a method with 4 args
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TNUMBER);
CLuaIHM::checkArgTypeUCString(ls, funcName, 3);
CLuaIHM::checkArgTypeUCString(ls, funcName, 4);
CLuaIHM::checkArgTypeUCString(ls, funcName, 5);
CItemSheet *item = dynamic_cast<CItemSheet *>(SheetMngr.get(CSheetId((uint32) ls.toNumber(2))));
if (!item || item->Family != ITEMFAMILY::SCROLL_R2)
{
CLuaIHM::fails(ls, "%s : bad sheet, r2 plot item required", funcName);
}
R2::TMissionItem mi;
mi.SheetId = (uint32) ls.toNumber(2);
CLuaIHM::getUCStringOnStack(ls, 3, mi.Name);
CLuaIHM::getUCStringOnStack(ls, 4, mi.Description);
CLuaIHM::getUCStringOnStack(ls, 5, mi.Comment);
getEditor().setPlotItemInfos(mi);
return 0;
}
// *********************************************************************************************************
int CEditor::luaIsCurrentSelectionPlayer(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaIsCurrentSelectionPlayer)
const char *funcName = "isCurrentSelectionPlayer";
CLuaIHM::checkArgCount(ls, funcName, 1); // method with 0 args
CPlayerCL *player = dynamic_cast<CPlayerCL *>(EntitiesMngr.entity(UserEntity->getTargetSlotNoLag()));
ls.push(player != NULL);
return 1;
}
// *********************************************************************************************************
int CEditor::luaFindEmptyPlace(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaFindEmptyPlace)
const char *funcName = "findEmptyPlace";
CLuaIHM::checkArgCount(ls, funcName, 3); // method with 2 args
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TNUMBER);
CLuaIHM::checkArgType(ls, funcName, 3, LUA_TNUMBER);
CVector result;
if (getEditor().getIslandCollision().findEmptyPlace(CVector2f((float) ls.toNumber(2), (float) ls.toNumber(3)), result))
{
ls.push(result.x);
ls.push(result.y);
ls.push(result.z);
return 3;
}
return 0;
}
// *********************************************************************************************************
int CEditor::luaIsInIslandRect(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaIsInIslandRect)
const char *funcName = "isInIslandRect";
CLuaIHM::checkArgCount(ls, funcName, 3); // method with 2 args
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TNUMBER);
CLuaIHM::checkArgType(ls, funcName, 3, LUA_TNUMBER);
const R2::CScenarioEntryPoints::CCompleteIsland *island = getEditor().getIslandCollision().getCurrIslandDesc();
if (!island)
{
ls.push(true);
return 1;
}
float x = (float) ls.toNumber(2);
float y = (float) ls.toNumber(3);
ls.push(x >= island->XMin && y >= island->YMin &&
x <= island->XMax && y <= island->YMax);
return 1;
}
// *********************************************************************************************************
int CEditor::luaGetCurrentIslandName(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaGetCurrentIslandName)
const char *funcName = "getCurrentIslandName";
CLuaIHM::checkArgCount(ls, funcName, 1); // method with no args
R2::CScenarioEntryPoints::CCompleteIsland *ci = getEditor().getIslandCollision().getCurrIslandDesc();
ls.push(ci ? ci->Island.c_str() : "");
return 1;
}
// *********************************************************************************************************
int CEditor::luaKickCharacter(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaKickCharacter)
const char *funcName = "kickCharacter";
CLuaIHM::checkArgCount(ls, funcName, 2); // this is a method (self + 1 params)
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TNUMBER);
CSessionBrowserImpl &sb = CSessionBrowserImpl::getInstance();
sb.kickCharacter(sb.getCharId(), R2::getEditor().getDMC().getEditionModule().getCurrentAdventureId(),
(uint32)ls.toNumber(2));
if(!sb.waitOneMessage(sb.getMessageName("on_invokeResult")))
nlwarning("kickCharacter callback return false");
return 0;
}
// *********************************************************************************************************
int CEditor::luaUnkickCharacter(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaUnkickCharacter)
const char *funcName = "unkickCharacter";
CLuaIHM::checkArgCount(ls, funcName, 2); // this is a method (self + 1 params)
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TNUMBER);
CSessionBrowserImpl &sb = CSessionBrowserImpl::getInstance();
sb.unkickCharacter(sb.getCharId(), R2::getEditor().getDMC().getEditionModule().getCurrentAdventureId(),
(uint32)ls.toNumber(2));
if(!sb.waitOneMessage(sb.getMessageName("on_invokeResult")))
nlwarning("unkickCharacter callback return false");
return 0;
}
// *********************************************************************************************************
int CEditor::luaTeleportToCharacter(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaTeleportToCharacter)
const char *funcName = "teleportToCharacter";
CLuaIHM::checkArgCount(ls, funcName, 2); // this is a method (self + 1 params)
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TNUMBER);
CClientEditionModule & cem = R2::getEditor().getDMC().getEditionModule();
cem.requestTeleportOneCharacterToAnother(cem.getCurrentAdventureId(), CSessionBrowserImpl::getInstance().getCharId(),
(uint32)ls.toNumber(2));
return 0;
}
// *********************************************************************************************************
int CEditor::luaWaitScenarioScreen(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaWaitScenarioScreen)
const char *funcName = "waitScenarioScreen";
CLuaIHM::checkArgCount(ls, funcName, 1); // method with no args
if (!ClientCfg.Local)
{
getEditor()._WaitScenarioScreenWanted = true;
}
return 0;
}
// *********************************************************************************************************
int CEditor::luaIsScenarioUpdating(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaIsScenarioUpdating)
const char *funcName = "isScenarioUpdating";
CLuaIHM::checkArgCount(ls, funcName, 1); // method with no args
ls.push( (double)getEditor()._UpdatingScenario );
return 1;
}
/*
// *********************************************************************************************************
int CEditor::luaIsValidPoly(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaIsValidPoly)
const char *funcName = "isValidPoly";
CLuaIHM::checkArgCount(ls, funcName, 2); // method with no args
CLuaIHM::checkArgType(ls, funcName, 3, LUA_TTABLE); // method with no args
ls.pushValue(3);
CLuaObject poly;
NLMISC::CPolygon2D poly2D;
poly.pop(luaState);
ENUM_LUA_TABLE(table, it)
{
NLMISC::CVector2f pos;
if (!CLuaIHM::pop(ls, pos))
{
CLuaIHM::fails(ls, "%s expects CVector2f for poly coordinates);
}
poly2D.Vertices.push_back(pos);
}
ls.push(getEditor().getIslandCollision().isValidPoly(poly2D);
return 1;
}
// *********************************************************************************************************
int CEditor::luaIsValidPosition(CLuaState &ls)
{
CHECK_EDITOR
const char *funcName = "snapPosToGround";
CLuaIHM::checkArgCount(ls, funcName, 3);
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TNUMBER);
CLuaIHM::checkArgType(ls, funcName, 3, LUA_TNUMBER);
//
/*NLPACS::UGlobalPosition gpos = GR->retrievePosition(CVector((float) ls.toNumber(2), (float) ls.toNumber(3), 2000.f));
if (gpos.InstanceId != -1)
{
CVector snappedPos = GR->getGlobalPosition(gpos);
ls.push((double) snappedPos.x);
ls.push((double) snappedPos.y);
ls.push((double) snappedPos.z);
return 3;
}
else
{
// default z to 0
ls.push(0.f);
return 3;
}
// new algo : to avoid to snapping to a cliff, start from the approximate (valid) height found in the height map
CVector inter;
if (CTool::computeWorldMapIntersection((float) ls.toNumber(2), (float) ls.toNumber(3), inter) != CTool::NoIntersection)
{
ls.push((double) inter.x);
ls.push((double) inter.y);
ls.push((double) inter.z);
}
*/
// *********************************************************************************************************
int CEditor::luaCanUndo(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaCanUndo)
const char *funcName = "canUndo";
CLuaIHM::checkArgCount(ls, funcName, 1); // method with no args
ls.push(getEditor().getDMC().getActionHistoric().canUndo());
return 1;
}
// *********************************************************************************************************
int CEditor::luaCanRedo(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaCanRedo)
const char *funcName = "canRedo";
CLuaIHM::checkArgCount(ls, funcName, 1); // method with no args
ls.push(getEditor().getDMC().getActionHistoric().canRedo());
return 1;
}
// *********************************************************************************************************
int CEditor::luaGetUserEntityName(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaGetUserEntityName)
const char *funcName = "getUserEntityName";
CLuaIHM::checkArgCount(ls, funcName, 1); // this is a method
if (UserEntity)
{
ucstring name = UserEntity->getEntityName()+PlayerSelectedHomeShardNameWithParenthesis;
ls.push( std::string( name.toUtf8() ) );
}
else
{
ls.push(std::string(""));
}
return 1;
}
static CLuaString lstr_Next("next");
// *********************************************************************************************************
int CEditor::luaEnumInstances(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaEnumInstances)
CLuaStackChecker lsc(&ls, 1);
// return an enumerator that allows to iterate over instance, by kind
const char *funcName = "enumInstances";
CLuaIHM::checkArgCount(ls, funcName, 2); // method with no args
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TSTRING);
struct CInstanceEnumerator
{
CSortedInstances *InstMap;
TInstanceByName::iterator Current;
static int next(CLuaState &ls)
{
const char *funcName = "InstanceEnumerator:next";
CLuaIHM::checkArgCount(ls, funcName, 1);
CLuaIHM::checkArgType(ls, funcName, 1, LUA_TUSERDATA);
CInstanceEnumerator *ie = (CInstanceEnumerator *) ls.toUserData(1);
CEditor &ed = getEditor();
while(ie->Current != ie->InstMap->end())
{
CInstance *inst = ie->Current->second;
if (!inst->getGhost())
{
// object must be in current act or base act
CInstance *parentAct = inst->getParentAct();
if (!parentAct || parentAct == ed.getCurrentAct() || parentAct == ed.getBaseAct())
{
inst->getLuaProjection().push();
++ ie->Current;
return 1;
}
}
++ ie->Current;
}
return 0;
}
//
static int gc(CLuaState &ls)
{
const char *funcName = "InstanceEnumerator:next";
CLuaIHM::checkArgCount(ls, funcName, 1);
CLuaIHM::checkArgType(ls, funcName, 1, LUA_TUSERDATA);
CInstanceEnumerator *ie = (CInstanceEnumerator *) ls.toUserData(1);
ie->~CInstanceEnumerator(); // no real effect for now...
return 0;
}
static int index(CLuaState &ls)
{
const char *funcName = "InstanceEnumerator.__index";
CLuaIHM::checkArgCount(ls, funcName, 2);
CLuaIHM::checkArgType(ls, funcName, 1, LUA_TUSERDATA);
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TSTRING);
if (ls.toString(2) == lstr_Next)
{
ls.push(CInstanceEnumerator::next);
return 1;
}
return 0;
}
};
sint classIndex = getEditor().classToIndex(ls.toString(2));
if (classIndex < 0)
{
CLuaIHM::fails(ls, "Trying to iterate over unknown class : %s", ls.toString(2));
}
ls.newTable();
CLuaObject mt;
mt.pop(ls);
ls.push(CInstanceEnumerator::index);
mt.setValue("__index", CLuaObject(ls));
ls.push(CInstanceEnumerator::gc);
mt.setValue("__gc", CLuaObject(ls));
//
void *newIter = ls.newUserData(sizeof(CInstanceEnumerator));
CInstanceEnumerator *ie = new (newIter) CInstanceEnumerator;
ie->InstMap = &getEditor()._InstancesByDispName[classIndex];
ie->Current = ie->InstMap->begin();
mt.push();
ls.setMetaTable(-2);
//
return 1;
}
// *********************************************************************************************************
void CEditor::waitScenarioScreen()
{
//H_AUTO(R2_CEditor_waitScenarioScreen)
if (ClientCfg.Local) return;
if (_Mode==EditionMode)
{
setMode(GoingToEditionMode);
}
getUI().hideAllWindows();
CInterfaceGroup *waitScreen = dynamic_cast<CInterfaceGroup *>(getUI().getElementFromId("ui:interface:r2ed_connecting"));
if (waitScreen)
{
waitScreen->setActive(true);
getUI().setTopWindow(waitScreen);
}
//
enum TState { WaitingScenario, WaitingTP, DoExit };
TState state = WaitingScenario;
//
getEditor()._ScenarioReceivedFlag = false;
getEditor()._TPReceivedFlag = false;
bool firewallTimeout = false;
//
ActionsContext.setContext("waiting_network");
TGameCycle serverTick = NetMngr.getCurrentServerTick();
getUI().setCaptureKeyboard(NULL);
getUI().setDefaultCaptureKeyboard(NULL);
loadBackgroundBitmap (StartBackground);
// patch for the 'sys info that pop' prb (cause unknown for now ...)
CInterfaceElement *sysInfo = getUI().getElementFromId("ui:interface:system_info");
bool sysInfoActive = false;
if (sysInfo) sysInfoActive = sysInfo->getActive();
for (;;)
{
switch(state)
{
case WaitingScenario:
if (getEditor()._ScenarioReceivedFlag)
{
if (getEditor()._WillTP)
{
state = WaitingTP;
}
else
{
state = DoExit;
}
}
break;
case WaitingTP:
// no-op
break;
default:
nlassert(0);
break;
}
if (state == DoExit)
{
break;
}
// Update network.
try
{
if ( ! firewallTimeout )
NetMngr.update();
}
catch ( EBlockedByFirewall& )
{
if ( NetMngr.getConnectionState() == CNetManager::Disconnect )
{
firewallTimeout = true;
}
else
{
// Display the firewall alert string
CViewText *pVT = dynamic_cast<CViewText*>(getUI().getElementFromId("ui:interface:r2ed_connecting:title"));
if (pVT != NULL)
pVT->setText(CI18N::get("uiFirewallAlert")+ucstring("..."));
// The mouse and fullscreen mode should be unlocked for the user to set the firewall permission
nlSleep( 30 ); // 'nice' the client, and prevent to make too many send attempts
}
}
CCDBNodeBranch::flushObserversCalls();
// check if we can send another dated block
if (NetMngr.getCurrentServerTick() != serverTick)
{
//
serverTick = NetMngr.getCurrentServerTick();
NetMngr.send(serverTick);
}
else
{
// Send dummy info
NetMngr.send();
}
// update module manager
NLNET::IModuleManager::getInstance().updateModules();
// Update the DT T0 and T1 global variables
updateClientTime();
CInputHandlerManager::getInstance()->pumpEvents();
Driver->clearBuffers(CRGBA::Black);
Driver->setMatrixMode2D11();
drawLoadingBitmap (1);
if ( state==WaitingTP && getEditor()._TPReceivedFlag)
{
break; // don't want an additionnal swap buffer here, so exit now (the loading bitmap is good,
// has just been reloaded by the tp screen...
}
// Interface handling & displaying (processes clicks...)
getUI().updateFrameEvents();
if (waitScreen)
{
getUI().setTopWindow(waitScreen);
}
if (sysInfo) sysInfo->setActive(false);
getUI().updateFrameViews(NULL);
CCDBNodeBranch::flushObserversCalls();
// Movie shooter
globalMenuMovieShooter();
// Force the client to sleep a bit.
if(ClientCfg.Sleep >= 0)
{
nlSleep(ClientCfg.Sleep);
}
//
if (!waitScreen)
{
TextContext->setShaded(true);
TextContext->setFontSize(40);
TextContext->setColor(CRGBA::White);
// TOP LEFT //
//----------//
TextContext->setHotSpot(NL3D::UTextContext::MiddleMiddle);
TextContext->printfAt(0.5f, 0.5f, "(UI NOT FOUND) Waiting scenario from server.");
}
// Display
Driver->swapBuffers();
//
if (NetMngr.getConnectionState() == CNetManager::Disconnect)
{
if ( firewallTimeout )
{
// Display the firewall error string instead of the normal failure string
CViewText *pVT = dynamic_cast<CViewText*>(getUI().getElementFromId("ui:interface:r2ed_connecting:title"));
if (pVT != NULL)
{
pVT->setMultiLine( true );
pVT->setText(CI18N::get("uiFirewallFail")+ucstring(".\n")+
CI18N::get("uiFirewallAlert")+ucstring("."));
}
}
}
}
if (sysInfo) sysInfo->setActive(sysInfoActive);
destroyLoadingBitmap ();
if (_Mode == GoingToEditionMode)
{
setMode(EditionMode);
}
return;
}
// *********************************************************************************************************
// lua instance observer
class CInstanceObserverLua : public CEditor::IInstanceObserver
{
public:
CInstanceObserverLua(CLuaObject &receiver);
~CInstanceObserverLua();
CLuaObject &getReceiver() { return _Receiver; }
// from IInstanceObserver
virtual void onInstanceCreated(CInstance &instance);
virtual void onInstanceErased(CInstance &instance);
virtual void onInstancePreHrcMove(CInstance &instance);
virtual void onInstancePostHrcMove(CInstance &instance);
virtual void onPreHrcMove(CInstance &instance);
virtual void onPostHrcMove(CInstance &instance);
virtual void onAttrModified(CInstance &instance, const std::string &attrName, sint32 attrIndex);
private:
CLuaObject _Receiver; // table that will receive the notifications
static uint _Count;
};
uint CInstanceObserverLua::_Count = 0;
CInstanceObserverLua::CInstanceObserverLua(CLuaObject &receiver) : _Receiver(receiver)
{
++ _Count;
}
CInstanceObserverLua::~CInstanceObserverLua()
{
#ifdef NL_DEBUG
nlassert(!getEditor().isInstanceObserver(this));
#endif
nlassert(_Count >= 0);
-- _Count;
}
void CInstanceObserverLua::onInstanceCreated(CInstance &instance)
{
//H_AUTO(R2_CInstanceObserverLua_onInstanceCreated)
nlassert(_Receiver.isValid());
if (_Receiver["onInstanceCreated"].isNil()) return; // no-op if not handled
getEditor().projectInLua(instance.getObjectTable());
_Receiver.callMethodByNameNoThrow("onInstanceCreated", 1, 0);
}
void CInstanceObserverLua::onInstanceErased(CInstance &instance)
{
//H_AUTO(R2_CInstanceObserverLua_onInstanceErased)
nlassert(_Receiver.isValid());
if (_Receiver["onInstanceErased"].isNil()) return; // no-op if not handled
getEditor().projectInLua(instance.getObjectTable());
_Receiver.callMethodByNameNoThrow("onInstanceErased", 1, 0);
}
void CInstanceObserverLua::onInstancePreHrcMove(CInstance &instance)
{
//H_AUTO(R2_CInstanceObserverLua_onInstancePreHrcMove)
nlassert(_Receiver.isValid());
if (_Receiver["onInstanceCreated"].isNil()) return; // no-op if not handled
getEditor().projectInLua(instance.getObjectTable());
_Receiver.callMethodByNameNoThrow("onInstancePreHrcMove", 1, 0);
}
void CInstanceObserverLua::onInstancePostHrcMove(CInstance &instance)
{
//H_AUTO(R2_CInstanceObserverLua_onInstancePostHrcMove)
nlassert(_Receiver.isValid());
if (_Receiver["onInstanceCreated"].isNil()) return; // no-op if not handled
getEditor().projectInLua(instance.getObjectTable());
_Receiver.callMethodByNameNoThrow("onInstancePostHrcMove", 1, 0);
}
void CInstanceObserverLua::onPreHrcMove(CInstance &instance)
{
//H_AUTO(R2_CInstanceObserverLua_onPreHrcMove)
nlassert(_Receiver.isValid());
if (_Receiver["onPreHrcMove"].isNil()) return; // no-op if not handled
getEditor().projectInLua(instance.getObjectTable());
_Receiver.callMethodByNameNoThrow("onPreHrcMove", 1, 0);
}
void CInstanceObserverLua::onPostHrcMove(CInstance &instance)
{
//H_AUTO(R2_CInstanceObserverLua_onPostHrcMove)
nlassert(_Receiver.isValid());
if (_Receiver["onPostHrcMove"].isNil()) return; // no-op if not handled
getEditor().projectInLua(instance.getObjectTable());
_Receiver.callMethodByNameNoThrow("onPostHrcMove", 1, 0);
}
void CInstanceObserverLua::onAttrModified(CInstance &instance, const std::string &attrName, sint32 attrIndex)
{
//H_AUTO(R2_CInstanceObserverLua_onAttrModified)
nlassert(_Receiver.isValid());
if (_Receiver["onAttrModified"].isNil()) return; // no-op if not handled
getEditor().projectInLua(instance.getObjectTable());
getEditor().getLua().push(attrName);
getEditor().getLua().push((double) attrIndex);
_Receiver.callMethodByNameNoThrow("onAttrModified", 3, 0);
}
// *********************************************************************************************************
int CEditor::luaAddInstanceObserver(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaAddInstanceObserver)
CHECK_EDITOR
const char *funcName = "addInstanceObserver";
CLuaIHM::checkArgCount(ls, funcName, 3); // this is a method (self + 2 params)
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TSTRING); // instance id
CLuaIHM::checkArgType(ls, funcName, 3, LUA_TTABLE); // receiver
CLuaObject receiver(ls); // pop the receiver
ls.push((double) getEditor().addInstanceObserver(ls.toString(2), new CInstanceObserverLua(receiver)));
return 1;
}
// *********************************************************************************************************
int CEditor::luaRemoveInstanceObserver(CLuaState &ls)
{
//H_AUTO(R2_CEditor_luaRemoveInstanceObserver)
CHECK_EDITOR
const char *funcName = "removeInstanceObserver";
CLuaIHM::checkArgCount(ls, funcName, 2); // this is a method (self + 1 params)
CLuaIHM::checkArgType(ls, funcName, 2, LUA_TNUMBER); // instance id
IInstanceObserver *observer = getEditor().getInstanceObserver((TInstanceObserverHandle) ls.toNumber(2));
if (observer == NULL)
{
CLuaIHM::fails(ls, "Instance observer not found for handle = %d", (int) ls.toNumber(2));
}
CInstanceObserverLua *luaObserver = dynamic_cast<CInstanceObserverLua *>(observer);
if (luaObserver == NULL)
{
CLuaIHM::fails(ls, "Instance observer found for handle %d, but has bad type, it wasn't registered from lua.", (int) ls.toNumber(2));
}
getEditor().removeInstanceObserver((TInstanceObserverHandle) ls.toNumber(2));
CLuaObject receiver = luaObserver->getReceiver();
delete luaObserver;
receiver.push();
return 1;
}
// *********************************************************************************************************
bool CEditor::isInstanceObserver(IInstanceObserver *observer) const
{
//H_AUTO(R2_CEditor_isInstanceObserver)
for(TInstanceObserverMap::const_iterator it = _InstanceObservers.begin(); it != _InstanceObservers.end(); ++it)
{
if (it->second == observer) return true;
}
return false;
}
// *********************************************************************************************************
const CObjectTable *CEditor::getObjectTableFromId(const TInstanceId &id) const
{
//H_AUTO(R2_CEditor_getObjectTableFromId)
CHECK_EDITOR
nlassert(_DMC);
const CObject *obj = _DMC->find(id);
if (!obj) return NULL;
nlassert(obj->isTable());
return (CObjectTable *) obj;
}
// *********************************************************************************************************
CInstance *CEditor::getInstanceFromObject(const CObject *obj) const
{
//H_AUTO(R2_CEditor_getInstanceFromObject)
CHECK_EDITOR
if (!obj->isTable()) return NULL;
TInstanceMap::const_iterator it = _Instances.find((CObjectTable *) obj);
if (it != _Instances.end())
{
return it->second;
}
return NULL;
}
// *********************************************************************************************************
void CEditor::registerEnvMethod(const char *name,TLuaWrappedFunction func)
{
//H_AUTO(R2_CEditor_registerEnvMethod)
CHECK_EDITOR
nlassert(name);
getEnv().setValue(name, func);
}
// *********************************************************************************************************
void CEditor::registerEnvFunction(const char *name, TLuaWrappedFunction func)
{
//H_AUTO(R2_CEditor_registerEnvFunction)
CHECK_EDITOR
nlassert(name);
getEnv().setValue(name, func);
}
// *********************************************************************************************************
void CEditor::registerLuaFunc()
{
//H_AUTO(R2_CEditor_registerLuaFunc)
CHECK_EDITOR
registerEnvMethod("getSelectedInstanceId", luaGetSelectedInstanceId);
registerEnvMethod("setSelectedInstanceId", luaSetSelectedInstanceId);
registerEnvMethod("setCurrentTool", luaSetCurrentTool);
registerEnvMethod("getCurrentTool", luaGetCurrentTool);
registerEnvMethod("getVisualPropertiesFromInstanceId", luaGetVisualPropertiesFromInstanceId);
registerEnvMethod("getInstanceFromId", luaGetInstanceFromId);
registerEnvMethod("displayContextMenu", luaDisplayContextMenu);
registerEnvMethod("getSelectedInstance", luaGetSelectedInstance);
registerEnvMethod("connectAsCreator", luaConnectAsCreator);
registerEnvMethod("doFile", luaDofile);
registerEnvMethod("tryFile", luaTryFile);
registerEnvMethod("setEntityCustomSelectBox", luaSetEntityCustomSelectBox);
registerEnvMethod("getEntityCustomSelectBox", luaGetEntityCustomSelectBox);
registerEnvMethod("choosePos", luaChoosePos);
registerEnvMethod("snapPosToGround", luaSnapPosToGround);
registerEnvMethod("getUserEntityPosition", luaGetUserEntityPosition);
registerEnvMethod("getUserEntityFront", luaGetUserEntityFront);
registerEnvMethod("setCurrentActFromId", luaSetCurrentActFromId);
registerEnvMethod("getCurrentAct", luaGetCurrentAct);
registerEnvMethod("genInstanceName", luaGenInstanceName);
registerEnvMethod("isPostFixedByNumber", luaIsPostFixedByNumber);
registerEnvMethod("setCookie", luaSetCookie);
registerEnvMethod("addInstanceObserver", luaAddInstanceObserver);
registerEnvMethod("removeInstanceObserver", luaRemoveInstanceObserver);
registerEnvFunction("requestSetLocalNode", luaRequestSetLocalNode);
registerEnvFunction("requestCommitLocalNode", luaRequestCommitLocalNode);
registerEnvFunction("requestRollbackLocalNode", luaRequestRollbackLocalNode);
registerEnvMethod("isCreature", luaIsCreature);
registerEnvMethod("setEditorSeason", luaSetEditorSeason);
registerEnvMethod("setFixedLighting", luaSetFixedLighting);
registerEnvMethod("getFixedLighting", luaGetFixedLighting);
registerEnvMethod("setPlotItemInfos", luaSetPlotItemInfos);
registerEnvMethod("isCurrentSelectionPlayer", luaIsCurrentSelectionPlayer);
registerEnvMethod("findEmptyPlace", luaFindEmptyPlace);
registerEnvMethod("isInIslandRect", luaIsInIslandRect);
registerEnvMethod("getCurrentIslandName", luaGetCurrentIslandName);
registerEnvMethod("waitScenarioScreen", luaWaitScenarioScreen);
registerEnvMethod("isScenarioUpdating", luaIsScenarioUpdating);
registerEnvMethod("canUndo", luaCanUndo);
registerEnvMethod("canRedo", luaCanRedo);
registerEnvMethod("getUserEntityName", luaGetUserEntityName);
registerEnvMethod("getStartingAnimationFilename", luaGetStartingAnimationFilename);
registerEnvMethod("kickCharacter", luaKickCharacter);
registerEnvMethod("unkickCharacter", luaUnkickCharacter);
registerEnvMethod("teleportToCharacter", luaTeleportToCharacter);
registerEnvMethod("enumInstances", luaEnumInstances);
registerEnvMethod("isClearingContent", luaIsClearingContent);
}
/*
static void polyUnitTest()
{
CPolygon2D poly;
poly.Vertices.push_back(CVector2f(0.f, 0.f));
poly.Vertices.push_back(CVector2f(1.f, 0.f));
poly.Vertices.push_back(CVector2f(1.f, 1.f));
poly.Vertices.push_back(CVector2f(0.f, 1.f));
nlassert(poly.contains(CVector2f(0.f, 0.f), false));
nlassert(poly.contains(CVector2f(1.f, 0.f), false));
nlassert(poly.contains(CVector2f(1.f, 1.f), false));
nlassert(poly.contains(CVector2f(0.f, 1.f), false));
nlassert(poly.contains(CVector2f(0.5f, 0.f), false))
nlassert(poly.contains(CVector2f(1.f, 0.5f), false));
nlassert(poly.contains(CVector2f(0.5f, 1.f), false));
nlassert(poly.contains(CVector2f(0.f, 0.5f), false));
nlassert(poly.contains(CVector2f(0.5f, 0.5f), false));
//
nlassert(!poly.contains(CVector2f(-0.5f, 0.5f), false));
nlassert(!poly.contains(CVector2f(1.5f, 0.5f), false));
nlassert(!poly.contains(CVector2f(0.5f, -0.5f), false));
nlassert(!poly.contains(CVector2f(0.5f, 1.5f), false));
}
*/
// *********************************************************************************************************
void CEditor::loadLanguageFile()
{
//H_AUTO(R2_CEditor_loadLanguageFile)
CHECK_EDITOR
static bool reload = false;
CI18N::ILoadProxy *oldLoadProxy = CI18N::getLoadProxy();
CI18N::setLoadProxy(NULL);
string lc = ClientCfg.LanguageCode;
nlinfo("Reloading the r2_%s.uxt", lc.c_str());
CI18N::loadFromFilename("r2_"+lc+".uxt", reload);
CI18N::setLoadProxy(oldLoadProxy);
reload = true;
}
// *********************************************************************************************************
void CEditor::loadKeySet(const std::string &keySet)
{
//H_AUTO(R2_CEditor_loadKeySet)
CHECK_EDITOR
if (keySet.empty()) return;
CVerboseClock clock("Parsing of keyset " + keySet);
// remove all existing keys
ActionsContext.removeAllCombos();
CMacroCmdManager::getInstance()->removeAllMacros();
// parse keys that are specific to r2ed
std::vector<string> xmlFilesToParse;
// Does the r2ed keys file exist ?
std::string userKeyFileName = "save/" + keySet + "_"+PlayerSelectedFileName+".xml";
if (CFile::fileExists(userKeyFileName) && CFile::getFileSize(userKeyFileName) > 0)
{
// Load the user key file
xmlFilesToParse.push_back (userKeyFileName);
}
// Load the default key (but don't replace existings bounds, see keys.xml "key_def_no_replace")
xmlFilesToParse.push_back (keySet + ".xml");
if (!CInterfaceManager::getInstance()->parseInterface (xmlFilesToParse, true))
{
badXMLParseMessageBox();
}
CActionsManager *actionManager = ActionsContext.getActionsManager();
if (actionManager)
{
// if some mode are unavailable, forbid the associated shortcuts
/*if (!_ModeEnabled[EditionMode])
{
actionManager->removeBaseAction("r2ed_stop_test");
}
if (!_ModeEnabled[TestMode] || !_ModeEnabled[DMMode])
{
actionManager->removeBaseAction("r2ed_try_go_test");
}*/
}
}
// *********************************************************************************************************
void CEditor::saveCurrentKeySet()
{
//H_AUTO(R2_CEditor_saveCurrentKeySet)
CHECK_EDITOR
std::string prefix = getKeySetPrefix(getMode());
if (prefix.empty()) return;
getUI().saveKeys ("save/" + prefix + "_" + PlayerSelectedFileName + ".xml");
}
// *********************************************************************************************************
std::string CEditor::getKeySetPrefix(TMode mode)
{
//H_AUTO(R2_CEditor_getKeySetPrefix)
CHECK_EDITOR
switch(mode)
{
case EditionMode: return "keys_r2ed";
case GoingToDMMode:
case GoingToEditionMode:
case AnimationModeLoading:
case AnimationModeWaitingForLoading:
case AnimationModeGoingToDm:
case AnimationModeGoingToPlay:
return "";
case AnimationModePlay:
return "keys";
default:
return "keys";
}
return std::string();
}
// *********************************************************************************************************
void CEditor::setUIMode(uint8 mode)
{
//H_AUTO(R2_CEditor_setUIMode)
CHECK_EDITOR
getUI().setMode(mode);
if (_ForceDesktopReset[mode])
{
// force to call reset when reloading the ui
getLua().push((double) mode);
callEnvMethod("resetDesktop", 1, 0);
_ForceDesktopReset[mode] = false;
}
getLua().push((double) mode);
callEnvMethod("onChangeDesktop", 1, 0);
}
// *********************************************************************************************************
void CEditor::loadStandardUI()
{
//H_AUTO(R2_CEditor_loadStandardUI)
getUI().setMode(0);
// force to reload the standard interface
ClientCfg.R2EDEnabled = false;
loadUIConfig("");
ClientCfg.R2EDEnabled = true;
getUI().updateAllLocalisedElements();
}
// *********************************************************************************************************
bool CEditor::isDMing() const
{
//H_AUTO(R2_CEditor_isDMing)
if (getMode() == AnimationModePlay) return false; // masterless mode -> not a dm
return getMode() == DMMode ||
getMode() == AnimationModeDm ||
(
getMode() != NotInitialized &&
getAccessMode() == AccessDM
);
}
// *********************************************************************************************************
void CEditor::setMode(TMode mode)
{
switch(mode)
{
case NotInitialized: nlwarning("*R2* Setting mode to NotInitialized"); break;
case EditionMode: nlwarning("*R2* Setting mode to EditionMode"); break;
case GoingToDMMode: nlwarning("*R2* Setting mode to GoingToDMMode"); break;
case GoingToEditionMode: nlwarning("*R2* Setting mode to GoingToEditionMode"); break;
case TestMode: nlwarning("*R2* Setting mode to TestMode"); break;
case DMMode: nlwarning("*R2* Setting mode to DMMode"); break;
case AnimationModeLoading: nlwarning("*R2* Setting mode to AnimationModeLoading"); break;
case AnimationModeWaitingForLoading: nlwarning("*R2* Setting mode to AnimationModeWaitingForLoading"); break;
case AnimationModeDm: nlwarning("*R2* Setting mode to AnimationModeDm"); break;
case AnimationModePlay: nlwarning("*R2* Setting mode to AnimationModePlay"); break;
case AnimationModeGoingToDm: nlwarning("*R2* Setting mode to AnimationModeGoingToDm"); break;
case AnimationModeGoingToPlay: nlwarning("*R2* Setting mode to AnimationModeGoingToPlay"); break;
default:
nlwarning("Setting mode to 'unknown'");
break;
}
if (mode != EditionMode)
{
delete _EntitySorter;
_EntitySorter = NULL;
}
//H_AUTO(R2_CEditor_setMode)
CHECK_EDITOR
//nlassert(_ModeEnabled[mode]);
if (mode == _Mode && !_ForceDesktopReset[mode]) return;
if (mode != _Mode)
{
if ((_Mode == AnimationModeDm || _Mode == AnimationModePlay)
&& !((mode == AnimationModeDm || mode == AnimationModePlay)))
{
// remove the chat windows when leaving animation mode
PeopleInterraction.removeAllFreeTellers();
}
CVerboseClock clock("Saving windows & key at a mode change ...");
if (_Mode != NotInitialized)
{
saveCurrentKeySet();
}
//if (_Mode != NotInitialized) saveUIConfig();
if (_Mode == EditionMode || _Mode == AnimationModeLoading)
{
getDMC().getActionHistoric().clear(); // no backup of undo/redo for now
clearContent(); // when leaving edition, remove content from tree
}
else if (mode == EditionMode || mode == GoingToEditionMode)
{
removeAllEntitySlots(); // when leaving test or dm, remove the entities slots
}
if (mode == DMMode || mode == AnimationModeDm)
{
getEditor().getDMC().getEditionModule().askUpdateCharMode(TCharMode::Dm);
}
else if (mode == EditionMode)
{
getEditor().getDMC().getEditionModule().askUpdateCharMode(TCharMode::Editer);
}
else if (mode == AnimationModePlay)
{
if(_Mode != NotInitialized)
{
if (_SerializeUIConfig)
{
saveUIConfig();
}
}
getEditor().getDMC().getEditionModule().askUpdateCharMode(TCharMode::Player);
// if access mode is DM, then load the whole ui
if (_AccessMode == AccessDM)
{
loadStandardUI();
}
}
else if (mode == TestMode)
{
getEditor().getDMC().getEditionModule().askUpdateCharMode(TCharMode::Tester);
}
else // Other like going to dm ...
{
}
}
// reset any selection
if ((!IngameDbMngr.initInProgress()) && UserEntity) // prevent from calling selection() if coming from a FarTP, otherwise the server will mess up the action counter because it can't know our datasetrow so early (initInProgress(): heuristic to ensure the setTarget() will work on the EGS)
{
UserEntity->selection(CLFECOMMON::INVALID_SLOT);
}
//
ContextCur.release();
_Mode = mode;
loadKeySet(getKeySetPrefix(_Mode));
getUI().disableModalWindow();
getUI().setCapturePointerLeft(NULL);
getUI().setCapturePointerRight(NULL);
getUI().setCaptureKeyboard(NULL);
// Season is now unknown, until server force it (in test mode), or first set act set it (in edit mode)
_Season = UnknownSeason;
//
switch(_Mode)
{
case EditionMode:
if (!_EntitySorter)
{
_EntitySorter = new CEntitySorter;
}
ActionsContext.setContext("r2ed");
// set new mode in lua
_Env.setValue("Mode", "Edit");
setUIMode(0);
::IgnoreEntityDbUpdates = true;
::SlotUnderCursor = CLFECOMMON::INVALID_SLOT;
/**
* nb we manage mouse bitmap ourselves, so there's only one cursor context here, and it never changes:
* its sole purpose is to route messages to the current tool
*/
ContextCur.add(false, "STAND BY", DEFAULT_CURSOR, 0.0f, CEditor::checkCursor, CEditor::mouseClick);
forceSetSelectedInstance(NULL);
initReferencePlotItems();
// force speed to "run" ("walk" not available in edition)
if (!UserEntity->running())
{
UserEntity->switchVelocity();
}
break;
case TestMode:
ActionsContext.setContext("r2ed_anim_test");
// set new mode in lua
_Env.setValue("Mode", "Test");
setUIMode(1);
::IgnoreEntityDbUpdates = false;
::initContextualCursor();
nlassert(CDisplayerBase::ObjCount == 0);
resetPlotItems();
break;
case DMMode:
ActionsContext.setContext("r2ed_anim_dm");
// set new mode in lua
_Env.setValue("Mode", "DM");
setUIMode(2);
::IgnoreEntityDbUpdates = false;
::initContextualCursor();
nlassert(CDisplayerBase::ObjCount == 0);
// when in local mode, init fake plot item to test the 'dm gift' interface
initDummyPlotItems();
break;
case GoingToDMMode:
ActionsContext.setContext("waiting_network");
// set new mode in lua
_Env.setValue("Mode", "GoingToDM");
setUIMode(3);
connexionMsg(_ConnexionMsg); // update connexion window
::IgnoreEntityDbUpdates = false;
::initContextualCursor();
nlassert(CDisplayerBase::ObjCount == 0);
resetPlotItems();
break;
case GoingToEditionMode:
nlassert(CDisplayerBase::ObjCount == 0);
ActionsContext.setContext("waiting_network");
// set new mode in lua
_Env.setValue("Mode", "BackToEditing");
setUIMode(3);
connexionMsg(_ConnexionMsg); // update connexion window
::IgnoreEntityDbUpdates = true;
::initContextualCursor();
resetPlotItems();
break;
case AnimationModeLoading:
nlassert(CDisplayerBase::ObjCount == 0);
ActionsContext.setContext("waiting_network");
// set new mode in lua
_Env.setValue("Mode", "AnimationModeLoading");
setUIMode(3);
connexionMsg(_ConnexionMsg); // update connexion window
// ::IgnoreEntityDbUpdates = true;
::initContextualCursor();
resetPlotItems();
break;
case AnimationModeWaitingForLoading:
nlassert(CDisplayerBase::ObjCount == 0);
ActionsContext.setContext("waiting_network");
// set new mode in lua
_Env.setValue("Mode", "AnimationModeWaitingForLoading");
setUIMode(3);
connexionMsg(_ConnexionMsg); // update connexion window
// ::IgnoreEntityDbUpdates = true;
::initContextualCursor();
resetPlotItems();
break;
case AnimationModeDm:
PeopleInterraction.updateAllFreeTellerHeaders(); // if invitation were received during loading, add 'invite' button to the new windows
nlassert(CDisplayerBase::ObjCount == 0);
ActionsContext.setContext("r2ed_anim_dm");
// set new mode in lua
_Env.setValue("Mode", "AnimationModeDm");
setUIMode(2);
::IgnoreEntityDbUpdates = false;
::initContextualCursor();
resetPlotItems();
break;
case AnimationModePlay:
{
// animation mode play is still visible in masterless mode
nlassert(CDisplayerBase::ObjCount == 0);
// no op here, just a player
ActionsContext.setContext("game");
//ActionsContext.setContext("r2ed_anim_test");
// set new mode in lua
_Env.setValue("Mode", "r2ed_anim_test");
// hide all ring windows that may remain
hideRingWindows();
// setUIMode(1); // don't change desktop here, because virtual desktops are allowed in this mode
getUI().setMode(0); // default to desktop 0 as with standard game
::IgnoreEntityDbUpdates = false;
::initContextualCursor();
resetPlotItems();
if (getDMC().getEditionModule().isSessionOwner())
{
// if we are not a player, pop the ui to pop control player window & to quit
callEnvMethod("popDMToolbarWindow", 0, 0);
}
else
{
callEnvMethod("playerModeUIFix", 0, 0);
}
}
break;
case AnimationModeGoingToDm:
nlassert(CDisplayerBase::ObjCount == 0);
ActionsContext.setContext("waiting_network");
//set new mode in lua
_Env.setValue("Mode", "AnimationModeGoingToDM");
setUIMode(3);
connexionMsg(_ConnexionMsg); // update connexion window
::IgnoreEntityDbUpdates = false;
::initContextualCursor();
resetPlotItems();
break;
default:
nlassert(0);
break;
}
ClientCfg.ManualWeatherSetup = false;
ContextCur.context("STAND BY");
_Mode = mode;
// warn lua that mode has changed
callEnvMethod("onModeChanged", 0, 0);
//
setCurrentTool(NULL);
// display the bars over the player (not available during edition, but available at test time)
UserEntity->buildInSceneInterface();
if (_Mode == EditionMode)
{
// special context menu for edition
GameContextMenu.init("game_context_menu_edition");
}
else
{
GameContextMenu.init("");
}
}
// *********************************************************************************************************
void CEditor::hideRingWindows()
{
//H_AUTO(R2_CEditor_hideRingWindows)
static const char *ringWindows[] =
{
"r2ed_palette",
"r2ed_connect",
"r2ed_property_sheet_no_selection",
"r2ed_property_sheet_no_properties",
"r2ed_bbox_edit",
"r2ed_toolbar",
"r2ed_windows_dm_bar",
"r2ed_toolbar_window",
"r2ed_windows_bar",
"r2ed_main_bl",
"r2ed_animation_loading",
"r2ed_animation_waiting",
"dm_controlled_entities",
"r2ed_current_session",
"r2ed_toolbar_admin",
"lua_inspector",
"r2ed_npc",
"r2ed_select_bar",
"r2ed_main_menu_button",
"r2ed_contextual_toolbar_new",
"feature_help",
"r2ed_logic_entities",
"r2ed_events",
"r2ed_activities",
"r2ed_edit_activity_sequence",
"r2ed_dialogs",
"r2ed_edit_chat_sequence",
"r2ed_mini_activity_view",
"r2ed_acts",
"r2ed_connecting",
"r2ed_scenario",
"r2ed_dm_gift",
"r2ed_scenario_filter",
"r2ed_testbar"
};
for (uint k = 0; k < sizeofarray(ringWindows); ++k)
{
std::string id = "ui:interface:" + std::string(ringWindows[k]);
CInterfaceElement *grp = getUI().getElementFromId(id);
if (grp)
{
grp->setActive(false);
}
else
{
nlwarning("Can't find group with id : %s", id.c_str());
}
}
}
// *********************************************************************************************************
void CEditor::init(TMode initialMode, TAccessMode accessMode)
{
nlwarning("*R2* init");
_EntitySorter = NULL;
//H_AUTO(R2_CEditor_init)
CNiceInputAuto niceInputs;
nlassert((uint) initialMode < ModeCount);
nlassert((uint) accessMode < AccessModeCount);
//nlassert(_ModeEnabled[initialMode]; // mode not enabled ...
CHECK_EDITOR
CVerboseClock clock("Init of editor ");
_Initialized = true;
_AccessMode = accessMode;
// add a list of files for which modifications will trigger R2ED reset
for(uint k = 0; k < ClientCfg.R2EDReloadFiles.size(); ++k)
{
CFile::removeFileChangeCallback(ClientCfg.R2EDReloadFiles[k]); // avoid to add twice when a reset ...
CFile::addFileChangeCallback(ClientCfg.R2EDReloadFiles[k], reloadEditorCallback);
}
//
CLuaStackChecker lsc(&getLua());
getLua().pushValue(LUA_GLOBALSINDEX);
_Globals.pop(getLua());
getLua().pushValue(LUA_REGISTRYINDEX);
_Registry.pop(getLua());
// create R2 environment, and keep a reference on it
_Env = _Globals.newTable(R2_LUA_PATH);
_Config = _Env.newTable("Config");
//
_Env.setValue("InClient", true); // TMP : signal to the script that it is initialised by the client
//
switch (accessMode)
{
case AccessEditor: _Env.setValue("AccessMode", "Editor"); break;
case AccessDM: _Env.setValue("AccessMode", "DM"); break;
case AccessOutlandOwner: _Env.setValue("AccessMode", "OutlandOwner"); break;
default:
nlassert(0);
break;
}
// Load the UI files
if (ReloadUIFlag)
{
loadLanguageFile();
reloadUI();
// setForceDesktopReset is done below
ReloadUIFlag = false;
}
//
registerDisplayers();
registerTools();
//
std::vector<std::string> emptyArgsList;
// Calling window init proc
registerLuaFunc();
//
_DMC->init(getLua().getStatePointer());
//
initObjectProjectionMetatable(); // init system to access to scenary objects from lua
// init client/server stuffs
CConfigVarBase::getConfigFileTimeStamp() ++; // invalidate all var taken from the config file
// load r2 features & components
{
CVerboseClock clock("Execution of r2_core.lua");
doLuaScript("r2_core.lua", "r2ed common functions and definitions");
}
initClassInheritanceTable();
//
{
CLuaStackChecker lsc4(&getLua());
setCurrentTool(NULL); // force the default tool (select / move)
}
initDecals();
// TMP nico : unit test for newpoly stuff
//polyUnitTest();
_Initialized = true;
_LuaUIMainLoop = _Env["UIMainLoop"];
if (_SerializeUIConfig)
{
// load virtual desktops configs
bool configLoaded = false;
{
CVerboseClock clock("Load of ui config from disk ");
// if access mode is DM, then load the whole ui (else it has been previuously loaded)
if (initialMode == AnimationModePlay)
{
loadStandardUI();
}
else
{
configLoaded = loadUIConfig(getUIPrefix(initialMode));
}
}
// if not found then reset all virtual desktops (one for each mode of the editor)
if (!configLoaded)
{
nlwarning("No interface config found, resetting editor windows to their default");
setForceDesktopReset(true);
}
}
setMode(initialMode);
{
CVerboseClock clock("Update of localized elements");
getUI().updateAllLocalisedElements();
}
}
// *********************************************************************************************************
std::string CEditor::getUIPrefix(TMode mode) const
{
//H_AUTO(R2_CEditor_getUIPrefix)
switch(mode)
{
case AnimationModePlay:
return "";
break;
default:
return "r2ed_interface_";
break;
}
}
// *********************************************************************************************************
uint CEditor::getMaxNumPlotItems()
{
//H_AUTO(R2_CEditor_getMaxNumPlotItems)
uint ret;
fromString(getUI().getDefine("r2ed_max_num_plot_item_sheets"), ret);
return ret;
}
// *********************************************************************************************************
CCDBNodeLeaf *CEditor::getRefPlotItemSheetDBLeaf(uint index)
{
//H_AUTO(R2_CEditor_getRefPlotItemSheetDBLeaf)
return getUI().getDbProp(toString("LOCAL:R2:REFERENCE_PLOT_ITEMS:%d:SHEET", (int) index), false);
}
// *********************************************************************************************************
CCDBNodeLeaf *CEditor::getPlotItemSheetDBLeaf(uint index)
{
//H_AUTO(R2_CEditor_getPlotItemSheetDBLeaf)
return getUI().getDbProp(toString("LOCAL:R2:PLOT_ITEMS:%d:SHEET", (int) index), false);
}
// *********************************************************************************************************
void CEditor::setReferencePlotItemSheet(uint index, uint32 sheetId)
{
//H_AUTO(R2_CEditor_setReferencePlotItemSheet)
CCDBNodeLeaf *leaf = getRefPlotItemSheetDBLeaf(index);
if (leaf)
{
leaf->setValue32(sheetId);
}
leaf = getUI().getDbProp(toString("LOCAL:R2:AVAILABLE_PLOT_ITEMS:%d:SHEET", (int) index), false);
if (leaf)
{
leaf->setValue32(sheetId);
}
}
// *********************************************************************************************************
void CEditor::initDummyPlotItems()
{
//H_AUTO(R2_CEditor_initDummyPlotItems)
uint maxNumPlotItems = getMaxNumPlotItems();
uint numItems = std::min(10u, maxNumPlotItems);
uint k;
for(k = 0; k < numItems; ++k)
{
CCDBNodeLeaf *destLeaf = getPlotItemSheetDBLeaf(k);
if (!destLeaf) continue;
CCDBNodeLeaf *srcLeaf = getRefPlotItemSheetDBLeaf(k);
destLeaf->setValue32(srcLeaf ? srcLeaf->getValue32() : 0u);
}
for(; k < maxNumPlotItems; ++k)
{
CCDBNodeLeaf *destLeaf = getPlotItemSheetDBLeaf(k);
if (!destLeaf) continue;
destLeaf->setValue32(0);
}
}
// *********************************************************************************************************
void CEditor::resetPlotItems()
{
//H_AUTO(R2_CEditor_resetPlotItems)
uint maxNumPlotItems = getMaxNumPlotItems();
// uint numItems = std::min(10u, maxNumPlotItems);
for(uint k = 0; k < maxNumPlotItems; ++k)
{
CCDBNodeLeaf *destLeaf = getPlotItemSheetDBLeaf(k);
if (!destLeaf) continue;
destLeaf->setValue32(0);
}
_PlotItemInfos.clear();
}
// *********************************************************************************************************
void CEditor::initReferencePlotItems()
{
//H_AUTO(R2_CEditor_initReferencePlotItems)
{
CVerboseClock clock("InitReferencePlotItems");
uint maxNumPlotItems = getMaxNumPlotItems();
clamp(maxNumPlotItems, 0u, 1024u);
uint currIndex = 0;
// look for plot item sheets
const CSheetManager::TEntitySheetMap &sheets = SheetMngr.getSheets();
for(CSheetManager::TEntitySheetMap::const_iterator it = sheets.begin(); it != sheets.end(); ++it)
{
if (it->second.EntitySheet && it->second.EntitySheet->Type == CEntitySheet::ITEM)
{
std::string name = it->second.EntitySheet->Id.toString();
if (strstr(name.c_str(), "r2_plot_item"))
{
uint32 sheetId = it->second.EntitySheet->Id.asInt();
setReferencePlotItemSheet(currIndex, sheetId);
++ currIndex;
}
}
}
for (; currIndex < maxNumPlotItems; ++currIndex)
{
setReferencePlotItemSheet(currIndex, 0);
}
resetPlotItems();
}
}
// *********************************************************************************************************
bool CEditor::loadUIConfig(const std::string &prefix)
{
//H_AUTO(R2_CEditor_loadUIConfig)
CHECK_EDITOR
return getUI().loadConfig("save/" + prefix + PlayerSelectedFileName + ".icfg");
}
// *********************************************************************************************************
void CEditor::saveUIConfig()
{
//H_AUTO(R2_CEditor_saveUIConfig)
CHECK_EDITOR
// if there are dirt desktop, update all of them
if (_Mode != AnimationModePlay)
{
bool restoreMode = false;
uint8 oldMode = getUI().getMode();
for (uint k = 0; k < 4; ++k)
{
if (_ForceDesktopReset[k])
{
setUIMode(k);
restoreMode = true;
_ForceDesktopReset[k] = false;
}
if (restoreMode)
{
setUIMode(oldMode);
}
}
}
getUI().saveConfig("save/" + getUIPrefix(_Mode) + PlayerSelectedFileName + ".icfg");
}
// *********************************************************************************************************
void CEditor::initDecals()
{
//H_AUTO(R2_CEditor_initDecals)
CHECK_EDITOR
// highlight
CLuaObject config = getEnv()["Config"];
CLuaObject objHighlight = config["HightlightDecalLook"];
CLuaObject objSelect = config["SelectDecalLook"];
CLuaObject objSelecting = config["SelectingDecalLook"];
CLuaObject objPioneer = config["PionneerDecalLook"];
_HighlightDecalAnim.buildFromLuaTable(objHighlight);
_SelectDecalAnim.buildFromLuaTable(objSelect);
_SelectingDecalAnim.buildFromLuaTable(objSelecting);
_PionneerDecalAnim.buildFromLuaTable(objPioneer);
_HighlightDecal.setClipDownFacing(true);
_SelectDecal.setClipDownFacing(true);
_PionneerDecal.setClipDownFacing(true);
//
CPrimLook boxLook;
boxLook.init(getEnv()["PrimRender"]["SelectBoxLook"]);
_SelectBox.setLook(boxLook);
boxLook.init(getEnv()["PrimRender"]["HighlightBoxLook"]);
_HighlightBox.setLook(boxLook);
}
// *********************************************************************************************************
void CEditor::showPrimRender(CPrimRender &dest, const NLMISC::CAABBox &localBox, const NLMISC::CMatrix &worldMat, const CDecalAnim &refDecalAnim)
{
//H_AUTO(R2_CEditor_showPrimRender)
CHECK_EDITOR
static NLMISC::CPolygon2D poly2D;
poly2D.Vertices.resize(4);
CVector pmin = localBox.getMin();
CVector pmax = localBox.getMax();
poly2D.Vertices[0] = worldMat * CVector(pmin.x, pmin.y, pmin.z);
poly2D.Vertices[1] = worldMat * CVector(pmax.x, pmin.y, pmin.z);
poly2D.Vertices[2] = worldMat * CVector(pmax.x, pmax.y, pmin.z);
poly2D.Vertices[3] = worldMat * CVector(pmin.x, pmax.y, pmin.z);
NLMISC::CAABBox bbox;
CLandscapePolyDrawer::computeBBoxFromPolygon(poly2D, bbox);
dest.setVertices(poly2D.Vertices);
//dest.setWorldMapPolyColor(polyColor);
float animRatio = refDecalAnim.DurationInMs == 0 ? 0.f : (T1 % refDecalAnim.DurationInMs) / (float) refDecalAnim.DurationInMs;
animRatio = 0.5f * cosf(2.f * (float) Pi * animRatio) + 0.5f;
CRGBA color = blend(refDecalAnim.StartDiffuse, refDecalAnim.EndDiffuse, animRatio);
CPrimLook pl = dest.getLook();
pl.EdgeLook.DecalColor = pl.VertexLook.DecalColor = pl.FirstVertexLook.DecalColor = color;
dest.setLook(pl);
dest.addDecalsToRenderList();
static volatile bool showPoly = true;
if (showPoly)
{
CLandscapePolyDrawer::getInstance().addPoly(poly2D, dest.getLook().EdgeLook.WorldMapColor, bbox);
}
}
// *********************************************************************************************************
void CEditor::showSelectBox(const NLMISC::CAABBox &localBox, const NLMISC::CMatrix &worldMat)
{
//H_AUTO(R2_CEditor_showSelectBox)
CHECK_EDITOR
showPrimRender(_SelectBox, localBox, worldMat, _SelectDecalAnim);
}
// *********************************************************************************************************
void CEditor::showHighlightBox(const NLMISC::CAABBox &localBox, const NLMISC::CMatrix &worldMat)
{
//H_AUTO(R2_CEditor_showHighlightBox)
CHECK_EDITOR
showPrimRender(_HighlightBox, localBox, worldMat, _HighlightDecalAnim);
}
// *********************************************************************************************************
void CEditor::showSelectDecal(const NLMISC::CVector &pos, float scale)
{
//H_AUTO(R2_CEditor_showSelectDecal)
CHECK_EDITOR
updateDecalBlendRegion(_SelectDecal, pos);
showDecal(pos, scale, _SelectDecal, _SelectDecalAnim);
}
// *********************************************************************************************************
void CEditor::showHighlightDecal(const NLMISC::CVector &pos, float scale)
{
//H_AUTO(R2_CEditor_showHighlightDecal)
CHECK_EDITOR
updateDecalBlendRegion(_HighlightDecal, pos);
showDecal(pos, scale, _HighlightDecal, _HighlightDecalAnim);
}
// *********************************************************************************************************
void CEditor::showDecal(const NLMISC::CVector2f &pos, float scale, CDecal &decal, const CDecalAnim &decalAnim)
{
//H_AUTO(R2_CEditor_showDecal)
CHECK_EDITOR
float animRatio = decalAnim.DurationInMs == 0 ? 0.f : ((T1 - _DecalRefTime) % decalAnim.DurationInMs) / (float) decalAnim.DurationInMs;
animRatio = 0.5f * cosf(2.f * (float) Pi * animRatio) + 0.5f;
decalAnim.updateDecal(pos, animRatio, decal, scale);
decal.addToRenderList();
}
// *********************************************************************************************************
void CEditor::addSelectingDecal(const NLMISC::CVector &pos, float scale)
{
//H_AUTO(R2_CEditor_addSelectingDecal)
CHECK_EDITOR
CSelectingDecal *decal = NULL;
// see if there's an unused decal in the list
for(uint k = 0; k < _SelectingDecals.size(); ++k)
{
if (_SelectingDecals[k]->EndDate < T1)
{
decal = _SelectingDecals[k];
break;
}
}
if (decal == NULL)
{
_SelectingDecals.push_back(new CSelectingDecal);
decal = _SelectingDecals.back();
}
decal->EndDate = T1 + _SelectingDecalAnim.DurationInMs;
decal->Pos = pos;
decal->Scale = scale;
//
updateDecalBlendRegion(decal->Decal, pos);
_DecalRefTime = T1;
}
// *********************************************************************************************************
void CEditor::updateSelectingDecals()
{
//H_AUTO(R2_CEditor_updateSelectingDecals)
CHECK_EDITOR
if (_SelectingDecalAnim.DurationInMs == 0) return;
for(uint k = 0; k < _SelectingDecals.size(); ++k)
{
CSelectingDecal *decal = _SelectingDecals[k];
if (decal)
{
if (decal->EndDate > T1)
{
float animRatio = (float) (T1 - (decal->EndDate - _SelectingDecalAnim.DurationInMs)) / (float) _SelectingDecalAnim.DurationInMs;
_SelectingDecalAnim.updateDecal(decal->Pos, animRatio, decal->Decal, decal->Scale);
decal->Decal.addToRenderList();
}
}
}
}
// *******************************************************************************************************
void CEditor::reset()
{
//H_AUTO(R2_CEditor_reset)
if (ClientCfg.R2EDExtendedDebug)
Driver->setWindowTitle(ucstring("Resetting R2ED editor ..."));
CHECK_EDITOR
_SerializeUIConfig = false; // prevent reloading of ui for speed
_WantedActOnInit.clear();
if (getCurrentAct())
{
R2::CObjectTable* act = getCurrentAct()->getObjectTable();
if (act->isString("Name"))
{
_WantedActOnInit = act->toString("Name");
}
else
{
_WantedActOnInit = act->toString("Title"); //obsolete
}
}
CVerboseClock clock("Reset of editor ");
TMode oldMode = _Mode;
TAccessMode oldAccessMode = _AccessMode;
if (_Mode == EditionMode)
{
R2::getEditor().getLua().executeScriptNoThrow("r2.Version.save(\"data/r2_buffer.dat\")");
}
release();
clearDebugWindow();
if (ReloadUIFlag)
{
loadLanguageFile();
reloadUI();
setForceDesktopReset(true); // next changes of desktop should force a reset, since ui has been reloaded
ReloadUIFlag = false;
}
init(oldMode, oldAccessMode);
if (oldMode == EditionMode)
{
CVerboseClock clock("Request reconnection ");
getLua().executeScriptNoThrow("r2.requestReconnection()");
}
// try to return to the act with the same title
_SerializeUIConfig = true;
if (ClientCfg.R2EDExtendedDebug)
{
Driver->setWindowTitle(CI18N::get("TheSagaOfRyzom"));
// Show the window
Driver->showWindow();
}
getUI().displaySystemInfo(CI18N::get("uiR2EDEditorReseted"), "BC");
}
// *********************************************************************************************************
void CEditor::reloadUI()
{
//H_AUTO(R2_CEditor_reloadUI)
CHECK_EDITOR
CVerboseClock clock("Reload ui");
// reload same list than at startup (given in client.cfg)
for (uint k = 0; k < ClientCfg.XMLR2EDInterfaceFiles.size(); ++k)
{
reloadUI(ClientCfg.XMLR2EDInterfaceFiles[k].c_str());
}
}
// *********************************************************************************************************
void CEditor::reloadUI(const char *filename)
{
//H_AUTO(R2_CEditor_reloadUI)
CHECK_EDITOR
CVerboseClock clock("Loading ui " + std::string(filename));
CLuaStackChecker ls(&getLua());
std::string path = CPath::lookup(filename, false);
if (path.empty())
{
nlwarning("File %s not found, r2 ui not (re)loaded.", filename);
return;
}
uint32 fileDate = CFile::getFileModificationDate(path);
if (_LastUIModificationDate.count(filename) == 0)
{
_LastUIModificationDate[filename] = 0;
}
uint32 &lastUIModificationDate = _LastUIModificationDate[filename];
if (fileDate > lastUIModificationDate || !ClientCfg.R2EDDontReparseUnchangedUIFiles) // TMP (init not complete else ...)
{
// reload the ui files
NLMISC::ICommand::execute(toString("loadui %s", filename).c_str(), g_log);
lastUIModificationDate = fileDate;
}
else
{
if (!ClientCfg.R2EDDontReparseUnchangedUIFiles)
{
nlwarning("ui file %s date unchanged, not reloading.", filename);
}
}
}
// *********************************************************************************************************
void CEditor::connectAsCreator()
{
//H_AUTO(R2_CEditor_connectAsCreator)
CHECK_EDITOR
nlassert(_DMC);
{
CLuaStackChecker lsc2(&getLua());
_DMC->testConnectionAsCreator();
}
}
// string cached into lua for fast comparison
static CLuaString lstr_isNil("isNil");
static CLuaString lstr_Parent("Parent");
static CLuaString lstr_ParentInstance("ParentInstance");
static CLuaString lstr_IndexInParent("IndexInParent");
static CLuaString lstr_User("User");
static CLuaString lstr_Size("Size");
static CLuaString lstr_DisplayerUI("DisplayerUI");
static CLuaString lstr_DisplayerVisual("DisplayerVisual");
static CLuaString lstr_DisplayerProperties("DisplayerProperties");
// *********************************************************************************************************
void CEditor::initObjectProjectionMetatable()
{
//H_AUTO(R2_CEditor_initObjectProjectionMetatable)
CHECK_EDITOR
_ObjectProjectionMetatable = getEnv().newTable("_instance_projection_metatable");
// projection functions
struct CProjector
{
static bool checkTag(CLuaState &ls)
{
CLuaStackChecker lsc(&ls);
ls.getMetaTable(1);
ls.push("tag");
ls.getTable(-2);
bool ok = false;
if (ls.isString(-1))
{
if (strcmp(ls.toString(-1), "CObjectTable") == 0)
{
ok = true;
}
}
ls.pop(2);
if (!ok)
{
nlwarning("metatable error : Editor object expected");
}
return ok;
}
static int index(CLuaState &ls)
{
nlassert(ls.getTop() == 2);
#ifdef NL_DEBUG
if (!checkTag(ls)) return false;
#endif
CObjectTable::TRefPtrConst &obj= *(CObjectTable::TRefPtrConst *) ls.toUserData(1);
if (OPERATOR_EQUAL(obj, NULL))
//NLMISC::operator==(obj, NULL))
{
std::string index = ls.toString(2);
// special members
if ( index == "isNil")
//if (std::operator==(index "isNil")
{
ls.push(true);
return 1;
}
CLuaIHM::dumpCallStack();
// object has been deleted but the script maintains a reference on it
throw ELuaWrappedFunctionException(&ls, "Attempt to access an erased object");
}
if (ls.isNumber(2))
{
// index is a number
const CObject *other = obj->getValue((uint32) ls.toNumber(2));
if (other)
{
pushValue(ls, other);
return 1;
}
else
{
// 'bad index' message already printed by CObject::getValue
CLuaIHM::dumpCallStack();
}
}
if (!ls.isString(2))
{
nlwarning("String expected when accessing an object property, %s found instead", ls.getTypename(2));
CLuaIHM::dumpCallStack(0);
return 0;
}
const char *strId = ls.toString(2); // strings are shared in lua, so a pointer comparison is sufficient
// special members
// NB nico : this was added before the R2::CInstance class derived from CReflectable,
// -> see if some of the following properties may better fit into R2::CInstance exported properties
if (OPERATOR_EQUAL( lstr_isNil, strId))
{
ls.push(false);
return 1;
}
else
if (OPERATOR_EQUAL(strId, lstr_Parent))
{
if (!obj->getParent())
{
ls.pushNil();
return 1;
}
nlassert(obj->getParent()->isTable());
getEditor().projectInLua((CObjectTable *) obj->getParent());
return 1;
}
else if (OPERATOR_EQUAL(strId, lstr_ParentInstance))
{
// look for parent instance (that is, not the
CObject *parent = obj->getParent();
while (parent)
{
if (parent->findAttr("InstanceId"))
{
getEditor().projectInLua((CObjectTable *) parent);
return 1;
}
parent = parent->getParent();
}
ls.pushNil();
return 1;
}
if (OPERATOR_EQUAL(strId, lstr_IndexInParent))
{
if (!obj->getParent())
{
ls.pushNil();
return 1;
}
sint32 index = obj->getParent()->findIndex(obj);
if (obj->getParent()->getKey(index).empty())
{
ls.push((double) index);
}
else
{
ls.push((double) -1);
}
return 1;
}
else if (OPERATOR_EQUAL(strId, lstr_User))
{
// push user table on stack
CEditor::getLuaUserTableFromObject(ls, *obj);
return 1;
}
else if (OPERATOR_EQUAL(strId, lstr_Size))
{
if (obj->isTable())
{
ls.push((double) obj->getSize());
}
else
{
ls.pushNil();
}
return 1;
}
else
if (OPERATOR_EQUAL(strId, lstr_DisplayerUI))
{
CInstance *instance = getEditor().getInstanceFromObject(obj);
if (!instance)
{
ls.pushNil();
return 1;
}
else
{
if (instance->getDisplayerUI())
{
instance->getDisplayerUI()->pushLuaAccess(ls);
}
else
{
ls.pushNil();
}
return 1;
}
}
else
if (OPERATOR_EQUAL(strId, lstr_DisplayerVisual))
{
CInstance *instance = getEditor().getInstanceFromObject(obj);
if (!instance)
{
ls.pushNil();
return 1;
}
else
{
if (instance->getDisplayerVisual())
{
instance->getDisplayerVisual()->pushLuaAccess(ls);
}
else
{
ls.pushNil();
}
return 1;
}
}
else
if (OPERATOR_EQUAL(strId, lstr_DisplayerProperties))
{
CInstance *instance = getEditor().getInstanceFromObject(obj);
if (!instance)
{
ls.pushNil();
return 1;
}
else
{
if (instance->getDisplayerProperties())
{
instance->getDisplayerProperties()->pushLuaAccess(ls);
}
else
{
ls.pushNil();
}
return 1;
}
}
// else tries with a string
std::string index = ls.toString(2);
const CObject *other = getObject(obj, index);
if (other)
{
pushValue(ls, other);
return 1;
}
// this is not an attribute of the class
// -> maybe this is a method ?
// look in the parent class is there is a function with the wanted name
// NOTE : calling a method on an editor object in the script can be done by doing
// r2:getClass(instance).methodName(instance, parameters ...)
// with this trick, one can do instance:methodName(parameters ...)
CObject *className = obj->findAttr("Class");
if (className)
{
CLuaObject method = getEditor().getClasses()[className->toString()][index];
if (method.isFunction())
{
method.push();
return 1;
}
}
// Try with a property exported from CInstance
// Is is ok to test thos properties after those defined in lua classes definition
// because the init check that no "native" property is declared by the user
CInstance *instance = getEditor().getInstanceFromObject(obj);
if (instance)
{
const CReflectedProperty *prop = instance->getReflectedProperty(index, false);
if (prop)
{
CLuaIHM::luaValueFromReflectedProperty(ls, *instance, *prop);
return 1;
}
}
// unknown attribute
ls.pushNil();
return 1;
}
static int newIndex(CLuaState &ls)
{
nlassert(ls.getTop() == 3);
#ifdef NL_DEBUG
if (!checkTag(ls)) return false;
#endif
CObjectTable::TRefPtrConst &obj= *(CObjectTable::TRefPtrConst *) ls.toUserData(1);
if (obj == NULL)
{
throw ELuaWrappedFunctionException(&ls, "Trying to set a property in an erased object, returning nil");
}
#ifdef NL_DEBUG
if (!checkTag(ls)) return false;
#endif
// try with a native (local ..) property exported from CInstance
CInstance *instance = getEditor().getInstanceFromObject(obj);
if (instance)
{
const CReflectedProperty *prop = instance->getReflectedProperty(ls.toString(2));
if (prop)
{
CLuaIHM::luaValueToReflectedProperty(ls, 3, *instance, *prop);
return 1;
}
}
// other instances properties are read only !!
throw ELuaWrappedFunctionException(&ls, "Property %s of editor object is read-only. You must use 'r2.requestSetNode' function to change it", ls.toString(2));
}
static int gc(CLuaState &ls)
{
#ifdef NL_DEBUG
if (!checkTag(ls)) return false;
#endif
// TODO : maybe not useful ...
CObjectTable::TRefPtrConst &obj= *(CObjectTable::TRefPtrConst *) ls.toUserData(1);
typedef CObjectTable::TRefPtrConst TRefPtrConst;
obj.~TRefPtrConst();
return 0;
}
// tool function used by 'next'
// TODO : put this code in a better place ...
static void pushValue(CLuaState &ls, const CObject *obj)
{
if (!obj)
{
ls.pushNil();
return;
}
if (obj->isString())
{
ls.push(obj->toString());
}
else if (obj->isNumber())
{
ls.push(obj->toNumber());
}
else if (obj->isTable())
{
getEditor().projectInLua((CObjectTable *) obj);
}
else
{
nlassert(0); // type not supported ...
}
}
// helper function used by 'next' : push a key
static void pushKey(CLuaState &ls, const CObject *obj, sint32 index)
{
if (!obj)
{
ls.pushNil();
return;
}
if (obj->getKey(index).empty())
{
ls.push((double) index);
}
else
{
ls.push(obj->getKey(index));
}
}
// special : we redefine the 'next' global function found in the standard
// lua library in order to allow fields traversal for the projected object.
// This is done in lua in the 'r2_core.lua' script
// The new 'next' function will look into the object metatable
// If a __next function is found, then it will be called for traversal
// otherwise, the standard 'next' function is called instead.
// This allow to have a traversal for objects that are not tables
static int next(CLuaState &ls)
{
if (ls.getTop() != 2)
{
CLuaIHM::fails(ls, "__next metamethod require 2 arguments (table & key)");
}
#ifdef NL_DEBUG
if (!checkTag(ls)) return false;
#endif
CObjectTable::TRefPtrConst &obj= *(CObjectTable::TRefPtrConst *) ls.toUserData(1);
if (obj == NULL)
{
throw ELuaWrappedFunctionException(&ls, "editor object '__next' metatmethod : Attempt to access an erased object");
}
if (ls.isNil(2))
{
// key is nil -> start of traversal
if(obj->getSize() == 0)
{
ls.remove(-2);
ls.pushNil();
return 2; // let (nil, nil) on the stack
}
else
{
// look for duplicated keys (ignoring empty keys, because in this case th index is the key)
std::set<std::string> keys;
for(uint k = 0; k < obj->getSize(); ++k)
{
std::string key = obj->getKey((sint32) k);
if (!key.empty())
{
if (keys.count(key))
{
nlwarning("Duplicated key of type string found while attempting to enumerate an instance content.");
nlwarning("key is %s", key.c_str());
CLuaIHM::dumpCallStack(1);
CLuaIHM::fails(ls, "Aborting to avoid infinite loop.");
}
keys.insert(key);
}
}
ls.pop(2);
pushKey(ls, obj, 0);
pushValue(ls, obj->getValue(0));
return 2;
}
}
else
{
// continuation of traversal
// -> retrieve index from the key
sint32 index;
if (ls.isNumber(2))
{
index = (uint32) ls.toNumber(2);
}
else
{
if (!ls.isString(2))
{
nlwarning("__next metamethod : string expected");
ls.pop();
ls.pushNil();
return 2;
}
else
{
index = obj->findIndex(ls.toString(2));
}
}
if (index == -1)
{
nlwarning("__next metamethod : key not found");
ls.pop(2);
ls.pushNil();
ls.pushNil();
return 2;
}
// retrieve next key
sint32 newIndex = (uint32) (index + 1);
if (newIndex == (sint32) obj->getSize())
{
// this was the last element, returns nil
ls.pop(2);
ls.pushNil();
ls.pushNil();
return 2;
}
else
{
ls.pop(2);
pushKey(ls, obj, newIndex);
pushValue(ls, obj->getValue(newIndex));
return 2;
}
}
return 0;
}
static int equal(CLuaState &ls)
{
// TMP TMP TMP
static volatile bool from = false;
if (from)
{
CLuaIHM::dumpCallStack(0);
}
nlassert(ls.getTop() == 2);
if (!checkTag(ls)) return false;
CObjectTable::TRefPtrConst &lhs= *(CObjectTable::TRefPtrConst *) ls.toUserData(1);
CObjectTable::TRefPtrConst &rhs= *(CObjectTable::TRefPtrConst *) ls.toUserData(2);
ls.push(OPERATOR_EQUAL(lhs, rhs));
return 1;
}
};
// affect functions to the metatable
getLua().push(CProjector::index);
_ObjectProjectionMetatable.setValue("__index", CLuaObject(getLua()));
getLua().push(CProjector::newIndex);
_ObjectProjectionMetatable.setValue("__newindex", CLuaObject(getLua()));
getLua().push(CProjector::gc);
_ObjectProjectionMetatable.setValue("__gc", CLuaObject(getLua()));
getLua().push(CProjector::next);
_ObjectProjectionMetatable.setValue("__next", CLuaObject(getLua()));
getLua().push(CProjector::equal);
_ObjectProjectionMetatable.setValue("__eq", CLuaObject(getLua()));
// tag to mark that the user data is a CObjectTable ref ptr
_ObjectProjectionMetatable.setValue("tag", std::string("CObjectTable"));
}
// *********************************************************************************************************
void CEditor::getLuaUserTableFromObject(CLuaState &ls, const CObjectTable &table)
{
//H_AUTO(R2_CEditor_getLuaUserTableFromObject)
CHECK_EDITOR
CLuaStackChecker lsc(&ls, 1);
// read write environement accessible from lua
ls.pushLightUserData((void *) &table);
ls.getTable(LUA_REGISTRYINDEX);
if (ls.isNil())
{
ls.pop();
// no user table created yet ?... create it
ls.pushLightUserData((void *) &table);
ls.newTable();
ls.pushValue(-1); // saves the table
ls.insert(-3);
ls.setTable(LUA_REGISTRYINDEX);
// table remains on the stack
}
}
// *********************************************************************************************************
void CEditor::setCurrentAct(CInstance *act)
{
//H_AUTO(R2_CEditor_setCurrentAct)
CHECK_EDITOR
if (!_ScenarioInstance) return;
if (_ClearingContent) return;
bool currentActSelected = _SelectedInstance && (_SelectedInstance == _CurrentAct);
CInstance* previousAct = _CurrentAct;
_CurrentAct = act ? act : getBaseAct();
//
struct CPreActChangedVisitor : public IInstanceVisitor
{
virtual void visit(CInstance &inst)
{
inst.onPreActChanged();
}
};
CPreActChangedVisitor preActChangedVisitor;
_ScenarioInstance->visit(preActChangedVisitor);
struct CActChangedVisitor : public IInstanceVisitor
{
virtual void visit(CInstance &inst)
{
inst.onActChanged();
}
};
CActChangedVisitor actChangedVisitor;
_ScenarioInstance->visit(actChangedVisitor);
//TP if necesary
// warn lua that current act has changed
if (!previousAct)
{
getLua().pushNil();
}
else
{
previousAct->getLuaProjection().push();
}
if (!_CurrentAct)
{
getLua().pushNil();
}
else
{
_CurrentAct->getLuaProjection().push();
}
callEnvMethod("onActChanged", 2, 0);
setSelectedInstance(currentActSelected ? _CurrentAct : NULL);
setCurrentTool(NULL);
}
// *********************************************************************************************************
void CEditor::setCurrentActFromTitle(const std::string &wantedTitle)
{
//H_AUTO(R2_CEditor_setCurrentActFromTitle)
CHECK_EDITOR
if (!_Scenario)
{
nlwarning("Scenario initialisation failed, can't set current act");
return;
}
CObject *actTable = _Scenario->getAttr("Acts");
if (actTable->isTable())
{
for(uint k = 0; k < actTable->getSize(); ++k)
{
R2::CObject *act = actTable->getValue(k);
nlassert(act);
std::string actTitle;
if (act->isString("Name"))
{
actTitle = act->toString("Name");
}
else
{
actTitle = act->toString("Title"); //obsolete
}
if (actTitle == wantedTitle)
{
setCurrentAct(getInstanceFromObject(act));
break;
}
}
}
else
{
nlwarning("'Acts' field in scenario should be a table");
}
}
// *********************************************************************************************************
void CEditor::projectInLua(const CObjectTable *table)
{
//H_AUTO(R2_CEditor_projectInLua)
CHECK_EDITOR
nlassert(table);
const CObjectTableClient *otc = NLMISC::safe_cast<const CObjectTableClient *>(table);
otc->pushOnLuaStack(getLua(), _ObjectProjectionMetatable);
}
// *********************************************************************************************************
void CEditor::backupRequestCommands()
{
//H_AUTO(R2_CEditor_backupRequestCommands)
nlassert(!_OldLuaRequestInsertNode.isValid()); // restoreRequestCommands not called ?
nlassert(!_OldLuaRequestInsertGhostNode.isValid());
nlassert(!_OldLuaRequestSetNode.isValid());
nlassert(!_OldLuaRequestEraseNode.isValid());
nlassert(!_OldLuaRequestMoveNode.isValid());
_OldLuaRequestInsertNode = getEnv()["requestInsertNode"];
_OldLuaRequestInsertGhostNode = getEnv()["requestInsertGhostNode"];
_OldLuaRequestSetNode = getEnv()["requestSetNode"];
_OldLuaRequestEraseNode = getEnv()["requestEraseNode"];
_OldLuaRequestMoveNode = getEnv()["requestMoveNode"];
struct CIgnoreRequestCall
{
static int requestSomething(CLuaState &)
{
nlwarning("PAS BIEN C MAL !!!");
/*
CLuaIHM::debugInfo("Can't call a 'r2.request' command while object are erased in displayers, or are just being created (in this case, use onPostCreate instead) !");
CLuaIHM::debugInfo("Callstack is :");
CLuaIHM::dumpCallStack(1);*/
return 0;
}
};
// While we delete the object tree, displayers will be triggered
// Make sure that they don't call any r2.requestSomething commands, end if so,
// print an error message
getEnv().setValue("requestInsertNode", CIgnoreRequestCall::requestSomething);
getEnv().setValue("requestInsertGhostNode", CIgnoreRequestCall::requestSomething);
getEnv().setValue("requestSetNode", CIgnoreRequestCall::requestSomething);
getEnv().setValue("requestEraseNode", CIgnoreRequestCall::requestSomething);
getEnv().setValue("requestMoveNode", CIgnoreRequestCall::requestSomething);
}
// *********************************************************************************************************
void CEditor::restoreRequestCommands()
{
//H_AUTO(R2_CEditor_restoreRequestCommands)
nlassert(_OldLuaRequestInsertNode.isValid()); // backupRequestCommands not called ?
nlassert(_OldLuaRequestInsertGhostNode.isValid());
nlassert(_OldLuaRequestSetNode.isValid());
nlassert(_OldLuaRequestEraseNode.isValid());
nlassert(_OldLuaRequestMoveNode.isValid());
// restore old functions
getEnv().setValue("requestInsertNode", _OldLuaRequestInsertNode);
getEnv().setValue("requestInsertGhostNode", _OldLuaRequestInsertGhostNode);
getEnv().setValue("requestSetNode", _OldLuaRequestSetNode);
getEnv().setValue("requestEraseNode", _OldLuaRequestEraseNode);
getEnv().setValue("requestMoveNode", _OldLuaRequestMoveNode);
//
_OldLuaRequestInsertNode.release();
_OldLuaRequestInsertGhostNode.release();
_OldLuaRequestSetNode.release();
_OldLuaRequestEraseNode.release();
_OldLuaRequestMoveNode.release();
}
// *********************************************************************************************************
void CEditor::clearContent()
{
nlwarning("*R2* clear content");
setMaxVisibleEntityExceededFlag(false);
//H_AUTO(R2_CEditor_clearContent)
if (_Mode != EditionMode && _Mode != AnimationModeLoading) return;
_ClearingContent = true;
CHECK_EDITOR
setSelectedInstance(NULL);
setFocusedInstance(NULL);
// backup all "requestCommand"
if (_CurrentTool) _CurrentTool->cancel();
CTool::releaseMouse();
delete _NewScenario; // will be not NULL only if quit is called during the scenario connection screen
_NewScenario = NULL;
eraseScenario();
_InstancesByDispName.clear();
_InstancesByDispName.resize(_ClassNameToIndex.size());
_InstanceObservers.clear();
_InstanceObserverHandles.clear();
_Cookies.clear();
_LocalGeneratedNames.clear();
_Instances.clear();
_BaseAct = NULL;
_CurrentAct = NULL;
_SelectedInstance = NULL;
_FocusedInstance = NULL;
_ScenarioInstance = NULL;
_Scenario = NULL;
_SelectingDecals.clear();
_InstanceObservers.clear();
_InstanceObserverHandles.clear();
_LastInstanceUnderPos = NULL;
_CurrentTool = NULL;
removeAllEntitySlots(); // for safety...
_DMC->CDynamicMapClient::scenarioUpdated(NULL, false, 1); // clear scenario for real
_ClearingContent = false;
_PlotItemInfos.clear();
PeopleInterraction.removeAllFreeTellers();
}
// *********************************************************************************************************
void CEditor::setPlotItemInfos(const TMissionItem &infos)
{
//H_AUTO(R2_CEditor_setPlotItemInfos)
_PlotItemInfos[infos.SheetId.asInt()] = infos;
}
// *********************************************************************************************************
const TMissionItem *CEditor::getPlotItemInfos(uint32 sheetId) const
{
//H_AUTO(R2_CEditor_getPlotItemInfos)
std::map<uint32, TMissionItem>::const_iterator it = _PlotItemInfos.find(sheetId);
if (it == _PlotItemInfos.end()) return NULL;
return &it->second;
}
// *********************************************************************************************************
void CEditor::release()
{
nlwarning("*R2* release");
//H_AUTO(R2_CEditor_release)
GameContextMenu.init("");
setFixedLighting(false);
CHECK_EDITOR
if (_Mode == NotInitialized)
{
// nothing more to do there
return;
}
// warn lua of final release
callEnvMethod("onFinalRelease", 0, 0);
if (_SerializeUIConfig)
{
// serialization disabled when resetting the editor (for speed)
saveCurrentKeySet();
saveUIConfig();
}
getUI().hideAllWindows(); // make sure all action handlers are called while the r2 lua environment is still active
clearContent();
_EntityCustomSelectBoxMap.clear();
// must do this after clearContent, which sets the default tool
if (_CurrentTool)
{
_CurrentTool->cancel();
_CurrentTool = NULL;
}
// clear the environment
if (getUI().getLuaState())
{
getLua().push(R2_LUA_PATH);
getLua().pushNil();
getLua().setTable(LUA_GLOBALSINDEX);
_Globals.release();
_Registry.release();
_ObjectProjectionMetatable.release(); // AJM
_Env.release();
_Config.release();
CEditorCheck::EditorCreated = false;
// force a garbage collection to free the mem for real
getLua().setGCThreshold(0);
}
// stop at login
if (_DMC)
{
_DMC->release();
}
//
nlassert(CDisplayerBase::ObjCount == 0);
_Initialized = false;
CEditorCheck::EditorCreated = false; // AJM
_Mode = NotInitialized;
_AccessMode = AccessModeUnknown;
::IgnoreEntityDbUpdates = false;
//
_IslandCollision.release();
//
_LastUIModificationDate.clear();
_InstancesByDispName.clear();
//
delete _EntitySorter;
_EntitySorter = NULL;
}
// *********************************************************************************************************
void CEditor::removeAllEntitySlots()
{
//H_AUTO(R2_CEditor_removeAllEntitySlots)
CHECK_EDITOR
for (uint k = 1; k < 255; ++k)
{
if (EntitiesMngr.entity(k) != NULL)
{
EntitiesMngr.remove(k, false);
}
}
}
// *********************************************************************************************************
void CEditor::registerDisplayers()
{
//H_AUTO(R2_CEditor_registerDisplayers)
CHECK_EDITOR
static bool registered = false;
if (registered) return;
NLMISC_REGISTER_CLASS(R2::CDisplayerVisualActivitySequence);
NLMISC_REGISTER_CLASS(R2::CDisplayerVisualEntity);
NLMISC_REGISTER_CLASS(R2::CDisplayerVisualGroup)
NLMISC_REGISTER_CLASS(R2::CDisplayerVisualShape)
NLMISC_REGISTER_CLASS(R2::CDisplayerLua)
registered = true;
}
// *********************************************************************************************************
void CEditor::registerTools()
{
//H_AUTO(R2_CEditor_registerTools)
CHECK_EDITOR
static bool registered = false;
if (registered) return;
NLMISC_REGISTER_CLASS(R2::CToolDrawPrim);
NLMISC_REGISTER_CLASS(R2::CToolSelectMove);
NLMISC_REGISTER_CLASS(R2::CToolSelectRotate);
NLMISC_REGISTER_CLASS(R2::CToolNewVertex);
registered = true;
}
// *********************************************************************************************************
void CEditor::setSelectedInstance(CInstance *inst)
{
//H_AUTO(R2_CEditor_setSelectedInstance)
CHECK_EDITOR
if (inst == _SelectedInstance) return;
forceSetSelectedInstance(inst);
}
// *********************************************************************************************************
void CEditor::forceSetSelectedInstance(CInstance *inst)
{
//H_AUTO(R2_CEditor_forceSetSelectedInstance)
CHECK_EDITOR
if (_EnteredInSetSelectedInstance) return; // prevent recursive call (may happen because selection highlight an item in a menu, which trigger selection in turn)
if (inst)
{
nlassert(inst->getSelectableFromRoot());
}
_EnteredInSetSelectedInstance = true;
if (inst)
{
nlassert(hasInstance(inst)); // must have been added in the editor !
}
if (_SelectedInstance)
{
_SelectedInstance->onSelect(false);
}
_SelectedInstance = inst;
if (_SelectedInstance)
{
_SelectedInstance->onSelect(true);
}
// call r2 method 'onSelectInstance'
if (!_SelectedInstance)
{
getLua().pushNil();
}
else
{
_SelectedInstance->getLuaProjection().push();
}
callEnvMethod("onSelectInstance", 1, 0);
_EnteredInSetSelectedInstance = false;
}
// *********************************************************************************************************
CInstance *CEditor::getSelectedInstance() const
{
//H_AUTO(R2_CEditor_getSelectedInstance)
CHECK_EDITOR
return _SelectedInstance;
}
// *********************************************************************************************************
void CEditor::setFocusedInstance(CInstance *inst)
{
//H_AUTO(R2_CEditor_setFocusedInstance)
CHECK_EDITOR
if (inst == _FocusedInstance) return;
if (inst)
{
nlassert(hasInstance(inst)); // must have been added in the editor !
}
if (_FocusedInstance)
{
_FocusedInstance->onFocus(false);
}
_FocusedInstance = inst;
if (_FocusedInstance)
{
_FocusedInstance->onFocus(true);
}
}
// *********************************************************************************************************
CInstance *CEditor::getFocusedInstance() const
{
//H_AUTO(R2_CEditor_getFocusedInstance)
CHECK_EDITOR
return _FocusedInstance;
}
// *********************************************************************************************************
bool CEditor::hasInstance(const CInstance *instance) const
{
//H_AUTO(R2_CEditor_hasInstance)
CHECK_EDITOR
// TMP : slow test for debug
for(TInstanceMap::const_iterator it = _Instances.begin(); it != _Instances.end(); ++it)
{
if (it->second == instance) return true;
}
return false;
}
// *********************************************************************************************************
bool CEditor::hasInstanceWithId(const TInstanceId &id) const
{
//H_AUTO(R2_CEditor_hasInstanceWithId)
CHECK_EDITOR
return getInstanceFromId(id) != NULL;
}
// *********************************************************************************************************
CInstance *CEditor::getInstanceFromId(const TInstanceId &id) const
{
//H_AUTO(R2_CEditor_getInstanceFromId)
CHECK_EDITOR
const CObjectTable *objTable = getObjectTableFromId(id);
if (!objTable) return NULL;
return getInstanceFromObject(objTable);
}
// *********************************************************************************************************
sint CEditor::getGeneratedNameIndex(const std::string &nameUtf8, const std::string &baseNameUtf8)
{
//H_AUTO(R2_CEditor_getGeneratedNameIndex)
CHECK_EDITOR
if (nameUtf8.size() >= baseNameUtf8.size() + 2)
{
if (nameUtf8.substr(0, baseNameUtf8.size()) == baseNameUtf8)
{
std::string::const_iterator strIt = nameUtf8.begin() + baseNameUtf8.size();
std::string::const_iterator endStrIt = nameUtf8.end();
if (*strIt == ' ')
{
++ strIt;
const char *numberStart = &*strIt;
for (; strIt != endStrIt && isdigit(*strIt); ++strIt) {}
if (strIt == endStrIt)
{
sint ret;
fromString(numberStart, ret);
return ret;
}
}
}
}
return -1;
}
// *********************************************************************************************************
bool CEditor::isPostFixedByNumber(const ucstring &baseName)
{
//H_AUTO(R2_CEditor_isPostFixedByNumber)
// strip number & spaces at the end of the name
sint lastIndex = (sint)baseName.length() - 1;
while (lastIndex > 0)
{
int currChar = (int) baseName[lastIndex];
if (!isdigit(currChar) &&
currChar != ' ' &&
currChar != '\t')
{
break;
}
-- lastIndex;
}
return lastIndex != (sint) baseName.length() - 1;
}
// *********************************************************************************************************
ucstring CEditor::genInstanceName(const ucstring &baseName)
{
//H_AUTO(R2_CEditor_genInstanceName)
CHECK_EDITOR
uint maxIndex = 0;
// strip number & spaces at the end of the name
ucstring strippedName = baseName;
sint lastIndex = (sint)strippedName.length() - 1;
while (lastIndex > 0)
{
int currChar = (int) strippedName[lastIndex];
if (!isdigit(currChar) &&
currChar != ' ' &&
currChar != '\t')
{
break;
}
-- lastIndex;
}
strippedName = strippedName.substr(0, lastIndex + 1);
std::string baseNameUtf8 = strippedName.toUtf8();
//
for(TInstanceMap::const_iterator it = _Instances.begin(); it != _Instances.end(); ++it)
{
maxIndex = (uint) std::max((sint) maxIndex, getGeneratedNameIndex(it->second->getName(), baseNameUtf8));
}
nlinfo("CEditor::genInstanceName newVersion");
// Old part : used when there was a delay between net command and effect
// did not work in some parts if genInstance is not followed by creation of an instance with that name
// cf RT 11560
/*
// now, look in local generated names (instance that have not been already added to the scene)
TGeneratedNameMap::iterator nameSetIt = _LocalGeneratedNames.find(baseNameUtf8);
if (nameSetIt != _LocalGeneratedNames.end())
{
if (!nameSetIt->second.empty())
{
maxIndex = std::max(maxIndex, *(nameSetIt->second.rbegin())); // take bigger element of the set (the last element)
}
}
// add max index in the map (will be removed when object is truly created)
_LocalGeneratedNames[baseNameUtf8].insert(maxIndex + 1);
*/
return strippedName + " " + toString(maxIndex + 1);
}
// *********************************************************************************************************
void CEditor::setCookie(const TInstanceId &instanceId, const std::string &key, CLuaObject &value)
{
//H_AUTO(R2_CEditor_setCookie)
CHECK_EDITOR
TCookieList &cl = _Cookies[instanceId];
cl.push_front(CCookie());
CCookie &cookie = cl.front(); // don't build struct and push to avoid costly copy of a CLuaObject
cookie.Key = key;
cookie.Value = value;
}
// *********************************************************************************************************
void CEditor::setCookie(const TInstanceId &instanceId, const std::string &key, bool value)
{
//H_AUTO(R2_CEditor_setCookie)
CHECK_EDITOR
getLua().push(value);
CLuaObject obj(getLua());
setCookie(instanceId, key, obj);
}
// *********************************************************************************************************
void CEditor::setCurrentTool(CTool *tool)
{
//H_AUTO(R2_CEditor_setCurrentTool)
CHECK_EDITOR
if (_CurrentTool)
{
_CurrentTool->cancel();
}
else
{
if (tool)
{
// setting a tool in non R2 mode force to change the contextual cursor
ContextCur.release();
ContextCur.add(false, "STAND BY", DEFAULT_CURSOR, 0.0f, CEditor::checkCursor, CEditor::mouseClick);
ContextCur.context("STAND BY");
}
}
CTool::setContextHelp(ucstring(""));
if (tool == NULL)
{
if (_Mode == EditionMode)
{
_CurrentTool = new CToolSelectMove;
}
else
{
_CurrentTool = NULL;
ContextCur.release();
::initContextualCursor();
ContextCur.context("STAND BY");
CTool::setMouseCursor(DEFAULT_CURSOR);
}
}
else
{
_CurrentTool = tool;
}
if (_CurrentTool)
{
_CurrentTool->onActivate();
}
{
CLuaStackChecker lsc(&getLua());
// activate tool in the ui
getLua().push(_CurrentTool ? _CurrentTool->getToolUIName() : "");
getEnv()["ToolUI"].callMethodByNameNoThrow("setActiveToolUIByName", 1, 0);
}
}
// *********************************************************************************************************
CLuaObject CEditor::getClasses() throw(ELuaError)
{
//H_AUTO(R2_getClasses_throw)
CHECK_EDITOR
return getEnv().at("Classes");
}
// *********************************************************************************************************
bool CEditor::callEnvFunc(const char *funcName, int numArgs, int numRet /*=0*/)
{
//H_AUTO(R2_CEditor_callEnvFunc)
CHECK_EDITOR
static volatile bool dumpStackWanted = false;
if (dumpStackWanted) getLua().dumpStack();
nlassert(funcName);
nlassert(getLua().getTop() >= numArgs);
int initialStackSize = getLua().getTop();
getEnv().push();
if (dumpStackWanted) getLua().dumpStack();
getLua().insert(-1 - numArgs); // put the table before the args
if (dumpStackWanted) getLua().dumpStack();
int result = getLua().pcallByName(funcName, numArgs, numRet, -1 - numArgs);
if (dumpStackWanted) getLua().dumpStack();
if (result != 0)
{
nlwarning("Error while calling function %s : %s", funcName, getLua().toString());
getLua().setTop(initialStackSize - numArgs + numRet); // clean the stack
return false;
}
// remove the R2 table from the stack
int newSize = getLua().getTop();
if (dumpStackWanted) getLua().dumpStack();
nlassert(newSize == initialStackSize + 1 - numArgs + numRet); // -1 is because of the r2 table
if (dumpStackWanted) getLua().dumpStack();
getLua().remove(- numRet - 1); // remove the 'R2' table
if (dumpStackWanted) getLua().dumpStack();
return true; // results remains on the stack
}
// *********************************************************************************************************
bool CEditor::callEnvMethod(const char *funcName, int numArgs, int numRet /*= 0*/)
{
//H_AUTO(R2_CEditor_callEnvMethod)
CHECK_EDITOR
nlassert(funcName);
nlassert(getLua().getTop() >= numArgs);
getEnv().push();
getLua().insert(-1 - numArgs); // put the table before the args (self parameter)
return callEnvFunc(funcName, numArgs + 1, numRet);
}
// *********************************************************************************************************
bool CEditor::doLuaScript(const char *filename, const char *fileDescText)
{
//H_AUTO(R2_CEditor_doLuaScript)
CHECK_EDITOR
CVerboseClock clock("parsing of " + std::string(filename));
// load the classes definition file
std::string filePath = NLMISC::CPath::lookup(filename, false, true);
if (filePath.empty())
{
nlwarning("Can't find %s : %s", fileDescText, filename);
return false;
}
if( 0 && FINAL_VERSION == 1) // deactivated for the moment because there are lua file that must be loaded from example
{
const static std::string path = "data_common.bnp@";
const static std::string::size_type len= path.size();
if (filePath.size() < len || filePath.substr(0, len) != path)
{
nlwarning("Can't find %s : %s in ('%s')", fileDescText, filename, path.c_str());
return false;
}
}
try
{
if (!getLua().executeFile(filePath))
{
nlwarning("Couldn't open file %s : %s for R2 is not loaded", filename, fileDescText);
}
CLuaStackChecker ls(&getLua());
return true;
}
catch(NLMISC::EStream &e)
{
nlwarning("Error while loading R2 %s (file = %s) : %s", fileDescText, filename, e.what());
}
catch(ELuaError &e)
{
//char filename[MAX_PATH];
std::string msg = e.what();
if (testWildCard(msg, "*.lua:*:*")) // TODO nico : more accurate testing for filename format
{
std::string::size_type extPos = msg.find(".lua");
nlassert(extPos != std::string::npos);
std::string filename = msg.substr(0, extPos + 4); // extract filename including extension
int line;
fromString(&*(msg.begin() + extPos + 5), line); // line number follows
nlwarning((CLuaIHM::createGotoFileButtonTag(filename.c_str(), line) + e.what()).c_str());
}
else
{
// no filename / line in the result, so maybe the parse error was in an 'inline' script
nlwarning("Error in R2 %s (file = %s) : %s", fileDescText, filename, e.what());
}
}
return false;
}
///////////////////////
// CONTEXTUAL CURSOR //
///////////////////////
// *********************************************************************************************************
void CEditor::checkCursor()
{
//H_AUTO(R2_CEditor_checkCursor)
CHECK_EDITOR
// no-op there : we delegate management of mouse to the CTool derived classes
}
// *********************************************************************************************************
void CEditor::mouseClick(bool rightButton, bool /* dblClick */)
{
//H_AUTO(R2_CEditor_mouseClick)
CHECK_EDITOR
CTool *currentTool = getEditor().getCurrentTool();
if (!currentTool) return;
if (!rightButton)
{
// nb : result not handled there (no defaut action for left click)
currentTool->onMouseLeftButtonClicked();
}
else
{
bool handled = currentTool->onMouseRightButtonClicked();
if (!handled)
{
getEditor().displayContextMenu();
}
}
}
// *********************************************************************************************************
void CEditor::updatePreCamera()
{
//H_AUTO(R2_CEditor_updatePreCamera)
if (_Mode == EditionMode)
{
static uint32 loop = 0;
++loop;
if (loop % 200 == 0) // minimal wait between to save = 20 seconds
{
if ( (CTime::getLocalTime() -_LastAutoSaveTime)/1000 > ClientCfg.R2EDAutoSaveWait) // 5 minutes if not change in Confile
{
autoSave();
}
loop = 0;
}
// if there's no pending action, then start a new one at each frame
if (!_DMC->getActionHistoric().isPendingActionInProgress())
{
_DMC->newAction(CI18N::get("uiR2EDUnamedAction"));
}
else
{
_DMC->getActionHistoric().flushPendingAction();
}
}
}
// *********************************************************************************************************
void CEditor::updatePrimitiveContextualVisibility()
{
//H_AUTO(R2_CEditor_updatePrimitiveContextualVisibility)
if (!_SelectedInstance) return;
// if selected instance remains in last contextual prim list, then they may remain visible
bool ok = false;
for(uint k = 0; k < _LastContextualPrims.size(); ++k)
{
if (_LastContextualPrims[k] == _SelectedInstance ||
_SelectedInstance->isSonOf(_LastContextualPrims[k])
)
{
ok = true;
break;
}
}
_LastContextualPrims.clear();
bool isLogicEntity = _SelectedInstance->isKindOf("LogicEntity");
if (!ok && !isLogicEntity) return;
if (isLogicEntity)
{
_LastContextualLogicEntity = _SelectedInstance;
}
if (!_LastContextualLogicEntity) return;
//
CObject *seq = _LastContextualLogicEntity->getGroupSelectedSequence();
if (!seq) return;
if (seq)
{
CObjectTable *activities = seq->toTable("Components");
if (activities)
{
// get first world object parent to get start position
for(uint k = 0; k < activities->getSize(); ++k)
{
// search next zone of activity
CObjectTable *activity = activities->getValue(k)->toTable();
if (!activity) continue;
std::string zoneId = getString(activity, "ActivityZoneId");
CInstance *primitive = getInstanceFromId(zoneId);
if (primitive)
{
CDisplayerVisualGroup *primDisp = dynamic_cast<CDisplayerVisualGroup *>(primitive->getDisplayerVisual());
if (primDisp)
{
_LastContextualPrims.push_back(primitive);
primDisp->setContextualVisibilityDate(T1);
}
}
}
}
}
}
// *********************************************************************************************************
CEntitySorter *CEditor::getEntitySorter() const
{
return _EntitySorter;
}
// *********************************************************************************************************
void CEditor::updateBeforeRender()
{
//H_AUTO(R2_CEditor_updateBeforeRender)
CHECK_EDITOR
//
_IslandCollision.updateCurrPackedIsland();
// update contextual visibility of primitive from current selection
updatePrimitiveContextualVisibility();
//
if (ConnectionWanted) return; // TMP special case for connection
if (_CurrentTool)
{
_CurrentTool->updateBeforeRender();
}
for(TInstanceMap::iterator it = _Instances.begin(); it != _Instances.end(); ++it)
{
if (it->second->getDisplayerVisual())
{
it->second->getDisplayerVisual()->onPreRender();
}
}
updateSelectingDecals();
// hide or show user depending on the mode
if (UserEntity)
{
if (_Mode == EditionMode || isDMing())
{
updateDecalBlendRegion(_PionneerDecal, UserEntity->pos());
showDecal(CVector2f((float) UserEntity->pos().x, (float) UserEntity->pos().y), 1.f, _PionneerDecal, _PionneerDecalAnim);
}
}
if ((_SelectedInstance && _SelectedInstance->maxVisibleEntityExceeded()) ||
(_FocusedInstance && _FocusedInstance->maxVisibleEntityExceeded()))
{
setMaxVisibleEntityExceededFlag(true);
}
else
{
setMaxVisibleEntityExceededFlag(false);
}
}
// *********************************************************************************************************
void CEditor::updateDecalBlendRegion(CDecal &decal, const NLMISC::CVector &pos)
{
//H_AUTO(R2_CEditor_updateDecalBlendRegion)
float topBlendDist = CV_DecalTopBlendStartDist.get();
float bottomBlendDist = CV_DecalBottomBlendStartDist.get();
float blendLength = CV_DecalBlendLength.get();
decal.setBottomBlend(pos.z - bottomBlendDist - blendLength,
pos.z - bottomBlendDist);
decal.setTopBlend(pos.z + topBlendDist,
pos.z + topBlendDist + blendLength);
}
// *********************************************************************************************************
void CEditor::updateBeforeSwapBuffer()
{
//H_AUTO(R2_CEditor_updateBeforeSwapBuffer)
CHECK_EDITOR
if (_WaitScenarioScreenWanted)
{
waitScenario();
_WaitScenarioScreenWanted= false;
}
}
// *********************************************************************************************************
void CEditor::waitScenario()
{
//H_AUTO(R2_CEditor_waitScenario)
_EditionModeDisconnectedFlag = false;
_WaitScenarioScreenActive = true;
waitScenarioScreen();
_WaitScenarioScreenActive = false;
if (_PostponeScenarioUpdated)
{
scenarioUpdated(_NewScenario, false, _NewScenarioInitialAct);
_PostponeScenarioUpdated = false;
_NewScenario = NULL;
}
}
// *********************************************************************************************************
void CEditor::updateAfterRender()
{
if (_EntitySorter) _EntitySorter->clipEntitiesByDist();
//H_AUTO(R2_CEditor_updateAfterRender)
_IslandCollision.updateCurrPackedIsland();
CHECK_EDITOR
if (ConnectionWanted)
{
connect();
return;
}
//
if (_CurrentTool)
{
_CurrentTool->updateAfterRender();
}
if (!_LuaUIMainLoop.isNil())
{
_LuaUIMainLoop.callMethodByNameNoThrow("onPostSceneRender", 0, 0);
}
//
for(TInstanceMap::iterator it = _Instances.begin(); it != _Instances.end(); ++it)
{
if (it->second->getDisplayerVisual())
{
it->second->getDisplayerVisual()->onPostRender();
}
}
}
// *********************************************************************************************************
void CEditor::autoSave()
{
//H_AUTO(R2_CEditor_autoSave)
_LastAutoSaveTime = NLMISC::CTime::getLocalTime();
uint32 maxAutoSave = ClientCfg.R2EDAutoSaveSlot;
std::string lastFile = toString("autosave_%u", maxAutoSave);
uint32 i = maxAutoSave -1;
for ( ; i != 0 ; --i )
{
std::string current = NLMISC::toString("autosave_%02u.r2", i);
std::string next = NLMISC::toString("autosave_%02u.r2", i+1);
if (CFile::fileExists(current))
{
if (CFile::fileExists(next)) // true only for i = maxAutoSave -1
{
CFile::deleteFile(next);
}
CFile::moveFile(next.c_str(), current.c_str());
}
}
if (CFile::fileExists("data/r2_buffer.dat"))
{
CFile::copyFile("autosave_01.r2", "data/r2_buffer.dat");
}
R2::getEditor().getLua().executeScriptNoThrow("r2.Version.save(\"data/r2_buffer.dat\")");
}
// *********************************************************************************************************
bool CEditor::handleEvent (const CEventDescriptor &eventDesc)
{
//H_AUTO(R2_CEditor_handleEvent )
CHECK_EDITOR
if (ConnectionWanted || !_CurrentTool) return false; // TMP special case for connection
if (eventDesc.getType() == CEventDescriptor::system)
{
const CEventDescriptorSystem &eds = (const CEventDescriptorSystem &) eventDesc;
if (eds.getEventTypeExtended() == CEventDescriptorSystem::setfocus)
{
const CEventDescriptorSetFocus &edsf = (const CEventDescriptorSetFocus &) eds;
if (edsf.hasFocus() == false)
{
// cancel current tool
setCurrentTool(NULL);
return true;
}
}
}
if (_CurrentTool)
{
return _CurrentTool->handleEvent(eventDesc);
}
return false;
}
// *********************************************************************************************************
void CEditor::copy()
{
//H_AUTO(R2_CEditor_copy)
// forward to lua
callEnvMethod("copy", 0);
}
// *********************************************************************************************************
void CEditor::paste()
{
//H_AUTO(R2_CEditor_paste)
// forward to lua
callEnvMethod("paste", 0);
}
// *********************************************************************************************************
/*
void CEditor::updateEvents()
{
if (!_CurrentTool) return;
//
if(EventsListener.isMouseButtonPushed(leftButton))
{
nlwarning("onMouseLeftButtonDown");
_CurrentTool->onMouseLeftButtonDown();
}
if(EventsListener.isMouseButtonReleased(leftButton))
{
nlwarning("isMouseButtonReleased");
_CurrentTool->onMouseLeftButtonUp();
}
if(EventsListener.isMouseButtonPushed(rightButton))
{
nlwarning("isMouseButtonPushed");
_CurrentTool->onMouseRightButtonDown();
}
if(EventsListener.isMouseButtonReleased(rightButton))
{
nlwarning("onMouseRightButtonUp");
_CurrentTool->onMouseRightButtonUp();
}
}*/
// *********************************************************************************************************
CEntityCL *CEditor::createEntity(uint slot, const NLMISC::CSheetId &sheetId, const NLMISC::CVector &pos, float heading, const std::string & permanentStatutIcon)
{
//H_AUTO(R2_CEditor_createEntity)
CHECK_EDITOR
if (sheetId == NLMISC::CSheetId::Unknown) return NULL;
CInterfaceManager *im = CInterfaceManager::getInstance();
if (EntitiesMngr.entity(slot))
{
EntitiesMngr.remove(slot, false);
}
// Create the temporary entity in the entity manager
TNewEntityInfo emptyEntityInfo;
emptyEntityInfo.reset();
CEntityCL *entity = EntitiesMngr.create(slot, sheetId.asInt(), emptyEntityInfo);
if (!entity)
{
nlwarning("Can't create entity");
return NULL;
}
// Set the permanent statut icon
entity->setPermanentStatutIcon(permanentStatutIcon);
// TMP TMP : code taken from /entity command
sint64 *prop = 0;
CCDBNodeLeaf *node = 0;
// Set The property 'CLFECOMMON::PROPERTY_POSITION'.
node = im->getDbProp("SERVER:Entities:E" + NLMISC::toString("%d", slot)+":P" + NLMISC::toString("%d", CLFECOMMON::PROPERTY_POSX), false);
if(node)
{
sint64 x = (sint64)(pos.x*1000.0);
sint64 y = (sint64)(pos.y*1000.0);
sint64 z = (sint64)(pos.z*1000.0);
node->setValue64(x);
node = im->getDbProp("SERVER:Entities:E"+toString("%d", slot)+":P"+toString("%d", CLFECOMMON::PROPERTY_POSY), false);
if(node)
{
node->setValue64(y);
node = im->getDbProp("SERVER:Entities:E"+toString("%d", slot)+":P"+toString("%d", CLFECOMMON::PROPERTY_POSZ), false);
if(node)
node->setValue64(z);
}
}
// Set The property 'PROPERTY_ORIENTATION'.
node = im->getDbProp("SERVER:Entities:E"+toString("%d", slot)+":P"+toString("%d", CLFECOMMON::PROPERTY_ORIENTATION), false);
if(node)
{
union
{
uint64 heading64;
float headingFloat;
};
headingFloat = heading;
node->setValue64(heading64);
}
// Set Mode
node = im->getDbProp("SERVER:Entities:E"+toString("%d", slot)+":P"+toString("%d", CLFECOMMON::PROPERTY_MODE), false);
if(node)
{
MBEHAV::EMode m = MBEHAV::NORMAL;
prop = (sint64 *)&m;
node->setValue64(*prop);
EntitiesMngr.updateVisualProperty(0, slot, CLFECOMMON::PROPERTY_MODE);
}
// Set Visual Properties
SPropVisualA visualA;
//visualA.PropertySubData.LTrail = 1;
// Set alternate look
SAltLookProp altLookProp;
// fill infos for look
prop = (sint64 *)&visualA;
if(dynamic_cast<CPlayerCL *>(entity))
{
// visual property A depends on the type of the entity
visualA.PropertySubData.Sex = ClientCfg.Sex;
}
else if(dynamic_cast<CPlayerR2CL *>(entity) == NULL)
{
// Get the database entry.
// Get the old value (not useful since we change the whole property).
altLookProp.Summary = 0;
altLookProp.Element.ColorTop = 0;
altLookProp.Element.ColorBot = 2;
altLookProp.Element.WeaponRightHand = 0;
altLookProp.Element.WeaponLeftHand = 0;
altLookProp.Element.Seed = 100;
altLookProp.Element.ColorHair = 4;
altLookProp.Element.Hat = 0;
// old colors
altLookProp.Element.ColorGlove = altLookProp.Element.ColorTop;
altLookProp.Element.ColorArm = altLookProp.Element.ColorTop;
altLookProp.Element.ColorBoot = altLookProp.Element.ColorBot;
// fill alt infos for look
prop = (sint64 *)&altLookProp.Summary;
}
// Set the database.
im->getDbProp("SERVER:Entities:E"+toString("%d", slot)+":P"+toString("%d", CLFECOMMON::PROPERTY_VPA))->setValue64(*prop);
// Set Visual Properties
SPropVisualB visualB;
visualB.PropertySubData.LTrail = 1;
// fill infos for look
sint64 *propB = 0;
propB = (sint64 *)&visualB;
// Set the database.
im->getDbProp("SERVER:Entities:E"+toString("%d", slot)+":P"+toString("%d", CLFECOMMON::PROPERTY_VPB))->setValue64(*propB);
// Apply Changes.
EntitiesMngr.updateVisualProperty(0, slot, CLFECOMMON::PROPERTY_VPA);
EntitiesMngr.updateVisualProperty(0, slot, CLFECOMMON::PROPERTY_VPB);
return entity;
}
// *********************************************************************************************************
CInstance *CEditor::getInstanceFromEntity(CEntityCL *entity) const
{
//H_AUTO(R2_CEditor_getInstanceFromEntity)
CHECK_EDITOR
for(TInstanceMap::const_iterator it = _Instances.begin(); it != _Instances.end(); ++it)
{
if (it->second->getEntity() == entity) return it->second;
}
return NULL;
}
// *********************************************************************************************************
void CEditor::displayContextMenu()
{
//H_AUTO(R2_CEditor_displayContextMenu)
CHECK_EDITOR
if (!_SelectedInstance)
{
// launch standard context menu (for free look & move options)
getUI().launchContextMenuInGame("ui:interface:game_context_menu_edition");
return;
}
if (getCurrentTool() && getCurrentTool()->isCreationTool())
{
setCurrentTool(NULL);
}
// retrieve the context menu depending on the type of the entity
CLuaObject classDesc = _SelectedInstance->getClass();
if (classDesc.isNil())
{
nlwarning("Can't retrieve object class");
return;
}
std::string menu = classDesc["Menu"];
if (menu.empty()) return; // no menu for that instance ?
//
CLuaObject menuSetupFunction = classDesc["onSetupMenu"];
if (menuSetupFunction.isFunction())
{
_SelectedInstance->getLuaProjection().push();
menuSetupFunction.callNoThrow(1, 0);
}
//
getUI().launchContextMenuInGame(menu);
}
// *********************************************************************************************************
void CEditor::triggerInstanceObserver(const TInstanceId &id, IObserverAction &action)
{
//H_AUTO(R2_CEditor_triggerInstanceObserver)
TInstanceObserverMap::iterator lb = _InstanceObservers.lower_bound(id);
TInstanceObserverMap::iterator ub = _InstanceObservers.upper_bound(id);
if (lb == ub) return;
// must do a copy, because an observer may erase himself from the list when it is triggered
static std::vector<IInstanceObserver::TRefPtr> dest;
dest.clear();
for (TInstanceObserverMap::iterator it = lb; it != ub; ++it)
{
dest.push_back(it->second);
}
for (std::vector<IInstanceObserver::TRefPtr>::iterator it = dest.begin(); it != dest.end(); ++it)
{
if (*it) action.doAction(**it);
}
}
// *********************************************************************************************************
void CEditor::onErase(CObject *object)
{
//H_AUTO(R2_CEditor_onErase)
bool dummyFoundInBase;
std::string dummyNameInParent;
onErase(object, dummyFoundInBase, dummyNameInParent);
}
// *********************************************************************************************************
void CEditor::onErase(CObject *root, bool &foundInBase, std::string &nameInParent)
{
//H_AUTO(R2_CEditor_onErase)
foundInBase = false;
CHECK_EDITOR
if (!root) return;
CInstance *inst = getInstanceFromObject(root);
// sons
if (root->isTable())
{
for(uint k = 0; k < root->getSize(); ++k)
{
CObject *obj = root->getValue(k);
if (obj->isTable())
{
onErase(obj);
}
}
}
//
CObject *parent = NULL;
//
if (inst)
{
// Add an 'Erased' flag to the object so that
// any pending property in the property sheet
// won't send a 'requestSetNode' when it is closed
try
{
(*inst).getLuaProjection()["User"].setValue("Erased", true);
}
catch (ELuaNotATable &e)
{
nlwarning(e.what());
}
// if object is selected or focused, then clear these flags
if (inst == getSelectedInstance())
{
setSelectedInstance(NULL);
}
if (inst == getFocusedInstance())
{
setFocusedInstance(NULL);
}
}
// if object can be found in its base, then not really a deletion, but
// rather a change of attribute to the "default value' (may happen in an undo operation)
parent = root->getParent();
if (parent)
{
sint32 sonIndex = parent->findIndex(root);
if (sonIndex != -1)
{
nameInParent = parent->getKey(sonIndex);
if (!nameInParent.empty() && getDMC().getPropertyAccessor().hasValueInBase(parent, nameInParent))
{
foundInBase = true;
}
}
}
if (inst && !foundInBase)
{
// send event to instance & displayers
inst->onErase();
// trigger observers
class CEraseNotification : public IObserverAction
{
public:
CEraseNotification(CInstance &instance) : Instance(instance) {}
virtual void doAction(IInstanceObserver &obs)
{
obs.onInstanceErased(Instance);
}
CInstance &Instance;
};
CEraseNotification eraseNotification(*inst);
triggerInstanceObserver(inst->getId(), eraseNotification);
}
// if object has a user environment attached to it, remove it from lua
if (inst)
{
CLuaStackChecker lsc(&getLua());
getLua().pushLightUserData((void *) root);
getLua().getTable(LUA_REGISTRYINDEX);
if (!getLua().isNil())
{
getLua().pushLightUserData((void *) root); // key
getLua().pushNil(); // value
getLua().setTable(LUA_REGISTRYINDEX); // erase
}
getLua().pop();
}
if (root->isTable())
{
// special patch: ref ids under this object should not be triggered any more for that object
CObjectTable *rootTable = root->toTable();
for (uint32 k = 0; k < rootTable->getSize(); ++k)
{
CObject *obj = rootTable->getValue(k);
CObjectRefIdClient *objRefId = dynamic_cast<CObjectRefIdClient *>(obj);
if (objRefId)
{
objRefId->enable(false); // don't observe anything
}
}
}
if (!_ClearingContent)
{
if (inst)
{
nlassert(_Instances.count((const CObjectTable *) root) == 1);
//nlwarning("Instance with id %s deleted, but not inserted", inst->getId().c_str());
}
}
// really remove object
//nlwarning("Removing instance with id %s (table = 0x%s)", inst->getId().c_str(), (int) root);
if (inst)
{
_Instances.erase((const CObjectTable *) root);
}
}
// *********************************************************************************************************
void CEditor::nodeErased(const std::string& instanceId, const std::string& attrName, sint32 position)
{
//H_AUTO(R2_CEditor_nodeErased)
CHECK_EDITOR
CObject *obj = _DMC->find(instanceId, attrName, position);
if (!obj) return;
bool foundInBase;
std::string nameInParent;
onErase(obj, foundInBase, nameInParent);
CObject *parent = obj->getParent();
_DMC->CDynamicMapClient::nodeErased(instanceId, attrName, position);
if (foundInBase)
{
nlassert(parent);
// erased, but reading in the base will give a new value, so the real
// action from observers standpoint is 'modified'
CInstance *parentInstance = getInstanceFromObject(parent);
if (parentInstance)
{
onAttrModified(*parentInstance, nameInParent);
}
else
{
nlwarning("Can't found instance in which %s was modified", nameInParent.c_str());
}
}
// warn the parent that it has been modified
onAttrModified(parent);
// NB : msg for deleted attribute does not exist yet, so the parent is warned that it is modified, but
// 'onTableModified' is not called (because key doesn't exist any more)
}
void CEditor::onResetEditionMode()
{
//H_AUTO(R2_CEditor_onResetEditionMode)
CHECK_EDITOR
// called when a scenario just before a scenario is created
// no-op, to clean
}
void CEditor::onEditionModeConnected( uint32 userSlotId, uint32 adventureId, CObject* highLevel, const std::string& versionName, bool willTP, uint32 initialActIndex)
{
//H_AUTO(R2_CEditor_onEditionModeConnected)
CHECK_EDITOR
if (!_WaitScenarioScreenActive)
{
setMode(EditionMode);
}
CInterfaceGroup *currentSessionGroup = dynamic_cast<CInterfaceGroup *>(getUI().getElementFromId("ui:interface:r2ed_current_session"));
if (currentSessionGroup)
{
CViewText *text = dynamic_cast<CViewText *>(currentSessionGroup->getView("current_session"));
if (text)
{
text->setText(toString("Edition Session = %d (%s)", adventureId, versionName.c_str()));
}
}
setDefaultChatWindow(PeopleInterraction.ChatGroup.Window);
_DMC->CDynamicMapClient::onEditionModeConnected(userSlotId, adventureId, highLevel, versionName, willTP, initialActIndex);
}
void CEditor::setAccessMode(TAccessMode mode)
{
//H_AUTO(R2_CEditor_setAccessMode)
_AccessMode = mode;
switch ( mode)
{
case AccessEditor: _Env.setValue("AccessMode", "Editor"); break;
case AccessDM:
_Env.setValue("AccessMode", "DM");
if (_Mode == AnimationModePlay)
{
loadStandardUI();
}
break;
case AccessOutlandOwner: _Env.setValue("AccessMode", "OutlandOwner"); break;
default:
nlassert(0);
break;
}
}
void CEditor::onAnimationModeConnected(const CClientMessageAdventureUserConnection& connected)
{
//H_AUTO(R2_CEditor_onAnimationModeConnected)
_ScenarioReceivedFlag = true; // end wait screen
// TMP PATCH
// _WillTP = true;
CHECK_EDITOR
switch(connected.Mode)
{
case 0: setMode(AnimationModeLoading); break;
case 1: setMode(AnimationModeWaitingForLoading); break;
case 2: setMode(AnimationModeDm); break;
case 3: setMode(AnimationModePlay); break;
default: nlwarning("Unhandled %u in Animation Session", connected.Mode);
}
CInterfaceGroup *currentSessionGroup = dynamic_cast<CInterfaceGroup *>(getUI().getElementFromId("ui:interface:r2ed_current_session"));
if (currentSessionGroup)
{
CViewText *text = dynamic_cast<CViewText *>(currentSessionGroup->getView("current_session"));
if (text)
{
text->setText(toString("Animation Session = %d (%s)", connected.SessionId.asInt(), connected.VersionName.c_str()));
}
}
setDefaultChatWindow(PeopleInterraction.ChatGroup.Window);
_DMC->CDynamicMapClient::onAnimationModeConnected(connected);
}
void CEditor::onEditionModeDisconnected()
{
//H_AUTO(R2_CEditor_onEditionModeDisconnected)
_EditionModeDisconnectedFlag = true;
delete _NewScenario;
_NewScenario = NULL;
CHECK_EDITOR
// Usefull only for the pionner that does not do requestTranslateFeatures()
// Because avec using the button the currentScenario = 0
try
{
R2::getEditor().getLua().executeScript("r2.Version.save(\"data/r2_buffer.dat\")");
}
catch (const std::exception& e)
{
nlwarning("Can't start Edition Mode", e.what());
}
_DMC->CDynamicMapClient::onEditionModeDisconnected();
}
void CEditor::onTestModeConnected()
{
//H_AUTO(R2_CEditor_onTestModeConnected)
CHECK_EDITOR
// TODO nico : change the name of the function : should rather be 'onAnimationModeConnected'
// start as a GM
getUI().runActionHandler("r2ed_anim_dm_mode", NULL, "");
_DMC->CDynamicMapClient::onTestModeConnected();
}
void CEditor::onTestModeDisconnected(TSessionId sessionId, uint32 lastAct, TScenarioSessionType sessionType)
{
//H_AUTO(R2_CEditor_onTestModeDisconnected)
CHECK_EDITOR
_DMC->CDynamicMapClient::onTestModeDisconnected(sessionId, lastAct, sessionType);
}
// *********************************************************************************************************
void CEditor::nodeInserted(const std::string& instanceId, const std::string& attrName, sint32 position, const std::string& key, CObject* value)
{
//H_AUTO(R2_CEditor_nodeInserted)
CHECK_EDITOR
_DMC->CDynamicMapClient::nodeInserted(instanceId, attrName, position, key, value);
if (value->isTable())
{
std::string id = getString(value, "InstanceId");
if (!id.empty())
{
const CObject *clonedValue = _DMC->find(id);
// bad insert
if (clonedValue)
{
createNewInstanceForObjectTable(clonedValue);
onAttrModified(clonedValue->getParent()); // parent table is modified
// so -> only parent of parent table will be notified of change
// maybe we need another msg "onInsert" that is sent
// to the parent table.
}
}
}
}
// *********************************************************************************************************
void CEditor::createNewInstanceForObjectTable(const CObject *obj)
{
//H_AUTO(R2_CEditor_createNewInstanceForObjectTable)
CHECK_EDITOR
createNewInstanceForObjectTableInternal(obj);
onPostCreate(obj);
}
// *********************************************************************************************************
void CEditor::notifyInstanceObserversOfCreation(CInstance &inst)
{
//H_AUTO(R2_CEditor_notifyInstanceObserversOfCreation)
CHECK_EDITOR
// if there are observers watching this instance, warn them
const std::string &id = inst.getId();
// trigger observers
class CCreationNotification : public IObserverAction
{
public:
CCreationNotification(CInstance &instance) : Instance(instance) {}
virtual void doAction(IInstanceObserver &obs)
{
obs.onInstanceCreated(Instance);
}
CInstance &Instance;
};
CCreationNotification creationNotification(inst);
triggerInstanceObserver(id, creationNotification);
}
// *********************************************************************************************************
void CEditor::onPostCreate(const CObject *obj)
{
//H_AUTO(R2_CEditor_onPostCreate)
CHECK_EDITOR
struct CPostCreateVisitor : public IObjectVisitor
{
virtual void visit(CObjectTable &obj)
{
// if table has a "InstanceId" field there is a matching editor object
CInstance *inst = getEditor().getInstanceFromObject(&obj);
if (inst)
{
inst->onPostCreate();
getEditor().notifyInstanceObserversOfCreation(*inst);
}
}
};
if (obj)
{
CPostCreateVisitor postCreateVisitor;
const_cast<CObject *>(obj)->visit(postCreateVisitor);
}
}
// *********************************************************************************************************
sint CEditor::getLeftQuota()
{
//H_AUTO(R2_CEditor_getLeftQuota)
CLuaState &ls = getLua();
CLuaStackChecker lsc(&ls);
callEnvMethod("getLeftQuota", 0, 1);
if (!ls.isNumber(-1))
{
ls.pop(1);
return 0;
}
sint result = (sint) ls.toNumber(-1);
ls.pop(1);
return result;
}
// *********************************************************************************************************
bool CEditor::checkRoomLeft()
{
//H_AUTO(R2_CEditor_checkRoomLeft)
return getLeftQuota() > 0;
}
// *********************************************************************************************************
void CEditor::makeRoomMsg()
{
//H_AUTO(R2_CEditor_makeRoomMsg)
// delegate ui display to lua
callEnvMethod("makeRoomMsg", 0, 0);
}
// *********************************************************************************************************
bool CEditor::verifyRoomLeft(uint aiCost, uint staticCost)
{
//H_AUTO(R2_CEditor_verifyRoomLeft)
CLuaState &ls = getLua();
if (aiCost)
{
CLuaStackChecker lsc(&ls);
getEditor().getLua().push((lua_Number)aiCost);
callEnvMethod("checkAiQuota", 1, 1);
if (!ls.isBoolean(-1))
{
ls.pop(1);
return false;
}
sint result = (sint) ls.toBoolean(-1);
ls.pop(1);
return result != 0;
}
if (staticCost)
{
CLuaStackChecker lsc(&ls);
getEditor().getLua().push((lua_Number)staticCost);
callEnvMethod("checkStaticQuota", 1, 1);
if (!ls.isBoolean(-1))
{
ls.pop(1);
return false;
}
sint result = (sint) ls.toBoolean(-1);
ls.pop(1);
return result != 0;
}
return true;
}
// *********************************************************************************************************
void CEditor::createNewInstanceForObjectTableInternal(const CObject *obj)
{
//H_AUTO(R2_CEditor_createNewInstanceForObjectTableInternal)
CHECK_EDITOR
if (!obj) return;
if (!obj->isTable()) return; // not a table ...
const CObjectTable *table = (const CObjectTable *) obj;
std::string id = getString(obj, "InstanceId");
if (!id.empty())
{
// CInstance is created only for objects with an instance id
CInstance *inst = new CInstance(table, getLua());
nlassert(_Instances.count(table) == 0);
_Instances[table] = inst;
// if a cookie was created, add in in the instance lua 'User' table
TCookieMap::iterator cookieList = _Cookies.find(id);
if (cookieList != _Cookies.end())
{
if (obj->isTable())
{
CLuaState &ls = getLua();
getLuaUserTableFromObject(ls, * (CObjectTable *) obj);
CLuaObject userTable(ls); // pop the table into a CLuaObject for convenience
for (TCookieList::iterator it = cookieList->second.begin(); it != cookieList->second.end(); ++it)
{
userTable.setValue(it->Key, it->Value);
}
}
_Cookies.erase(cookieList);
}
// create displayers
CLuaObject classDesc = inst->getClass();
if (!classDesc.isNil())
{
CLuaStackChecker lsc(&getLua());
CDisplayerVisual *dispViz= createObjectFromClassName<CDisplayerVisual>(classDesc["DisplayerVisual"].toString());
if (dispViz)
{
bool ok = dispViz->init(classDesc["DisplayerVisualParams"]);
if (!ok)
{
nlwarning("Error when calling init on visual displayer of class %s", classDesc["Name"].toString().c_str());
}
}
inst->setDisplayerVisual(dispViz);
//
CDisplayerBase *dispUI = createObjectFromClassName<CDisplayerBase>(classDesc["DisplayerUI"].toString());
if (dispUI)
{
bool ok = dispUI->init(classDesc["DisplayerUIParams"]);
if (!ok)
{
nlwarning("Error when calling init on ui displayer of class %s", classDesc["Name"].toString().c_str());
}
}
inst->setDisplayerUI(dispUI);
//
CDisplayerBase *dispProp = createObjectFromClassName<CDisplayerBase>(classDesc["DisplayerProperties"].toString());
if (dispProp)
{
bool ok = dispProp->init(classDesc["DisplayerPropertiesParams"]);
if (!ok)
{
nlwarning("Error when calling init on property displayer of class %s", classDesc["Name"].toString().c_str());
}
}
inst->setDisplayerProperties(dispProp);
}
// prevent completion of the tree while an instance is being created
//backupRequestCommands();
inst->onCreate();
//restoreRequestCommands();
//
// if a name was generated locally, erase from the local name map.
std::string className = getString(obj, "Class");
std::string name = getString(obj, "Name");
if (!className.empty() && !name.empty())
{
for (TGeneratedNameMap::iterator it = _LocalGeneratedNames.begin(); it != _LocalGeneratedNames.end(); ++it)
{
sint index = getGeneratedNameIndex(name, it->first);
if (index != -1)
{
it->second.erase(index);
break;
}
}
}
}
// do the same on sons
for(uint k = 0; k < table->getSize(); ++k)
{
createNewInstanceForObjectTableInternal(table->getValue(k));
}
}
// *********************************************************************************************************
void CEditor::onAttrModified(CInstance &parentInstance, const std::string &attrName, sint32 indexInArray)
{
//H_AUTO(R2_CEditor_onAttrModified)
parentInstance.onAttrModified(attrName, indexInArray);
class CAttrModifiedNotification : public IObserverAction
{
public:
CAttrModifiedNotification(CInstance &instance, const std::string &key, sint32 indexInArray)
: Instance(instance), Key(key), IndexInArray(indexInArray) {}
virtual void doAction(IInstanceObserver &obs)
{
obs.onAttrModified(Instance, Key, IndexInArray);
}
CInstance &Instance;
const std::string &Key;
sint32 IndexInArray;
};
CAttrModifiedNotification attrModifiedNotification(parentInstance, attrName, indexInArray);
triggerInstanceObserver(parentInstance.getId(), attrModifiedNotification);
}
// *********************************************************************************************************
void CEditor::onAttrModified(const CObject *value)
{
//H_AUTO(R2_CEditor_onAttrModified)
CHECK_EDITOR
if (!value) return;
const CObject *son = value;
const CObject *parent = value->getParent();
sint32 indexInArray = -1;
while (parent)
{
CInstance *parentInstance = getInstanceFromObject(parent);
sint32 indexInParent = parent->findIndex(son);
nlassert(indexInParent != -1);
if (parentInstance)
{
// we are in an instance (a CObjectTable with an instance id)
// TODO nico : a cache for the 'name' in the parent like with CObjectRefId ...
std::string key = parent->getKey(indexInParent);
onAttrModified(*parentInstance, key, indexInArray);
indexInArray = -1;
}
else
{
// we are in an array in an instance -> memorize index in that array for next call to "onAttrModified"...
indexInArray = indexInParent;
}
son = parent;
parent = parent->getParent();
}
}
// *********************************************************************************************************
void CEditor::nodeSet(const std::string& instanceId, const std::string& attrName, CObject* value)
{
//H_AUTO(R2_CEditor_nodeSet)
CHECK_EDITOR
CObject *obj = _DMC->find(instanceId);
if (!obj) return;
// erase previous object
nlassert(obj->isTable());
//
if (!attrName.empty())
{
obj = obj->findAttr(attrName);
}
if (obj)
{
onErase(obj);
}
// change the actual value
_DMC->CDynamicMapClient::nodeSet(instanceId, attrName, value);
//
nlassert(!getInstanceFromObject(value)); // this must be a new object...
//
if (value->isTable())
{
std::string id = getString(value, "InstanceId");
if (!id.empty())
{
const CObject *clonedValue = _DMC->find(id);
if (clonedValue)
{
createNewInstanceForObjectTable(clonedValue); // if the created object is a table, warn him that he has been created
onAttrModified(clonedValue);
return;
}
}
}
// no instance id for object, retrieve object pointer from parent
const CObject *parent = _DMC->find(instanceId);
if (!parent) return;
onAttrModified(parent->getAttr(attrName));
}
// *********************************************************************************************************
void CEditor::eraseScenario()
{
//H_AUTO(R2_CEditor_eraseScenario)
if (_Scenario)
{
backupRequestCommands();
onErase(_Scenario);
restoreRequestCommands();
_Scenario = NULL;
}
}
// *********************************************************************************************************
void CEditor::scenarioUpdated(CObject* highLevel, bool willTP, uint32 initialActIndex)
{
//H_AUTO(R2_CEditor_scenarioUpdated)
CHECK_EDITOR
_ScenarioReceivedFlag = true;
if (_WaitScenarioScreenActive)
{
// defer scenario update to the end of the wait screen
_NewScenario = highLevel ? highLevel->clone() : NULL;
_NewScenarioInitialAct = initialActIndex;
_PostponeScenarioUpdated = true;
return;
}
if (!highLevel)
{
_IsStartingScenario = false;
_WillTP = false;
callEnvFunc("onEmptyScenarioUpdated", 0);
return;
}
// _WillTP = willTP;
_WillTP = false; // TMP TMP
_UpdatingScenario = true;
_BaseAct = NULL;
_CurrentAct = NULL;
CLuaStackRestorer lsc(&getLua(), getLua().getTop());
//nlwarning("Scenario updated, start highlevel = ");
//highLevel->dump();
eraseScenario();
_IsStartingScenario = false;
if (_DMC->getEditionModule().getMustStartScenario())
{
_IsStartingScenario = true;
}
nlassert(_Instances.empty());
nlassert(CDisplayerBase::ObjCount == 0);
//
_DMC->CDynamicMapClient::scenarioUpdated(highLevel, willTP, initialActIndex);
//nlwarning("Scenario updated, content is = ");
/*
if (_DMC->getHighLevel())
{
_DMC->getHighLevel()->dump();
}
else
{
nlwarning("NULL");
}
*/
createNewInstanceForObjectTableInternal(_DMC->getHighLevel());
nlassert(highLevel->isTable());
_Scenario = (CObjectTable *) _DMC->getHighLevel();
if (_Scenario->getAttr("InstanceId"))
{
_ScenarioInstance = getInstanceFromId(_Scenario->getAttr("InstanceId")->toString());
}
else
{
_ScenarioInstance = NULL;
nlwarning("Can't retrieve scenario (no instance id)");
_Scenario->dump();
}
static volatile bool forceDump = false;
if (forceDump)
{
_Scenario->dump();
}
//
projectInLua(_Scenario); // push on the lua stack
getLua().push(float(initialActIndex)); // example reconnect after test in act4
// update value in the framework
callEnvFunc("onScenarioUpdated", 2);
//nlwarning("Instance list now is :");
//dumpInstances();
CObject *acts = _Scenario->getAttr("Acts");
if (acts)
{
CObject *baseAct = acts->getValue(0);
if (baseAct)
{
_BaseAct = getInstanceFromId(baseAct->toString("InstanceId"));
}
}
if (!_BaseAct)
{
nlwarning("Base act not found at scenario update");
}
if (_WantedActOnInit.empty())
{
if (!_CurrentAct) // if act not currently setted at scenario creation ...
{
setCurrentAct(_BaseAct); // ...then default to the base act
}
}
else
{
setCurrentActFromTitle(_WantedActOnInit);
_WantedActOnInit.clear();
}
_DMC->getActionHistoric().clear(highLevel); // reinit the undo / redo stack
onPostCreate(_Scenario); // post creation require that current act has been set
//TP is done via the onTpPositionSimulated message
/*
// teleport in good island
if (ClientCfg.Local)
{
sint locationId = (uint) _ScenarioInstance->getLuaProjection()["Description"]["LocationId"].toNumber();
CScenarioEntryPoints &sep = CScenarioEntryPoints::getInstance();
_IslandCollision.loadEntryPoints();
if (sep.getCompleteIslands().empty())
{
nlwarning("Entry points not loaded, teleport not done (local mode)");
}
else if (locationId >= (sint) sep.getCompleteIslands().size())
{
nlwarning("Bad location id %d", locationId);
}
else
{
// check if already in this entry point (no tp if so)
const CScenarioEntryPoints::CCompleteIsland &ci = sep.getCompleteIslands()[locationId];
CVectorD playerPos = UserEntity->pos();
//
if (playerPos.x <= ci.XMin ||
playerPos.x >= ci.XMax ||
playerPos.y <= ci.YMin ||
playerPos.y >= ci.YMax)
{
if(ci.EntryPoints.size()>0)
{
const CScenarioEntryPoints::CShortEntryPoint & shortEntryPoint = ci.EntryPoints[0];
CVector dest((float) shortEntryPoint.X, (float) shortEntryPoint.Y, 0.f);
UserEntity->pos(dest); // change position in pacs
// Select the closest continent from the new position.
beginLoading (LoadingBackground);
#define BAR_STEP_TP 2 // fixme : this define is duplicated....
ProgressBar.reset (BAR_STEP_TP);
ucstring nmsg("Loading...");
ProgressBar.newMessage ( ClientCfg.buildLoadingString(nmsg) );
ProgressBar.progress(0);
ContinentMngr.select(dest, ProgressBar);
endLoading();
// Teleport the User.
UserEntity->tp(dest);
}
}
}
}
*/
_UpdatingScenario = false;
if (_DMC->getEditionModule().getMustStartScenario()
&& _DMC->getEditionModule().getScenarioUpToDate())
{
_DMC->getEditionModule().setMustStartScenario( false);
if ( _DMC->getCurrentScenario()->getHighLevel())
{
bool result = true;
if (ClientCfg.R2EDMustVerifyRingAccessWhileLoadingAnimation)
{
CLuaState &ls = getLua();
CLuaStackChecker lsc(&ls);
callEnvFunc( "verifyScenario", 0, 1);
if (!ls.isBoolean(-1))
{
nlassert(0 && "verifyScenario return wrong type");
}
result = ls.toBoolean(-1);
ls.pop();
}
if (result) //Start scenario only if allowed
{
ConnectionWanted = true; // ugly
}
}
}
}
// *********************************************************************************************************
CInstance *CEditor::getDefaultFeature(CInstance *act)
{
//H_AUTO(R2_CEditor_getDefaultFeature)
CHECK_EDITOR
if (!act) return NULL;
CObject *defaultFeature = act->getObjectTable()->getAttr("Features");
if (!defaultFeature) return NULL;
defaultFeature = defaultFeature->getValue(0);
if (!defaultFeature) return NULL; // 0 should be the default feature
CInstance *result = getInstanceFromId(defaultFeature->toString("InstanceId"));
if (!result) return NULL;
if (!result->isKindOf("DefaultFeature"))
{
nlwarning("Can't retrieve default feature.");
}
return result;
}
// *********************************************************************************************************
CInstance *CEditor::getDefaultFeature()
{
//H_AUTO(R2_CEditor_getDefaultFeature)
CHECK_EDITOR
return getDefaultFeature(getCurrentAct());
}
// *********************************************************************************************************
void CEditor::nodeMoved(const std::string& instanceId, const std::string& attrName, sint32 position, const std::string& destInstanceId, const std::string& destAttrName, sint32 destPosition)
{
//H_AUTO(R2_CEditor_nodeMoved)
CHECK_EDITOR
const CObject *src = _DMC->find(instanceId, attrName, position);
if (!src) return;
//
CObject *oldParent = src->getParent();
//
CInstance *inst = getInstanceFromObject(src);
// tells object that he is about to move
if (inst)
{
inst->onPreHrcMove();
// notify possible observers that this instance will move
class CPreHrcMoveNotification : public IObserverAction
{
public:
CPreHrcMoveNotification(CInstance &instance) : Instance(instance) {}
virtual void doAction(IInstanceObserver &obs)
{
obs.onPreHrcMove(Instance);
}
CInstance &Instance;
};
CPreHrcMoveNotification preHrcMoveNotification(*inst);
triggerInstanceObserver(inst->getId(), preHrcMoveNotification);
}
// do the move
_DMC->CDynamicMapClient::nodeMoved(instanceId, attrName, position, destInstanceId, destAttrName, destPosition);
// warn the previous parent that it has been modified
if (src->getParent() != oldParent)
{
onAttrModified(oldParent); // the old parent is modified so send appropriate msg
}
// else ... if new parent is the same then send message only at the end (else object believe it has been modified twice in a row)
// tells object that he has moved
if (inst)
{
inst->onPostHrcMove();
// notify possible observers that this instance has moved
class CPostHrcMoveNotification : public IObserverAction
{
public:
CPostHrcMoveNotification(CInstance &instance) : Instance(instance) {}
virtual void doAction(IInstanceObserver &obs)
{
obs.onPostHrcMove(Instance);
}
CInstance &Instance;
};
CPostHrcMoveNotification postHrcMoveNotification(*inst);
triggerInstanceObserver(inst->getId(), postHrcMoveNotification);
}
onAttrModified(src); // the new parent is modified so send appropriate msg
}
// *********************************************************************************************************
void CEditor::dumpInstances()
{
//H_AUTO(R2_CEditor_dumpInstances)
CHECK_EDITOR
for(TInstanceMap::iterator it = _Instances.begin(); it != _Instances.end(); ++it)
{
nlwarning("Obj = %p, id = %s, instance = %p", it->first, it->second->getId().c_str(), &(*it->second));
}
}
// ***************************************************************
struct CSortSelectableObject
{
ISelectableObject *Object;
float Depth;
CVector RayStart, RayStartClipped, RayEndClipped;
NLMISC::CAABBox SelectBox;
bool SelectBoxInWorld;
bool operator<(const CSortSelectableObject &o) const
{
return Depth<o.Depth;
}
};
// special selectable object to detect that mouse is over user
class CSelectableUser : public ISelectableObject
{
public:
virtual bool isSelectable() const { return true; }
virtual bool getLastClip() const { return UserEntity->getLastClip(); }
virtual NLMISC::CAABBox getSelectBox() const
{
return UserEntity->localSelectBox();
}
virtual const NLMISC::CMatrix &getInvertedMatrix() const
{
static CMatrix invertedMatrix;
invertedMatrix = UserEntity->dirMatrix();
invertedMatrix.setPos(UserEntity->pos());
invertedMatrix.invert();
return invertedMatrix;
}
virtual float preciseIntersectionTest(const NLMISC::CVector &worldRayStart, const NLMISC::CVector &worldRayDir) const
{
if (!UserEntity) return FLT_MAX;
return CEditor::preciseEntityIntersectionTest(*UserEntity, worldRayStart, worldRayDir);
}
virtual CInstance *getInstanceInEditor() const { return NULL; }
};
// ***************************************************************
static CSelectableUser SelectableUser;
//
CInstance *CEditor::getInstanceUnderPos(float x, float y, float distSelection, bool &isPlayerUnderCursor)
{
static volatile bool ignore = false;
if (ignore)
{
return NULL;
}
//H_AUTO(R2_CEditor_getInstanceUnderPos)
CHECK_EDITOR
// TODO nico: this code was copied from CEntityManager::getEntityUnderPos
// then modified, so some factoring could be made ...
uint i;
// valid only if bbox still intersect
CInstance *precInstanceUnderPos= _LastInstanceUnderPos;
bool precInstanceUnderPosValid= false;
// reset result
isPlayerUnderCursor= false;
_LastInstanceUnderPos = NULL;
// build the ray
CMatrix camMatrix = MainCam.getMatrix();
NL3D::CFrustum camFrust = MainCam.getFrustum();
NL3D::CViewport viewport = Driver->getViewport();
// Get the Ray made by the mouse.
CTool::CWorldViewRay worldViewRay;
worldViewRay.OnMiniMap = false;
worldViewRay.Valid = true;
//
viewport.getRayWithPoint(x, y, worldViewRay.Origin, worldViewRay.Dir, camMatrix, camFrust);
worldViewRay.Dir.normalize();
worldViewRay.Right = camMatrix.getI().normed();
worldViewRay.Up = camMatrix.getK().normed();
// **** Get entities with box intersecting the ray.
static std::vector<ISelectableObject *> validObjects;
validObjects.clear();
static std::vector<CSortSelectableObject> intersectedObjects;
intersectedObjects.clear();
ISelectableObject *precSelectableObject = NULL;
validObjects.push_back(&SelectableUser); // add fake object for test with user entity
for(TInstanceMap::iterator it = _Instances.begin(); it != _Instances.end(); ++it)
{
CInstance *instance = it->second;
CDisplayerVisual *vd = instance->getDisplayerVisual();
if (!vd) continue;
if (!vd->isSelectable()) continue;
if (instance == getEditor().getSelectedInstance())
{
precSelectableObject = vd;
}
validObjects.push_back(vd);
}
// compute intersection of mouse with landscape / batiments for test with projected object like region & decals
CVector sceneInter;
CTool::TRayIntersectionType rayInterType = CTool::computeLandscapeRayIntersection(worldViewRay, sceneInter);
CSortSelectableObject selectObj;
ISelectableObject *borderSelected = NULL;
for(uint k = 0; k < validObjects.size(); ++k)
{
ISelectableObject *object = validObjects[k];
// if entity not visible, skip
if(object->getLastClip())
continue;
float depth = FLT_MAX;
ISelectableObject::TSelectionType selectionType = object->getSelectionType();
bool borderSelection = false;
switch(selectionType)
{
case ISelectableObject::GroundProjected:
if (rayInterType != CTool::NoIntersection)
{
CVector2f testPos(sceneInter.x, sceneInter.y);
borderSelection = object->isInProjectionBorder(testPos);
if (borderSelection || object->isInProjection(testPos))
{
depth = (worldViewRay.Origin - sceneInter).norm();
}
}
break;
case ISelectableObject::LocalSelectBox:
case ISelectableObject::WorldSelectBox:
{
const CMatrix &rayMatrix = (selectionType == ISelectableObject::LocalSelectBox) ? object->getInvertedMatrix()
: CMatrix::Identity;
selectObj.RayStart = rayMatrix * worldViewRay.Origin;
selectObj.RayStartClipped = selectObj.RayStart;
selectObj.RayEndClipped = rayMatrix * (worldViewRay.Origin + worldViewRay.Dir * distSelection);
// if intersect the bbox
selectObj.SelectBox = object->getSelectBox();
selectObj.SelectBoxInWorld = false;
if (selectObj.SelectBox.clipSegment(selectObj.RayStartClipped, selectObj.RayEndClipped))
{
depth = (selectObj.RayStartClipped - selectObj.RayStart).norm();
// is it the last entity under pos?
if(object->getInstanceInEditor() ==precInstanceUnderPos)
precInstanceUnderPosValid= true;
}
}
break;
default:
nlassert(0);
break;
}
if (depth != FLT_MAX)
{
selectObj.Object = object;
selectObj.Depth = depth;
if (borderSelection)
{
borderSelected = object;
}
// add this entity to the list of possible entities
intersectedObjects.push_back(selectObj);
}
}
// if no intersected entities, quit
if(intersectedObjects.empty())
return NULL;
// Compute startDistBox: nearest entity distance, but the user
float startDistBox;
if(intersectedObjects[0].Object == &SelectableUser)
{
// if the nearest entity is the user, set res
isPlayerUnderCursor= true;
// if only player intersected, return NULL!
if(intersectedObjects.size()==1)
return NULL;
// so take the second for startDistBox
startDistBox= intersectedObjects[1].Depth;
}
else
{
// ok, take it.
startDistBox= intersectedObjects[0].Depth;
}
/*static std::vector<ISelectableObject *> projectedObjects;
projectedObjects.clear();*/
// **** get best entity according to distance face-camera or box-ray if no face intersection
ISelectableObject *objectSelected= NULL;
float bestDistBox= FLT_MAX;
float bestDistZ= FLT_MAX;
for(i=0;i<intersectedObjects.size();i++)
{
ISelectableObject *object = intersectedObjects[i].Object;
// If this entity is the UserEntity, skip!!
if(object == &SelectableUser)
continue;
float distZ = 0.f;
bool preciseInterFound = false;
ISelectableObject::TSelectionType selectionType = object->getSelectionType();
switch(selectionType)
{
case ISelectableObject::GroundProjected:
{
// if current selection remains selected, keep it
bool keepSelection = objectSelected &&
objectSelected->getSelectionType() == ISelectableObject::GroundProjected &&
objectSelected == precSelectableObject;
if (!keepSelection)
{
preciseInterFound = true;
distZ = intersectedObjects[i].Depth;
}
}
break;
case ISelectableObject::LocalSelectBox:
case ISelectableObject::WorldSelectBox:
// if (!object->intersectionTest(bbox, pos, dir, dist2D, distZ, distSelection)) continue;
distZ = object->preciseIntersectionTest(worldViewRay.Origin, worldViewRay.Dir);
if (distZ != FLT_MAX)
{
preciseInterFound = true;
}
break;
default:
nlassert(0);
break;
}
// *** if intersect face, then take the best face-intersection, else use box-ray cost
// true face-col found?
if(preciseInterFound)
{
// yes, get the nearest
if(distZ<=bestDistZ)
{
bestDistBox= 0;
bestDistZ= distZ;
objectSelected= object;
/*if (selectionType == ISelectableObject::GroundProjected)
{
projectedObjects.push_back(object);
}*/
}
}
// else
else
{
// if a true face-intersection has not been found for others entities
if(bestDistZ==FLT_MAX)
{
// get the "distance to camera" contribution.
// NB: ray & select box are in the same space (local or world)
CVector c= selectObj.SelectBox.getCenter();
float distCamCost= intersectedObjects[i].Depth;
// get relative to the nearest intersected entity
distCamCost-= startDistBox;
// take the middle of the clipped segment. suppose that this middle is the "nearest ray point to center"
// This is false, but gives better results.
CVector m= (selectObj.RayStartClipped + selectObj.RayEndClipped) / 2;
// get the distance to center. NB: small entities are preferred since smaller mean lower cost
float outBBoxCost= (m - c).norm();
// the final cost is a weighted sum of the both. NB: distCamCost is in meter,
// and outBBBoxCost is meters. Hence ClientCfg.SelectionOutBBoxWeight is a factor
float boxCost= distCamCost + outBBoxCost * ClientCfg.SelectionOutBBoxWeight;
// take the lowest cost
if(boxCost<bestDistBox)
{
objectSelected= object;
bestDistBox= boxCost;
}
}
}
}
// If precise intersection not found
if(bestDistZ==FLT_MAX)
{
// if the last entity under pos is valid, prefer it among all other approximate ones
if(precInstanceUnderPos && precInstanceUnderPosValid)
objectSelected= precInstanceUnderPos->getDisplayerVisual();
}
if (objectSelected->getSelectionType() == ISelectableObject::GroundProjected)
{
if (borderSelected && borderSelected != objectSelected)
{
if (!objectSelected || !objectSelected->isInProjectionBorder(CVector2f(sceneInter.x, sceneInter.y)))
{
// when mouse over zone border, preffered over other zones
objectSelected = borderSelected;
}
}
// TODO nico: list not really needed here (comes from old code)
/*uint k;
for(k = 0; k < projectedObjects.size(); ++k)
{
if (projectedObjects[k] == objectSelected) break;
}
objectSelected = projectedObjects[(k + 1) % projectedObjects.size()];
*/
}
// return the best entity
_LastInstanceUnderPos = objectSelected ? objectSelected->getInstanceInEditor() : NULL;
return _LastInstanceUnderPos;
}// getEntityUnderPos //
// *********************************************************************************************************
float CEditor::preciseEntityIntersectionTest(CEntityCL &entity, const NLMISC::CVector &worldRayStart, const NLMISC::CVector &worldRayDir)
{
//H_AUTO(R2_CEditor_preciseEntityIntersectionTest)
CHECK_EDITOR
// if entity skeleton model was clipped, skip
NL3D::USkeleton *skeleton = entity.skeleton();
if(!ClientCfg.Light && skeleton && !skeleton->getLastClippedState())
return FLT_MAX;
H_AUTO(RZ_Client_GEUP_face_intersect)
// *** Try get face-intersection, result in distZ
// if the entity support fast and precise intersection (and if it succeeds)
// bool trueIntersectComputed= false;
float dist2D, distZ;
if(!ClientCfg.Light)
{
if(skeleton)
{
if(skeleton->supportFastIntersect() && skeleton->fastIntersect(worldRayStart, worldRayDir, dist2D, distZ, false))
{
if (dist2D == 0.f)
{
return distZ;
}
}
}
// get the intersection with the instance (bot object)
else if(!entity.instances().empty() && !entity.instances()[0].Current.empty())
{
NL3D::UInstance inst= entity.instances()[0].Current;
if(inst.supportFastIntersect())
{
if (inst.fastIntersect(worldRayStart, worldRayDir, dist2D, distZ, false))
{
if (dist2D == 0.f)
{
return distZ;
}
}
}
else
{
// TMP
// precise test was asked, but there's no better test than bbox, so return the dist for bbox middle
return (entity.pos().asVector() + entity.localSelectBox().getCenter() - worldRayStart).norm();
}
}
}
return FLT_MAX;
}
// *********************************************************************************************************
const NLMISC::CAABBox &CEditor::getLocalSelectBox(CEntityCL &entity) const
{
//H_AUTO(R2_CEditor_getLocalSelectBox)
CHECK_EDITOR
const TEntityCustomSelectBoxMap &boxMap = getEditor().getEntityCustomSelectBoxMap();
TEntityCustomSelectBoxMap::const_iterator it = boxMap.find(CSheetId(entity.sheetId()).toString());
if (it != boxMap.end())
{
if (it->second.Enabled)
{
return it->second.Box;
}
}
return entity.localSelectBox();
}
// *********************************************************************************************************
NLMISC::CAABBox CEditor::getSelectBox(CEntityCL &entity) const
{
//H_AUTO(R2_CEditor_getSelectBox)
CHECK_EDITOR
const TEntityCustomSelectBoxMap &boxMap = getEditor().getEntityCustomSelectBoxMap();
TEntityCustomSelectBoxMap::const_iterator it = boxMap.find(CSheetId(entity.sheetId()).toString());
if (it != boxMap.end())
{
if (it->second.Enabled)
{
// box is local, transform in world
CMatrix modelMatrix;
modelMatrix = entity.dirMatrix();
modelMatrix.setPos(entity.pos().asVector());
return CAABBox::transformAABBox(modelMatrix, it->second.Box);
}
}
return entity.selectBox();
}
// *********************************************************************************************************
void CEditor::connexionMsg(const std::string &stringId)
{
//H_AUTO(R2_CEditor_connexionMsg)
CHECK_EDITOR
getEditor()._ConnexionMsg = stringId;
// ignore if current ui desktop is not the third
if (getUI().getMode() != 3) return;
// show the connection window
CInterfaceGroup *r2ConnectWindow = dynamic_cast<CInterfaceGroup *>(getUI().getElementFromId("ui:interface:r2ed_connect"));
if (!r2ConnectWindow) return;
if (stringId.empty())
{
r2ConnectWindow->setActive(false);
return;
}
else
{
if (!r2ConnectWindow->getActive())
{
// center for the first display
r2ConnectWindow->setActive(true);
r2ConnectWindow->center();
}
CViewText *vt = dynamic_cast<CViewText *>(r2ConnectWindow->getView("connexionMsg"));
if (vt)
{
vt->setText(CI18N::get(stringId));
}
}
}
// *********************************************************************************************************
void CEditor::connect()
{
//H_AUTO(R2_CEditor_connect)
CHECK_EDITOR
if (_Mode == EditionMode || _Mode == AnimationModeLoading)
{
bool ok = false;
if (_ScenarioInstance)
{
try
{
if (_ScenarioInstance->getLuaProjection().callMethodByNameNoThrow("validateForTesting", 0, 1))
{
if (getLua().toBoolean(-1) == true)
{
R2::getEditor().getDMC().getEditionModule().requestStartScenario();
ok = true;
}
}
}
catch (const std::exception& )
{
// 'ok' still == false ...
}
}
if (!ok)
{
nlwarning("Can't go live");
}
}
else if (_Mode == TestMode || _Mode == DMMode)
{
try
{
R2::getEditor().getLua().executeScript("r2.requestStopLive()");
if (ClientCfg.Local)
{
R2::getEditor().setMode(CEditor::EditionMode);
getLua().executeScriptNoThrow("r2.requestReconnection()");
}
else
{
R2::getEditor().setMode(CEditor::GoingToEditionMode);
}
CEditor::connexionMsg("uimR2EDGoToEditingMode");
}
catch (const std::exception& e)
{
nlwarning("Can't go live: %s", e.what());
}
}
// TODO Nico : reset the good capture keyboard
setDefaultChatWindow(PeopleInterraction.ChatGroup.Window);
ConnectionWanted = false;
return;
}
// *********************************************************************************************************
CEditor::TInstanceObserverHandle CEditor::addInstanceObserver(const TInstanceId &instanceId, IInstanceObserver *observer)
{
//H_AUTO(R2_CEditor_addInstanceObserver)
CHECK_EDITOR
//nlwarning("#adding instance observer 0x%x", (int) observer);
nlassert(_InstanceObservers.size() == _InstanceObserverHandles.size());
const TInstanceObserverMap::const_iterator lb = _InstanceObservers.lower_bound(instanceId);
const TInstanceObserverMap::const_iterator ub = _InstanceObservers.upper_bound(instanceId);
// NB nico : removed the inserted twice stuff below because observer pointers are not used as keys,
// so sharing is possible
// see if not inserted twice
/*
for (TInstanceObserverMap::const_iterator it = lb; it != ub; ++it)
{
if (it->second == observer)
{
nlwarning("addInstanceObserver : Instance observer inserted twice");
return BadInstanceObserverHandle;
}
}
*/
// insert the handle
TInstanceObserverHandle handle = _InstanceObserverHandleCounter++;
if (_InstanceObserverHandleCounter == BadInstanceObserverHandle)
{
++ _InstanceObserverHandleCounter; // avoid bad handle
}
_InstanceObserverHandles[handle] = _InstanceObservers.insert(TInstanceObserverMap::value_type(instanceId, observer));
nlassert(_InstanceObservers.size() == _InstanceObserverHandles.size());
return handle;
}
// *********************************************************************************************************
CEditor::IInstanceObserver *CEditor::removeInstanceObserver(TInstanceObserverHandle handle)
{
//H_AUTO(R2_CEditor_removeInstanceObserver)
CHECK_EDITOR
nlassert(_InstanceObservers.size() == _InstanceObserverHandles.size());
TInstanceObserverHandleMap::iterator it = _InstanceObserverHandles.find(handle);
if (it == _InstanceObserverHandles.end())
{
nlwarning("removeInstanceObserver : Instance observer handle not found : %d", (int) handle);
return NULL;
}
IInstanceObserver *observer = it->second->second;
//nlwarning("#removing instance observer 0x%x", (int) observer);
_InstanceObservers.erase(it->second); // remove from observer map
_InstanceObserverHandles.erase(it);
nlassert(_InstanceObservers.size() == _InstanceObserverHandles.size());
return observer;
}
// *********************************************************************************************************
CEditor::IInstanceObserver *CEditor::getInstanceObserver(TInstanceObserverHandle handle)
{
//H_AUTO(R2_CEditor_getInstanceObserver)
CHECK_EDITOR
nlassert(_InstanceObservers.size() == _InstanceObserverHandles.size());
TInstanceObserverHandleMap::iterator it = _InstanceObserverHandles.find(handle);
if (it == _InstanceObserverHandles.end())
{
return NULL;
}
return it->second->second;
}
// *********************************************************************************************************
CEditor::TSeason CEditor::getSeason() const
{
//H_AUTO(R2_CEditor_getSeason)
if (_IsWaitingTPForSeasonChange) return UnknownSeason; // as long at the teleport message hasn't been received, don't change
// the season for nothing -> pretend that we don't know the season so that it remains unchnged
return _Season;
}
// *********************************************************************************************************
void CEditor::tpReceived()
{
//H_AUTO(R2_CEditor_tpReceived)
_IsWaitingTPForSeasonChange = false; // season can be changed now
_TPReceivedFlag = true;
}
// *********************************************************************************************************
void CEditor::checkMissingCollisions()
{
//H_AUTO(R2_CEditor_checkMissingCollisions)
CScenarioEntryPoints &sep = CScenarioEntryPoints::getInstance();
sep.loadCompleteIslands();
const CScenarioEntryPoints::TCompleteIslands &islands = sep.getCompleteIslands();
for(uint k = 0; k < islands.size(); ++k)
{
bool found = !(CPath::lookup(islands[k].Island + ".packed_island", false, false).empty());
if (!found)
{
nlwarning("ISLAND COLLISION MISSING FOR : %s", islands[k].Island.c_str());
}
found = !(CPath::lookup(islands[k].Island + ".island_hm", false, false).empty());
if (!found)
{
nlwarning("ISLAND HEIGHTMAP MISSING FOR : %s", islands[k].Island.c_str());
}
}
}
// *********************************************************************************************************
// move
class CAHEdContextMenu : public IActionHandler
{
virtual void execute(CCtrlBase * /* pCaller */, const std::string &/* sParams */)
{
getEditor().displayContextMenu();
}
};
REGISTER_ACTION_HANDLER(CAHEdContextMenu, "r2ed_context_menu");
// *********************************************************************************************************
NLMISC::CVector getVector(const CObject *obj)
{
return getVectorD(obj).asVector();
}
// *********************************************************************************************************
NLMISC::CVectorD getVectorD(const CObject *obj)
{
return CVectorD(getNumber(obj, "x"), getNumber(obj, "y"), getNumber(obj, "z"));
}
// *********************************************************************************************************
CObject *buildVector(const NLMISC::CVectorD &vector, const std::string &instanceId /*= ""*/)
{
CObject *table;
if (instanceId == "")
{
table = getEditor().getDMC().newComponent("Position");
table->set("x", vector.x);
table->set("y", vector.y);
table->set("z", vector.z);
}
else
{
table = new CObjectTableClient;
table->insert("InstanceId", new CObjectString(instanceId));
table->insert("Class", new CObjectString("Position"));
table->insert("x", new CObjectNumber(vector.x));
table->insert("y", new CObjectNumber(vector.y));
table->insert("z", new CObjectNumber(vector.z));
}
return table;
}
// *********************************************************************************************************
const CObject *getObject(const CObject *obj,const std::string &attrName)
{
if (!obj) return NULL;
return getEditor().getDMC().getPropertyAccessor().getPropertyValue(obj, attrName);
}
// *********************************************************************************************************
std::string getString(const CObject *obj, const std::string &attrName)
{
obj = getObject(obj, attrName);
if (!obj) return "";
return obj->isString() ? obj->toString() : "";
}
// *********************************************************************************************************
double getNumber(const CObject *obj, const std::string &attrName)
{
obj = getObject(obj, attrName);
if (!obj) return 0;
return obj->isNumber() ? obj->toNumber() : 0;
}
bool isEditionCurrent()
{
CEditor &ed = getEditor();
return ClientCfg.R2EDEnabled && ed.getMode() == CEditor::EditionMode;
}
// *********************************************************************************************************
void CEditor::onContinentChanged()
{
//H_AUTO(R2_CEditor_onContinentChanged)
if (_Mode != EditionMode) return;
// refresh all collisions
if (_ScenarioInstance)
{
struct CContinentChangedVisitor : public IInstanceVisitor
{
virtual void visit(CInstance &inst)
{
inst.onContinentChanged();
}
};
CContinentChangedVisitor continentChangedVisitor;
_ScenarioInstance->visit(continentChangedVisitor);
}
}
//-----------------------------
bool CEditor::getVisualPropertiesFromObject(CObject* object, SPropVisualA& vA, SPropVisualB& vB, SPropVisualC& vC)
{
//H_AUTO(R2_CEditor_getVisualPropertiesFromObject)
std::string sheetClient = getString(object, "SheetClient");
const CEntitySheet *entitySheet = SheetMngr.get(CSheetId(sheetClient));
if (!entitySheet)
{
nlwarning("Can't find client sheet %s", sheetClient.c_str());
return false;
}
CSheetId sheetId(sheetClient);
if (sheetId == CSheetId::Unknown)
{
nlwarning("Can't get sheet");
return false;
}
//-------------------------random init npc visual properties
std::map< std::string, double > visualProps;
static const char* keys[] = { "GabaritHeight", "GabaritTorsoWidth", "GabaritArmsWidth", "GabaritLegsWidth", "GabaritBreastSize"
, "HairType", "HairColor", "Tattoo", "EyesColor"
, "MorphTarget1", "MorphTarget2", "MorphTarget3", "MorphTarget4"
, "MorphTarget5", "MorphTarget6", "MorphTarget7", "MorphTarget8"
, "JacketModel", "TrouserModel", "FeetModel", "HandsModel"
, "ArmModel", "WeaponRightHand", "WeaponLeftHand"
, "JacketColor", "ArmColor", "HandsColor"
, "TrouserColor", "FeetColor"};
unsigned int first = 0;
unsigned int last = sizeof(keys) / sizeof(keys[0]);
for (; first != last; ++first)
{
visualProps[keys[first]] = getNumber(object, keys[first]);
}
//vA.PropertySubData.Sex = (uint) visualProps["Sex"];
if(entitySheet)
{
const CCharacterSheet *chSheet = dynamic_cast<const CCharacterSheet *>(entitySheet);
if(chSheet)
{
vA.PropertySubData.Sex = (chSheet->Gender == GSGENDER::female);
}
}
vC.PropertySubData.CharacterHeight = (uint) visualProps["GabaritHeight"];
vC.PropertySubData.ArmsWidth = (uint) visualProps["GabaritArmsWidth"];
vC.PropertySubData.TorsoWidth = (uint) visualProps["GabaritTorsoWidth"];
vC.PropertySubData.LegsWidth = (uint) visualProps["GabaritLegsWidth"];
vC.PropertySubData.BreastSize = (uint) visualProps["GabaritBreastSize"];
int itemNb = (int) visualProps["HairType"];
std::string itemFileName;
if(itemNb>0)
{
itemFileName = CSheetId(itemNb).toString();
vA.PropertySubData.HatModel = (uint) SheetMngr.getVSIndex(itemFileName, SLOTTYPE::HEAD_SLOT);
}
else
{
vA.PropertySubData.HatModel = 0;
}
vA.PropertySubData.HatColor = (uint) visualProps["HairColor"];
vC.PropertySubData.Tattoo = (uint) visualProps["Tattoo"];
vC.PropertySubData.EyesColor = (uint) visualProps["EyesColor"];
vC.PropertySubData.MorphTarget1 = (uint) visualProps["MorphTarget1"];
vC.PropertySubData.MorphTarget2 = (uint) visualProps["MorphTarget2"];
vC.PropertySubData.MorphTarget3 = (uint) visualProps["MorphTarget3"];
vC.PropertySubData.MorphTarget4 = (uint) visualProps["MorphTarget4"];
vC.PropertySubData.MorphTarget5 = (uint) visualProps["MorphTarget5"];
vC.PropertySubData.MorphTarget6 = (uint) visualProps["MorphTarget6"];
vC.PropertySubData.MorphTarget7 = (uint) visualProps["MorphTarget7"];
vC.PropertySubData.MorphTarget8 = (uint) visualProps["MorphTarget8"];
itemNb = (int) visualProps["JacketModel"];
if(itemNb>0)
{
itemFileName = CSheetId(itemNb).toString();
vA.PropertySubData.JacketModel = (uint) SheetMngr.getVSIndex(itemFileName, SLOTTYPE::CHEST_SLOT);
}
else
{
vA.PropertySubData.JacketModel = 0;
}
itemNb = (int) visualProps["TrouserModel"];
if(itemNb>0)
{
itemFileName = CSheetId(itemNb).toString();
vA.PropertySubData.TrouserModel = (uint) SheetMngr.getVSIndex(itemFileName, SLOTTYPE::LEGS_SLOT);
}
else
{
vA.PropertySubData.TrouserModel = 0;
}
itemNb = (int) visualProps["FeetModel"];
if(itemNb>0)
{
itemFileName = CSheetId(itemNb).toString();
vB.PropertySubData.FeetModel = (uint) SheetMngr.getVSIndex(itemFileName, SLOTTYPE::FEET_SLOT);
}
else
{
vB.PropertySubData.FeetModel = 0;
}
itemNb = (int) visualProps["HandsModel"];
if(itemNb>0)
{
itemFileName = CSheetId(itemNb).toString();
vB.PropertySubData.HandsModel = (uint) SheetMngr.getVSIndex(itemFileName, SLOTTYPE::HANDS_SLOT);
}
else
{
vB.PropertySubData.HandsModel = 0;
}
itemNb = (int) visualProps["ArmModel"];
if(itemNb>0)
{
itemFileName = CSheetId(itemNb).toString();
vA.PropertySubData.ArmModel = (uint) SheetMngr.getVSIndex(itemFileName, SLOTTYPE::ARMS_SLOT);
}
else
{
vA.PropertySubData.ArmModel = 0;
}
vA.PropertySubData.JacketColor = (uint) visualProps["JacketColor"];
vA.PropertySubData.TrouserColor = (uint) visualProps["TrouserColor"];
vB.PropertySubData.FeetColor = (uint) visualProps["FeetColor"];
vB.PropertySubData.HandsColor = (uint) visualProps["HandsColor"];
vA.PropertySubData.ArmColor = (uint) visualProps["ArmColor"];
itemNb = (int) visualProps["WeaponRightHand"];
if(itemNb>0)
{
itemFileName = CSheetId(itemNb).toString();
vA.PropertySubData.WeaponRightHand = (uint) SheetMngr.getVSIndex(itemFileName, SLOTTYPE::RIGHT_HAND_SLOT);
}
else
{
vA.PropertySubData.WeaponRightHand = 0;
}
itemNb = (int) visualProps["WeaponLeftHand"];
if(itemNb>0)
{
itemFileName = CSheetId(itemNb).toString();
vA.PropertySubData.WeaponLeftHand = (uint) SheetMngr.getVSIndex(itemFileName, SLOTTYPE::LEFT_HAND_SLOT);
}
else
{
vA.PropertySubData.WeaponLeftHand = 0;
}
return true;
}
// *********************************************************************************************************
bool CEditor::CSortedInstances::contains(CInstance *inst) const
{
TInstanceToIter::const_iterator it = _InstanceToIter.find(inst);
#ifdef NL_DEBUG
if (it == _InstanceToIter.end())
{
for(TInstanceByName::const_iterator it = _ByName.begin(); it != _ByName.end(); ++it)
{
nlassert(it->second != inst); // should be empty
}
}
#endif
return it != _InstanceToIter.end();
}
// *********************************************************************************************************
void CEditor::CSortedInstances::insert(const ucstring &name, CInstance *inst)
{
nlassert(inst);
#ifdef NL_DEBUG
static volatile bool doTest = true;
if (doTest)
{
nlassert(!contains(inst)); // inserted twice !!
}
#endif
_InstanceToIter[inst] = _ByName.insert(std::make_pair(name, inst));
}
// *********************************************************************************************************
void CEditor::CSortedInstances::remove(CInstance *inst)
{
nlassert(inst);
TInstanceToIter::iterator it = _InstanceToIter.find(inst);
nlassert(it != _InstanceToIter.end());
_ByName.erase(it->second);
_InstanceToIter.erase(it);
#ifdef NL_DEBUG
nlassert(!contains(inst));
#endif
}
// *********************************************************************************************************
bool CEditor::isRegisteredByDispName(CInstance *inst) const
{
nlassert(inst);
for (uint k = 0; k < _InstancesByDispName.size(); ++k)
{
if (_InstancesByDispName[k].contains(inst)) return true; // registered twice !!
}
return false;
}
// *********************************************************************************************************
void CEditor::registerInstanceDispName(const ucstring &displayName, CInstance *inst)
{
nlassert(inst);
sint currClass = inst->getClassIndex();
if (currClass < 0)
{
nlwarning("Classindex not found for class %s", inst->getClassName().c_str());
return;
}
#ifdef NL_DEBUG
nlassert(!isRegisteredByDispName(inst));
#endif
// for each class & subclass of the object, insert in the matching list
while (currClass >= 0)
{
nlassert(currClass < (sint) _InstancesByDispName.size());
_InstancesByDispName[currClass].insert(displayName, inst);
currClass = getBaseClass(currClass);
}
//
#ifdef NL_DEBUG
nlassert(isRegisteredByDispName(inst));
#endif
}
// *********************************************************************************************************
void CEditor::unregisterInstanceDispName(CInstance *inst)
{
nlassert(inst);
sint currClass = inst->getClassIndex();
if (currClass < 0)
{
nlwarning("Classindex not found for class %s", inst->getClassName().c_str());
return;
}
#ifdef NL_DEBUG
nlassert(isRegisteredByDispName(inst));
#endif
while (currClass >= 0)
{
nlassert(currClass < (sint) _InstancesByDispName.size());
_InstancesByDispName[currClass].remove(inst);
currClass = getBaseClass(currClass);
}
#ifdef NL_DEBUG
nlassert(!isRegisteredByDispName(inst));
#endif
}
// *********************************************************************************************************
//
// Creation of a new entity in scene
//
class CAHCreateEntity : public IActionHandler
{
virtual void execute(CCtrlBase * /* pCaller */, const std::string &sParams)
{
CHECK_EDITOR
if (getEditor().getMode() != CEditor::EditionMode)
{
nlwarning("Can't modify scenario while testing");
return;
}
CInterfaceManager *im = CInterfaceManager::getInstance();
// Retrieve sheet for entity that is to be created.
std::string paletteId = getParam(sParams, "PaletteId");
CObject *paletteNode = getEditor().getDMC().getPaletteElement(paletteId);
if (!paletteNode)
{
nlwarning("Can't retrieve palette node for id %s", paletteId.c_str());
return;
}
if (!paletteNode->isTable())
{
nlwarning("Bad type for palette node %s (should be a table)", paletteId.c_str());
return;
}
std::string sheetClient = getString(paletteNode, "SheetClient");
const CEntitySheet *entitySheet = SheetMngr.get(CSheetId(sheetClient));
if (!entitySheet)
{
nlwarning("Can't find client sheet %s", sheetClient.c_str());
return;
}
const CCharacterSheet *chSheet = dynamic_cast<const CCharacterSheet *>(entitySheet);
if(chSheet->R2Npc)
{
getEditor().getLua().push(sheetClient);
if (getEditor().getEnv().callMethodByNameNoThrow("randomNPCSex", 1, 1))
{
CLuaObject result(getEditor().getLua());
sheetClient = result.toString();
}
}
CSheetId sheetId(sheetClient);
if (sheetId == CSheetId::Unknown)
{
nlwarning("Can't get sheet");
return;
}
getEditor().setCurrentTool(NULL); // remove current to avoid to have ghost removed by that tool if it was a "CToolCreateEntity" too.
uint ghostSlot = 1; // TMP TMP
CEntityCL * entity = NULL;
if (!(entity=CEditor::createEntity(ghostSlot, sheetId, CVector::Null, 0.f)))
{
return;
}
//-------------------------random init npc visual properties
if(dynamic_cast<CPlayerR2CL *>(entity))
{
// push equipment id
getEditor().getLua().push(getString(paletteNode, "Equipment"));
// push race
std::string race;
switch(entity->people())
{
case EGSPD::CPeople::Fyros:
race = "Fyros";
break;
case EGSPD::CPeople::Matis:
race = "Matis";
break;
case EGSPD::CPeople::Tryker:
race = "Tryker";
break;
case EGSPD::CPeople::Zorai:
race = "Zorai";
break;
default:
nlwarning("CAHCreateEntity::execute unknown people");
}
getEditor().getLua().push(race);
if (getEditor().getEnv().callMethodByNameNoThrow("randomNPCProperties", 2, 1))
{
CLuaObject result(getEditor().getLua());
std::map< std::string, double > visualProps;
ENUM_LUA_TABLE(result, it)
{
visualProps[it.nextKey().toString()] = it.nextValue().toNumber();
}
// visual property A depends on the type of the entity
SPropVisualA vA;
SPropVisualB vB;
SPropVisualC vC;
sint64 *prop = 0;
//vA.PropertySubData.Sex = (uint) visualProps["Sex"];
const CEntitySheet *entitySheet = SheetMngr.get((CSheetId)sheetId.asInt());
if(entitySheet)
{
const CCharacterSheet *chSheet = dynamic_cast<const CCharacterSheet *>(entitySheet);
if(chSheet)
{
vA.PropertySubData.Sex = (chSheet->Gender == GSGENDER::female);
}
}
vC.PropertySubData.CharacterHeight = (uint) visualProps["GabaritHeight"];
vC.PropertySubData.ArmsWidth = (uint) visualProps["GabaritArmsWidth"];
vC.PropertySubData.TorsoWidth = (uint) visualProps["GabaritTorsoWidth"];
vC.PropertySubData.LegsWidth = (uint) visualProps["GabaritLegsWidth"];
vC.PropertySubData.BreastSize = (uint) visualProps["GabaritBreastSize"];
int itemNb = (int) visualProps["HairType"];
std::string itemFileName;
if(itemNb>0)
{
itemFileName = CSheetId(itemNb).toString();
vA.PropertySubData.HatModel = (uint) SheetMngr.getVSIndex(itemFileName, SLOTTYPE::HEAD_SLOT);
}
else
{
vA.PropertySubData.HatModel = 0;
}
vA.PropertySubData.HatColor = (uint) visualProps["HairColor"];
vC.PropertySubData.Tattoo = (uint) visualProps["Tattoo"];
vC.PropertySubData.EyesColor = (uint) visualProps["EyesColor"];
vC.PropertySubData.MorphTarget1 = (uint) visualProps["MorphTarget1"];
vC.PropertySubData.MorphTarget2 = (uint) visualProps["MorphTarget2"];
vC.PropertySubData.MorphTarget3 = (uint) visualProps["MorphTarget3"];
vC.PropertySubData.MorphTarget4 = (uint) visualProps["MorphTarget4"];
vC.PropertySubData.MorphTarget5 = (uint) visualProps["MorphTarget5"];
vC.PropertySubData.MorphTarget6 = (uint) visualProps["MorphTarget6"];
vC.PropertySubData.MorphTarget7 = (uint) visualProps["MorphTarget7"];
vC.PropertySubData.MorphTarget8 = (uint) visualProps["MorphTarget8"];
itemNb = (int) visualProps["JacketModel"];
if(itemNb>0)
{
itemFileName = CSheetId(itemNb).toString();
vA.PropertySubData.JacketModel = (uint) SheetMngr.getVSIndex(itemFileName, SLOTTYPE::CHEST_SLOT);
}
else
{
vA.PropertySubData.JacketModel = 0;
}
itemNb = (int) visualProps["TrouserModel"];
if(itemNb>0)
{
itemFileName = CSheetId(itemNb).toString();
vA.PropertySubData.TrouserModel = (uint) SheetMngr.getVSIndex(itemFileName, SLOTTYPE::LEGS_SLOT);
}
else
{
vA.PropertySubData.TrouserModel = 0;
}
itemNb = (int) visualProps["FeetModel"];
if(itemNb>0)
{
itemFileName = CSheetId(itemNb).toString();
vB.PropertySubData.FeetModel = (uint) SheetMngr.getVSIndex(itemFileName, SLOTTYPE::FEET_SLOT);
}
else
{
vB.PropertySubData.FeetModel = 0;
}
itemNb = (int) visualProps["HandsModel"];
if(itemNb>0)
{
itemFileName = CSheetId(itemNb).toString();
vB.PropertySubData.HandsModel = (uint) SheetMngr.getVSIndex(itemFileName, SLOTTYPE::HANDS_SLOT);
}
else
{
vB.PropertySubData.HandsModel = 0;
}
itemNb = (int) visualProps["ArmModel"];
if(itemNb>0)
{
itemFileName = CSheetId(itemNb).toString();
vA.PropertySubData.ArmModel = (uint) SheetMngr.getVSIndex(itemFileName, SLOTTYPE::ARMS_SLOT);
}
else
{
vA.PropertySubData.ArmModel = 0;
}
vA.PropertySubData.JacketColor = (uint) visualProps["JacketColor"];
vA.PropertySubData.TrouserColor = (uint) visualProps["TrouserColor"];
vB.PropertySubData.FeetColor = (uint) visualProps["FeetColor"];
vB.PropertySubData.HandsColor = (uint) visualProps["HandsColor"];
vA.PropertySubData.ArmColor = (uint) visualProps["ArmColor"];
itemNb = (int) visualProps["WeaponRightHand"];
if(itemNb>0)
{
itemFileName = CSheetId(itemNb).toString();
vA.PropertySubData.WeaponRightHand = (uint) SheetMngr.getVSIndex(itemFileName, SLOTTYPE::RIGHT_HAND_SLOT);
}
else
{
vA.PropertySubData.WeaponRightHand = 0;
}
itemNb = (int) visualProps["WeaponLeftHand"];
if(itemNb>0)
{
itemFileName = CSheetId(itemNb).toString();
vA.PropertySubData.WeaponLeftHand = (uint) SheetMngr.getVSIndex(itemFileName, SLOTTYPE::LEFT_HAND_SLOT);
}
else
{
vA.PropertySubData.WeaponLeftHand = 0;
}
prop = (sint64 *)&vA;
im->getDbProp("SERVER:Entities:E"+toString("%d", entity->slot())+":P"+toString("%d", CLFECOMMON::PROPERTY_VPA))->setValue64(*prop);
prop = (sint64 *)&vB;
im->getDbProp("SERVER:Entities:E"+toString("%d", entity->slot())+":P"+toString("%d", CLFECOMMON::PROPERTY_VPB))->setValue64(*prop);
prop = (sint64 *)&vC;
im->getDbProp("SERVER:Entities:E"+toString("%d", entity->slot())+":P"+toString("%d", CLFECOMMON::PROPERTY_VPC))->setValue64(*prop);
EntitiesMngr.updateVisualProperty(0, entity->slot(), CLFECOMMON::PROPERTY_VPA);
EntitiesMngr.updateVisualProperty(0, entity->slot(), CLFECOMMON::PROPERTY_VPB);
EntitiesMngr.updateVisualProperty(0, entity->slot(), CLFECOMMON::PROPERTY_VPC);
}
}
getEditor().setCurrentTool(new CToolCreateEntity(ghostSlot, paletteId, im->getDbProp("UI:TEMP:R2_DRAW_ARRAY")->getValueBool()));
}
};
REGISTER_ACTION_HANDLER(CAHCreateEntity, "r2ed_create_entity");
// *********************************************************************************************************
sint CEditor::classToIndex(const std::string &className) const
{
CHashMap<std::string, uint>::const_iterator it = _ClassNameToIndex.find(className);
if (it == _ClassNameToIndex.end()) return -1;
return it->second;
}
// *********************************************************************************************************
bool CEditor::isKindOf(sint testedClassIndex, sint kindClassIndex) const
{
if (testedClassIndex < 0 || kindClassIndex < 0) return false;
return _KindOfTable(testedClassIndex, kindClassIndex) != 0;
}
// *********************************************************************************************************
void CEditor::initClassInheritanceTable()
{
std::vector<std::string> classes;
std::map<std::string, std::string> baseClasses;
ENUM_LUA_TABLE(getClasses(), it)
{
CLuaObject name = it.nextValue()["Name"];
CLuaObject base = it.nextValue()["BaseClass"];
std::string baseName;
if (base.isString())
{
baseName = base.toString();
}
if (name.isString())
{
_ClassNameToIndex[name.toString()] = (uint)classes.size();
classes.push_back(name.toString());
baseClasses[name.toString()] = baseName;
}
}
_KindOfTable.init((uint)classes.size(), (uint)classes.size(), 0);
_BaseClassIndices.resize(classes.size());
for (uint k = 0; k < classes.size(); ++k)
{
//nlwarning("Class %d = %s", (int) k, classes[k].c_str());
_BaseClassIndices[k] = classToIndex(baseClasses[classes[k]]);
for (uint l = 0; l < classes.size(); ++l)
{
std::string currClass = classes[k];
while(!currClass.empty())
{
if (currClass == classes[l])
{
_KindOfTable(k, l) = 1;
break;
}
currClass = baseClasses[currClass];
}
}
}
_InstancesByDispName.clear();
_InstancesByDispName.resize(classes.size());
}
// *********************************************************************************************************
sint CEditor::getBaseClass(sint derivedClass) const
{
if (derivedClass < 0 || derivedClass >= (sint) _BaseClassIndices.size()) return -1;
return _BaseClassIndices[derivedClass];
}
// *********************************************************************************************************
void CEditor::setMaxVisibleEntityExceededFlag(bool on)
{
if (on == _MaxVisibleEntityExceededFlag) return;
_MaxVisibleEntityExceededFlag = on;
// lua ui update
CLuaStackChecker lsc(&getLua(), 0);
getLua().push(on);
callEnvMethod("setMaxVisibleEntityExceededFlag", 1, 0);
}
// *********************************************************************************************************
class CAHGoTest : public IActionHandler
{
virtual void execute(CCtrlBase * /* pCaller */, const std::string &/* sParams */)
{
CHECK_EDITOR
ConnectionWanted = true;
}
};
REGISTER_ACTION_HANDLER(CAHGoTest, "r2ed_go_test");
// *********************************************************************************************************
class CAHStopTest : public IActionHandler
{
virtual void execute(CCtrlBase * /* pCaller */, const std::string &/* sParams */)
{
CHECK_EDITOR
if (getEditor().getAccessMode() != CEditor::AccessEditor) return;
ConnectionWanted = true;
}
};
REGISTER_ACTION_HANDLER(CAHStopTest, "r2ed_stop_test");
// *********************************************************************************************************
class CAHOpenScenarioControl : public IActionHandler
{
virtual void execute(CCtrlBase * /* pCaller */, const std::string &sParams)
{
CHECK_EDITOR
bool openUI = true;
bool showHide = (getParam(sParams, "showHide")=="1");
CInterfaceManager *pIM = CInterfaceManager::getInstance();
CInterfaceGroup* wnd = NULL;
if(!R2::getEditor().isInitialized())
wnd = dynamic_cast<CInterfaceGroup*>(pIM->getElementFromId("ui:interface:ring_scenario_loading_window"));
else
wnd = dynamic_cast<CInterfaceGroup*>(pIM->getElementFromId("ui:interface:r2ed_scenario_control"));
if(wnd)
{
if(showHide)
openUI = !wnd->getActive();
wnd->setActive(openUI);
}
//}
}
};
REGISTER_ACTION_HANDLER(CAHOpenScenarioControl, "open_scenario_control");
// *********************************************************************************************************
class CAHR2StopLive : public IActionHandler
{
virtual void execute(CCtrlBase * /* pCaller */, const std::string &sParams)
{
CHECK_EDITOR
bool confirmStopLive = getParam(sParams, "confirmStopLive")=="1";
if(!confirmStopLive)
{
CInterfaceManager *pIM = CInterfaceManager::getInstance();
pIM->validMessageBox(CInterfaceManager::QuestionIconMsg, CI18N::get("uiR2EDconfirmStopLive"), "r2_stop_live", "confirmStopLive=1", "", "", "ui:interface");
}
else
{
CSessionBrowserImpl &sb = CSessionBrowserImpl::getInstance();
sb.closeSession(sb.getCharId(), R2::getEditor().getDMC().getEditionModule().getCurrentAdventureId());
if(sb.waitOneMessage(sb.getMessageName("on_invokeResult")))
{
if(R2::getEditor().getDMC().getEditionModule().getEditSessionLink()!=0)
{
// Now we expect to receive an impulsion FAR_TP to mainland, triggered
// by the DSS. We'll disobey it by FarTPing to the Edition session instead!
nldebug( "Will return to editing session %u", R2::getEditor().getDMC().getEditionModule().getEditSessionLink() );
FarTP.hookNextFarTPForEditor();
}
// otherwise, let accomplish the FAR_TP to mainland (returnToPreviousSession)
// triggered by the DSS.
}
else
{
nlwarning("closeSession callback return false");
}
}
}
};
REGISTER_ACTION_HANDLER(CAHR2StopLive, "r2_stop_live");
// *********************************************************************************************************
class CAHInviteCharacter : public IActionHandler
{
virtual void execute(CCtrlBase *pCaller, const std::string &/* sParams */)
{
CHECK_EDITOR
CInterfaceManager *pIM = CInterfaceManager::getInstance();
pIM->runActionHandler("leave_modal", pCaller, "");
if(pCaller)
{
CInterfaceGroup *fatherGC = pCaller->getParent();
if (fatherGC)
{
// Look for the root parent
for(;;)
{
CInterfaceGroup *parent = fatherGC->getParent();
if (!parent || (parent->getId()=="ui:interface"))
break;
fatherGC = parent;
}
// Get the modal edit box
CGroupEditBox *geb = dynamic_cast<CGroupEditBox *>(fatherGC->getGroup("add_contact_eb:eb"));
if (geb && !geb->getInputString().empty())
{
string charName = geb->getInputString().toString();
CSessionBrowserImpl & sessionBrowser = CSessionBrowserImpl::getInstance();
sessionBrowser.inviteCharacterByName(sessionBrowser.getCharId(), charName);
if(!sessionBrowser.waitOneMessage(sessionBrowser.getMessageName("on_invokeResult")))
{
nlwarning("inviteCharacterByName callback return false");
}
if(sessionBrowser._LastInvokeResult == 14)
{
CViewText* pVT = dynamic_cast<CViewText*>(pIM->getElementFromId("ui:interface:warning_free_trial:text"));
if (pVT != NULL)
pVT->setText(CI18N::get("uiRingWarningInviteFreeTrial"));
pIM->runActionHandler("enter_modal", pCaller, "group=ui:interface:warning_free_trial");
}
else if(sessionBrowser._LastInvokeResult == 12)
{
pIM->runActionHandler("enter_modal", pCaller, "group=ui:interface:warning_newcomer");
}
geb->setInputString(ucstring(""));
}
}
}
}
};
REGISTER_ACTION_HANDLER(CAHInviteCharacter, "r2ed_invite_character");
// *********************************************************************************************************
class CAHTryGoTest : public IActionHandler
{
virtual void execute(CCtrlBase * /* pCaller */, const std::string &/* sParams */)
{
CHECK_EDITOR
getEditor().callEnvMethod("tryGoTest", 0, 0);
}
};
REGISTER_ACTION_HANDLER(CAHTryGoTest, "r2ed_try_go_test");
// *********************************************************************************************************
class CAHCancelTool : public IActionHandler
{
virtual void execute(CCtrlBase * /* pCaller */, const std::string &/* sParams */)
{
if (!ClientCfg.R2EDEnabled || !getEditor().isInitialized()) return;
CHECK_EDITOR
getEditor().setCurrentTool(NULL);
}
};
REGISTER_ACTION_HANDLER(CAHCancelTool, "r2ed_cancel_tool");
//////////////////////////////////////////
// SWITCH DM / TESTER AT ANIMATION TIME //
//////////////////////////////////////////
// *********************************************************************************************************
class CAHAnimTestMode : public IActionHandler
{
virtual void execute(CCtrlBase * /* pCaller */, const std::string &/* sParams */)
{
CHECK_EDITOR
if (getEditor().getAccessMode() == CEditor::AccessDM) return;
getEditor().setMode(CEditor::TestMode);
}
};
REGISTER_ACTION_HANDLER(CAHAnimTestMode, "r2ed_anim_test_mode");
// *********************************************************************************************************
class CAHAnimDMMode : public IActionHandler
{
virtual void execute(CCtrlBase * /* pCaller */, const std::string &/* sParams */)
{
CHECK_EDITOR
getEditor().setMode(CEditor::DMMode);
}
};
REGISTER_ACTION_HANDLER(CAHAnimDMMode, "r2ed_anim_dm_mode");
// *********************************************************************************************************
class CAHR2ContextCommand : public IActionHandler
{
virtual void execute(CCtrlBase * /* pCaller */, const std::string &sParams)
{
CHECK_EDITOR
// forward the call to lua
// first param is the id of the command
std::string commandId = getParam(sParams, "commandId");
// if there's a current active tool, then see if it wants to handle it first
if (commandId == "delete")
{
if (getEditor().getCurrentTool() && getEditor().getCurrentTool()->onDeleteCmd())
return;
}
getEditor().getLua().push(commandId);
getEditor().callEnvMethod("execContextCommand", 1, 0);
}
};
REGISTER_ACTION_HANDLER(CAHR2ContextCommand, "r2ed_context_command");
// *********************************************************************************************************
class CAHR2Teleport : public IActionHandler
{
virtual void execute(CCtrlBase * /* pCaller */, const std::string &/* sParams */)
{
if (!ClientCfg.R2EDEnabled) return;
if (R2::getEditor().getMode() != CEditor::EditionMode &&
R2::getEditor().getMode() != CEditor::DMMode &&
R2::getEditor().getMode() != CEditor::AnimationModeDm) return;
CHECK_EDITOR
// just forward to lua
getEditor().callEnvMethod("activeTeleportTool", 0, 0);
}
};
REGISTER_ACTION_HANDLER(CAHR2Teleport, "r2ed_teleport");
// *********************************************************************************************************
class CAHR2Undo : public IActionHandler
{
virtual void execute(CCtrlBase * /* pCaller */, const std::string &/* sParams */)
{
// if an edit box currently has focus, then try undo on it first
CGroupEditBox *eb = dynamic_cast<CGroupEditBox *>(getEditor().getUI().getCaptureKeyboard());
if (eb && eb->undo())
{
return;
}
CActionHistoric &historic = getEditor().getDMC().getActionHistoric();
if (historic.canUndo())
{
getEditor().setCurrentTool(NULL);
const ucstring *actionName = historic.getPreviousActionName();
nlassert(actionName);
CLuaIHM::push(getEditor().getLua(), *actionName);
historic.undo();
getEditor().callEnvMethod("onUndo", 1, 0);
}
else
{
getEditor().callEnvMethod("onCantUndo", 0, 0);
}
}
};
REGISTER_ACTION_HANDLER(CAHR2Undo, "r2ed_undo");
// *********************************************************************************************************
class CAHR2Redo : public IActionHandler
{
virtual void execute(CCtrlBase * /* pCaller */, const std::string &/* sParams */)
{
// if an edit box currently has focus, then try redo on it first
CGroupEditBox *eb = dynamic_cast<CGroupEditBox *>(getEditor().getUI().getCaptureKeyboard());
if (eb && eb->redo())
{
return;
}
CActionHistoric &historic = getEditor().getDMC().getActionHistoric();
if (historic.canRedo())
{
getEditor().setCurrentTool(NULL);
const ucstring *actionName = historic.getNextActionName();
nlassert(actionName);
CLuaIHM::push(getEditor().getLua(), *actionName);
historic.redo();
getEditor().callEnvMethod("onRedo", 1, 0);
}
else
{
getEditor().callEnvMethod("onCantRedo", 0, 0);
}
}
};
REGISTER_ACTION_HANDLER(CAHR2Redo, "r2ed_redo");
// *********************************************************************************************************
// signal the server that a dm gift has begun, so that the server can take note of the target entity (the current target)
class CAHR2DMGiftBegin : public IActionHandler
{
virtual void execute(CCtrlBase * /* pCaller */, const std::string &/* sParams */)
{
CBitMemStream out;
if(GenericMsgHeaderMngr.pushNameToStream("DM_GIFT:BEGIN", out))
{
NetMngr.push(out);
}
else
nlwarning("<CHandlerInvalidateExchange::execute> unknown message name 'DM_GIFT:BEGIN");
}
};
REGISTER_ACTION_HANDLER(CAHR2DMGiftBegin, "r2ed_dm_gift_begin");
// *********************************************************************************************************
// signal the server that a dm gift has begun, so that the server can take note of the target entity (the current target)
class CAHR2DMGiftValidate : public IActionHandler
{
virtual void execute(CCtrlBase * /* pCaller */, const std::string &/* sParams */)
{
CInterfaceManager *im = CInterfaceManager::getInstance();
CBitMemStream out;
if(GenericMsgHeaderMngr.pushNameToStream("DM_GIFT:VALIDATE", out))
{
for(uint k = 0; k < 8; ++k)
{
uint32 sheetId = 0;
uint8 quantity = 0;
CCDBNodeLeaf *sheetLeaf = im->getDbProp(toString("LOCAL:R2:DM_GIFT:%d:SHEET", (int) k));
if (sheetLeaf) sheetId = (uint32) sheetLeaf->getValue32();
CCDBNodeLeaf *quantityLeaf = im->getDbProp(toString("LOCAL:R2:DM_GIFT:%d:QUANTITY", (int) k));
if (quantityLeaf) quantity = (uint8) quantityLeaf->getValue8();
out.serial(sheetId);
out.serial(quantity);
}
NetMngr.push(out);
}
else
nlwarning("<CHandlerInvalidateExchange::execute> unknown message name 'DM_GIFT:VALIDATE");
}
};
REGISTER_ACTION_HANDLER(CAHR2DMGiftValidate, "r2ed_dm_gift_validate");
// *********************************************************************************************************
// freeze / unfreeze bot objects
class CAHR2FreezeUnfreezeBotObjects : public IActionHandler
{
virtual void execute(CCtrlBase * /* pCaller */, const std::string &/* sParams */)
{
CInterfaceManager *im = CInterfaceManager::getInstance();
im->executeLuaScript("r2:freezeUnfreezeBotObjects()");
}
};
REGISTER_ACTION_HANDLER(CAHR2FreezeUnfreezeBotObjects, "r2ed_freeze_unfreeze_botobjects");
} // R2
// *********************************************************************************************************
// COMMAND TO REINITIALIZE THE EDITOR : useful to check scripts error
NLMISC_COMMAND(resetEditor, "reset R2 editor, reload all scripts (except ui xmls)", "")
{
if (!args.empty()) return false;
R2::ResetWanted = true;
R2::ReloadUIFlag = false;
return true;
}
// *********************************************************************************************************
// COMMAND TO REINITIALIZE THE EDITOR : useful to check scripts error
NLMISC_COMMAND(resetEditorAndReloadUI, "reset R2 editor, reload all scripts & xml (possibly preserving scenario)", "")
{
if (!args.empty()) return false;
R2::ResetWanted = true;
R2::ReloadUIFlag = true;
return true;
}
// *********************************************************************************************************
// write scenario rtdata to a file for debugging or stress testing by simulator_service
NLMISC_COMMAND(saveScenarioRtData, "save scenario RtData to file", "<filename>")
{
const R2::CObject *scenario = R2::getEditor().getDMC().getHighLevel();
if( !scenario )
{
nlinfo("No current scenario -- can't save rtdata");
return false;
}
std::string fileName = "outpout";
if( args.empty() )
{
// try to name file from title, then area+instance
R2::CObject *description = scenario->findAttr("Description");
if( description )
{
R2::CObject *name = description->findAttr("Name");
R2::CObject *title = 0;
if (name)
{
title = name;
}
else
{
title = description->findAttr("Title"); //obsolete
}
R2::CObject *locationId = description->findAttr("LocationId");
R2::CObject *instanceId = description->findAttr("InstanceId");
if( title )
{
string sTitle = title->toString();
if( !sTitle.empty() )
fileName = sTitle;
}
else if( locationId && instanceId )
fileName = "area" + locationId->toString() + "_" + instanceId->toString();
}
}
else // filename given
{
fileName = args[0];
}
nlinfo("translating current scenario");
R2::CObject *pHighLevel = scenario->clone();
nlassert( pHighLevel );
R2::CObject *pRtData = R2::getEditor().getDMC().translateScenario( pHighLevel );
delete pHighLevel;
if( !pRtData )
{
nlwarning("Failed to translate high-level scenario into rtdata");
return false;
}
// binary
string fullFileName = fileName;
fullFileName += ".rt.bin";
nlinfo("writing rtdata to %s", fullFileName.c_str());
COFile output;
output.open(fullFileName);
R2::CObjectSerializerClient serializer( pRtData );
serializer.serial(output);
output.flush();
output.close();
if( true ) // debug
{
// text
fullFileName = fileName;
fullFileName += ".rt.txt";
COFile output;
//std::stringstream ss;
std::string ss;
output.open(fullFileName);
pRtData->serialize(ss);
//std::string str = ss.str();
output.serial(ss);
output.flush();
output.close();
}
return true;
}
NLMISC_COMMAND(dumpValidPositions, "dump valid position to create objects from the current camera viewpoint", "<filename>")
{
if (args.size() != 1) return false;
try
{
COFile output(args[0]);
CBitmap result;
Driver->getBuffer(result);
if (result.getWidth() == 0 || result.getHeight() == 0) return false;
CMatrix camMatrix = MainCam.getMatrix();
NL3D::CFrustum fru = MainCam.getFrustum();
R2::CTool::CWorldViewRay wvr;
wvr.Right = camMatrix.getI().normed();
wvr.Up = camMatrix.getK().normed();
wvr.OnMiniMap = false;
wvr.Origin = camMatrix.getPos();
wvr.Valid = true;
for (uint x = 0; x < result.getWidth(); ++x)
{
for (uint y = 0; y < result.getHeight(); ++y)
{
CVector ray(fru.Left + (fru.Right - fru.Left) * (x / (float) result.getWidth()), fru.Near,
fru.Top + (fru.Bottom - fru.Top) * (y / (float) result.getHeight()));
ray.normalize();
ray = camMatrix.mulVector(ray);
CVector inter;
wvr.Dir = ray;
R2::CTool::TRayIntersectionType rit = R2::CTool::computeLandscapeRayIntersection(wvr, inter);
CRGBA resultCol;
switch(rit)
{
case R2::CTool::NoIntersection: resultCol = CRGBA(0, 0, 0, 0); break;
case R2::CTool::ValidPacsPos: resultCol = CRGBA(0, 0, 255, 100); break;
default: resultCol = CRGBA(255, 0, 0, 100); break;
}
CRGBA *dest = ((CRGBA *) &result.getPixels(0)[0]) + x + y * result.getWidth();
dest->blendFromui(*dest, resultCol, resultCol.A);
}
}
result.writeTGA(output, 32, false);
}
catch(...)
{
}
return true;
}
/*
// *********************************************************************************************************
NLMISC_COMMAND(resetScenario, "reset R2 editor, reload all scripts & xml", "")
{
if (!args.empty()) return false;
R2::ResetScenarioWanted = true;
return true;
}
// *********************************************************************************************************
NLMISC_COMMAND(reloadScenario, "reset R2 editor, reload all scripts & xml (possibly preserving scenario)", "")
{
if (!args.empty()) return false;
//
R2::ReloadScenarioWanted = true;
return true;
}
*/
bool IsInRingMode()
{
return R2::getEditor().getMode() != R2::CEditor::NotInitialized;
}