#include "renderer.hpp" #include "fader.hpp" #include "particles.hpp" #include #include #include "OgreRoot.h" #include "OgreRenderWindow.h" #include "OgreLogManager.h" #include "OgreLog.h" #include "OgreTextureManager.h" #include "OgreTexture.h" #include "OgreHardwarePixelBuffer.h" #include #include "OgreParticleAffectorFactory.h" #include #include #include #include #include #if defined(__APPLE__) && !defined(__LP64__) #include #endif using namespace Ogre; using namespace OEngine::Render; #if defined(__APPLE__) && !defined(__LP64__) CustomRoot::CustomRoot(const Ogre::String& pluginFileName, const Ogre::String& configFileName, const Ogre::String& logFileName) : Ogre::Root(pluginFileName, configFileName, logFileName) {} bool CustomRoot::isQueuedEnd() const { return mQueuedEnd; } #endif void OgreRenderer::cleanup() { delete mFader; mFader = NULL; delete mRoot; mRoot = NULL; SDL_DestroyWindow(mSDLWindow); mSDLWindow = NULL; unloadPlugins(); } void OgreRenderer::start() { //TODO: Check if we still need to do this if we're using SDL's //message pump #if defined(__APPLE__) && !defined(__LP64__) // OSX Carbon Message Pump do { EventRef event = NULL; EventTargetRef targetWindow; targetWindow = GetEventDispatcherTarget(); // If we are unable to get the target then we no longer care about events. if (!targetWindow) return; // Grab the next event while possible while (ReceiveNextEvent(0, NULL, kEventDurationNoWait, true, &event) == noErr) { // Dispatch the event SendEventToEventTarget(event, targetWindow); ReleaseEvent(event); } if (!mRoot->renderOneFrame()) { break; } } while (!mRoot->isQueuedEnd()); #else mRoot->startRendering(); #endif } void OgreRenderer::loadPlugins() { #ifdef ENABLE_PLUGIN_GL mGLPlugin = new Ogre::GLPlugin(); mRoot->installPlugin(mGLPlugin); #endif #ifdef ENABLE_PLUGIN_Direct3D9 mD3D9Plugin = new Ogre::D3D9Plugin(); mRoot->installPlugin(mD3D9Plugin); #endif #ifdef ENABLE_PLUGIN_CgProgramManager mCgPlugin = new Ogre::CgPlugin(); mRoot->installPlugin(mCgPlugin); #endif #ifdef ENABLE_PLUGIN_OctreeSceneManager mOctreePlugin = new Ogre::OctreePlugin(); mRoot->installPlugin(mOctreePlugin); #endif #ifdef ENABLE_PLUGIN_ParticleFX mParticleFXPlugin = new Ogre::ParticleFXPlugin(); mRoot->installPlugin(mParticleFXPlugin); #endif } void OgreRenderer::unloadPlugins() { std::vector::iterator ei; for(ei = mEmitterFactories.begin();ei != mEmitterFactories.end();ei++) OGRE_DELETE (*ei); mEmitterFactories.clear(); std::vector::iterator ai; for(ai = mAffectorFactories.begin();ai != mAffectorFactories.end();ai++) OGRE_DELETE (*ai); mAffectorFactories.clear(); #ifdef ENABLE_PLUGIN_GL delete mGLPlugin; mGLPlugin = NULL; #endif #ifdef ENABLE_PLUGIN_Direct3D9 delete mD3D9Plugin; mD3D9Plugin = NULL; #endif #ifdef ENABLE_PLUGIN_CgProgramManager delete mCgPlugin; mCgPlugin = NULL; #endif #ifdef ENABLE_PLUGIN_OctreeSceneManager delete mOctreePlugin; mOctreePlugin = NULL; #endif #ifdef ENABLE_PLUGIN_ParticleFX delete mParticleFXPlugin; mParticleFXPlugin = NULL; #endif } void OgreRenderer::update(float dt) { mFader->update(dt); } void OgreRenderer::screenshot(const std::string &file) { mWindow->writeContentsToFile(file); } float OgreRenderer::getFPS() { return mWindow->getLastFPS(); } void OgreRenderer::configure(const std::string &logPath, const std::string& renderSystem, const std::string& rttMode, bool _logging) { // Set up logging first new LogManager; Log *log = LogManager::getSingleton().createLog(logPath + std::string("Ogre.log")); logging = _logging; if(logging) // Full log detail log->setLogDetail(LL_BOREME); else // Disable logging log->setDebugOutputEnabled(false); #if defined(__APPLE__) && !defined(__LP64__) mRoot = new CustomRoot("", "", ""); #else mRoot = new Root("", "", ""); #endif #if defined(ENABLE_PLUGIN_GL) || defined(ENABLE_PLUGIN_Direct3D9) || defined(ENABLE_PLUGIN_CgProgramManager) || defined(ENABLE_PLUGIN_OctreeSceneManager) || defined(ENABLE_PLUGIN_ParticleFX) loadPlugins(); #endif std::string pluginDir; const char* pluginEnv = getenv("OPENMW_OGRE_PLUGIN_DIR"); if (pluginEnv) pluginDir = pluginEnv; else { #if OGRE_PLATFORM == OGRE_PLATFORM_WIN32 pluginDir = ".\\"; #endif #if OGRE_PLATFORM == OGRE_PLATFORM_APPLE pluginDir = OGRE_PLUGIN_DIR; #endif #if OGRE_PLATFORM == OGRE_PLATFORM_LINUX pluginDir = OGRE_PLUGIN_DIR_REL; #endif } boost::filesystem::path absPluginPath = boost::filesystem::absolute(boost::filesystem::path(pluginDir)); pluginDir = absPluginPath.string(); Files::loadOgrePlugin(pluginDir, "RenderSystem_GL", *mRoot); Files::loadOgrePlugin(pluginDir, "RenderSystem_GLES2", *mRoot); Files::loadOgrePlugin(pluginDir, "RenderSystem_GL3Plus", *mRoot); Files::loadOgrePlugin(pluginDir, "RenderSystem_Direct3D9", *mRoot); Files::loadOgrePlugin(pluginDir, "Plugin_CgProgramManager", *mRoot); Files::loadOgrePlugin(pluginDir, "Plugin_ParticleFX", *mRoot); Ogre::ParticleEmitterFactory *emitter; emitter = OGRE_NEW NifEmitterFactory(); Ogre::ParticleSystemManager::getSingleton().addEmitterFactory(emitter); mEmitterFactories.push_back(emitter); Ogre::ParticleAffectorFactory *affector; affector = OGRE_NEW GrowFadeAffectorFactory(); Ogre::ParticleSystemManager::getSingleton().addAffectorFactory(affector); mAffectorFactories.push_back(affector); affector = OGRE_NEW GravityAffectorFactory(); Ogre::ParticleSystemManager::getSingleton().addAffectorFactory(affector); mAffectorFactories.push_back(affector); RenderSystem* rs = mRoot->getRenderSystemByName(renderSystem); if (rs == 0) throw std::runtime_error ("RenderSystem with name " + renderSystem + " not found, make sure the plugins are loaded"); mRoot->setRenderSystem(rs); if (rs->getName().find("OpenGL") != std::string::npos) rs->setConfigOption ("RTT Preferred Mode", rttMode); } void OgreRenderer::recreateWindow(const std::string &title, const WindowSettings &settings) { Ogre::ColourValue viewportBG = mView->getBackgroundColour(); mRoot->destroyRenderTarget(mWindow); NameValuePairList params; params.insert(std::make_pair("title", title)); params.insert(std::make_pair("FSAA", settings.fsaa)); params.insert(std::make_pair("vsync", settings.vsync ? "true" : "false")); mWindow = mRoot->createRenderWindow(title, settings.window_x, settings.window_y, settings.fullscreen, ¶ms); // Create one viewport, entire window mView = mWindow->addViewport(mCamera); mView->setBackgroundColour(viewportBG); adjustViewport(); } void OgreRenderer::createWindow(const std::string &title, const WindowSettings& settings) { assert(mRoot); mRoot->initialise(false); NameValuePairList params; params.insert(std::make_pair("title", title)); params.insert(std::make_pair("FSAA", settings.fsaa)); params.insert(std::make_pair("vsync", settings.vsync ? "true" : "false")); // Create an application window with the following settings: mSDLWindow = SDL_CreateWindow( "OpenMW", // window title SDL_WINDOWPOS_UNDEFINED, // initial x position SDL_WINDOWPOS_UNDEFINED, // initial y position settings.window_x, // width, in pixels settings.window_y, // height, in pixels SDL_WINDOW_SHOWN|SDL_WINDOW_RESIZABLE | (settings.fullscreen ? SDL_WINDOW_FULLSCREEN_DESKTOP : 0) ); //get the native whnd struct SDL_SysWMinfo wmInfo; SDL_VERSION(&wmInfo.version); if(-1 == SDL_GetWindowWMInfo(mSDLWindow, &wmInfo)) throw std::runtime_error("Couldn't get WM Info!"); Ogre::String winHandle; switch(wmInfo.subsystem) { #ifdef WIN32 case SDL_SYSWM_WINDOWS: // Windows code winHandle = Ogre::StringConverter::toString((unsigned long)wmInfo.info.win.window); break; #elif MACOS case SDL_SYSWM_COCOA: //required to make OGRE play nice with our window params.insert(std::make_pair("macAPI", "cocoa")); params.insert(std::make_pair("macAPICocoaUseNSView", "true")); winHandle = Ogre::StringConverter::toString((unsigned long)wmInfo.info.cocoa.window); break; #else case SDL_SYSWM_X11: winHandle = Ogre::StringConverter::toString((unsigned long)wmInfo.info.x11.display); winHandle += ":0:"; winHandle += Ogre::StringConverter::toString((unsigned long)wmInfo.info.x11.window); break; #endif default: throw std::runtime_error("Unexpected WM!"); break; } params.insert(std::make_pair("externalWindowHandle", winHandle)); mWindow = mRoot->createRenderWindow(title, settings.window_x, settings.window_y, settings.fullscreen, ¶ms); // create the semi-transparent black background texture used by the GUI. // has to be created in code with TU_DYNAMIC_WRITE_ONLY param // so that it can be modified at runtime. Ogre::TextureManager::getSingleton().createManual( "transparent.png", Ogre::ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME, Ogre::TEX_TYPE_2D, 1, 1, 0, Ogre::PF_A8R8G8B8, Ogre::TU_WRITE_ONLY); } void OgreRenderer::createScene(const std::string& camName, float fov, float nearClip) { assert(mRoot); assert(mWindow); // Get the SceneManager, in this case a generic one mScene = mRoot->createSceneManager(ST_GENERIC); // Create the camera mCamera = mScene->createCamera(camName); mCamera->setNearClipDistance(nearClip); mCamera->setFOVy(Degree(fov)); // Create one viewport, entire window mView = mWindow->addViewport(mCamera); // Alter the camera aspect ratio to match the viewport mCamera->setAspectRatio(Real(mView->getActualWidth()) / Real(mView->getActualHeight())); mFader = new Fader(mScene); } void OgreRenderer::adjustViewport() { // Alter the camera aspect ratio to match the viewport if(mCamera != NULL) { mView->setDimensions(0, 0, 1, 1); mCamera->setAspectRatio(Real(mView->getActualWidth()) / Real(mView->getActualHeight())); } } void OgreRenderer::setWindowEventListener(Ogre::WindowEventListener* listener) { Ogre::WindowEventUtilities::addWindowEventListener(mWindow, listener); } void OgreRenderer::removeWindowEventListener(Ogre::WindowEventListener* listener) { Ogre::WindowEventUtilities::removeWindowEventListener(mWindow, listener); } void OgreRenderer::setFov(float fov) { mCamera->setFOVy(Degree(fov)); }