// 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_player.h" #include "ai_script_data_manager.h" #include "ais_control.h" using std::string; using std::vector; using namespace NLMISC; using namespace AIVM; using namespace AICOMP; using namespace AITYPES; using namespace RYAI_MAP_CRUNCH; // a big bad global var ! extern CAIEntityPhysical *TempSpeaker; extern CBotPlayer *TempPlayer; /****************************************************************************/ /* CGroupNpc methods */ /****************************************************************************/ //---------------------------------------------------------------------------- // setfactionProp /** @page code @subsection setFactionProp_ss_ Set a property of the faction profile. Valid values for Property are: - faction - ennemyFaction - friendFaction There are special faction that are automatically attributed to entities. Special factions are: - Player - Predator - Famous where faction is the name of a Ryzom faction with uppercase initials and without underscores (ie: tribe_beachcombers -> FamousTribeBeachcombers), it represents players with a positive fame with the specified faction - outpost:: where id is the outpost alias as an int, and side is either attacker ou defender (these faction may see their format change soon and shouldn't be used without prior discussion with the responsible coder), it represents players and squads fighting in an outpost conflict Arguments: s(Property),s(Content) -> @param Property is the property to modify @param Content is a '|' seperated list of faction names @code ()setFactionProp("ennemyFaction", "Player"); @endcode */ // CGroupNpc void setFactionProp_ss_(CStateInstance* entity, CScriptStack& stack) { std::string params = stack.top(); // ToChange stack.pop(); TStringId const factionType = CStringMapper::map(stack.top()); stack.pop(); CGroupNpc* const grpNpc = dynamic_cast(entity->getGroup()); if (!grpNpc) { nlwarning("setFactionProp on a non Npc Group, doesnt work"); return; } static TStringId factionStrId = CStringMapper::map("faction"); static TStringId ennemyFactionStrId = CStringMapper::map("ennemyFaction"); static TStringId friendFactionStrId = CStringMapper::map("friendFaction"); breakable { CPropertySetWithExtraList tmpSet; CStringSeparator const sep(params,"|"); while (sep.hasNext()) tmpSet.addProperty(AITYPES::CPropertyId::create(sep.get())); if (factionType==factionStrId) { grpNpc->faction() = tmpSet; break; } if (factionType==ennemyFactionStrId) { grpNpc->ennemyFaction() = tmpSet; break; } if (factionType==friendFactionStrId) { grpNpc->friendFaction() = tmpSet; break; } nlwarning("'%s' is not a correct faction name", CStringMapper::unmap(factionType).c_str()); return; } } void setOupostMode_ss_(CStateInstance* entity, CScriptStack& stack) { std::string sideStr = stack.top(); stack.pop(); std::string aliasStr = stack.top(); CGroupNpc* const npcGroup = dynamic_cast(entity->getGroup()); if ( ! npcGroup) { nlwarning("setOutpostMode on a non Npc Group, doesnt work"); return; } OUTPOSTENUMS::TPVPSide side; if (sideStr == "attacker") { side = OUTPOSTENUMS::OutpostAttacker; } else if (sideStr == "owner") { side = OUTPOSTENUMS::OutpostOwner; } else { nlwarning("setOutpostMode: invalid side"); } npcGroup->setOutpostSide(side); npcGroup->setOutpostFactions(side); FOREACH(botIt, CCont, npcGroup->bots()) { CBot* bot = *botIt; CBotNpc* botNpc = NLMISC::safe_cast(bot); if (botNpc) { CSpawnBotNpc* spawnBotNpc = botNpc->getSpawn(); if (spawnBotNpc) { spawnBotNpc->setOutpostSide(side); spawnBotNpc->setOutpostAlias(LigoConfig.aliasFromString(aliasStr)); } } } if (npcGroup->isSpawned()) npcGroup->getSpawnObj()->sendInfoToEGS(); } //---------------------------------------------------------------------------- /** @page code @subsection moveToZone_ss_ Moves the current group from one zone to another. Arguments: s(From), s(To) -> @param[in] From is a zone name @param[in] To is a zone name @code ()moveToZone($zone1, $zone2); // Move the current group from $zone1 to $zone2 @endcode */ // Spawned CGroupNpc not in a family behaviour void moveToZone_ss_(CStateInstance* entity, CScriptStack& stack) { TStringId const zoneDest = CStringMapper::map(stack.top()); stack.pop(); TStringId const zoneSrc = CStringMapper::map(stack.top()); stack.pop(); IManagerParent* const managerParent = entity->getGroup()->getOwner()->getOwner(); CAIInstance* const aiInstance = dynamic_cast(managerParent); if (!aiInstance) return; if (!entity) { nlwarning("moveToZone failed: no state instance!"); nlwarning(" - from %s", CStringMapper::unmap(zoneSrc).c_str()); nlwarning(" - to %s", CStringMapper::unmap(zoneDest).c_str()); return; } CGroupNpc* group = dynamic_cast(entity->getGroup()); if (!group) { nlwarning("moveToZone failed: no NPC group!"); nlwarning(" - from %s", CStringMapper::unmap(zoneSrc).c_str()); nlwarning(" - to %s", CStringMapper::unmap(zoneDest).c_str()); return; } CSpawnGroupNpc* spawnGroup = group->getSpawnObj(); if (!spawnGroup) { nlwarning("moveToZone failed: no spawned group!"); nlwarning(" - from %s", CStringMapper::unmap(zoneSrc).c_str()); nlwarning(" - to %s", CStringMapper::unmap(zoneDest).c_str()); return; } CNpcZone const* const destZone = aiInstance->getZone(zoneDest); CNpcZone const* const srcZone = aiInstance->getZone(zoneSrc); if (!destZone) { nlwarning("moveToZone: getZone destination failed!"); nlwarning(" - from %s", CStringMapper::unmap(zoneSrc).c_str()); nlwarning(" - to %s", CStringMapper::unmap(zoneDest).c_str()); } if (!srcZone) { nlwarning("moveToZone: getZone source failed!"); nlwarning(" - from %s", CStringMapper::unmap(zoneSrc).c_str()); nlwarning(" - to %s", CStringMapper::unmap(zoneDest).c_str()); } if (!destZone || !srcZone) return; if (destZone==srcZone) { nlwarning("Trying to find a path between two same zones in %s, aborting moveToZone", entity->getActiveState()->getAliasFullName().c_str()); nlwarning(" - from %s", CStringMapper::unmap(zoneSrc).c_str()); nlwarning(" - to %s", CStringMapper::unmap(zoneDest).c_str()); return; } spawnGroup->movingProfile().setAIProfile(new CGrpProfileDynFollowPath(spawnGroup, srcZone, destZone, AITYPES::CPropertySet())); } //---------------------------------------------------------------------------- /** @page code @subsection setActivity_s_ Changes the activity of the group. Valid activities are: - "no_change" - "escorted" - "guard" - "guard_escorted" - "normal" - "faction" - "faction_no_assist" - "bandit" Arguments: s(Activity) -> @param[in] Activity is an activity name the group will take @code ()setActivity("bandit"); // Gives a bandit activity to the group @endcode */ // Spawned CGroupNpc not in a family behaviour void setActivity_s_(CStateInstance* entity, CScriptStack& stack) { string activity = stack.top(); stack.pop(); IManagerParent* const managerParent = entity->getGroup()->getOwner()->getOwner(); CAIInstance* const aiInstance = dynamic_cast(managerParent); if (!aiInstance) return; if (!entity) { nlwarning("setActivity failed!"); return; } CGroupNpc* group = dynamic_cast(entity->getGroup()); if (!group) { nlwarning("setActivity failed: no NPC group"); return; } CSpawnGroupNpc* spawnGroup = group->getSpawnObj(); if (!spawnGroup) { nlwarning("setActivity failed: no spawned group"); return; } breakable { if (activity=="no_change") break; if (activity=="escorted") { spawnGroup->activityProfile().setAIProfile(new CGrpProfileEscorted(spawnGroup)); break; } if (activity=="guard") { spawnGroup->activityProfile().setAIProfile(new CGrpProfileGuard(spawnGroup)); break; } if (activity=="guard_escorted") { spawnGroup->activityProfile().setAIProfile(new CGrpProfileGuardEscorted(spawnGroup)); break; } if (activity=="normal") { spawnGroup->activityProfile().setAIProfile(new CGrpProfileNormal(spawnGroup)); break; } if (activity=="faction") { spawnGroup->activityProfile().setAIProfile(new CGrpProfileFaction(spawnGroup)); break; } if (activity=="faction_no_assist") { CGrpProfileFaction * grpPro = new CGrpProfileFaction(spawnGroup); grpPro->noAssist(); spawnGroup->activityProfile().setAIProfile(grpPro); break; } if (activity=="bandit") { spawnGroup->activityProfile().setAIProfile(new CGrpProfileBandit(spawnGroup)); break; } nlwarning("trying to set activity profile to an unknown profile name"); } } //---------------------------------------------------------------------------- /** @page code @subsection waitInZone_s_ Makes the group wander in the specified zone. It's useful to prevent the group from looping previous movement profile). Arguments: s(Zone) -> @param[in] Zone is a zone name @code ()waitInZone($zone); // Makes the group wander in $zone @endcode */ // Spawned CGroupNpc not in a family behaviour void waitInZone_s_(CStateInstance* entity, CScriptStack& stack) { std::string zoneName = stack.top(); TStringId const zoneDest = CStringMapper::map(zoneName); stack.pop(); if (!entity) { nlwarning("waitInZone failed!"); return; } IManagerParent* const managerParent = entity->getGroup()->getOwner()->getOwner(); CAIInstance* const aiInstance = dynamic_cast(managerParent); if (!aiInstance) { // :TODO: Check is that warning was a flood //nlwarning("waitInZone failed!"); return; } CGroupNpc* group = dynamic_cast(entity->getGroup()); if (!group) { nlwarning("waitInZone failed: no NPC group"); return; } CSpawnGroupNpc* spawnGroup = group->getSpawnObj(); if (!spawnGroup) { nlwarning("waitInZone failed: no spawned group"); return; } CNpcZone const* const destZone = aiInstance->getZone(zoneDest); if (!destZone) { nlwarning("waitInZone: getZone destination failed! (%s)", zoneName.c_str()); return; } spawnGroup->movingProfile().setAIProfile(new CGrpProfileWander(spawnGroup, destZone)); } //---------------------------------------------------------------------------- /** @page code @subsection stopMoving__ Makes the group stop moving and stand at its current position. Arguments: -> @code ()stopMoving(); // Makes the group stop moving @endcode */ // Spawned CGroupNpc not in a family behaviour void stopMoving__(CStateInstance* entity, CScriptStack& stack) { if (!entity) { nlwarning("stopMoving failed!"); return; } IManagerParent* const managerParent = entity->getGroup()->getOwner()->getOwner(); CAIInstance* const aiInstance = dynamic_cast(managerParent); if (!aiInstance) return; CGroupNpc* group = dynamic_cast(entity->getGroup()); if (!group) { nlwarning("stopMoving failed: no NPC group"); return; } CSpawnGroupNpc* spawnGroup = group->getSpawnObj(); if (!spawnGroup) { nlwarning("stopMoving failed: no spawned group"); return; } spawnGroup->movingProfile().setAIProfile(new CGrpProfileIdle(spawnGroup)); } //---------------------------------------------------------------------------- /** @page code @subsection startWander_f_ Set activity to wander in current pos: Arguments: f(Radius) -> @param[in] Radius dispersion of wander activity @code ()startWander(100); // Gives a wander activity to the group with dispersion of 100 @endcode */ // Spawned CGroupNpc not in a family behaviour void startWander_f_(CStateInstance* entity, CScriptStack& stack) { uint32 dispersionRadius = (uint32)(float&)stack.top(); stack.pop(); IManagerParent* const managerParent = entity->getGroup()->getOwner()->getOwner(); CAIInstance* const aiInstance = dynamic_cast(managerParent); if (!aiInstance) return; if (!entity) { nlwarning("setActivity failed!"); return; } CGroupNpc* group = dynamic_cast(entity->getGroup()); if (!group) { nlwarning("startWander failed: no NPC group"); return; } CSpawnGroupNpc* spawnGroup = group->getSpawnObj(); if (!spawnGroup) { nlwarning("startWander failed: no spawned group"); return; } CAIVector centerPos; if (!spawnGroup->calcCenterPos(centerPos)) // true if there's some bots in the group. { nlwarning("startWander failed: no center pos"); return; } NLMISC::CSmartPtr destZone = NLMISC::CSmartPtr(new CNpcZonePlaceNoPrim()); destZone->setPosAndRadius(AITYPES::vp_auto, CAIPos(centerPos, 0, 0), (uint32)(dispersionRadius*1000.)); spawnGroup->movingProfile().setAIProfile(new CGrpProfileWanderNoPrim(spawnGroup, destZone)); } //---------------------------------------------------------------------------- /** @page code @subsection startMoving_fff_ Set activity to wander in current pos: Arguments: f(Radius) -> @param[in] Radius dispersion of wander activity @code ()startMoving(100,-100,10); // Moves the group to 100,-100 with radius of 10 @endcode */ // Spawned CGroupNpc not in a family behaviour void startMoving_fff_(CStateInstance* entity, CScriptStack& stack) { uint32 dispersionRadius = (uint32)(float&)stack.top(); stack.pop(); float const y = (float&)stack.top(); stack.pop(); float const x = (float&)stack.top(); stack.pop(); IManagerParent* const managerParent = entity->getGroup()->getOwner()->getOwner(); CAIInstance* const aiInstance = dynamic_cast(managerParent); if (!aiInstance) return; if (!entity) { nlwarning("setActivity failed!"); return; } CGroupNpc* group = dynamic_cast(entity->getGroup()); if (!group) { nlwarning("setActivity failed: no NPC group"); return; } CSpawnGroupNpc* spawnGroup = group->getSpawnObj(); if (!spawnGroup) { nlwarning("setActivity failed: no spawned group"); return; } NLMISC::CSmartPtr destZone = NLMISC::CSmartPtr(new CNpcZonePlaceNoPrim()); destZone->setPosAndRadius(AITYPES::vp_auto, CAIPos(CAIVector(x, y), 0, 0), (uint32)(dispersionRadius*1000.)); spawnGroup->movingProfile().setAIProfile(new CGrpProfileWanderNoPrim(spawnGroup, destZone)); return; } //---------------------------------------------------------------------------- /** @page code @subsection followPlayer_sf_ Set activity to follow the given player Arguments: s(PlayerEid) f(Radius) -> @param[in] PlayerEid id of player to follow @param[in] Radius dispersion of wander activity @code ()followPlayer("(0x0002015bb4:01:88:88)",10); @endcode */ // Spawned CGroupNpc not in a family behaviour void followPlayer_sf_(CStateInstance* entity, CScriptStack& stack) { uint32 dispersionRadius = (uint32)(float&)stack.top(); stack.pop(); NLMISC::CEntityId playerId = NLMISC::CEntityId((std::string)stack.top()); IManagerParent* const managerParent = entity->getGroup()->getOwner()->getOwner(); CAIInstance* const aiInstance = dynamic_cast(managerParent); if (!aiInstance) return; if (!entity) { nlwarning("followPlayer failed!"); return; } CGroupNpc* group = dynamic_cast(entity->getGroup()); if (!group) { nlwarning("followPlayer failed: no NPC group"); return; } CSpawnGroupNpc* spawnGroup = group->getSpawnObj(); if (!spawnGroup) { nlwarning("followPlayer failed: no spawned group"); return; } if (playerId == CEntityId::Unknown) { nlwarning("followPlayer failed: unknown player"); DEBUG_STOP; return; } spawnGroup->movingProfile().setAIProfile(new CGrpProfileFollowPlayer(spawnGroup, TheDataset.getDataSetRow(playerId), dispersionRadius)); return; } //---------------------------------------------------------------------------- /** @page code @subsection wander__ Makes the group wander in the current npc state zone. Arguments: -> @code ()wander(); // Makes the group wander @endcode */ // Spawned CGroupNpc not in a family behaviour void wander__(CStateInstance* entity, CScriptStack& stack) { if (!entity) { nlwarning("wander failed!"); return; } IManagerParent* const managerParent = entity->getGroup()->getOwner()->getOwner(); CAIInstance* const aiInstance = dynamic_cast(managerParent); if (!aiInstance) return; CGroupNpc* group = dynamic_cast(entity->getGroup()); if (!group) { nlwarning("wander failed: no NPC group"); return; } CSpawnGroupNpc* spawnGroup = group->getSpawnObj(); if (!spawnGroup) { nlwarning("wander failed: no spawned group"); return; } spawnGroup->movingProfile().setAIProfile(new CGrpProfileWander(spawnGroup)); } //---------------------------------------------------------------------------- /** @page code @subsection setAttackable_f_ Sets the group as being attackable (or not) by bots and players. 0 means not attackable, 1 means attackable. Arguments: f(Attackable) -> @param[in] Attackable tells whether the group is attackable @code ()setAttackable(1); // Make the group attackable by players and bots @endcode */ // CGroupNpc void setAttackable_f_(CStateInstance* entity, CScriptStack& stack) { bool const attackable = (float&)stack.top()!=0.0f; stack.pop(); CGroup* group = entity->getGroup(); CGroupNpc* npcGroup = NLMISC::safe_cast(group); npcGroup->setPlayerAttackable(attackable); npcGroup->setBotAttackable(attackable); if (npcGroup->isSpawned()) npcGroup->getSpawnObj()->sendInfoToEGS(); } //---------------------------------------------------------------------------- /** @page code @subsection setPlayerAttackable_f_ Sets the group as being attackable (or not) by players. 0 means not attackable, 1 means attackable. Arguments: f(Attackable) -> @param[in] Attackable tells whether the group is attackable @code ()setPlayerAttackable(1); // Make the group attackable by players @endcode */ // CGroupNpc void setPlayerAttackable_f_(CStateInstance* entity, CScriptStack& stack) { bool attackable = (float&)stack.top()!=0.0f; stack.pop(); CGroup* group = entity->getGroup(); CGroupNpc* npcGroup = NLMISC::safe_cast(group); npcGroup->setPlayerAttackable(attackable); if (npcGroup->isSpawned()) npcGroup->getSpawnObj()->sendInfoToEGS(); } //---------------------------------------------------------------------------- // setBotAttackable_f_ // Arguments: f(Attackable) -> /** @page code @subsection setBotAttackable_f_ Sets the group as being attackable (or not) by bots. 0 means not attackable, 1 means attackable. Arguments: f(Attackable) -> @param[in] Attackable tells whether the group is attackable @code ()setBotAttackable(0); // Make the group not attackable by bots @endcode */ // CGroupNpc void setBotAttackable_f_(CStateInstance* entity, CScriptStack& stack) { bool attackable = (float&)stack.top()!=0.0f; stack.pop(); CGroup* group = entity->getGroup(); CGroupNpc* npcGroup = NLMISC::safe_cast(group); npcGroup->setBotAttackable(attackable); if (npcGroup->isSpawned()) npcGroup->getSpawnObj()->sendInfoToEGS(); } //---------------------------------------------------------------------------- // setFactionAttackableAbove_sff_ // Arguments: s(Faction),f(Threshold),f(Attackable) -> /** @page code @subsection setFactionAttackableAbove_sff_ Sets the group as being attackable (or not) by players having their fame matching the specified condition. If player fame for Faction is above specified Threshold the bot will be attackable by that player or not depending on Attackable parameter, 0 meaning not attackable, 1 meaning attackable. Note: Bots must not be player attackable for the faction to be taken into account. Arguments: s(Faction),f(Threshold),f(Attackable) -> @param[in] Faction tells the faction against which player fame will be tested @param[in] Threshold is the test threshold above which player will be able to attack @param[in] Attackable tells whether the group is attackable or not @code ()setFactionAttackableAbove("TribeNightTurners", 0, 1); // Make the group attackable by players with positive tribe_night_turners fame @endcode */ // CGroupNpc void setFactionAttackableAbove_sff_(CStateInstance* entity, CScriptStack& stack) { bool attackable = (float&)stack.top()!=0.0f; stack.pop(); sint32 threshold = (sint32)(float&)stack.top(); stack.pop(); std::string faction = (std::string&)stack.top(); stack.pop(); CGroup* group = entity->getGroup(); CGroupNpc* npcGroup = NLMISC::safe_cast(group); npcGroup->setFactionAttackableAbove(faction, threshold, attackable); if (npcGroup->isSpawned()) npcGroup->getSpawnObj()->sendInfoToEGS(); } //---------------------------------------------------------------------------- // setFactionAttackableBelow_sff_ // Arguments: s(Faction),f(Threshold),f(Attackable) -> /** @page code @subsection setFactionAttackableBelow_sff_ Sets the group as being attackable (or not) by players having their fame matching the specified condition. If player fame for Faction is below specified Threshold the bot will be attackable by that player or not depending on Attackable parameter, 0 meaning not attackable, 1 meaning attackable. Note: Bots must not be player attackable for the faction to be taken into account. Arguments: s(Faction),f(Threshold),f(Attackable) -> @param[in] Faction tells the faction against which player fame will be tested @param[in] Threshold is the test threshold below which player will be able to attack @param[in] Attackable tells whether the group is attackable or not @code ()setFactionAttackableBelow("TribeMatisianBorderGuards", 0, 1); // Make the group attackable by players with negative tribe_matisian_border_guards fame @endcode */ // CGroupNpc void setFactionAttackableBelow_sff_(CStateInstance* entity, CScriptStack& stack) { bool attackable = (float&)stack.top()!=0.0f; stack.pop(); sint32 threshold = (sint32)(float&)stack.top(); stack.pop(); std::string faction = (std::string&)stack.top(); stack.pop(); CGroup* group = entity->getGroup(); CGroupNpc* npcGroup = NLMISC::safe_cast(group); npcGroup->setFactionAttackableBelow(faction, threshold, attackable); if (npcGroup->isSpawned()) npcGroup->getSpawnObj()->sendInfoToEGS(); } //---------------------------------------------------------------------------- /** @page code @subsection addBotChat_s_ Add an entry in the botchat menu of every bot of the group. Specified text ID can be created dynamically with setSimplePhrase (@ref setSimplePhrase_ss_). Arguments: s(BotChat) -> @param[in] BotChat is a text ID @code ()addBotChat("menu:QUESTION:REPONSE"); @endcode */ // CGroupNpc void addBotChat_s_(CStateInstance* entity, CScriptStack& stack) { string const botChat = (string)stack.top(); stack.pop(); CGroup* group = entity->getGroup(); CGroupNpc* npcGroup = NLMISC::safe_cast(group); FOREACH(botIt, CCont, group->bots()) { CBot* bot = *botIt; CBotNpc* botNpc = NLMISC::safe_cast(bot); if (botNpc) { if (!botNpc->getChat()) botNpc->newChat(); botNpc->getChat()->add(botNpc->getAIInstance(), botChat); CSpawnBotNpc* spawnBotNpc = botNpc->getSpawn(); if (spawnBotNpc) { spawnBotNpc->setCurrentChatProfile(botNpc->getChat()); } } } if (npcGroup->isSpawned()) npcGroup->getSpawnObj()->sendInfoToEGS(); } //---------------------------------------------------------------------------- /** @page code @subsection clearBotChat__ Removes all entries from the botchat menu of every bot of the group. Arguments: -> @code ()clearBotChat(); @endcode */ // CGroupNpc void clearBotChat__(CStateInstance* entity, CScriptStack& stack) { CGroup* group = entity->getGroup(); CGroupNpc* npcGroup = NLMISC::safe_cast(group); FOREACH(botIt, CCont, group->bots()) { CBot* bot = *botIt; CBotNpc* botNpc = NLMISC::safe_cast(bot); if (botNpc) { if (!botNpc->getChat()) botNpc->newChat(); botNpc->getChat()->clear(); CSpawnBotNpc* spawnBotNpc = botNpc->getSpawn(); if (spawnBotNpc) { spawnBotNpc->setCurrentChatProfile(botNpc->getChat()); } } } if (npcGroup->isSpawned()) npcGroup->getSpawnObj()->sendInfoToEGS(); } //---------------------------------------------------------------------------- /** @page code @subsection ignoreOffensiveActions_f_ Make the bots of the group ignore offensive actions issued on them. Arguments: f(IgnoreOffensiveActions) -> @param[in] IgnoreOffensiveActions is tells whethere to ignore offensive actions (1) or not (0) @code ()ignoreOffensiveActions(1); @endcode */ // CGroup void ignoreOffensiveActions_f_(CStateInstance* entity, CScriptStack& stack) { bool const ignoreOffensiveActions = ((float)stack.top())!=0.f; stack.pop(); CGroup* group = entity->getGroup(); FOREACH(botIt, CCont, group->bots()) { CBot* bot = *botIt; if (bot) bot->setIgnoreOffensiveActions(ignoreOffensiveActions); } } //---------------------------------------------------------------------------- /** @page code @subsection setDespawnTime_f_ Sets the time before current group being despawned. Arguments: f(DespawnTime) -> @param[in] DespawnTime is the despawn time in ticks (-1 will set "pseudo-infinite") @code ()setDespawnTime(80); ()setDespawnTime(-1); @endcode */ // CGroupNpc void setDespawnTime_f_(CStateInstance* entity, CScriptStack& stack) { float time = stack.top(); stack.pop(); CGroup* group = entity->getGroup(); CGroupNpc* npcGroup = NLMISC::safe_cast(group); uint32 despawntime = (uint32)(sint32)time; // -1 => ~0 // TODO:kxu: fix CAITimer! if (despawntime > 0x7fffffff) despawntime = 0x7fffffff; // AI timers do not treat properly delta times greater than the max signed int npcGroup->despawnTime() = despawntime; } /** @page code @subsection despawnBotByAlias_s_ Despawn a specific Bot by its alias Arguments: f(alias), -> @param[in] alias is the alias of the bot @code ()despawnBotByAlias('(A:1000:10560)'); @endcode */ // CGroup void despawnBotByAlias_s_(CStateInstance* entity, CScriptStack& stack) { uint32 alias = LigoConfig.aliasFromString((string)stack.top()); stack.pop(); CGroup* group = entity->getGroup(); CGroupNpc* npcGroup = NLMISC::safe_cast(group); FOREACH(botIt, CCont, group->bots()) { CBot* bot = *botIt; if (bot->getAlias() != alias) { continue; } if (!bot->isSpawned()) return; if (bot->getRyzomType() == RYZOMID::npc) { CBotNpc* botNpc = NLMISC::safe_cast(bot); CSpawnBotNpc* spawnBotNpc = botNpc->getSpawn(); spawnBotNpc->spawnGrp().addBotToDespawnAndRespawnTime(&spawnBotNpc->getPersistent(), 1, spawnBotNpc->spawnGrp().getPersistent().respawnTime()); } } } //---------------------------------------------------------------------------- /** @page code @subsection setRespawnTime_f_ Sets the time in game cycles before current group being respawned. Arguments: f(RespawnTime) -> @param[in] RespawnTime is the respawn time in ticks (-1 will set "pseudo-infinite") @code ()setRespawnTime(80); ()setRespawnTime(-1); @endcode */ // CGroupNpc void setRespawnTime_f_(CStateInstance* entity, CScriptStack& stack) { float const time = stack.top(); stack.pop(); CGroup* group = entity->getGroup(); CGroupNpc* npcGroup = NLMISC::safe_cast(group); uint32 respawntime = (uint32)(sint32)time; // -1 => ~0 // TODO:kxu: fix CAITimer! if (respawntime > 0x7fffffff) respawntime = 0x7fffffff; // AI timers do not treat properly delta times greater than the max signed int npcGroup->respawnTime() = respawntime; } //---------------------------------------------------------------------------- /** @page code @subsection addHpUpTrigger_ff_ Registers a trigger on HP increases. Whenever the HP level of a bot upcross the threshold it triggers the specified user event. Several triggers can be registered on the same group, even with the same threshold and event. Arguments: f(threshold),f(user_event_n) -> @param[in] threshold is a HP threshold @param[in] user_event_n is the user event to trigger @code ()addHpUpTrigger(0.5, 4); @endcode */ // CGroupNpc void addHpUpTrigger_ff_(CStateInstance* entity, CScriptStack& stack) { int eventId = (int)(float)stack.top(); stack.pop(); float threshold = (float)stack.top(); stack.pop(); CGroupNpc* const grpNpc = dynamic_cast(entity->getGroup()); if (!grpNpc) { nlwarning("Trying to add a hp up trigger (%f) listener (user event %d) in a group which is not an NPC group.", threshold, eventId); return; } grpNpc->addHpUpTrigger(threshold, eventId); } //---------------------------------------------------------------------------- /** @page code @subsection delHpUpTrigger_ff_ Unregisters a trigger on HP increases. The same values used when registering the trigger must be passed. If several triggers were defined with the same parameters only one is removed. Arguments: f(threshold),f(user_event_n) -> @param[in] threshold is a HP threshold @param[in] user_event_n is the user event to trigger @code ()delHpUpTrigger(0.5, 4); @endcode */ // CGroupNpc void delHpUpTrigger_ff_(CStateInstance* entity, CScriptStack& stack) { int eventId = (int)(float)stack.top(); stack.pop(); float threshold = (float)stack.top(); stack.pop(); CGroupNpc* const grpNpc = dynamic_cast(entity->getGroup()); if (!grpNpc) { nlwarning("Trying to delete a hp up trigger (%f) listener (user event %d) in a group which is not an NPC group.", threshold, eventId); return; } grpNpc->delHpUpTrigger(threshold, eventId); } //---------------------------------------------------------------------------- /** @page code @subsection addHpDownTrigger_ff_ Registers a trigger on HP decreases. Whenever the HP level of a bot downcross the threshold it triggers the specified user event. Several triggers can be registered on the same group, even with the same threshold and event. Arguments: f(threshold),f(user_event_n) -> @param[in] threshold is a HP threshold @param[in] user_event_n is the user event to trigger @code ()addHpDownTrigger(0.5, 5); @endcode */ // CGroupNpc void addHpDownTrigger_ff_(CStateInstance* entity, CScriptStack& stack) { int eventId = (int)(float)stack.top(); stack.pop(); float threshold = (float)stack.top(); stack.pop(); CGroupNpc* const grpNpc = dynamic_cast(entity->getGroup()); if (!grpNpc) { nlwarning("Trying to add a hp down trigger (%f) listener (user event %d) in a group which is not an NPC group.", threshold, eventId); return; } grpNpc->addHpDownTrigger(threshold, eventId); } //---------------------------------------------------------------------------- /** @page code @subsection delHpDownTrigger_ff_ Unregisters a trigger on HP decreases. The same values used when registering the trigger must be passed. If several triggers were defined with the same parameters only one is removed. Arguments: f(threshold),f(user_event_n) -> @param[in] threshold is a HP threshold @param[in] user_event_n is the user event to trigger @code ()delHpDownTrigger(0.5, 5); @endcode */ // CGroupNpc void delHpDownTrigger_ff_(CStateInstance* entity, CScriptStack& stack) { int eventId = (int)(float)stack.top(); stack.pop(); float threshold = (float)stack.top(); stack.pop(); CGroupNpc* const grpNpc = dynamic_cast(entity->getGroup()); if (!grpNpc) { nlwarning("Trying to delete a hp down trigger (%f) listener (user event %d) in a group which is not an NPC group.", threshold, eventId); return; } grpNpc->delHpDownTrigger(threshold, eventId); } //---------------------------------------------------------------------------- /** @page code @subsection addHpUpTrigger_fs_ @sa @ref addHpUpTrigger_ff_ These triggers call a script function instead of trigger a user event. Arguments: f(threshold),s(callback) -> @param[in] threshold is a HP threshold @param[in] callback is the script callback to trigger @code ()addHpUpTrigger(0.5, "onHPIncrease"); @endcode */ // CGroupNpc void addHpUpTrigger_fs_(CStateInstance* entity, CScriptStack& stack) { std::string cbFunc = (std::string)stack.top(); stack.pop(); float threshold = (float)stack.top(); stack.pop(); CGroupNpc* const grpNpc = dynamic_cast(entity->getGroup()); if (!grpNpc) { nlwarning("Trying to add a hp up trigger (%f) listener (%s) in a group which is not an NPC group.", threshold, cbFunc.c_str()); return; } grpNpc->addHpUpTrigger(threshold, cbFunc); } //---------------------------------------------------------------------------- /** @page code @subsection delHpUpTrigger_fs_ @sa @ref delHpUpTrigger_ff_ This function is used to remove script function triggers. Arguments: f(threshold),s(callback) -> @param[in] threshold is a HP threshold @param[in] callback is the script callback to trigger @code ()delHpUpTrigger(0.5, "onHPIncrease"); @endcode */ // CGroupNpc void delHpUpTrigger_fs_(CStateInstance* entity, CScriptStack& stack) { std::string cbFunc = (std::string)stack.top(); stack.pop(); float threshold = (float)stack.top(); stack.pop(); CGroupNpc* const grpNpc = dynamic_cast(entity->getGroup()); if (!grpNpc) { nlwarning("Trying to delete a hp up trigger (%f) listener (%s) in a group which is not an NPC group.", threshold, cbFunc.c_str()); return; } grpNpc->delHpUpTrigger(threshold, cbFunc); } //---------------------------------------------------------------------------- /** @page code @subsection addHpDownTrigger_fs_ @sa @ref addHpDownTrigger_ff_ These triggers call a script function instead of trigger a user event. Arguments: f(threshold),s(callback) -> @param[in] threshold is a HP threshold @param[in] callback is the script callback to trigger @code ()addHpDownTrigger(0.5, "onHPDecrease"); @endcode */ // CGroupNpc void addHpDownTrigger_fs_(CStateInstance* entity, CScriptStack& stack) { std::string cbFunc = (std::string)stack.top(); stack.pop(); float threshold = (float)stack.top(); stack.pop(); CGroupNpc* const grpNpc = dynamic_cast(entity->getGroup()); if (!grpNpc) { nlwarning("Trying to add a hp down trigger (%f) listener (%s) in a group which is not an NPC group.", threshold, cbFunc.c_str()); return; } grpNpc->addHpDownTrigger(threshold, cbFunc); } //---------------------------------------------------------------------------- /** @page code @subsection delHpDownTrigger_fs_ @sa @ref delHpDownTrigger_ff_ This function is used to remove script function triggers. Arguments: f(threshold),s(callback) -> @param[in] threshold is a HP threshold @param[in] callback is the script callback to trigger @code ()delHpDownTrigger(0.5, "onHPDecrease"); @endcode */ // CGroupNpc void delHpDownTrigger_fs_(CStateInstance* entity, CScriptStack& stack) { std::string cbFunc = (std::string)stack.top(); stack.pop(); float threshold = (float)stack.top(); stack.pop(); CGroupNpc* const grpNpc = dynamic_cast(entity->getGroup()); if (!grpNpc) { nlwarning("Trying to delete a hp down trigger (%f) listener (%s) in a group which is not an NPC group.", threshold, cbFunc.c_str()); return; } grpNpc->delHpDownTrigger(threshold, cbFunc); } //---------------------------------------------------------------------------- /** @page code @subsection addNamedEntityListener_ssf_ Associates a listeners with a named entity property. Whenever that field changes the specified user event is triggered. Valid property names are: - state - param1 - param2 Name of the entity cannot be listened, because it cannot change. Several listeners (even with the same parameters) can be associated to each property. Arguments: s(name),s(prop),f(event) -> @param[in] name is a the name of the named entity to listen @param[in] prop is a the property of the named entity to listen @param[in] event is a the user event to trigger @code ()addNamedEntityListener("Invasion", "state", 6); @endcode */ // CGroupNpc void addNamedEntityListener_ssf_(CStateInstance* entity, CScriptStack& stack) { int event = (int)(float)stack.top(); stack.pop(); std::string prop = (std::string)stack.top(); stack.pop(); std::string name = (std::string)stack.top(); stack.pop(); CGroupNpc* const grpNpc = dynamic_cast(entity->getGroup()); if (!grpNpc) { nlwarning("Trying to create a named entity (%s:%s) listener (user event %d) in a group which is not an NPC group.", name.c_str(), prop.c_str(), event); return; } grpNpc->addNamedEntityListener(name, prop, event); } //---------------------------------------------------------------------------- /** @page code @subsection delNamedEntityListener_ssf_ Removes a listener from a named entity property. If several listeners with the same parameters are registered, only one is removed. Arguments: s(name),s(prop),f(event) -> @param[in] name is a the name of the named entity to listen @param[in] prop is a the property of the named entity to listen @param[in] event is a the user event to trigger @code ()delNamedEntityListener("Invasion", "state", 6); @endcode */ // CGroupNpc void delNamedEntityListener_ssf_(CStateInstance* entity, CScriptStack& stack) { int event = (int)(float)stack.top(); stack.pop(); std::string prop = (std::string)stack.top(); stack.pop(); std::string name = (std::string)stack.top(); stack.pop(); CGroupNpc* const grpNpc = dynamic_cast(entity->getGroup()); if (!grpNpc) { nlwarning("Trying to delete a named entity (%s:%s) listener (user event %d) in a group which is not an NPC group.", name.c_str(), prop.c_str(), event); return; } grpNpc->delNamedEntityListener(name, prop, event); } //---------------------------------------------------------------------------- /** @page code @subsection addNamedEntityListener_sss_ @sa @ref addNamedEntityListener_ssf_ These listeners call a script function instead of triggering a user event. Arguments: s(name),s(prop),s(cbFunc) -> @param[in] name is a the name of the named entity to listen @param[in] prop is a the property of the named entity to listen @param[in] cbFunc is a the callback to trigger @code ()addNamedEntityListener("Invasion", "state", "onStateChange"); @endcode */ // CGroupNpc void addNamedEntityListener_sss_(CStateInstance* entity, CScriptStack& stack) { std::string cbFunc = (std::string)stack.top(); stack.pop(); std::string prop = (std::string)stack.top(); stack.pop(); std::string name = (std::string)stack.top(); stack.pop(); CGroupNpc* const grpNpc = dynamic_cast(entity->getGroup()); if (!grpNpc) { nlwarning("Trying to create a named entity (%s:%s) listener (%s) in a group which is not an NPC group.", name.c_str(), prop.c_str(), cbFunc.c_str()); return; } grpNpc->addNamedEntityListener(name, prop, cbFunc); } //---------------------------------------------------------------------------- /** @page code @subsection delNamedEntityListener_sss_ @sa @ref delNamedEntityListener_ssf_ This function removes script function listeners. Arguments: s(name),s(prop),s(cbFunc) -> @param[in] name is a the name of the named entity to listen @param[in] prop is a the property of the named entity to listen @param[in] cbFunc is a the callback to trigger @code ()delNamedEntityListener("Invasion", "state", "onStateChange"); @endcode */ // CGroupNpc void delNamedEntityListener_sss_(CStateInstance* entity, CScriptStack& stack) { std::string cbFunc = (std::string)stack.top(); stack.pop(); std::string prop = (std::string)stack.top(); stack.pop(); std::string name = (std::string)stack.top(); stack.pop(); CGroupNpc* const grpNpc = dynamic_cast(entity->getGroup()); if (!grpNpc) { nlwarning("Trying to delete a named entity (%s:%s) listener (%s) in a group which is not an NPC group.", name.c_str(), prop.c_str(), cbFunc.c_str()); return; } grpNpc->delNamedEntityListener(name, prop, cbFunc); } //---------------------------------------------------------------------------- /** @page code @subsection setPlayerController_ss_ Make a player control a npc. Arguments: s(botId),s(playerId) -> @param[in] botId is the entity id of the bot the player will control @param[in] playerId is the entity id of the player that will control the bot @code ()setPlayerController("(0x0002015bb4:01:88:88)", "(0x0000004880:00:00:00)"); @endcode */ // CSpawnBotNpc void setPlayerController_ss_(CStateInstance* entity, CScriptStack& stack) { NLMISC::CEntityId playerId = NLMISC::CEntityId((std::string)stack.top()); stack.pop(); NLMISC::CEntityId botId = NLMISC::CEntityId((std::string)stack.top()); stack.pop(); if (botId!=NLMISC::CEntityId::Unknown) { CGroup* grp = NULL; CSpawnBotNpc* bot = NULL; CAIEntityPhysicalLocator *inst = CAIEntityPhysicalLocator::getInstance(); if (inst) { CAIEntityPhysical *botEntity = inst->getEntity(botId); if (botEntity) { if ((bot = dynamic_cast(botEntity)) && (&bot->getPersistent().getGroup())==entity->getGroup()) { CBotPlayer* player = NULL; if (playerId!=NLMISC::CEntityId::Unknown && (player = dynamic_cast(CAIEntityPhysicalLocator::getInstance()->getEntity(playerId)))) { bot->setPlayerController(player); } else bot->setPlayerController(NULL); } } else nlwarning("Bot entity not found"); } else nlwarning("Instance not found"); } } //---------------------------------------------------------------------------- /** @page code @subsection clearPlayerController_s_ Stop the control of a npc by a player. Arguments: s(botId) -> @param[in] botId is the entity id of the bot @code ()clearPlayerController("(0x0002015bb4:01:88:88)"); @endcode */ // CSpawnBotNpc void clearPlayerController_s_(CStateInstance* entity, CScriptStack& stack) { NLMISC::CEntityId botId = NLMISC::CEntityId((std::string)stack.top()); stack.pop(); if (botId!=NLMISC::CEntityId::Unknown) { CGroup* grp = NULL; CSpawnBotNpc* bot = NULL; if ((bot = dynamic_cast(CAIEntityPhysicalLocator::getInstance()->getEntity(botId))) && (&bot->getPersistent().getGroup())==entity->getGroup()) { bot->setPlayerController(NULL); } } } //---------------------------------------------------------------------------- /** @page code @subsection activateEasterEgg_fffsffffsss_ Call the EGS function CCharacterControl::activateEasterEgg Arguments: f(easterEggId), f(scenarioId), f(actId), s(items), f(x), f(y), f(z),f(heading), f(groupname), f(name), f(clientSheet)-> @param[in] items is the sheet/quantity vector (a string of format "item1:qty1;item2:qty2;...") @code ()activateEasterEgg(2, 1601, 4, "toto.sitem:2;tata.sitem:1;titi.sitem:3", 1247, 4627, 0, 0); @endcode */ void activateEasterEgg_fffsffffsss_(CStateInstance* si, CScriptStack& stack) { std::string clientSheet = (std::string)stack.top(); stack.pop(); std::string name = (std::string)stack.top(); stack.pop(); std::string grpCtrl = (std::string)stack.top(); stack.pop(); float heading = (float)stack.top(); stack.pop(); float z = (float)stack.top(); stack.pop(); float y = (float)stack.top(); stack.pop(); float x = (float)stack.top(); stack.pop(); string items = (string)stack.top(); stack.pop(); uint32 actId = static_cast( (float)stack.top()); stack.pop(); TSessionId scenarioId( static_cast( (float)stack.top())); stack.pop(); uint32 easterEgg = static_cast( (float)stack.top()); stack.pop(); IAisControl::getInstance()->activateEasterEgg(easterEgg, scenarioId, actId, items, x, y, z, heading, grpCtrl, name, clientSheet); } //---------------------------------------------------------------------------- /** @page code @subsection deactivateEasterEgg_fff_ Call the EGS function CCharacterControl::deactivateEasterEgg Arguments: f(easterEggId), f(scenarioId), f(actId) -> @code ()deactivateEasterEgg(0, 4); @endcode */ void deactivateEasterEgg_fff_(CStateInstance* si, CScriptStack& stack) { uint32 actId = static_cast( (float)stack.top()); stack.pop(); TSessionId scenarioId( static_cast( (float)stack.top())); stack.pop(); uint32 easterEgg = static_cast( (float)stack.top()); stack.pop(); IAisControl::getInstance()->deactivateEasterEgg(easterEgg, scenarioId, actId); } //---------------------------------------------------------------------------- /** @page code @subsection receiveMissionItems_ssc_ The npc will ask mission items to the targeter player. A new entry of the npc contextual menu will propose to the targeter player to give mission items to the npc if he has the requested items. Then user events are triggered on the group to inform it about what happens: - user_event_1: triggered if the player has the requested mission items. - user_event_2: triggered if the player hasn't the requested mission items. - user_event_3: triggered after the player has given the mission items to the npc. Warning: this function can only be called after the event "player_target_npc". Warning: only works on an R2 shard for R2 plot items. Arguments: s(missionItems), s(missionText), c(groupToNotify) -> @param[in] missionItems is the list of mission items, the string format is "item1:qty1;item2:qty2;...". @param[in] missionText is the text which will appear in the npc contextual menu. @param[in] groupToNotify is the npc group which will receive the user events. @code (@groupToNotify)group_name.context(); ()receiveMissionItems("toto.sitem:2;tata.sitem:1;titi.sitem:3", "Mission text", @groupToNotify); @endcode */ // CSpawnBotNpc void receiveMissionItems_ssc_(CStateInstance* entity, CScriptStack& stack) { CGroupNpc* const groupToNotify = dynamic_cast( (IScriptContext*)stack.top() ); stack.pop(); string missionText = (string)stack.top(); stack.pop(); string missionItems = (string)stack.top(); stack.pop(); if (groupToNotify == NULL) { nlwarning("receiveMissionItems failed: groupToNotify is NULL"); DEBUG_STOP; return; } if (missionItems.empty()) { nlwarning("receiveMissionItems failed: missionItems is empty"); DEBUG_STOP; return; } nlassert(entity != NULL); CGroup* const group = entity->getGroup(); IManagerParent* const managerParent = group->getOwner()->getOwner(); CAIInstance* const aiInstance = dynamic_cast(managerParent); if (aiInstance == NULL) { nlwarning("receiveMissionItems failed: the AI instance of the entity is NULL"); DEBUG_STOP; return; } CBotPlayer* const talkingPlayer = TempPlayer; CSpawnBotNpc* const talkingNpc = dynamic_cast(TempSpeaker); if ( talkingPlayer == NULL || talkingNpc == NULL || talkingNpc->getPersistent().getOwner() != group) // check if the talking npc is in this group { nlwarning("receiveMissionItems failed: invalid interlocutors"); DEBUG_STOP; return; } // do nothing if one of them is dead if (!talkingPlayer->isAlive() || !talkingNpc->isAlive()) { return; } // turn the npc to face the player talkingNpc->setTheta(talkingNpc->pos().angleTo(talkingPlayer->pos())); // send the request to the EGS CGiveItemRequestMsg msg; msg.InstanceId = aiInstance->getInstanceNumber(); msg.GroupAlias = groupToNotify->getAlias(); msg.CharacterRowId = talkingPlayer->dataSetRow(); msg.CreatureRowId = talkingNpc->dataSetRow(); msg.MissionText = missionText; // extract items and quantities from the missionItems string std::vector itemAndQtyList; NLMISC::splitString(missionItems, ";", itemAndQtyList); if (itemAndQtyList.empty()) { nlwarning("receiveMissionItems failed: the provided mission items string is invalid"); DEBUG_STOP; return; } FOREACHC(it, std::vector, itemAndQtyList) { std::vector itemAndQty; NLMISC::splitString(*it, ":", itemAndQty); if (itemAndQty.size() != 2) { nlwarning("receiveMissionItems failed: the provided mission items string is invalid"); DEBUG_STOP; return; } const CSheetId sheetId(itemAndQty[0]); if (sheetId == CSheetId::Unknown) { nlwarning("receiveMissionItems failed: invalid mission item sheet '%s'", itemAndQty[0].c_str()); DEBUG_STOP; return; } uint32 quantity; NLMISC::fromString(itemAndQty[1], quantity); if (quantity == 0) { nlwarning("receiveMissionItems failed: invalid quantity '%s'", itemAndQty[1].c_str()); DEBUG_STOP; return; } msg.Items.push_back(sheetId); msg.Quantities.push_back(quantity); } nlassert(!msg.Items.empty()); nlassert(msg.Items.size() == msg.Quantities.size()); msg.send("EGS"); } //---------------------------------------------------------------------------- /** @page code @subsection giveMissionItems_ssc_ The npc will give mission items to the targeter player. A new entry of the npc contextual menu will propose to the targeter player to receive mission items from the npc. Then user events are triggered on the group to inform it about what happens: - user_event_1: triggered after the player has received the mission items from the npc. Warning: this function can only be called after the event "player_target_npc". Warning: only works on an R2 shard for R2 plot items. Arguments: s(missionItems), s(missionText), c(groupToNotify) -> @param[in] missionItems is the list of mission items, the string format is "item1:qty1;item2:qty2;...". @param[in] missionText is the text which will appear in the npc contextual menu. @param[in] groupToNotify is the npc group which will receive the user events. @code (@groupToNotify)group_name.context(); ()giveMissionItems("toto.sitem:2;tata.sitem:1;titi.sitem:3", "Mission text", @groupToNotify); @endcode */ // CSpawnBotNpc void giveMissionItems_ssc_(CStateInstance* entity, CScriptStack& stack) { CGroupNpc* const groupToNotify = dynamic_cast( (IScriptContext*)stack.top() ); stack.pop(); string missionText = (string)stack.top(); stack.pop(); string missionItems = (string)stack.top(); stack.pop(); if (groupToNotify == NULL) { nlwarning("giveMissionItems failed: groupToNotify is NULL"); DEBUG_STOP; return; } if (missionItems.empty()) { nlwarning("giveMissionItems failed: missionItems is empty"); DEBUG_STOP; return; } nlassert(entity != NULL); CGroup* const group = entity->getGroup(); IManagerParent* const managerParent = group->getOwner()->getOwner(); CAIInstance* const aiInstance = dynamic_cast(managerParent); if (aiInstance == NULL) { nlwarning("giveMissionItems failed: the AI instance of the entity is NULL"); DEBUG_STOP; return; } CBotPlayer* const talkingPlayer = TempPlayer; CSpawnBotNpc* const talkingNpc = dynamic_cast(TempSpeaker); if ( talkingPlayer == NULL || talkingNpc == NULL || talkingNpc->getPersistent().getOwner() != group) // check if the talking npc is in this group { nlwarning("giveMissionItems failed: invalid interlocutors"); DEBUG_STOP; return; } // do nothing if one of them is dead if (!talkingPlayer->isAlive() || !talkingNpc->isAlive()) { return; } // turn the npc to face the player talkingNpc->setTheta(talkingNpc->pos().angleTo(talkingPlayer->pos())); // send the request to the EGS CReceiveItemRequestMsg msg; msg.InstanceId = aiInstance->getInstanceNumber(); msg.GroupAlias = groupToNotify->getAlias(); msg.CharacterRowId = talkingPlayer->dataSetRow(); msg.CreatureRowId = talkingNpc->dataSetRow(); msg.MissionText = missionText; // extract items and quantities from the missionItems string std::vector itemAndQtyList; NLMISC::splitString(missionItems, ";", itemAndQtyList); if (itemAndQtyList.empty()) { nlwarning("giveMissionItems failed: the provided mission items string is invalid"); DEBUG_STOP; return; } FOREACHC(it, std::vector, itemAndQtyList) { std::vector itemAndQty; NLMISC::splitString(*it, ":", itemAndQty); if (itemAndQty.size() != 2) { nlwarning("giveMissionItems failed: the provided mission items string is invalid"); DEBUG_STOP; return; } const CSheetId sheetId(itemAndQty[0]); if (sheetId == CSheetId::Unknown) { nlwarning("giveMissionItems failed: invalid mission item sheet '%s'", itemAndQty[0].c_str()); DEBUG_STOP; return; } uint32 quantity; NLMISC::fromString(itemAndQty[1], quantity); if (quantity == 0) { nlwarning("giveMissionItems failed: invalid quantity '%s'", itemAndQty[1].c_str()); DEBUG_STOP; return; } msg.Items.push_back(sheetId); msg.Quantities.push_back(quantity); } nlassert(!msg.Items.empty()); nlassert(msg.Items.size() == msg.Quantities.size()); msg.send("EGS"); } //---------------------------------------------------------------------------- /** @page code @subsection talkTo_sc_ A new entry of the npc contextual menu will propose to the targeter player to talk to the npc. Then user events are triggered on the group to inform it about what happens: - user_event_1: triggered each time (because of the TRICK). - user_event_3: triggered after the player has talked to the npc. Warning: this function can only be called after the event "player_target_npc". Arguments: s(missionText), c(groupToNotify) -> @param[in] missionText is the text which will appear in the npc contextual menu. @param[in] groupToNotify is the npc group which will receive the user events. @code (@groupToNotify)group_name.context(); ()talkTo("Mission text", @groupToNotify); @endcode */ // CSpawnBotNpc void talkTo_sc_(CStateInstance* entity, CScriptStack& stack) { CGroupNpc* const groupToNotify = dynamic_cast( (IScriptContext*)stack.top() ); stack.pop(); string missionText = (string)stack.top(); stack.pop(); if (groupToNotify == NULL) { nlwarning("talkTo failed: groupToNotify is NULL"); DEBUG_STOP; return; } nlassert(entity != NULL); CGroup* const group = entity->getGroup(); IManagerParent* const managerParent = group->getOwner()->getOwner(); CAIInstance* const aiInstance = dynamic_cast(managerParent); if (aiInstance == NULL) { nlwarning("talkTo failed: the AI instance of the entity is NULL"); DEBUG_STOP; return; } CBotPlayer* const talkingPlayer = TempPlayer; CSpawnBotNpc* const talkingNpc = dynamic_cast(TempSpeaker); if ( talkingPlayer == NULL || talkingNpc == NULL || talkingNpc->getPersistent().getOwner() != group) // check if the talking npc is in this group { nlwarning("talkTo failed: invalid interlocutors"); DEBUG_STOP; return; } // do nothing if one of them is dead if (!talkingPlayer->isAlive() || !talkingNpc->isAlive()) { return; } // turn the npc to face the player talkingNpc->setTheta(talkingNpc->pos().angleTo(talkingPlayer->pos())); // send the request to the EGS // TRICK: here we use an empty give item request CGiveItemRequestMsg msg; msg.InstanceId = aiInstance->getInstanceNumber(); msg.GroupAlias = groupToNotify->getAlias(); msg.CharacterRowId = talkingPlayer->dataSetRow(); msg.CreatureRowId = talkingNpc->dataSetRow(); msg.MissionText = missionText; nlassert(msg.Items.empty()); nlassert(msg.Quantities.empty()); msg.send("EGS"); } //give_reward void giveReward_ssssc_(CStateInstance* entity, CScriptStack& stack) { CGroupNpc* const groupToNotify = dynamic_cast( (IScriptContext*)stack.top() ); stack.pop(); string notEnoughPointsText = (string)stack.top(); stack.pop(); string inventoryFullText = (string)stack.top(); stack.pop(); string rareRewardText = (string)stack.top(); stack.pop(); string rewardText = (string)stack.top(); stack.pop(); if (groupToNotify == NULL) { nlwarning("giveReward failed: groupToNotify is NULL"); DEBUG_STOP; return; } nlassert(entity != NULL); CGroup* const group = entity->getGroup(); IManagerParent* const managerParent = group->getOwner()->getOwner(); CAIInstance* const aiInstance = dynamic_cast(managerParent); if (aiInstance == NULL) { nlwarning("giveReward failed: the AI instance of the entity is NULL"); DEBUG_STOP; return; } CEntityId charEid(group->getEventParamString(0)); // must be player char EID CEntityId npcEid(group->getEventParamString(1)); // must be NPC EID if (charEid == CEntityId::Unknown || npcEid == CEntityId::Unknown) { nlwarning("giveReward failed: the chareter or npc can't be retrreived from the event parameter"); DEBUG_STOP; return; } // retrieve the CBotPlayer CAIEntityPhysical *charEntity=CAIS::instance().getEntityPhysical(TheDataset.getDataSetRow(charEid)); if (!charEntity) return; CBotPlayer *talkingPlayer=NLMISC::safe_cast(charEntity); if (!talkingPlayer) return; // retrieve the CSpawnBotNpcBotPlayer CAIEntityPhysical *npcEntity=CAIS::instance().getEntityPhysical(TheDataset.getDataSetRow(npcEid)); if (!npcEntity) return; CSpawnBotNpc *talkingNpc=NLMISC::safe_cast(npcEntity); if (!talkingNpc) return; // CBotPlayer* const talkingPlayer = TempPlayer; // CSpawnBotNpc* const talkingNpc = dynamic_cast(TempSpeaker); if ( talkingPlayer == NULL || talkingNpc == NULL) { nlwarning("giveReward failed: invalid interlocutors"); DEBUG_STOP; return; } // do nothing if one of them is dead if (!talkingPlayer->isAlive() || !talkingNpc->isAlive()) { return; } // turn the npc to face the player talkingNpc->setTheta(talkingNpc->pos().angleTo(talkingPlayer->pos())); TDataSetRow CharacterRowId = talkingPlayer->dataSetRow(); TDataSetRow CreatureRowId = talkingNpc->dataSetRow(); IAisControl::getInstance()->giveRewardMessage(CharacterRowId, CreatureRowId, rewardText, rareRewardText, inventoryFullText, notEnoughPointsText); } //give_reward void teleportNear_fffc_(CStateInstance* entity, CScriptStack& stack) { CGroupNpc* const groupToNotify = dynamic_cast( (IScriptContext*)stack.top() ); stack.pop(); float z = (float)stack.top(); stack.pop(); float y = (float)stack.top(); stack.pop(); float x = (float)stack.top(); stack.pop(); if (groupToNotify == NULL) { nlwarning("teleportNear failed: groupToNotify is NULL"); DEBUG_STOP; return; } nlassert(entity != NULL); CGroup* const group = entity->getGroup(); IManagerParent* const managerParent = group->getOwner()->getOwner(); CAIInstance* const aiInstance = dynamic_cast(managerParent); if (aiInstance == NULL) { nlwarning("giveReward failed: the AI instance of the entity is NULL"); DEBUG_STOP; return; } CEntityId charEid(group->getEventParamString(0)); // must be player char EID CEntityId npcEid(group->getEventParamString(1)); // must be NPC EID if (charEid == CEntityId::Unknown || npcEid == CEntityId::Unknown) { nlwarning("giveReward failed: the character or npc can't be retreived from the event parameter"); DEBUG_STOP; return; } // retrieve the CBotPlayer CAIEntityPhysical *charEntity=CAIS::instance().getEntityPhysical(TheDataset.getDataSetRow(charEid)); if (!charEntity) return; CBotPlayer *talkingPlayer=NLMISC::safe_cast(charEntity); if (!talkingPlayer) return; // retrieve the CSpawnBotNpcBotPlayer CAIEntityPhysical *npcEntity=CAIS::instance().getEntityPhysical(TheDataset.getDataSetRow(npcEid)); if (!npcEntity) return; CSpawnBotNpc *talkingNpc=NLMISC::safe_cast(npcEntity); if (!talkingNpc) return; // CBotPlayer* const talkingPlayer = TempPlayer; // CSpawnBotNpc* const talkingNpc = dynamic_cast(TempSpeaker); if ( talkingPlayer == NULL || talkingNpc == NULL) { nlwarning("giveReward failed: invalid interlocutors"); DEBUG_STOP; return; } // do nothing if one of them is dead if (!talkingPlayer->isAlive() || !talkingNpc->isAlive()) { return; } // turn the npc to face the player /// talkingNpc->setTheta(talkingNpc->pos().angleTo(talkingPlayer->pos())); TDataSetRow CharacterRowId = talkingPlayer->dataSetRow(); TDataSetRow CreatureRowId = talkingNpc->dataSetRow(); NLMISC::CEntityId player = CMirrors::getEntityId(CharacterRowId); if (player != NLMISC::CEntityId::Unknown) { IAisControl::getInstance()->teleportNearMessage(player, x, y, z); } } // Return an "alive" bot from a group by the bot name static CSpawnBot* getSpawnBotFromGroupByName(CGroupNpc* const group, const std::string& botname) { // Group exist if (!group) { return 0; } // Group is in a valid AIInstance { if (!group->getOwner()) { return 0; } IManagerParent* const managerParent = group->getOwner()->getOwner(); CAIInstance* const aiInstance = dynamic_cast(managerParent); if (!aiInstance) { return 0; } } // Bot is spawn and alive { CSpawnBot* spawnBot=0; CAliasCont const& bots = group->bots(); CCont & cc_bots = group->bots(); CBot* child; if (botname == "") child = bots.getFirstChild(); else child = bots.getChildByName(botname); if (!child) { return 0; } spawnBot = child->getSpawnObj(); if (!spawnBot || !spawnBot->isAlive()) { return 0; } return spawnBot; } nlassert(0); //this path is never used return 0; } //---------------------------------------------------------------------------- /** @page code @subsection facing_cscs_ The npc1 will turn to npc2 Arguments: c(group1), s(botname1), c(group2), s(botname2), -> @param[in] group1 is the npc group of the boot that turn to the other bot @param[in] botname1 is the name of the bot @param[in] group2 is the npc group of the boot that is the target @param[in] botname2 is the name of the bot that is targeted @code (@group1)group_name1.context(); (@group1)group_name2.context(); ()facing(@group1, "bob", @group2, "bobette"); @endcode */ // CSpawnBotNpc void facing_cscs_(CStateInstance* entity, CScriptStack& stack) { string botname2 = (string)stack.top(); stack.pop(); CGroupNpc* const group2 = dynamic_cast( (IScriptContext*)stack.top() ); stack.pop(); string botname1 = (string)stack.top(); stack.pop(); CGroupNpc* const group1 = dynamic_cast( (IScriptContext*)stack.top() ); stack.pop(); CSpawnBot* bot1 = getSpawnBotFromGroupByName(group1, botname1); CSpawnBot* bot2 = getSpawnBotFromGroupByName(group2, botname2); if (!bot1 || !bot2) { return; } CSpawnBotNpc* bot = dynamic_cast(bot1); if (!bot) { return; } // Same than setTheta but initial theta is restored few secondes later bot->setFacing(bot1->pos().angleTo(bot2->pos())); // bot1->setTheta(bot1->pos().angleTo(bot2->pos())); } //---------------------------------------------------------------------------- /** @page code @subsection npcSay_css_ Make a npc say a text There are 3 type of text - Classic StringId (IOS does traduction) - RING "complete text (no traduction) - RING StringId (DSS does traduction) Arguments: c(group), s(botname), s(text), -> @param[in] group is the npc group of the boot that does the emote @param[in] botname is the name of the bot @param[in] text is the name of the emote @code (@group)group_name.context(); ()npcSay(@group, "bob", "DSS_1601 RtEntryText_6") ;// Send To dss ()npcSay(@group, "bob", "RAW Ca farte?"); // phrase direcly send to IOS as raw (for debug) ()npcSay(@group, "bob", "answer_group_no_m"); //phrase id @endcode */ // CSpawnBotNpc #include "game_share/chat_group.h" #include "game_share/send_chat.h" void execSayHelper(CSpawnBot *spawnBot, NLMISC::CSString text, CChatGroup::TGroupType mode = CChatGroup::say) { if (spawnBot) { NLMISC::CSString prefix = text.left(4); if (prefix=="DSS_") { NLMISC::CSString phrase = text.right(text.length() - 4); NLMISC::CSString idStr = phrase.strtok(" ",false,false,false,false); uint32 scenarioId = atoi(idStr.c_str()); forwardToDss(spawnBot->dataSetRow(), mode, phrase, scenarioId); return; } if (prefix=="RAW ") { std::string phrase = text.right(text.length()-4); ucstring ucstr = phrase; npcChatToChannelSentence(spawnBot->dataSetRow(), mode, ucstr); return; } //Classic phrase ID npcChatToChannel(spawnBot->dataSetRow(), mode, text); } } void npcSay_css_(CStateInstance* entity, CScriptStack& stack) { string text = (string)stack.top(); stack.pop(); string botname = (string)stack.top(); stack.pop(); CGroupNpc* const group = dynamic_cast( (IScriptContext*)stack.top() ); stack.pop(); CSpawnBot* spawnBot = getSpawnBotFromGroupByName(group, botname); if (!spawnBot) { return; } execSayHelper(spawnBot, text); } //---------------------------------------------------------------------------- /** @page code @subsection npcSay_ss_ Make a npc say a text Arguments: s(text), s(mode) -> @param[in] text is the text to say. prefix with ID: to use an id @param[in] mode is the mode to use (say, shout) @code ()npcSay("Hello!","say"); // phrase direcly send to IOS as raw ()npcSay("ID:answer_group_no_m","shout"); // phrase id @endcode */ void npcSay_ss_(CStateInstance* entity, CScriptStack& stack) { std::string sMode = (std::string)stack.top(); stack.pop(); std::string text = (std::string)stack.top(); stack.pop(); CChatGroup::TGroupType mode = CChatGroup::say; mode = CChatGroup::stringToGroupType(sMode); CGroup* group = entity->getGroup(); if (group->isSpawned()) { FOREACH(itBot, CCont, group->bots()) { CBot* bot = *itBot; if (bot) { if (bot->isSpawned()) { CSpawnBot *spawnBot = bot->getSpawnObj(); std::string prefix = NLMISC::CSString(text).left(3); if (NLMISC::nlstricmp(prefix.c_str(), "id:") == 0) { text = NLMISC::CSString(text).right(text.length()-3); execSayHelper(spawnBot, text, mode); } else { execSayHelper(spawnBot, "RAW " + text, mode); } } } } } } //---------------------------------------------------------------------------- /** @page code @subsection deactivateEasterEgg_fff_ Call the DSS function CAnimationModule::dssMessage Arguments: f(easterEggId), f(scenarioId), f(actId) -> @code ()dssMessage(114, "BC", "Bob", "Life is harde"); @endcode */ void dssMessage_fsss_(CStateInstance* entity, CScriptStack& stack) { string msg = (string)stack.top(); stack.pop(); string who = (string)stack.top(); stack.pop(); string mode = (string)stack.top(); stack.pop(); float instance = (float)stack.top(); stack.pop(); IAisControl::getInstance()->dssMessage(TSessionId(uint32(instance)), mode, who, msg); return; } //---------------------------------------------------------------------------- /** @page code @subsection setScenarioPoints Call the DSS function CAnimationModule::setScenarioPoints Arguments: f(scenarioInstance), f(scenarioPoints) -> @code ()setScenarioPoints(114, 42); @endcode */ void setScenarioPoints_ff_(CStateInstance* entity, CScriptStack& stack) { float scenarioPoints = (float)stack.top(); stack.pop(); float instance = (float)stack.top(); stack.pop(); IAisControl::getInstance()->setScenarioPoints(TSessionId(uint32(instance)), scenarioPoints); return; } //---------------------------------------------------------------------------- /** @page code @subsection startScenarioTiming Call the DSS function CAnimationModule::startScenarioTiming Arguments: f(scenarioInstance), @code ()startScenarioTiming(114, 42); @endcode */ void startScenarioTiming_f_(CStateInstance* entity, CScriptStack& stack) { float instance = (float)stack.top(); stack.pop(); IAisControl::getInstance()->startScenarioTiming(TSessionId(uint32(instance))); return; } //---------------------------------------------------------------------------- /** @page code @subsection endScenarioTiming Call the DSS function CAnimationModule::endScenarioTiming Arguments: f(scenarioInstance), @code ()endScenarioTiming(114, 42); @endcode */ void endScenarioTiming_f_(CStateInstance* entity, CScriptStack& stack) { float instance = (float)stack.top(); stack.pop(); IAisControl::getInstance()->endScenarioTiming(TSessionId(uint32(instance))); return; } //---------------------------------------------------------------------------- /** @page code @subsection emote_css_ Make a npc launch a emote Arguments: c(group1), s(botname1), c(group2), s(botname2), -> @param[in] group1 is the npc group of the boot that does the emote @param[in] botname1 is the name of the bot @param[in] emote is the name of the emote @code (@group1)group_name.context(); ()emote(@group1, "bob", "sad") @endcode */ // CSpawnBotNpc /****************************************************************************/ void emote_css_(CStateInstance* entity, CScriptStack& stack) { string emoteName = (string)stack.top(); stack.pop(); string botname = (string)stack.top(); stack.pop(); CGroupNpc* const group = dynamic_cast( (IScriptContext*)stack.top() ); stack.pop(); CSpawnBot* spawnBot = getSpawnBotFromGroupByName(group, botname); if (!spawnBot) { return; } //CBot& bot = spawnBot->getPersistent(); // The entity Id must be valid (we know that the bot is alive so its entity Id must be ok) NLMISC::CEntityId entityId=spawnBot->getEntityId(); if (entityId == NLMISC::CEntityId::Unknown) { return; } // Is the emote valid uint32 emoteId = CAIS::instance().getEmotNumber(emoteName); if (emoteId == ~0) { return; } // Get the behaviour Id MBEHAV::EBehaviour behaviourId = (MBEHAV::EBehaviour)(emoteId + MBEHAV::EMOTE_BEGIN); // Change the behaviour NLNET::CMessage msgout("SET_BEHAVIOUR"); msgout.serial(entityId); MBEHAV::CBehaviour bh(behaviourId); bh.Data = (uint16)(CTimeInterface::gameCycle()); msgout.serial(bh); NLNET::CUnifiedNetwork::getInstance()->send( "EGS", msgout ); } void emote_ss_(CStateInstance* entity, CScriptStack& stack) { string emoteName = (string)stack.top(); stack.pop(); NLMISC::CEntityId botId = NLMISC::CEntityId((std::string)stack.top()); if (botId == NLMISC::CEntityId::Unknown) { return; } // Is the emote valid uint32 emoteId = CAIS::instance().getEmotNumber(emoteName); if (emoteId == ~0) { return; } // Get the behaviour Id MBEHAV::EBehaviour behaviourId = (MBEHAV::EBehaviour)(emoteId + MBEHAV::EMOTE_BEGIN); // Change the behaviour NLNET::CMessage msgout("SET_BEHAVIOUR"); msgout.serial(botId); MBEHAV::CBehaviour bh(behaviourId); bh.Data = (uint16)(CTimeInterface::gameCycle()); msgout.serial(bh); NLNET::CUnifiedNetwork::getInstance()->send( "EGS", msgout ); } void emote_s_(CStateInstance* entity, CScriptStack& stack) { string emoteName = (string)stack.top(); stack.pop(); // Is the emote valid uint32 emoteId = CAIS::instance().getEmotNumber(emoteName); if (emoteId == ~0) return; // Get the behaviour Id MBEHAV::EBehaviour behaviourId = (MBEHAV::EBehaviour)(emoteId + MBEHAV::EMOTE_BEGIN); CGroup* group = entity->getGroup(); if (group->isSpawned()) { FOREACH(itBot, CCont, group->bots()) { CBot* bot = *itBot; if (bot) { // Change the behaviour if (bot->isSpawned()) { CSpawnBot *spawnBot = bot->getSpawnObj(); if (spawnBot) { CEntityId botId = spawnBot->getEntityId(); NLNET::CMessage msgout("SET_BEHAVIOUR"); msgout.serial(botId); MBEHAV::CBehaviour bh(behaviourId); bh.Data = (uint16)(CTimeInterface::gameCycle()); msgout.serial(bh); NLNET::CUnifiedNetwork::getInstance()->send( "EGS", msgout ); } } } } } } void rename_s_(CStateInstance* entity, CScriptStack& stack) { string newName = (string)stack.top(); stack.pop(); CGroup* group = entity->getGroup(); if (group->isSpawned()) { FOREACH(itBot, CCont, group->bots()) { CBot* bot = *itBot; if (bot) { if (bot->isSpawned()) { CSpawnBot *spawnBot = bot->getSpawnObj(); if (spawnBot) { TDataSetRow row = spawnBot->dataSetRow(); ucstring name; name.fromUtf8(newName); NLNET::CMessage msgout("CHARACTER_NAME"); msgout.serial(row); msgout.serial(name); sendMessageViaMirror("IOS", msgout); bot->setCustomName(name); } } } } } } void vpx_s_(CStateInstance* entity, CScriptStack& stack) { string vpx = (string)stack.top(); stack.pop(); CGroup* group = entity->getGroup(); if (group->isSpawned()) { FOREACH(itBot, CCont, group->bots()) { CBotNpc* bot = NLMISC::safe_cast(*itBot); if (bot) { bot->setVisualProperties(vpx); bot->sendVisualProperties(); } } } } //---------------------------------------------------------------------------- /** @page code @subsection maxHitRange_f_ Sets the max hit range possible for player, in meters Arguments: f(MaxHitRange) -> @param[in] MaxHitRange set the max range for player can hit this npc group @code ()maxHitRange(50); // Set the max hit range in 50 meters all npc in group @endcode */ // CBotNpc void maxHitRange_f_(CStateInstance* entity, CScriptStack& stack) { float maxHitRange = (float&)stack.top(); stack.pop(); CGroup* group = entity->getGroup(); CGroupNpc* npcGroup = NLMISC::safe_cast(group); FOREACH(botIt, CCont, group->bots()) { CBot* bot = *botIt; if (!bot->isSpawned()) return; if (bot->getRyzomType() == RYZOMID::npc) { CBotNpc* botNpc = NLMISC::safe_cast(bot); botNpc->setMaxHitRangeForPlayer(maxHitRange); } } } ////---------------------------------------------------------------------------- ///** @page code // //@subsection hideMissionStepIcon_b_ //Allows to hide icons for current missions steps having interactions with this NPC (default: shown) // //Arguments: b(hide) -> //@param[in] hide true to hide the icon for all mission steps relating to this NPC (default: false) // //*/ //// CBotNpc //void hideMissionStepIcon_b_(CStateInstance* entity, CScriptStack& stack) //{ // bool b = (bool&)stack.top(); stack.pop(); // CGroup* group = entity->getGroup(); // CGroupNpc* npcGroup = NLMISC::safe_cast(group); // // FOREACH(botIt, CCont, group->bots()) // { // CBot* bot = *botIt; // // if (!bot->isSpawned()) return; // if (bot->getRyzomType() == RYZOMID::npc) // { // CBotNpc* botNpc = NLMISC::safe_cast(bot); // botNpc->setMissionStepIconHidden(b); // } // } //} // ////---------------------------------------------------------------------------- ///** @page code // //@subsection hideMissionGiverIcon_b_ //Allows to hide icons for missions proposed by this NPC (default: shown) // //Arguments: b(hide) -> //@param[in] hide true to hide the icon for all missions propsed by this NPC (default: false) // //*/ //// CBotNpc //void hideMissionGiverIcon_b_(CStateInstance* entity, CScriptStack& stack) //{ // bool b = (bool&)stack.top(); stack.pop(); // CGroup* group = entity->getGroup(); // CGroupNpc* npcGroup = NLMISC::safe_cast(group); // // FOREACH(botIt, CCont, group->bots()) // { // CBot* bot = *botIt; // // if (!bot->isSpawned()) return; // if (bot->getRyzomType() == RYZOMID::npc) // { // CBotNpc* botNpc = NLMISC::safe_cast(bot); // botNpc->setMissionGiverIconHidden(b); // } // } //} std::map nfGetNpcGroupNativeFunctions() { std::map functions; #define REGISTER_NATIVE_FUNC(cont, func) cont.insert(std::make_pair(std::string(#func), &func)) REGISTER_NATIVE_FUNC(functions, setFactionProp_ss_); REGISTER_NATIVE_FUNC(functions, setOupostMode_ss_); REGISTER_NATIVE_FUNC(functions, moveToZone_ss_); REGISTER_NATIVE_FUNC(functions, setActivity_s_); REGISTER_NATIVE_FUNC(functions, startWander_f_); REGISTER_NATIVE_FUNC(functions, startMoving_fff_); REGISTER_NATIVE_FUNC(functions, waitInZone_s_); REGISTER_NATIVE_FUNC(functions, stopMoving__); REGISTER_NATIVE_FUNC(functions, followPlayer_sf_); REGISTER_NATIVE_FUNC(functions, wander__); REGISTER_NATIVE_FUNC(functions, setAttackable_f_); REGISTER_NATIVE_FUNC(functions, setPlayerAttackable_f_); REGISTER_NATIVE_FUNC(functions, setBotAttackable_f_); REGISTER_NATIVE_FUNC(functions, setFactionAttackableAbove_sff_); REGISTER_NATIVE_FUNC(functions, setFactionAttackableBelow_sff_); REGISTER_NATIVE_FUNC(functions, addBotChat_s_); REGISTER_NATIVE_FUNC(functions, clearBotChat__); REGISTER_NATIVE_FUNC(functions, ignoreOffensiveActions_f_); REGISTER_NATIVE_FUNC(functions, setDespawnTime_f_); REGISTER_NATIVE_FUNC(functions, setRespawnTime_f_); REGISTER_NATIVE_FUNC(functions, addHpUpTrigger_ff_); REGISTER_NATIVE_FUNC(functions, delHpUpTrigger_ff_); REGISTER_NATIVE_FUNC(functions, addHpDownTrigger_ff_); REGISTER_NATIVE_FUNC(functions, delHpDownTrigger_ff_); REGISTER_NATIVE_FUNC(functions, addHpUpTrigger_fs_); REGISTER_NATIVE_FUNC(functions, delHpUpTrigger_fs_); REGISTER_NATIVE_FUNC(functions, addHpDownTrigger_fs_); REGISTER_NATIVE_FUNC(functions, delHpDownTrigger_fs_); REGISTER_NATIVE_FUNC(functions, addNamedEntityListener_ssf_); REGISTER_NATIVE_FUNC(functions, delNamedEntityListener_ssf_); REGISTER_NATIVE_FUNC(functions, addNamedEntityListener_sss_); REGISTER_NATIVE_FUNC(functions, delNamedEntityListener_sss_); REGISTER_NATIVE_FUNC(functions, setPlayerController_ss_); REGISTER_NATIVE_FUNC(functions, clearPlayerController_s_); REGISTER_NATIVE_FUNC(functions, activateEasterEgg_fffsffffsss_); REGISTER_NATIVE_FUNC(functions, deactivateEasterEgg_fff_); REGISTER_NATIVE_FUNC(functions, receiveMissionItems_ssc_); REGISTER_NATIVE_FUNC(functions, giveMissionItems_ssc_); REGISTER_NATIVE_FUNC(functions, talkTo_sc_); REGISTER_NATIVE_FUNC(functions, facing_cscs_); REGISTER_NATIVE_FUNC(functions, emote_css_); REGISTER_NATIVE_FUNC(functions, emote_ss_); REGISTER_NATIVE_FUNC(functions, emote_s_); REGISTER_NATIVE_FUNC(functions, rename_s_); REGISTER_NATIVE_FUNC(functions, vpx_s_); REGISTER_NATIVE_FUNC(functions, npcSay_css_); REGISTER_NATIVE_FUNC(functions, npcSay_ss_); REGISTER_NATIVE_FUNC(functions, dssMessage_fsss_); REGISTER_NATIVE_FUNC(functions, despawnBotByAlias_s_); REGISTER_NATIVE_FUNC(functions, giveReward_ssssc_); REGISTER_NATIVE_FUNC(functions, teleportNear_fffc_); REGISTER_NATIVE_FUNC(functions, setScenarioPoints_ff_); REGISTER_NATIVE_FUNC(functions, startScenarioTiming_f_); REGISTER_NATIVE_FUNC(functions, endScenarioTiming_f_); REGISTER_NATIVE_FUNC(functions, maxHitRange_f_); // REGISTER_NATIVE_FUNC(functions, hideMissionStepIcon_b_); // REGISTER_NATIVE_FUNC(functions, hideMissionGiverIcon_b_); #undef REGISTER_NATIVE_FUNC return functions; }