2012-09-11 14:37:54 +00:00
|
|
|
#include "loadingscreen.hpp"
|
|
|
|
|
|
|
|
#include <OgreRenderWindow.h>
|
2014-02-19 17:40:29 +00:00
|
|
|
#include <OgreMaterialManager.h>
|
|
|
|
#include <OgreTechnique.h>
|
|
|
|
#include <OgreRectangle2D.h>
|
|
|
|
#include <OgreSceneNode.h>
|
|
|
|
#include <OgreTextureManager.h>
|
|
|
|
#include <OgreViewport.h>
|
2014-03-06 03:01:25 +00:00
|
|
|
#include <OgreHardwarePixelBuffer.h>
|
2015-01-10 02:56:06 +00:00
|
|
|
#include <OgreSceneManager.h>
|
2012-09-11 17:18:26 +00:00
|
|
|
|
2015-01-10 01:50:43 +00:00
|
|
|
#include <MyGUI_RenderManager.h>
|
|
|
|
#include <MyGUI_ScrollBar.h>
|
|
|
|
#include <MyGUI_Gui.h>
|
|
|
|
#include <MyGUI_TextBox.h>
|
|
|
|
|
2015-01-31 22:27:34 +00:00
|
|
|
#include <components/settings/settings.hpp>
|
|
|
|
|
2012-09-11 14:37:54 +00:00
|
|
|
#include "../mwbase/environment.hpp"
|
2012-10-16 18:25:50 +00:00
|
|
|
#include "../mwbase/world.hpp"
|
2014-06-02 21:26:43 +00:00
|
|
|
#include "../mwbase/statemanager.hpp"
|
2012-09-11 14:37:54 +00:00
|
|
|
|
2012-09-12 17:15:29 +00:00
|
|
|
#include "../mwbase/windowmanager.hpp"
|
2013-06-16 17:43:59 +00:00
|
|
|
#include "../mwbase/inputmanager.hpp"
|
2012-09-12 17:15:29 +00:00
|
|
|
|
2014-03-30 21:04:12 +00:00
|
|
|
#include "backgroundimage.hpp"
|
|
|
|
|
2012-09-11 14:37:54 +00:00
|
|
|
namespace MWGui
|
|
|
|
{
|
|
|
|
|
2013-04-10 18:46:21 +00:00
|
|
|
LoadingScreen::LoadingScreen(Ogre::SceneManager* sceneMgr, Ogre::RenderWindow* rw)
|
2012-09-11 14:37:54 +00:00
|
|
|
: mSceneMgr(sceneMgr)
|
|
|
|
, mWindow(rw)
|
2013-04-10 18:46:21 +00:00
|
|
|
, WindowBase("openmw_loading_screen.layout")
|
2015-03-08 00:07:29 +00:00
|
|
|
, mLastRenderTime(0)
|
|
|
|
, mLastWallpaperChangeTime(0)
|
2013-08-27 13:48:13 +00:00
|
|
|
, mProgress(0)
|
2013-10-13 15:52:14 +00:00
|
|
|
, mVSyncWasEnabled(false)
|
2012-09-11 14:37:54 +00:00
|
|
|
{
|
2014-08-01 15:15:28 +00:00
|
|
|
mMainWidget->setSize(MyGUI::RenderManager::getInstance().getViewSize());
|
|
|
|
|
2012-09-11 14:37:54 +00:00
|
|
|
getWidget(mLoadingText, "LoadingText");
|
2012-09-11 14:53:49 +00:00
|
|
|
getWidget(mProgressBar, "ProgressBar");
|
2014-12-19 23:04:24 +00:00
|
|
|
getWidget(mLoadingBox, "LoadingBox");
|
2012-09-11 17:18:26 +00:00
|
|
|
|
2013-08-27 13:48:13 +00:00
|
|
|
mProgressBar->setScrollViewPage(1);
|
2012-09-11 17:18:26 +00:00
|
|
|
|
2014-03-30 21:04:12 +00:00
|
|
|
mBackgroundImage = MyGUI::Gui::getInstance().createWidgetReal<BackgroundImage>("ImageBox", 0,0,1,1,
|
|
|
|
MyGUI::Align::Stretch, "Menu");
|
|
|
|
|
|
|
|
setVisible(false);
|
2012-09-11 14:37:54 +00:00
|
|
|
}
|
|
|
|
|
2013-08-27 13:48:13 +00:00
|
|
|
void LoadingScreen::setLabel(const std::string &label)
|
|
|
|
{
|
|
|
|
mLoadingText->setCaptionWithReplacing(label);
|
2014-12-19 23:04:24 +00:00
|
|
|
int padding = mLoadingBox->getWidth() - mLoadingText->getWidth();
|
|
|
|
MyGUI::IntSize size(mLoadingText->getTextSize().width+padding, mLoadingBox->getHeight());
|
|
|
|
size.width = std::max(300, size.width);
|
|
|
|
mLoadingBox->setSize(size);
|
|
|
|
mLoadingBox->setPosition(mMainWidget->getWidth()/2 - mLoadingBox->getWidth()/2, mLoadingBox->getTop());
|
2013-08-27 13:48:13 +00:00
|
|
|
}
|
|
|
|
|
2012-09-11 14:37:54 +00:00
|
|
|
LoadingScreen::~LoadingScreen()
|
|
|
|
{
|
2014-03-30 21:04:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void LoadingScreen::setVisible(bool visible)
|
|
|
|
{
|
|
|
|
WindowBase::setVisible(visible);
|
|
|
|
mBackgroundImage->setVisible(visible);
|
2012-09-11 17:18:26 +00:00
|
|
|
}
|
|
|
|
|
2013-08-27 13:48:13 +00:00
|
|
|
void LoadingScreen::loadingOn()
|
2012-09-11 14:37:54 +00:00
|
|
|
{
|
2014-03-05 21:24:05 +00:00
|
|
|
// Early-out if already on
|
2014-03-30 21:04:12 +00:00
|
|
|
if (mMainWidget->getVisible())
|
2014-03-05 21:24:05 +00:00
|
|
|
return;
|
|
|
|
|
2013-10-13 15:52:14 +00:00
|
|
|
// Temporarily turn off VSync, we want to do actual loading rather than waiting for the screen to sync.
|
|
|
|
// Threaded loading would be even better, of course - especially because some drivers force VSync to on and we can't change it.
|
|
|
|
mVSyncWasEnabled = mWindow->isVSyncEnabled();
|
|
|
|
mWindow->setVSyncEnabled(false);
|
|
|
|
|
2014-06-02 21:26:43 +00:00
|
|
|
bool showWallpaper = (MWBase::Environment::get().getStateManager()->getState()
|
|
|
|
== MWBase::StateManager::State_NoGame);
|
|
|
|
|
|
|
|
|
|
|
|
if (!showWallpaper)
|
2014-03-05 21:24:05 +00:00
|
|
|
{
|
2014-03-06 03:01:25 +00:00
|
|
|
mBackgroundImage->setImageTexture("");
|
|
|
|
int width = mWindow->getWidth();
|
|
|
|
int height = mWindow->getHeight();
|
|
|
|
const std::string textureName = "@loading_background";
|
|
|
|
Ogre::TexturePtr texture;
|
|
|
|
texture = Ogre::TextureManager::getSingleton().getByName(textureName);
|
|
|
|
if (texture.isNull())
|
|
|
|
{
|
|
|
|
texture = Ogre::TextureManager::getSingleton().createManual(textureName,
|
|
|
|
Ogre::ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME,
|
|
|
|
Ogre::TEX_TYPE_2D,
|
|
|
|
width, height, 0, mWindow->suggestPixelFormat(), Ogre::TU_DYNAMIC_WRITE_ONLY);
|
|
|
|
}
|
|
|
|
texture->unload();
|
|
|
|
texture->setWidth(width);
|
|
|
|
texture->setHeight(height);
|
|
|
|
texture->createInternalResources();
|
|
|
|
mWindow->copyContentsToMemory(texture->getBuffer()->lock(Ogre::Image::Box(0,0,width,height), Ogre::HardwareBuffer::HBL_DISCARD));
|
|
|
|
texture->getBuffer()->unlock();
|
2014-03-30 21:04:12 +00:00
|
|
|
mBackgroundImage->setBackgroundImage(texture->getName(), false, false);
|
2014-03-05 21:24:05 +00:00
|
|
|
}
|
|
|
|
|
2013-08-27 13:48:13 +00:00
|
|
|
setVisible(true);
|
2012-09-11 15:36:20 +00:00
|
|
|
|
2014-06-02 21:26:43 +00:00
|
|
|
if (showWallpaper)
|
2012-09-11 14:37:54 +00:00
|
|
|
{
|
2013-08-27 13:48:13 +00:00
|
|
|
changeWallpaper();
|
2012-09-11 14:37:54 +00:00
|
|
|
}
|
|
|
|
|
2014-06-02 21:26:43 +00:00
|
|
|
MWBase::Environment::get().getWindowManager()->pushGuiMode(showWallpaper ? GM_LoadingWallpaper : GM_Loading);
|
2013-08-27 13:48:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void LoadingScreen::loadingOff()
|
|
|
|
{
|
2013-10-13 15:52:14 +00:00
|
|
|
// Re-enable vsync now.
|
|
|
|
mWindow->setVSyncEnabled(mVSyncWasEnabled);
|
|
|
|
|
2013-08-27 13:48:13 +00:00
|
|
|
setVisible(false);
|
|
|
|
|
|
|
|
MWBase::Environment::get().getWindowManager()->removeGuiMode(GM_Loading);
|
|
|
|
MWBase::Environment::get().getWindowManager()->removeGuiMode(GM_LoadingWallpaper);
|
|
|
|
}
|
2012-09-11 14:37:54 +00:00
|
|
|
|
2013-08-27 13:48:13 +00:00
|
|
|
void LoadingScreen::changeWallpaper ()
|
|
|
|
{
|
|
|
|
if (mResources.empty())
|
|
|
|
{
|
|
|
|
Ogre::StringVector groups = Ogre::ResourceGroupManager::getSingleton().getResourceGroups ();
|
|
|
|
for (Ogre::StringVector::iterator it = groups.begin(); it != groups.end(); ++it)
|
|
|
|
{
|
2014-06-07 15:10:02 +00:00
|
|
|
Ogre::StringVectorPtr resourcesInThisGroup = Ogre::ResourceGroupManager::getSingleton ().findResourceNames (*it, "Splash/*.tga");
|
2013-08-27 13:48:13 +00:00
|
|
|
mResources.insert(mResources.end(), resourcesInThisGroup->begin(), resourcesInThisGroup->end());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!mResources.empty())
|
|
|
|
{
|
|
|
|
std::string const & randomSplash = mResources.at (rand() % mResources.size());
|
|
|
|
|
2014-03-30 21:04:12 +00:00
|
|
|
Ogre::TextureManager::getSingleton ().load (randomSplash, Ogre::ResourceGroupManager::AUTODETECT_RESOURCE_GROUP_NAME);
|
2013-08-27 13:48:13 +00:00
|
|
|
|
2014-03-30 21:04:12 +00:00
|
|
|
// TODO: add option (filename pattern?) to use image aspect ratio instead of 4:3
|
2015-01-27 23:02:05 +00:00
|
|
|
// we can't do this by default, because the Morrowind splash screens are 1024x1024, but should be displayed as 4:3
|
|
|
|
bool stretch = Settings::Manager::getBool("stretch menu background", "GUI");
|
|
|
|
mBackgroundImage->setBackgroundImage(randomSplash, true, stretch);
|
2013-08-27 13:48:13 +00:00
|
|
|
}
|
2012-09-11 14:37:54 +00:00
|
|
|
else
|
2013-08-27 13:48:13 +00:00
|
|
|
std::cerr << "No loading screens found!" << std::endl;
|
|
|
|
}
|
2012-09-11 15:36:20 +00:00
|
|
|
|
2013-08-27 13:48:13 +00:00
|
|
|
void LoadingScreen::setProgressRange (size_t range)
|
|
|
|
{
|
|
|
|
mProgressBar->setScrollRange(range+1);
|
|
|
|
mProgressBar->setScrollPosition(0);
|
|
|
|
mProgressBar->setTrackSize(0);
|
|
|
|
mProgress = 0;
|
|
|
|
}
|
2012-09-11 15:36:20 +00:00
|
|
|
|
2013-08-27 13:48:13 +00:00
|
|
|
void LoadingScreen::setProgress (size_t value)
|
|
|
|
{
|
|
|
|
if (value - mProgress < mProgressBar->getScrollRange()/100.f)
|
|
|
|
return;
|
|
|
|
mProgress = value;
|
|
|
|
mProgressBar->setScrollPosition(0);
|
2015-03-08 00:07:29 +00:00
|
|
|
mProgressBar->setTrackSize(static_cast<int>(value / (float)(mProgressBar->getScrollRange()) * mProgressBar->getLineSize()));
|
2013-08-27 13:48:13 +00:00
|
|
|
draw();
|
|
|
|
}
|
2012-09-11 15:36:20 +00:00
|
|
|
|
2013-08-27 13:48:13 +00:00
|
|
|
void LoadingScreen::increaseProgress (size_t increase)
|
|
|
|
{
|
|
|
|
mProgressBar->setScrollPosition(0);
|
|
|
|
size_t value = mProgress + increase;
|
|
|
|
mProgress = value;
|
2015-03-08 00:07:29 +00:00
|
|
|
mProgressBar->setTrackSize(static_cast<int>(value / (float)(mProgressBar->getScrollRange()) * mProgressBar->getLineSize()));
|
2013-08-27 13:48:13 +00:00
|
|
|
draw();
|
|
|
|
}
|
2012-09-11 14:37:54 +00:00
|
|
|
|
2013-08-27 13:48:13 +00:00
|
|
|
void LoadingScreen::indicateProgress()
|
|
|
|
{
|
|
|
|
float time = (mTimer.getMilliseconds() % 2001) / 1000.f;
|
|
|
|
if (time > 1)
|
|
|
|
time = (time-2)*-1;
|
2012-09-11 14:37:54 +00:00
|
|
|
|
2013-08-27 13:48:13 +00:00
|
|
|
mProgressBar->setTrackSize(50);
|
2015-03-08 00:07:29 +00:00
|
|
|
mProgressBar->setScrollPosition(static_cast<size_t>(time * (mProgressBar->getScrollRange() - 1)));
|
2013-08-27 13:48:13 +00:00
|
|
|
draw();
|
|
|
|
}
|
|
|
|
|
|
|
|
void LoadingScreen::draw()
|
|
|
|
{
|
|
|
|
const float loadingScreenFps = 20.f;
|
2012-09-11 14:37:54 +00:00
|
|
|
|
2012-09-11 14:53:49 +00:00
|
|
|
if (mTimer.getMilliseconds () > mLastRenderTime + (1.f/loadingScreenFps) * 1000.f)
|
2012-09-11 14:37:54 +00:00
|
|
|
{
|
|
|
|
mLastRenderTime = mTimer.getMilliseconds ();
|
|
|
|
|
2014-06-02 21:26:43 +00:00
|
|
|
bool showWallpaper = (MWBase::Environment::get().getStateManager()->getState()
|
|
|
|
== MWBase::StateManager::State_NoGame);
|
|
|
|
|
|
|
|
if (showWallpaper && mTimer.getMilliseconds () > mLastWallpaperChangeTime + 5000*1)
|
2012-09-13 10:33:09 +00:00
|
|
|
{
|
|
|
|
mLastWallpaperChangeTime = mTimer.getMilliseconds ();
|
|
|
|
changeWallpaper();
|
|
|
|
}
|
2012-09-11 14:37:54 +00:00
|
|
|
|
|
|
|
// 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)
|
|
|
|
{
|
2012-09-11 17:18:26 +00:00
|
|
|
if (i > 0 && i < 96)
|
2012-09-11 14:37:54 +00:00
|
|
|
mSceneMgr->addSpecialCaseRenderQueue(i);
|
|
|
|
}
|
|
|
|
mSceneMgr->setSpecialCaseRenderQueueMode(Ogre::SceneManager::SCRQM_EXCLUDE);
|
|
|
|
|
2014-03-27 18:51:48 +00:00
|
|
|
MWBase::Environment::get().getInputManager()->update(0, true, true);
|
2012-09-11 14:37:54 +00:00
|
|
|
|
2013-08-27 13:48:13 +00:00
|
|
|
// First, swap buffers from last draw, then, queue an update of the
|
|
|
|
// window contents, but don't swap buffers (which would have
|
|
|
|
// caused a sync / flush and would be expensive).
|
|
|
|
// We're doing this so we can do some actual loading while the GPU is busy with the render.
|
|
|
|
// This means the render is lagging a frame behind, but this is hardly noticable.
|
2013-10-13 15:52:14 +00:00
|
|
|
mWindow->swapBuffers();
|
2013-10-13 16:28:22 +00:00
|
|
|
|
2013-08-27 13:48:13 +00:00
|
|
|
mWindow->update(false);
|
2012-09-11 17:18:26 +00:00
|
|
|
|
2012-09-11 14:37:54 +00:00
|
|
|
// resume 3d rendering
|
|
|
|
mSceneMgr->clearSpecialCaseRenderQueues();
|
|
|
|
mSceneMgr->setSpecialCaseRenderQueueMode(Ogre::SceneManager::SCRQM_EXCLUDE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|