mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-31 18:56:38 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			782 lines
		
	
	
	
		
			30 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			782 lines
		
	
	
	
		
			30 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|   Copyright (C) 2016, 2018-2021 cc9cii
 | |
| 
 | |
|   This software is provided 'as-is', without any express or implied
 | |
|   warranty.  In no event will the authors be held liable for any damages
 | |
|   arising from the use of this software.
 | |
| 
 | |
|   Permission is granted to anyone to use this software for any purpose,
 | |
|   including commercial applications, and to alter it and redistribute it
 | |
|   freely, subject to the following restrictions:
 | |
| 
 | |
|   1. The origin of this software must not be misrepresented; you must not
 | |
|      claim that you wrote the original software. If you use this software
 | |
|      in a product, an acknowledgment in the product documentation would be
 | |
|      appreciated but is not required.
 | |
|   2. Altered source versions must be plainly marked as such, and must not be
 | |
|      misrepresented as being the original software.
 | |
|   3. This notice may not be removed or altered from any source distribution.
 | |
| 
 | |
|   cc9cii cc9c@iinet.net.au
 | |
| 
 | |
|   Much of the information on the data structures are based on the information
 | |
|   from Tes4Mod:Mod_File_Format and Tes5Mod:File_Formats but also refined by
 | |
|   trial & error.  See http://en.uesp.net/wiki for details.
 | |
| 
 | |
| */
 | |
| #include "loadrace.hpp"
 | |
| 
 | |
| #include <cstring>
 | |
| #include <stdexcept>
 | |
| 
 | |
| #include "reader.hpp"
 | |
| //#include "writer.hpp"
 | |
| 
 | |
| void ESM4::Race::load(ESM4::Reader& reader)
 | |
| {
 | |
|     mId = reader.getFormIdFromHeader();
 | |
|     mFlags = reader.hdr().record.flags;
 | |
| 
 | |
|     std::uint32_t esmVer = reader.esmVersion();
 | |
|     bool isTES4 = (esmVer == ESM::VER_080 || esmVer == ESM::VER_100) && !reader.hasFormVersion();
 | |
|     bool isFONV = esmVer == ESM::VER_132 || esmVer == ESM::VER_133 || esmVer == ESM::VER_134;
 | |
|     bool isFO3 = false;
 | |
| 
 | |
|     bool isMale = false;
 | |
|     int currPart = -1; // 0 = head, 1 = body, 2 = egt, 3 = hkx
 | |
|     std::uint32_t currentIndex = 0xffffffff;
 | |
| 
 | |
|     while (reader.getSubRecordHeader())
 | |
|     {
 | |
|         const ESM4::SubRecordHeader& subHdr = reader.subRecordHeader();
 | |
|         // std::cout << "RACE " << ESM::printName(subHdr.typeId) << std::endl;
 | |
|         switch (subHdr.typeId)
 | |
|         {
 | |
|             case ESM::fourCC("EDID"):
 | |
|             {
 | |
|                 reader.getZString(mEditorId);
 | |
|                 // TES4
 | |
|                 // Sheogorath  0x0005308E
 | |
|                 // GoldenSaint 0x0001208F
 | |
|                 // DarkSeducer 0x0001208E
 | |
|                 // VampireRace 0x00000019
 | |
|                 // Dremora     0x00038010
 | |
|                 // Argonian    0x00023FE9
 | |
|                 // Nord        0x000224FD
 | |
|                 // Breton      0x000224FC
 | |
|                 // WoodElf     0x000223C8
 | |
|                 // Khajiit     0x000223C7
 | |
|                 // DarkElf     0x000191C1
 | |
|                 // Orc         0x000191C0
 | |
|                 // HighElf     0x00019204
 | |
|                 // Redguard    0x00000D43
 | |
|                 // Imperial    0x00000907
 | |
|                 break;
 | |
|             }
 | |
|             case ESM::fourCC("FULL"):
 | |
|                 reader.getLocalizedString(mFullName);
 | |
|                 break;
 | |
|             case ESM::fourCC("DESC"):
 | |
|             {
 | |
|                 if (subHdr.dataSize == 1) // FO3?
 | |
|                 {
 | |
|                     reader.skipSubRecordData();
 | |
|                     break;
 | |
|                 }
 | |
| 
 | |
|                 reader.getLocalizedString(mDesc);
 | |
|                 break;
 | |
|             }
 | |
|             case ESM::fourCC("SPLO"): // bonus spell formid (TES5 may have SPCT and multiple SPLO)
 | |
|                 reader.getFormId(mBonusSpells.emplace_back());
 | |
|                 break;
 | |
|             case ESM::fourCC("DATA"): // ?? different length for TES5
 | |
|             {
 | |
| // DATA:size 128
 | |
| // 0f 0f ff 00 ff 00 ff 00 ff 00 ff 00 ff 00 00 00
 | |
| // 9a 99 99 3f 00 00 80 3f 00 00 80 3f 00 00 80 3f
 | |
| // 48 89 10 00 00 00 40 41 00 00 00 00 00 00 48 43
 | |
| // 00 00 48 43 00 00 80 3f 9a 99 19 3f 00 00 00 40
 | |
| // 01 00 00 00 ff ff ff ff ff ff ff ff 00 00 00 00
 | |
| // ff ff ff ff 00 00 00 00 00 00 20 41 00 00 a0 40
 | |
| // 00 00 a0 40 00 00 80 42 ff ff ff ff 00 00 00 00
 | |
| // 00 00 00 00 9a 99 99 3e 00 00 a0 40 02 00 00 00
 | |
| #if 0
 | |
|                 unsigned char mDataBuf[256/*bufSize*/];
 | |
|                 reader.get(mDataBuf, subHdr.dataSize);
 | |
| 
 | |
|                 std::ostringstream ss;
 | |
|                 ss << ESM::printName(subHdr.typeId) << ":size " << subHdr.dataSize << "\n";
 | |
|                 for (unsigned int i = 0; i < subHdr.dataSize; ++i)
 | |
|                 {
 | |
|                     //if (mDataBuf[i] > 64 && mDataBuf[i] < 91)
 | |
|                         //ss << (char)(mDataBuf[i]) << " ";
 | |
|                     //else
 | |
|                         ss << std::setfill('0') << std::setw(2) << std::hex << (int)(mDataBuf[i]);
 | |
|                     if ((i & 0x000f) == 0xf)
 | |
|                         ss << "\n";
 | |
|                     else if (i < 256/*bufSize*/-1)
 | |
|                         ss << " ";
 | |
|                 }
 | |
|                 std::cout << ss.str() << std::endl;
 | |
| #else
 | |
|                 if (subHdr.dataSize == 36) // TES4/FO3/FONV
 | |
|                 {
 | |
|                     if (!isTES4 && !isFONV && !mIsTES5)
 | |
|                         isFO3 = true;
 | |
| 
 | |
|                     std::uint8_t skill;
 | |
|                     std::uint8_t bonus;
 | |
|                     for (unsigned int i = 0; i < 8; ++i)
 | |
|                     {
 | |
|                         reader.get(skill);
 | |
|                         reader.get(bonus);
 | |
|                         mSkillBonus[static_cast<SkillIndex>(skill)] = bonus;
 | |
|                     }
 | |
|                     reader.get(mHeightMale);
 | |
|                     reader.get(mHeightFemale);
 | |
|                     reader.get(mWeightMale);
 | |
|                     reader.get(mWeightFemale);
 | |
|                     reader.get(mRaceFlags);
 | |
|                 }
 | |
|                 else if (subHdr.dataSize == 128 || subHdr.dataSize == 164) // TES5
 | |
|                 {
 | |
|                     mIsTES5 = true;
 | |
| 
 | |
|                     std::uint8_t skill;
 | |
|                     std::uint8_t bonus;
 | |
|                     for (unsigned int i = 0; i < 7; ++i)
 | |
|                     {
 | |
|                         reader.get(skill);
 | |
|                         reader.get(bonus);
 | |
|                         mSkillBonus[static_cast<SkillIndex>(skill)] = bonus;
 | |
|                     }
 | |
|                     std::uint16_t unknown;
 | |
|                     reader.get(unknown);
 | |
| 
 | |
|                     reader.get(mHeightMale);
 | |
|                     reader.get(mHeightFemale);
 | |
|                     reader.get(mWeightMale);
 | |
|                     reader.get(mWeightFemale);
 | |
|                     reader.get(mRaceFlags);
 | |
| 
 | |
|                     // FIXME
 | |
|                     float dummy;
 | |
|                     reader.get(dummy); // starting health
 | |
|                     reader.get(dummy); // starting magicka
 | |
|                     reader.get(dummy); // starting stamina
 | |
|                     reader.get(dummy); // base carry weight
 | |
|                     reader.get(dummy); // base mass
 | |
|                     reader.get(dummy); // accleration rate
 | |
|                     reader.get(dummy); // decleration rate
 | |
| 
 | |
|                     uint32_t dummy2;
 | |
|                     reader.get(dummy2); // size
 | |
|                     reader.get(dummy2); // head biped object
 | |
|                     reader.get(dummy2); // hair biped object
 | |
|                     reader.get(dummy); // injured health % (0.f..1.f)
 | |
|                     reader.get(dummy2); // shield biped object
 | |
|                     reader.get(dummy); // health regen
 | |
|                     reader.get(dummy); // magicka regen
 | |
|                     reader.get(dummy); // stamina regen
 | |
|                     reader.get(dummy); // unarmed damage
 | |
|                     reader.get(dummy); // unarmed reach
 | |
|                     reader.get(dummy2); // body biped object
 | |
|                     reader.get(dummy); // aim angle tolerance
 | |
|                     reader.get(dummy2); // unknown
 | |
|                     reader.get(dummy); // angular accleration rate
 | |
|                     reader.get(dummy); // angular tolerance
 | |
|                     reader.get(dummy2); // flags
 | |
| 
 | |
|                     if (subHdr.dataSize == 164)
 | |
|                     {
 | |
|                         reader.get(dummy2); // unknown 1
 | |
|                         reader.get(dummy2); // unknown 2
 | |
|                         reader.get(dummy2); // unknown 3
 | |
|                         reader.get(dummy2); // unknown 4
 | |
|                         reader.get(dummy2); // unknown 5
 | |
|                         reader.get(dummy2); // unknown 6
 | |
|                         reader.get(dummy2); // unknown 7
 | |
|                         reader.get(dummy2); // unknown 8
 | |
|                         reader.get(dummy2); // unknown 9
 | |
|                     }
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     reader.skipSubRecordData();
 | |
|                     // std::cout << "RACE " << ESM::printName(subHdr.typeId) << " skipping..." << subHdr.dataSize <<
 | |
|                     // std::endl;
 | |
|                 }
 | |
| #endif
 | |
|                 break;
 | |
|             }
 | |
|             case ESM::fourCC("DNAM"):
 | |
|             {
 | |
|                 reader.getFormId(mDefaultHair[0]); // male
 | |
|                 reader.getFormId(mDefaultHair[1]); // female
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|             case ESM::fourCC("CNAM"):
 | |
|                 //              CNAM       SNAM                     VNAM
 | |
|                 // Sheogorath   0x0  0000  98 2b  10011000 00101011
 | |
|                 // Golden Saint 0x3  0011  26 46  00100110 01000110
 | |
|                 // Dark Seducer 0xC  1100  df 55  11011111 01010101
 | |
|                 // Vampire Race 0x0  0000  77 44  01110111 10001000
 | |
|                 // Dremora      0x7  0111  bf 32  10111111 00110010
 | |
|                 // Argonian     0x0  0000  dc 3c  11011100 00111100
 | |
|                 // Nord         0x5  0101  b6 03  10110110 00000011
 | |
|                 // Breton       0x5  0101  48 1d  01001000 00011101 00000000 00000907 (Imperial)
 | |
|                 // Wood Elf     0xD  1101  2e 4a  00101110 01001010 00019204 00019204 (HighElf)
 | |
|                 // khajiit      0x5  0101  54 5b  01010100 01011011 00023FE9 00023FE9 (Argonian)
 | |
|                 // Dark Elf     0x0  0000  72 54  01110010 01010100 00019204 00019204 (HighElf)
 | |
|                 // Orc          0xC  1100  74 09  01110100 00001001 000224FD 000224FD (Nord)
 | |
|                 // High Elf     0xF  1111  e6 21  11100110 00100001
 | |
|                 // Redguard     0xD  1101  a9 61  10101001 01100001
 | |
|                 // Imperial     0xD  1101  8e 35  10001110 00110101
 | |
|                 {
 | |
|                     reader.skipSubRecordData();
 | |
|                     break;
 | |
|                 }
 | |
|             case ESM::fourCC("PNAM"):
 | |
|                 reader.get(mFaceGenMainClamp);
 | |
|                 break; // 0x40A00000 = 5.f
 | |
|             case ESM::fourCC("UNAM"):
 | |
|                 reader.get(mFaceGenFaceClamp);
 | |
|                 break; // 0x40400000 = 3.f
 | |
|             case ESM::fourCC("ATTR"): // Only in TES4?
 | |
|             {
 | |
|                 if (subHdr.dataSize == 2) // FO3?
 | |
|                 {
 | |
|                     reader.skipSubRecordData();
 | |
|                     break;
 | |
|                 }
 | |
|                 reader.get(mAttribMale.strength);
 | |
|                 reader.get(mAttribMale.intelligence);
 | |
|                 reader.get(mAttribMale.willpower);
 | |
|                 reader.get(mAttribMale.agility);
 | |
|                 reader.get(mAttribMale.speed);
 | |
|                 reader.get(mAttribMale.endurance);
 | |
|                 reader.get(mAttribMale.personality);
 | |
|                 reader.get(mAttribMale.luck);
 | |
|                 reader.get(mAttribFemale.strength);
 | |
|                 reader.get(mAttribFemale.intelligence);
 | |
|                 reader.get(mAttribFemale.willpower);
 | |
|                 reader.get(mAttribFemale.agility);
 | |
|                 reader.get(mAttribFemale.speed);
 | |
|                 reader.get(mAttribFemale.endurance);
 | |
|                 reader.get(mAttribFemale.personality);
 | |
|                 reader.get(mAttribFemale.luck);
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|             //        [0..9]-> ICON
 | |
|             // NAM0 -> INDX -> MODL --+
 | |
|             //          ^   -> MODB   |
 | |
|             //          |             |
 | |
|             //          +-------------+
 | |
|             //
 | |
|             case ESM::fourCC("NAM0"): // start marker head data /* 1 */
 | |
|             {
 | |
|                 currPart = 0; // head part
 | |
| 
 | |
|                 if (isFO3 || isFONV)
 | |
|                 {
 | |
|                     mHeadParts.resize(8);
 | |
|                     mHeadPartsFemale.resize(8);
 | |
|                 }
 | |
|                 else if (isTES4)
 | |
|                     mHeadParts.resize(9); // assumed based on Construction Set
 | |
|                 else // Optimized for TES5
 | |
|                 {
 | |
|                     mHeadPartIdsMale.resize(5);
 | |
|                     mHeadPartIdsFemale.resize(5);
 | |
|                 }
 | |
| 
 | |
|                 currentIndex = 0xffffffff;
 | |
|                 break;
 | |
|             }
 | |
|             case ESM::fourCC("INDX"):
 | |
|             {
 | |
|                 reader.get(currentIndex);
 | |
|                 // FIXME: below check is rather useless
 | |
|                 // if (headpart)
 | |
|                 //{
 | |
|                 //    if (currentIndex > 8)
 | |
|                 //        throw std::runtime_error("ESM4::RACE::load - too many head part " + currentIndex);
 | |
|                 //}
 | |
|                 // else // bodypart
 | |
|                 //{
 | |
|                 //    if (currentIndex > 4)
 | |
|                 //        throw std::runtime_error("ESM4::RACE::load - too many body part " + currentIndex);
 | |
|                 //}
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|             case ESM::fourCC("MODL"):
 | |
|             {
 | |
|                 if (currentIndex == 0xffffffff)
 | |
|                 {
 | |
|                     reader.skipSubRecordData();
 | |
|                 }
 | |
|                 else if (currPart == 0) // head part
 | |
|                 {
 | |
|                     if (isMale || isTES4)
 | |
|                         reader.getZString(mHeadParts[currentIndex].mesh);
 | |
|                     else
 | |
|                         reader.getZString(mHeadPartsFemale[currentIndex].mesh); // TODO: check TES4
 | |
| 
 | |
|                     // TES5 keeps head part formid in mHeadPartIdsMale and mHeadPartIdsFemale
 | |
|                 }
 | |
|                 else if (currPart == 1) // body part
 | |
|                 {
 | |
|                     if (isMale)
 | |
|                         reader.getZString(mBodyPartsMale[currentIndex].mesh);
 | |
|                     else
 | |
|                         reader.getZString(mBodyPartsFemale[currentIndex].mesh);
 | |
| 
 | |
|                     // TES5 seems to have no body parts at all, instead keep EGT models
 | |
|                 }
 | |
|                 // else if (curr_part == 2) // egt
 | |
|                 // {
 | |
|                 //     // std::cout << mEditorId << " egt " << currentIndex << std::endl; // FIXME
 | |
|                 //     reader.skipSubRecordData(); // FIXME TES5 egt
 | |
|                 // }
 | |
|                 else
 | |
|                 {
 | |
|                     // std::cout << mEditorId << " hkx " << currentIndex << std::endl; // FIXME
 | |
|                     reader.skipSubRecordData(); // FIXME TES5 hkx
 | |
|                 }
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|             case ESM::fourCC("MODB"):
 | |
|                 reader.skipSubRecordData();
 | |
|                 break; // always 0x0000?
 | |
|             case ESM::fourCC("ICON"):
 | |
|             {
 | |
|                 if (currentIndex == 0xffffffff)
 | |
|                 {
 | |
|                     reader.skipSubRecordData();
 | |
|                 }
 | |
|                 else if (currPart == 0) // head part
 | |
|                 {
 | |
|                     if (isMale || isTES4)
 | |
|                         reader.getZString(mHeadParts[currentIndex].texture);
 | |
|                     else
 | |
|                         reader.getZString(mHeadPartsFemale[currentIndex].texture); // TODO: check TES4
 | |
|                 }
 | |
|                 else if (currPart == 1) // body part
 | |
|                 {
 | |
|                     if (isMale)
 | |
|                         reader.getZString(mBodyPartsMale[currentIndex].texture);
 | |
|                     else
 | |
|                         reader.getZString(mBodyPartsFemale[currentIndex].texture);
 | |
|                 }
 | |
|                 else
 | |
|                     reader.skipSubRecordData(); // FIXME TES5
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|             //
 | |
|             case ESM::fourCC("NAM1"): // start marker body data /* 4 */
 | |
|             {
 | |
| 
 | |
|                 if (isFO3 || isFONV)
 | |
|                 {
 | |
|                     currPart = 1; // body part
 | |
| 
 | |
|                     mBodyPartsMale.resize(4);
 | |
|                     mBodyPartsFemale.resize(4);
 | |
|                 }
 | |
|                 else if (isTES4)
 | |
|                 {
 | |
|                     currPart = 1; // body part
 | |
| 
 | |
|                     mBodyPartsMale.resize(5); // 0 = upper body, 1 = legs, 2 = hands, 3 = feet, 4 = tail
 | |
|                     mBodyPartsFemale.resize(5); // 0 = upper body, 1 = legs, 2 = hands, 3 = feet, 4 = tail
 | |
|                 }
 | |
|                 else // TES5
 | |
|                     currPart = 2; // for TES5 NAM1 indicates the start of EGT model
 | |
| 
 | |
|                 if (isTES4)
 | |
|                     currentIndex = 4; // FIXME: argonian tail mesh without preceeding INDX
 | |
|                 else
 | |
|                     currentIndex = 0xffffffff;
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|             case ESM::fourCC("MNAM"):
 | |
|                 isMale = true;
 | |
|                 break; /* 2, 5, 7 */
 | |
|             case ESM::fourCC("FNAM"):
 | |
|                 isMale = false;
 | |
|                 break; /* 3, 6, 8 */
 | |
|             //
 | |
|             case ESM::fourCC("HNAM"):
 | |
|             {
 | |
|                 // FIXME: this is a texture name in FO4
 | |
|                 if (subHdr.dataSize % sizeof(ESM::FormId32) != 0)
 | |
|                     reader.skipSubRecordData();
 | |
|                 else
 | |
|                 {
 | |
|                     std::size_t numHairChoices = subHdr.dataSize / sizeof(ESM::FormId32);
 | |
|                     mHairChoices.resize(numHairChoices);
 | |
|                     for (unsigned int i = 0; i < numHairChoices; ++i)
 | |
|                         reader.getFormId(mHairChoices.at(i));
 | |
|                 }
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|             case ESM::fourCC("ENAM"):
 | |
|             {
 | |
|                 std::size_t numEyeChoices = subHdr.dataSize / sizeof(ESM::FormId32);
 | |
|                 mEyeChoices.resize(numEyeChoices);
 | |
|                 for (unsigned int i = 0; i < numEyeChoices; ++i)
 | |
|                     reader.getFormId(mEyeChoices.at(i));
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|             case ESM::fourCC("FGGS"):
 | |
|             {
 | |
|                 if (isMale || isTES4)
 | |
|                 {
 | |
|                     mSymShapeModeCoefficients.resize(50);
 | |
|                     for (std::size_t i = 0; i < 50; ++i)
 | |
|                         reader.get(mSymShapeModeCoefficients.at(i));
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     mSymShapeModeCoeffFemale.resize(50);
 | |
|                     for (std::size_t i = 0; i < 50; ++i)
 | |
|                         reader.get(mSymShapeModeCoeffFemale.at(i));
 | |
|                 }
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|             case ESM::fourCC("FGGA"):
 | |
|             {
 | |
|                 if (isMale || isTES4)
 | |
|                 {
 | |
|                     mAsymShapeModeCoefficients.resize(30);
 | |
|                     for (std::size_t i = 0; i < 30; ++i)
 | |
|                         reader.get(mAsymShapeModeCoefficients.at(i));
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     mAsymShapeModeCoeffFemale.resize(30);
 | |
|                     for (std::size_t i = 0; i < 30; ++i)
 | |
|                         reader.get(mAsymShapeModeCoeffFemale.at(i));
 | |
|                 }
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|             case ESM::fourCC("FGTS"):
 | |
|             {
 | |
|                 if (isMale || isTES4)
 | |
|                 {
 | |
|                     mSymTextureModeCoefficients.resize(50);
 | |
|                     for (std::size_t i = 0; i < 50; ++i)
 | |
|                         reader.get(mSymTextureModeCoefficients.at(i));
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     mSymTextureModeCoeffFemale.resize(50);
 | |
|                     for (std::size_t i = 0; i < 50; ++i)
 | |
|                         reader.get(mSymTextureModeCoeffFemale.at(i));
 | |
|                 }
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|             //
 | |
|             case ESM::fourCC("SNAM"): // skipping...2 // only in TES4?
 | |
|             {
 | |
|                 reader.skipSubRecordData();
 | |
|                 break;
 | |
|             }
 | |
|             case ESM::fourCC("XNAM"):
 | |
|             {
 | |
|                 ESM::FormId race;
 | |
|                 std::int32_t adjustment;
 | |
|                 reader.getFormId(race);
 | |
|                 reader.get(adjustment);
 | |
|                 mDisposition[race] = adjustment;
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|             case ESM::fourCC("VNAM"):
 | |
|             {
 | |
|                 if (subHdr.dataSize == 8) // TES4
 | |
|                 {
 | |
|                     reader.getFormId(mVNAM[0]); // For TES4 seems to be 2 race formids
 | |
|                     reader.getFormId(mVNAM[1]);
 | |
|                 }
 | |
|                 else if (subHdr.dataSize == 4) // TES5
 | |
|                 {
 | |
|                     // equipment type flags meant to be uint32 ???  GLOB reference? shows up in
 | |
|                     // SCRO in sript records and CTDA in INFO records
 | |
|                     uint32_t dummy;
 | |
|                     reader.get(dummy);
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     reader.skipSubRecordData();
 | |
|                     // std::cout << "RACE " << ESM::printName(subHdr.typeId) << " skipping..." << subHdr.dataSize <<
 | |
|                     // std::endl;
 | |
|                 }
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|             //
 | |
|             case ESM::fourCC("ANAM"): // TES5
 | |
|             {
 | |
|                 if (isMale)
 | |
|                     reader.getZString(mModelMale);
 | |
|                 else
 | |
|                     reader.getZString(mModelFemale);
 | |
|                 break;
 | |
|             }
 | |
|             case ESM::fourCC("KSIZ"):
 | |
|                 reader.get(mNumKeywords);
 | |
|                 break;
 | |
|             case ESM::fourCC("KWDA"):
 | |
|             {
 | |
|                 ESM::FormId formid;
 | |
|                 for (unsigned int i = 0; i < mNumKeywords; ++i)
 | |
|                     reader.getFormId(formid);
 | |
|                 break;
 | |
|             }
 | |
|             //
 | |
|             case ESM::fourCC("WNAM"): // ARMO FormId
 | |
|             {
 | |
|                 reader.getFormId(mSkin);
 | |
|                 // std::cout << mEditorId << " skin " << formIdToString(mSkin) << std::endl; // FIXME
 | |
|                 break;
 | |
|             }
 | |
|             case ESM::fourCC("BODT"): // body template
 | |
|             {
 | |
|                 reader.get(mBodyTemplate.bodyPart);
 | |
|                 reader.get(mBodyTemplate.flags);
 | |
|                 reader.get(mBodyTemplate.unknown1); // probably padding
 | |
|                 reader.get(mBodyTemplate.unknown2); // probably padding
 | |
|                 reader.get(mBodyTemplate.unknown3); // probably padding
 | |
|                 reader.get(mBodyTemplate.type);
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|             case ESM::fourCC("BOD2"):
 | |
|             {
 | |
|                 if (subHdr.dataSize == 8 || subHdr.dataSize == 4) // TES5, FO4
 | |
|                 {
 | |
|                     reader.get(mBodyTemplate.bodyPart);
 | |
|                     mBodyTemplate.flags = 0;
 | |
|                     mBodyTemplate.unknown1 = 0; // probably padding
 | |
|                     mBodyTemplate.unknown2 = 0; // probably padding
 | |
|                     mBodyTemplate.unknown3 = 0; // probably padding
 | |
|                     mBodyTemplate.type = 0;
 | |
|                     if (subHdr.dataSize == 8)
 | |
|                         reader.get(mBodyTemplate.type);
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     reader.skipSubRecordData();
 | |
|                 }
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|             case ESM::fourCC("HEAD"): // TES5
 | |
|             {
 | |
|                 ESM::FormId formId;
 | |
|                 reader.getFormId(formId);
 | |
| 
 | |
|                 if (currentIndex != 0xffffffff)
 | |
|                 {
 | |
|                     // FIXME: no order? head, mouth, eyes, brow, hair
 | |
|                     if (isMale)
 | |
|                     {
 | |
|                         if (currentIndex >= mHeadPartIdsMale.size())
 | |
|                             mHeadPartIdsMale.resize(currentIndex + 1);
 | |
|                         mHeadPartIdsMale[currentIndex] = formId;
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         if (currentIndex >= mHeadPartIdsFemale.size())
 | |
|                             mHeadPartIdsFemale.resize(currentIndex + 1);
 | |
|                         mHeadPartIdsFemale[currentIndex] = formId;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 // std::cout << mEditorId << (isMale ? " male head " : " female head ")
 | |
|                 // << formIdToString(formId) << " " << currentIndex << std::endl; // FIXME
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|             case ESM::fourCC("NAM3"): // start of hkx model
 | |
|             {
 | |
|                 currPart = 3; // for TES5 NAM3 indicates the start of hkx model
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|             // Not sure for what this is used - maybe to indicate which slots are in use? e.g.:
 | |
|             //
 | |
|             // ManakinRace HEAD
 | |
|             // ManakinRace Hair
 | |
|             // ManakinRace BODY
 | |
|             // ManakinRace Hands
 | |
|             // ManakinRace Forearms
 | |
|             // ManakinRace Amulet
 | |
|             // ManakinRace Ring
 | |
|             // ManakinRace Feet
 | |
|             // ManakinRace Calves
 | |
|             // ManakinRace SHIELD
 | |
|             // ManakinRace
 | |
|             // ManakinRace LongHair
 | |
|             // ManakinRace Circlet
 | |
|             // ManakinRace
 | |
|             // ManakinRace
 | |
|             // ManakinRace
 | |
|             // ManakinRace
 | |
|             // ManakinRace
 | |
|             // ManakinRace
 | |
|             // ManakinRace
 | |
|             // ManakinRace DecapitateHead
 | |
|             // ManakinRace Decapitate
 | |
|             // ManakinRace
 | |
|             // ManakinRace
 | |
|             // ManakinRace
 | |
|             // ManakinRace
 | |
|             // ManakinRace
 | |
|             // ManakinRace
 | |
|             // ManakinRace
 | |
|             // ManakinRace
 | |
|             // ManakinRace
 | |
|             // ManakinRace FX0
 | |
|             case ESM::fourCC("NAME"): // TES5 biped object names (x32)
 | |
|             {
 | |
|                 std::string name;
 | |
|                 reader.getZString(name);
 | |
|                 // std::cout << mEditorId << " " << name << std::endl;
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|             case ESM::fourCC("MTNM"): // movement type
 | |
|             case ESM::fourCC("ATKD"): // attack data
 | |
|             case ESM::fourCC("ATKE"): // attach event
 | |
|             case ESM::fourCC("GNAM"): // body part data
 | |
|             case ESM::fourCC("NAM4"): // material type
 | |
|             case ESM::fourCC("NAM5"): // unarmed impact?
 | |
|             case ESM::fourCC("LNAM"): // close loot sound
 | |
|             case ESM::fourCC("QNAM"): // equipment slot formid
 | |
|             case ESM::fourCC("HCLF"): // default hair colour
 | |
|             case ESM::fourCC("UNES"): // unarmed equipment slot formid
 | |
|             case ESM::fourCC("TINC"):
 | |
|             case ESM::fourCC("TIND"):
 | |
|             case ESM::fourCC("TINI"):
 | |
|             case ESM::fourCC("TINL"):
 | |
|             case ESM::fourCC("TINP"):
 | |
|             case ESM::fourCC("TINT"):
 | |
|             case ESM::fourCC("TINV"):
 | |
|             case ESM::fourCC("TIRS"):
 | |
|             case ESM::fourCC("PHWT"):
 | |
|             case ESM::fourCC("AHCF"):
 | |
|             case ESM::fourCC("AHCM"):
 | |
|             case ESM::fourCC("MPAI"):
 | |
|             case ESM::fourCC("MPAV"):
 | |
|             case ESM::fourCC("DFTF"):
 | |
|             case ESM::fourCC("DFTM"):
 | |
|             case ESM::fourCC("FLMV"):
 | |
|             case ESM::fourCC("FTSF"):
 | |
|             case ESM::fourCC("FTSM"):
 | |
|             case ESM::fourCC("MTYP"):
 | |
|             case ESM::fourCC("NAM7"):
 | |
|             case ESM::fourCC("NAM8"):
 | |
|             case ESM::fourCC("PHTN"):
 | |
|             case ESM::fourCC("RNAM"):
 | |
|             case ESM::fourCC("RNMV"):
 | |
|             case ESM::fourCC("RPRF"):
 | |
|             case ESM::fourCC("RPRM"):
 | |
|             case ESM::fourCC("SNMV"):
 | |
|             case ESM::fourCC("SPCT"):
 | |
|             case ESM::fourCC("SPED"):
 | |
|             case ESM::fourCC("SWMV"):
 | |
|             case ESM::fourCC("WKMV"):
 | |
|             case ESM::fourCC("SPMV"):
 | |
|             case ESM::fourCC("ATKR"):
 | |
|             case ESM::fourCC("CTDA"):
 | |
|             case ESM::fourCC("CIS1"):
 | |
|             case ESM::fourCC("CIS2"):
 | |
|             case ESM::fourCC("MODT"): // Model data
 | |
|             case ESM::fourCC("MODC"):
 | |
|             case ESM::fourCC("MODS"):
 | |
|             case ESM::fourCC("MODF"): // Model data end
 | |
|             //
 | |
|             case ESM::fourCC("YNAM"): // FO3
 | |
|             case ESM::fourCC("NAM2"): // FO3
 | |
|             case ESM::fourCC("VTCK"): // FO3
 | |
|             case ESM::fourCC("MODD"): // FO3
 | |
|             case ESM::fourCC("ONAM"): // FO3
 | |
|             case ESM::fourCC("APPR"): // FO4
 | |
|             case ESM::fourCC("ATKS"): // FO4
 | |
|             case ESM::fourCC("ATKT"): // FO4
 | |
|             case ESM::fourCC("ATKW"): // FO4
 | |
|             case ESM::fourCC("BMMP"): // FO4
 | |
|             case ESM::fourCC("BSMB"): // FO4
 | |
|             case ESM::fourCC("BSMP"): // FO4
 | |
|             case ESM::fourCC("BSMS"): // FO4
 | |
| 
 | |
|             case ESM::fourCC("FMRI"): // FO4
 | |
|             case ESM::fourCC("FMRN"): // FO4
 | |
|             case ESM::fourCC("HLTX"): // FO4
 | |
|             case ESM::fourCC("MLSI"): // FO4
 | |
|             case ESM::fourCC("MPGN"): // FO4
 | |
|             case ESM::fourCC("MPGS"): // FO4
 | |
|             case ESM::fourCC("MPPC"): // FO4
 | |
|             case ESM::fourCC("MPPF"): // FO4
 | |
|             case ESM::fourCC("MPPI"): // FO4
 | |
|             case ESM::fourCC("MPPK"): // FO4
 | |
|             case ESM::fourCC("MPPM"): // FO4
 | |
|             case ESM::fourCC("MPPN"): // FO4
 | |
|             case ESM::fourCC("MPPT"): // FO4
 | |
|             case ESM::fourCC("MSID"): // FO4
 | |
|             case ESM::fourCC("MSM0"): // FO4
 | |
|             case ESM::fourCC("MSM1"): // FO4
 | |
|             case ESM::fourCC("NNAM"): // FO4
 | |
|             case ESM::fourCC("NTOP"): // FO4
 | |
|             case ESM::fourCC("PRPS"): // FO4
 | |
|             case ESM::fourCC("PTOP"): // FO4
 | |
|             case ESM::fourCC("QSTI"): // FO4
 | |
|             case ESM::fourCC("RBPC"): // FO4
 | |
|             case ESM::fourCC("SADD"): // FO4
 | |
|             case ESM::fourCC("SAKD"): // FO4
 | |
|             case ESM::fourCC("SAPT"): // FO4
 | |
|             case ESM::fourCC("SGNM"): // FO4
 | |
|             case ESM::fourCC("SRAC"): // FO4
 | |
|             case ESM::fourCC("SRAF"): // FO4
 | |
|             case ESM::fourCC("STCP"): // FO4
 | |
|             case ESM::fourCC("STKD"): // FO4
 | |
|             case ESM::fourCC("TETI"): // FO4
 | |
|             case ESM::fourCC("TTEB"): // FO4
 | |
|             case ESM::fourCC("TTEC"): // FO4
 | |
|             case ESM::fourCC("TTED"): // FO4
 | |
|             case ESM::fourCC("TTEF"): // FO4
 | |
|             case ESM::fourCC("TTET"): // FO4
 | |
|             case ESM::fourCC("TTGE"): // FO4
 | |
|             case ESM::fourCC("TTGP"): // FO4
 | |
|             case ESM::fourCC("UNWP"): // FO4
 | |
|             case ESM::fourCC("WMAP"): // FO4
 | |
|             case ESM::fourCC("ZNAM"): // FO4
 | |
|                 reader.skipSubRecordData();
 | |
|                 break;
 | |
|             default:
 | |
|                 throw std::runtime_error("ESM4::RACE::load - Unknown subrecord " + ESM::printName(subHdr.typeId));
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| // void ESM4::Race::save(ESM4::Writer& writer) const
 | |
| //{
 | |
| // }
 | |
| 
 | |
| // void ESM4::Race::blank()
 | |
| //{
 | |
| // }
 |