Merge pull request #205 from OpenMW/master

Add OpenMW commits up to 23 Apr 2017
0.6.1
David Cernat 8 years ago committed by GitHub
commit ef634a1c52

@ -726,6 +726,19 @@ void NpcAnimation::removePartGroup(int group)
} }
} }
bool NpcAnimation::isFirstPersonPart(const ESM::BodyPart* bodypart)
{
return (bodypart->mId.size() >= 3)
&& bodypart->mId[bodypart->mId.size()-3] == '1'
&& bodypart->mId[bodypart->mId.size()-2] == 's'
&& bodypart->mId[bodypart->mId.size()-1] == 't';
}
bool NpcAnimation::isFemalePart(const ESM::BodyPart* bodypart)
{
return bodypart->mData.mFlags & ESM::BodyPart::BPF_Female;
}
bool NpcAnimation::addOrReplaceIndividualPart(ESM::PartReferenceType type, int group, int priority, const std::string &mesh, bool enchantedGlow, osg::Vec4f* glowColor) bool NpcAnimation::addOrReplaceIndividualPart(ESM::PartReferenceType type, int group, int priority, const std::string &mesh, bool enchantedGlow, osg::Vec4f* glowColor)
{ {
if(priority <= mPartPriorities[type]) if(priority <= mPartPriorities[type])
@ -1096,46 +1109,79 @@ const std::vector<const ESM::BodyPart *>& NpcAnimation::getBodyParts(const std::
if (!Misc::StringUtils::ciEqual(bodypart.mRace, race)) if (!Misc::StringUtils::ciEqual(bodypart.mRace, race))
continue; continue;
bool partFirstPerson = (bodypart.mId.size() >= 3) bool partFirstPerson = isFirstPersonPart(&bodypart);
&& bodypart.mId[bodypart.mId.size()-3] == '1'
&& bodypart.mId[bodypart.mId.size()-2] == 's' bool isHand = bodypart.mData.mPart == ESM::BodyPart::MP_Hand ||
&& bodypart.mId[bodypart.mId.size()-1] == 't'; bodypart.mData.mPart == ESM::BodyPart::MP_Wrist ||
if(partFirstPerson != (firstPerson)) bodypart.mData.mPart == ESM::BodyPart::MP_Forearm ||
bodypart.mData.mPart == ESM::BodyPart::MP_Upperarm;
bool isSameGender = isFemalePart(&bodypart) == female;
/* A fallback for the arms if 1st person is missing:
1. Try to use 3d person skin for same gender
2. Try to use 1st person skin for male, if female == true
3. Try to use 3d person skin for male, if female == true
A fallback in another cases: allow to use male bodyparts, if female == true
*/
if (firstPerson && isHand && !partFirstPerson)
{ {
if(firstPerson && (bodypart.mData.mPart == ESM::BodyPart::MP_Hand || // Allow 3rd person skins as a fallback for the arms if 1st person is missing
bodypart.mData.mPart == ESM::BodyPart::MP_Wrist || BodyPartMapType::const_iterator bIt = sBodyPartMap.lower_bound(BodyPartMapType::key_type(bodypart.mData.mPart));
bodypart.mData.mPart == ESM::BodyPart::MP_Forearm || while(bIt != sBodyPartMap.end() && bIt->first == bodypart.mData.mPart)
bodypart.mData.mPart == ESM::BodyPart::MP_Upperarm))
{ {
/* Allow 3rd person skins as a fallback for the arms if 1st person is missing. */ // If we have no fallback bodypart now and bodypart is for same gender (1)
BodyPartMapType::const_iterator bIt = sBodyPartMap.lower_bound(BodyPartMapType::key_type(bodypart.mData.mPart)); if(!parts[bIt->second] && isSameGender)
while(bIt != sBodyPartMap.end() && bIt->first == bodypart.mData.mPart) parts[bIt->second] = &bodypart;
{
if(!parts[bIt->second]) // If we have fallback bodypart for other gender and found fallback for current gender (1)
parts[bIt->second] = &*it; else if(isSameGender && isFemalePart(parts[bIt->second]) != female)
++bIt; parts[bIt->second] = &bodypart;
}
// If we have no fallback bodypart and searching for female bodyparts (3)
else if(!parts[bIt->second] && female)
parts[bIt->second] = &bodypart;
++bIt;
} }
continue; continue;
} }
if ((female) != (bodypart.mData.mFlags & ESM::BodyPart::BPF_Female)) // Don't allow to use podyparts for a different view
if (partFirstPerson != firstPerson)
continue;
if (female && !isFemalePart(&bodypart))
{ {
// Allow opposite gender's parts as fallback if parts for our gender are missing // Allow male parts as fallback for females if female parts are missing
BodyPartMapType::const_iterator bIt = sBodyPartMap.lower_bound(BodyPartMapType::key_type(bodypart.mData.mPart)); BodyPartMapType::const_iterator bIt = sBodyPartMap.lower_bound(BodyPartMapType::key_type(bodypart.mData.mPart));
while(bIt != sBodyPartMap.end() && bIt->first == bodypart.mData.mPart) while(bIt != sBodyPartMap.end() && bIt->first == bodypart.mData.mPart)
{ {
// If we have no fallback bodypart now
if(!parts[bIt->second]) if(!parts[bIt->second])
parts[bIt->second] = &*it; parts[bIt->second] = &bodypart;
// If we have 3d person fallback bodypart for hand and 1st person fallback found (2)
else if(isHand && !isFirstPersonPart(parts[bIt->second]) && partFirstPerson)
parts[bIt->second] = &bodypart;
++bIt; ++bIt;
} }
continue; continue;
} }
// Don't allow to use podyparts for another gender
if (female != isFemalePart(&bodypart))
continue;
// Use properly found bodypart, replacing fallbacks
BodyPartMapType::const_iterator bIt = sBodyPartMap.lower_bound(BodyPartMapType::key_type(bodypart.mData.mPart)); BodyPartMapType::const_iterator bIt = sBodyPartMap.lower_bound(BodyPartMapType::key_type(bodypart.mData.mPart));
while(bIt != sBodyPartMap.end() && bIt->first == bodypart.mData.mPart) while(bIt != sBodyPartMap.end() && bIt->first == bodypart.mData.mPart)
{ {
parts[bIt->second] = &*it; parts[bIt->second] = &bodypart;
++bIt; ++bIt;
} }
} }

@ -90,6 +90,9 @@ private:
osg::ref_ptr<NeckController> mFirstPersonNeckController; osg::ref_ptr<NeckController> mFirstPersonNeckController;
static bool isFirstPersonPart(const ESM::BodyPart* bodypart);
static bool isFemalePart(const ESM::BodyPart* bodypart);
protected: protected:
virtual void addControllers(); virtual void addControllers();

@ -1773,8 +1773,10 @@ namespace NifOsg
} }
if (!hasMatCtrl && mat->getColorMode() == osg::Material::OFF if (!hasMatCtrl && mat->getColorMode() == osg::Material::OFF
&& mat->getEmission(osg::Material::FRONT_AND_BACK) == osg::Vec4f(0,0,0,1)
&& mat->getDiffuse(osg::Material::FRONT_AND_BACK) == osg::Vec4f(1,1,1,1) && mat->getDiffuse(osg::Material::FRONT_AND_BACK) == osg::Vec4f(1,1,1,1)
&& mat->getAmbient(osg::Material::FRONT_AND_BACK) == osg::Vec4f(1,1,1,1) && mat->getAmbient(osg::Material::FRONT_AND_BACK) == osg::Vec4f(1,1,1,1)
&& mat->getShininess(osg::Material::FRONT_AND_BACK) == 0
&& mat->getSpecular(osg::Material::FRONT_AND_BACK) == osg::Vec4f(0.f, 0.f, 0.f, 0.f)) && mat->getSpecular(osg::Material::FRONT_AND_BACK) == osg::Vec4f(0.f, 0.f, 0.f, 0.f))
{ {
// default state, skip // default state, skip

Loading…
Cancel
Save