window-basic-transform.cpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385
  1. #include <QMessageBox>
  2. #include "window-basic-transform.hpp"
  3. #include "window-basic-main.hpp"
  4. Q_DECLARE_METATYPE(OBSScene);
  5. Q_DECLARE_METATYPE(OBSSceneItem);
  6. static bool find_sel(obs_scene_t *, obs_sceneitem_t *item, void *param)
  7. {
  8. OBSSceneItem &dst = *reinterpret_cast<OBSSceneItem *>(param);
  9. if (obs_sceneitem_selected(item)) {
  10. dst = item;
  11. return false;
  12. }
  13. if (obs_sceneitem_is_group(item)) {
  14. obs_sceneitem_group_enum_items(item, find_sel, param);
  15. if (!!dst) {
  16. return false;
  17. }
  18. }
  19. return true;
  20. };
  21. static OBSSceneItem FindASelectedItem(obs_scene_t *scene)
  22. {
  23. OBSSceneItem item;
  24. obs_scene_enum_items(scene, find_sel, &item);
  25. return item;
  26. }
  27. #define COMBO_CHANGED &QComboBox::currentIndexChanged
  28. #define ISCROLL_CHANGED &QSpinBox::valueChanged
  29. #define DSCROLL_CHANGED &QDoubleSpinBox::valueChanged
  30. OBSBasicTransform::OBSBasicTransform(OBSSceneItem item, OBSBasic *parent)
  31. : QDialog(parent),
  32. ui(new Ui::OBSBasicTransform),
  33. main(parent)
  34. {
  35. setWindowFlags(windowFlags() & ~Qt::WindowContextHelpButtonHint);
  36. ui->setupUi(this);
  37. HookWidget(ui->positionX, DSCROLL_CHANGED,
  38. &OBSBasicTransform::OnControlChanged);
  39. HookWidget(ui->positionY, DSCROLL_CHANGED,
  40. &OBSBasicTransform::OnControlChanged);
  41. HookWidget(ui->rotation, DSCROLL_CHANGED,
  42. &OBSBasicTransform::OnControlChanged);
  43. HookWidget(ui->sizeX, DSCROLL_CHANGED,
  44. &OBSBasicTransform::OnControlChanged);
  45. HookWidget(ui->sizeY, DSCROLL_CHANGED,
  46. &OBSBasicTransform::OnControlChanged);
  47. HookWidget(ui->align, COMBO_CHANGED,
  48. &OBSBasicTransform::OnControlChanged);
  49. HookWidget(ui->boundsType, COMBO_CHANGED,
  50. &OBSBasicTransform::OnBoundsType);
  51. HookWidget(ui->boundsAlign, COMBO_CHANGED,
  52. &OBSBasicTransform::OnControlChanged);
  53. HookWidget(ui->boundsWidth, DSCROLL_CHANGED,
  54. &OBSBasicTransform::OnControlChanged);
  55. HookWidget(ui->boundsHeight, DSCROLL_CHANGED,
  56. &OBSBasicTransform::OnControlChanged);
  57. HookWidget(ui->cropLeft, ISCROLL_CHANGED,
  58. &OBSBasicTransform::OnCropChanged);
  59. HookWidget(ui->cropRight, ISCROLL_CHANGED,
  60. &OBSBasicTransform::OnCropChanged);
  61. HookWidget(ui->cropTop, ISCROLL_CHANGED,
  62. &OBSBasicTransform::OnCropChanged);
  63. HookWidget(ui->cropBottom, ISCROLL_CHANGED,
  64. &OBSBasicTransform::OnCropChanged);
  65. ui->buttonBox->button(QDialogButtonBox::Close)->setDefault(true);
  66. connect(ui->buttonBox->button(QDialogButtonBox::Reset),
  67. &QPushButton::clicked, main,
  68. &OBSBasic::on_actionResetTransform_triggered);
  69. installEventFilter(CreateShortcutFilter());
  70. OBSScene scene = obs_sceneitem_get_scene(item);
  71. SetScene(scene);
  72. SetItem(item);
  73. std::string name = obs_source_get_name(obs_sceneitem_get_source(item));
  74. setWindowTitle(QTStr("Basic.TransformWindow.Title").arg(name.c_str()));
  75. OBSDataAutoRelease wrapper =
  76. obs_scene_save_transform_states(main->GetCurrentScene(), false);
  77. undo_data = std::string(obs_data_get_json(wrapper));
  78. channelChangedSignal.Connect(obs_get_signal_handler(), "channel_change",
  79. OBSChannelChanged, this);
  80. }
  81. OBSBasicTransform::~OBSBasicTransform()
  82. {
  83. OBSDataAutoRelease wrapper =
  84. obs_scene_save_transform_states(main->GetCurrentScene(), false);
  85. auto undo_redo = [](const std::string &data) {
  86. OBSDataAutoRelease dat =
  87. obs_data_create_from_json(data.c_str());
  88. OBSSourceAutoRelease source = obs_get_source_by_uuid(
  89. obs_data_get_string(dat, "scene_uuid"));
  90. reinterpret_cast<OBSBasic *>(App()->GetMainWindow())
  91. ->SetCurrentScene(source.Get(), true);
  92. obs_scene_load_transform_states(data.c_str());
  93. };
  94. std::string redo_data(obs_data_get_json(wrapper));
  95. if (undo_data.compare(redo_data) != 0)
  96. main->undo_s.add_action(
  97. QTStr("Undo.Transform")
  98. .arg(obs_source_get_name(obs_scene_get_source(
  99. main->GetCurrentScene()))),
  100. undo_redo, undo_redo, undo_data, redo_data);
  101. }
  102. void OBSBasicTransform::SetScene(OBSScene scene)
  103. {
  104. transformSignal.Disconnect();
  105. selectSignal.Disconnect();
  106. deselectSignal.Disconnect();
  107. removeSignal.Disconnect();
  108. if (scene) {
  109. OBSSource source = obs_scene_get_source(scene);
  110. signal_handler_t *signal =
  111. obs_source_get_signal_handler(source);
  112. transformSignal.Connect(signal, "item_transform",
  113. OBSSceneItemTransform, this);
  114. removeSignal.Connect(signal, "item_remove", OBSSceneItemRemoved,
  115. this);
  116. selectSignal.Connect(signal, "item_select", OBSSceneItemSelect,
  117. this);
  118. deselectSignal.Connect(signal, "item_deselect",
  119. OBSSceneItemDeselect, this);
  120. }
  121. }
  122. void OBSBasicTransform::SetItem(OBSSceneItem newItem)
  123. {
  124. QMetaObject::invokeMethod(this, "SetItemQt",
  125. Q_ARG(OBSSceneItem, OBSSceneItem(newItem)));
  126. }
  127. void OBSBasicTransform::SetItemQt(OBSSceneItem newItem)
  128. {
  129. item = newItem;
  130. if (item)
  131. RefreshControls();
  132. bool enable = !!item;
  133. ui->container->setEnabled(enable);
  134. ui->buttonBox->button(QDialogButtonBox::Reset)->setEnabled(enable);
  135. }
  136. void OBSBasicTransform::OBSChannelChanged(void *param, calldata_t *data)
  137. {
  138. OBSBasicTransform *window =
  139. reinterpret_cast<OBSBasicTransform *>(param);
  140. uint32_t channel = (uint32_t)calldata_int(data, "channel");
  141. OBSSource source = (obs_source_t *)calldata_ptr(data, "source");
  142. if (channel == 0) {
  143. OBSScene scene = obs_scene_from_source(source);
  144. window->SetScene(scene);
  145. if (!scene)
  146. window->SetItem(nullptr);
  147. else
  148. window->SetItem(FindASelectedItem(scene));
  149. }
  150. }
  151. void OBSBasicTransform::OBSSceneItemTransform(void *param, calldata_t *data)
  152. {
  153. OBSBasicTransform *window =
  154. reinterpret_cast<OBSBasicTransform *>(param);
  155. OBSSceneItem item = (obs_sceneitem_t *)calldata_ptr(data, "item");
  156. if (item == window->item && !window->ignoreTransformSignal)
  157. QMetaObject::invokeMethod(window, "RefreshControls");
  158. }
  159. void OBSBasicTransform::OBSSceneItemRemoved(void *param, calldata_t *data)
  160. {
  161. OBSBasicTransform *window =
  162. reinterpret_cast<OBSBasicTransform *>(param);
  163. obs_scene_t *scene = (obs_scene_t *)calldata_ptr(data, "scene");
  164. obs_sceneitem_t *item = (obs_sceneitem_t *)calldata_ptr(data, "item");
  165. if (item == window->item)
  166. window->SetItem(FindASelectedItem(scene));
  167. }
  168. void OBSBasicTransform::OBSSceneItemSelect(void *param, calldata_t *data)
  169. {
  170. OBSBasicTransform *window =
  171. reinterpret_cast<OBSBasicTransform *>(param);
  172. OBSSceneItem item = (obs_sceneitem_t *)calldata_ptr(data, "item");
  173. if (item != window->item)
  174. window->SetItem(item);
  175. }
  176. void OBSBasicTransform::OBSSceneItemDeselect(void *param, calldata_t *data)
  177. {
  178. OBSBasicTransform *window =
  179. reinterpret_cast<OBSBasicTransform *>(param);
  180. obs_scene_t *scene = (obs_scene_t *)calldata_ptr(data, "scene");
  181. obs_sceneitem_t *item = (obs_sceneitem_t *)calldata_ptr(data, "item");
  182. if (item == window->item) {
  183. window->setWindowTitle(
  184. QTStr("Basic.TransformWindow.NoSelectedSource"));
  185. window->SetItem(FindASelectedItem(scene));
  186. }
  187. }
  188. static const uint32_t listToAlign[] = {OBS_ALIGN_TOP | OBS_ALIGN_LEFT,
  189. OBS_ALIGN_TOP,
  190. OBS_ALIGN_TOP | OBS_ALIGN_RIGHT,
  191. OBS_ALIGN_LEFT,
  192. OBS_ALIGN_CENTER,
  193. OBS_ALIGN_RIGHT,
  194. OBS_ALIGN_BOTTOM | OBS_ALIGN_LEFT,
  195. OBS_ALIGN_BOTTOM,
  196. OBS_ALIGN_BOTTOM | OBS_ALIGN_RIGHT};
  197. static int AlignToList(uint32_t align)
  198. {
  199. int index = 0;
  200. for (uint32_t curAlign : listToAlign) {
  201. if (curAlign == align)
  202. return index;
  203. index++;
  204. }
  205. return 0;
  206. }
  207. void OBSBasicTransform::RefreshControls()
  208. {
  209. if (!item)
  210. return;
  211. obs_transform_info osi;
  212. obs_sceneitem_crop crop;
  213. obs_sceneitem_get_info(item, &osi);
  214. obs_sceneitem_get_crop(item, &crop);
  215. obs_source_t *source = obs_sceneitem_get_source(item);
  216. uint32_t source_cx = obs_source_get_width(source);
  217. uint32_t source_cy = obs_source_get_height(source);
  218. float width = float(source_cx);
  219. float height = float(source_cy);
  220. int alignIndex = AlignToList(osi.alignment);
  221. int boundsAlignIndex = AlignToList(osi.bounds_alignment);
  222. ignoreItemChange = true;
  223. ui->positionX->setValue(osi.pos.x);
  224. ui->positionY->setValue(osi.pos.y);
  225. ui->rotation->setValue(osi.rot);
  226. ui->sizeX->setValue(osi.scale.x * width);
  227. ui->sizeY->setValue(osi.scale.y * height);
  228. ui->align->setCurrentIndex(alignIndex);
  229. bool valid_size = source_cx != 0 && source_cy != 0;
  230. ui->sizeX->setEnabled(valid_size);
  231. ui->sizeY->setEnabled(valid_size);
  232. ui->boundsType->setCurrentIndex(int(osi.bounds_type));
  233. ui->boundsAlign->setCurrentIndex(boundsAlignIndex);
  234. ui->boundsWidth->setValue(osi.bounds.x);
  235. ui->boundsHeight->setValue(osi.bounds.y);
  236. ui->cropLeft->setValue(int(crop.left));
  237. ui->cropRight->setValue(int(crop.right));
  238. ui->cropTop->setValue(int(crop.top));
  239. ui->cropBottom->setValue(int(crop.bottom));
  240. ignoreItemChange = false;
  241. std::string name = obs_source_get_name(source);
  242. setWindowTitle(QTStr("Basic.TransformWindow.Title").arg(name.c_str()));
  243. }
  244. void OBSBasicTransform::OnBoundsType(int index)
  245. {
  246. if (index == -1)
  247. return;
  248. obs_bounds_type type = (obs_bounds_type)index;
  249. bool enable = (type != OBS_BOUNDS_NONE);
  250. ui->boundsAlign->setEnabled(enable);
  251. ui->boundsWidth->setEnabled(enable);
  252. ui->boundsHeight->setEnabled(enable);
  253. if (!ignoreItemChange) {
  254. obs_bounds_type lastType = obs_sceneitem_get_bounds_type(item);
  255. if (lastType == OBS_BOUNDS_NONE) {
  256. OBSSource source = obs_sceneitem_get_source(item);
  257. int width = (int)obs_source_get_width(source);
  258. int height = (int)obs_source_get_height(source);
  259. ui->boundsWidth->setValue(width);
  260. ui->boundsHeight->setValue(height);
  261. }
  262. }
  263. OnControlChanged();
  264. }
  265. void OBSBasicTransform::OnControlChanged()
  266. {
  267. if (ignoreItemChange)
  268. return;
  269. obs_source_t *source = obs_sceneitem_get_source(item);
  270. uint32_t source_cx = obs_source_get_width(source);
  271. uint32_t source_cy = obs_source_get_height(source);
  272. double width = double(source_cx);
  273. double height = double(source_cy);
  274. obs_transform_info oti;
  275. obs_sceneitem_get_info(item, &oti);
  276. /* do not scale a source if it has 0 width/height */
  277. if (source_cx != 0 && source_cy != 0) {
  278. oti.scale.x = float(ui->sizeX->value() / width);
  279. oti.scale.y = float(ui->sizeY->value() / height);
  280. }
  281. oti.pos.x = float(ui->positionX->value());
  282. oti.pos.y = float(ui->positionY->value());
  283. oti.rot = float(ui->rotation->value());
  284. oti.alignment = listToAlign[ui->align->currentIndex()];
  285. oti.bounds_type = (obs_bounds_type)ui->boundsType->currentIndex();
  286. oti.bounds_alignment = listToAlign[ui->boundsAlign->currentIndex()];
  287. oti.bounds.x = float(ui->boundsWidth->value());
  288. oti.bounds.y = float(ui->boundsHeight->value());
  289. ignoreTransformSignal = true;
  290. obs_sceneitem_set_info(item, &oti);
  291. ignoreTransformSignal = false;
  292. }
  293. void OBSBasicTransform::OnCropChanged()
  294. {
  295. if (ignoreItemChange)
  296. return;
  297. obs_sceneitem_crop crop;
  298. crop.left = uint32_t(ui->cropLeft->value());
  299. crop.right = uint32_t(ui->cropRight->value());
  300. crop.top = uint32_t(ui->cropTop->value());
  301. crop.bottom = uint32_t(ui->cropBottom->value());
  302. ignoreTransformSignal = true;
  303. obs_sceneitem_set_crop(item, &crop);
  304. ignoreTransformSignal = false;
  305. }
  306. template<typename T> static T GetOBSRef(QListWidgetItem *item)
  307. {
  308. return item->data(static_cast<int>(QtDataRole::OBSRef)).value<T>();
  309. }
  310. void OBSBasicTransform::OnSceneChanged(QListWidgetItem *current,
  311. QListWidgetItem *)
  312. {
  313. if (!current)
  314. return;
  315. OBSScene scene = GetOBSRef<OBSScene>(current);
  316. this->SetScene(scene);
  317. }