| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465 |
- #include "OBSBasicTransform.hpp"
- #include <widgets/OBSBasic.hpp>
- #include "moc_OBSBasicTransform.cpp"
- namespace {
- static bool find_sel(obs_scene_t *, obs_sceneitem_t *item, void *param)
- {
- OBSSceneItem &dst = *static_cast<OBSSceneItem *>(param);
- if (obs_sceneitem_selected(item)) {
- dst = item;
- return false;
- }
- if (obs_sceneitem_is_group(item)) {
- obs_sceneitem_group_enum_items(item, find_sel, param);
- if (!!dst) {
- return false;
- }
- }
- return true;
- };
- static OBSSceneItem FindASelectedItem(obs_scene_t *scene)
- {
- OBSSceneItem item;
- obs_scene_enum_items(scene, find_sel, &item);
- return item;
- }
- static vec2 getAlignmentConversion(uint32_t alignment)
- {
- vec2 ratio = {0.5f, 0.5f};
- if (alignment & OBS_ALIGN_RIGHT) {
- ratio.x = 1.0f;
- }
- if (alignment & OBS_ALIGN_LEFT) {
- ratio.x = 0.0f;
- }
- if (alignment & OBS_ALIGN_BOTTOM) {
- ratio.y = 1.0f;
- }
- if (alignment & OBS_ALIGN_TOP) {
- ratio.y = 0.0f;
- }
- return ratio;
- }
- } // namespace
- #define COMBO_CHANGED &QComboBox::currentIndexChanged
- #define ALIGN_CHANGED &AlignmentSelector::currentIndexChanged
- #define ISCROLL_CHANGED &QSpinBox::valueChanged
- #define DSCROLL_CHANGED &QDoubleSpinBox::valueChanged
- OBSBasicTransform::OBSBasicTransform(OBSSceneItem item, OBSBasic *parent)
- : QDialog(parent),
- ui(new Ui::OBSBasicTransform),
- main(parent)
- {
- setWindowFlags(windowFlags() & ~Qt::WindowContextHelpButtonHint);
- ui->setupUi(this);
- positionAlignment = new AlignmentSelector(this);
- positionAlignment->setAccessibleName(QTStr("Basic.TransformWindow.Alignment"));
- ui->alignmentLayout->addWidget(positionAlignment);
- positionAlignment->setAlignment(Qt::AlignTop | Qt::AlignLeft);
- boundsAlignment = new AlignmentSelector(this);
- boundsAlignment->setAccessibleName(QTStr("Basic.TransformWindow.BoundsAlignment"));
- boundsAlignment->setEnabled(false);
- ui->boundsAlignmentLayout->addWidget(boundsAlignment);
- boundsAlignment->setAlignment(Qt::AlignTop | Qt::AlignLeft);
- setTabOrder(ui->rotation, positionAlignment);
- setTabOrder(ui->boundsType, boundsAlignment);
- hookWidget(ui->positionX, DSCROLL_CHANGED, &OBSBasicTransform::onControlChanged);
- hookWidget(ui->positionY, DSCROLL_CHANGED, &OBSBasicTransform::onControlChanged);
- hookWidget(ui->rotation, DSCROLL_CHANGED, &OBSBasicTransform::onControlChanged);
- hookWidget(ui->sizeX, DSCROLL_CHANGED, &OBSBasicTransform::onControlChanged);
- hookWidget(ui->sizeY, DSCROLL_CHANGED, &OBSBasicTransform::onControlChanged);
- hookWidget(positionAlignment.get(), ALIGN_CHANGED, &OBSBasicTransform::onAlignChanged);
- hookWidget(ui->boundsType, COMBO_CHANGED, &OBSBasicTransform::onBoundsType);
- hookWidget(boundsAlignment.get(), ALIGN_CHANGED, &OBSBasicTransform::onControlChanged);
- hookWidget(ui->boundsWidth, DSCROLL_CHANGED, &OBSBasicTransform::onControlChanged);
- hookWidget(ui->boundsHeight, DSCROLL_CHANGED, &OBSBasicTransform::onControlChanged);
- hookWidget(ui->cropLeft, ISCROLL_CHANGED, &OBSBasicTransform::onCropChanged);
- hookWidget(ui->cropRight, ISCROLL_CHANGED, &OBSBasicTransform::onCropChanged);
- hookWidget(ui->cropTop, ISCROLL_CHANGED, &OBSBasicTransform::onCropChanged);
- hookWidget(ui->cropBottom, ISCROLL_CHANGED, &OBSBasicTransform::onCropChanged);
- #if QT_VERSION >= QT_VERSION_CHECK(6, 7, 0)
- hookWidget(ui->cropToBounds, &QCheckBox::checkStateChanged, &OBSBasicTransform::onControlChanged);
- #else
- hookWidget(ui->cropToBounds, &QCheckBox::stateChanged, &OBSBasicTransform::onControlChanged);
- #endif
- ui->buttonBox->button(QDialogButtonBox::Close)->setDefault(true);
- connect(ui->buttonBox->button(QDialogButtonBox::Reset), &QPushButton::clicked, main,
- &OBSBasic::on_actionResetTransform_triggered);
- installEventFilter(CreateShortcutFilter());
- OBSScene scene = obs_sceneitem_get_scene(item);
- setScene(scene);
- setItem(item);
- std::string name = obs_source_get_name(obs_sceneitem_get_source(item));
- setWindowTitle(QTStr("Basic.TransformWindow.Title").arg(name.c_str()));
- OBSDataAutoRelease wrapper = obs_scene_save_transform_states(main->GetCurrentScene(), false);
- undo_data = std::string(obs_data_get_json(wrapper));
- adjustSize();
- setMinimumSize(size());
- setMaximumSize(size());
- }
- OBSBasicTransform::~OBSBasicTransform()
- {
- OBSDataAutoRelease wrapper = obs_scene_save_transform_states(main->GetCurrentScene(), false);
- auto undo_redo = [](const std::string &data) {
- OBSDataAutoRelease dat = obs_data_create_from_json(data.c_str());
- OBSSourceAutoRelease source = obs_get_source_by_uuid(obs_data_get_string(dat, "scene_uuid"));
- OBSBasic::Get()->SetCurrentScene(source.Get(), true);
- obs_scene_load_transform_states(data.c_str());
- };
- std::string redo_data(obs_data_get_json(wrapper));
- if (undo_data.compare(redo_data) != 0)
- main->undo_s.add_action(
- QTStr("Undo.Transform").arg(obs_source_get_name(obs_scene_get_source(main->GetCurrentScene()))),
- undo_redo, undo_redo, undo_data, redo_data);
- }
- void OBSBasicTransform::setScene(OBSScene scene)
- {
- sigs.clear();
- if (scene) {
- OBSSource source = obs_scene_get_source(scene);
- signal_handler_t *signal = obs_source_get_signal_handler(source);
- sigs.emplace_back(signal, "item_transform", OBSSceneItemTransform, this);
- sigs.emplace_back(signal, "item_remove", OBSSceneItemRemoved, this);
- sigs.emplace_back(signal, "item_select", OBSSceneItemSelect, this);
- sigs.emplace_back(signal, "item_deselect", OBSSceneItemDeselect, this);
- sigs.emplace_back(signal, "item_locked", OBSSceneItemLocked, this);
- }
- }
- void OBSBasicTransform::setItem(OBSSceneItem newItem)
- {
- QMetaObject::invokeMethod(this, "setItemQt", Q_ARG(OBSSceneItem, OBSSceneItem(newItem)));
- }
- void OBSBasicTransform::setEnabled(bool enable)
- {
- ui->transformSettings->setEnabled(enable);
- ui->boundsSettings->setEnabled(enable);
- ui->cropSettings->setEnabled(enable);
- ui->buttonBox->button(QDialogButtonBox::Reset)->setEnabled(enable);
- }
- void OBSBasicTransform::setItemQt(OBSSceneItem newItem)
- {
- item = newItem;
- if (item)
- refreshControls();
- bool enable = !!item && !obs_sceneitem_locked(item);
- setEnabled(enable);
- }
- void OBSBasicTransform::OBSSceneItemTransform(void *param, calldata_t *data)
- {
- OBSBasicTransform *window = static_cast<OBSBasicTransform *>(param);
- OBSSceneItem item = (obs_sceneitem_t *)calldata_ptr(data, "item");
- if (item == window->item && !window->ignoreTransformSignal)
- QMetaObject::invokeMethod(window, "refreshControls");
- }
- void OBSBasicTransform::OBSSceneItemRemoved(void *param, calldata_t *data)
- {
- OBSBasicTransform *window = static_cast<OBSBasicTransform *>(param);
- obs_scene_t *scene = (obs_scene_t *)calldata_ptr(data, "scene");
- obs_sceneitem_t *item = (obs_sceneitem_t *)calldata_ptr(data, "item");
- if (item == window->item)
- window->setItem(FindASelectedItem(scene));
- }
- void OBSBasicTransform::OBSSceneItemSelect(void *param, calldata_t *data)
- {
- OBSBasicTransform *window = static_cast<OBSBasicTransform *>(param);
- OBSSceneItem item = (obs_sceneitem_t *)calldata_ptr(data, "item");
- if (item != window->item)
- window->setItem(item);
- }
- void OBSBasicTransform::OBSSceneItemDeselect(void *param, calldata_t *data)
- {
- OBSBasicTransform *window = static_cast<OBSBasicTransform *>(param);
- obs_scene_t *scene = (obs_scene_t *)calldata_ptr(data, "scene");
- obs_sceneitem_t *item = (obs_sceneitem_t *)calldata_ptr(data, "item");
- if (item == window->item) {
- window->setWindowTitle(QTStr("Basic.TransformWindow.NoSelectedSource"));
- window->setItem(FindASelectedItem(scene));
- }
- }
- void OBSBasicTransform::OBSSceneItemLocked(void *param, calldata_t *data)
- {
- OBSBasicTransform *window = static_cast<OBSBasicTransform *>(param);
- bool locked = calldata_bool(data, "locked");
- QMetaObject::invokeMethod(window, "setEnabled", Q_ARG(bool, !locked));
- }
- static const uint32_t indexToAlign[] = {OBS_ALIGN_TOP | OBS_ALIGN_LEFT,
- OBS_ALIGN_TOP,
- OBS_ALIGN_TOP | OBS_ALIGN_RIGHT,
- OBS_ALIGN_LEFT,
- OBS_ALIGN_CENTER,
- OBS_ALIGN_RIGHT,
- OBS_ALIGN_BOTTOM | OBS_ALIGN_LEFT,
- OBS_ALIGN_BOTTOM,
- OBS_ALIGN_BOTTOM | OBS_ALIGN_RIGHT};
- static int alignToIndex(uint32_t align)
- {
- int index = 0;
- for (uint32_t curAlign : indexToAlign) {
- if (curAlign == align)
- return index;
- index++;
- }
- return 0;
- }
- void OBSBasicTransform::refreshControls()
- {
- if (!item)
- return;
- obs_transform_info oti;
- obs_sceneitem_crop crop;
- obs_sceneitem_get_info2(item, &oti);
- obs_sceneitem_get_crop(item, &crop);
- obs_source_t *source = obs_sceneitem_get_source(item);
- uint32_t source_cx = obs_source_get_width(source);
- uint32_t source_cy = obs_source_get_height(source);
- float width = float(source_cx);
- float height = float(source_cy);
- int alignIndex = alignToIndex(oti.alignment);
- int boundsAlignIndex = alignToIndex(oti.bounds_alignment);
- ignoreItemChange = true;
- ui->positionX->setValue(oti.pos.x);
- ui->positionY->setValue(oti.pos.y);
- ui->rotation->setValue(oti.rot);
- ui->sizeX->setValue(oti.scale.x * width);
- ui->sizeY->setValue(oti.scale.y * height);
- positionAlignment->setCurrentIndex(alignIndex);
- bool valid_size = source_cx != 0 && source_cy != 0;
- ui->sizeX->setEnabled(valid_size);
- ui->sizeY->setEnabled(valid_size);
- ui->boundsType->setCurrentIndex(int(oti.bounds_type));
- boundsAlignment->setCurrentIndex(boundsAlignIndex);
- ui->boundsWidth->setValue(oti.bounds.x);
- ui->boundsHeight->setValue(oti.bounds.y);
- ui->cropToBounds->setChecked(oti.crop_to_bounds);
- ui->cropLeft->setValue(int(crop.left));
- ui->cropRight->setValue(int(crop.right));
- ui->cropTop->setValue(int(crop.top));
- ui->cropBottom->setValue(int(crop.bottom));
- ignoreItemChange = false;
- std::string name = obs_source_get_name(source);
- setWindowTitle(QTStr("Basic.TransformWindow.Title").arg(name.c_str()));
- }
- void OBSBasicTransform::onAlignChanged(int index)
- {
- uint32_t alignment = indexToAlign[index];
- vec2 flipRatio = getAlignmentConversion(alignment);
- obs_transform_info oti;
- obs_sceneitem_crop crop;
- obs_sceneitem_get_info2(item, &oti);
- obs_sceneitem_get_crop(item, &crop);
- obs_source_t *source = obs_sceneitem_get_source(item);
- uint32_t sourceWidth = obs_source_get_width(source);
- uint32_t sourceHeight = obs_source_get_height(source);
- uint32_t widthForFlip = sourceWidth - crop.left - crop.right;
- uint32_t heightForFlip = sourceHeight - crop.top - crop.bottom;
- if (oti.bounds_type != OBS_BOUNDS_NONE) {
- widthForFlip = oti.bounds.x;
- heightForFlip = oti.bounds.y;
- }
- vec2 currentRatio = getAlignmentConversion(oti.alignment);
- float shiftX = (currentRatio.x - flipRatio.x) * widthForFlip * oti.scale.x;
- float shiftY = (currentRatio.y - flipRatio.y) * heightForFlip * oti.scale.y;
- bool previousIgnoreState = ignoreItemChange;
- ignoreItemChange = true;
- ui->positionX->setValue(oti.pos.x - shiftX);
- ui->positionY->setValue(oti.pos.y - shiftY);
- ignoreItemChange = previousIgnoreState;
- onControlChanged();
- }
- void OBSBasicTransform::onBoundsType(int index)
- {
- if (index == -1)
- return;
- obs_bounds_type type = (obs_bounds_type)index;
- bool enable = (type != OBS_BOUNDS_NONE);
- boundsAlignment->setEnabled(enable && type != OBS_BOUNDS_STRETCH);
- ui->boundsWidth->setEnabled(enable);
- ui->boundsHeight->setEnabled(enable);
- bool isCoverBounds = type == OBS_BOUNDS_SCALE_OUTER || type == OBS_BOUNDS_SCALE_TO_WIDTH ||
- type == OBS_BOUNDS_SCALE_TO_HEIGHT;
- ui->cropToBounds->setEnabled(isCoverBounds);
- if (!ignoreItemChange) {
- obs_bounds_type lastType = obs_sceneitem_get_bounds_type(item);
- if (lastType == OBS_BOUNDS_NONE) {
- OBSSource source = obs_sceneitem_get_source(item);
- int width = (int)obs_source_get_width(source);
- int height = (int)obs_source_get_height(source);
- vec2 scale;
- obs_sceneitem_get_scale(item, &scale);
- obs_sceneitem_crop crop;
- obs_sceneitem_get_crop(item, &crop);
- ui->sizeX->setValue(width);
- ui->sizeY->setValue(height);
- ui->boundsWidth->setValue((width - crop.left - crop.right) * scale.x);
- ui->boundsHeight->setValue((height - crop.top - crop.bottom) * scale.y);
- } else if (type == OBS_BOUNDS_NONE) {
- OBSSource source = obs_sceneitem_get_source(item);
- int width = (int)obs_source_get_width(source);
- int height = (int)obs_source_get_height(source);
- matrix4 draw;
- obs_sceneitem_get_draw_transform(item, &draw);
- ui->sizeX->setValue(width * draw.x.x);
- ui->sizeY->setValue(height * draw.y.y);
- obs_transform_info oti;
- obs_sceneitem_get_info2(item, &oti);
- // We use the draw transform values here which is always a top left coordinate origin.
- vec2 currentRatio = getAlignmentConversion(OBS_ALIGN_TOP | OBS_ALIGN_LEFT);
- vec2 flipRatio = getAlignmentConversion(oti.alignment);
- float drawX = draw.t.x;
- float drawY = draw.t.y;
- obs_sceneitem_crop crop;
- obs_sceneitem_get_crop(item, &crop);
- uint32_t widthForFlip = width - crop.left - crop.right;
- uint32_t heightForFlip = height - crop.top - crop.bottom;
- float shiftX = (currentRatio.x - flipRatio.x) * (widthForFlip * draw.x.x);
- float shiftY = (currentRatio.y - flipRatio.y) * (heightForFlip * draw.y.y);
- ui->positionX->setValue(oti.pos.x - (oti.pos.x - drawX) - shiftX);
- ui->positionY->setValue(oti.pos.y - (oti.pos.y - drawY) - shiftY);
- }
- }
- onControlChanged();
- }
- void OBSBasicTransform::onControlChanged()
- {
- if (ignoreItemChange)
- return;
- obs_source_t *source = obs_sceneitem_get_source(item);
- uint32_t source_cx = obs_source_get_width(source);
- uint32_t source_cy = obs_source_get_height(source);
- double width = double(source_cx);
- double height = double(source_cy);
- obs_transform_info oti;
- obs_sceneitem_get_info2(item, &oti);
- /* do not scale a source if it has 0 width/height */
- if (source_cx != 0 && source_cy != 0) {
- oti.scale.x = float(ui->sizeX->value() / width);
- oti.scale.y = float(ui->sizeY->value() / height);
- }
- oti.pos.x = float(ui->positionX->value());
- oti.pos.y = float(ui->positionY->value());
- oti.rot = float(ui->rotation->value());
- oti.alignment = indexToAlign[positionAlignment->currentIndex()];
- oti.bounds_type = (obs_bounds_type)ui->boundsType->currentIndex();
- oti.bounds_alignment = indexToAlign[boundsAlignment->currentIndex()];
- oti.bounds.x = float(ui->boundsWidth->value());
- oti.bounds.y = float(ui->boundsHeight->value());
- oti.crop_to_bounds = ui->cropToBounds->isChecked();
- ignoreTransformSignal = true;
- obs_sceneitem_set_info2(item, &oti);
- ignoreTransformSignal = false;
- }
- void OBSBasicTransform::onCropChanged()
- {
- if (ignoreItemChange)
- return;
- obs_sceneitem_crop crop;
- crop.left = uint32_t(ui->cropLeft->value());
- crop.right = uint32_t(ui->cropRight->value());
- crop.top = uint32_t(ui->cropTop->value());
- crop.bottom = uint32_t(ui->cropBottom->value());
- ignoreTransformSignal = true;
- obs_sceneitem_set_crop(item, &crop);
- ignoreTransformSignal = false;
- }
- void OBSBasicTransform::onSceneChanged(QListWidgetItem *current, QListWidgetItem *)
- {
- if (!current)
- return;
- OBSScene scene = GetOBSRef<OBSScene>(current);
- this->setScene(scene);
- }
|