2012-09-20 11:56:37 +00:00
|
|
|
#include "globalmap.hpp"
|
|
|
|
|
2012-09-20 16:23:27 +00:00
|
|
|
#include <boost/filesystem.hpp>
|
2012-11-16 18:34:09 +00:00
|
|
|
#include <boost/lexical_cast.hpp>
|
2012-09-20 16:23:27 +00:00
|
|
|
|
2012-09-20 11:56:37 +00:00
|
|
|
#include <OgreImage.h>
|
|
|
|
#include <OgreTextureManager.h>
|
|
|
|
#include <OgreColourValue.h>
|
|
|
|
#include <OgreHardwareVertexBuffer.h>
|
|
|
|
#include <OgreRoot.h>
|
2012-11-04 11:13:04 +00:00
|
|
|
#include <OgreHardwarePixelBuffer.h>
|
2012-09-20 11:56:37 +00:00
|
|
|
|
2013-08-27 13:48:13 +00:00
|
|
|
#include <components/loadinglistener/loadinglistener.hpp>
|
2015-01-31 22:27:34 +00:00
|
|
|
#include <components/settings/settings.hpp>
|
2013-08-27 13:48:13 +00:00
|
|
|
|
2014-01-25 17:20:17 +00:00
|
|
|
#include <components/esm/globalmap.hpp>
|
|
|
|
|
2012-09-20 11:56:37 +00:00
|
|
|
#include "../mwbase/environment.hpp"
|
|
|
|
#include "../mwbase/world.hpp"
|
|
|
|
|
2012-10-01 15:17:04 +00:00
|
|
|
#include "../mwworld/esmstore.hpp"
|
2012-09-20 11:56:37 +00:00
|
|
|
|
|
|
|
namespace MWRender
|
|
|
|
{
|
|
|
|
|
|
|
|
GlobalMap::GlobalMap(const std::string &cacheDir)
|
|
|
|
: mCacheDir(cacheDir)
|
2013-07-31 16:46:32 +00:00
|
|
|
, mWidth(0)
|
|
|
|
, mHeight(0)
|
2015-05-01 00:24:27 +00:00
|
|
|
, mMinX(0), mMaxX(0)
|
|
|
|
, mMinY(0), mMaxY(0)
|
|
|
|
|
2012-09-20 11:56:37 +00:00
|
|
|
{
|
2014-09-26 10:47:33 +00:00
|
|
|
mCellSize = Settings::Manager::getInt("global map cell size", "Map");
|
2012-09-20 11:56:37 +00:00
|
|
|
}
|
|
|
|
|
2014-08-14 17:01:03 +00:00
|
|
|
GlobalMap::~GlobalMap()
|
|
|
|
{
|
|
|
|
Ogre::TextureManager::getSingleton().remove(mOverlayTexture->getName());
|
|
|
|
}
|
2012-09-20 11:56:37 +00:00
|
|
|
|
2013-08-27 13:48:13 +00:00
|
|
|
void GlobalMap::render (Loading::Listener* loadingListener)
|
2012-09-20 11:56:37 +00:00
|
|
|
{
|
2012-09-20 16:23:27 +00:00
|
|
|
Ogre::TexturePtr tex;
|
2012-09-20 11:56:37 +00:00
|
|
|
|
2012-11-06 07:53:00 +00:00
|
|
|
const MWWorld::ESMStore &esmStore =
|
|
|
|
MWBase::Environment::get().getWorld()->getStore();
|
|
|
|
|
2012-09-21 14:26:04 +00:00
|
|
|
// get the size of the world
|
2012-11-06 07:53:00 +00:00
|
|
|
MWWorld::Store<ESM::Cell>::iterator it = esmStore.get<ESM::Cell>().extBegin();
|
2012-11-06 09:14:03 +00:00
|
|
|
for (; it != esmStore.get<ESM::Cell>().extEnd(); ++it)
|
2012-09-20 11:56:37 +00:00
|
|
|
{
|
2012-11-06 07:53:00 +00:00
|
|
|
if (it->getGridX() < mMinX)
|
|
|
|
mMinX = it->getGridX();
|
|
|
|
if (it->getGridX() > mMaxX)
|
|
|
|
mMaxX = it->getGridX();
|
|
|
|
if (it->getGridY() < mMinY)
|
|
|
|
mMinY = it->getGridY();
|
|
|
|
if (it->getGridY() > mMaxY)
|
|
|
|
mMaxY = it->getGridY();
|
2012-09-21 14:26:04 +00:00
|
|
|
}
|
2012-09-20 11:56:37 +00:00
|
|
|
|
2014-09-26 10:47:33 +00:00
|
|
|
mWidth = mCellSize*(mMaxX-mMinX+1);
|
|
|
|
mHeight = mCellSize*(mMaxY-mMinY+1);
|
2012-09-20 11:56:37 +00:00
|
|
|
|
2013-08-27 13:48:13 +00:00
|
|
|
loadingListener->loadingOn();
|
|
|
|
loadingListener->setLabel("Creating map");
|
|
|
|
loadingListener->setProgressRange((mMaxX-mMinX+1) * (mMaxY-mMinY+1));
|
|
|
|
loadingListener->setProgress(0);
|
|
|
|
|
2015-01-29 01:23:49 +00:00
|
|
|
std::vector<Ogre::uchar> data (mWidth * mHeight * 3);
|
2012-09-20 11:56:37 +00:00
|
|
|
|
2015-01-29 01:23:49 +00:00
|
|
|
for (int x = mMinX; x <= mMaxX; ++x)
|
|
|
|
{
|
|
|
|
for (int y = mMinY; y <= mMaxY; ++y)
|
2012-09-20 11:56:37 +00:00
|
|
|
{
|
2015-01-29 01:23:49 +00:00
|
|
|
ESM::Land* land = esmStore.get<ESM::Land>().search (x,y);
|
2012-09-20 11:56:37 +00:00
|
|
|
|
2015-01-29 01:23:49 +00:00
|
|
|
if (land)
|
|
|
|
{
|
2015-01-29 02:30:07 +00:00
|
|
|
int mask = ESM::Land::DATA_WNAM;
|
2015-01-29 01:23:49 +00:00
|
|
|
if (!land->isDataLoaded(mask))
|
|
|
|
land->loadData(mask);
|
|
|
|
}
|
2012-09-20 11:56:37 +00:00
|
|
|
|
2015-01-29 01:23:49 +00:00
|
|
|
for (int cellY=0; cellY<mCellSize; ++cellY)
|
|
|
|
{
|
|
|
|
for (int cellX=0; cellX<mCellSize; ++cellX)
|
2012-09-20 11:56:37 +00:00
|
|
|
{
|
2015-03-08 00:07:29 +00:00
|
|
|
int vertexX = static_cast<int>(float(cellX)/float(mCellSize) * 9);
|
|
|
|
int vertexY = static_cast<int>(float(cellY) / float(mCellSize) * 9);
|
2012-09-20 11:56:37 +00:00
|
|
|
|
|
|
|
|
2015-01-29 01:23:49 +00:00
|
|
|
int texelX = (x-mMinX) * mCellSize + cellX;
|
|
|
|
int texelY = (mHeight-1) - ((y-mMinY) * mCellSize + cellY);
|
2012-09-20 11:56:37 +00:00
|
|
|
|
2015-01-29 01:23:49 +00:00
|
|
|
unsigned char r,g,b;
|
2012-09-20 11:56:37 +00:00
|
|
|
|
2015-01-29 01:23:49 +00:00
|
|
|
float y = 0;
|
|
|
|
if (land && land->mDataTypes & ESM::Land::DATA_WNAM)
|
|
|
|
y = (land->mLandData->mWnam[vertexY * 9 + vertexX] << 4) / 2048.f;
|
|
|
|
else
|
|
|
|
y = (SCHAR_MIN << 4) / 2048.f;
|
|
|
|
if (y < 0)
|
|
|
|
{
|
2015-03-08 00:07:29 +00:00
|
|
|
r = static_cast<unsigned char>(14 * y + 38);
|
|
|
|
g = static_cast<unsigned char>(20 * y + 56);
|
|
|
|
b = static_cast<unsigned char>(18 * y + 51);
|
2015-01-29 01:23:49 +00:00
|
|
|
}
|
|
|
|
else if (y < 0.3f)
|
|
|
|
{
|
|
|
|
if (y < 0.1f)
|
|
|
|
y *= 8.f;
|
2012-09-20 11:56:37 +00:00
|
|
|
else
|
|
|
|
{
|
2015-03-08 00:07:29 +00:00
|
|
|
y -= 0.1f;
|
|
|
|
y += 0.8f;
|
2012-09-20 11:56:37 +00:00
|
|
|
}
|
2015-03-08 00:07:29 +00:00
|
|
|
r = static_cast<unsigned char>(66 - 32 * y);
|
|
|
|
g = static_cast<unsigned char>(48 - 23 * y);
|
|
|
|
b = static_cast<unsigned char>(33 - 16 * y);
|
2012-09-20 11:56:37 +00:00
|
|
|
}
|
2015-01-29 01:23:49 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
y -= 0.3f;
|
|
|
|
y *= 1.428f;
|
2015-03-08 00:07:29 +00:00
|
|
|
r = static_cast<unsigned char>(34 - 29 * y);
|
|
|
|
g = static_cast<unsigned char>(25 - 20 * y);
|
|
|
|
b = static_cast<unsigned char>(17 - 12 * y);
|
2015-01-29 01:23:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
data[texelY * mWidth * 3 + texelX * 3] = r;
|
|
|
|
data[texelY * mWidth * 3 + texelX * 3+1] = g;
|
|
|
|
data[texelY * mWidth * 3 + texelX * 3+2] = b;
|
2012-09-20 11:56:37 +00:00
|
|
|
}
|
|
|
|
}
|
2015-01-29 01:23:49 +00:00
|
|
|
loadingListener->increaseProgress();
|
2015-01-29 02:30:07 +00:00
|
|
|
if (land)
|
|
|
|
land->unloadData();
|
2012-09-20 11:56:37 +00:00
|
|
|
}
|
2015-01-29 01:23:49 +00:00
|
|
|
}
|
2012-09-20 11:56:37 +00:00
|
|
|
|
2015-01-29 01:23:49 +00:00
|
|
|
Ogre::DataStreamPtr stream(new Ogre::MemoryDataStream(&data[0], data.size()));
|
2012-09-20 11:56:37 +00:00
|
|
|
|
2015-01-29 01:23:49 +00:00
|
|
|
tex = Ogre::TextureManager::getSingleton ().createManual ("GlobalMap.png", Ogre::ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME,
|
|
|
|
Ogre::TEX_TYPE_2D, mWidth, mHeight, 0, Ogre::PF_B8G8R8, Ogre::TU_STATIC);
|
|
|
|
tex->loadRawData(stream, mWidth, mHeight, Ogre::PF_B8G8R8);
|
2012-09-20 16:23:27 +00:00
|
|
|
|
2012-09-20 11:56:37 +00:00
|
|
|
tex->load();
|
2012-11-16 18:34:09 +00:00
|
|
|
|
|
|
|
mOverlayTexture = Ogre::TextureManager::getSingleton().createManual("GlobalMapOverlay", Ogre::ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME,
|
2014-08-14 17:01:03 +00:00
|
|
|
Ogre::TEX_TYPE_2D, mWidth, mHeight, 0, Ogre::PF_A8B8G8R8, Ogre::TU_DYNAMIC, this);
|
2012-11-16 18:34:09 +00:00
|
|
|
|
2014-01-25 12:34:56 +00:00
|
|
|
clear();
|
2013-08-27 13:48:13 +00:00
|
|
|
|
|
|
|
loadingListener->loadingOff();
|
2012-09-20 11:56:37 +00:00
|
|
|
}
|
|
|
|
|
2012-09-21 14:26:04 +00:00
|
|
|
void GlobalMap::worldPosToImageSpace(float x, float z, float& imageX, float& imageY)
|
|
|
|
{
|
|
|
|
imageX = float(x / 8192.f - mMinX) / (mMaxX - mMinX + 1);
|
|
|
|
|
2013-02-26 12:52:01 +00:00
|
|
|
imageY = 1.f-float(z / 8192.f - mMinY) / (mMaxY - mMinY + 1);
|
2012-09-21 14:26:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GlobalMap::cellTopLeftCornerToImageSpace(int x, int y, float& imageX, float& imageY)
|
|
|
|
{
|
|
|
|
imageX = float(x - mMinX) / (mMaxX - mMinX + 1);
|
|
|
|
|
|
|
|
// NB y + 1, because we want the top left corner, not bottom left where the origin of the cell is
|
|
|
|
imageY = 1.f-float(y - mMinY + 1) / (mMaxY - mMinY + 1);
|
|
|
|
}
|
|
|
|
|
2012-11-04 11:13:04 +00:00
|
|
|
void GlobalMap::exploreCell(int cellX, int cellY)
|
|
|
|
{
|
2015-03-08 00:07:29 +00:00
|
|
|
float originX = static_cast<float>((cellX - mMinX) * mCellSize);
|
2012-11-16 18:34:09 +00:00
|
|
|
// NB y + 1, because we want the top left corner, not bottom left where the origin of the cell is
|
2015-03-08 00:07:29 +00:00
|
|
|
float originY = static_cast<float>(mHeight - (cellY + 1 - mMinY) * mCellSize);
|
2012-11-04 11:13:04 +00:00
|
|
|
|
2012-11-16 18:34:09 +00:00
|
|
|
if (cellX > mMaxX || cellX < mMinX || cellY > mMaxY || cellY < mMinY)
|
|
|
|
return;
|
2012-09-21 14:26:04 +00:00
|
|
|
|
2012-11-16 18:34:09 +00:00
|
|
|
Ogre::TexturePtr localMapTexture = Ogre::TextureManager::getSingleton().getByName("Cell_"
|
|
|
|
+ boost::lexical_cast<std::string>(cellX) + "_" + boost::lexical_cast<std::string>(cellY));
|
2012-11-04 11:13:04 +00:00
|
|
|
|
2012-11-16 18:34:09 +00:00
|
|
|
if (!localMapTexture.isNull())
|
2012-11-16 21:26:00 +00:00
|
|
|
{
|
2014-12-23 00:54:37 +00:00
|
|
|
int mapWidth = localMapTexture->getWidth();
|
|
|
|
int mapHeight = localMapTexture->getHeight();
|
2014-08-14 17:01:03 +00:00
|
|
|
mOverlayTexture->load();
|
2014-12-23 00:54:37 +00:00
|
|
|
mOverlayTexture->getBuffer()->blit(localMapTexture->getBuffer(), Ogre::Image::Box(0,0,mapWidth,mapHeight),
|
2015-03-08 00:07:29 +00:00
|
|
|
Ogre::Image::Box(static_cast<Ogre::uint32>(originX), static_cast<Ogre::uint32>(originY),
|
|
|
|
static_cast<Ogre::uint32>(originX + mCellSize), static_cast<Ogre::uint32>(originY + mCellSize)));
|
2014-08-14 17:01:03 +00:00
|
|
|
|
|
|
|
Ogre::Image backup;
|
|
|
|
std::vector<Ogre::uchar> data;
|
2014-09-26 10:47:33 +00:00
|
|
|
data.resize(mCellSize*mCellSize*4, 0);
|
|
|
|
backup.loadDynamicImage(&data[0], mCellSize, mCellSize, Ogre::PF_A8B8G8R8);
|
2014-08-14 17:01:03 +00:00
|
|
|
|
2014-12-23 00:54:37 +00:00
|
|
|
localMapTexture->getBuffer()->blitToMemory(Ogre::Image::Box(0,0,mapWidth,mapHeight), backup.getPixelBox());
|
2014-08-14 17:01:03 +00:00
|
|
|
|
2014-09-26 10:47:33 +00:00
|
|
|
for (int x=0; x<mCellSize; ++x)
|
|
|
|
for (int y=0; y<mCellSize; ++y)
|
2014-08-14 17:01:03 +00:00
|
|
|
{
|
|
|
|
assert (originX+x < mOverlayImage.getWidth());
|
|
|
|
assert (originY+y < mOverlayImage.getHeight());
|
|
|
|
assert (x < int(backup.getWidth()));
|
|
|
|
assert (y < int(backup.getHeight()));
|
2015-03-08 00:07:29 +00:00
|
|
|
mOverlayImage.setColourAt(backup.getColourAt(x, y, 0), static_cast<size_t>(originX + x), static_cast<size_t>(originY + y), 0);
|
2014-08-14 17:01:03 +00:00
|
|
|
}
|
2012-11-16 21:26:00 +00:00
|
|
|
}
|
2012-11-04 11:13:04 +00:00
|
|
|
}
|
2014-01-25 12:34:56 +00:00
|
|
|
|
|
|
|
void GlobalMap::clear()
|
|
|
|
{
|
2014-08-14 17:01:03 +00:00
|
|
|
Ogre::uchar* buffer = OGRE_ALLOC_T(Ogre::uchar, mWidth * mHeight * 4, Ogre::MEMCATEGORY_GENERAL);
|
|
|
|
memset(buffer, 0, mWidth * mHeight * 4);
|
2014-01-25 12:34:56 +00:00
|
|
|
|
2014-08-14 17:01:03 +00:00
|
|
|
mOverlayImage.loadDynamicImage(&buffer[0], mWidth, mHeight, 1, Ogre::PF_A8B8G8R8, true); // pass ownership of buffer to image
|
2014-01-25 12:34:56 +00:00
|
|
|
|
2014-08-14 17:01:03 +00:00
|
|
|
mOverlayTexture->load();
|
|
|
|
}
|
|
|
|
|
|
|
|
void GlobalMap::loadResource(Ogre::Resource *resource)
|
|
|
|
{
|
2015-01-12 18:57:54 +00:00
|
|
|
Ogre::Texture* tex = static_cast<Ogre::Texture*>(resource);
|
2014-08-14 17:01:03 +00:00
|
|
|
Ogre::ConstImagePtrList list;
|
|
|
|
list.push_back(&mOverlayImage);
|
|
|
|
tex->_loadImages(list);
|
2014-01-25 12:34:56 +00:00
|
|
|
}
|
2014-01-25 17:20:17 +00:00
|
|
|
|
2014-04-26 11:42:32 +00:00
|
|
|
void GlobalMap::write(ESM::GlobalMap& map)
|
2014-01-25 17:20:17 +00:00
|
|
|
{
|
|
|
|
map.mBounds.mMinX = mMinX;
|
|
|
|
map.mBounds.mMaxX = mMaxX;
|
|
|
|
map.mBounds.mMinY = mMinY;
|
|
|
|
map.mBounds.mMaxY = mMaxY;
|
|
|
|
|
2014-08-14 17:01:03 +00:00
|
|
|
Ogre::DataStreamPtr encoded = mOverlayImage.encode("png");
|
2014-01-25 17:20:17 +00:00
|
|
|
map.mImageData.resize(encoded->size());
|
|
|
|
encoded->read(&map.mImageData[0], encoded->size());
|
|
|
|
}
|
|
|
|
|
2014-04-26 11:42:32 +00:00
|
|
|
void GlobalMap::read(ESM::GlobalMap& map)
|
2014-01-25 17:20:17 +00:00
|
|
|
{
|
2014-04-26 11:42:32 +00:00
|
|
|
const ESM::GlobalMap::Bounds& bounds = map.mBounds;
|
2014-01-26 14:18:31 +00:00
|
|
|
|
2015-01-29 01:23:49 +00:00
|
|
|
if (bounds.mMaxX-bounds.mMinX < 0)
|
2014-04-26 11:42:32 +00:00
|
|
|
return;
|
2015-01-29 01:23:49 +00:00
|
|
|
if (bounds.mMaxY-bounds.mMinY < 0)
|
2014-04-26 11:42:32 +00:00
|
|
|
return;
|
2014-01-25 17:20:17 +00:00
|
|
|
|
2014-04-26 11:42:32 +00:00
|
|
|
if (bounds.mMinX > bounds.mMaxX
|
|
|
|
|| bounds.mMinY > bounds.mMaxY)
|
|
|
|
throw std::runtime_error("invalid map bounds");
|
2014-01-25 17:20:17 +00:00
|
|
|
|
2014-04-26 11:42:32 +00:00
|
|
|
Ogre::Image image;
|
|
|
|
Ogre::DataStreamPtr stream(new Ogre::MemoryDataStream(&map.mImageData[0], map.mImageData.size()));
|
|
|
|
image.load(stream, "png");
|
|
|
|
|
|
|
|
int xLength = (bounds.mMaxX-bounds.mMinX+1);
|
|
|
|
int yLength = (bounds.mMaxY-bounds.mMinY+1);
|
|
|
|
|
|
|
|
// Size of one cell in image space
|
|
|
|
int cellImageSizeSrc = image.getWidth() / xLength;
|
|
|
|
if (int(image.getHeight() / yLength) != cellImageSizeSrc)
|
|
|
|
throw std::runtime_error("cell size must be quadratic");
|
|
|
|
|
|
|
|
// If cell bounds of the currently loaded content and the loaded savegame do not match,
|
|
|
|
// we need to resize source/dest boxes to accommodate
|
|
|
|
// This means nonexisting cells will be dropped silently
|
2014-09-26 10:47:33 +00:00
|
|
|
int cellImageSizeDst = mCellSize;
|
2014-04-26 11:42:32 +00:00
|
|
|
|
|
|
|
// Completely off-screen? -> no need to blit anything
|
|
|
|
if (bounds.mMaxX < mMinX
|
|
|
|
|| bounds.mMaxY < mMinY
|
|
|
|
|| bounds.mMinX > mMaxX
|
|
|
|
|| bounds.mMinY > mMaxY)
|
|
|
|
return;
|
2014-01-25 17:20:17 +00:00
|
|
|
|
2014-04-26 11:42:32 +00:00
|
|
|
int leftDiff = (mMinX - bounds.mMinX);
|
|
|
|
int topDiff = (bounds.mMaxY - mMaxY);
|
|
|
|
int rightDiff = (bounds.mMaxX - mMaxX);
|
|
|
|
int bottomDiff = (mMinY - bounds.mMinY);
|
|
|
|
Ogre::Image::Box srcBox ( std::max(0, leftDiff * cellImageSizeSrc),
|
|
|
|
std::max(0, topDiff * cellImageSizeSrc),
|
|
|
|
std::min(image.getWidth(), image.getWidth() - rightDiff * cellImageSizeSrc),
|
|
|
|
std::min(image.getHeight(), image.getHeight() - bottomDiff * cellImageSizeSrc));
|
|
|
|
|
|
|
|
Ogre::Image::Box destBox ( std::max(0, -leftDiff * cellImageSizeDst),
|
|
|
|
std::max(0, -topDiff * cellImageSizeDst),
|
|
|
|
std::min(mOverlayTexture->getWidth(), mOverlayTexture->getWidth() + rightDiff * cellImageSizeDst),
|
|
|
|
std::min(mOverlayTexture->getHeight(), mOverlayTexture->getHeight() + bottomDiff * cellImageSizeDst));
|
|
|
|
|
|
|
|
// Looks like there is no interface for blitting from memory with src/dst boxes.
|
|
|
|
// So we create a temporary texture for blitting.
|
|
|
|
Ogre::TexturePtr tex = Ogre::TextureManager::getSingleton().createManual("@temp",
|
|
|
|
Ogre::ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME, Ogre::TEX_TYPE_2D, image.getWidth(),
|
|
|
|
image.getHeight(), 0, Ogre::PF_A8B8G8R8);
|
|
|
|
tex->loadImage(image);
|
|
|
|
|
2014-08-14 17:01:03 +00:00
|
|
|
mOverlayTexture->load();
|
2014-04-26 11:42:32 +00:00
|
|
|
mOverlayTexture->getBuffer()->blit(tex->getBuffer(), srcBox, destBox);
|
|
|
|
|
2014-08-14 17:01:03 +00:00
|
|
|
if (srcBox.left == destBox.left && srcBox.right == destBox.right
|
2014-09-25 13:28:02 +00:00
|
|
|
&& srcBox.top == destBox.top && srcBox.bottom == destBox.bottom
|
|
|
|
&& int(image.getWidth()) == mWidth && int(image.getHeight()) == mHeight)
|
2014-08-14 17:01:03 +00:00
|
|
|
mOverlayImage = image;
|
|
|
|
else
|
|
|
|
mOverlayTexture->convertToImage(mOverlayImage);
|
|
|
|
|
2014-04-26 11:42:32 +00:00
|
|
|
Ogre::TextureManager::getSingleton().remove("@temp");
|
2014-01-25 17:20:17 +00:00
|
|
|
}
|
2012-09-20 11:56:37 +00:00
|
|
|
}
|