#include "loadland.hpp" namespace ESM { void Land::LandData::save(ESMWriter &esm) { // TODO: Make this actually work. esm.writeHNT("VNML", normals, sizeof(VNML)); esm.writeHNT("VHGT", heights, sizeof(VHGT)); //esm.writeHNT("WNAM", 0, 81); esm.startSubRecord("WNAM"); for (int i = 0; i < 81; i++) esm.writeT((char)0x80, 1); esm.endRecord("WNAM"); if (dataTypes & Land::DATA_VCLR) esm.writeHNT("VCLR", colours, 3*LAND_NUM_VERTS); if (dataTypes & Land::DATA_VTEX) esm.writeHNT("VTEX", textures, 512); } Land::Land() : flags(0) , X(0) , Y(0) , mEsm(NULL) // , hasData(false) , dataTypes(0) , dataLoaded(false) , landData(NULL) { } Land::~Land() { delete landData; } void Land::load(ESMReader &esm) { mEsm = &esm; // Get the grid location esm.getSubNameIs("INTV"); esm.getSubHeaderIs(8); esm.getT(X); esm.getT(Y); esm.getHNT(flags, "DATA"); // Store the file position context = esm.getContext(); hasData = false; // Skip these here. Load the actual data when the cell is loaded. if (esm.isNextSub("VNML")) { esm.skipHSubSize(12675); dataTypes |= DATA_VNML; } if (esm.isNextSub("VHGT")) { esm.skipHSubSize(4232); dataTypes |= DATA_VHGT; } if (esm.isNextSub("WNAM")) { esm.skipHSubSize(81); dataTypes |= DATA_WNAM; } if (esm.isNextSub("VCLR")) { esm.skipHSubSize(12675); dataTypes |= DATA_VCLR; } if (esm.isNextSub("VTEX")) { esm.skipHSubSize(512); dataTypes |= DATA_VTEX; } // We need all three of VNML, VHGT and VTEX in order to use the // landscape. (Though Morrowind seems to accept terrain without VTEX/VCLR entries) hasData = dataTypes & (DATA_VNML|DATA_VHGT|DATA_WNAM); dataLoaded = false; landData = NULL; } void Land::save(ESMWriter &esm) { esm.startSubRecord("INTV"); esm.writeT(X); esm.writeT(Y); esm.endRecord("INTV"); esm.writeHNT("DATA", flags); // TODO: Land! bool wasLoaded = dataLoaded; if (hasData) loadData(); // I think it might be a good idea to have // the data loaded before trying to save it if (dataLoaded) landData->save(esm); if (!wasLoaded) unloadData(); // Don't need to keep the data loaded if it wasn't already } void Land::loadData() { if (dataLoaded) { return; } landData = new LandData; if (hasData) { mEsm->restoreContext(context); memset(landData->normals, 0, LAND_NUM_VERTS * 3); //esm.getHNExact(landData->normals, sizeof(VNML), "VNML"); if (mEsm->isNextSub("VNML")) { mEsm->skipHSubSize(12675); } VHGT rawHeights; mEsm->getHNExact(&rawHeights, sizeof(VHGT), "VHGT"); int currentHeightOffset = rawHeights.heightOffset; for (int y = 0; y < LAND_SIZE; y++) { currentHeightOffset += rawHeights.heightData[y * LAND_SIZE]; landData->heights[y * LAND_SIZE] = currentHeightOffset * HEIGHT_SCALE; int tempOffset = currentHeightOffset; for (int x = 1; x < LAND_SIZE; x++) { tempOffset += rawHeights.heightData[y * LAND_SIZE + x]; landData->heights[x + y * LAND_SIZE] = tempOffset * HEIGHT_SCALE; } } if (mEsm->isNextSub("WNAM")) { mEsm->skipHSubSize(81); } if (mEsm->isNextSub("VCLR")) { landData->usingColours = true; mEsm->getHExact(&landData->colours, 3*LAND_NUM_VERTS); }else{ landData->usingColours = false; } if (mEsm->isNextSub("VTEX")) { //TODO fix magic numbers uint16_t vtex[512]; mEsm->getHExact(&vtex, 512); int readPos = 0; //bit ugly, but it works for ( int y1 = 0; y1 < 4; y1++ ) for ( int x1 = 0; x1 < 4; x1++ ) for ( int y2 = 0; y2 < 4; y2++) for ( int x2 = 0; x2 < 4; x2++ ) landData->textures[(y1*4+y2)*16+(x1*4+x2)] = vtex[readPos++]; } } else { landData->usingColours = false; memset(&landData->textures, 0, 512 * sizeof(uint16_t)); for (int i = 0; i < LAND_NUM_VERTS; i++) { landData->heights[i] = -256.0f * HEIGHT_SCALE; } } landData->dataTypes = dataTypes; dataLoaded = true; } void Land::unloadData() { if (dataLoaded) { delete landData; landData = NULL; dataLoaded = false; } } }