Make non-actors glow when they cast spells

pull/35/head
Allofich 9 years ago
parent dca7b4beb7
commit 3841a8fb40

@ -1531,10 +1531,11 @@ void CharacterController::update(float duration)
osg::Vec3f movement(0.f, 0.f, 0.f); osg::Vec3f movement(0.f, 0.f, 0.f);
float speed = 0.f; float speed = 0.f;
updateMagicEffects();
if(!cls.isActor()) if(!cls.isActor())
{ {
updateMagicEffects();
MWRender::Animation* animation = MWBase::Environment::get().getWorld()->getAnimation(mPtr);
animation->updateSpellGlow(duration);
if(mAnimQueue.size() > 1) if(mAnimQueue.size() > 1)
{ {
if(mAnimation->isPlaying(mAnimQueue.front().mGroup) == false) if(mAnimation->isPlaying(mAnimQueue.front().mGroup) == false)

@ -943,6 +943,8 @@ namespace MWMechanics
const ESM::MagicEffect *effect; const ESM::MagicEffect *effect;
effect = store.get<ESM::MagicEffect>().find(effectentry.mEffectID); effect = store.get<ESM::MagicEffect>().find(effectentry.mEffectID);
MWRender::Animation* animation = MWBase::Environment::get().getWorld()->getAnimation(mCaster);
if (mCaster.getClass().isActor()) // TODO: Non-actors (except for large statics?) should also create a visual casting effect if (mCaster.getClass().isActor()) // TODO: Non-actors (except for large statics?) should also create a visual casting effect
{ {
const ESM::Static* castStatic; const ESM::Static* castStatic;
@ -951,8 +953,17 @@ namespace MWMechanics
else else
castStatic = store.get<ESM::Static>().find ("VFX_DefaultCast"); castStatic = store.get<ESM::Static>().find ("VFX_DefaultCast");
MWBase::Environment::get().getWorld()->getAnimation(mCaster)->addEffect( animation->addEffect("meshes\\" + castStatic->mModel, effect->mIndex);
"meshes\\" + castStatic->mModel, effect->mIndex); }
if (!mCaster.getClass().isActor())
{
osg::Vec4f glowcolor(1,1,1,1);
glowcolor.x() = effect->mData.mRed / 255.f;
glowcolor.y() = effect->mData.mGreen / 255.f;
glowcolor.z() = effect->mData.mBlue / 255.f;
animation->addSpellCastGlow(glowcolor);
} }
static const std::string schools[] = { static const std::string schools[] = {

@ -38,6 +38,7 @@
#include "../mwworld/esmstore.hpp" #include "../mwworld/esmstore.hpp"
#include "../mwworld/class.hpp" #include "../mwworld/class.hpp"
#include "../mwworld/cellstore.hpp" #include "../mwworld/cellstore.hpp"
#include "../mwbase/mechanicsmanager.hpp"
#include "../mwmechanics/character.hpp" // FIXME: for MWMechanics::Priority #include "../mwmechanics/character.hpp" // FIXME: for MWMechanics::Priority
@ -89,10 +90,11 @@ namespace
class GlowUpdater : public SceneUtil::StateSetUpdater class GlowUpdater : public SceneUtil::StateSetUpdater
{ {
public: public:
GlowUpdater(int texUnit, osg::Vec4f color, const std::vector<osg::ref_ptr<osg::Texture2D> >& textures) GlowUpdater(int texUnit, osg::Vec4f color, const std::vector<osg::ref_ptr<osg::Texture2D> >& textures, bool hasDuration)
: mTexUnit(texUnit) : mTexUnit(texUnit)
, mColor(color) , mColor(color)
, mTextures(textures) , mTextures(textures)
, mHasDuration(hasDuration)
{ {
} }
@ -122,10 +124,26 @@ namespace
stateset->setTextureAttribute(mTexUnit, mTextures[index], osg::StateAttribute::ON|osg::StateAttribute::OVERRIDE); stateset->setTextureAttribute(mTexUnit, mTextures[index], osg::StateAttribute::ON|osg::StateAttribute::OVERRIDE);
} }
bool const getHasDuration()
{
return mHasDuration;
}
std::vector<osg::ref_ptr<osg::Texture2D> > const getTextures()
{
return mTextures;
}
int const getTexUnit()
{
return mTexUnit;
}
private: private:
int mTexUnit; int mTexUnit;
osg::Vec4f mColor; osg::Vec4f mColor;
std::vector<osg::ref_ptr<osg::Texture2D> > mTextures; std::vector<osg::ref_ptr<osg::Texture2D> > mTextures;
bool mHasDuration;
}; };
class NodeMapVisitor : public osg::NodeVisitor class NodeMapVisitor : public osg::NodeVisitor
@ -340,6 +358,7 @@ namespace MWRender
, mHeadYawRadians(0.f) , mHeadYawRadians(0.f)
, mHeadPitchRadians(0.f) , mHeadPitchRadians(0.f)
, mAlpha(1.f) , mAlpha(1.f)
, mSpellGlowDuration(0.f)
{ {
for(size_t i = 0;i < sNumBlendMasks;i++) for(size_t i = 0;i < sNumBlendMasks;i++)
mAnimationTimePtr[i].reset(new AnimationTime); mAnimationTimePtr[i].reset(new AnimationTime);
@ -1106,7 +1125,44 @@ namespace MWRender
int mLowestUnusedTexUnit; int mLowestUnusedTexUnit;
}; };
void Animation::addGlow(osg::ref_ptr<osg::Node> node, osg::Vec4f glowColor) void Animation::addSpellCastGlow(osg::Vec4f glowColor){
addGlow(mObjectRoot, glowColor, true);
MWBase::Environment::get().getMechanicsManager()->add(mPtr);
}
void Animation::updateSpellGlow(float duration){
if (mGlowUpdater != NULL && (mGlowUpdater->getHasDuration()))
mSpellGlowDuration += duration;
if (mSpellGlowDuration >= 1.5f) // length of spell glow effect was measured from original game as around 1.5 seconds
removeSpellGlow();
}
void Animation::removeSpellGlow()
{
osg::ref_ptr<osg::StateSet> writableStateSet = NULL;
if (!mObjectRoot->getStateSet())
writableStateSet = mObjectRoot->getOrCreateStateSet();
else
writableStateSet = osg::clone(mObjectRoot->getStateSet(), osg::CopyOp::SHALLOW_COPY);
std::vector<osg::ref_ptr<osg::Texture2D> > Textures = mGlowUpdater->getTextures();
int TexUnit = mGlowUpdater->getTexUnit();
mObjectRoot->removeUpdateCallback(mGlowUpdater);
mGlowUpdater = NULL;
for (size_t index = 0; index < Textures.size(); index++)
{
writableStateSet->setTextureAttribute(TexUnit, Textures[index], osg::StateAttribute::OFF|osg::StateAttribute::OVERRIDE);
writableStateSet->removeTextureAttribute(TexUnit, Textures[index]);
}
mObjectRoot->setStateSet(writableStateSet);
writableStateSet->removeUniform(mUniform);
mResourceSystem->getSceneManager()->recreateShaders(mObjectRoot);
mSpellGlowDuration = 0.f;
}
void Animation::addGlow(osg::ref_ptr<osg::Node> node, osg::Vec4f glowColor, bool hasDuration)
{ {
std::vector<osg::ref_ptr<osg::Texture2D> > textures; std::vector<osg::ref_ptr<osg::Texture2D> > textures;
for (int i=0; i<32; ++i) for (int i=0; i<32; ++i)
@ -1130,8 +1186,9 @@ namespace MWRender
FindLowestUnusedTexUnitVisitor findLowestUnusedTexUnitVisitor; FindLowestUnusedTexUnitVisitor findLowestUnusedTexUnitVisitor;
node->accept(findLowestUnusedTexUnitVisitor); node->accept(findLowestUnusedTexUnitVisitor);
int texUnit = findLowestUnusedTexUnitVisitor.mLowestUnusedTexUnit; int texUnit = findLowestUnusedTexUnitVisitor.mLowestUnusedTexUnit;
osg::ref_ptr<GlowUpdater> glowupdater (new GlowUpdater(texUnit, glowColor, textures)); mGlowUpdater = new GlowUpdater(texUnit, glowColor, textures, hasDuration);
node->addUpdateCallback(glowupdater); //osg::ref_ptr<GlowUpdater> glowupdater (new GlowUpdater(texUnit, glowColor, textures));
node->addUpdateCallback(mGlowUpdater);
// set a texture now so that the ShaderVisitor can find it // set a texture now so that the ShaderVisitor can find it
osg::ref_ptr<osg::StateSet> writableStateSet = NULL; osg::ref_ptr<osg::StateSet> writableStateSet = NULL;
@ -1143,7 +1200,8 @@ namespace MWRender
node->setStateSet(writableStateSet); node->setStateSet(writableStateSet);
} }
writableStateSet->setTextureAttributeAndModes(texUnit, textures.front(), osg::StateAttribute::ON); writableStateSet->setTextureAttributeAndModes(texUnit, textures.front(), osg::StateAttribute::ON);
writableStateSet->addUniform(new osg::Uniform("envMapColor", glowColor)); mUniform = new osg::Uniform("envMapColor", glowColor);
writableStateSet->addUniform(mUniform);
mResourceSystem->getSceneManager()->recreateShaders(node); mResourceSystem->getSceneManager()->recreateShaders(node);
} }

@ -5,6 +5,11 @@
#include <components/sceneutil/controller.hpp> #include <components/sceneutil/controller.hpp>
namespace
{
class GlowUpdater;
}
namespace ESM namespace ESM
{ {
struct Light; struct Light;
@ -263,6 +268,10 @@ protected:
osg::ref_ptr<SceneUtil::LightSource> mGlowLight; osg::ref_ptr<SceneUtil::LightSource> mGlowLight;
float mAlpha; float mAlpha;
float mSpellGlowDuration;
osg::ref_ptr<GlowUpdater> mGlowUpdater;
osg::Uniform* mUniform;
const NodeMap& getNodeMap() const; const NodeMap& getNodeMap() const;
@ -317,7 +326,7 @@ protected:
osg::Vec4f getEnchantmentColor(const MWWorld::ConstPtr& item) const; osg::Vec4f getEnchantmentColor(const MWWorld::ConstPtr& item) const;
void addGlow(osg::ref_ptr<osg::Node> node, osg::Vec4f glowColor); void addGlow(osg::ref_ptr<osg::Node> node, osg::Vec4f glowColor, bool hasDuration = false);
/// Set the render bin for this animation's object root. May be customized by subclasses. /// Set the render bin for this animation's object root. May be customized by subclasses.
virtual void setRenderBin(); virtual void setRenderBin();
@ -351,6 +360,10 @@ public:
void removeEffect (int effectId); void removeEffect (int effectId);
void getLoopingEffects (std::vector<int>& out) const; void getLoopingEffects (std::vector<int>& out) const;
void addSpellCastGlow(osg::Vec4f glowColor);
void updateSpellGlow(float duration);
void removeSpellGlow();
virtual void updatePtr(const MWWorld::Ptr &ptr); virtual void updatePtr(const MWWorld::Ptr &ptr);
bool hasAnimation(const std::string &anim) const; bool hasAnimation(const std::string &anim) const;

Loading…
Cancel
Save