Use "hair" as filter for PRT_Hair parts (Fixes #2218)

This commit is contained in:
scrawl 2014-12-22 19:58:18 +01:00
parent d55fe43fc9
commit 59f21c6105
7 changed files with 20 additions and 13 deletions

View file

@ -1270,7 +1270,7 @@ void Animation::addEffect(const std::string &model, int effectId, bool loop, con
if (bonename.empty()) if (bonename.empty())
params.mObjects = NifOgre::Loader::createObjects(mInsert, model); params.mObjects = NifOgre::Loader::createObjects(mInsert, model);
else else
params.mObjects = NifOgre::Loader::createObjects(mSkelBase, bonename, mInsert, model); params.mObjects = NifOgre::Loader::createObjects(mSkelBase, bonename, "", mInsert, model);
// TODO: turn off shadow casting // TODO: turn off shadow casting
setRenderProperties(params.mObjects, RV_Misc, setRenderProperties(params.mObjects, RV_Misc,

View file

@ -106,7 +106,7 @@ void CreatureWeaponAnimation::updatePart(NifOgre::ObjectScenePtr& scene, int slo
else else
bonename = "Shield Bone"; bonename = "Shield Bone";
scene = NifOgre::Loader::createObjects(mSkelBase, bonename, mInsert, item.getClass().getModel(item)); scene = NifOgre::Loader::createObjects(mSkelBase, bonename, bonename, mInsert, item.getClass().getModel(item));
Ogre::Vector3 glowColor = getEnchantmentColor(item); Ogre::Vector3 glowColor = getEnchantmentColor(item);
setRenderProperties(scene, RV_Actors, RQG_Main, RQG_Alpha, 0, setRenderProperties(scene, RV_Actors, RQG_Main, RQG_Alpha, 0,

View file

@ -160,7 +160,7 @@ static NpcAnimation::PartBoneMap createPartListMap()
{ {
NpcAnimation::PartBoneMap result; NpcAnimation::PartBoneMap result;
result.insert(std::make_pair(ESM::PRT_Head, "Head")); result.insert(std::make_pair(ESM::PRT_Head, "Head"));
result.insert(std::make_pair(ESM::PRT_Hair, "Head")); result.insert(std::make_pair(ESM::PRT_Hair, "Head")); // note it uses "Head" as attach bone, but "Hair" as filter
result.insert(std::make_pair(ESM::PRT_Neck, "Neck")); result.insert(std::make_pair(ESM::PRT_Neck, "Neck"));
result.insert(std::make_pair(ESM::PRT_Cuirass, "Chest")); result.insert(std::make_pair(ESM::PRT_Cuirass, "Chest"));
result.insert(std::make_pair(ESM::PRT_Groin, "Groin")); result.insert(std::make_pair(ESM::PRT_Groin, "Groin"));
@ -574,9 +574,9 @@ public:
} }
}; };
NifOgre::ObjectScenePtr NpcAnimation::insertBoundedPart(const std::string &model, int group, const std::string &bonename, bool enchantedGlow, Ogre::Vector3* glowColor) NifOgre::ObjectScenePtr NpcAnimation::insertBoundedPart(const std::string &model, int group, const std::string &bonename, const std::string &bonefilter, bool enchantedGlow, Ogre::Vector3* glowColor)
{ {
NifOgre::ObjectScenePtr objects = NifOgre::Loader::createObjects(mSkelBase, bonename, mInsert, model); NifOgre::ObjectScenePtr objects = NifOgre::Loader::createObjects(mSkelBase, bonename, bonefilter, mInsert, model);
setRenderProperties(objects, (mViewMode == VM_FirstPerson) ? RV_FirstPerson : mVisibilityFlags, RQG_Main, RQG_Alpha, 0, setRenderProperties(objects, (mViewMode == VM_FirstPerson) ? RV_FirstPerson : mVisibilityFlags, RQG_Main, RQG_Alpha, 0,
enchantedGlow, glowColor); enchantedGlow, glowColor);
@ -690,7 +690,10 @@ bool NpcAnimation::addOrReplaceIndividualPart(ESM::PartReferenceType type, int g
mPartPriorities[type] = priority; mPartPriorities[type] = priority;
try try
{ {
mObjectParts[type] = insertBoundedPart(mesh, group, sPartList.at(type), enchantedGlow, glowColor); const std::string& bonename = sPartList.at(type);
// PRT_Hair seems to be the only type that breaks consistency and uses a filter that's different from the attachment bone
const std::string bonefilter = (type == ESM::PRT_Hair) ? "hair" : bonename;
mObjectParts[type] = insertBoundedPart(mesh, group, bonename, bonefilter, enchantedGlow, glowColor);
} }
catch (std::exception& e) catch (std::exception& e)
{ {

View file

@ -106,6 +106,7 @@ private:
void updateNpcBase(); void updateNpcBase();
NifOgre::ObjectScenePtr insertBoundedPart(const std::string &model, int group, const std::string &bonename, NifOgre::ObjectScenePtr insertBoundedPart(const std::string &model, int group, const std::string &bonename,
const std::string &bonefilter,
bool enchantedGlow, Ogre::Vector3* glowColor=NULL); bool enchantedGlow, Ogre::Vector3* glowColor=NULL);
void removeIndividualPart(ESM::PartReferenceType type); void removeIndividualPart(ESM::PartReferenceType type);

View file

@ -55,8 +55,11 @@ void WeaponAnimation::attachArrow(MWWorld::Ptr actor)
return; return;
std::string model = ammo->getClass().getModel(*ammo); std::string model = ammo->getClass().getModel(*ammo);
assert(weapon->mSkelBase && "Need a skeleton to attach the arrow to"); if (!weapon->mSkelBase)
mAmmunition = NifOgre::Loader::createObjects(weapon->mSkelBase, "ArrowBone", weapon->mSkelBase->getParentSceneNode(), model); throw std::runtime_error("Need a skeleton to attach the arrow to");
const std::string bonename = "ArrowBone";
mAmmunition = NifOgre::Loader::createObjects(weapon->mSkelBase, bonename, bonename, weapon->mSkelBase->getParentSceneNode(), model);
configureAddedObject(mAmmunition, *ammo, MWWorld::InventoryStore::Slot_Ammunition); configureAddedObject(mAmmunition, *ammo, MWWorld::InventoryStore::Slot_Ammunition);
} }
} }

View file

@ -1383,6 +1383,7 @@ ObjectScenePtr Loader::createObjects(Ogre::SceneNode *parentNode, std::string na
} }
ObjectScenePtr Loader::createObjects(Ogre::Entity *parent, const std::string &bonename, ObjectScenePtr Loader::createObjects(Ogre::Entity *parent, const std::string &bonename,
const std::string& bonefilter,
Ogre::SceneNode *parentNode, Ogre::SceneNode *parentNode,
std::string name, const std::string &group) std::string name, const std::string &group)
{ {
@ -1408,11 +1409,9 @@ ObjectScenePtr Loader::createObjects(Ogre::Entity *parent, const std::string &bo
if(isskinned) if(isskinned)
{ {
// Apparently both are allowed. Sigh. // accepts anything named "filter*" or "tri filter*"
// This could also mean that filters are supposed to work on the actual node std::string filter = "@shape=tri "+bonefilter;
// hierarchy, rather than just trishapes, and the 'tri ' should be omitted? std::string filter2 = "@shape="+bonefilter;
std::string filter = "@shape=tri "+bonename;
std::string filter2 = "@shape="+bonename;
Misc::StringUtils::toLower(filter); Misc::StringUtils::toLower(filter);
Misc::StringUtils::toLower(filter2); Misc::StringUtils::toLower(filter2);
for(size_t i = 0;i < scene->mEntities.size();i++) for(size_t i = 0;i < scene->mEntities.size();i++)

View file

@ -97,6 +97,7 @@ class Loader
{ {
public: public:
static ObjectScenePtr createObjects(Ogre::Entity *parent, const std::string &bonename, static ObjectScenePtr createObjects(Ogre::Entity *parent, const std::string &bonename,
const std::string& filter,
Ogre::SceneNode *parentNode, Ogre::SceneNode *parentNode,
std::string name, std::string name,
const std::string &group="General"); const std::string &group="General");