khanat-opennel-code/code/ryzom/client/src/r2/editor.h
2010-11-12 14:26:38 +01:00

1086 lines
44 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/>.
#ifndef R2_EDITOR_H
#define R2_EDITOR_H
#include "../interface_v3/lua_object.h"
#include "instance.h"
#include "tool.h"
#include "../decal.h"
#include "../decal_anim.h"
#include "entity_custom_select_box.h"
#include "island_collision.h"
#include "prim_render.h"
//
#include "nel/misc/singleton.h"
#include "nel/misc/class_registry.h"
#include "nel/misc/time_nl.h"
#include "nel/misc/sstring.h"
#include "nel/misc/array_2d.h"
//
#include "dmc/dmc.h"
//
#include "game_share/object.h"
#include "game_share/scenario_entry_points.h"
#include "game_share/r2_types.h"
class CGroupTree;
class CEntityCL;
namespace NL3D
{
class CPackedWorld;
}
namespace R2
{
class CDynamicMapService;
class CUserComponentsManager;
class CEntitySorter;
// texture for the default mouse cursor
extern const char *DEFAULT_CURSOR;
class CDisplayerVisualEntity;
/** Ryzom Ring in game editor
*
* ///////////////////////////////////////////////////////////
* // TMP OVERVIEW OF R2 UI BEHAVIOR (WORK IN PROGRESS ...) //
* ///////////////////////////////////////////////////////////
*
*
* The ui can be in one of the following states :
*
* I CREATION MODE
* ===============
*
* No tool is displayed as "highlighted"
*
* 1) Mouse in 3D view
* --------------------
* The mouse cursor has a little star to indicate "creation mode".
* If creation is not possible at the mouse position, then there's is a little 'stop' icon shown
* No instance selection is done
*
* EVENTS :
* --------
* LCLICK -> create
* RCLICK -> cancel & restore default tool
*
* NB : we don't do creation on LDOWN or RUP in order to allow camera manipulation
*
* 2) Mouse over UI
* -----------------
* The mouse cursor still has a little star to indicate "creation mode".
* Ideally on some actions, the creation is canceled, and the default tool is backuped;
* NB nico : this is unclear to me atm, is it for button clicks only ?
* (for example just moving a slider do not seem a good reason to cancel the tool...)
* so no-op for now
* (TODO : exception for the minimap, if we want to be able to create by clicking on it ?)
*
* II CONTEXT MENU MODE
* ====================
*
* The standard mouse cursor is displayed.
*
* EVENTS :
* --------
* LDOWN / RDOWN -> cancel the menu
*
* III CAPTURED BY UI MODE
* =======================
* ... same as 'context menu mode'
*
* IV TOOL MODE
* =============
* The current tool icon is highlighted in the toolbar
*
* std tool includes :
* - Move
* - Rotate
* ...
*
* 1) Mouse in 3D view
* --------------------
* a ) Mouse over empty space in scene
* -----------------------------------
* The standard mouse cursor is displayed
*
* EVENTS :
* --------
* nothing to do here, camera event are handled by the 'UserControls' global object
* If the right button is up without being preceded by a camera move, then the context
* menu is shown
*
*
* b ) Mouse over an instance in scene that is not the selection
* -------------------------------------------------------------
* The mouse with a little circle is displayed
*
* EVENTS :
* --------
* RDOWN -> Test and see which one is best :
* - a new slection is done and the context menu is shown ?
* - nothing ? (e.g no selection and the context menu is trigerred by
* the 'UserControls' global object as usual ?
* RDOWN while maintained action : cancel the current action and restore
* state (same behaviour as most apps)
* LDOWN -> selection + tool dependent (instant action or maintained action)
* a maintained action (such as moving an entity) will usually
* capture the mouse.
* others -> tool dependent
*
* c ) Mouse over an instance in scene that the selection
* -------------------------------------------------------------
* The mouse shows the action that can be performed
* RDOWN -> same as previous
* others -> tool dependent
* RDOWN while maintained action : cancel the current action and restore
* state (same behaviour as most apps)
*
* NB : a tool may want to capture the mouse, in this case it must call CTool::captureMouse
* and CTool::releaseMouse when it is done
*
* REMARKS:
*
* - Maybe it would be cool to make camera manipulation totally orthogonal to the system,
* and transparently layered on top of it.
* This would requires using something like the middle button or some shift / control combination
* to avoid conflict with the previous events ...
*
* 2) Mouse over UI
* -----------------
* The mouse is dipslayed as usual
* Events are taken in account by the UI, not by the tool
* This is transparent for CTool derivers, because mouse events won't reach them
* Clicking in the interface doesn't change the current tool
*
*
* \author Nicolas Vizerie
* \author Nevrax France
* \date 5/2005
*/
class CEditor : public NLMISC::CSingleton<CEditor>
{
public:
enum TMode { NotInitialized = 0, EditionMode, GoingToDMMode, GoingToEditionMode, TestMode, DMMode,
AnimationModeLoading, AnimationModeWaitingForLoading, AnimationModeDm, AnimationModePlay,
AnimationModeGoingToDm, AnimationModeGoingToPlay, ModeCount };
enum TAccessMode { AccessEditor = 0, AccessDM, AccessOutlandOwner, AccessModeCount, AccessModeUnknown = AccessModeCount };
////////////
// OBJECT //
////////////
CEditor();
~CEditor();
// Init what's need to be initialized depending on the configuration
void autoConfigInit(bool serverIsRingSession);
// Release what's need to be released depending on the configuration
void autoConfigRelease(bool serverIsRingSession);
// initialisation of modules & gw
void initModules(bool connectDMC);
void releaseModules();
// wait first scenario to be received (may be an empty one ...)
void waitScenario();
TAccessMode getAccessMode() const { return _AccessMode; }
void setAccessMode(TAccessMode mode);
// Initialisation of the editor
void init(TMode initialMode, TAccessMode accessMode);
// reload xml, lua & translation & reset. scenario is preserved
void reset();
// reload xml, lua & translation & reset. scenario is reseted
void resetScenario();
// reload xml, lua & translation & reset. Last save scenario is reloaded
void reloadScenario();
// load / reload the work language file
void loadLanguageFile();
//
void release();
// clear all content in the map, client side
void clearContent();
//
bool isInitialized() const { return _Initialized; }
// (re)load the ui if its date changed
void reloadUI();
// set current mode of the editor
void setMode(TMode mode);
TMode getMode() const { return _Mode; }
// test if user is DMing (be it masterless or not)
bool isDMing() const;
/////////////////////
// UPDATE & EVENTS //
/////////////////////
/** Events handling
* Handle an user input event
* \return true if the event was handled by the editor
*/
bool handleEvent (const CEventDescriptor &eventDesc);
// Handle copy, reaches the editor if no edit box is currently active
void copy();
// Handle paste, reaches the editor if no edit box is currently active
void paste();
// An entity has been selected with the standard selection system (usually in test mode)
void inGameSelection(const CLFECOMMON::TCLEntityId &slot);
void onContinentChanged();
/** update of editor just before precamera call for entities
*/
void updatePreCamera();
/** update of editor (after update of entities just before render)
*/
void updateBeforeRender();
/** Update of editor (just after render of the scene and entities update and before rendering of the interface)
* This is the place to display additionnal visual infos (slection bbox etc ...)
*/
void updateAfterRender();
// called after main loop 'swap buffer'
void updateBeforeSwapBuffer();
/////////////////////////////////////////////
// CLIENT OBJECTS (yet named 'instances' ) //
/////////////////////////////////////////////
// NB about this tmp mess
// 'CInstance' are client counterparts of 'CObjectTable'
// set current selected instance in the editor
void setSelectedInstance(CInstance *inst);
void forceSetSelectedInstance(CInstance *inst);
// get current selected instance in the editor
CInstance *getSelectedInstance() const;
// set current highlighted instance in the editor
void setFocusedInstance(CInstance *inst);
// get current higlighted entity in the editor (with mouseoverit)
CInstance *getFocusedInstance() const;
//
bool hasInstance(const CInstance *instance) const;
// Check if there's an instance with the given id
bool hasInstanceWithId(const TInstanceId &id) const;
// Get an instance from its id
CInstance *getInstanceFromId(const TInstanceId &id) const;
/** Create Visual Properties for an instance
* Return false if can not retrieve VisualProperties
*/
bool getVisualPropertiesFromObject(CObject* instance, SPropVisualA& vA, SPropVisualB& vB, SPropVisualC& vC);
/** Generate a new user readable name for an object in the scene
* If the name if already postfixed by a number, it will be stripped.
* Example : Npc 1, Npc 2, Group 1, Bandit camp 1 etc.
* Localised base name is given (for example "road" if using english)
* The function looks in all road instances and finds the first free number.
*
*
* \param baseClass Filter base class for the search : only class that derived from (or are of ) that class will
* be considered when looking for a new name
*/
ucstring genInstanceName(const ucstring &baseName);
// test if an instance name is post fixed by a number
static bool isPostFixedByNumber(const ucstring &baseName);
/** test if a ucstring has the format of a generated name that is : baseName + " " + index
* Return index on success or -1 on failure
*/
static sint getGeneratedNameIndex(const std::string &nameUtf8, const std::string &baseNameUtf8);
/** Set a local user value (we name it a "cookie") that should be attached to an instance at creation time
* (that is, when the object will be created on client -> when the server creation msg is received)
* The value will be stored in the 'User' lua table.
* The 'User' table is attached to each instances in the editor and is Read/Write
* (other properties in client and server and are read only, and should be modified by sending a requestxxx network message)
* This table is local only (that is, not seen by the server, nor by other clients editing the same map).
*
* Example of use :
*
* We want that when a road is created, a dialog pop to enter the name of the road.
*
* First we must be able to distinguish between roads created by ourselves and road created by
* other persons editing the map.
*
* Secondly we don't want that all roads to have this behaviour (for example, a road in a more complex system
* would be named automatically)
*
*
* In lua script it would look like :
*
* At creation request time :
* ==========================
*
* local road = r2.newComponent("Road")
* r2.setCookie(road.InstanceId, "AskName", true)
* r2.requestInsertNode(... -> send newtork creation message for the road
*
*
* When creation message is received (may be placed in the 'onCreate' method of a displayer attached to that instance)
* ==================================================================================================================
*
* function roadDisplayer:onCreate(road)
* if road.User.AskName == true then
* -- road was created by this client
* -- dialog to enter the name was asked
* -- => popup the dialog
* ....
* end
* end
*
* Other example : when an act is created by the user it becomes the current act
*
*/
void setCookie(const TInstanceId &instanceId, const std::string &key, CLuaObject &value);
// helpers to add cookie of predefined type (to be completed ..)
void setCookie(const TInstanceId &instanceId, const std::string &key, bool value);
struct IInstanceObserver : public NLMISC::CRefCount // refptr'ed observers
{
virtual ~IInstanceObserver() {}
typedef NLMISC::CRefPtr<IInstanceObserver> TRefPtr;
// called when the watched instance has been created
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 onInstanceEraseRequest(CInstance &/* instance */) {}
virtual void onAttrModified(CInstance &/* instance */, const std::string &/* attrName */, sint32 /* attrIndex */) {}
};
typedef sint TInstanceObserverHandle;
enum { BadInstanceObserverHandle = -1 };
/** add a watch to know when an instance is created (the instance is identified by its instance id)
* returns a handle for removal, or 'BadInstanceObserverHandle' if creation failed
*/
TInstanceObserverHandle addInstanceObserver(const TInstanceId &instanceId, IInstanceObserver *observer);
// Get a pointer to an observer from its handle (or NULL if not found)
IInstanceObserver *getInstanceObserver(TInstanceObserverHandle handle);
// Remove an instance observer from its handle (but do not delete it). Return the pointer to the observer
IInstanceObserver *removeInstanceObserver(TInstanceObserverHandle handle);
/** Test from a pointer if the object is currently observing an instance
* NB : slow because of linear search
*/
bool isInstanceObserver(IInstanceObserver *observer) const;
////////////////////////
// LUA R2 ENVIRONMENT //
////////////////////////
// get table for global variables in lua environment (equivallent to _G lua variable)
CLuaObject &getGlobals() { return _Globals; }
// get table for registry in lua environment
CLuaObject &getRegistry() { return _Registry; }
// get lua classes (the r2.Classes table)
CLuaObject getClasses() throw(ELuaError);
// get R2 environment (the 'r2' table into lua global environment)
CLuaObject &getEnv();
// get the config table (that is the 'r2.Config' table)
CLuaObject getConfig();
// get a reference to the lua state object
CLuaState &getLua();
/** Project a CObjectTable into lua (accessor is pushed onto the lua stack)
* property of the table, which is a C++ object, will be accessible from lua (by using metatable)
*/
void projectInLua(const CObjectTable *table);
// get the default feature for the current selected act
CInstance *getDefaultFeature();
// get the default feature for the given act
CInstance *getDefaultFeature(CInstance *act);
// set the current act (NULL being synonimous with the base act)
void setCurrentAct(CInstance *act);
void setCurrentActFromTitle(const std::string &title);
CInstance *getCurrentAct() const { return _CurrentAct; }
CInstance *getBaseAct() const { return _BaseAct; }
/** helper : calls a function in the lua r2ed environment
* Arguments must have already been pushed on the stack
* If the call fails then an error msg is printed in the log, and no arguments are returned
*
* \param funcName name of the function to call (must resides in the r2ed table)
* \param numArgs, numbers of arguments that the functions will receive
* \param numRet : Number of parameters that the function returns
* As usual the stack will be adjusted to that size after the call
*/
bool callEnvFunc(const char *funcName, int numArgs, int numRet = 0);
// Behave like 'callEnvFunc', but call a method instead (r2 is passed as the 'self' parameter, that is)
bool callEnvMethod(const char *funcName, int numArgs, int numRet = 0);
//////////////////////
// SERVER COMMANDS //
//////////////////////
// access to interface with server
CDynamicMapClient &getDMC() const { nlassert(_DMC); return *_DMC; }
/** Set a property of an object locally. No network msg is sent, but modification events are triggered to signal that
* the object has been modified (rooted to one of the CDisplayerBase derived object, attached to the object being modified. see displayer_base.h).
* Changes to local value must be commited or cancelled when edition is finish by calling 'requestCommitLocalNode' or
* 'requestRollbackLocalNode'
* Typical use is by the slider widget : real value is sent accross the network only when the user release the mouse, but
* local update of object property related to the slider is done each time the slider is modified before release.
* During the maintained action, no network message are desireable.
* NB : a copy of input parameter 'value' will be done. Caller is responsible for deleting 'value' after use.
*/
void requestSetLocalNode(const std::string& instanceId, const std::string& attrName, const CObject *value);
void requestCommitLocalNode(const std::string& instanceId, const std::string& attrName);
void requestRollbackLocalNode(const std::string& instanceId, const std::string& attrName);
/////////////
// DECALS //
/////////////
void showHighlightDecal(const NLMISC::CVector &pos, float scale);
void showSelectDecal(const NLMISC::CVector &pos, float scale);
void addSelectingDecal(const NLMISC::CVector &pos, float scale);
void showSelectBox(const NLMISC::CAABBox &localBox, const NLMISC::CMatrix &worldMat);
void showHighlightBox(const NLMISC::CAABBox &localBox, const NLMISC::CMatrix &worldMat);
/////////////////
// COLLISIONS //
/////////////////
CIslandCollision &getIslandCollision() { return _IslandCollision; }
///////////
// MISC //
///////////
// Shortcut to the GUI
static CInterfaceManager &getUI();
//
/** Set current edition tool. NULL will reset to the default tool (selection tool)
*/
void setCurrentTool(CTool *tool);
// Get current tool for edition
CTool *getCurrentTool() const { return _CurrentTool; }
/** Helper to execute a lua script
* \param filename name of the lua script file
* \param fileDescText short description of the script function (for error messages)
* \return true on success
*/
bool doLuaScript(const char *filename, const char *fileDescText);
// helper : create an entity in scene at the given slot & position, replacing any previous entity in that slot
static CEntityCL *createEntity(uint slot, const NLMISC::CSheetId &sheetId, const NLMISC::CVector &pos, float heading, const std::string & permanentStatutIcon=std::string(""));
// helper : get an instance from a CEntityCL pointer
CInstance *getInstanceFromEntity(CEntityCL *entity) const;
// helper : clear content of the debug window
void clearDebugWindow();
// display the editor contextual menu
void displayContextMenu();
// tmp, for debug
void dumpInstances();
// helper : return the entity under the mouse
CInstance *getInstanceUnderPos(float x, float y, float distSelection, bool &isPlayerUnderCursor);
// helper : test intersection between an entity and a ray
static float preciseEntityIntersectionTest(CEntityCL &entity, const NLMISC::CVector &worldRayStart, const NLMISC::CVector &worldRayDir);
// Tmp show the connexion window and display a msg in it. An empty msg will close the window
static void connexionMsg(const std::string &stringId);
TEntityCustomSelectBoxMap &getEntityCustomSelectBoxMap() { return _EntityCustomSelectBoxMap; }
// from a pointer on an entity, retrieve its local selection bbox (possibly redefined in r2_ui_custom_boxes_data.lua
const NLMISC::CAABBox &getLocalSelectBox(CEntityCL &entity) const;
// from a pointer on an entity, retrieve its local selection bbox (possibly redefined in r2_ui_custom_boxes_data.lua
NLMISC::CAABBox getSelectBox(CEntityCL &entity) const;
// check if there's room left to create new objects in the scene
sint getLeftQuota();
bool checkRoomLeft();
// display an error msg to prompt the user to make room for new objects in its scenario
void makeRoomMsg();
// check if there is room in specific category if not display an error msg
// verify ther is at least size object in category StaticObject or AiObject
bool verifyRoomLeft(uint aiCost, uint staticCost);
// rename delete auto_save9, rename auto_save1.r2 to auto_save2.r2 and so until auto_save8.r2, copy auto_save.r2 to auto_save1.r2,
void autoSave();
/** Season driven from editor
* This value is usually 'Unknwown' unless the mode is "edit'.
* In this case, the value depends on the act being edited
*/
enum TSeason { Automatic = 0, Spring, Summer, Autumn, Winter, UnknownSeason };
TSeason getSeason() const;
void tpReceived();
bool getFixedLighting() const { return _FixedLighting; }
void setFixedLighting(bool enabled);
/** Get current infos of a plot item plot display in the editor
* Plot items are items with SCROLL_R2 as family. Each sheet can be used only once in
* a scenario (meaning that each icon can only be seen for a single plot item).
* The name of a plot item can also be change by using the lua command r2:setPlotItemName(sheetId, ucName)
*/
void setPlotItemInfos(const TMissionItem &infos);
const TMissionItem *getPlotItemInfos(uint32 sheetId) const;
// convert name of a class in R2 to a unique index (-1 if not found)
sint classToIndex(const std::string &className) const;
// this if one class id 'indexOf' another from their index
bool isKindOf(sint testedClassIndex, sint kindClassIndex) const;
// from one class index, returns the index to the derived class (-1 if not found)
sint getBaseClass(sint derivedClass) const;
///////////////////////////////
// > 254 entities management //
///////////////////////////////
CEntitySorter *getEntitySorter() const;
private:
CEntitySorter *_EntitySorter;
// mapping from sheet id to plot item name
std::map<uint32, TMissionItem> _PlotItemInfos;
CIslandCollision _IslandCollision;
//
bool _SerializeUIConfig;
TMode _Mode;
TAccessMode _AccessMode;
typedef std::map<const CObjectTable *, CInstance::TSmartPtr> TInstanceMap;
private:
CLuaObject _Globals; // match to the '_G' lua global variable
CLuaObject _Registry;
CLuaObject _Env;
CLuaObject _Config;
CLuaObject _ObjectProjectionMetatable;
CLuaObject _LuaUIMainLoop;
//
TInstanceMap _Instances;
CInstance *_SelectedInstance;
CInstance *_FocusedInstance;
CHashMap<std::string, uint> _ClassNameToIndex; // Map each class name to an unique index (filled at init)
NLMISC::CArray2D<uint8> _KindOfTable; // Table to test if one class derives from another (filled at init)
std::vector<sint> _BaseClassIndices; // for each class, give index of thebase class (vector ordered by classes indices)
typedef CHashMultiMap<ucstring, CInstance *, NLMISC::CUCStringHashMapTraits> TInstanceByName;
bool _MaxVisibleEntityExceededFlag;
// instance sorted by name
class CSortedInstances
{
public:
void insert(const ucstring &name, CInstance *inst);
void remove(CInstance *inst);
bool contains(CInstance *inst) const;
TInstanceByName::iterator begin() { return _ByName.begin(); }
TInstanceByName::iterator end() { return _ByName.end(); }
private:
typedef std::map<CInstance *, TInstanceByName::iterator> TInstanceToIter;
//
TInstanceByName _ByName;
TInstanceToIter _InstanceToIter;
};
// typedef TInstanceByDispName; // for usage by CInstance
// list of instances for each classes (ordered by class index)
std::vector<CSortedInstances> _InstancesByDispName; // instances sorted by their display name (private use)
//priv for debug
bool isRegisteredByDispName(CInstance *inst) const;
// Cookies (local objects attached to instances at creation time, see setCookie)
struct CCookie
{
std::string Key;
CLuaObject Value;
};
typedef std::list<CCookie> TCookieList; // for a single instance, map each key to its value
typedef std::map<TInstanceId, TCookieList> TCookieMap;
TCookieMap _Cookies;
//
CInstance::TRefPtr _CurrentAct;
CInstance::TRefPtr _BaseAct;
CInstance::TRefPtr _ScenarioInstance;
CObjectTable *_Scenario;
std::string _WantedActOnInit;
//
friend class CDynamicMapClientEventForwarder;
friend class CAHR2QuitConnectingScreen;
CDynamicMapClient *_DMC; // replication of server map state on that client
CDynamicMapService *_DMS; // the server (hosted in local for now)
CInstance::TRefPtr _LastInstanceUnderPos;
//
CTool::TSmartPtr _CurrentTool;
static bool _ReloadWanted;
//
CDecal _HighlightDecal;
CDecalAnim _HighlightDecalAnim;
CDecal _SelectDecal;
CDecalAnim _SelectDecalAnim;
CDecalAnim _SelectingDecalAnim;
CDecal _PionneerDecal;
// alternative selection for huge element like particle systems, display a box on ground rather than
// the selection circle
CPrimRender _SelectBox;
CPrimRender _HighlightBox;
//
NLMISC::TTime _LastAutoSaveTime;
CDecalAnim _PionneerDecalAnim;
struct CSelectingDecal : public NLMISC::CRefCount
{
CDecal Decal;
sint64 EndDate;
NLMISC::CVector Pos;
float Scale;
};
std::vector<NLMISC::CSmartPtr<CSelectingDecal> > _SelectingDecals;
sint64 _DecalRefTime; // reference time for "decals" animation
//
bool _EnteredInSetSelectedInstance; // prevent recursive call from CEditor::setSelectedInstance
bool _Initialized;
bool _ForceDesktopReset[4];
// cache to avoid to reparse the ui : last modification date of ui files
std::map<std::string, uint32> _LastUIModificationDate;
TEntityCustomSelectBoxMap _EntityCustomSelectBoxMap;
/** system for local generation of name : for each kind of name, gives the locally allocated ids
* -> Used to generate name for instances, taking in account instance that have not been added to the scene yet.
* (that is, requestSetNode message has been sent, but server has not added object to the scenario yet)
*/
typedef std::set<uint> TNameSet;
typedef std::map<std::string, TNameSet> TGeneratedNameMap;
TGeneratedNameMap _LocalGeneratedNames;
// instance observers
typedef std::multimap<TInstanceId, IInstanceObserver::TRefPtr> TInstanceObserverMap;
TInstanceObserverMap _InstanceObservers;
typedef std::map<TInstanceObserverHandle, TInstanceObserverMap::iterator> TInstanceObserverHandleMap;
TInstanceObserverHandleMap _InstanceObserverHandles; // map each observer handle into an entry in the map
TInstanceObserverHandle _InstanceObserverHandleCounter; // current handle to generate when adding a new observer
TSeason _Season;
bool _FixedLighting;
bool _IsWaitingTPForSeasonChange;
bool _UpdatingScenario;
bool _WillTP; // true if a teleport should be expected after the scenario has been updated
// in this case, first season change is ignored, because it is done during the teleport
bool _ScenarioReceivedFlag; // for the wait screen ...
bool _TPReceivedFlag; // for the wait screen ...
bool _WaitScenarioScreenWanted; // lua requests that we display the 'wait scenario' screen
bool _WaitScenarioScreenActive; // the 'wait scenario' screen is being displayed
bool _EditionModeDisconnectedFlag;
CObject *_NewScenario; // new scenario that will be updated just after the wait screen is over
uint32 _NewScenarioInitialAct;// the start at which the user start an edition session (can be ~= from 1 after a test session)
bool _PostponeScenarioUpdated;
// Contextual selection
//
// We keep track of the last selected 'logic entity' (npc most of the time),
// and the last list of primitive through which its current activity sequence goes
// This way we can handle the following scenario :
// - create a npc
// - make him wander in zone A then zone B
// - select contextual visibility for primitive
// - click on npc : both zone shows
// - click on zone A : zone B dissapear -> strange
// By keeping the last list of contextual primitive, we ensure that contextual selection
// remains visible when one click on one element in the currently displayed sequence
CInstance::TRefPtr _LastContextualLogicEntity;
std::vector<CInstance::TRefPtr> _LastContextualPrims;
private:
/////////////////////////////
// NETWORK EVENTS HANDLING //
/////////////////////////////
void nodeErased(const std::string& instanceId, const std::string& attrName, sint32 position);
void nodeInserted(const std::string& instanceId, const std::string& attrName, sint32 position,
const std::string& key, CObject* value);
virtual void nodeSet(const std::string& instanceId, const std::string& attrName, CObject* value);
void nodeMoved(const std::string& instanceId, const std::string& attrName, sint32 position,
const std::string& destInstanceId, const std::string& destAttrName, sint32 destPosition);
void scenarioUpdated(CObject* highLevel, bool willTP, uint32 startingActIndex);
// send the needed events to tell that the attr at 'attrName' inside 'parentInstance' (possibly at position (indexInArray') has been modified
void onAttrModified(CInstance &parentInstance, const std::string &attrName, sint32 indexInArray = -1);
// send the needed events to tell that an object has been modified (& propagate to parents)
void onAttrModified(const CObject *value);
void onResetEditionMode();
void onEditionModeConnected( uint32 userSlotId, uint32 adventureId, CObject* highLevel, const std::string& versionName, bool willTP, uint32 initialActIndex);
void onAnimationModeConnected(const CClientMessageAdventureUserConnection& connected);
void onEditionModeDisconnected();
void onTestModeConnected();
// deconnect from test or play
void onTestModeDisconnected(TSessionId sessionId, uint32 lasAct, TScenarioSessionType sessionType);
///////////////////////////////////////////////////////
// EDITOR OBJECTS (instances) / CObjectTable mapping //
///////////////////////////////////////////////////////
public:
// Get a CObjectTable from its id
const CObjectTable *getObjectTableFromId(const TInstanceId &id) const;
private:
// Erase the current Scenario (and block outgoing message)
void eraseScenario();
void onErase(CObject *object);
void onErase(CObject *object, bool &foundInBase, std::string &nameInParent);
/** Create a new CInstance for the given CObject
* - Displayer are attached
* - Object is inserted in the object map
* - 'onCreate' msg is sent
* 'obj' must be a table or it fails
*/
void createNewInstanceForObjectTable(const CObject *obj);
void createNewInstanceForObjectTableInternal(const CObject *obj);
void onPostCreate(const CObject *obj);
void waitScenarioScreen();
public:
/** private : retrieve lua 'User' table attached to an object (a read / write table)
* The table is pushed on stack
*/
static void getLuaUserTableFromObject(CLuaState &ls, const CObjectTable &table);
private:
//////////////////////////////////////
// EDITOR FUNCTIONS EXPORTED TO LUA //
//////////////////////////////////////
static int luaGetVisualPropertiesFromInstanceId(CLuaState &ls);
static int luaGetSelectedInstanceId(CLuaState &ls);
static int luaGetSelectedInstance(CLuaState &ls);
static int luaSetSelectedInstanceId(CLuaState &ls);
static int luaSetCurrentTool(CLuaState &ls);
static int luaGetCurrentTool(CLuaState &ls);
static int luaGetInstanceFromId(CLuaState &ls);
static int luaDisplayContextMenu(CLuaState &ls);
static int luaConnectAsCreator(CLuaState &ls);
static int luaDofile(CLuaState &ls); // equivalent of the 'dofile' lua function, but with more info output
static int luaTryFile(CLuaState &ls); // same as try file, but do not throw an exception on error, just return an error message and return false
static int luaSetEntityCustomSelectBox(CLuaState &ls);
static int luaGetEntityCustomSelectBox(CLuaState &ls);
static int luaChoosePos(CLuaState &ls);
static int luaSnapPosToGround(CLuaState &ls); // takes x, y, z as parameters and return the snapped position
static int luaGetUserEntityPosition(CLuaState &ls);
static int luaGetUserEntityFront(CLuaState &ls);
static int luaRequestSetLocalNode(CLuaState &ls);
static int luaRequestCommitLocalNode(CLuaState &ls);
static int luaRequestRollbackLocalNode(CLuaState &ls);
static int luaSetCurrentActFromId(CLuaState &ls);
static int luaGetCurrentAct(CLuaState &ls);
static int luaAddInstanceObserver(CLuaState &ls); // param 1 = instance id of the instance to observe
// param 2 = table of an object that will receive notifications. The table should contain methods
// with the same names than those found in 'IInstanceObserver' (plus parameters are the same)
// the method returns a handle for future deletion
static int luaRemoveInstanceObserver(CLuaState &ls); // remove an observer that was previously added by a 'addInstanceObserver'
// param 1 = the handle returned by 'addInstanceObserver' at the registration time
// returns a reference to the observer that was registered
static int luaGenInstanceName(CLuaState &ls); // calls CEditor::genInstanceName, same parameters
// NB : return has type 'ucstring', so may need to call :toUtf8() in the lua script
static int luaIsPostFixedByNumber(CLuaState &ls);
static int luaIsClearingContent(CLuaState &ls);
static int luaSetCookie(CLuaState &ls); // same than CEditor::setCookie
static int luaIsCreature(CLuaState &ls);
static int luaSetEditorSeason(CLuaState &ls); // set the weather to display when editing
static int luaSetFixedLighting(CLuaState &ls);
static int luaGetFixedLighting(CLuaState &ls);
static int luaSetPlotItemInfos(CLuaState &ls);
static int luaIsCurrentSelectionPlayer(CLuaState &ls);
static int luaFindEmptyPlace(CLuaState &ls);
static int luaIsInIslandRect(CLuaState &ls); // test if pos is in the current island (not necessarily a valid pos, but inside the island rect)
// takes x, y as entry, returns true on success
static int luaGetCurrentIslandName(CLuaState &ls);
static int luaWaitScenarioScreen(CLuaState &ls); // display the wait screen after the scenario creation has been launched
static int luaIsScenarioUpdating(CLuaState &ls);
// undo / redo possible ?
static int luaCanUndo(CLuaState &ls);
static int luaCanRedo(CLuaState &ls);
// return the name of the editer
static sint luaGetUserEntityName(CLuaState &ls);
static int luaGetStartingAnimationFilename(CLuaState &ls);
static int luaKickCharacter(CLuaState &ls);
static int luaUnkickCharacter(CLuaState &ls);
static int luaTeleportToCharacter(CLuaState &ls);
static int luaEnumInstances(CLuaState &ls);
void connect();
// remove all object from the entity manager (but the player)
static void removeAllEntitySlots();
////////////////
// PLOT ITEMS //
////////////////
public:
static uint getMaxNumPlotItems();
static CCDBNodeLeaf *getPlotItemSheetDBLeaf(uint index);
static bool getIsStartingScenario() { return _IsStartingScenario; }
bool isClearingContent() const { return _ClearingContent; }
private:
void initPlotItems();
void initReferencePlotItems();
static void initDummyPlotItems();
void resetPlotItems();
static void setReferencePlotItemSheet(uint index, uint32 sheetId);
static CCDBNodeLeaf *getRefPlotItemSheetDBLeaf(uint index);
//////////
// MISC //
//////////
void setMaxVisibleEntityExceededFlag(bool on);
void backupRequestCommands();
void restoreRequestCommands();
void setForceDesktopReset(bool force);
void setUIMode(uint8 mode);
bool loadUIConfig(const std::string &prefix);
void loadStandardUI();
void saveUIConfig();
void hideRingWindows();
std::string getUIPrefix(TMode mode) const;
void loadKeySet(const std::string &keySet);
static std::string getKeySetPrefix(TMode mode);
void saveCurrentKeySet();
void reloadUI(const char *filename);
void initHighlightDecal();
void updateDecalBlendRegion(CDecal &decal, const NLMISC::CVector &pos);
void initPalette();
void initObjectProjectionMetatable();
void registerDisplayers();
void registerTools();
void registerLuaFunc();
// add a C++ method in the environement
void registerEnvMethod(const char *name, TLuaWrappedFunction func);
void registerEnvFunction(const char *name, TLuaWrappedFunction func);
// Initialisation of contextual cursor.
void initDecals();
void showDecal(const NLMISC::CVector2f &pos, float scale, CDecal &decal, const CDecalAnim &decalAnim);
void updatePrimitiveContextualVisibility();
void initClassInheritanceTable();
// contextual mouse handling
static void checkCursor();
// Forward click from contextual cursor to current tool
static void mouseClick(bool rightButton, bool dblClick);
// callback to reload the editor when one of the config files changed
static void reloadEditorCallback(const std::string &filename);
// update the display of decals created when the player select an instance in the scene
void updateSelectingDecals();
// display of highlight or select box (for selection of huge objects)
// the CDecalAnim is used to mimic the color cycle seen when standard selection circle is displayed
// (no CPrimRenderAnim for now)
void showPrimRender(CPrimRender &dest, const NLMISC::CAABBox &localBox, const NLMISC::CMatrix &worldMat, const CDecalAnim &refDecalAnim);
CLuaObject _OldLuaRequestInsertNode;
CLuaObject _OldLuaRequestInsertGhostNode;
CLuaObject _OldLuaRequestSetNode;
CLuaObject _OldLuaRequestEraseNode;
CLuaObject _OldLuaRequestMoveNode;
bool _ClearingContent;
std::string _ConnexionMsg;
static std::string _ScenarioToLoadWhenEntreringIntoAnimation;
static bool _IsStartingScenario; // the scenario is an animation scenario launch from the ring access point
//bool _ModeEnabled[ModeCount];
public:
// private method
CInstance *getInstanceFromObject(const CObject *obj) const; // nb : only table have associated instance
void connectAsCreator();
// notify obervers of an instance that it has been created
void notifyInstanceObserversOfCreation(CInstance &inst);
// TMP for debug : dump missing collisions in the log
void checkMissingCollisions();
// trigger an instance observers for the given instance id. A copy of the observer list is made, thus
// allowing for safe removal of observer when they are triggered
struct IObserverAction
{
virtual ~IObserverAction() { }
virtual void doAction(IInstanceObserver &obs) = 0;
};
void triggerInstanceObserver(const TInstanceId &id, IObserverAction &action);
static void setStartingAnimationFilename(const std::string& filename);
// for CInstance usage : allows to keep a list of instances sorted by their display name
void registerInstanceDispName(const ucstring &displayName, CInstance *inst);
void unregisterInstanceDispName(CInstance *inst);
};
// shortcut function to get the editor
inline CEditor &getEditor() { return CEditor::getInstance(); }
// test whether editor is currently enabled and is in EDITION mode. (In this mode, selection is managed by the editor rather than by the entity manager)
bool isEditionCurrent();
/** helper to create a class from the registry
* \TODO a true factory class!
* \TODO or put this in a better place (NLMISC ? )
*/
template <class T>
T *createObjectFromClassName(const std::string &className)
{
if (className.empty()) return NULL;
try
{
NLMISC::IClassable *obj = NLMISC::CClassRegistry::create(className);
if (!obj)
{
nlwarning("Couldn't create object of class %s", className.c_str());
return NULL;
}
T *inst = dynamic_cast<T *>(obj);
if (!inst)
{
nlwarning("<R2::createObjectFromClassName> class %s found in the registry, but does not match the expected class.",
obj->getClassName().c_str());
delete obj;
return NULL;
}
return inst;
}
catch(const NLMISC::ERegistry &)
{
return NULL;
}
}
extern bool ResetWanted;
extern bool ResetScenarioWanted;
extern bool ReloadScenarioWanted;
extern bool ConnectionWanted;
//
// helper : get a NLMISC::CVector from a DMC::CObject
NLMISC::CVector getVector(const CObject *obj);
NLMISC::CVectorD getVectorD(const CObject *obj);
//helper : build a CObject from a NLMISC::CVectorD
CObject *buildVector(const NLMISC::CVectorD &v, const std::string &instanceId = "");
// DMC helpers:
std::string getString(const CObject *obj, const std::string &attrName);
double getNumber(const CObject *obj, const std::string &attrName);
const CObject *getObject(const CObject *obj, const std::string &attrName);
/** Helper : read an enum from a lua string, printing necessary error if no match is found.
* nil found -> no op
*/
template <class T>
void enumFromLua(const CLuaObject &value,
std::pair<std::string, T> *enumTable,
uint numEnum,
T &dest,
const std::string &errorMsgPrefix
)
{
if (value.isNil())
{
return;
}
if (!value.isString())
{
nlwarning("%s : String expected when reading '%s'", errorMsgPrefix.c_str(), value.getId().c_str());
return;
}
for(uint k = 0; k < numEnum; ++k)
{
if (value.toString() == enumTable[k].first)
{
dest = enumTable[k].second;
return;
}
}
nlwarning("%s : Unknown enum %s read from object %s", errorMsgPrefix.c_str(),
value.toString().c_str(),
value.getId().c_str());
}
} // R2
bool IsInRingMode();
#endif