Applies transformations correctly to copy/pasted sources when copy and pasting multiple. Previously the transformation/crop state was stored in a single variable on copy so it would only apply the transformations to one of the selected sources on paste instead of what the state was when copied.
409 lines
11 KiB
C++
409 lines
11 KiB
C++
/******************************************************************************
|
|
Copyright (C) 2014 by Hugh Bailey <obs.jim@gmail.com>
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
******************************************************************************/
|
|
|
|
#include <QMessageBox>
|
|
#include "window-basic-main.hpp"
|
|
#include "window-basic-source-select.hpp"
|
|
#include "qt-wrappers.hpp"
|
|
#include "obs-app.hpp"
|
|
|
|
struct AddSourceData {
|
|
obs_source_t *source;
|
|
bool visible;
|
|
obs_transform_info *transform = nullptr;
|
|
obs_sceneitem_crop *crop = nullptr;
|
|
};
|
|
|
|
bool OBSBasicSourceSelect::EnumSources(void *data, obs_source_t *source)
|
|
{
|
|
if (obs_source_is_hidden(source))
|
|
return false;
|
|
|
|
OBSBasicSourceSelect *window =
|
|
static_cast<OBSBasicSourceSelect *>(data);
|
|
const char *name = obs_source_get_name(source);
|
|
const char *id = obs_source_get_unversioned_id(source);
|
|
|
|
if (strcmp(id, window->id) == 0)
|
|
window->ui->sourceList->addItem(QT_UTF8(name));
|
|
|
|
return true;
|
|
}
|
|
|
|
bool OBSBasicSourceSelect::EnumGroups(void *data, obs_source_t *source)
|
|
{
|
|
OBSBasicSourceSelect *window =
|
|
static_cast<OBSBasicSourceSelect *>(data);
|
|
const char *name = obs_source_get_name(source);
|
|
const char *id = obs_source_get_unversioned_id(source);
|
|
|
|
if (strcmp(id, window->id) == 0) {
|
|
OBSBasic *main =
|
|
reinterpret_cast<OBSBasic *>(App()->GetMainWindow());
|
|
OBSScene scene = main->GetCurrentScene();
|
|
|
|
obs_sceneitem_t *existing = obs_scene_get_group(scene, name);
|
|
if (!existing)
|
|
window->ui->sourceList->addItem(QT_UTF8(name));
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void OBSBasicSourceSelect::OBSSourceAdded(void *data, calldata_t *calldata)
|
|
{
|
|
OBSBasicSourceSelect *window =
|
|
static_cast<OBSBasicSourceSelect *>(data);
|
|
obs_source_t *source = (obs_source_t *)calldata_ptr(calldata, "source");
|
|
|
|
QMetaObject::invokeMethod(window, "SourceAdded",
|
|
Q_ARG(OBSSource, source));
|
|
}
|
|
|
|
void OBSBasicSourceSelect::OBSSourceRemoved(void *data, calldata_t *calldata)
|
|
{
|
|
OBSBasicSourceSelect *window =
|
|
static_cast<OBSBasicSourceSelect *>(data);
|
|
obs_source_t *source = (obs_source_t *)calldata_ptr(calldata, "source");
|
|
|
|
QMetaObject::invokeMethod(window, "SourceRemoved",
|
|
Q_ARG(OBSSource, source));
|
|
}
|
|
|
|
void OBSBasicSourceSelect::SourceAdded(OBSSource source)
|
|
{
|
|
const char *name = obs_source_get_name(source);
|
|
const char *sourceId = obs_source_get_unversioned_id(source);
|
|
|
|
if (strcmp(sourceId, id) != 0)
|
|
return;
|
|
|
|
ui->sourceList->addItem(name);
|
|
}
|
|
|
|
void OBSBasicSourceSelect::SourceRemoved(OBSSource source)
|
|
{
|
|
const char *name = obs_source_get_name(source);
|
|
const char *sourceId = obs_source_get_unversioned_id(source);
|
|
|
|
if (strcmp(sourceId, id) != 0)
|
|
return;
|
|
|
|
QList<QListWidgetItem *> items =
|
|
ui->sourceList->findItems(name, Qt::MatchFixedString);
|
|
|
|
if (!items.count())
|
|
return;
|
|
|
|
delete items[0];
|
|
}
|
|
|
|
static void AddSource(void *_data, obs_scene_t *scene)
|
|
{
|
|
AddSourceData *data = (AddSourceData *)_data;
|
|
obs_sceneitem_t *sceneitem;
|
|
|
|
sceneitem = obs_scene_add(scene, data->source);
|
|
|
|
if (data->transform != nullptr)
|
|
obs_sceneitem_set_info(sceneitem, data->transform);
|
|
|
|
if (data->crop != nullptr)
|
|
obs_sceneitem_set_crop(sceneitem, data->crop);
|
|
|
|
obs_sceneitem_set_visible(sceneitem, data->visible);
|
|
}
|
|
|
|
static char *get_new_source_name(const char *name)
|
|
{
|
|
struct dstr new_name = {0};
|
|
int inc = 0;
|
|
|
|
dstr_copy(&new_name, name);
|
|
|
|
for (;;) {
|
|
obs_source_t *existing_source =
|
|
obs_get_source_by_name(new_name.array);
|
|
if (!existing_source)
|
|
break;
|
|
|
|
obs_source_release(existing_source);
|
|
|
|
dstr_printf(&new_name, "%s %d", name, ++inc + 1);
|
|
}
|
|
|
|
return new_name.array;
|
|
}
|
|
|
|
static void AddExisting(OBSSource source, bool visible, bool duplicate,
|
|
obs_transform_info *transform, obs_sceneitem_crop *crop)
|
|
{
|
|
OBSBasic *main = reinterpret_cast<OBSBasic *>(App()->GetMainWindow());
|
|
OBSScene scene = main->GetCurrentScene();
|
|
if (!scene)
|
|
return;
|
|
|
|
if (duplicate) {
|
|
OBSSource from = source;
|
|
char *new_name =
|
|
get_new_source_name(obs_source_get_name(source));
|
|
source = obs_source_duplicate(from, new_name, false);
|
|
obs_source_release(source);
|
|
bfree(new_name);
|
|
|
|
if (!source)
|
|
return;
|
|
}
|
|
|
|
AddSourceData data;
|
|
data.source = source;
|
|
data.visible = visible;
|
|
data.transform = transform;
|
|
data.crop = crop;
|
|
|
|
obs_enter_graphics();
|
|
obs_scene_atomic_update(scene, AddSource, &data);
|
|
obs_leave_graphics();
|
|
}
|
|
|
|
static void AddExisting(const char *name, bool visible, bool duplicate,
|
|
obs_transform_info *transform, obs_sceneitem_crop *crop)
|
|
{
|
|
obs_source_t *source = obs_get_source_by_name(name);
|
|
if (source) {
|
|
AddExisting(source, visible, duplicate, transform, crop);
|
|
obs_source_release(source);
|
|
}
|
|
}
|
|
|
|
bool AddNew(QWidget *parent, const char *id, const char *name,
|
|
const bool visible, OBSSource &newSource)
|
|
{
|
|
OBSBasic *main = reinterpret_cast<OBSBasic *>(App()->GetMainWindow());
|
|
OBSScene scene = main->GetCurrentScene();
|
|
bool success = false;
|
|
if (!scene)
|
|
return false;
|
|
|
|
obs_source_t *source = obs_get_source_by_name(name);
|
|
if (source && parent) {
|
|
OBSMessageBox::information(parent, QTStr("NameExists.Title"),
|
|
QTStr("NameExists.Text"));
|
|
|
|
} else {
|
|
const char *v_id = obs_get_latest_input_type_id(id);
|
|
source = obs_source_create(v_id, name, NULL, nullptr);
|
|
|
|
if (source) {
|
|
AddSourceData data;
|
|
data.source = source;
|
|
data.visible = visible;
|
|
|
|
obs_enter_graphics();
|
|
obs_scene_atomic_update(scene, AddSource, &data);
|
|
obs_leave_graphics();
|
|
|
|
newSource = source;
|
|
|
|
/* set monitoring if source monitors by default */
|
|
uint32_t flags = obs_source_get_output_flags(source);
|
|
if ((flags & OBS_SOURCE_MONITOR_BY_DEFAULT) != 0) {
|
|
obs_source_set_monitoring_type(
|
|
source,
|
|
OBS_MONITORING_TYPE_MONITOR_ONLY);
|
|
}
|
|
|
|
success = true;
|
|
}
|
|
}
|
|
|
|
obs_source_release(source);
|
|
return success;
|
|
}
|
|
|
|
void OBSBasicSourceSelect::on_buttonBox_accepted()
|
|
{
|
|
bool useExisting = ui->selectExisting->isChecked();
|
|
bool visible = ui->sourceVisible->isChecked();
|
|
|
|
if (useExisting) {
|
|
QListWidgetItem *item = ui->sourceList->currentItem();
|
|
if (!item)
|
|
return;
|
|
|
|
AddExisting(QT_TO_UTF8(item->text()), visible, false, nullptr,
|
|
nullptr);
|
|
} else {
|
|
if (ui->sourceName->text().isEmpty()) {
|
|
OBSMessageBox::warning(this,
|
|
QTStr("NoNameEntered.Title"),
|
|
QTStr("NoNameEntered.Text"));
|
|
return;
|
|
}
|
|
|
|
if (!AddNew(this, id, QT_TO_UTF8(ui->sourceName->text()),
|
|
visible, newSource))
|
|
return;
|
|
|
|
OBSBasic *main =
|
|
reinterpret_cast<OBSBasic *>(App()->GetMainWindow());
|
|
std::string scene_name =
|
|
obs_source_get_name(main->GetCurrentSceneSource());
|
|
auto undo = [scene_name, main](const std::string &data) {
|
|
obs_source_t *source =
|
|
obs_get_source_by_name(data.c_str());
|
|
obs_source_release(source);
|
|
obs_source_remove(source);
|
|
|
|
obs_source_t *scene_source =
|
|
obs_get_source_by_name(scene_name.c_str());
|
|
main->SetCurrentScene(scene_source, true);
|
|
obs_source_release(scene_source);
|
|
|
|
main->RefreshSources(main->GetCurrentScene());
|
|
};
|
|
obs_data_t *wrapper = obs_data_create();
|
|
obs_data_set_string(wrapper, "id", id);
|
|
obs_sceneitem_t *item = obs_scene_sceneitem_from_source(
|
|
main->GetCurrentScene(), newSource);
|
|
obs_data_set_int(wrapper, "item_id",
|
|
obs_sceneitem_get_id(item));
|
|
obs_data_set_string(
|
|
wrapper, "name",
|
|
ui->sourceName->text().toUtf8().constData());
|
|
obs_data_set_bool(wrapper, "visible", visible);
|
|
|
|
auto redo = [scene_name, main](const std::string &data) {
|
|
obs_source_t *scene_source =
|
|
obs_get_source_by_name(scene_name.c_str());
|
|
main->SetCurrentScene(scene_source, true);
|
|
obs_source_release(scene_source);
|
|
|
|
obs_data_t *dat =
|
|
obs_data_create_from_json(data.c_str());
|
|
OBSSource source;
|
|
AddNew(NULL, obs_data_get_string(dat, "id"),
|
|
obs_data_get_string(dat, "name"),
|
|
obs_data_get_bool(dat, "visible"), source);
|
|
obs_sceneitem_t *item = obs_scene_sceneitem_from_source(
|
|
main->GetCurrentScene(), source);
|
|
obs_sceneitem_set_id(item, (int64_t)obs_data_get_int(
|
|
dat, "item_id"));
|
|
|
|
main->RefreshSources(main->GetCurrentScene());
|
|
obs_data_release(dat);
|
|
obs_sceneitem_release(item);
|
|
};
|
|
undo_s.add_action(QTStr("Undo.Add").arg(ui->sourceName->text()),
|
|
undo, redo,
|
|
std::string(obs_source_get_name(newSource)),
|
|
std::string(obs_data_get_json(wrapper)));
|
|
obs_data_release(wrapper);
|
|
obs_sceneitem_release(item);
|
|
}
|
|
|
|
done(DialogCode::Accepted);
|
|
}
|
|
|
|
void OBSBasicSourceSelect::on_buttonBox_rejected()
|
|
{
|
|
done(DialogCode::Rejected);
|
|
}
|
|
|
|
static inline const char *GetSourceDisplayName(const char *id)
|
|
{
|
|
if (strcmp(id, "scene") == 0)
|
|
return Str("Basic.Scene");
|
|
const char *v_id = obs_get_latest_input_type_id(id);
|
|
return obs_source_get_display_name(v_id);
|
|
}
|
|
|
|
Q_DECLARE_METATYPE(OBSScene);
|
|
|
|
template<typename T> static inline T GetOBSRef(QListWidgetItem *item)
|
|
{
|
|
return item->data(static_cast<int>(QtDataRole::OBSRef)).value<T>();
|
|
}
|
|
|
|
OBSBasicSourceSelect::OBSBasicSourceSelect(OBSBasic *parent, const char *id_,
|
|
undo_stack &undo_s)
|
|
: QDialog(parent),
|
|
ui(new Ui::OBSBasicSourceSelect),
|
|
id(id_),
|
|
undo_s(undo_s)
|
|
{
|
|
setWindowFlags(windowFlags() & ~Qt::WindowContextHelpButtonHint);
|
|
|
|
ui->setupUi(this);
|
|
|
|
ui->sourceList->setAttribute(Qt::WA_MacShowFocusRect, false);
|
|
|
|
QString placeHolderText{QT_UTF8(GetSourceDisplayName(id))};
|
|
|
|
QString text{placeHolderText};
|
|
int i = 2;
|
|
obs_source_t *source = nullptr;
|
|
while ((source = obs_get_source_by_name(QT_TO_UTF8(text)))) {
|
|
obs_source_release(source);
|
|
text = QString("%1 %2").arg(placeHolderText).arg(i++);
|
|
}
|
|
|
|
ui->sourceName->setText(text);
|
|
ui->sourceName->setFocus(); //Fixes deselect of text.
|
|
ui->sourceName->selectAll();
|
|
|
|
installEventFilter(CreateShortcutFilter());
|
|
|
|
if (strcmp(id_, "scene") == 0) {
|
|
OBSBasic *main =
|
|
reinterpret_cast<OBSBasic *>(App()->GetMainWindow());
|
|
OBSSource curSceneSource = main->GetCurrentSceneSource();
|
|
|
|
ui->selectExisting->setChecked(true);
|
|
ui->createNew->setChecked(false);
|
|
ui->createNew->setEnabled(false);
|
|
ui->sourceName->setEnabled(false);
|
|
|
|
int count = main->ui->scenes->count();
|
|
for (int i = 0; i < count; i++) {
|
|
QListWidgetItem *item = main->ui->scenes->item(i);
|
|
OBSScene scene = GetOBSRef<OBSScene>(item);
|
|
OBSSource sceneSource = obs_scene_get_source(scene);
|
|
|
|
if (curSceneSource == sceneSource)
|
|
continue;
|
|
|
|
const char *name = obs_source_get_name(sceneSource);
|
|
ui->sourceList->addItem(QT_UTF8(name));
|
|
}
|
|
} else if (strcmp(id_, "group") == 0) {
|
|
obs_enum_sources(EnumGroups, this);
|
|
} else {
|
|
obs_enum_sources(EnumSources, this);
|
|
}
|
|
}
|
|
|
|
void OBSBasicSourceSelect::SourcePaste(SourceCopyInfo &info, bool dup)
|
|
{
|
|
OBSSource source = OBSGetStrongRef(info.weak_source);
|
|
if (!source)
|
|
return;
|
|
|
|
AddExisting(source, info.visible, dup, info.transform.get(),
|
|
info.crop.get());
|
|
}
|