window-basic-transform.cpp 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309
  1. #include <QMessageBox>
  2. #include "window-basic-transform.hpp"
  3. #include "window-basic-main.hpp"
  4. Q_DECLARE_METATYPE(OBSSceneItem);
  5. static bool find_sel(obs_scene_t *, obs_sceneitem_t *item, void *param)
  6. {
  7. OBSSceneItem &dst = *reinterpret_cast<OBSSceneItem*>(param);
  8. if (obs_sceneitem_selected(item)) {
  9. dst = item;
  10. return false;
  11. }
  12. if (obs_sceneitem_is_group(item)) {
  13. obs_sceneitem_group_enum_items(item, find_sel, param);
  14. if (!!dst) {
  15. return false;
  16. }
  17. }
  18. return true;
  19. };
  20. static OBSSceneItem FindASelectedItem(OBSScene scene)
  21. {
  22. OBSSceneItem item;
  23. obs_scene_enum_items(scene, find_sel, &item);
  24. return item;
  25. }
  26. void OBSBasicTransform::HookWidget(QWidget *widget, const char *signal,
  27. const char *slot)
  28. {
  29. QObject::connect(widget, signal, this, slot);
  30. }
  31. #define COMBO_CHANGED SIGNAL(currentIndexChanged(int))
  32. #define ISCROLL_CHANGED SIGNAL(valueChanged(int))
  33. #define DSCROLL_CHANGED SIGNAL(valueChanged(double))
  34. OBSBasicTransform::OBSBasicTransform(OBSBasic *parent)
  35. : QDialog (parent),
  36. ui (new Ui::OBSBasicTransform),
  37. main (parent)
  38. {
  39. ui->setupUi(this);
  40. setWindowFlags(windowFlags() & ~Qt::WindowContextHelpButtonHint);
  41. HookWidget(ui->positionX, DSCROLL_CHANGED, SLOT(OnControlChanged()));
  42. HookWidget(ui->positionY, DSCROLL_CHANGED, SLOT(OnControlChanged()));
  43. HookWidget(ui->rotation, DSCROLL_CHANGED, SLOT(OnControlChanged()));
  44. HookWidget(ui->sizeX, DSCROLL_CHANGED, SLOT(OnControlChanged()));
  45. HookWidget(ui->sizeY, DSCROLL_CHANGED, SLOT(OnControlChanged()));
  46. HookWidget(ui->align, COMBO_CHANGED, SLOT(OnControlChanged()));
  47. HookWidget(ui->boundsType, COMBO_CHANGED, SLOT(OnBoundsType(int)));
  48. HookWidget(ui->boundsAlign, COMBO_CHANGED, SLOT(OnControlChanged()));
  49. HookWidget(ui->boundsWidth, DSCROLL_CHANGED, SLOT(OnControlChanged()));
  50. HookWidget(ui->boundsHeight, DSCROLL_CHANGED, SLOT(OnControlChanged()));
  51. HookWidget(ui->cropLeft, ISCROLL_CHANGED, SLOT(OnCropChanged()));
  52. HookWidget(ui->cropRight, ISCROLL_CHANGED, SLOT(OnCropChanged()));
  53. HookWidget(ui->cropTop, ISCROLL_CHANGED, SLOT(OnCropChanged()));
  54. HookWidget(ui->cropBottom, ISCROLL_CHANGED, SLOT(OnCropChanged()));
  55. ui->buttonBox->button(QDialogButtonBox::Close)->setDefault(true);
  56. connect(ui->buttonBox->button(QDialogButtonBox::Reset),
  57. SIGNAL(clicked()), this, SLOT(on_resetButton_clicked()));
  58. installEventFilter(CreateShortcutFilter());
  59. OBSSceneItem item = FindASelectedItem(main->GetCurrentScene());
  60. OBSScene scene = obs_sceneitem_get_scene(item);
  61. SetScene(scene);
  62. SetItem(item);
  63. channelChangedSignal.Connect(obs_get_signal_handler(), "channel_change",
  64. OBSChannelChanged, this);
  65. }
  66. void OBSBasicTransform::SetScene(OBSScene scene)
  67. {
  68. transformSignal.Disconnect();
  69. selectSignal.Disconnect();
  70. deselectSignal.Disconnect();
  71. removeSignal.Disconnect();
  72. if (scene) {
  73. OBSSource source = obs_scene_get_source(scene);
  74. signal_handler_t *signal = obs_source_get_signal_handler(source);
  75. transformSignal.Connect(signal, "item_transform",
  76. OBSSceneItemTransform, this);
  77. removeSignal.Connect(signal, "item_remove",
  78. OBSSceneItemRemoved, this);
  79. selectSignal.Connect(signal, "item_select",
  80. OBSSceneItemSelect, this);
  81. deselectSignal.Connect(signal, "item_deselect",
  82. OBSSceneItemDeselect, this);
  83. }
  84. }
  85. void OBSBasicTransform::SetItem(OBSSceneItem newItem)
  86. {
  87. QMetaObject::invokeMethod(this, "SetItemQt",
  88. Q_ARG(OBSSceneItem, OBSSceneItem(newItem)));
  89. }
  90. void OBSBasicTransform::SetItemQt(OBSSceneItem newItem)
  91. {
  92. item = newItem;
  93. if (item)
  94. RefreshControls();
  95. setEnabled(!!item);
  96. }
  97. void OBSBasicTransform::OBSChannelChanged(void *param, calldata_t *data)
  98. {
  99. OBSBasicTransform *window = reinterpret_cast<OBSBasicTransform*>(param);
  100. uint32_t channel = (uint32_t)calldata_int(data, "channel");
  101. OBSSource source = (obs_source_t*)calldata_ptr(data, "source");
  102. if (channel == 0) {
  103. OBSScene scene = obs_scene_from_source(source);
  104. window->SetScene(scene);
  105. if (!scene)
  106. window->SetItem(nullptr);
  107. else
  108. window->SetItem(FindASelectedItem(scene));
  109. }
  110. }
  111. void OBSBasicTransform::OBSSceneItemTransform(void *param, calldata_t *data)
  112. {
  113. OBSBasicTransform *window = reinterpret_cast<OBSBasicTransform*>(param);
  114. OBSSceneItem item = (obs_sceneitem_t*)calldata_ptr(data, "item");
  115. if (item == window->item && !window->ignoreTransformSignal)
  116. QMetaObject::invokeMethod(window, "RefreshControls");
  117. }
  118. void OBSBasicTransform::OBSSceneItemRemoved(void *param, calldata_t *data)
  119. {
  120. OBSBasicTransform *window = reinterpret_cast<OBSBasicTransform*>(param);
  121. OBSScene scene = (obs_scene_t*)calldata_ptr(data, "scene");
  122. OBSSceneItem item = (obs_sceneitem_t*)calldata_ptr(data, "item");
  123. if (item == window->item)
  124. window->SetItem(FindASelectedItem(scene));
  125. }
  126. void OBSBasicTransform::OBSSceneItemSelect(void *param, calldata_t *data)
  127. {
  128. OBSBasicTransform *window = reinterpret_cast<OBSBasicTransform*>(param);
  129. OBSSceneItem item = (obs_sceneitem_t*)calldata_ptr(data, "item");
  130. if (item != window->item)
  131. window->SetItem(item);
  132. }
  133. void OBSBasicTransform::OBSSceneItemDeselect(void *param, calldata_t *data)
  134. {
  135. OBSBasicTransform *window = reinterpret_cast<OBSBasicTransform*>(param);
  136. OBSScene scene = (obs_scene_t*)calldata_ptr(data, "scene");
  137. OBSSceneItem item = (obs_sceneitem_t*)calldata_ptr(data, "item");
  138. if (item == window->item)
  139. window->SetItem(FindASelectedItem(scene));
  140. }
  141. static const uint32_t listToAlign[] = {
  142. OBS_ALIGN_TOP | OBS_ALIGN_LEFT,
  143. OBS_ALIGN_TOP,
  144. OBS_ALIGN_TOP | OBS_ALIGN_RIGHT,
  145. OBS_ALIGN_LEFT,
  146. OBS_ALIGN_CENTER,
  147. OBS_ALIGN_RIGHT,
  148. OBS_ALIGN_BOTTOM | OBS_ALIGN_LEFT,
  149. OBS_ALIGN_BOTTOM,
  150. OBS_ALIGN_BOTTOM | OBS_ALIGN_RIGHT
  151. };
  152. static int AlignToList(uint32_t align)
  153. {
  154. int index = 0;
  155. for (uint32_t curAlign : listToAlign) {
  156. if (curAlign == align)
  157. return index;
  158. index++;
  159. }
  160. return 0;
  161. }
  162. void OBSBasicTransform::RefreshControls()
  163. {
  164. if (!item)
  165. return;
  166. obs_transform_info osi;
  167. obs_sceneitem_crop crop;
  168. obs_sceneitem_get_info(item, &osi);
  169. obs_sceneitem_get_crop(item, &crop);
  170. obs_source_t *source = obs_sceneitem_get_source(item);
  171. float width = float(obs_source_get_width(source));
  172. float height = float(obs_source_get_height(source));
  173. int alignIndex = AlignToList(osi.alignment);
  174. int boundsAlignIndex = AlignToList(osi.bounds_alignment);
  175. ignoreItemChange = true;
  176. ui->positionX->setValue(osi.pos.x);
  177. ui->positionY->setValue(osi.pos.y);
  178. ui->rotation->setValue(osi.rot);
  179. ui->sizeX->setValue(osi.scale.x * width);
  180. ui->sizeY->setValue(osi.scale.y * height);
  181. ui->align->setCurrentIndex(alignIndex);
  182. ui->boundsType->setCurrentIndex(int(osi.bounds_type));
  183. ui->boundsAlign->setCurrentIndex(boundsAlignIndex);
  184. ui->boundsWidth->setValue(osi.bounds.x);
  185. ui->boundsHeight->setValue(osi.bounds.y);
  186. ui->cropLeft->setValue(int(crop.left));
  187. ui->cropRight->setValue(int(crop.right));
  188. ui->cropTop->setValue(int(crop.top));
  189. ui->cropBottom->setValue(int(crop.bottom));
  190. ignoreItemChange = false;
  191. }
  192. void OBSBasicTransform::OnBoundsType(int index)
  193. {
  194. if (index == -1)
  195. return;
  196. obs_bounds_type type = (obs_bounds_type)index;
  197. bool enable = (type != OBS_BOUNDS_NONE);
  198. ui->boundsAlign->setEnabled(enable);
  199. ui->boundsWidth->setEnabled(enable);
  200. ui->boundsHeight->setEnabled(enable);
  201. if (!ignoreItemChange) {
  202. obs_bounds_type lastType = obs_sceneitem_get_bounds_type(item);
  203. if (lastType == OBS_BOUNDS_NONE) {
  204. OBSSource source = obs_sceneitem_get_source(item);
  205. int width = (int)obs_source_get_width(source);
  206. int height = (int)obs_source_get_height(source);
  207. ui->boundsWidth->setValue(width);
  208. ui->boundsHeight->setValue(height);
  209. }
  210. }
  211. OnControlChanged();
  212. }
  213. void OBSBasicTransform::OnControlChanged()
  214. {
  215. if (ignoreItemChange)
  216. return;
  217. obs_source_t *source = obs_sceneitem_get_source(item);
  218. double width = double(obs_source_get_width(source));
  219. double height = double(obs_source_get_height(source));
  220. obs_transform_info oti;
  221. oti.pos.x = float(ui->positionX->value());
  222. oti.pos.y = float(ui->positionY->value());
  223. oti.rot = float(ui->rotation->value());
  224. oti.scale.x = float(ui->sizeX->value() / width);
  225. oti.scale.y = float(ui->sizeY->value() / height);
  226. oti.alignment = listToAlign[ui->align->currentIndex()];
  227. oti.bounds_type = (obs_bounds_type)ui->boundsType->currentIndex();
  228. oti.bounds_alignment = listToAlign[ui->boundsAlign->currentIndex()];
  229. oti.bounds.x = float(ui->boundsWidth->value());
  230. oti.bounds.y = float(ui->boundsHeight->value());
  231. ignoreTransformSignal = true;
  232. obs_sceneitem_set_info(item, &oti);
  233. ignoreTransformSignal = false;
  234. }
  235. void OBSBasicTransform::OnCropChanged()
  236. {
  237. if (ignoreItemChange)
  238. return;
  239. obs_sceneitem_crop crop;
  240. crop.left = uint32_t(ui->cropLeft->value());
  241. crop.right = uint32_t(ui->cropRight->value());
  242. crop.top = uint32_t(ui->cropTop->value());
  243. crop.bottom = uint32_t(ui->cropBottom->value());
  244. ignoreTransformSignal = true;
  245. obs_sceneitem_set_crop(item, &crop);
  246. ignoreTransformSignal = false;
  247. }
  248. void OBSBasicTransform::on_resetButton_clicked()
  249. {
  250. main->on_actionResetTransform_triggered();
  251. }