1
0
Fork 1
mirror of https://github.com/TES3MP/openmw-tes3mp.git synced 2025-01-16 17:19:56 +00:00
openmw-tes3mp/apps/openmw/mwrender/localmap.cpp

430 lines
15 KiB
C++
Raw Normal View History

#include "localmap.hpp"
#include <OgreMaterialManager.h>
#include <OgreHardwarePixelBuffer.h>
#include <OgreSceneManager.h>
#include <OgreSceneNode.h>
#include <OgreCamera.h>
#include <OgreTextureManager.h>
2012-10-01 15:17:04 +00:00
#include "../mwworld/esmstore.hpp"
#include "../mwbase/environment.hpp"
#include "../mwbase/world.hpp"
#include "../mwbase/windowmanager.hpp"
2012-03-17 12:59:51 +00:00
#include "renderconst.hpp"
#include "renderingmanager.hpp"
using namespace MWRender;
using namespace Ogre;
LocalMap::LocalMap(OEngine::Render::OgreRenderer* rend, MWRender::RenderingManager* rendering) :
2012-03-29 17:20:09 +00:00
mInterior(false), mCellX(0), mCellY(0)
{
mRendering = rend;
2012-04-02 17:37:24 +00:00
mRenderingManager = rendering;
2012-03-29 16:16:11 +00:00
mCameraPosNode = mRendering->getScene()->getRootSceneNode()->createChildSceneNode();
mCameraRotNode = mCameraPosNode->createChildSceneNode();
2012-03-29 16:16:11 +00:00
mCameraNode = mCameraRotNode->createChildSceneNode();
mCellCamera = mRendering->getScene()->createCamera("CellCamera");
mCellCamera->setProjectionType(PT_ORTHOGRAPHIC);
2012-03-29 16:16:11 +00:00
mCameraNode->attachObject(mCellCamera);
mLight = mRendering->getScene()->createLight();
mLight->setType (Ogre::Light::LT_DIRECTIONAL);
2013-02-27 08:20:42 +00:00
mLight->setDirection (Ogre::Vector3(0.3, 0.3, -0.7));
mLight->setVisible (false);
mLight->setDiffuseColour (ColourValue(0.7,0.7,0.7));
2012-03-14 16:44:19 +00:00
}
LocalMap::~LocalMap()
{
deleteBuffers();
}
2012-04-02 17:37:24 +00:00
const Ogre::Vector2 LocalMap::rotatePoint(const Ogre::Vector2& p, const Ogre::Vector2& c, const float angle)
{
return Vector2( Math::Cos(angle) * (p.x - c.x) - Math::Sin(angle) * (p.y - c.y) + c.x,
Math::Sin(angle) * (p.x - c.x) + Math::Cos(angle) * (p.y - c.y) + c.y);
}
2012-03-14 16:44:19 +00:00
void LocalMap::deleteBuffers()
{
mBuffers.clear();
}
void LocalMap::saveTexture(const std::string& texname, const std::string& filename)
{
TexturePtr tex = TextureManager::getSingleton().getByName(texname);
if (tex.isNull()) return;
HardwarePixelBufferSharedPtr readbuffer = tex->getBuffer();
readbuffer->lock(HardwareBuffer::HBL_NORMAL );
const PixelBox &readrefpb = readbuffer->getCurrentLock();
uchar *readrefdata = static_cast<uchar*>(readrefpb.data);
Image img;
img = img.loadDynamicImage (readrefdata, tex->getWidth(),
tex->getHeight(), tex->getFormat());
img.save("./" + filename);
readbuffer->unlock();
}
2012-03-14 16:44:19 +00:00
std::string LocalMap::coordStr(const int x, const int y)
{
return StringConverter::toString(x) + "_" + StringConverter::toString(y);
}
void LocalMap::saveFogOfWar(MWWorld::Ptr::CellStore* cell)
{
if (!mInterior)
{
/*saveTexture("Cell_"+coordStr(mCellX, mCellY)+"_fog",
"Cell_"+coordStr(mCellX, mCellY)+"_fog.png");*/
}
else
{
2013-02-26 12:39:10 +00:00
Vector2 min(mBounds.getMinimum().x, mBounds.getMinimum().y);
Vector2 max(mBounds.getMaximum().x, mBounds.getMaximum().y);
Vector2 length = max-min;
// divide into segments
2012-03-18 19:44:56 +00:00
const int segsX = std::ceil( length.x / sSize );
const int segsY = std::ceil( length.y / sSize );
for (int x=0; x<segsX; ++x)
{
for (int y=0; y<segsY; ++y)
{
/*saveTexture(
mInteriorName + "_" + coordStr(x,y) + "_fog",
mInteriorName + "_" + coordStr(x,y) + "_fog.png");*/
}
}
}
}
void LocalMap::requestMap(MWWorld::Ptr::CellStore* cell, float zMin, float zMax)
{
2012-03-14 13:51:58 +00:00
mInterior = false;
2012-03-29 17:20:09 +00:00
mCameraRotNode->setOrientation(Quaternion::IDENTITY);
2013-02-26 12:39:10 +00:00
mCellCamera->setOrientation(Quaternion(Ogre::Math::Cos(Ogre::Degree(0)/2.f), 0, 0, -Ogre::Math::Sin(Ogre::Degree(0)/2.f)));
2012-03-29 17:20:09 +00:00
2012-11-05 12:07:59 +00:00
int x = cell->mCell->getGridX();
int y = cell->mCell->getGridY();
2012-11-05 12:07:59 +00:00
std::string name = "Cell_"+coordStr(x, y);
2012-03-14 13:51:58 +00:00
mCameraPosNode->setPosition(Vector3(0,0,0));
render((x+0.5)*sSize, (y+0.5)*sSize, zMin, zMax, sSize, sSize, name);
}
void LocalMap::requestMap(MWWorld::Ptr::CellStore* cell,
AxisAlignedBox bounds)
{
// if we're in an empty cell, don't bother rendering anything
if (bounds.isNull ())
return;
2012-03-14 13:51:58 +00:00
mInterior = true;
2012-03-14 16:44:19 +00:00
mBounds = bounds;
2012-03-29 16:16:11 +00:00
2013-02-26 12:39:10 +00:00
float zMin = mBounds.getMinimum().z;
float zMax = mBounds.getMaximum().z;
const Vector2& north = MWBase::Environment::get().getWorld()->getNorthVector(cell);
2013-02-26 12:39:10 +00:00
Radian angle = Ogre::Math::ATan2 (north.x, north.y);
2012-04-02 17:37:24 +00:00
mAngle = angle.valueRadians();
2013-02-26 12:39:10 +00:00
mCellCamera->setOrientation(Quaternion::IDENTITY);
mCameraRotNode->setOrientation(Quaternion(Math::Cos(mAngle/2.f), 0, 0, -Math::Sin(mAngle/2.f)));
2012-03-29 16:16:11 +00:00
2012-04-03 13:34:13 +00:00
// rotate the cell and merge the rotated corners to the bounding box
2013-02-26 12:39:10 +00:00
Vector2 _center(bounds.getCenter().x, bounds.getCenter().y);
Vector3 _c1 = bounds.getCorner(AxisAlignedBox::FAR_LEFT_BOTTOM);
Vector3 _c2 = bounds.getCorner(AxisAlignedBox::FAR_RIGHT_BOTTOM);
Vector3 _c3 = bounds.getCorner(AxisAlignedBox::FAR_LEFT_TOP);
Vector3 _c4 = bounds.getCorner(AxisAlignedBox::FAR_RIGHT_TOP);
Vector2 c1(_c1.x, _c1.y);
Vector2 c2(_c2.x, _c2.y);
Vector2 c3(_c3.x, _c3.y);
Vector2 c4(_c4.x, _c4.y);
2012-04-03 13:34:13 +00:00
c1 = rotatePoint(c1, _center, mAngle);
c2 = rotatePoint(c2, _center, mAngle);
c3 = rotatePoint(c3, _center, mAngle);
c4 = rotatePoint(c4, _center, mAngle);
2013-02-26 12:39:10 +00:00
mBounds.merge(Vector3(c1.x, c1.y, 0));
mBounds.merge(Vector3(c2.x, c2.y, 0));
mBounds.merge(Vector3(c3.x, c3.y, 0));
mBounds.merge(Vector3(c4.x, c4.y, 0));
2012-04-02 17:37:24 +00:00
// apply a little padding
mBounds.setMinimum (mBounds.getMinimum() - Vector3(500,500,0));
mBounds.setMaximum (mBounds.getMaximum() + Vector3(500,500,0));
2013-02-26 12:39:10 +00:00
Vector2 center(mBounds.getCenter().x, mBounds.getCenter().y);
2013-02-26 12:39:10 +00:00
Vector2 min(mBounds.getMinimum().x, mBounds.getMinimum().y);
Vector2 max(mBounds.getMaximum().x, mBounds.getMaximum().y);
Vector2 length = max-min;
2013-02-26 12:39:10 +00:00
mCameraPosNode->setPosition(Vector3(center.x, center.y, 0));
// divide into segments
2012-03-18 19:44:56 +00:00
const int segsX = std::ceil( length.x / sSize );
const int segsY = std::ceil( length.y / sSize );
2012-11-05 12:07:59 +00:00
mInteriorName = cell->mCell->mName;
2012-03-14 16:44:19 +00:00
for (int x=0; x<segsX; ++x)
{
for (int y=0; y<segsY; ++y)
{
2012-03-18 19:44:56 +00:00
Vector2 start = min + Vector2(sSize*x,sSize*y);
Vector2 newcenter = start + 4096;
2013-02-26 12:39:10 +00:00
render(newcenter.x - center.x, newcenter.y - center.y, zMin, zMax, sSize, sSize,
2012-11-05 12:07:59 +00:00
cell->mCell->mName + "_" + coordStr(x,y));
}
}
}
void LocalMap::render(const float x, const float y,
const float zlow, const float zhigh,
const float xw, const float yw, const std::string& texture)
{
mCellCamera->setFarClipDistance( (zhigh-zlow) + 2000 );
mCellCamera->setNearClipDistance(50);
mCellCamera->setOrthoWindow(xw, yw);
mCameraNode->setPosition(Vector3(x, y, zhigh+1000));
2013-02-04 23:39:56 +00:00
// disable fog (only necessary for fixed function, the shader based
// materials already do this through local_map material configuration)
float oldFogStart = mRendering->getScene()->getFogStart();
float oldFogEnd = mRendering->getScene()->getFogEnd();
Ogre::ColourValue oldFogColour = mRendering->getScene()->getFogColour();
mRendering->getScene()->setFog(FOG_NONE);
// set up lighting
Ogre::ColourValue oldAmbient = mRendering->getScene()->getAmbientLight();
mRendering->getScene()->setAmbientLight(Ogre::ColourValue(0.3, 0.3, 0.3));
mRenderingManager->disableLights(true);
mLight->setVisible(true);
TexturePtr tex;
// try loading from memory
tex = TextureManager::getSingleton().getByName(texture);
if (tex.isNull())
{
// try loading from disk
//if (boost::filesystem::exists(texture+".jpg"))
//{
/// \todo
//}
//else
{
// render
tex = TextureManager::getSingleton().createManual(
texture,
ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME,
TEX_TYPE_2D,
xw*sMapResolution/sSize, yw*sMapResolution/sSize,
0,
PF_R8G8B8,
TU_RENDERTARGET);
RenderTarget* rtt = tex->getBuffer()->getRenderTarget();
2013-02-03 19:06:03 +00:00
rtt->setAutoUpdated(false);
Viewport* vp = rtt->addViewport(mCellCamera);
vp->setOverlaysEnabled(false);
vp->setShadowsEnabled(false);
vp->setBackgroundColour(ColourValue(0, 0, 0));
2012-04-03 13:13:47 +00:00
vp->setVisibilityMask(RV_Map);
2012-07-09 17:46:36 +00:00
vp->setMaterialScheme("local_map");
2013-02-04 23:39:56 +00:00
rtt->update();
2012-03-14 13:51:58 +00:00
// create "fog of war" texture
TexturePtr tex2 = TextureManager::getSingleton().createManual(
texture + "_fog",
ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME,
TEX_TYPE_2D,
xw*sFogOfWarResolution/sSize, yw*sFogOfWarResolution/sSize,
2012-03-14 13:51:58 +00:00
0,
PF_A8R8G8B8,
2012-03-16 16:09:31 +00:00
TU_DYNAMIC_WRITE_ONLY_DISCARDABLE);
2012-03-14 16:44:19 +00:00
// create a buffer to use for dynamic operations
2012-03-26 22:18:09 +00:00
std::vector<uint32> buffer;
buffer.resize(sFogOfWarResolution*sFogOfWarResolution);
2012-03-14 16:44:19 +00:00
// initialize to (0, 0, 0, 1)
2012-03-18 19:44:56 +00:00
for (int p=0; p<sFogOfWarResolution*sFogOfWarResolution; ++p)
2012-03-14 16:44:19 +00:00
{
2012-03-26 22:18:09 +00:00
buffer[p] = (255 << 24);
2012-03-14 16:44:19 +00:00
}
2012-03-26 22:18:09 +00:00
memcpy(tex2->getBuffer()->lock(HardwareBuffer::HBL_DISCARD), &buffer[0], sFogOfWarResolution*sFogOfWarResolution*4);
2012-03-14 16:44:19 +00:00
tex2->getBuffer()->unlock();
mBuffers[texture] = buffer;
// save to cache for next time
//rtt->writeContentsToFile("./" + texture + ".jpg");
}
}
mRenderingManager->enableLights(true);
mLight->setVisible(false);
2013-02-03 19:06:03 +00:00
// re-enable fog
2013-02-04 23:39:56 +00:00
mRendering->getScene()->setFog(FOG_LINEAR, oldFogColour, 0, oldFogStart, oldFogEnd);
mRendering->getScene()->setAmbientLight(oldAmbient);
}
2012-08-28 15:30:34 +00:00
void LocalMap::getInteriorMapPosition (Ogre::Vector2 pos, float& nX, float& nY, int& x, int& y)
{
2013-02-26 12:39:10 +00:00
pos = rotatePoint(pos, Vector2(mBounds.getCenter().x, mBounds.getCenter().y), mAngle);
2012-08-28 15:30:34 +00:00
2013-02-26 12:39:10 +00:00
Vector2 min(mBounds.getMinimum().x, mBounds.getMinimum().y);
2012-08-28 15:30:34 +00:00
x = std::ceil((pos.x - min.x)/sSize)-1;
y = std::ceil((pos.y - min.y)/sSize)-1;
nX = (pos.x - min.x - sSize*x)/sSize;
2013-02-26 12:39:10 +00:00
nY = 1.0-(pos.y - min.y - sSize*y)/sSize;
2012-08-28 15:30:34 +00:00
}
bool LocalMap::isPositionExplored (float nX, float nY, int x, int y, bool interior)
{
std::string texName = (interior ? mInteriorName + "_" : "Cell_") + coordStr(x, y);
if (mBuffers.find(texName) == mBuffers.end())
return false;
int texU = (sFogOfWarResolution-1) * nX;
int texV = (sFogOfWarResolution-1) * nY;
Ogre::uint32 clr = mBuffers[texName][texV * sFogOfWarResolution + texU];
uint8 alpha = (clr >> 24);
return alpha < 200;
}
void LocalMap::updatePlayer (const Ogre::Vector3& position, const Ogre::Quaternion& orientation)
2012-03-14 13:51:58 +00:00
{
2012-03-18 19:44:56 +00:00
if (sFogOfWarSkip != 0)
{
static int count=0;
if (++count % sFogOfWarSkip != 0)
return;
}
2012-03-14 16:44:19 +00:00
// retrieve the x,y grid coordinates the player is in
2012-03-14 13:51:58 +00:00
int x,y;
2012-08-28 15:30:34 +00:00
float u,v;
2013-02-26 12:39:10 +00:00
Vector2 pos(position.x, position.y);
2012-08-28 15:30:34 +00:00
if (mInterior)
getInteriorMapPosition(pos, u,v, x,y);
2012-03-29 17:20:09 +00:00
2013-02-26 12:39:10 +00:00
Vector3 playerdirection = mCameraRotNode->convertWorldToLocalOrientation(orientation).yAxis();
2012-03-29 17:20:09 +00:00
2013-02-26 12:39:10 +00:00
Vector2 min(mBounds.getMinimum().x, mBounds.getMinimum().y);
2012-04-02 17:37:24 +00:00
2012-03-14 13:51:58 +00:00
if (!mInterior)
{
2012-03-18 19:44:56 +00:00
x = std::ceil(pos.x / sSize)-1;
2013-02-26 12:39:10 +00:00
y = std::ceil(pos.y / sSize)-1;
mCellX = x;
mCellY = y;
2012-03-14 13:51:58 +00:00
}
else
{
MWBase::Environment::get().getWindowManager()->setInteriorMapTexture(x,y);
2012-03-14 13:51:58 +00:00
}
// convert from world coordinates to texture UV coordinates
2012-05-26 22:39:33 +00:00
std::string texBaseName;
2012-03-14 13:51:58 +00:00
if (!mInterior)
{
2012-03-18 19:44:56 +00:00
u = std::abs((pos.x - (sSize*x))/sSize);
2013-02-26 12:39:10 +00:00
v = 1.0-std::abs((pos.y - (sSize*y))/sSize);
2012-05-26 22:39:33 +00:00
texBaseName = "Cell_";
2012-03-14 13:51:58 +00:00
}
else
{
2012-05-26 22:39:33 +00:00
texBaseName = mInteriorName + "_";
2012-03-14 13:51:58 +00:00
}
MWBase::Environment::get().getWindowManager()->setPlayerPos(u, v);
2013-02-26 12:39:10 +00:00
MWBase::Environment::get().getWindowManager()->setPlayerDir(playerdirection.x, playerdirection.y);
2012-03-14 13:51:58 +00:00
// explore radius (squared)
2012-05-28 03:03:50 +00:00
const float sqrExploreRadius = (mInterior ? 0.01 : 0.09) * sFogOfWarResolution*sFogOfWarResolution;
const float exploreRadius = (mInterior ? 0.1 : 0.3) * sFogOfWarResolution; // explore radius from 0 to sFogOfWarResolution
2012-05-26 22:39:33 +00:00
const float exploreRadiusUV = exploreRadius / sFogOfWarResolution; // explore radius from 0 to 1 (UV space)
2012-03-14 13:51:58 +00:00
2012-05-26 22:39:33 +00:00
// change the affected fog of war textures (in a 3x3 grid around the player)
for (int mx = -1; mx<2; ++mx)
2012-03-14 13:51:58 +00:00
{
2012-05-26 22:39:33 +00:00
for (int my = -1; my<2; ++my)
2012-03-14 13:51:58 +00:00
{
2012-05-26 22:39:33 +00:00
// is this texture affected at all?
bool affected = false;
if (mx == 0 && my == 0) // the player is always in the center of the 3x3 grid
affected = true;
else
2012-03-14 16:44:19 +00:00
{
2012-05-26 22:39:33 +00:00
bool affectsX = (mx > 0)? (u + exploreRadiusUV > 1) : (u - exploreRadiusUV < 0);
bool affectsY = (my > 0)? (v + exploreRadiusUV > 1) : (v - exploreRadiusUV < 0);
affected = (affectsX && (my == 0)) || (affectsY && mx == 0) || (affectsX && affectsY);
}
if (!affected)
continue;
2012-03-14 16:44:19 +00:00
2013-02-26 12:39:10 +00:00
std::string texName = texBaseName + coordStr(x+mx,y+my*-1);
2012-05-26 22:39:33 +00:00
TexturePtr tex = TextureManager::getSingleton().getByName(texName+"_fog");
if (!tex.isNull())
{
// get its buffer
if (mBuffers.find(texName) == mBuffers.end()) return;
int i=0;
for (int texV = 0; texV<sFogOfWarResolution; ++texV)
{
for (int texU = 0; texU<sFogOfWarResolution; ++texU)
{
2012-05-28 03:03:50 +00:00
// fix into range of 0 ... sFogOfWarResolution
int _texU = texU * (float(sFogOfWarResolution+1) / float(sFogOfWarResolution));
int _texV = texV * (float(sFogOfWarResolution+1) / float(sFogOfWarResolution));
float sqrDist = Math::Sqr((_texU + mx*sFogOfWarResolution) - u*sFogOfWarResolution) + Math::Sqr((_texV + my*sFogOfWarResolution) - v*sFogOfWarResolution);
2012-05-26 22:39:33 +00:00
uint32 clr = mBuffers[texName][i];
uint8 alpha = (clr >> 24);
alpha = std::min( alpha, (uint8) (std::max(0.f, std::min(1.f, (sqrDist/sqrExploreRadius)))*255) );
mBuffers[texName][i] = (uint32) (alpha << 24);
++i;
}
}
// copy to the texture
memcpy(tex->getBuffer()->lock(HardwareBuffer::HBL_DISCARD), &mBuffers[texName][0], sFogOfWarResolution*sFogOfWarResolution*4);
tex->getBuffer()->unlock();
2012-03-14 16:44:19 +00:00
}
2012-03-14 13:51:58 +00:00
}
}
}