#include "graphicspage.hpp" #include #include #include #include #include #include #include #include #include #include "settings/graphicssettings.hpp" QString getAspect(int x, int y) { int gcd = boost::math::gcd (x, y); int xaspect = x / gcd; int yaspect = y / gcd; // special case: 8 : 5 is usually referred to as 16:10 if (xaspect == 8 && yaspect == 5) return QString("16:10"); return QString(QString::number(xaspect) + ":" + QString::number(yaspect)); } GraphicsPage::GraphicsPage(Files::ConfigurationManager &cfg, GraphicsSettings &graphicsSetting, QWidget *parent) : mCfgMgr(cfg) , mGraphicsSettings(graphicsSetting) , QWidget(parent) { setupUi(this); // Set the maximum res we can set in windowed mode QRect res = getMaximumResolution(); customWidthSpinBox->setMaximum(res.width()); customHeightSpinBox->setMaximum(res.height()); connect(rendererComboBox, SIGNAL(currentIndexChanged(const QString&)), this, SLOT(rendererChanged(const QString&))); connect(fullScreenCheckBox, SIGNAL(stateChanged(int)), this, SLOT(slotFullScreenChanged(int))); connect(standardRadioButton, SIGNAL(toggled(bool)), this, SLOT(slotStandardToggled(bool))); connect(screenComboBox, SIGNAL(currentIndexChanged(const QString&)), this, SLOT(screenChanged(const QString&))); } bool GraphicsPage::setupOgre() { // Create a log manager so we can surpress debug text to stdout/stderr Ogre::LogManager* logMgr = OGRE_NEW Ogre::LogManager; logMgr->createLog((mCfgMgr.getLogPath().string() + "/launcherOgre.log"), true, false, false); try { mOgre = new Ogre::Root("", "", "./launcherOgre.log"); } catch(Ogre::Exception &ex) { QString ogreError = QString::fromStdString(ex.getFullDescription().c_str()); QMessageBox msgBox; msgBox.setWindowTitle("Error creating Ogre::Root"); msgBox.setIcon(QMessageBox::Critical); msgBox.setStandardButtons(QMessageBox::Ok); msgBox.setText(tr("
Failed to create the Ogre::Root object

\ Press \"Show Details...\" for more information.
")); msgBox.setDetailedText(ogreError); msgBox.exec(); qCritical("Error creating Ogre::Root, the error reported was:\n %s", qPrintable(ogreError)); return false; } 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 } QDir dir(QString::fromStdString(pluginDir)); pluginDir = dir.absolutePath().toStdString(); Files::loadOgrePlugin(pluginDir, "RenderSystem_GL", *mOgre); Files::loadOgrePlugin(pluginDir, "RenderSystem_GL3Plus", *mOgre); Files::loadOgrePlugin(pluginDir, "RenderSystem_Direct3D9", *mOgre); #ifdef ENABLE_PLUGIN_GL mGLPlugin = new Ogre::GLPlugin(); mOgre->installPlugin(mGLPlugin); #endif #ifdef ENABLE_PLUGIN_Direct3D9 mD3D9Plugin = new Ogre::D3D9Plugin(); mOgre->installPlugin(mD3D9Plugin); #endif // Get the available renderers and put them in the combobox const Ogre::RenderSystemList &renderers = mOgre->getAvailableRenderers(); for (Ogre::RenderSystemList::const_iterator r = renderers.begin(); r != renderers.end(); ++r) { mSelectedRenderSystem = *r; rendererComboBox->addItem((*r)->getName().c_str()); } QString openGLName = QString("OpenGL Rendering Subsystem"); QString direct3DName = QString("Direct3D9 Rendering Subsystem"); // Create separate rendersystems mOpenGLRenderSystem = mOgre->getRenderSystemByName(openGLName.toStdString()); mDirect3DRenderSystem = mOgre->getRenderSystemByName(direct3DName.toStdString()); if (!mOpenGLRenderSystem && !mDirect3DRenderSystem) { QMessageBox msgBox; msgBox.setWindowTitle(tr("Error creating renderer")); msgBox.setIcon(QMessageBox::Critical); msgBox.setStandardButtons(QMessageBox::Ok); msgBox.setText(tr("
Could not select a valid render system

\ Please make sure the plugins.cfg file exists and contains a valid rendering plugin.
")); msgBox.exec(); return false; } // Now fill the GUI elements int index = rendererComboBox->findText(mGraphicsSettings.value(QString("Video/render system"))); if ( index != -1) { rendererComboBox->setCurrentIndex(index); } else { #if OGRE_PLATFORM == OGRE_PLATFORM_WIN32 rendererComboBox->setCurrentIndex(rendererComboBox->findText(direct3DName)); #else rendererComboBox->setCurrentIndex(rendererComboBox->findText(openGLName)); #endif } antiAliasingComboBox->clear(); antiAliasingComboBox->addItems(getAvailableOptions(QString("FSAA"), mSelectedRenderSystem)); return true; } bool GraphicsPage::setupSDL() { // FIXME: do setupSDLWordaround here instead. // seems like Qt, SDL and Ogre don't like each other // results in a segfault if SDL is initialized after Qt QStringList screens; for (int i = 0; i < mScreenCount; i++) { screens.append(QString("Screen ") + QString::number(i + 1)); } screenComboBox->addItems(screens); return true; } std::vector GraphicsPage::mVideoModes; int GraphicsPage::mScreenCount; bool GraphicsPage::setupSDLWordaround() { if (SDL_Init(SDL_INIT_VIDEO) != 0) { std::cout << "SDL_Init failed: " << SDL_GetError() << std::endl; return false; } SDL_DisplayMode mode; int displayIndex, modeIndex, displays = SDL_GetNumVideoDisplays(); mScreenCount = displays; if(displays < 0) { std::cout << "SDL_GetNumVideoDisplays failed: " << SDL_GetError() << std::endl; SDL_Quit(); return false; } for (displayIndex = 0; displayIndex < displays; displayIndex++) { int modes = SDL_GetNumDisplayModes(displayIndex); if(modes < 0) { std::cout << "SDL_GetNumDisplayModes failed: " << SDL_GetError() << std::endl; SDL_Quit(); return false; } for (modeIndex = 0; modeIndex < modes; modeIndex++) { if (SDL_GetDisplayMode(displayIndex, modeIndex, &mode) < 0) { std::cout << "SDL_GetDisplayMode failed: " << SDL_GetError() << std::endl; SDL_Quit(); return false; } VideoMode vmode; vmode.w = mode.w; vmode.h = mode.h; vmode.screen = displayIndex; mVideoModes.push_back(vmode); } } SDL_Quit(); return true; } bool GraphicsPage::loadSettings() { if (!setupSDL()) return false; if (!setupOgre()) return false; if (mGraphicsSettings.value(QString("Video/vsync")) == QLatin1String("true")) vSyncCheckBox->setCheckState(Qt::Checked); if (mGraphicsSettings.value(QString("Video/fullscreen")) == QLatin1String("true")) fullScreenCheckBox->setCheckState(Qt::Checked); int aaIndex = antiAliasingComboBox->findText(mGraphicsSettings.value(QString("Video/antialiasing"))); if (aaIndex != -1) antiAliasingComboBox->setCurrentIndex(aaIndex); QString width = mGraphicsSettings.value(QString("Video/resolution x")); QString height = mGraphicsSettings.value(QString("Video/resolution y")); QString resolution = width + QString(" x ") + height; QString screen = mGraphicsSettings.value(QString("Video/screen")); int screenIndex = screenComboBox->findText(QString("Screen ") + screen); if (screenIndex != -1) screenComboBox->setCurrentIndex(screenIndex); int resIndex = resolutionComboBox->findText(resolution, Qt::MatchStartsWith); if (resIndex != -1) { standardRadioButton->toggle(); resolutionComboBox->setCurrentIndex(resIndex); } else { customRadioButton->toggle(); customWidthSpinBox->setValue(width.toInt()); customHeightSpinBox->setValue(height.toInt()); } return true; } void GraphicsPage::saveSettings() { vSyncCheckBox->checkState() ? mGraphicsSettings.setValue(QString("Video/vsync"), QString("true")) : mGraphicsSettings.setValue(QString("Video/vsync"), QString("false")); fullScreenCheckBox->checkState() ? mGraphicsSettings.setValue(QString("Video/fullscreen"), QString("true")) : mGraphicsSettings.setValue(QString("Video/fullscreen"), QString("false")); mGraphicsSettings.setValue(QString("Video/antialiasing"), antiAliasingComboBox->currentText()); mGraphicsSettings.setValue(QString("Video/render system"), rendererComboBox->currentText()); if (standardRadioButton->isChecked()) { QRegExp resolutionRe(QString("(\\d+) x (\\d+).*")); if (resolutionRe.exactMatch(resolutionComboBox->currentText().simplified())) { mGraphicsSettings.setValue(QString("Video/resolution x"), resolutionRe.cap(1)); mGraphicsSettings.setValue(QString("Video/resolution y"), resolutionRe.cap(2)); } } else { mGraphicsSettings.setValue(QString("Video/resolution x"), QString::number(customWidthSpinBox->value())); mGraphicsSettings.setValue(QString("Video/resolution y"), QString::number(customHeightSpinBox->value())); } QRegExp screenRe(QString(".*(\\d+)")); if(screenRe.exactMatch(screenComboBox->currentText())) { mGraphicsSettings.setValue(QString("Video/screen"), screenRe.cap(1)); } } QStringList GraphicsPage::getAvailableOptions(const QString &key, Ogre::RenderSystem *renderer) { QStringList result; uint row = 0; Ogre::ConfigOptionMap options = renderer->getConfigOptions(); for (Ogre::ConfigOptionMap::iterator i = options.begin (); i != options.end (); i++, row++) { Ogre::StringVector::iterator opt_it; uint idx = 0; for (opt_it = i->second.possibleValues.begin(); opt_it != i->second.possibleValues.end(); opt_it++, idx++) { if (strcmp (key.toStdString().c_str(), i->first.c_str()) == 0) { result << ((key == "FSAA") ? QString("MSAA ") : QString("")) + QString::fromStdString((*opt_it).c_str()).simplified(); } } } // Sort ascending qSort(result.begin(), result.end(), naturalSortLessThanCI); // Replace the zero option with Off int index = result.indexOf("MSAA 0"); if (index != -1) result.replace(index, tr("Off")); return result; } #if 1 QStringList GraphicsPage::getAvailableResolutions(int screen) { QStringList result; for (std::vector::iterator it = mVideoModes.begin(); it != mVideoModes.end(); it++) { VideoMode mode = *it; if(mode.screen != screen) continue; QString aspect = getAspect(mode.w, mode.h); QString resolution = QString::number(mode.w) + QString(" x ") + QString::number(mode.h); if (aspect == QLatin1String("16:9") || aspect == QLatin1String("16:10")) { resolution.append(tr("\t(Wide ") + aspect + ")"); } else if (aspect == QLatin1String("4:3")) { resolution.append(tr("\t(Standard 4:3)")); } result.append(resolution); } return result; } #endif #if 0 QStringList GraphicsPage::getAvailableResolutions(Ogre::RenderSystem *renderer) { QString key("Video Mode"); QStringList result; uint row = 0; Ogre::ConfigOptionMap options = renderer->getConfigOptions(); for (Ogre::ConfigOptionMap::iterator i = options.begin (); i != options.end (); i++, row++) { if (key.toStdString() != i->first) continue; Ogre::StringVector::iterator opt_it; uint idx = 0; for (opt_it = i->second.possibleValues.begin (); opt_it != i->second.possibleValues.end (); opt_it++, idx++) { QRegExp resolutionRe(QString("(\\d+) x (\\d+).*")); QString resolution = QString::fromStdString(*opt_it).simplified(); if (resolutionRe.exactMatch(resolution)) { int width = resolutionRe.cap(1).toInt(); int height = resolutionRe.cap(2).toInt(); QString aspect = getAspect(width, height); QString cleanRes = resolutionRe.cap(1) + QString(" x ") + resolutionRe.cap(2); if (aspect == QLatin1String("16:9") || aspect == QLatin1String("16:10")) { cleanRes.append(tr("\t(Wide ") + aspect + ")"); } else if (aspect == QLatin1String("4:3")) { cleanRes.append(tr("\t(Standard 4:3)")); } // do not add duplicate resolutions if (!result.contains(cleanRes)) result.append(cleanRes); } } } // Sort the resolutions in descending order qSort(result.begin(), result.end(), naturalSortGreaterThanCI); return result; } #endif QRect GraphicsPage::getMaximumResolution() { QRect max; int i, screens = QApplication::desktop()->screenCount(); for (i = 0; i < screens; i++) { QRect res = QApplication::desktop()->screenGeometry(i); if (res.width() > max.width()) max.setWidth(res.width()); if (res.height() > max.height()) max.setHeight(res.height()); } return max; } void GraphicsPage::rendererChanged(const QString &renderer) { mSelectedRenderSystem = mOgre->getRenderSystemByName(renderer.toStdString()); antiAliasingComboBox->clear(); antiAliasingComboBox->addItems(getAvailableOptions(QString("FSAA"), mSelectedRenderSystem)); } void GraphicsPage::screenChanged(const QString &screen) { QRegExp screenRe(QString(".*(\\d+)")); if(screenRe.exactMatch(screen)) { resolutionComboBox->clear(); resolutionComboBox->addItems(getAvailableResolutions(screenRe.cap(1).toInt() - 1)); } } void GraphicsPage::slotFullScreenChanged(int state) { if (state == Qt::Checked) { standardRadioButton->toggle(); customRadioButton->setEnabled(false); customWidthSpinBox->setEnabled(false); customHeightSpinBox->setEnabled(false); } else { customRadioButton->setEnabled(true); customWidthSpinBox->setEnabled(true); customHeightSpinBox->setEnabled(true); } } void GraphicsPage::slotStandardToggled(bool checked) { if (checked) { resolutionComboBox->setEnabled(true); customWidthSpinBox->setEnabled(false); customHeightSpinBox->setEnabled(false); } else { resolutionComboBox->setEnabled(false); customWidthSpinBox->setEnabled(true); customHeightSpinBox->setEnabled(true); } }