2014-05-16 13:09:23 +02:00
|
|
|
#include "projectilemanager.hpp"
|
|
|
|
|
2016-09-05 02:52:00 +09:00
|
|
|
#include <iomanip>
|
2016-09-05 00:49:22 +09:00
|
|
|
|
2015-06-01 21:41:13 +02:00
|
|
|
#include <osg/PositionAttitudeTransform>
|
2014-05-16 13:09:23 +02:00
|
|
|
|
2015-07-07 19:16:32 +02:00
|
|
|
#include <components/esm/esmwriter.hpp>
|
2014-05-17 05:21:17 +02:00
|
|
|
#include <components/esm/projectilestate.hpp>
|
2015-06-01 21:41:13 +02:00
|
|
|
#include <components/resource/resourcesystem.hpp>
|
|
|
|
#include <components/resource/scenemanager.hpp>
|
|
|
|
#include <components/sceneutil/controller.hpp>
|
2015-12-05 00:04:23 +01:00
|
|
|
#include <components/sceneutil/visitor.hpp>
|
2015-12-05 00:44:04 +01:00
|
|
|
#include <components/sceneutil/lightmanager.hpp>
|
2014-05-17 05:21:17 +02:00
|
|
|
|
2014-05-16 13:09:23 +02:00
|
|
|
#include "../mwworld/manualref.hpp"
|
|
|
|
#include "../mwworld/class.hpp"
|
2015-02-09 15:01:49 +01:00
|
|
|
#include "../mwworld/esmstore.hpp"
|
2014-05-17 02:52:10 +02:00
|
|
|
#include "../mwworld/inventorystore.hpp"
|
2014-05-16 13:09:23 +02:00
|
|
|
|
|
|
|
#include "../mwbase/soundmanager.hpp"
|
|
|
|
#include "../mwbase/world.hpp"
|
|
|
|
#include "../mwbase/environment.hpp"
|
|
|
|
|
|
|
|
#include "../mwmechanics/combat.hpp"
|
|
|
|
#include "../mwmechanics/creaturestats.hpp"
|
|
|
|
#include "../mwmechanics/spellcasting.hpp"
|
2015-08-21 21:12:39 +12:00
|
|
|
#include "../mwmechanics/actorutil.hpp"
|
2014-05-16 13:09:23 +02:00
|
|
|
|
|
|
|
#include "../mwrender/effectmanager.hpp"
|
2014-08-18 15:32:52 +02:00
|
|
|
#include "../mwrender/animation.hpp"
|
2015-06-01 21:41:13 +02:00
|
|
|
#include "../mwrender/vismask.hpp"
|
2015-12-04 23:28:11 +01:00
|
|
|
#include "../mwrender/renderingmanager.hpp"
|
2014-05-16 13:09:23 +02:00
|
|
|
|
|
|
|
#include "../mwsound/sound.hpp"
|
|
|
|
|
2015-06-01 21:41:13 +02:00
|
|
|
#include "../mwphysics/physicssystem.hpp"
|
|
|
|
|
|
|
|
|
2014-05-16 13:09:23 +02:00
|
|
|
namespace MWWorld
|
|
|
|
{
|
|
|
|
|
2015-12-04 23:28:11 +01:00
|
|
|
ProjectileManager::ProjectileManager(osg::Group* parent, Resource::ResourceSystem* resourceSystem,
|
|
|
|
MWRender::RenderingManager* rendering, MWPhysics::PhysicsSystem* physics)
|
2015-06-01 21:41:13 +02:00
|
|
|
: mParent(parent)
|
|
|
|
, mResourceSystem(resourceSystem)
|
2015-12-04 23:28:11 +01:00
|
|
|
, mRendering(rendering)
|
2015-06-01 21:41:13 +02:00
|
|
|
, mPhysics(physics)
|
2014-05-16 13:09:23 +02:00
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2015-12-05 00:03:25 +01:00
|
|
|
/// Rotates an osg::PositionAttitudeTransform over time.
|
|
|
|
class RotateCallback : public osg::NodeCallback
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
RotateCallback(const osg::Vec3f& axis = osg::Vec3f(0,-1,0), float rotateSpeed = osg::PI*2)
|
|
|
|
: mAxis(axis)
|
|
|
|
, mRotateSpeed(rotateSpeed)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void operator()(osg::Node* node, osg::NodeVisitor* nv)
|
|
|
|
{
|
|
|
|
osg::PositionAttitudeTransform* transform = static_cast<osg::PositionAttitudeTransform*>(node);
|
|
|
|
|
|
|
|
double time = nv->getFrameStamp()->getSimulationTime();
|
|
|
|
|
|
|
|
osg::Quat orient = osg::Quat(time * mRotateSpeed, mAxis);
|
|
|
|
transform->setAttitude(orient);
|
|
|
|
|
|
|
|
traverse(node, nv);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
osg::Vec3f mAxis;
|
|
|
|
float mRotateSpeed;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
void ProjectileManager::createModel(State &state, const std::string &model, const osg::Vec3f& pos, const osg::Quat& orient, bool rotate)
|
2014-05-16 13:09:23 +02:00
|
|
|
{
|
2015-06-01 21:41:13 +02:00
|
|
|
state.mNode = new osg::PositionAttitudeTransform;
|
|
|
|
state.mNode->setNodeMask(MWRender::Mask_Effect);
|
2015-06-11 16:40:26 +02:00
|
|
|
state.mNode->setPosition(pos);
|
|
|
|
state.mNode->setAttitude(orient);
|
2014-08-18 15:32:52 +02:00
|
|
|
|
2015-12-05 00:03:25 +01:00
|
|
|
osg::Group* attachTo = state.mNode;
|
|
|
|
|
|
|
|
if (rotate)
|
|
|
|
{
|
|
|
|
osg::ref_ptr<osg::PositionAttitudeTransform> rotateNode (new osg::PositionAttitudeTransform);
|
|
|
|
rotateNode->addUpdateCallback(new RotateCallback());
|
|
|
|
state.mNode->addChild(rotateNode);
|
|
|
|
attachTo = rotateNode;
|
|
|
|
}
|
|
|
|
|
2016-09-05 00:49:22 +09:00
|
|
|
osg::ref_ptr<osg::Node> ptr = mResourceSystem->getSceneManager()->getInstance(model, attachTo);
|
2015-12-05 00:03:25 +01:00
|
|
|
|
2016-09-04 17:39:14 +09:00
|
|
|
if (state.mIdMagic.size() > 1)
|
|
|
|
for (size_t iter = 1; iter != state.mIdMagic.size(); ++iter)
|
|
|
|
{
|
2016-09-05 02:52:00 +09:00
|
|
|
std::ostringstream nodeName;
|
|
|
|
nodeName << "Dummy" << std::setw(2) << std::setfill('0') << iter;
|
2016-09-04 17:39:14 +09:00
|
|
|
const ESM::Weapon* weapon = MWBase::Environment::get().getWorld()->getStore().get<ESM::Weapon>().find (state.mIdMagic.at(iter));
|
2016-09-05 02:59:33 +09:00
|
|
|
SceneUtil::FindByNameVisitor findVisitor(nodeName.str());
|
|
|
|
attachTo->accept(findVisitor);
|
|
|
|
if (findVisitor.mFoundNode)
|
|
|
|
mResourceSystem->getSceneManager()->getInstance("meshes\\" + weapon->mModel, findVisitor.mFoundNode);
|
2016-09-04 17:39:14 +09:00
|
|
|
}
|
|
|
|
|
2015-12-05 00:04:23 +01:00
|
|
|
SceneUtil::DisableFreezeOnCullVisitor disableFreezeOnCullVisitor;
|
|
|
|
state.mNode->accept(disableFreezeOnCullVisitor);
|
|
|
|
|
2015-12-05 00:44:04 +01:00
|
|
|
state.mNode->addCullCallback(new SceneUtil::LightListCallback);
|
|
|
|
|
2015-12-05 00:03:25 +01:00
|
|
|
mParent->addChild(state.mNode);
|
2015-06-01 21:41:13 +02:00
|
|
|
|
|
|
|
state.mEffectAnimationTime.reset(new MWRender::EffectAnimationTime);
|
|
|
|
|
|
|
|
SceneUtil::AssignControllerSourcesVisitor assignVisitor (state.mEffectAnimationTime);
|
|
|
|
state.mNode->accept(assignVisitor);
|
2014-05-16 13:09:23 +02:00
|
|
|
}
|
|
|
|
|
2015-06-01 21:41:13 +02:00
|
|
|
void ProjectileManager::update(State& state, float duration)
|
2014-05-16 13:09:23 +02:00
|
|
|
{
|
2015-06-01 21:41:13 +02:00
|
|
|
state.mEffectAnimationTime->addTime(duration);
|
2014-05-16 13:09:23 +02:00
|
|
|
}
|
|
|
|
|
2016-09-04 17:39:14 +09:00
|
|
|
void ProjectileManager::launchMagicBolt(const std::vector<std::string> &projectileIDs, const std::vector<std::string> &sounds,
|
2014-05-16 13:09:23 +02:00
|
|
|
const std::string &spellId, float speed, bool stack,
|
2014-06-18 01:41:07 +02:00
|
|
|
const ESM::EffectList &effects, const Ptr &caster, const std::string &sourceName,
|
2015-06-01 21:41:13 +02:00
|
|
|
const osg::Vec3f& fallbackDirection)
|
2014-05-16 13:09:23 +02:00
|
|
|
{
|
2016-06-10 23:33:47 +02:00
|
|
|
osg::Vec3f pos = caster.getRefData().getPosition().asVec3();
|
2016-03-05 15:56:54 +01:00
|
|
|
if (caster.getClass().isActor())
|
2016-06-10 23:33:47 +02:00
|
|
|
{
|
|
|
|
// Spawn at 0.75 * ActorHeight
|
|
|
|
// Note: we ignore the collision box offset, this is required to make some flying creatures work as intended.
|
|
|
|
pos.z() += mPhysics->getHalfExtents(caster).z() * 2 * 0.75;
|
|
|
|
}
|
2014-05-16 13:09:23 +02:00
|
|
|
|
2014-06-28 14:49:07 +02:00
|
|
|
if (MWBase::Environment::get().getWorld()->isUnderwater(caster.getCell(), pos)) // Underwater casting not possible
|
|
|
|
return;
|
|
|
|
|
2015-06-01 21:41:13 +02:00
|
|
|
osg::Quat orient;
|
2014-06-18 01:41:07 +02:00
|
|
|
if (caster.getClass().isActor())
|
2015-06-01 21:41:13 +02:00
|
|
|
orient = osg::Quat(caster.getRefData().getPosition().rot[0], osg::Vec3f(-1,0,0))
|
|
|
|
* osg::Quat(caster.getRefData().getPosition().rot[2], osg::Vec3f(0,0,-1));
|
2014-06-18 01:41:07 +02:00
|
|
|
else
|
2015-06-01 21:41:13 +02:00
|
|
|
orient.makeRotate(osg::Vec3f(0,1,0), osg::Vec3f(fallbackDirection));
|
2014-05-16 13:09:23 +02:00
|
|
|
|
|
|
|
MagicBoltState state;
|
|
|
|
state.mSourceName = sourceName;
|
2014-05-17 05:21:17 +02:00
|
|
|
state.mSpellId = spellId;
|
2015-06-01 21:41:13 +02:00
|
|
|
state.mCasterHandle = caster;
|
2014-06-18 01:41:07 +02:00
|
|
|
if (caster.getClass().isActor())
|
|
|
|
state.mActorId = caster.getClass().getCreatureStats(caster).getActorId();
|
|
|
|
else
|
|
|
|
state.mActorId = -1;
|
2014-05-16 13:09:23 +02:00
|
|
|
state.mSpeed = speed;
|
|
|
|
state.mStack = stack;
|
2016-09-04 17:39:14 +09:00
|
|
|
state.mIdMagic = projectileIDs;
|
2016-09-05 03:31:48 +09:00
|
|
|
state.mSoundIds = sounds;
|
2014-05-16 13:09:23 +02:00
|
|
|
|
2016-09-04 01:54:09 +09:00
|
|
|
// Should have already had non-projectile effects removed
|
|
|
|
state.mEffects = effects;
|
2014-05-16 13:09:23 +02:00
|
|
|
|
2016-09-04 17:39:14 +09:00
|
|
|
MWWorld::ManualRef ref(MWBase::Environment::get().getWorld()->getStore(), projectileIDs.at(0));
|
|
|
|
MWWorld::Ptr ptr = ref.getPtr();
|
2014-05-16 13:09:23 +02:00
|
|
|
|
2016-09-04 17:39:14 +09:00
|
|
|
createModel(state, ptr.getClass().getModel(ptr), pos, orient, true);
|
2014-05-16 13:09:23 +02:00
|
|
|
|
2016-09-04 17:39:14 +09:00
|
|
|
MWBase::SoundManager *sndMgr = MWBase::Environment::get().getSoundManager();
|
2016-09-04 23:57:06 +09:00
|
|
|
for (size_t it = 0; it != sounds.size(); it++)
|
|
|
|
{
|
2016-09-05 03:31:48 +09:00
|
|
|
state.mSounds.push_back(sndMgr->playSound3D(pos, sounds.at(it), 1.0f, 1.0f, MWBase::SoundManager::Play_TypeSfx, MWBase::SoundManager::Play_Loop));
|
2016-09-04 23:57:06 +09:00
|
|
|
}
|
|
|
|
|
2014-05-16 13:09:23 +02:00
|
|
|
mMagicBolts.push_back(state);
|
|
|
|
}
|
|
|
|
|
2015-12-18 17:13:54 +01:00
|
|
|
void ProjectileManager::launchProjectile(Ptr actor, ConstPtr projectile, const osg::Vec3f &pos, const osg::Quat &orient, Ptr bow, float speed, float attackStrength)
|
2014-05-16 13:09:23 +02:00
|
|
|
{
|
|
|
|
ProjectileState state;
|
|
|
|
state.mActorId = actor.getClass().getCreatureStats(actor).getActorId();
|
2014-05-25 14:13:07 +02:00
|
|
|
state.mBowId = bow.getCellRef().getRefId();
|
2015-06-01 21:41:13 +02:00
|
|
|
state.mVelocity = orient * osg::Vec3f(0,1,0) * speed;
|
2016-09-04 01:54:09 +09:00
|
|
|
state.mIdArrow = projectile.getCellRef().getRefId();
|
2015-06-01 21:41:13 +02:00
|
|
|
state.mCasterHandle = actor;
|
2015-06-26 05:15:07 +02:00
|
|
|
state.mAttackStrength = attackStrength;
|
2014-05-16 13:09:23 +02:00
|
|
|
|
2014-05-25 14:13:07 +02:00
|
|
|
MWWorld::ManualRef ref(MWBase::Environment::get().getWorld()->getStore(), projectile.getCellRef().getRefId());
|
2014-05-16 13:09:23 +02:00
|
|
|
MWWorld::Ptr ptr = ref.getPtr();
|
|
|
|
|
2015-12-05 00:03:25 +01:00
|
|
|
createModel(state, ptr.getClass().getModel(ptr), pos, orient, false);
|
2014-05-16 13:09:23 +02:00
|
|
|
|
|
|
|
mProjectiles.push_back(state);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ProjectileManager::update(float dt)
|
|
|
|
{
|
|
|
|
moveProjectiles(dt);
|
|
|
|
moveMagicBolts(dt);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ProjectileManager::moveMagicBolts(float duration)
|
|
|
|
{
|
|
|
|
for (std::vector<MagicBoltState>::iterator it = mMagicBolts.begin(); it != mMagicBolts.end();)
|
|
|
|
{
|
2015-06-01 21:41:13 +02:00
|
|
|
osg::Quat orient = it->mNode->getAttitude();
|
2014-05-16 13:09:23 +02:00
|
|
|
static float fTargetSpellMaxSpeed = MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>()
|
|
|
|
.find("fTargetSpellMaxSpeed")->getFloat();
|
|
|
|
float speed = fTargetSpellMaxSpeed * it->mSpeed;
|
|
|
|
|
2015-06-01 21:41:13 +02:00
|
|
|
osg::Vec3f direction = orient * osg::Vec3f(0,1,0);
|
|
|
|
direction.normalize();
|
|
|
|
osg::Vec3f pos(it->mNode->getPosition());
|
|
|
|
osg::Vec3f newPos = pos + direction * duration * speed;
|
2014-05-16 13:09:23 +02:00
|
|
|
|
2016-09-05 03:31:48 +09:00
|
|
|
for (size_t soundIter = 0; soundIter != it->mSounds.size(); soundIter++)
|
2016-09-04 23:57:06 +09:00
|
|
|
{
|
2016-09-05 03:31:48 +09:00
|
|
|
it->mSounds.at(soundIter)->setPosition(newPos);
|
2016-09-04 23:57:06 +09:00
|
|
|
}
|
2014-05-16 13:09:23 +02:00
|
|
|
|
|
|
|
it->mNode->setPosition(newPos);
|
|
|
|
|
2015-06-01 21:41:13 +02:00
|
|
|
update(*it, duration);
|
|
|
|
|
|
|
|
MWWorld::Ptr caster = it->getCaster();
|
2014-05-16 13:09:23 +02:00
|
|
|
|
|
|
|
// Check for impact
|
|
|
|
// TODO: use a proper btRigidBody / btGhostObject?
|
2015-06-01 21:41:13 +02:00
|
|
|
MWPhysics::PhysicsSystem::RayResult result = mPhysics->castRay(pos, newPos, caster, 0xff, MWPhysics::CollisionType_Projectile);
|
2014-05-16 13:09:23 +02:00
|
|
|
|
2015-06-01 21:41:13 +02:00
|
|
|
bool hit = false;
|
|
|
|
if (result.mHit)
|
2014-05-16 13:09:23 +02:00
|
|
|
{
|
2015-06-01 21:41:13 +02:00
|
|
|
hit = true;
|
|
|
|
if (result.mHitObject.isEmpty())
|
2014-05-16 13:09:23 +02:00
|
|
|
{
|
2015-06-01 21:41:13 +02:00
|
|
|
// terrain
|
2014-05-16 13:09:23 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-06-01 21:41:13 +02:00
|
|
|
MWMechanics::CastSpell cast(caster, result.mHitObject);
|
2014-05-16 13:09:23 +02:00
|
|
|
cast.mHitPosition = pos;
|
2014-05-17 05:21:17 +02:00
|
|
|
cast.mId = it->mSpellId;
|
2014-05-16 13:09:23 +02:00
|
|
|
cast.mSourceName = it->mSourceName;
|
|
|
|
cast.mStack = it->mStack;
|
2015-06-01 21:41:13 +02:00
|
|
|
cast.inflict(result.mHitObject, caster, it->mEffects, ESM::RT_Target, false, true);
|
2014-05-16 13:09:23 +02:00
|
|
|
}
|
|
|
|
}
|
2014-06-28 14:49:07 +02:00
|
|
|
|
|
|
|
// Explodes when hitting water
|
2015-08-21 21:12:39 +12:00
|
|
|
if (MWBase::Environment::get().getWorld()->isUnderwater(MWMechanics::getPlayer().getCell(), newPos))
|
2014-06-28 14:49:07 +02:00
|
|
|
hit = true;
|
|
|
|
|
2014-05-16 13:09:23 +02:00
|
|
|
if (hit)
|
|
|
|
{
|
2016-01-12 00:31:34 +01:00
|
|
|
MWBase::Environment::get().getWorld()->explodeSpell(pos, it->mEffects, caster, result.mHitObject,
|
|
|
|
ESM::RT_Target, it->mSpellId, it->mSourceName);
|
2014-05-16 13:09:23 +02:00
|
|
|
|
2016-09-05 03:31:48 +09:00
|
|
|
for (size_t soundIter = 0; soundIter != it->mSounds.size(); soundIter++)
|
2016-09-04 23:57:06 +09:00
|
|
|
{
|
2016-09-05 03:31:48 +09:00
|
|
|
MWBase::Environment::get().getSoundManager()->stopSound(it->mSounds.at(soundIter));
|
2016-09-04 23:57:06 +09:00
|
|
|
}
|
|
|
|
|
2015-06-01 21:41:13 +02:00
|
|
|
mParent->removeChild(it->mNode);
|
2014-05-16 13:09:23 +02:00
|
|
|
|
|
|
|
it = mMagicBolts.erase(it);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ProjectileManager::moveProjectiles(float duration)
|
|
|
|
{
|
|
|
|
for (std::vector<ProjectileState>::iterator it = mProjectiles.begin(); it != mProjectiles.end();)
|
|
|
|
{
|
|
|
|
// gravity constant - must be way lower than the gravity affecting actors, since we're not
|
|
|
|
// simulating aerodynamics at all
|
2015-06-01 21:41:13 +02:00
|
|
|
it->mVelocity -= osg::Vec3f(0, 0, 627.2f * 0.1f) * duration;
|
2014-05-16 13:09:23 +02:00
|
|
|
|
2015-06-01 21:41:13 +02:00
|
|
|
osg::Vec3f pos(it->mNode->getPosition());
|
|
|
|
osg::Vec3f newPos = pos + it->mVelocity * duration;
|
2014-05-16 13:09:23 +02:00
|
|
|
|
2015-06-01 21:41:13 +02:00
|
|
|
osg::Quat orient;
|
|
|
|
orient.makeRotate(osg::Vec3f(0,1,0), it->mVelocity);
|
|
|
|
it->mNode->setAttitude(orient);
|
2014-05-16 13:09:23 +02:00
|
|
|
it->mNode->setPosition(newPos);
|
|
|
|
|
2015-06-01 21:41:13 +02:00
|
|
|
update(*it, duration);
|
|
|
|
|
|
|
|
MWWorld::Ptr caster = it->getCaster();
|
2014-05-16 13:09:23 +02:00
|
|
|
|
|
|
|
// Check for impact
|
|
|
|
// TODO: use a proper btRigidBody / btGhostObject?
|
2015-06-01 21:41:13 +02:00
|
|
|
MWPhysics::PhysicsSystem::RayResult result = mPhysics->castRay(pos, newPos, caster, 0xff, MWPhysics::CollisionType_Projectile);
|
2014-05-16 13:09:23 +02:00
|
|
|
|
2015-12-04 23:28:11 +01:00
|
|
|
bool underwater = MWBase::Environment::get().getWorld()->isUnderwater(MWMechanics::getPlayer().getCell(), newPos);
|
|
|
|
if (result.mHit || underwater)
|
2014-05-16 13:09:23 +02:00
|
|
|
{
|
2015-12-04 23:28:11 +01:00
|
|
|
if (result.mHit)
|
2014-05-16 13:09:23 +02:00
|
|
|
{
|
2016-09-04 01:54:09 +09:00
|
|
|
MWWorld::ManualRef projectileRef(MWBase::Environment::get().getWorld()->getStore(), it->mIdArrow);
|
2015-12-04 23:28:11 +01:00
|
|
|
|
|
|
|
// Try to get a Ptr to the bow that was used. It might no longer exist.
|
|
|
|
MWWorld::Ptr bow = projectileRef.getPtr();
|
|
|
|
if (!caster.isEmpty())
|
|
|
|
{
|
|
|
|
MWWorld::InventoryStore& inv = caster.getClass().getInventoryStore(caster);
|
|
|
|
MWWorld::ContainerStoreIterator invIt = inv.getSlot(MWWorld::InventoryStore::Slot_CarriedRight);
|
|
|
|
if (invIt != inv.end() && Misc::StringUtils::ciEqual(invIt->getCellRef().getRefId(), it->mBowId))
|
|
|
|
bow = *invIt;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (caster.isEmpty())
|
|
|
|
caster = result.mHitObject;
|
|
|
|
|
|
|
|
MWMechanics::projectileHit(caster, result.mHitObject, bow, projectileRef.getPtr(), result.mHitPos, it->mAttackStrength);
|
2014-05-16 13:09:23 +02:00
|
|
|
}
|
2014-08-02 23:14:17 +02:00
|
|
|
|
2015-12-04 23:28:11 +01:00
|
|
|
if (underwater)
|
|
|
|
mRendering->emitWaterRipple(newPos);
|
2014-08-02 23:14:17 +02:00
|
|
|
|
2015-06-01 21:41:13 +02:00
|
|
|
mParent->removeChild(it->mNode);
|
2014-05-16 13:09:23 +02:00
|
|
|
it = mProjectiles.erase(it);
|
|
|
|
continue;
|
|
|
|
}
|
2015-12-04 23:28:11 +01:00
|
|
|
|
|
|
|
++it;
|
2014-05-16 13:09:23 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ProjectileManager::clear()
|
|
|
|
{
|
|
|
|
for (std::vector<ProjectileState>::iterator it = mProjectiles.begin(); it != mProjectiles.end(); ++it)
|
|
|
|
{
|
2015-06-01 21:41:13 +02:00
|
|
|
mParent->removeChild(it->mNode);
|
2014-05-16 13:09:23 +02:00
|
|
|
}
|
|
|
|
mProjectiles.clear();
|
|
|
|
for (std::vector<MagicBoltState>::iterator it = mMagicBolts.begin(); it != mMagicBolts.end(); ++it)
|
|
|
|
{
|
2015-06-01 21:41:13 +02:00
|
|
|
mParent->removeChild(it->mNode);
|
2016-09-05 03:31:48 +09:00
|
|
|
for (size_t soundIter = 0; soundIter != it->mSounds.size(); soundIter++)
|
2016-09-04 23:57:06 +09:00
|
|
|
{
|
2016-09-05 03:31:48 +09:00
|
|
|
MWBase::Environment::get().getSoundManager()->stopSound(it->mSounds.at(soundIter));
|
2016-09-04 23:57:06 +09:00
|
|
|
}
|
2014-05-16 13:09:23 +02:00
|
|
|
}
|
|
|
|
mMagicBolts.clear();
|
|
|
|
}
|
|
|
|
|
2014-05-17 05:21:17 +02:00
|
|
|
void ProjectileManager::write(ESM::ESMWriter &writer, Loading::Listener &progress) const
|
|
|
|
{
|
|
|
|
for (std::vector<ProjectileState>::const_iterator it = mProjectiles.begin(); it != mProjectiles.end(); ++it)
|
|
|
|
{
|
|
|
|
writer.startRecord(ESM::REC_PROJ);
|
|
|
|
|
|
|
|
ESM::ProjectileState state;
|
2016-09-04 01:54:09 +09:00
|
|
|
state.mId = it->mIdArrow;
|
2015-06-01 21:41:13 +02:00
|
|
|
state.mPosition = ESM::Vector3(osg::Vec3f(it->mNode->getPosition()));
|
|
|
|
state.mOrientation = ESM::Quaternion(osg::Quat(it->mNode->getAttitude()));
|
2014-05-17 05:21:17 +02:00
|
|
|
state.mActorId = it->mActorId;
|
|
|
|
|
|
|
|
state.mBowId = it->mBowId;
|
|
|
|
state.mVelocity = it->mVelocity;
|
2015-06-26 02:32:41 +02:00
|
|
|
state.mAttackStrength = it->mAttackStrength;
|
2014-05-17 05:21:17 +02:00
|
|
|
|
|
|
|
state.save(writer);
|
|
|
|
|
|
|
|
writer.endRecord(ESM::REC_PROJ);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (std::vector<MagicBoltState>::const_iterator it = mMagicBolts.begin(); it != mMagicBolts.end(); ++it)
|
|
|
|
{
|
|
|
|
writer.startRecord(ESM::REC_MPRJ);
|
|
|
|
|
|
|
|
ESM::MagicBoltState state;
|
2016-09-04 01:54:09 +09:00
|
|
|
state.mId = it->mIdMagic.at(0);
|
2015-06-01 21:41:13 +02:00
|
|
|
state.mPosition = ESM::Vector3(osg::Vec3f(it->mNode->getPosition()));
|
|
|
|
state.mOrientation = ESM::Quaternion(osg::Quat(it->mNode->getAttitude()));
|
2014-05-17 05:21:17 +02:00
|
|
|
state.mActorId = it->mActorId;
|
|
|
|
|
|
|
|
state.mSpellId = it->mSpellId;
|
|
|
|
state.mEffects = it->mEffects;
|
2016-09-05 03:31:48 +09:00
|
|
|
state.mSound = it->mSoundIds.at(0);
|
2014-05-17 05:21:17 +02:00
|
|
|
state.mSourceName = it->mSourceName;
|
|
|
|
state.mSpeed = it->mSpeed;
|
|
|
|
state.mStack = it->mStack;
|
|
|
|
|
|
|
|
state.save(writer);
|
|
|
|
|
|
|
|
writer.endRecord(ESM::REC_MPRJ);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-22 19:04:59 +01:00
|
|
|
bool ProjectileManager::readRecord(ESM::ESMReader &reader, uint32_t type)
|
2014-05-17 05:21:17 +02:00
|
|
|
{
|
|
|
|
if (type == ESM::REC_PROJ)
|
|
|
|
{
|
|
|
|
ESM::ProjectileState esm;
|
|
|
|
esm.load(reader);
|
|
|
|
|
|
|
|
ProjectileState state;
|
|
|
|
state.mActorId = esm.mActorId;
|
|
|
|
state.mBowId = esm.mBowId;
|
|
|
|
state.mVelocity = esm.mVelocity;
|
2016-09-04 01:54:09 +09:00
|
|
|
state.mIdArrow = esm.mId;
|
2015-06-26 02:32:41 +02:00
|
|
|
state.mAttackStrength = esm.mAttackStrength;
|
2014-05-17 05:21:17 +02:00
|
|
|
|
|
|
|
std::string model;
|
|
|
|
try
|
|
|
|
{
|
|
|
|
MWWorld::ManualRef ref(MWBase::Environment::get().getWorld()->getStore(), esm.mId);
|
|
|
|
MWWorld::Ptr ptr = ref.getPtr();
|
|
|
|
model = ptr.getClass().getModel(ptr);
|
|
|
|
}
|
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-12-05 00:03:25 +01:00
|
|
|
createModel(state, model, osg::Vec3f(esm.mPosition), osg::Quat(esm.mOrientation), false);
|
2014-05-17 05:21:17 +02:00
|
|
|
|
|
|
|
mProjectiles.push_back(state);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else if (type == ESM::REC_MPRJ)
|
|
|
|
{
|
|
|
|
ESM::MagicBoltState esm;
|
|
|
|
esm.load(reader);
|
|
|
|
|
|
|
|
MagicBoltState state;
|
|
|
|
state.mSourceName = esm.mSourceName;
|
2016-09-04 01:54:09 +09:00
|
|
|
state.mIdMagic.push_back(esm.mId);
|
2014-05-17 05:21:17 +02:00
|
|
|
state.mSpellId = esm.mSpellId;
|
|
|
|
state.mActorId = esm.mActorId;
|
|
|
|
state.mSpeed = esm.mSpeed;
|
|
|
|
state.mStack = esm.mStack;
|
|
|
|
state.mEffects = esm.mEffects;
|
|
|
|
|
2016-09-05 03:31:48 +09:00
|
|
|
std::string projectileID;
|
|
|
|
std::vector<std::string> projectileIDs;
|
|
|
|
|
|
|
|
if (esm.mEffects.mList.size() > 1)
|
|
|
|
{
|
|
|
|
std::ostringstream ID;
|
|
|
|
ID << "VFX_Multiple" << esm.mEffects.mList.size();
|
|
|
|
state.mIdMagic.push_back(ID.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
for (std::vector<ESM::ENAMstruct>::const_iterator iter (esm.mEffects.mList.begin());
|
|
|
|
iter != esm.mEffects.mList.end(); ++iter)
|
|
|
|
{
|
|
|
|
const ESM::MagicEffect *magicEffect = MWBase::Environment::get().getWorld()->getStore().get<ESM::MagicEffect>().find (
|
|
|
|
iter->mEffectID);
|
|
|
|
|
|
|
|
projectileID = magicEffect->mBolt;
|
|
|
|
if (projectileID.empty())
|
|
|
|
projectileID = "VFX_DefaultBolt";
|
|
|
|
state.mIdMagic.push_back(projectileID);
|
|
|
|
|
|
|
|
static const std::string schools[] = {
|
|
|
|
"alteration", "conjuration", "destruction", "illusion", "mysticism", "restoration"
|
|
|
|
};
|
|
|
|
|
|
|
|
if (!magicEffect->mBoltSound.empty())
|
|
|
|
state.mSoundIds.push_back(magicEffect->mBoltSound);
|
|
|
|
else
|
|
|
|
state.mSoundIds.push_back(schools[magicEffect->mData.mSchool] + " bolt");
|
|
|
|
}
|
|
|
|
|
2014-05-17 05:21:17 +02:00
|
|
|
std::string model;
|
|
|
|
try
|
|
|
|
{
|
2016-09-05 03:31:48 +09:00
|
|
|
MWWorld::ManualRef ref(MWBase::Environment::get().getWorld()->getStore(), state.mIdMagic.at(0));
|
2014-05-17 05:21:17 +02:00
|
|
|
MWWorld::Ptr ptr = ref.getPtr();
|
|
|
|
model = ptr.getClass().getModel(ptr);
|
|
|
|
}
|
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-12-05 00:03:25 +01:00
|
|
|
createModel(state, model, osg::Vec3f(esm.mPosition), osg::Quat(esm.mOrientation), true);
|
2014-05-17 05:21:17 +02:00
|
|
|
|
|
|
|
MWBase::SoundManager *sndMgr = MWBase::Environment::get().getSoundManager();
|
2016-09-05 03:31:48 +09:00
|
|
|
|
|
|
|
for (size_t soundIter = 0; soundIter != state.mSoundIds.size(); soundIter++)
|
|
|
|
{
|
|
|
|
state.mSounds.push_back(sndMgr->playSound3D(esm.mPosition, state.mSoundIds.at(soundIter), 1.0f, 1.0f,
|
|
|
|
MWBase::SoundManager::Play_TypeSfx, MWBase::SoundManager::Play_Loop));
|
|
|
|
}
|
2014-05-17 05:21:17 +02:00
|
|
|
|
|
|
|
mMagicBolts.push_back(state);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ProjectileManager::countSavedGameRecords() const
|
|
|
|
{
|
|
|
|
return mMagicBolts.size() + mProjectiles.size();
|
|
|
|
}
|
|
|
|
|
2015-06-01 21:41:13 +02:00
|
|
|
MWWorld::Ptr ProjectileManager::State::getCaster()
|
|
|
|
{
|
|
|
|
if (!mCasterHandle.isEmpty())
|
|
|
|
return mCasterHandle;
|
|
|
|
|
|
|
|
return MWBase::Environment::get().getWorld()->searchPtrViaActorId(mActorId);
|
|
|
|
}
|
|
|
|
|
2014-05-16 13:09:23 +02:00
|
|
|
}
|