mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-27 12:35:46 +00:00
302 lines
10 KiB
C++
302 lines
10 KiB
C++
#include "textoverlay.hpp"
|
|
|
|
#include <OgreCamera.h>
|
|
#include <OgreMaterialManager.h>
|
|
#include <OgreTechnique.h>
|
|
|
|
#include <Overlay/OgreOverlayManager.h>
|
|
#include <Overlay/OgreOverlayContainer.h>
|
|
#include <Overlay/OgreFontManager.h>
|
|
#include <Overlay/OgreTextAreaOverlayElement.h>
|
|
#include <OgreEntity.h>
|
|
#include <OgreViewport.h>
|
|
#include <OgreRoot.h>
|
|
#include "OgreHardwarePixelBuffer.h"
|
|
|
|
namespace CSVRender
|
|
{
|
|
|
|
static const int sMargin = 5; // margin around overlay text in pixels
|
|
|
|
// Things to do:
|
|
// - configurable font size in pixels (automatically calulate everything else from it)
|
|
// - configurable texture to use
|
|
// - configurable toggle key to enable/disable the text overlays (currntly fixed 'm')
|
|
// - try material script
|
|
// - setup common resources such as font in a central place, so that they are not
|
|
// repeated for each text overlay (also destroy once)
|
|
// - decide whether to use QPaint
|
|
|
|
// http://www.ogre3d.org/tikiwiki/tiki-index.php?page=ObjectTextDisplay
|
|
// http://www.ogre3d.org/tikiwiki/tiki-index.php?page=MovableTextOverlay
|
|
// http://www.ogre3d.org/tikiwiki/tiki-index.php?page=Creating+dynamic+textures
|
|
// http://www.ogre3d.org/tikiwiki/ManualObject
|
|
TextOverlay::TextOverlay(const Ogre::MovableObject* obj, const Ogre::Camera* camera, const Ogre::String& id)
|
|
: mOverlay(0), mCaption(""), mEnabled(false), mCamera(camera), mObj(obj), mId(id), mOnScreen(false)
|
|
, mFontHeight(16) // FIXME: make this configurable
|
|
{
|
|
if(id == "" || !camera || !obj)
|
|
throw std::runtime_error("TextOverlay could not be created.");
|
|
|
|
// setup font
|
|
if (Ogre::FontManager::getSingleton().resourceExists("DejaVuLGC"))
|
|
mFont = Ogre::FontManager::getSingleton().getByName("DejaVuLGC","General");
|
|
else
|
|
{
|
|
Ogre::ResourceGroupManager::getSingleton().addResourceLocation("resources\\mygui", "FileSystem");
|
|
mFont = Ogre::FontManager::getSingleton().create("DejaVuLGC","General");
|
|
mFont->setType(Ogre::FT_TRUETYPE);
|
|
mFont->setSource("DejaVuLGCSansMono.ttf");
|
|
mFont->setTrueTypeSize(mFontHeight);
|
|
mFont->setTrueTypeResolution(96);
|
|
}
|
|
if(!mFont.isNull())
|
|
mFont->load();
|
|
else
|
|
throw std::runtime_error("TextOverlay font not loaded.");
|
|
|
|
// setup overlay
|
|
Ogre::OverlayManager &overlayMgr = Ogre::OverlayManager::getSingleton();
|
|
mOverlay = overlayMgr.getByName("CellIDPanel");
|
|
if(!mOverlay)
|
|
mOverlay = overlayMgr.create("CellIDPanel");
|
|
|
|
// create texture
|
|
Ogre::TexturePtr texture = Ogre::TextureManager::getSingleton().getByName("DynamicTransBlue");
|
|
if(texture.isNull())
|
|
{
|
|
texture = Ogre::TextureManager::getSingleton().createManual(
|
|
"DynamicTransBlue", // name
|
|
Ogre::ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME,
|
|
Ogre::TEX_TYPE_2D, // type
|
|
8, 8, // width & height
|
|
0, // number of mipmaps
|
|
Ogre::PF_BYTE_BGRA, // pixel format
|
|
Ogre::TU_DEFAULT); // usage; should be TU_DYNAMIC_WRITE_ONLY_DISCARDABLE for
|
|
// textures updated very often (e.g. each frame)
|
|
|
|
Ogre::HardwarePixelBufferSharedPtr pixelBuffer = texture->getBuffer();
|
|
pixelBuffer->lock(Ogre::HardwareBuffer::HBL_NORMAL);
|
|
const Ogre::PixelBox& pixelBox = pixelBuffer->getCurrentLock();
|
|
|
|
uint8_t* pDest = static_cast<uint8_t*>(pixelBox.data);
|
|
|
|
// Fill in some pixel data. This will give a semi-transparent blue,
|
|
// but this is of course dependent on the chosen pixel format.
|
|
for (size_t j = 0; j < 8; j++)
|
|
{
|
|
for(size_t i = 0; i < 8; i++)
|
|
{
|
|
*pDest++ = 255; // B
|
|
*pDest++ = 0; // G
|
|
*pDest++ = 0; // R
|
|
*pDest++ = 63; // A
|
|
}
|
|
|
|
pDest += pixelBox.getRowSkip() * Ogre::PixelUtil::getNumElemBytes(pixelBox.format);
|
|
}
|
|
pixelBuffer->unlock();
|
|
}
|
|
|
|
// setup material for containers
|
|
Ogre::MaterialPtr mQuadMaterial = Ogre::MaterialManager::getSingleton().create("TransOverlayMaterial",
|
|
Ogre::ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME, true );
|
|
Ogre::Pass *pass = mQuadMaterial->getTechnique( 0 )->getPass( 0 );
|
|
pass->setLightingEnabled( false );
|
|
pass->setDepthWriteEnabled( false );
|
|
pass->setSceneBlending( Ogre::SBT_TRANSPARENT_ALPHA );
|
|
|
|
Ogre::TextureUnitState *tex = pass->createTextureUnitState("MyCustomState", 0);
|
|
tex->setTextureName("DynamicTransBlue");
|
|
tex->setTextureFiltering( Ogre::TFO_ANISOTROPIC );
|
|
mQuadMaterial->load();
|
|
|
|
mContainer = static_cast<Ogre::OverlayContainer*>(overlayMgr.createOverlayElement("Panel", "container"+mId));
|
|
mContainer->setMaterialName("TransOverlayMaterial");
|
|
mOverlay->add2D(mContainer);
|
|
|
|
// setup text area overlay element
|
|
mElement = static_cast<Ogre::TextAreaOverlayElement*>(overlayMgr.createOverlayElement("TextArea", "text"+mId));
|
|
mElement->setMetricsMode(Ogre::GMM_RELATIVE);
|
|
mElement->setDimensions(1.0, 1.0);
|
|
mElement->setMetricsMode(Ogre::GMM_PIXELS);
|
|
mElement->setPosition(sMargin*2, sMargin*1.3); // 1.3 & 2 = fudge factor
|
|
|
|
mElement->setFontName("DejaVuLGC");
|
|
mElement->setCharHeight(fontHeight()); // NOTE: seems that this is required as well as font->setTrueTypeSize()
|
|
mElement->setHorizontalAlignment(Ogre::GHA_LEFT);
|
|
//mElement->setColour(Ogre::ColourValue(1.0, 1.0, 1.0)); // R, G, B
|
|
mElement->setColour(Ogre::ColourValue(1.0, 1.0, 0)); // yellow
|
|
|
|
mContainer->addChild(mElement);
|
|
mOverlay->show();
|
|
}
|
|
|
|
void TextOverlay::getScreenCoordinates(const Ogre::Vector3& position, Ogre::Real& x, Ogre::Real& y)
|
|
{
|
|
Ogre::Vector3 hcsPosition = mCamera->getProjectionMatrix() * (mCamera->getViewMatrix() * position);
|
|
|
|
x = 1.0f - ((hcsPosition.x * 0.5f) + 0.5f); // 0 <= x <= 1 // left := 0,right := 1
|
|
y = ((hcsPosition.y * 0.5f) + 0.5f); // 0 <= y <= 1 // bottom := 0,top := 1
|
|
}
|
|
|
|
void TextOverlay::getMinMaxEdgesOfTopAABBIn2D(float& MinX, float& MinY, float& MaxX, float& MaxY)
|
|
{
|
|
MinX = 0, MinY = 0, MaxX = 0, MaxY = 0;
|
|
float X[4]; // the 2D dots of the AABB in screencoordinates
|
|
float Y[4];
|
|
|
|
if (!mObj->isInScene())
|
|
return;
|
|
|
|
const Ogre::AxisAlignedBox &AABB = mObj->getWorldBoundingBox(true); // the AABB of the target
|
|
const Ogre::Vector3 CornersOfTopAABB[4] = { AABB.getCorner(Ogre::AxisAlignedBox::FAR_LEFT_TOP),
|
|
AABB.getCorner(Ogre::AxisAlignedBox::FAR_RIGHT_TOP),
|
|
AABB.getCorner(Ogre::AxisAlignedBox::NEAR_LEFT_TOP),
|
|
AABB.getCorner(Ogre::AxisAlignedBox::NEAR_RIGHT_TOP)};
|
|
|
|
//The normal vector of the plaine.this points directly infront of the cam
|
|
Ogre::Vector3 CameraPlainNormal = mCamera->getDerivedOrientation().zAxis();
|
|
|
|
//the plaine that devides the space bevor and behin the cam
|
|
Ogre::Plane CameraPlain = Ogre::Plane(CameraPlainNormal, mCamera->getDerivedPosition());
|
|
|
|
for (int i = 0; i < 4; i++)
|
|
{
|
|
X[i] = 0;
|
|
Y[i] = 0;
|
|
|
|
getScreenCoordinates(CornersOfTopAABB[i],X[i],Y[i]); // transfor into 2d dots
|
|
|
|
if (CameraPlain.getSide(CornersOfTopAABB[i]) == Ogre::Plane::NEGATIVE_SIDE)
|
|
{
|
|
if (i == 0) // accept the first set of values, no matter how bad it might be.
|
|
{
|
|
MinX = X[i];
|
|
MinY = Y[i];
|
|
MaxX = X[i];
|
|
MaxY = Y[i];
|
|
}
|
|
else // now compare if you get "better" values
|
|
{
|
|
if (MinX > X[i]) MinX = X[i];
|
|
if (MinY > Y[i]) MinY = Y[i];
|
|
if (MaxX < X[i]) MaxX = X[i];
|
|
if (MaxY < Y[i]) MaxY = Y[i];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
MinX = 0;
|
|
MinY = 0;
|
|
MaxX = 0;
|
|
MaxY = 0;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
TextOverlay::~TextOverlay()
|
|
{
|
|
Ogre::OverlayManager::OverlayMapIterator iter = Ogre::OverlayManager::getSingleton().getOverlayIterator();
|
|
if(!iter.hasMoreElements())
|
|
mOverlay->hide();
|
|
|
|
Ogre::OverlayManager *overlayMgr = Ogre::OverlayManager::getSingletonPtr();
|
|
mContainer->removeChild("text"+mId);
|
|
mOverlay->remove2D(mContainer);
|
|
overlayMgr->destroyOverlayElement(mElement);
|
|
overlayMgr->destroyOverlayElement(mContainer);
|
|
|
|
if(!iter.hasMoreElements())
|
|
overlayMgr->destroy(mOverlay);
|
|
}
|
|
|
|
void TextOverlay::enable(bool enable)
|
|
{
|
|
mEnabled = enable;
|
|
if (enable)
|
|
mOverlay->show();
|
|
else
|
|
mOverlay->hide();
|
|
}
|
|
|
|
void TextOverlay::setCaption(const Ogre::String& text)
|
|
{
|
|
if(mCaption == text)
|
|
return;
|
|
|
|
mCaption = text;
|
|
mElement->setCaption(text);
|
|
}
|
|
|
|
Ogre::FontPtr TextOverlay::getFont()
|
|
{
|
|
return mFont;
|
|
}
|
|
|
|
int TextOverlay::textWidth()
|
|
{
|
|
float textWidth = 0;
|
|
|
|
// NOTE: assumed fixed width font, i.e. no compensation for narrow glyphs
|
|
for(Ogre::String::const_iterator i = mCaption.begin(); i < mCaption.end(); ++i)
|
|
textWidth += getFont()->getGlyphAspectRatio(*i);
|
|
|
|
textWidth *= fontHeight();
|
|
|
|
return (int) textWidth;
|
|
}
|
|
|
|
int TextOverlay::fontHeight()
|
|
{
|
|
return mFontHeight;
|
|
}
|
|
|
|
void TextOverlay::update(bool toggleOverlay)
|
|
{
|
|
if(toggleOverlay)
|
|
mEnabled = !mEnabled;
|
|
|
|
if (!mEnabled)
|
|
{
|
|
mOverlay->hide();
|
|
Ogre::Root::getSingleton().renderOneFrame();
|
|
return;
|
|
}
|
|
else
|
|
mOverlay->show();
|
|
|
|
float min_x, max_x, min_y, max_y;
|
|
getMinMaxEdgesOfTopAABBIn2D(min_x, min_y, max_x, max_y);
|
|
|
|
if ((min_x>0.0) && (max_x<1.0) && (min_y>0.0) && (max_y<1.0))
|
|
{
|
|
mOnScreen = true;
|
|
mContainer->show();
|
|
}
|
|
else
|
|
{
|
|
mOnScreen = false;
|
|
mContainer->hide();
|
|
Ogre::Root::getSingleton().renderOneFrame();
|
|
return;
|
|
}
|
|
|
|
Ogre::OverlayManager &overlayMgr = Ogre::OverlayManager::getSingleton();
|
|
float viewportWidth = std::max(overlayMgr.getViewportWidth(), 1); // zero at the start
|
|
float viewportHeight = std::max(overlayMgr.getViewportHeight(), 1); // zero at the start
|
|
|
|
float relTextWidth = (2*sMargin + textWidth() + sMargin) / viewportWidth; // 2 = fudge factor
|
|
float relTextHeight = (sMargin + fontHeight() + sMargin) / viewportHeight;
|
|
|
|
mContainer->setMetricsMode(Ogre::GMM_RELATIVE);
|
|
mContainer->setPosition(1-(min_x + max_x + relTextWidth)/2, 1-max_y-(relTextHeight-sMargin/viewportHeight));
|
|
|
|
mContainer->setDimensions(relTextWidth, relTextHeight);
|
|
|
|
Ogre::Root::getSingleton().renderOneFrame();
|
|
}
|
|
|
|
}
|