mirror of
https://github.com/TES3MP/openmw-tes3mp.git
synced 2025-01-15 16:49:55 +00:00
418 lines
11 KiB
C++
418 lines
11 KiB
C++
#include "myguirendermanager.hpp"
|
|
|
|
#include <stdexcept>
|
|
|
|
#include <MyGUI_Gui.h>
|
|
#include <MyGUI_Timer.h>
|
|
|
|
#include <osg/Drawable>
|
|
#include <osg/Geode>
|
|
#include <osg/PolygonMode>
|
|
#include <osg/BlendFunc>
|
|
#include <osg/Depth>
|
|
#include <osg/TexEnv>
|
|
#include <osg/Texture2D>
|
|
|
|
#include <osgViewer/Viewer>
|
|
|
|
#include <osgGA/GUIEventHandler>
|
|
|
|
#include <components/resource/texturemanager.hpp>
|
|
|
|
#include "myguitexture.hpp"
|
|
|
|
#define MYGUI_PLATFORM_LOG_SECTION "Platform"
|
|
#define MYGUI_PLATFORM_LOG(level, text) MYGUI_LOGGING(MYGUI_PLATFORM_LOG_SECTION, level, text)
|
|
|
|
#define MYGUI_PLATFORM_EXCEPT(dest) do { \
|
|
MYGUI_PLATFORM_LOG(Critical, dest); \
|
|
MYGUI_DBG_BREAK;\
|
|
std::ostringstream stream; \
|
|
stream << dest << "\n"; \
|
|
MYGUI_BASE_EXCEPT(stream.str().c_str(), "MyGUI"); \
|
|
} while(0)
|
|
|
|
#define MYGUI_PLATFORM_ASSERT(exp, dest) do { \
|
|
if ( ! (exp) ) \
|
|
{ \
|
|
MYGUI_PLATFORM_LOG(Critical, dest); \
|
|
MYGUI_DBG_BREAK;\
|
|
std::ostringstream stream; \
|
|
stream << dest << "\n"; \
|
|
MYGUI_BASE_EXCEPT(stream.str().c_str(), "MyGUI"); \
|
|
} \
|
|
} while(0)
|
|
|
|
namespace
|
|
{
|
|
|
|
// Proxy to forward a Drawable's draw call to RenderManager::drawFrame
|
|
class Renderable : public osg::Drawable {
|
|
osgMyGUI::RenderManager *mParent;
|
|
|
|
virtual void drawImplementation(osg::RenderInfo &renderInfo) const
|
|
{ mParent->drawFrame(renderInfo); }
|
|
|
|
public:
|
|
Renderable(osgMyGUI::RenderManager *parent=nullptr) : mParent(parent) { }
|
|
Renderable(const Renderable ©, const osg::CopyOp ©op=osg::CopyOp::SHALLOW_COPY)
|
|
: osg::Drawable(copy, copyop)
|
|
, mParent(copy.mParent)
|
|
{ }
|
|
|
|
META_Object(osgMyGUI, Renderable)
|
|
};
|
|
|
|
// Proxy to forward an OSG resize event to RenderManager::setViewSize
|
|
class ResizeHandler : public osgGA::GUIEventHandler {
|
|
osgMyGUI::RenderManager *mParent;
|
|
|
|
virtual bool handle(const osgGA::GUIEventAdapter &ea, osgGA::GUIActionAdapter &aa)
|
|
{
|
|
if(ea.getEventType() == osgGA::GUIEventAdapter::RESIZE)
|
|
{
|
|
int width = ea.getWindowWidth();
|
|
int height = ea.getWindowHeight();
|
|
mParent->setViewSize(width, height);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public:
|
|
ResizeHandler(osgMyGUI::RenderManager *parent=nullptr) : mParent(parent) { }
|
|
ResizeHandler(const ResizeHandler ©, const osg::CopyOp ©op=osg::CopyOp::SHALLOW_COPY)
|
|
: osg::Object(copy, copyop), osgGA::GUIEventHandler(copy, copyop)
|
|
, mParent(copy.mParent)
|
|
{ }
|
|
|
|
META_Object(osgMyGUI, ResizeHandler)
|
|
};
|
|
|
|
}
|
|
|
|
|
|
namespace osgMyGUI
|
|
{
|
|
|
|
class OSGVertexBuffer : public MyGUI::IVertexBuffer
|
|
{
|
|
osg::ref_ptr<osg::VertexBufferObject> mBuffer;
|
|
osg::ref_ptr<osg::UByteArray> mVertexArray;
|
|
std::vector<MyGUI::Vertex> mLockedData;
|
|
|
|
size_t mNeedVertexCount;
|
|
|
|
public:
|
|
OSGVertexBuffer();
|
|
virtual ~OSGVertexBuffer();
|
|
|
|
virtual void setVertexCount(size_t count);
|
|
virtual size_t getVertexCount();
|
|
|
|
virtual MyGUI::Vertex *lock();
|
|
virtual void unlock();
|
|
|
|
/*internal:*/
|
|
void destroy();
|
|
void create();
|
|
|
|
osg::VertexBufferObject *getBuffer() const { return mBuffer.get(); }
|
|
};
|
|
|
|
OSGVertexBuffer::OSGVertexBuffer()
|
|
: mNeedVertexCount(0)
|
|
{
|
|
}
|
|
|
|
OSGVertexBuffer::~OSGVertexBuffer()
|
|
{
|
|
destroy();
|
|
}
|
|
|
|
void OSGVertexBuffer::setVertexCount(size_t count)
|
|
{
|
|
if(count == mNeedVertexCount)
|
|
return;
|
|
|
|
mNeedVertexCount = count;
|
|
destroy();
|
|
create();
|
|
}
|
|
|
|
size_t OSGVertexBuffer::getVertexCount()
|
|
{
|
|
return mNeedVertexCount;
|
|
}
|
|
|
|
MyGUI::Vertex *OSGVertexBuffer::lock()
|
|
{
|
|
MYGUI_PLATFORM_ASSERT(mBuffer.valid(), "Vertex buffer is not created");
|
|
|
|
mVertexArray->resize(mNeedVertexCount * sizeof(MyGUI::Vertex));
|
|
return (MyGUI::Vertex*)&(*mVertexArray)[0];
|
|
}
|
|
|
|
void OSGVertexBuffer::unlock()
|
|
{
|
|
mVertexArray->dirty();
|
|
mBuffer->dirty();
|
|
}
|
|
|
|
void OSGVertexBuffer::destroy()
|
|
{
|
|
mBuffer = nullptr;
|
|
mVertexArray = nullptr;
|
|
}
|
|
|
|
void OSGVertexBuffer::create()
|
|
{
|
|
MYGUI_PLATFORM_ASSERT(!mBuffer.valid(), "Vertex buffer already exist");
|
|
|
|
mVertexArray = new osg::UByteArray(mNeedVertexCount*sizeof(MyGUI::Vertex));
|
|
|
|
mBuffer = new osg::VertexBufferObject;
|
|
mBuffer->setDataVariance(osg::Object::DYNAMIC);
|
|
mBuffer->setUsage(GL_DYNAMIC_DRAW);
|
|
mBuffer->setArray(0, mVertexArray.get());
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
RenderManager::RenderManager(osgViewer::Viewer *viewer, osg::Group *sceneroot, Resource::TextureManager* textureManager)
|
|
: mViewer(viewer)
|
|
, mSceneRoot(sceneroot)
|
|
, mTextureManager(textureManager)
|
|
, mUpdate(false)
|
|
, mIsInitialise(false)
|
|
{
|
|
}
|
|
|
|
RenderManager::~RenderManager()
|
|
{
|
|
MYGUI_PLATFORM_LOG(Info, "* Shutdown: "<<getClassTypeName());
|
|
|
|
if(mGuiRoot.valid())
|
|
mSceneRoot->removeChild(mGuiRoot.get());
|
|
mGuiRoot = nullptr;
|
|
mSceneRoot = nullptr;
|
|
mViewer = nullptr;
|
|
|
|
destroyAllResources();
|
|
|
|
MYGUI_PLATFORM_LOG(Info, getClassTypeName()<<" successfully shutdown");
|
|
mIsInitialise = false;
|
|
}
|
|
|
|
|
|
void RenderManager::initialise()
|
|
{
|
|
MYGUI_PLATFORM_ASSERT(!mIsInitialise, getClassTypeName()<<" initialised twice");
|
|
MYGUI_PLATFORM_LOG(Info, "* Initialise: "<<getClassTypeName());
|
|
|
|
mVertexFormat = MyGUI::VertexColourType::ColourABGR;
|
|
|
|
mUpdate = false;
|
|
|
|
osg::ref_ptr<osg::Drawable> drawable = new Renderable(this);
|
|
drawable->setSupportsDisplayList(false);
|
|
drawable->setUseVertexBufferObjects(true);
|
|
drawable->setDataVariance(osg::Object::DYNAMIC);
|
|
|
|
osg::ref_ptr<osg::Geode> geode = new osg::Geode;
|
|
geode->addDrawable(drawable.get());
|
|
|
|
osg::ref_ptr<osg::Camera> camera = new osg::Camera();
|
|
camera->setReferenceFrame(osg::Transform::ABSOLUTE_RF);
|
|
camera->setProjectionResizePolicy(osg::Camera::FIXED);
|
|
camera->setProjectionMatrix(osg::Matrix::identity());
|
|
camera->setViewMatrix(osg::Matrix::identity());
|
|
camera->setRenderOrder(osg::Camera::POST_RENDER);
|
|
camera->setClearMask(GL_NONE);
|
|
osg::StateSet *state = new osg::StateSet;
|
|
state->setTextureMode(0, GL_TEXTURE_2D, osg::StateAttribute::ON);
|
|
state->setMode(GL_DEPTH_TEST, osg::StateAttribute::OFF);
|
|
state->setMode(GL_BLEND, osg::StateAttribute::ON);
|
|
state->setAttribute(new osg::BlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA));
|
|
geode->setStateSet(state);
|
|
geode->setCullingActive(false);
|
|
camera->addChild(geode.get());
|
|
|
|
mGuiRoot = camera;
|
|
mSceneRoot->addChild(mGuiRoot.get());
|
|
mViewer->addEventHandler(new ResizeHandler(this));
|
|
|
|
osg::ref_ptr<osg::Viewport> vp = mViewer->getCamera()->getViewport();
|
|
setViewSize(vp->width(), vp->height());
|
|
|
|
MYGUI_PLATFORM_LOG(Info, getClassTypeName()<<" successfully initialized");
|
|
mIsInitialise = true;
|
|
}
|
|
|
|
void RenderManager::shutdown()
|
|
{
|
|
}
|
|
|
|
MyGUI::IVertexBuffer* RenderManager::createVertexBuffer()
|
|
{
|
|
return new OSGVertexBuffer();
|
|
}
|
|
|
|
void RenderManager::destroyVertexBuffer(MyGUI::IVertexBuffer *buffer)
|
|
{
|
|
delete buffer;
|
|
}
|
|
|
|
|
|
void RenderManager::begin()
|
|
{
|
|
osg::State *state = mRenderInfo->getState();
|
|
state->disableAllVertexArrays();
|
|
state->setClientActiveTextureUnit(0);
|
|
glEnableClientState(GL_VERTEX_ARRAY);
|
|
glEnableClientState(GL_TEXTURE_COORD_ARRAY);
|
|
glEnableClientState(GL_COLOR_ARRAY);
|
|
}
|
|
|
|
void RenderManager::doRender(MyGUI::IVertexBuffer *buffer, MyGUI::ITexture *texture, size_t count)
|
|
{
|
|
osg::State *state = mRenderInfo->getState();
|
|
osg::VertexBufferObject *vbo = static_cast<OSGVertexBuffer*>(buffer)->getBuffer();
|
|
MYGUI_PLATFORM_ASSERT(vbo, "Vertex buffer is not created");
|
|
|
|
if(texture)
|
|
{
|
|
osg::Texture2D *tex = static_cast<OSGTexture*>(texture)->getTexture();
|
|
MYGUI_PLATFORM_ASSERT(tex, "Texture is not created");
|
|
state->applyTextureAttribute(0, tex);
|
|
}
|
|
|
|
osg::GLBufferObject* bufferobject = state->isVertexBufferObjectSupported() ? vbo->getOrCreateGLBufferObject(state->getContextID()) : 0;
|
|
if (bufferobject)
|
|
{
|
|
state->bindVertexBufferObject(bufferobject);
|
|
|
|
glVertexPointer(3, GL_FLOAT, sizeof(MyGUI::Vertex), (char*)NULL);
|
|
glColorPointer(4, GL_UNSIGNED_BYTE, sizeof(MyGUI::Vertex), (char*)NULL + 12);
|
|
glTexCoordPointer(2, GL_FLOAT, sizeof(MyGUI::Vertex), (char*)NULL + 16);
|
|
}
|
|
else
|
|
{
|
|
glVertexPointer(3, GL_FLOAT, sizeof(MyGUI::Vertex), (char*)vbo->getArray(0)->getDataPointer());
|
|
glColorPointer(4, GL_UNSIGNED_BYTE, sizeof(MyGUI::Vertex), (char*)vbo->getArray(0)->getDataPointer() + 12);
|
|
glTexCoordPointer(2, GL_FLOAT, sizeof(MyGUI::Vertex), (char*)vbo->getArray(0)->getDataPointer() + 16);
|
|
}
|
|
|
|
glDrawArrays(GL_TRIANGLES, 0, count);
|
|
}
|
|
|
|
void RenderManager::end()
|
|
{
|
|
osg::State *state = mRenderInfo->getState();
|
|
state->unbindVertexBufferObject();
|
|
state->dirtyAllVertexArrays();
|
|
state->disableAllVertexArrays();
|
|
}
|
|
|
|
void RenderManager::drawFrame(osg::RenderInfo &renderInfo)
|
|
{
|
|
MyGUI::Gui *gui = MyGUI::Gui::getInstancePtr();
|
|
if(gui == nullptr) return;
|
|
|
|
mRenderInfo = &renderInfo;
|
|
|
|
static MyGUI::Timer timer;
|
|
static unsigned long last_time = timer.getMilliseconds();
|
|
unsigned long now_time = timer.getMilliseconds();
|
|
unsigned long time = now_time - last_time;
|
|
|
|
onFrameEvent((float)((double)(time) / (double)1000));
|
|
|
|
last_time = now_time;
|
|
|
|
begin();
|
|
onRenderToTarget(this, mUpdate);
|
|
end();
|
|
|
|
mUpdate = false;
|
|
}
|
|
|
|
void RenderManager::setViewSize(int width, int height)
|
|
{
|
|
if(width < 1) width = 1;
|
|
if(height < 1) height = 1;
|
|
|
|
mGuiRoot->setViewport(0, 0, width, height);
|
|
mViewSize.set(width, height);
|
|
|
|
mInfo.maximumDepth = 1;
|
|
mInfo.hOffset = 0;
|
|
mInfo.vOffset = 0;
|
|
mInfo.aspectCoef = float(mViewSize.height) / float(mViewSize.width);
|
|
mInfo.pixScaleX = 1.0f / float(mViewSize.width);
|
|
mInfo.pixScaleY = 1.0f / float(mViewSize.height);
|
|
|
|
onResizeView(mViewSize);
|
|
mUpdate = true;
|
|
}
|
|
|
|
|
|
bool RenderManager::isFormatSupported(MyGUI::PixelFormat /*format*/, MyGUI::TextureUsage /*usage*/)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
MyGUI::ITexture* RenderManager::createTexture(const std::string &name)
|
|
{
|
|
MapTexture::const_iterator item = mTextures.find(name);
|
|
MYGUI_PLATFORM_ASSERT(item == mTextures.end(), "Texture '"<<name<<"' already exist");
|
|
|
|
OSGTexture* texture = new OSGTexture(name, mTextureManager);
|
|
mTextures.insert(std::make_pair(name, texture));
|
|
return texture;
|
|
}
|
|
|
|
void RenderManager::destroyTexture(MyGUI::ITexture *texture)
|
|
{
|
|
if(texture == nullptr)
|
|
return;
|
|
|
|
MapTexture::iterator item = mTextures.find(texture->getName());
|
|
MYGUI_PLATFORM_ASSERT(item != mTextures.end(), "Texture '"<<texture->getName()<<"' not found");
|
|
|
|
mTextures.erase(item);
|
|
delete texture;
|
|
}
|
|
|
|
MyGUI::ITexture* RenderManager::getTexture(const std::string &name)
|
|
{
|
|
if (name.empty())
|
|
return NULL;
|
|
|
|
MapTexture::const_iterator item = mTextures.find(name);
|
|
if(item == mTextures.end())
|
|
{
|
|
MyGUI::ITexture* tex = createTexture(name);
|
|
tex->loadFromFile(name);
|
|
return tex;
|
|
}
|
|
return item->second;
|
|
}
|
|
|
|
void RenderManager::destroyAllResources()
|
|
{
|
|
for (MapTexture::iterator it = mTextures.begin(); it != mTextures.end(); ++it)
|
|
delete it->second;
|
|
mTextures.clear();
|
|
}
|
|
|
|
bool RenderManager::checkTexture(MyGUI::ITexture* _texture)
|
|
{
|
|
for (MapTexture::const_iterator item = mTextures.begin(); item != mTextures.end(); ++item)
|
|
{
|
|
if (item->second == _texture)
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
}
|