mirror of
https://github.com/WarmUpTill/SceneSwitcher.git
synced 2026-03-22 09:54:54 -05:00
* Add variable support to process condition * Display unresolved variable value in Window action * Add tooltips to indicate that variables are supported
439 lines
12 KiB
C++
439 lines
12 KiB
C++
#include "macro-condition-window.hpp"
|
|
#include "layout-helpers.hpp"
|
|
#include "plugin-state-helpers.hpp"
|
|
#include "platform-funcs.hpp"
|
|
|
|
#include <regex>
|
|
|
|
namespace advss {
|
|
|
|
const std::string MacroConditionWindow::id = "window";
|
|
|
|
bool MacroConditionWindow::_registered = MacroConditionFactory::Register(
|
|
MacroConditionWindow::id,
|
|
{MacroConditionWindow::Create, MacroConditionWindowEdit::Create,
|
|
"AdvSceneSwitcher.condition.window"});
|
|
|
|
static bool windowContainsText(const std::string &window,
|
|
const std::string &matchText,
|
|
const RegexConfig ®ex)
|
|
{
|
|
auto text = GetTextInWindow(window);
|
|
if (!text.has_value()) {
|
|
return false;
|
|
}
|
|
|
|
if (regex.Enabled()) {
|
|
return regex.Matches(*text, matchText);
|
|
}
|
|
return text == matchText;
|
|
}
|
|
|
|
bool MacroConditionWindow::WindowMatchesRequirements(
|
|
const std::string &window) const
|
|
{
|
|
const bool focusCheckOK =
|
|
(!_focus || window == ForegroundWindowTitle());
|
|
if (!focusCheckOK) {
|
|
return false;
|
|
}
|
|
const bool fullscreenCheckOK = (!_fullscreen || IsFullscreen(window));
|
|
if (!fullscreenCheckOK) {
|
|
return false;
|
|
}
|
|
const bool maxCheckOK = (!_maximized || IsMaximized(window));
|
|
if (!maxCheckOK) {
|
|
return false;
|
|
}
|
|
const bool textCheckOK =
|
|
(!_checkText || windowContainsText(window, _text, _textRegex));
|
|
if (!textCheckOK) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool MacroConditionWindow::WindowMatches(
|
|
const std::vector<std::string> &windowList)
|
|
{
|
|
bool match = !_checkTitle ||
|
|
std::find(windowList.begin(), windowList.end(),
|
|
std::string(_window)) != windowList.end();
|
|
match = match && WindowMatchesRequirements(_window);
|
|
SetVariableValueBasedOnMatch(_window);
|
|
return match;
|
|
}
|
|
|
|
#ifdef _WIN32
|
|
std::string GetWindowClassByWindowTitle(const std::string &window);
|
|
#endif
|
|
|
|
bool MacroConditionWindow::WindowRegexMatches(
|
|
const std::vector<std::string> &windowList)
|
|
{
|
|
// No need to test if checking for window title is required as if the
|
|
// user has disabled window title matching the option will always be
|
|
// enabled in the backend and use the regular expression ".*".
|
|
|
|
for (const auto &window : windowList) {
|
|
if (_windowRegex.Matches(window, _window) &&
|
|
WindowMatchesRequirements(window)) {
|
|
SetVariableValueBasedOnMatch(window);
|
|
return true;
|
|
}
|
|
}
|
|
SetVariableValueBasedOnMatch("");
|
|
return false;
|
|
}
|
|
|
|
void MacroConditionWindow::SetVariableValueBasedOnMatch(
|
|
const std::string &matchWindow)
|
|
{
|
|
SetTempVarValue("window", matchWindow);
|
|
#ifdef _WIN32
|
|
SetTempVarValue("windowClass",
|
|
GetWindowClassByWindowTitle(matchWindow));
|
|
#endif
|
|
if (!IsReferencedInVars()) {
|
|
return;
|
|
}
|
|
if (_checkText) {
|
|
auto text = GetTextInWindow(matchWindow);
|
|
SetVariableValue(text.value_or(""));
|
|
} else {
|
|
SetVariableValue(ForegroundWindowTitle());
|
|
}
|
|
}
|
|
|
|
static bool foregroundWindowChanged()
|
|
{
|
|
return ForegroundWindowTitle() != PreviousForegroundWindowTitle();
|
|
}
|
|
|
|
bool MacroConditionWindow::CheckCondition()
|
|
{
|
|
std::vector<std::string> windowList;
|
|
GetWindowList(windowList);
|
|
bool match = false;
|
|
if (_windowRegex.Enabled()) {
|
|
match = WindowRegexMatches(windowList);
|
|
} else {
|
|
match = WindowMatches(windowList);
|
|
}
|
|
match = match && (!_windowFocusChanged || foregroundWindowChanged());
|
|
return match;
|
|
}
|
|
|
|
bool MacroConditionWindow::Save(obs_data_t *obj) const
|
|
{
|
|
MacroCondition::Save(obj);
|
|
obs_data_set_bool(obj, "checkTitle", _checkTitle);
|
|
_window.Save(obj, "window");
|
|
_windowRegex.Save(obj, "windowRegexConfig");
|
|
obs_data_set_bool(obj, "fullscreen", _fullscreen);
|
|
obs_data_set_bool(obj, "maximized", _maximized);
|
|
obs_data_set_bool(obj, "focus", _focus);
|
|
obs_data_set_bool(obj, "windowFocusChanged", _windowFocusChanged);
|
|
obs_data_set_bool(obj, "checkWindowText", _checkText);
|
|
_text.Save(obj, "text");
|
|
_textRegex.Save(obj, "textRegexConfig");
|
|
obs_data_set_int(obj, "version", 1);
|
|
return true;
|
|
}
|
|
|
|
bool MacroConditionWindow::Load(obs_data_t *obj)
|
|
{
|
|
MacroCondition::Load(obj);
|
|
if (!obs_data_has_user_value(obj, "version")) {
|
|
_checkTitle = true;
|
|
_windowRegex.CreateBackwardsCompatibleRegex(true, true);
|
|
} else {
|
|
_checkTitle = obs_data_get_bool(obj, "checkTitle");
|
|
_windowRegex.Load(obj, "windowRegexConfig");
|
|
}
|
|
_window.Load(obj, "window");
|
|
_fullscreen = obs_data_get_bool(obj, "fullscreen");
|
|
_maximized = obs_data_get_bool(obj, "maximized");
|
|
_focus = obs_data_get_bool(obj, "focus");
|
|
_windowFocusChanged = obs_data_get_bool(obj, "windowFocusChanged");
|
|
#ifdef _WIN32
|
|
_checkText = obs_data_get_bool(obj, "checkWindowText");
|
|
#else
|
|
_checkText = false;
|
|
#endif
|
|
_text.Load(obj, "text");
|
|
_textRegex.Load(obj, "textRegexConfig");
|
|
return true;
|
|
}
|
|
|
|
std::string MacroConditionWindow::GetShortDesc() const
|
|
{
|
|
return _window;
|
|
}
|
|
|
|
void MacroConditionWindow::SetupTempVars()
|
|
{
|
|
MacroCondition::SetupTempVars();
|
|
AddTempvar(
|
|
"window",
|
|
obs_module_text("AdvSceneSwitcher.tempVar.window.window"),
|
|
obs_module_text(
|
|
"AdvSceneSwitcher.tempVar.window.window.description"));
|
|
#ifdef _WIN32
|
|
AddTempvar(
|
|
"windowClass",
|
|
obs_module_text("AdvSceneSwitcher.tempVar.window.windowClass"),
|
|
obs_module_text(
|
|
"AdvSceneSwitcher.tempVar.window.windowClass.description"));
|
|
#endif
|
|
}
|
|
|
|
MacroConditionWindowEdit::MacroConditionWindowEdit(
|
|
QWidget *parent, std::shared_ptr<MacroConditionWindow> entryData)
|
|
: QWidget(parent),
|
|
_windowSelection(new WindowSelectionWidget(this)),
|
|
_windowRegex(new RegexConfigWidget(this)),
|
|
_checkTitle(new QCheckBox()),
|
|
_fullscreen(new QCheckBox()),
|
|
_maximized(new QCheckBox()),
|
|
_focused(new QCheckBox()),
|
|
_windowFocusChanged(new QCheckBox()),
|
|
_checkText(new QCheckBox()),
|
|
_text(new VariableTextEdit(this)),
|
|
_textRegex(new RegexConfigWidget(this)),
|
|
_focusWindow(new QLabel()),
|
|
_currentFocusLayout(new QHBoxLayout())
|
|
{
|
|
_checkText->setToolTip(obs_module_text(
|
|
"AdvSceneSwitcher.condition.window.entry.text.note"));
|
|
_text->setToolTip(obs_module_text(
|
|
"AdvSceneSwitcher.condition.window.entry.text.note"));
|
|
|
|
_windowSelection->setToolTip(
|
|
obs_module_text("AdvSceneSwitcher.tooltip.availableVariables"));
|
|
|
|
QWidget::connect(_windowSelection,
|
|
SIGNAL(currentTextChanged(const QString &)), this,
|
|
SLOT(WindowChanged(const QString &)));
|
|
QWidget::connect(_windowRegex,
|
|
SIGNAL(RegexConfigChanged(const RegexConfig &)), this,
|
|
SLOT(WindowRegexChanged(const RegexConfig &)));
|
|
QWidget::connect(_checkTitle, SIGNAL(stateChanged(int)), this,
|
|
SLOT(CheckTitleChanged(int)));
|
|
QWidget::connect(_fullscreen, SIGNAL(stateChanged(int)), this,
|
|
SLOT(FullscreenChanged(int)));
|
|
QWidget::connect(_maximized, SIGNAL(stateChanged(int)), this,
|
|
SLOT(MaximizedChanged(int)));
|
|
QWidget::connect(_focused, SIGNAL(stateChanged(int)), this,
|
|
SLOT(FocusedChanged(int)));
|
|
QWidget::connect(_windowFocusChanged, SIGNAL(stateChanged(int)), this,
|
|
SLOT(WindowFocusChanged(int)));
|
|
QWidget::connect(_checkText, SIGNAL(stateChanged(int)), this,
|
|
SLOT(CheckTextChanged(int)));
|
|
QWidget::connect(_text, SIGNAL(textChanged()), this,
|
|
SLOT(WindowTextChanged()));
|
|
QWidget::connect(_textRegex,
|
|
SIGNAL(RegexConfigChanged(const RegexConfig &)), this,
|
|
SLOT(TextRegexChanged(const RegexConfig &)));
|
|
QWidget::connect(&_timer, SIGNAL(timeout()), this,
|
|
SLOT(UpdateFocusWindow()));
|
|
|
|
const std::unordered_map<std::string, QWidget *> widgetPlaceholders = {
|
|
{"{{windows}}", _windowSelection},
|
|
{"{{windowRegex}}", _windowRegex},
|
|
{"{{checkTitle}}", _checkTitle},
|
|
{"{{fullscreen}}", _fullscreen},
|
|
{"{{maximized}}", _maximized},
|
|
{"{{focused}}", _focused},
|
|
{"{{windowFocusChanged}}", _windowFocusChanged},
|
|
{"{{focusWindow}}", _focusWindow},
|
|
{"{{checkText}}", _checkText},
|
|
{"{{windowText}}", _text},
|
|
{"{{textRegex}}", _textRegex},
|
|
};
|
|
|
|
auto titleLayout = new QHBoxLayout;
|
|
titleLayout->setContentsMargins(0, 0, 0, 0);
|
|
PlaceWidgets(obs_module_text(
|
|
"AdvSceneSwitcher.condition.window.entry.window"),
|
|
titleLayout, widgetPlaceholders);
|
|
auto fullscreenLayout = new QHBoxLayout;
|
|
fullscreenLayout->setContentsMargins(0, 0, 0, 0);
|
|
PlaceWidgets(
|
|
obs_module_text(
|
|
"AdvSceneSwitcher.condition.window.entry.fullscreen"),
|
|
fullscreenLayout, widgetPlaceholders);
|
|
auto maximizedLayout = new QHBoxLayout;
|
|
maximizedLayout->setContentsMargins(0, 0, 0, 0);
|
|
PlaceWidgets(
|
|
obs_module_text(
|
|
"AdvSceneSwitcher.condition.window.entry.maximized"),
|
|
maximizedLayout, widgetPlaceholders);
|
|
auto focusedLayout = new QHBoxLayout;
|
|
focusedLayout->setContentsMargins(0, 0, 0, 0);
|
|
PlaceWidgets(obs_module_text(
|
|
"AdvSceneSwitcher.condition.window.entry.focused"),
|
|
focusedLayout, widgetPlaceholders);
|
|
auto focusChangedLayout = new QHBoxLayout;
|
|
focusChangedLayout->setContentsMargins(0, 0, 0, 0);
|
|
PlaceWidgets(
|
|
obs_module_text(
|
|
"AdvSceneSwitcher.condition.window.entry.focusedChange"),
|
|
focusChangedLayout, widgetPlaceholders);
|
|
auto textLayout = new QHBoxLayout;
|
|
textLayout->setContentsMargins(0, 0, 0, 0);
|
|
PlaceWidgets(
|
|
obs_module_text("AdvSceneSwitcher.condition.window.entry.text"),
|
|
textLayout, widgetPlaceholders);
|
|
_text->setSizePolicy(QSizePolicy::MinimumExpanding,
|
|
QSizePolicy::Preferred);
|
|
textLayout->setStretchFactor(_text, 10);
|
|
_currentFocusLayout->setContentsMargins(0, 0, 0, 0);
|
|
PlaceWidgets(
|
|
obs_module_text(
|
|
"AdvSceneSwitcher.condition.window.entry.currentFocus"),
|
|
_currentFocusLayout, widgetPlaceholders);
|
|
|
|
auto mainLayout = new QVBoxLayout;
|
|
mainLayout->addLayout(titleLayout);
|
|
mainLayout->addLayout(fullscreenLayout);
|
|
mainLayout->addLayout(maximizedLayout);
|
|
mainLayout->addLayout(focusedLayout);
|
|
mainLayout->addLayout(focusChangedLayout);
|
|
mainLayout->addLayout(textLayout);
|
|
#ifndef _WIN32
|
|
SetLayoutVisible(textLayout, false);
|
|
#endif
|
|
mainLayout->addLayout(_currentFocusLayout);
|
|
setLayout(mainLayout);
|
|
|
|
_entryData = entryData;
|
|
UpdateEntryData();
|
|
_loading = false;
|
|
|
|
_timer.start(1000);
|
|
}
|
|
|
|
void MacroConditionWindowEdit::WindowChanged(const QString &text)
|
|
{
|
|
GUARD_LOADING_AND_LOCK();
|
|
_entryData->_window = text.toStdString();
|
|
emit HeaderInfoChanged(
|
|
QString::fromStdString(_entryData->GetShortDesc()));
|
|
}
|
|
|
|
void MacroConditionWindowEdit::WindowRegexChanged(const RegexConfig &conf)
|
|
{
|
|
GUARD_LOADING_AND_LOCK();
|
|
_entryData->_windowRegex = conf;
|
|
adjustSize();
|
|
updateGeometry();
|
|
}
|
|
|
|
void MacroConditionWindowEdit::CheckTitleChanged(int state)
|
|
{
|
|
GUARD_LOADING_AND_LOCK();
|
|
const QSignalBlocker b1(_windowSelection);
|
|
const QSignalBlocker b2(_windowRegex);
|
|
if (!state) {
|
|
_entryData->_window = ".*";
|
|
_entryData->_windowRegex.SetEnabled(true);
|
|
_windowSelection->setCurrentText(".*");
|
|
_windowRegex->EnableChanged(true);
|
|
}
|
|
_entryData->_checkTitle = state;
|
|
SetWidgetVisibility();
|
|
}
|
|
|
|
void MacroConditionWindowEdit::CheckTextChanged(int state)
|
|
{
|
|
GUARD_LOADING_AND_LOCK();
|
|
_entryData->_checkText = state;
|
|
SetWidgetVisibility();
|
|
}
|
|
|
|
void MacroConditionWindowEdit::WindowTextChanged()
|
|
{
|
|
GUARD_LOADING_AND_LOCK();
|
|
_entryData->_text = _text->toPlainText().toStdString();
|
|
adjustSize();
|
|
updateGeometry();
|
|
}
|
|
|
|
void MacroConditionWindowEdit::TextRegexChanged(const RegexConfig &conf)
|
|
{
|
|
GUARD_LOADING_AND_LOCK();
|
|
_entryData->_textRegex = conf;
|
|
adjustSize();
|
|
updateGeometry();
|
|
}
|
|
|
|
void MacroConditionWindowEdit::FullscreenChanged(int state)
|
|
{
|
|
GUARD_LOADING_AND_LOCK();
|
|
_entryData->_fullscreen = state;
|
|
}
|
|
|
|
void MacroConditionWindowEdit::MaximizedChanged(int state)
|
|
{
|
|
GUARD_LOADING_AND_LOCK();
|
|
_entryData->_maximized = state;
|
|
}
|
|
|
|
void MacroConditionWindowEdit::FocusedChanged(int state)
|
|
{
|
|
GUARD_LOADING_AND_LOCK();
|
|
_entryData->_focus = state;
|
|
SetWidgetVisibility();
|
|
}
|
|
|
|
void MacroConditionWindowEdit::WindowFocusChanged(int state)
|
|
{
|
|
GUARD_LOADING_AND_LOCK();
|
|
_entryData->_windowFocusChanged = state;
|
|
SetWidgetVisibility();
|
|
}
|
|
|
|
void MacroConditionWindowEdit::UpdateFocusWindow()
|
|
{
|
|
_focusWindow->setText(QString::fromStdString(ForegroundWindowTitle()));
|
|
}
|
|
|
|
void MacroConditionWindowEdit::SetWidgetVisibility()
|
|
{
|
|
if (!_entryData) {
|
|
return;
|
|
}
|
|
SetLayoutVisible(_currentFocusLayout,
|
|
_entryData->_focus || _entryData->_windowFocusChanged);
|
|
_windowSelection->setVisible(_entryData->_checkTitle);
|
|
_windowRegex->setVisible(_entryData->_checkTitle);
|
|
_textRegex->setVisible(_entryData->_checkText);
|
|
_text->setVisible(_entryData->_checkText);
|
|
adjustSize();
|
|
updateGeometry();
|
|
}
|
|
|
|
void MacroConditionWindowEdit::UpdateEntryData()
|
|
{
|
|
if (!_entryData) {
|
|
return;
|
|
}
|
|
|
|
_windowSelection->setCurrentText(
|
|
_entryData->_window.UnresolvedValue().c_str());
|
|
_windowRegex->SetRegexConfig(_entryData->_windowRegex);
|
|
_checkTitle->setChecked(_entryData->_checkTitle);
|
|
_fullscreen->setChecked(_entryData->_fullscreen);
|
|
_maximized->setChecked(_entryData->_maximized);
|
|
_focused->setChecked(_entryData->_focus);
|
|
_windowFocusChanged->setChecked(_entryData->_windowFocusChanged);
|
|
_checkText->setChecked(_entryData->_checkText);
|
|
_text->setPlainText(_entryData->_text);
|
|
_textRegex->SetRegexConfig(_entryData->_textRegex);
|
|
SetWidgetVisibility();
|
|
}
|
|
|
|
} // namespace advss
|