mirror of
https://github.com/TES3MP/openmw-tes3mp.git
synced 2025-01-22 03:53:52 +00:00
29556a1802
A Warning indicates a potential problem in the content file(s) that the user told OpenMW to load. E.g. this might cause an object to not display at all or as intended, however the rest of the game will run fine. An Error, however, is more likely to be a bug with the engine itself - it means that basic assumptions have been violated and the engine might not run correctly anymore. The above mostly applies to errors/warnings during game-play; startup issues are handled differently: when a file is completely invalid/corrupted to the point that the engine can not start, that might cause messages that are worded as Error due to the severity of the issue but are not necessarily the engine's fault. Hopefully, being a little more consistent here will alleviate confusion among users as to when a log message should be reported and to whom.
416 lines
16 KiB
C++
416 lines
16 KiB
C++
#include "converter.hpp"
|
|
|
|
#include <stdexcept>
|
|
|
|
#include <osgDB/WriteFile>
|
|
|
|
#include <components/esm/creaturestate.hpp>
|
|
#include <components/esm/containerstate.hpp>
|
|
|
|
#include "convertcrec.hpp"
|
|
#include "convertcntc.hpp"
|
|
#include "convertscri.hpp"
|
|
|
|
namespace
|
|
{
|
|
|
|
void convertImage(char* data, int size, int width, int height, GLenum pf, const std::string& out)
|
|
{
|
|
osg::ref_ptr<osg::Image> image (new osg::Image);
|
|
image->allocateImage(width, height, 1, pf, GL_UNSIGNED_BYTE);
|
|
memcpy(image->data(), data, size);
|
|
image->flipVertical();
|
|
|
|
osgDB::writeImageFile(*image, out);
|
|
}
|
|
|
|
|
|
void convertCellRef(const ESSImport::CellRef& cellref, ESM::ObjectState& objstate)
|
|
{
|
|
objstate.mEnabled = cellref.mEnabled;
|
|
objstate.mPosition = cellref.mPos;
|
|
objstate.mRef.mRefNum = cellref.mRefNum;
|
|
if (cellref.mDeleted)
|
|
objstate.mCount = 0;
|
|
convertSCRI(cellref.mSCRI, objstate.mLocals);
|
|
objstate.mHasLocals = !objstate.mLocals.mVariables.empty();
|
|
|
|
if (cellref.mHasANIS)
|
|
convertANIS(cellref.mANIS, objstate.mAnimationState);
|
|
}
|
|
|
|
bool isIndexedRefId(const std::string& indexedRefId)
|
|
{
|
|
if (indexedRefId.size() <= 8)
|
|
return false;
|
|
|
|
if (indexedRefId.find_first_not_of("0123456789") == std::string::npos)
|
|
return false; // entirely numeric refid, this is a reference to
|
|
// a dynamically created record e.g. player-enchanted weapon
|
|
|
|
std::string index = indexedRefId.substr(indexedRefId.size()-8);
|
|
if(index.find_first_not_of("0123456789ABCDEF") == std::string::npos )
|
|
return true;
|
|
return false;
|
|
}
|
|
}
|
|
|
|
namespace ESSImport
|
|
{
|
|
|
|
|
|
struct MAPH
|
|
{
|
|
unsigned int size;
|
|
unsigned int value;
|
|
};
|
|
|
|
void ConvertFMAP::read(ESM::ESMReader &esm)
|
|
{
|
|
MAPH maph;
|
|
esm.getHNT(maph, "MAPH");
|
|
std::vector<char> data;
|
|
esm.getSubNameIs("MAPD");
|
|
esm.getSubHeader();
|
|
data.resize(esm.getSubSize());
|
|
esm.getExact(&data[0], data.size());
|
|
|
|
mGlobalMapImage = new osg::Image;
|
|
mGlobalMapImage->allocateImage(maph.size, maph.size, 1, GL_RGB, GL_UNSIGNED_BYTE);
|
|
memcpy(mGlobalMapImage->data(), &data[0], data.size());
|
|
|
|
// to match openmw size
|
|
// FIXME: filtering?
|
|
mGlobalMapImage->scaleImage(maph.size*2, maph.size*2, 1, GL_UNSIGNED_BYTE);
|
|
}
|
|
|
|
void ConvertFMAP::write(ESM::ESMWriter &esm)
|
|
{
|
|
int numcells = mGlobalMapImage->s() / 18; // NB truncating, doesn't divide perfectly
|
|
// with the 512x512 map the game has by default
|
|
int cellSize = mGlobalMapImage->s()/numcells;
|
|
|
|
// Note the upper left corner of the (0,0) cell should be at (width/2, height/2)
|
|
|
|
mContext->mGlobalMapState.mBounds.mMinX = -numcells/2;
|
|
mContext->mGlobalMapState.mBounds.mMaxX = (numcells-1)/2;
|
|
mContext->mGlobalMapState.mBounds.mMinY = -(numcells-1)/2;
|
|
mContext->mGlobalMapState.mBounds.mMaxY = numcells/2;
|
|
|
|
osg::ref_ptr<osg::Image> image2 (new osg::Image);
|
|
int width = cellSize*numcells;
|
|
int height = cellSize*numcells;
|
|
std::vector<unsigned char> data;
|
|
data.resize(width*height*4, 0);
|
|
|
|
image2->allocateImage(width, height, 1, GL_RGBA, GL_UNSIGNED_BYTE);
|
|
memcpy(image2->data(), &data[0], data.size());
|
|
|
|
for (std::set<std::pair<int, int> >::const_iterator it = mContext->mExploredCells.begin(); it != mContext->mExploredCells.end(); ++it)
|
|
{
|
|
if (it->first > mContext->mGlobalMapState.mBounds.mMaxX
|
|
|| it->first < mContext->mGlobalMapState.mBounds.mMinX
|
|
|| it->second > mContext->mGlobalMapState.mBounds.mMaxY
|
|
|| it->second < mContext->mGlobalMapState.mBounds.mMinY)
|
|
{
|
|
// out of bounds, I think this could happen, since the original engine had a fixed-size map
|
|
continue;
|
|
}
|
|
|
|
int imageLeftSrc = mGlobalMapImage->s()/2;
|
|
int imageTopSrc = mGlobalMapImage->t()/2;
|
|
imageLeftSrc += it->first * cellSize;
|
|
imageTopSrc -= it->second * cellSize;
|
|
int imageLeftDst = width/2;
|
|
int imageTopDst = height/2;
|
|
imageLeftDst += it->first * cellSize;
|
|
imageTopDst -= it->second * cellSize;
|
|
for (int x=0; x<cellSize; ++x)
|
|
for (int y=0; y<cellSize; ++y)
|
|
{
|
|
unsigned int col = *(unsigned int*)mGlobalMapImage->data(imageLeftSrc+x, imageTopSrc+y, 0);
|
|
*(unsigned int*)image2->data(imageLeftDst+x, imageTopDst+y, 0) = col;
|
|
}
|
|
}
|
|
|
|
std::stringstream ostream;
|
|
osgDB::ReaderWriter* readerwriter = osgDB::Registry::instance()->getReaderWriterForExtension("png");
|
|
if (!readerwriter)
|
|
{
|
|
std::cerr << "Error: can't write global map image, no png readerwriter found" << std::endl;
|
|
return;
|
|
}
|
|
|
|
image2->flipVertical();
|
|
|
|
osgDB::ReaderWriter::WriteResult result = readerwriter->writeImage(*image2, ostream);
|
|
if (!result.success())
|
|
{
|
|
std::cerr << "Error: can't write global map image: " << result.message() << " code " << result.status() << std::endl;
|
|
return;
|
|
}
|
|
|
|
std::string outData = ostream.str();
|
|
mContext->mGlobalMapState.mImageData = std::vector<char>(outData.begin(), outData.end());
|
|
|
|
esm.startRecord(ESM::REC_GMAP);
|
|
mContext->mGlobalMapState.save(esm);
|
|
esm.endRecord(ESM::REC_GMAP);
|
|
}
|
|
|
|
void ConvertCell::read(ESM::ESMReader &esm)
|
|
{
|
|
ESM::Cell cell;
|
|
bool isDeleted = false;
|
|
|
|
cell.load(esm, isDeleted, false);
|
|
|
|
// I wonder what 0x40 does?
|
|
if (cell.isExterior() && cell.mData.mFlags & 0x20)
|
|
{
|
|
mContext->mGlobalMapState.mMarkers.insert(std::make_pair(cell.mData.mX, cell.mData.mY));
|
|
}
|
|
|
|
// note if the player is in a nameless exterior cell, we will assign the cellId later based on player position
|
|
if (cell.mName == mContext->mPlayerCellName)
|
|
{
|
|
mContext->mPlayer.mCellId = cell.getCellId();
|
|
}
|
|
|
|
Cell newcell;
|
|
newcell.mCell = cell;
|
|
|
|
// fog of war
|
|
// seems to be a 1-bit pixel format, 16*16 pixels
|
|
// TODO: add bleeding of FOW into neighbouring cells (openmw handles this by writing to the textures,
|
|
// MW handles it when rendering only)
|
|
unsigned char nam8[32];
|
|
// exterior has 1 NAM8, interior can have multiple ones, and have an extra 4 byte flag at the start
|
|
// (probably offset of that specific fog texture?)
|
|
while (esm.isNextSub("NAM8"))
|
|
{
|
|
if (cell.isExterior()) // TODO: NAM8 occasionally exists for cells that haven't been explored.
|
|
// are there any flags marking explored cells?
|
|
mContext->mExploredCells.insert(std::make_pair(cell.mData.mX, cell.mData.mY));
|
|
|
|
esm.getSubHeader();
|
|
|
|
if (esm.getSubSize() == 36)
|
|
{
|
|
// flag on interiors
|
|
esm.skip(4);
|
|
}
|
|
|
|
esm.getExact(nam8, 32);
|
|
|
|
newcell.mFogOfWar.reserve(16*16);
|
|
for (int x=0; x<16; ++x)
|
|
{
|
|
for (int y=0; y<16; ++y)
|
|
{
|
|
size_t pos = x*16+y;
|
|
size_t bytepos = pos/8;
|
|
assert(bytepos<32);
|
|
int bit = pos%8;
|
|
newcell.mFogOfWar.push_back(((nam8[bytepos] >> bit) & (0x1)) ? 0xffffffff : 0x000000ff);
|
|
}
|
|
}
|
|
|
|
if (cell.isExterior())
|
|
{
|
|
std::ostringstream filename;
|
|
filename << "fog_" << cell.mData.mX << "_" << cell.mData.mY << ".tga";
|
|
|
|
convertImage((char*)&newcell.mFogOfWar[0], newcell.mFogOfWar.size()*4, 16, 16, GL_RGBA, filename.str());
|
|
}
|
|
}
|
|
|
|
// moved reference, not handled yet
|
|
// NOTE: MVRF can also occur in within normal references (importcellref.cpp)?
|
|
// this does not match the ESM file implementation,
|
|
// verify if that can happen with ESM files too
|
|
while (esm.isNextSub("MVRF"))
|
|
{
|
|
esm.skipHSub(); // skip MVRF
|
|
esm.getSubName();
|
|
esm.skipHSub(); // skip CNDT
|
|
}
|
|
|
|
std::vector<CellRef> cellrefs;
|
|
while (esm.hasMoreSubs() && esm.isNextSub("FRMR"))
|
|
{
|
|
CellRef ref;
|
|
ref.load (esm);
|
|
cellrefs.push_back(ref);
|
|
}
|
|
|
|
while (esm.isNextSub("MPCD"))
|
|
{
|
|
float notepos[3];
|
|
esm.getHT(notepos, 3*sizeof(float));
|
|
|
|
// Markers seem to be arranged in a 32*32 grid, notepos has grid-indices.
|
|
// This seems to be the reason markers can't be placed everywhere in interior cells,
|
|
// i.e. when the grid is exceeded.
|
|
// Converting the interior markers correctly could be rather tricky, but is probably similar logic
|
|
// as used for the FoW texture placement, which we need to figure out anyway
|
|
notepos[1] += 31.f;
|
|
notepos[0] += 0.5;
|
|
notepos[1] += 0.5;
|
|
notepos[0] = 8192 * notepos[0] / 32.f;
|
|
notepos[1] = 8192 * notepos[1] / 32.f;
|
|
if (cell.isExterior())
|
|
{
|
|
notepos[0] += 8192 * cell.mData.mX;
|
|
notepos[1] += 8192 * cell.mData.mY;
|
|
}
|
|
// TODO: what encoding is this in?
|
|
std::string note = esm.getHNString("MPNT");
|
|
ESM::CustomMarker marker;
|
|
marker.mWorldX = notepos[0];
|
|
marker.mWorldY = notepos[1];
|
|
marker.mNote = note;
|
|
marker.mCell = cell.getCellId();
|
|
mMarkers.push_back(marker);
|
|
}
|
|
|
|
newcell.mRefs = cellrefs;
|
|
|
|
|
|
if (cell.isExterior())
|
|
mExtCells[std::make_pair(cell.mData.mX, cell.mData.mY)] = newcell;
|
|
else
|
|
mIntCells[cell.mName] = newcell;
|
|
}
|
|
|
|
void ConvertCell::writeCell(const Cell &cell, ESM::ESMWriter& esm)
|
|
{
|
|
ESM::Cell esmcell = cell.mCell;
|
|
esm.startRecord(ESM::REC_CSTA);
|
|
ESM::CellState csta;
|
|
csta.mHasFogOfWar = 0;
|
|
csta.mId = esmcell.getCellId();
|
|
csta.mId.save(esm);
|
|
// TODO csta.mLastRespawn;
|
|
// shouldn't be needed if we respawn on global schedule like in original MW
|
|
csta.mWaterLevel = esmcell.mWater;
|
|
csta.save(esm);
|
|
|
|
for (std::vector<CellRef>::const_iterator refIt = cell.mRefs.begin(); refIt != cell.mRefs.end(); ++refIt)
|
|
{
|
|
const CellRef& cellref = *refIt;
|
|
ESM::CellRef out (cellref);
|
|
|
|
// TODO: use mContext->mCreatures/mNpcs
|
|
|
|
if (!isIndexedRefId(cellref.mIndexedRefId))
|
|
{
|
|
// non-indexed RefNum, i.e. no CREC/NPCC/CNTC record associated with it
|
|
// this could be any type of object really (even creatures/npcs too)
|
|
out.mRefID = cellref.mIndexedRefId;
|
|
std::string idLower = Misc::StringUtils::lowerCase(out.mRefID);
|
|
|
|
ESM::ObjectState objstate;
|
|
objstate.blank();
|
|
objstate.mRef = out;
|
|
objstate.mRef.mRefID = idLower;
|
|
objstate.mHasCustomState = false;
|
|
convertCellRef(cellref, objstate);
|
|
esm.writeHNT ("OBJE", 0);
|
|
objstate.save(esm);
|
|
continue;
|
|
}
|
|
else
|
|
{
|
|
std::stringstream stream;
|
|
stream << std::hex << cellref.mIndexedRefId.substr(cellref.mIndexedRefId.size()-8,8);
|
|
int refIndex;
|
|
stream >> refIndex;
|
|
|
|
out.mRefID = cellref.mIndexedRefId.substr(0,cellref.mIndexedRefId.size()-8);
|
|
std::string idLower = Misc::StringUtils::lowerCase(out.mRefID);
|
|
|
|
std::map<std::pair<int, std::string>, NPCC>::const_iterator npccIt = mContext->mNpcChanges.find(
|
|
std::make_pair(refIndex, out.mRefID));
|
|
if (npccIt != mContext->mNpcChanges.end())
|
|
{
|
|
ESM::NpcState objstate;
|
|
objstate.blank();
|
|
objstate.mRef = out;
|
|
objstate.mRef.mRefID = idLower;
|
|
// TODO: need more micromanagement here so we don't overwrite values
|
|
// from the ESM with default values
|
|
if (cellref.mHasACDT)
|
|
convertACDT(cellref.mACDT, objstate.mCreatureStats);
|
|
if (cellref.mHasACSC)
|
|
convertACSC(cellref.mACSC, objstate.mCreatureStats);
|
|
convertNpcData(cellref, objstate.mNpcStats);
|
|
convertNPCC(npccIt->second, objstate);
|
|
convertCellRef(cellref, objstate);
|
|
esm.writeHNT ("OBJE", ESM::REC_NPC_);
|
|
objstate.save(esm);
|
|
continue;
|
|
}
|
|
|
|
std::map<std::pair<int, std::string>, CNTC>::const_iterator cntcIt = mContext->mContainerChanges.find(
|
|
std::make_pair(refIndex, out.mRefID));
|
|
if (cntcIt != mContext->mContainerChanges.end())
|
|
{
|
|
ESM::ContainerState objstate;
|
|
objstate.blank();
|
|
objstate.mRef = out;
|
|
objstate.mRef.mRefID = idLower;
|
|
convertCNTC(cntcIt->second, objstate);
|
|
convertCellRef(cellref, objstate);
|
|
esm.writeHNT ("OBJE", ESM::REC_CONT);
|
|
objstate.save(esm);
|
|
continue;
|
|
}
|
|
|
|
std::map<std::pair<int, std::string>, CREC>::const_iterator crecIt = mContext->mCreatureChanges.find(
|
|
std::make_pair(refIndex, out.mRefID));
|
|
if (crecIt != mContext->mCreatureChanges.end())
|
|
{
|
|
ESM::CreatureState objstate;
|
|
objstate.blank();
|
|
objstate.mRef = out;
|
|
objstate.mRef.mRefID = idLower;
|
|
// TODO: need more micromanagement here so we don't overwrite values
|
|
// from the ESM with default values
|
|
if (cellref.mHasACDT)
|
|
convertACDT(cellref.mACDT, objstate.mCreatureStats);
|
|
if (cellref.mHasACSC)
|
|
convertACSC(cellref.mACSC, objstate.mCreatureStats);
|
|
convertCREC(crecIt->second, objstate);
|
|
convertCellRef(cellref, objstate);
|
|
esm.writeHNT ("OBJE", ESM::REC_CREA);
|
|
objstate.save(esm);
|
|
continue;
|
|
}
|
|
|
|
std::stringstream error;
|
|
error << "Can't find type for " << cellref.mIndexedRefId << std::endl;
|
|
throw std::runtime_error(error.str());
|
|
}
|
|
}
|
|
|
|
esm.endRecord(ESM::REC_CSTA);
|
|
}
|
|
|
|
void ConvertCell::write(ESM::ESMWriter &esm)
|
|
{
|
|
for (std::map<std::string, Cell>::const_iterator it = mIntCells.begin(); it != mIntCells.end(); ++it)
|
|
writeCell(it->second, esm);
|
|
|
|
for (std::map<std::pair<int, int>, Cell>::const_iterator it = mExtCells.begin(); it != mExtCells.end(); ++it)
|
|
writeCell(it->second, esm);
|
|
|
|
for (std::vector<ESM::CustomMarker>::const_iterator it = mMarkers.begin(); it != mMarkers.end(); ++it)
|
|
{
|
|
esm.startRecord(ESM::REC_MARK);
|
|
it->save(esm);
|
|
esm.endRecord(ESM::REC_MARK);
|
|
}
|
|
}
|
|
|
|
}
|