forked from mirror/openmw-tes3mp
238 lines
8.2 KiB
C++
238 lines
8.2 KiB
C++
#include "loadingscreen.hpp"
|
|
|
|
#include <OgreRenderWindow.h>
|
|
#include <OgreCompositorManager.h>
|
|
#include <OgreCompositorChain.h>
|
|
|
|
#include <openengine/ogre/fader.hpp>
|
|
|
|
#include "../mwbase/environment.hpp"
|
|
#include "../mwbase/world.hpp"
|
|
|
|
#include "../mwbase/windowmanager.hpp"
|
|
#include "../mwbase/inputmanager.hpp"
|
|
|
|
namespace MWGui
|
|
{
|
|
|
|
LoadingScreen::LoadingScreen(Ogre::SceneManager* sceneMgr, Ogre::RenderWindow* rw)
|
|
: mSceneMgr(sceneMgr)
|
|
, mWindow(rw)
|
|
, WindowBase("openmw_loading_screen.layout")
|
|
, mLoadingOn(false)
|
|
, mLastRenderTime(0.f)
|
|
, mLastWallpaperChangeTime(0.f)
|
|
, mFirstLoad(true)
|
|
, mTotalRefsLoading(0)
|
|
, mCurrentCellLoading(0)
|
|
, mTotalCellsLoading(0)
|
|
, mCurrentRefLoading(0)
|
|
, mCurrentRefList(0)
|
|
{
|
|
getWidget(mLoadingText, "LoadingText");
|
|
getWidget(mProgressBar, "ProgressBar");
|
|
getWidget(mBackgroundImage, "BackgroundImage");
|
|
|
|
|
|
mBackgroundMaterial = Ogre::MaterialManager::getSingleton().create("BackgroundMaterial", Ogre::ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME);
|
|
mBackgroundMaterial->getTechnique(0)->getPass(0)->setLightingEnabled(false);
|
|
mBackgroundMaterial->getTechnique(0)->getPass(0)->setDepthCheckEnabled(false);
|
|
mBackgroundMaterial->getTechnique(0)->getPass(0)->createTextureUnitState("");
|
|
|
|
mRectangle = new Ogre::Rectangle2D(true);
|
|
mRectangle->setCorners(-1.0, 1.0, 1.0, -1.0);
|
|
mRectangle->setMaterial("BackgroundMaterial");
|
|
// Render the background before everything else
|
|
mRectangle->setRenderQueueGroup(Ogre::RENDER_QUEUE_OVERLAY-1);
|
|
// Use infinite AAB to always stay visible
|
|
Ogre::AxisAlignedBox aabInf;
|
|
aabInf.setInfinite();
|
|
mRectangle->setBoundingBox(aabInf);
|
|
// Attach background to the scene
|
|
Ogre::SceneNode* node = mSceneMgr->getRootSceneNode()->createChildSceneNode();
|
|
node->attachObject(mRectangle);
|
|
mRectangle->setVisible(false);
|
|
}
|
|
|
|
LoadingScreen::~LoadingScreen()
|
|
{
|
|
delete mRectangle;
|
|
}
|
|
|
|
void LoadingScreen::onResChange(int w, int h)
|
|
{
|
|
setCoord(0,0,w,h);
|
|
}
|
|
|
|
void LoadingScreen::setLoadingProgress (const std::string& stage, int depth, int current, int total)
|
|
{
|
|
if (!mLoadingOn)
|
|
loadingOn();
|
|
|
|
const int numRefLists = 20;
|
|
|
|
if (depth == 0)
|
|
{
|
|
mCurrentCellLoading = current;
|
|
mTotalCellsLoading = total;
|
|
|
|
mCurrentRefLoading = 0;
|
|
mCurrentRefList = 0;
|
|
}
|
|
else if (depth == 1)
|
|
{
|
|
mCurrentRefLoading = current;
|
|
mTotalRefsLoading = total;
|
|
}
|
|
|
|
assert (mTotalCellsLoading != 0);
|
|
|
|
float refProgress;
|
|
if (mTotalRefsLoading <= 1)
|
|
refProgress = 1;
|
|
else
|
|
refProgress = float(mCurrentRefLoading) / float(mTotalRefsLoading-1);
|
|
refProgress += mCurrentRefList;
|
|
refProgress /= numRefLists;
|
|
|
|
assert(refProgress <= 1 && refProgress >= 0);
|
|
|
|
if (depth == 1 && mCurrentRefLoading == mTotalRefsLoading-1)
|
|
++mCurrentRefList;
|
|
|
|
float progress = (float(mCurrentCellLoading)+refProgress) / float(mTotalCellsLoading);
|
|
assert(progress <= 1 && progress >= 0);
|
|
|
|
mLoadingText->setCaption(stage);
|
|
mProgressBar->setProgressPosition (static_cast<size_t>(progress * 1000));
|
|
|
|
static float loadingScreenFps = 30.f;
|
|
|
|
if (mTimer.getMilliseconds () > mLastRenderTime + (1.f/loadingScreenFps) * 1000.f)
|
|
{
|
|
float dt = mTimer.getMilliseconds () - mLastRenderTime;
|
|
mLastRenderTime = mTimer.getMilliseconds ();
|
|
|
|
if (mFirstLoad && mTimer.getMilliseconds () > mLastWallpaperChangeTime + 3000*1)
|
|
{
|
|
mLastWallpaperChangeTime = mTimer.getMilliseconds ();
|
|
changeWallpaper();
|
|
}
|
|
|
|
// Turn off rendering except the GUI
|
|
mSceneMgr->clearSpecialCaseRenderQueues();
|
|
// SCRQM_INCLUDE with RENDER_QUEUE_OVERLAY does not work.
|
|
for (int i = 0; i < Ogre::RENDER_QUEUE_MAX; ++i)
|
|
{
|
|
if (i > 0 && i < 96)
|
|
mSceneMgr->addSpecialCaseRenderQueue(i);
|
|
}
|
|
mSceneMgr->setSpecialCaseRenderQueueMode(Ogre::SceneManager::SCRQM_EXCLUDE);
|
|
|
|
// always update input before rendering something, otherwise mygui goes crazy when something was entered in the frame before
|
|
// (e.g. when using "coc" console command, it would enter an infinite loop and crash due to overflow)
|
|
MWBase::Environment::get().getInputManager()->update(0, true);
|
|
|
|
Ogre::CompositorChain* chain = Ogre::CompositorManager::getSingleton().getCompositorChain(mWindow->getViewport(0));
|
|
|
|
bool hasCompositor = chain->getCompositor ("gbufferFinalizer");
|
|
|
|
|
|
if (!hasCompositor)
|
|
{
|
|
mWindow->getViewport(0)->setClearEveryFrame(false);
|
|
}
|
|
else
|
|
{
|
|
if (!mFirstLoad)
|
|
{
|
|
mBackgroundMaterial->getTechnique(0)->getPass(0)->getTextureUnitState(0)->setTextureName(chain->getCompositor ("gbufferFinalizer")->getTextureInstance ("no_mrt_output", 0)->getName());
|
|
mRectangle->setVisible(true);
|
|
}
|
|
|
|
for (unsigned int i = 0; i<chain->getNumCompositors(); ++i)
|
|
{
|
|
Ogre::CompositorManager::getSingleton().setCompositorEnabled(mWindow->getViewport(0), chain->getCompositor(i)->getCompositor()->getName(), false);
|
|
}
|
|
}
|
|
|
|
MWBase::Environment::get().getWorld ()->getFader ()->update (dt);
|
|
|
|
mWindow->update();
|
|
|
|
if (!hasCompositor)
|
|
mWindow->getViewport(0)->setClearEveryFrame(true);
|
|
else
|
|
{
|
|
for (unsigned int i = 0; i<chain->getNumCompositors(); ++i)
|
|
{
|
|
Ogre::CompositorManager::getSingleton().setCompositorEnabled(mWindow->getViewport(0), chain->getCompositor(i)->getCompositor()->getName(), true);
|
|
}
|
|
}
|
|
|
|
mRectangle->setVisible(false);
|
|
|
|
// resume 3d rendering
|
|
mSceneMgr->clearSpecialCaseRenderQueues();
|
|
mSceneMgr->setSpecialCaseRenderQueueMode(Ogre::SceneManager::SCRQM_EXCLUDE);
|
|
}
|
|
}
|
|
|
|
void LoadingScreen::loadingDone()
|
|
{
|
|
loadingOff();
|
|
}
|
|
|
|
void LoadingScreen::loadingOn()
|
|
{
|
|
setVisible(true);
|
|
mLoadingOn = true;
|
|
|
|
if (mFirstLoad)
|
|
{
|
|
changeWallpaper();
|
|
|
|
MWBase::Environment::get().getWindowManager()->pushGuiMode(GM_LoadingWallpaper);
|
|
}
|
|
else
|
|
{
|
|
mBackgroundImage->setImageTexture("");
|
|
MWBase::Environment::get().getWindowManager()->pushGuiMode(GM_Loading);
|
|
}
|
|
}
|
|
|
|
|
|
void LoadingScreen::loadingOff()
|
|
{
|
|
setVisible(false);
|
|
mLoadingOn = false;
|
|
mFirstLoad = false;
|
|
|
|
MWBase::Environment::get().getWindowManager()->removeGuiMode(GM_Loading);
|
|
MWBase::Environment::get().getWindowManager()->removeGuiMode(GM_LoadingWallpaper);
|
|
}
|
|
|
|
void LoadingScreen::changeWallpaper ()
|
|
{
|
|
if (mResources.empty())
|
|
{
|
|
Ogre::StringVector groups = Ogre::ResourceGroupManager::getSingleton().getResourceGroups ();
|
|
for (Ogre::StringVector::iterator it = groups.begin(); it != groups.end(); ++it)
|
|
{
|
|
Ogre::StringVectorPtr resourcesInThisGroup = Ogre::ResourceGroupManager::getSingleton ().findResourceNames (*it, "Splash_*.tga");
|
|
mResources.insert(mResources.end(), resourcesInThisGroup->begin(), resourcesInThisGroup->end());
|
|
}
|
|
}
|
|
|
|
if (!mResources.empty())
|
|
{
|
|
std::string const & randomSplash = mResources.at (rand() % mResources.size());
|
|
|
|
Ogre::TexturePtr tex = Ogre::TextureManager::getSingleton ().load (randomSplash, Ogre::ResourceGroupManager::AUTODETECT_RESOURCE_GROUP_NAME);
|
|
|
|
mBackgroundImage->setImageTexture (randomSplash);
|
|
}
|
|
else
|
|
std::cerr << "No loading screens found!" << std::endl;
|
|
}
|
|
}
|