openspades/Sources/Gui/StartupScreenHelper.cpp
yvt da7e61b25c
Change std::regex static local variables to global
Removes the runtime cost caused by lazy initialization.
2019-07-23 22:25:12 +09:00

776 lines
25 KiB
C++

/*
Copyright (c) 2013 yvt
Portion of the code is based on Serverbrowser.cpp.
This file is part of OpenSpades.
OpenSpades is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
OpenSpades is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with OpenSpades. If not, see <http://www.gnu.org/licenses/>.
*/
#include <algorithm>
#include <cctype>
#include <utility>
#include <regex>
#include <Imports/OpenGL.h> //for gpu info
#include <Imports/SDL.h>
#include <json/json.h>
#include "StartupScreenHelper.h"
#include "StartupScreen.h"
#include <Audio/ALDevice.h>
#include <Audio/YsrDevice.h>
#include <Core/FileManager.h>
#include <Core/Settings.h>
#include <OpenSpades.h>
#include <Core/ShellApi.h>
#include <Gui/Main.h>
#include <Gui/Icon.h>
#include <Gui/PackageUpdateManager.h>
SPADES_SETTING(r_bloom);
SPADES_SETTING(r_lens);
SPADES_SETTING(r_cameraBlur);
SPADES_SETTING(r_softParticles);
SPADES_SETTING(r_mapSoftShadow);
SPADES_SETTING(r_modelShadows);
SPADES_SETTING(r_radiosity);
SPADES_SETTING(r_dlights);
SPADES_SETTING(r_water);
SPADES_SETTING(r_multisamples);
SPADES_SETTING(r_fxaa);
SPADES_SETTING(r_videoWidth);
SPADES_SETTING(r_videoHeight);
SPADES_SETTING(r_fullscreen);
SPADES_SETTING(r_fogShadow);
SPADES_SETTING(r_lensFlare);
SPADES_SETTING(r_lensFlareDynamic);
SPADES_SETTING(r_blitFramebuffer);
SPADES_SETTING(r_srgb);
SPADES_SETTING(r_shadowMapSize);
SPADES_SETTING(s_maxPolyphonics);
SPADES_SETTING(s_eax);
SPADES_SETTING(r_maxAnisotropy);
SPADES_SETTING(r_colorCorrection);
SPADES_SETTING(r_physicalLighting);
SPADES_SETTING(r_occlusionQuery);
SPADES_SETTING(r_depthOfField);
SPADES_SETTING(r_vsync);
SPADES_SETTING(r_renderer);
SPADES_SETTING(r_swUndersampling);
SPADES_SETTING(r_hdr);
namespace spades {
namespace gui {
StartupScreenHelper::StartupScreenHelper()
: scr(nullptr),
shaderHighCapable(false),
postFilterHighCapable(false),
particleHighCapable(false) {
SPADES_MARK_FUNCTION();
}
StartupScreenHelper::~StartupScreenHelper() { SPADES_MARK_FUNCTION(); }
void StartupScreenHelper::StartupScreenDestroyed() {
SPADES_MARK_FUNCTION();
scr = nullptr;
}
namespace {
std::regex const localeInfoRegex("[-a-zA-Z0-9_]+\\.json");
}
void StartupScreenHelper::ExamineSystem() {
SPADES_MARK_FUNCTION();
// clear capability report
// (this function can be called multiple times via StartupScreenHelper::FixConfig)
reportLines.clear();
report.clear();
// check installed locales
SPLog("Checking installed locales");
auto localeDirectories = FileManager::EnumFiles("Locales");
locales.clear();
for (const std::string &localeInfoName : localeDirectories) {
if (!std::regex_match(localeInfoName, localeInfoRegex)) {
continue;
}
std::string locale = localeInfoName.substr(0, localeInfoName.size() - 5);
std::string localeInfoPath = "Locales/" + localeInfoName;
try {
std::string buffer = FileManager::ReadAllBytes(localeInfoPath.c_str());
LocaleInfo info;
Json::Reader reader;
Json::Value root;
if (!reader.parse(buffer.c_str(), root, false)) {
SPRaise("Failed to parse LocaleInfo.json: %s",
reader.getFormatedErrorMessages().c_str());
}
info.name = locale;
info.descriptionEnglish = root["descriptionEnglish"].asString();
info.descriptionNative = root["description"].asString();
locales.push_back(std::move(info));
SPLog("Locale '%s' found.", locale.c_str());
} catch (const std::exception &ex) {
SPLog("Error while reading the locale info for '%s': %s", locale.c_str(),
ex.what());
}
}
// check audio device availability
// Note: this only checks whether these libraries can be loaded.
SPLog("Checking YSR availability");
if (!audio::YsrDevice::TryLoadYsr()) {
incapableConfigs.insert(
std::make_pair("s_audioDriver", [](std::string value) -> std::string {
if (EqualsIgnoringCase(value, "ysr")) {
return "YSR library couldn't be loaded.";
} else {
return std::string();
}
}));
}
SPLog("Checking OpenAL availability");
if (!audio::ALDevice::TryLoad()) {
incapableConfigs.insert(
std::make_pair("s_audioDriver", [](std::string value) -> std::string {
if (EqualsIgnoringCase(value, "openal")) {
return "OpenAL library couldn't be loaded.";
} else {
return std::string();
}
}));
}
// check GL capabilities
SPLog("Performing ecapability query");
int idDisplay = 0;
int numDisplayMode = SDL_GetNumDisplayModes(idDisplay);
SDL_DisplayMode mode;
modes.clear();
if (numDisplayMode > 0) {
std::set<std::pair<int, int>> foundModes;
for (int i = 0; i < numDisplayMode; i++) {
SDL_GetDisplayMode(idDisplay, i, &mode);
if (mode.w < 800 || mode.h < 600)
continue;
if (foundModes.find(std::make_pair(mode.w, mode.h)) != foundModes.end())
continue;
foundModes.insert(std::make_pair(mode.w, mode.h));
modes.push_back(spades::IntVector3::Make(mode.w, mode.h, 0));
SPLog("Video Mode Found: %dx%d", mode.w, mode.h);
}
} else {
SPLog("Failed to get video mode list. Presetting default list");
modes.push_back(spades::IntVector3::Make(800, 600, 0));
modes.push_back(spades::IntVector3::Make(1024, 768, 0));
modes.push_back(spades::IntVector3::Make(1280, 720, 0));
modes.push_back(spades::IntVector3::Make(1920, 1080, 0));
}
bool capable = true;
SDL_Window *window = SDL_CreateWindow("OpenSpades: Please wait...", 1, 1, 1, 1,
SDL_WINDOW_OPENGL | SDL_WINDOW_BORDERLESS);
if (window == nullptr) {
SPLog("Failed to create SDL window: %s", SDL_GetError());
}
SDL_GLContext context = window ? SDL_GL_CreateContext(window) : nullptr;
if (window != nullptr && context == nullptr) {
SPLog("Failed to create OpenGL context: %s", SDL_GetError());
}
#ifdef __APPLE__
#elif __unix
SDL_Surface *icon = nullptr;
SDL_RWops *icon_rw = nullptr;
icon_rw = SDL_RWFromConstMem(g_appIconData, GetAppIconDataSize());
if (icon_rw != nullptr) {
icon = IMG_LoadPNG_RW(icon_rw);
SDL_FreeRW(icon_rw);
}
if (icon == nullptr) {
std::string msg = SDL_GetError();
SPLog("Failed to load icon: %s", msg.c_str());
} else {
SDL_SetWindowIcon(window, icon);
SDL_FreeSurface(icon);
}
#endif
if (!context) {
// OpenGL initialization failed!
std::string err = SDL_GetError();
SPLog("SDL_SetVideoMode failed: %s", err.c_str());
AddReport("OpenGL-capable graphics accelerator is unavailable.",
MakeVector4(1.f, 0.5f, 0.5f, 1.f));
AddReport();
AddReport("OpenGL/SDL couldn't be initialized.");
AddReport("Falling back to the software renderer.");
AddReport();
AddReport("Message from SDL:", MakeVector4(1.f, 1.f, 1.f, 0.7f));
AddReport(err, MakeVector4(1.f, 1.f, 1.f, 0.7f));
capable = false;
shaderHighCapable = false;
postFilterHighCapable = false;
particleHighCapable = false;
} else {
SDL_GL_MakeCurrent(window, context);
shaderHighCapable = true;
postFilterHighCapable = true;
particleHighCapable = true;
const char *str;
GLint maxTextureSize;
GLint max3DTextureSize;
GLint maxCombinedTextureUnits;
GLint maxVertexTextureUnits;
GLint maxVaryingComponents;
SPLog("--- OpenGL Renderer Info ---");
AddReport("OpenGL-capable graphics accelerator is available.");
if ((str = (const char *)glGetString(GL_VENDOR)) != NULL) {
SPLog("Vendor: %s", str);
AddReport(std::string("Vendor: ") + str, MakeVector4(1.f, 1.f, 1.f, 0.7f));
}
if ((str = (const char *)glGetString(GL_RENDERER)) != NULL) {
SPLog("Name: %s", str);
AddReport(std::string("Name: ") + str, MakeVector4(1.f, 1.f, 1.f, 0.7f));
}
if ((str = (const char *)glGetString(GL_VERSION)) != NULL) {
AddReport(std::string("Version: ") + str, MakeVector4(1.f, 1.f, 1.f, 0.7f));
SPLog("Version: %s", str);
}
if ((str = (const char *)glGetString(GL_SHADING_LANGUAGE_VERSION)) != NULL) {
AddReport(std::string("GLSL Version: ") + str,
MakeVector4(1.f, 1.f, 1.f, 0.7f));
SPLog("Shading Language Version: %s", str);
}
AddReport();
maxTextureSize = 0;
glGetIntegerv(GL_MAX_TEXTURE_SIZE, &maxTextureSize);
if (maxTextureSize > 0) {
SPLog("Max Texture Size: %d", (int)maxTextureSize);
}
max3DTextureSize = 0;
glGetIntegerv(GL_MAX_3D_TEXTURE_SIZE, &max3DTextureSize);
if (max3DTextureSize > 0) {
SPLog("Max 3D Texture Size: %d", (int)max3DTextureSize);
}
maxCombinedTextureUnits = 0;
glGetIntegerv(GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS, &maxCombinedTextureUnits);
if (maxCombinedTextureUnits > 0) {
SPLog("Max Combined Texture Image Units: %d", (int)maxCombinedTextureUnits);
}
maxVertexTextureUnits = 0;
glGetIntegerv(GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS, &maxVertexTextureUnits);
if (maxVertexTextureUnits > 0) {
SPLog("Max Vertex Texture Image Units: %d", (int)maxVertexTextureUnits);
}
maxVaryingComponents = 0;
glGetIntegerv(GL_MAX_VARYING_COMPONENTS, &maxVaryingComponents);
if (maxVaryingComponents > 0) {
SPLog("Max Varying Components: %d", (int)maxVaryingComponents);
}
str = (const char *)glGetString(GL_EXTENSIONS);
std::string extensions;
if (str)
extensions = str;
const char *const requiredExtensions[] = {"GL_ARB_multitexture",
"GL_ARB_shader_objects",
"GL_ARB_shading_language_100",
"GL_ARB_texture_non_power_of_two",
"GL_ARB_vertex_buffer_object",
"GL_EXT_framebuffer_object",
NULL};
SPLog("--- Extensions ---");
std::vector<std::string> strs = spades::Split(str, " ");
for (size_t i = 0; i < strs.size(); i++) {
SPLog("%s", strs[i].c_str());
}
SPLog("------------------");
for (size_t i = 0; requiredExtensions[i]; i++) {
const char *ex = requiredExtensions[i];
if (extensions.find(ex) == std::string::npos) {
// extension not found
AddReport(std::string(ex) + " is NOT SUPPORTED!",
MakeVector4(1.f, 0.5f, 0.5f, 1.f));
capable = false;
}
}
// non-requred extensions
if (extensions.find("GL_ARB_framebuffer_sRGB") == std::string::npos) {
if (r_srgb) {
r_srgb = 0;
SPLog("Disabling r_srgb: no GL_ARB_framebuffer_sRGB");
}
incapableConfigs.insert(
std::make_pair("r_srgb", [](std::string value) -> std::string {
if (std::stoi(value) != 0) {
return "SRGB framebuffer is disabled because your video card doesn't "
"support GL_ARB_framebuffer_sRGB.";
} else {
return std::string();
}
}));
AddReport("GL_ARB_framebuffer_sRGB is NOT SUPPORTED",
MakeVector4(1.f, 1.f, 0.5f, 1.f));
AddReport(" r_srgb is disabled.", MakeVector4(1.f, 1.f, 1.f, 0.7f));
}
if (extensions.find("GL_EXT_framebuffer_blit") == std::string::npos) {
if (r_blitFramebuffer) {
r_blitFramebuffer = 0;
SPLog("Disabling r_blitFramebuffer: no GL_EXT_framebuffer_blit");
}
if (r_multisamples) {
r_multisamples = 0;
SPLog("Disabling r_multisamples: no GL_EXT_framebuffer_blit");
}
incapableConfigs.insert(
std::make_pair("r_blitFramebuffer", [](std::string value) -> std::string {
if (std::stoi(value) != 0) {
return "r_blitFramebuffer is disabled because your video card "
"doesn't support GL_EXT_framebuffer_blit.";
} else {
return std::string();
}
}));
AddReport("GL_EXT_framebuffer_blit is NOT SUPPORTED",
MakeVector4(1.f, 1.f, 0.5f, 1.f));
AddReport(" r_blitFramebuffer is disabled.", MakeVector4(1.f, 1.f, 1.f, 0.7f));
}
if (extensions.find("GL_EXT_texture_filter_anisotropic") == std::string::npos) {
if ((float)r_maxAnisotropy > 1.1f) {
r_maxAnisotropy = 1;
SPLog("Setting r_maxAnisotropy to 1: no GL_EXT_texture_filter_anisotropic");
}
incapableConfigs.insert(
std::make_pair("r_maxAnisotropy", [](std::string value) -> std::string {
if (std::stof(value) > 1.001f) {
return "Anisotropic texture filtering is disabled because your video "
"card doesn't support GL_EXT_texture_filter_anisotropic.";
} else {
return std::string();
}
}));
AddReport("GL_EXT_texture_filter_anisotropic is NOT SUPPORTED",
MakeVector4(1.f, 1.f, 0.5f, 1.f));
AddReport(" r_maxAnisotropy is disabled.", MakeVector4(1.f, 1.f, 1.f, 0.7f));
}
if (extensions.find("GL_ARB_occlusion_query") == std::string::npos) {
if (r_occlusionQuery) {
r_occlusionQuery = 0;
SPLog("Disabling r_occlusionQuery: no GL_ARB_occlusion_query");
}
incapableConfigs.insert(
std::make_pair("r_occlusionQuery", [](std::string value) -> std::string {
if (std::stoi(value)) {
return "Occlusion query is disabled because your video card doesn't "
"support GL_ARB_occlusion_query.";
} else {
return std::string();
}
}));
AddReport("GL_ARB_occlusion_query is NOT SUPPORTED",
MakeVector4(1.f, 1.f, 0.5f, 1.f));
AddReport(" r_occlusionQuery is disabled.", MakeVector4(1.f, 1.f, 1.f, 0.7f));
}
if (extensions.find("GL_NV_conditional_render") == std::string::npos) {
if (r_occlusionQuery) {
r_occlusionQuery = 0;
SPLog("Disabling r_occlusionQuery: no GL_NV_conditional_render");
}
incapableConfigs.insert(
std::make_pair("r_occlusionQuery", [](std::string value) -> std::string {
if (std::stoi(value)) {
return "Occlusion query is disabled because your video card doesn't "
"support GL_NV_conditional_render.";
} else {
return std::string();
}
}));
AddReport("GL_NV_conditional_render is NOT SUPPORTED",
MakeVector4(1.f, 1.f, 0.5f, 1.f));
AddReport(" r_occlusionQuery is disabled.", MakeVector4(1.f, 1.f, 1.f, 0.7f));
}
if (extensions.find("GL_ARB_color_buffer_float") == std::string::npos) {
if (r_hdr) {
r_hdr = 0;
SPLog("Disabling r_hdr: no GL_ARB_color_buffer_float");
}
incapableConfigs.insert(
std::make_pair("r_hdr", [](std::string value) -> std::string {
if (std::stoi(value)) {
return "HDR Rendering is disabled because your video card doesn't "
"support GL_ARB_color_buffer_float.";
} else {
return std::string();
}
}));
AddReport("GL_ARB_color_buffer_float is NOT SUPPORTED",
MakeVector4(1.f, 1.f, 0.5f, 1.f));
AddReport(" r_hdr is disabled.", MakeVector4(1.f, 1.f, 1.f, 0.7f));
}
if (extensions.find("GL_EXT_texture_array") == std::string::npos) {
if ((int)r_water >= 2) {
r_water = 1;
SPLog("Disabling Water 2: no GL_EXT_texture_array");
}
shaderHighCapable = false;
incapableConfigs.insert(
std::make_pair("r_water", [](std::string value) -> std::string {
if (std::stoi(value) >= 2) {
return "Water 2 is disabled because your video card doesn't "
"support GL_EXT_texture_array.";
} else {
return std::string();
}
}));
AddReport("GL_EXT_texture_array is NOT SUPPORTED",
MakeVector4(1.f, 1.f, 0.5f, 1.f));
AddReport(" Water 2 is disabled.", MakeVector4(1.f, 1.f, 1.f, 0.7f));
}
AddReport("Max Texture Size: " + std::to_string(maxTextureSize),
MakeVector4(1.f, 1.f, 1.f, 0.7f));
if (maxTextureSize < 1024) {
capable = false;
AddReport(" TOO SMALL (1024 required)", MakeVector4(1.f, 0.5f, 0.5f, 1.f));
}
if ((int)r_shadowMapSize > maxTextureSize) {
SPLog("Changed r_shadowMapSize from %d to %d: too small GL_MAX_TEXTURE_SIZE",
(int)r_shadowMapSize, maxTextureSize);
r_shadowMapSize = maxTextureSize;
}
AddReport("Max 3D Texture Size: " + std::to_string(max3DTextureSize),
MakeVector4(1.f, 1.f, 1.f, 0.7f));
if (max3DTextureSize < 512) {
AddReport(" Global Illumination is disabled (512 required)",
MakeVector4(1.f, 1.f, 0.5f, 1.f));
incapableConfigs.insert(
std::make_pair("r_radiosity", [](std::string value) -> std::string {
if (std::stoi(value)) {
return "Global illumination is disabled because your video card "
"doesn't support a 3D texture of at least 512x512x64.";
} else {
return std::string();
}
}));
if (r_radiosity) {
r_radiosity = 0;
SPLog("Disabling r_radiosity: too small GL_MAX_3D_TEXTURE_SIZE");
}
}
AddReport("Max Combined Texture Image Units: " +
std::to_string(maxCombinedTextureUnits),
MakeVector4(1.f, 1.f, 1.f, 0.7f));
if (maxCombinedTextureUnits < 12) {
AddReport(" Global Illumination is disabled (12 required)",
MakeVector4(1.f, 1.f, 0.5f, 1.f));
incapableConfigs.insert(
std::make_pair("r_radiosity", [](std::string value) -> std::string {
if (std::stoi(value)) {
return "Global illumination is disabled because your video card "
"supports too few combined texture image units.";
} else {
return std::string();
}
}));
if (r_radiosity) {
r_radiosity = 0;
SPLog(
"Disabling r_radiosity: too small GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS");
}
}
if (maxCombinedTextureUnits < 15) {
AddReport(" Water 2 is disabled (15 required)",
MakeVector4(1.f, 1.f, 0.5f, 1.f));
shaderHighCapable = false;
incapableConfigs.insert(
std::make_pair("r_water", [](std::string value) -> std::string {
if (std::stoi(value) >= 2) {
return "Water 2 is disabled because your video card supports too few "
"combined texture image units.";
} else {
return std::string();
}
}));
if ((int)r_water >= 2) {
r_water = 1;
SPLog("Disabling Water 2: too small GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS");
}
}
AddReport("Max Vertex Texture Image Units: " +
std::to_string(maxVertexTextureUnits),
MakeVector4(1.f, 1.f, 1.f, 0.7f));
if (maxVertexTextureUnits < 3) {
AddReport(" Water 2 is disabled (3 required)",
MakeVector4(1.f, 1.f, 0.5f, 1.f));
shaderHighCapable = false;
incapableConfigs.insert(
std::make_pair("r_water", [](std::string value) -> std::string {
if (std::stoi(value) >= 2) {
return "Water 2 is disabled because your video card supports too few "
"vertex texture image units.";
} else {
return std::string();
}
}));
if ((int)r_water >= 2) {
r_water = 1;
SPLog("Disabling Water 2: too small GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS");
}
}
AddReport("Max Varying Components: " + std::to_string(maxVaryingComponents),
MakeVector4(1.f, 1.f, 1.f, 0.7f));
if (maxVaryingComponents < 37) {
AddReport(" Shaded Particle is disabled (37 required)",
MakeVector4(1.f, 1.f, 0.5f, 1.f));
particleHighCapable = false;
incapableConfigs.insert(
std::make_pair("r_softParticles", [](std::string value) -> std::string {
if (std::stoi(value) >= 2) {
return "Shaded particle is disabled because your video card supports "
"too few varying fragment shader input components.";
} else {
return std::string();
}
}));
if ((int)r_softParticles >= 2) {
r_softParticles = 1;
SPLog(
"Disabling shaded particle: too small GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS");
}
}
AddReport();
if (capable) {
AddReport("Your video card supports all "
"required OpenGL extensions/features.",
MakeVector4(0.5f, 1.f, 0.5f, 1.f));
} else {
AddReport("Your video card/driver doesn't support "
"at least one of required OpenGL extensions/features."
" Falling back to the software renderer.",
MakeVector4(1.f, 0.5f, 0.5f, 1.f));
}
SDL_GL_DeleteContext(context);
SDL_DestroyWindow(window);
SPLog("SDL Capability Query Window finalized");
}
SPLog("OpenGL driver is OpenSpades capable: %s", capable ? "YES" : "NO");
openGLCapable = capable;
if (!openGLCapable) {
if (spades::EqualsIgnoringCase(r_renderer, "gl")) {
SPLog("Switched to software renderer");
r_renderer = "sw";
}
incapableConfigs.insert(
std::make_pair("r_renderer", [](std::string value) -> std::string {
if (spades::EqualsIgnoringCase(value, "gl")) {
return "OpenGL renderer is disabled because "
"your video card/driver doesn't support "
"at least one of required OpenGL extensions/features.";
} else {
return std::string();
}
}));
}
}
void StartupScreenHelper::FixConfigs() { ExamineSystem(); }
std::string StartupScreenHelper::GetOperatingSystemType() {
#if defined(OS_PLATFORM_LINUX)
return "Linux";
#elif defined(TARGET_OS_MAC)
return "Mac";
#elif defined(OS_PLATFORM_WINDOWS)
return "Windows";
#elif defined(__FreeBSD__)
return "FreeBSD";
#elif defined(__OpenBSD__)
return "OpenBSD";
#else
return std::string{};
#endif
}
PackageUpdateManager& StartupScreenHelper::GetPackageUpdateManager() {
return PackageUpdateManager::GetInstance();
}
bool StartupScreenHelper::OpenUpdateInfoURL() {
std::string url = GetPackageUpdateManager().GetLatestVersionInfoPageURL();
if (url.find("http:") != 0 && url.find("https:") != 0) {
return false;
}
return OpenURLInBrowser(url);
}
bool StartupScreenHelper::BrowseUserDirectory() {
std::string path = g_userResourceDirectory;
if (path.empty()) {
SPLog("Cannot open the user resource directory: g_userResourceDirectory is empty.");
return false;
}
return ShowDirectoryInShell(path);
}
void StartupScreenHelper::Start() {
if (scr == nullptr) {
return;
}
scr->Start();
}
int StartupScreenHelper::GetNumVideoModes() { return static_cast<int>(modes.size()); }
int StartupScreenHelper::GetVideoModeWidth(int index) {
if (index < 0 || index >= GetNumVideoModes())
SPInvalidArgument("index");
return modes[index].x;
}
int StartupScreenHelper::GetVideoModeHeight(int index) {
if (index < 0 || index >= GetNumVideoModes())
SPInvalidArgument("index");
return modes[index].y;
}
int StartupScreenHelper::GetNumReportLines() {
return static_cast<int>(reportLines.size());
}
std::string StartupScreenHelper::GetReportLineText(int index) {
if (index < 0 || index >= GetNumReportLines())
SPInvalidArgument("index");
return reportLines[index].text;
}
Vector4 StartupScreenHelper::GetReportLineColor(int index) {
if (index < 0 || index >= GetNumReportLines())
SPInvalidArgument("index");
return reportLines[index].color;
}
void StartupScreenHelper::AddReport(const std::string &text, Vector4 color) {
ReportLine l = {text, color};
reportLines.push_back(l);
report += text;
report += '\n';
}
int StartupScreenHelper::GetNumLocales() { return static_cast<int>(locales.size()); }
std::string StartupScreenHelper::GetLocale(int index) {
if (index < 0 || index >= GetNumLocales())
SPInvalidArgument("index");
return locales[index].name;
}
std::string StartupScreenHelper::GetLocaleDescriptionNative(int index) {
if (index < 0 || index >= GetNumLocales())
SPInvalidArgument("index");
return locales[index].descriptionNative;
}
std::string StartupScreenHelper::GetLocaleDescriptionEnglish(int index) {
if (index < 0 || index >= GetNumLocales())
SPInvalidArgument("index");
return locales[index].descriptionEnglish;
}
std::string StartupScreenHelper::CheckConfigCapability(const std::string &cfg,
const std::string &value) {
auto range = incapableConfigs.equal_range(cfg);
std::string ret;
bool hasMulti = false;
for (auto it = range.first; it != range.second; it++) {
auto &f = it->second;
auto err = f(value);
if (err.size() > 0) {
if (ret.size() == 0) {
ret = err;
} else {
if (!hasMulti) {
ret = "- " + ret;
hasMulti = true;
}
ret += "\n- " + err;
}
}
}
return ret;
}
}
}