2015-04-24 19:55:30 +00:00
|
|
|
#include "myguirendermanager.hpp"
|
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
#include <regex>
|
|
|
|
|
2015-04-24 19:55:30 +00:00
|
|
|
#include <MyGUI_Gui.h>
|
|
|
|
#include <MyGUI_Timer.h>
|
2020-03-29 23:25:23 +00:00
|
|
|
#include <MyGUI_LayerManager.h>
|
|
|
|
#include <MyGUI_LayerNode.h>
|
|
|
|
#include <MyGUI_Enumerator.h>
|
2015-04-24 19:55:30 +00:00
|
|
|
|
|
|
|
#include <osg/Drawable>
|
|
|
|
#include <osg/BlendFunc>
|
|
|
|
#include <osg/Texture2D>
|
2016-02-05 19:23:41 +00:00
|
|
|
#include <osg/TexMat>
|
2015-04-24 19:55:30 +00:00
|
|
|
|
|
|
|
#include <osgViewer/Viewer>
|
|
|
|
|
|
|
|
#include <osgGA/GUIEventHandler>
|
|
|
|
|
2016-02-05 22:03:53 +00:00
|
|
|
#include <components/resource/imagemanager.hpp>
|
2015-04-24 19:55:30 +00:00
|
|
|
|
2015-04-30 22:59:41 +00:00
|
|
|
#include "myguitexture.hpp"
|
|
|
|
|
2015-04-24 19:55:30 +00:00
|
|
|
#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); \
|
|
|
|
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); \
|
|
|
|
std::ostringstream stream; \
|
|
|
|
stream << dest << "\n"; \
|
|
|
|
MYGUI_BASE_EXCEPT(stream.str().c_str(), "MyGUI"); \
|
|
|
|
} \
|
|
|
|
} while(0)
|
|
|
|
|
2015-04-30 23:15:25 +00:00
|
|
|
namespace osgMyGUI
|
2015-04-24 19:55:30 +00:00
|
|
|
{
|
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
class GUICamera;
|
|
|
|
|
2015-05-04 20:17:28 +00:00
|
|
|
class Drawable : public osg::Drawable {
|
2020-05-01 19:37:01 +00:00
|
|
|
osgMyGUI::RenderManager *mManager;
|
2015-11-07 16:21:03 +00:00
|
|
|
osg::ref_ptr<osg::StateSet> mStateSet;
|
|
|
|
|
|
|
|
public:
|
2015-05-04 20:17:28 +00:00
|
|
|
|
|
|
|
// Stage 0: update widget animations and controllers. Run during the Update traversal.
|
|
|
|
class FrameUpdate : public osg::Drawable::UpdateCallback
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
FrameUpdate()
|
2018-10-09 06:21:12 +00:00
|
|
|
: mRenderManager(nullptr)
|
2015-05-04 20:17:28 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void setRenderManager(osgMyGUI::RenderManager* renderManager)
|
|
|
|
{
|
|
|
|
mRenderManager = renderManager;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void update(osg::NodeVisitor*, osg::Drawable*)
|
|
|
|
{
|
|
|
|
if (mRenderManager)
|
|
|
|
mRenderManager->update();
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
osgMyGUI::RenderManager* mRenderManager;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Stage 1: collect draw calls. Run during the Cull traversal.
|
|
|
|
class CollectDrawCalls : public osg::Drawable::CullCallback
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
CollectDrawCalls()
|
2020-05-01 19:37:01 +00:00
|
|
|
: mCamera(nullptr)
|
|
|
|
, mFilter("")
|
2015-05-04 20:17:28 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
void setCamera(osgMyGUI::GUICamera* camera)
|
2015-05-04 20:17:28 +00:00
|
|
|
{
|
2020-05-01 19:37:01 +00:00
|
|
|
mCamera = camera;
|
2015-05-04 20:17:28 +00:00
|
|
|
}
|
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
void setFilter(std::string filter)
|
2015-05-04 20:17:28 +00:00
|
|
|
{
|
2020-05-01 19:37:01 +00:00
|
|
|
mFilter = filter;
|
2015-05-04 20:17:28 +00:00
|
|
|
}
|
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
virtual bool cull(osg::NodeVisitor*, osg::Drawable*, osg::State*) const;
|
|
|
|
|
2015-05-04 20:17:28 +00:00
|
|
|
private:
|
2020-05-01 19:37:01 +00:00
|
|
|
GUICamera* mCamera;
|
|
|
|
std::string mFilter;
|
2015-05-04 20:17:28 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// Stage 2: execute the draw calls. Run during the Draw traversal. May run in parallel with the update traversal of the next frame.
|
|
|
|
virtual void drawImplementation(osg::RenderInfo &renderInfo) const
|
|
|
|
{
|
|
|
|
osg::State *state = renderInfo.getState();
|
2015-11-07 16:21:03 +00:00
|
|
|
|
|
|
|
state->pushStateSet(mStateSet);
|
|
|
|
state->apply();
|
|
|
|
|
2015-05-04 20:17:28 +00:00
|
|
|
state->disableAllVertexArrays();
|
|
|
|
state->setClientActiveTextureUnit(0);
|
|
|
|
glEnableClientState(GL_VERTEX_ARRAY);
|
|
|
|
glEnableClientState(GL_TEXTURE_COORD_ARRAY);
|
|
|
|
glEnableClientState(GL_COLOR_ARRAY);
|
|
|
|
|
2015-06-08 01:26:36 +00:00
|
|
|
mReadFrom = (mReadFrom+1)%sNumBuffers;
|
2015-05-05 00:38:59 +00:00
|
|
|
const std::vector<Batch>& vec = mBatchVector[mReadFrom];
|
|
|
|
for (std::vector<Batch>::const_iterator it = vec.begin(); it != vec.end(); ++it)
|
2015-05-04 20:17:28 +00:00
|
|
|
{
|
|
|
|
const Batch& batch = *it;
|
|
|
|
osg::VertexBufferObject *vbo = batch.mVertexBuffer;
|
2015-11-07 16:21:03 +00:00
|
|
|
|
|
|
|
if (batch.mStateSet)
|
|
|
|
{
|
|
|
|
state->pushStateSet(batch.mStateSet);
|
|
|
|
state->apply();
|
|
|
|
}
|
|
|
|
|
2015-05-04 20:17:28 +00:00
|
|
|
osg::Texture2D* texture = batch.mTexture;
|
|
|
|
if(texture)
|
|
|
|
state->applyTextureAttribute(0, texture);
|
|
|
|
|
2016-03-11 00:35:28 +00:00
|
|
|
osg::GLBufferObject* bufferobject = state->isVertexBufferObjectSupported() ? vbo->getOrCreateGLBufferObject(state->getContextID()) : 0;
|
|
|
|
if (bufferobject)
|
2015-05-04 20:17:28 +00:00
|
|
|
{
|
|
|
|
state->bindVertexBufferObject(bufferobject);
|
|
|
|
|
2018-11-08 17:10:23 +00:00
|
|
|
glVertexPointer(3, GL_FLOAT, sizeof(MyGUI::Vertex), reinterpret_cast<char*>(0));
|
|
|
|
glColorPointer(4, GL_UNSIGNED_BYTE, sizeof(MyGUI::Vertex), reinterpret_cast<char*>(12));
|
|
|
|
glTexCoordPointer(2, GL_FLOAT, sizeof(MyGUI::Vertex), reinterpret_cast<char*>(16));
|
2015-05-04 20:17:28 +00:00
|
|
|
}
|
|
|
|
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, batch.mVertexCount);
|
2015-11-07 16:21:03 +00:00
|
|
|
|
|
|
|
if (batch.mStateSet)
|
|
|
|
{
|
|
|
|
state->popStateSet();
|
|
|
|
state->apply();
|
|
|
|
}
|
2015-05-04 20:17:28 +00:00
|
|
|
}
|
|
|
|
|
2015-05-08 23:06:55 +00:00
|
|
|
glDisableClientState(GL_VERTEX_ARRAY);
|
|
|
|
glDisableClientState(GL_TEXTURE_COORD_ARRAY);
|
|
|
|
glDisableClientState(GL_COLOR_ARRAY);
|
|
|
|
|
2015-11-07 16:21:03 +00:00
|
|
|
state->popStateSet();
|
|
|
|
|
2015-05-04 20:17:28 +00:00
|
|
|
state->unbindVertexBufferObject();
|
|
|
|
state->dirtyAllVertexArrays();
|
|
|
|
state->disableAllVertexArrays();
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
2020-05-01 19:37:01 +00:00
|
|
|
Drawable(std::string filter = "", osgMyGUI::RenderManager *manager = nullptr, osgMyGUI::GUICamera* camera = nullptr)
|
|
|
|
: mManager(manager)
|
2015-05-05 00:38:59 +00:00
|
|
|
, mWriteTo(0)
|
|
|
|
, mReadFrom(0)
|
2015-05-04 20:17:28 +00:00
|
|
|
{
|
|
|
|
setSupportsDisplayList(false);
|
|
|
|
|
|
|
|
osg::ref_ptr<CollectDrawCalls> collectDrawCalls = new CollectDrawCalls;
|
2020-05-01 19:37:01 +00:00
|
|
|
collectDrawCalls->setCamera(camera);
|
|
|
|
collectDrawCalls->setFilter(filter);
|
2015-05-04 20:17:28 +00:00
|
|
|
setCullCallback(collectDrawCalls);
|
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
if (mManager)
|
|
|
|
{
|
|
|
|
osg::ref_ptr<FrameUpdate> frameUpdate = new FrameUpdate;
|
|
|
|
frameUpdate->setRenderManager(mManager);
|
|
|
|
setUpdateCallback(frameUpdate);
|
|
|
|
}
|
2015-11-07 16:21:03 +00:00
|
|
|
|
|
|
|
mStateSet = new osg::StateSet;
|
|
|
|
mStateSet->setMode(GL_LIGHTING, osg::StateAttribute::OFF);
|
|
|
|
mStateSet->setTextureMode(0, GL_TEXTURE_2D, osg::StateAttribute::ON);
|
|
|
|
mStateSet->setMode(GL_DEPTH_TEST, osg::StateAttribute::OFF);
|
|
|
|
mStateSet->setMode(GL_BLEND, osg::StateAttribute::ON);
|
2016-02-05 19:23:41 +00:00
|
|
|
|
|
|
|
// need to flip tex coords since MyGUI uses DirectX convention of top left image origin
|
|
|
|
osg::Matrix flipMat;
|
|
|
|
flipMat.preMultTranslate(osg::Vec3f(0,1,0));
|
|
|
|
flipMat.preMultScale(osg::Vec3f(1,-1,1));
|
|
|
|
mStateSet->setTextureAttribute(0, new osg::TexMat(flipMat), osg::StateAttribute::ON);
|
2015-05-04 20:17:28 +00:00
|
|
|
}
|
|
|
|
Drawable(const Drawable ©, const osg::CopyOp ©op=osg::CopyOp::SHALLOW_COPY)
|
|
|
|
: osg::Drawable(copy, copyop)
|
2020-05-01 19:37:01 +00:00
|
|
|
, mManager(copy.mManager)
|
2015-11-07 16:21:03 +00:00
|
|
|
, mStateSet(copy.mStateSet)
|
2015-07-01 01:42:04 +00:00
|
|
|
, mWriteTo(0)
|
|
|
|
, mReadFrom(0)
|
2015-05-04 20:17:28 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
// Defines the necessary information for a draw call
|
|
|
|
struct Batch
|
|
|
|
{
|
|
|
|
// May be empty
|
|
|
|
osg::ref_ptr<osg::Texture2D> mTexture;
|
|
|
|
|
|
|
|
osg::ref_ptr<osg::VertexBufferObject> mVertexBuffer;
|
2015-05-05 00:38:59 +00:00
|
|
|
// need to hold on to this too as the mVertexBuffer does not hold a ref to its own array
|
2017-01-19 02:17:49 +00:00
|
|
|
osg::ref_ptr<osg::Array> mArray;
|
2015-05-05 00:38:59 +00:00
|
|
|
|
2015-11-07 16:21:03 +00:00
|
|
|
// optional
|
|
|
|
osg::ref_ptr<osg::StateSet> mStateSet;
|
|
|
|
|
2015-05-04 20:17:28 +00:00
|
|
|
size_t mVertexCount;
|
|
|
|
};
|
|
|
|
|
|
|
|
void addBatch(const Batch& batch)
|
|
|
|
{
|
2015-05-05 00:38:59 +00:00
|
|
|
mBatchVector[mWriteTo].push_back(batch);
|
2015-05-04 20:17:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void clear()
|
|
|
|
{
|
2015-06-08 01:26:36 +00:00
|
|
|
mWriteTo = (mWriteTo+1)%sNumBuffers;
|
2015-05-05 00:38:59 +00:00
|
|
|
mBatchVector[mWriteTo].clear();
|
2015-05-04 20:17:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
META_Object(osgMyGUI, Drawable)
|
|
|
|
|
|
|
|
private:
|
2015-06-08 01:26:36 +00:00
|
|
|
// 2 would be enough in most cases, use 4 to get stereo working
|
|
|
|
static const int sNumBuffers = 4;
|
|
|
|
|
2015-06-02 23:18:03 +00:00
|
|
|
// double buffering approach, to avoid the need for synchronization with the draw thread
|
2015-06-08 01:26:36 +00:00
|
|
|
std::vector<Batch> mBatchVector[sNumBuffers];
|
2015-06-02 23:18:03 +00:00
|
|
|
|
2015-05-05 00:38:59 +00:00
|
|
|
int mWriteTo;
|
|
|
|
mutable int mReadFrom;
|
2015-05-04 20:17:28 +00:00
|
|
|
};
|
|
|
|
|
2015-04-24 19:55:30 +00:00
|
|
|
class OSGVertexBuffer : public MyGUI::IVertexBuffer
|
|
|
|
{
|
2017-01-19 02:17:49 +00:00
|
|
|
osg::ref_ptr<osg::VertexBufferObject> mBuffer[2];
|
|
|
|
osg::ref_ptr<osg::UByteArray> mVertexArray[2];
|
2015-04-24 19:55:30 +00:00
|
|
|
|
|
|
|
size_t mNeedVertexCount;
|
|
|
|
|
2017-01-19 02:17:49 +00:00
|
|
|
unsigned int mCurrentBuffer;
|
|
|
|
bool mUsed; // has the mCurrentBuffer been submitted to the rendering thread
|
2015-06-07 16:57:26 +00:00
|
|
|
|
|
|
|
void destroy();
|
2017-01-19 02:17:49 +00:00
|
|
|
osg::UByteArray* create();
|
2015-06-07 16:57:26 +00:00
|
|
|
|
2015-04-24 19:55:30 +00:00
|
|
|
public:
|
|
|
|
OSGVertexBuffer();
|
2017-01-19 02:17:49 +00:00
|
|
|
virtual ~OSGVertexBuffer() {}
|
2015-04-24 19:55:30 +00:00
|
|
|
|
2017-01-19 02:17:49 +00:00
|
|
|
void markUsed();
|
|
|
|
|
|
|
|
osg::Array* getVertexArray();
|
|
|
|
osg::VertexBufferObject* getVertexBuffer();
|
2015-06-07 16:57:26 +00:00
|
|
|
|
2015-04-24 19:55:30 +00:00
|
|
|
virtual void setVertexCount(size_t count);
|
|
|
|
virtual size_t getVertexCount();
|
|
|
|
|
|
|
|
virtual MyGUI::Vertex *lock();
|
|
|
|
virtual void unlock();
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
OSGVertexBuffer::OSGVertexBuffer()
|
|
|
|
: mNeedVertexCount(0)
|
2017-01-19 02:17:49 +00:00
|
|
|
, mCurrentBuffer(0)
|
|
|
|
, mUsed(false)
|
2015-04-24 19:55:30 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2017-01-19 02:17:49 +00:00
|
|
|
void OSGVertexBuffer::markUsed()
|
2015-06-07 16:57:26 +00:00
|
|
|
{
|
2017-01-19 02:17:49 +00:00
|
|
|
mUsed = true;
|
2015-06-07 16:57:26 +00:00
|
|
|
}
|
|
|
|
|
2015-04-24 19:55:30 +00:00
|
|
|
void OSGVertexBuffer::setVertexCount(size_t count)
|
|
|
|
{
|
|
|
|
if(count == mNeedVertexCount)
|
|
|
|
return;
|
|
|
|
|
|
|
|
mNeedVertexCount = count;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t OSGVertexBuffer::getVertexCount()
|
|
|
|
{
|
|
|
|
return mNeedVertexCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
MyGUI::Vertex *OSGVertexBuffer::lock()
|
|
|
|
{
|
2017-01-19 02:17:49 +00:00
|
|
|
if (mUsed)
|
2015-06-07 16:57:26 +00:00
|
|
|
{
|
2017-01-19 02:17:49 +00:00
|
|
|
mCurrentBuffer = (mCurrentBuffer+1)%2;
|
|
|
|
mUsed = false;
|
2015-06-07 16:57:26 +00:00
|
|
|
}
|
2017-01-19 02:17:49 +00:00
|
|
|
osg::UByteArray* array = mVertexArray[mCurrentBuffer];
|
|
|
|
if (!array)
|
2015-06-07 16:57:26 +00:00
|
|
|
{
|
2017-01-19 02:17:49 +00:00
|
|
|
array = create();
|
|
|
|
}
|
|
|
|
else if (array->size() != mNeedVertexCount * sizeof(MyGUI::Vertex))
|
|
|
|
{
|
|
|
|
array->resize(mNeedVertexCount * sizeof(MyGUI::Vertex));
|
2015-06-07 16:57:26 +00:00
|
|
|
}
|
2015-05-05 00:38:59 +00:00
|
|
|
|
2017-01-19 02:17:49 +00:00
|
|
|
return (MyGUI::Vertex*)&(*array)[0];
|
2015-04-24 19:55:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void OSGVertexBuffer::unlock()
|
|
|
|
{
|
2017-01-19 02:17:49 +00:00
|
|
|
mVertexArray[mCurrentBuffer]->dirty();
|
|
|
|
mBuffer[mCurrentBuffer]->dirty();
|
2015-04-24 19:55:30 +00:00
|
|
|
}
|
|
|
|
|
2017-01-19 02:17:49 +00:00
|
|
|
osg::UByteArray* OSGVertexBuffer::create()
|
2015-04-24 19:55:30 +00:00
|
|
|
{
|
2017-01-19 02:17:49 +00:00
|
|
|
mVertexArray[mCurrentBuffer] = new osg::UByteArray(mNeedVertexCount*sizeof(MyGUI::Vertex));
|
|
|
|
|
|
|
|
mBuffer[mCurrentBuffer] = new osg::VertexBufferObject;
|
|
|
|
mBuffer[mCurrentBuffer]->setDataVariance(osg::Object::DYNAMIC);
|
|
|
|
mBuffer[mCurrentBuffer]->setUsage(GL_DYNAMIC_DRAW);
|
|
|
|
// NB mBuffer does not own the array
|
|
|
|
mBuffer[mCurrentBuffer]->setArray(0, mVertexArray[mCurrentBuffer].get());
|
|
|
|
|
|
|
|
return mVertexArray[mCurrentBuffer];
|
2015-04-24 19:55:30 +00:00
|
|
|
}
|
|
|
|
|
2017-01-19 02:17:49 +00:00
|
|
|
osg::Array* OSGVertexBuffer::getVertexArray()
|
2015-04-24 19:55:30 +00:00
|
|
|
{
|
2017-01-19 02:17:49 +00:00
|
|
|
return mVertexArray[mCurrentBuffer];
|
|
|
|
}
|
2015-04-24 19:55:30 +00:00
|
|
|
|
2017-01-19 02:17:49 +00:00
|
|
|
osg::VertexBufferObject* OSGVertexBuffer::getVertexBuffer()
|
|
|
|
{
|
|
|
|
return mBuffer[mCurrentBuffer];
|
2015-04-24 19:55:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
/// Camera used to draw a MyGUI layer
|
|
|
|
class GUICamera : public osg::Camera, public StateInjectableRenderTarget
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
GUICamera(osg::Camera::RenderOrder order, RenderManager* parent, std::string filter)
|
|
|
|
: mParent(parent)
|
|
|
|
, mUpdate(false)
|
2020-05-12 20:13:01 +00:00
|
|
|
, mFilter(filter)
|
2020-05-01 19:37:01 +00:00
|
|
|
{
|
|
|
|
setReferenceFrame(osg::Transform::ABSOLUTE_RF);
|
|
|
|
setProjectionResizePolicy(osg::Camera::FIXED);
|
|
|
|
setProjectionMatrix(osg::Matrix::identity());
|
|
|
|
setViewMatrix(osg::Matrix::identity());
|
|
|
|
setRenderOrder(order);
|
|
|
|
setClearMask(GL_NONE);
|
|
|
|
setName("GUI Camera");
|
|
|
|
mDrawable = new Drawable(filter, parent, this);
|
|
|
|
mDrawable->setName("GUI Drawable");
|
|
|
|
addChild(mDrawable.get());
|
|
|
|
mDrawable->setCullingActive(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
~GUICamera()
|
|
|
|
{
|
|
|
|
mParent->deleteGUICamera(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Called by the cull traversal
|
|
|
|
/** @see IRenderTarget::begin */
|
|
|
|
void begin() override;
|
|
|
|
void end() override;
|
|
|
|
|
|
|
|
/** @see IRenderTarget::doRender */
|
|
|
|
void doRender(MyGUI::IVertexBuffer* buffer, MyGUI::ITexture* texture, size_t count) override;
|
|
|
|
|
|
|
|
|
|
|
|
void collectDrawCalls();
|
|
|
|
void collectDrawCalls(std::string filter);
|
|
|
|
|
|
|
|
void setViewSize(MyGUI::IntSize viewSize);
|
|
|
|
|
|
|
|
/** @see IRenderTarget::getInfo */
|
|
|
|
const MyGUI::RenderTargetInfo& getInfo() override { return mInfo; }
|
|
|
|
|
|
|
|
RenderManager* mParent;
|
|
|
|
osg::ref_ptr<Drawable> mDrawable;
|
|
|
|
MyGUI::RenderTargetInfo mInfo;
|
|
|
|
bool mUpdate;
|
2020-05-12 20:13:01 +00:00
|
|
|
std::string mFilter;
|
2020-05-01 19:37:01 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
void GUICamera::begin()
|
|
|
|
{
|
|
|
|
mDrawable->clear();
|
|
|
|
// variance will be recomputed based on textures being rendered in this frame
|
|
|
|
mDrawable->setDataVariance(osg::Object::STATIC);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Drawable::CollectDrawCalls::cull(osg::NodeVisitor*, osg::Drawable*, osg::State*) const
|
|
|
|
{
|
|
|
|
if (!mCamera)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (mFilter.empty())
|
|
|
|
mCamera->collectDrawCalls();
|
|
|
|
else
|
|
|
|
mCamera->collectDrawCalls(mFilter);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
RenderManager::RenderManager(osgViewer::Viewer *viewer, osg::Group *sceneroot, Resource::ImageManager* imageManager, float scalingFactor)
|
2015-04-24 19:55:30 +00:00
|
|
|
: mViewer(viewer)
|
|
|
|
, mSceneRoot(sceneroot)
|
2016-02-05 22:10:27 +00:00
|
|
|
, mImageManager(imageManager)
|
2015-04-24 19:55:30 +00:00
|
|
|
, mIsInitialise(false)
|
2015-05-14 22:41:21 +00:00
|
|
|
, mInvScalingFactor(1.f)
|
2015-04-24 19:55:30 +00:00
|
|
|
{
|
2015-05-14 22:41:21 +00:00
|
|
|
if (scalingFactor != 0.f)
|
|
|
|
mInvScalingFactor = 1.f / scalingFactor;
|
2020-05-12 20:13:01 +00:00
|
|
|
|
|
|
|
|
|
|
|
osg::ref_ptr<osg::Viewport> vp = mViewer->getCamera()->getViewport();
|
|
|
|
setViewSize(vp->width(), vp->height());
|
2015-04-24 19:55:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
RenderManager::~RenderManager()
|
|
|
|
{
|
|
|
|
MYGUI_PLATFORM_LOG(Info, "* Shutdown: "<<getClassTypeName());
|
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
for (auto guiCamera : mGuiCameras)
|
|
|
|
mSceneRoot->removeChild(guiCamera);
|
|
|
|
mGuiCameras.clear();
|
2015-04-24 19:55:30 +00:00
|
|
|
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;
|
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
mSceneRoot->addChild(createGUICamera(osg::Camera::POST_RENDER, ""));
|
2015-04-24 19:55:30 +00:00
|
|
|
|
|
|
|
MYGUI_PLATFORM_LOG(Info, getClassTypeName()<<" successfully initialized");
|
|
|
|
mIsInitialise = true;
|
|
|
|
}
|
|
|
|
|
2015-04-28 14:02:29 +00:00
|
|
|
void RenderManager::shutdown()
|
|
|
|
{
|
2020-05-01 19:37:01 +00:00
|
|
|
// TODO: Is this method meaningful? Why not just let the destructor handle everything?
|
|
|
|
for (auto guiCamera : mGuiCameras)
|
|
|
|
{
|
|
|
|
guiCamera->removeChildren(0, guiCamera->getNumChildren());
|
|
|
|
mSceneRoot->removeChild(guiCamera);
|
|
|
|
}
|
2015-04-28 14:02:29 +00:00
|
|
|
}
|
|
|
|
|
2015-04-24 19:55:30 +00:00
|
|
|
MyGUI::IVertexBuffer* RenderManager::createVertexBuffer()
|
|
|
|
{
|
|
|
|
return new OSGVertexBuffer();
|
|
|
|
}
|
|
|
|
|
|
|
|
void RenderManager::destroyVertexBuffer(MyGUI::IVertexBuffer *buffer)
|
|
|
|
{
|
|
|
|
delete buffer;
|
|
|
|
}
|
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
void GUICamera::doRender(MyGUI::IVertexBuffer *buffer, MyGUI::ITexture *texture, size_t count)
|
2015-04-24 19:55:30 +00:00
|
|
|
{
|
2015-05-04 20:17:28 +00:00
|
|
|
Drawable::Batch batch;
|
|
|
|
batch.mVertexCount = count;
|
2017-01-19 02:17:49 +00:00
|
|
|
batch.mVertexBuffer = static_cast<OSGVertexBuffer*>(buffer)->getVertexBuffer();
|
|
|
|
batch.mArray = static_cast<OSGVertexBuffer*>(buffer)->getVertexArray();
|
|
|
|
static_cast<OSGVertexBuffer*>(buffer)->markUsed();
|
2015-05-04 20:17:28 +00:00
|
|
|
if (texture)
|
2015-05-08 22:13:22 +00:00
|
|
|
{
|
2015-05-04 20:17:28 +00:00
|
|
|
batch.mTexture = static_cast<OSGTexture*>(texture)->getTexture();
|
2015-05-08 22:13:22 +00:00
|
|
|
if (batch.mTexture->getDataVariance() == osg::Object::DYNAMIC)
|
|
|
|
mDrawable->setDataVariance(osg::Object::DYNAMIC); // only for this frame, reset in begin()
|
|
|
|
}
|
2020-05-01 19:37:01 +00:00
|
|
|
|
2015-11-07 16:21:03 +00:00
|
|
|
if (mInjectState)
|
|
|
|
batch.mStateSet = mInjectState;
|
2015-04-24 19:55:30 +00:00
|
|
|
|
2015-05-04 20:17:28 +00:00
|
|
|
mDrawable->addBatch(batch);
|
2015-04-24 19:55:30 +00:00
|
|
|
}
|
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
void StateInjectableRenderTarget::setInjectState(osg::StateSet* stateSet)
|
2015-11-07 16:21:03 +00:00
|
|
|
{
|
|
|
|
mInjectState = stateSet;
|
|
|
|
}
|
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
void GUICamera::end()
|
2015-04-24 19:55:30 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-05-04 20:17:28 +00:00
|
|
|
void RenderManager::update()
|
2015-04-24 19:55:30 +00:00
|
|
|
{
|
|
|
|
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;
|
2015-05-04 20:17:28 +00:00
|
|
|
}
|
2015-04-24 19:55:30 +00:00
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
void GUICamera::collectDrawCalls()
|
2015-05-04 20:17:28 +00:00
|
|
|
{
|
2015-05-08 22:13:22 +00:00
|
|
|
begin();
|
2020-05-01 19:37:01 +00:00
|
|
|
MyGUI::LayerManager* myGUILayers = MyGUI::LayerManager::getInstancePtr();
|
|
|
|
if (myGUILayers != nullptr)
|
|
|
|
{
|
|
|
|
for (unsigned i = 0; i < myGUILayers->getLayerCount(); i++)
|
|
|
|
{
|
|
|
|
auto layer = myGUILayers->getLayer(i);
|
|
|
|
layer->renderToTarget(this, mUpdate);
|
|
|
|
}
|
|
|
|
}
|
2015-05-08 22:13:22 +00:00
|
|
|
end();
|
2015-04-24 19:55:30 +00:00
|
|
|
|
|
|
|
mUpdate = false;
|
|
|
|
}
|
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
void GUICamera::collectDrawCalls(std::string filter)
|
2015-04-24 19:55:30 +00:00
|
|
|
{
|
2020-05-01 19:37:01 +00:00
|
|
|
begin();
|
|
|
|
MyGUI::LayerManager* myGUILayers = MyGUI::LayerManager::getInstancePtr();
|
|
|
|
if (myGUILayers != nullptr)
|
|
|
|
{
|
|
|
|
for (unsigned i = 0; i < myGUILayers->getLayerCount(); i++)
|
|
|
|
{
|
|
|
|
auto layer = myGUILayers->getLayer(i);
|
|
|
|
auto name = layer->getName();
|
2015-05-14 22:41:21 +00:00
|
|
|
|
2020-05-17 14:25:51 +00:00
|
|
|
if (filter.find(name) != std::string::npos)
|
|
|
|
{
|
2020-05-01 19:37:01 +00:00
|
|
|
layer->renderToTarget(this, mUpdate);
|
2020-05-17 14:25:51 +00:00
|
|
|
}
|
2020-05-01 19:37:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
end();
|
2015-04-24 19:55:30 +00:00
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
mUpdate = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GUICamera::setViewSize(MyGUI::IntSize viewSize)
|
|
|
|
{
|
2015-04-24 19:55:30 +00:00
|
|
|
mInfo.maximumDepth = 1;
|
|
|
|
mInfo.hOffset = 0;
|
|
|
|
mInfo.vOffset = 0;
|
2020-05-01 19:37:01 +00:00
|
|
|
mInfo.aspectCoef = float(viewSize.height) / float(viewSize.width);
|
|
|
|
mInfo.pixScaleX = 1.0f / float(viewSize.width);
|
|
|
|
mInfo.pixScaleY = 1.0f / float(viewSize.height);
|
|
|
|
mUpdate = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RenderManager::setViewSize(int width, int height)
|
|
|
|
{
|
|
|
|
if(width < 1) width = 1;
|
|
|
|
if(height < 1) height = 1;
|
2015-04-24 21:30:30 +00:00
|
|
|
|
2020-05-01 19:37:01 +00:00
|
|
|
mViewSize.set(width * mInvScalingFactor, height * mInvScalingFactor);
|
|
|
|
|
|
|
|
for (auto* camera : mGuiCameras)
|
|
|
|
{
|
|
|
|
GUICamera* guiCamera = static_cast<GUICamera*>(camera);
|
|
|
|
guiCamera->setViewport(0, 0, width, height);
|
|
|
|
guiCamera->setViewSize(mViewSize);
|
|
|
|
}
|
2015-04-24 19:55:30 +00:00
|
|
|
onResizeView(mViewSize);
|
2020-05-01 19:37:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
osg::ref_ptr<osg::Camera> RenderManager::createGUICamera(int order, std::string layerFilter)
|
|
|
|
{
|
|
|
|
osg::ref_ptr<GUICamera> camera = new GUICamera(static_cast<osg::Camera::RenderOrder>(order), this, layerFilter);
|
|
|
|
mGuiCameras.insert(camera);
|
2020-05-12 20:13:01 +00:00
|
|
|
camera->setViewport(0, 0, mViewSize.width, mViewSize.height);
|
|
|
|
camera->setViewSize(mViewSize);
|
2020-05-01 19:37:01 +00:00
|
|
|
return camera;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RenderManager::deleteGUICamera(GUICamera* camera)
|
|
|
|
{
|
|
|
|
mGuiCameras.erase(camera);
|
2015-04-24 19:55:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool RenderManager::isFormatSupported(MyGUI::PixelFormat /*format*/, MyGUI::TextureUsage /*usage*/)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
MyGUI::ITexture* RenderManager::createTexture(const std::string &name)
|
|
|
|
{
|
2015-06-03 16:00:45 +00:00
|
|
|
MapTexture::iterator item = mTextures.find(name);
|
|
|
|
if (item != mTextures.end())
|
|
|
|
{
|
|
|
|
delete item->second;
|
|
|
|
mTextures.erase(item);
|
|
|
|
}
|
2015-04-24 19:55:30 +00:00
|
|
|
|
2016-02-05 22:10:27 +00:00
|
|
|
OSGTexture* texture = new OSGTexture(name, mImageManager);
|
2015-04-24 19:55:30 +00:00
|
|
|
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)
|
|
|
|
{
|
2015-05-02 16:06:54 +00:00
|
|
|
if (name.empty())
|
2018-10-09 06:21:12 +00:00
|
|
|
return nullptr;
|
2015-05-02 16:06:54 +00:00
|
|
|
|
2015-04-24 19:55:30 +00:00
|
|
|
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)
|
|
|
|
{
|
2015-07-10 01:03:17 +00:00
|
|
|
// We support external textures that aren't registered via this manager, so can't implement this method sensibly.
|
|
|
|
return true;
|
2015-04-24 19:55:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|