299 lines
7.3 KiB
C++
299 lines
7.3 KiB
C++
/******************************************************************************
|
|
Copyright (C) 2013 by Hugh Bailey <obs.jim@gmail.com>
|
|
|
|
This program 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.
|
|
|
|
This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
|
|
******************************************************************************/
|
|
|
|
#include <sstream>
|
|
#include <util/bmem.h>
|
|
#include <util/dstr.h>
|
|
#include <util/platform.h>
|
|
#include <obs.hpp>
|
|
|
|
#include <QProxyStyle>
|
|
|
|
#include "qt-wrappers.hpp"
|
|
#include "obs-app.hpp"
|
|
#include "window-basic-main.hpp"
|
|
#include "platform.hpp"
|
|
|
|
#ifdef _WIN32
|
|
#include <windows.h>
|
|
#endif
|
|
|
|
using namespace std;
|
|
|
|
static void do_log(enum log_type type, const char *msg, va_list args)
|
|
{
|
|
#ifdef _WIN32
|
|
char bla[4096];
|
|
vsnprintf(bla, 4095, msg, args);
|
|
|
|
OutputDebugStringA(bla);
|
|
OutputDebugStringA("\n");
|
|
|
|
if (type >= LOG_WARNING && IsDebuggerPresent())
|
|
__debugbreak();
|
|
#else
|
|
vprintf(msg, args);
|
|
printf("\n");
|
|
#endif
|
|
|
|
UNUSED_PARAMETER(type);
|
|
}
|
|
|
|
bool OBSApp::InitGlobalConfigDefaults()
|
|
{
|
|
config_set_default_string(globalConfig, "General", "Language", "en");
|
|
config_set_default_int(globalConfig, "Window", "PosX", -1);
|
|
config_set_default_int(globalConfig, "Window", "PosY", -1);
|
|
config_set_default_int(globalConfig, "Window", "SizeX", -1);
|
|
config_set_default_int(globalConfig, "Window", "SizeY", -1);
|
|
|
|
vector<MonitorInfo> monitors;
|
|
GetMonitors(monitors);
|
|
|
|
if (!monitors.size()) {
|
|
OBSErrorBox(NULL, "There appears to be no monitors. Er, this "
|
|
"technically shouldn't be possible.");
|
|
return false;
|
|
}
|
|
|
|
uint32_t cx = monitors[0].cx;
|
|
uint32_t cy = monitors[0].cy;
|
|
|
|
#if _WIN32
|
|
config_set_default_string(globalConfig, "Video", "Renderer",
|
|
"Direct3D 11");
|
|
#else
|
|
config_set_default_string(globalConfig, "Video", "Renderer",
|
|
"OpenGL");
|
|
#endif
|
|
|
|
config_set_default_uint(globalConfig, "Video", "BaseCX", cx);
|
|
config_set_default_uint(globalConfig, "Video", "BaseCY", cy);
|
|
|
|
cx = cx * 10 / 15;
|
|
cy = cy * 10 / 15;
|
|
config_set_default_uint(globalConfig, "Video", "OutputCX", cx);
|
|
config_set_default_uint(globalConfig, "Video", "OutputCY", cy);
|
|
|
|
config_set_default_uint(globalConfig, "Video", "FPSType", 0);
|
|
config_set_default_string(globalConfig, "Video", "FPSCommon", "30");
|
|
config_set_default_uint(globalConfig, "Video", "FPSInt", 30);
|
|
config_set_default_uint(globalConfig, "Video", "FPSNum", 30);
|
|
config_set_default_uint(globalConfig, "Video", "FPSDen", 1);
|
|
config_set_default_uint(globalConfig, "Video", "FPSNS", 33333333);
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool do_mkdir(const char *path)
|
|
{
|
|
if (os_mkdir(path) == MKDIR_ERROR) {
|
|
OBSErrorBox(NULL, "Failed to create directory %s", path);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool MakeUserDirs()
|
|
{
|
|
BPtr<char> configPath(os_get_config_path("obs-studio"));
|
|
return do_mkdir(configPath);
|
|
}
|
|
|
|
bool OBSApp::InitGlobalConfig()
|
|
{
|
|
BPtr<char> path(os_get_config_path("obs-studio/global.ini"));
|
|
|
|
int errorcode = globalConfig.Open(path, CONFIG_OPEN_ALWAYS);
|
|
if (errorcode != CONFIG_SUCCESS) {
|
|
OBSErrorBox(NULL, "Failed to open global.ini: %d", errorcode);
|
|
return false;
|
|
}
|
|
|
|
return InitGlobalConfigDefaults();
|
|
}
|
|
|
|
#define DEFAULT_LANG "en"
|
|
|
|
bool OBSApp::InitLocale()
|
|
{
|
|
const char *lang = config_get_string(globalConfig, "General",
|
|
"Language");
|
|
|
|
locale = lang;
|
|
|
|
stringstream file;
|
|
file << "locale/" << lang << ".txt";
|
|
|
|
string englishPath;
|
|
if (!GetDataFilePath("locale/" DEFAULT_LANG ".txt", englishPath)) {
|
|
OBSErrorBox(NULL, "Failed to find locale/" DEFAULT_LANG ".txt");
|
|
return false;
|
|
}
|
|
|
|
textLookup = text_lookup_create(englishPath.c_str());
|
|
if (!textLookup) {
|
|
OBSErrorBox(NULL, "Failed to create locale from file '%s'",
|
|
englishPath.c_str());
|
|
return false;
|
|
}
|
|
|
|
if (astrcmpi(lang, DEFAULT_LANG) == 0)
|
|
return true;
|
|
|
|
string path;
|
|
if (GetDataFilePath(file.str().c_str(), path)) {
|
|
if (!text_lookup_add(textLookup, path.c_str()))
|
|
blog(LOG_WARNING, "Failed to add locale file '%s'",
|
|
path.c_str());
|
|
} else {
|
|
blog(LOG_WARNING, "Could not find locale file '%s'",
|
|
file.str().c_str());
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
OBSApp::OBSApp(int &argc, char **argv)
|
|
: QApplication(argc, argv)
|
|
{
|
|
if (!InitApplicationBundle())
|
|
throw "Failed to initialize application bundle";
|
|
if (!MakeUserDirs())
|
|
throw "Failed to created required user directories";
|
|
if (!InitGlobalConfig())
|
|
throw "Failed to initialize global config";
|
|
if (!InitLocale())
|
|
throw "Failed to load locale";
|
|
|
|
mainWindow = move(unique_ptr<OBSBasic>(new OBSBasic()));
|
|
}
|
|
|
|
void OBSApp::GetFPSCommon(uint32_t &num, uint32_t &den) const
|
|
{
|
|
const char *val = config_get_string(globalConfig, "Video", "FPSCommon");
|
|
|
|
if (strcmp(val, "10") == 0) {
|
|
num = 30;
|
|
den = 1;
|
|
} else if (strcmp(val, "20") == 0) {
|
|
num = 20;
|
|
den = 1;
|
|
} else if (strcmp(val, "25") == 0) {
|
|
num = 25;
|
|
den = 1;
|
|
} else if (strcmp(val, "29.97") == 0) {
|
|
num = 30000;
|
|
den = 1001;
|
|
} else if (strcmp(val, "48") == 0) {
|
|
num = 48;
|
|
den = 1;
|
|
} else if (strcmp(val, "59.94") == 0) {
|
|
num = 60000;
|
|
den = 1001;
|
|
} else if (strcmp(val, "60") == 0) {
|
|
num = 60;
|
|
den = 1;
|
|
} else {
|
|
num = 30;
|
|
den = 1;
|
|
}
|
|
}
|
|
|
|
void OBSApp::GetFPSInteger(uint32_t &num, uint32_t &den) const
|
|
{
|
|
num = (uint32_t)config_get_uint(globalConfig, "Video", "FPSInt");
|
|
den = 1;
|
|
}
|
|
|
|
void OBSApp::GetFPSFraction(uint32_t &num, uint32_t &den) const
|
|
{
|
|
num = (uint32_t)config_get_uint(globalConfig, "Video", "FPSNum");
|
|
den = (uint32_t)config_get_uint(globalConfig, "Video", "FPSDen");
|
|
}
|
|
|
|
void OBSApp::GetFPSNanoseconds(uint32_t &num, uint32_t &den) const
|
|
{
|
|
num = 1000000000;
|
|
den = (uint32_t)config_get_uint(globalConfig, "Video", "FPSNS");
|
|
}
|
|
|
|
void OBSApp::GetConfigFPS(uint32_t &num, uint32_t &den) const
|
|
{
|
|
uint32_t type = config_get_uint(globalConfig, "Video", "FPSType");
|
|
|
|
if (type == 1) //"Integer"
|
|
GetFPSInteger(num, den);
|
|
else if (type == 2) //"Fraction"
|
|
GetFPSFraction(num, den);
|
|
else if (false) //"Nanoseconds", currently not implemented
|
|
GetFPSNanoseconds(num, den);
|
|
else
|
|
GetFPSCommon(num, den);
|
|
}
|
|
|
|
const char *OBSApp::GetRenderModule() const
|
|
{
|
|
const char *renderer = config_get_string(globalConfig, "Video",
|
|
"Renderer");
|
|
|
|
if (astrcmpi(renderer, "Direct3D 11") == 0)
|
|
return "libobs-d3d11";
|
|
else
|
|
return "libobs-opengl";
|
|
}
|
|
|
|
void OBSApp::OBSInit()
|
|
{
|
|
mainWindow->OBSInit();
|
|
}
|
|
|
|
struct NoFocusFrameStyle : QProxyStyle
|
|
{
|
|
void drawControl(ControlElement element, const QStyleOption *option,
|
|
QPainter *painter, const QWidget *widget=nullptr)
|
|
const override
|
|
{
|
|
if (element == CE_FocusFrame)
|
|
return;
|
|
|
|
QProxyStyle::drawControl(element, option, painter, widget);
|
|
}
|
|
};
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
int ret = -1;
|
|
QCoreApplication::addLibraryPath(".");
|
|
base_set_log_handler(do_log);
|
|
|
|
try {
|
|
OBSApp program(argc, argv);
|
|
program.setStyle(new NoFocusFrameStyle);
|
|
program.OBSInit();
|
|
ret = program.exec();
|
|
|
|
} catch (const char *error) {
|
|
blog(LOG_ERROR, "%s", error);
|
|
}
|
|
|
|
blog(LOG_INFO, "Number of memory leaks: %llu",
|
|
(unsigned long long int)bnum_allocs());
|
|
return ret;
|
|
}
|