mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-25 02:26:42 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			247 lines
		
	
	
	
		
			9.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			247 lines
		
	
	
	
		
			9.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|   Copyright (C) 2015-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 "loadcell.hpp"
 | |
| 
 | |
| #ifdef NDEBUG // FIXME: debuggigng only
 | |
| #undef NDEBUG
 | |
| #endif
 | |
| 
 | |
| #include <cassert>
 | |
| #include <cfloat> // FLT_MAX for gcc
 | |
| #include <stdexcept>
 | |
| 
 | |
| #include <iostream> // FIXME: debug only
 | |
| 
 | |
| #include "reader.hpp"
 | |
| //#include "writer.hpp"
 | |
| 
 | |
| // TODO: Try loading only EDID and XCLC (along with mFormId, mFlags and mParent)
 | |
| //
 | |
| // But, for external cells we may be scanning the whole record since we don't know if there is
 | |
| // going to be an EDID subrecord. And the vast majority of cells are these kinds.
 | |
| //
 | |
| // So perhaps some testing needs to be done to see if scanning and skipping takes
 | |
| // longer/shorter/same as loading the subrecords.
 | |
| void ESM4::Cell::load(ESM4::Reader& reader)
 | |
| {
 | |
|     mFormId = reader.hdr().record.id;
 | |
|     reader.adjustFormId(mFormId);
 | |
|     mFlags = reader.hdr().record.flags;
 | |
|     mParent = reader.currWorld();
 | |
| 
 | |
|     reader.clearCellGrid(); // clear until XCLC FIXME: somehow do this automatically?
 | |
| 
 | |
|     // Sometimes cell 0,0 does not have an XCLC sub record (e.g. ToddLand 000009BF)
 | |
|     // To workaround this issue put a default value if group is "exterior sub cell" and its
 | |
|     // grid from label is "0 0".  Note the reversed X/Y order (no matter since they're both 0
 | |
|     // anyway).
 | |
|     if (reader.grp().type == ESM4::Grp_ExteriorSubCell && reader.grp().label.grid[1] == 0
 | |
|         && reader.grp().label.grid[0] == 0)
 | |
|     {
 | |
|         ESM4::CellGrid currCellGrid;
 | |
|         currCellGrid.grid.x = 0;
 | |
|         currCellGrid.grid.y = 0;
 | |
|         reader.setCurrCellGrid(currCellGrid); // side effect: sets mCellGridValid  true
 | |
|     }
 | |
| 
 | |
|     // WARN: we need to call setCurrCell (and maybe setCurrCellGrid?) again before loading
 | |
|     // cell child groups if we are loading them after restoring the context
 | |
|     // (may be easier to update the context before saving?)
 | |
|     reader.setCurrCell(mFormId); // save for LAND (and other children) to access later
 | |
|     std::uint32_t esmVer = reader.esmVersion();
 | |
|     bool isFONV = esmVer == ESM::VER_132 || esmVer == ESM::VER_133 || esmVer == ESM::VER_134;
 | |
| 
 | |
|     while (reader.getSubRecordHeader())
 | |
|     {
 | |
|         const ESM4::SubRecordHeader& subHdr = reader.subRecordHeader();
 | |
|         switch (subHdr.typeId)
 | |
|         {
 | |
|             case ESM4::SUB_EDID:
 | |
|             {
 | |
|                 if (!reader.getZString(mEditorId))
 | |
|                     throw std::runtime_error("CELL EDID data read error");
 | |
| #if 0
 | |
|                 std::string padding;
 | |
|                 padding.insert(0, reader.stackSize()*2, ' ');
 | |
|                 std::cout << padding << "CELL Editor ID: " << mEditorId << std::endl;
 | |
| #endif
 | |
|                 break;
 | |
|             }
 | |
|             case ESM4::SUB_XCLC:
 | |
|             {
 | |
|                 //(X, Y) grid location of the cell followed by flags. Always in
 | |
|                 // exterior cells and never in interior cells.
 | |
|                 //
 | |
|                 //    int32 - X
 | |
|                 //    int32 - Y
 | |
|                 //    uint32 - flags (high bits look random)
 | |
|                 //
 | |
|                 //        0x1 - Force Hide Land Quad 1
 | |
|                 //        0x2 - Force Hide Land Quad 2
 | |
|                 //        0x4 - Force Hide Land Quad 3
 | |
|                 //        0x8 - Force Hide Land Quad 4
 | |
|                 uint32_t flags;
 | |
|                 reader.get(mX);
 | |
|                 reader.get(mY);
 | |
| #if 0
 | |
|                 std::string padding;
 | |
|                 padding.insert(0, reader.stackSize()*2, ' ');
 | |
|                 std::cout << padding << "CELL group " << ESM4::printLabel(reader.grp().label, reader.grp().type) << std::endl;
 | |
|                 std::cout << padding << "CELL formId " << std::hex << reader.hdr().record.id << std::endl;
 | |
|                 std::cout << padding << "CELL X " << std::dec << mX << ", Y " << mY << std::endl;
 | |
| #endif
 | |
|                 if (esmVer == ESM::VER_094 || esmVer == ESM::VER_170 || isFONV)
 | |
|                     if (subHdr.dataSize == 12)
 | |
|                         reader.get(flags); // not in Obvlivion, nor FO3/FONV
 | |
| 
 | |
|                 // Remember cell grid for later (loading LAND, NAVM which should be CELL temporary children)
 | |
|                 // Note that grids only apply for external cells.  For interior cells use the cell's formid.
 | |
|                 ESM4::CellGrid currCell;
 | |
|                 currCell.grid.x = (int16_t)mX;
 | |
|                 currCell.grid.y = (int16_t)mY;
 | |
|                 reader.setCurrCellGrid(currCell);
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|             case ESM4::SUB_FULL:
 | |
|                 reader.getLocalizedString(mFullName);
 | |
|                 break;
 | |
|             case ESM4::SUB_DATA:
 | |
|             {
 | |
|                 if (esmVer == ESM::VER_094 || esmVer == ESM::VER_170 || isFONV)
 | |
|                     if (subHdr.dataSize == 2)
 | |
|                         reader.get(mCellFlags);
 | |
|                     else
 | |
|                     {
 | |
|                         assert(subHdr.dataSize == 1 && "CELL unexpected DATA flag size");
 | |
|                         reader.get(&mCellFlags, sizeof(std::uint8_t));
 | |
|                     }
 | |
|                 else
 | |
|                 {
 | |
|                     reader.get((std::uint8_t&)mCellFlags); // 8 bits in Obvlivion
 | |
|                 }
 | |
| #if 0
 | |
|                 std::string padding;
 | |
|                 padding.insert(0, reader.stackSize()*2, ' ');
 | |
|                 std::cout << padding  << "flags: " << std::hex << mCellFlags << std::endl;
 | |
| #endif
 | |
|                 break;
 | |
|             }
 | |
|             case ESM4::SUB_XCLR: // for exterior cells
 | |
|             {
 | |
|                 mRegions.resize(subHdr.dataSize / sizeof(FormId));
 | |
|                 for (std::vector<FormId>::iterator it = mRegions.begin(); it != mRegions.end(); ++it)
 | |
|                 {
 | |
|                     reader.getFormId(*it);
 | |
| #if 0
 | |
|                     std::string padding;
 | |
|                     padding.insert(0, reader.stackSize()*2, ' ');
 | |
|                     std::cout << padding  << "region: " << std::hex << *it << std::endl;
 | |
| #endif
 | |
|                 }
 | |
|                 break;
 | |
|             }
 | |
|             case ESM4::SUB_XOWN:
 | |
|                 reader.getFormId(mOwner);
 | |
|                 break;
 | |
|             case ESM4::SUB_XGLB:
 | |
|                 reader.getFormId(mGlobal);
 | |
|                 break; // Oblivion only?
 | |
|             case ESM4::SUB_XCCM:
 | |
|                 reader.getFormId(mClimate);
 | |
|                 break;
 | |
|             case ESM4::SUB_XCWT:
 | |
|                 reader.getFormId(mWater);
 | |
|                 break;
 | |
|             case ESM4::SUB_XCLW:
 | |
|                 reader.get(mWaterHeight);
 | |
|                 break;
 | |
|             case ESM4::SUB_XCLL:
 | |
|             {
 | |
|                 if (esmVer == ESM::VER_094 || esmVer == ESM::VER_170 || isFONV)
 | |
|                 {
 | |
|                     if (subHdr.dataSize == 40) // FO3/FONV
 | |
|                         reader.get(mLighting);
 | |
|                     else if (subHdr.dataSize == 92) // TES5
 | |
|                     {
 | |
|                         reader.get(mLighting);
 | |
|                         reader.skipSubRecordData(52); // FIXME
 | |
|                     }
 | |
|                     else
 | |
|                         reader.skipSubRecordData();
 | |
|                 }
 | |
|                 else
 | |
|                     reader.get(&mLighting, 36); // TES4
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|             case ESM4::SUB_XCMT:
 | |
|                 reader.get(mMusicType);
 | |
|                 break; // Oblivion only?
 | |
|             case ESM4::SUB_LTMP:
 | |
|                 reader.getFormId(mLightingTemplate);
 | |
|                 break;
 | |
|             case ESM4::SUB_LNAM:
 | |
|                 reader.get(mLightingTemplateFlags);
 | |
|                 break; // seems to always follow LTMP
 | |
|             case ESM4::SUB_XCMO:
 | |
|                 reader.getFormId(mMusic);
 | |
|                 break;
 | |
|             case ESM4::SUB_XCAS:
 | |
|                 reader.getFormId(mAcousticSpace);
 | |
|                 break;
 | |
|             case ESM4::SUB_TVDT:
 | |
|             case ESM4::SUB_MHDT:
 | |
|             case ESM4::SUB_XCGD:
 | |
|             case ESM4::SUB_XNAM:
 | |
|             case ESM4::SUB_XLCN:
 | |
|             case ESM4::SUB_XWCS:
 | |
|             case ESM4::SUB_XWCU:
 | |
|             case ESM4::SUB_XWCN:
 | |
|             case ESM4::SUB_XCIM:
 | |
|             case ESM4::SUB_XEZN:
 | |
|             case ESM4::SUB_XWEM:
 | |
|             case ESM4::SUB_XILL:
 | |
|             case ESM4::SUB_XRNK: // Oblivion only?
 | |
|             case ESM4::SUB_XCET: // FO3
 | |
|             case ESM4::SUB_IMPF: // FO3 Zeta
 | |
|             {
 | |
|                 // std::cout << "CELL " << ESM::printName(subHdr.typeId) << " skipping..." << std::endl;
 | |
|                 reader.skipSubRecordData();
 | |
|                 break;
 | |
|             }
 | |
|             default:
 | |
|                 throw std::runtime_error("ESM4::CELL::load - Unknown subrecord " + ESM::printName(subHdr.typeId));
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| // void ESM4::Cell::save(ESM4::Writer& writer) const
 | |
| //{
 | |
| // }
 | |
| 
 | |
| void ESM4::Cell::blank() {}
 |