2013-09-01 14:27:21 +00:00
|
|
|
/*
|
|
|
|
** config.cpp
|
|
|
|
**
|
|
|
|
** This file is part of mkxp.
|
|
|
|
**
|
2021-09-06 18:50:44 +00:00
|
|
|
** Copyright (C) 2021 Amaryllis Kulla <amaryllis.kulla@protonmail.com>
|
2013-09-01 14:27:21 +00:00
|
|
|
**
|
|
|
|
** mkxp 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 2 of the License, or
|
|
|
|
** (at your option) any later version.
|
|
|
|
**
|
|
|
|
** mkxp 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 mkxp. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
2013-12-11 19:46:54 +00:00
|
|
|
#include <boost/program_options/options_description.hpp>
|
|
|
|
#include <boost/program_options/parsers.hpp>
|
|
|
|
#include <boost/program_options/variables_map.hpp>
|
2014-01-01 02:03:53 +00:00
|
|
|
|
2014-01-25 08:24:55 +00:00
|
|
|
#include <SDL_filesystem.h>
|
|
|
|
|
2013-12-11 19:46:54 +00:00
|
|
|
#include <fstream>
|
2014-08-20 03:20:07 +00:00
|
|
|
#include <stdint.h>
|
2013-12-11 19:46:54 +00:00
|
|
|
|
|
|
|
#include "debugwriter.h"
|
|
|
|
#include "util.h"
|
2014-12-24 04:30:43 +00:00
|
|
|
#include "sdl-util.h"
|
2018-09-25 17:58:10 +00:00
|
|
|
#include "iniconfig.h"
|
2013-12-11 19:46:54 +00:00
|
|
|
|
2014-08-20 03:20:07 +00:00
|
|
|
#ifdef INI_ENCODING
|
|
|
|
extern "C" {
|
|
|
|
#include <libguess.h>
|
|
|
|
}
|
|
|
|
#include <iconv.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* http://stackoverflow.com/a/1031773 */
|
|
|
|
static bool validUtf8(const char *string)
|
|
|
|
{
|
|
|
|
const uint8_t *bytes = (uint8_t*) string;
|
|
|
|
|
|
|
|
while(*bytes)
|
|
|
|
{
|
|
|
|
if( (/* ASCII
|
|
|
|
* use bytes[0] <= 0x7F to allow ASCII control characters */
|
|
|
|
bytes[0] == 0x09 ||
|
|
|
|
bytes[0] == 0x0A ||
|
|
|
|
bytes[0] == 0x0D ||
|
|
|
|
(0x20 <= bytes[0] && bytes[0] <= 0x7E)
|
|
|
|
)
|
|
|
|
) {
|
|
|
|
bytes += 1;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( (/* non-overlong 2-byte */
|
|
|
|
(0xC2 <= bytes[0] && bytes[0] <= 0xDF) &&
|
|
|
|
(0x80 <= bytes[1] && bytes[1] <= 0xBF)
|
|
|
|
)
|
|
|
|
) {
|
|
|
|
bytes += 2;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( (/* excluding overlongs */
|
|
|
|
bytes[0] == 0xE0 &&
|
|
|
|
(0xA0 <= bytes[1] && bytes[1] <= 0xBF) &&
|
|
|
|
(0x80 <= bytes[2] && bytes[2] <= 0xBF)
|
|
|
|
) ||
|
|
|
|
(/* straight 3-byte */
|
|
|
|
((0xE1 <= bytes[0] && bytes[0] <= 0xEC) ||
|
|
|
|
bytes[0] == 0xEE ||
|
|
|
|
bytes[0] == 0xEF) &&
|
|
|
|
(0x80 <= bytes[1] && bytes[1] <= 0xBF) &&
|
|
|
|
(0x80 <= bytes[2] && bytes[2] <= 0xBF)
|
|
|
|
) ||
|
|
|
|
(/* excluding surrogates */
|
|
|
|
bytes[0] == 0xED &&
|
|
|
|
(0x80 <= bytes[1] && bytes[1] <= 0x9F) &&
|
|
|
|
(0x80 <= bytes[2] && bytes[2] <= 0xBF)
|
|
|
|
)
|
|
|
|
) {
|
|
|
|
bytes += 3;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( (/* planes 1-3 */
|
|
|
|
bytes[0] == 0xF0 &&
|
|
|
|
(0x90 <= bytes[1] && bytes[1] <= 0xBF) &&
|
|
|
|
(0x80 <= bytes[2] && bytes[2] <= 0xBF) &&
|
|
|
|
(0x80 <= bytes[3] && bytes[3] <= 0xBF)
|
|
|
|
) ||
|
|
|
|
(/* planes 4-15 */
|
|
|
|
(0xF1 <= bytes[0] && bytes[0] <= 0xF3) &&
|
|
|
|
(0x80 <= bytes[1] && bytes[1] <= 0xBF) &&
|
|
|
|
(0x80 <= bytes[2] && bytes[2] <= 0xBF) &&
|
|
|
|
(0x80 <= bytes[3] && bytes[3] <= 0xBF)
|
|
|
|
) ||
|
|
|
|
(/* plane 16 */
|
|
|
|
bytes[0] == 0xF4 &&
|
|
|
|
(0x80 <= bytes[1] && bytes[1] <= 0x8F) &&
|
|
|
|
(0x80 <= bytes[2] && bytes[2] <= 0xBF) &&
|
|
|
|
(0x80 <= bytes[3] && bytes[3] <= 0xBF)
|
|
|
|
)
|
|
|
|
) {
|
|
|
|
bytes += 4;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-01-25 08:24:55 +00:00
|
|
|
static std::string prefPath(const char *org, const char *app)
|
|
|
|
{
|
|
|
|
char *path = SDL_GetPrefPath(org, app);
|
2014-12-23 19:19:28 +00:00
|
|
|
|
|
|
|
if (!path)
|
|
|
|
return std::string();
|
|
|
|
|
2014-01-25 08:24:55 +00:00
|
|
|
std::string str(path);
|
|
|
|
SDL_free(path);
|
|
|
|
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2017-03-03 18:37:19 +00:00
|
|
|
template<typename T>
|
|
|
|
std::set<T> setFromVec(const std::vector<T> &vec)
|
|
|
|
{
|
|
|
|
return std::set<T>(vec.begin(), vec.end());
|
|
|
|
}
|
|
|
|
|
2013-12-11 19:46:54 +00:00
|
|
|
typedef std::vector<std::string> StringVec;
|
|
|
|
namespace po = boost::program_options;
|
2013-09-01 14:27:21 +00:00
|
|
|
|
2014-09-23 17:23:11 +00:00
|
|
|
#define CONF_FILE "mkxp.conf"
|
|
|
|
|
2013-09-01 14:27:21 +00:00
|
|
|
Config::Config()
|
2015-01-03 18:28:23 +00:00
|
|
|
{}
|
2013-09-01 14:27:21 +00:00
|
|
|
|
2014-01-10 19:16:34 +00:00
|
|
|
void Config::read(int argc, char *argv[])
|
2013-09-01 14:27:21 +00:00
|
|
|
{
|
2013-12-11 19:46:54 +00:00
|
|
|
#define PO_DESC_ALL \
|
2015-01-03 18:28:23 +00:00
|
|
|
PO_DESC(rgssVersion, int, 0) \
|
|
|
|
PO_DESC(debugMode, bool, false) \
|
|
|
|
PO_DESC(printFPS, bool, false) \
|
|
|
|
PO_DESC(winResizable, bool, false) \
|
|
|
|
PO_DESC(fullscreen, bool, false) \
|
|
|
|
PO_DESC(fixedAspectRatio, bool, true) \
|
2015-07-16 16:41:48 +00:00
|
|
|
PO_DESC(smoothScaling, bool, true) \
|
2015-01-03 18:28:23 +00:00
|
|
|
PO_DESC(vsync, bool, false) \
|
|
|
|
PO_DESC(defScreenW, int, 0) \
|
|
|
|
PO_DESC(defScreenH, int, 0) \
|
2017-12-10 23:48:35 +00:00
|
|
|
PO_DESC(windowTitle, std::string, "") \
|
2015-01-03 18:28:23 +00:00
|
|
|
PO_DESC(fixedFramerate, int, 0) \
|
|
|
|
PO_DESC(frameSkip, bool, true) \
|
|
|
|
PO_DESC(syncToRefreshrate, bool, false) \
|
|
|
|
PO_DESC(solidFonts, bool, false) \
|
|
|
|
PO_DESC(subImageFix, bool, false) \
|
2017-04-23 12:32:11 +00:00
|
|
|
PO_DESC(enableBlitting, bool, true) \
|
2017-04-23 10:28:34 +00:00
|
|
|
PO_DESC(maxTextureSize, int, 0) \
|
2021-12-07 06:10:45 +00:00
|
|
|
PO_DESC(integerScaling.active, bool, false) \
|
|
|
|
PO_DESC(integerScaling.lastMileScaling, bool, true) \
|
2015-01-03 18:28:23 +00:00
|
|
|
PO_DESC(gameFolder, std::string, ".") \
|
|
|
|
PO_DESC(anyAltToggleFS, bool, false) \
|
|
|
|
PO_DESC(enableReset, bool, true) \
|
|
|
|
PO_DESC(allowSymlinks, bool, false) \
|
|
|
|
PO_DESC(dataPathOrg, std::string, "") \
|
|
|
|
PO_DESC(dataPathApp, std::string, "") \
|
|
|
|
PO_DESC(iconPath, std::string, "") \
|
2015-02-19 01:23:23 +00:00
|
|
|
PO_DESC(execName, std::string, "Game") \
|
2015-01-03 18:28:23 +00:00
|
|
|
PO_DESC(titleLanguage, std::string, "") \
|
|
|
|
PO_DESC(midi.soundFont, std::string, "") \
|
|
|
|
PO_DESC(midi.chorus, bool, false) \
|
|
|
|
PO_DESC(midi.reverb, bool, false) \
|
|
|
|
PO_DESC(SE.sourceCount, int, 6) \
|
2021-09-29 22:28:37 +00:00
|
|
|
PO_DESC(volume.bgs, float, 1.0f) \
|
|
|
|
PO_DESC(volume.se, float, 1.0f) \
|
2015-01-03 18:28:23 +00:00
|
|
|
PO_DESC(customScript, std::string, "") \
|
|
|
|
PO_DESC(pathCache, bool, true) \
|
|
|
|
PO_DESC(useScriptNames, bool, false)
|
2013-09-03 09:07:56 +00:00
|
|
|
|
2014-01-01 02:03:53 +00:00
|
|
|
// Not gonna take your shit boost
|
|
|
|
#define GUARD_ALL( exp ) try { exp } catch(...) {}
|
|
|
|
|
2018-02-22 07:37:47 +00:00
|
|
|
editor.debug = false;
|
|
|
|
editor.battleTest = false;
|
|
|
|
|
|
|
|
/* Read arguments sent from the editor */
|
|
|
|
if (argc > 1)
|
|
|
|
{
|
|
|
|
std::string argv1 = argv[1];
|
|
|
|
/* RGSS1 uses "debug", 2 and 3 use "test" */
|
|
|
|
if (argv1 == "debug" || argv1 == "test")
|
|
|
|
editor.debug = true;
|
|
|
|
else if (argv1 == "btest")
|
|
|
|
editor.battleTest = true;
|
|
|
|
|
|
|
|
/* Fix offset */
|
|
|
|
if (editor.debug || editor.battleTest)
|
|
|
|
{
|
|
|
|
argc--;
|
|
|
|
argv++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-03 18:28:23 +00:00
|
|
|
#define PO_DESC(key, type, def) (#key, po::value< type >()->default_value(def))
|
2013-09-03 09:07:56 +00:00
|
|
|
|
2013-12-11 19:46:54 +00:00
|
|
|
po::options_description podesc;
|
|
|
|
podesc.add_options()
|
|
|
|
PO_DESC_ALL
|
2014-08-24 05:27:36 +00:00
|
|
|
("preloadScript", po::value<StringVec>()->composing())
|
2014-01-10 19:16:34 +00:00
|
|
|
("RTP", po::value<StringVec>()->composing())
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 11:37:14 +00:00
|
|
|
("fontSub", po::value<StringVec>()->composing())
|
2014-07-24 21:11:12 +00:00
|
|
|
("rubyLoadpath", po::value<StringVec>()->composing())
|
2013-12-11 19:46:54 +00:00
|
|
|
;
|
|
|
|
|
2014-01-10 19:16:34 +00:00
|
|
|
po::variables_map vm;
|
|
|
|
|
|
|
|
/* Parse command line options */
|
2014-09-23 17:23:11 +00:00
|
|
|
try
|
|
|
|
{
|
|
|
|
po::parsed_options cmdPo =
|
|
|
|
po::command_line_parser(argc, argv).options(podesc).run();
|
|
|
|
po::store(cmdPo, vm);
|
|
|
|
}
|
|
|
|
catch (po::error &error)
|
|
|
|
{
|
|
|
|
Debug() << "Command line:" << error.what();
|
|
|
|
}
|
2014-01-10 19:16:34 +00:00
|
|
|
|
2014-09-23 17:23:11 +00:00
|
|
|
/* Parse configuration file */
|
2014-12-24 04:30:43 +00:00
|
|
|
SDLRWStream confFile(CONF_FILE, "r");
|
2013-12-11 19:46:54 +00:00
|
|
|
|
2014-01-01 16:01:05 +00:00
|
|
|
if (confFile)
|
|
|
|
{
|
2014-09-23 17:23:11 +00:00
|
|
|
try
|
|
|
|
{
|
2014-12-24 04:30:43 +00:00
|
|
|
po::store(po::parse_config_file(confFile.stream(), podesc, true), vm);
|
2014-09-23 17:23:11 +00:00
|
|
|
po::notify(vm);
|
|
|
|
}
|
|
|
|
catch (po::error &error)
|
|
|
|
{
|
|
|
|
Debug() << CONF_FILE":" << error.what();
|
|
|
|
}
|
|
|
|
}
|
2013-12-11 19:46:54 +00:00
|
|
|
|
|
|
|
#undef PO_DESC
|
2015-01-03 18:28:23 +00:00
|
|
|
#define PO_DESC(key, type, def) GUARD_ALL( key = vm[#key].as< type >(); )
|
2013-12-11 19:46:54 +00:00
|
|
|
|
|
|
|
PO_DESC_ALL;
|
2013-09-03 09:07:56 +00:00
|
|
|
|
2017-03-03 18:37:19 +00:00
|
|
|
GUARD_ALL( preloadScripts = setFromVec(vm["preloadScript"].as<StringVec>()); );
|
2014-08-24 05:27:36 +00:00
|
|
|
|
2013-12-11 19:46:54 +00:00
|
|
|
GUARD_ALL( rtps = vm["RTP"].as<StringVec>(); );
|
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 11:37:14 +00:00
|
|
|
GUARD_ALL( fontSubs = vm["fontSub"].as<StringVec>(); );
|
|
|
|
|
2014-08-28 21:11:10 +00:00
|
|
|
GUARD_ALL( rubyLoadpaths = vm["rubyLoadpath"].as<StringVec>(); );
|
2014-07-24 21:11:12 +00:00
|
|
|
|
2013-12-11 19:46:54 +00:00
|
|
|
#undef PO_DESC
|
|
|
|
#undef PO_DESC_ALL
|
2014-08-25 03:28:46 +00:00
|
|
|
|
2014-08-29 07:42:13 +00:00
|
|
|
rgssVersion = clamp(rgssVersion, 0, 3);
|
|
|
|
|
2014-08-25 03:28:46 +00:00
|
|
|
SE.sourceCount = clamp(SE.sourceCount, 1, 64);
|
2014-01-25 08:24:55 +00:00
|
|
|
|
|
|
|
if (!dataPathOrg.empty() && !dataPathApp.empty())
|
|
|
|
customDataPath = prefPath(dataPathOrg.c_str(), dataPathApp.c_str());
|
|
|
|
|
|
|
|
commonDataPath = prefPath(".", "mkxp");
|
2014-11-02 23:23:37 +00:00
|
|
|
|
|
|
|
/* parse cropped textures config */
|
|
|
|
std::string cropTexFile;
|
|
|
|
if (!readFile("croptextures", cropTexFile))
|
|
|
|
{
|
|
|
|
Debug() << "Couldn't read croptextures config file";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t lineStart = 0;
|
|
|
|
bool atEnd = false;
|
|
|
|
|
|
|
|
while (!atEnd)
|
|
|
|
{
|
|
|
|
CropTexture tex;
|
|
|
|
|
|
|
|
size_t fnEnd = cropTexFile.find('"', lineStart+1);
|
|
|
|
size_t numSep = cropTexFile.find(' ', fnEnd+2);
|
|
|
|
size_t lineEnd = cropTexFile.find('\n', numSep);
|
|
|
|
|
|
|
|
if (fnEnd == std::string::npos || numSep == std::string::npos)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (lineEnd == std::string::npos)
|
|
|
|
{
|
|
|
|
lineEnd = cropTexFile.size();
|
|
|
|
atEnd = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
tex.filename = cropTexFile.substr(lineStart+1, fnEnd-(lineStart+1));
|
|
|
|
tex.w = atoi(cropTexFile.substr(fnEnd+2, numSep-(fnEnd+2)).c_str());
|
|
|
|
tex.h = atoi(cropTexFile.substr(numSep+1, lineEnd-(numSep+1)).c_str());
|
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
Debug() << "Couldn't parse croptextures config file";
|
|
|
|
cropTexs.clear();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
cropTexs.push_back(tex);
|
|
|
|
Debug() << "Parsed:" << tex.filename << tex.w << tex.h;
|
|
|
|
|
|
|
|
lineStart = lineEnd+1;
|
|
|
|
}
|
2013-09-01 14:27:21 +00:00
|
|
|
}
|
|
|
|
|
2014-01-05 09:31:15 +00:00
|
|
|
static std::string baseName(const std::string &path)
|
|
|
|
{
|
2014-01-06 17:45:01 +00:00
|
|
|
size_t pos = path.find_last_of("/\\");
|
2014-01-05 09:31:15 +00:00
|
|
|
|
|
|
|
if (pos == path.npos)
|
2014-01-06 17:45:01 +00:00
|
|
|
return path;
|
2014-01-05 09:31:15 +00:00
|
|
|
|
2014-01-06 17:45:01 +00:00
|
|
|
return path.substr(pos + 1);
|
2014-01-05 09:31:15 +00:00
|
|
|
}
|
|
|
|
|
2014-09-05 19:54:16 +00:00
|
|
|
static void setupScreenSize(Config &conf)
|
|
|
|
{
|
|
|
|
if (conf.defScreenW <= 0)
|
|
|
|
conf.defScreenW = (conf.rgssVersion == 1 ? 640 : 544);
|
|
|
|
|
|
|
|
if (conf.defScreenH <= 0)
|
|
|
|
conf.defScreenH = (conf.rgssVersion == 1 ? 480 : 416);
|
|
|
|
}
|
|
|
|
|
2013-09-01 14:27:21 +00:00
|
|
|
void Config::readGameINI()
|
|
|
|
{
|
2013-12-11 19:46:54 +00:00
|
|
|
if (!customScript.empty())
|
2013-09-01 14:27:21 +00:00
|
|
|
{
|
2014-01-05 09:31:15 +00:00
|
|
|
game.title = baseName(customScript);
|
2014-09-05 19:54:16 +00:00
|
|
|
|
|
|
|
if (rgssVersion == 0)
|
|
|
|
rgssVersion = 1;
|
|
|
|
|
|
|
|
setupScreenSize(*this);
|
|
|
|
|
2013-09-01 14:27:21 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-02-19 01:23:23 +00:00
|
|
|
std::string iniFilename = execName + ".ini";
|
|
|
|
SDLRWStream iniFile(iniFilename.c_str(), "r");
|
2013-12-11 19:46:54 +00:00
|
|
|
|
2014-12-24 04:30:43 +00:00
|
|
|
if (iniFile)
|
|
|
|
{
|
2018-09-25 17:58:10 +00:00
|
|
|
INIConfiguration ic;
|
|
|
|
if(ic.load(iniFile.stream()))
|
2014-12-24 04:30:43 +00:00
|
|
|
{
|
2018-09-25 17:58:10 +00:00
|
|
|
GUARD_ALL( game.title = ic.getStringProperty("Game", "Title"); );
|
|
|
|
GUARD_ALL( game.scripts = ic.getStringProperty("Game", "Scripts"); );
|
|
|
|
|
|
|
|
strReplace(game.scripts, '\\', '/');
|
|
|
|
|
|
|
|
if (game.title.empty())
|
|
|
|
{
|
|
|
|
Debug() << iniFilename + ": Could not find Game.Title property";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (game.scripts.empty())
|
|
|
|
{
|
|
|
|
Debug() << iniFilename + ": Could not find Game.Scripts property";
|
|
|
|
}
|
2014-12-24 04:30:43 +00:00
|
|
|
}
|
2018-09-25 17:58:10 +00:00
|
|
|
else
|
2014-12-24 04:30:43 +00:00
|
|
|
{
|
2018-09-25 17:58:10 +00:00
|
|
|
Debug() << iniFilename + ": Failed to parse ini file";
|
2014-12-24 04:30:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-02-19 01:23:23 +00:00
|
|
|
Debug() << "FAILED to open" << iniFilename;
|
2014-12-24 04:30:43 +00:00
|
|
|
}
|
2013-12-11 19:46:54 +00:00
|
|
|
|
2014-08-20 03:20:07 +00:00
|
|
|
#ifdef INI_ENCODING
|
|
|
|
/* Can add more later */
|
|
|
|
const char *languages[] =
|
|
|
|
{
|
|
|
|
titleLanguage.c_str(),
|
|
|
|
GUESS_REGION_JP, /* Japanese */
|
|
|
|
GUESS_REGION_KR, /* Korean */
|
|
|
|
GUESS_REGION_CN, /* Chinese */
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
bool convSuccess = true;
|
|
|
|
|
|
|
|
/* Verify that the game title is UTF-8, and if not,
|
|
|
|
* try to determine the encoding and convert to UTF-8 */
|
|
|
|
if (!validUtf8(game.title.c_str()))
|
|
|
|
{
|
|
|
|
const char *encoding = 0;
|
|
|
|
convSuccess = false;
|
|
|
|
|
|
|
|
for (size_t i = 0; languages[i]; ++i)
|
|
|
|
{
|
|
|
|
encoding = libguess_determine_encoding(game.title.c_str(),
|
|
|
|
game.title.size(),
|
|
|
|
languages[i]);
|
|
|
|
if (encoding)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (encoding)
|
|
|
|
{
|
|
|
|
iconv_t cd = iconv_open("UTF-8", encoding);
|
|
|
|
|
|
|
|
size_t inLen = game.title.size();
|
|
|
|
size_t outLen = inLen * 4;
|
|
|
|
std::string buf(outLen, '\0');
|
|
|
|
char *inPtr = const_cast<char*>(game.title.c_str());
|
|
|
|
char *outPtr = const_cast<char*>(buf.c_str());
|
|
|
|
|
|
|
|
errno = 0;
|
|
|
|
size_t result = iconv(cd, &inPtr, &inLen, &outPtr, &outLen);
|
|
|
|
|
|
|
|
iconv_close(cd);
|
|
|
|
|
|
|
|
if (result != (size_t) -1 && errno == 0)
|
|
|
|
{
|
|
|
|
buf.resize(buf.size()-outLen);
|
|
|
|
game.title = buf;
|
|
|
|
convSuccess = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!convSuccess)
|
|
|
|
game.title.clear();
|
|
|
|
#else
|
|
|
|
if (!validUtf8(game.title.c_str()))
|
|
|
|
game.title.clear();
|
|
|
|
#endif
|
|
|
|
|
2013-12-11 19:46:54 +00:00
|
|
|
if (game.title.empty())
|
2014-01-05 09:31:15 +00:00
|
|
|
game.title = baseName(gameFolder);
|
2014-08-28 21:11:10 +00:00
|
|
|
|
|
|
|
if (rgssVersion == 0)
|
|
|
|
{
|
|
|
|
/* Try to guess RGSS version based on Data/Scripts extension */
|
|
|
|
rgssVersion = 1;
|
|
|
|
|
|
|
|
if (!game.scripts.empty())
|
|
|
|
{
|
|
|
|
const char *p = &game.scripts[game.scripts.size()];
|
|
|
|
const char *head = &game.scripts[0];
|
|
|
|
|
|
|
|
while (--p != head)
|
|
|
|
if (*p == '.')
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (!strcmp(p, ".rvdata"))
|
|
|
|
rgssVersion = 2;
|
|
|
|
else if (!strcmp(p, ".rvdata2"))
|
|
|
|
rgssVersion = 3;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-05 19:54:16 +00:00
|
|
|
setupScreenSize(*this);
|
2013-09-01 14:27:21 +00:00
|
|
|
}
|