2013-11-07 16:45:03 -07:00
|
|
|
/******************************************************************************
|
2015-02-19 11:02:54 -08:00
|
|
|
Copyright (C) 2013-2015 by Hugh Bailey <obs.jim@gmail.com>
|
2014-05-15 14:04:18 -07:00
|
|
|
Zachary Lund <admin@computerquip.com>
|
2015-02-19 11:02:54 -08:00
|
|
|
Philippe Groarke <philippe.groarke@gmail.com>
|
2013-11-07 16:45:03 -07:00
|
|
|
|
|
|
|
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
|
2013-12-02 22:24:38 -07:00
|
|
|
the Free Software Foundation, either version 2 of the License, or
|
2013-11-07 16:45:03 -07:00
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
2013-11-09 00:19:38 -07:00
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
2013-11-07 16:45:03 -07:00
|
|
|
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/>.
|
|
|
|
******************************************************************************/
|
|
|
|
|
2018-04-05 05:45:01 -03:00
|
|
|
#include <ctime>
|
2013-12-22 17:42:02 -07:00
|
|
|
#include <obs.hpp>
|
2016-10-03 23:50:13 -07:00
|
|
|
#include <QGuiApplication>
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
#include <QMessageBox>
|
2014-01-25 09:08:56 -07:00
|
|
|
#include <QShowEvent>
|
2014-08-22 03:18:42 +02:00
|
|
|
#include <QDesktopServices>
|
2014-02-10 10:22:35 -07:00
|
|
|
#include <QFileDialog>
|
2016-07-04 22:34:56 -07:00
|
|
|
#include <QDesktopWidget>
|
2016-10-03 23:50:13 -07:00
|
|
|
#include <QScreen>
|
2018-08-02 08:23:12 +12:00
|
|
|
#include <QColorDialog>
|
|
|
|
#include <QSizePolicy>
|
2019-08-27 20:08:43 +12:00
|
|
|
#include <QScrollBar>
|
2020-03-18 20:37:56 -07:00
|
|
|
#include <QTextStream>
|
2013-12-29 09:17:00 -07:00
|
|
|
|
2014-05-18 17:44:10 -07:00
|
|
|
#include <util/dstr.h>
|
2014-03-06 21:08:12 -07:00
|
|
|
#include <util/util.hpp>
|
|
|
|
#include <util/platform.h>
|
2015-07-11 08:04:12 +02:00
|
|
|
#include <util/profiler.hpp>
|
2017-05-15 12:03:00 +02:00
|
|
|
#include <util/dstr.hpp>
|
2014-03-06 21:08:12 -07:00
|
|
|
|
2013-11-23 23:38:52 -07:00
|
|
|
#include "obs-app.hpp"
|
2014-03-06 21:08:12 -07:00
|
|
|
#include "platform.hpp"
|
2015-03-22 20:51:21 -07:00
|
|
|
#include "visibility-item-widget.hpp"
|
2015-06-27 18:18:40 -07:00
|
|
|
#include "item-widget-helpers.hpp"
|
2014-01-24 21:19:50 -07:00
|
|
|
#include "window-basic-settings.hpp"
|
2013-12-29 08:54:06 -07:00
|
|
|
#include "window-namedialog.hpp"
|
2017-05-06 12:13:56 -07:00
|
|
|
#include "window-basic-auto-config.hpp"
|
2014-05-10 18:47:48 -07:00
|
|
|
#include "window-basic-source-select.hpp"
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
#include "window-basic-main.hpp"
|
2017-05-10 19:37:45 -07:00
|
|
|
#include "window-basic-stats.hpp"
|
2015-02-06 03:17:33 -08:00
|
|
|
#include "window-basic-main-outputs.hpp"
|
2014-05-18 17:44:10 -07:00
|
|
|
#include "window-log-reply.hpp"
|
2015-04-04 01:40:15 -07:00
|
|
|
#include "window-projector.hpp"
|
2014-09-03 04:11:55 +02:00
|
|
|
#include "window-remux.hpp"
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
#include "qt-wrappers.hpp"
|
2018-08-23 20:43:44 -05:00
|
|
|
#include "context-bar-controls.hpp"
|
2014-03-23 01:07:54 -07:00
|
|
|
#include "display-helpers.hpp"
|
2014-05-03 22:54:38 -07:00
|
|
|
#include "volume-control.hpp"
|
UI: Replace Qt5Network classes with libcurl
The Qt5Network classes seem to only support OpenSSL, and because OpenSSL
isn't available on windows, we would have to distribute it with the
program to get SSL access working. The problem with that is that
OpenSSL is not GPL-compatible, so we cannot distribute OpenSSL with the
program, which means we have to find a better (and preferably superior)
library for accessing remote files that can use the windows SSPI for our
SSL needs, which comes with the operating system.
Fortunately, libcurl is probably the best library out there, and can be
compiled with SSPI instead of OpenSSL, so we're just going to switch to
libcurl instead. Originally I thought it didn't support SSPI, otherwise
I would have implemented it sooner.
As a side note, this will make it so we'll able to get files from the
internet via plugins, which will be quite useful.
2015-05-23 23:39:39 -07:00
|
|
|
#include "remote-text.hpp"
|
2019-11-19 16:57:49 -08:00
|
|
|
#include "ui-validation.hpp"
|
2018-08-23 20:43:44 -05:00
|
|
|
#include "media-controls.hpp"
|
2018-04-05 05:45:01 -03:00
|
|
|
#include <fstream>
|
|
|
|
#include <sstream>
|
2014-01-04 13:53:36 -07:00
|
|
|
|
2018-05-12 17:01:22 -07:00
|
|
|
#ifdef _WIN32
|
2017-02-20 04:46:29 -08:00
|
|
|
#include "win-update/win-update.hpp"
|
2019-02-24 11:00:57 +11:00
|
|
|
#include "windows.h"
|
2017-02-20 04:46:29 -08:00
|
|
|
#endif
|
|
|
|
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
#include "ui_OBSBasic.h"
|
2018-08-02 08:23:12 +12:00
|
|
|
#include "ui_ColorSelect.h"
|
2014-01-09 19:51:51 -06:00
|
|
|
|
2014-05-18 17:44:10 -07:00
|
|
|
#include <fstream>
|
Implement RTMP module (still needs drop code)
- Implement the RTMP output module. This time around, we just use a
simple FLV muxer, then just write to the stream with RTMP_Write.
Easy and effective.
- Fix the FLV muxer, the muxer now outputs proper FLV packets.
- Output API:
* When using encoders, automatically interleave encoded packets
before sending it to the output.
* Pair encoders and have them automatically wait for the other to
start to ensure sync.
* Change 'obs_output_signal_start_fail' to 'obs_output_signal_stop'
because it was a bit confusing, and doing this makes a lot more
sense for outputs that need to stop suddenly (disconnections/etc).
- Encoder API:
* Remove some unnecessary encoder functions from the actual API and
make them internal. Most of the encoder functions are handled
automatically by outputs anyway, so there's no real need to expose
them and end up inadvertently confusing plugin writers.
* Have audio encoders wait for the video encoder to get a frame, then
start at the exact data point that the first video frame starts to
ensure the most accrate sync of video/audio possible.
* Add a required 'frame_size' callback for audio encoders that
returns the expected number of frames desired to encode with. This
way, the libobs encoder API can handle the circular buffering
internally automatically for the encoder modules, so encoder
writers don't have to do it themselves.
- Fix a few bugs in the serializer interface. It was passing the wrong
variable for the data in a few cases.
- If a source has video, make obs_source_update defer the actual update
callback until the tick function is called to prevent threading
issues.
2014-04-07 22:00:10 -07:00
|
|
|
#include <sstream>
|
|
|
|
|
2014-04-16 22:35:01 +02:00
|
|
|
#include <QScreen>
|
|
|
|
#include <QWindow>
|
|
|
|
|
2018-07-27 21:35:08 -07:00
|
|
|
#include <json11.hpp>
|
|
|
|
|
|
|
|
using namespace json11;
|
2013-12-29 08:54:06 -07:00
|
|
|
using namespace std;
|
2013-11-22 16:20:52 -07:00
|
|
|
|
2018-11-17 12:01:55 -08:00
|
|
|
#ifdef BROWSER_AVAILABLE
|
2018-09-18 08:54:30 -07:00
|
|
|
#include <browser-panel.hpp>
|
2018-07-27 21:35:08 -07:00
|
|
|
#endif
|
|
|
|
|
2019-02-06 22:44:28 -08:00
|
|
|
#include "ui-config.h"
|
|
|
|
|
2019-02-06 21:38:34 -08:00
|
|
|
struct QCef;
|
2019-02-06 22:10:08 -08:00
|
|
|
struct QCefCookieManager;
|
|
|
|
|
|
|
|
QCef *cef = nullptr;
|
|
|
|
QCefCookieManager *panel_cookies = nullptr;
|
|
|
|
|
|
|
|
void DestroyPanelCookieManager();
|
2019-02-06 21:38:34 -08:00
|
|
|
|
2015-12-05 05:53:38 -08:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
template<typename OBSRef> struct SignalContainer {
|
|
|
|
OBSRef ref;
|
|
|
|
vector<shared_ptr<OBSSignal>> handlers;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-03-26 22:01:22 -07:00
|
|
|
extern volatile long insideEventLoop;
|
|
|
|
|
2014-02-02 14:26:23 -07:00
|
|
|
Q_DECLARE_METATYPE(OBSScene);
|
|
|
|
Q_DECLARE_METATYPE(OBSSceneItem);
|
2014-10-29 15:53:59 +01:00
|
|
|
Q_DECLARE_METATYPE(OBSSource);
|
2014-08-02 01:33:53 -07:00
|
|
|
Q_DECLARE_METATYPE(obs_order_movement);
|
2015-12-05 05:53:38 -08:00
|
|
|
Q_DECLARE_METATYPE(SignalContainer<OBSScene>);
|
2014-02-02 14:26:23 -07:00
|
|
|
|
2019-08-27 20:08:43 +12:00
|
|
|
QDataStream &operator<<(QDataStream &out, const SignalContainer<OBSScene> &v)
|
|
|
|
{
|
|
|
|
out << v.ref;
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
QDataStream &operator>>(QDataStream &in, SignalContainer<OBSScene> &v)
|
|
|
|
{
|
|
|
|
in >> v.ref;
|
|
|
|
return in;
|
|
|
|
}
|
|
|
|
|
2015-06-28 03:21:45 +02:00
|
|
|
template<typename T> static T GetOBSRef(QListWidgetItem *item)
|
|
|
|
{
|
|
|
|
return item->data(static_cast<int>(QtDataRole::OBSRef)).value<T>();
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T> static void SetOBSRef(QListWidgetItem *item, T &&val)
|
|
|
|
{
|
|
|
|
item->setData(static_cast<int>(QtDataRole::OBSRef),
|
|
|
|
QVariant::fromValue(val));
|
|
|
|
}
|
|
|
|
|
2014-07-27 12:48:14 -07:00
|
|
|
static void AddExtraModulePaths()
|
|
|
|
{
|
2015-01-15 23:44:38 -08:00
|
|
|
char base_module_dir[512];
|
2016-07-06 01:35:12 -07:00
|
|
|
#if defined(_WIN32) || defined(__APPLE__)
|
|
|
|
int ret = GetProgramDataPath(base_module_dir, sizeof(base_module_dir),
|
|
|
|
"obs-studio/plugins/%module%");
|
|
|
|
#else
|
2015-06-01 16:11:57 -07:00
|
|
|
int ret = GetConfigPath(base_module_dir, sizeof(base_module_dir),
|
2015-01-15 23:44:38 -08:00
|
|
|
"obs-studio/plugins/%module%");
|
2016-07-06 01:35:12 -07:00
|
|
|
#endif
|
2014-08-31 16:28:36 +02:00
|
|
|
|
2015-01-15 23:44:38 -08:00
|
|
|
if (ret <= 0)
|
2014-07-27 12:48:14 -07:00
|
|
|
return;
|
|
|
|
|
2019-08-17 08:49:08 -07:00
|
|
|
string path = base_module_dir;
|
2015-07-14 18:05:26 -07:00
|
|
|
#if defined(__APPLE__)
|
2014-07-27 12:48:14 -07:00
|
|
|
obs_add_module_path((path + "/bin").c_str(), (path + "/data").c_str());
|
2017-05-28 12:27:58 -05:00
|
|
|
|
|
|
|
BPtr<char> config_bin =
|
|
|
|
os_get_config_path_ptr("obs-studio/plugins/%module%/bin");
|
|
|
|
BPtr<char> config_data =
|
|
|
|
os_get_config_path_ptr("obs-studio/plugins/%module%/data");
|
|
|
|
obs_add_module_path(config_bin, config_data);
|
|
|
|
|
2015-07-14 18:05:26 -07:00
|
|
|
#elif ARCH_BITS == 64
|
|
|
|
obs_add_module_path((path + "/bin/64bit").c_str(),
|
|
|
|
(path + "/data").c_str());
|
|
|
|
#else
|
|
|
|
obs_add_module_path((path + "/bin/32bit").c_str(),
|
|
|
|
(path + "/data").c_str());
|
|
|
|
#endif
|
2014-07-27 12:48:14 -07:00
|
|
|
}
|
|
|
|
|
2018-06-01 22:42:11 -07:00
|
|
|
extern obs_frontend_callbacks *InitializeAPIInterface(OBSBasic *main);
|
|
|
|
|
2019-02-06 13:30:02 -08:00
|
|
|
void assignDockToggle(QDockWidget *dock, QAction *action)
|
|
|
|
{
|
|
|
|
auto handleWindowToggle = [action](bool vis) {
|
|
|
|
action->blockSignals(true);
|
|
|
|
action->setChecked(vis);
|
|
|
|
action->blockSignals(false);
|
|
|
|
};
|
|
|
|
auto handleMenuToggle = [dock](bool check) {
|
|
|
|
dock->blockSignals(true);
|
|
|
|
dock->setVisible(check);
|
|
|
|
dock->blockSignals(false);
|
|
|
|
};
|
|
|
|
|
|
|
|
dock->connect(dock->toggleViewAction(), &QAction::toggled,
|
|
|
|
handleWindowToggle);
|
|
|
|
dock->connect(action, &QAction::toggled, handleMenuToggle);
|
|
|
|
}
|
|
|
|
|
2019-02-06 22:45:06 -08:00
|
|
|
extern void RegisterTwitchAuth();
|
2019-02-14 20:35:47 +02:00
|
|
|
extern void RegisterRestreamAuth();
|
2019-02-06 22:44:28 -08:00
|
|
|
|
2014-02-02 15:23:38 -07:00
|
|
|
OBSBasic::OBSBasic(QWidget *parent)
|
2014-05-20 23:27:27 -07:00
|
|
|
: OBSMainWindow(parent), ui(new Ui::OBSBasic)
|
2014-02-02 15:23:38 -07:00
|
|
|
{
|
2020-05-15 07:34:54 -05:00
|
|
|
/* setup log viewer */
|
|
|
|
logView = new OBSLogViewer();
|
|
|
|
|
2019-08-27 20:08:43 +12:00
|
|
|
qRegisterMetaTypeStreamOperators<SignalContainer<OBSScene>>(
|
|
|
|
"SignalContainer<OBSScene>");
|
|
|
|
|
2017-01-25 00:45:22 -02:00
|
|
|
setAttribute(Qt::WA_NativeWindow);
|
|
|
|
|
2019-02-06 22:45:06 -08:00
|
|
|
#if TWITCH_ENABLED
|
|
|
|
RegisterTwitchAuth();
|
|
|
|
#endif
|
2019-02-14 20:35:47 +02:00
|
|
|
#if RESTREAM_ENABLED
|
|
|
|
RegisterRestreamAuth();
|
|
|
|
#endif
|
2019-02-06 22:44:28 -08:00
|
|
|
|
2016-09-26 12:40:23 -07:00
|
|
|
setAcceptDrops(true);
|
|
|
|
|
2020-02-14 23:30:34 +11:00
|
|
|
setContextMenuPolicy(Qt::CustomContextMenu);
|
|
|
|
connect(this, SIGNAL(customContextMenuRequested(const QPoint &)), this,
|
|
|
|
SLOT(on_customContextMenuRequested(const QPoint &)));
|
|
|
|
|
2018-06-01 22:42:11 -07:00
|
|
|
api = InitializeAPIInterface(this);
|
|
|
|
|
2014-02-02 15:23:38 -07:00
|
|
|
ui->setupUi(this);
|
2019-07-12 11:11:49 -05:00
|
|
|
ui->previewDisabledWidget->setVisible(false);
|
2015-04-02 21:35:46 -07:00
|
|
|
|
2017-07-27 02:40:41 -07:00
|
|
|
startingDockLayout = saveState();
|
|
|
|
|
2019-03-05 14:37:01 -08:00
|
|
|
statsDock = new OBSDock();
|
2018-09-16 10:02:43 -07:00
|
|
|
statsDock->setObjectName(QStringLiteral("statsDock"));
|
2020-06-07 15:31:13 -07:00
|
|
|
statsDock->setFeatures(QDockWidget::DockWidgetClosable |
|
|
|
|
QDockWidget::DockWidgetMovable |
|
|
|
|
QDockWidget::DockWidgetFloatable);
|
2018-09-16 10:02:43 -07:00
|
|
|
statsDock->setWindowTitle(QTStr("Basic.Stats"));
|
|
|
|
addDockWidget(Qt::BottomDockWidgetArea, statsDock);
|
|
|
|
statsDock->setVisible(false);
|
|
|
|
statsDock->setFloating(true);
|
|
|
|
statsDock->resize(700, 200);
|
|
|
|
|
2015-02-17 02:45:34 -05:00
|
|
|
copyActionsDynamicProperties();
|
2014-04-16 22:35:01 +02:00
|
|
|
|
2015-01-16 03:06:54 -05:00
|
|
|
char styleSheetPath[512];
|
2015-06-23 19:38:01 -07:00
|
|
|
int ret = GetProfilePath(styleSheetPath, sizeof(styleSheetPath),
|
|
|
|
"stylesheet.qss");
|
2015-01-16 03:06:54 -05:00
|
|
|
if (ret > 0) {
|
2015-01-17 19:24:00 +02:00
|
|
|
if (QFile::exists(styleSheetPath)) {
|
|
|
|
QString path =
|
|
|
|
QString("file:///") + QT_UTF8(styleSheetPath);
|
|
|
|
App()->setStyleSheet(path);
|
|
|
|
}
|
2015-01-16 03:06:54 -05:00
|
|
|
}
|
|
|
|
|
2014-10-29 15:54:45 +01:00
|
|
|
qRegisterMetaType<OBSScene>("OBSScene");
|
|
|
|
qRegisterMetaType<OBSSceneItem>("OBSSceneItem");
|
|
|
|
qRegisterMetaType<OBSSource>("OBSSource");
|
2014-11-01 21:48:58 +01:00
|
|
|
qRegisterMetaType<obs_hotkey_id>("obs_hotkey_id");
|
2019-06-01 13:13:24 +10:00
|
|
|
qRegisterMetaType<SavedProjectorInfo *>("SavedProjectorInfo *");
|
2014-10-29 15:54:45 +01:00
|
|
|
|
2015-06-27 04:06:34 +02:00
|
|
|
qRegisterMetaTypeStreamOperators<std::vector<std::shared_ptr<OBSSignal>>>(
|
|
|
|
"std::vector<std::shared_ptr<OBSSignal>>");
|
2015-06-27 10:48:58 +02:00
|
|
|
qRegisterMetaTypeStreamOperators<OBSScene>("OBSScene");
|
2015-06-27 03:39:39 +02:00
|
|
|
qRegisterMetaTypeStreamOperators<OBSSceneItem>("OBSSceneItem");
|
2015-06-27 10:48:58 +02:00
|
|
|
|
2014-10-29 17:18:00 +01:00
|
|
|
ui->scenes->setAttribute(Qt::WA_MacShowFocusRect, false);
|
|
|
|
ui->sources->setAttribute(Qt::WA_MacShowFocusRect, false);
|
|
|
|
|
2019-08-27 20:08:43 +12:00
|
|
|
bool sceneGrid = config_get_bool(App()->GlobalConfig(), "BasicWindow",
|
|
|
|
"gridMode");
|
|
|
|
ui->scenes->SetGridMode(sceneGrid);
|
|
|
|
|
2019-04-07 03:34:21 -04:00
|
|
|
ui->scenes->setItemDelegate(new SceneRenameDelegate(ui->scenes));
|
|
|
|
|
2016-01-09 14:47:58 -08:00
|
|
|
auto displayResize = [this]() {
|
2014-04-16 22:35:01 +02:00
|
|
|
struct obs_video_info ovi;
|
|
|
|
|
|
|
|
if (obs_get_video_info(&ovi))
|
|
|
|
ResizePreview(ovi.base_width, ovi.base_height);
|
2016-01-09 14:47:58 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
connect(windowHandle(), &QWindow::screenChanged, displayResize);
|
|
|
|
connect(ui->preview, &OBSQTDisplay::DisplayResized, displayResize);
|
2014-05-15 14:04:18 -07:00
|
|
|
|
2020-04-19 09:31:18 -05:00
|
|
|
delete shortcutFilter;
|
|
|
|
shortcutFilter = CreateShortcutFilter();
|
2019-01-31 03:33:15 +01:00
|
|
|
installEventFilter(shortcutFilter);
|
2014-11-01 21:48:58 +01:00
|
|
|
|
2015-07-05 23:41:36 -07:00
|
|
|
stringstream name;
|
2016-08-21 11:26:15 -07:00
|
|
|
name << "OBS " << App()->GetVersionString();
|
2015-07-05 23:41:36 -07:00
|
|
|
blog(LOG_INFO, "%s", name.str().c_str());
|
|
|
|
blog(LOG_INFO, "---------------------------------");
|
|
|
|
|
2015-07-03 17:55:55 -07:00
|
|
|
UpdateTitleBar();
|
2014-06-30 00:06:01 -07:00
|
|
|
|
|
|
|
connect(ui->scenes->itemDelegate(),
|
|
|
|
SIGNAL(closeEditor(QWidget *,
|
|
|
|
QAbstractItemDelegate::EndEditHint)),
|
|
|
|
this,
|
|
|
|
SLOT(SceneNameEdited(QWidget *,
|
|
|
|
QAbstractItemDelegate::EndEditHint)));
|
|
|
|
|
2014-07-06 16:18:16 -07:00
|
|
|
cpuUsageInfo = os_cpu_usage_info_start();
|
2014-07-06 16:19:27 -07:00
|
|
|
cpuUsageTimer = new QTimer(this);
|
2018-07-31 21:11:31 -07:00
|
|
|
connect(cpuUsageTimer.data(), SIGNAL(timeout()), ui->statusbar,
|
2014-07-06 16:19:27 -07:00
|
|
|
SLOT(UpdateCPUUsage()));
|
|
|
|
cpuUsageTimer->start(3000);
|
2014-07-13 21:46:23 +02:00
|
|
|
|
2019-06-12 04:56:00 -05:00
|
|
|
diskFullTimer = new QTimer(this);
|
|
|
|
connect(diskFullTimer, SIGNAL(timeout()), this,
|
|
|
|
SLOT(CheckDiskSpaceRemaining()));
|
|
|
|
|
2018-08-31 21:42:44 -03:00
|
|
|
QAction *renameScene = new QAction(ui->scenesDock);
|
|
|
|
renameScene->setShortcutContext(Qt::WidgetWithChildrenShortcut);
|
|
|
|
connect(renameScene, SIGNAL(triggered()), this, SLOT(EditSceneName()));
|
|
|
|
ui->scenesDock->addAction(renameScene);
|
|
|
|
|
|
|
|
QAction *renameSource = new QAction(ui->sourcesDock);
|
|
|
|
renameSource->setShortcutContext(Qt::WidgetWithChildrenShortcut);
|
|
|
|
connect(renameSource, SIGNAL(triggered()), this,
|
|
|
|
SLOT(EditSceneItemName()));
|
|
|
|
ui->sourcesDock->addAction(renameSource);
|
|
|
|
|
2014-07-13 21:46:23 +02:00
|
|
|
#ifdef __APPLE__
|
2018-08-31 21:42:44 -03:00
|
|
|
renameScene->setShortcut({Qt::Key_Return});
|
|
|
|
renameSource->setShortcut({Qt::Key_Return});
|
|
|
|
|
2018-03-03 02:54:12 -03:00
|
|
|
ui->actionRemoveSource->setShortcuts({Qt::Key_Backspace});
|
|
|
|
ui->actionRemoveScene->setShortcuts({Qt::Key_Backspace});
|
2014-07-21 03:06:02 +02:00
|
|
|
|
|
|
|
ui->action_Settings->setMenuRole(QAction::PreferencesRole);
|
|
|
|
ui->actionE_xit->setMenuRole(QAction::QuitRole);
|
2018-08-31 21:42:44 -03:00
|
|
|
#else
|
|
|
|
renameScene->setShortcut({Qt::Key_F2});
|
|
|
|
renameSource->setShortcut({Qt::Key_F2});
|
2014-07-13 21:46:23 +02:00
|
|
|
#endif
|
2015-04-02 23:09:13 -07:00
|
|
|
|
|
|
|
auto addNudge = [this](const QKeySequence &seq, const char *s) {
|
|
|
|
QAction *nudge = new QAction(ui->preview);
|
|
|
|
nudge->setShortcut(seq);
|
|
|
|
nudge->setShortcutContext(Qt::WidgetShortcut);
|
|
|
|
ui->preview->addAction(nudge);
|
|
|
|
connect(nudge, SIGNAL(triggered()), this, s);
|
|
|
|
};
|
|
|
|
|
|
|
|
addNudge(Qt::Key_Up, SLOT(NudgeUp()));
|
|
|
|
addNudge(Qt::Key_Down, SLOT(NudgeDown()));
|
|
|
|
addNudge(Qt::Key_Left, SLOT(NudgeLeft()));
|
|
|
|
addNudge(Qt::Key_Right, SLOT(NudgeRight()));
|
2017-07-27 02:40:41 -07:00
|
|
|
|
|
|
|
assignDockToggle(ui->scenesDock, ui->toggleScenes);
|
|
|
|
assignDockToggle(ui->sourcesDock, ui->toggleSources);
|
|
|
|
assignDockToggle(ui->mixerDock, ui->toggleMixer);
|
|
|
|
assignDockToggle(ui->transitionsDock, ui->toggleTransitions);
|
|
|
|
assignDockToggle(ui->controlsDock, ui->toggleControls);
|
2018-09-16 10:02:43 -07:00
|
|
|
assignDockToggle(statsDock, ui->toggleStats);
|
2017-08-12 19:45:25 +02:00
|
|
|
|
|
|
|
//hide all docking panes
|
|
|
|
ui->toggleScenes->setChecked(false);
|
|
|
|
ui->toggleSources->setChecked(false);
|
|
|
|
ui->toggleMixer->setChecked(false);
|
|
|
|
ui->toggleTransitions->setChecked(false);
|
|
|
|
ui->toggleControls->setChecked(false);
|
2018-08-28 07:39:50 -07:00
|
|
|
ui->toggleStats->setChecked(false);
|
2017-08-12 19:45:25 +02:00
|
|
|
|
2018-09-16 10:02:43 -07:00
|
|
|
QPoint curPos;
|
|
|
|
|
2018-08-23 20:43:44 -05:00
|
|
|
UpdateContextBar();
|
|
|
|
|
2017-08-12 19:45:25 +02:00
|
|
|
//restore parent window geometry
|
|
|
|
const char *geometry = config_get_string(App()->GlobalConfig(),
|
|
|
|
"BasicWindow", "geometry");
|
|
|
|
if (geometry != NULL) {
|
|
|
|
QByteArray byteArray =
|
|
|
|
QByteArray::fromBase64(QByteArray(geometry));
|
|
|
|
restoreGeometry(byteArray);
|
|
|
|
|
|
|
|
QRect windowGeometry = normalGeometry();
|
|
|
|
if (!WindowPositionValid(windowGeometry)) {
|
|
|
|
QRect rect = App()->desktop()->geometry();
|
|
|
|
setGeometry(QStyle::alignedRect(Qt::LeftToRight,
|
|
|
|
Qt::AlignCenter, size(),
|
|
|
|
rect));
|
|
|
|
}
|
2018-09-16 10:02:43 -07:00
|
|
|
|
|
|
|
curPos = pos();
|
|
|
|
} else {
|
|
|
|
QRect desktopRect =
|
|
|
|
QGuiApplication::primaryScreen()->geometry();
|
|
|
|
QSize adjSize = desktopRect.size() / 2 - size() / 2;
|
|
|
|
curPos = QPoint(adjSize.width(), adjSize.height());
|
2017-08-12 19:45:25 +02:00
|
|
|
}
|
2018-09-16 10:02:43 -07:00
|
|
|
|
|
|
|
QPoint curSize(width(), height());
|
2020-05-15 07:34:54 -05:00
|
|
|
|
2018-09-16 10:02:43 -07:00
|
|
|
QPoint statsDockSize(statsDock->width(), statsDock->height());
|
|
|
|
QPoint statsDockPos = curSize / 2 - statsDockSize / 2;
|
|
|
|
QPoint newPos = curPos + statsDockPos;
|
|
|
|
statsDock->move(newPos);
|
2019-02-28 15:29:41 -06:00
|
|
|
|
|
|
|
ui->previewLabel->setProperty("themeID", "previewProgramLabels");
|
2020-04-04 18:57:31 -07:00
|
|
|
ui->previewLabel->style()->polish(ui->previewLabel);
|
2019-02-28 15:29:41 -06:00
|
|
|
|
|
|
|
bool labels = config_get_bool(GetGlobalConfig(), "BasicWindow",
|
|
|
|
"StudioModeLabels");
|
|
|
|
|
|
|
|
if (!previewProgramMode)
|
|
|
|
ui->previewLabel->setHidden(true);
|
|
|
|
else
|
|
|
|
ui->previewLabel->setHidden(!labels);
|
2019-07-12 11:11:49 -05:00
|
|
|
|
|
|
|
ui->previewDisabledWidget->setContextMenuPolicy(Qt::CustomContextMenu);
|
|
|
|
connect(ui->previewDisabledWidget,
|
|
|
|
SIGNAL(customContextMenuRequested(const QPoint &)), this,
|
|
|
|
SLOT(PreviewDisabledMenu(const QPoint &)));
|
|
|
|
connect(ui->enablePreviewButton, SIGNAL(clicked()), this,
|
|
|
|
SLOT(TogglePreview()));
|
2019-10-10 20:12:20 -05:00
|
|
|
|
2020-08-21 03:39:42 -05:00
|
|
|
connect(ui->scenes, SIGNAL(scenesReordered()), this,
|
|
|
|
SLOT(ScenesReordered()));
|
2014-02-02 15:23:38 -07:00
|
|
|
}
|
|
|
|
|
2015-12-05 05:58:05 -08:00
|
|
|
static void SaveAudioDevice(const char *name, int channel, obs_data_t *parent,
|
|
|
|
vector<OBSSource> &audioSources)
|
2014-05-03 22:54:38 -07:00
|
|
|
{
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_source_t *source = obs_get_output_source(channel);
|
2014-05-03 22:54:38 -07:00
|
|
|
if (!source)
|
|
|
|
return;
|
|
|
|
|
2015-12-05 05:58:05 -08:00
|
|
|
audioSources.push_back(source);
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_data_t *data = obs_save_source(source);
|
2014-05-03 22:54:38 -07:00
|
|
|
|
2014-08-05 11:09:29 -07:00
|
|
|
obs_data_set_obj(parent, name, data);
|
2014-05-03 22:54:38 -07:00
|
|
|
|
|
|
|
obs_data_release(data);
|
|
|
|
obs_source_release(source);
|
|
|
|
}
|
|
|
|
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
static obs_data_t *GenerateSaveData(obs_data_array_t *sceneOrder,
|
|
|
|
obs_data_array_t *quickTransitionData,
|
|
|
|
int transitionDuration,
|
2016-02-27 02:50:04 -08:00
|
|
|
obs_data_array_t *transitions,
|
2016-12-29 09:21:53 -06:00
|
|
|
OBSScene &scene, OBSSource &curProgramScene,
|
2018-02-28 17:49:47 -03:00
|
|
|
obs_data_array_t *savedProjectorList)
|
2014-04-26 23:47:50 -07:00
|
|
|
{
|
2015-12-05 05:58:05 -08:00
|
|
|
obs_data_t *saveData = obs_data_create();
|
|
|
|
|
|
|
|
vector<OBSSource> audioSources;
|
2020-02-25 00:24:05 +01:00
|
|
|
audioSources.reserve(6);
|
2015-12-05 05:58:05 -08:00
|
|
|
|
|
|
|
SaveAudioDevice(DESKTOP_AUDIO_1, 1, saveData, audioSources);
|
|
|
|
SaveAudioDevice(DESKTOP_AUDIO_2, 2, saveData, audioSources);
|
|
|
|
SaveAudioDevice(AUX_AUDIO_1, 3, saveData, audioSources);
|
|
|
|
SaveAudioDevice(AUX_AUDIO_2, 4, saveData, audioSources);
|
|
|
|
SaveAudioDevice(AUX_AUDIO_3, 5, saveData, audioSources);
|
2018-06-26 20:41:32 +02:00
|
|
|
SaveAudioDevice(AUX_AUDIO_4, 6, saveData, audioSources);
|
2015-12-05 05:58:05 -08:00
|
|
|
|
2018-07-15 18:50:33 -07:00
|
|
|
/* -------------------------------- */
|
|
|
|
/* save non-group sources */
|
|
|
|
|
2015-12-05 05:58:05 -08:00
|
|
|
auto FilterAudioSources = [&](obs_source_t *source) {
|
2018-07-15 18:50:33 -07:00
|
|
|
if (obs_source_is_group(source))
|
|
|
|
return false;
|
|
|
|
|
2015-12-05 05:58:05 -08:00
|
|
|
return find(begin(audioSources), end(audioSources), source) ==
|
|
|
|
end(audioSources);
|
|
|
|
};
|
|
|
|
using FilterAudioSources_t = decltype(FilterAudioSources);
|
|
|
|
|
|
|
|
obs_data_array_t *sourcesArray = obs_save_sources_filtered(
|
|
|
|
[](void *data, obs_source_t *source) {
|
|
|
|
return (*static_cast<FilterAudioSources_t *>(data))(
|
|
|
|
source);
|
|
|
|
},
|
|
|
|
static_cast<void *>(&FilterAudioSources));
|
|
|
|
|
2018-07-15 18:50:33 -07:00
|
|
|
/* -------------------------------- */
|
|
|
|
/* save group sources separately */
|
|
|
|
|
|
|
|
/* saving separately ensures they won't be loaded in older versions */
|
|
|
|
obs_data_array_t *groupsArray = obs_save_sources_filtered(
|
|
|
|
[](void *, obs_source_t *source) {
|
|
|
|
return obs_source_is_group(source);
|
|
|
|
},
|
|
|
|
nullptr);
|
|
|
|
|
|
|
|
/* -------------------------------- */
|
|
|
|
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
obs_source_t *transition = obs_get_output_source(0);
|
|
|
|
obs_source_t *currentScene = obs_scene_get_source(scene);
|
2015-12-05 05:58:05 -08:00
|
|
|
const char *sceneName = obs_source_get_name(currentScene);
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
const char *programName = obs_source_get_name(curProgramScene);
|
2014-04-26 23:47:50 -07:00
|
|
|
|
2015-06-23 19:29:07 -07:00
|
|
|
const char *sceneCollection = config_get_string(
|
|
|
|
App()->GlobalConfig(), "Basic", "SceneCollection");
|
|
|
|
|
2014-08-05 11:09:29 -07:00
|
|
|
obs_data_set_string(saveData, "current_scene", sceneName);
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
obs_data_set_string(saveData, "current_program_scene", programName);
|
2015-06-25 03:53:48 -07:00
|
|
|
obs_data_set_array(saveData, "scene_order", sceneOrder);
|
2015-06-23 19:29:07 -07:00
|
|
|
obs_data_set_string(saveData, "name", sceneCollection);
|
2014-08-05 11:09:29 -07:00
|
|
|
obs_data_set_array(saveData, "sources", sourcesArray);
|
2018-07-15 18:50:33 -07:00
|
|
|
obs_data_set_array(saveData, "groups", groupsArray);
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
obs_data_set_array(saveData, "quick_transitions", quickTransitionData);
|
2016-02-27 02:50:04 -08:00
|
|
|
obs_data_set_array(saveData, "transitions", transitions);
|
2016-12-29 09:21:53 -06:00
|
|
|
obs_data_set_array(saveData, "saved_projectors", savedProjectorList);
|
2014-04-26 23:47:50 -07:00
|
|
|
obs_data_array_release(sourcesArray);
|
2018-07-15 18:50:33 -07:00
|
|
|
obs_data_array_release(groupsArray);
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
|
|
|
|
obs_data_set_string(saveData, "current_transition",
|
|
|
|
obs_source_get_name(transition));
|
|
|
|
obs_data_set_int(saveData, "transition_duration", transitionDuration);
|
|
|
|
obs_source_release(transition);
|
2014-04-26 23:47:50 -07:00
|
|
|
|
|
|
|
return saveData;
|
|
|
|
}
|
|
|
|
|
2015-02-17 02:45:34 -05:00
|
|
|
void OBSBasic::copyActionsDynamicProperties()
|
|
|
|
{
|
|
|
|
// Themes need the QAction dynamic properties
|
|
|
|
for (QAction *x : ui->scenesToolbar->actions()) {
|
|
|
|
QWidget *temp = ui->scenesToolbar->widgetForAction(x);
|
|
|
|
|
|
|
|
for (QByteArray &y : x->dynamicPropertyNames()) {
|
|
|
|
temp->setProperty(y, x->property(y));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (QAction *x : ui->sourcesToolbar->actions()) {
|
|
|
|
QWidget *temp = ui->sourcesToolbar->widgetForAction(x);
|
|
|
|
|
|
|
|
for (QByteArray &y : x->dynamicPropertyNames()) {
|
|
|
|
temp->setProperty(y, x->property(y));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-09 18:45:20 -02:00
|
|
|
void OBSBasic::UpdateVolumeControlsDecayRate()
|
|
|
|
{
|
|
|
|
double meterDecayRate =
|
|
|
|
config_get_double(basicConfig, "Audio", "MeterDecayRate");
|
|
|
|
|
|
|
|
for (size_t i = 0; i < volumes.size(); i++) {
|
|
|
|
volumes[i]->SetMeterDecayRate(meterDecayRate);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-19 22:57:36 +01:00
|
|
|
void OBSBasic::UpdateVolumeControlsPeakMeterType()
|
|
|
|
{
|
|
|
|
uint32_t peakMeterTypeIdx =
|
|
|
|
config_get_uint(basicConfig, "Audio", "PeakMeterType");
|
|
|
|
|
|
|
|
enum obs_peak_meter_type peakMeterType;
|
|
|
|
switch (peakMeterTypeIdx) {
|
|
|
|
case 0:
|
|
|
|
peakMeterType = SAMPLE_PEAK_METER;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
peakMeterType = TRUE_PEAK_METER;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
peakMeterType = SAMPLE_PEAK_METER;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = 0; i < volumes.size(); i++) {
|
|
|
|
volumes[i]->setPeakMeterType(peakMeterType);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-03 22:54:38 -07:00
|
|
|
void OBSBasic::ClearVolumeControls()
|
|
|
|
{
|
2018-05-16 05:53:15 +08:00
|
|
|
for (VolControl *vol : volumes)
|
|
|
|
delete vol;
|
2014-05-03 22:54:38 -07:00
|
|
|
|
|
|
|
volumes.clear();
|
|
|
|
}
|
|
|
|
|
2015-06-25 03:53:48 -07:00
|
|
|
obs_data_array_t *OBSBasic::SaveSceneListOrder()
|
|
|
|
{
|
|
|
|
obs_data_array_t *sceneOrder = obs_data_array_create();
|
|
|
|
|
|
|
|
for (int i = 0; i < ui->scenes->count(); i++) {
|
|
|
|
obs_data_t *data = obs_data_create();
|
|
|
|
obs_data_set_string(data, "name",
|
|
|
|
QT_TO_UTF8(ui->scenes->item(i)->text()));
|
|
|
|
obs_data_array_push_back(sceneOrder, data);
|
|
|
|
obs_data_release(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
return sceneOrder;
|
|
|
|
}
|
|
|
|
|
2016-12-29 09:21:53 -06:00
|
|
|
obs_data_array_t *OBSBasic::SaveProjectors()
|
|
|
|
{
|
2018-02-28 17:49:47 -03:00
|
|
|
obs_data_array_t *savedProjectors = obs_data_array_create();
|
2016-12-29 09:21:53 -06:00
|
|
|
|
2018-02-28 17:49:47 -03:00
|
|
|
auto saveProjector = [savedProjectors](OBSProjector *projector) {
|
|
|
|
if (!projector)
|
|
|
|
return;
|
2017-11-08 22:38:44 -05:00
|
|
|
|
|
|
|
obs_data_t *data = obs_data_create();
|
2018-02-28 17:49:47 -03:00
|
|
|
ProjectorType type = projector->GetProjectorType();
|
2020-04-27 23:26:20 +02:00
|
|
|
|
2018-03-01 02:10:32 -03:00
|
|
|
switch (type) {
|
|
|
|
case ProjectorType::Scene:
|
|
|
|
case ProjectorType::Source: {
|
|
|
|
obs_source_t *source = projector->GetSource();
|
|
|
|
const char *name = obs_source_get_name(source);
|
|
|
|
obs_data_set_string(data, "name", name);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2020-04-27 23:26:20 +02:00
|
|
|
|
2018-02-28 17:49:47 -03:00
|
|
|
obs_data_set_int(data, "monitor", projector->GetMonitor());
|
|
|
|
obs_data_set_int(data, "type", static_cast<int>(type));
|
2018-03-01 02:10:32 -03:00
|
|
|
obs_data_set_string(
|
|
|
|
data, "geometry",
|
|
|
|
projector->saveGeometry().toBase64().constData());
|
2020-04-27 23:26:20 +02:00
|
|
|
|
|
|
|
if (projector->IsAlwaysOnTopOverridden())
|
|
|
|
obs_data_set_bool(data, "alwaysOnTop",
|
|
|
|
projector->IsAlwaysOnTop());
|
|
|
|
|
|
|
|
obs_data_set_bool(data, "alwaysOnTopOverridden",
|
|
|
|
projector->IsAlwaysOnTopOverridden());
|
|
|
|
|
2018-02-28 17:49:47 -03:00
|
|
|
obs_data_array_push_back(savedProjectors, data);
|
2017-11-08 22:38:44 -05:00
|
|
|
obs_data_release(data);
|
2018-02-28 17:49:47 -03:00
|
|
|
};
|
2017-11-08 22:38:44 -05:00
|
|
|
|
2019-10-08 00:43:10 -05:00
|
|
|
for (size_t i = 0; i < projectors.size(); i++)
|
|
|
|
saveProjector(static_cast<OBSProjector *>(projectors[i]));
|
2017-10-21 15:51:01 -02:00
|
|
|
|
2018-02-28 17:49:47 -03:00
|
|
|
return savedProjectors;
|
2017-10-21 15:51:01 -02:00
|
|
|
}
|
|
|
|
|
2014-04-26 23:47:50 -07:00
|
|
|
void OBSBasic::Save(const char *file)
|
|
|
|
{
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
OBSScene scene = GetCurrentScene();
|
|
|
|
OBSSource curProgramScene = OBSGetStrongRef(programScene);
|
|
|
|
if (!curProgramScene)
|
|
|
|
curProgramScene = obs_scene_get_source(scene);
|
|
|
|
|
2015-06-25 03:53:48 -07:00
|
|
|
obs_data_array_t *sceneOrder = SaveSceneListOrder();
|
2016-02-27 02:50:04 -08:00
|
|
|
obs_data_array_t *transitions = SaveTransitions();
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
obs_data_array_t *quickTrData = SaveQuickTransitions();
|
2016-12-29 09:21:53 -06:00
|
|
|
obs_data_array_t *savedProjectorList = SaveProjectors();
|
2017-11-08 22:38:44 -05:00
|
|
|
obs_data_t *saveData = GenerateSaveData(
|
2016-02-27 02:50:04 -08:00
|
|
|
sceneOrder, quickTrData, ui->transitionDuration->value(),
|
2018-02-28 17:49:47 -03:00
|
|
|
transitions, scene, curProgramScene, savedProjectorList);
|
2015-08-21 17:56:37 -07:00
|
|
|
|
2016-07-26 01:32:43 -07:00
|
|
|
obs_data_set_bool(saveData, "preview_locked", ui->preview->Locked());
|
2017-05-13 13:13:55 -04:00
|
|
|
obs_data_set_bool(saveData, "scaling_enabled",
|
|
|
|
ui->preview->IsFixedScaling());
|
|
|
|
obs_data_set_int(saveData, "scaling_level",
|
|
|
|
ui->preview->GetScalingLevel());
|
|
|
|
obs_data_set_double(saveData, "scaling_off_x",
|
|
|
|
ui->preview->GetScrollX());
|
|
|
|
obs_data_set_double(saveData, "scaling_off_y",
|
|
|
|
ui->preview->GetScrollY());
|
2016-07-26 01:32:43 -07:00
|
|
|
|
2016-08-28 14:24:14 -07:00
|
|
|
if (api) {
|
|
|
|
obs_data_t *moduleObj = obs_data_create();
|
|
|
|
api->on_save(moduleObj);
|
|
|
|
obs_data_set_obj(saveData, "modules", moduleObj);
|
|
|
|
obs_data_release(moduleObj);
|
|
|
|
}
|
|
|
|
|
2015-08-21 17:56:37 -07:00
|
|
|
if (!obs_data_save_json_safe(saveData, file, "tmp", "bak"))
|
|
|
|
blog(LOG_ERROR, "Could not save scene data to %s", file);
|
2014-04-26 23:47:50 -07:00
|
|
|
|
|
|
|
obs_data_release(saveData);
|
2015-06-25 03:53:48 -07:00
|
|
|
obs_data_array_release(sceneOrder);
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
obs_data_array_release(quickTrData);
|
2016-02-27 02:50:04 -08:00
|
|
|
obs_data_array_release(transitions);
|
2016-12-29 09:21:53 -06:00
|
|
|
obs_data_array_release(savedProjectorList);
|
2014-04-26 23:47:50 -07:00
|
|
|
}
|
|
|
|
|
2018-02-04 23:36:56 +02:00
|
|
|
void OBSBasic::DeferSaveBegin()
|
|
|
|
{
|
|
|
|
os_atomic_inc_long(&disableSaving);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::DeferSaveEnd()
|
|
|
|
{
|
|
|
|
long result = os_atomic_dec_long(&disableSaving);
|
|
|
|
if (result == 0) {
|
|
|
|
SaveProject();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-07 21:11:29 +02:00
|
|
|
static void LogFilter(obs_source_t *, obs_source_t *filter, void *v_val);
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
static void LoadAudioDevice(const char *name, int channel, obs_data_t *parent)
|
2014-05-03 22:54:38 -07:00
|
|
|
{
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_data_t *data = obs_data_get_obj(parent, name);
|
2014-05-03 22:54:38 -07:00
|
|
|
if (!data)
|
|
|
|
return;
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_source_t *source = obs_load_source(data);
|
2014-05-03 22:54:38 -07:00
|
|
|
if (source) {
|
|
|
|
obs_set_output_source(channel, source);
|
2020-04-07 21:11:29 +02:00
|
|
|
|
|
|
|
const char *name = obs_source_get_name(source);
|
|
|
|
blog(LOG_INFO, "[Loaded global audio device]: '%s'", name);
|
|
|
|
obs_source_enum_filters(source, LogFilter, (void *)(intptr_t)1);
|
2020-05-16 11:10:04 +02:00
|
|
|
obs_monitoring_type monitoring_type =
|
|
|
|
obs_source_get_monitoring_type(source);
|
|
|
|
if (monitoring_type != OBS_MONITORING_TYPE_NONE) {
|
|
|
|
const char *type = (monitoring_type ==
|
|
|
|
OBS_MONITORING_TYPE_MONITOR_ONLY)
|
|
|
|
? "monitor only"
|
|
|
|
: "monitor and output";
|
|
|
|
|
|
|
|
blog(LOG_INFO, " - monitoring: %s", type);
|
|
|
|
}
|
2014-05-03 22:54:38 -07:00
|
|
|
obs_source_release(source);
|
|
|
|
}
|
|
|
|
|
|
|
|
obs_data_release(data);
|
|
|
|
}
|
|
|
|
|
2015-06-23 18:47:22 -07:00
|
|
|
static inline bool HasAudioDevices(const char *source_id)
|
|
|
|
{
|
|
|
|
const char *output_id = source_id;
|
2015-12-29 15:25:45 -08:00
|
|
|
obs_properties_t *props = obs_get_source_properties(output_id);
|
2015-06-23 18:47:22 -07:00
|
|
|
size_t count = 0;
|
|
|
|
|
|
|
|
if (!props)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
obs_property_t *devices = obs_properties_get(props, "device_id");
|
|
|
|
if (devices)
|
|
|
|
count = obs_property_list_item_count(devices);
|
|
|
|
|
|
|
|
obs_properties_destroy(props);
|
|
|
|
|
|
|
|
return count != 0;
|
|
|
|
}
|
|
|
|
|
2015-09-23 08:41:00 -07:00
|
|
|
void OBSBasic::CreateFirstRunSources()
|
2014-05-03 22:54:38 -07:00
|
|
|
{
|
2015-06-23 18:47:22 -07:00
|
|
|
bool hasDesktopAudio = HasAudioDevices(App()->OutputAudioSource());
|
|
|
|
bool hasInputAudio = HasAudioDevices(App()->InputAudioSource());
|
|
|
|
|
|
|
|
if (hasDesktopAudio)
|
|
|
|
ResetAudioDevice(App()->OutputAudioSource(), "default",
|
|
|
|
Str("Basic.DesktopDevice1"), 1);
|
|
|
|
if (hasInputAudio)
|
|
|
|
ResetAudioDevice(App()->InputAudioSource(), "default",
|
|
|
|
Str("Basic.AuxDevice1"), 3);
|
2015-09-11 21:49:21 -07:00
|
|
|
}
|
|
|
|
|
2015-09-11 21:51:09 -07:00
|
|
|
void OBSBasic::CreateDefaultScene(bool firstStart)
|
2015-09-11 21:49:21 -07:00
|
|
|
{
|
|
|
|
disableSaving++;
|
|
|
|
|
|
|
|
ClearSceneData();
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
InitDefaultTransitions();
|
|
|
|
CreateDefaultQuickTransitions();
|
|
|
|
ui->transitionDuration->setValue(300);
|
|
|
|
SetTransition(fadeTransition);
|
2015-09-11 21:49:21 -07:00
|
|
|
|
|
|
|
obs_scene_t *scene = obs_scene_create(Str("Basic.Scene"));
|
|
|
|
|
2015-09-11 21:51:09 -07:00
|
|
|
if (firstStart)
|
2015-09-23 08:41:00 -07:00
|
|
|
CreateFirstRunSources();
|
2015-09-11 21:49:21 -07:00
|
|
|
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
SetCurrentScene(scene, true);
|
2015-09-11 21:49:21 -07:00
|
|
|
obs_scene_release(scene);
|
2015-06-30 05:49:31 -07:00
|
|
|
|
|
|
|
disableSaving--;
|
2014-05-03 22:54:38 -07:00
|
|
|
}
|
|
|
|
|
2015-06-25 03:53:48 -07:00
|
|
|
static void ReorderItemByName(QListWidget *lw, const char *name, int newIndex)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < lw->count(); i++) {
|
|
|
|
QListWidgetItem *item = lw->item(i);
|
|
|
|
|
|
|
|
if (strcmp(name, QT_TO_UTF8(item->text())) == 0) {
|
|
|
|
if (newIndex != i) {
|
|
|
|
item = lw->takeItem(i);
|
|
|
|
lw->insertItem(newIndex, item);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::LoadSceneListOrder(obs_data_array_t *array)
|
|
|
|
{
|
|
|
|
size_t num = obs_data_array_count(array);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < num; i++) {
|
|
|
|
obs_data_t *data = obs_data_array_item(array, i);
|
|
|
|
const char *name = obs_data_get_string(data, "name");
|
|
|
|
|
|
|
|
ReorderItemByName(ui->scenes, name, (int)i);
|
|
|
|
|
|
|
|
obs_data_release(data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-29 09:21:53 -06:00
|
|
|
void OBSBasic::LoadSavedProjectors(obs_data_array_t *array)
|
|
|
|
{
|
2018-11-13 14:39:56 +08:00
|
|
|
for (SavedProjectorInfo *info : savedProjectorsArray) {
|
|
|
|
delete info;
|
|
|
|
}
|
|
|
|
savedProjectorsArray.clear();
|
|
|
|
|
2016-12-29 09:21:53 -06:00
|
|
|
size_t num = obs_data_array_count(array);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < num; i++) {
|
|
|
|
obs_data_t *data = obs_data_array_item(array, i);
|
|
|
|
|
2018-02-28 17:49:47 -03:00
|
|
|
SavedProjectorInfo *info = new SavedProjectorInfo();
|
|
|
|
info->monitor = obs_data_get_int(data, "monitor");
|
|
|
|
info->type = static_cast<ProjectorType>(
|
|
|
|
obs_data_get_int(data, "type"));
|
2018-03-01 02:10:32 -03:00
|
|
|
info->geometry =
|
|
|
|
std::string(obs_data_get_string(data, "geometry"));
|
|
|
|
info->name = std::string(obs_data_get_string(data, "name"));
|
2020-04-27 23:26:20 +02:00
|
|
|
info->alwaysOnTop = obs_data_get_bool(data, "alwaysOnTop");
|
|
|
|
info->alwaysOnTopOverridden =
|
|
|
|
obs_data_get_bool(data, "alwaysOnTopOverridden");
|
|
|
|
|
2018-02-28 17:49:47 -03:00
|
|
|
savedProjectorsArray.emplace_back(info);
|
2017-10-21 15:51:01 -02:00
|
|
|
|
|
|
|
obs_data_release(data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-05 17:43:30 -07:00
|
|
|
static void LogFilter(obs_source_t *, obs_source_t *filter, void *v_val)
|
|
|
|
{
|
|
|
|
const char *name = obs_source_get_name(filter);
|
|
|
|
const char *id = obs_source_get_id(filter);
|
|
|
|
int val = (int)(intptr_t)v_val;
|
|
|
|
string indent;
|
|
|
|
|
|
|
|
for (int i = 0; i < val; i++)
|
|
|
|
indent += " ";
|
|
|
|
|
|
|
|
blog(LOG_INFO, "%s- filter: '%s' (%s)", indent.c_str(), name, id);
|
|
|
|
}
|
|
|
|
|
2019-03-31 21:30:09 +11:00
|
|
|
static bool LogSceneItem(obs_scene_t *, obs_sceneitem_t *item, void *v_val)
|
2016-08-05 17:43:30 -07:00
|
|
|
{
|
|
|
|
obs_source_t *source = obs_sceneitem_get_source(item);
|
|
|
|
const char *name = obs_source_get_name(source);
|
|
|
|
const char *id = obs_source_get_id(source);
|
2019-03-31 21:30:09 +11:00
|
|
|
int indent_count = (int)(intptr_t)v_val;
|
|
|
|
string indent;
|
|
|
|
|
|
|
|
for (int i = 0; i < indent_count; i++)
|
|
|
|
indent += " ";
|
2016-08-05 17:43:30 -07:00
|
|
|
|
2019-03-31 21:30:09 +11:00
|
|
|
blog(LOG_INFO, "%s- source: '%s' (%s)", indent.c_str(), name, id);
|
2016-08-05 17:43:30 -07:00
|
|
|
|
2017-03-02 13:05:38 -08:00
|
|
|
obs_monitoring_type monitoring_type =
|
|
|
|
obs_source_get_monitoring_type(source);
|
|
|
|
|
|
|
|
if (monitoring_type != OBS_MONITORING_TYPE_NONE) {
|
|
|
|
const char *type =
|
|
|
|
(monitoring_type == OBS_MONITORING_TYPE_MONITOR_ONLY)
|
|
|
|
? "monitor only"
|
|
|
|
: "monitor and output";
|
|
|
|
|
2019-03-31 21:30:09 +11:00
|
|
|
blog(LOG_INFO, " %s- monitoring: %s", indent.c_str(), type);
|
2017-03-02 13:05:38 -08:00
|
|
|
}
|
2019-03-31 21:30:09 +11:00
|
|
|
int child_indent = 1 + indent_count;
|
|
|
|
obs_source_enum_filters(source, LogFilter,
|
|
|
|
(void *)(intptr_t)child_indent);
|
|
|
|
if (obs_sceneitem_is_group(item))
|
|
|
|
obs_sceneitem_group_enum_items(item, LogSceneItem,
|
|
|
|
(void *)(intptr_t)child_indent);
|
2016-08-05 17:43:30 -07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::LogScenes()
|
|
|
|
{
|
|
|
|
blog(LOG_INFO, "------------------------------------------------");
|
|
|
|
blog(LOG_INFO, "Loaded scenes:");
|
|
|
|
|
|
|
|
for (int i = 0; i < ui->scenes->count(); i++) {
|
|
|
|
QListWidgetItem *item = ui->scenes->item(i);
|
|
|
|
OBSScene scene = GetOBSRef<OBSScene>(item);
|
|
|
|
|
|
|
|
obs_source_t *source = obs_scene_get_source(scene);
|
|
|
|
const char *name = obs_source_get_name(source);
|
|
|
|
|
|
|
|
blog(LOG_INFO, "- scene '%s':", name);
|
2019-03-31 21:30:09 +11:00
|
|
|
obs_scene_enum_items(scene, LogSceneItem, (void *)(intptr_t)1);
|
2016-08-05 17:43:30 -07:00
|
|
|
obs_source_enum_filters(source, LogFilter, (void *)(intptr_t)1);
|
|
|
|
}
|
|
|
|
|
|
|
|
blog(LOG_INFO, "------------------------------------------------");
|
|
|
|
}
|
|
|
|
|
2014-04-26 23:47:50 -07:00
|
|
|
void OBSBasic::Load(const char *file)
|
|
|
|
{
|
2015-08-21 17:56:37 -07:00
|
|
|
disableSaving++;
|
|
|
|
|
|
|
|
obs_data_t *data = obs_data_create_from_json_file_safe(file, "bak");
|
|
|
|
if (!data) {
|
|
|
|
disableSaving--;
|
2017-05-04 18:17:42 -07:00
|
|
|
blog(LOG_INFO, "No scene file found, creating default scene");
|
2015-09-11 21:51:09 -07:00
|
|
|
CreateDefaultScene(true);
|
2015-07-02 16:15:51 -07:00
|
|
|
SaveProject();
|
2014-04-26 23:47:50 -07:00
|
|
|
return;
|
2014-05-03 22:54:38 -07:00
|
|
|
}
|
2014-04-26 23:47:50 -07:00
|
|
|
|
2015-07-02 16:17:55 -07:00
|
|
|
ClearSceneData();
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
InitDefaultTransitions();
|
2015-07-02 16:17:55 -07:00
|
|
|
|
2017-12-25 13:23:39 -08:00
|
|
|
obs_data_t *modulesObj = obs_data_get_obj(data, "modules");
|
|
|
|
if (api)
|
|
|
|
api->on_preload(modulesObj);
|
|
|
|
|
2015-06-25 03:53:48 -07:00
|
|
|
obs_data_array_t *sceneOrder = obs_data_get_array(data, "scene_order");
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_data_array_t *sources = obs_data_get_array(data, "sources");
|
2018-07-15 18:50:33 -07:00
|
|
|
obs_data_array_t *groups = obs_data_get_array(data, "groups");
|
2016-02-27 02:50:04 -08:00
|
|
|
obs_data_array_t *transitions = obs_data_get_array(data, "transitions");
|
2014-08-05 11:09:29 -07:00
|
|
|
const char *sceneName = obs_data_get_string(data, "current_scene");
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
const char *programSceneName =
|
|
|
|
obs_data_get_string(data, "current_program_scene");
|
|
|
|
const char *transitionName =
|
|
|
|
obs_data_get_string(data, "current_transition");
|
|
|
|
|
2016-04-13 18:59:27 -07:00
|
|
|
if (!opt_starting_scene.empty()) {
|
|
|
|
programSceneName = opt_starting_scene.c_str();
|
|
|
|
if (!IsPreviewProgramMode())
|
|
|
|
sceneName = opt_starting_scene.c_str();
|
|
|
|
}
|
|
|
|
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
int newDuration = obs_data_get_int(data, "transition_duration");
|
|
|
|
if (!newDuration)
|
|
|
|
newDuration = 300;
|
|
|
|
|
|
|
|
if (!transitionName)
|
|
|
|
transitionName = obs_source_get_name(fadeTransition);
|
2015-06-23 19:29:07 -07:00
|
|
|
|
|
|
|
const char *curSceneCollection = config_get_string(
|
|
|
|
App()->GlobalConfig(), "Basic", "SceneCollection");
|
|
|
|
|
|
|
|
obs_data_set_default_string(data, "name", curSceneCollection);
|
|
|
|
|
|
|
|
const char *name = obs_data_get_string(data, "name");
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_source_t *curScene;
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
obs_source_t *curProgramScene;
|
|
|
|
obs_source_t *curTransition;
|
2014-04-26 23:47:50 -07:00
|
|
|
|
2015-06-23 19:29:07 -07:00
|
|
|
if (!name || !*name)
|
|
|
|
name = curSceneCollection;
|
|
|
|
|
2014-05-03 22:54:38 -07:00
|
|
|
LoadAudioDevice(DESKTOP_AUDIO_1, 1, data);
|
|
|
|
LoadAudioDevice(DESKTOP_AUDIO_2, 2, data);
|
|
|
|
LoadAudioDevice(AUX_AUDIO_1, 3, data);
|
|
|
|
LoadAudioDevice(AUX_AUDIO_2, 4, data);
|
|
|
|
LoadAudioDevice(AUX_AUDIO_3, 5, data);
|
2018-06-26 20:41:32 +02:00
|
|
|
LoadAudioDevice(AUX_AUDIO_4, 6, data);
|
2014-05-03 22:54:38 -07:00
|
|
|
|
2018-07-15 18:50:33 -07:00
|
|
|
if (!sources) {
|
|
|
|
sources = groups;
|
|
|
|
groups = nullptr;
|
|
|
|
} else {
|
|
|
|
obs_data_array_push_back_array(sources, groups);
|
|
|
|
}
|
|
|
|
|
2018-05-04 11:47:54 -07:00
|
|
|
obs_load_sources(sources, nullptr, nullptr);
|
2014-04-26 23:47:50 -07:00
|
|
|
|
2016-02-27 02:50:04 -08:00
|
|
|
if (transitions)
|
|
|
|
LoadTransitions(transitions);
|
2015-06-25 03:53:48 -07:00
|
|
|
if (sceneOrder)
|
|
|
|
LoadSceneListOrder(sceneOrder);
|
|
|
|
|
2016-02-27 02:50:04 -08:00
|
|
|
obs_data_array_release(transitions);
|
|
|
|
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
curTransition = FindTransition(transitionName);
|
|
|
|
if (!curTransition)
|
|
|
|
curTransition = fadeTransition;
|
|
|
|
|
|
|
|
ui->transitionDuration->setValue(newDuration);
|
|
|
|
SetTransition(curTransition);
|
|
|
|
|
2016-04-13 18:59:27 -07:00
|
|
|
retryScene:
|
2014-04-26 23:47:50 -07:00
|
|
|
curScene = obs_get_source_by_name(sceneName);
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
curProgramScene = obs_get_source_by_name(programSceneName);
|
2016-04-13 18:59:27 -07:00
|
|
|
|
|
|
|
/* if the starting scene command line parameter is bad at all,
|
|
|
|
* fall back to original settings */
|
|
|
|
if (!opt_starting_scene.empty() && (!curScene || !curProgramScene)) {
|
|
|
|
sceneName = obs_data_get_string(data, "current_scene");
|
|
|
|
programSceneName =
|
|
|
|
obs_data_get_string(data, "current_program_scene");
|
|
|
|
obs_source_release(curScene);
|
|
|
|
obs_source_release(curProgramScene);
|
|
|
|
opt_starting_scene.clear();
|
|
|
|
goto retryScene;
|
|
|
|
}
|
|
|
|
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
if (!curProgramScene) {
|
|
|
|
curProgramScene = curScene;
|
|
|
|
obs_source_addref(curScene);
|
|
|
|
}
|
|
|
|
|
|
|
|
SetCurrentScene(curScene, true);
|
|
|
|
if (IsPreviewProgramMode())
|
|
|
|
TransitionToScene(curProgramScene, true);
|
2014-04-26 23:47:50 -07:00
|
|
|
obs_source_release(curScene);
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
obs_source_release(curProgramScene);
|
2014-04-26 23:47:50 -07:00
|
|
|
|
|
|
|
obs_data_array_release(sources);
|
2018-07-15 18:50:33 -07:00
|
|
|
obs_data_array_release(groups);
|
2015-06-25 03:53:48 -07:00
|
|
|
obs_data_array_release(sceneOrder);
|
2015-06-23 19:29:07 -07:00
|
|
|
|
2018-02-27 20:26:22 -03:00
|
|
|
/* ------------------- */
|
|
|
|
|
|
|
|
bool projectorSave = config_get_bool(GetGlobalConfig(), "BasicWindow",
|
|
|
|
"SaveProjectors");
|
|
|
|
|
|
|
|
if (projectorSave) {
|
|
|
|
obs_data_array_t *savedProjectors =
|
|
|
|
obs_data_get_array(data, "saved_projectors");
|
|
|
|
|
2018-04-18 19:15:52 -07:00
|
|
|
if (savedProjectors) {
|
2018-02-27 20:26:22 -03:00
|
|
|
LoadSavedProjectors(savedProjectors);
|
2018-04-18 19:15:52 -07:00
|
|
|
OpenSavedProjectors();
|
|
|
|
activateWindow();
|
|
|
|
}
|
2018-02-27 20:26:22 -03:00
|
|
|
|
|
|
|
obs_data_array_release(savedProjectors);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ------------------- */
|
|
|
|
|
2015-06-23 19:29:07 -07:00
|
|
|
std::string file_base = strrchr(file, '/') + 1;
|
|
|
|
file_base.erase(file_base.size() - 5, 5);
|
|
|
|
|
|
|
|
config_set_string(App()->GlobalConfig(), "Basic", "SceneCollection",
|
|
|
|
name);
|
|
|
|
config_set_string(App()->GlobalConfig(), "Basic", "SceneCollectionFile",
|
|
|
|
file_base.c_str());
|
|
|
|
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
obs_data_array_t *quickTransitionData =
|
|
|
|
obs_data_get_array(data, "quick_transitions");
|
|
|
|
LoadQuickTransitions(quickTransitionData);
|
|
|
|
obs_data_array_release(quickTransitionData);
|
|
|
|
|
|
|
|
RefreshQuickTransitions();
|
|
|
|
|
2016-07-26 01:32:43 -07:00
|
|
|
bool previewLocked = obs_data_get_bool(data, "preview_locked");
|
|
|
|
ui->preview->SetLocked(previewLocked);
|
|
|
|
ui->actionLockPreview->setChecked(previewLocked);
|
|
|
|
|
2017-05-13 13:13:55 -04:00
|
|
|
/* ---------------------- */
|
|
|
|
|
|
|
|
bool fixedScaling = obs_data_get_bool(data, "scaling_enabled");
|
|
|
|
int scalingLevel = (int)obs_data_get_int(data, "scaling_level");
|
|
|
|
float scrollOffX = (float)obs_data_get_double(data, "scaling_off_x");
|
|
|
|
float scrollOffY = (float)obs_data_get_double(data, "scaling_off_y");
|
|
|
|
|
|
|
|
if (fixedScaling) {
|
|
|
|
ui->preview->SetScalingLevel(scalingLevel);
|
|
|
|
ui->preview->SetScrollingOffset(scrollOffX, scrollOffY);
|
2016-11-05 12:48:46 -04:00
|
|
|
}
|
2017-05-13 13:13:55 -04:00
|
|
|
ui->preview->SetFixedScaling(fixedScaling);
|
2020-03-11 13:48:44 -07:00
|
|
|
emit ui->preview->DisplayResized();
|
2016-11-05 12:48:46 -04:00
|
|
|
|
2017-05-13 13:13:55 -04:00
|
|
|
/* ---------------------- */
|
2016-11-05 12:48:46 -04:00
|
|
|
|
2017-12-25 13:23:39 -08:00
|
|
|
if (api)
|
2016-08-28 14:24:14 -07:00
|
|
|
api->on_load(modulesObj);
|
|
|
|
|
2017-12-25 13:23:39 -08:00
|
|
|
obs_data_release(modulesObj);
|
2014-04-26 23:47:50 -07:00
|
|
|
obs_data_release(data);
|
2015-06-30 05:49:31 -07:00
|
|
|
|
2016-04-13 18:59:27 -07:00
|
|
|
if (!opt_starting_scene.empty())
|
|
|
|
opt_starting_scene.clear();
|
|
|
|
|
2016-04-13 19:00:12 -07:00
|
|
|
if (opt_start_streaming) {
|
2017-10-12 17:20:19 +02:00
|
|
|
blog(LOG_INFO, "Starting stream due to command line parameter");
|
2016-04-13 19:00:12 -07:00
|
|
|
QMetaObject::invokeMethod(this, "StartStreaming",
|
|
|
|
Qt::QueuedConnection);
|
|
|
|
opt_start_streaming = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (opt_start_recording) {
|
2017-10-12 17:20:19 +02:00
|
|
|
blog(LOG_INFO,
|
|
|
|
"Starting recording due to command line parameter");
|
2016-04-13 19:00:12 -07:00
|
|
|
QMetaObject::invokeMethod(this, "StartRecording",
|
|
|
|
Qt::QueuedConnection);
|
|
|
|
opt_start_recording = false;
|
|
|
|
}
|
|
|
|
|
2017-01-11 13:19:17 -06:00
|
|
|
if (opt_start_replaybuffer) {
|
|
|
|
QMetaObject::invokeMethod(this, "StartReplayBuffer",
|
|
|
|
Qt::QueuedConnection);
|
|
|
|
opt_start_replaybuffer = false;
|
|
|
|
}
|
|
|
|
|
2020-06-20 07:42:14 -07:00
|
|
|
if (opt_start_virtualcam) {
|
|
|
|
QMetaObject::invokeMethod(this, "StartVirtualCam",
|
|
|
|
Qt::QueuedConnection);
|
|
|
|
opt_start_virtualcam = false;
|
|
|
|
}
|
|
|
|
|
2019-10-04 22:33:06 -07:00
|
|
|
copyStrings.clear();
|
2019-04-23 02:26:17 -05:00
|
|
|
copyFiltersString = nullptr;
|
|
|
|
|
2016-08-05 17:43:30 -07:00
|
|
|
LogScenes();
|
|
|
|
|
2015-06-30 05:49:31 -07:00
|
|
|
disableSaving--;
|
2017-12-25 13:38:02 -08:00
|
|
|
|
2018-08-29 16:05:29 -07:00
|
|
|
if (api) {
|
2017-12-25 13:38:02 -08:00
|
|
|
api->on_event(OBS_FRONTEND_EVENT_SCENE_CHANGED);
|
2018-03-04 12:07:13 -08:00
|
|
|
api->on_event(OBS_FRONTEND_EVENT_PREVIEW_SCENE_CHANGED);
|
|
|
|
}
|
2014-04-26 23:47:50 -07:00
|
|
|
}
|
|
|
|
|
2015-06-23 19:38:01 -07:00
|
|
|
#define SERVICE_PATH "service.json"
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
|
|
|
|
void OBSBasic::SaveService()
|
|
|
|
{
|
|
|
|
if (!service)
|
|
|
|
return;
|
|
|
|
|
2015-01-15 23:44:38 -08:00
|
|
|
char serviceJsonPath[512];
|
2015-06-23 19:38:01 -07:00
|
|
|
int ret = GetProfilePath(serviceJsonPath, sizeof(serviceJsonPath),
|
2015-01-15 23:44:38 -08:00
|
|
|
SERVICE_PATH);
|
|
|
|
if (ret <= 0)
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
return;
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_data_t *data = obs_data_create();
|
|
|
|
obs_data_t *settings = obs_service_get_settings(service);
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
|
2015-02-07 08:19:34 -08:00
|
|
|
obs_data_set_string(data, "type", obs_service_get_type(service));
|
2014-08-05 11:09:29 -07:00
|
|
|
obs_data_set_obj(data, "settings", settings);
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
|
2015-08-21 17:56:37 -07:00
|
|
|
if (!obs_data_save_json_safe(data, serviceJsonPath, "tmp", "bak"))
|
|
|
|
blog(LOG_WARNING, "Failed to save service");
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
|
|
|
|
obs_data_release(settings);
|
|
|
|
obs_data_release(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool OBSBasic::LoadService()
|
|
|
|
{
|
|
|
|
const char *type;
|
|
|
|
|
2015-01-15 23:44:38 -08:00
|
|
|
char serviceJsonPath[512];
|
2015-06-23 19:38:01 -07:00
|
|
|
int ret = GetProfilePath(serviceJsonPath, sizeof(serviceJsonPath),
|
2015-01-15 23:44:38 -08:00
|
|
|
SERVICE_PATH);
|
|
|
|
if (ret <= 0)
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
return false;
|
|
|
|
|
2015-08-21 17:56:37 -07:00
|
|
|
obs_data_t *data =
|
|
|
|
obs_data_create_from_json_file_safe(serviceJsonPath, "bak");
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
|
2020-06-18 23:45:52 +02:00
|
|
|
if (!data)
|
|
|
|
return false;
|
|
|
|
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
obs_data_set_default_string(data, "type", "rtmp_common");
|
2014-08-05 11:09:29 -07:00
|
|
|
type = obs_data_get_string(data, "type");
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_data_t *settings = obs_data_get_obj(data, "settings");
|
2015-04-05 00:03:40 +02:00
|
|
|
obs_data_t *hotkey_data = obs_data_get_obj(data, "hotkeys");
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
|
2014-11-01 21:41:17 +01:00
|
|
|
service = obs_service_create(type, "default_service", settings,
|
2015-04-05 00:03:40 +02:00
|
|
|
hotkey_data);
|
2015-05-04 02:07:43 +02:00
|
|
|
obs_service_release(service);
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
|
2015-04-05 00:03:40 +02:00
|
|
|
obs_data_release(hotkey_data);
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
obs_data_release(settings);
|
|
|
|
obs_data_release(data);
|
|
|
|
|
|
|
|
return !!service;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool OBSBasic::InitService()
|
|
|
|
{
|
2015-07-11 08:04:12 +02:00
|
|
|
ProfileScope("OBSBasic::InitService");
|
|
|
|
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
if (LoadService())
|
|
|
|
return true;
|
|
|
|
|
2014-11-01 21:41:17 +01:00
|
|
|
service = obs_service_create("rtmp_common", "default_service", nullptr,
|
|
|
|
nullptr);
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
if (!service)
|
|
|
|
return false;
|
2015-05-04 02:07:43 +02:00
|
|
|
obs_service_release(service);
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-07-03 10:26:18 -07:00
|
|
|
static const double scaled_vals[] = {1.0, 1.25, (1.0 / 0.75), 1.5,
|
|
|
|
(1.0 / 0.6), 1.75, 2.0, 2.25,
|
|
|
|
2.5, 2.75, 3.0, 0.0};
|
|
|
|
|
2019-02-06 22:10:08 -08:00
|
|
|
extern void CheckExistingCookieId();
|
|
|
|
|
2014-03-06 21:08:12 -07:00
|
|
|
bool OBSBasic::InitBasicConfigDefaults()
|
|
|
|
{
|
2016-10-03 23:50:13 -07:00
|
|
|
QList<QScreen *> screens = QGuiApplication::screens();
|
2014-03-06 21:08:12 -07:00
|
|
|
|
2016-10-03 23:50:13 -07:00
|
|
|
if (!screens.size()) {
|
2014-03-06 21:08:12 -07:00
|
|
|
OBSErrorBox(NULL, "There appears to be no monitors. Er, this "
|
|
|
|
"technically shouldn't be possible.");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-10-03 23:50:13 -07:00
|
|
|
QScreen *primaryScreen = QGuiApplication::primaryScreen();
|
|
|
|
|
|
|
|
uint32_t cx = primaryScreen->size().width();
|
|
|
|
uint32_t cy = primaryScreen->size().height();
|
2014-03-06 21:08:12 -07:00
|
|
|
|
2017-04-28 18:02:03 -07:00
|
|
|
bool oldResolutionDefaults = config_get_bool(
|
|
|
|
App()->GlobalConfig(), "General", "Pre19Defaults");
|
|
|
|
|
|
|
|
/* use 1920x1080 for new default base res if main monitor is above
|
|
|
|
* 1920x1080, but don't apply for people from older builds -- only to
|
|
|
|
* new users */
|
|
|
|
if (!oldResolutionDefaults && (cx * cy) > (1920 * 1080)) {
|
|
|
|
cx = 1920;
|
|
|
|
cy = 1080;
|
|
|
|
}
|
|
|
|
|
2018-10-04 20:06:54 -07:00
|
|
|
bool changed = false;
|
|
|
|
|
|
|
|
/* ----------------------------------------------------- */
|
|
|
|
/* move over old FFmpeg track settings */
|
|
|
|
if (config_has_user_value(basicConfig, "AdvOut", "FFAudioTrack") &&
|
|
|
|
!config_has_user_value(basicConfig, "AdvOut", "Pre22.1Settings")) {
|
|
|
|
|
|
|
|
int track = (int)config_get_int(basicConfig, "AdvOut",
|
|
|
|
"FFAudioTrack");
|
|
|
|
config_set_int(basicConfig, "AdvOut", "FFAudioMixes",
|
|
|
|
1LL << (track - 1));
|
|
|
|
config_set_bool(basicConfig, "AdvOut", "Pre22.1Settings", true);
|
|
|
|
changed = true;
|
|
|
|
}
|
|
|
|
|
2015-07-03 09:49:22 -07:00
|
|
|
/* ----------------------------------------------------- */
|
|
|
|
/* move over mixer values in advanced if older config */
|
|
|
|
if (config_has_user_value(basicConfig, "AdvOut", "RecTrackIndex") &&
|
|
|
|
!config_has_user_value(basicConfig, "AdvOut", "RecTracks")) {
|
|
|
|
|
|
|
|
uint64_t track =
|
|
|
|
config_get_uint(basicConfig, "AdvOut", "RecTrackIndex");
|
|
|
|
track = 1ULL << (track - 1);
|
|
|
|
config_set_uint(basicConfig, "AdvOut", "RecTracks", track);
|
|
|
|
config_remove_value(basicConfig, "AdvOut", "RecTrackIndex");
|
2018-10-04 20:06:54 -07:00
|
|
|
changed = true;
|
2015-07-03 09:49:22 -07:00
|
|
|
}
|
|
|
|
|
2019-10-10 03:29:09 +02:00
|
|
|
/* ----------------------------------------------------- */
|
|
|
|
/* set twitch chat extensions to "both" if prev version */
|
|
|
|
/* is under 24.1 */
|
|
|
|
if (config_get_bool(GetGlobalConfig(), "General", "Pre24.1Defaults") &&
|
|
|
|
!config_has_user_value(basicConfig, "Twitch", "AddonChoice")) {
|
|
|
|
config_set_int(basicConfig, "Twitch", "AddonChoice", 3);
|
|
|
|
changed = true;
|
|
|
|
}
|
|
|
|
|
2015-07-03 09:49:22 -07:00
|
|
|
/* ----------------------------------------------------- */
|
|
|
|
|
2018-10-04 20:06:54 -07:00
|
|
|
if (changed)
|
|
|
|
config_save_safe(basicConfig, "tmp", nullptr);
|
|
|
|
|
|
|
|
/* ----------------------------------------------------- */
|
|
|
|
|
2015-06-25 01:06:46 -07:00
|
|
|
config_set_default_string(basicConfig, "Output", "Mode", "Simple");
|
2015-02-06 03:17:33 -08:00
|
|
|
|
2014-05-20 23:27:27 -07:00
|
|
|
config_set_default_string(basicConfig, "SimpleOutput", "FilePath",
|
|
|
|
GetDefaultVideoSavePath().c_str());
|
2015-05-29 09:45:54 -07:00
|
|
|
config_set_default_string(basicConfig, "SimpleOutput", "RecFormat",
|
2019-07-11 15:38:29 +10:00
|
|
|
"mkv");
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
config_set_default_uint(basicConfig, "SimpleOutput", "VBitrate", 2500);
|
2015-06-05 08:57:01 -07:00
|
|
|
config_set_default_uint(basicConfig, "SimpleOutput", "ABitrate", 160);
|
2014-08-25 07:48:51 -07:00
|
|
|
config_set_default_bool(basicConfig, "SimpleOutput", "UseAdvanced",
|
|
|
|
false);
|
2016-04-10 03:10:36 -07:00
|
|
|
config_set_default_bool(basicConfig, "SimpleOutput", "EnforceBitrate",
|
|
|
|
true);
|
2014-08-25 07:48:51 -07:00
|
|
|
config_set_default_string(basicConfig, "SimpleOutput", "Preset",
|
|
|
|
"veryfast");
|
2019-02-12 14:20:53 -08:00
|
|
|
config_set_default_string(basicConfig, "SimpleOutput", "NVENCPreset",
|
|
|
|
"hq");
|
UI: Add recording presets to simple output
So certain high-profile individuals were complaining that it was
difficult to configure recording settings for quality in OBS. So, I
decided to add a very easy-to-use auto-configuration for high quality
encoding -- including lossless encoding. This feature will
automatically configure ideal recording settings based upon a specified
quality level.
Recording quality presets added to simple output:
- Same as stream: Copies the encoded streaming data with no extra usage
hit.
- High quality: uses a higher CRF value (starting at 23) if using x264.
- Indistinguishable quality: uses a low CRF value (starting at 16) if
using x264.
- Lossless will spawn an FFmpeg output that uses huffyuv encoding. If a
user tries to select lossless, they will be warned both via a dialog
prompt and a warning message in the settings window to ensure they
understand that it requires tremendous amounts of free space. It will
always use the AVI file format.
Extra Notes:
- When High/Indistinguishable quality is set, it will allow you to
select the recording encoder. Currently, it just allows you to select
x264 (at either veryfast or ultrafast). Later on, it'll be useful to
be able to set up pre-configured presets for hardware encoders once
more are implemented and tested.
- I decided to allow the use of x264 at both veryfast or ultrafast
presets. The reasoning is two-fold:
1.) ultrafast is perfectly viable even for near indistinguishable
quality as long as it has the appropriate CRF value. It's nice if you
want to record but would like to or need to reduce the impact of
encoding on the CPU. It will automatically compensate for the preset at
the cost of larger file size.
2.) It was suggested to just always use ultrafast, but ultrafast
requires 2-4x as much disk space for the same CRF (most likely due to
x264 compensating for the preset). Providing veryfast is important if
you really want to reduce file size and/or reduce blocking at lower
quality levels.
- When a recording preset is used, a secondary audio encoder is also
spawned at 192 bitrate to ensure high quality audio. I chose 192
because that's the limit of the media foundation aac encoder on
windows, which I want to make sure is used if available due to its
high performance.
- The CRF calculation is based upon resolution, quality, and whether
it's set to ultrafast. First, quality sets the base CRF, 23 for
"good" quality, 16 for "very high" quality. If set to ultrafast,
it'll subtract 2 points from the CRF value to help compensate. Lower
resolutions will also lower the CRF value to help improve higher
details with a smaller pixel ratio.
2015-09-18 22:29:36 -07:00
|
|
|
config_set_default_string(basicConfig, "SimpleOutput", "RecQuality",
|
|
|
|
"Stream");
|
2016-12-07 05:21:44 -08:00
|
|
|
config_set_default_bool(basicConfig, "SimpleOutput", "RecRB", false);
|
|
|
|
config_set_default_int(basicConfig, "SimpleOutput", "RecRBTime", 20);
|
|
|
|
config_set_default_int(basicConfig, "SimpleOutput", "RecRBSize", 512);
|
2016-12-09 14:42:14 -08:00
|
|
|
config_set_default_string(basicConfig, "SimpleOutput", "RecRBPrefix",
|
|
|
|
"Replay");
|
2014-03-10 13:10:35 -07:00
|
|
|
|
2015-02-10 20:06:00 -08:00
|
|
|
config_set_default_bool(basicConfig, "AdvOut", "ApplyServiceSettings",
|
|
|
|
true);
|
2015-01-26 13:41:22 -08:00
|
|
|
config_set_default_bool(basicConfig, "AdvOut", "UseRescale", false);
|
|
|
|
config_set_default_uint(basicConfig, "AdvOut", "TrackIndex", 1);
|
|
|
|
config_set_default_string(basicConfig, "AdvOut", "Encoder", "obs_x264");
|
|
|
|
|
|
|
|
config_set_default_string(basicConfig, "AdvOut", "RecType", "Standard");
|
|
|
|
|
|
|
|
config_set_default_string(basicConfig, "AdvOut", "RecFilePath",
|
|
|
|
GetDefaultVideoSavePath().c_str());
|
2019-07-11 15:38:29 +10:00
|
|
|
config_set_default_string(basicConfig, "AdvOut", "RecFormat", "mkv");
|
2015-01-26 13:41:22 -08:00
|
|
|
config_set_default_bool(basicConfig, "AdvOut", "RecUseRescale", false);
|
2015-05-30 21:45:14 -07:00
|
|
|
config_set_default_uint(basicConfig, "AdvOut", "RecTracks", (1 << 0));
|
2015-01-26 13:41:22 -08:00
|
|
|
config_set_default_string(basicConfig, "AdvOut", "RecEncoder", "none");
|
2019-07-23 07:20:13 -05:00
|
|
|
config_set_default_uint(basicConfig, "AdvOut", "FLVTrack", 1);
|
2019-06-22 22:13:45 -07:00
|
|
|
|
2015-08-18 20:58:24 -07:00
|
|
|
config_set_default_bool(basicConfig, "AdvOut", "FFOutputToFile", true);
|
|
|
|
config_set_default_string(basicConfig, "AdvOut", "FFFilePath",
|
|
|
|
GetDefaultVideoSavePath().c_str());
|
|
|
|
config_set_default_string(basicConfig, "AdvOut", "FFExtension", "mp4");
|
2015-01-26 13:41:22 -08:00
|
|
|
config_set_default_uint(basicConfig, "AdvOut", "FFVBitrate", 2500);
|
2017-02-17 17:25:21 +01:00
|
|
|
config_set_default_uint(basicConfig, "AdvOut", "FFVGOPSize", 250);
|
2015-01-26 13:41:22 -08:00
|
|
|
config_set_default_bool(basicConfig, "AdvOut", "FFUseRescale", false);
|
2017-02-17 18:07:32 +01:00
|
|
|
config_set_default_bool(basicConfig, "AdvOut", "FFIgnoreCompat", false);
|
2015-01-26 13:41:22 -08:00
|
|
|
config_set_default_uint(basicConfig, "AdvOut", "FFABitrate", 160);
|
2018-10-04 20:06:54 -07:00
|
|
|
config_set_default_uint(basicConfig, "AdvOut", "FFAudioMixes", 1);
|
2019-06-22 22:13:45 -07:00
|
|
|
|
2015-01-26 13:41:22 -08:00
|
|
|
config_set_default_uint(basicConfig, "AdvOut", "Track1Bitrate", 160);
|
|
|
|
config_set_default_uint(basicConfig, "AdvOut", "Track2Bitrate", 160);
|
|
|
|
config_set_default_uint(basicConfig, "AdvOut", "Track3Bitrate", 160);
|
|
|
|
config_set_default_uint(basicConfig, "AdvOut", "Track4Bitrate", 160);
|
2016-12-22 06:54:05 -08:00
|
|
|
config_set_default_uint(basicConfig, "AdvOut", "Track5Bitrate", 160);
|
|
|
|
config_set_default_uint(basicConfig, "AdvOut", "Track6Bitrate", 160);
|
2019-06-22 22:13:45 -07:00
|
|
|
|
2017-09-05 20:01:49 -04:00
|
|
|
config_set_default_bool(basicConfig, "AdvOut", "RecRB", false);
|
|
|
|
config_set_default_uint(basicConfig, "AdvOut", "RecRBTime", 20);
|
|
|
|
config_set_default_int(basicConfig, "AdvOut", "RecRBSize", 512);
|
2019-06-22 22:13:45 -07:00
|
|
|
|
2014-03-06 21:08:12 -07:00
|
|
|
config_set_default_uint(basicConfig, "Video", "BaseCX", cx);
|
|
|
|
config_set_default_uint(basicConfig, "Video", "BaseCY", cy);
|
|
|
|
|
2017-04-28 18:00:11 -07:00
|
|
|
/* don't allow BaseCX/BaseCY to be susceptible to defaults changing */
|
|
|
|
if (!config_has_user_value(basicConfig, "Video", "BaseCX") ||
|
|
|
|
!config_has_user_value(basicConfig, "Video", "BaseCY")) {
|
|
|
|
config_set_uint(basicConfig, "Video", "BaseCX", cx);
|
|
|
|
config_set_uint(basicConfig, "Video", "BaseCY", cy);
|
|
|
|
config_save_safe(basicConfig, "tmp", nullptr);
|
|
|
|
}
|
|
|
|
|
2016-03-25 02:43:38 -07:00
|
|
|
config_set_default_string(basicConfig, "Output", "FilenameFormatting",
|
2016-04-02 16:28:04 -07:00
|
|
|
"%CCYY-%MM-%DD %hh-%mm-%ss");
|
2016-03-25 02:43:38 -07:00
|
|
|
|
2015-09-06 16:12:03 -07:00
|
|
|
config_set_default_bool(basicConfig, "Output", "DelayEnable", false);
|
|
|
|
config_set_default_uint(basicConfig, "Output", "DelaySec", 20);
|
|
|
|
config_set_default_bool(basicConfig, "Output", "DelayPreserve", true);
|
|
|
|
|
2015-09-10 19:10:40 -07:00
|
|
|
config_set_default_bool(basicConfig, "Output", "Reconnect", true);
|
|
|
|
config_set_default_uint(basicConfig, "Output", "RetryDelay", 10);
|
|
|
|
config_set_default_uint(basicConfig, "Output", "MaxRetries", 20);
|
|
|
|
|
2016-07-29 08:30:54 -07:00
|
|
|
config_set_default_string(basicConfig, "Output", "BindIP", "default");
|
2017-02-22 02:05:45 +01:00
|
|
|
config_set_default_bool(basicConfig, "Output", "NewSocketLoopEnable",
|
|
|
|
false);
|
|
|
|
config_set_default_bool(basicConfig, "Output", "LowLatencyEnable",
|
|
|
|
false);
|
2016-07-29 08:30:54 -07:00
|
|
|
|
2015-07-03 10:26:18 -07:00
|
|
|
int i = 0;
|
|
|
|
uint32_t scale_cx = cx;
|
|
|
|
uint32_t scale_cy = cy;
|
|
|
|
|
|
|
|
/* use a default scaled resolution that has a pixel count no higher
|
|
|
|
* than 1280x720 */
|
|
|
|
while (((scale_cx * scale_cy) > (1280 * 720)) && scaled_vals[i] > 0.0) {
|
|
|
|
double scale = scaled_vals[i++];
|
|
|
|
scale_cx = uint32_t(double(cx) / scale);
|
|
|
|
scale_cy = uint32_t(double(cy) / scale);
|
|
|
|
}
|
|
|
|
|
|
|
|
config_set_default_uint(basicConfig, "Video", "OutputCX", scale_cx);
|
|
|
|
config_set_default_uint(basicConfig, "Video", "OutputCY", scale_cy);
|
2014-03-06 21:08:12 -07:00
|
|
|
|
2017-04-28 18:00:11 -07:00
|
|
|
/* don't allow OutputCX/OutputCY to be susceptible to defaults
|
|
|
|
* changing */
|
|
|
|
if (!config_has_user_value(basicConfig, "Video", "OutputCX") ||
|
|
|
|
!config_has_user_value(basicConfig, "Video", "OutputCY")) {
|
|
|
|
config_set_uint(basicConfig, "Video", "OutputCX", scale_cx);
|
|
|
|
config_set_uint(basicConfig, "Video", "OutputCY", scale_cy);
|
|
|
|
config_save_safe(basicConfig, "tmp", nullptr);
|
|
|
|
}
|
|
|
|
|
2014-03-06 21:08:12 -07:00
|
|
|
config_set_default_uint(basicConfig, "Video", "FPSType", 0);
|
|
|
|
config_set_default_string(basicConfig, "Video", "FPSCommon", "30");
|
|
|
|
config_set_default_uint(basicConfig, "Video", "FPSInt", 30);
|
|
|
|
config_set_default_uint(basicConfig, "Video", "FPSNum", 30);
|
|
|
|
config_set_default_uint(basicConfig, "Video", "FPSDen", 1);
|
2014-12-15 01:08:46 -08:00
|
|
|
config_set_default_string(basicConfig, "Video", "ScaleType", "bicubic");
|
2015-01-09 20:16:30 -08:00
|
|
|
config_set_default_string(basicConfig, "Video", "ColorFormat", "NV12");
|
2020-01-25 13:03:24 -08:00
|
|
|
config_set_default_string(basicConfig, "Video", "ColorSpace", "sRGB");
|
2015-01-09 20:16:30 -08:00
|
|
|
config_set_default_string(basicConfig, "Video", "ColorRange",
|
|
|
|
"Partial");
|
2014-03-06 21:08:12 -07:00
|
|
|
|
2017-02-05 21:51:50 -08:00
|
|
|
config_set_default_string(basicConfig, "Audio", "MonitoringDeviceId",
|
|
|
|
"default");
|
|
|
|
config_set_default_string(
|
|
|
|
basicConfig, "Audio", "MonitoringDeviceName",
|
|
|
|
Str("Basic.Settings.Advanced.Audio.MonitoringDevice"
|
|
|
|
".Default"));
|
2020-08-19 13:59:57 +02:00
|
|
|
config_set_default_uint(basicConfig, "Audio", "SampleRate", 48000);
|
2014-03-06 21:08:12 -07:00
|
|
|
config_set_default_string(basicConfig, "Audio", "ChannelSetup",
|
|
|
|
"Stereo");
|
2018-01-09 18:45:20 -02:00
|
|
|
config_set_default_double(basicConfig, "Audio", "MeterDecayRate",
|
|
|
|
VOLUME_METER_DECAY_FAST);
|
2018-01-19 22:57:36 +01:00
|
|
|
config_set_default_uint(basicConfig, "Audio", "PeakMeterType", 0);
|
2014-03-06 21:08:12 -07:00
|
|
|
|
2019-02-06 22:10:08 -08:00
|
|
|
CheckExistingCookieId();
|
|
|
|
|
2014-03-06 21:08:12 -07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-02-05 18:05:19 -08:00
|
|
|
extern bool EncoderAvailable(const char *encoder);
|
|
|
|
|
|
|
|
void OBSBasic::InitBasicConfigDefaults2()
|
|
|
|
{
|
|
|
|
bool oldEncDefaults = config_get_bool(App()->GlobalConfig(), "General",
|
|
|
|
"Pre23Defaults");
|
|
|
|
bool useNV = EncoderAvailable("ffmpeg_nvenc") && !oldEncDefaults;
|
|
|
|
|
|
|
|
config_set_default_string(basicConfig, "SimpleOutput", "StreamEncoder",
|
|
|
|
useNV ? SIMPLE_ENCODER_NVENC
|
|
|
|
: SIMPLE_ENCODER_X264);
|
|
|
|
config_set_default_string(basicConfig, "SimpleOutput", "RecEncoder",
|
|
|
|
useNV ? SIMPLE_ENCODER_NVENC
|
|
|
|
: SIMPLE_ENCODER_X264);
|
|
|
|
}
|
|
|
|
|
2014-03-06 21:08:12 -07:00
|
|
|
bool OBSBasic::InitBasicConfig()
|
|
|
|
{
|
2015-07-11 08:04:12 +02:00
|
|
|
ProfileScope("OBSBasic::InitBasicConfig");
|
|
|
|
|
2015-01-15 23:44:38 -08:00
|
|
|
char configPath[512];
|
2015-06-23 19:38:01 -07:00
|
|
|
|
|
|
|
int ret = GetProfilePath(configPath, sizeof(configPath), "");
|
|
|
|
if (ret <= 0) {
|
|
|
|
OBSErrorBox(nullptr, "Failed to get profile path");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (os_mkdir(configPath) == MKDIR_ERROR) {
|
|
|
|
OBSErrorBox(nullptr, "Failed to create profile path");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = GetProfilePath(configPath, sizeof(configPath), "basic.ini");
|
2015-01-15 23:44:38 -08:00
|
|
|
if (ret <= 0) {
|
|
|
|
OBSErrorBox(nullptr, "Failed to get base.ini path");
|
|
|
|
return false;
|
|
|
|
}
|
2014-03-06 21:08:12 -07:00
|
|
|
|
2014-05-12 15:30:36 -07:00
|
|
|
int code = basicConfig.Open(configPath, CONFIG_OPEN_ALWAYS);
|
|
|
|
if (code != CONFIG_SUCCESS) {
|
|
|
|
OBSErrorBox(NULL, "Failed to open basic.ini: %d", code);
|
2014-03-06 21:08:12 -07:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-06-23 19:38:01 -07:00
|
|
|
if (config_get_string(basicConfig, "General", "Name") == nullptr) {
|
|
|
|
const char *curName = config_get_string(App()->GlobalConfig(),
|
|
|
|
"Basic", "Profile");
|
|
|
|
|
|
|
|
config_set_string(basicConfig, "General", "Name", curName);
|
2015-08-21 17:56:37 -07:00
|
|
|
basicConfig.SaveSafe("tmp");
|
2015-06-23 19:38:01 -07:00
|
|
|
}
|
|
|
|
|
2014-03-06 21:08:12 -07:00
|
|
|
return InitBasicConfigDefaults();
|
|
|
|
}
|
|
|
|
|
2014-05-03 22:54:38 -07:00
|
|
|
void OBSBasic::InitOBSCallbacks()
|
|
|
|
{
|
2015-07-11 08:04:12 +02:00
|
|
|
ProfileScope("OBSBasic::InitOBSCallbacks");
|
|
|
|
|
2020-02-24 21:51:02 +01:00
|
|
|
signalHandlers.reserve(signalHandlers.size() + 7);
|
2018-05-04 11:47:54 -07:00
|
|
|
signalHandlers.emplace_back(obs_get_signal_handler(), "source_create",
|
|
|
|
OBSBasic::SourceCreated, this);
|
2015-06-27 09:29:17 +02:00
|
|
|
signalHandlers.emplace_back(obs_get_signal_handler(), "source_remove",
|
2014-05-03 22:54:38 -07:00
|
|
|
OBSBasic::SourceRemoved, this);
|
2015-06-27 09:29:17 +02:00
|
|
|
signalHandlers.emplace_back(obs_get_signal_handler(), "source_activate",
|
2014-05-03 22:54:38 -07:00
|
|
|
OBSBasic::SourceActivated, this);
|
2015-06-27 09:29:17 +02:00
|
|
|
signalHandlers.emplace_back(obs_get_signal_handler(),
|
|
|
|
"source_deactivate",
|
2014-05-03 22:54:38 -07:00
|
|
|
OBSBasic::SourceDeactivated, this);
|
2019-09-19 23:38:53 -07:00
|
|
|
signalHandlers.emplace_back(obs_get_signal_handler(),
|
|
|
|
"source_audio_activate",
|
|
|
|
OBSBasic::SourceAudioActivated, this);
|
|
|
|
signalHandlers.emplace_back(obs_get_signal_handler(),
|
|
|
|
"source_audio_deactivate",
|
|
|
|
OBSBasic::SourceAudioDeactivated, this);
|
2015-06-27 09:29:17 +02:00
|
|
|
signalHandlers.emplace_back(obs_get_signal_handler(), "source_rename",
|
2014-06-30 00:06:01 -07:00
|
|
|
OBSBasic::SourceRenamed, this);
|
2014-05-03 22:54:38 -07:00
|
|
|
}
|
|
|
|
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
void OBSBasic::InitPrimitives()
|
|
|
|
{
|
2015-07-11 08:04:12 +02:00
|
|
|
ProfileScope("OBSBasic::InitPrimitives");
|
|
|
|
|
2014-08-04 05:48:58 -07:00
|
|
|
obs_enter_graphics();
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
|
2014-08-07 23:42:07 -07:00
|
|
|
gs_render_start(true);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
gs_vertex2f(0.0f, 0.0f);
|
|
|
|
gs_vertex2f(0.0f, 1.0f);
|
|
|
|
gs_vertex2f(1.0f, 0.0f);
|
2020-03-19 23:25:24 +13:00
|
|
|
gs_vertex2f(1.0f, 1.0f);
|
2014-08-07 23:42:07 -07:00
|
|
|
box = gs_render_save();
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
|
2016-03-30 18:44:49 -07:00
|
|
|
gs_render_start(true);
|
|
|
|
gs_vertex2f(0.0f, 0.0f);
|
|
|
|
gs_vertex2f(0.0f, 1.0f);
|
|
|
|
boxLeft = gs_render_save();
|
|
|
|
|
|
|
|
gs_render_start(true);
|
|
|
|
gs_vertex2f(0.0f, 0.0f);
|
|
|
|
gs_vertex2f(1.0f, 0.0f);
|
|
|
|
boxTop = gs_render_save();
|
|
|
|
|
|
|
|
gs_render_start(true);
|
|
|
|
gs_vertex2f(1.0f, 0.0f);
|
|
|
|
gs_vertex2f(1.0f, 1.0f);
|
|
|
|
boxRight = gs_render_save();
|
|
|
|
|
|
|
|
gs_render_start(true);
|
|
|
|
gs_vertex2f(0.0f, 1.0f);
|
|
|
|
gs_vertex2f(1.0f, 1.0f);
|
|
|
|
boxBottom = gs_render_save();
|
|
|
|
|
2014-08-07 23:42:07 -07:00
|
|
|
gs_render_start(true);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
for (int i = 0; i <= 360; i += (360 / 20)) {
|
|
|
|
float pos = RAD(float(i));
|
|
|
|
gs_vertex2f(cosf(pos), sinf(pos));
|
|
|
|
}
|
2014-08-07 23:42:07 -07:00
|
|
|
circle = gs_render_save();
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
|
2014-08-04 05:48:58 -07:00
|
|
|
obs_leave_graphics();
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
}
|
|
|
|
|
2016-12-09 14:40:04 -08:00
|
|
|
void OBSBasic::ReplayBufferClicked()
|
|
|
|
{
|
|
|
|
if (outputHandler->ReplayBufferActive())
|
|
|
|
StopReplayBuffer();
|
|
|
|
else
|
|
|
|
StartReplayBuffer();
|
|
|
|
};
|
|
|
|
|
2020-06-20 07:42:14 -07:00
|
|
|
void OBSBasic::AddVCamButton()
|
|
|
|
{
|
|
|
|
vcamButton = new ReplayBufferButton(QTStr("Basic.Main.StartVirtualCam"),
|
|
|
|
this);
|
|
|
|
vcamButton->setCheckable(true);
|
|
|
|
connect(vcamButton.data(), &QPushButton::clicked, this,
|
|
|
|
&OBSBasic::VCamButtonClicked);
|
|
|
|
|
|
|
|
vcamButton->setProperty("themeID", "vcamButton");
|
|
|
|
ui->buttonsVLayout->insertWidget(2, vcamButton);
|
|
|
|
}
|
|
|
|
|
2015-02-06 03:17:33 -08:00
|
|
|
void OBSBasic::ResetOutputs()
|
|
|
|
{
|
2015-07-11 08:04:12 +02:00
|
|
|
ProfileScope("OBSBasic::ResetOutputs");
|
|
|
|
|
2015-01-26 13:41:22 -08:00
|
|
|
const char *mode = config_get_string(basicConfig, "Output", "Mode");
|
|
|
|
bool advOut = astrcmpi(mode, "Advanced") == 0;
|
|
|
|
|
2015-02-06 03:17:33 -08:00
|
|
|
if (!outputHandler || !outputHandler->Active()) {
|
|
|
|
outputHandler.reset();
|
2015-01-26 13:41:22 -08:00
|
|
|
outputHandler.reset(advOut ? CreateAdvancedOutputHandler(this)
|
|
|
|
: CreateSimpleOutputHandler(this));
|
2016-12-07 05:21:44 -08:00
|
|
|
|
2016-12-09 14:40:04 -08:00
|
|
|
delete replayBufferButton;
|
2019-12-12 17:07:26 -08:00
|
|
|
delete replayLayout;
|
2016-12-09 14:40:04 -08:00
|
|
|
|
|
|
|
if (outputHandler->replayBuffer) {
|
2019-12-12 17:07:26 -08:00
|
|
|
replayBufferButton = new ReplayBufferButton(
|
2016-12-09 14:40:04 -08:00
|
|
|
QTStr("Basic.Main.StartReplayBuffer"), this);
|
2018-06-27 22:15:15 -05:00
|
|
|
replayBufferButton->setCheckable(true);
|
2016-12-11 21:04:58 -08:00
|
|
|
connect(replayBufferButton.data(),
|
2019-09-12 13:55:31 -07:00
|
|
|
&QPushButton::clicked, this,
|
2016-12-09 14:40:04 -08:00
|
|
|
&OBSBasic::ReplayBufferClicked);
|
|
|
|
|
2020-02-05 04:04:53 -06:00
|
|
|
replayBufferButton->setSizePolicy(QSizePolicy::Ignored,
|
|
|
|
QSizePolicy::Fixed);
|
|
|
|
|
2019-12-12 17:07:26 -08:00
|
|
|
replayLayout = new QHBoxLayout(this);
|
|
|
|
replayLayout->addWidget(replayBufferButton);
|
|
|
|
|
2017-06-13 14:34:05 -05:00
|
|
|
replayBufferButton->setProperty("themeID",
|
|
|
|
"replayBufferButton");
|
2019-12-12 17:07:26 -08:00
|
|
|
ui->buttonsVLayout->insertLayout(2, replayLayout);
|
2016-12-09 14:40:04 -08:00
|
|
|
}
|
2016-12-07 05:21:44 -08:00
|
|
|
|
2016-12-09 14:40:04 -08:00
|
|
|
if (sysTrayReplayBuffer)
|
|
|
|
sysTrayReplayBuffer->setEnabled(
|
|
|
|
!!outputHandler->replayBuffer);
|
2015-02-06 03:17:33 -08:00
|
|
|
} else {
|
|
|
|
outputHandler->Update();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-02 16:11:04 -08:00
|
|
|
static void AddProjectorMenuMonitors(QMenu *parent, QObject *target,
|
|
|
|
const char *slot);
|
|
|
|
|
2016-08-05 17:44:12 -07:00
|
|
|
#define STARTUP_SEPARATOR \
|
|
|
|
"==== Startup complete ==============================================="
|
|
|
|
#define SHUTDOWN_SEPARATOR \
|
|
|
|
"==== Shutting down =================================================="
|
2015-07-05 23:42:46 -07:00
|
|
|
|
2016-12-21 19:17:39 -08:00
|
|
|
#define UNSUPPORTED_ERROR \
|
|
|
|
"Failed to initialize video:\n\nRequired graphics API functionality " \
|
|
|
|
"not found. Your GPU may not be supported."
|
|
|
|
|
|
|
|
#define UNKNOWN_ERROR \
|
2016-12-22 06:57:26 -08:00
|
|
|
"Failed to initialize video. Your GPU may not be supported, " \
|
|
|
|
"or your graphics drivers may need to be updated."
|
2016-12-21 19:17:39 -08:00
|
|
|
|
2014-02-02 15:23:38 -07:00
|
|
|
void OBSBasic::OBSInit()
|
|
|
|
{
|
2015-07-11 08:04:12 +02:00
|
|
|
ProfileScope("OBSBasic::OBSInit");
|
|
|
|
|
2015-06-23 19:29:07 -07:00
|
|
|
const char *sceneCollection = config_get_string(
|
|
|
|
App()->GlobalConfig(), "Basic", "SceneCollectionFile");
|
2015-01-15 23:44:38 -08:00
|
|
|
char savePath[512];
|
2015-06-23 19:29:07 -07:00
|
|
|
char fileName[512];
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (!sceneCollection)
|
|
|
|
throw "Failed to get scene collection name";
|
|
|
|
|
|
|
|
ret = snprintf(fileName, 512, "obs-studio/basic/scenes/%s.json",
|
|
|
|
sceneCollection);
|
2015-01-15 23:44:38 -08:00
|
|
|
if (ret <= 0)
|
2015-06-23 19:29:07 -07:00
|
|
|
throw "Failed to create scene collection file name";
|
|
|
|
|
|
|
|
ret = GetConfigPath(savePath, sizeof(savePath), fileName);
|
|
|
|
if (ret <= 0)
|
|
|
|
throw "Failed to get scene collection json file path";
|
2014-05-03 22:54:38 -07:00
|
|
|
|
2014-03-06 21:08:12 -07:00
|
|
|
if (!InitBasicConfig())
|
|
|
|
throw "Failed to load basic.ini";
|
2014-02-22 20:14:19 -07:00
|
|
|
if (!ResetAudio())
|
2014-02-02 15:23:38 -07:00
|
|
|
throw "Failed to initialize audio";
|
|
|
|
|
2015-01-15 23:44:38 -08:00
|
|
|
ret = ResetVideo();
|
2014-07-20 17:40:57 -07:00
|
|
|
|
|
|
|
switch (ret) {
|
|
|
|
case OBS_VIDEO_MODULE_NOT_FOUND:
|
|
|
|
throw "Failed to initialize video: Graphics module not found";
|
|
|
|
case OBS_VIDEO_NOT_SUPPORTED:
|
2016-12-21 19:17:39 -08:00
|
|
|
throw UNSUPPORTED_ERROR;
|
2014-07-20 17:40:57 -07:00
|
|
|
case OBS_VIDEO_INVALID_PARAM:
|
|
|
|
throw "Failed to initialize video: Invalid parameters";
|
|
|
|
default:
|
|
|
|
if (ret != OBS_VIDEO_SUCCESS)
|
2016-12-21 19:17:39 -08:00
|
|
|
throw UNKNOWN_ERROR;
|
2014-07-20 17:40:57 -07:00
|
|
|
}
|
|
|
|
|
2017-03-02 11:38:55 -08:00
|
|
|
/* load audio monitoring */
|
2017-10-16 17:28:50 -05:00
|
|
|
#if defined(_WIN32) || defined(__APPLE__) || HAVE_PULSEAUDIO
|
2017-03-02 11:38:55 -08:00
|
|
|
const char *device_name =
|
|
|
|
config_get_string(basicConfig, "Audio", "MonitoringDeviceName");
|
|
|
|
const char *device_id =
|
|
|
|
config_get_string(basicConfig, "Audio", "MonitoringDeviceId");
|
|
|
|
|
|
|
|
obs_set_audio_monitoring_device(device_name, device_id);
|
2017-03-02 12:35:37 -08:00
|
|
|
|
|
|
|
blog(LOG_INFO, "Audio monitoring device:\n\tname: %s\n\tid: %s",
|
|
|
|
device_name, device_id);
|
2017-03-02 11:38:55 -08:00
|
|
|
#endif
|
|
|
|
|
2014-05-03 22:54:38 -07:00
|
|
|
InitOBSCallbacks();
|
2014-11-01 21:48:58 +01:00
|
|
|
InitHotkeys();
|
2014-02-02 15:23:38 -07:00
|
|
|
|
2014-07-27 12:48:14 -07:00
|
|
|
AddExtraModulePaths();
|
2016-08-05 17:46:00 -07:00
|
|
|
blog(LOG_INFO, "---------------------------------");
|
(API Change) Refactor module handling
Changed API:
- char *obs_find_plugin_file(const char *sub_path);
Changed to: char *obs_module_file(const char *file);
Cahnge it so you no longer need to specify a sub-path such as:
obs_find_plugin_file("module_name/file.ext")
Instead, now automatically handle the module data path so all you need
to do is:
obs_module_file("file.ext")
- int obs_load_module(const char *name);
Changed to: int obs_open_module(obs_module_t *module,
const char *path,
const char *data_path);
bool obs_init_module(obs_module_t module);
Change the module loading API so that if the front-end chooses, it can
load modules directly from a specified path, and associate a data
directory with it on the spot.
The module will not be initialized immediately; obs_init_module must
be called on the module pointer in order to fully initialize the
module. This is done so a module can be disabled by the front-end if
the it so chooses.
New API:
- void obs_add_module_path(const char *bin, const char *data);
These functions allow you to specify new module search paths to add,
and allow you to search through them, or optionally just load all
modules from them. If the string %module% is included, it will
replace it with the module's name when that string is used as a
lookup. Data paths are now directly added to the module's internal
storage structure, and when obs_find_module_file is used, it will look
up the pointer to the obs_module structure and get its data directory
that way.
Example:
obs_add_module_path("/opt/obs/my-modules/%module%/bin",
"/opt/obs/my-modules/%module%/data");
This would cause it to additionally look for the binary of a
hypthetical module named "foo" at /opt/obs/my-modules/foo/bin/foo.so
(or libfoo.so), and then look for the data in
/opt/obs/my-modules/foo/data.
This gives the front-end more flexibility for handling third-party
plugin modules, or handling all plugin modules in a custom way.
- void obs_find_modules(obs_find_module_callback_t callback, void
*param);
This searches the existing paths for modules and calls the callback
function when any are found. Useful for plugin management and custom
handling of the paths by the front-end if desired.
- void obs_load_all_modules(void);
Search through the paths and both loads and initializes all modules
automatically without custom handling.
- void obs_enum_modules(obs_enum_module_callback_t callback,
void *param);
Enumerates currently opened modules.
2014-07-27 12:00:11 -07:00
|
|
|
obs_load_all_modules();
|
2016-08-05 17:46:00 -07:00
|
|
|
blog(LOG_INFO, "---------------------------------");
|
|
|
|
obs_log_loaded_modules();
|
2017-07-14 23:26:53 -04:00
|
|
|
blog(LOG_INFO, "---------------------------------");
|
|
|
|
obs_post_load_modules();
|
2014-03-07 17:03:34 -07:00
|
|
|
|
2018-11-17 12:01:55 -08:00
|
|
|
#ifdef BROWSER_AVAILABLE
|
2019-02-06 21:38:34 -08:00
|
|
|
cef = obs_browser_init_panel();
|
2018-07-27 21:35:08 -07:00
|
|
|
#endif
|
|
|
|
|
2020-06-20 07:42:14 -07:00
|
|
|
obs_data_t *obsData = obs_get_private_data();
|
|
|
|
vcamEnabled = obs_data_get_bool(obsData, "vcamEnabled");
|
|
|
|
if (vcamEnabled) {
|
|
|
|
AddVCamButton();
|
|
|
|
}
|
|
|
|
obs_data_release(obsData);
|
|
|
|
|
2019-02-05 18:05:19 -08:00
|
|
|
InitBasicConfigDefaults2();
|
|
|
|
|
2018-01-20 13:56:33 -08:00
|
|
|
CheckForSimpleModeX264Fallback();
|
|
|
|
|
2016-08-05 17:44:12 -07:00
|
|
|
blog(LOG_INFO, STARTUP_SEPARATOR);
|
2015-07-05 23:42:46 -07:00
|
|
|
|
2015-02-06 03:17:33 -08:00
|
|
|
ResetOutputs();
|
2014-11-01 21:53:25 +01:00
|
|
|
CreateHotkeys();
|
2015-02-06 03:17:33 -08:00
|
|
|
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
if (!InitService())
|
|
|
|
throw "Failed to initialize service";
|
|
|
|
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
InitPrimitives();
|
|
|
|
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
sceneDuplicationMode = config_get_bool(
|
|
|
|
App()->GlobalConfig(), "BasicWindow", "SceneDuplicationMode");
|
|
|
|
swapScenesMode = config_get_bool(App()->GlobalConfig(), "BasicWindow",
|
|
|
|
"SwapScenesMode");
|
|
|
|
editPropertiesMode = config_get_bool(
|
|
|
|
App()->GlobalConfig(), "BasicWindow", "EditPropertiesMode");
|
2017-01-11 13:19:17 -06:00
|
|
|
|
2016-06-30 14:52:56 -07:00
|
|
|
#define SET_VISIBILITY(name, control) \
|
|
|
|
do { \
|
|
|
|
if (config_has_user_value(App()->GlobalConfig(), \
|
|
|
|
"BasicWindow", name)) { \
|
|
|
|
bool visible = config_get_bool(App()->GlobalConfig(), \
|
|
|
|
"BasicWindow", name); \
|
|
|
|
ui->control->setChecked(visible); \
|
|
|
|
} \
|
|
|
|
} while (false)
|
|
|
|
|
|
|
|
SET_VISIBILITY("ShowListboxToolbars", toggleListboxToolbars);
|
|
|
|
SET_VISIBILITY("ShowStatusBar", toggleStatusBar);
|
|
|
|
#undef SET_VISIBILITY
|
|
|
|
|
2020-01-20 23:18:46 -08:00
|
|
|
bool sourceIconsVisible = config_get_bool(
|
|
|
|
GetGlobalConfig(), "BasicWindow", "ShowSourceIcons");
|
|
|
|
ui->toggleSourceIcons->setChecked(sourceIconsVisible);
|
|
|
|
|
2018-08-23 20:43:44 -05:00
|
|
|
if (config_has_user_value(App()->GlobalConfig(), "BasicWindow",
|
|
|
|
"ShowContextToolbars")) {
|
|
|
|
bool visible = config_get_bool(App()->GlobalConfig(),
|
|
|
|
"BasicWindow",
|
|
|
|
"ShowContextToolbars");
|
|
|
|
ui->toggleContextBar->setChecked(visible);
|
|
|
|
ui->contextContainer->setVisible(visible);
|
|
|
|
} else {
|
|
|
|
ui->toggleContextBar->setChecked(true);
|
|
|
|
ui->contextContainer->setVisible(true);
|
|
|
|
}
|
|
|
|
|
2018-05-01 12:01:11 -07:00
|
|
|
{
|
|
|
|
ProfileScope("OBSBasic::Load");
|
|
|
|
disableSaving--;
|
|
|
|
Load(savePath);
|
|
|
|
disableSaving++;
|
|
|
|
}
|
|
|
|
|
2014-07-13 23:56:28 -07:00
|
|
|
TimedCheckForUpdates();
|
2014-06-16 19:41:36 -07:00
|
|
|
loaded = true;
|
2015-01-04 08:20:15 -08:00
|
|
|
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
previewEnabled = config_get_bool(App()->GlobalConfig(), "BasicWindow",
|
2015-04-02 21:35:46 -07:00
|
|
|
"PreviewEnabled");
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
|
|
|
|
if (!previewEnabled && !IsPreviewProgramMode())
|
|
|
|
QMetaObject::invokeMethod(this, "EnablePreviewDisplay",
|
|
|
|
Qt::QueuedConnection,
|
|
|
|
Q_ARG(bool, previewEnabled));
|
2015-05-25 01:37:13 -07:00
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
uint32_t winVer = GetWindowsVersion();
|
|
|
|
if (winVer > 0 && winVer < 0x602) {
|
|
|
|
bool disableAero =
|
|
|
|
config_get_bool(basicConfig, "Video", "DisableAero");
|
|
|
|
SetAeroEnabled(!disableAero);
|
|
|
|
}
|
|
|
|
#endif
|
2015-06-30 05:49:31 -07:00
|
|
|
|
2018-05-01 12:01:11 -07:00
|
|
|
RefreshSceneCollections();
|
2015-06-23 19:38:01 -07:00
|
|
|
RefreshProfiles();
|
2015-06-30 05:49:31 -07:00
|
|
|
disableSaving--;
|
2015-08-02 00:02:58 -07:00
|
|
|
|
|
|
|
auto addDisplay = [this](OBSQTDisplay *window) {
|
|
|
|
obs_display_add_draw_callback(window->GetDisplay(),
|
|
|
|
OBSBasic::RenderMain, this);
|
|
|
|
|
|
|
|
struct obs_video_info ovi;
|
|
|
|
if (obs_get_video_info(&ovi))
|
|
|
|
ResizePreview(ovi.base_width, ovi.base_height);
|
|
|
|
};
|
|
|
|
|
|
|
|
connect(ui->preview, &OBSQTDisplay::DisplayCreated, addDisplay);
|
|
|
|
|
2016-01-22 07:33:29 -08:00
|
|
|
#ifdef _WIN32
|
2016-09-26 12:40:23 -07:00
|
|
|
SetWin32DropStyle(this);
|
2016-01-22 07:33:29 -08:00
|
|
|
show();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
bool alwaysOnTop = config_get_bool(App()->GlobalConfig(), "BasicWindow",
|
|
|
|
"AlwaysOnTop");
|
2017-04-30 07:47:19 -07:00
|
|
|
if (alwaysOnTop || opt_always_on_top) {
|
2016-01-22 07:33:29 -08:00
|
|
|
SetAlwaysOnTop(this, true);
|
|
|
|
ui->actionAlwaysOnTop->setChecked(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef _WIN32
|
2015-08-02 00:02:58 -07:00
|
|
|
show();
|
2016-01-22 07:33:29 -08:00
|
|
|
#endif
|
2016-01-09 14:20:46 -08:00
|
|
|
|
2018-08-28 07:39:50 -07:00
|
|
|
/* setup stats dock */
|
2018-09-16 10:02:43 -07:00
|
|
|
OBSBasicStats *statsDlg = new OBSBasicStats(statsDock, false);
|
|
|
|
statsDock->setWidget(statsDlg);
|
2018-08-28 07:39:50 -07:00
|
|
|
|
2019-08-08 03:27:45 -07:00
|
|
|
/* ----------------------------- */
|
|
|
|
/* add custom browser docks */
|
|
|
|
|
|
|
|
#ifdef BROWSER_AVAILABLE
|
|
|
|
if (cef) {
|
|
|
|
QAction *action = new QAction(QTStr("Basic.MainMenu."
|
|
|
|
"View.Docks."
|
2020-02-19 21:29:26 -06:00
|
|
|
"CustomBrowserDocks"),
|
|
|
|
this);
|
2019-08-08 03:27:45 -07:00
|
|
|
ui->viewMenuDocks->insertAction(ui->toggleScenes, action);
|
|
|
|
connect(action, &QAction::triggered, this,
|
|
|
|
&OBSBasic::ManageExtraBrowserDocks);
|
|
|
|
ui->viewMenuDocks->insertSeparator(ui->toggleScenes);
|
|
|
|
|
|
|
|
LoadExtraBrowserDocks();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2017-07-27 02:40:41 -07:00
|
|
|
const char *dockStateStr = config_get_string(
|
|
|
|
App()->GlobalConfig(), "BasicWindow", "DockState");
|
2018-08-23 20:43:44 -05:00
|
|
|
|
2017-07-27 02:40:41 -07:00
|
|
|
if (!dockStateStr) {
|
|
|
|
on_resetUI_triggered();
|
2016-01-09 14:20:46 -08:00
|
|
|
} else {
|
2017-07-27 02:40:41 -07:00
|
|
|
QByteArray dockState =
|
|
|
|
QByteArray::fromBase64(QByteArray(dockStateStr));
|
|
|
|
if (!restoreState(dockState))
|
|
|
|
on_resetUI_triggered();
|
2016-01-09 14:20:46 -08:00
|
|
|
}
|
|
|
|
|
2019-02-09 00:43:05 -08:00
|
|
|
bool pre23Defaults = config_get_bool(App()->GlobalConfig(), "General",
|
|
|
|
"Pre23Defaults");
|
|
|
|
if (pre23Defaults) {
|
|
|
|
bool resetDockLock23 = config_get_bool(
|
|
|
|
App()->GlobalConfig(), "General", "ResetDockLock23");
|
|
|
|
if (!resetDockLock23) {
|
|
|
|
config_set_bool(App()->GlobalConfig(), "General",
|
|
|
|
"ResetDockLock23", true);
|
|
|
|
config_remove_value(App()->GlobalConfig(),
|
|
|
|
"BasicWindow", "DocksLocked");
|
|
|
|
config_save_safe(App()->GlobalConfig(), "tmp", nullptr);
|
|
|
|
}
|
|
|
|
}
|
2017-07-27 02:40:41 -07:00
|
|
|
|
|
|
|
bool docksLocked = config_get_bool(App()->GlobalConfig(), "BasicWindow",
|
|
|
|
"DocksLocked");
|
|
|
|
on_lockUI_toggled(docksLocked);
|
|
|
|
ui->lockUI->blockSignals(true);
|
|
|
|
ui->lockUI->setChecked(docksLocked);
|
|
|
|
ui->lockUI->blockSignals(false);
|
2016-08-13 09:36:17 -05:00
|
|
|
|
2018-08-17 03:50:25 -07:00
|
|
|
#ifndef __APPLE__
|
2016-08-13 09:36:17 -05:00
|
|
|
SystemTray(true);
|
2018-08-17 03:50:25 -07:00
|
|
|
#endif
|
2016-12-29 09:21:53 -06:00
|
|
|
|
2017-05-06 12:13:56 -07:00
|
|
|
bool has_last_version = config_has_user_value(App()->GlobalConfig(),
|
|
|
|
"General", "LastVersion");
|
|
|
|
bool first_run =
|
|
|
|
config_get_bool(App()->GlobalConfig(), "General", "FirstRun");
|
|
|
|
|
|
|
|
if (!first_run) {
|
|
|
|
config_set_bool(App()->GlobalConfig(), "General", "FirstRun",
|
|
|
|
true);
|
|
|
|
config_save_safe(App()->GlobalConfig(), "tmp", nullptr);
|
|
|
|
}
|
|
|
|
|
2020-04-17 23:59:20 -05:00
|
|
|
if (!first_run && !has_last_version && !Active())
|
|
|
|
QMetaObject::invokeMethod(this, "on_autoConfigure_triggered",
|
|
|
|
Qt::QueuedConnection);
|
2017-05-13 20:47:47 -07:00
|
|
|
|
2018-06-12 22:52:08 -03:00
|
|
|
ToggleMixerLayout(config_get_bool(App()->GlobalConfig(), "BasicWindow",
|
|
|
|
"VerticalVolControl"));
|
2018-04-27 23:49:48 -03:00
|
|
|
|
2017-05-13 20:47:47 -07:00
|
|
|
if (config_get_bool(basicConfig, "General", "OpenStatsOnStartup"))
|
|
|
|
on_stats_triggered();
|
2017-06-19 15:33:12 -07:00
|
|
|
|
|
|
|
OBSBasicStats::InitializeValues();
|
2017-12-02 16:11:04 -08:00
|
|
|
|
|
|
|
/* ----------------------- */
|
|
|
|
/* Add multiview menu */
|
|
|
|
|
|
|
|
ui->viewMenu->addSeparator();
|
|
|
|
|
2020-04-19 09:31:18 -05:00
|
|
|
multiviewProjectorMenu = new QMenu(QTStr("MultiviewProjector"));
|
2017-12-02 16:11:04 -08:00
|
|
|
ui->viewMenu->addMenu(multiviewProjectorMenu);
|
2018-04-05 05:54:24 -03:00
|
|
|
AddProjectorMenuMonitors(multiviewProjectorMenu, this,
|
|
|
|
SLOT(OpenMultiviewProjector()));
|
2020-04-19 09:31:18 -05:00
|
|
|
connect(ui->viewMenu->menuAction(), &QAction::hovered, this,
|
|
|
|
&OBSBasic::UpdateMultiviewProjectorMenu);
|
2017-12-02 16:11:04 -08:00
|
|
|
ui->viewMenu->addAction(QTStr("MultiviewWindowed"), this,
|
|
|
|
SLOT(OpenMultiviewWindow()));
|
2018-03-12 19:56:07 -05:00
|
|
|
|
2019-07-27 23:59:16 -05:00
|
|
|
ui->sources->UpdateIcons();
|
|
|
|
|
2020-01-13 22:40:03 -06:00
|
|
|
if (!opt_studio_mode) {
|
|
|
|
SetPreviewProgramMode(config_get_bool(App()->GlobalConfig(),
|
|
|
|
"BasicWindow",
|
|
|
|
"PreviewProgramMode"));
|
|
|
|
} else {
|
|
|
|
SetPreviewProgramMode(true);
|
|
|
|
opt_studio_mode = false;
|
|
|
|
}
|
|
|
|
|
2018-03-12 19:56:07 -05:00
|
|
|
#if !defined(_WIN32) && !defined(__APPLE__)
|
2018-03-16 16:27:49 -07:00
|
|
|
delete ui->actionShowCrashLogs;
|
|
|
|
delete ui->actionUploadLastCrashLog;
|
|
|
|
delete ui->menuCrashLogs;
|
2018-03-12 19:56:07 -05:00
|
|
|
delete ui->actionCheckForUpdates;
|
2018-03-16 16:27:49 -07:00
|
|
|
ui->actionShowCrashLogs = nullptr;
|
|
|
|
ui->actionUploadLastCrashLog = nullptr;
|
|
|
|
ui->menuCrashLogs = nullptr;
|
2018-03-12 19:56:07 -05:00
|
|
|
ui->actionCheckForUpdates = nullptr;
|
2020-07-18 21:33:41 +02:00
|
|
|
#elif _WIN32 || __APPLE__
|
|
|
|
if (App()->IsUpdaterDisabled())
|
|
|
|
ui->actionCheckForUpdates->setEnabled(false);
|
2018-03-12 19:56:07 -05:00
|
|
|
#endif
|
2018-04-12 13:11:33 -07:00
|
|
|
|
UI: Remove mac browser workarounds, improve stability
The workarounds were made because of conflicts with running multiple UI
threads at once on macOS, which macOS can't do very well, and would be
susceptible to crashes. This would cause crashes not only on startup
but seemingly at random when using the browser source on macOS. The
original "fix" was a hack to try to minimize UI code and browser UI code
from executing at the same time. The macOS initial scene loading was
deferred until all Qt-related and main window initialization was
completed. Although this worked to some extent to prevent conflicts, it
made it so that there was an initial period on startup where the entire
UI seemed "blank" for users, and it was still possible for the main UI
thread and the browser UI thread to clash, causing crashes seemingly at
random for users.
The external message pump method of CEF is the solution to the problem,
which is the method which allows the main UI thread to share events with
CEF. To do this, all CEF operations need to be performed in the UI
thread (Qt's main thread), and CefDoMessageLoopWork() needs to be called
when CefApp::OnScheduleMessagePumpWork callback is triggered. A number
of other issues had to be solved as well, such as CefBrowser references
getting "stuck" in the Qt event queue.
With this, macOS no longer needs to do the "deferred load" hack, and
browsers are now much more stable and no longer as susceptible to
seemingly random crashes, improving overall program stability when
browsers are used.
2019-05-01 12:13:35 -07:00
|
|
|
OnFirstLoad();
|
|
|
|
|
2020-05-11 22:43:54 +02:00
|
|
|
activateWindow();
|
|
|
|
|
2018-05-01 12:01:11 -07:00
|
|
|
#ifdef __APPLE__
|
UI: Remove mac browser workarounds, improve stability
The workarounds were made because of conflicts with running multiple UI
threads at once on macOS, which macOS can't do very well, and would be
susceptible to crashes. This would cause crashes not only on startup
but seemingly at random when using the browser source on macOS. The
original "fix" was a hack to try to minimize UI code and browser UI code
from executing at the same time. The macOS initial scene loading was
deferred until all Qt-related and main window initialization was
completed. Although this worked to some extent to prevent conflicts, it
made it so that there was an initial period on startup where the entire
UI seemed "blank" for users, and it was still possible for the main UI
thread and the browser UI thread to clash, causing crashes seemingly at
random for users.
The external message pump method of CEF is the solution to the problem,
which is the method which allows the main UI thread to share events with
CEF. To do this, all CEF operations need to be performed in the UI
thread (Qt's main thread), and CefDoMessageLoopWork() needs to be called
when CefApp::OnScheduleMessagePumpWork callback is triggered. A number
of other issues had to be solved as well, such as CefBrowser references
getting "stuck" in the Qt event queue.
With this, macOS no longer needs to do the "deferred load" hack, and
browsers are now much more stable and no longer as susceptible to
seemingly random crashes, improving overall program stability when
browsers are used.
2019-05-01 12:13:35 -07:00
|
|
|
QMetaObject::invokeMethod(this, "DeferredSysTrayLoad",
|
2018-04-12 13:11:33 -07:00
|
|
|
Qt::QueuedConnection, Q_ARG(int, 10));
|
2018-05-01 12:01:11 -07:00
|
|
|
#endif
|
2018-04-12 13:11:33 -07:00
|
|
|
}
|
|
|
|
|
2018-06-25 18:54:32 -05:00
|
|
|
void OBSBasic::OnFirstLoad()
|
|
|
|
{
|
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_FINISHED_LOADING);
|
2018-07-27 21:35:08 -07:00
|
|
|
|
2019-05-01 12:12:03 -07:00
|
|
|
#if defined(BROWSER_AVAILABLE) && defined(_WIN32)
|
2018-07-27 21:35:08 -07:00
|
|
|
/* Attempt to load init screen if available */
|
2019-02-06 21:38:34 -08:00
|
|
|
if (cef) {
|
2018-07-27 21:35:08 -07:00
|
|
|
WhatsNewInfoThread *wnit = new WhatsNewInfoThread();
|
|
|
|
if (wnit) {
|
|
|
|
connect(wnit, &WhatsNewInfoThread::Result, this,
|
|
|
|
&OBSBasic::ReceivedIntroJson);
|
|
|
|
}
|
|
|
|
if (wnit) {
|
2018-07-31 21:11:31 -07:00
|
|
|
introCheckThread.reset(wnit);
|
2018-07-27 21:35:08 -07:00
|
|
|
introCheckThread->start();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2019-02-06 22:24:25 -08:00
|
|
|
|
|
|
|
Auth::Load();
|
2020-05-15 07:34:54 -05:00
|
|
|
|
|
|
|
if (logView && logView->ShowOnStartup())
|
|
|
|
logView->show();
|
2018-06-25 18:54:32 -05:00
|
|
|
}
|
|
|
|
|
UI: Remove mac browser workarounds, improve stability
The workarounds were made because of conflicts with running multiple UI
threads at once on macOS, which macOS can't do very well, and would be
susceptible to crashes. This would cause crashes not only on startup
but seemingly at random when using the browser source on macOS. The
original "fix" was a hack to try to minimize UI code and browser UI code
from executing at the same time. The macOS initial scene loading was
deferred until all Qt-related and main window initialization was
completed. Although this worked to some extent to prevent conflicts, it
made it so that there was an initial period on startup where the entire
UI seemed "blank" for users, and it was still possible for the main UI
thread and the browser UI thread to clash, causing crashes seemingly at
random for users.
The external message pump method of CEF is the solution to the problem,
which is the method which allows the main UI thread to share events with
CEF. To do this, all CEF operations need to be performed in the UI
thread (Qt's main thread), and CefDoMessageLoopWork() needs to be called
when CefApp::OnScheduleMessagePumpWork callback is triggered. A number
of other issues had to be solved as well, such as CefBrowser references
getting "stuck" in the Qt event queue.
With this, macOS no longer needs to do the "deferred load" hack, and
browsers are now much more stable and no longer as susceptible to
seemingly random crashes, improving overall program stability when
browsers are used.
2019-05-01 12:13:35 -07:00
|
|
|
void OBSBasic::DeferredSysTrayLoad(int requeueCount)
|
2018-04-12 13:11:33 -07:00
|
|
|
{
|
|
|
|
if (--requeueCount > 0) {
|
UI: Remove mac browser workarounds, improve stability
The workarounds were made because of conflicts with running multiple UI
threads at once on macOS, which macOS can't do very well, and would be
susceptible to crashes. This would cause crashes not only on startup
but seemingly at random when using the browser source on macOS. The
original "fix" was a hack to try to minimize UI code and browser UI code
from executing at the same time. The macOS initial scene loading was
deferred until all Qt-related and main window initialization was
completed. Although this worked to some extent to prevent conflicts, it
made it so that there was an initial period on startup where the entire
UI seemed "blank" for users, and it was still possible for the main UI
thread and the browser UI thread to clash, causing crashes seemingly at
random for users.
The external message pump method of CEF is the solution to the problem,
which is the method which allows the main UI thread to share events with
CEF. To do this, all CEF operations need to be performed in the UI
thread (Qt's main thread), and CefDoMessageLoopWork() needs to be called
when CefApp::OnScheduleMessagePumpWork callback is triggered. A number
of other issues had to be solved as well, such as CefBrowser references
getting "stuck" in the Qt event queue.
With this, macOS no longer needs to do the "deferred load" hack, and
browsers are now much more stable and no longer as susceptible to
seemingly random crashes, improving overall program stability when
browsers are used.
2019-05-01 12:13:35 -07:00
|
|
|
QMetaObject::invokeMethod(this, "DeferredSysTrayLoad",
|
2018-04-12 13:11:33 -07:00
|
|
|
Qt::QueuedConnection,
|
|
|
|
Q_ARG(int, requeueCount));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-08-17 03:50:25 -07:00
|
|
|
/* Minimizng to tray on initial startup does not work on mac
|
|
|
|
* unless it is done in the deferred load */
|
|
|
|
SystemTray(true);
|
2014-02-02 15:23:38 -07:00
|
|
|
}
|
|
|
|
|
2018-07-27 21:35:08 -07:00
|
|
|
/* shows a "what's new" page on startup of new versions using CEF */
|
|
|
|
void OBSBasic::ReceivedIntroJson(const QString &text)
|
|
|
|
{
|
2018-09-18 08:54:30 -07:00
|
|
|
#ifdef BROWSER_AVAILABLE
|
2018-07-27 21:35:08 -07:00
|
|
|
#ifdef _WIN32
|
2020-03-16 08:46:47 -07:00
|
|
|
if (closing)
|
|
|
|
return;
|
|
|
|
|
2018-07-27 21:35:08 -07:00
|
|
|
std::string err;
|
|
|
|
Json json = Json::parse(QT_TO_UTF8(text), err);
|
|
|
|
if (!err.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
std::string info_url;
|
|
|
|
int info_increment = -1;
|
|
|
|
|
|
|
|
/* check to see if there's an info page for this version */
|
|
|
|
const Json::array &items = json.array_items();
|
|
|
|
for (const Json &item : items) {
|
|
|
|
const std::string &version = item["version"].string_value();
|
|
|
|
const std::string &url = item["url"].string_value();
|
|
|
|
int increment = item["increment"].int_value();
|
2018-08-13 00:26:02 -07:00
|
|
|
int rc = item["RC"].int_value();
|
2018-07-27 21:35:08 -07:00
|
|
|
|
|
|
|
int major = 0;
|
|
|
|
int minor = 0;
|
|
|
|
|
|
|
|
sscanf(version.c_str(), "%d.%d", &major, &minor);
|
2018-08-13 00:26:02 -07:00
|
|
|
#if OBS_RELEASE_CANDIDATE > 0
|
|
|
|
if (major == OBS_RELEASE_CANDIDATE_MAJOR &&
|
|
|
|
minor == OBS_RELEASE_CANDIDATE_MINOR &&
|
|
|
|
rc == OBS_RELEASE_CANDIDATE) {
|
|
|
|
#else
|
2018-07-27 21:35:08 -07:00
|
|
|
if (major == LIBOBS_API_MAJOR_VER &&
|
2018-08-13 00:26:02 -07:00
|
|
|
minor == LIBOBS_API_MINOR_VER && rc == 0) {
|
|
|
|
#endif
|
2018-07-27 21:35:08 -07:00
|
|
|
info_url = url;
|
|
|
|
info_increment = increment;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* this version was not found, or no info for this version */
|
|
|
|
if (info_increment == -1) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-08-13 00:26:02 -07:00
|
|
|
#if OBS_RELEASE_CANDIDATE > 0
|
|
|
|
uint32_t lastVersion = config_get_int(App()->GlobalConfig(), "General",
|
|
|
|
"LastRCVersion");
|
|
|
|
#else
|
2018-07-27 21:35:08 -07:00
|
|
|
uint32_t lastVersion =
|
|
|
|
config_get_int(App()->GlobalConfig(), "General", "LastVersion");
|
2018-08-13 00:26:02 -07:00
|
|
|
#endif
|
2018-07-27 21:35:08 -07:00
|
|
|
|
|
|
|
int current_version_increment = -1;
|
|
|
|
|
2018-08-13 00:26:02 -07:00
|
|
|
#if OBS_RELEASE_CANDIDATE > 0
|
|
|
|
if (lastVersion < OBS_RELEASE_CANDIDATE_VER) {
|
|
|
|
#else
|
2019-03-06 15:36:19 -08:00
|
|
|
if ((lastVersion & ~0xFFFF) < (LIBOBS_API_VER & ~0xFFFF)) {
|
2018-08-13 00:26:02 -07:00
|
|
|
#endif
|
2018-07-27 21:35:08 -07:00
|
|
|
config_set_int(App()->GlobalConfig(), "General",
|
|
|
|
"InfoIncrement", -1);
|
|
|
|
} else {
|
|
|
|
current_version_increment = config_get_int(
|
|
|
|
App()->GlobalConfig(), "General", "InfoIncrement");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (info_increment <= current_version_increment) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
config_set_int(App()->GlobalConfig(), "General", "InfoIncrement",
|
|
|
|
info_increment);
|
|
|
|
|
2019-02-26 12:00:27 -08:00
|
|
|
/* Don't show What's New dialog for new users */
|
|
|
|
#if !defined(OBS_RELEASE_CANDIDATE) || OBS_RELEASE_CANDIDATE == 0
|
|
|
|
if (!lastVersion) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
cef->init_browser();
|
2020-03-16 08:46:47 -07:00
|
|
|
|
|
|
|
WhatsNewBrowserInitThread *wnbit =
|
|
|
|
new WhatsNewBrowserInitThread(QT_UTF8(info_url.c_str()));
|
|
|
|
if (wnbit) {
|
|
|
|
connect(wnbit, &WhatsNewBrowserInitThread::Result, this,
|
|
|
|
&OBSBasic::ShowWhatsNew);
|
|
|
|
}
|
|
|
|
if (wnbit) {
|
|
|
|
whatsNewInitThread.reset(wnbit);
|
|
|
|
whatsNewInitThread->start();
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
UNUSED_PARAMETER(text);
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
UNUSED_PARAMETER(text);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::ShowWhatsNew(const QString &url)
|
|
|
|
{
|
|
|
|
#ifdef BROWSER_AVAILABLE
|
|
|
|
#ifdef _WIN32
|
|
|
|
if (closing)
|
|
|
|
return;
|
|
|
|
|
|
|
|
std::string info_url = QT_TO_UTF8(url);
|
2019-02-26 12:00:27 -08:00
|
|
|
|
2019-02-10 16:27:51 -08:00
|
|
|
QDialog *dlg = new QDialog(this);
|
|
|
|
dlg->setAttribute(Qt::WA_DeleteOnClose, true);
|
|
|
|
dlg->setWindowTitle("What's New");
|
|
|
|
dlg->resize(700, 600);
|
2018-07-27 21:35:08 -07:00
|
|
|
|
2019-02-10 22:31:59 -08:00
|
|
|
Qt::WindowFlags flags = dlg->windowFlags();
|
|
|
|
Qt::WindowFlags helpFlag = Qt::WindowContextHelpButtonHint;
|
|
|
|
dlg->setWindowFlags(flags & (~helpFlag));
|
|
|
|
|
2019-02-06 21:38:34 -08:00
|
|
|
QCefWidget *cefWidget = cef->create_widget(nullptr, info_url);
|
2018-07-27 21:35:08 -07:00
|
|
|
if (!cefWidget) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
connect(cefWidget, SIGNAL(titleChanged(const QString &)), dlg,
|
2019-02-10 16:27:51 -08:00
|
|
|
SLOT(setWindowTitle(const QString &)));
|
2018-07-27 21:35:08 -07:00
|
|
|
|
|
|
|
QPushButton *close = new QPushButton(QTStr("Close"));
|
|
|
|
connect(close, &QAbstractButton::clicked, dlg, &QDialog::accept);
|
|
|
|
|
|
|
|
QHBoxLayout *bottomLayout = new QHBoxLayout();
|
|
|
|
bottomLayout->addStretch();
|
|
|
|
bottomLayout->addWidget(close);
|
|
|
|
bottomLayout->addStretch();
|
|
|
|
|
2019-02-10 16:27:51 -08:00
|
|
|
QVBoxLayout *topLayout = new QVBoxLayout(dlg);
|
2018-07-27 21:35:08 -07:00
|
|
|
topLayout->addWidget(cefWidget);
|
|
|
|
topLayout->addLayout(bottomLayout);
|
|
|
|
|
2019-02-10 16:27:51 -08:00
|
|
|
dlg->show();
|
2018-07-27 21:35:08 -07:00
|
|
|
#else
|
2020-03-16 08:46:47 -07:00
|
|
|
UNUSED_PARAMETER(url);
|
2018-07-27 21:35:08 -07:00
|
|
|
#endif
|
2018-09-18 08:54:30 -07:00
|
|
|
#else
|
2020-03-16 08:46:47 -07:00
|
|
|
UNUSED_PARAMETER(url);
|
2018-09-18 08:54:30 -07:00
|
|
|
#endif
|
2018-07-27 21:35:08 -07:00
|
|
|
}
|
|
|
|
|
2020-04-19 09:31:18 -05:00
|
|
|
void OBSBasic::UpdateMultiviewProjectorMenu()
|
|
|
|
{
|
|
|
|
multiviewProjectorMenu->clear();
|
|
|
|
AddProjectorMenuMonitors(multiviewProjectorMenu, this,
|
|
|
|
SLOT(OpenMultiviewProjector()));
|
|
|
|
}
|
|
|
|
|
2014-11-01 21:48:58 +01:00
|
|
|
void OBSBasic::InitHotkeys()
|
|
|
|
{
|
2015-07-11 08:04:12 +02:00
|
|
|
ProfileScope("OBSBasic::InitHotkeys");
|
|
|
|
|
2014-11-01 21:48:58 +01:00
|
|
|
struct obs_hotkeys_translations t = {};
|
|
|
|
t.insert = Str("Hotkeys.Insert");
|
|
|
|
t.del = Str("Hotkeys.Delete");
|
|
|
|
t.home = Str("Hotkeys.Home");
|
|
|
|
t.end = Str("Hotkeys.End");
|
|
|
|
t.page_up = Str("Hotkeys.PageUp");
|
|
|
|
t.page_down = Str("Hotkeys.PageDown");
|
|
|
|
t.num_lock = Str("Hotkeys.NumLock");
|
|
|
|
t.scroll_lock = Str("Hotkeys.ScrollLock");
|
|
|
|
t.caps_lock = Str("Hotkeys.CapsLock");
|
|
|
|
t.backspace = Str("Hotkeys.Backspace");
|
|
|
|
t.tab = Str("Hotkeys.Tab");
|
|
|
|
t.print = Str("Hotkeys.Print");
|
|
|
|
t.pause = Str("Hotkeys.Pause");
|
|
|
|
t.left = Str("Hotkeys.Left");
|
|
|
|
t.right = Str("Hotkeys.Right");
|
|
|
|
t.up = Str("Hotkeys.Up");
|
|
|
|
t.down = Str("Hotkeys.Down");
|
|
|
|
#ifdef _WIN32
|
|
|
|
t.meta = Str("Hotkeys.Windows");
|
|
|
|
#else
|
|
|
|
t.meta = Str("Hotkeys.Super");
|
|
|
|
#endif
|
|
|
|
t.menu = Str("Hotkeys.Menu");
|
|
|
|
t.space = Str("Hotkeys.Space");
|
|
|
|
t.numpad_num = Str("Hotkeys.NumpadNum");
|
|
|
|
t.numpad_multiply = Str("Hotkeys.NumpadMultiply");
|
|
|
|
t.numpad_divide = Str("Hotkeys.NumpadDivide");
|
|
|
|
t.numpad_plus = Str("Hotkeys.NumpadAdd");
|
|
|
|
t.numpad_minus = Str("Hotkeys.NumpadSubtract");
|
|
|
|
t.numpad_decimal = Str("Hotkeys.NumpadDecimal");
|
|
|
|
t.apple_keypad_num = Str("Hotkeys.AppleKeypadNum");
|
|
|
|
t.apple_keypad_multiply = Str("Hotkeys.AppleKeypadMultiply");
|
|
|
|
t.apple_keypad_divide = Str("Hotkeys.AppleKeypadDivide");
|
|
|
|
t.apple_keypad_plus = Str("Hotkeys.AppleKeypadAdd");
|
|
|
|
t.apple_keypad_minus = Str("Hotkeys.AppleKeypadSubtract");
|
|
|
|
t.apple_keypad_decimal = Str("Hotkeys.AppleKeypadDecimal");
|
|
|
|
t.apple_keypad_equal = Str("Hotkeys.AppleKeypadEqual");
|
|
|
|
t.mouse_num = Str("Hotkeys.MouseButton");
|
2019-03-31 05:09:43 -05:00
|
|
|
t.escape = Str("Hotkeys.Escape");
|
2014-11-01 21:48:58 +01:00
|
|
|
obs_hotkeys_set_translations(&t);
|
|
|
|
|
|
|
|
obs_hotkeys_set_audio_hotkeys_translations(Str("Mute"), Str("Unmute"),
|
2015-05-12 21:36:19 +02:00
|
|
|
Str("Push-to-mute"),
|
|
|
|
Str("Push-to-talk"));
|
2014-11-01 21:48:58 +01:00
|
|
|
|
|
|
|
obs_hotkeys_set_sceneitem_hotkeys_translations(Str("SceneItemShow"),
|
|
|
|
Str("SceneItemHide"));
|
|
|
|
|
|
|
|
obs_hotkey_enable_callback_rerouting(true);
|
|
|
|
obs_hotkey_set_callback_routing_func(OBSBasic::HotkeyTriggered, this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::ProcessHotkey(obs_hotkey_id id, bool pressed)
|
|
|
|
{
|
|
|
|
obs_hotkey_trigger_routed_callback(id, pressed);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::HotkeyTriggered(void *data, obs_hotkey_id id, bool pressed)
|
|
|
|
{
|
|
|
|
OBSBasic &basic = *static_cast<OBSBasic *>(data);
|
|
|
|
QMetaObject::invokeMethod(&basic, "ProcessHotkey",
|
|
|
|
Q_ARG(obs_hotkey_id, id),
|
|
|
|
Q_ARG(bool, pressed));
|
|
|
|
}
|
|
|
|
|
2014-11-01 21:53:25 +01:00
|
|
|
void OBSBasic::CreateHotkeys()
|
|
|
|
{
|
2015-07-11 08:04:12 +02:00
|
|
|
ProfileScope("OBSBasic::CreateHotkeys");
|
|
|
|
|
2014-11-01 21:53:25 +01:00
|
|
|
auto LoadHotkeyData = [&](const char *name) -> OBSData {
|
|
|
|
const char *info =
|
|
|
|
config_get_string(basicConfig, "Hotkeys", name);
|
|
|
|
if (!info)
|
|
|
|
return {};
|
|
|
|
|
|
|
|
obs_data_t *data = obs_data_create_from_json(info);
|
|
|
|
if (!data)
|
|
|
|
return {};
|
|
|
|
|
|
|
|
OBSData res = data;
|
|
|
|
obs_data_release(data);
|
|
|
|
return res;
|
|
|
|
};
|
|
|
|
|
2015-09-06 16:19:53 -07:00
|
|
|
auto LoadHotkey = [&](obs_hotkey_id id, const char *name) {
|
|
|
|
obs_data_array_t *array =
|
|
|
|
obs_data_get_array(LoadHotkeyData(name), "bindings");
|
|
|
|
|
|
|
|
obs_hotkey_load(id, array);
|
|
|
|
obs_data_array_release(array);
|
|
|
|
};
|
|
|
|
|
2014-11-01 21:53:25 +01:00
|
|
|
auto LoadHotkeyPair = [&](obs_hotkey_pair_id id, const char *name0,
|
|
|
|
const char *name1) {
|
|
|
|
obs_data_array_t *array0 =
|
|
|
|
obs_data_get_array(LoadHotkeyData(name0), "bindings");
|
|
|
|
obs_data_array_t *array1 =
|
|
|
|
obs_data_get_array(LoadHotkeyData(name1), "bindings");
|
|
|
|
|
|
|
|
obs_hotkey_pair_load(id, array0, array1);
|
|
|
|
obs_data_array_release(array0);
|
|
|
|
obs_data_array_release(array1);
|
|
|
|
};
|
|
|
|
|
2017-10-12 17:20:19 +02:00
|
|
|
#define MAKE_CALLBACK(pred, method, log_action) \
|
2014-11-01 21:53:25 +01:00
|
|
|
[](void *data, obs_hotkey_pair_id, obs_hotkey_t *, bool pressed) { \
|
|
|
|
OBSBasic &basic = *static_cast<OBSBasic *>(data); \
|
2018-07-28 15:18:43 +02:00
|
|
|
if ((pred) && pressed) { \
|
2017-10-12 17:20:19 +02:00
|
|
|
blog(LOG_INFO, log_action " due to hotkey"); \
|
2014-11-01 21:53:25 +01:00
|
|
|
method(); \
|
|
|
|
return true; \
|
|
|
|
} \
|
|
|
|
return false; \
|
|
|
|
}
|
|
|
|
|
|
|
|
streamingHotkeys = obs_hotkey_pair_register_frontend(
|
2016-12-09 14:40:04 -08:00
|
|
|
"OBSBasic.StartStreaming", Str("Basic.Main.StartStreaming"),
|
|
|
|
"OBSBasic.StopStreaming", Str("Basic.Main.StopStreaming"),
|
2018-07-28 15:18:43 +02:00
|
|
|
MAKE_CALLBACK(!basic.outputHandler->StreamingActive() &&
|
|
|
|
basic.ui->streamButton->isEnabled(),
|
2017-10-12 17:20:19 +02:00
|
|
|
basic.StartStreaming, "Starting stream"),
|
2018-07-28 15:18:43 +02:00
|
|
|
MAKE_CALLBACK(basic.outputHandler->StreamingActive() &&
|
|
|
|
basic.ui->streamButton->isEnabled(),
|
2017-10-12 17:20:19 +02:00
|
|
|
basic.StopStreaming, "Stopping stream"),
|
2014-11-01 21:53:25 +01:00
|
|
|
this, this);
|
|
|
|
LoadHotkeyPair(streamingHotkeys, "OBSBasic.StartStreaming",
|
|
|
|
"OBSBasic.StopStreaming");
|
2019-06-22 22:13:45 -07:00
|
|
|
|
2015-09-06 16:19:53 -07:00
|
|
|
auto cb = [](void *data, obs_hotkey_id, obs_hotkey_t *, bool pressed) {
|
|
|
|
OBSBasic &basic = *static_cast<OBSBasic *>(data);
|
|
|
|
if (basic.outputHandler->StreamingActive() && pressed) {
|
|
|
|
basic.ForceStopStreaming();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
forceStreamingStopHotkey = obs_hotkey_register_frontend(
|
|
|
|
"OBSBasic.ForceStopStreaming",
|
|
|
|
Str("Basic.Main.ForceStopStreaming"), cb, this);
|
|
|
|
LoadHotkey(forceStreamingStopHotkey, "OBSBasic.ForceStopStreaming");
|
|
|
|
|
2014-11-01 21:53:25 +01:00
|
|
|
recordingHotkeys = obs_hotkey_pair_register_frontend(
|
2016-12-09 14:40:04 -08:00
|
|
|
"OBSBasic.StartRecording", Str("Basic.Main.StartRecording"),
|
|
|
|
"OBSBasic.StopRecording", Str("Basic.Main.StopRecording"),
|
2018-07-28 15:18:43 +02:00
|
|
|
MAKE_CALLBACK(!basic.outputHandler->RecordingActive() &&
|
|
|
|
!basic.ui->recordButton->isChecked(),
|
2017-10-12 17:20:19 +02:00
|
|
|
basic.StartRecording, "Starting recording"),
|
2018-07-28 15:18:43 +02:00
|
|
|
MAKE_CALLBACK(basic.outputHandler->RecordingActive() &&
|
|
|
|
basic.ui->recordButton->isChecked(),
|
2017-10-12 17:20:19 +02:00
|
|
|
basic.StopRecording, "Stopping recording"),
|
2014-11-01 21:53:25 +01:00
|
|
|
this, this);
|
|
|
|
LoadHotkeyPair(recordingHotkeys, "OBSBasic.StartRecording",
|
|
|
|
"OBSBasic.StopRecording");
|
2016-12-09 14:40:04 -08:00
|
|
|
|
2019-07-07 15:47:29 -07:00
|
|
|
pauseHotkeys = obs_hotkey_pair_register_frontend(
|
|
|
|
"OBSBasic.PauseRecording", Str("Basic.Main.PauseRecording"),
|
|
|
|
"OBSBasic.UnpauseRecording", Str("Basic.Main.UnpauseRecording"),
|
|
|
|
MAKE_CALLBACK(basic.pause && !basic.pause->isChecked(),
|
|
|
|
basic.PauseRecording, "Pausing recording"),
|
|
|
|
MAKE_CALLBACK(basic.pause && basic.pause->isChecked(),
|
|
|
|
basic.UnpauseRecording, "Unpausing recording"),
|
|
|
|
this, this);
|
|
|
|
LoadHotkeyPair(pauseHotkeys, "OBSBasic.PauseRecording",
|
|
|
|
"OBSBasic.UnpauseRecording");
|
|
|
|
|
2016-12-09 14:40:04 -08:00
|
|
|
replayBufHotkeys = obs_hotkey_pair_register_frontend(
|
|
|
|
"OBSBasic.StartReplayBuffer",
|
|
|
|
Str("Basic.Main.StartReplayBuffer"),
|
|
|
|
"OBSBasic.StopReplayBuffer", Str("Basic.Main.StopReplayBuffer"),
|
|
|
|
MAKE_CALLBACK(!basic.outputHandler->ReplayBufferActive(),
|
2017-10-12 17:49:02 +02:00
|
|
|
basic.StartReplayBuffer,
|
|
|
|
"Starting replay buffer"),
|
2016-12-09 14:40:04 -08:00
|
|
|
MAKE_CALLBACK(basic.outputHandler->ReplayBufferActive(),
|
2017-10-12 17:49:02 +02:00
|
|
|
basic.StopReplayBuffer, "Stopping replay buffer"),
|
2016-12-09 14:40:04 -08:00
|
|
|
this, this);
|
|
|
|
LoadHotkeyPair(replayBufHotkeys, "OBSBasic.StartReplayBuffer",
|
|
|
|
"OBSBasic.StopReplayBuffer");
|
2019-03-04 17:29:37 -06:00
|
|
|
|
2020-06-20 07:42:14 -07:00
|
|
|
if (vcamEnabled) {
|
|
|
|
vcamHotkeys = obs_hotkey_pair_register_frontend(
|
|
|
|
"OBSBasic.StartVirtualCam",
|
|
|
|
Str("Basic.Main.StartVirtualCam"),
|
|
|
|
"OBSBasic.StopVirtualCam",
|
|
|
|
Str("Basic.Main.StopVirtualCam"),
|
|
|
|
MAKE_CALLBACK(!basic.outputHandler->VirtualCamActive(),
|
|
|
|
basic.StartVirtualCam,
|
|
|
|
"Starting virtual camera"),
|
|
|
|
MAKE_CALLBACK(basic.outputHandler->VirtualCamActive(),
|
|
|
|
basic.StopVirtualCam,
|
|
|
|
"Stopping virtual camera"),
|
|
|
|
this, this);
|
|
|
|
LoadHotkeyPair(vcamHotkeys, "OBSBasic.StartVirtualCam",
|
|
|
|
"OBSBasic.StopVirtualCam");
|
|
|
|
}
|
|
|
|
|
2019-03-04 17:29:37 -06:00
|
|
|
togglePreviewHotkeys = obs_hotkey_pair_register_frontend(
|
|
|
|
"OBSBasic.EnablePreview",
|
|
|
|
Str("Basic.Main.PreviewConextMenu.Enable"),
|
|
|
|
"OBSBasic.DisablePreview", Str("Basic.Main.Preview.Disable"),
|
|
|
|
MAKE_CALLBACK(!basic.previewEnabled, basic.EnablePreview,
|
|
|
|
"Enabling preview"),
|
|
|
|
MAKE_CALLBACK(basic.previewEnabled, basic.DisablePreview,
|
|
|
|
"Disabling preview"),
|
|
|
|
this, this);
|
|
|
|
LoadHotkeyPair(togglePreviewHotkeys, "OBSBasic.EnablePreview",
|
|
|
|
"OBSBasic.DisablePreview");
|
2018-08-23 20:43:44 -05:00
|
|
|
|
|
|
|
contextBarHotkeys = obs_hotkey_pair_register_frontend(
|
|
|
|
"OBSBasic.ShowContextBar", Str("Basic.Main.ShowContextBar"),
|
|
|
|
"OBSBasic.HideContextBar", Str("Basic.Main.HideContextBar"),
|
|
|
|
MAKE_CALLBACK(!basic.ui->contextContainer->isVisible(),
|
|
|
|
basic.ShowContextBar, "Showing Context Bar"),
|
|
|
|
MAKE_CALLBACK(basic.ui->contextContainer->isVisible(),
|
|
|
|
basic.HideContextBar, "Hiding Context Bar"),
|
|
|
|
this, this);
|
|
|
|
LoadHotkeyPair(contextBarHotkeys, "OBSBasic.ShowContextBar",
|
|
|
|
"OBSBasic.HideContextBar");
|
2014-11-01 21:53:25 +01:00
|
|
|
#undef MAKE_CALLBACK
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
|
|
|
|
auto togglePreviewProgram = [](void *data, obs_hotkey_id,
|
|
|
|
obs_hotkey_t *, bool pressed) {
|
|
|
|
if (pressed)
|
|
|
|
QMetaObject::invokeMethod(static_cast<OBSBasic *>(data),
|
|
|
|
"on_modeSwitch_clicked",
|
|
|
|
Qt::QueuedConnection);
|
|
|
|
};
|
|
|
|
|
|
|
|
togglePreviewProgramHotkey = obs_hotkey_register_frontend(
|
|
|
|
"OBSBasic.TogglePreviewProgram",
|
|
|
|
Str("Basic.TogglePreviewProgramMode"), togglePreviewProgram,
|
|
|
|
this);
|
|
|
|
LoadHotkey(togglePreviewProgramHotkey, "OBSBasic.TogglePreviewProgram");
|
|
|
|
|
|
|
|
auto transition = [](void *data, obs_hotkey_id, obs_hotkey_t *,
|
|
|
|
bool pressed) {
|
|
|
|
if (pressed)
|
|
|
|
QMetaObject::invokeMethod(static_cast<OBSBasic *>(data),
|
|
|
|
"TransitionClicked",
|
|
|
|
Qt::QueuedConnection);
|
|
|
|
};
|
|
|
|
|
|
|
|
transitionHotkey = obs_hotkey_register_frontend(
|
|
|
|
"OBSBasic.Transition", Str("Transition"), transition, this);
|
|
|
|
LoadHotkey(transitionHotkey, "OBSBasic.Transition");
|
2019-10-15 23:20:35 -05:00
|
|
|
|
|
|
|
auto resetStats = [](void *data, obs_hotkey_id, obs_hotkey_t *,
|
|
|
|
bool pressed) {
|
|
|
|
if (pressed)
|
|
|
|
QMetaObject::invokeMethod(static_cast<OBSBasic *>(data),
|
|
|
|
"ResetStatsHotkey",
|
|
|
|
Qt::QueuedConnection);
|
|
|
|
};
|
|
|
|
|
|
|
|
statsHotkey = obs_hotkey_register_frontend(
|
|
|
|
"OBSBasic.ResetStats", Str("Basic.Stats.ResetStats"),
|
|
|
|
resetStats, this);
|
|
|
|
LoadHotkey(statsHotkey, "OBSBasic.ResetStats");
|
2020-07-23 08:53:21 -05:00
|
|
|
|
|
|
|
auto screenshot = [](void *data, obs_hotkey_id, obs_hotkey_t *,
|
|
|
|
bool pressed) {
|
|
|
|
if (pressed)
|
|
|
|
QMetaObject::invokeMethod(static_cast<OBSBasic *>(data),
|
|
|
|
"Screenshot",
|
|
|
|
Qt::QueuedConnection);
|
|
|
|
};
|
|
|
|
|
|
|
|
screenshotHotkey = obs_hotkey_register_frontend(
|
|
|
|
"OBSBasic.Screenshot", Str("Screenshot"), screenshot, this);
|
|
|
|
LoadHotkey(screenshotHotkey, "OBSBasic.Screenshot");
|
|
|
|
|
|
|
|
auto screenshotSource = [](void *data, obs_hotkey_id, obs_hotkey_t *,
|
|
|
|
bool pressed) {
|
|
|
|
if (pressed)
|
|
|
|
QMetaObject::invokeMethod(static_cast<OBSBasic *>(data),
|
|
|
|
"ScreenshotSelectedSource",
|
|
|
|
Qt::QueuedConnection);
|
|
|
|
};
|
|
|
|
|
|
|
|
sourceScreenshotHotkey = obs_hotkey_register_frontend(
|
|
|
|
"OBSBasic.SelectedSourceScreenshot",
|
|
|
|
Str("Screenshot.SourceHotkey"), screenshotSource, this);
|
|
|
|
LoadHotkey(sourceScreenshotHotkey, "OBSBasic.SelectedSourceScreenshot");
|
2014-11-01 21:53:25 +01:00
|
|
|
}
|
|
|
|
|
2015-06-29 00:47:07 -07:00
|
|
|
void OBSBasic::ClearHotkeys()
|
|
|
|
{
|
|
|
|
obs_hotkey_pair_unregister(streamingHotkeys);
|
|
|
|
obs_hotkey_pair_unregister(recordingHotkeys);
|
2019-07-07 15:47:29 -07:00
|
|
|
obs_hotkey_pair_unregister(pauseHotkeys);
|
2016-12-09 14:40:04 -08:00
|
|
|
obs_hotkey_pair_unregister(replayBufHotkeys);
|
2019-05-23 04:15:22 -07:00
|
|
|
obs_hotkey_pair_unregister(togglePreviewHotkeys);
|
2015-09-06 16:19:53 -07:00
|
|
|
obs_hotkey_unregister(forceStreamingStopHotkey);
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
obs_hotkey_unregister(togglePreviewProgramHotkey);
|
|
|
|
obs_hotkey_unregister(transitionHotkey);
|
2019-10-15 23:20:35 -05:00
|
|
|
obs_hotkey_unregister(statsHotkey);
|
2020-07-23 08:53:21 -05:00
|
|
|
obs_hotkey_unregister(screenshotHotkey);
|
|
|
|
obs_hotkey_unregister(sourceScreenshotHotkey);
|
2015-06-29 00:47:07 -07:00
|
|
|
}
|
|
|
|
|
2014-02-02 15:23:38 -07:00
|
|
|
OBSBasic::~OBSBasic()
|
|
|
|
{
|
2020-03-14 10:39:24 -07:00
|
|
|
/* clear out UI event queue */
|
|
|
|
QApplication::sendPostedEvents(App());
|
|
|
|
|
2017-02-20 04:46:29 -08:00
|
|
|
if (updateCheckThread && updateCheckThread->isRunning())
|
|
|
|
updateCheckThread->wait();
|
|
|
|
|
2020-07-23 08:53:21 -05:00
|
|
|
delete screenshotData;
|
2020-05-15 07:34:54 -05:00
|
|
|
delete logView;
|
2020-04-19 09:31:18 -05:00
|
|
|
delete multiviewProjectorMenu;
|
|
|
|
delete previewProjector;
|
|
|
|
delete studioProgramProjector;
|
|
|
|
delete previewProjectorSource;
|
|
|
|
delete previewProjectorMain;
|
|
|
|
delete sourceProjector;
|
|
|
|
delete sceneProjectorMenu;
|
|
|
|
delete scaleFilteringMenu;
|
|
|
|
delete colorMenu;
|
|
|
|
delete colorWidgetAction;
|
|
|
|
delete colorSelect;
|
|
|
|
delete deinterlaceMenu;
|
|
|
|
delete perSceneTransitionMenu;
|
|
|
|
delete shortcutFilter;
|
|
|
|
delete trayMenu;
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
delete programOptions;
|
|
|
|
delete program;
|
2015-04-02 21:35:46 -07:00
|
|
|
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
/* XXX: any obs data must be released before calling obs_shutdown.
|
|
|
|
* currently, we can't automate this with C++ RAII because of the
|
|
|
|
* delicate nature of obs_shutdown needing to be freed before the UI
|
|
|
|
* can be freed, and we have no control over the destruction order of
|
|
|
|
* the Qt UI stuff, so we have to manually clear any references to
|
|
|
|
* libobs. */
|
2014-07-06 16:19:27 -07:00
|
|
|
delete cpuUsageTimer;
|
2014-07-06 16:18:16 -07:00
|
|
|
os_cpu_usage_info_destroy(cpuUsageInfo);
|
|
|
|
|
2014-11-01 21:48:58 +01:00
|
|
|
obs_hotkey_set_callback_routing_func(nullptr, nullptr);
|
2015-06-29 00:47:07 -07:00
|
|
|
ClearHotkeys();
|
2014-11-01 21:48:58 +01:00
|
|
|
|
2015-05-04 02:07:43 +02:00
|
|
|
service = nullptr;
|
2015-02-06 03:17:33 -08:00
|
|
|
outputHandler.reset();
|
|
|
|
|
2014-09-15 18:16:16 -05:00
|
|
|
if (interaction)
|
|
|
|
delete interaction;
|
|
|
|
|
2014-07-22 22:16:57 +02:00
|
|
|
if (properties)
|
|
|
|
delete properties;
|
|
|
|
|
2015-02-25 21:23:57 -08:00
|
|
|
if (filters)
|
|
|
|
delete filters;
|
|
|
|
|
2014-07-22 22:16:57 +02:00
|
|
|
if (transformWindow)
|
|
|
|
delete transformWindow;
|
2014-03-23 01:07:54 -07:00
|
|
|
|
2014-12-28 00:38:00 -08:00
|
|
|
if (advAudioWindow)
|
|
|
|
delete advAudioWindow;
|
|
|
|
|
2018-08-18 01:13:20 -05:00
|
|
|
if (about)
|
|
|
|
delete about;
|
|
|
|
|
2015-08-02 00:02:58 -07:00
|
|
|
obs_display_remove_draw_callback(ui->preview->GetDisplay(),
|
|
|
|
OBSBasic::RenderMain, this);
|
|
|
|
|
2014-08-04 05:48:58 -07:00
|
|
|
obs_enter_graphics();
|
2014-08-07 23:42:07 -07:00
|
|
|
gs_vertexbuffer_destroy(box);
|
2016-03-30 18:44:49 -07:00
|
|
|
gs_vertexbuffer_destroy(boxLeft);
|
|
|
|
gs_vertexbuffer_destroy(boxTop);
|
|
|
|
gs_vertexbuffer_destroy(boxRight);
|
|
|
|
gs_vertexbuffer_destroy(boxBottom);
|
2014-08-07 23:42:07 -07:00
|
|
|
gs_vertexbuffer_destroy(circle);
|
2014-08-04 05:48:58 -07:00
|
|
|
obs_leave_graphics();
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
|
2015-03-18 06:31:08 -07:00
|
|
|
/* When shutting down, sometimes source references can get in to the
|
|
|
|
* event queue, and if we don't forcibly process those events they
|
|
|
|
* won't get processed until after obs_shutdown has been called. I
|
|
|
|
* really wish there were a more elegant way to deal with this via C++,
|
|
|
|
* but Qt doesn't use C++ in a normal way, so you can't really rely on
|
|
|
|
* normal C++ behavior for your data to be freed in the order that you
|
|
|
|
* expect or want it to. */
|
|
|
|
QApplication::sendPostedEvents(this);
|
|
|
|
|
2014-07-13 12:38:58 -07:00
|
|
|
config_set_int(App()->GlobalConfig(), "General", "LastVersion",
|
|
|
|
LIBOBS_API_VER);
|
2018-08-13 00:26:02 -07:00
|
|
|
#if OBS_RELEASE_CANDIDATE > 0
|
|
|
|
config_set_int(App()->GlobalConfig(), "General", "LastRCVersion",
|
|
|
|
OBS_RELEASE_CANDIDATE_VER);
|
|
|
|
#endif
|
2015-03-18 19:37:34 -07:00
|
|
|
|
2016-01-22 07:33:29 -08:00
|
|
|
bool alwaysOnTop = IsAlwaysOnTop(this);
|
2015-03-18 19:37:34 -07:00
|
|
|
|
2015-04-02 21:35:46 -07:00
|
|
|
config_set_bool(App()->GlobalConfig(), "BasicWindow", "PreviewEnabled",
|
|
|
|
previewEnabled);
|
2016-01-22 07:33:29 -08:00
|
|
|
config_set_bool(App()->GlobalConfig(), "BasicWindow", "AlwaysOnTop",
|
|
|
|
alwaysOnTop);
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
config_set_bool(App()->GlobalConfig(), "BasicWindow",
|
|
|
|
"SceneDuplicationMode", sceneDuplicationMode);
|
|
|
|
config_set_bool(App()->GlobalConfig(), "BasicWindow", "SwapScenesMode",
|
|
|
|
swapScenesMode);
|
|
|
|
config_set_bool(App()->GlobalConfig(), "BasicWindow",
|
|
|
|
"EditPropertiesMode", editPropertiesMode);
|
|
|
|
config_set_bool(App()->GlobalConfig(), "BasicWindow",
|
|
|
|
"PreviewProgramMode", IsPreviewProgramMode());
|
2017-07-27 02:40:41 -07:00
|
|
|
config_set_bool(App()->GlobalConfig(), "BasicWindow", "DocksLocked",
|
|
|
|
ui->lockUI->isChecked());
|
2015-08-21 17:56:37 -07:00
|
|
|
config_save_safe(App()->GlobalConfig(), "tmp", nullptr);
|
2015-05-25 01:37:13 -07:00
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
uint32_t winVer = GetWindowsVersion();
|
|
|
|
if (winVer > 0 && winVer < 0x602) {
|
|
|
|
bool disableAero =
|
|
|
|
config_get_bool(basicConfig, "Video", "DisableAero");
|
|
|
|
if (disableAero) {
|
|
|
|
SetAeroEnabled(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2019-02-06 21:38:34 -08:00
|
|
|
|
|
|
|
#ifdef BROWSER_AVAILABLE
|
2019-02-06 22:10:08 -08:00
|
|
|
DestroyPanelCookieManager();
|
2019-02-06 21:38:34 -08:00
|
|
|
delete cef;
|
|
|
|
cef = nullptr;
|
|
|
|
#endif
|
2014-02-02 15:23:38 -07:00
|
|
|
}
|
|
|
|
|
2015-07-09 10:51:22 -07:00
|
|
|
void OBSBasic::SaveProjectNow()
|
|
|
|
{
|
|
|
|
if (disableSaving)
|
|
|
|
return;
|
|
|
|
|
|
|
|
projectChanged = true;
|
|
|
|
SaveProjectDeferred();
|
|
|
|
}
|
|
|
|
|
2015-01-04 08:16:59 -08:00
|
|
|
void OBSBasic::SaveProject()
|
|
|
|
{
|
2015-06-30 05:49:31 -07:00
|
|
|
if (disableSaving)
|
|
|
|
return;
|
|
|
|
|
2015-07-06 09:02:13 -07:00
|
|
|
projectChanged = true;
|
|
|
|
QMetaObject::invokeMethod(this, "SaveProjectDeferred",
|
|
|
|
Qt::QueuedConnection);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::SaveProjectDeferred()
|
|
|
|
{
|
|
|
|
if (disableSaving)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!projectChanged)
|
|
|
|
return;
|
|
|
|
|
|
|
|
projectChanged = false;
|
|
|
|
|
2015-06-23 19:29:07 -07:00
|
|
|
const char *sceneCollection = config_get_string(
|
|
|
|
App()->GlobalConfig(), "Basic", "SceneCollectionFile");
|
2015-01-15 23:44:38 -08:00
|
|
|
char savePath[512];
|
2015-06-23 19:29:07 -07:00
|
|
|
char fileName[512];
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (!sceneCollection)
|
|
|
|
return;
|
|
|
|
|
|
|
|
ret = snprintf(fileName, 512, "obs-studio/basic/scenes/%s.json",
|
|
|
|
sceneCollection);
|
|
|
|
if (ret <= 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
ret = GetConfigPath(savePath, sizeof(savePath), fileName);
|
2015-01-15 23:44:38 -08:00
|
|
|
if (ret <= 0)
|
|
|
|
return;
|
|
|
|
|
2015-01-04 08:16:59 -08:00
|
|
|
Save(savePath);
|
|
|
|
}
|
|
|
|
|
2017-10-21 15:51:01 -02:00
|
|
|
OBSSource OBSBasic::GetProgramSource()
|
|
|
|
{
|
|
|
|
return OBSGetStrongRef(programScene);
|
|
|
|
}
|
|
|
|
|
2014-02-02 14:26:23 -07:00
|
|
|
OBSScene OBSBasic::GetCurrentScene()
|
2014-01-04 13:53:36 -07:00
|
|
|
{
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
QListWidgetItem *item = ui->scenes->currentItem();
|
2015-06-28 03:21:45 +02:00
|
|
|
return item ? GetOBSRef<OBSScene>(item) : nullptr;
|
2014-01-04 13:53:36 -07:00
|
|
|
}
|
|
|
|
|
2014-09-15 18:04:22 -05:00
|
|
|
OBSSceneItem OBSBasic::GetSceneItem(QListWidgetItem *item)
|
2014-01-30 01:31:52 -07:00
|
|
|
{
|
2015-06-28 03:21:45 +02:00
|
|
|
return item ? GetOBSRef<OBSSceneItem>(item) : nullptr;
|
2014-01-30 01:31:52 -07:00
|
|
|
}
|
|
|
|
|
2014-09-15 18:04:22 -05:00
|
|
|
OBSSceneItem OBSBasic::GetCurrentSceneItem()
|
|
|
|
{
|
2018-06-02 09:45:01 -07:00
|
|
|
return ui->sources->Get(GetTopSelectedSourceItem());
|
2014-09-15 18:04:22 -05:00
|
|
|
}
|
|
|
|
|
2016-11-05 12:48:46 -04:00
|
|
|
void OBSBasic::UpdatePreviewScalingMenu()
|
|
|
|
{
|
2017-05-13 13:13:55 -04:00
|
|
|
bool fixedScaling = ui->preview->IsFixedScaling();
|
|
|
|
float scalingAmount = ui->preview->GetScalingAmount();
|
|
|
|
if (!fixedScaling) {
|
|
|
|
ui->actionScaleWindow->setChecked(true);
|
|
|
|
ui->actionScaleCanvas->setChecked(false);
|
|
|
|
ui->actionScaleOutput->setChecked(false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
obs_video_info ovi;
|
|
|
|
obs_get_video_info(&ovi);
|
|
|
|
|
|
|
|
ui->actionScaleWindow->setChecked(false);
|
|
|
|
ui->actionScaleCanvas->setChecked(scalingAmount == 1.0f);
|
2016-11-05 12:48:46 -04:00
|
|
|
ui->actionScaleOutput->setChecked(scalingAmount ==
|
2017-05-13 13:13:55 -04:00
|
|
|
float(ovi.output_width) /
|
|
|
|
float(ovi.base_width));
|
2016-11-05 12:48:46 -04:00
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
void OBSBasic::CreateInteractionWindow(obs_source_t *source)
|
2014-09-15 18:16:16 -05:00
|
|
|
{
|
|
|
|
if (interaction)
|
|
|
|
interaction->close();
|
|
|
|
|
|
|
|
interaction = new OBSBasicInteraction(this, source);
|
|
|
|
interaction->Init();
|
|
|
|
interaction->setAttribute(Qt::WA_DeleteOnClose, true);
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
void OBSBasic::CreatePropertiesWindow(obs_source_t *source)
|
2014-07-22 22:16:57 +02:00
|
|
|
{
|
|
|
|
if (properties)
|
|
|
|
properties->close();
|
|
|
|
|
|
|
|
properties = new OBSBasicProperties(this, source);
|
|
|
|
properties->Init();
|
|
|
|
properties->setAttribute(Qt::WA_DeleteOnClose, true);
|
2014-03-01 05:54:55 -07:00
|
|
|
}
|
|
|
|
|
2015-02-25 21:23:57 -08:00
|
|
|
void OBSBasic::CreateFiltersWindow(obs_source_t *source)
|
|
|
|
{
|
|
|
|
if (filters)
|
|
|
|
filters->close();
|
|
|
|
|
|
|
|
filters = new OBSBasicFilters(this, source);
|
|
|
|
filters->Init();
|
|
|
|
filters->setAttribute(Qt::WA_DeleteOnClose, true);
|
|
|
|
}
|
|
|
|
|
2014-02-02 17:03:55 -07:00
|
|
|
/* Qt callbacks for invokeMethod */
|
|
|
|
|
|
|
|
void OBSBasic::AddScene(OBSSource source)
|
2013-12-28 05:33:16 -07:00
|
|
|
{
|
2014-08-04 08:41:15 -07:00
|
|
|
const char *name = obs_source_get_name(source);
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_scene_t *scene = obs_scene_from_source(source);
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
|
|
|
|
QListWidgetItem *item = new QListWidgetItem(QT_UTF8(name));
|
2015-06-28 03:21:45 +02:00
|
|
|
SetOBSRef(item, OBSScene(scene));
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
ui->scenes->addItem(item);
|
2014-01-04 13:53:36 -07:00
|
|
|
|
2014-11-01 21:53:05 +01:00
|
|
|
obs_hotkey_register_source(
|
|
|
|
source, "OBSBasic.SelectScene",
|
|
|
|
Str("Basic.Hotkeys.SelectScene"),
|
|
|
|
[](void *data, obs_hotkey_id, obs_hotkey_t *, bool pressed) {
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
OBSBasic *main = reinterpret_cast<OBSBasic *>(
|
|
|
|
App()->GetMainWindow());
|
2019-06-22 22:13:45 -07:00
|
|
|
|
2014-11-01 21:53:05 +01:00
|
|
|
auto potential_source =
|
|
|
|
static_cast<obs_source_t *>(data);
|
|
|
|
auto source = obs_source_get_ref(potential_source);
|
|
|
|
if (source && pressed)
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
main->SetCurrentScene(source);
|
2014-11-01 21:53:05 +01:00
|
|
|
obs_source_release(source);
|
|
|
|
},
|
|
|
|
static_cast<obs_source_t *>(source));
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
signal_handler_t *handler = obs_source_get_signal_handler(source);
|
2015-06-27 09:28:40 +02:00
|
|
|
|
2015-12-05 05:53:38 -08:00
|
|
|
SignalContainer<OBSScene> container;
|
|
|
|
container.ref = scene;
|
|
|
|
container.handlers.assign({
|
2015-06-27 09:28:40 +02:00
|
|
|
std::make_shared<OBSSignal>(handler, "item_add",
|
|
|
|
OBSBasic::SceneItemAdded, this),
|
|
|
|
std::make_shared<OBSSignal>(handler, "reorder",
|
|
|
|
OBSBasic::SceneReordered, this),
|
2020-02-07 16:44:10 -08:00
|
|
|
std::make_shared<OBSSignal>(handler, "refresh",
|
|
|
|
OBSBasic::SceneRefreshed, this),
|
2015-12-05 05:53:38 -08:00
|
|
|
});
|
2015-06-27 09:28:40 +02:00
|
|
|
|
|
|
|
item->setData(static_cast<int>(QtDataRole::OBSSignals),
|
2015-12-05 05:53:38 -08:00
|
|
|
QVariant::fromValue(container));
|
2015-06-30 05:49:31 -07:00
|
|
|
|
2015-08-28 15:01:39 -07:00
|
|
|
/* if the scene already has items (a duplicated scene) add them */
|
|
|
|
auto addSceneItem = [this](obs_sceneitem_t *item) {
|
|
|
|
AddSceneItem(item);
|
|
|
|
};
|
|
|
|
|
|
|
|
using addSceneItem_t = decltype(addSceneItem);
|
|
|
|
|
|
|
|
obs_scene_enum_items(
|
|
|
|
scene,
|
|
|
|
[](obs_scene_t *, obs_sceneitem_t *item, void *param) {
|
|
|
|
addSceneItem_t *func;
|
|
|
|
func = reinterpret_cast<addSceneItem_t *>(param);
|
|
|
|
(*func)(item);
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
&addSceneItem);
|
|
|
|
|
2015-06-30 05:49:31 -07:00
|
|
|
SaveProject();
|
2016-08-05 18:43:04 -07:00
|
|
|
|
|
|
|
if (!disableSaving) {
|
|
|
|
obs_source_t *source = obs_scene_get_source(scene);
|
|
|
|
blog(LOG_INFO, "User added scene '%s'",
|
|
|
|
obs_source_get_name(source));
|
2017-10-21 15:51:01 -02:00
|
|
|
|
|
|
|
OBSProjector::UpdateMultiviewProjectors();
|
2016-08-05 18:43:04 -07:00
|
|
|
}
|
2016-12-02 19:05:57 +01:00
|
|
|
|
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_SCENE_LIST_CHANGED);
|
2013-12-28 05:33:16 -07:00
|
|
|
}
|
|
|
|
|
2014-02-02 17:03:55 -07:00
|
|
|
void OBSBasic::RemoveScene(OBSSource source)
|
2013-12-28 22:29:13 -07:00
|
|
|
{
|
2015-09-24 10:11:38 +02:00
|
|
|
obs_scene_t *scene = obs_scene_from_source(source);
|
|
|
|
|
|
|
|
QListWidgetItem *sel = nullptr;
|
|
|
|
int count = ui->scenes->count();
|
2017-03-13 12:17:01 -05:00
|
|
|
|
2015-09-24 10:11:38 +02:00
|
|
|
for (int i = 0; i < count; i++) {
|
|
|
|
auto item = ui->scenes->item(i);
|
|
|
|
auto cur_scene = GetOBSRef<OBSScene>(item);
|
|
|
|
if (cur_scene != scene)
|
|
|
|
continue;
|
2013-12-28 22:29:13 -07:00
|
|
|
|
2015-09-24 10:11:38 +02:00
|
|
|
sel = item;
|
|
|
|
break;
|
|
|
|
}
|
2014-01-06 20:20:18 -07:00
|
|
|
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
if (sel != nullptr) {
|
2015-09-24 10:11:38 +02:00
|
|
|
if (sel == ui->scenes->currentItem())
|
2018-06-02 09:45:01 -07:00
|
|
|
ui->sources->Clear();
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
delete sel;
|
2014-01-06 20:20:18 -07:00
|
|
|
}
|
2015-06-30 05:49:31 -07:00
|
|
|
|
|
|
|
SaveProject();
|
2016-08-05 18:43:04 -07:00
|
|
|
|
|
|
|
if (!disableSaving) {
|
|
|
|
blog(LOG_INFO, "User Removed scene '%s'",
|
|
|
|
obs_source_get_name(source));
|
2017-10-21 15:51:01 -02:00
|
|
|
|
|
|
|
OBSProjector::UpdateMultiviewProjectors();
|
2016-08-05 18:43:04 -07:00
|
|
|
}
|
2017-02-23 00:43:20 -08:00
|
|
|
|
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_SCENE_LIST_CHANGED);
|
2014-01-04 13:53:36 -07:00
|
|
|
}
|
|
|
|
|
2018-08-26 22:02:11 -07:00
|
|
|
static bool select_one(obs_scene_t *scene, obs_sceneitem_t *item, void *param)
|
|
|
|
{
|
|
|
|
obs_sceneitem_t *selectedItem =
|
|
|
|
reinterpret_cast<obs_sceneitem_t *>(param);
|
|
|
|
if (obs_sceneitem_is_group(item))
|
|
|
|
obs_sceneitem_group_enum_items(item, select_one, param);
|
|
|
|
|
|
|
|
obs_sceneitem_select(item, (selectedItem == item));
|
|
|
|
|
|
|
|
UNUSED_PARAMETER(scene);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-02-02 17:03:55 -07:00
|
|
|
void OBSBasic::AddSceneItem(OBSSceneItem item)
|
2014-01-04 13:53:36 -07:00
|
|
|
{
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_scene_t *scene = obs_sceneitem_get_scene(item);
|
2014-01-06 20:20:18 -07:00
|
|
|
|
2014-03-01 05:54:55 -07:00
|
|
|
if (GetCurrentScene() == scene)
|
2018-06-02 09:45:01 -07:00
|
|
|
ui->sources->Add(item);
|
2014-01-06 20:20:18 -07:00
|
|
|
|
2015-06-30 05:49:31 -07:00
|
|
|
SaveProject();
|
2016-08-05 18:43:04 -07:00
|
|
|
|
|
|
|
if (!disableSaving) {
|
|
|
|
obs_source_t *sceneSource = obs_scene_get_source(scene);
|
|
|
|
obs_source_t *itemSource = obs_sceneitem_get_source(item);
|
|
|
|
blog(LOG_INFO, "User added source '%s' (%s) to scene '%s'",
|
|
|
|
obs_source_get_name(itemSource),
|
|
|
|
obs_source_get_id(itemSource),
|
|
|
|
obs_source_get_name(sceneSource));
|
2019-06-22 22:13:45 -07:00
|
|
|
|
2018-08-26 22:02:11 -07:00
|
|
|
obs_scene_enum_items(scene, select_one,
|
|
|
|
(obs_sceneitem_t *)item);
|
2016-08-05 18:43:04 -07:00
|
|
|
}
|
2014-01-04 13:53:36 -07:00
|
|
|
}
|
|
|
|
|
2014-02-02 17:03:55 -07:00
|
|
|
void OBSBasic::UpdateSceneSelection(OBSSource source)
|
2014-01-04 13:53:36 -07:00
|
|
|
{
|
|
|
|
if (source) {
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_scene_t *scene = obs_scene_from_source(source);
|
2014-08-04 08:41:15 -07:00
|
|
|
const char *name = obs_source_get_name(source);
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
|
2014-04-26 23:47:50 -07:00
|
|
|
if (!scene)
|
|
|
|
return;
|
|
|
|
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
QList<QListWidgetItem *> items =
|
|
|
|
ui->scenes->findItems(QT_UTF8(name), Qt::MatchExactly);
|
|
|
|
|
2014-02-20 22:04:14 -07:00
|
|
|
if (items.count()) {
|
|
|
|
sceneChanging = true;
|
|
|
|
ui->scenes->setCurrentItem(items.first());
|
|
|
|
sceneChanging = false;
|
|
|
|
|
2018-03-04 12:07:13 -08:00
|
|
|
OBSScene curScene =
|
|
|
|
GetOBSRef<OBSScene>(ui->scenes->currentItem());
|
|
|
|
if (api && scene != curScene)
|
|
|
|
api->on_event(
|
|
|
|
OBS_FRONTEND_EVENT_PREVIEW_SCENE_CHANGED);
|
2014-01-04 13:53:36 -07:00
|
|
|
}
|
2013-12-28 22:29:13 -07:00
|
|
|
}
|
2014-01-04 13:53:36 -07:00
|
|
|
}
|
|
|
|
|
2014-06-30 00:06:01 -07:00
|
|
|
static void RenameListValues(QListWidget *listWidget, const QString &newName,
|
|
|
|
const QString &prevName)
|
|
|
|
{
|
|
|
|
QList<QListWidgetItem *> items =
|
|
|
|
listWidget->findItems(prevName, Qt::MatchExactly);
|
|
|
|
|
|
|
|
for (int i = 0; i < items.count(); i++)
|
|
|
|
items[i]->setText(newName);
|
|
|
|
}
|
|
|
|
|
2017-10-21 15:51:01 -02:00
|
|
|
void OBSBasic::RenameSources(OBSSource source, QString newName,
|
|
|
|
QString prevName)
|
2014-06-30 00:06:01 -07:00
|
|
|
{
|
|
|
|
RenameListValues(ui->scenes, newName, prevName);
|
2014-07-08 11:58:48 -07:00
|
|
|
|
|
|
|
for (size_t i = 0; i < volumes.size(); i++) {
|
|
|
|
if (volumes[i]->GetName().compare(prevName) == 0)
|
|
|
|
volumes[i]->SetName(newName);
|
|
|
|
}
|
2015-06-30 05:49:31 -07:00
|
|
|
|
2019-10-08 00:43:10 -05:00
|
|
|
for (size_t i = 0; i < projectors.size(); i++) {
|
|
|
|
if (projectors[i]->GetSource() == source)
|
|
|
|
projectors[i]->RenameProjector(prevName, newName);
|
|
|
|
}
|
2016-12-29 09:21:53 -06:00
|
|
|
|
2015-06-30 05:49:31 -07:00
|
|
|
SaveProject();
|
2017-10-21 15:51:01 -02:00
|
|
|
|
|
|
|
obs_scene_t *scene = obs_scene_from_source(source);
|
|
|
|
if (scene)
|
|
|
|
OBSProjector::UpdateMultiviewProjectors();
|
2018-08-23 20:43:44 -05:00
|
|
|
|
|
|
|
UpdateContextBar();
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::UpdateContextBar()
|
|
|
|
{
|
|
|
|
OBSSceneItem item = GetCurrentSceneItem();
|
|
|
|
|
|
|
|
QLayoutItem *la = ui->emptySpace->layout()->itemAt(0);
|
|
|
|
if (la) {
|
|
|
|
delete la->widget();
|
|
|
|
ui->emptySpace->layout()->removeItem(la);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (item) {
|
|
|
|
obs_source_t *source = obs_sceneitem_get_source(item);
|
|
|
|
const char *id = obs_source_get_unversioned_id(source);
|
|
|
|
uint32_t flags = obs_source_get_output_flags(source);
|
|
|
|
|
|
|
|
if (flags & OBS_SOURCE_CONTROLLABLE_MEDIA) {
|
|
|
|
MediaControls *mediaControls =
|
|
|
|
new MediaControls(ui->emptySpace);
|
|
|
|
mediaControls->SetSource(source);
|
|
|
|
|
|
|
|
ui->emptySpace->layout()->addWidget(mediaControls);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strcmp(id, "browser_source") == 0) {
|
|
|
|
BrowserToolbar *c =
|
|
|
|
new BrowserToolbar(ui->emptySpace, source);
|
|
|
|
ui->emptySpace->layout()->addWidget(c);
|
|
|
|
|
|
|
|
} else if (strcmp(id, "wasapi_input_capture") == 0 ||
|
|
|
|
strcmp(id, "wasapi_output_capture") == 0 ||
|
|
|
|
strcmp(id, "coreaudio_input_capture") == 0 ||
|
|
|
|
strcmp(id, "coreaudio_output_capture") == 0 ||
|
|
|
|
strcmp(id, "pulse_input_capture") == 0 ||
|
|
|
|
strcmp(id, "pulse_output_capture") == 0 ||
|
|
|
|
strcmp(id, "alsa_input_capture") == 0) {
|
|
|
|
AudioCaptureToolbar *c =
|
|
|
|
new AudioCaptureToolbar(ui->emptySpace, source);
|
|
|
|
c->Init();
|
|
|
|
ui->emptySpace->layout()->addWidget(c);
|
|
|
|
|
|
|
|
} else if (strcmp(id, "window_capture") == 0 ||
|
|
|
|
strcmp(id, "xcomposite_input") == 0) {
|
|
|
|
WindowCaptureToolbar *c = new WindowCaptureToolbar(
|
|
|
|
ui->emptySpace, source);
|
|
|
|
c->Init();
|
|
|
|
ui->emptySpace->layout()->addWidget(c);
|
|
|
|
|
|
|
|
} else if (strcmp(id, "monitor_capture") == 0 ||
|
|
|
|
strcmp(id, "display_capture") == 0 ||
|
|
|
|
strcmp(id, "xshm_input") == 0) {
|
|
|
|
DisplayCaptureToolbar *c = new DisplayCaptureToolbar(
|
|
|
|
ui->emptySpace, source);
|
|
|
|
c->Init();
|
|
|
|
ui->emptySpace->layout()->addWidget(c);
|
|
|
|
|
|
|
|
} else if (strcmp(id, "dshow_input") == 0 ||
|
|
|
|
strcmp(id, "av_capture_input") == 0 ||
|
|
|
|
strcmp(id, "v4l2_input") == 0) {
|
|
|
|
DeviceCaptureToolbar *c = new DeviceCaptureToolbar(
|
|
|
|
ui->emptySpace, source);
|
|
|
|
c->Init();
|
|
|
|
ui->emptySpace->layout()->addWidget(c);
|
|
|
|
|
|
|
|
} else if (strcmp(id, "game_capture") == 0) {
|
|
|
|
GameCaptureToolbar *c =
|
|
|
|
new GameCaptureToolbar(ui->emptySpace, source);
|
|
|
|
ui->emptySpace->layout()->addWidget(c);
|
|
|
|
|
|
|
|
} else if (strcmp(id, "image_source") == 0) {
|
|
|
|
ImageSourceToolbar *c =
|
|
|
|
new ImageSourceToolbar(ui->emptySpace, source);
|
|
|
|
ui->emptySpace->layout()->addWidget(c);
|
|
|
|
|
|
|
|
} else if (strcmp(id, "color_source") == 0) {
|
|
|
|
ColorSourceToolbar *c =
|
|
|
|
new ColorSourceToolbar(ui->emptySpace, source);
|
|
|
|
ui->emptySpace->layout()->addWidget(c);
|
|
|
|
|
|
|
|
} else if (strcmp(id, "text_ft2_source") == 0 ||
|
|
|
|
strcmp(id, "text_gdiplus") == 0) {
|
|
|
|
TextSourceToolbar *c =
|
|
|
|
new TextSourceToolbar(ui->emptySpace, source);
|
|
|
|
ui->emptySpace->layout()->addWidget(c);
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *name = obs_source_get_name(source);
|
|
|
|
ui->contextSourceLabel->setText(name);
|
|
|
|
|
|
|
|
ui->sourceFiltersButton->setEnabled(true);
|
|
|
|
ui->sourcePropertiesButton->setEnabled(true);
|
|
|
|
} else {
|
|
|
|
ui->contextSourceLabel->setText(
|
|
|
|
QTStr("ContextBar.NoSelectedSource"));
|
|
|
|
|
|
|
|
ui->sourceFiltersButton->setEnabled(false);
|
|
|
|
ui->sourcePropertiesButton->setEnabled(false);
|
|
|
|
}
|
2014-06-30 00:06:01 -07:00
|
|
|
}
|
|
|
|
|
2017-09-13 21:10:45 -07:00
|
|
|
static inline bool SourceMixerHidden(obs_source_t *source)
|
|
|
|
{
|
|
|
|
obs_data_t *priv_settings = obs_source_get_private_settings(source);
|
|
|
|
bool hidden = obs_data_get_bool(priv_settings, "mixer_hidden");
|
|
|
|
obs_data_release(priv_settings);
|
|
|
|
|
|
|
|
return hidden;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void SetSourceMixerHidden(obs_source_t *source, bool hidden)
|
|
|
|
{
|
|
|
|
obs_data_t *priv_settings = obs_source_get_private_settings(source);
|
|
|
|
obs_data_set_bool(priv_settings, "mixer_hidden", hidden);
|
|
|
|
obs_data_release(priv_settings);
|
|
|
|
}
|
|
|
|
|
2015-06-20 18:57:06 -07:00
|
|
|
void OBSBasic::GetAudioSourceFilters()
|
|
|
|
{
|
|
|
|
QAction *action = reinterpret_cast<QAction *>(sender());
|
|
|
|
VolControl *vol = action->property("volControl").value<VolControl *>();
|
|
|
|
obs_source_t *source = vol->GetSource();
|
|
|
|
|
|
|
|
CreateFiltersWindow(source);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::GetAudioSourceProperties()
|
|
|
|
{
|
|
|
|
QAction *action = reinterpret_cast<QAction *>(sender());
|
|
|
|
VolControl *vol = action->property("volControl").value<VolControl *>();
|
|
|
|
obs_source_t *source = vol->GetSource();
|
|
|
|
|
|
|
|
CreatePropertiesWindow(source);
|
|
|
|
}
|
|
|
|
|
2017-09-13 21:10:45 -07:00
|
|
|
void OBSBasic::HideAudioControl()
|
|
|
|
{
|
|
|
|
QAction *action = reinterpret_cast<QAction *>(sender());
|
|
|
|
VolControl *vol = action->property("volControl").value<VolControl *>();
|
|
|
|
obs_source_t *source = vol->GetSource();
|
|
|
|
|
|
|
|
if (!SourceMixerHidden(source)) {
|
|
|
|
SetSourceMixerHidden(source, true);
|
|
|
|
DeactivateAudioSource(source);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::UnhideAllAudioControls()
|
|
|
|
{
|
|
|
|
auto UnhideAudioMixer = [this](obs_source_t *source) /* -- */
|
|
|
|
{
|
|
|
|
if (!obs_source_active(source))
|
|
|
|
return true;
|
|
|
|
if (!SourceMixerHidden(source))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
SetSourceMixerHidden(source, false);
|
|
|
|
ActivateAudioSource(source);
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
using UnhideAudioMixer_t = decltype(UnhideAudioMixer);
|
|
|
|
|
|
|
|
auto PreEnum = [](void *data, obs_source_t *source) -> bool /* -- */
|
|
|
|
{ return (*reinterpret_cast<UnhideAudioMixer_t *>(data))(source); };
|
|
|
|
|
|
|
|
obs_enum_sources(PreEnum, &UnhideAudioMixer);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::ToggleHideMixer()
|
|
|
|
{
|
|
|
|
OBSSceneItem item = GetCurrentSceneItem();
|
|
|
|
OBSSource source = obs_sceneitem_get_source(item);
|
|
|
|
|
|
|
|
if (!SourceMixerHidden(source)) {
|
|
|
|
SetSourceMixerHidden(source, true);
|
|
|
|
DeactivateAudioSource(source);
|
|
|
|
} else {
|
|
|
|
SetSourceMixerHidden(source, false);
|
|
|
|
ActivateAudioSource(source);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-20 14:23:07 -05:00
|
|
|
void OBSBasic::MixerRenameSource()
|
|
|
|
{
|
|
|
|
QAction *action = reinterpret_cast<QAction *>(sender());
|
|
|
|
VolControl *vol = action->property("volControl").value<VolControl *>();
|
|
|
|
OBSSource source = vol->GetSource();
|
|
|
|
|
|
|
|
const char *prevName = obs_source_get_name(source);
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
string name;
|
|
|
|
bool accepted = NameDialog::AskForName(
|
|
|
|
this, QTStr("Basic.Main.MixerRename.Title"),
|
|
|
|
QTStr("Basic.Main.MixerRename.Text"), name,
|
|
|
|
QT_UTF8(prevName));
|
|
|
|
if (!accepted)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (name.empty()) {
|
2019-03-30 15:44:44 +01:00
|
|
|
OBSMessageBox::warning(this,
|
2017-10-20 14:23:07 -05:00
|
|
|
QTStr("NoNameEntered.Title"),
|
|
|
|
QTStr("NoNameEntered.Text"));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-11-26 02:30:26 -08:00
|
|
|
OBSSource sourceTest = obs_get_source_by_name(name.c_str());
|
|
|
|
obs_source_release(sourceTest);
|
2017-10-20 14:23:07 -05:00
|
|
|
|
|
|
|
if (sourceTest) {
|
2019-03-30 15:44:44 +01:00
|
|
|
OBSMessageBox::warning(this, QTStr("NameExists.Title"),
|
2017-10-20 14:23:07 -05:00
|
|
|
QTStr("NameExists.Text"));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
obs_source_set_name(source, name.c_str());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-16 18:32:00 -06:00
|
|
|
static inline bool SourceVolumeLocked(obs_source_t *source)
|
|
|
|
{
|
|
|
|
obs_data_t *priv_settings = obs_source_get_private_settings(source);
|
|
|
|
bool lock = obs_data_get_bool(priv_settings, "volume_locked");
|
|
|
|
obs_data_release(priv_settings);
|
|
|
|
|
|
|
|
return lock;
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::LockVolumeControl(bool lock)
|
|
|
|
{
|
|
|
|
QAction *action = reinterpret_cast<QAction *>(sender());
|
|
|
|
VolControl *vol = action->property("volControl").value<VolControl *>();
|
|
|
|
obs_source_t *source = vol->GetSource();
|
|
|
|
|
|
|
|
obs_data_t *priv_settings = obs_source_get_private_settings(source);
|
|
|
|
obs_data_set_bool(priv_settings, "volume_locked", lock);
|
|
|
|
obs_data_release(priv_settings);
|
|
|
|
|
|
|
|
vol->EnableSlider(!lock);
|
|
|
|
}
|
|
|
|
|
2015-06-20 18:57:06 -07:00
|
|
|
void OBSBasic::VolControlContextMenu()
|
|
|
|
{
|
|
|
|
VolControl *vol = reinterpret_cast<VolControl *>(sender());
|
|
|
|
|
2017-09-13 21:10:45 -07:00
|
|
|
/* ------------------- */
|
|
|
|
|
2020-02-16 18:32:00 -06:00
|
|
|
QAction lockAction(QTStr("LockVolume"), this);
|
|
|
|
lockAction.setCheckable(true);
|
|
|
|
lockAction.setChecked(SourceVolumeLocked(vol->GetSource()));
|
|
|
|
|
2017-09-13 21:10:45 -07:00
|
|
|
QAction hideAction(QTStr("Hide"), this);
|
|
|
|
QAction unhideAllAction(QTStr("UnhideAll"), this);
|
2017-10-20 14:23:07 -05:00
|
|
|
QAction mixerRenameAction(QTStr("Rename"), this);
|
2017-09-13 21:10:45 -07:00
|
|
|
|
2019-02-26 06:58:53 -06:00
|
|
|
QAction copyFiltersAction(QTStr("Copy.Filters"), this);
|
|
|
|
QAction pasteFiltersAction(QTStr("Paste.Filters"), this);
|
|
|
|
|
2015-06-20 18:57:06 -07:00
|
|
|
QAction filtersAction(QTStr("Filters"), this);
|
|
|
|
QAction propertiesAction(QTStr("Properties"), this);
|
2017-08-05 21:20:08 -07:00
|
|
|
QAction advPropAction(QTStr("Basic.MainMenu.Edit.AdvAudio"), this);
|
2015-06-20 18:57:06 -07:00
|
|
|
|
2018-04-27 23:49:48 -03:00
|
|
|
QAction toggleControlLayoutAction(QTStr("VerticalLayout"), this);
|
|
|
|
toggleControlLayoutAction.setCheckable(true);
|
|
|
|
toggleControlLayoutAction.setChecked(config_get_bool(
|
|
|
|
GetGlobalConfig(), "BasicWindow", "VerticalVolControl"));
|
|
|
|
|
2017-09-13 21:10:45 -07:00
|
|
|
/* ------------------- */
|
|
|
|
|
|
|
|
connect(&hideAction, &QAction::triggered, this,
|
|
|
|
&OBSBasic::HideAudioControl, Qt::DirectConnection);
|
|
|
|
connect(&unhideAllAction, &QAction::triggered, this,
|
|
|
|
&OBSBasic::UnhideAllAudioControls, Qt::DirectConnection);
|
2020-02-16 18:32:00 -06:00
|
|
|
connect(&lockAction, &QAction::toggled, this,
|
|
|
|
&OBSBasic::LockVolumeControl, Qt::DirectConnection);
|
2017-10-20 14:23:07 -05:00
|
|
|
connect(&mixerRenameAction, &QAction::triggered, this,
|
|
|
|
&OBSBasic::MixerRenameSource, Qt::DirectConnection);
|
2019-06-22 22:13:45 -07:00
|
|
|
|
2019-02-26 06:58:53 -06:00
|
|
|
connect(©FiltersAction, &QAction::triggered, this,
|
|
|
|
&OBSBasic::AudioMixerCopyFilters, Qt::DirectConnection);
|
|
|
|
connect(&pasteFiltersAction, &QAction::triggered, this,
|
|
|
|
&OBSBasic::AudioMixerPasteFilters, Qt::DirectConnection);
|
2019-06-22 22:13:45 -07:00
|
|
|
|
2015-06-20 18:57:06 -07:00
|
|
|
connect(&filtersAction, &QAction::triggered, this,
|
|
|
|
&OBSBasic::GetAudioSourceFilters, Qt::DirectConnection);
|
|
|
|
connect(&propertiesAction, &QAction::triggered, this,
|
|
|
|
&OBSBasic::GetAudioSourceProperties, Qt::DirectConnection);
|
2017-08-05 21:20:08 -07:00
|
|
|
connect(&advPropAction, &QAction::triggered, this,
|
|
|
|
&OBSBasic::on_actionAdvAudioProperties_triggered,
|
|
|
|
Qt::DirectConnection);
|
2015-06-20 18:57:06 -07:00
|
|
|
|
2017-09-13 21:10:45 -07:00
|
|
|
/* ------------------- */
|
|
|
|
|
2018-04-27 23:49:48 -03:00
|
|
|
connect(&toggleControlLayoutAction, &QAction::changed, this,
|
|
|
|
&OBSBasic::ToggleVolControlLayout, Qt::DirectConnection);
|
|
|
|
|
|
|
|
/* ------------------- */
|
|
|
|
|
2017-09-13 21:10:45 -07:00
|
|
|
hideAction.setProperty("volControl",
|
|
|
|
QVariant::fromValue<VolControl *>(vol));
|
2020-02-16 18:32:00 -06:00
|
|
|
lockAction.setProperty("volControl",
|
|
|
|
QVariant::fromValue<VolControl *>(vol));
|
2017-10-20 14:23:07 -05:00
|
|
|
mixerRenameAction.setProperty("volControl",
|
|
|
|
QVariant::fromValue<VolControl *>(vol));
|
2017-09-13 21:10:45 -07:00
|
|
|
|
2019-02-26 06:58:53 -06:00
|
|
|
copyFiltersAction.setProperty("volControl",
|
|
|
|
QVariant::fromValue<VolControl *>(vol));
|
|
|
|
pasteFiltersAction.setProperty("volControl",
|
|
|
|
QVariant::fromValue<VolControl *>(vol));
|
|
|
|
|
2015-06-20 18:57:06 -07:00
|
|
|
filtersAction.setProperty("volControl",
|
|
|
|
QVariant::fromValue<VolControl *>(vol));
|
|
|
|
propertiesAction.setProperty("volControl",
|
|
|
|
QVariant::fromValue<VolControl *>(vol));
|
|
|
|
|
2017-09-13 21:10:45 -07:00
|
|
|
/* ------------------- */
|
|
|
|
|
2019-02-26 06:58:53 -06:00
|
|
|
if (copyFiltersString == nullptr)
|
|
|
|
pasteFiltersAction.setEnabled(false);
|
|
|
|
else
|
|
|
|
pasteFiltersAction.setEnabled(true);
|
|
|
|
|
2018-08-19 04:57:16 -03:00
|
|
|
QMenu popup;
|
2020-02-16 18:32:00 -06:00
|
|
|
popup.addAction(&lockAction);
|
|
|
|
popup.addSeparator();
|
2017-09-13 21:10:45 -07:00
|
|
|
popup.addAction(&unhideAllAction);
|
|
|
|
popup.addAction(&hideAction);
|
2017-10-20 14:23:07 -05:00
|
|
|
popup.addAction(&mixerRenameAction);
|
2017-09-13 21:10:45 -07:00
|
|
|
popup.addSeparator();
|
2019-02-26 06:58:53 -06:00
|
|
|
popup.addAction(©FiltersAction);
|
|
|
|
popup.addAction(&pasteFiltersAction);
|
|
|
|
popup.addSeparator();
|
2018-04-27 23:49:48 -03:00
|
|
|
popup.addAction(&toggleControlLayoutAction);
|
|
|
|
popup.addSeparator();
|
2015-06-20 18:57:06 -07:00
|
|
|
popup.addAction(&filtersAction);
|
|
|
|
popup.addAction(&propertiesAction);
|
2017-08-05 21:20:08 -07:00
|
|
|
popup.addAction(&advPropAction);
|
2015-06-20 18:57:06 -07:00
|
|
|
popup.exec(QCursor::pos());
|
|
|
|
}
|
|
|
|
|
2018-04-27 23:49:48 -03:00
|
|
|
void OBSBasic::on_hMixerScrollArea_customContextMenuRequested()
|
|
|
|
{
|
|
|
|
StackedMixerAreaContextMenuRequested();
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_vMixerScrollArea_customContextMenuRequested()
|
|
|
|
{
|
|
|
|
StackedMixerAreaContextMenuRequested();
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::StackedMixerAreaContextMenuRequested()
|
2017-09-13 21:10:45 -07:00
|
|
|
{
|
|
|
|
QAction unhideAllAction(QTStr("UnhideAll"), this);
|
2017-09-14 15:01:15 +02:00
|
|
|
|
|
|
|
QAction advPropAction(QTStr("Basic.MainMenu.Edit.AdvAudio"), this);
|
|
|
|
|
2018-04-27 23:49:48 -03:00
|
|
|
QAction toggleControlLayoutAction(QTStr("VerticalLayout"), this);
|
|
|
|
toggleControlLayoutAction.setCheckable(true);
|
|
|
|
toggleControlLayoutAction.setChecked(config_get_bool(
|
|
|
|
GetGlobalConfig(), "BasicWindow", "VerticalVolControl"));
|
|
|
|
|
2017-09-14 15:01:15 +02:00
|
|
|
/* ------------------- */
|
|
|
|
|
2017-09-13 21:10:45 -07:00
|
|
|
connect(&unhideAllAction, &QAction::triggered, this,
|
|
|
|
&OBSBasic::UnhideAllAudioControls, Qt::DirectConnection);
|
|
|
|
|
2017-09-14 15:01:15 +02:00
|
|
|
connect(&advPropAction, &QAction::triggered, this,
|
|
|
|
&OBSBasic::on_actionAdvAudioProperties_triggered,
|
|
|
|
Qt::DirectConnection);
|
|
|
|
|
|
|
|
/* ------------------- */
|
|
|
|
|
2018-04-27 23:49:48 -03:00
|
|
|
connect(&toggleControlLayoutAction, &QAction::changed, this,
|
|
|
|
&OBSBasic::ToggleVolControlLayout, Qt::DirectConnection);
|
|
|
|
|
|
|
|
/* ------------------- */
|
|
|
|
|
2018-08-19 04:57:16 -03:00
|
|
|
QMenu popup;
|
2017-09-13 21:10:45 -07:00
|
|
|
popup.addAction(&unhideAllAction);
|
2017-09-14 15:01:15 +02:00
|
|
|
popup.addSeparator();
|
2018-04-27 23:49:48 -03:00
|
|
|
popup.addAction(&toggleControlLayoutAction);
|
|
|
|
popup.addSeparator();
|
2017-09-14 15:01:15 +02:00
|
|
|
popup.addAction(&advPropAction);
|
2017-09-13 21:10:45 -07:00
|
|
|
popup.exec(QCursor::pos());
|
|
|
|
}
|
|
|
|
|
2018-06-12 22:52:08 -03:00
|
|
|
void OBSBasic::ToggleMixerLayout(bool vertical)
|
|
|
|
{
|
|
|
|
if (vertical) {
|
|
|
|
ui->stackedMixerArea->setMinimumSize(180, 220);
|
|
|
|
ui->stackedMixerArea->setCurrentIndex(1);
|
|
|
|
} else {
|
|
|
|
ui->stackedMixerArea->setMinimumSize(220, 0);
|
|
|
|
ui->stackedMixerArea->setCurrentIndex(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-27 23:49:48 -03:00
|
|
|
void OBSBasic::ToggleVolControlLayout()
|
|
|
|
{
|
|
|
|
bool vertical = !config_get_bool(GetGlobalConfig(), "BasicWindow",
|
|
|
|
"VerticalVolControl");
|
|
|
|
config_set_bool(GetGlobalConfig(), "BasicWindow", "VerticalVolControl",
|
|
|
|
vertical);
|
2018-06-12 22:52:08 -03:00
|
|
|
ToggleMixerLayout(vertical);
|
2018-04-27 23:49:48 -03:00
|
|
|
|
|
|
|
// We need to store it so we can delete current and then add
|
|
|
|
// at the right order
|
|
|
|
vector<OBSSource> sources;
|
|
|
|
for (size_t i = 0; i != volumes.size(); i++)
|
|
|
|
sources.emplace_back(volumes[i]->GetSource());
|
|
|
|
|
|
|
|
ClearVolumeControls();
|
|
|
|
|
|
|
|
for (const auto &source : sources)
|
|
|
|
ActivateAudioSource(source);
|
|
|
|
}
|
|
|
|
|
2014-05-03 22:54:38 -07:00
|
|
|
void OBSBasic::ActivateAudioSource(OBSSource source)
|
|
|
|
{
|
2017-09-13 21:10:45 -07:00
|
|
|
if (SourceMixerHidden(source))
|
|
|
|
return;
|
2019-09-19 23:38:53 -07:00
|
|
|
if (!obs_source_audio_active(source))
|
|
|
|
return;
|
2017-09-13 21:10:45 -07:00
|
|
|
|
2018-04-27 23:49:48 -03:00
|
|
|
bool vertical = config_get_bool(GetGlobalConfig(), "BasicWindow",
|
|
|
|
"VerticalVolControl");
|
|
|
|
VolControl *vol = new VolControl(source, true, vertical);
|
2015-06-20 18:57:06 -07:00
|
|
|
|
2020-02-16 18:32:00 -06:00
|
|
|
vol->EnableSlider(!SourceVolumeLocked(source));
|
|
|
|
|
2018-01-09 18:45:20 -02:00
|
|
|
double meterDecayRate =
|
|
|
|
config_get_double(basicConfig, "Audio", "MeterDecayRate");
|
|
|
|
vol->SetMeterDecayRate(meterDecayRate);
|
2018-01-19 22:57:36 +01:00
|
|
|
|
|
|
|
uint32_t peakMeterTypeIdx =
|
|
|
|
config_get_uint(basicConfig, "Audio", "PeakMeterType");
|
|
|
|
|
|
|
|
enum obs_peak_meter_type peakMeterType;
|
|
|
|
switch (peakMeterTypeIdx) {
|
|
|
|
case 0:
|
|
|
|
peakMeterType = SAMPLE_PEAK_METER;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
peakMeterType = TRUE_PEAK_METER;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
peakMeterType = SAMPLE_PEAK_METER;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
vol->setPeakMeterType(peakMeterType);
|
|
|
|
|
2017-09-13 21:08:27 -07:00
|
|
|
vol->setContextMenuPolicy(Qt::CustomContextMenu);
|
|
|
|
|
|
|
|
connect(vol, &QWidget::customContextMenuRequested, this,
|
|
|
|
&OBSBasic::VolControlContextMenu);
|
2015-06-20 18:57:06 -07:00
|
|
|
connect(vol, &VolControl::ConfigClicked, this,
|
|
|
|
&OBSBasic::VolControlContextMenu);
|
2014-05-03 22:54:38 -07:00
|
|
|
|
2018-03-18 19:29:13 +04:00
|
|
|
InsertQObjectByName(volumes, vol);
|
|
|
|
|
|
|
|
for (auto volume : volumes) {
|
2018-04-27 23:49:48 -03:00
|
|
|
if (vertical)
|
|
|
|
ui->vVolControlLayout->addWidget(volume);
|
|
|
|
else
|
|
|
|
ui->hVolControlLayout->addWidget(volume);
|
2018-03-18 19:29:13 +04:00
|
|
|
}
|
2014-05-03 22:54:38 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::DeactivateAudioSource(OBSSource source)
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < volumes.size(); i++) {
|
|
|
|
if (volumes[i]->GetSource() == source) {
|
|
|
|
delete volumes[i];
|
|
|
|
volumes.erase(volumes.begin() + i);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
bool OBSBasic::QueryRemoveSource(obs_source_t *source)
|
2014-06-30 01:13:32 -07:00
|
|
|
{
|
2018-06-02 09:45:01 -07:00
|
|
|
if (obs_source_get_type(source) == OBS_SOURCE_TYPE_SCENE &&
|
2018-07-15 18:18:25 -07:00
|
|
|
!obs_source_is_group(source)) {
|
2017-03-25 07:36:46 -07:00
|
|
|
int count = ui->scenes->count();
|
|
|
|
|
|
|
|
if (count == 1) {
|
2017-05-13 14:06:32 -07:00
|
|
|
OBSMessageBox::information(this,
|
2017-03-25 07:36:46 -07:00
|
|
|
QTStr("FinalScene.Title"),
|
|
|
|
QTStr("FinalScene.Text"));
|
|
|
|
return false;
|
|
|
|
}
|
2017-03-13 12:17:01 -05:00
|
|
|
}
|
|
|
|
|
2014-08-04 08:41:15 -07:00
|
|
|
const char *name = obs_source_get_name(source);
|
2014-06-30 13:45:58 -07:00
|
|
|
|
|
|
|
QString text = QTStr("ConfirmRemove.Text");
|
|
|
|
text.replace("$1", QT_UTF8(name));
|
2014-06-30 01:13:32 -07:00
|
|
|
|
2015-01-16 11:21:11 +02:00
|
|
|
QMessageBox remove_source(this);
|
2014-08-24 08:56:50 -07:00
|
|
|
remove_source.setText(text);
|
2020-06-25 00:38:35 +01:00
|
|
|
QPushButton *Yes =
|
2014-08-24 08:56:50 -07:00
|
|
|
remove_source.addButton(QTStr("Yes"), QMessageBox::YesRole);
|
2020-06-25 00:38:35 +01:00
|
|
|
remove_source.setDefaultButton(Yes);
|
2014-08-15 19:07:00 +02:00
|
|
|
remove_source.addButton(QTStr("No"), QMessageBox::NoRole);
|
|
|
|
remove_source.setIcon(QMessageBox::Question);
|
|
|
|
remove_source.setWindowTitle(QTStr("ConfirmRemove.Title"));
|
|
|
|
remove_source.exec();
|
|
|
|
|
|
|
|
return Yes == remove_source.clickedButton();
|
2014-06-30 13:45:58 -07:00
|
|
|
}
|
2014-06-30 01:13:32 -07:00
|
|
|
|
2014-07-13 23:56:28 -07:00
|
|
|
#define UPDATE_CHECK_INTERVAL (60 * 60 * 24 * 4) /* 4 days */
|
|
|
|
|
2014-10-06 00:32:18 +02:00
|
|
|
#ifdef UPDATE_SPARKLE
|
|
|
|
void init_sparkle_updater(bool update_to_undeployed);
|
|
|
|
void trigger_sparkle_update();
|
|
|
|
#endif
|
|
|
|
|
2014-07-13 23:56:28 -07:00
|
|
|
void OBSBasic::TimedCheckForUpdates()
|
|
|
|
{
|
2020-07-18 21:33:41 +02:00
|
|
|
if (App()->IsUpdaterDisabled())
|
|
|
|
return;
|
2017-02-20 04:46:29 -08:00
|
|
|
if (!config_get_bool(App()->GlobalConfig(), "General",
|
|
|
|
"EnableAutoUpdates"))
|
|
|
|
return;
|
|
|
|
|
2014-10-06 00:32:18 +02:00
|
|
|
#ifdef UPDATE_SPARKLE
|
|
|
|
init_sparkle_updater(config_get_bool(App()->GlobalConfig(), "General",
|
|
|
|
"UpdateToUndeployed"));
|
2018-05-12 17:01:22 -07:00
|
|
|
#elif _WIN32
|
2014-07-13 23:56:28 -07:00
|
|
|
long long lastUpdate = config_get_int(App()->GlobalConfig(), "General",
|
|
|
|
"LastUpdateCheck");
|
|
|
|
uint32_t lastVersion =
|
|
|
|
config_get_int(App()->GlobalConfig(), "General", "LastVersion");
|
|
|
|
|
|
|
|
if (lastVersion < LIBOBS_API_VER) {
|
|
|
|
lastUpdate = 0;
|
|
|
|
config_set_int(App()->GlobalConfig(), "General",
|
|
|
|
"LastUpdateCheck", 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
long long t = (long long)time(nullptr);
|
|
|
|
long long secs = t - lastUpdate;
|
|
|
|
|
|
|
|
if (secs > UPDATE_CHECK_INTERVAL)
|
2017-02-20 04:46:29 -08:00
|
|
|
CheckForUpdates(false);
|
2014-10-06 00:32:18 +02:00
|
|
|
#endif
|
2014-07-13 23:56:28 -07:00
|
|
|
}
|
|
|
|
|
2017-02-20 04:46:29 -08:00
|
|
|
void OBSBasic::CheckForUpdates(bool manualUpdate)
|
2014-07-13 23:56:28 -07:00
|
|
|
{
|
2014-10-06 00:32:18 +02:00
|
|
|
#ifdef UPDATE_SPARKLE
|
|
|
|
trigger_sparkle_update();
|
2018-05-12 17:01:22 -07:00
|
|
|
#elif _WIN32
|
2014-07-13 23:56:28 -07:00
|
|
|
ui->actionCheckForUpdates->setEnabled(false);
|
|
|
|
|
2017-02-20 04:46:29 -08:00
|
|
|
if (updateCheckThread && updateCheckThread->isRunning())
|
|
|
|
return;
|
2014-07-14 08:51:57 -07:00
|
|
|
|
2018-07-31 21:11:31 -07:00
|
|
|
updateCheckThread.reset(new AutoUpdateThread(manualUpdate));
|
UI: Replace Qt5Network classes with libcurl
The Qt5Network classes seem to only support OpenSSL, and because OpenSSL
isn't available on windows, we would have to distribute it with the
program to get SSL access working. The problem with that is that
OpenSSL is not GPL-compatible, so we cannot distribute OpenSSL with the
program, which means we have to find a better (and preferably superior)
library for accessing remote files that can use the windows SSPI for our
SSL needs, which comes with the operating system.
Fortunately, libcurl is probably the best library out there, and can be
compiled with SSPI instead of OpenSSL, so we're just going to switch to
libcurl instead. Originally I thought it didn't support SSPI, otherwise
I would have implemented it sooner.
As a side note, this will make it so we'll able to get files from the
internet via plugins, which will be quite useful.
2015-05-23 23:39:39 -07:00
|
|
|
updateCheckThread->start();
|
2014-10-06 00:32:18 +02:00
|
|
|
#endif
|
2017-06-01 03:58:42 -03:00
|
|
|
|
|
|
|
UNUSED_PARAMETER(manualUpdate);
|
2014-07-13 23:56:28 -07:00
|
|
|
}
|
|
|
|
|
2017-02-20 04:46:29 -08:00
|
|
|
void OBSBasic::updateCheckFinished()
|
2014-07-13 23:56:28 -07:00
|
|
|
{
|
|
|
|
ui->actionCheckForUpdates->setEnabled(true);
|
|
|
|
}
|
|
|
|
|
2015-08-28 15:01:39 -07:00
|
|
|
void OBSBasic::DuplicateSelectedScene()
|
|
|
|
{
|
|
|
|
OBSScene curScene = GetCurrentScene();
|
|
|
|
|
|
|
|
if (!curScene)
|
|
|
|
return;
|
|
|
|
|
|
|
|
OBSSource curSceneSource = obs_scene_get_source(curScene);
|
|
|
|
QString format{obs_source_get_name(curSceneSource)};
|
|
|
|
format += " %1";
|
|
|
|
|
|
|
|
int i = 2;
|
|
|
|
QString placeHolderText = format.arg(i);
|
|
|
|
obs_source_t *source = nullptr;
|
|
|
|
while ((source = obs_get_source_by_name(QT_TO_UTF8(placeHolderText)))) {
|
|
|
|
obs_source_release(source);
|
|
|
|
placeHolderText = format.arg(++i);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
string name;
|
|
|
|
bool accepted = NameDialog::AskForName(
|
|
|
|
this, QTStr("Basic.Main.AddSceneDlg.Title"),
|
|
|
|
QTStr("Basic.Main.AddSceneDlg.Text"), name,
|
|
|
|
placeHolderText);
|
|
|
|
if (!accepted)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (name.empty()) {
|
2019-03-30 15:44:44 +01:00
|
|
|
OBSMessageBox::warning(this,
|
2015-08-28 15:01:39 -07:00
|
|
|
QTStr("NoNameEntered.Title"),
|
|
|
|
QTStr("NoNameEntered.Text"));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
obs_source_t *source = obs_get_source_by_name(name.c_str());
|
|
|
|
if (source) {
|
2019-03-30 15:44:44 +01:00
|
|
|
OBSMessageBox::warning(this, QTStr("NameExists.Title"),
|
2015-08-28 15:01:39 -07:00
|
|
|
QTStr("NameExists.Text"));
|
|
|
|
|
|
|
|
obs_source_release(source);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
obs_scene_t *scene = obs_scene_duplicate(curScene, name.c_str(),
|
2016-01-12 17:04:01 -08:00
|
|
|
OBS_SCENE_DUP_REFS);
|
2015-08-28 15:01:39 -07:00
|
|
|
source = obs_scene_get_source(scene);
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
SetCurrentScene(source, true);
|
2015-08-28 15:01:39 -07:00
|
|
|
obs_scene_release(scene);
|
2016-08-28 14:24:14 -07:00
|
|
|
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
break;
|
2015-08-28 15:01:39 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-30 13:45:58 -07:00
|
|
|
void OBSBasic::RemoveSelectedScene()
|
|
|
|
{
|
|
|
|
OBSScene scene = GetCurrentScene();
|
|
|
|
if (scene) {
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_source_t *source = obs_scene_get_source(scene);
|
2016-08-28 14:24:14 -07:00
|
|
|
if (QueryRemoveSource(source)) {
|
2014-06-30 13:45:58 -07:00
|
|
|
obs_source_remove(source);
|
2016-08-28 14:24:14 -07:00
|
|
|
|
|
|
|
if (api)
|
|
|
|
api->on_event(
|
|
|
|
OBS_FRONTEND_EVENT_SCENE_LIST_CHANGED);
|
|
|
|
}
|
2014-06-30 13:45:58 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::RemoveSelectedSceneItem()
|
|
|
|
{
|
|
|
|
OBSSceneItem item = GetCurrentSceneItem();
|
|
|
|
if (item) {
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_source_t *source = obs_sceneitem_get_source(item);
|
2014-06-30 13:45:58 -07:00
|
|
|
if (QueryRemoveSource(source))
|
2014-06-30 01:13:32 -07:00
|
|
|
obs_sceneitem_remove(item);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-19 11:02:54 -08:00
|
|
|
void OBSBasic::ReorderSources(OBSScene scene)
|
|
|
|
{
|
2015-06-27 04:00:24 +02:00
|
|
|
if (scene != GetCurrentScene() || ui->sources->IgnoreReorder())
|
2015-02-19 11:02:54 -08:00
|
|
|
return;
|
|
|
|
|
2018-06-02 09:45:01 -07:00
|
|
|
ui->sources->ReorderItems();
|
2015-06-30 05:49:31 -07:00
|
|
|
SaveProject();
|
2015-02-19 11:02:54 -08:00
|
|
|
}
|
|
|
|
|
2020-02-07 16:44:10 -08:00
|
|
|
void OBSBasic::RefreshSources(OBSScene scene)
|
|
|
|
{
|
|
|
|
if (scene != GetCurrentScene() || ui->sources->IgnoreReorder())
|
|
|
|
return;
|
|
|
|
|
|
|
|
ui->sources->RefreshItems();
|
|
|
|
SaveProject();
|
|
|
|
}
|
|
|
|
|
2014-01-04 13:53:36 -07:00
|
|
|
/* OBS Callbacks */
|
|
|
|
|
2015-02-19 11:02:54 -08:00
|
|
|
void OBSBasic::SceneReordered(void *data, calldata_t *params)
|
|
|
|
{
|
|
|
|
OBSBasic *window = static_cast<OBSBasic *>(data);
|
|
|
|
|
|
|
|
obs_scene_t *scene = (obs_scene_t *)calldata_ptr(params, "scene");
|
|
|
|
|
|
|
|
QMetaObject::invokeMethod(window, "ReorderSources",
|
|
|
|
Q_ARG(OBSScene, OBSScene(scene)));
|
|
|
|
}
|
|
|
|
|
2020-02-07 16:44:10 -08:00
|
|
|
void OBSBasic::SceneRefreshed(void *data, calldata_t *params)
|
|
|
|
{
|
|
|
|
OBSBasic *window = static_cast<OBSBasic *>(data);
|
|
|
|
|
|
|
|
obs_scene_t *scene = (obs_scene_t *)calldata_ptr(params, "scene");
|
|
|
|
|
|
|
|
QMetaObject::invokeMethod(window, "RefreshSources",
|
|
|
|
Q_ARG(OBSScene, OBSScene(scene)));
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
void OBSBasic::SceneItemAdded(void *data, calldata_t *params)
|
2014-01-04 13:53:36 -07:00
|
|
|
{
|
|
|
|
OBSBasic *window = static_cast<OBSBasic *>(data);
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_sceneitem_t *item = (obs_sceneitem_t *)calldata_ptr(params, "item");
|
2013-12-28 22:29:13 -07:00
|
|
|
|
2014-02-02 17:03:55 -07:00
|
|
|
QMetaObject::invokeMethod(window, "AddSceneItem",
|
|
|
|
Q_ARG(OBSSceneItem, OBSSceneItem(item)));
|
2013-12-28 22:29:13 -07:00
|
|
|
}
|
|
|
|
|
2018-05-04 11:47:54 -07:00
|
|
|
void OBSBasic::SourceCreated(void *data, calldata_t *params)
|
2014-01-04 13:53:36 -07:00
|
|
|
{
|
2018-05-04 11:47:54 -07:00
|
|
|
obs_source_t *source = (obs_source_t *)calldata_ptr(params, "source");
|
2013-12-28 05:33:16 -07:00
|
|
|
|
2014-08-04 08:41:15 -07:00
|
|
|
if (obs_scene_from_source(source) != NULL)
|
2018-05-04 11:47:54 -07:00
|
|
|
QMetaObject::invokeMethod(static_cast<OBSBasic *>(data),
|
2018-05-04 15:33:35 -07:00
|
|
|
"AddScene", WaitConnection(),
|
2014-02-02 17:03:55 -07:00
|
|
|
Q_ARG(OBSSource, OBSSource(source)));
|
2013-12-28 05:33:16 -07:00
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
void OBSBasic::SourceRemoved(void *data, calldata_t *params)
|
2013-12-28 05:33:16 -07:00
|
|
|
{
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_source_t *source = (obs_source_t *)calldata_ptr(params, "source");
|
2013-12-28 05:33:16 -07:00
|
|
|
|
2014-08-04 08:41:15 -07:00
|
|
|
if (obs_scene_from_source(source) != NULL)
|
2014-02-02 17:03:55 -07:00
|
|
|
QMetaObject::invokeMethod(static_cast<OBSBasic *>(data),
|
|
|
|
"RemoveScene",
|
|
|
|
Q_ARG(OBSSource, OBSSource(source)));
|
2013-12-28 05:33:16 -07:00
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
void OBSBasic::SourceActivated(void *data, calldata_t *params)
|
2014-05-03 22:54:38 -07:00
|
|
|
{
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_source_t *source = (obs_source_t *)calldata_ptr(params, "source");
|
2014-05-03 22:54:38 -07:00
|
|
|
uint32_t flags = obs_source_get_output_flags(source);
|
|
|
|
|
|
|
|
if (flags & OBS_SOURCE_AUDIO)
|
|
|
|
QMetaObject::invokeMethod(static_cast<OBSBasic *>(data),
|
|
|
|
"ActivateAudioSource",
|
|
|
|
Q_ARG(OBSSource, OBSSource(source)));
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
void OBSBasic::SourceDeactivated(void *data, calldata_t *params)
|
2014-05-03 22:54:38 -07:00
|
|
|
{
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_source_t *source = (obs_source_t *)calldata_ptr(params, "source");
|
2014-05-03 22:54:38 -07:00
|
|
|
uint32_t flags = obs_source_get_output_flags(source);
|
|
|
|
|
|
|
|
if (flags & OBS_SOURCE_AUDIO)
|
|
|
|
QMetaObject::invokeMethod(static_cast<OBSBasic *>(data),
|
|
|
|
"DeactivateAudioSource",
|
|
|
|
Q_ARG(OBSSource, OBSSource(source)));
|
|
|
|
}
|
|
|
|
|
2019-09-19 23:38:53 -07:00
|
|
|
void OBSBasic::SourceAudioActivated(void *data, calldata_t *params)
|
|
|
|
{
|
|
|
|
obs_source_t *source = (obs_source_t *)calldata_ptr(params, "source");
|
|
|
|
|
|
|
|
if (obs_source_active(source))
|
|
|
|
QMetaObject::invokeMethod(static_cast<OBSBasic *>(data),
|
|
|
|
"ActivateAudioSource",
|
|
|
|
Q_ARG(OBSSource, OBSSource(source)));
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::SourceAudioDeactivated(void *data, calldata_t *params)
|
|
|
|
{
|
|
|
|
obs_source_t *source = (obs_source_t *)calldata_ptr(params, "source");
|
|
|
|
QMetaObject::invokeMethod(static_cast<OBSBasic *>(data),
|
|
|
|
"DeactivateAudioSource",
|
|
|
|
Q_ARG(OBSSource, OBSSource(source)));
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
void OBSBasic::SourceRenamed(void *data, calldata_t *params)
|
2014-06-30 00:06:01 -07:00
|
|
|
{
|
2017-10-21 15:51:01 -02:00
|
|
|
obs_source_t *source = (obs_source_t *)calldata_ptr(params, "source");
|
2014-06-30 00:06:01 -07:00
|
|
|
const char *newName = calldata_string(params, "new_name");
|
|
|
|
const char *prevName = calldata_string(params, "prev_name");
|
|
|
|
|
|
|
|
QMetaObject::invokeMethod(static_cast<OBSBasic *>(data),
|
2017-10-21 15:51:01 -02:00
|
|
|
"RenameSources", Q_ARG(OBSSource, source),
|
2014-06-30 00:06:01 -07:00
|
|
|
Q_ARG(QString, QT_UTF8(newName)),
|
|
|
|
Q_ARG(QString, QT_UTF8(prevName)));
|
2016-08-05 18:43:04 -07:00
|
|
|
|
|
|
|
blog(LOG_INFO, "Source '%s' renamed to '%s'", prevName, newName);
|
2014-06-30 00:06:01 -07:00
|
|
|
}
|
|
|
|
|
2014-06-15 19:48:02 -07:00
|
|
|
void OBSBasic::DrawBackdrop(float cx, float cy)
|
|
|
|
{
|
|
|
|
if (!box)
|
|
|
|
return;
|
|
|
|
|
2019-04-02 23:23:37 -07:00
|
|
|
GS_DEBUG_MARKER_BEGIN(GS_DEBUG_COLOR_DEFAULT, "DrawBackdrop");
|
|
|
|
|
2015-10-16 07:31:52 -07:00
|
|
|
gs_effect_t *solid = obs_get_base_effect(OBS_EFFECT_SOLID);
|
2014-09-25 17:44:05 -07:00
|
|
|
gs_eparam_t *color = gs_effect_get_param_by_name(solid, "color");
|
|
|
|
gs_technique_t *tech = gs_effect_get_technique(solid, "Solid");
|
2014-06-15 19:48:02 -07:00
|
|
|
|
|
|
|
vec4 colorVal;
|
|
|
|
vec4_set(&colorVal, 0.0f, 0.0f, 0.0f, 1.0f);
|
2014-08-07 23:42:07 -07:00
|
|
|
gs_effect_set_vec4(color, &colorVal);
|
2014-06-15 19:48:02 -07:00
|
|
|
|
2014-08-07 23:42:07 -07:00
|
|
|
gs_technique_begin(tech);
|
|
|
|
gs_technique_begin_pass(tech, 0);
|
2014-06-15 19:48:02 -07:00
|
|
|
gs_matrix_push();
|
|
|
|
gs_matrix_identity();
|
|
|
|
gs_matrix_scale3f(float(cx), float(cy), 1.0f);
|
|
|
|
|
|
|
|
gs_load_vertexbuffer(box);
|
|
|
|
gs_draw(GS_TRISTRIP, 0, 0);
|
|
|
|
|
|
|
|
gs_matrix_pop();
|
2014-08-07 23:42:07 -07:00
|
|
|
gs_technique_end_pass(tech);
|
|
|
|
gs_technique_end(tech);
|
2014-06-15 19:48:02 -07:00
|
|
|
|
|
|
|
gs_load_vertexbuffer(nullptr);
|
2019-04-02 23:23:37 -07:00
|
|
|
|
|
|
|
GS_DEBUG_MARKER_END();
|
2014-06-15 19:48:02 -07:00
|
|
|
}
|
|
|
|
|
2014-02-13 08:58:31 -07:00
|
|
|
void OBSBasic::RenderMain(void *data, uint32_t cx, uint32_t cy)
|
|
|
|
{
|
2019-04-02 23:23:37 -07:00
|
|
|
GS_DEBUG_MARKER_BEGIN(GS_DEBUG_COLOR_DEFAULT, "RenderMain");
|
|
|
|
|
2014-03-07 10:19:03 -07:00
|
|
|
OBSBasic *window = static_cast<OBSBasic *>(data);
|
2014-04-22 11:24:05 -07:00
|
|
|
obs_video_info ovi;
|
|
|
|
|
|
|
|
obs_get_video_info(&ovi);
|
|
|
|
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
window->previewCX = int(window->previewScale * float(ovi.base_width));
|
|
|
|
window->previewCY = int(window->previewScale * float(ovi.base_height));
|
2014-04-22 11:24:05 -07:00
|
|
|
|
|
|
|
gs_viewport_push();
|
|
|
|
gs_projection_push();
|
2014-06-15 19:48:02 -07:00
|
|
|
|
2019-05-09 23:30:37 -07:00
|
|
|
obs_display_t *display = window->ui->preview->GetDisplay();
|
|
|
|
uint32_t width, height;
|
|
|
|
obs_display_size(display, &width, &height);
|
|
|
|
float right = float(width) - window->previewX;
|
|
|
|
float bottom = float(height) - window->previewY;
|
2019-01-15 14:58:20 +13:00
|
|
|
|
|
|
|
gs_ortho(-window->previewX, right, -window->previewY, bottom, -100.0f,
|
|
|
|
100.0f);
|
|
|
|
|
|
|
|
window->ui->preview->DrawOverflow();
|
|
|
|
|
2014-06-15 19:48:02 -07:00
|
|
|
/* --------------------------------------- */
|
|
|
|
|
2014-04-22 11:24:05 -07:00
|
|
|
gs_ortho(0.0f, float(ovi.base_width), 0.0f, float(ovi.base_height),
|
|
|
|
-100.0f, 100.0f);
|
2014-08-07 23:42:07 -07:00
|
|
|
gs_set_viewport(window->previewX, window->previewY, window->previewCX,
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
window->previewCY);
|
2014-04-22 11:24:05 -07:00
|
|
|
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
if (window->IsPreviewProgramMode()) {
|
2019-07-18 08:44:09 -07:00
|
|
|
window->DrawBackdrop(float(ovi.base_width),
|
|
|
|
float(ovi.base_height));
|
|
|
|
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
OBSScene scene = window->GetCurrentScene();
|
|
|
|
obs_source_t *source = obs_scene_get_source(scene);
|
|
|
|
if (source)
|
|
|
|
obs_source_video_render(source);
|
|
|
|
} else {
|
2019-07-18 08:44:09 -07:00
|
|
|
obs_render_main_texture_src_color_only();
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
}
|
2014-06-15 19:48:02 -07:00
|
|
|
gs_load_vertexbuffer(nullptr);
|
2014-04-22 11:24:05 -07:00
|
|
|
|
2014-06-15 19:48:02 -07:00
|
|
|
/* --------------------------------------- */
|
|
|
|
|
|
|
|
gs_ortho(-window->previewX, right, -window->previewY, bottom, -100.0f,
|
|
|
|
100.0f);
|
2014-08-07 23:42:07 -07:00
|
|
|
gs_reset_viewport();
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
|
|
|
|
window->ui->preview->DrawSceneEditing();
|
|
|
|
|
2014-06-15 19:48:02 -07:00
|
|
|
/* --------------------------------------- */
|
|
|
|
|
2014-04-22 11:24:05 -07:00
|
|
|
gs_projection_pop();
|
|
|
|
gs_viewport_pop();
|
2014-02-14 15:13:36 -07:00
|
|
|
|
2019-04-02 23:23:37 -07:00
|
|
|
GS_DEBUG_MARKER_END();
|
|
|
|
|
2014-02-14 15:13:36 -07:00
|
|
|
UNUSED_PARAMETER(cx);
|
|
|
|
UNUSED_PARAMETER(cy);
|
2014-02-13 08:58:31 -07:00
|
|
|
}
|
|
|
|
|
2014-01-04 13:53:36 -07:00
|
|
|
/* Main class functions */
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_service_t *OBSBasic::GetService()
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
{
|
2015-05-04 02:07:43 +02:00
|
|
|
if (!service) {
|
2014-11-01 21:41:17 +01:00
|
|
|
service =
|
|
|
|
obs_service_create("rtmp_common", NULL, NULL, nullptr);
|
2015-05-04 02:07:43 +02:00
|
|
|
obs_service_release(service);
|
|
|
|
}
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
return service;
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
void OBSBasic::SetService(obs_service_t *newService)
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
{
|
2015-05-04 02:07:43 +02:00
|
|
|
if (newService)
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
service = newService;
|
|
|
|
}
|
|
|
|
|
2019-05-12 17:13:38 +12:00
|
|
|
int OBSBasic::GetTransitionDuration()
|
|
|
|
{
|
|
|
|
return ui->transitionDuration->value();
|
|
|
|
}
|
|
|
|
|
2016-07-01 10:27:27 -07:00
|
|
|
bool OBSBasic::StreamingActive() const
|
2015-02-07 08:09:57 -08:00
|
|
|
{
|
|
|
|
if (!outputHandler)
|
|
|
|
return false;
|
|
|
|
return outputHandler->StreamingActive();
|
|
|
|
}
|
|
|
|
|
2016-07-01 10:27:27 -07:00
|
|
|
bool OBSBasic::Active() const
|
|
|
|
{
|
|
|
|
if (!outputHandler)
|
|
|
|
return false;
|
|
|
|
return outputHandler->Active();
|
|
|
|
}
|
|
|
|
|
2014-07-20 17:49:54 -07:00
|
|
|
#ifdef _WIN32
|
|
|
|
#define IS_WIN32 1
|
|
|
|
#else
|
|
|
|
#define IS_WIN32 0
|
|
|
|
#endif
|
|
|
|
|
2014-07-20 17:49:09 -07:00
|
|
|
static inline int AttemptToResetVideo(struct obs_video_info *ovi)
|
|
|
|
{
|
2015-08-02 00:02:58 -07:00
|
|
|
return obs_reset_video(ovi);
|
2014-07-20 17:49:09 -07:00
|
|
|
}
|
|
|
|
|
2014-12-15 01:08:46 -08:00
|
|
|
static inline enum obs_scale_type GetScaleType(ConfigFile &basicConfig)
|
|
|
|
{
|
|
|
|
const char *scaleTypeStr =
|
|
|
|
config_get_string(basicConfig, "Video", "ScaleType");
|
|
|
|
|
|
|
|
if (astrcmpi(scaleTypeStr, "bilinear") == 0)
|
|
|
|
return OBS_SCALE_BILINEAR;
|
|
|
|
else if (astrcmpi(scaleTypeStr, "lanczos") == 0)
|
|
|
|
return OBS_SCALE_LANCZOS;
|
2019-08-14 22:29:30 -07:00
|
|
|
else if (astrcmpi(scaleTypeStr, "area") == 0)
|
|
|
|
return OBS_SCALE_AREA;
|
2014-12-15 01:08:46 -08:00
|
|
|
else
|
|
|
|
return OBS_SCALE_BICUBIC;
|
|
|
|
}
|
|
|
|
|
2015-01-09 20:16:30 -08:00
|
|
|
static inline enum video_format GetVideoFormatFromName(const char *name)
|
|
|
|
{
|
|
|
|
if (astrcmpi(name, "I420") == 0)
|
|
|
|
return VIDEO_FORMAT_I420;
|
|
|
|
else if (astrcmpi(name, "NV12") == 0)
|
|
|
|
return VIDEO_FORMAT_NV12;
|
2015-04-17 02:46:48 -07:00
|
|
|
else if (astrcmpi(name, "I444") == 0)
|
|
|
|
return VIDEO_FORMAT_I444;
|
2015-01-09 20:16:30 -08:00
|
|
|
#if 0 //currently unsupported
|
|
|
|
else if (astrcmpi(name, "YVYU") == 0)
|
|
|
|
return VIDEO_FORMAT_YVYU;
|
|
|
|
else if (astrcmpi(name, "YUY2") == 0)
|
|
|
|
return VIDEO_FORMAT_YUY2;
|
|
|
|
else if (astrcmpi(name, "UYVY") == 0)
|
|
|
|
return VIDEO_FORMAT_UYVY;
|
|
|
|
#endif
|
|
|
|
else
|
2015-04-15 18:42:06 -07:00
|
|
|
return VIDEO_FORMAT_RGBA;
|
2015-01-09 20:16:30 -08:00
|
|
|
}
|
|
|
|
|
2017-11-08 23:44:22 -05:00
|
|
|
void OBSBasic::ResetUI()
|
|
|
|
{
|
|
|
|
bool studioPortraitLayout = config_get_bool(
|
|
|
|
GetGlobalConfig(), "BasicWindow", "StudioPortraitLayout");
|
|
|
|
|
2019-02-28 15:29:41 -06:00
|
|
|
bool labels = config_get_bool(GetGlobalConfig(), "BasicWindow",
|
|
|
|
"StudioModeLabels");
|
|
|
|
|
2017-11-08 23:44:22 -05:00
|
|
|
if (studioPortraitLayout)
|
|
|
|
ui->previewLayout->setDirection(QBoxLayout::TopToBottom);
|
|
|
|
else
|
|
|
|
ui->previewLayout->setDirection(QBoxLayout::LeftToRight);
|
2019-02-28 15:29:41 -06:00
|
|
|
|
|
|
|
if (previewProgramMode)
|
|
|
|
ui->previewLabel->setHidden(!labels);
|
|
|
|
|
|
|
|
if (programLabel)
|
|
|
|
programLabel->setHidden(!labels);
|
2017-11-08 23:44:22 -05:00
|
|
|
}
|
|
|
|
|
2014-07-20 17:40:57 -07:00
|
|
|
int OBSBasic::ResetVideo()
|
2013-12-22 23:40:07 -07:00
|
|
|
{
|
2017-05-15 14:13:12 -07:00
|
|
|
if (outputHandler && outputHandler->Active())
|
|
|
|
return OBS_VIDEO_CURRENTLY_ACTIVE;
|
|
|
|
|
2015-07-11 08:04:12 +02:00
|
|
|
ProfileScope("OBSBasic::ResetVideo");
|
|
|
|
|
2013-12-22 23:40:07 -07:00
|
|
|
struct obs_video_info ovi;
|
2014-07-20 17:40:57 -07:00
|
|
|
int ret;
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
|
2014-03-06 21:08:12 -07:00
|
|
|
GetConfigFPS(ovi.fps_num, ovi.fps_den);
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
|
2015-01-09 20:16:30 -08:00
|
|
|
const char *colorFormat =
|
|
|
|
config_get_string(basicConfig, "Video", "ColorFormat");
|
|
|
|
const char *colorSpace =
|
|
|
|
config_get_string(basicConfig, "Video", "ColorSpace");
|
|
|
|
const char *colorRange =
|
|
|
|
config_get_string(basicConfig, "Video", "ColorRange");
|
|
|
|
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
ovi.graphics_module = App()->GetRenderModule();
|
2014-03-06 21:08:12 -07:00
|
|
|
ovi.base_width =
|
|
|
|
(uint32_t)config_get_uint(basicConfig, "Video", "BaseCX");
|
|
|
|
ovi.base_height =
|
|
|
|
(uint32_t)config_get_uint(basicConfig, "Video", "BaseCY");
|
|
|
|
ovi.output_width =
|
|
|
|
(uint32_t)config_get_uint(basicConfig, "Video", "OutputCX");
|
|
|
|
ovi.output_height =
|
|
|
|
(uint32_t)config_get_uint(basicConfig, "Video", "OutputCY");
|
2015-01-09 20:16:30 -08:00
|
|
|
ovi.output_format = GetVideoFormatFromName(colorFormat);
|
2020-01-25 10:34:51 -08:00
|
|
|
ovi.colorspace = astrcmpi(colorSpace, "601") == 0
|
|
|
|
? VIDEO_CS_601
|
|
|
|
: (astrcmpi(colorSpace, "709") == 0
|
|
|
|
? VIDEO_CS_709
|
|
|
|
: VIDEO_CS_SRGB);
|
2015-01-09 20:16:30 -08:00
|
|
|
ovi.range = astrcmpi(colorRange, "Full") == 0 ? VIDEO_RANGE_FULL
|
|
|
|
: VIDEO_RANGE_PARTIAL;
|
2017-10-07 03:02:14 -07:00
|
|
|
ovi.adapter =
|
|
|
|
config_get_uint(App()->GlobalConfig(), "Video", "AdapterIdx");
|
2014-02-16 19:28:21 -07:00
|
|
|
ovi.gpu_conversion = true;
|
2014-12-15 01:08:46 -08:00
|
|
|
ovi.scale_type = GetScaleType(basicConfig);
|
2014-01-09 19:51:51 -06:00
|
|
|
|
2020-05-24 00:36:50 +02:00
|
|
|
if (ovi.base_width < 8 || ovi.base_height < 8) {
|
2016-02-04 14:42:46 -08:00
|
|
|
ovi.base_width = 1920;
|
|
|
|
ovi.base_height = 1080;
|
|
|
|
config_set_uint(basicConfig, "Video", "BaseCX", 1920);
|
|
|
|
config_set_uint(basicConfig, "Video", "BaseCY", 1080);
|
|
|
|
}
|
|
|
|
|
2020-05-24 00:36:50 +02:00
|
|
|
if (ovi.output_width < 8 || ovi.output_height < 8) {
|
2016-02-04 14:42:46 -08:00
|
|
|
ovi.output_width = ovi.base_width;
|
|
|
|
ovi.output_height = ovi.base_height;
|
|
|
|
config_set_uint(basicConfig, "Video", "OutputCX",
|
|
|
|
ovi.base_width);
|
|
|
|
config_set_uint(basicConfig, "Video", "OutputCY",
|
|
|
|
ovi.base_height);
|
|
|
|
}
|
|
|
|
|
2014-07-20 17:49:09 -07:00
|
|
|
ret = AttemptToResetVideo(&ovi);
|
2014-07-20 17:49:54 -07:00
|
|
|
if (IS_WIN32 && ret != OBS_VIDEO_SUCCESS) {
|
2017-05-15 14:13:12 -07:00
|
|
|
if (ret == OBS_VIDEO_CURRENTLY_ACTIVE) {
|
|
|
|
blog(LOG_WARNING, "Tried to reset when "
|
|
|
|
"already active");
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-07-20 17:49:54 -07:00
|
|
|
/* Try OpenGL if DirectX fails on windows */
|
2015-01-09 20:19:22 +01:00
|
|
|
if (astrcmpi(ovi.graphics_module, DL_OPENGL) != 0) {
|
2014-10-14 18:07:49 +02:00
|
|
|
blog(LOG_WARNING,
|
|
|
|
"Failed to initialize obs video (%d) "
|
|
|
|
"with graphics_module='%s', retrying "
|
|
|
|
"with graphics_module='%s'",
|
|
|
|
ret, ovi.graphics_module, DL_OPENGL);
|
2015-01-09 20:19:22 +01:00
|
|
|
ovi.graphics_module = DL_OPENGL;
|
2014-07-20 17:49:54 -07:00
|
|
|
ret = AttemptToResetVideo(&ovi);
|
|
|
|
}
|
2015-08-08 00:55:46 -07:00
|
|
|
} else if (ret == OBS_VIDEO_SUCCESS) {
|
|
|
|
ResizePreview(ovi.base_width, ovi.base_height);
|
2016-01-30 16:59:22 -08:00
|
|
|
if (program)
|
|
|
|
ResizeProgram(ovi.base_width, ovi.base_height);
|
2014-07-20 17:49:54 -07:00
|
|
|
}
|
|
|
|
|
2018-03-14 06:05:13 -03:00
|
|
|
if (ret == OBS_VIDEO_SUCCESS) {
|
2017-06-24 09:14:46 -07:00
|
|
|
OBSBasicStats::InitializeValues();
|
2018-03-14 06:05:13 -03:00
|
|
|
OBSProjector::UpdateMultiviewProjectors();
|
|
|
|
}
|
2017-06-19 15:33:12 -07:00
|
|
|
|
2014-07-20 17:40:57 -07:00
|
|
|
return ret;
|
2014-01-09 19:08:20 -07:00
|
|
|
}
|
2013-12-31 04:02:07 -07:00
|
|
|
|
2014-02-22 20:14:19 -07:00
|
|
|
bool OBSBasic::ResetAudio()
|
2014-01-09 19:08:20 -07:00
|
|
|
{
|
2015-07-11 08:04:12 +02:00
|
|
|
ProfileScope("OBSBasic::ResetAudio");
|
|
|
|
|
2015-03-07 04:47:12 -08:00
|
|
|
struct obs_audio_info ai;
|
2014-03-06 21:08:12 -07:00
|
|
|
ai.samples_per_sec =
|
|
|
|
config_get_uint(basicConfig, "Audio", "SampleRate");
|
2014-02-23 16:27:19 -07:00
|
|
|
|
2014-03-06 21:08:12 -07:00
|
|
|
const char *channelSetupStr =
|
|
|
|
config_get_string(basicConfig, "Audio", "ChannelSetup");
|
2014-02-23 16:27:19 -07:00
|
|
|
|
|
|
|
if (strcmp(channelSetupStr, "Mono") == 0)
|
|
|
|
ai.speakers = SPEAKERS_MONO;
|
libobs: Add surround sound audio support
(This commit also modifies the following modules: UI,
deps/media-playback, coreaudio-encoder, decklink, linux-alsa,
linux-pulseaudio, mac-capture, obs-ffmpeg, obs-filters, obs-libfdk,
obs-outputs, win-dshow, and win-wasapi)
Adds surround sound audio support to the core, core plugins, and user
interface.
Compatible streaming services: Twitch, FB 360 live
Compatible protocols: rtmp / mpeg-ts tcp udp
Compatible file formats: mkv mp4 ts (others untested)
Compatible codecs: ffmpeg aac, fdk_aac, CoreAudio aac,
opus, vorbis, pcm (others untested).
Tested streaming servers: wowza, nginx
HLS, mpeg-dash : surround passthrough
Html5 players tested with live surround:
videojs, mediaelement, viblast (hls+dash), hls.js
Decklink: on win32, swap channels order for 5.1 7.1
(due to different channel mapping on wav, mpeg, ffmpeg)
Audio filters: surround working.
Monitoring: surround working (win macOs linux (pulse-audio)).
VST: stereo plugins keep in general only the first two channels.
surround plugins should work (e.g. mcfx does).
OS: win, macOs, linux (alsa, pulse-audio).
Misc: larger audio bitrates unlocked to accommodate more channels
NB: mf-aac only supports mono and stereo + 5.1 on win 10
(not implemented due to lack of usefulness)
Closes jp9000/obs-studio#968
2017-05-27 02:15:54 +02:00
|
|
|
else if (strcmp(channelSetupStr, "2.1") == 0)
|
|
|
|
ai.speakers = SPEAKERS_2POINT1;
|
2017-12-01 17:10:05 +01:00
|
|
|
else if (strcmp(channelSetupStr, "4.0") == 0)
|
|
|
|
ai.speakers = SPEAKERS_4POINT0;
|
libobs: Add surround sound audio support
(This commit also modifies the following modules: UI,
deps/media-playback, coreaudio-encoder, decklink, linux-alsa,
linux-pulseaudio, mac-capture, obs-ffmpeg, obs-filters, obs-libfdk,
obs-outputs, win-dshow, and win-wasapi)
Adds surround sound audio support to the core, core plugins, and user
interface.
Compatible streaming services: Twitch, FB 360 live
Compatible protocols: rtmp / mpeg-ts tcp udp
Compatible file formats: mkv mp4 ts (others untested)
Compatible codecs: ffmpeg aac, fdk_aac, CoreAudio aac,
opus, vorbis, pcm (others untested).
Tested streaming servers: wowza, nginx
HLS, mpeg-dash : surround passthrough
Html5 players tested with live surround:
videojs, mediaelement, viblast (hls+dash), hls.js
Decklink: on win32, swap channels order for 5.1 7.1
(due to different channel mapping on wav, mpeg, ffmpeg)
Audio filters: surround working.
Monitoring: surround working (win macOs linux (pulse-audio)).
VST: stereo plugins keep in general only the first two channels.
surround plugins should work (e.g. mcfx does).
OS: win, macOs, linux (alsa, pulse-audio).
Misc: larger audio bitrates unlocked to accommodate more channels
NB: mf-aac only supports mono and stereo + 5.1 on win 10
(not implemented due to lack of usefulness)
Closes jp9000/obs-studio#968
2017-05-27 02:15:54 +02:00
|
|
|
else if (strcmp(channelSetupStr, "4.1") == 0)
|
|
|
|
ai.speakers = SPEAKERS_4POINT1;
|
|
|
|
else if (strcmp(channelSetupStr, "5.1") == 0)
|
|
|
|
ai.speakers = SPEAKERS_5POINT1;
|
|
|
|
else if (strcmp(channelSetupStr, "7.1") == 0)
|
|
|
|
ai.speakers = SPEAKERS_7POINT1;
|
2014-02-23 16:27:19 -07:00
|
|
|
else
|
|
|
|
ai.speakers = SPEAKERS_STEREO;
|
|
|
|
|
2014-01-09 19:08:20 -07:00
|
|
|
return obs_reset_audio(&ai);
|
2013-12-22 23:40:07 -07:00
|
|
|
}
|
|
|
|
|
2015-06-23 18:47:22 -07:00
|
|
|
void OBSBasic::ResetAudioDevice(const char *sourceId, const char *deviceId,
|
2014-05-12 15:30:36 -07:00
|
|
|
const char *deviceDesc, int channel)
|
2014-03-07 17:03:34 -07:00
|
|
|
{
|
2017-06-21 00:03:14 -07:00
|
|
|
bool disable = deviceId && strcmp(deviceId, "disabled") == 0;
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_source_t *source;
|
|
|
|
obs_data_t *settings;
|
2014-03-07 17:03:34 -07:00
|
|
|
|
|
|
|
source = obs_get_output_source(channel);
|
|
|
|
if (source) {
|
2017-06-21 00:03:14 -07:00
|
|
|
if (disable) {
|
|
|
|
obs_set_output_source(channel, nullptr);
|
|
|
|
} else {
|
|
|
|
settings = obs_source_get_settings(source);
|
|
|
|
const char *oldId =
|
|
|
|
obs_data_get_string(settings, "device_id");
|
|
|
|
if (strcmp(oldId, deviceId) != 0) {
|
|
|
|
obs_data_set_string(settings, "device_id",
|
|
|
|
deviceId);
|
|
|
|
obs_source_update(source, settings);
|
|
|
|
}
|
|
|
|
obs_data_release(settings);
|
|
|
|
}
|
2014-03-07 17:03:34 -07:00
|
|
|
|
|
|
|
obs_source_release(source);
|
|
|
|
|
2017-06-21 00:03:14 -07:00
|
|
|
} else if (!disable) {
|
|
|
|
settings = obs_data_create();
|
2014-08-05 11:09:29 -07:00
|
|
|
obs_data_set_string(settings, "device_id", deviceId);
|
2015-12-29 15:25:45 -08:00
|
|
|
source = obs_source_create(sourceId, deviceDesc, settings,
|
|
|
|
nullptr);
|
2014-03-07 17:03:34 -07:00
|
|
|
obs_data_release(settings);
|
|
|
|
|
|
|
|
obs_set_output_source(channel, source);
|
|
|
|
obs_source_release(source);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-31 04:02:07 -07:00
|
|
|
void OBSBasic::ResizePreview(uint32_t cx, uint32_t cy)
|
2013-12-06 06:39:19 -07:00
|
|
|
{
|
2014-01-23 17:00:42 -07:00
|
|
|
QSize targetSize;
|
2017-05-13 13:13:55 -04:00
|
|
|
bool isFixedScaling;
|
2016-11-05 12:48:46 -04:00
|
|
|
obs_video_info ovi;
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
|
2013-12-06 09:16:33 -07:00
|
|
|
/* resize preview panel to fix to the top section of the window */
|
2014-04-16 20:28:02 +02:00
|
|
|
targetSize = GetPixelSize(ui->preview);
|
2016-11-05 12:48:46 -04:00
|
|
|
|
2017-05-13 13:13:55 -04:00
|
|
|
isFixedScaling = ui->preview->IsFixedScaling();
|
2016-11-05 12:48:46 -04:00
|
|
|
obs_get_video_info(&ovi);
|
|
|
|
|
2017-05-13 13:13:55 -04:00
|
|
|
if (isFixedScaling) {
|
|
|
|
previewScale = ui->preview->GetScalingAmount();
|
2016-11-05 12:48:46 -04:00
|
|
|
GetCenterPosFromFixedScale(
|
|
|
|
int(cx), int(cy),
|
|
|
|
targetSize.width() - PREVIEW_EDGE_SIZE * 2,
|
|
|
|
targetSize.height() - PREVIEW_EDGE_SIZE * 2, previewX,
|
|
|
|
previewY, previewScale);
|
2017-05-13 13:13:55 -04:00
|
|
|
previewX += ui->preview->GetScrollX();
|
|
|
|
previewY += ui->preview->GetScrollY();
|
2016-11-05 12:48:46 -04:00
|
|
|
|
|
|
|
} else {
|
|
|
|
GetScaleAndCenterPos(int(cx), int(cy),
|
|
|
|
targetSize.width() - PREVIEW_EDGE_SIZE * 2,
|
|
|
|
targetSize.height() -
|
|
|
|
PREVIEW_EDGE_SIZE * 2,
|
|
|
|
previewX, previewY, previewScale);
|
|
|
|
}
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
|
2014-06-15 19:48:02 -07:00
|
|
|
previewX += float(PREVIEW_EDGE_SIZE);
|
|
|
|
previewY += float(PREVIEW_EDGE_SIZE);
|
2013-12-31 04:02:07 -07:00
|
|
|
}
|
|
|
|
|
2015-07-02 16:12:47 -07:00
|
|
|
void OBSBasic::CloseDialogs()
|
|
|
|
{
|
|
|
|
QList<QDialog *> childDialogs = this->findChildren<QDialog *>();
|
|
|
|
if (!childDialogs.isEmpty()) {
|
|
|
|
for (int i = 0; i < childDialogs.size(); ++i) {
|
|
|
|
childDialogs.at(i)->close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-02 18:15:10 +02:00
|
|
|
if (!stats.isNull())
|
|
|
|
stats->close(); //call close to save Stats geometry
|
2018-04-19 01:10:10 -05:00
|
|
|
if (!remux.isNull())
|
|
|
|
remux->close();
|
2015-07-02 16:12:47 -07:00
|
|
|
}
|
|
|
|
|
2017-01-25 00:39:18 -08:00
|
|
|
void OBSBasic::EnumDialogs()
|
|
|
|
{
|
|
|
|
visDialogs.clear();
|
|
|
|
modalDialogs.clear();
|
|
|
|
visMsgBoxes.clear();
|
|
|
|
|
|
|
|
/* fill list of Visible dialogs and Modal dialogs */
|
|
|
|
QList<QDialog *> dialogs = findChildren<QDialog *>();
|
|
|
|
for (QDialog *dialog : dialogs) {
|
|
|
|
if (dialog->isVisible())
|
|
|
|
visDialogs.append(dialog);
|
|
|
|
if (dialog->isModal())
|
|
|
|
modalDialogs.append(dialog);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* fill list of Visible message boxes */
|
|
|
|
QList<QMessageBox *> msgBoxes = findChildren<QMessageBox *>();
|
|
|
|
for (QMessageBox *msgbox : msgBoxes) {
|
|
|
|
if (msgbox->isVisible())
|
|
|
|
visMsgBoxes.append(msgbox);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-23 10:30:19 -07:00
|
|
|
void OBSBasic::ClearProjectors()
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < projectors.size(); i++) {
|
|
|
|
if (projectors[i])
|
|
|
|
delete projectors[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
projectors.clear();
|
|
|
|
}
|
|
|
|
|
2015-07-02 16:12:47 -07:00
|
|
|
void OBSBasic::ClearSceneData()
|
|
|
|
{
|
2015-06-30 05:49:31 -07:00
|
|
|
disableSaving++;
|
|
|
|
|
2015-07-02 16:12:47 -07:00
|
|
|
CloseDialogs();
|
|
|
|
|
|
|
|
ClearVolumeControls();
|
|
|
|
ClearListItems(ui->scenes);
|
2018-06-02 09:45:01 -07:00
|
|
|
ui->sources->Clear();
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
ClearQuickTransitions();
|
|
|
|
ui->transitions->clear();
|
2015-07-02 16:12:47 -07:00
|
|
|
|
2020-08-23 10:30:19 -07:00
|
|
|
ClearProjectors();
|
2019-10-08 00:43:10 -05:00
|
|
|
|
2015-07-02 16:12:47 -07:00
|
|
|
obs_set_output_source(0, nullptr);
|
|
|
|
obs_set_output_source(1, nullptr);
|
|
|
|
obs_set_output_source(2, nullptr);
|
|
|
|
obs_set_output_source(3, nullptr);
|
|
|
|
obs_set_output_source(4, nullptr);
|
|
|
|
obs_set_output_source(5, nullptr);
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
lastScene = nullptr;
|
|
|
|
swapScene = nullptr;
|
|
|
|
programScene = nullptr;
|
2015-07-02 16:12:47 -07:00
|
|
|
|
|
|
|
auto cb = [](void *unused, obs_source_t *source) {
|
|
|
|
obs_source_remove(source);
|
|
|
|
UNUSED_PARAMETER(unused);
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
2020-04-18 08:28:08 +02:00
|
|
|
obs_enum_scenes(cb, nullptr);
|
2015-07-02 16:12:47 -07:00
|
|
|
obs_enum_sources(cb, nullptr);
|
|
|
|
|
2017-12-25 13:44:23 -08:00
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_SCENE_COLLECTION_CLEANUP);
|
|
|
|
|
2015-06-30 05:49:31 -07:00
|
|
|
disableSaving--;
|
2015-07-05 23:39:05 -07:00
|
|
|
|
|
|
|
blog(LOG_INFO, "All scene data cleared");
|
|
|
|
blog(LOG_INFO, "------------------------------------------------");
|
2015-07-02 16:12:47 -07:00
|
|
|
}
|
|
|
|
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
void OBSBasic::closeEvent(QCloseEvent *event)
|
2013-12-31 04:02:07 -07:00
|
|
|
{
|
2019-03-26 22:01:22 -07:00
|
|
|
/* Do not close window if inside of a temporary event loop because we
|
|
|
|
* could be inside of an Auth::LoadUI call. Keep trying once per
|
|
|
|
* second until we've exit any known sub-loops. */
|
|
|
|
if (os_atomic_load_long(&insideEventLoop) != 0) {
|
|
|
|
QTimer::singleShot(1000, this, SLOT(close()));
|
|
|
|
event->ignore();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-09-28 01:56:10 -07:00
|
|
|
if (isVisible())
|
|
|
|
config_set_string(App()->GlobalConfig(), "BasicWindow",
|
|
|
|
"geometry",
|
|
|
|
saveGeometry().toBase64().constData());
|
|
|
|
|
2015-02-07 03:39:16 -08:00
|
|
|
if (outputHandler && outputHandler->Active()) {
|
2016-08-13 09:36:17 -05:00
|
|
|
SetShowing(true);
|
|
|
|
|
2017-05-13 14:06:32 -07:00
|
|
|
QMessageBox::StandardButton button = OBSMessageBox::question(
|
2015-02-07 03:39:16 -08:00
|
|
|
this, QTStr("ConfirmExit.Title"),
|
|
|
|
QTStr("ConfirmExit.Text"));
|
|
|
|
|
|
|
|
if (button == QMessageBox::No) {
|
|
|
|
event->ignore();
|
2020-02-16 15:15:33 -06:00
|
|
|
restart = false;
|
2015-02-07 03:39:16 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-17 17:18:51 +02:00
|
|
|
QWidget::closeEvent(event);
|
|
|
|
if (!event->isAccepted())
|
|
|
|
return;
|
|
|
|
|
2016-08-21 11:29:56 -07:00
|
|
|
blog(LOG_INFO, SHUTDOWN_SEPARATOR);
|
|
|
|
|
2020-03-16 08:46:47 -07:00
|
|
|
closing = true;
|
|
|
|
|
2018-07-27 21:35:08 -07:00
|
|
|
if (introCheckThread)
|
|
|
|
introCheckThread->wait();
|
2020-03-16 08:46:47 -07:00
|
|
|
if (whatsNewInitThread)
|
|
|
|
whatsNewInitThread->wait();
|
UI: Replace Qt5Network classes with libcurl
The Qt5Network classes seem to only support OpenSSL, and because OpenSSL
isn't available on windows, we would have to distribute it with the
program to get SSL access working. The problem with that is that
OpenSSL is not GPL-compatible, so we cannot distribute OpenSSL with the
program, which means we have to find a better (and preferably superior)
library for accessing remote files that can use the windows SSPI for our
SSL needs, which comes with the operating system.
Fortunately, libcurl is probably the best library out there, and can be
compiled with SSPI instead of OpenSSL, so we're just going to switch to
libcurl instead. Originally I thought it didn't support SSPI, otherwise
I would have implemented it sooner.
As a side note, this will make it so we'll able to get files from the
internet via plugins, which will be quite useful.
2015-05-23 23:39:39 -07:00
|
|
|
if (updateCheckThread)
|
|
|
|
updateCheckThread->wait();
|
|
|
|
if (logUploadThread)
|
|
|
|
logUploadThread->wait();
|
|
|
|
|
2015-06-27 09:29:17 +02:00
|
|
|
signalHandlers.clear();
|
|
|
|
|
2019-02-06 22:24:25 -08:00
|
|
|
Auth::Save();
|
2015-07-09 10:51:22 -07:00
|
|
|
SaveProjectNow();
|
2019-02-06 22:24:25 -08:00
|
|
|
auth.reset();
|
|
|
|
|
2019-08-08 03:27:45 -07:00
|
|
|
delete extraBrowsers;
|
|
|
|
|
2019-02-06 22:24:25 -08:00
|
|
|
config_set_string(App()->GlobalConfig(), "BasicWindow", "DockState",
|
|
|
|
saveState().toBase64().constData());
|
2016-08-28 14:24:14 -07:00
|
|
|
|
2019-08-08 03:27:45 -07:00
|
|
|
#ifdef BROWSER_AVAILABLE
|
|
|
|
SaveExtraBrowserDocks();
|
|
|
|
ClearExtraBrowserDocks();
|
|
|
|
#endif
|
|
|
|
|
2016-08-28 14:24:14 -07:00
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_EXIT);
|
|
|
|
|
2015-06-30 05:49:31 -07:00
|
|
|
disableSaving++;
|
2015-03-18 07:02:58 -07:00
|
|
|
|
2015-07-02 16:12:47 -07:00
|
|
|
/* Clear all scene data (dialogs, widgets, widget sub-items, scenes,
|
|
|
|
* sources, etc) so that all references are released before shutdown */
|
|
|
|
ClearSceneData();
|
2017-05-14 13:03:46 -07:00
|
|
|
|
|
|
|
App()->quit();
|
2013-12-31 07:10:47 -07:00
|
|
|
}
|
|
|
|
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
void OBSBasic::changeEvent(QEvent *event)
|
2013-12-31 07:10:47 -07:00
|
|
|
{
|
2020-02-13 22:54:22 +01:00
|
|
|
if (event->type() == QEvent::WindowStateChange) {
|
|
|
|
QWindowStateChangeEvent *stateEvent =
|
|
|
|
(QWindowStateChangeEvent *)event;
|
|
|
|
|
|
|
|
if (isMinimized()) {
|
|
|
|
if (trayIcon && trayIcon->isVisible() &&
|
|
|
|
sysTrayMinimizeToTray()) {
|
|
|
|
ToggleShowHide();
|
|
|
|
}
|
2017-01-24 23:01:24 -08:00
|
|
|
|
2020-02-13 22:54:22 +01:00
|
|
|
if (previewEnabled)
|
|
|
|
EnablePreviewDisplay(false);
|
2020-03-01 16:38:37 +01:00
|
|
|
} else if (stateEvent->oldState() & Qt::WindowMinimized &&
|
2020-02-13 22:54:22 +01:00
|
|
|
isVisible()) {
|
|
|
|
if (previewEnabled)
|
|
|
|
EnablePreviewDisplay(true);
|
|
|
|
}
|
2017-01-24 23:01:24 -08:00
|
|
|
}
|
2013-11-23 23:38:52 -07:00
|
|
|
}
|
|
|
|
|
2014-08-22 03:18:42 +02:00
|
|
|
void OBSBasic::on_actionShow_Recordings_triggered()
|
|
|
|
{
|
2015-06-04 20:15:56 +02:00
|
|
|
const char *mode = config_get_string(basicConfig, "Output", "Mode");
|
2018-08-20 01:36:17 +02:00
|
|
|
const char *type = config_get_string(basicConfig, "AdvOut", "RecType");
|
|
|
|
const char *adv_path =
|
|
|
|
strcmp(type, "Standard")
|
|
|
|
? config_get_string(basicConfig, "AdvOut", "FFFilePath")
|
|
|
|
: config_get_string(basicConfig, "AdvOut",
|
|
|
|
"RecFilePath");
|
2015-06-04 20:15:56 +02:00
|
|
|
const char *path = strcmp(mode, "Advanced")
|
2018-08-20 01:36:17 +02:00
|
|
|
? config_get_string(basicConfig,
|
|
|
|
"SimpleOutput",
|
|
|
|
"FilePath")
|
|
|
|
: adv_path;
|
2014-08-22 03:18:42 +02:00
|
|
|
QDesktopServices::openUrl(QUrl::fromLocalFile(path));
|
|
|
|
}
|
|
|
|
|
2014-09-03 04:11:55 +02:00
|
|
|
void OBSBasic::on_actionRemux_triggered()
|
|
|
|
{
|
2018-04-19 01:10:10 -05:00
|
|
|
if (!remux.isNull()) {
|
|
|
|
remux->show();
|
|
|
|
remux->raise();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-06-04 20:15:56 +02:00
|
|
|
const char *mode = config_get_string(basicConfig, "Output", "Mode");
|
|
|
|
const char *path = strcmp(mode, "Advanced")
|
|
|
|
? config_get_string(basicConfig,
|
|
|
|
"SimpleOutput",
|
|
|
|
"FilePath")
|
|
|
|
: config_get_string(basicConfig, "AdvOut",
|
|
|
|
"RecFilePath");
|
2018-04-19 01:10:10 -05:00
|
|
|
|
|
|
|
OBSRemux *remuxDlg;
|
|
|
|
remuxDlg = new OBSRemux(path, this);
|
|
|
|
remuxDlg->show();
|
|
|
|
remux = remuxDlg;
|
2014-09-03 04:11:55 +02:00
|
|
|
}
|
|
|
|
|
2014-04-15 14:19:59 +02:00
|
|
|
void OBSBasic::on_action_Settings_triggered()
|
|
|
|
{
|
2019-03-26 22:01:22 -07:00
|
|
|
static bool settings_already_executing = false;
|
|
|
|
|
|
|
|
/* Do not load settings window if inside of a temporary event loop
|
|
|
|
* because we could be inside of an Auth::LoadUI call. Keep trying
|
|
|
|
* once per second until we've exit any known sub-loops. */
|
|
|
|
if (os_atomic_load_long(&insideEventLoop) != 0) {
|
|
|
|
QTimer::singleShot(1000, this,
|
|
|
|
SLOT(on_action_Settings_triggered()));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (settings_already_executing) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
settings_already_executing = true;
|
|
|
|
|
2020-03-02 23:21:26 +01:00
|
|
|
{
|
|
|
|
OBSBasicSettings settings(this);
|
|
|
|
settings.exec();
|
|
|
|
}
|
|
|
|
|
2016-08-13 09:36:17 -05:00
|
|
|
SystemTray(false);
|
2019-03-26 22:01:22 -07:00
|
|
|
|
|
|
|
settings_already_executing = false;
|
2020-02-16 15:15:33 -06:00
|
|
|
|
|
|
|
if (restart) {
|
|
|
|
QMessageBox::StandardButton button = OBSMessageBox::question(
|
|
|
|
this, QTStr("Restart"), QTStr("NeedsRestart"));
|
|
|
|
|
|
|
|
if (button == QMessageBox::Yes)
|
|
|
|
close();
|
|
|
|
else
|
|
|
|
restart = false;
|
|
|
|
}
|
2014-04-15 14:19:59 +02:00
|
|
|
}
|
|
|
|
|
2014-12-28 00:38:00 -08:00
|
|
|
void OBSBasic::on_actionAdvAudioProperties_triggered()
|
|
|
|
{
|
2015-02-13 13:06:55 +01:00
|
|
|
if (advAudioWindow != nullptr) {
|
|
|
|
advAudioWindow->raise();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-01-25 22:48:38 +11:00
|
|
|
bool iconsVisible = config_get_bool(App()->GlobalConfig(),
|
|
|
|
"BasicWindow", "ShowSourceIcons");
|
|
|
|
|
2014-12-28 00:38:00 -08:00
|
|
|
advAudioWindow = new OBSBasicAdvAudio(this);
|
|
|
|
advAudioWindow->show();
|
|
|
|
advAudioWindow->setAttribute(Qt::WA_DeleteOnClose, true);
|
2020-01-25 22:48:38 +11:00
|
|
|
advAudioWindow->SetIconsVisible(iconsVisible);
|
2015-02-13 13:06:55 +01:00
|
|
|
|
|
|
|
connect(advAudioWindow, SIGNAL(destroyed()), this,
|
|
|
|
SLOT(on_advAudioProps_destroyed()));
|
2014-12-28 00:38:00 -08:00
|
|
|
}
|
|
|
|
|
2015-02-07 02:11:42 -08:00
|
|
|
void OBSBasic::on_advAudioProps_clicked()
|
|
|
|
{
|
|
|
|
on_actionAdvAudioProperties_triggered();
|
|
|
|
}
|
|
|
|
|
2015-02-13 13:06:55 +01:00
|
|
|
void OBSBasic::on_advAudioProps_destroyed()
|
|
|
|
{
|
|
|
|
advAudioWindow = nullptr;
|
|
|
|
}
|
|
|
|
|
2014-02-20 22:04:14 -07:00
|
|
|
void OBSBasic::on_scenes_currentItemChanged(QListWidgetItem *current,
|
|
|
|
QListWidgetItem *prev)
|
2013-12-30 01:17:57 -07:00
|
|
|
{
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_source_t *source = NULL;
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
|
2014-02-20 22:04:14 -07:00
|
|
|
if (sceneChanging)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (current) {
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_scene_t *scene;
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
|
2015-06-28 03:21:45 +02:00
|
|
|
scene = GetOBSRef<OBSScene>(current);
|
2014-08-04 08:41:15 -07:00
|
|
|
source = obs_scene_get_source(scene);
|
2013-12-30 01:17:57 -07:00
|
|
|
}
|
|
|
|
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
SetCurrentScene(source);
|
2014-02-20 22:04:14 -07:00
|
|
|
|
2018-03-04 12:07:13 -08:00
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_PREVIEW_SCENE_CHANGED);
|
|
|
|
|
2018-08-23 20:43:44 -05:00
|
|
|
UpdateContextBar();
|
|
|
|
|
2014-02-20 22:04:14 -07:00
|
|
|
UNUSED_PARAMETER(prev);
|
2013-12-30 01:17:57 -07:00
|
|
|
}
|
|
|
|
|
2014-06-30 01:13:32 -07:00
|
|
|
void OBSBasic::EditSceneName()
|
|
|
|
{
|
2014-10-15 22:05:21 +02:00
|
|
|
QListWidgetItem *item = ui->scenes->currentItem();
|
|
|
|
Qt::ItemFlags flags = item->flags();
|
|
|
|
|
|
|
|
item->setFlags(flags | Qt::ItemIsEditable);
|
|
|
|
ui->scenes->editItem(item);
|
|
|
|
item->setFlags(flags);
|
2014-06-30 01:13:32 -07:00
|
|
|
}
|
|
|
|
|
2019-10-08 00:43:10 -05:00
|
|
|
void OBSBasic::AddProjectorMenuMonitors(QMenu *parent, QObject *target,
|
|
|
|
const char *slot)
|
2015-04-04 01:40:15 -07:00
|
|
|
{
|
|
|
|
QAction *action;
|
2016-10-03 23:50:13 -07:00
|
|
|
QList<QScreen *> screens = QGuiApplication::screens();
|
|
|
|
for (int i = 0; i < screens.size(); i++) {
|
2019-02-24 11:00:57 +11:00
|
|
|
QScreen *screen = screens[i];
|
|
|
|
QRect screenGeometry = screen->geometry();
|
|
|
|
QString name = "";
|
|
|
|
#ifdef _WIN32
|
2020-03-18 20:37:56 -07:00
|
|
|
QTextStream fullname(&name);
|
|
|
|
fullname << GetMonitorName(screen->name());
|
|
|
|
fullname << " (";
|
|
|
|
fullname << (i + 1);
|
|
|
|
fullname << ")";
|
2019-02-24 11:00:57 +11:00
|
|
|
#elif defined(__APPLE__)
|
|
|
|
name = screen->name();
|
|
|
|
#elif QT_VERSION >= QT_VERSION_CHECK(5, 9, 0)
|
|
|
|
name = screen->model().simplified();
|
|
|
|
|
|
|
|
if (name.length() > 1 && name.endsWith("-"))
|
|
|
|
name.chop(1);
|
|
|
|
#endif
|
|
|
|
name = name.simplified();
|
|
|
|
|
|
|
|
if (name.length() == 0) {
|
|
|
|
name = QString("%1 %2")
|
|
|
|
.arg(QTStr("Display"))
|
|
|
|
.arg(QString::number(i + 1));
|
|
|
|
}
|
2015-04-04 01:40:15 -07:00
|
|
|
QString str =
|
2019-02-24 11:00:57 +11:00
|
|
|
QString("%1: %2x%3 @ %4,%5")
|
|
|
|
.arg(name,
|
2018-03-14 21:39:14 -03:00
|
|
|
QString::number(screenGeometry.width()),
|
|
|
|
QString::number(screenGeometry.height()),
|
|
|
|
QString::number(screenGeometry.x()),
|
|
|
|
QString::number(screenGeometry.y()));
|
2015-04-04 01:40:15 -07:00
|
|
|
|
|
|
|
action = parent->addAction(str, target, slot);
|
|
|
|
action->setProperty("monitor", i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
void OBSBasic::on_scenes_customContextMenuRequested(const QPoint &pos)
|
2013-11-07 16:45:03 -07:00
|
|
|
{
|
2014-06-30 16:03:12 -07:00
|
|
|
QListWidgetItem *item = ui->scenes->itemAt(pos);
|
|
|
|
|
2015-01-16 11:21:11 +02:00
|
|
|
QMenu popup(this);
|
2015-06-25 03:53:48 -07:00
|
|
|
QMenu order(QTStr("Basic.MainMenu.Edit.Order"), this);
|
2019-08-27 20:08:43 +12:00
|
|
|
|
2014-06-30 16:03:12 -07:00
|
|
|
popup.addAction(QTStr("Add"), this,
|
|
|
|
SLOT(on_actionAddScene_triggered()));
|
|
|
|
|
2014-10-15 22:04:31 +02:00
|
|
|
if (item) {
|
2019-04-23 02:26:17 -05:00
|
|
|
QAction *pasteFilters =
|
|
|
|
new QAction(QTStr("Paste.Filters"), this);
|
|
|
|
pasteFilters->setEnabled(copyFiltersString);
|
|
|
|
connect(pasteFilters, SIGNAL(triggered()), this,
|
|
|
|
SLOT(ScenePasteFilters()));
|
|
|
|
|
2014-10-15 22:04:31 +02:00
|
|
|
popup.addSeparator();
|
2015-08-28 15:01:39 -07:00
|
|
|
popup.addAction(QTStr("Duplicate"), this,
|
|
|
|
SLOT(DuplicateSelectedScene()));
|
2019-04-23 02:26:17 -05:00
|
|
|
popup.addAction(QTStr("Copy.Filters"), this,
|
|
|
|
SLOT(SceneCopyFilters()));
|
|
|
|
popup.addAction(pasteFilters);
|
|
|
|
popup.addSeparator();
|
2014-10-15 22:04:31 +02:00
|
|
|
popup.addAction(QTStr("Rename"), this, SLOT(EditSceneName()));
|
2014-06-30 16:03:12 -07:00
|
|
|
popup.addAction(QTStr("Remove"), this,
|
2018-03-03 02:54:12 -03:00
|
|
|
SLOT(RemoveSelectedScene()));
|
2015-03-18 15:09:44 -07:00
|
|
|
popup.addSeparator();
|
2015-06-25 03:53:48 -07:00
|
|
|
|
|
|
|
order.addAction(QTStr("Basic.MainMenu.Edit.Order.MoveUp"), this,
|
|
|
|
SLOT(on_actionSceneUp_triggered()));
|
|
|
|
order.addAction(QTStr("Basic.MainMenu.Edit.Order.MoveDown"),
|
|
|
|
this, SLOT(on_actionSceneDown_triggered()));
|
|
|
|
order.addSeparator();
|
|
|
|
order.addAction(QTStr("Basic.MainMenu.Edit.Order.MoveToTop"),
|
|
|
|
this, SLOT(MoveSceneToTop()));
|
|
|
|
order.addAction(QTStr("Basic.MainMenu.Edit.Order.MoveToBottom"),
|
|
|
|
this, SLOT(MoveSceneToBottom()));
|
|
|
|
popup.addMenu(&order);
|
|
|
|
|
|
|
|
popup.addSeparator();
|
2017-10-21 15:51:01 -02:00
|
|
|
|
2020-04-19 09:31:18 -05:00
|
|
|
delete sceneProjectorMenu;
|
|
|
|
sceneProjectorMenu = new QMenu(QTStr("SceneProjector"));
|
2015-04-04 01:40:15 -07:00
|
|
|
AddProjectorMenuMonitors(sceneProjectorMenu, this,
|
|
|
|
SLOT(OpenSceneProjector()));
|
|
|
|
popup.addMenu(sceneProjectorMenu);
|
2017-03-01 22:08:49 -06:00
|
|
|
|
|
|
|
QAction *sceneWindow = popup.addAction(
|
|
|
|
QTStr("SceneWindow"), this, SLOT(OpenSceneWindow()));
|
|
|
|
|
|
|
|
popup.addAction(sceneWindow);
|
2020-07-23 08:53:21 -05:00
|
|
|
popup.addAction(QTStr("Screenshot.Scene"), this,
|
|
|
|
SLOT(ScreenshotScene()));
|
2015-04-04 01:40:15 -07:00
|
|
|
popup.addSeparator();
|
2015-03-18 15:09:44 -07:00
|
|
|
popup.addAction(QTStr("Filters"), this,
|
|
|
|
SLOT(OpenSceneFilters()));
|
2017-11-30 05:05:50 -08:00
|
|
|
|
|
|
|
popup.addSeparator();
|
|
|
|
|
2020-04-19 09:31:18 -05:00
|
|
|
delete perSceneTransitionMenu;
|
|
|
|
perSceneTransitionMenu = CreatePerSceneTransitionMenu();
|
2019-01-31 03:33:15 +01:00
|
|
|
popup.addMenu(perSceneTransitionMenu);
|
2017-10-21 15:51:01 -02:00
|
|
|
|
|
|
|
/* ---------------------- */
|
|
|
|
|
2017-12-02 16:11:04 -08:00
|
|
|
QAction *multiviewAction =
|
|
|
|
popup.addAction(QTStr("ShowInMultiview"));
|
2017-10-21 15:51:01 -02:00
|
|
|
|
2017-12-02 16:11:04 -08:00
|
|
|
OBSSource source = GetCurrentSceneSource();
|
|
|
|
OBSData data = obs_source_get_private_settings(source);
|
|
|
|
obs_data_release(data);
|
2017-10-21 15:51:01 -02:00
|
|
|
|
2017-12-02 16:11:04 -08:00
|
|
|
obs_data_set_default_bool(data, "show_in_multiview", true);
|
|
|
|
bool show = obs_data_get_bool(data, "show_in_multiview");
|
2017-10-21 15:51:01 -02:00
|
|
|
|
2017-12-02 16:11:04 -08:00
|
|
|
multiviewAction->setCheckable(true);
|
|
|
|
multiviewAction->setChecked(show);
|
2017-10-21 15:51:01 -02:00
|
|
|
|
2018-06-03 04:12:42 -07:00
|
|
|
auto showInMultiview = [](OBSData data) {
|
2017-12-02 16:11:04 -08:00
|
|
|
bool show =
|
|
|
|
obs_data_get_bool(data, "show_in_multiview");
|
|
|
|
obs_data_set_bool(data, "show_in_multiview", !show);
|
|
|
|
OBSProjector::UpdateMultiviewProjectors();
|
|
|
|
};
|
|
|
|
|
|
|
|
connect(multiviewAction, &QAction::triggered,
|
|
|
|
std::bind(showInMultiview, data));
|
2014-10-15 22:04:31 +02:00
|
|
|
}
|
2014-06-30 16:03:12 -07:00
|
|
|
|
2019-08-27 20:08:43 +12:00
|
|
|
popup.addSeparator();
|
|
|
|
|
|
|
|
bool grid = ui->scenes->GetGridMode();
|
|
|
|
|
|
|
|
QAction *gridAction = new QAction(grid ? QTStr("Basic.Main.ListMode")
|
|
|
|
: QTStr("Basic.Main.GridMode"),
|
|
|
|
this);
|
|
|
|
connect(gridAction, SIGNAL(triggered()), this,
|
|
|
|
SLOT(on_actionGridMode_triggered()));
|
|
|
|
popup.addAction(gridAction);
|
|
|
|
|
2014-06-30 16:03:12 -07:00
|
|
|
popup.exec(QCursor::pos());
|
2013-11-07 16:45:03 -07:00
|
|
|
}
|
|
|
|
|
2019-08-27 20:08:43 +12:00
|
|
|
void OBSBasic::on_actionGridMode_triggered()
|
|
|
|
{
|
|
|
|
bool gridMode = !ui->scenes->GetGridMode();
|
|
|
|
ui->scenes->SetGridMode(gridMode);
|
|
|
|
}
|
|
|
|
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
void OBSBasic::on_actionAddScene_triggered()
|
2013-11-07 16:45:03 -07:00
|
|
|
{
|
2013-12-29 08:54:06 -07:00
|
|
|
string name;
|
2014-05-09 18:09:17 -04:00
|
|
|
QString format{QTStr("Basic.Main.DefaultSceneName.Text")};
|
2014-05-14 20:58:15 +02:00
|
|
|
|
2017-02-26 08:23:31 -08:00
|
|
|
int i = 2;
|
2014-05-14 20:58:15 +02:00
|
|
|
QString placeHolderText = format.arg(i);
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_source_t *source = nullptr;
|
2014-05-14 22:20:08 +02:00
|
|
|
while ((source = obs_get_source_by_name(QT_TO_UTF8(placeHolderText)))) {
|
|
|
|
obs_source_release(source);
|
2014-05-14 20:58:15 +02:00
|
|
|
placeHolderText = format.arg(++i);
|
2014-05-14 22:20:08 +02:00
|
|
|
}
|
2014-05-09 18:09:17 -04:00
|
|
|
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
bool accepted = NameDialog::AskForName(
|
2014-05-12 15:30:36 -07:00
|
|
|
this, QTStr("Basic.Main.AddSceneDlg.Title"),
|
|
|
|
QTStr("Basic.Main.AddSceneDlg.Text"), name, placeHolderText);
|
2013-12-29 08:54:06 -07:00
|
|
|
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
if (accepted) {
|
2014-03-10 13:39:51 -07:00
|
|
|
if (name.empty()) {
|
2019-03-30 15:44:44 +01:00
|
|
|
OBSMessageBox::warning(this,
|
2014-08-12 15:09:19 -04:00
|
|
|
QTStr("NoNameEntered.Title"),
|
|
|
|
QTStr("NoNameEntered.Text"));
|
2014-03-10 13:39:51 -07:00
|
|
|
on_actionAddScene_triggered();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_source_t *source = obs_get_source_by_name(name.c_str());
|
2013-12-29 09:17:00 -07:00
|
|
|
if (source) {
|
2019-03-30 15:44:44 +01:00
|
|
|
OBSMessageBox::warning(this, QTStr("NameExists.Title"),
|
2014-05-12 15:30:36 -07:00
|
|
|
QTStr("NameExists.Text"));
|
2013-12-29 09:17:00 -07:00
|
|
|
|
|
|
|
obs_source_release(source);
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
on_actionAddScene_triggered();
|
2013-12-29 09:17:00 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_scene_t *scene = obs_scene_create(name.c_str());
|
2014-08-04 08:41:15 -07:00
|
|
|
source = obs_scene_get_source(scene);
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
SetCurrentScene(source);
|
2015-12-05 05:58:05 -08:00
|
|
|
obs_scene_release(scene);
|
2013-12-29 08:54:06 -07:00
|
|
|
}
|
2013-11-07 16:45:03 -07:00
|
|
|
}
|
|
|
|
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
void OBSBasic::on_actionRemoveScene_triggered()
|
2013-11-07 16:45:03 -07:00
|
|
|
{
|
2014-06-30 13:45:58 -07:00
|
|
|
OBSScene scene = GetCurrentScene();
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_source_t *source = obs_scene_get_source(scene);
|
2014-06-30 13:45:58 -07:00
|
|
|
|
|
|
|
if (source && QueryRemoveSource(source))
|
|
|
|
obs_source_remove(source);
|
2013-11-07 16:45:03 -07:00
|
|
|
}
|
|
|
|
|
2015-06-25 03:53:48 -07:00
|
|
|
void OBSBasic::ChangeSceneIndex(bool relative, int offset, int invalidIdx)
|
|
|
|
{
|
|
|
|
int idx = ui->scenes->currentRow();
|
|
|
|
if (idx == -1 || idx == invalidIdx)
|
|
|
|
return;
|
|
|
|
|
|
|
|
sceneChanging = true;
|
|
|
|
|
|
|
|
QListWidgetItem *item = ui->scenes->takeItem(idx);
|
|
|
|
|
|
|
|
if (!relative)
|
|
|
|
idx = 0;
|
|
|
|
|
|
|
|
ui->scenes->insertItem(idx + offset, item);
|
|
|
|
ui->scenes->setCurrentRow(idx + offset);
|
|
|
|
item->setSelected(true);
|
|
|
|
|
|
|
|
sceneChanging = false;
|
2018-05-20 23:15:26 -03:00
|
|
|
|
|
|
|
OBSProjector::UpdateMultiviewProjectors();
|
2015-06-25 03:53:48 -07:00
|
|
|
}
|
|
|
|
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
void OBSBasic::on_actionSceneUp_triggered()
|
2013-11-07 16:45:03 -07:00
|
|
|
{
|
2015-06-25 03:53:48 -07:00
|
|
|
ChangeSceneIndex(true, -1, 0);
|
2013-11-07 16:45:03 -07:00
|
|
|
}
|
|
|
|
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
void OBSBasic::on_actionSceneDown_triggered()
|
2013-12-30 01:17:57 -07:00
|
|
|
{
|
2015-06-25 03:53:48 -07:00
|
|
|
ChangeSceneIndex(true, 1, ui->scenes->count() - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::MoveSceneToTop()
|
|
|
|
{
|
|
|
|
ChangeSceneIndex(false, 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::MoveSceneToBottom()
|
|
|
|
{
|
|
|
|
ChangeSceneIndex(false, ui->scenes->count() - 1,
|
|
|
|
ui->scenes->count() - 1);
|
2013-12-30 01:17:57 -07:00
|
|
|
}
|
|
|
|
|
2014-06-30 01:13:32 -07:00
|
|
|
void OBSBasic::EditSceneItemName()
|
|
|
|
{
|
2018-06-02 09:45:01 -07:00
|
|
|
int idx = GetTopSelectedSourceItem();
|
|
|
|
ui->sources->Edit(idx);
|
2014-06-30 01:13:32 -07:00
|
|
|
}
|
|
|
|
|
2016-03-15 20:55:40 -07:00
|
|
|
void OBSBasic::SetDeinterlacingMode()
|
|
|
|
{
|
|
|
|
QAction *action = reinterpret_cast<QAction *>(sender());
|
|
|
|
obs_deinterlace_mode mode =
|
|
|
|
(obs_deinterlace_mode)action->property("mode").toInt();
|
|
|
|
OBSSceneItem sceneItem = GetCurrentSceneItem();
|
|
|
|
obs_source_t *source = obs_sceneitem_get_source(sceneItem);
|
|
|
|
|
|
|
|
obs_source_set_deinterlace_mode(source, mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::SetDeinterlacingOrder()
|
|
|
|
{
|
|
|
|
QAction *action = reinterpret_cast<QAction *>(sender());
|
|
|
|
obs_deinterlace_field_order order =
|
|
|
|
(obs_deinterlace_field_order)action->property("order").toInt();
|
|
|
|
OBSSceneItem sceneItem = GetCurrentSceneItem();
|
|
|
|
obs_source_t *source = obs_sceneitem_get_source(sceneItem);
|
|
|
|
|
|
|
|
obs_source_set_deinterlace_field_order(source, order);
|
|
|
|
}
|
|
|
|
|
2019-01-31 03:33:15 +01:00
|
|
|
QMenu *OBSBasic::AddDeinterlacingMenu(QMenu *menu, obs_source_t *source)
|
2016-03-15 20:55:40 -07:00
|
|
|
{
|
|
|
|
obs_deinterlace_mode deinterlaceMode =
|
|
|
|
obs_source_get_deinterlace_mode(source);
|
|
|
|
obs_deinterlace_field_order deinterlaceOrder =
|
|
|
|
obs_source_get_deinterlace_field_order(source);
|
|
|
|
QAction *action;
|
|
|
|
|
|
|
|
#define ADD_MODE(name, mode) \
|
|
|
|
action = menu->addAction(QTStr("" name), this, \
|
|
|
|
SLOT(SetDeinterlacingMode())); \
|
|
|
|
action->setProperty("mode", (int)mode); \
|
|
|
|
action->setCheckable(true); \
|
|
|
|
action->setChecked(deinterlaceMode == mode);
|
|
|
|
|
|
|
|
ADD_MODE("Disable", OBS_DEINTERLACE_MODE_DISABLE);
|
|
|
|
ADD_MODE("Deinterlacing.Discard", OBS_DEINTERLACE_MODE_DISCARD);
|
|
|
|
ADD_MODE("Deinterlacing.Retro", OBS_DEINTERLACE_MODE_RETRO);
|
|
|
|
ADD_MODE("Deinterlacing.Blend", OBS_DEINTERLACE_MODE_BLEND);
|
|
|
|
ADD_MODE("Deinterlacing.Blend2x", OBS_DEINTERLACE_MODE_BLEND_2X);
|
|
|
|
ADD_MODE("Deinterlacing.Linear", OBS_DEINTERLACE_MODE_LINEAR);
|
|
|
|
ADD_MODE("Deinterlacing.Linear2x", OBS_DEINTERLACE_MODE_LINEAR_2X);
|
|
|
|
ADD_MODE("Deinterlacing.Yadif", OBS_DEINTERLACE_MODE_YADIF);
|
|
|
|
ADD_MODE("Deinterlacing.Yadif2x", OBS_DEINTERLACE_MODE_YADIF_2X);
|
|
|
|
#undef ADD_MODE
|
|
|
|
|
|
|
|
menu->addSeparator();
|
|
|
|
|
|
|
|
#define ADD_ORDER(name, order) \
|
|
|
|
action = menu->addAction(QTStr("Deinterlacing." name), this, \
|
|
|
|
SLOT(SetDeinterlacingOrder())); \
|
|
|
|
action->setProperty("order", (int)order); \
|
|
|
|
action->setCheckable(true); \
|
|
|
|
action->setChecked(deinterlaceOrder == order);
|
|
|
|
|
|
|
|
ADD_ORDER("TopFieldFirst", OBS_DEINTERLACE_FIELD_ORDER_TOP);
|
|
|
|
ADD_ORDER("BottomFieldFirst", OBS_DEINTERLACE_FIELD_ORDER_BOTTOM);
|
|
|
|
#undef ADD_ORDER
|
|
|
|
|
|
|
|
return menu;
|
|
|
|
}
|
|
|
|
|
2016-06-29 07:08:02 -07:00
|
|
|
void OBSBasic::SetScaleFilter()
|
|
|
|
{
|
|
|
|
QAction *action = reinterpret_cast<QAction *>(sender());
|
|
|
|
obs_scale_type mode = (obs_scale_type)action->property("mode").toInt();
|
|
|
|
OBSSceneItem sceneItem = GetCurrentSceneItem();
|
|
|
|
|
|
|
|
obs_sceneitem_set_scale_filter(sceneItem, mode);
|
|
|
|
}
|
|
|
|
|
2019-01-31 03:33:15 +01:00
|
|
|
QMenu *OBSBasic::AddScaleFilteringMenu(QMenu *menu, obs_sceneitem_t *item)
|
2016-06-29 07:08:02 -07:00
|
|
|
{
|
|
|
|
obs_scale_type scaleFilter = obs_sceneitem_get_scale_filter(item);
|
|
|
|
QAction *action;
|
|
|
|
|
|
|
|
#define ADD_MODE(name, mode) \
|
|
|
|
action = \
|
|
|
|
menu->addAction(QTStr("" name), this, SLOT(SetScaleFilter())); \
|
|
|
|
action->setProperty("mode", (int)mode); \
|
|
|
|
action->setCheckable(true); \
|
|
|
|
action->setChecked(scaleFilter == mode);
|
|
|
|
|
|
|
|
ADD_MODE("Disable", OBS_SCALE_DISABLE);
|
|
|
|
ADD_MODE("ScaleFiltering.Point", OBS_SCALE_POINT);
|
|
|
|
ADD_MODE("ScaleFiltering.Bilinear", OBS_SCALE_BILINEAR);
|
|
|
|
ADD_MODE("ScaleFiltering.Bicubic", OBS_SCALE_BICUBIC);
|
|
|
|
ADD_MODE("ScaleFiltering.Lanczos", OBS_SCALE_LANCZOS);
|
2019-03-06 20:53:15 -08:00
|
|
|
ADD_MODE("ScaleFiltering.Area", OBS_SCALE_AREA);
|
2016-06-29 07:08:02 -07:00
|
|
|
#undef ADD_MODE
|
|
|
|
|
|
|
|
return menu;
|
|
|
|
}
|
|
|
|
|
2019-01-31 03:33:15 +01:00
|
|
|
QMenu *OBSBasic::AddBackgroundColorMenu(QMenu *menu,
|
|
|
|
QWidgetAction *widgetAction,
|
|
|
|
ColorSelect *select,
|
|
|
|
obs_sceneitem_t *item)
|
2018-08-02 08:23:12 +12:00
|
|
|
{
|
|
|
|
QAction *action;
|
|
|
|
|
|
|
|
menu->setStyleSheet(QString(
|
|
|
|
"*[bgColor=\"1\"]{background-color:rgba(255,68,68,33%);}"
|
|
|
|
"*[bgColor=\"2\"]{background-color:rgba(255,255,68,33%);}"
|
|
|
|
"*[bgColor=\"3\"]{background-color:rgba(68,255,68,33%);}"
|
|
|
|
"*[bgColor=\"4\"]{background-color:rgba(68,255,255,33%);}"
|
|
|
|
"*[bgColor=\"5\"]{background-color:rgba(68,68,255,33%);}"
|
|
|
|
"*[bgColor=\"6\"]{background-color:rgba(255,68,255,33%);}"
|
|
|
|
"*[bgColor=\"7\"]{background-color:rgba(68,68,68,33%);}"
|
|
|
|
"*[bgColor=\"8\"]{background-color:rgba(255,255,255,33%);}"));
|
|
|
|
|
|
|
|
obs_data_t *privData = obs_sceneitem_get_private_settings(item);
|
|
|
|
obs_data_release(privData);
|
|
|
|
|
|
|
|
obs_data_set_default_int(privData, "color-preset", 0);
|
|
|
|
int preset = obs_data_get_int(privData, "color-preset");
|
|
|
|
|
|
|
|
action = menu->addAction(QTStr("Clear"), this, +SLOT(ColorChange()));
|
|
|
|
action->setCheckable(true);
|
|
|
|
action->setProperty("bgColor", 0);
|
|
|
|
action->setChecked(preset == 0);
|
|
|
|
|
|
|
|
action = menu->addAction(QTStr("CustomColor"), this,
|
|
|
|
+SLOT(ColorChange()));
|
|
|
|
action->setCheckable(true);
|
|
|
|
action->setProperty("bgColor", 1);
|
|
|
|
action->setChecked(preset == 1);
|
|
|
|
|
|
|
|
menu->addSeparator();
|
|
|
|
|
|
|
|
widgetAction->setDefaultWidget(select);
|
|
|
|
|
|
|
|
for (int i = 1; i < 9; i++) {
|
|
|
|
stringstream button;
|
|
|
|
button << "preset" << i;
|
|
|
|
QPushButton *colorButton =
|
|
|
|
select->findChild<QPushButton *>(button.str().c_str());
|
|
|
|
if (preset == i + 1)
|
|
|
|
colorButton->setStyleSheet("border: 2px solid black");
|
|
|
|
|
|
|
|
colorButton->setProperty("bgColor", i);
|
|
|
|
select->connect(colorButton, SIGNAL(released()), this,
|
|
|
|
SLOT(ColorChange()));
|
|
|
|
}
|
|
|
|
|
|
|
|
menu->addAction(widgetAction);
|
|
|
|
|
|
|
|
return menu;
|
|
|
|
}
|
|
|
|
|
2019-01-31 03:33:15 +01:00
|
|
|
ColorSelect::ColorSelect(QWidget *parent)
|
|
|
|
: QWidget(parent), ui(new Ui::ColorSelect)
|
|
|
|
{
|
|
|
|
ui->setupUi(this);
|
|
|
|
}
|
|
|
|
|
2018-06-02 09:45:01 -07:00
|
|
|
void OBSBasic::CreateSourcePopupMenu(int idx, bool preview)
|
2013-11-07 16:45:03 -07:00
|
|
|
{
|
2015-01-16 11:21:11 +02:00
|
|
|
QMenu popup(this);
|
2020-04-19 09:31:18 -05:00
|
|
|
delete previewProjectorSource;
|
|
|
|
delete sourceProjector;
|
|
|
|
delete scaleFilteringMenu;
|
|
|
|
delete colorMenu;
|
|
|
|
delete colorWidgetAction;
|
|
|
|
delete colorSelect;
|
|
|
|
delete deinterlaceMenu;
|
2015-04-02 21:35:46 -07:00
|
|
|
|
|
|
|
if (preview) {
|
|
|
|
QAction *action = popup.addAction(
|
|
|
|
QTStr("Basic.Main.PreviewConextMenu.Enable"), this,
|
|
|
|
SLOT(TogglePreview()));
|
|
|
|
action->setCheckable(true);
|
2015-08-02 00:02:58 -07:00
|
|
|
action->setChecked(
|
|
|
|
obs_display_enabled(ui->preview->GetDisplay()));
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
if (IsPreviewProgramMode())
|
|
|
|
action->setEnabled(false);
|
2015-04-02 21:35:46 -07:00
|
|
|
|
2016-11-05 12:48:46 -04:00
|
|
|
popup.addAction(ui->actionLockPreview);
|
|
|
|
popup.addMenu(ui->scalingMenu);
|
2016-07-26 01:32:43 -07:00
|
|
|
|
2020-04-19 09:31:18 -05:00
|
|
|
previewProjectorSource = new QMenu(QTStr("PreviewProjector"));
|
2019-01-31 03:33:15 +01:00
|
|
|
AddProjectorMenuMonitors(previewProjectorSource, this,
|
2017-11-30 06:17:07 -08:00
|
|
|
SLOT(OpenPreviewProjector()));
|
2015-04-04 01:40:15 -07:00
|
|
|
|
2019-01-31 03:33:15 +01:00
|
|
|
popup.addMenu(previewProjectorSource);
|
2015-04-04 01:40:15 -07:00
|
|
|
|
2017-03-01 22:08:49 -06:00
|
|
|
QAction *previewWindow =
|
|
|
|
popup.addAction(QTStr("PreviewWindow"), this,
|
2017-11-30 06:17:07 -08:00
|
|
|
SLOT(OpenPreviewWindow()));
|
2017-03-01 22:08:49 -06:00
|
|
|
|
|
|
|
popup.addAction(previewWindow);
|
|
|
|
|
2020-07-23 08:53:21 -05:00
|
|
|
popup.addAction(QTStr("Screenshot.Preview"), this,
|
|
|
|
SLOT(ScreenshotScene()));
|
|
|
|
|
2015-04-02 21:35:46 -07:00
|
|
|
popup.addSeparator();
|
|
|
|
}
|
|
|
|
|
2014-06-30 01:13:32 -07:00
|
|
|
QPointer<QMenu> addSourceMenu = CreateAddSourcePopupMenu();
|
|
|
|
if (addSourceMenu)
|
|
|
|
popup.addMenu(addSourceMenu);
|
|
|
|
|
2017-03-25 04:19:29 -07:00
|
|
|
ui->actionCopyFilters->setEnabled(false);
|
2018-01-31 21:01:46 +01:00
|
|
|
ui->actionCopySource->setEnabled(false);
|
2017-03-25 04:19:29 -07:00
|
|
|
|
2018-06-02 09:45:01 -07:00
|
|
|
if (ui->sources->MultipleBaseSelected()) {
|
|
|
|
popup.addSeparator();
|
|
|
|
popup.addAction(QTStr("Basic.Main.GroupItems"), ui->sources,
|
|
|
|
SLOT(GroupSelectedItems()));
|
|
|
|
|
|
|
|
} else if (ui->sources->GroupsSelected()) {
|
|
|
|
popup.addSeparator();
|
|
|
|
popup.addAction(QTStr("Basic.Main.Ungroup"), ui->sources,
|
|
|
|
SLOT(UngroupSelectedGroups()));
|
|
|
|
}
|
|
|
|
|
2017-03-25 04:19:29 -07:00
|
|
|
popup.addSeparator();
|
|
|
|
popup.addAction(ui->actionCopySource);
|
|
|
|
popup.addAction(ui->actionPasteRef);
|
|
|
|
popup.addAction(ui->actionPasteDup);
|
|
|
|
popup.addSeparator();
|
|
|
|
|
|
|
|
popup.addSeparator();
|
|
|
|
popup.addAction(ui->actionCopyFilters);
|
|
|
|
popup.addAction(ui->actionPasteFilters);
|
|
|
|
popup.addSeparator();
|
|
|
|
|
2018-06-02 09:45:01 -07:00
|
|
|
if (idx != -1) {
|
2014-06-30 01:13:32 -07:00
|
|
|
if (addSourceMenu)
|
|
|
|
popup.addSeparator();
|
|
|
|
|
2018-06-02 09:45:01 -07:00
|
|
|
OBSSceneItem sceneItem = ui->sources->Get(idx);
|
2020-02-17 13:01:04 -06:00
|
|
|
bool lock = obs_sceneitem_locked(sceneItem);
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_source_t *source = obs_sceneitem_get_source(sceneItem);
|
2016-03-15 20:55:40 -07:00
|
|
|
uint32_t flags = obs_source_get_output_flags(source);
|
|
|
|
bool isAsyncVideo = (flags & OBS_SOURCE_ASYNC_VIDEO) ==
|
|
|
|
OBS_SOURCE_ASYNC_VIDEO;
|
2017-09-13 21:10:45 -07:00
|
|
|
bool hasAudio = (flags & OBS_SOURCE_AUDIO) == OBS_SOURCE_AUDIO;
|
2014-09-15 18:16:16 -05:00
|
|
|
QAction *action;
|
|
|
|
|
2020-04-19 09:31:18 -05:00
|
|
|
colorMenu = new QMenu(QTStr("ChangeBG"));
|
|
|
|
colorWidgetAction = new QWidgetAction(colorMenu);
|
|
|
|
colorSelect = new ColorSelect(colorMenu);
|
2019-01-31 03:33:15 +01:00
|
|
|
popup.addMenu(AddBackgroundColorMenu(
|
|
|
|
colorMenu, colorWidgetAction, colorSelect, sceneItem));
|
2014-06-30 01:13:32 -07:00
|
|
|
popup.addAction(QTStr("Rename"), this,
|
|
|
|
SLOT(EditSceneItemName()));
|
2014-06-30 18:21:40 -07:00
|
|
|
popup.addAction(QTStr("Remove"), this,
|
2018-03-03 02:54:12 -03:00
|
|
|
SLOT(on_actionRemoveSource_triggered()));
|
2014-06-30 19:47:06 -07:00
|
|
|
popup.addSeparator();
|
|
|
|
popup.addMenu(ui->orderMenu);
|
2014-06-30 01:13:32 -07:00
|
|
|
popup.addMenu(ui->transformMenu);
|
2015-04-04 01:40:15 -07:00
|
|
|
|
2020-02-17 13:01:04 -06:00
|
|
|
ui->actionResetTransform->setEnabled(!lock);
|
|
|
|
ui->actionRotate90CW->setEnabled(!lock);
|
|
|
|
ui->actionRotate90CCW->setEnabled(!lock);
|
|
|
|
ui->actionRotate180->setEnabled(!lock);
|
|
|
|
ui->actionFlipHorizontal->setEnabled(!lock);
|
|
|
|
ui->actionFlipVertical->setEnabled(!lock);
|
|
|
|
ui->actionFitToScreen->setEnabled(!lock);
|
|
|
|
ui->actionStretchToScreen->setEnabled(!lock);
|
|
|
|
ui->actionCenterToScreen->setEnabled(!lock);
|
|
|
|
ui->actionVerticalCenter->setEnabled(!lock);
|
|
|
|
ui->actionHorizontalCenter->setEnabled(!lock);
|
|
|
|
|
2020-04-19 09:31:18 -05:00
|
|
|
sourceProjector = new QMenu(QTStr("SourceProjector"));
|
2015-04-04 01:40:15 -07:00
|
|
|
AddProjectorMenuMonitors(sourceProjector, this,
|
|
|
|
SLOT(OpenSourceProjector()));
|
|
|
|
|
2017-03-01 22:08:49 -06:00
|
|
|
QAction *sourceWindow = popup.addAction(
|
|
|
|
QTStr("SourceWindow"), this, SLOT(OpenSourceWindow()));
|
|
|
|
|
|
|
|
popup.addAction(sourceWindow);
|
|
|
|
|
2015-04-04 01:40:15 -07:00
|
|
|
popup.addSeparator();
|
2017-09-13 21:10:45 -07:00
|
|
|
|
|
|
|
if (hasAudio) {
|
|
|
|
QAction *actionHideMixer =
|
|
|
|
popup.addAction(QTStr("HideMixer"), this,
|
|
|
|
SLOT(ToggleHideMixer()));
|
|
|
|
actionHideMixer->setCheckable(true);
|
|
|
|
actionHideMixer->setChecked(SourceMixerHidden(source));
|
|
|
|
}
|
|
|
|
|
2016-03-15 20:55:40 -07:00
|
|
|
if (isAsyncVideo) {
|
2020-04-19 09:31:18 -05:00
|
|
|
deinterlaceMenu = new QMenu(QTStr("Deinterlacing"));
|
2019-01-31 03:33:15 +01:00
|
|
|
popup.addMenu(
|
|
|
|
AddDeinterlacingMenu(deinterlaceMenu, source));
|
2016-03-15 20:55:40 -07:00
|
|
|
popup.addSeparator();
|
|
|
|
}
|
2016-06-29 07:08:02 -07:00
|
|
|
|
2018-08-29 17:45:59 -05:00
|
|
|
QAction *resizeOutput =
|
|
|
|
popup.addAction(QTStr("ResizeOutputSizeOfSource"), this,
|
|
|
|
SLOT(ResizeOutputSizeOfSource()));
|
|
|
|
|
|
|
|
int width = obs_source_get_width(source);
|
|
|
|
int height = obs_source_get_height(source);
|
|
|
|
|
2019-02-19 19:26:37 -08:00
|
|
|
resizeOutput->setEnabled(!obs_video_active());
|
2018-08-29 17:45:59 -05:00
|
|
|
|
2020-05-24 00:36:50 +02:00
|
|
|
if (width < 8 || height < 8)
|
2018-08-29 17:45:59 -05:00
|
|
|
resizeOutput->setEnabled(false);
|
|
|
|
|
2020-04-19 09:31:18 -05:00
|
|
|
scaleFilteringMenu = new QMenu(QTStr("ScaleFiltering"));
|
2019-01-31 03:33:15 +01:00
|
|
|
popup.addMenu(
|
|
|
|
AddScaleFilteringMenu(scaleFilteringMenu, sceneItem));
|
2016-06-29 07:08:02 -07:00
|
|
|
popup.addSeparator();
|
|
|
|
|
2015-04-04 01:40:15 -07:00
|
|
|
popup.addMenu(sourceProjector);
|
2017-03-01 22:08:49 -06:00
|
|
|
popup.addAction(sourceWindow);
|
2020-07-23 08:53:21 -05:00
|
|
|
popup.addAction(QTStr("Screenshot.Source"), this,
|
|
|
|
SLOT(ScreenshotSelectedSource()));
|
2014-06-30 01:13:32 -07:00
|
|
|
popup.addSeparator();
|
2014-09-15 18:16:16 -05:00
|
|
|
|
|
|
|
action = popup.addAction(QTStr("Interact"), this,
|
|
|
|
SLOT(on_actionInteract_triggered()));
|
|
|
|
|
|
|
|
action->setEnabled(obs_source_get_output_flags(source) &
|
|
|
|
OBS_SOURCE_INTERACTION);
|
|
|
|
|
2015-02-25 21:23:57 -08:00
|
|
|
popup.addAction(QTStr("Filters"), this, SLOT(OpenFilters()));
|
2014-06-30 01:13:32 -07:00
|
|
|
popup.addAction(QTStr("Properties"), this,
|
|
|
|
SLOT(on_actionSourceProperties_triggered()));
|
2017-03-25 04:19:29 -07:00
|
|
|
|
|
|
|
ui->actionCopyFilters->setEnabled(true);
|
2018-01-31 21:01:46 +01:00
|
|
|
ui->actionCopySource->setEnabled(true);
|
2018-04-18 08:28:09 -07:00
|
|
|
} else {
|
|
|
|
ui->actionPasteFilters->setEnabled(false);
|
2014-06-30 01:13:32 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
popup.exec(QCursor::pos());
|
2013-11-07 16:45:03 -07:00
|
|
|
}
|
|
|
|
|
2015-04-02 21:35:46 -07:00
|
|
|
void OBSBasic::on_sources_customContextMenuRequested(const QPoint &pos)
|
|
|
|
{
|
2018-06-02 09:45:01 -07:00
|
|
|
if (ui->scenes->count()) {
|
|
|
|
QModelIndex idx = ui->sources->indexAt(pos);
|
|
|
|
CreateSourcePopupMenu(idx.row(), false);
|
|
|
|
}
|
2014-10-15 22:05:56 +02:00
|
|
|
}
|
|
|
|
|
2017-09-21 15:37:01 -05:00
|
|
|
void OBSBasic::on_scenes_itemDoubleClicked(QListWidgetItem *witem)
|
|
|
|
{
|
|
|
|
if (!witem)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (IsPreviewProgramMode()) {
|
|
|
|
bool doubleClickSwitch =
|
|
|
|
config_get_bool(App()->GlobalConfig(), "BasicWindow",
|
|
|
|
"TransitionOnDoubleClick");
|
|
|
|
|
2019-10-09 04:01:55 -05:00
|
|
|
if (doubleClickSwitch)
|
|
|
|
TransitionClicked();
|
2017-09-21 15:37:01 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-10 18:47:48 -07:00
|
|
|
void OBSBasic::AddSource(const char *id)
|
2013-12-30 06:56:39 -07:00
|
|
|
{
|
2014-06-29 17:33:40 -07:00
|
|
|
if (id && *id) {
|
|
|
|
OBSBasicSourceSelect sourceSelect(this, id);
|
|
|
|
sourceSelect.exec();
|
2018-07-23 18:44:02 -07:00
|
|
|
if (sourceSelect.newSource && strcmp(id, "group") != 0)
|
2015-10-24 16:07:05 +02:00
|
|
|
CreatePropertiesWindow(sourceSelect.newSource);
|
2014-06-29 17:33:40 -07:00
|
|
|
}
|
2013-12-30 06:56:39 -07:00
|
|
|
}
|
|
|
|
|
2014-06-29 17:33:40 -07:00
|
|
|
QMenu *OBSBasic::CreateAddSourcePopupMenu()
|
2013-11-07 16:45:03 -07:00
|
|
|
{
|
2020-03-09 06:03:10 -07:00
|
|
|
const char *unversioned_type;
|
2013-12-30 06:56:39 -07:00
|
|
|
const char *type;
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
bool foundValues = false;
|
2016-11-03 12:28:33 -07:00
|
|
|
bool foundDeprecated = false;
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
size_t idx = 0;
|
2013-12-30 06:56:39 -07:00
|
|
|
|
2015-01-16 11:21:11 +02:00
|
|
|
QMenu *popup = new QMenu(QTStr("Add"), this);
|
2016-11-03 12:28:33 -07:00
|
|
|
QMenu *deprecated = new QMenu(QTStr("Deprecated"), popup);
|
2014-04-26 23:47:50 -07:00
|
|
|
|
2016-09-14 16:27:04 -07:00
|
|
|
auto getActionAfter = [](QMenu *menu, const QString &name) {
|
|
|
|
QList<QAction *> actions = menu->actions();
|
|
|
|
|
|
|
|
for (QAction *menuAction : actions) {
|
|
|
|
if (menuAction->text().compare(name) >= 0)
|
|
|
|
return menuAction;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (QAction *)nullptr;
|
|
|
|
};
|
|
|
|
|
|
|
|
auto addSource = [this, getActionAfter](QMenu *popup, const char *type,
|
|
|
|
const char *name) {
|
|
|
|
QString qname = QT_UTF8(name);
|
|
|
|
QAction *popupItem = new QAction(qname, this);
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
popupItem->setData(QT_UTF8(type));
|
2014-06-29 17:33:40 -07:00
|
|
|
connect(popupItem, SIGNAL(triggered(bool)), this,
|
|
|
|
SLOT(AddSourceFromAction()));
|
2016-09-14 16:27:04 -07:00
|
|
|
|
2019-07-27 23:59:16 -05:00
|
|
|
QIcon icon;
|
|
|
|
|
|
|
|
if (strcmp(type, "scene") == 0)
|
|
|
|
icon = GetSceneIcon();
|
|
|
|
else
|
|
|
|
icon = GetSourceIcon(type);
|
|
|
|
|
|
|
|
popupItem->setIcon(icon);
|
|
|
|
|
2016-09-14 16:27:04 -07:00
|
|
|
QAction *after = getActionAfter(popup, qname);
|
|
|
|
popup->insertAction(after, popupItem);
|
2016-07-01 15:23:06 -07:00
|
|
|
};
|
2013-12-30 06:56:39 -07:00
|
|
|
|
2020-03-09 06:03:10 -07:00
|
|
|
while (obs_enum_input_types2(idx++, &type, &unversioned_type)) {
|
2016-07-01 15:23:06 -07:00
|
|
|
const char *name = obs_source_get_display_name(type);
|
2016-09-14 16:32:56 -07:00
|
|
|
uint32_t caps = obs_get_source_output_flags(type);
|
2016-07-01 15:23:06 -07:00
|
|
|
|
2017-12-25 13:51:06 -08:00
|
|
|
if ((caps & OBS_SOURCE_CAP_DISABLED) != 0)
|
|
|
|
continue;
|
|
|
|
|
2016-09-14 16:32:56 -07:00
|
|
|
if ((caps & OBS_SOURCE_DEPRECATED) == 0) {
|
2020-03-09 06:03:10 -07:00
|
|
|
addSource(popup, unversioned_type, name);
|
2016-11-03 12:28:33 -07:00
|
|
|
} else {
|
2020-03-09 06:03:10 -07:00
|
|
|
addSource(deprecated, unversioned_type, name);
|
2016-11-03 12:28:33 -07:00
|
|
|
foundDeprecated = true;
|
2016-09-14 16:32:56 -07:00
|
|
|
}
|
2016-11-03 12:28:33 -07:00
|
|
|
foundValues = true;
|
2013-12-30 06:56:39 -07:00
|
|
|
}
|
|
|
|
|
2016-09-14 16:27:04 -07:00
|
|
|
addSource(popup, "scene", Str("Basic.Scene"));
|
2016-07-01 15:23:06 -07:00
|
|
|
|
2018-06-02 09:45:01 -07:00
|
|
|
popup->addSeparator();
|
|
|
|
QAction *addGroup = new QAction(QTStr("Group"), this);
|
2018-07-15 18:50:33 -07:00
|
|
|
addGroup->setData(QT_UTF8("group"));
|
2019-07-27 23:59:16 -05:00
|
|
|
addGroup->setIcon(GetGroupIcon());
|
2018-06-02 09:45:01 -07:00
|
|
|
connect(addGroup, SIGNAL(triggered(bool)), this,
|
2018-07-15 18:50:33 -07:00
|
|
|
SLOT(AddSourceFromAction()));
|
2018-06-02 09:45:01 -07:00
|
|
|
popup->addAction(addGroup);
|
|
|
|
|
2016-11-03 12:28:33 -07:00
|
|
|
if (!foundDeprecated) {
|
|
|
|
delete deprecated;
|
|
|
|
deprecated = nullptr;
|
|
|
|
}
|
|
|
|
|
2014-06-29 17:33:40 -07:00
|
|
|
if (!foundValues) {
|
|
|
|
delete popup;
|
|
|
|
popup = nullptr;
|
2016-11-03 12:28:33 -07:00
|
|
|
|
|
|
|
} else if (foundDeprecated) {
|
2018-06-02 09:45:01 -07:00
|
|
|
popup->addSeparator();
|
2016-11-03 12:28:33 -07:00
|
|
|
popup->addMenu(deprecated);
|
2013-12-30 06:56:39 -07:00
|
|
|
}
|
2014-06-29 17:33:40 -07:00
|
|
|
|
|
|
|
return popup;
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::AddSourceFromAction()
|
|
|
|
{
|
|
|
|
QAction *action = qobject_cast<QAction *>(sender());
|
|
|
|
if (!action)
|
|
|
|
return;
|
|
|
|
|
|
|
|
AddSource(QT_TO_UTF8(action->data().toString()));
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::AddSourcePopupMenu(const QPoint &pos)
|
|
|
|
{
|
|
|
|
if (!GetCurrentScene()) {
|
|
|
|
// Tell the user he needs a scene first (help beginners).
|
2017-05-13 14:06:32 -07:00
|
|
|
OBSMessageBox::information(
|
2014-06-29 17:33:40 -07:00
|
|
|
this, QTStr("Basic.Main.AddSourceHelp.Title"),
|
|
|
|
QTStr("Basic.Main.AddSourceHelp.Text"));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-07-31 21:11:31 -07:00
|
|
|
QScopedPointer<QMenu> popup(CreateAddSourcePopupMenu());
|
2014-06-29 17:33:40 -07:00
|
|
|
if (popup)
|
|
|
|
popup->exec(pos);
|
2013-12-30 06:56:39 -07:00
|
|
|
}
|
|
|
|
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
void OBSBasic::on_actionAddSource_triggered()
|
2013-12-30 06:56:39 -07:00
|
|
|
{
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
AddSourcePopupMenu(QCursor::pos());
|
2013-11-07 16:45:03 -07:00
|
|
|
}
|
|
|
|
|
2018-06-02 09:45:01 -07:00
|
|
|
static bool remove_items(obs_scene_t *, obs_sceneitem_t *item, void *param)
|
|
|
|
{
|
|
|
|
vector<OBSSceneItem> &items =
|
|
|
|
*reinterpret_cast<vector<OBSSceneItem> *>(param);
|
|
|
|
|
|
|
|
if (obs_sceneitem_selected(item)) {
|
|
|
|
items.emplace_back(item);
|
|
|
|
} else if (obs_sceneitem_is_group(item)) {
|
|
|
|
obs_sceneitem_group_enum_items(item, remove_items, &items);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
void OBSBasic::on_actionRemoveSource_triggered()
|
2013-11-07 16:45:03 -07:00
|
|
|
{
|
2016-07-02 15:44:24 -07:00
|
|
|
vector<OBSSceneItem> items;
|
2014-06-30 13:45:58 -07:00
|
|
|
|
2018-06-02 09:45:01 -07:00
|
|
|
obs_scene_enum_items(GetCurrentScene(), remove_items, &items);
|
2016-07-02 15:44:24 -07:00
|
|
|
|
|
|
|
if (!items.size())
|
|
|
|
return;
|
|
|
|
|
|
|
|
auto removeMultiple = [this](size_t count) {
|
|
|
|
QString text = QTStr("ConfirmRemove.TextMultiple")
|
|
|
|
.arg(QString::number(count));
|
|
|
|
|
|
|
|
QMessageBox remove_items(this);
|
|
|
|
remove_items.setText(text);
|
2020-06-25 00:38:35 +01:00
|
|
|
QPushButton *Yes = remove_items.addButton(QTStr("Yes"),
|
|
|
|
QMessageBox::YesRole);
|
|
|
|
remove_items.setDefaultButton(Yes);
|
2016-07-02 15:44:24 -07:00
|
|
|
remove_items.addButton(QTStr("No"), QMessageBox::NoRole);
|
|
|
|
remove_items.setIcon(QMessageBox::Question);
|
|
|
|
remove_items.setWindowTitle(QTStr("ConfirmRemove.Title"));
|
|
|
|
remove_items.exec();
|
|
|
|
|
|
|
|
return Yes == remove_items.clickedButton();
|
|
|
|
};
|
|
|
|
|
|
|
|
if (items.size() == 1) {
|
|
|
|
OBSSceneItem &item = items[0];
|
|
|
|
obs_source_t *source = obs_sceneitem_get_source(item);
|
|
|
|
|
|
|
|
if (source && QueryRemoveSource(source))
|
|
|
|
obs_sceneitem_remove(item);
|
|
|
|
} else {
|
|
|
|
if (removeMultiple(items.size())) {
|
|
|
|
for (auto &item : items)
|
|
|
|
obs_sceneitem_remove(item);
|
|
|
|
}
|
|
|
|
}
|
2013-11-07 16:45:03 -07:00
|
|
|
}
|
|
|
|
|
2014-09-15 18:16:16 -05:00
|
|
|
void OBSBasic::on_actionInteract_triggered()
|
|
|
|
{
|
|
|
|
OBSSceneItem item = GetCurrentSceneItem();
|
|
|
|
OBSSource source = obs_sceneitem_get_source(item);
|
|
|
|
|
|
|
|
if (source)
|
|
|
|
CreateInteractionWindow(source);
|
|
|
|
}
|
|
|
|
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
void OBSBasic::on_actionSourceProperties_triggered()
|
2013-11-07 16:45:03 -07:00
|
|
|
{
|
2014-03-23 01:07:54 -07:00
|
|
|
OBSSceneItem item = GetCurrentSceneItem();
|
2014-08-03 14:39:19 -07:00
|
|
|
OBSSource source = obs_sceneitem_get_source(item);
|
2014-03-23 01:07:54 -07:00
|
|
|
|
2014-07-22 22:16:57 +02:00
|
|
|
if (source)
|
|
|
|
CreatePropertiesWindow(source);
|
2013-11-07 16:45:03 -07:00
|
|
|
}
|
|
|
|
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
void OBSBasic::on_actionSourceUp_triggered()
|
2013-11-07 16:45:03 -07:00
|
|
|
{
|
2014-05-15 17:40:53 -07:00
|
|
|
OBSSceneItem item = GetCurrentSceneItem();
|
2014-08-03 14:39:19 -07:00
|
|
|
obs_sceneitem_set_order(item, OBS_ORDER_MOVE_UP);
|
2013-11-07 16:45:03 -07:00
|
|
|
}
|
2013-12-10 11:22:33 -07:00
|
|
|
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
void OBSBasic::on_actionSourceDown_triggered()
|
2013-12-10 21:14:45 -07:00
|
|
|
{
|
2014-05-15 17:40:53 -07:00
|
|
|
OBSSceneItem item = GetCurrentSceneItem();
|
2014-08-03 14:39:19 -07:00
|
|
|
obs_sceneitem_set_order(item, OBS_ORDER_MOVE_DOWN);
|
2013-12-10 21:14:45 -07:00
|
|
|
}
|
|
|
|
|
2014-06-30 19:47:06 -07:00
|
|
|
void OBSBasic::on_actionMoveUp_triggered()
|
|
|
|
{
|
|
|
|
OBSSceneItem item = GetCurrentSceneItem();
|
2014-08-03 14:39:19 -07:00
|
|
|
obs_sceneitem_set_order(item, OBS_ORDER_MOVE_UP);
|
2014-06-30 19:47:06 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionMoveDown_triggered()
|
|
|
|
{
|
|
|
|
OBSSceneItem item = GetCurrentSceneItem();
|
2014-08-03 14:39:19 -07:00
|
|
|
obs_sceneitem_set_order(item, OBS_ORDER_MOVE_DOWN);
|
2014-06-30 19:47:06 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionMoveToTop_triggered()
|
|
|
|
{
|
|
|
|
OBSSceneItem item = GetCurrentSceneItem();
|
2014-08-03 14:39:19 -07:00
|
|
|
obs_sceneitem_set_order(item, OBS_ORDER_MOVE_TOP);
|
2014-06-30 19:47:06 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionMoveToBottom_triggered()
|
|
|
|
{
|
|
|
|
OBSSceneItem item = GetCurrentSceneItem();
|
2014-08-03 14:39:19 -07:00
|
|
|
obs_sceneitem_set_order(item, OBS_ORDER_MOVE_BOTTOM);
|
2014-06-30 19:47:06 -07:00
|
|
|
}
|
|
|
|
|
2018-03-16 16:22:37 -07:00
|
|
|
static BPtr<char> ReadLogFile(const char *subdir, const char *log)
|
2014-05-18 17:44:10 -07:00
|
|
|
{
|
2015-01-15 23:44:38 -08:00
|
|
|
char logDir[512];
|
2018-03-16 16:22:37 -07:00
|
|
|
if (GetConfigPath(logDir, sizeof(logDir), subdir) <= 0)
|
2015-01-15 23:44:38 -08:00
|
|
|
return nullptr;
|
2014-05-18 17:44:10 -07:00
|
|
|
|
2019-08-17 08:49:08 -07:00
|
|
|
string path = logDir;
|
2014-05-18 17:44:10 -07:00
|
|
|
path += "/";
|
|
|
|
path += log;
|
|
|
|
|
2014-08-24 02:02:45 +02:00
|
|
|
BPtr<char> file = os_quick_read_utf8_file(path.c_str());
|
2014-05-18 17:44:10 -07:00
|
|
|
if (!file)
|
|
|
|
blog(LOG_WARNING, "Failed to read log file %s", path.c_str());
|
|
|
|
|
|
|
|
return file;
|
|
|
|
}
|
|
|
|
|
2020-04-07 20:22:05 +10:00
|
|
|
void OBSBasic::UploadLog(const char *subdir, const char *file, const bool crash)
|
2014-05-18 17:44:10 -07:00
|
|
|
{
|
2018-03-16 16:22:37 -07:00
|
|
|
BPtr<char> fileString{ReadLogFile(subdir, file)};
|
2014-05-18 17:44:10 -07:00
|
|
|
|
2014-08-24 02:02:45 +02:00
|
|
|
if (!fileString)
|
2014-05-18 17:44:10 -07:00
|
|
|
return;
|
|
|
|
|
2014-08-24 02:02:45 +02:00
|
|
|
if (!*fileString)
|
2014-05-18 17:44:10 -07:00
|
|
|
return;
|
|
|
|
|
|
|
|
ui->menuLogFiles->setEnabled(false);
|
2020-04-07 20:22:05 +10:00
|
|
|
ui->menuCrashLogs->setEnabled(false);
|
2014-05-18 17:44:10 -07:00
|
|
|
|
2014-08-24 02:02:45 +02:00
|
|
|
stringstream ss;
|
|
|
|
ss << "OBS " << App()->GetVersionString() << " log file uploaded at "
|
2018-03-16 11:19:03 +13:00
|
|
|
<< CurrentDateTimeString() << "\n\n"
|
|
|
|
<< fileString;
|
2014-08-25 21:22:58 +02:00
|
|
|
|
UI: Replace Qt5Network classes with libcurl
The Qt5Network classes seem to only support OpenSSL, and because OpenSSL
isn't available on windows, we would have to distribute it with the
program to get SSL access working. The problem with that is that
OpenSSL is not GPL-compatible, so we cannot distribute OpenSSL with the
program, which means we have to find a better (and preferably superior)
library for accessing remote files that can use the windows SSPI for our
SSL needs, which comes with the operating system.
Fortunately, libcurl is probably the best library out there, and can be
compiled with SSPI instead of OpenSSL, so we're just going to switch to
libcurl instead. Originally I thought it didn't support SSPI, otherwise
I would have implemented it sooner.
As a side note, this will make it so we'll able to get files from the
internet via plugins, which will be quite useful.
2015-05-23 23:39:39 -07:00
|
|
|
if (logUploadThread) {
|
|
|
|
logUploadThread->wait();
|
|
|
|
}
|
2014-08-25 21:22:58 +02:00
|
|
|
|
UI: Replace Qt5Network classes with libcurl
The Qt5Network classes seem to only support OpenSSL, and because OpenSSL
isn't available on windows, we would have to distribute it with the
program to get SSL access working. The problem with that is that
OpenSSL is not GPL-compatible, so we cannot distribute OpenSSL with the
program, which means we have to find a better (and preferably superior)
library for accessing remote files that can use the windows SSPI for our
SSL needs, which comes with the operating system.
Fortunately, libcurl is probably the best library out there, and can be
compiled with SSPI instead of OpenSSL, so we're just going to switch to
libcurl instead. Originally I thought it didn't support SSPI, otherwise
I would have implemented it sooner.
As a side note, this will make it so we'll able to get files from the
internet via plugins, which will be quite useful.
2015-05-23 23:39:39 -07:00
|
|
|
RemoteTextThread *thread =
|
2018-05-18 15:33:28 +02:00
|
|
|
new RemoteTextThread("https://obsproject.com/logs/upload",
|
2018-03-16 11:19:03 +13:00
|
|
|
"text/plain", ss.str().c_str());
|
|
|
|
|
2018-07-31 21:11:31 -07:00
|
|
|
logUploadThread.reset(thread);
|
2020-04-07 20:22:05 +10:00
|
|
|
if (crash) {
|
|
|
|
connect(thread, &RemoteTextThread::Result, this,
|
|
|
|
&OBSBasic::crashUploadFinished);
|
|
|
|
} else {
|
|
|
|
connect(thread, &RemoteTextThread::Result, this,
|
|
|
|
&OBSBasic::logUploadFinished);
|
|
|
|
}
|
UI: Replace Qt5Network classes with libcurl
The Qt5Network classes seem to only support OpenSSL, and because OpenSSL
isn't available on windows, we would have to distribute it with the
program to get SSL access working. The problem with that is that
OpenSSL is not GPL-compatible, so we cannot distribute OpenSSL with the
program, which means we have to find a better (and preferably superior)
library for accessing remote files that can use the windows SSPI for our
SSL needs, which comes with the operating system.
Fortunately, libcurl is probably the best library out there, and can be
compiled with SSPI instead of OpenSSL, so we're just going to switch to
libcurl instead. Originally I thought it didn't support SSPI, otherwise
I would have implemented it sooner.
As a side note, this will make it so we'll able to get files from the
internet via plugins, which will be quite useful.
2015-05-23 23:39:39 -07:00
|
|
|
logUploadThread->start();
|
2014-05-18 17:44:10 -07:00
|
|
|
}
|
|
|
|
|
2014-08-22 03:19:19 +02:00
|
|
|
void OBSBasic::on_actionShowLogs_triggered()
|
|
|
|
{
|
2015-01-15 23:44:38 -08:00
|
|
|
char logDir[512];
|
2015-06-01 16:11:57 -07:00
|
|
|
if (GetConfigPath(logDir, sizeof(logDir), "obs-studio/logs") <= 0)
|
2015-01-15 23:44:38 -08:00
|
|
|
return;
|
|
|
|
|
2014-08-22 03:19:19 +02:00
|
|
|
QUrl url = QUrl::fromLocalFile(QT_UTF8(logDir));
|
|
|
|
QDesktopServices::openUrl(url);
|
|
|
|
}
|
|
|
|
|
2014-05-18 17:44:10 -07:00
|
|
|
void OBSBasic::on_actionUploadCurrentLog_triggered()
|
|
|
|
{
|
2020-04-07 20:22:05 +10:00
|
|
|
UploadLog("obs-studio/logs", App()->GetCurrentLog(), false);
|
2014-05-18 17:44:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionUploadLastLog_triggered()
|
|
|
|
{
|
2020-04-07 20:22:05 +10:00
|
|
|
UploadLog("obs-studio/logs", App()->GetLastLog(), false);
|
2014-05-18 17:44:10 -07:00
|
|
|
}
|
|
|
|
|
2015-02-23 23:04:19 -07:00
|
|
|
void OBSBasic::on_actionViewCurrentLog_triggered()
|
|
|
|
{
|
2020-05-15 07:34:54 -05:00
|
|
|
logView->setVisible(!logView->isVisible());
|
2015-02-23 23:04:19 -07:00
|
|
|
}
|
|
|
|
|
2018-03-16 16:27:49 -07:00
|
|
|
void OBSBasic::on_actionShowCrashLogs_triggered()
|
|
|
|
{
|
|
|
|
char logDir[512];
|
|
|
|
if (GetConfigPath(logDir, sizeof(logDir), "obs-studio/crashes") <= 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
QUrl url = QUrl::fromLocalFile(QT_UTF8(logDir));
|
|
|
|
QDesktopServices::openUrl(url);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionUploadLastCrashLog_triggered()
|
|
|
|
{
|
2020-04-07 20:22:05 +10:00
|
|
|
UploadLog("obs-studio/crashes", App()->GetLastCrashLog(), true);
|
2018-03-16 16:27:49 -07:00
|
|
|
}
|
|
|
|
|
2014-07-13 23:56:28 -07:00
|
|
|
void OBSBasic::on_actionCheckForUpdates_triggered()
|
|
|
|
{
|
2017-02-20 04:46:29 -08:00
|
|
|
CheckForUpdates(true);
|
2014-07-13 23:56:28 -07:00
|
|
|
}
|
|
|
|
|
UI: Replace Qt5Network classes with libcurl
The Qt5Network classes seem to only support OpenSSL, and because OpenSSL
isn't available on windows, we would have to distribute it with the
program to get SSL access working. The problem with that is that
OpenSSL is not GPL-compatible, so we cannot distribute OpenSSL with the
program, which means we have to find a better (and preferably superior)
library for accessing remote files that can use the windows SSPI for our
SSL needs, which comes with the operating system.
Fortunately, libcurl is probably the best library out there, and can be
compiled with SSPI instead of OpenSSL, so we're just going to switch to
libcurl instead. Originally I thought it didn't support SSPI, otherwise
I would have implemented it sooner.
As a side note, this will make it so we'll able to get files from the
internet via plugins, which will be quite useful.
2015-05-23 23:39:39 -07:00
|
|
|
void OBSBasic::logUploadFinished(const QString &text, const QString &error)
|
2014-05-18 17:44:10 -07:00
|
|
|
{
|
|
|
|
ui->menuLogFiles->setEnabled(true);
|
2020-04-07 20:22:05 +10:00
|
|
|
ui->menuCrashLogs->setEnabled(true);
|
|
|
|
|
|
|
|
if (text.isEmpty()) {
|
|
|
|
OBSMessageBox::critical(
|
|
|
|
this, QTStr("LogReturnDialog.ErrorUploadingLog"),
|
|
|
|
error);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
openLogDialog(text, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::crashUploadFinished(const QString &text, const QString &error)
|
|
|
|
{
|
|
|
|
ui->menuLogFiles->setEnabled(true);
|
|
|
|
ui->menuCrashLogs->setEnabled(true);
|
2014-05-18 17:44:10 -07:00
|
|
|
|
UI: Replace Qt5Network classes with libcurl
The Qt5Network classes seem to only support OpenSSL, and because OpenSSL
isn't available on windows, we would have to distribute it with the
program to get SSL access working. The problem with that is that
OpenSSL is not GPL-compatible, so we cannot distribute OpenSSL with the
program, which means we have to find a better (and preferably superior)
library for accessing remote files that can use the windows SSPI for our
SSL needs, which comes with the operating system.
Fortunately, libcurl is probably the best library out there, and can be
compiled with SSPI instead of OpenSSL, so we're just going to switch to
libcurl instead. Originally I thought it didn't support SSPI, otherwise
I would have implemented it sooner.
As a side note, this will make it so we'll able to get files from the
internet via plugins, which will be quite useful.
2015-05-23 23:39:39 -07:00
|
|
|
if (text.isEmpty()) {
|
2019-03-30 15:44:44 +01:00
|
|
|
OBSMessageBox::critical(
|
2014-05-18 17:44:10 -07:00
|
|
|
this, QTStr("LogReturnDialog.ErrorUploadingLog"),
|
UI: Replace Qt5Network classes with libcurl
The Qt5Network classes seem to only support OpenSSL, and because OpenSSL
isn't available on windows, we would have to distribute it with the
program to get SSL access working. The problem with that is that
OpenSSL is not GPL-compatible, so we cannot distribute OpenSSL with the
program, which means we have to find a better (and preferably superior)
library for accessing remote files that can use the windows SSPI for our
SSL needs, which comes with the operating system.
Fortunately, libcurl is probably the best library out there, and can be
compiled with SSPI instead of OpenSSL, so we're just going to switch to
libcurl instead. Originally I thought it didn't support SSPI, otherwise
I would have implemented it sooner.
As a side note, this will make it so we'll able to get files from the
internet via plugins, which will be quite useful.
2015-05-23 23:39:39 -07:00
|
|
|
error);
|
2014-05-18 17:44:10 -07:00
|
|
|
return;
|
|
|
|
}
|
2020-04-07 20:22:05 +10:00
|
|
|
openLogDialog(text, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::openLogDialog(const QString &text, const bool crash)
|
|
|
|
{
|
2014-05-18 17:44:10 -07:00
|
|
|
|
UI: Replace Qt5Network classes with libcurl
The Qt5Network classes seem to only support OpenSSL, and because OpenSSL
isn't available on windows, we would have to distribute it with the
program to get SSL access working. The problem with that is that
OpenSSL is not GPL-compatible, so we cannot distribute OpenSSL with the
program, which means we have to find a better (and preferably superior)
library for accessing remote files that can use the windows SSPI for our
SSL needs, which comes with the operating system.
Fortunately, libcurl is probably the best library out there, and can be
compiled with SSPI instead of OpenSSL, so we're just going to switch to
libcurl instead. Originally I thought it didn't support SSPI, otherwise
I would have implemented it sooner.
As a side note, this will make it so we'll able to get files from the
internet via plugins, which will be quite useful.
2015-05-23 23:39:39 -07:00
|
|
|
obs_data_t *returnData = obs_data_create_from_json(QT_TO_UTF8(text));
|
2018-05-18 15:33:28 +02:00
|
|
|
string resURL = obs_data_get_string(returnData, "url");
|
2018-03-16 11:19:03 +13:00
|
|
|
QString logURL = resURL.c_str();
|
2014-05-18 17:44:10 -07:00
|
|
|
obs_data_release(returnData);
|
|
|
|
|
2020-04-07 20:22:05 +10:00
|
|
|
OBSLogReply logDialog(this, logURL, crash);
|
2014-05-18 17:44:10 -07:00
|
|
|
logDialog.exec();
|
|
|
|
}
|
|
|
|
|
2014-08-25 10:10:58 -07:00
|
|
|
static void RenameListItem(OBSBasic *parent, QListWidget *listWidget,
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_source_t *source, const string &name)
|
2014-06-30 01:05:33 -07:00
|
|
|
{
|
2014-10-13 18:09:06 +02:00
|
|
|
const char *prevName = obs_source_get_name(source);
|
|
|
|
if (name == prevName)
|
|
|
|
return;
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_source_t *foundSource = obs_get_source_by_name(name.c_str());
|
2014-10-13 18:09:06 +02:00
|
|
|
QListWidgetItem *listItem = listWidget->currentItem();
|
2014-06-30 01:05:33 -07:00
|
|
|
|
2014-10-13 18:09:06 +02:00
|
|
|
if (foundSource || name.empty()) {
|
2014-06-30 01:05:33 -07:00
|
|
|
listItem->setText(QT_UTF8(prevName));
|
2014-08-12 15:09:19 -04:00
|
|
|
|
2014-10-13 18:09:06 +02:00
|
|
|
if (foundSource) {
|
2019-03-30 15:44:44 +01:00
|
|
|
OBSMessageBox::warning(parent,
|
2014-08-12 15:09:19 -04:00
|
|
|
QTStr("NameExists.Title"),
|
|
|
|
QTStr("NameExists.Text"));
|
|
|
|
} else if (name.empty()) {
|
2019-03-30 15:44:44 +01:00
|
|
|
OBSMessageBox::warning(parent,
|
2014-08-12 15:09:19 -04:00
|
|
|
QTStr("NoNameEntered.Title"),
|
|
|
|
QTStr("NoNameEntered.Text"));
|
|
|
|
}
|
|
|
|
|
2014-06-30 01:05:33 -07:00
|
|
|
obs_source_release(foundSource);
|
|
|
|
} else {
|
|
|
|
listItem->setText(QT_UTF8(name.c_str()));
|
2014-08-04 08:41:15 -07:00
|
|
|
obs_source_set_name(source, name.c_str());
|
2014-06-30 01:05:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-30 00:06:01 -07:00
|
|
|
void OBSBasic::SceneNameEdited(QWidget *editor,
|
|
|
|
QAbstractItemDelegate::EndEditHint endHint)
|
|
|
|
{
|
|
|
|
OBSScene scene = GetCurrentScene();
|
|
|
|
QLineEdit *edit = qobject_cast<QLineEdit *>(editor);
|
2014-06-30 01:05:33 -07:00
|
|
|
string text = QT_TO_UTF8(edit->text().trimmed());
|
2014-06-30 00:06:01 -07:00
|
|
|
|
|
|
|
if (!scene)
|
|
|
|
return;
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_source_t *source = obs_scene_get_source(scene);
|
2014-08-12 15:09:19 -04:00
|
|
|
RenameListItem(this, ui->scenes, source, text);
|
2014-06-30 00:06:01 -07:00
|
|
|
|
2016-08-28 14:24:14 -07:00
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_SCENE_LIST_CHANGED);
|
|
|
|
|
2014-06-30 00:06:01 -07:00
|
|
|
UNUSED_PARAMETER(endHint);
|
|
|
|
}
|
|
|
|
|
2015-02-25 21:23:57 -08:00
|
|
|
void OBSBasic::OpenFilters()
|
|
|
|
{
|
|
|
|
OBSSceneItem item = GetCurrentSceneItem();
|
|
|
|
OBSSource source = obs_sceneitem_get_source(item);
|
|
|
|
|
|
|
|
CreateFiltersWindow(source);
|
|
|
|
}
|
|
|
|
|
2015-03-18 15:09:44 -07:00
|
|
|
void OBSBasic::OpenSceneFilters()
|
|
|
|
{
|
|
|
|
OBSScene scene = GetCurrentScene();
|
|
|
|
OBSSource source = obs_scene_get_source(scene);
|
|
|
|
|
|
|
|
CreateFiltersWindow(source);
|
|
|
|
}
|
|
|
|
|
2015-06-30 07:58:29 -07:00
|
|
|
#define RECORDING_START \
|
|
|
|
"==== Recording Start ==============================================="
|
|
|
|
#define RECORDING_STOP \
|
|
|
|
"==== Recording Stop ================================================"
|
2016-12-09 14:40:04 -08:00
|
|
|
#define REPLAY_BUFFER_START \
|
|
|
|
"==== Replay Buffer Start ==========================================="
|
|
|
|
#define REPLAY_BUFFER_STOP \
|
|
|
|
"==== Replay Buffer Stop ============================================"
|
2015-06-30 07:58:29 -07:00
|
|
|
#define STREAMING_START \
|
|
|
|
"==== Streaming Start ==============================================="
|
|
|
|
#define STREAMING_STOP \
|
|
|
|
"==== Streaming Stop ================================================"
|
2020-06-20 07:42:14 -07:00
|
|
|
#define VIRTUAL_CAM_START \
|
|
|
|
"==== Virtual Camera Start =========================================="
|
|
|
|
#define VIRTUAL_CAM_STOP \
|
|
|
|
"==== Virtual Camera Stop ==========================================="
|
2015-06-30 07:58:29 -07:00
|
|
|
|
2014-10-26 02:03:13 +02:00
|
|
|
void OBSBasic::StartStreaming()
|
|
|
|
{
|
2016-08-28 14:24:14 -07:00
|
|
|
if (outputHandler->StreamingActive())
|
|
|
|
return;
|
2017-05-15 14:15:25 -07:00
|
|
|
if (disableOutputsRef)
|
2017-04-28 03:58:53 -07:00
|
|
|
return;
|
2016-08-28 14:24:14 -07:00
|
|
|
|
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_STREAMING_STARTING);
|
|
|
|
|
2014-10-26 02:03:13 +02:00
|
|
|
SaveProject();
|
|
|
|
|
2015-09-06 16:19:53 -07:00
|
|
|
ui->streamButton->setEnabled(false);
|
2020-07-04 19:27:48 +10:00
|
|
|
ui->streamButton->setChecked(false);
|
2015-09-06 16:19:53 -07:00
|
|
|
ui->streamButton->setText(QTStr("Basic.Main.Connecting"));
|
2016-10-19 05:29:34 -07:00
|
|
|
|
|
|
|
if (sysTrayStream) {
|
|
|
|
sysTrayStream->setEnabled(false);
|
|
|
|
sysTrayStream->setText(ui->streamButton->text());
|
|
|
|
}
|
2014-10-26 02:03:13 +02:00
|
|
|
|
2015-09-06 16:19:53 -07:00
|
|
|
if (!outputHandler->StartStreaming(service)) {
|
2018-12-12 15:20:48 -08:00
|
|
|
QString message =
|
|
|
|
!outputHandler->lastError.empty()
|
|
|
|
? QTStr(outputHandler->lastError.c_str())
|
|
|
|
: QTStr("Output.StartFailedGeneric");
|
2015-09-06 16:19:53 -07:00
|
|
|
ui->streamButton->setText(QTStr("Basic.Main.StartStreaming"));
|
|
|
|
ui->streamButton->setEnabled(true);
|
2017-10-06 14:51:03 -04:00
|
|
|
ui->streamButton->setChecked(false);
|
2016-10-19 05:29:34 -07:00
|
|
|
|
|
|
|
if (sysTrayStream) {
|
|
|
|
sysTrayStream->setText(ui->streamButton->text());
|
|
|
|
sysTrayStream->setEnabled(true);
|
|
|
|
}
|
2017-02-04 05:29:09 +01:00
|
|
|
|
2018-12-12 15:20:48 -08:00
|
|
|
QMessageBox::critical(this, QTStr("Output.StartStreamFailed"),
|
|
|
|
message);
|
2017-02-04 05:29:09 +01:00
|
|
|
return;
|
2014-10-26 02:03:13 +02:00
|
|
|
}
|
2016-06-16 12:59:36 -05:00
|
|
|
|
|
|
|
bool recordWhenStreaming = config_get_bool(
|
|
|
|
GetGlobalConfig(), "BasicWindow", "RecordWhenStreaming");
|
|
|
|
if (recordWhenStreaming)
|
|
|
|
StartRecording();
|
2017-01-04 16:11:52 -05:00
|
|
|
|
|
|
|
bool replayBufferWhileStreaming = config_get_bool(
|
|
|
|
GetGlobalConfig(), "BasicWindow", "ReplayBufferWhileStreaming");
|
|
|
|
if (replayBufferWhileStreaming)
|
|
|
|
StartReplayBuffer();
|
2014-10-26 02:03:13 +02:00
|
|
|
}
|
|
|
|
|
2016-07-01 10:27:27 -07:00
|
|
|
#ifdef _WIN32
|
|
|
|
static inline void UpdateProcessPriority()
|
|
|
|
{
|
|
|
|
const char *priority = config_get_string(App()->GlobalConfig(),
|
|
|
|
"General", "ProcessPriority");
|
|
|
|
if (priority && strcmp(priority, "Normal") != 0)
|
|
|
|
SetProcessPriority(priority);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void ClearProcessPriority()
|
|
|
|
{
|
|
|
|
const char *priority = config_get_string(App()->GlobalConfig(),
|
|
|
|
"General", "ProcessPriority");
|
|
|
|
if (priority && strcmp(priority, "Normal") != 0)
|
|
|
|
SetProcessPriority("Normal");
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
#define UpdateProcessPriority() \
|
|
|
|
do { \
|
|
|
|
} while (false)
|
|
|
|
#define ClearProcessPriority() \
|
|
|
|
do { \
|
|
|
|
} while (false)
|
|
|
|
#endif
|
|
|
|
|
2016-08-21 12:05:25 -07:00
|
|
|
inline void OBSBasic::OnActivate()
|
2014-10-26 02:03:13 +02:00
|
|
|
{
|
2016-08-21 12:05:25 -07:00
|
|
|
if (ui->profileMenu->isEnabled()) {
|
|
|
|
ui->profileMenu->setEnabled(false);
|
2017-05-06 12:13:56 -07:00
|
|
|
ui->autoConfigure->setEnabled(false);
|
2016-08-21 12:05:25 -07:00
|
|
|
App()->IncrementSleepInhibition();
|
|
|
|
UpdateProcessPriority();
|
2016-08-13 09:36:17 -05:00
|
|
|
|
2016-10-19 06:44:02 -07:00
|
|
|
if (trayIcon)
|
2019-03-11 10:52:01 -05:00
|
|
|
trayIcon->setIcon(QIcon::fromTheme(
|
|
|
|
"obs-tray-active",
|
|
|
|
QIcon(":/res/images/tray_active.png")));
|
2016-08-21 12:05:25 -07:00
|
|
|
}
|
|
|
|
}
|
2015-06-30 07:58:29 -07:00
|
|
|
|
2019-11-22 13:37:51 -05:00
|
|
|
extern volatile bool recording_paused;
|
|
|
|
extern volatile bool replaybuf_active;
|
|
|
|
|
2016-08-21 12:05:25 -07:00
|
|
|
inline void OBSBasic::OnDeactivate()
|
|
|
|
{
|
2015-11-16 09:08:55 -08:00
|
|
|
if (!outputHandler->Active() && !ui->profileMenu->isEnabled()) {
|
2015-06-23 19:38:01 -07:00
|
|
|
ui->profileMenu->setEnabled(true);
|
2017-05-06 12:13:56 -07:00
|
|
|
ui->autoConfigure->setEnabled(true);
|
2015-11-16 09:08:55 -08:00
|
|
|
App()->DecrementSleepInhibition();
|
2016-07-01 10:27:27 -07:00
|
|
|
ClearProcessPriority();
|
2016-08-13 09:36:17 -05:00
|
|
|
|
2016-10-19 06:44:02 -07:00
|
|
|
if (trayIcon)
|
2019-03-11 10:52:01 -05:00
|
|
|
trayIcon->setIcon(QIcon::fromTheme(
|
|
|
|
"obs-tray", QIcon(":/res/images/obs.png")));
|
2020-07-11 06:12:51 -07:00
|
|
|
} else if (trayIcon) {
|
|
|
|
if (os_atomic_load_bool(&recording_paused))
|
|
|
|
trayIcon->setIcon(QIcon(":/res/images/obs_paused.png"));
|
|
|
|
else
|
|
|
|
trayIcon->setIcon(
|
|
|
|
QIcon(":/res/images/tray_active.png"));
|
2015-06-30 07:58:29 -07:00
|
|
|
}
|
2016-08-21 12:05:25 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::StopStreaming()
|
|
|
|
{
|
|
|
|
SaveProject();
|
|
|
|
|
|
|
|
if (outputHandler->StreamingActive())
|
2016-09-09 07:37:54 -07:00
|
|
|
outputHandler->StopStreaming(streamingStopping);
|
2016-08-21 12:05:25 -07:00
|
|
|
|
|
|
|
OnDeactivate();
|
2016-06-16 12:59:36 -05:00
|
|
|
|
|
|
|
bool recordWhenStreaming = config_get_bool(
|
|
|
|
GetGlobalConfig(), "BasicWindow", "RecordWhenStreaming");
|
|
|
|
bool keepRecordingWhenStreamStops =
|
|
|
|
config_get_bool(GetGlobalConfig(), "BasicWindow",
|
|
|
|
"KeepRecordingWhenStreamStops");
|
|
|
|
if (recordWhenStreaming && !keepRecordingWhenStreamStops)
|
|
|
|
StopRecording();
|
2017-01-04 16:11:52 -05:00
|
|
|
|
|
|
|
bool replayBufferWhileStreaming = config_get_bool(
|
|
|
|
GetGlobalConfig(), "BasicWindow", "ReplayBufferWhileStreaming");
|
|
|
|
bool keepReplayBufferStreamStops =
|
|
|
|
config_get_bool(GetGlobalConfig(), "BasicWindow",
|
|
|
|
"KeepReplayBufferStreamStops");
|
|
|
|
if (replayBufferWhileStreaming && !keepReplayBufferStreamStops)
|
|
|
|
StopReplayBuffer();
|
2014-10-26 02:03:13 +02:00
|
|
|
}
|
|
|
|
|
2015-09-06 16:19:53 -07:00
|
|
|
void OBSBasic::ForceStopStreaming()
|
|
|
|
{
|
|
|
|
SaveProject();
|
|
|
|
|
|
|
|
if (outputHandler->StreamingActive())
|
2016-09-09 07:37:54 -07:00
|
|
|
outputHandler->StopStreaming(true);
|
2015-09-06 16:19:53 -07:00
|
|
|
|
2016-08-21 12:05:25 -07:00
|
|
|
OnDeactivate();
|
2016-06-16 12:59:36 -05:00
|
|
|
|
|
|
|
bool recordWhenStreaming = config_get_bool(
|
|
|
|
GetGlobalConfig(), "BasicWindow", "RecordWhenStreaming");
|
|
|
|
bool keepRecordingWhenStreamStops =
|
|
|
|
config_get_bool(GetGlobalConfig(), "BasicWindow",
|
|
|
|
"KeepRecordingWhenStreamStops");
|
|
|
|
if (recordWhenStreaming && !keepRecordingWhenStreamStops)
|
|
|
|
StopRecording();
|
2017-01-04 16:11:52 -05:00
|
|
|
|
|
|
|
bool replayBufferWhileStreaming = config_get_bool(
|
|
|
|
GetGlobalConfig(), "BasicWindow", "ReplayBufferWhileStreaming");
|
|
|
|
bool keepReplayBufferStreamStops =
|
|
|
|
config_get_bool(GetGlobalConfig(), "BasicWindow",
|
|
|
|
"KeepReplayBufferStreamStops");
|
|
|
|
if (replayBufferWhileStreaming && !keepReplayBufferStreamStops)
|
|
|
|
StopReplayBuffer();
|
2015-09-06 16:19:53 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::StreamDelayStarting(int sec)
|
|
|
|
{
|
|
|
|
ui->streamButton->setText(QTStr("Basic.Main.StopStreaming"));
|
|
|
|
ui->streamButton->setEnabled(true);
|
2017-10-06 14:51:03 -04:00
|
|
|
ui->streamButton->setChecked(true);
|
2016-10-19 05:29:34 -07:00
|
|
|
|
|
|
|
if (sysTrayStream) {
|
|
|
|
sysTrayStream->setText(ui->streamButton->text());
|
|
|
|
sysTrayStream->setEnabled(true);
|
|
|
|
}
|
2015-09-06 16:19:53 -07:00
|
|
|
|
2020-04-19 09:31:18 -05:00
|
|
|
if (!startStreamMenu.isNull())
|
|
|
|
startStreamMenu->deleteLater();
|
|
|
|
|
|
|
|
startStreamMenu = new QMenu();
|
2015-09-06 16:19:53 -07:00
|
|
|
startStreamMenu->addAction(QTStr("Basic.Main.StopStreaming"), this,
|
|
|
|
SLOT(StopStreaming()));
|
|
|
|
startStreamMenu->addAction(QTStr("Basic.Main.ForceStopStreaming"), this,
|
|
|
|
SLOT(ForceStopStreaming()));
|
|
|
|
ui->streamButton->setMenu(startStreamMenu);
|
|
|
|
|
|
|
|
ui->statusbar->StreamDelayStarting(sec);
|
2015-11-16 09:08:55 -08:00
|
|
|
|
2016-08-21 12:05:25 -07:00
|
|
|
OnActivate();
|
2015-09-06 16:19:53 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::StreamDelayStopping(int sec)
|
|
|
|
{
|
|
|
|
ui->streamButton->setText(QTStr("Basic.Main.StartStreaming"));
|
|
|
|
ui->streamButton->setEnabled(true);
|
2017-10-06 14:51:03 -04:00
|
|
|
ui->streamButton->setChecked(false);
|
2016-10-19 05:29:34 -07:00
|
|
|
|
|
|
|
if (sysTrayStream) {
|
|
|
|
sysTrayStream->setText(ui->streamButton->text());
|
|
|
|
sysTrayStream->setEnabled(true);
|
|
|
|
}
|
2015-09-06 16:19:53 -07:00
|
|
|
|
2020-04-19 09:31:18 -05:00
|
|
|
if (!startStreamMenu.isNull())
|
|
|
|
startStreamMenu->deleteLater();
|
|
|
|
|
|
|
|
startStreamMenu = new QMenu();
|
2015-09-06 16:19:53 -07:00
|
|
|
startStreamMenu->addAction(QTStr("Basic.Main.StartStreaming"), this,
|
|
|
|
SLOT(StartStreaming()));
|
|
|
|
startStreamMenu->addAction(QTStr("Basic.Main.ForceStopStreaming"), this,
|
|
|
|
SLOT(ForceStopStreaming()));
|
|
|
|
ui->streamButton->setMenu(startStreamMenu);
|
|
|
|
|
|
|
|
ui->statusbar->StreamDelayStopping(sec);
|
2018-10-17 18:24:26 -07:00
|
|
|
|
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_STREAMING_STOPPING);
|
2015-09-06 16:19:53 -07:00
|
|
|
}
|
|
|
|
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
void OBSBasic::StreamingStart()
|
Implement RTMP module (still needs drop code)
- Implement the RTMP output module. This time around, we just use a
simple FLV muxer, then just write to the stream with RTMP_Write.
Easy and effective.
- Fix the FLV muxer, the muxer now outputs proper FLV packets.
- Output API:
* When using encoders, automatically interleave encoded packets
before sending it to the output.
* Pair encoders and have them automatically wait for the other to
start to ensure sync.
* Change 'obs_output_signal_start_fail' to 'obs_output_signal_stop'
because it was a bit confusing, and doing this makes a lot more
sense for outputs that need to stop suddenly (disconnections/etc).
- Encoder API:
* Remove some unnecessary encoder functions from the actual API and
make them internal. Most of the encoder functions are handled
automatically by outputs anyway, so there's no real need to expose
them and end up inadvertently confusing plugin writers.
* Have audio encoders wait for the video encoder to get a frame, then
start at the exact data point that the first video frame starts to
ensure the most accrate sync of video/audio possible.
* Add a required 'frame_size' callback for audio encoders that
returns the expected number of frames desired to encode with. This
way, the libobs encoder API can handle the circular buffering
internally automatically for the encoder modules, so encoder
writers don't have to do it themselves.
- Fix a few bugs in the serializer interface. It was passing the wrong
variable for the data in a few cases.
- If a source has video, make obs_source_update defer the actual update
callback until the tick function is called to prevent threading
issues.
2014-04-07 22:00:10 -07:00
|
|
|
{
|
2014-07-12 05:34:23 -07:00
|
|
|
ui->streamButton->setText(QTStr("Basic.Main.StopStreaming"));
|
2014-05-15 14:04:18 -07:00
|
|
|
ui->streamButton->setEnabled(true);
|
2017-10-06 14:51:03 -04:00
|
|
|
ui->streamButton->setChecked(true);
|
2015-02-06 03:17:33 -08:00
|
|
|
ui->statusbar->StreamStarted(outputHandler->streamOutput);
|
2016-10-19 05:29:34 -07:00
|
|
|
|
|
|
|
if (sysTrayStream) {
|
|
|
|
sysTrayStream->setText(ui->streamButton->text());
|
|
|
|
sysTrayStream->setEnabled(true);
|
|
|
|
}
|
2015-11-16 09:08:55 -08:00
|
|
|
|
2016-08-28 14:24:14 -07:00
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_STREAMING_STARTED);
|
|
|
|
|
2016-08-21 12:05:25 -07:00
|
|
|
OnActivate();
|
2015-11-16 09:08:55 -08:00
|
|
|
|
2015-07-05 23:48:29 -07:00
|
|
|
blog(LOG_INFO, STREAMING_START);
|
2014-03-10 13:10:35 -07:00
|
|
|
}
|
|
|
|
|
2016-06-22 01:47:08 -07:00
|
|
|
void OBSBasic::StreamStopping()
|
|
|
|
{
|
|
|
|
ui->streamButton->setText(QTStr("Basic.Main.StoppingStreaming"));
|
2016-10-19 05:29:34 -07:00
|
|
|
|
|
|
|
if (sysTrayStream)
|
|
|
|
sysTrayStream->setText(ui->streamButton->text());
|
2016-08-28 14:24:14 -07:00
|
|
|
|
2016-09-09 07:37:54 -07:00
|
|
|
streamingStopping = true;
|
2016-08-28 14:24:14 -07:00
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_STREAMING_STOPPING);
|
2016-06-22 01:47:08 -07:00
|
|
|
}
|
|
|
|
|
2017-05-15 12:03:00 +02:00
|
|
|
void OBSBasic::StreamingStop(int code, QString last_error)
|
2014-03-10 13:10:35 -07:00
|
|
|
{
|
2019-05-17 01:19:36 -07:00
|
|
|
const char *errorDescription = "";
|
2017-05-15 12:03:00 +02:00
|
|
|
DStr errorMessage;
|
|
|
|
bool use_last_error = false;
|
2019-05-17 01:19:36 -07:00
|
|
|
bool encode_error = false;
|
2014-05-12 15:30:36 -07:00
|
|
|
|
|
|
|
switch (code) {
|
|
|
|
case OBS_OUTPUT_BAD_PATH:
|
2017-05-15 12:03:00 +02:00
|
|
|
errorDescription = Str("Output.ConnectFail.BadPath");
|
2014-05-12 15:30:36 -07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case OBS_OUTPUT_CONNECT_FAILED:
|
2017-05-15 12:03:00 +02:00
|
|
|
use_last_error = true;
|
|
|
|
errorDescription = Str("Output.ConnectFail.ConnectFailed");
|
2014-05-12 15:30:36 -07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case OBS_OUTPUT_INVALID_STREAM:
|
2017-05-15 12:03:00 +02:00
|
|
|
errorDescription = Str("Output.ConnectFail.InvalidStream");
|
2014-05-12 15:30:36 -07:00
|
|
|
break;
|
|
|
|
|
2019-05-17 01:19:36 -07:00
|
|
|
case OBS_OUTPUT_ENCODE_ERROR:
|
|
|
|
encode_error = true;
|
|
|
|
break;
|
|
|
|
|
2014-06-25 02:03:00 -07:00
|
|
|
default:
|
2014-05-12 15:30:36 -07:00
|
|
|
case OBS_OUTPUT_ERROR:
|
2017-05-15 12:03:00 +02:00
|
|
|
use_last_error = true;
|
|
|
|
errorDescription = Str("Output.ConnectFail.Error");
|
2014-05-12 15:30:36 -07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case OBS_OUTPUT_DISCONNECTED:
|
|
|
|
/* doesn't happen if output is set to reconnect. note that
|
|
|
|
* reconnects are handled in the output, not in the UI */
|
2017-05-15 12:03:00 +02:00
|
|
|
use_last_error = true;
|
|
|
|
errorDescription = Str("Output.ConnectFail.Disconnected");
|
2014-05-12 15:30:36 -07:00
|
|
|
}
|
|
|
|
|
2017-05-15 12:03:00 +02:00
|
|
|
if (use_last_error && !last_error.isEmpty())
|
|
|
|
dstr_printf(errorMessage, "%s\n\n%s", errorDescription,
|
|
|
|
QT_TO_UTF8(last_error));
|
|
|
|
else
|
|
|
|
dstr_copy(errorMessage, errorDescription);
|
|
|
|
|
2014-07-06 16:19:27 -07:00
|
|
|
ui->statusbar->StreamStopped();
|
2014-05-20 23:27:27 -07:00
|
|
|
|
2014-05-12 15:30:36 -07:00
|
|
|
ui->streamButton->setText(QTStr("Basic.Main.StartStreaming"));
|
2014-05-15 14:04:18 -07:00
|
|
|
ui->streamButton->setEnabled(true);
|
2017-10-06 14:51:03 -04:00
|
|
|
ui->streamButton->setChecked(false);
|
2016-10-19 05:29:34 -07:00
|
|
|
|
|
|
|
if (sysTrayStream) {
|
|
|
|
sysTrayStream->setText(ui->streamButton->text());
|
|
|
|
sysTrayStream->setEnabled(true);
|
|
|
|
}
|
2014-05-12 15:30:36 -07:00
|
|
|
|
2016-09-09 07:37:54 -07:00
|
|
|
streamingStopping = false;
|
2016-08-28 14:24:14 -07:00
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_STREAMING_STOPPED);
|
|
|
|
|
2016-08-21 12:05:25 -07:00
|
|
|
OnDeactivate();
|
2015-07-05 23:48:29 -07:00
|
|
|
|
|
|
|
blog(LOG_INFO, STREAMING_STOP);
|
2015-06-30 07:58:29 -07:00
|
|
|
|
2019-05-17 01:19:36 -07:00
|
|
|
if (encode_error) {
|
|
|
|
OBSMessageBox::information(
|
|
|
|
this, QTStr("Output.StreamEncodeError.Title"),
|
|
|
|
QTStr("Output.StreamEncodeError.Msg"));
|
|
|
|
|
|
|
|
} else if (code != OBS_OUTPUT_SUCCESS && isVisible()) {
|
2017-05-13 14:06:32 -07:00
|
|
|
OBSMessageBox::information(this,
|
2014-05-12 15:30:36 -07:00
|
|
|
QTStr("Output.ConnectFail.Title"),
|
|
|
|
QT_UTF8(errorMessage));
|
2019-05-17 01:19:36 -07:00
|
|
|
|
2016-08-13 09:36:17 -05:00
|
|
|
} else if (code != OBS_OUTPUT_SUCCESS && !isVisible()) {
|
2017-05-15 12:03:00 +02:00
|
|
|
SysTrayNotify(QT_UTF8(errorDescription),
|
|
|
|
QSystemTrayIcon::Warning);
|
2016-08-13 09:36:17 -05:00
|
|
|
}
|
2015-09-06 16:19:53 -07:00
|
|
|
|
2020-04-19 09:31:18 -05:00
|
|
|
if (!startStreamMenu.isNull()) {
|
|
|
|
ui->streamButton->setMenu(nullptr);
|
|
|
|
startStreamMenu->deleteLater();
|
|
|
|
startStreamMenu = nullptr;
|
|
|
|
}
|
2014-04-14 02:22:09 -07:00
|
|
|
}
|
|
|
|
|
2017-05-08 06:53:35 -05:00
|
|
|
void OBSBasic::AutoRemux()
|
|
|
|
{
|
|
|
|
const char *mode = config_get_string(basicConfig, "Output", "Mode");
|
2019-02-19 16:51:06 -08:00
|
|
|
bool advanced = astrcmpi(mode, "Advanced") == 0;
|
|
|
|
|
2019-02-19 16:07:58 -08:00
|
|
|
const char *path = !advanced ? config_get_string(basicConfig,
|
|
|
|
"SimpleOutput",
|
|
|
|
"FilePath")
|
|
|
|
: config_get_string(basicConfig, "AdvOut",
|
|
|
|
"RecFilePath");
|
|
|
|
|
2019-02-19 16:51:06 -08:00
|
|
|
/* do not save if using FFmpeg output in advanced output mode */
|
|
|
|
if (advanced) {
|
|
|
|
const char *type =
|
|
|
|
config_get_string(basicConfig, "AdvOut", "RecType");
|
|
|
|
if (astrcmpi(type, "FFmpeg") == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-19 16:32:51 -08:00
|
|
|
QString input;
|
|
|
|
input += path;
|
|
|
|
input += "/";
|
|
|
|
input += remuxFilename.c_str();
|
|
|
|
|
|
|
|
QFileInfo fi(remuxFilename.c_str());
|
|
|
|
|
2019-02-19 16:51:06 -08:00
|
|
|
/* do not remux if lossless */
|
|
|
|
if (fi.suffix().compare("avi", Qt::CaseInsensitive) == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-02-19 16:32:51 -08:00
|
|
|
QString output;
|
|
|
|
output += path;
|
|
|
|
output += "/";
|
|
|
|
output += fi.completeBaseName();
|
|
|
|
output += ".mp4";
|
2017-05-08 06:53:35 -05:00
|
|
|
|
|
|
|
OBSRemux *remux = new OBSRemux(path, this, true);
|
|
|
|
remux->show();
|
2019-02-19 16:32:51 -08:00
|
|
|
remux->AutoRemux(input, output);
|
2017-05-08 06:53:35 -05:00
|
|
|
}
|
|
|
|
|
2014-10-26 01:57:29 +02:00
|
|
|
void OBSBasic::StartRecording()
|
|
|
|
{
|
2016-06-16 12:59:36 -05:00
|
|
|
if (outputHandler->RecordingActive())
|
|
|
|
return;
|
2017-05-15 14:15:25 -07:00
|
|
|
if (disableOutputsRef)
|
2017-04-28 03:58:53 -07:00
|
|
|
return;
|
2014-10-26 01:57:29 +02:00
|
|
|
|
2019-12-27 17:12:31 +08:00
|
|
|
if (!OutputPathValid()) {
|
|
|
|
OutputPathInvalidMessage();
|
|
|
|
ui->recordButton->setChecked(false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-06-12 04:56:00 -05:00
|
|
|
if (LowDiskSpace()) {
|
|
|
|
DiskSpaceMessage();
|
|
|
|
ui->recordButton->setChecked(false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-08-28 14:24:14 -07:00
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_RECORDING_STARTING);
|
|
|
|
|
2016-06-16 12:59:36 -05:00
|
|
|
SaveProject();
|
2018-07-31 21:35:06 -07:00
|
|
|
|
|
|
|
if (!outputHandler->StartRecording())
|
|
|
|
ui->recordButton->setChecked(false);
|
2014-10-26 01:57:29 +02:00
|
|
|
}
|
|
|
|
|
2016-06-22 01:47:08 -07:00
|
|
|
void OBSBasic::RecordStopping()
|
|
|
|
{
|
2016-12-09 14:40:04 -08:00
|
|
|
ui->recordButton->setText(QTStr("Basic.Main.StoppingRecording"));
|
2016-10-19 05:29:34 -07:00
|
|
|
|
|
|
|
if (sysTrayRecord)
|
|
|
|
sysTrayRecord->setText(ui->recordButton->text());
|
2016-08-28 14:24:14 -07:00
|
|
|
|
2016-09-09 07:37:54 -07:00
|
|
|
recordingStopping = true;
|
2016-08-28 14:24:14 -07:00
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_RECORDING_STOPPING);
|
2016-06-22 01:47:08 -07:00
|
|
|
}
|
|
|
|
|
2014-10-26 01:57:29 +02:00
|
|
|
void OBSBasic::StopRecording()
|
|
|
|
{
|
|
|
|
SaveProject();
|
|
|
|
|
|
|
|
if (outputHandler->RecordingActive())
|
2016-09-09 07:37:54 -07:00
|
|
|
outputHandler->StopRecording(recordingStopping);
|
2015-06-30 07:58:29 -07:00
|
|
|
|
2016-08-21 12:05:25 -07:00
|
|
|
OnDeactivate();
|
2014-10-26 01:57:29 +02:00
|
|
|
}
|
|
|
|
|
2014-08-25 03:10:57 +02:00
|
|
|
void OBSBasic::RecordingStart()
|
|
|
|
{
|
2015-02-06 03:17:33 -08:00
|
|
|
ui->statusbar->RecordingStarted(outputHandler->fileOutput);
|
2016-12-09 14:40:04 -08:00
|
|
|
ui->recordButton->setText(QTStr("Basic.Main.StopRecording"));
|
2017-10-06 14:51:03 -04:00
|
|
|
ui->recordButton->setChecked(true);
|
2016-10-19 05:29:34 -07:00
|
|
|
|
|
|
|
if (sysTrayRecord)
|
|
|
|
sysTrayRecord->setText(ui->recordButton->text());
|
2015-11-16 09:08:55 -08:00
|
|
|
|
2016-09-09 07:37:54 -07:00
|
|
|
recordingStopping = false;
|
2016-08-28 14:24:14 -07:00
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_RECORDING_STARTED);
|
|
|
|
|
2019-06-12 04:56:00 -05:00
|
|
|
if (!diskFullTimer->isActive())
|
|
|
|
diskFullTimer->start(1000);
|
|
|
|
|
2016-08-21 12:05:25 -07:00
|
|
|
OnActivate();
|
2019-07-07 15:47:29 -07:00
|
|
|
UpdatePause();
|
2015-11-16 09:08:55 -08:00
|
|
|
|
2015-07-05 23:48:29 -07:00
|
|
|
blog(LOG_INFO, RECORDING_START);
|
2014-08-25 03:10:57 +02:00
|
|
|
}
|
|
|
|
|
2019-03-30 15:44:44 +01:00
|
|
|
void OBSBasic::RecordingStop(int code, QString last_error)
|
2014-03-10 13:10:35 -07:00
|
|
|
{
|
2014-08-25 03:10:57 +02:00
|
|
|
ui->statusbar->RecordingStopped();
|
2016-12-09 14:40:04 -08:00
|
|
|
ui->recordButton->setText(QTStr("Basic.Main.StartRecording"));
|
2017-10-06 14:51:03 -04:00
|
|
|
ui->recordButton->setChecked(false);
|
2016-10-19 05:29:34 -07:00
|
|
|
|
|
|
|
if (sysTrayRecord)
|
|
|
|
sysTrayRecord->setText(ui->recordButton->text());
|
2016-12-09 14:40:04 -08:00
|
|
|
|
2015-07-05 23:48:29 -07:00
|
|
|
blog(LOG_INFO, RECORDING_STOP);
|
2015-06-09 21:00:22 -07:00
|
|
|
|
2016-08-13 09:36:17 -05:00
|
|
|
if (code == OBS_OUTPUT_UNSUPPORTED && isVisible()) {
|
2015-06-09 21:00:22 -07:00
|
|
|
OBSMessageBox::critical(this, QTStr("Output.RecordFail.Title"),
|
|
|
|
QTStr("Output.RecordFail.Unsupported"));
|
2015-06-30 07:58:29 -07:00
|
|
|
|
2019-05-17 01:19:36 -07:00
|
|
|
} else if (code == OBS_OUTPUT_ENCODE_ERROR && isVisible()) {
|
|
|
|
OBSMessageBox::warning(
|
|
|
|
this, QTStr("Output.RecordError.Title"),
|
|
|
|
QTStr("Output.RecordError.EncodeErrorMsg"));
|
|
|
|
|
2016-08-13 09:36:17 -05:00
|
|
|
} else if (code == OBS_OUTPUT_NO_SPACE && isVisible()) {
|
2019-03-30 15:44:44 +01:00
|
|
|
OBSMessageBox::warning(this,
|
2015-09-18 22:20:55 -07:00
|
|
|
QTStr("Output.RecordNoSpace.Title"),
|
|
|
|
QTStr("Output.RecordNoSpace.Msg"));
|
|
|
|
|
2016-08-13 09:36:17 -05:00
|
|
|
} else if (code != OBS_OUTPUT_SUCCESS && isVisible()) {
|
2019-03-30 15:44:44 +01:00
|
|
|
|
|
|
|
const char *errorDescription;
|
|
|
|
DStr errorMessage;
|
|
|
|
bool use_last_error = true;
|
|
|
|
|
|
|
|
errorDescription = Str("Output.RecordError.Msg");
|
|
|
|
|
|
|
|
if (use_last_error && !last_error.isEmpty())
|
|
|
|
dstr_printf(errorMessage, "%s\n\n%s", errorDescription,
|
|
|
|
QT_TO_UTF8(last_error));
|
|
|
|
else
|
|
|
|
dstr_copy(errorMessage, errorDescription);
|
|
|
|
|
2015-09-18 22:20:55 -07:00
|
|
|
OBSMessageBox::critical(this, QTStr("Output.RecordError.Title"),
|
2019-03-30 15:44:44 +01:00
|
|
|
QT_UTF8(errorMessage));
|
2016-08-13 09:36:17 -05:00
|
|
|
|
|
|
|
} else if (code == OBS_OUTPUT_UNSUPPORTED && !isVisible()) {
|
|
|
|
SysTrayNotify(QTStr("Output.RecordFail.Unsupported"),
|
|
|
|
QSystemTrayIcon::Warning);
|
|
|
|
|
|
|
|
} else if (code == OBS_OUTPUT_NO_SPACE && !isVisible()) {
|
|
|
|
SysTrayNotify(QTStr("Output.RecordNoSpace.Msg"),
|
|
|
|
QSystemTrayIcon::Warning);
|
|
|
|
|
|
|
|
} else if (code != OBS_OUTPUT_SUCCESS && !isVisible()) {
|
|
|
|
SysTrayNotify(QTStr("Output.RecordError.Msg"),
|
|
|
|
QSystemTrayIcon::Warning);
|
2015-09-18 22:20:55 -07:00
|
|
|
}
|
|
|
|
|
2016-08-28 14:24:14 -07:00
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_RECORDING_STOPPED);
|
|
|
|
|
2019-06-12 04:56:00 -05:00
|
|
|
if (diskFullTimer->isActive())
|
|
|
|
diskFullTimer->stop();
|
|
|
|
|
2017-05-08 06:53:35 -05:00
|
|
|
if (remuxAfterRecord)
|
|
|
|
AutoRemux();
|
|
|
|
|
2016-08-21 12:05:25 -07:00
|
|
|
OnDeactivate();
|
2019-07-07 15:47:29 -07:00
|
|
|
UpdatePause(false);
|
2014-05-20 23:27:27 -07:00
|
|
|
}
|
obs-studio UI: Implement stream settings UI
- Updated the services API so that it links up with an output and
the output gets data from that service rather than via settings.
This allows the service context to have control over how an output is
used, and makes it so that the URL/key/etc isn't necessarily some
static setting.
Also, if the service is attached to an output, it will stick around
until the output is destroyed.
- The settings interface has been updated so that it can allow the
usage of service plugins. What this means is that now you can create
a service plugin that can control aspects of the stream, and it
allows each service to create their own user interface if they create
a service plugin module.
- Testing out saving of current service information. Saves/loads from
JSON in to obs_data_t, seems to be working quite nicely, and the
service object information is saved/preserved on exit, and loaded
again on startup.
- I agonized over the settings user interface for days, and eventually
I just decided that the only way that users weren't going to be
fumbling over options was to split up the settings in to simple/basic
output, pre-configured, and then advanced for advanced use (such as
multiple outputs or services, which I'll implement later).
This was particularly painful to really design right, I wanted more
features and wanted to include everything in one interface but
ultimately just realized from experience that users are just not
technically knowledgable about it and will end up fumbling with the
settings rather than getting things done.
Basically, what this means is that casual users only have to enter in
about 3 things to configure their stream: Stream key, audio bitrate,
and video bitrate. I am really happy with this interface for those
types of users, but it definitely won't be sufficient for advanced
usage or for custom outputs, so that stuff will have to be separated.
- Improved the JSON usage for the 'common streaming services' context,
I realized that JSON arrays are there to ensure sorting, while
forgetting that general items are optimized for hashing. So
basically I'm just using arrays now to sort items in it.
2014-04-24 01:49:07 -07:00
|
|
|
|
2019-07-07 15:47:29 -07:00
|
|
|
void OBSBasic::ShowReplayBufferPauseWarning()
|
|
|
|
{
|
|
|
|
auto msgBox = []() {
|
|
|
|
QMessageBox msgbox(App()->GetMainWindow());
|
|
|
|
msgbox.setWindowTitle(QTStr("Output.ReplayBuffer."
|
|
|
|
"PauseWarning.Title"));
|
|
|
|
msgbox.setText(QTStr("Output.ReplayBuffer."
|
|
|
|
"PauseWarning.Text"));
|
|
|
|
msgbox.setIcon(QMessageBox::Icon::Information);
|
|
|
|
msgbox.addButton(QMessageBox::Ok);
|
|
|
|
|
|
|
|
QCheckBox *cb = new QCheckBox(QTStr("DoNotShowAgain"));
|
|
|
|
msgbox.setCheckBox(cb);
|
|
|
|
|
|
|
|
msgbox.exec();
|
|
|
|
|
|
|
|
if (cb->isChecked()) {
|
|
|
|
config_set_bool(App()->GlobalConfig(), "General",
|
|
|
|
"WarnedAboutReplayBufferPausing", true);
|
|
|
|
config_save_safe(App()->GlobalConfig(), "tmp", nullptr);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
bool warned = config_get_bool(App()->GlobalConfig(), "General",
|
|
|
|
"WarnedAboutReplayBufferPausing");
|
|
|
|
if (!warned) {
|
|
|
|
QMetaObject::invokeMethod(App(), "Exec", Qt::QueuedConnection,
|
|
|
|
Q_ARG(VoidFunc, msgBox));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-09 14:40:04 -08:00
|
|
|
void OBSBasic::StartReplayBuffer()
|
|
|
|
{
|
|
|
|
if (!outputHandler || !outputHandler->replayBuffer)
|
|
|
|
return;
|
|
|
|
if (outputHandler->ReplayBufferActive())
|
|
|
|
return;
|
2017-05-15 14:15:25 -07:00
|
|
|
if (disableOutputsRef)
|
2017-04-28 03:58:53 -07:00
|
|
|
return;
|
2016-12-09 14:40:04 -08:00
|
|
|
|
2019-11-19 16:57:49 -08:00
|
|
|
if (!UIValidation::NoSourcesConfirmation(this)) {
|
2018-06-26 19:37:21 -05:00
|
|
|
replayBufferButton->setChecked(false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-06-12 04:56:00 -05:00
|
|
|
if (LowDiskSpace()) {
|
|
|
|
DiskSpaceMessage();
|
|
|
|
replayBufferButton->setChecked(false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-12-09 14:40:04 -08:00
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_REPLAY_BUFFER_STARTING);
|
|
|
|
|
|
|
|
SaveProject();
|
2019-07-07 15:47:29 -07:00
|
|
|
|
|
|
|
if (!outputHandler->StartReplayBuffer()) {
|
2018-07-31 21:35:06 -07:00
|
|
|
replayBufferButton->setChecked(false);
|
2019-07-07 15:47:29 -07:00
|
|
|
} else if (os_atomic_load_bool(&recording_paused)) {
|
|
|
|
ShowReplayBufferPauseWarning();
|
|
|
|
}
|
2016-12-09 14:40:04 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::ReplayBufferStopping()
|
|
|
|
{
|
|
|
|
if (!outputHandler || !outputHandler->replayBuffer)
|
|
|
|
return;
|
|
|
|
|
|
|
|
replayBufferButton->setText(QTStr("Basic.Main.StoppingReplayBuffer"));
|
|
|
|
|
|
|
|
if (sysTrayReplayBuffer)
|
|
|
|
sysTrayReplayBuffer->setText(replayBufferButton->text());
|
|
|
|
|
|
|
|
replayBufferStopping = true;
|
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_REPLAY_BUFFER_STOPPING);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::StopReplayBuffer()
|
|
|
|
{
|
|
|
|
if (!outputHandler || !outputHandler->replayBuffer)
|
|
|
|
return;
|
|
|
|
|
|
|
|
SaveProject();
|
|
|
|
|
|
|
|
if (outputHandler->ReplayBufferActive())
|
|
|
|
outputHandler->StopReplayBuffer(replayBufferStopping);
|
|
|
|
|
|
|
|
OnDeactivate();
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::ReplayBufferStart()
|
|
|
|
{
|
|
|
|
if (!outputHandler || !outputHandler->replayBuffer)
|
|
|
|
return;
|
|
|
|
|
|
|
|
replayBufferButton->setText(QTStr("Basic.Main.StopReplayBuffer"));
|
2019-02-19 19:27:54 -08:00
|
|
|
replayBufferButton->setChecked(true);
|
2016-12-09 14:40:04 -08:00
|
|
|
|
|
|
|
if (sysTrayReplayBuffer)
|
|
|
|
sysTrayReplayBuffer->setText(replayBufferButton->text());
|
|
|
|
|
|
|
|
replayBufferStopping = false;
|
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_REPLAY_BUFFER_STARTED);
|
|
|
|
|
|
|
|
OnActivate();
|
2019-12-12 17:07:26 -08:00
|
|
|
UpdateReplayBuffer();
|
2016-12-09 14:40:04 -08:00
|
|
|
|
|
|
|
blog(LOG_INFO, REPLAY_BUFFER_START);
|
|
|
|
}
|
|
|
|
|
2017-10-19 07:14:14 -04:00
|
|
|
void OBSBasic::ReplayBufferSave()
|
|
|
|
{
|
|
|
|
if (!outputHandler || !outputHandler->replayBuffer)
|
|
|
|
return;
|
|
|
|
if (!outputHandler->ReplayBufferActive())
|
|
|
|
return;
|
|
|
|
|
|
|
|
calldata_t cd = {0};
|
|
|
|
proc_handler_t *ph =
|
|
|
|
obs_output_get_proc_handler(outputHandler->replayBuffer);
|
|
|
|
proc_handler_call(ph, "save", &cd);
|
|
|
|
calldata_free(&cd);
|
|
|
|
}
|
|
|
|
|
2016-12-09 14:40:04 -08:00
|
|
|
void OBSBasic::ReplayBufferStop(int code)
|
|
|
|
{
|
|
|
|
if (!outputHandler || !outputHandler->replayBuffer)
|
|
|
|
return;
|
|
|
|
|
|
|
|
replayBufferButton->setText(QTStr("Basic.Main.StartReplayBuffer"));
|
2019-02-20 18:09:39 +02:00
|
|
|
replayBufferButton->setChecked(false);
|
2016-12-09 14:40:04 -08:00
|
|
|
|
|
|
|
if (sysTrayReplayBuffer)
|
|
|
|
sysTrayReplayBuffer->setText(replayBufferButton->text());
|
|
|
|
|
|
|
|
blog(LOG_INFO, REPLAY_BUFFER_STOP);
|
|
|
|
|
|
|
|
if (code == OBS_OUTPUT_UNSUPPORTED && isVisible()) {
|
|
|
|
OBSMessageBox::critical(this, QTStr("Output.RecordFail.Title"),
|
|
|
|
QTStr("Output.RecordFail.Unsupported"));
|
|
|
|
|
|
|
|
} else if (code == OBS_OUTPUT_NO_SPACE && isVisible()) {
|
2019-03-30 15:44:44 +01:00
|
|
|
OBSMessageBox::warning(this,
|
2016-12-09 14:40:04 -08:00
|
|
|
QTStr("Output.RecordNoSpace.Title"),
|
|
|
|
QTStr("Output.RecordNoSpace.Msg"));
|
|
|
|
|
|
|
|
} else if (code != OBS_OUTPUT_SUCCESS && isVisible()) {
|
|
|
|
OBSMessageBox::critical(this, QTStr("Output.RecordError.Title"),
|
|
|
|
QTStr("Output.RecordError.Msg"));
|
|
|
|
|
|
|
|
} else if (code == OBS_OUTPUT_UNSUPPORTED && !isVisible()) {
|
|
|
|
SysTrayNotify(QTStr("Output.RecordFail.Unsupported"),
|
|
|
|
QSystemTrayIcon::Warning);
|
|
|
|
|
|
|
|
} else if (code == OBS_OUTPUT_NO_SPACE && !isVisible()) {
|
|
|
|
SysTrayNotify(QTStr("Output.RecordNoSpace.Msg"),
|
|
|
|
QSystemTrayIcon::Warning);
|
|
|
|
|
|
|
|
} else if (code != OBS_OUTPUT_SUCCESS && !isVisible()) {
|
|
|
|
SysTrayNotify(QTStr("Output.RecordError.Msg"),
|
|
|
|
QSystemTrayIcon::Warning);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_REPLAY_BUFFER_STOPPED);
|
|
|
|
|
|
|
|
OnDeactivate();
|
2019-12-12 17:07:26 -08:00
|
|
|
UpdateReplayBuffer(false);
|
2016-12-09 14:40:04 -08:00
|
|
|
}
|
|
|
|
|
2020-06-20 07:42:14 -07:00
|
|
|
void OBSBasic::StartVirtualCam()
|
|
|
|
{
|
|
|
|
if (!outputHandler || !outputHandler->virtualCam)
|
|
|
|
return;
|
|
|
|
if (outputHandler->VirtualCamActive())
|
|
|
|
return;
|
|
|
|
if (disableOutputsRef)
|
|
|
|
return;
|
|
|
|
|
|
|
|
SaveProject();
|
|
|
|
|
|
|
|
if (!outputHandler->StartVirtualCam()) {
|
|
|
|
vcamButton->setChecked(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::StopVirtualCam()
|
|
|
|
{
|
|
|
|
if (!outputHandler || !outputHandler->virtualCam)
|
|
|
|
return;
|
|
|
|
|
|
|
|
SaveProject();
|
|
|
|
|
|
|
|
if (outputHandler->VirtualCamActive())
|
|
|
|
outputHandler->StopVirtualCam();
|
|
|
|
|
|
|
|
OnDeactivate();
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::OnVirtualCamStart()
|
|
|
|
{
|
|
|
|
if (!outputHandler || !outputHandler->virtualCam)
|
|
|
|
return;
|
|
|
|
|
|
|
|
vcamButton->setText(QTStr("Basic.Main.StopVirtualCam"));
|
|
|
|
vcamButton->setChecked(true);
|
|
|
|
|
|
|
|
OnActivate();
|
|
|
|
|
|
|
|
blog(LOG_INFO, VIRTUAL_CAM_START);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::OnVirtualCamStop(int)
|
|
|
|
{
|
|
|
|
if (!outputHandler || !outputHandler->virtualCam)
|
|
|
|
return;
|
|
|
|
|
|
|
|
vcamButton->setText(QTStr("Basic.Main.StartVirtualCam"));
|
|
|
|
vcamButton->setChecked(false);
|
|
|
|
|
|
|
|
blog(LOG_INFO, VIRTUAL_CAM_STOP);
|
|
|
|
|
|
|
|
OnDeactivate();
|
|
|
|
}
|
|
|
|
|
2014-05-20 23:27:27 -07:00
|
|
|
void OBSBasic::on_streamButton_clicked()
|
|
|
|
{
|
2015-02-06 03:17:33 -08:00
|
|
|
if (outputHandler->StreamingActive()) {
|
2016-01-22 08:09:04 -08:00
|
|
|
bool confirm = config_get_bool(GetGlobalConfig(), "BasicWindow",
|
|
|
|
"WarnBeforeStoppingStream");
|
|
|
|
|
2016-08-13 09:36:17 -05:00
|
|
|
if (confirm && isVisible()) {
|
2016-01-22 08:09:04 -08:00
|
|
|
QMessageBox::StandardButton button =
|
2017-05-13 14:06:32 -07:00
|
|
|
OBSMessageBox::question(
|
2016-01-22 08:09:04 -08:00
|
|
|
this, QTStr("ConfirmStop.Title"),
|
2019-11-27 12:30:15 +01:00
|
|
|
QTStr("ConfirmStop.Text"),
|
|
|
|
QMessageBox::Yes | QMessageBox::No,
|
|
|
|
QMessageBox::No);
|
2016-01-22 08:09:04 -08:00
|
|
|
|
2018-06-27 23:01:25 -05:00
|
|
|
if (button == QMessageBox::No) {
|
|
|
|
ui->streamButton->setChecked(true);
|
2016-01-22 08:09:04 -08:00
|
|
|
return;
|
2018-06-27 23:01:25 -05:00
|
|
|
}
|
2016-01-22 08:09:04 -08:00
|
|
|
}
|
|
|
|
|
2014-10-26 02:03:13 +02:00
|
|
|
StopStreaming();
|
2014-05-20 23:27:27 -07:00
|
|
|
} else {
|
2019-11-19 16:57:49 -08:00
|
|
|
if (!UIValidation::NoSourcesConfirmation(this)) {
|
2018-06-26 19:37:21 -05:00
|
|
|
ui->streamButton->setChecked(false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-11-26 14:55:46 -08:00
|
|
|
auto action =
|
|
|
|
UIValidation::StreamSettingsConfirmation(this, service);
|
|
|
|
switch (action) {
|
|
|
|
case StreamSettingsAction::ContinueStream:
|
|
|
|
break;
|
|
|
|
case StreamSettingsAction::OpenSettings:
|
|
|
|
on_action_Settings_triggered();
|
|
|
|
ui->streamButton->setChecked(false);
|
|
|
|
return;
|
|
|
|
case StreamSettingsAction::Cancel:
|
|
|
|
ui->streamButton->setChecked(false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-01-22 08:09:04 -08:00
|
|
|
bool confirm = config_get_bool(GetGlobalConfig(), "BasicWindow",
|
|
|
|
"WarnBeforeStartingStream");
|
|
|
|
|
2019-03-09 06:38:52 +01:00
|
|
|
obs_data_t *settings = obs_service_get_settings(service);
|
|
|
|
bool bwtest = obs_data_get_bool(settings, "bwtest");
|
|
|
|
obs_data_release(settings);
|
|
|
|
|
|
|
|
if (bwtest && isVisible()) {
|
|
|
|
QMessageBox::StandardButton button =
|
|
|
|
OBSMessageBox::question(
|
|
|
|
this, QTStr("ConfirmBWTest.Title"),
|
|
|
|
QTStr("ConfirmBWTest.Text"));
|
|
|
|
|
|
|
|
if (button == QMessageBox::No) {
|
|
|
|
ui->streamButton->setChecked(false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else if (confirm && isVisible()) {
|
2016-01-22 08:09:04 -08:00
|
|
|
QMessageBox::StandardButton button =
|
2017-05-13 14:06:32 -07:00
|
|
|
OBSMessageBox::question(
|
2016-01-22 08:09:04 -08:00
|
|
|
this, QTStr("ConfirmStart.Title"),
|
2019-11-27 12:30:15 +01:00
|
|
|
QTStr("ConfirmStart.Text"),
|
|
|
|
QMessageBox::Yes | QMessageBox::No,
|
|
|
|
QMessageBox::No);
|
2016-01-22 08:09:04 -08:00
|
|
|
|
2018-06-27 23:01:25 -05:00
|
|
|
if (button == QMessageBox::No) {
|
|
|
|
ui->streamButton->setChecked(false);
|
2016-01-22 08:09:04 -08:00
|
|
|
return;
|
2018-06-27 23:01:25 -05:00
|
|
|
}
|
2016-01-22 08:09:04 -08:00
|
|
|
}
|
|
|
|
|
2014-10-26 02:03:13 +02:00
|
|
|
StartStreaming();
|
2014-05-20 23:27:27 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_recordButton_clicked()
|
|
|
|
{
|
2018-06-26 19:37:21 -05:00
|
|
|
if (outputHandler->RecordingActive()) {
|
2019-06-17 15:14:14 +01:00
|
|
|
bool confirm = config_get_bool(GetGlobalConfig(), "BasicWindow",
|
|
|
|
"WarnBeforeStoppingRecord");
|
|
|
|
|
|
|
|
if (confirm && isVisible()) {
|
|
|
|
QMessageBox::StandardButton button =
|
|
|
|
OBSMessageBox::question(
|
|
|
|
this, QTStr("ConfirmStopRecord.Title"),
|
2019-11-27 12:30:15 +01:00
|
|
|
QTStr("ConfirmStopRecord.Text"),
|
|
|
|
QMessageBox::Yes | QMessageBox::No,
|
|
|
|
QMessageBox::No);
|
2019-06-17 15:14:14 +01:00
|
|
|
|
|
|
|
if (button == QMessageBox::No) {
|
|
|
|
ui->recordButton->setChecked(true);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2014-10-26 01:57:29 +02:00
|
|
|
StopRecording();
|
2018-06-26 19:37:21 -05:00
|
|
|
} else {
|
2019-11-19 16:57:49 -08:00
|
|
|
if (!UIValidation::NoSourcesConfirmation(this)) {
|
2018-06-26 19:37:21 -05:00
|
|
|
ui->recordButton->setChecked(false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-10-26 01:57:29 +02:00
|
|
|
StartRecording();
|
2018-06-26 19:37:21 -05:00
|
|
|
}
|
2014-02-10 10:22:35 -07:00
|
|
|
}
|
|
|
|
|
2020-06-20 07:42:14 -07:00
|
|
|
void OBSBasic::VCamButtonClicked()
|
|
|
|
{
|
|
|
|
if (outputHandler->VirtualCamActive()) {
|
|
|
|
StopVirtualCam();
|
|
|
|
} else {
|
|
|
|
if (!UIValidation::NoSourcesConfirmation(this)) {
|
|
|
|
vcamButton->setChecked(false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
StartVirtualCam();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Change the UI to Qt (work in progress)
--------------------------------------------------
Notes and details
--------------------------------------------------
Why was this done? Because wxWidgets was just lacking in many areas. I
know wxWidgets is designed to be used with native controls, and that's
great, but wxWidgets just is not a feature-complete toolkit for
multiplatform applications. It lacks in dialog editors, its code is
archaic and outdated, and I just feel frustrated every time I try to do
things with it.
Qt on the other hand.. I had to actually try Qt to realize how much
better it was as a toolkit. They've got everything from dialog editors,
to an IDE, a debugger, build tools, just everything, and it's all
top-notch and highly maintained. The focus of the toolkit is
application development, and they spend their time trying to help
people do exactly that: make programs. Great support, great tools,
and because of that, great toolkit. I just didn't want to alienate any
developers by being stubborn about native widgets.
There *are* some things that are rather lackluster about it and design
choices I disagree with though. For example, I realize that to have an
easy to use toolkit you have to have some level of code generation.
However, in my personal and humble opinion, moc just feels like a
terrible way to approach the problem. Even now I feel like there are a
variety of ways you could handle code generation and automatic
management of things like that. I don't like the idea of circumventing
the language itself like that. It feels like one giant massive hack.
--------------------------------------------------
Things that aren't working properly:
--------------------------------------------------
- Settings dialog is not implemented. The dialog is complete but the
code to handle the dialog hasn't been constructed yet.
- There is a problem with using Qt widgets as a device target on
windows, with at least OpenGL: if I have the preview widget
automatically resize itself, it seems to cause some sort of video
card failure that I don't understand.
- Because of the above, resizing the preview widget has been disabled
until I can figure out what's going on, so it's currently only a
32x32 area.
- Direct3D doesn't seem to render correctly either, seems that the
viewport is messed up or something. I'm sort of confused about
what's going on with it.
- The new main window seems to be triggering more race conditions than
the wxWidgets main window dialog did. I'm not entirely sure what's
going on here, but this may just be existing race conditions within
libobs itself that I just never spotted before (even though I tend to
be very thorough with race conditions any time I use variables
cross-thread)
2014-01-23 11:53:55 -07:00
|
|
|
void OBSBasic::on_settingsButton_clicked()
|
2013-12-10 11:22:33 -07:00
|
|
|
{
|
2016-09-21 21:20:06 -07:00
|
|
|
on_action_Settings_triggered();
|
2013-12-10 11:22:33 -07:00
|
|
|
}
|
2014-03-06 21:08:12 -07:00
|
|
|
|
2018-01-05 15:20:39 -06:00
|
|
|
void OBSBasic::on_actionHelpPortal_triggered()
|
|
|
|
{
|
|
|
|
QUrl url = QUrl("https://obsproject.com/help", QUrl::TolerantMode);
|
|
|
|
QDesktopServices::openUrl(url);
|
|
|
|
}
|
|
|
|
|
2015-06-25 01:02:47 -07:00
|
|
|
void OBSBasic::on_actionWebsite_triggered()
|
|
|
|
{
|
|
|
|
QUrl url = QUrl("https://obsproject.com", QUrl::TolerantMode);
|
|
|
|
QDesktopServices::openUrl(url);
|
|
|
|
}
|
|
|
|
|
2018-08-01 18:41:57 -07:00
|
|
|
void OBSBasic::on_actionDiscord_triggered()
|
|
|
|
{
|
2018-08-03 00:33:06 -07:00
|
|
|
QUrl url = QUrl("https://obsproject.com/discord", QUrl::TolerantMode);
|
2018-08-01 18:41:57 -07:00
|
|
|
QDesktopServices::openUrl(url);
|
|
|
|
}
|
|
|
|
|
2015-07-02 20:41:19 -07:00
|
|
|
void OBSBasic::on_actionShowSettingsFolder_triggered()
|
|
|
|
{
|
|
|
|
char path[512];
|
|
|
|
int ret = GetConfigPath(path, 512, "obs-studio");
|
|
|
|
if (ret <= 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
QDesktopServices::openUrl(QUrl::fromLocalFile(path));
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionShowProfileFolder_triggered()
|
|
|
|
{
|
|
|
|
char path[512];
|
|
|
|
int ret = GetProfilePath(path, 512, "");
|
|
|
|
if (ret <= 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
QDesktopServices::openUrl(QUrl::fromLocalFile(path));
|
|
|
|
}
|
|
|
|
|
2018-06-02 09:45:01 -07:00
|
|
|
int OBSBasic::GetTopSelectedSourceItem()
|
2016-01-23 11:02:22 -08:00
|
|
|
{
|
2018-06-02 09:45:01 -07:00
|
|
|
QModelIndexList selectedItems =
|
|
|
|
ui->sources->selectionModel()->selectedIndexes();
|
|
|
|
return selectedItems.count() ? selectedItems[0].row() : -1;
|
2016-01-23 11:02:22 -08:00
|
|
|
}
|
|
|
|
|
2019-10-04 22:33:06 -07:00
|
|
|
QModelIndexList OBSBasic::GetAllSelectedSourceItems()
|
|
|
|
{
|
|
|
|
return ui->sources->selectionModel()->selectedIndexes();
|
|
|
|
}
|
|
|
|
|
2015-04-02 21:35:46 -07:00
|
|
|
void OBSBasic::on_preview_customContextMenuRequested(const QPoint &pos)
|
|
|
|
{
|
2016-01-23 11:02:22 -08:00
|
|
|
CreateSourcePopupMenu(GetTopSelectedSourceItem(), true);
|
2015-04-02 21:35:46 -07:00
|
|
|
|
|
|
|
UNUSED_PARAMETER(pos);
|
|
|
|
}
|
|
|
|
|
2017-11-08 22:38:44 -05:00
|
|
|
void OBSBasic::on_program_customContextMenuRequested(const QPoint &)
|
|
|
|
{
|
|
|
|
QMenu popup(this);
|
|
|
|
QPointer<QMenu> studioProgramProjector;
|
|
|
|
|
|
|
|
studioProgramProjector = new QMenu(QTStr("StudioProgramProjector"));
|
|
|
|
AddProjectorMenuMonitors(studioProgramProjector, this,
|
2017-11-30 06:17:07 -08:00
|
|
|
SLOT(OpenStudioProgramProjector()));
|
2017-11-08 22:38:44 -05:00
|
|
|
|
|
|
|
popup.addMenu(studioProgramProjector);
|
|
|
|
|
|
|
|
QAction *studioProgramWindow =
|
|
|
|
popup.addAction(QTStr("StudioProgramWindow"), this,
|
2017-11-30 06:17:07 -08:00
|
|
|
SLOT(OpenStudioProgramWindow()));
|
2017-11-08 22:38:44 -05:00
|
|
|
|
|
|
|
popup.addAction(studioProgramWindow);
|
|
|
|
|
2020-07-23 08:53:21 -05:00
|
|
|
popup.addAction(QTStr("Screenshot.StudioProgram"), this,
|
|
|
|
SLOT(ScreenshotProgram()));
|
|
|
|
|
2017-11-08 22:38:44 -05:00
|
|
|
popup.exec(QCursor::pos());
|
|
|
|
}
|
|
|
|
|
2019-07-12 11:11:49 -05:00
|
|
|
void OBSBasic::PreviewDisabledMenu(const QPoint &pos)
|
2015-04-02 21:35:46 -07:00
|
|
|
{
|
|
|
|
QMenu popup(this);
|
2020-04-19 09:31:18 -05:00
|
|
|
delete previewProjectorMain;
|
2015-04-02 21:35:46 -07:00
|
|
|
|
|
|
|
QAction *action =
|
|
|
|
popup.addAction(QTStr("Basic.Main.PreviewConextMenu.Enable"),
|
|
|
|
this, SLOT(TogglePreview()));
|
|
|
|
action->setCheckable(true);
|
2015-08-02 00:02:58 -07:00
|
|
|
action->setChecked(obs_display_enabled(ui->preview->GetDisplay()));
|
2015-04-02 21:35:46 -07:00
|
|
|
|
2020-04-19 09:31:18 -05:00
|
|
|
previewProjectorMain = new QMenu(QTStr("PreviewProjector"));
|
2019-01-31 03:33:15 +01:00
|
|
|
AddProjectorMenuMonitors(previewProjectorMain, this,
|
2015-04-04 01:40:15 -07:00
|
|
|
SLOT(OpenPreviewProjector()));
|
|
|
|
|
2017-03-01 22:08:49 -06:00
|
|
|
QAction *previewWindow = popup.addAction(QTStr("PreviewWindow"), this,
|
|
|
|
SLOT(OpenPreviewWindow()));
|
|
|
|
|
2019-01-31 03:33:15 +01:00
|
|
|
popup.addMenu(previewProjectorMain);
|
2017-03-01 22:08:49 -06:00
|
|
|
popup.addAction(previewWindow);
|
2015-04-02 21:35:46 -07:00
|
|
|
popup.exec(QCursor::pos());
|
|
|
|
|
|
|
|
UNUSED_PARAMETER(pos);
|
|
|
|
}
|
|
|
|
|
2016-01-22 07:33:29 -08:00
|
|
|
void OBSBasic::on_actionAlwaysOnTop_triggered()
|
|
|
|
{
|
2018-02-24 20:53:51 -08:00
|
|
|
#ifndef _WIN32
|
2016-01-22 07:33:29 -08:00
|
|
|
/* Make sure all dialogs are safely and successfully closed before
|
|
|
|
* switching the always on top mode due to the fact that windows all
|
|
|
|
* have to be recreated, so queue the actual toggle to happen after
|
|
|
|
* all events related to closing the dialogs have finished */
|
2018-02-24 20:53:51 -08:00
|
|
|
CloseDialogs();
|
|
|
|
#endif
|
|
|
|
|
2016-01-22 07:33:29 -08:00
|
|
|
QMetaObject::invokeMethod(this, "ToggleAlwaysOnTop",
|
|
|
|
Qt::QueuedConnection);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::ToggleAlwaysOnTop()
|
|
|
|
{
|
|
|
|
bool isAlwaysOnTop = IsAlwaysOnTop(this);
|
|
|
|
|
|
|
|
ui->actionAlwaysOnTop->setChecked(!isAlwaysOnTop);
|
|
|
|
SetAlwaysOnTop(this, !isAlwaysOnTop);
|
|
|
|
|
|
|
|
show();
|
|
|
|
}
|
|
|
|
|
2014-03-06 21:08:12 -07:00
|
|
|
void OBSBasic::GetFPSCommon(uint32_t &num, uint32_t &den) const
|
|
|
|
{
|
|
|
|
const char *val = config_get_string(basicConfig, "Video", "FPSCommon");
|
|
|
|
|
|
|
|
if (strcmp(val, "10") == 0) {
|
|
|
|
num = 10;
|
|
|
|
den = 1;
|
|
|
|
} else if (strcmp(val, "20") == 0) {
|
|
|
|
num = 20;
|
|
|
|
den = 1;
|
2017-02-06 14:55:14 -08:00
|
|
|
} else if (strcmp(val, "24 NTSC") == 0) {
|
|
|
|
num = 24000;
|
|
|
|
den = 1001;
|
2019-03-10 08:31:08 +01:00
|
|
|
} else if (strcmp(val, "25 PAL") == 0) {
|
2014-03-06 21:08:12 -07:00
|
|
|
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;
|
2019-03-10 08:31:08 +01:00
|
|
|
} else if (strcmp(val, "50 PAL") == 0) {
|
|
|
|
num = 50;
|
|
|
|
den = 1;
|
2014-03-06 21:08:12 -07:00
|
|
|
} 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 OBSBasic::GetFPSInteger(uint32_t &num, uint32_t &den) const
|
|
|
|
{
|
|
|
|
num = (uint32_t)config_get_uint(basicConfig, "Video", "FPSInt");
|
|
|
|
den = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::GetFPSFraction(uint32_t &num, uint32_t &den) const
|
|
|
|
{
|
|
|
|
num = (uint32_t)config_get_uint(basicConfig, "Video", "FPSNum");
|
|
|
|
den = (uint32_t)config_get_uint(basicConfig, "Video", "FPSDen");
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::GetFPSNanoseconds(uint32_t &num, uint32_t &den) const
|
|
|
|
{
|
|
|
|
num = 1000000000;
|
|
|
|
den = (uint32_t)config_get_uint(basicConfig, "Video", "FPSNS");
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::GetConfigFPS(uint32_t &num, uint32_t &den) const
|
|
|
|
{
|
|
|
|
uint32_t type = config_get_uint(basicConfig, "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);
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
config_t *OBSBasic::Config() const
|
2014-03-06 21:08:12 -07:00
|
|
|
{
|
|
|
|
return basicConfig;
|
|
|
|
}
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
|
|
|
|
void OBSBasic::on_actionEditTransform_triggered()
|
|
|
|
{
|
2014-07-22 22:16:57 +02:00
|
|
|
if (transformWindow)
|
|
|
|
transformWindow->close();
|
|
|
|
|
2020-04-04 23:39:46 +11:00
|
|
|
if (!GetCurrentSceneItem())
|
|
|
|
return;
|
|
|
|
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
transformWindow = new OBSBasicTransform(this);
|
|
|
|
transformWindow->show();
|
2014-07-22 22:16:57 +02:00
|
|
|
transformWindow->setAttribute(Qt::WA_DeleteOnClose, true);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
}
|
|
|
|
|
2016-12-07 02:43:23 -05:00
|
|
|
static obs_transform_info copiedTransformInfo;
|
|
|
|
static obs_sceneitem_crop copiedCropInfo;
|
|
|
|
|
|
|
|
void OBSBasic::on_actionCopyTransform_triggered()
|
|
|
|
{
|
|
|
|
auto func = [](obs_scene_t *scene, obs_sceneitem_t *item, void *param) {
|
|
|
|
if (!obs_sceneitem_selected(item))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
obs_sceneitem_defer_update_begin(item);
|
|
|
|
obs_sceneitem_get_info(item, &copiedTransformInfo);
|
|
|
|
obs_sceneitem_get_crop(item, &copiedCropInfo);
|
|
|
|
obs_sceneitem_defer_update_end(item);
|
|
|
|
|
|
|
|
UNUSED_PARAMETER(scene);
|
|
|
|
UNUSED_PARAMETER(param);
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
obs_scene_enum_items(GetCurrentScene(), func, nullptr);
|
|
|
|
ui->actionPasteTransform->setEnabled(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionPasteTransform_triggered()
|
|
|
|
{
|
|
|
|
auto func = [](obs_scene_t *scene, obs_sceneitem_t *item, void *param) {
|
|
|
|
if (!obs_sceneitem_selected(item))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
obs_sceneitem_defer_update_begin(item);
|
|
|
|
obs_sceneitem_set_info(item, &copiedTransformInfo);
|
|
|
|
obs_sceneitem_set_crop(item, &copiedCropInfo);
|
|
|
|
obs_sceneitem_defer_update_end(item);
|
|
|
|
|
|
|
|
UNUSED_PARAMETER(scene);
|
|
|
|
UNUSED_PARAMETER(param);
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
obs_scene_enum_items(GetCurrentScene(), func, nullptr);
|
|
|
|
}
|
|
|
|
|
2018-08-15 10:01:48 -07:00
|
|
|
static bool reset_tr(obs_scene_t *scene, obs_sceneitem_t *item, void *param)
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
{
|
2018-08-15 10:01:48 -07:00
|
|
|
if (obs_sceneitem_is_group(item))
|
|
|
|
obs_sceneitem_group_enum_items(item, reset_tr, nullptr);
|
|
|
|
if (!obs_sceneitem_selected(item))
|
|
|
|
return true;
|
2020-02-17 13:01:04 -06:00
|
|
|
if (obs_sceneitem_locked(item))
|
|
|
|
return true;
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
|
2018-08-15 10:01:48 -07:00
|
|
|
obs_sceneitem_defer_update_begin(item);
|
2016-03-31 10:15:20 -07:00
|
|
|
|
2018-08-15 10:01:48 -07:00
|
|
|
obs_transform_info info;
|
|
|
|
vec2_set(&info.pos, 0.0f, 0.0f);
|
|
|
|
vec2_set(&info.scale, 1.0f, 1.0f);
|
|
|
|
info.rot = 0.0f;
|
|
|
|
info.alignment = OBS_ALIGN_TOP | OBS_ALIGN_LEFT;
|
|
|
|
info.bounds_type = OBS_BOUNDS_NONE;
|
|
|
|
info.bounds_alignment = OBS_ALIGN_CENTER;
|
|
|
|
vec2_set(&info.bounds, 0.0f, 0.0f);
|
|
|
|
obs_sceneitem_set_info(item, &info);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
|
2018-08-15 10:01:48 -07:00
|
|
|
obs_sceneitem_crop crop = {};
|
|
|
|
obs_sceneitem_set_crop(item, &crop);
|
2016-03-31 10:15:20 -07:00
|
|
|
|
2018-08-15 10:01:48 -07:00
|
|
|
obs_sceneitem_defer_update_end(item);
|
2016-03-31 10:15:20 -07:00
|
|
|
|
2018-08-15 10:01:48 -07:00
|
|
|
UNUSED_PARAMETER(scene);
|
|
|
|
UNUSED_PARAMETER(param);
|
|
|
|
return true;
|
|
|
|
}
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
|
2018-08-15 10:01:48 -07:00
|
|
|
void OBSBasic::on_actionResetTransform_triggered()
|
|
|
|
{
|
|
|
|
obs_scene_enum_items(GetCurrentScene(), reset_tr, nullptr);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
static void GetItemBox(obs_sceneitem_t *item, vec3 &tl, vec3 &br)
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
{
|
|
|
|
matrix4 boxTransform;
|
|
|
|
obs_sceneitem_get_box_transform(item, &boxTransform);
|
|
|
|
|
|
|
|
vec3_set(&tl, M_INFINITE, M_INFINITE, 0.0f);
|
2014-06-22 23:49:57 -07:00
|
|
|
vec3_set(&br, -M_INFINITE, -M_INFINITE, 0.0f);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
|
2014-06-22 23:49:57 -07:00
|
|
|
auto GetMinPos = [&](float x, float y) {
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
vec3 pos;
|
|
|
|
vec3_set(&pos, x, y, 0.0f);
|
|
|
|
vec3_transform(&pos, &pos, &boxTransform);
|
2014-06-22 23:49:57 -07:00
|
|
|
vec3_min(&tl, &tl, &pos);
|
|
|
|
vec3_max(&br, &br, &pos);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
};
|
|
|
|
|
2014-06-22 23:49:57 -07:00
|
|
|
GetMinPos(0.0f, 0.0f);
|
|
|
|
GetMinPos(1.0f, 0.0f);
|
|
|
|
GetMinPos(0.0f, 1.0f);
|
|
|
|
GetMinPos(1.0f, 1.0f);
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
static vec3 GetItemTL(obs_sceneitem_t *item)
|
2014-06-22 23:49:57 -07:00
|
|
|
{
|
|
|
|
vec3 tl, br;
|
|
|
|
GetItemBox(item, tl, br);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
return tl;
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
static void SetItemTL(obs_sceneitem_t *item, const vec3 &tl)
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
{
|
|
|
|
vec3 newTL;
|
|
|
|
vec2 pos;
|
|
|
|
|
2014-08-03 14:39:19 -07:00
|
|
|
obs_sceneitem_get_pos(item, &pos);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
newTL = GetItemTL(item);
|
|
|
|
pos.x += tl.x - newTL.x;
|
|
|
|
pos.y += tl.y - newTL.y;
|
2014-08-03 14:39:19 -07:00
|
|
|
obs_sceneitem_set_pos(item, &pos);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
static bool RotateSelectedSources(obs_scene_t *scene, obs_sceneitem_t *item,
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
void *param)
|
|
|
|
{
|
2018-08-16 08:44:42 -07:00
|
|
|
if (obs_sceneitem_is_group(item))
|
|
|
|
obs_sceneitem_group_enum_items(item, RotateSelectedSources,
|
|
|
|
param);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
if (!obs_sceneitem_selected(item))
|
|
|
|
return true;
|
2020-02-17 13:01:04 -06:00
|
|
|
if (obs_sceneitem_locked(item))
|
|
|
|
return true;
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
|
|
|
|
float rot = *reinterpret_cast<float *>(param);
|
|
|
|
|
|
|
|
vec3 tl = GetItemTL(item);
|
|
|
|
|
2014-08-03 14:39:19 -07:00
|
|
|
rot += obs_sceneitem_get_rot(item);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
if (rot >= 360.0f)
|
|
|
|
rot -= 360.0f;
|
|
|
|
else if (rot <= -360.0f)
|
|
|
|
rot += 360.0f;
|
2014-08-03 14:39:19 -07:00
|
|
|
obs_sceneitem_set_rot(item, rot);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
|
2018-08-16 08:44:42 -07:00
|
|
|
obs_sceneitem_force_update_transform(item);
|
|
|
|
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
SetItemTL(item, tl);
|
|
|
|
|
|
|
|
UNUSED_PARAMETER(scene);
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
void OBSBasic::on_actionRotate90CW_triggered()
|
|
|
|
{
|
|
|
|
float f90CW = 90.0f;
|
|
|
|
obs_scene_enum_items(GetCurrentScene(), RotateSelectedSources, &f90CW);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionRotate90CCW_triggered()
|
|
|
|
{
|
|
|
|
float f90CCW = -90.0f;
|
|
|
|
obs_scene_enum_items(GetCurrentScene(), RotateSelectedSources, &f90CCW);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionRotate180_triggered()
|
|
|
|
{
|
|
|
|
float f180 = 180.0f;
|
|
|
|
obs_scene_enum_items(GetCurrentScene(), RotateSelectedSources, &f180);
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
static bool MultiplySelectedItemScale(obs_scene_t *scene, obs_sceneitem_t *item,
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
void *param)
|
|
|
|
{
|
|
|
|
vec2 &mul = *reinterpret_cast<vec2 *>(param);
|
|
|
|
|
2018-08-16 08:44:42 -07:00
|
|
|
if (obs_sceneitem_is_group(item))
|
|
|
|
obs_sceneitem_group_enum_items(item, MultiplySelectedItemScale,
|
|
|
|
param);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
if (!obs_sceneitem_selected(item))
|
|
|
|
return true;
|
2020-02-17 13:01:04 -06:00
|
|
|
if (obs_sceneitem_locked(item))
|
|
|
|
return true;
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
|
|
|
|
vec3 tl = GetItemTL(item);
|
|
|
|
|
|
|
|
vec2 scale;
|
2014-08-03 14:39:19 -07:00
|
|
|
obs_sceneitem_get_scale(item, &scale);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
vec2_mul(&scale, &scale, &mul);
|
2014-08-03 14:39:19 -07:00
|
|
|
obs_sceneitem_set_scale(item, &scale);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
|
2018-08-16 08:44:42 -07:00
|
|
|
obs_sceneitem_force_update_transform(item);
|
|
|
|
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
SetItemTL(item, tl);
|
2014-06-16 17:55:48 -07:00
|
|
|
|
|
|
|
UNUSED_PARAMETER(scene);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionFlipHorizontal_triggered()
|
|
|
|
{
|
2014-06-16 17:55:48 -07:00
|
|
|
vec2 scale;
|
|
|
|
vec2_set(&scale, -1.0f, 1.0f);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
obs_scene_enum_items(GetCurrentScene(), MultiplySelectedItemScale,
|
|
|
|
&scale);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionFlipVertical_triggered()
|
|
|
|
{
|
2014-06-16 17:55:48 -07:00
|
|
|
vec2 scale;
|
|
|
|
vec2_set(&scale, 1.0f, -1.0f);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
obs_scene_enum_items(GetCurrentScene(), MultiplySelectedItemScale,
|
|
|
|
&scale);
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
static bool CenterAlignSelectedItems(obs_scene_t *scene, obs_sceneitem_t *item,
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
void *param)
|
|
|
|
{
|
|
|
|
obs_bounds_type boundsType =
|
|
|
|
*reinterpret_cast<obs_bounds_type *>(param);
|
|
|
|
|
2018-08-16 12:42:23 -07:00
|
|
|
if (obs_sceneitem_is_group(item))
|
|
|
|
obs_sceneitem_group_enum_items(item, CenterAlignSelectedItems,
|
|
|
|
param);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
if (!obs_sceneitem_selected(item))
|
|
|
|
return true;
|
2020-02-17 13:01:04 -06:00
|
|
|
if (obs_sceneitem_locked(item))
|
|
|
|
return true;
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
|
|
|
|
obs_video_info ovi;
|
|
|
|
obs_get_video_info(&ovi);
|
|
|
|
|
2014-08-01 23:33:45 -07:00
|
|
|
obs_transform_info itemInfo;
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
vec2_set(&itemInfo.pos, 0.0f, 0.0f);
|
|
|
|
vec2_set(&itemInfo.scale, 1.0f, 1.0f);
|
|
|
|
itemInfo.alignment = OBS_ALIGN_LEFT | OBS_ALIGN_TOP;
|
|
|
|
itemInfo.rot = 0.0f;
|
|
|
|
|
|
|
|
vec2_set(&itemInfo.bounds, float(ovi.base_width),
|
|
|
|
float(ovi.base_height));
|
|
|
|
itemInfo.bounds_type = boundsType;
|
|
|
|
itemInfo.bounds_alignment = OBS_ALIGN_CENTER;
|
|
|
|
|
|
|
|
obs_sceneitem_set_info(item, &itemInfo);
|
|
|
|
|
|
|
|
UNUSED_PARAMETER(scene);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionFitToScreen_triggered()
|
|
|
|
{
|
|
|
|
obs_bounds_type boundsType = OBS_BOUNDS_SCALE_INNER;
|
|
|
|
obs_scene_enum_items(GetCurrentScene(), CenterAlignSelectedItems,
|
|
|
|
&boundsType);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionStretchToScreen_triggered()
|
|
|
|
{
|
|
|
|
obs_bounds_type boundsType = OBS_BOUNDS_STRETCH;
|
|
|
|
obs_scene_enum_items(GetCurrentScene(), CenterAlignSelectedItems,
|
|
|
|
&boundsType);
|
|
|
|
}
|
|
|
|
|
2019-03-28 00:00:12 -05:00
|
|
|
enum class CenterType {
|
|
|
|
Scene,
|
|
|
|
Vertical,
|
|
|
|
Horizontal,
|
|
|
|
};
|
|
|
|
|
|
|
|
static bool center_to_scene(obs_scene_t *, obs_sceneitem_t *item, void *param)
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
{
|
2019-03-28 00:00:12 -05:00
|
|
|
CenterType centerType = *reinterpret_cast<CenterType *>(param);
|
|
|
|
|
2018-08-16 12:42:23 -07:00
|
|
|
vec3 tl, br, itemCenter, screenCenter, offset;
|
|
|
|
obs_video_info ovi;
|
2019-03-28 00:00:12 -05:00
|
|
|
obs_transform_info oti;
|
2014-06-22 23:49:57 -07:00
|
|
|
|
2018-08-16 12:42:23 -07:00
|
|
|
if (obs_sceneitem_is_group(item))
|
2019-03-28 00:00:12 -05:00
|
|
|
obs_sceneitem_group_enum_items(item, center_to_scene,
|
|
|
|
¢erType);
|
2018-08-16 12:42:23 -07:00
|
|
|
if (!obs_sceneitem_selected(item))
|
|
|
|
return true;
|
2020-02-17 13:01:04 -06:00
|
|
|
if (obs_sceneitem_locked(item))
|
|
|
|
return true;
|
2014-06-22 23:49:57 -07:00
|
|
|
|
2018-08-16 12:42:23 -07:00
|
|
|
obs_get_video_info(&ovi);
|
2019-03-28 00:00:12 -05:00
|
|
|
obs_sceneitem_get_info(item, &oti);
|
|
|
|
|
|
|
|
if (centerType == CenterType::Scene)
|
|
|
|
vec3_set(&screenCenter, float(ovi.base_width),
|
|
|
|
float(ovi.base_height), 0.0f);
|
|
|
|
else if (centerType == CenterType::Vertical)
|
|
|
|
vec3_set(&screenCenter, float(oti.bounds.x),
|
|
|
|
float(ovi.base_height), 0.0f);
|
|
|
|
else if (centerType == CenterType::Horizontal)
|
|
|
|
vec3_set(&screenCenter, float(ovi.base_width),
|
|
|
|
float(oti.bounds.y), 0.0f);
|
2014-06-22 23:49:57 -07:00
|
|
|
|
2018-08-16 12:42:23 -07:00
|
|
|
vec3_mulf(&screenCenter, &screenCenter, 0.5f);
|
2014-06-22 23:49:57 -07:00
|
|
|
|
2018-08-16 12:42:23 -07:00
|
|
|
GetItemBox(item, tl, br);
|
2014-06-22 23:49:57 -07:00
|
|
|
|
2018-08-16 12:42:23 -07:00
|
|
|
vec3_sub(&itemCenter, &br, &tl);
|
|
|
|
vec3_mulf(&itemCenter, &itemCenter, 0.5f);
|
|
|
|
vec3_add(&itemCenter, &itemCenter, &tl);
|
2014-06-22 23:49:57 -07:00
|
|
|
|
2018-08-16 12:42:23 -07:00
|
|
|
vec3_sub(&offset, &screenCenter, &itemCenter);
|
|
|
|
vec3_add(&tl, &tl, &offset);
|
2014-06-22 23:49:57 -07:00
|
|
|
|
2019-03-28 00:00:12 -05:00
|
|
|
if (centerType == CenterType::Vertical)
|
|
|
|
tl.x = oti.pos.x;
|
|
|
|
else if (centerType == CenterType::Horizontal)
|
|
|
|
tl.y = oti.pos.y;
|
|
|
|
|
2018-08-16 12:42:23 -07:00
|
|
|
SetItemTL(item, tl);
|
|
|
|
return true;
|
|
|
|
};
|
2014-06-22 23:49:57 -07:00
|
|
|
|
2018-08-16 12:42:23 -07:00
|
|
|
void OBSBasic::on_actionCenterToScreen_triggered()
|
|
|
|
{
|
2019-03-28 00:00:12 -05:00
|
|
|
CenterType centerType = CenterType::Scene;
|
|
|
|
obs_scene_enum_items(GetCurrentScene(), center_to_scene, ¢erType);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionVerticalCenter_triggered()
|
|
|
|
{
|
|
|
|
CenterType centerType = CenterType::Vertical;
|
|
|
|
obs_scene_enum_items(GetCurrentScene(), center_to_scene, ¢erType);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionHorizontalCenter_triggered()
|
|
|
|
{
|
|
|
|
CenterType centerType = CenterType::Horizontal;
|
|
|
|
obs_scene_enum_items(GetCurrentScene(), center_to_scene, ¢erType);
|
UI: Add scene editing
So, scene editing was interesting (and by interesting I mean
excruciating). I almost implemented 'manipulator' visuals (ala 3dsmax
for example), and used 3 modes for controlling position/rotation/size,
but in a 2D editing, it felt clunky, so I defaulted back to simply
click-and-drag for movement, and then took a similar though slightly
different looking approach for handling scaling and reszing.
I also added a number of menu item helpers related to positioning,
scaling, rotating, flipping, and resetting the transform back to
default.
There is also a new 'transform' dialog (accessible via menu) which will
allow you to manually edit every single transform variable of a scene
item directly if desired.
If a scene item does not have bounds active, pulling on the sides of a
source will cause it to resize it via base scale rather than by the
bounding box system (if the source resizes that scale will apply). If
bounds are active, it will modify the bounding box only instead.
How a source scales when a bounding box is active depends on the type of
bounds being used. You can set it to scale to the inner bounds, the
outer bounds, scale to bounds width only, scale to bounds height only,
and a setting to stretch to bounds (which forces a source to always draw
at the bounding box size rather than be affected by its internal size).
You can also set it to be used as a 'maximum' size, so that the source
doesn't necessarily get scaled unless it extends beyond the bounds.
Like in OBS1, objects will snap to the edges unless the control key is
pressed. However, this will now happen even if the object is rotated or
oriented in any strange way. Snapping will also occur when stretching
or changing the bounding box size.
2014-06-15 00:54:48 -07:00
|
|
|
}
|
2015-04-02 21:35:46 -07:00
|
|
|
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
void OBSBasic::EnablePreviewDisplay(bool enable)
|
|
|
|
{
|
|
|
|
obs_display_set_enabled(ui->preview->GetDisplay(), enable);
|
|
|
|
ui->preview->setVisible(enable);
|
2019-07-12 11:11:49 -05:00
|
|
|
ui->previewDisabledWidget->setVisible(!enable);
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
}
|
|
|
|
|
2015-04-02 21:35:46 -07:00
|
|
|
void OBSBasic::TogglePreview()
|
|
|
|
{
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
previewEnabled = !previewEnabled;
|
|
|
|
EnablePreviewDisplay(previewEnabled);
|
2015-04-02 21:35:46 -07:00
|
|
|
}
|
2015-04-02 23:09:13 -07:00
|
|
|
|
2019-03-04 17:29:37 -06:00
|
|
|
void OBSBasic::EnablePreview()
|
|
|
|
{
|
|
|
|
if (previewProgramMode)
|
|
|
|
return;
|
|
|
|
|
|
|
|
previewEnabled = true;
|
|
|
|
EnablePreviewDisplay(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::DisablePreview()
|
|
|
|
{
|
|
|
|
if (previewProgramMode)
|
|
|
|
return;
|
|
|
|
|
|
|
|
previewEnabled = false;
|
|
|
|
EnablePreviewDisplay(false);
|
|
|
|
}
|
|
|
|
|
2018-06-02 06:52:23 -07:00
|
|
|
static bool nudge_callback(obs_scene_t *, obs_sceneitem_t *item, void *param)
|
2015-04-02 23:09:13 -07:00
|
|
|
{
|
2018-06-02 06:52:23 -07:00
|
|
|
if (obs_sceneitem_locked(item))
|
|
|
|
return true;
|
2015-04-02 23:09:13 -07:00
|
|
|
|
2018-06-02 06:52:23 -07:00
|
|
|
struct vec2 &offset = *reinterpret_cast<struct vec2 *>(param);
|
|
|
|
struct vec2 pos;
|
2017-06-17 19:10:42 -05:00
|
|
|
|
2018-06-02 06:52:23 -07:00
|
|
|
if (!obs_sceneitem_selected(item)) {
|
2018-06-02 09:45:01 -07:00
|
|
|
if (obs_sceneitem_is_group(item)) {
|
|
|
|
struct vec3 offset3;
|
|
|
|
vec3_set(&offset3, offset.x, offset.y, 0.0f);
|
|
|
|
|
|
|
|
struct matrix4 matrix;
|
|
|
|
obs_sceneitem_get_draw_transform(item, &matrix);
|
|
|
|
vec4_set(&matrix.t, 0.0f, 0.0f, 0.0f, 1.0f);
|
|
|
|
matrix4_inv(&matrix, &matrix);
|
|
|
|
vec3_transform(&offset3, &offset3, &matrix);
|
|
|
|
|
|
|
|
struct vec2 new_offset;
|
|
|
|
vec2_set(&new_offset, offset3.x, offset3.y);
|
|
|
|
obs_sceneitem_group_enum_items(item, nudge_callback,
|
|
|
|
&new_offset);
|
|
|
|
}
|
|
|
|
|
2018-06-02 06:52:23 -07:00
|
|
|
return true;
|
|
|
|
}
|
2015-04-02 23:09:13 -07:00
|
|
|
|
2018-06-02 06:52:23 -07:00
|
|
|
obs_sceneitem_get_pos(item, &pos);
|
|
|
|
vec2_add(&pos, &pos, &offset);
|
|
|
|
obs_sceneitem_set_pos(item, &pos);
|
|
|
|
return true;
|
|
|
|
}
|
2015-04-10 10:07:56 -07:00
|
|
|
|
2018-06-02 06:52:23 -07:00
|
|
|
void OBSBasic::Nudge(int dist, MoveDir dir)
|
|
|
|
{
|
|
|
|
if (ui->preview->Locked())
|
|
|
|
return;
|
2015-04-02 23:09:13 -07:00
|
|
|
|
2018-06-02 06:52:23 -07:00
|
|
|
struct vec2 offset;
|
|
|
|
vec2_set(&offset, 0.0f, 0.0f);
|
2015-04-02 23:09:13 -07:00
|
|
|
|
2018-06-02 06:52:23 -07:00
|
|
|
switch (dir) {
|
|
|
|
case MoveDir::Up:
|
|
|
|
offset.y = (float)-dist;
|
|
|
|
break;
|
|
|
|
case MoveDir::Down:
|
|
|
|
offset.y = (float)dist;
|
|
|
|
break;
|
|
|
|
case MoveDir::Left:
|
|
|
|
offset.x = (float)-dist;
|
|
|
|
break;
|
|
|
|
case MoveDir::Right:
|
|
|
|
offset.x = (float)dist;
|
|
|
|
break;
|
|
|
|
}
|
2015-04-02 23:09:13 -07:00
|
|
|
|
2018-06-02 06:52:23 -07:00
|
|
|
obs_scene_enum_items(GetCurrentScene(), nudge_callback, &offset);
|
2015-04-02 23:09:13 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::NudgeUp()
|
|
|
|
{
|
|
|
|
Nudge(1, MoveDir::Up);
|
|
|
|
}
|
|
|
|
void OBSBasic::NudgeDown()
|
|
|
|
{
|
|
|
|
Nudge(1, MoveDir::Down);
|
|
|
|
}
|
|
|
|
void OBSBasic::NudgeLeft()
|
|
|
|
{
|
|
|
|
Nudge(1, MoveDir::Left);
|
|
|
|
}
|
|
|
|
void OBSBasic::NudgeRight()
|
|
|
|
{
|
|
|
|
Nudge(1, MoveDir::Right);
|
|
|
|
}
|
2015-04-04 01:40:15 -07:00
|
|
|
|
2019-10-08 00:43:10 -05:00
|
|
|
void OBSBasic::DeleteProjector(OBSProjector *projector)
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < projectors.size(); i++) {
|
|
|
|
if (projectors[i] == projector) {
|
|
|
|
projectors[i]->deleteLater();
|
|
|
|
projectors.erase(projectors.begin() + i);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-28 23:34:11 -03:00
|
|
|
OBSProjector *OBSBasic::OpenProjector(obs_source_t *source, int monitor,
|
2019-10-08 00:43:10 -05:00
|
|
|
ProjectorType type)
|
2015-04-04 01:40:15 -07:00
|
|
|
{
|
|
|
|
/* seriously? 10 monitors? */
|
2016-12-29 09:21:53 -06:00
|
|
|
if (monitor > 9 || monitor > QGuiApplication::screens().size() - 1)
|
2018-02-28 23:34:11 -03:00
|
|
|
return nullptr;
|
2016-12-29 09:21:53 -06:00
|
|
|
|
2020-06-13 19:54:28 -05:00
|
|
|
if (monitor > -1) {
|
|
|
|
for (size_t i = 0; i < projectors.size(); i++) {
|
|
|
|
if (projectors[i]->GetMonitor() == monitor) {
|
|
|
|
DeleteProjector(projectors[i]);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-28 23:34:11 -03:00
|
|
|
OBSProjector *projector =
|
2019-10-08 00:43:10 -05:00
|
|
|
new OBSProjector(nullptr, source, monitor, type);
|
2017-03-01 22:08:49 -06:00
|
|
|
|
2019-10-08 00:43:10 -05:00
|
|
|
if (projector)
|
|
|
|
projectors.emplace_back(projector);
|
2018-02-28 23:34:11 -03:00
|
|
|
|
|
|
|
return projector;
|
2015-04-04 01:40:15 -07:00
|
|
|
}
|
|
|
|
|
2017-11-08 22:38:44 -05:00
|
|
|
void OBSBasic::OpenStudioProgramProjector()
|
|
|
|
{
|
|
|
|
int monitor = sender()->property("monitor").toInt();
|
2019-10-08 00:43:10 -05:00
|
|
|
OpenProjector(nullptr, monitor, ProjectorType::StudioProgram);
|
2017-11-08 22:38:44 -05:00
|
|
|
}
|
|
|
|
|
2015-04-04 01:40:15 -07:00
|
|
|
void OBSBasic::OpenPreviewProjector()
|
|
|
|
{
|
|
|
|
int monitor = sender()->property("monitor").toInt();
|
2019-10-08 00:43:10 -05:00
|
|
|
OpenProjector(nullptr, monitor, ProjectorType::Preview);
|
2015-04-04 01:40:15 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::OpenSourceProjector()
|
|
|
|
{
|
|
|
|
int monitor = sender()->property("monitor").toInt();
|
|
|
|
OBSSceneItem item = GetCurrentSceneItem();
|
|
|
|
if (!item)
|
|
|
|
return;
|
|
|
|
|
2019-10-08 00:43:10 -05:00
|
|
|
OpenProjector(obs_sceneitem_get_source(item), monitor,
|
2018-02-28 23:34:11 -03:00
|
|
|
ProjectorType::Source);
|
2015-04-04 01:40:15 -07:00
|
|
|
}
|
|
|
|
|
2017-10-21 15:51:01 -02:00
|
|
|
void OBSBasic::OpenMultiviewProjector()
|
|
|
|
{
|
|
|
|
int monitor = sender()->property("monitor").toInt();
|
2019-10-08 00:43:10 -05:00
|
|
|
OpenProjector(nullptr, monitor, ProjectorType::Multiview);
|
2017-10-21 15:51:01 -02:00
|
|
|
}
|
|
|
|
|
2015-04-04 01:40:15 -07:00
|
|
|
void OBSBasic::OpenSceneProjector()
|
|
|
|
{
|
|
|
|
int monitor = sender()->property("monitor").toInt();
|
|
|
|
OBSScene scene = GetCurrentScene();
|
|
|
|
if (!scene)
|
|
|
|
return;
|
|
|
|
|
2019-10-08 00:43:10 -05:00
|
|
|
OpenProjector(obs_scene_get_source(scene), monitor,
|
2018-02-28 17:49:47 -03:00
|
|
|
ProjectorType::Scene);
|
2017-03-01 22:08:49 -06:00
|
|
|
}
|
|
|
|
|
2017-11-08 22:38:44 -05:00
|
|
|
void OBSBasic::OpenStudioProgramWindow()
|
|
|
|
{
|
2019-10-08 00:43:10 -05:00
|
|
|
OpenProjector(nullptr, -1, ProjectorType::StudioProgram);
|
2017-11-08 22:38:44 -05:00
|
|
|
}
|
|
|
|
|
2017-03-01 22:08:49 -06:00
|
|
|
void OBSBasic::OpenPreviewWindow()
|
|
|
|
{
|
2019-10-08 00:43:10 -05:00
|
|
|
OpenProjector(nullptr, -1, ProjectorType::Preview);
|
2017-03-01 22:08:49 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::OpenSourceWindow()
|
|
|
|
{
|
|
|
|
OBSSceneItem item = GetCurrentSceneItem();
|
|
|
|
if (!item)
|
|
|
|
return;
|
|
|
|
|
|
|
|
OBSSource source = obs_sceneitem_get_source(item);
|
2018-02-28 23:34:11 -03:00
|
|
|
|
2019-10-08 00:43:10 -05:00
|
|
|
OpenProjector(obs_sceneitem_get_source(item), -1,
|
2018-02-28 23:34:11 -03:00
|
|
|
ProjectorType::Source);
|
2017-03-01 22:08:49 -06:00
|
|
|
}
|
|
|
|
|
2017-10-21 15:51:01 -02:00
|
|
|
void OBSBasic::OpenMultiviewWindow()
|
|
|
|
{
|
2019-10-08 00:43:10 -05:00
|
|
|
OpenProjector(nullptr, -1, ProjectorType::Multiview);
|
2017-10-21 15:51:01 -02:00
|
|
|
}
|
|
|
|
|
2017-03-01 22:08:49 -06:00
|
|
|
void OBSBasic::OpenSceneWindow()
|
|
|
|
{
|
|
|
|
OBSScene scene = GetCurrentScene();
|
|
|
|
if (!scene)
|
|
|
|
return;
|
|
|
|
|
|
|
|
OBSSource source = obs_scene_get_source(scene);
|
2018-02-28 23:34:11 -03:00
|
|
|
|
2019-10-08 00:43:10 -05:00
|
|
|
OpenProjector(obs_scene_get_source(scene), -1, ProjectorType::Scene);
|
2015-04-04 01:40:15 -07:00
|
|
|
}
|
2015-07-03 17:55:55 -07:00
|
|
|
|
2016-12-29 09:21:53 -06:00
|
|
|
void OBSBasic::OpenSavedProjectors()
|
|
|
|
{
|
2018-02-28 17:49:47 -03:00
|
|
|
for (SavedProjectorInfo *info : savedProjectorsArray) {
|
2019-06-01 06:51:39 +10:00
|
|
|
OpenSavedProjector(info);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::OpenSavedProjector(SavedProjectorInfo *info)
|
|
|
|
{
|
|
|
|
if (info) {
|
2018-03-01 02:10:32 -03:00
|
|
|
OBSProjector *projector = nullptr;
|
2018-02-28 17:49:47 -03:00
|
|
|
switch (info->type) {
|
2018-03-01 02:10:32 -03:00
|
|
|
case ProjectorType::Source:
|
2018-02-28 17:49:47 -03:00
|
|
|
case ProjectorType::Scene: {
|
|
|
|
OBSSource source =
|
|
|
|
obs_get_source_by_name(info->name.c_str());
|
|
|
|
if (!source)
|
2019-06-01 06:51:39 +10:00
|
|
|
return;
|
2016-12-29 09:21:53 -06:00
|
|
|
|
2019-10-08 00:43:10 -05:00
|
|
|
projector = OpenProjector(source, info->monitor,
|
2018-03-01 02:10:32 -03:00
|
|
|
info->type);
|
2018-02-28 17:49:47 -03:00
|
|
|
|
|
|
|
obs_source_release(source);
|
|
|
|
break;
|
2017-10-21 15:51:01 -02:00
|
|
|
}
|
2019-10-08 00:43:10 -05:00
|
|
|
default: {
|
2018-03-01 02:10:32 -03:00
|
|
|
projector = OpenProjector(nullptr, info->monitor,
|
2019-10-08 00:43:10 -05:00
|
|
|
info->type);
|
2018-02-28 17:49:47 -03:00
|
|
|
break;
|
|
|
|
}
|
2016-12-29 09:21:53 -06:00
|
|
|
}
|
2018-03-01 02:10:32 -03:00
|
|
|
|
2018-10-22 00:32:39 -05:00
|
|
|
if (projector && !info->geometry.empty() && info->monitor < 0) {
|
2018-03-01 02:10:32 -03:00
|
|
|
QByteArray byteArray = QByteArray::fromBase64(
|
|
|
|
QByteArray(info->geometry.c_str()));
|
|
|
|
projector->restoreGeometry(byteArray);
|
2017-10-21 15:51:01 -02:00
|
|
|
|
2018-03-01 02:10:32 -03:00
|
|
|
if (!WindowPositionValid(projector->normalGeometry())) {
|
|
|
|
QRect rect = App()->desktop()->geometry();
|
|
|
|
projector->setGeometry(QStyle::alignedRect(
|
|
|
|
Qt::LeftToRight, Qt::AlignCenter,
|
|
|
|
size(), rect));
|
2016-12-29 09:21:53 -06:00
|
|
|
}
|
2020-04-27 23:26:20 +02:00
|
|
|
|
|
|
|
if (info->alwaysOnTopOverridden)
|
|
|
|
projector->SetIsAlwaysOnTop(info->alwaysOnTop,
|
|
|
|
true);
|
2016-12-29 09:21:53 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-04 08:21:43 -06:00
|
|
|
void OBSBasic::on_actionFullscreenInterface_triggered()
|
|
|
|
{
|
2019-10-21 15:17:26 -05:00
|
|
|
if (!isFullScreen())
|
2017-03-04 08:21:43 -06:00
|
|
|
showFullScreen();
|
|
|
|
else
|
|
|
|
showNormal();
|
|
|
|
}
|
|
|
|
|
2015-07-03 17:55:55 -07:00
|
|
|
void OBSBasic::UpdateTitleBar()
|
|
|
|
{
|
|
|
|
stringstream name;
|
|
|
|
|
2015-06-23 19:38:01 -07:00
|
|
|
const char *profile =
|
|
|
|
config_get_string(App()->GlobalConfig(), "Basic", "Profile");
|
2015-06-23 19:29:07 -07:00
|
|
|
const char *sceneCollection = config_get_string(
|
|
|
|
App()->GlobalConfig(), "Basic", "SceneCollection");
|
|
|
|
|
UI: Implement transitions and preview/program mode
Implements transitions, and introduces "Studio Mode" which allows live
editing of the same or different scenes while preserving what's
currently being displayed.
Studio Mode offers a number of new features:
- The ability to edit different scenes or the same scene without
modifying what's currently being displayed (of course)
- The ability to set up "quick transitions" with a desired transition
and duration that can be assigned hotkeys
- The option to create full copies of all sources in the program scene
to allow editing of source properties of the same scene live without
modifying the output, or (by default) just use references. (Note
however that certain sources cannot be duplicated, such as capture
sources, media sources, and device sources)
- Swap Mode (enabled by default) which swaps the program scene with
the preview scene when a transition completes
Currently, only non-configurable transitions (transitions without
properties) are listed, and the only transitions available as of this
writing are fade and cut. In future versions more transitions will be
added, such as swipe, stingers, and many other various sort of
transitions, and the UI will support being able to add/configure/remove
those sort of configurable transitions.
2016-01-23 11:19:29 -08:00
|
|
|
name << "OBS ";
|
|
|
|
if (previewProgramMode)
|
|
|
|
name << "Studio ";
|
|
|
|
|
|
|
|
name << App()->GetVersionString();
|
2016-10-09 20:53:45 -04:00
|
|
|
if (App()->IsPortableMode())
|
|
|
|
name << " - Portable Mode";
|
|
|
|
|
2015-06-23 19:38:01 -07:00
|
|
|
name << " - " << Str("TitleBar.Profile") << ": " << profile;
|
2015-06-23 19:29:07 -07:00
|
|
|
name << " - " << Str("TitleBar.Scenes") << ": " << sceneCollection;
|
2015-07-03 17:55:55 -07:00
|
|
|
|
|
|
|
setWindowTitle(QT_UTF8(name.str().c_str()));
|
|
|
|
}
|
2015-06-23 19:38:01 -07:00
|
|
|
|
|
|
|
int OBSBasic::GetProfilePath(char *path, size_t size, const char *file) const
|
|
|
|
{
|
|
|
|
char profiles_path[512];
|
|
|
|
const char *profile =
|
|
|
|
config_get_string(App()->GlobalConfig(), "Basic", "ProfileDir");
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (!profile)
|
|
|
|
return -1;
|
|
|
|
if (!path)
|
|
|
|
return -1;
|
|
|
|
if (!file)
|
|
|
|
file = "";
|
|
|
|
|
|
|
|
ret = GetConfigPath(profiles_path, 512, "obs-studio/basic/profiles");
|
|
|
|
if (ret <= 0)
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
if (!*file)
|
|
|
|
return snprintf(path, size, "%s/%s", profiles_path, profile);
|
|
|
|
|
|
|
|
return snprintf(path, size, "%s/%s/%s", profiles_path, profile, file);
|
|
|
|
}
|
2016-06-30 14:52:56 -07:00
|
|
|
|
2017-07-27 02:40:41 -07:00
|
|
|
void OBSBasic::on_resetUI_triggered()
|
2016-06-30 14:52:56 -07:00
|
|
|
{
|
2019-02-06 13:57:59 -08:00
|
|
|
/* prune deleted extra docks */
|
|
|
|
for (int i = extraDocks.size() - 1; i >= 0; i--) {
|
|
|
|
if (!extraDocks[i]) {
|
|
|
|
extraDocks.removeAt(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (extraDocks.size()) {
|
|
|
|
QMessageBox::StandardButton button = QMessageBox::question(
|
|
|
|
this, QTStr("ResetUIWarning.Title"),
|
|
|
|
QTStr("ResetUIWarning.Text"));
|
|
|
|
|
|
|
|
if (button == QMessageBox::No)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* undock/hide/center extra docks */
|
|
|
|
for (int i = extraDocks.size() - 1; i >= 0; i--) {
|
|
|
|
if (extraDocks[i]) {
|
|
|
|
extraDocks[i]->setVisible(true);
|
|
|
|
extraDocks[i]->setFloating(true);
|
|
|
|
extraDocks[i]->move(frameGeometry().topLeft() +
|
|
|
|
rect().center() -
|
|
|
|
extraDocks[i]->rect().center());
|
|
|
|
extraDocks[i]->setVisible(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-27 02:40:41 -07:00
|
|
|
restoreState(startingDockLayout);
|
2016-06-30 14:52:56 -07:00
|
|
|
|
2017-07-27 22:50:25 -07:00
|
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 6, 0)
|
2017-07-27 02:40:41 -07:00
|
|
|
int cx = width();
|
|
|
|
int cy = height();
|
|
|
|
|
|
|
|
int cx22_5 = cx * 225 / 1000;
|
|
|
|
int cx5 = cx * 5 / 100;
|
|
|
|
|
|
|
|
cy = cy * 225 / 1000;
|
|
|
|
|
|
|
|
int mixerSize = cx - (cx22_5 * 2 + cx5 * 2);
|
|
|
|
|
|
|
|
QList<QDockWidget *> docks{ui->scenesDock, ui->sourcesDock,
|
|
|
|
ui->mixerDock, ui->transitionsDock,
|
2018-09-16 10:02:43 -07:00
|
|
|
ui->controlsDock};
|
2017-07-27 02:40:41 -07:00
|
|
|
|
|
|
|
QList<int> sizes{cx22_5, cx22_5, mixerSize, cx5, cx5};
|
|
|
|
|
|
|
|
ui->scenesDock->setVisible(true);
|
|
|
|
ui->sourcesDock->setVisible(true);
|
|
|
|
ui->mixerDock->setVisible(true);
|
|
|
|
ui->transitionsDock->setVisible(true);
|
|
|
|
ui->controlsDock->setVisible(true);
|
2018-09-16 10:02:43 -07:00
|
|
|
statsDock->setVisible(false);
|
|
|
|
statsDock->setFloating(true);
|
2017-07-27 02:40:41 -07:00
|
|
|
|
|
|
|
resizeDocks(docks, {cy, cy, cy, cy, cy}, Qt::Vertical);
|
|
|
|
resizeDocks(docks, sizes, Qt::Horizontal);
|
2017-07-27 22:50:25 -07:00
|
|
|
#endif
|
2020-05-11 22:43:54 +02:00
|
|
|
|
|
|
|
activateWindow();
|
2017-07-27 02:40:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_lockUI_toggled(bool lock)
|
|
|
|
{
|
|
|
|
QDockWidget::DockWidgetFeatures features =
|
|
|
|
lock ? QDockWidget::NoDockWidgetFeatures
|
2020-06-07 15:31:13 -07:00
|
|
|
: (QDockWidget::DockWidgetClosable |
|
|
|
|
QDockWidget::DockWidgetMovable |
|
|
|
|
QDockWidget::DockWidgetFloatable);
|
2017-07-27 02:40:41 -07:00
|
|
|
|
2019-04-02 12:23:10 -07:00
|
|
|
QDockWidget::DockWidgetFeatures mainFeatures = features;
|
|
|
|
mainFeatures &= ~QDockWidget::QDockWidget::DockWidgetClosable;
|
|
|
|
|
|
|
|
ui->scenesDock->setFeatures(mainFeatures);
|
|
|
|
ui->sourcesDock->setFeatures(mainFeatures);
|
|
|
|
ui->mixerDock->setFeatures(mainFeatures);
|
|
|
|
ui->transitionsDock->setFeatures(mainFeatures);
|
|
|
|
ui->controlsDock->setFeatures(mainFeatures);
|
2018-09-16 10:02:43 -07:00
|
|
|
statsDock->setFeatures(features);
|
2019-02-06 13:57:59 -08:00
|
|
|
|
|
|
|
for (int i = extraDocks.size() - 1; i >= 0; i--) {
|
|
|
|
if (!extraDocks[i]) {
|
|
|
|
extraDocks.removeAt(i);
|
|
|
|
} else {
|
|
|
|
extraDocks[i]->setFeatures(features);
|
|
|
|
}
|
|
|
|
}
|
2016-06-30 14:52:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_toggleListboxToolbars_toggled(bool visible)
|
|
|
|
{
|
|
|
|
ui->sourcesToolbar->setVisible(visible);
|
|
|
|
ui->scenesToolbar->setVisible(visible);
|
|
|
|
|
|
|
|
config_set_bool(App()->GlobalConfig(), "BasicWindow",
|
|
|
|
"ShowListboxToolbars", visible);
|
|
|
|
}
|
|
|
|
|
2018-08-23 20:43:44 -05:00
|
|
|
void OBSBasic::ShowContextBar()
|
|
|
|
{
|
|
|
|
on_toggleContextBar_toggled(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::HideContextBar()
|
|
|
|
{
|
|
|
|
on_toggleContextBar_toggled(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_toggleContextBar_toggled(bool visible)
|
|
|
|
{
|
|
|
|
config_set_bool(App()->GlobalConfig(), "BasicWindow",
|
|
|
|
"ShowContextToolbars", visible);
|
|
|
|
this->ui->contextContainer->setVisible(visible);
|
|
|
|
}
|
|
|
|
|
2016-06-30 14:52:56 -07:00
|
|
|
void OBSBasic::on_toggleStatusBar_toggled(bool visible)
|
|
|
|
{
|
|
|
|
ui->statusbar->setVisible(visible);
|
|
|
|
|
|
|
|
config_set_bool(App()->GlobalConfig(), "BasicWindow", "ShowStatusBar",
|
|
|
|
visible);
|
|
|
|
}
|
2016-07-26 01:32:43 -07:00
|
|
|
|
2020-01-20 23:18:46 -08:00
|
|
|
void OBSBasic::on_toggleSourceIcons_toggled(bool visible)
|
|
|
|
{
|
|
|
|
ui->sources->SetIconsVisible(visible);
|
2020-01-25 22:48:38 +11:00
|
|
|
if (advAudioWindow != nullptr)
|
|
|
|
advAudioWindow->SetIconsVisible(visible);
|
2020-01-20 23:18:46 -08:00
|
|
|
|
|
|
|
config_set_bool(App()->GlobalConfig(), "BasicWindow", "ShowSourceIcons",
|
|
|
|
visible);
|
|
|
|
}
|
|
|
|
|
2016-07-26 01:32:43 -07:00
|
|
|
void OBSBasic::on_actionLockPreview_triggered()
|
|
|
|
{
|
|
|
|
ui->preview->ToggleLocked();
|
|
|
|
ui->actionLockPreview->setChecked(ui->preview->Locked());
|
|
|
|
}
|
2016-08-13 09:36:17 -05:00
|
|
|
|
2016-11-05 12:48:46 -04:00
|
|
|
void OBSBasic::on_scalingMenu_aboutToShow()
|
|
|
|
{
|
|
|
|
obs_video_info ovi;
|
|
|
|
obs_get_video_info(&ovi);
|
|
|
|
|
|
|
|
QAction *action = ui->actionScaleCanvas;
|
|
|
|
QString text = QTStr("Basic.MainMenu.Edit.Scale.Canvas");
|
|
|
|
text = text.arg(QString::number(ovi.base_width),
|
|
|
|
QString::number(ovi.base_height));
|
|
|
|
action->setText(text);
|
|
|
|
|
|
|
|
action = ui->actionScaleOutput;
|
|
|
|
text = QTStr("Basic.MainMenu.Edit.Scale.Output");
|
|
|
|
text = text.arg(QString::number(ovi.output_width),
|
|
|
|
QString::number(ovi.output_height));
|
|
|
|
action->setText(text);
|
2017-05-13 13:13:55 -04:00
|
|
|
action->setVisible(!(ovi.output_width == ovi.base_width &&
|
|
|
|
ovi.output_height == ovi.base_height));
|
2016-11-05 12:48:46 -04:00
|
|
|
|
|
|
|
UpdatePreviewScalingMenu();
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionScaleWindow_triggered()
|
|
|
|
{
|
2017-05-13 13:13:55 -04:00
|
|
|
ui->preview->SetFixedScaling(false);
|
2016-11-05 12:48:46 -04:00
|
|
|
ui->preview->ResetScrollingOffset();
|
|
|
|
emit ui->preview->DisplayResized();
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionScaleCanvas_triggered()
|
|
|
|
{
|
2017-05-13 13:13:55 -04:00
|
|
|
ui->preview->SetFixedScaling(true);
|
|
|
|
ui->preview->SetScalingLevel(0);
|
2016-11-05 12:48:46 -04:00
|
|
|
emit ui->preview->DisplayResized();
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionScaleOutput_triggered()
|
|
|
|
{
|
2017-05-13 13:13:55 -04:00
|
|
|
obs_video_info ovi;
|
|
|
|
obs_get_video_info(&ovi);
|
|
|
|
|
|
|
|
ui->preview->SetFixedScaling(true);
|
|
|
|
float scalingAmount = float(ovi.output_width) / float(ovi.base_width);
|
|
|
|
// log base ZOOM_SENSITIVITY of x = log(x) / log(ZOOM_SENSITIVITY)
|
|
|
|
int32_t approxScalingLevel =
|
|
|
|
int32_t(round(log(scalingAmount) / log(ZOOM_SENSITIVITY)));
|
|
|
|
ui->preview->SetScalingLevel(approxScalingLevel);
|
|
|
|
ui->preview->SetScalingAmount(scalingAmount);
|
2016-11-05 12:48:46 -04:00
|
|
|
emit ui->preview->DisplayResized();
|
|
|
|
}
|
|
|
|
|
2016-08-13 09:36:17 -05:00
|
|
|
void OBSBasic::SetShowing(bool showing)
|
|
|
|
{
|
|
|
|
if (!showing && isVisible()) {
|
|
|
|
config_set_string(App()->GlobalConfig(), "BasicWindow",
|
|
|
|
"geometry",
|
|
|
|
saveGeometry().toBase64().constData());
|
|
|
|
|
2017-01-25 00:39:18 -08:00
|
|
|
/* hide all visible child dialogs */
|
|
|
|
visDlgPositions.clear();
|
|
|
|
if (!visDialogs.isEmpty()) {
|
|
|
|
for (QDialog *dlg : visDialogs) {
|
|
|
|
visDlgPositions.append(dlg->pos());
|
|
|
|
dlg->hide();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-19 06:44:02 -07:00
|
|
|
if (showHide)
|
|
|
|
showHide->setText(QTStr("Basic.SystemTray.Show"));
|
2016-08-13 09:36:17 -05:00
|
|
|
QTimer::singleShot(250, this, SLOT(hide()));
|
|
|
|
|
|
|
|
setVisible(false);
|
|
|
|
|
2018-08-17 03:50:25 -07:00
|
|
|
#ifdef __APPLE__
|
|
|
|
EnableOSXDockIcon(false);
|
|
|
|
#endif
|
|
|
|
|
2016-08-13 09:36:17 -05:00
|
|
|
} else if (showing && !isVisible()) {
|
2016-10-19 06:44:02 -07:00
|
|
|
if (showHide)
|
|
|
|
showHide->setText(QTStr("Basic.SystemTray.Hide"));
|
2016-08-13 09:36:17 -05:00
|
|
|
QTimer::singleShot(250, this, SLOT(show()));
|
|
|
|
|
|
|
|
setVisible(true);
|
2017-01-24 23:01:24 -08:00
|
|
|
|
2018-08-17 03:50:25 -07:00
|
|
|
#ifdef __APPLE__
|
|
|
|
EnableOSXDockIcon(true);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* raise and activate window to ensure it is on top */
|
|
|
|
raise();
|
|
|
|
activateWindow();
|
|
|
|
|
2017-01-25 00:39:18 -08:00
|
|
|
/* show all child dialogs that was visible earlier */
|
|
|
|
if (!visDialogs.isEmpty()) {
|
|
|
|
for (int i = 0; i < visDialogs.size(); ++i) {
|
|
|
|
QDialog *dlg = visDialogs[i];
|
|
|
|
dlg->move(visDlgPositions[i]);
|
|
|
|
dlg->show();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-24 23:01:24 -08:00
|
|
|
/* Unminimize window if it was hidden to tray instead of task
|
|
|
|
* bar. */
|
|
|
|
if (sysTrayMinimizeToTray()) {
|
|
|
|
Qt::WindowStates state;
|
|
|
|
state = windowState() & ~Qt::WindowMinimized;
|
|
|
|
state |= Qt::WindowActive;
|
|
|
|
setWindowState(state);
|
|
|
|
}
|
2016-08-13 09:36:17 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-25 00:39:18 -08:00
|
|
|
void OBSBasic::ToggleShowHide()
|
|
|
|
{
|
|
|
|
bool showing = isVisible();
|
|
|
|
if (showing) {
|
|
|
|
/* check for modal dialogs */
|
|
|
|
EnumDialogs();
|
|
|
|
if (!modalDialogs.isEmpty() || !visMsgBoxes.isEmpty())
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
SetShowing(!showing);
|
|
|
|
}
|
|
|
|
|
2016-10-19 06:49:12 -07:00
|
|
|
void OBSBasic::SystemTrayInit()
|
|
|
|
{
|
2019-03-11 10:52:01 -05:00
|
|
|
trayIcon.reset(new QSystemTrayIcon(
|
|
|
|
QIcon::fromTheme("obs-tray", QIcon(":/res/images/obs.png")),
|
|
|
|
this));
|
2016-08-13 09:36:17 -05:00
|
|
|
trayIcon->setToolTip("OBS Studio");
|
|
|
|
|
|
|
|
showHide = new QAction(QTStr("Basic.SystemTray.Show"), trayIcon.data());
|
|
|
|
sysTrayStream = new QAction(QTStr("Basic.Main.StartStreaming"),
|
2018-07-31 21:11:31 -07:00
|
|
|
trayIcon.data());
|
2016-08-13 09:36:17 -05:00
|
|
|
sysTrayRecord = new QAction(QTStr("Basic.Main.StartRecording"),
|
2018-07-31 21:11:31 -07:00
|
|
|
trayIcon.data());
|
2016-12-09 14:40:04 -08:00
|
|
|
sysTrayReplayBuffer = new QAction(QTStr("Basic.Main.StartReplayBuffer"),
|
2018-07-31 21:11:31 -07:00
|
|
|
trayIcon.data());
|
2016-08-13 09:36:17 -05:00
|
|
|
exit = new QAction(QTStr("Exit"), trayIcon.data());
|
|
|
|
|
2020-04-19 09:31:18 -05:00
|
|
|
trayMenu = new QMenu;
|
|
|
|
previewProjector = new QMenu(QTStr("PreviewProjector"));
|
|
|
|
studioProgramProjector = new QMenu(QTStr("StudioProgramProjector"));
|
2018-04-05 05:54:24 -03:00
|
|
|
AddProjectorMenuMonitors(previewProjector, this,
|
|
|
|
SLOT(OpenPreviewProjector()));
|
|
|
|
AddProjectorMenuMonitors(studioProgramProjector, this,
|
|
|
|
SLOT(OpenStudioProgramProjector()));
|
|
|
|
trayMenu->addAction(showHide);
|
|
|
|
trayMenu->addMenu(previewProjector);
|
|
|
|
trayMenu->addMenu(studioProgramProjector);
|
|
|
|
trayMenu->addAction(sysTrayStream);
|
|
|
|
trayMenu->addAction(sysTrayRecord);
|
|
|
|
trayMenu->addAction(sysTrayReplayBuffer);
|
|
|
|
trayMenu->addAction(exit);
|
|
|
|
trayIcon->setContextMenu(trayMenu);
|
2019-06-11 20:22:24 -07:00
|
|
|
trayIcon->show();
|
2018-04-05 05:54:24 -03:00
|
|
|
|
2016-12-09 14:40:04 -08:00
|
|
|
if (outputHandler && !outputHandler->replayBuffer)
|
|
|
|
sysTrayReplayBuffer->setEnabled(false);
|
2016-12-07 05:21:44 -08:00
|
|
|
|
2018-07-31 21:11:31 -07:00
|
|
|
connect(trayIcon.data(),
|
|
|
|
SIGNAL(activated(QSystemTrayIcon::ActivationReason)), this,
|
2016-08-13 09:36:17 -05:00
|
|
|
SLOT(IconActivated(QSystemTrayIcon::ActivationReason)));
|
|
|
|
connect(showHide, SIGNAL(triggered()), this, SLOT(ToggleShowHide()));
|
|
|
|
connect(sysTrayStream, SIGNAL(triggered()), this,
|
|
|
|
SLOT(on_streamButton_clicked()));
|
|
|
|
connect(sysTrayRecord, SIGNAL(triggered()), this,
|
|
|
|
SLOT(on_recordButton_clicked()));
|
2016-12-11 21:04:58 -08:00
|
|
|
connect(sysTrayReplayBuffer.data(), &QAction::triggered, this,
|
2016-12-09 14:40:04 -08:00
|
|
|
&OBSBasic::ReplayBufferClicked);
|
2016-08-13 09:36:17 -05:00
|
|
|
connect(exit, SIGNAL(triggered()), this, SLOT(close()));
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::IconActivated(QSystemTrayIcon::ActivationReason reason)
|
|
|
|
{
|
2020-04-19 09:31:18 -05:00
|
|
|
// Refresh projector list
|
|
|
|
previewProjector->clear();
|
|
|
|
studioProgramProjector->clear();
|
|
|
|
AddProjectorMenuMonitors(previewProjector, this,
|
|
|
|
SLOT(OpenPreviewProjector()));
|
|
|
|
AddProjectorMenuMonitors(studioProgramProjector, this,
|
|
|
|
SLOT(OpenStudioProgramProjector()));
|
|
|
|
|
2020-03-19 05:31:29 -07:00
|
|
|
if (reason == QSystemTrayIcon::Trigger) {
|
|
|
|
EnablePreviewDisplay(previewEnabled && !isVisible());
|
2016-08-13 09:36:17 -05:00
|
|
|
ToggleShowHide();
|
2020-03-19 05:31:29 -07:00
|
|
|
}
|
2016-08-13 09:36:17 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::SysTrayNotify(const QString &text,
|
|
|
|
QSystemTrayIcon::MessageIcon n)
|
|
|
|
{
|
2019-05-03 02:21:29 -04:00
|
|
|
if (trayIcon && QSystemTrayIcon::supportsMessages()) {
|
2016-08-13 09:36:17 -05:00
|
|
|
QSystemTrayIcon::MessageIcon icon =
|
|
|
|
QSystemTrayIcon::MessageIcon(n);
|
|
|
|
trayIcon->showMessage("OBS Studio", text, icon, 10000);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::SystemTray(bool firstStarted)
|
|
|
|
{
|
|
|
|
if (!QSystemTrayIcon::isSystemTrayAvailable())
|
|
|
|
return;
|
2019-05-04 09:23:21 -05:00
|
|
|
if (!trayIcon && !firstStarted)
|
2019-05-01 12:11:05 -07:00
|
|
|
return;
|
2016-08-13 09:36:17 -05:00
|
|
|
|
|
|
|
bool sysTrayWhenStarted = config_get_bool(
|
|
|
|
GetGlobalConfig(), "BasicWindow", "SysTrayWhenStarted");
|
|
|
|
bool sysTrayEnabled = config_get_bool(GetGlobalConfig(), "BasicWindow",
|
|
|
|
"SysTrayEnabled");
|
|
|
|
|
|
|
|
if (firstStarted)
|
|
|
|
SystemTrayInit();
|
|
|
|
|
|
|
|
if (!sysTrayWhenStarted && !sysTrayEnabled) {
|
|
|
|
trayIcon->hide();
|
2017-01-11 13:19:17 -06:00
|
|
|
} else if ((sysTrayWhenStarted && sysTrayEnabled) ||
|
|
|
|
opt_minimize_tray) {
|
2016-08-13 09:36:17 -05:00
|
|
|
trayIcon->show();
|
|
|
|
if (firstStarted) {
|
|
|
|
QTimer::singleShot(50, this, SLOT(hide()));
|
|
|
|
EnablePreviewDisplay(false);
|
|
|
|
setVisible(false);
|
2018-08-17 03:50:25 -07:00
|
|
|
#ifdef __APPLE__
|
|
|
|
EnableOSXDockIcon(false);
|
|
|
|
#endif
|
2017-01-11 13:19:17 -06:00
|
|
|
opt_minimize_tray = false;
|
2016-08-13 09:36:17 -05:00
|
|
|
}
|
|
|
|
} else if (sysTrayEnabled) {
|
|
|
|
trayIcon->show();
|
|
|
|
} else if (!sysTrayEnabled) {
|
|
|
|
trayIcon->hide();
|
|
|
|
} else if (!sysTrayWhenStarted && sysTrayEnabled) {
|
|
|
|
trayIcon->hide();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isVisible())
|
|
|
|
showHide->setText(QTStr("Basic.SystemTray.Hide"));
|
|
|
|
else
|
|
|
|
showHide->setText(QTStr("Basic.SystemTray.Show"));
|
|
|
|
}
|
2017-01-24 23:01:24 -08:00
|
|
|
|
|
|
|
bool OBSBasic::sysTrayMinimizeToTray()
|
|
|
|
{
|
|
|
|
return config_get_bool(GetGlobalConfig(), "BasicWindow",
|
|
|
|
"SysTrayMinimizeToTray");
|
|
|
|
}
|
2020-07-11 06:05:35 -07:00
|
|
|
|
2017-03-25 04:19:29 -07:00
|
|
|
void OBSBasic::on_actionCopySource_triggered()
|
|
|
|
{
|
2019-10-04 22:33:06 -07:00
|
|
|
copyStrings.clear();
|
|
|
|
bool allowPastingDuplicate = true;
|
2018-06-02 09:45:01 -07:00
|
|
|
|
2019-10-04 22:33:06 -07:00
|
|
|
for (auto &selectedSource : GetAllSelectedSourceItems()) {
|
|
|
|
OBSSceneItem item = ui->sources->Get(selectedSource.row());
|
|
|
|
if (!item)
|
|
|
|
continue;
|
2017-03-25 04:19:29 -07:00
|
|
|
|
2019-10-04 22:33:06 -07:00
|
|
|
on_actionCopyTransform_triggered();
|
2017-03-25 04:19:29 -07:00
|
|
|
|
2019-10-04 22:33:06 -07:00
|
|
|
OBSSource source = obs_sceneitem_get_source(item);
|
2017-03-25 04:19:29 -07:00
|
|
|
|
2019-10-04 22:33:06 -07:00
|
|
|
copyStrings.push_front(obs_source_get_name(source));
|
2017-10-14 01:49:37 -04:00
|
|
|
|
2019-10-04 22:33:06 -07:00
|
|
|
copyVisible = obs_sceneitem_visible(item);
|
|
|
|
|
|
|
|
uint32_t output_flags = obs_source_get_output_flags(source);
|
2020-04-08 12:38:42 -04:00
|
|
|
if (output_flags & OBS_SOURCE_DO_NOT_DUPLICATE)
|
2019-10-04 22:33:06 -07:00
|
|
|
allowPastingDuplicate = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
ui->actionPasteRef->setEnabled(true);
|
|
|
|
ui->actionPasteDup->setEnabled(allowPastingDuplicate);
|
2017-03-25 04:19:29 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionPasteRef_triggered()
|
|
|
|
{
|
2019-10-04 22:33:06 -07:00
|
|
|
for (auto ©String : copyStrings) {
|
|
|
|
/* do not allow duplicate refs of the same group in the same scene */
|
|
|
|
OBSScene scene = GetCurrentScene();
|
|
|
|
if (!!obs_scene_get_group(scene, copyString))
|
|
|
|
continue;
|
2018-07-15 18:58:53 -07:00
|
|
|
|
2019-10-04 22:33:06 -07:00
|
|
|
OBSBasicSourceSelect::SourcePaste(copyString, copyVisible,
|
|
|
|
false);
|
|
|
|
on_actionPasteTransform_triggered();
|
|
|
|
}
|
2017-03-25 04:19:29 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionPasteDup_triggered()
|
|
|
|
{
|
2019-10-04 22:33:06 -07:00
|
|
|
for (auto ©String : copyStrings) {
|
|
|
|
OBSBasicSourceSelect::SourcePaste(copyString, copyVisible,
|
|
|
|
true);
|
|
|
|
on_actionPasteTransform_triggered();
|
|
|
|
}
|
2017-03-25 04:19:29 -07:00
|
|
|
}
|
|
|
|
|
2019-02-26 06:58:53 -06:00
|
|
|
void OBSBasic::AudioMixerCopyFilters()
|
|
|
|
{
|
|
|
|
QAction *action = reinterpret_cast<QAction *>(sender());
|
|
|
|
VolControl *vol = action->property("volControl").value<VolControl *>();
|
|
|
|
obs_source_t *source = vol->GetSource();
|
|
|
|
|
|
|
|
copyFiltersString = obs_source_get_name(source);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::AudioMixerPasteFilters()
|
|
|
|
{
|
|
|
|
QAction *action = reinterpret_cast<QAction *>(sender());
|
|
|
|
VolControl *vol = action->property("volControl").value<VolControl *>();
|
|
|
|
obs_source_t *dstSource = vol->GetSource();
|
|
|
|
|
|
|
|
OBSSource source = obs_get_source_by_name(copyFiltersString);
|
|
|
|
obs_source_release(source);
|
|
|
|
|
|
|
|
if (source == dstSource)
|
|
|
|
return;
|
|
|
|
|
|
|
|
obs_source_copy_filters(dstSource, source);
|
|
|
|
}
|
|
|
|
|
2019-04-23 02:26:17 -05:00
|
|
|
void OBSBasic::SceneCopyFilters()
|
|
|
|
{
|
|
|
|
copyFiltersString = obs_source_get_name(GetCurrentSceneSource());
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::ScenePasteFilters()
|
|
|
|
{
|
|
|
|
OBSSource source = obs_get_source_by_name(copyFiltersString);
|
|
|
|
obs_source_release(source);
|
|
|
|
|
|
|
|
OBSSource dstSource = GetCurrentSceneSource();
|
|
|
|
|
|
|
|
if (source == dstSource)
|
|
|
|
return;
|
|
|
|
|
|
|
|
obs_source_copy_filters(dstSource, source);
|
|
|
|
}
|
|
|
|
|
2017-03-25 04:19:29 -07:00
|
|
|
void OBSBasic::on_actionCopyFilters_triggered()
|
|
|
|
{
|
|
|
|
OBSSceneItem item = GetCurrentSceneItem();
|
|
|
|
|
|
|
|
if (!item)
|
|
|
|
return;
|
|
|
|
|
|
|
|
OBSSource source = obs_sceneitem_get_source(item);
|
|
|
|
|
|
|
|
copyFiltersString = obs_source_get_name(source);
|
|
|
|
|
|
|
|
ui->actionPasteFilters->setEnabled(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_actionPasteFilters_triggered()
|
|
|
|
{
|
|
|
|
OBSSource source = obs_get_source_by_name(copyFiltersString);
|
2017-12-04 17:34:42 +02:00
|
|
|
obs_source_release(source);
|
2017-03-25 04:19:29 -07:00
|
|
|
|
2017-12-04 17:34:42 +02:00
|
|
|
OBSSceneItem sceneItem = GetCurrentSceneItem();
|
2017-03-25 04:19:29 -07:00
|
|
|
OBSSource dstSource = obs_sceneitem_get_source(sceneItem);
|
|
|
|
|
|
|
|
if (source == dstSource)
|
|
|
|
return;
|
|
|
|
|
|
|
|
obs_source_copy_filters(dstSource, source);
|
|
|
|
}
|
2017-05-06 12:13:56 -07:00
|
|
|
|
2018-08-02 08:23:12 +12:00
|
|
|
static void ConfirmColor(SourceTree *sources, const QColor &color,
|
|
|
|
QModelIndexList selectedItems)
|
|
|
|
{
|
|
|
|
for (int x = 0; x < selectedItems.count(); x++) {
|
|
|
|
SourceTreeItem *treeItem =
|
|
|
|
sources->GetItemWidget(selectedItems[x].row());
|
|
|
|
treeItem->setStyleSheet("background: " +
|
|
|
|
color.name(QColor::HexArgb));
|
|
|
|
treeItem->style()->unpolish(treeItem);
|
|
|
|
treeItem->style()->polish(treeItem);
|
|
|
|
|
|
|
|
OBSSceneItem sceneItem = sources->Get(selectedItems[x].row());
|
|
|
|
obs_data_t *privData =
|
|
|
|
obs_sceneitem_get_private_settings(sceneItem);
|
|
|
|
obs_data_set_int(privData, "color-preset", 1);
|
|
|
|
obs_data_set_string(privData, "color",
|
|
|
|
QT_TO_UTF8(color.name(QColor::HexArgb)));
|
|
|
|
obs_data_release(privData);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::ColorChange()
|
|
|
|
{
|
|
|
|
QModelIndexList selectedItems =
|
|
|
|
ui->sources->selectionModel()->selectedIndexes();
|
|
|
|
QAction *action = qobject_cast<QAction *>(sender());
|
|
|
|
QPushButton *colorButton = qobject_cast<QPushButton *>(sender());
|
|
|
|
|
|
|
|
if (selectedItems.count() == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (colorButton) {
|
|
|
|
int preset = colorButton->property("bgColor").value<int>();
|
|
|
|
|
|
|
|
for (int x = 0; x < selectedItems.count(); x++) {
|
|
|
|
SourceTreeItem *treeItem = ui->sources->GetItemWidget(
|
|
|
|
selectedItems[x].row());
|
|
|
|
treeItem->setStyleSheet("");
|
|
|
|
treeItem->setProperty("bgColor", preset);
|
|
|
|
treeItem->style()->unpolish(treeItem);
|
|
|
|
treeItem->style()->polish(treeItem);
|
|
|
|
|
|
|
|
OBSSceneItem sceneItem =
|
|
|
|
ui->sources->Get(selectedItems[x].row());
|
|
|
|
obs_data_t *privData =
|
|
|
|
obs_sceneitem_get_private_settings(sceneItem);
|
|
|
|
obs_data_set_int(privData, "color-preset", preset + 1);
|
|
|
|
obs_data_set_string(privData, "color", "");
|
|
|
|
obs_data_release(privData);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 1; i < 9; i++) {
|
|
|
|
stringstream button;
|
|
|
|
button << "preset" << i;
|
|
|
|
QPushButton *cButton =
|
|
|
|
colorButton->parentWidget()
|
|
|
|
->findChild<QPushButton *>(
|
|
|
|
button.str().c_str());
|
|
|
|
cButton->setStyleSheet("border: 1px solid black");
|
|
|
|
}
|
|
|
|
|
|
|
|
colorButton->setStyleSheet("border: 2px solid black");
|
|
|
|
} else if (action) {
|
|
|
|
int preset = action->property("bgColor").value<int>();
|
|
|
|
|
|
|
|
if (preset == 1) {
|
|
|
|
OBSSceneItem curSceneItem = GetCurrentSceneItem();
|
|
|
|
SourceTreeItem *curTreeItem =
|
|
|
|
GetItemWidgetFromSceneItem(curSceneItem);
|
|
|
|
obs_data_t *curPrivData =
|
|
|
|
obs_sceneitem_get_private_settings(
|
|
|
|
curSceneItem);
|
|
|
|
|
|
|
|
int oldPreset =
|
|
|
|
obs_data_get_int(curPrivData, "color-preset");
|
|
|
|
const QString oldSheet = curTreeItem->styleSheet();
|
|
|
|
|
|
|
|
auto liveChangeColor = [=](const QColor &color) {
|
|
|
|
if (color.isValid()) {
|
|
|
|
curTreeItem->setStyleSheet(
|
|
|
|
"background: " +
|
|
|
|
color.name(QColor::HexArgb));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
auto changedColor = [=](const QColor &color) {
|
|
|
|
if (color.isValid()) {
|
|
|
|
ConfirmColor(ui->sources, color,
|
|
|
|
selectedItems);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
auto rejected = [=]() {
|
|
|
|
if (oldPreset == 1) {
|
|
|
|
curTreeItem->setStyleSheet(oldSheet);
|
|
|
|
curTreeItem->setProperty("bgColor", 0);
|
|
|
|
} else if (oldPreset == 0) {
|
|
|
|
curTreeItem->setStyleSheet(
|
|
|
|
"background: none");
|
|
|
|
curTreeItem->setProperty("bgColor", 0);
|
|
|
|
} else {
|
|
|
|
curTreeItem->setStyleSheet("");
|
|
|
|
curTreeItem->setProperty("bgColor",
|
|
|
|
oldPreset - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
curTreeItem->style()->unpolish(curTreeItem);
|
|
|
|
curTreeItem->style()->polish(curTreeItem);
|
|
|
|
};
|
|
|
|
|
|
|
|
QColorDialog::ColorDialogOptions options =
|
|
|
|
QColorDialog::ShowAlphaChannel;
|
|
|
|
|
|
|
|
const char *oldColor =
|
|
|
|
obs_data_get_string(curPrivData, "color");
|
|
|
|
const char *customColor = *oldColor != 0 ? oldColor
|
|
|
|
: "#55FF0000";
|
2020-03-28 23:15:25 +11:00
|
|
|
#ifndef _WIN32
|
2018-08-02 08:23:12 +12:00
|
|
|
options |= QColorDialog::DontUseNativeDialog;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
QColorDialog *colorDialog = new QColorDialog(this);
|
|
|
|
colorDialog->setOptions(options);
|
|
|
|
colorDialog->setCurrentColor(QColor(customColor));
|
|
|
|
connect(colorDialog, &QColorDialog::currentColorChanged,
|
|
|
|
liveChangeColor);
|
|
|
|
connect(colorDialog, &QColorDialog::colorSelected,
|
|
|
|
changedColor);
|
|
|
|
connect(colorDialog, &QColorDialog::rejected, rejected);
|
|
|
|
colorDialog->open();
|
|
|
|
|
|
|
|
obs_data_release(curPrivData);
|
|
|
|
} else {
|
|
|
|
for (int x = 0; x < selectedItems.count(); x++) {
|
|
|
|
SourceTreeItem *treeItem =
|
|
|
|
ui->sources->GetItemWidget(
|
|
|
|
selectedItems[x].row());
|
|
|
|
treeItem->setStyleSheet("background: none");
|
|
|
|
treeItem->setProperty("bgColor", preset);
|
|
|
|
treeItem->style()->unpolish(treeItem);
|
|
|
|
treeItem->style()->polish(treeItem);
|
|
|
|
|
|
|
|
OBSSceneItem sceneItem = ui->sources->Get(
|
|
|
|
selectedItems[x].row());
|
|
|
|
obs_data_t *privData =
|
|
|
|
obs_sceneitem_get_private_settings(
|
|
|
|
sceneItem);
|
|
|
|
obs_data_set_int(privData, "color-preset",
|
|
|
|
preset);
|
|
|
|
obs_data_set_string(privData, "color", "");
|
|
|
|
obs_data_release(privData);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SourceTreeItem *OBSBasic::GetItemWidgetFromSceneItem(obs_sceneitem_t *sceneItem)
|
|
|
|
{
|
|
|
|
int i = 0;
|
|
|
|
SourceTreeItem *treeItem = ui->sources->GetItemWidget(i);
|
|
|
|
OBSSceneItem item = ui->sources->Get(i);
|
|
|
|
int64_t id = obs_sceneitem_get_id(sceneItem);
|
|
|
|
while (treeItem && obs_sceneitem_get_id(item) != id) {
|
|
|
|
i++;
|
|
|
|
treeItem = ui->sources->GetItemWidget(i);
|
|
|
|
item = ui->sources->Get(i);
|
|
|
|
}
|
|
|
|
if (treeItem)
|
|
|
|
return treeItem;
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2017-05-06 12:13:56 -07:00
|
|
|
void OBSBasic::on_autoConfigure_triggered()
|
|
|
|
{
|
|
|
|
AutoConfig test(this);
|
|
|
|
test.setModal(true);
|
|
|
|
test.show();
|
|
|
|
test.exec();
|
|
|
|
}
|
2017-05-10 19:37:45 -07:00
|
|
|
|
|
|
|
void OBSBasic::on_stats_triggered()
|
|
|
|
{
|
2017-05-13 20:45:48 -07:00
|
|
|
if (!stats.isNull()) {
|
|
|
|
stats->show();
|
|
|
|
stats->raise();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-05-10 19:37:45 -07:00
|
|
|
OBSBasicStats *statsDlg;
|
|
|
|
statsDlg = new OBSBasicStats(nullptr);
|
|
|
|
statsDlg->show();
|
|
|
|
stats = statsDlg;
|
|
|
|
}
|
2018-08-02 08:23:12 +12:00
|
|
|
|
2018-08-18 01:13:20 -05:00
|
|
|
void OBSBasic::on_actionShowAbout_triggered()
|
|
|
|
{
|
|
|
|
if (about)
|
|
|
|
about->close();
|
|
|
|
|
|
|
|
about = new OBSAbout(this);
|
|
|
|
about->show();
|
|
|
|
|
|
|
|
about->setAttribute(Qt::WA_DeleteOnClose, true);
|
|
|
|
}
|
|
|
|
|
2018-08-29 17:45:59 -05:00
|
|
|
void OBSBasic::ResizeOutputSizeOfSource()
|
|
|
|
{
|
2019-07-15 04:50:42 -05:00
|
|
|
if (obs_video_active())
|
2018-08-29 17:45:59 -05:00
|
|
|
return;
|
|
|
|
|
|
|
|
QMessageBox resize_output(this);
|
|
|
|
resize_output.setText(QTStr("ResizeOutputSizeOfSource.Text") + "\n\n" +
|
|
|
|
QTStr("ResizeOutputSizeOfSource.Continue"));
|
|
|
|
QAbstractButton *Yes =
|
|
|
|
resize_output.addButton(QTStr("Yes"), QMessageBox::YesRole);
|
|
|
|
resize_output.addButton(QTStr("No"), QMessageBox::NoRole);
|
|
|
|
resize_output.setIcon(QMessageBox::Warning);
|
|
|
|
resize_output.setWindowTitle(QTStr("ResizeOutputSizeOfSource"));
|
|
|
|
resize_output.exec();
|
|
|
|
|
|
|
|
if (resize_output.clickedButton() != Yes)
|
|
|
|
return;
|
|
|
|
|
|
|
|
OBSSource source = obs_sceneitem_get_source(GetCurrentSceneItem());
|
|
|
|
|
|
|
|
int width = obs_source_get_width(source);
|
|
|
|
int height = obs_source_get_height(source);
|
|
|
|
|
|
|
|
config_set_uint(basicConfig, "Video", "BaseCX", width);
|
|
|
|
config_set_uint(basicConfig, "Video", "BaseCY", height);
|
|
|
|
config_set_uint(basicConfig, "Video", "OutputCX", width);
|
|
|
|
config_set_uint(basicConfig, "Video", "OutputCY", height);
|
|
|
|
|
|
|
|
ResetVideo();
|
|
|
|
on_actionFitToScreen_triggered();
|
|
|
|
}
|
2019-02-06 13:28:15 -08:00
|
|
|
|
2019-02-06 13:57:59 -08:00
|
|
|
QAction *OBSBasic::AddDockWidget(QDockWidget *dock)
|
|
|
|
{
|
|
|
|
QAction *action = ui->viewMenuDocks->addAction(dock->windowTitle());
|
|
|
|
action->setCheckable(true);
|
|
|
|
assignDockToggle(dock, action);
|
|
|
|
extraDocks.push_back(dock);
|
|
|
|
|
|
|
|
bool lock = ui->lockUI->isChecked();
|
|
|
|
QDockWidget::DockWidgetFeatures features =
|
|
|
|
lock ? QDockWidget::NoDockWidgetFeatures
|
2020-06-07 15:31:13 -07:00
|
|
|
: (QDockWidget::DockWidgetClosable |
|
|
|
|
QDockWidget::DockWidgetMovable |
|
|
|
|
QDockWidget::DockWidgetFloatable);
|
2019-02-06 13:57:59 -08:00
|
|
|
|
|
|
|
dock->setFeatures(features);
|
|
|
|
|
|
|
|
/* prune deleted docks */
|
|
|
|
for (int i = extraDocks.size() - 1; i >= 0; i--) {
|
|
|
|
if (!extraDocks[i]) {
|
|
|
|
extraDocks.removeAt(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return action;
|
|
|
|
}
|
|
|
|
|
2019-02-06 13:28:15 -08:00
|
|
|
OBSBasic *OBSBasic::Get()
|
|
|
|
{
|
|
|
|
return reinterpret_cast<OBSBasic *>(App()->GetMainWindow());
|
|
|
|
}
|
2019-02-18 19:28:26 -08:00
|
|
|
|
|
|
|
bool OBSBasic::StreamingActive()
|
|
|
|
{
|
|
|
|
if (!outputHandler)
|
|
|
|
return false;
|
|
|
|
return outputHandler->StreamingActive();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool OBSBasic::RecordingActive()
|
|
|
|
{
|
|
|
|
if (!outputHandler)
|
|
|
|
return false;
|
|
|
|
return outputHandler->RecordingActive();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool OBSBasic::ReplayBufferActive()
|
|
|
|
{
|
|
|
|
if (!outputHandler)
|
|
|
|
return false;
|
|
|
|
return outputHandler->ReplayBufferActive();
|
|
|
|
}
|
2019-04-07 03:34:21 -04:00
|
|
|
|
|
|
|
SceneRenameDelegate::SceneRenameDelegate(QObject *parent)
|
|
|
|
: QStyledItemDelegate(parent)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void SceneRenameDelegate::setEditorData(QWidget *editor,
|
|
|
|
const QModelIndex &index) const
|
|
|
|
{
|
|
|
|
QStyledItemDelegate::setEditorData(editor, index);
|
|
|
|
QLineEdit *lineEdit = qobject_cast<QLineEdit *>(editor);
|
|
|
|
if (lineEdit)
|
|
|
|
lineEdit->selectAll();
|
|
|
|
}
|
2019-04-23 15:56:30 +08:00
|
|
|
|
|
|
|
bool SceneRenameDelegate::eventFilter(QObject *editor, QEvent *event)
|
|
|
|
{
|
|
|
|
if (event->type() == QEvent::KeyPress) {
|
|
|
|
QKeyEvent *keyEvent = static_cast<QKeyEvent *>(event);
|
|
|
|
if (keyEvent->key() == Qt::Key_Escape) {
|
|
|
|
QLineEdit *lineEdit = qobject_cast<QLineEdit *>(editor);
|
|
|
|
if (lineEdit)
|
|
|
|
lineEdit->undo();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return QStyledItemDelegate::eventFilter(editor, event);
|
|
|
|
}
|
2019-05-05 20:00:06 -07:00
|
|
|
|
|
|
|
void OBSBasic::UpdatePatronJson(const QString &text, const QString &error)
|
|
|
|
{
|
|
|
|
if (!error.isEmpty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
patronJson = QT_TO_UTF8(text);
|
|
|
|
}
|
2019-07-07 15:47:29 -07:00
|
|
|
|
|
|
|
void OBSBasic::PauseRecording()
|
|
|
|
{
|
|
|
|
if (!pause || !outputHandler || !outputHandler->fileOutput)
|
|
|
|
return;
|
|
|
|
|
|
|
|
obs_output_t *output = outputHandler->fileOutput;
|
|
|
|
|
|
|
|
if (obs_output_pause(output, true)) {
|
2019-08-20 06:37:22 -05:00
|
|
|
pause->setAccessibleName(QTStr("Basic.Main.UnpauseRecording"));
|
|
|
|
pause->setToolTip(QTStr("Basic.Main.UnpauseRecording"));
|
2019-09-12 12:06:09 -07:00
|
|
|
pause->blockSignals(true);
|
2019-07-07 15:47:29 -07:00
|
|
|
pause->setChecked(true);
|
2019-09-12 12:06:09 -07:00
|
|
|
pause->blockSignals(false);
|
2019-10-02 19:44:47 -04:00
|
|
|
|
2020-04-14 20:16:33 +02:00
|
|
|
ui->statusbar->RecordingPaused();
|
|
|
|
|
2019-10-02 19:44:47 -04:00
|
|
|
if (trayIcon)
|
|
|
|
trayIcon->setIcon(QIcon(":/res/images/obs_paused.png"));
|
|
|
|
|
2019-07-07 15:47:29 -07:00
|
|
|
os_atomic_set_bool(&recording_paused, true);
|
|
|
|
|
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_RECORDING_PAUSED);
|
|
|
|
|
|
|
|
if (os_atomic_load_bool(&replaybuf_active))
|
|
|
|
ShowReplayBufferPauseWarning();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::UnpauseRecording()
|
|
|
|
{
|
|
|
|
if (!pause || !outputHandler || !outputHandler->fileOutput)
|
|
|
|
return;
|
|
|
|
|
|
|
|
obs_output_t *output = outputHandler->fileOutput;
|
|
|
|
|
|
|
|
if (obs_output_pause(output, false)) {
|
2019-08-20 06:37:22 -05:00
|
|
|
pause->setAccessibleName(QTStr("Basic.Main.PauseRecording"));
|
|
|
|
pause->setToolTip(QTStr("Basic.Main.PauseRecording"));
|
2019-09-12 12:06:09 -07:00
|
|
|
pause->blockSignals(true);
|
2019-07-07 15:47:29 -07:00
|
|
|
pause->setChecked(false);
|
2019-09-12 12:06:09 -07:00
|
|
|
pause->blockSignals(false);
|
2019-10-02 19:44:47 -04:00
|
|
|
|
2020-04-14 20:16:33 +02:00
|
|
|
ui->statusbar->RecordingUnpaused();
|
|
|
|
|
2019-10-02 19:44:47 -04:00
|
|
|
if (trayIcon)
|
|
|
|
trayIcon->setIcon(
|
|
|
|
QIcon(":/res/images/tray_active.png"));
|
|
|
|
|
2019-07-07 15:47:29 -07:00
|
|
|
os_atomic_set_bool(&recording_paused, false);
|
|
|
|
|
|
|
|
if (api)
|
|
|
|
api->on_event(OBS_FRONTEND_EVENT_RECORDING_UNPAUSED);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::PauseToggled()
|
|
|
|
{
|
|
|
|
if (!pause || !outputHandler || !outputHandler->fileOutput)
|
|
|
|
return;
|
|
|
|
|
|
|
|
obs_output_t *output = outputHandler->fileOutput;
|
|
|
|
bool enable = !obs_output_paused(output);
|
|
|
|
|
2019-08-20 06:37:48 -05:00
|
|
|
if (enable)
|
|
|
|
PauseRecording();
|
|
|
|
else
|
|
|
|
UnpauseRecording();
|
2019-07-07 15:47:29 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::UpdatePause(bool activate)
|
|
|
|
{
|
|
|
|
if (!activate || !outputHandler || !outputHandler->RecordingActive()) {
|
|
|
|
pause.reset();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *mode = config_get_string(basicConfig, "Output", "Mode");
|
|
|
|
bool adv = astrcmpi(mode, "Advanced") == 0;
|
|
|
|
bool shared;
|
|
|
|
|
|
|
|
if (adv) {
|
|
|
|
const char *recType =
|
|
|
|
config_get_string(basicConfig, "AdvOut", "RecType");
|
|
|
|
|
|
|
|
if (astrcmpi(recType, "FFmpeg") == 0) {
|
|
|
|
shared = config_get_bool(basicConfig, "AdvOut",
|
|
|
|
"FFOutputToFile");
|
|
|
|
} else {
|
|
|
|
const char *recordEncoder = config_get_string(
|
|
|
|
basicConfig, "AdvOut", "RecEncoder");
|
|
|
|
shared = astrcmpi(recordEncoder, "none") == 0;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
const char *quality = config_get_string(
|
|
|
|
basicConfig, "SimpleOutput", "RecQuality");
|
|
|
|
shared = strcmp(quality, "Stream") == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!shared) {
|
|
|
|
pause.reset(new QPushButton());
|
|
|
|
pause->setAccessibleName(QTStr("Basic.Main.PauseRecording"));
|
|
|
|
pause->setToolTip(QTStr("Basic.Main.PauseRecording"));
|
|
|
|
pause->setCheckable(true);
|
|
|
|
pause->setChecked(false);
|
|
|
|
pause->setProperty("themeID",
|
|
|
|
QVariant(QStringLiteral("pauseIconSmall")));
|
2020-06-24 09:22:58 -07:00
|
|
|
|
|
|
|
QSizePolicy sp;
|
|
|
|
sp.setHeightForWidth(true);
|
|
|
|
pause->setSizePolicy(sp);
|
|
|
|
|
2019-09-12 13:55:31 -07:00
|
|
|
connect(pause.data(), &QAbstractButton::clicked, this,
|
2019-07-07 15:47:29 -07:00
|
|
|
&OBSBasic::PauseToggled);
|
|
|
|
ui->recordingLayout->addWidget(pause.data());
|
|
|
|
} else {
|
|
|
|
pause.reset();
|
|
|
|
}
|
|
|
|
}
|
2019-06-12 04:56:00 -05:00
|
|
|
|
2019-12-12 17:07:26 -08:00
|
|
|
void OBSBasic::UpdateReplayBuffer(bool activate)
|
|
|
|
{
|
|
|
|
if (!activate || !outputHandler ||
|
|
|
|
!outputHandler->ReplayBufferActive()) {
|
|
|
|
replay.reset();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
replay.reset(new QPushButton());
|
|
|
|
replay->setAccessibleName(QTStr("Basic.Main.SaveReplay"));
|
|
|
|
replay->setToolTip(QTStr("Basic.Main.SaveReplay"));
|
|
|
|
replay->setCheckable(true);
|
|
|
|
replay->setChecked(false);
|
|
|
|
replay->setProperty("themeID",
|
|
|
|
QVariant(QStringLiteral("replayIconSmall")));
|
2020-06-24 09:22:58 -07:00
|
|
|
|
|
|
|
QSizePolicy sp;
|
|
|
|
sp.setHeightForWidth(true);
|
|
|
|
replay->setSizePolicy(sp);
|
|
|
|
|
2019-12-12 17:07:26 -08:00
|
|
|
connect(replay.data(), &QAbstractButton::clicked, this,
|
|
|
|
&OBSBasic::ReplayBufferSave);
|
|
|
|
replayLayout->addWidget(replay.data());
|
|
|
|
}
|
|
|
|
|
2019-06-12 04:56:00 -05:00
|
|
|
#define MBYTE (1024ULL * 1024ULL)
|
|
|
|
#define MBYTES_LEFT_STOP_REC 50ULL
|
|
|
|
#define MAX_BYTES_LEFT (MBYTES_LEFT_STOP_REC * MBYTE)
|
|
|
|
|
2019-09-27 15:04:07 +02:00
|
|
|
const char *OBSBasic::GetCurrentOutputPath()
|
|
|
|
{
|
|
|
|
const char *path = nullptr;
|
|
|
|
const char *mode = config_get_string(Config(), "Output", "Mode");
|
|
|
|
|
|
|
|
if (strcmp(mode, "Advanced") == 0) {
|
|
|
|
const char *advanced_mode =
|
|
|
|
config_get_string(Config(), "AdvOut", "RecType");
|
|
|
|
|
|
|
|
if (strcmp(advanced_mode, "FFmpeg") == 0) {
|
|
|
|
path = config_get_string(Config(), "AdvOut",
|
|
|
|
"FFFilePath");
|
|
|
|
} else {
|
|
|
|
path = config_get_string(Config(), "AdvOut",
|
|
|
|
"RecFilePath");
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
path = config_get_string(Config(), "SimpleOutput", "FilePath");
|
|
|
|
}
|
|
|
|
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
|
2019-12-27 17:12:31 +08:00
|
|
|
void OBSBasic::OutputPathInvalidMessage()
|
|
|
|
{
|
|
|
|
blog(LOG_ERROR, "Recording stopped because of bad output path");
|
|
|
|
|
|
|
|
OBSMessageBox::critical(this, QTStr("Output.BadPath.Title"),
|
|
|
|
QTStr("Output.BadPath.Text"));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool OBSBasic::OutputPathValid()
|
|
|
|
{
|
|
|
|
const char *path = GetCurrentOutputPath();
|
|
|
|
return path && *path && QDir(path).exists();
|
|
|
|
}
|
|
|
|
|
2019-06-12 04:56:00 -05:00
|
|
|
void OBSBasic::DiskSpaceMessage()
|
|
|
|
{
|
|
|
|
blog(LOG_ERROR, "Recording stopped because of low disk space");
|
|
|
|
|
|
|
|
OBSMessageBox::critical(this, QTStr("Output.RecordNoSpace.Title"),
|
|
|
|
QTStr("Output.RecordNoSpace.Msg"));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool OBSBasic::LowDiskSpace()
|
|
|
|
{
|
2019-09-27 15:04:07 +02:00
|
|
|
const char *path;
|
|
|
|
|
|
|
|
path = GetCurrentOutputPath();
|
|
|
|
if (!path)
|
|
|
|
return false;
|
2019-06-12 04:56:00 -05:00
|
|
|
|
|
|
|
uint64_t num_bytes = os_get_free_disk_space(path);
|
|
|
|
|
|
|
|
if (num_bytes < (MAX_BYTES_LEFT))
|
|
|
|
return true;
|
|
|
|
else
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::CheckDiskSpaceRemaining()
|
|
|
|
{
|
|
|
|
if (LowDiskSpace()) {
|
|
|
|
StopRecording();
|
|
|
|
StopReplayBuffer();
|
|
|
|
|
|
|
|
DiskSpaceMessage();
|
|
|
|
}
|
|
|
|
}
|
2019-10-10 20:12:20 -05:00
|
|
|
|
2020-08-21 03:39:42 -05:00
|
|
|
void OBSBasic::ScenesReordered()
|
2019-10-10 20:12:20 -05:00
|
|
|
{
|
|
|
|
OBSProjector::UpdateMultiviewProjectors();
|
|
|
|
}
|
2019-10-15 23:20:35 -05:00
|
|
|
|
|
|
|
void OBSBasic::ResetStatsHotkey()
|
|
|
|
{
|
|
|
|
QList<OBSBasicStats *> list = findChildren<OBSBasicStats *>();
|
|
|
|
|
|
|
|
foreach(OBSBasicStats * s, list) s->Reset();
|
|
|
|
}
|
2020-02-14 23:30:34 +11:00
|
|
|
|
|
|
|
void OBSBasic::on_customContextMenuRequested(const QPoint &pos)
|
|
|
|
{
|
|
|
|
QWidget *widget = childAt(pos);
|
|
|
|
const char *className = nullptr;
|
|
|
|
if (widget != nullptr)
|
|
|
|
className = widget->metaObject()->className();
|
|
|
|
|
|
|
|
if (!className || strstr(className, "Dock") != nullptr)
|
|
|
|
ui->viewMenuDocks->exec(mapToGlobal(pos));
|
|
|
|
}
|
2020-06-13 21:00:55 -05:00
|
|
|
|
|
|
|
void OBSBasic::UpdateProjectorHideCursor()
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < projectors.size(); i++)
|
|
|
|
projectors[i]->SetHideCursor();
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::UpdateProjectorAlwaysOnTop(bool top)
|
|
|
|
{
|
|
|
|
for (size_t i = 0; i < projectors.size(); i++)
|
|
|
|
SetAlwaysOnTop(projectors[i], top);
|
|
|
|
}
|
2018-08-23 20:43:44 -05:00
|
|
|
|
|
|
|
void OBSBasic::on_sourcePropertiesButton_clicked()
|
|
|
|
{
|
|
|
|
on_actionSourceProperties_triggered();
|
|
|
|
}
|
|
|
|
|
|
|
|
void OBSBasic::on_sourceFiltersButton_clicked()
|
|
|
|
{
|
|
|
|
OpenFilters();
|
|
|
|
}
|