// Ryzom - MMORPG Framework // Copyright (C) 2010 Winch Gate Property Limited // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero General Public License as // published by the Free Software Foundation, either version 3 of the // License, or (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . #include "stdpch.h" #include "script_compiler.h" #include "ai_grp_npc.h" #include "group_profile.h" #include "ai_generic_fight.h" #include "server_share/msg_brick_service.h" #include "continent_inline.h" #include "dyn_grp_inline.h" #include "ai_script_data_manager.h" #include "ai_logic_action.h" using std::string; using std::vector; using namespace NLMISC; using namespace AIVM; using namespace AICOMP; using namespace AITYPES; using namespace RYAI_MAP_CRUNCH; //---------------------------------------------------------------------------- /** @page code @subsection setEvent_f_ Triggers a user event. Arguments: f(EventId) -> @param EventId is the user event id to be triggered @code ()setEvent(0); // Triggers the event 0 @endcode */ // CStateInstance void setEvent_f_(CStateInstance* entity, CScriptStack& stack) { size_t const eventIndex = (size_t)(float)stack.top(); stack.pop(); if (!entity) { nlwarning("sendEvent failed"); return; } nlassert(eventIndex<10); if (IsRingShard.get()) { if ( entity->isUserEventBlocked( eventIndex) ) { return; } // Do not allow uservent recursion on ring shard entity->blockUserEvent(eventIndex); } entity->processStateEvent(entity->getPersistentStateInstance()->getEventContainer().EventUserEvent[eventIndex]); if (IsRingShard.get()) { entity->unblockUserEvent(eventIndex); } } //---------------------------------------------------------------------------- /** @page code @subsection setTimer_ff_ Sets a timer delay and start it. Arguments: f(DeltaTime), f(TimerId) -> @param[in] DeltaTime is the time (in ticks) before the timer event is triggered @param[in] TimerId is the timer ID @code ()setTimer(200, 0); // Sets the timer t0 to 200 ticks @endcode */ // CStateInstance void setTimer_ff_(CStateInstance* si, CScriptStack& stack) { size_t const timerId = (int)(float)stack.top(); stack.pop(); size_t const deltaTime = (int)(float)stack.top(); stack.pop(); if (!si) { nlwarning("setTimer failed"); return; } if (timerId<0 || timerId>3) { string errStr("critical, native function setTimer used with out of bound parameter (0-3) equals to "); errStr += toString(timerId); nlwarning(errStr.c_str()); nlassert(false); } si->timerUser(timerId).set(deltaTime); } //---------------------------------------------------------------------------- /** @page code @subsection timerSetRyzomDaytime_fff_ Set a timer at a specified hour in Ryzom time and start it. Arguments: f(TimerId), f(Hour), f(Minute) -> @param[in] TimerId is the timer ID @param[in] Hour is the hour we want the timer to trigger @param[in] Minute is the minute we want the timer to trigger @code ()timerSetRyzomDaytime(0, 13, 45); // create a timer at 13h45 (Ryzom time) @endcode */ // CStateInstance void timerSetRyzomDaytime_fff_(CStateInstance* si, CScriptStack& stack) { float const minute = (float)stack.top(); stack.pop(); float const hour = (float)stack.top(); stack.pop(); size_t const timerId = (int)(float)stack.top(); stack.pop(); if ( !(0 <= minute && minute <=59 && 0 <= hour && hour <= 23) ) { return; } float datime = hour + (minute * 100.f)/(60.f*100.f); float currentTime = CTimeInterface::getRyzomTime().getRyzomTime(); if (datime < currentTime) { // advance to next day currentTime -= 24; } float deltaTime = datime - currentTime; // convert to ticks uint32 timeTicks = uint32(deltaTime * RYZOM_HOURS_IN_TICKS); if (!si) { nlwarning("timerSetRyzomDaytime failed"); return; } if (timerId<0 || timerId>3) { string errStr("critical, native function timerSetRyzomDaytime used with out of bound parameter (0-3) equals to "); errStr += toString(timerId); nlwarning(errStr.c_str()); return; } si->timerUser(timerId).set(timeTicks); } //---------------------------------------------------------------------------- /** @page code @subsection timerIsEnabled_f_f Test if a timer is enabled. Arguments: f(TimerId) -> f(IsEnabled) @param[in] TimerId is the timer ID @param[out] IsEnabled 1 if the timer is enabled @code (isEnabled)timerIsEnabled(0); // test if the timer t0 is enabled and put the ret in isEnabled variable @endcode */ // CStateInstance void timerIsEnabled_f_f(CStateInstance* si, CScriptStack& stack) { size_t const timerId = (int)(float)stack.top(); stack.pop(); if (!si) { nlwarning("timerIsEnabled failed"); return; } if (timerId<0 || timerId>3) { string errStr("critical, native function timerIsEnabled used with out of bound parameter (0-3) equals to "); errStr += toString(timerId); nlwarning(errStr.c_str()); return; } float isEnabled = si->timerUser(timerId).isEnabled(); stack.push(isEnabled); } //---------------------------------------------------------------------------- /** @page code @subsection timerIsSuspended_f_f Test if a timer is suspended. Arguments: f(TimerId) -> f(IsSuspended) @param[in] TimerId is the timer ID @param[out] IsSuspended 1 if the timer is suspended @code (isSuspended)timerIsSuspended(0); // test if the timer t0 is suspended and put the ret in isSuspended variable @endcode */ // CStateInstance void timerIsSuspended_f_f(CStateInstance* si, CScriptStack& stack) { size_t const timerId = (int)(float)stack.top(); stack.pop(); if (!si) { nlwarning("timerIsSuspended failed"); return; } if (timerId<0 || timerId>3) { string errStr("critical, native function timerIsSuspended used with out of bound parameter (0-3) equals to "); errStr += toString(timerId); nlwarning(errStr.c_str()); return; } float isSuspended = si->timerUser(timerId).isSuspended(); stack.push(isSuspended); } //---------------------------------------------------------------------------- /** @page code @subsection timerSuspend_f_ Suspend a timer. Arguments: f(TimerId) -> @param[in] TimerId is the timer ID @code ()timerSuspend(0); // suspend the timer t0 @endcode */ // CStateInstance void timerSuspend_f_(CStateInstance* si, CScriptStack& stack) { size_t const timerId = (int)(float)stack.top(); stack.pop(); if (!si) { nlwarning("timerSuspend failed"); return; } if (timerId<0 || timerId>3) { string errStr("critical, native function timerSuspend used with out of bound parameter (0-3) equals to "); errStr += toString(timerId); nlwarning(errStr.c_str()); return; } si->timerUser(timerId).suspend(); } //---------------------------------------------------------------------------- /** @page code @subsection timerDisable_f_ Disable a timer. Arguments: f(TimerId) -> @param[in] TimerId is the timer ID @code ()timerDisable(0); // disable the timer t0 @endcode */ // CStateInstance void timerDisable_f_(CStateInstance* si, CScriptStack& stack) { size_t const timerId = (int)(float)stack.top(); stack.pop(); if (!si) { nlwarning("timerDisable failed"); return; } if (timerId<0 || timerId>3) { string errStr("critical, native function timerDisable used with out of bound parameter (0-3) equals to "); errStr += toString(timerId); nlwarning(errStr.c_str()); return; } si->timerUser(timerId).disable(); } //---------------------------------------------------------------------------- /** @page code @subsection timerResume_f_ Resume a timer. Arguments: f(TimerId) -> @param[in] TimerId is the timer ID @code ()timerResume(0); // resume the timer t0 @endcode */ // CStateInstance void timerResume_f_(CStateInstance* si, CScriptStack& stack) { size_t const timerId = (int)(float)stack.top(); stack.pop(); if (!si) { nlwarning("timerResume failed"); return; } if (timerId<0 || timerId>3) { string errStr("critical, native function timerResume used with out of bound parameter (0-3) equals to "); errStr += toString(timerId); nlwarning(errStr.c_str()); return; } si->timerUser(timerId).resume(); } //---------------------------------------------------------------------------- /** @page code @subsection timerAdd_ff_ Add a delta time to a timer. Arguments: f(TimerId), f(DeltaTime) -> @param[in] TimerId is the timer ID @param[in] DeltaTime is the delta time in ticks @code ()timerAdd(0, 41); // add 41 ticks to the next trigger ()timerAdd(0, -10); // remove 10 ticks to the next trigger @endcode */ // CStateInstance void timerAdd_ff_(CStateInstance* si, CScriptStack& stack) { float const dt = (float)stack.top(); stack.pop(); size_t const timerId = (int)(float)stack.top(); stack.pop(); if (!si) { nlwarning("timerAdd failed"); return; } if (timerId<0 || timerId>3) { string errStr("critical, native function timerAdd used with out of bound parameter (0-3) equals to "); errStr += toString(timerId); nlwarning(errStr.c_str()); return; } if (dt > 0) { uint32 dt2 = static_cast(dt); si->timerUser(timerId).add(dt2); } else { uint32 dt2 = static_cast(-dt); si->timerUser(timerId).sub(dt2); } } //---------------------------------------------------------------------------- /** @page code @subsection dssStartAct_ff_ Request the start of a dss Act. Arguments: f(SessionId), f(ActId)-> @param[in] SessionId is the id session @param[in] ActId is the id of the act (0 = permanent content) @code ()dssStartAct(42,2); // Start the 2ond act of the session 42 @endcode */ // CStateInstance void dssStartAct_ff_(CStateInstance* si, CScriptStack& stack) { float const actId = (float)stack.top(); stack.pop(); float const sessionId= (float)stack.top(); stack.pop(); if ( !(0 <= actId && 0 <= sessionId) ) { return; } CAILogicActionDssStartActHelper::dssStartAct(TSessionId(static_cast(sessionId)), static_cast(actId)); } //---------------------------------------------------------------------------- /** @page code @subsection postNextState_s_ Triggers a state change. Arguments: s(StateName) -> @param[in] StateName is the name of the next state @code ()postNextState("state_invasion_2"); // Post the next state named 'state_invasion_2' if found in this state machine ()postNextState($name_var); // Post the next state named by name_var if found in this state machine @endcode */ // CStateInstance void postNextState_s_(CStateInstance* si, CScriptStack& stack) { string NextStateName = stack.top(); stack.pop(); if (!si) { nlwarning("postNextState failed!"); return; } CAIState* const state = si->getPersistentStateInstance()->getEventContainer().cstStates().getChildByName(NextStateName); if (!state) { string errStr; if (si->getGroup()) errStr="state "+NextStateName+" not found for "+si->getGroup()->getName(); else errStr="state "+NextStateName+" not found for unnamed group"; nlwarning(errStr.c_str()); return; } si->setNextState(state); } //---------------------------------------------------------------------------- /** @page code @subsection import_s_ Imports (executes in current context as a script function) a script defined in a script rep. Arguments: s(libName) -> @param[in] libName is the library name @code ()import("script_boss"); @endcode */ // CStateInstance void import_s_(CStateInstance* entity, CScriptStack& stack) { string LibName = stack.top(); stack.pop(); CSmartPtr const& codePtr = AIVM::CLibrary::getInstance().getLib(LibName); if (codePtr!=NULL) entity->interpretCode(NULL, codePtr); else nlwarning("unknown library %s", LibName.c_str()); } //---------------------------------------------------------------------------- /** @page code @subsection setNelVar_sf_ Sets the content of a NeL Variable. The variable is created if it doesn't exist. Arguments: s(varId),f(value) -> @param[in] varId is a the name of the variable to set @param[in] value is a the value to set @code ()setNelVar("BotCount", 32); @endcode */ // CStateInstance void setNelVar_sf_(CStateInstance* entity, CScriptStack& stack) { float value = (float)stack.top(); stack.pop(); std::string varId = (std::string)stack.top(); stack.pop(); entity->setNelVar(varId, value); } void setGlobalNelVar_sf_(CStateInstance* entity, CScriptStack& stack) { float value = (float)stack.top(); stack.pop(); std::string varId = (std::string)stack.top(); stack.pop(); CStateInstance::setGlobalNelVar(varId, value); } //---------------------------------------------------------------------------- /** @page code @subsection getNelVar_s_f Returns the content of a NeL Variable. The variable is created if it doesn't exist. Arguments: s(varId) -> f(value) @param[in] varId is a the name of the variable to set @param[out] value is a the value of the variable @code (botCount)getNelVar("BotCount"); @endcode */ // CStateInstance void getNelVar_s_f(CStateInstance* entity, CScriptStack& stack) { std::string varId = (std::string)stack.top(); stack.top() = entity->getNelVar(varId); } //---------------------------------------------------------------------------- /** @page code @subsection delNelVar_ss_ Detetes a NeL Variable. Passed value is used to determine the type of the variable. Content of that value is ignored. Arguments: s(varId) -> @param[in] varId is a the name of the variable to delete @param[in] value is a a value of the same type of the variable @code ()delNelVar("BotCount", 0); @endcode */ // CStateInstance void delNelVar_sf_(CStateInstance* entity, CScriptStack& stack) { stack.pop(); std::string varId = (std::string)stack.top(); stack.pop(); entity->delNelVar(varId); } //---------------------------------------------------------------------------- /** @page code @subsection setNelVar_ss_ Sets the content of a NeL Variable. The variable is created if it doesn't exist. Arguments: s(varId),s(value) -> @param[in] varId is a the name of the variable to set @param[in] value is a the value to set @code ()setNelVar("BotFamily", "the_killers"); @endcode */ // CStateInstance void setNelVar_ss_(CStateInstance* entity, CScriptStack& stack) { std::string value = (std::string)stack.top(); stack.pop(); std::string varId = (std::string)stack.top(); stack.pop(); entity->setStrNelVar(varId, value); } void setGlobalNelVar_ss_(CStateInstance* entity, CScriptStack& stack) { std::string value = (std::string)stack.top(); stack.pop(); std::string varId = (std::string)stack.top(); stack.pop(); CStateInstance::setGlobalNelVar(varId, value); } //---------------------------------------------------------------------------- /** @page code @subsection getNelVar_s_s Returns the content of a NeL Variable. The variable is created if it doesn't exist. Arguments: s(varId) -> s(value) @param[in] varId is a the name of the variable to set @param[out] value is a the value of the variable @code (botFamily)getNelVar("BotFamily"); @endcode */ // CStateInstance void getNelVar_s_s(CStateInstance* entity, CScriptStack& stack) { std::string varId = (std::string)stack.top(); stack.top() = entity->getStrNelVar(varId); } //---------------------------------------------------------------------------- /** @page code @subsection delNelVar_ss_ Detetes a NeL Variable. Passed value is used to determine the type of the variable. Content of that value is ignored. Arguments: s(varId) -> @param[in] varId is a the name of the variable to delete @param[in] value is a a value of the same type of the variable @code ()delNelVar("BotFamily", ""); @endcode */ // CStateInstance void delNelVar_ss_(CStateInstance* entity, CScriptStack& stack) { stack.pop(); std::string varId = (std::string)stack.top(); stack.pop(); entity->delStrNelVar(varId); } //---------------------------------------------------------------------------- /** @page code @subsection getStateName__s Returns the name of the current state. Arguments: -> s(StateName) @param[out] StateName is the name of the current state @code ($name)getStateName(); @endcode */ // CStateInstance void getStateName__s(CStateInstance* si, CScriptStack& stack) { string str; breakable { if (!si) break; CAIState const* const curState = si->getState(); if (!curState) break; str = curState->getName(); } stack.push(str); } ////////////////////////////////////////////////////////////////////////////// // Undocumented methods // ////////////////////////////////////////////////////////////////////////////// //---------------------------------------------------------------------------- // CStateInstance void loadFile_s_(CStateInstance* entity, CScriptStack& stack) { string fileName = stack.top(); stack.pop(); try { NLMISC::CIFile file(NLMISC::CPath::lookup(fileName)); vector lines; while (!file.eof()) { const size_t bufferSize = 4*1024; char buffer[bufferSize]; file.getline(buffer, bufferSize); lines.push_back(buffer); } // Compile the buffer CSmartPtr codePtr = CCompiler::getInstance().compileCode(lines, fileName); // Interpret the code for the group entity->interpretCode(NULL, codePtr); } catch (EPathNotFound e) { nlwarning("Path not found while loading AIS script %s", fileName.c_str()); } } /****************************************************************************/ std::map nfGetStateInstanceNativeFunctions() { std::map functions; #define REGISTER_NATIVE_FUNC(cont, func) cont.insert(std::make_pair(std::string(#func), &func)) REGISTER_NATIVE_FUNC(functions, setEvent_f_); REGISTER_NATIVE_FUNC(functions, setTimer_ff_); REGISTER_NATIVE_FUNC(functions, timerSetRyzomDaytime_fff_); REGISTER_NATIVE_FUNC(functions, timerDisable_f_); REGISTER_NATIVE_FUNC(functions, timerSuspend_f_); REGISTER_NATIVE_FUNC(functions, timerResume_f_); REGISTER_NATIVE_FUNC(functions, timerAdd_ff_); REGISTER_NATIVE_FUNC(functions, timerIsEnabled_f_f); REGISTER_NATIVE_FUNC(functions, timerIsSuspended_f_f); REGISTER_NATIVE_FUNC(functions, dssStartAct_ff_); REGISTER_NATIVE_FUNC(functions, postNextState_s_); REGISTER_NATIVE_FUNC(functions, import_s_); REGISTER_NATIVE_FUNC(functions, setNelVar_sf_); REGISTER_NATIVE_FUNC(functions, getNelVar_s_f); REGISTER_NATIVE_FUNC(functions, delNelVar_sf_); REGISTER_NATIVE_FUNC(functions, setNelVar_ss_); REGISTER_NATIVE_FUNC(functions, getNelVar_s_s); REGISTER_NATIVE_FUNC(functions, delNelVar_ss_); REGISTER_NATIVE_FUNC(functions, setGlobalNelVar_sf_); REGISTER_NATIVE_FUNC(functions, setGlobalNelVar_ss_); REGISTER_NATIVE_FUNC(functions, getStateName__s); REGISTER_NATIVE_FUNC(functions, loadFile_s_); #undef REGISTER_NATIVE_FUNC return functions; }