|
|
@ -5,6 +5,8 @@
|
|
|
|
#include <OgreImage.h>
|
|
|
|
#include <OgreImage.h>
|
|
|
|
#include <OgreColourValue.h>
|
|
|
|
#include <OgreColourValue.h>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#include <osgDB/WriteFile>
|
|
|
|
|
|
|
|
|
|
|
|
#include <components/esm/creaturestate.hpp>
|
|
|
|
#include <components/esm/creaturestate.hpp>
|
|
|
|
#include <components/esm/containerstate.hpp>
|
|
|
|
#include <components/esm/containerstate.hpp>
|
|
|
|
|
|
|
|
|
|
|
@ -15,12 +17,14 @@
|
|
|
|
namespace
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
void convertImage(char* data, int size, int width, int height, Ogre::PixelFormat pf, const std::string& out)
|
|
|
|
void convertImage(char* data, int size, int width, int height, GLenum pf, const std::string& out)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
Ogre::Image screenshot;
|
|
|
|
osg::ref_ptr<osg::Image> image (new osg::Image);
|
|
|
|
Ogre::DataStreamPtr stream (new Ogre::MemoryDataStream(data, size));
|
|
|
|
image->allocateImage(width, height, 1, pf, GL_UNSIGNED_BYTE);
|
|
|
|
screenshot.loadRawData(stream, width, height, 1, pf);
|
|
|
|
memcpy(image->data(), data, size);
|
|
|
|
screenshot.save(out);
|
|
|
|
image->flipVertical();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
osgDB::writeImageFile(*image, out);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@ -71,17 +75,20 @@ namespace ESSImport
|
|
|
|
data.resize(esm.getSubSize());
|
|
|
|
data.resize(esm.getSubSize());
|
|
|
|
esm.getExact(&data[0], data.size());
|
|
|
|
esm.getExact(&data[0], data.size());
|
|
|
|
|
|
|
|
|
|
|
|
Ogre::DataStreamPtr stream (new Ogre::MemoryDataStream(&data[0], data.size()));
|
|
|
|
mGlobalMapImage = new osg::Image;
|
|
|
|
mGlobalMapImage.loadRawData(stream, maph.size, maph.size, 1, Ogre::PF_BYTE_RGB);
|
|
|
|
mGlobalMapImage->allocateImage(maph.size, maph.size, 1, GL_RGB, GL_UNSIGNED_BYTE);
|
|
|
|
|
|
|
|
memcpy(mGlobalMapImage->data(), &data[0], data.size());
|
|
|
|
|
|
|
|
|
|
|
|
// to match openmw size
|
|
|
|
// to match openmw size
|
|
|
|
mGlobalMapImage.resize(maph.size*2, maph.size*2, Ogre::Image::FILTER_BILINEAR);
|
|
|
|
// FIXME: filtering?
|
|
|
|
|
|
|
|
mGlobalMapImage->scaleImage(maph.size*2, maph.size*2, 1, GL_UNSIGNED_BYTE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ConvertFMAP::write(ESM::ESMWriter &esm)
|
|
|
|
void ConvertFMAP::write(ESM::ESMWriter &esm)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
int numcells = mGlobalMapImage.getWidth() / 18; // NB truncating, doesn't divide perfectly
|
|
|
|
int numcells = mGlobalMapImage->s() / 18; // NB truncating, doesn't divide perfectly
|
|
|
|
// with the 512x512 map the game has by default
|
|
|
|
// with the 512x512 map the game has by default
|
|
|
|
int cellSize = mGlobalMapImage.getWidth()/numcells;
|
|
|
|
int cellSize = mGlobalMapImage->s()/numcells;
|
|
|
|
|
|
|
|
|
|
|
|
// Note the upper left corner of the (0,0) cell should be at (width/2, height/2)
|
|
|
|
// Note the upper left corner of the (0,0) cell should be at (width/2, height/2)
|
|
|
|
|
|
|
|
|
|
|
@ -90,12 +97,14 @@ namespace ESSImport
|
|
|
|
mContext->mGlobalMapState.mBounds.mMinY = -(numcells-1)/2;
|
|
|
|
mContext->mGlobalMapState.mBounds.mMinY = -(numcells-1)/2;
|
|
|
|
mContext->mGlobalMapState.mBounds.mMaxY = numcells/2;
|
|
|
|
mContext->mGlobalMapState.mBounds.mMaxY = numcells/2;
|
|
|
|
|
|
|
|
|
|
|
|
Ogre::Image image2;
|
|
|
|
osg::ref_ptr<osg::Image> image2 (new osg::Image);
|
|
|
|
std::vector<Ogre::uint8> data;
|
|
|
|
|
|
|
|
int width = cellSize*numcells;
|
|
|
|
int width = cellSize*numcells;
|
|
|
|
int height = cellSize*numcells;
|
|
|
|
int height = cellSize*numcells;
|
|
|
|
|
|
|
|
std::vector<unsigned char> data;
|
|
|
|
data.resize(width*height*4, 0);
|
|
|
|
data.resize(width*height*4, 0);
|
|
|
|
image2.loadDynamicImage(&data[0], width, height, Ogre::PF_BYTE_RGBA);
|
|
|
|
|
|
|
|
|
|
|
|
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)
|
|
|
|
for (std::set<std::pair<int, int> >::const_iterator it = mContext->mExploredCells.begin(); it != mContext->mExploredCells.end(); ++it)
|
|
|
|
{
|
|
|
|
{
|
|
|
@ -108,8 +117,8 @@ namespace ESSImport
|
|
|
|
continue;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int imageLeftSrc = mGlobalMapImage.getWidth()/2;
|
|
|
|
int imageLeftSrc = mGlobalMapImage->s()/2;
|
|
|
|
int imageTopSrc = mGlobalMapImage.getHeight()/2;
|
|
|
|
int imageTopSrc = mGlobalMapImage->t()/2;
|
|
|
|
imageLeftSrc += it->first * cellSize;
|
|
|
|
imageLeftSrc += it->first * cellSize;
|
|
|
|
imageTopSrc -= it->second * cellSize;
|
|
|
|
imageTopSrc -= it->second * cellSize;
|
|
|
|
int imageLeftDst = width/2;
|
|
|
|
int imageLeftDst = width/2;
|
|
|
@ -118,13 +127,31 @@ namespace ESSImport
|
|
|
|
imageTopDst -= it->second * cellSize;
|
|
|
|
imageTopDst -= it->second * cellSize;
|
|
|
|
for (int x=0; x<cellSize; ++x)
|
|
|
|
for (int x=0; x<cellSize; ++x)
|
|
|
|
for (int y=0; y<cellSize; ++y)
|
|
|
|
for (int y=0; y<cellSize; ++y)
|
|
|
|
image2.setColourAt(mGlobalMapImage.getColourAt(imageLeftSrc+x, imageTopSrc+y, 0)
|
|
|
|
{
|
|
|
|
, imageLeftDst+x, imageTopDst+y, 0);
|
|
|
|
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 << "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 << "can't write global map image: " << result.message() << std::endl;
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Ogre::DataStreamPtr encoded = image2.encode("png");
|
|
|
|
std::string outData = ostream.str();
|
|
|
|
mContext->mGlobalMapState.mImageData.resize(encoded->size());
|
|
|
|
mContext->mGlobalMapState.mImageData = std::vector<char>(outData.begin(), outData.end());
|
|
|
|
encoded->read(&mContext->mGlobalMapState.mImageData[0], encoded->size());
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
esm.startRecord(ESM::REC_GMAP);
|
|
|
|
esm.startRecord(ESM::REC_GMAP);
|
|
|
|
mContext->mGlobalMapState.save(esm);
|
|
|
|
mContext->mGlobalMapState.save(esm);
|
|
|
@ -194,7 +221,7 @@ namespace ESSImport
|
|
|
|
std::ostringstream filename;
|
|
|
|
std::ostringstream filename;
|
|
|
|
filename << "fog_" << cell.mData.mX << "_" << cell.mData.mY << ".tga";
|
|
|
|
filename << "fog_" << cell.mData.mX << "_" << cell.mData.mY << ".tga";
|
|
|
|
|
|
|
|
|
|
|
|
convertImage((char*)&newcell.mFogOfWar[0], newcell.mFogOfWar.size()*4, 16, 16, Ogre::PF_BYTE_RGBA, filename.str());
|
|
|
|
convertImage((char*)&newcell.mFogOfWar[0], newcell.mFogOfWar.size()*4, 16, 16, GL_RGBA, filename.str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|