2015-01-05 17:52:37 +00:00
|
|
|
#include "summoning.hpp"
|
|
|
|
|
2015-12-18 01:58:38 +00:00
|
|
|
#include <iostream>
|
|
|
|
|
2015-01-05 17:52:37 +00:00
|
|
|
#include "../mwbase/environment.hpp"
|
|
|
|
#include "../mwbase/world.hpp"
|
|
|
|
|
2015-01-22 03:12:08 +00:00
|
|
|
#include "../mwmechanics/spellcasting.hpp"
|
|
|
|
|
2015-01-05 17:52:37 +00:00
|
|
|
#include "../mwworld/esmstore.hpp"
|
|
|
|
#include "../mwworld/class.hpp"
|
|
|
|
#include "../mwworld/manualref.hpp"
|
|
|
|
#include "../mwworld/inventorystore.hpp"
|
|
|
|
|
2015-05-29 23:00:24 +00:00
|
|
|
#include "../mwrender/animation.hpp"
|
2015-01-05 17:52:37 +00:00
|
|
|
|
|
|
|
#include "creaturestats.hpp"
|
|
|
|
#include "aifollow.hpp"
|
|
|
|
|
|
|
|
|
|
|
|
namespace MWMechanics
|
|
|
|
{
|
|
|
|
|
|
|
|
void cleanupSummonedCreature (MWMechanics::CreatureStats& casterStats, int creatureActorId)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = MWBase::Environment::get().getWorld()->searchPtrViaActorId(creatureActorId);
|
|
|
|
if (!ptr.isEmpty())
|
|
|
|
{
|
|
|
|
// TODO: Show death animation before deleting? We shouldn't allow looting the corpse while the animation
|
|
|
|
// plays though, which is a rather lame exploit in vanilla.
|
|
|
|
MWBase::Environment::get().getWorld()->deleteObject(ptr);
|
|
|
|
|
|
|
|
const ESM::Static* fx = MWBase::Environment::get().getWorld()->getStore().get<ESM::Static>()
|
|
|
|
.search("VFX_Summon_End");
|
|
|
|
if (fx)
|
|
|
|
MWBase::Environment::get().getWorld()->spawnEffect("meshes\\" + fx->mModel,
|
2015-04-19 15:55:56 +00:00
|
|
|
"", ptr.getRefData().getPosition().asVec3());
|
2015-01-05 17:52:37 +00:00
|
|
|
}
|
2015-12-18 01:36:34 +00:00
|
|
|
else if (creatureActorId != -1)
|
2015-01-05 17:52:37 +00:00
|
|
|
{
|
|
|
|
// We didn't find the creature. It's probably in an inactive cell.
|
|
|
|
// Add to graveyard so we can delete it when the cell becomes active.
|
|
|
|
std::vector<int>& graveyard = casterStats.getSummonedCreatureGraveyard();
|
|
|
|
graveyard.push_back(creatureActorId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
UpdateSummonedCreatures::UpdateSummonedCreatures(const MWWorld::Ptr &actor)
|
|
|
|
: mActor(actor)
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2015-02-16 05:41:53 +00:00
|
|
|
UpdateSummonedCreatures::~UpdateSummonedCreatures()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-01-05 17:52:37 +00:00
|
|
|
void UpdateSummonedCreatures::visit(EffectKey key, const std::string &sourceName, const std::string &sourceId, int casterActorId, float magnitude, float remainingTime, float totalTime)
|
|
|
|
{
|
2015-01-22 03:12:08 +00:00
|
|
|
if (isSummoningEffect(key.mId) && magnitude > 0)
|
2015-01-05 17:52:37 +00:00
|
|
|
{
|
|
|
|
mActiveEffects.insert(std::make_pair(key.mId, sourceId));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void UpdateSummonedCreatures::finish()
|
|
|
|
{
|
|
|
|
static std::map<int, std::string> summonMap;
|
|
|
|
if (summonMap.empty())
|
|
|
|
{
|
|
|
|
summonMap[ESM::MagicEffect::SummonAncestralGhost] = "sMagicAncestralGhostID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonBonelord] = "sMagicBonelordID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonBonewalker] = "sMagicLeastBonewalkerID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonCenturionSphere] = "sMagicCenturionSphereID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonClannfear] = "sMagicClannfearID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonDaedroth] = "sMagicDaedrothID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonDremora] = "sMagicDremoraID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonFabricant] = "sMagicFabricantID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonFlameAtronach] = "sMagicFlameAtronachID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonFrostAtronach] = "sMagicFrostAtronachID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonGoldenSaint] = "sMagicGoldenSaintID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonGreaterBonewalker] = "sMagicGreaterBonewalkerID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonHunger] = "sMagicHungerID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonScamp] = "sMagicScampID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonSkeletalMinion] = "sMagicSkeletalMinionID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonStormAtronach] = "sMagicStormAtronachID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonWingedTwilight] = "sMagicWingedTwilightID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonWolf] = "sMagicCreature01ID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonBear] = "sMagicCreature02ID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonBonewolf] = "sMagicCreature03ID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonCreature04] = "sMagicCreature04ID";
|
|
|
|
summonMap[ESM::MagicEffect::SummonCreature05] = "sMagicCreature05ID";
|
|
|
|
}
|
|
|
|
|
|
|
|
MWMechanics::CreatureStats& creatureStats = mActor.getClass().getCreatureStats(mActor);
|
|
|
|
|
|
|
|
// Update summon effects
|
|
|
|
std::map<CreatureStats::SummonKey, int>& creatureMap = creatureStats.getSummonedCreatureMap();
|
|
|
|
for (std::map<CreatureStats::SummonKey, int>::iterator it = creatureMap.begin(); it != creatureMap.end(); )
|
|
|
|
{
|
|
|
|
bool found = mActiveEffects.find(it->first) != mActiveEffects.end();
|
|
|
|
if (!found)
|
|
|
|
{
|
|
|
|
// Effect has ended
|
|
|
|
cleanupSummonedCreature(creatureStats, it->second);
|
|
|
|
creatureMap.erase(it++);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (std::set<std::pair<int, std::string> >::iterator it = mActiveEffects.begin(); it != mActiveEffects.end(); ++it)
|
|
|
|
{
|
|
|
|
bool found = creatureMap.find(std::make_pair(it->first, it->second)) != creatureMap.end();
|
|
|
|
if (!found)
|
|
|
|
{
|
|
|
|
ESM::Position ipos = mActor.getRefData().getPosition();
|
2015-06-03 19:37:21 +00:00
|
|
|
osg::Vec3f pos(ipos.asVec3());
|
|
|
|
|
|
|
|
osg::Quat rot (-ipos.rot[2], osg::Vec3f(0,0,1));
|
2015-01-05 17:52:37 +00:00
|
|
|
const float distance = 50;
|
2015-06-03 19:37:21 +00:00
|
|
|
pos = pos + (rot * osg::Vec3f(0,1,0)) * distance;
|
|
|
|
ipos.pos[0] = pos.x();
|
|
|
|
ipos.pos[1] = pos.y();
|
|
|
|
ipos.pos[2] = pos.z();
|
2015-01-05 17:52:37 +00:00
|
|
|
ipos.rot[0] = 0;
|
|
|
|
ipos.rot[1] = 0;
|
|
|
|
ipos.rot[2] = 0;
|
|
|
|
|
|
|
|
const std::string& creatureGmst = summonMap[it->first];
|
|
|
|
std::string creatureID =
|
|
|
|
MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>().find(creatureGmst)->getString();
|
|
|
|
|
|
|
|
if (!creatureID.empty())
|
|
|
|
{
|
|
|
|
MWWorld::CellStore* store = mActor.getCell();
|
2015-12-18 01:36:34 +00:00
|
|
|
int creatureActorId = -1;
|
|
|
|
try
|
|
|
|
{
|
|
|
|
MWWorld::ManualRef ref(MWBase::Environment::get().getWorld()->getStore(), creatureID, 1);
|
|
|
|
ref.getPtr().getCellRef().setPosition(ipos);
|
|
|
|
|
|
|
|
MWMechanics::CreatureStats& summonedCreatureStats = ref.getPtr().getClass().getCreatureStats(ref.getPtr());
|
|
|
|
|
|
|
|
// Make the summoned creature follow its master and help in fights
|
|
|
|
AiFollow package(mActor.getCellRef().getRefId());
|
|
|
|
summonedCreatureStats.getAiSequence().stack(package, ref.getPtr());
|
|
|
|
creatureActorId = summonedCreatureStats.getActorId();
|
|
|
|
|
2016-02-29 16:05:18 +00:00
|
|
|
MWWorld::Ptr placed = MWBase::Environment::get().getWorld()->placeObject(ref.getPtr(),store,ipos);
|
2015-12-18 01:36:34 +00:00
|
|
|
|
|
|
|
MWRender::Animation* anim = MWBase::Environment::get().getWorld()->getAnimation(placed);
|
|
|
|
if (anim)
|
|
|
|
{
|
|
|
|
const ESM::Static* fx = MWBase::Environment::get().getWorld()->getStore().get<ESM::Static>()
|
|
|
|
.search("VFX_Summon_Start");
|
|
|
|
if (fx)
|
|
|
|
anim->addEffect("meshes\\" + fx->mModel, -1, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (std::exception& e)
|
2015-01-05 17:52:37 +00:00
|
|
|
{
|
2015-12-18 01:36:34 +00:00
|
|
|
std::cerr << "Failed to spawn summoned creature: " << e.what() << std::endl;
|
|
|
|
// still insert into creatureMap so we don't try to spawn again every frame, that would spam the warning log
|
2015-01-05 17:52:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
creatureMap.insert(std::make_pair(*it, creatureActorId));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (std::map<CreatureStats::SummonKey, int>::iterator it = creatureMap.begin(); it != creatureMap.end(); )
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = MWBase::Environment::get().getWorld()->searchPtrViaActorId(it->second);
|
|
|
|
if (!ptr.isEmpty() && ptr.getClass().getCreatureStats(ptr).isDead())
|
|
|
|
{
|
|
|
|
// Purge the magic effect so a new creature can be summoned if desired
|
|
|
|
creatureStats.getActiveSpells().purgeEffect(it->first.first, it->first.second);
|
|
|
|
if (mActor.getClass().hasInventoryStore(ptr))
|
|
|
|
mActor.getClass().getInventoryStore(mActor).purgeEffect(it->first.first, it->first.second);
|
|
|
|
|
|
|
|
cleanupSummonedCreature(creatureStats, it->second);
|
|
|
|
creatureMap.erase(it++);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<int>& graveyard = creatureStats.getSummonedCreatureGraveyard();
|
|
|
|
for (std::vector<int>::iterator it = graveyard.begin(); it != graveyard.end(); )
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptr = MWBase::Environment::get().getWorld()->searchPtrViaActorId(*it);
|
|
|
|
if (!ptr.isEmpty())
|
|
|
|
{
|
|
|
|
it = graveyard.erase(it);
|
|
|
|
|
|
|
|
const ESM::Static* fx = MWBase::Environment::get().getWorld()->getStore().get<ESM::Static>()
|
|
|
|
.search("VFX_Summon_End");
|
|
|
|
if (fx)
|
|
|
|
MWBase::Environment::get().getWorld()->spawnEffect("meshes\\" + fx->mModel,
|
2015-04-19 15:55:56 +00:00
|
|
|
"", ptr.getRefData().getPosition().asVec3());
|
2015-01-05 17:52:37 +00:00
|
|
|
|
|
|
|
MWBase::Environment::get().getWorld()->deleteObject(ptr);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|