obs.hpp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342
  1. /******************************************************************************
  2. Copyright (C) 2023 by Lain Bailey <[email protected]>
  3. This program is free software: you can redistribute it and/or modify
  4. it under the terms of the GNU General Public License as published by
  5. the Free Software Foundation, either version 2 of the License, or
  6. (at your option) any later version.
  7. This program is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. GNU General Public License for more details.
  11. You should have received a copy of the GNU General Public License
  12. along with this program. If not, see <http://www.gnu.org/licenses/>.
  13. ******************************************************************************/
  14. /* Useful C++ classes and bindings for base obs data */
  15. #pragma once
  16. #include "obs.h"
  17. /* RAII wrappers */
  18. template<typename T, void release(T)> class OBSRefAutoRelease;
  19. template<typename T, void addref(T), void release(T)> class OBSRef;
  20. template<typename T, T getref(T), void release(T)> class OBSSafeRef;
  21. using OBSObject = OBSSafeRef<obs_object_t *, obs_object_get_ref, obs_object_release>;
  22. using OBSSource = OBSSafeRef<obs_source_t *, obs_source_get_ref, obs_source_release>;
  23. using OBSScene = OBSSafeRef<obs_scene_t *, obs_scene_get_ref, obs_scene_release>;
  24. using OBSSceneItem = OBSRef<obs_sceneitem_t *, obs_sceneitem_addref, obs_sceneitem_release>;
  25. using OBSData = OBSRef<obs_data_t *, obs_data_addref, obs_data_release>;
  26. using OBSDataArray = OBSRef<obs_data_array_t *, obs_data_array_addref, obs_data_array_release>;
  27. using OBSOutput = OBSSafeRef<obs_output_t *, obs_output_get_ref, obs_output_release>;
  28. using OBSEncoder = OBSSafeRef<obs_encoder_t *, obs_encoder_get_ref, obs_encoder_release>;
  29. using OBSService = OBSSafeRef<obs_service_t *, obs_service_get_ref, obs_service_release>;
  30. using OBSCanvas = OBSSafeRef<obs_canvas_t *, obs_canvas_get_ref, obs_canvas_release>;
  31. using OBSWeakObject = OBSRef<obs_weak_object_t *, obs_weak_object_addref, obs_weak_object_release>;
  32. using OBSWeakSource = OBSRef<obs_weak_source_t *, obs_weak_source_addref, obs_weak_source_release>;
  33. using OBSWeakOutput = OBSRef<obs_weak_output_t *, obs_weak_output_addref, obs_weak_output_release>;
  34. using OBSWeakEncoder = OBSRef<obs_weak_encoder_t *, obs_weak_encoder_addref, obs_weak_encoder_release>;
  35. using OBSWeakService = OBSRef<obs_weak_service_t *, obs_weak_service_addref, obs_weak_service_release>;
  36. using OBSWeakCanvas = OBSRef<obs_weak_canvas_t *, obs_weak_canvas_addref, obs_weak_canvas_release>;
  37. #define OBS_AUTORELEASE
  38. using OBSObjectAutoRelease = OBSRefAutoRelease<obs_object_t *, obs_object_release>;
  39. using OBSSourceAutoRelease = OBSRefAutoRelease<obs_source_t *, obs_source_release>;
  40. using OBSSceneAutoRelease = OBSRefAutoRelease<obs_scene_t *, obs_scene_release>;
  41. using OBSSceneItemAutoRelease = OBSRefAutoRelease<obs_sceneitem_t *, obs_sceneitem_release>;
  42. using OBSDataAutoRelease = OBSRefAutoRelease<obs_data_t *, obs_data_release>;
  43. using OBSDataArrayAutoRelease = OBSRefAutoRelease<obs_data_array_t *, obs_data_array_release>;
  44. using OBSOutputAutoRelease = OBSRefAutoRelease<obs_output_t *, obs_output_release>;
  45. using OBSEncoderAutoRelease = OBSRefAutoRelease<obs_encoder_t *, obs_encoder_release>;
  46. using OBSServiceAutoRelease = OBSRefAutoRelease<obs_service_t *, obs_service_release>;
  47. using OBSCanvasAutoRelease = OBSRefAutoRelease<obs_canvas_t *, obs_canvas_release>;
  48. using OBSWeakObjectAutoRelease = OBSRefAutoRelease<obs_weak_object_t *, obs_weak_object_release>;
  49. using OBSWeakSourceAutoRelease = OBSRefAutoRelease<obs_weak_source_t *, obs_weak_source_release>;
  50. using OBSWeakOutputAutoRelease = OBSRefAutoRelease<obs_weak_output_t *, obs_weak_output_release>;
  51. using OBSWeakEncoderAutoRelease = OBSRefAutoRelease<obs_weak_encoder_t *, obs_weak_encoder_release>;
  52. using OBSWeakServiceAutoRelease = OBSRefAutoRelease<obs_weak_service_t *, obs_weak_service_release>;
  53. using OBSWeakCanvasAutoRelease = OBSRefAutoRelease<obs_weak_canvas_t *, obs_weak_canvas_release>;
  54. template<typename T, void release(T)> class OBSRefAutoRelease {
  55. protected:
  56. T val;
  57. public:
  58. inline OBSRefAutoRelease() : val(nullptr) {}
  59. inline OBSRefAutoRelease(T val_) : val(val_) {}
  60. OBSRefAutoRelease(const OBSRefAutoRelease &ref) = delete;
  61. inline OBSRefAutoRelease(OBSRefAutoRelease &&ref) : val(ref.val) { ref.val = nullptr; }
  62. inline ~OBSRefAutoRelease() { release(val); }
  63. inline operator T() const { return val; }
  64. inline T Get() const { return val; }
  65. inline bool operator==(T p) const { return val == p; }
  66. inline bool operator!=(T p) const { return val != p; }
  67. inline OBSRefAutoRelease &operator=(OBSRefAutoRelease &&ref)
  68. {
  69. if (this != &ref) {
  70. release(val);
  71. val = ref.val;
  72. ref.val = nullptr;
  73. }
  74. return *this;
  75. }
  76. inline OBSRefAutoRelease &operator=(T new_val)
  77. {
  78. release(val);
  79. val = new_val;
  80. return *this;
  81. }
  82. };
  83. template<typename T, void addref(T), void release(T)> class OBSRef : public OBSRefAutoRelease<T, release> {
  84. inline OBSRef &Replace(T valIn)
  85. {
  86. addref(valIn);
  87. release(this->val);
  88. this->val = valIn;
  89. return *this;
  90. }
  91. struct TakeOwnership {};
  92. inline OBSRef(T val_, TakeOwnership) : OBSRefAutoRelease<T, release>::OBSRefAutoRelease(val_) {}
  93. public:
  94. inline OBSRef() : OBSRefAutoRelease<T, release>::OBSRefAutoRelease(nullptr) {}
  95. inline OBSRef(const OBSRef &ref) : OBSRefAutoRelease<T, release>::OBSRefAutoRelease(ref.val)
  96. {
  97. addref(this->val);
  98. }
  99. inline OBSRef(T val_) : OBSRefAutoRelease<T, release>::OBSRefAutoRelease(val_) { addref(this->val); }
  100. inline OBSRef &operator=(const OBSRef &ref) { return Replace(ref.val); }
  101. inline OBSRef &operator=(T valIn) { return Replace(valIn); }
  102. friend OBSWeakObject OBSGetWeakRef(obs_object_t *object);
  103. friend OBSWeakSource OBSGetWeakRef(obs_source_t *source);
  104. friend OBSWeakOutput OBSGetWeakRef(obs_output_t *output);
  105. friend OBSWeakEncoder OBSGetWeakRef(obs_encoder_t *encoder);
  106. friend OBSWeakService OBSGetWeakRef(obs_service_t *service);
  107. friend OBSWeakCanvas OBSGetWeakRef(obs_canvas_t *canvas);
  108. };
  109. template<typename T, T getref(T), void release(T)> class OBSSafeRef : public OBSRefAutoRelease<T, release> {
  110. inline OBSSafeRef &Replace(T valIn)
  111. {
  112. T newVal = getref(valIn);
  113. release(this->val);
  114. this->val = newVal;
  115. return *this;
  116. }
  117. struct TakeOwnership {};
  118. inline OBSSafeRef(T val_, TakeOwnership) : OBSRefAutoRelease<T, release>::OBSRefAutoRelease(val_) {}
  119. public:
  120. inline OBSSafeRef() : OBSRefAutoRelease<T, release>::OBSRefAutoRelease(nullptr) {}
  121. inline OBSSafeRef(const OBSSafeRef &ref) : OBSRefAutoRelease<T, release>::OBSRefAutoRelease(ref.val)
  122. {
  123. this->val = getref(ref.val);
  124. }
  125. inline OBSSafeRef(T val_) : OBSRefAutoRelease<T, release>::OBSRefAutoRelease(val_)
  126. {
  127. this->val = getref(this->val);
  128. }
  129. inline OBSSafeRef &operator=(const OBSSafeRef &ref) { return Replace(ref.val); }
  130. inline OBSSafeRef &operator=(T valIn) { return Replace(valIn); }
  131. friend OBSObject OBSGetStrongRef(obs_weak_object_t *weak);
  132. friend OBSSource OBSGetStrongRef(obs_weak_source_t *weak);
  133. friend OBSOutput OBSGetStrongRef(obs_weak_output_t *weak);
  134. friend OBSEncoder OBSGetStrongRef(obs_weak_encoder_t *weak);
  135. friend OBSService OBSGetStrongRef(obs_weak_service_t *weak);
  136. friend OBSCanvas OBSGetStrongRef(obs_weak_canvas_t *weak);
  137. };
  138. inline OBSObject OBSGetStrongRef(obs_weak_object_t *weak)
  139. {
  140. return {obs_weak_object_get_object(weak), OBSObject::TakeOwnership()};
  141. }
  142. inline OBSWeakObject OBSGetWeakRef(obs_object_t *object)
  143. {
  144. return {obs_object_get_weak_object(object), OBSWeakObject::TakeOwnership()};
  145. }
  146. inline OBSSource OBSGetStrongRef(obs_weak_source_t *weak)
  147. {
  148. return {obs_weak_source_get_source(weak), OBSSource::TakeOwnership()};
  149. }
  150. inline OBSWeakSource OBSGetWeakRef(obs_source_t *source)
  151. {
  152. return {obs_source_get_weak_source(source), OBSWeakSource::TakeOwnership()};
  153. }
  154. inline OBSOutput OBSGetStrongRef(obs_weak_output_t *weak)
  155. {
  156. return {obs_weak_output_get_output(weak), OBSOutput::TakeOwnership()};
  157. }
  158. inline OBSWeakOutput OBSGetWeakRef(obs_output_t *output)
  159. {
  160. return {obs_output_get_weak_output(output), OBSWeakOutput::TakeOwnership()};
  161. }
  162. inline OBSEncoder OBSGetStrongRef(obs_weak_encoder_t *weak)
  163. {
  164. return {obs_weak_encoder_get_encoder(weak), OBSEncoder::TakeOwnership()};
  165. }
  166. inline OBSWeakEncoder OBSGetWeakRef(obs_encoder_t *encoder)
  167. {
  168. return {obs_encoder_get_weak_encoder(encoder), OBSWeakEncoder::TakeOwnership()};
  169. }
  170. inline OBSService OBSGetStrongRef(obs_weak_service_t *weak)
  171. {
  172. return {obs_weak_service_get_service(weak), OBSService::TakeOwnership()};
  173. }
  174. inline OBSWeakService OBSGetWeakRef(obs_service_t *service)
  175. {
  176. return {obs_service_get_weak_service(service), OBSWeakService::TakeOwnership()};
  177. }
  178. inline OBSCanvas OBSGetStrongRef(obs_weak_canvas_t *canvas)
  179. {
  180. return {obs_weak_canvas_get_canvas(canvas), OBSCanvas::TakeOwnership()};
  181. }
  182. inline OBSWeakCanvas OBSGetWeakRef(obs_canvas_t *canvas)
  183. {
  184. return {obs_canvas_get_weak_canvas(canvas), OBSWeakCanvas::TakeOwnership()};
  185. }
  186. /* objects that are not meant to be instanced */
  187. template<typename T, void destroy(T)> class OBSPtr {
  188. T obj;
  189. public:
  190. inline OBSPtr() : obj(nullptr) {}
  191. inline OBSPtr(T obj_) : obj(obj_) {}
  192. inline OBSPtr(const OBSPtr &) = delete;
  193. inline OBSPtr(OBSPtr &&other) : obj(other.obj) { other.obj = nullptr; }
  194. inline ~OBSPtr() { destroy(obj); }
  195. inline OBSPtr &operator=(T obj_)
  196. {
  197. if (obj_ != obj)
  198. destroy(obj);
  199. obj = obj_;
  200. return *this;
  201. }
  202. inline OBSPtr &operator=(const OBSPtr &) = delete;
  203. inline OBSPtr &operator=(OBSPtr &&other)
  204. {
  205. if (obj)
  206. destroy(obj);
  207. obj = other.obj;
  208. other.obj = nullptr;
  209. return *this;
  210. }
  211. inline operator T() const { return obj; }
  212. inline bool operator==(T p) const { return obj == p; }
  213. inline bool operator!=(T p) const { return obj != p; }
  214. };
  215. using OBSDisplay = OBSPtr<obs_display_t *, obs_display_destroy>;
  216. using OBSView = OBSPtr<obs_view_t *, obs_view_destroy>;
  217. using OBSFader = OBSPtr<obs_fader_t *, obs_fader_destroy>;
  218. using OBSVolMeter = OBSPtr<obs_volmeter_t *, obs_volmeter_destroy>;
  219. /* signal handler connection */
  220. class OBSSignal {
  221. signal_handler_t *handler;
  222. const char *signal;
  223. signal_callback_t callback;
  224. void *param;
  225. public:
  226. inline OBSSignal() : handler(nullptr), signal(nullptr), callback(nullptr), param(nullptr) {}
  227. inline OBSSignal(signal_handler_t *handler_, const char *signal_, signal_callback_t callback_, void *param_)
  228. : handler(handler_),
  229. signal(signal_),
  230. callback(callback_),
  231. param(param_)
  232. {
  233. signal_handler_connect_ref(handler, signal, callback, param);
  234. }
  235. inline void Disconnect()
  236. {
  237. signal_handler_disconnect(handler, signal, callback, param);
  238. handler = nullptr;
  239. signal = nullptr;
  240. callback = nullptr;
  241. param = nullptr;
  242. }
  243. inline ~OBSSignal() { Disconnect(); }
  244. inline void Connect(signal_handler_t *handler_, const char *signal_, signal_callback_t callback_, void *param_)
  245. {
  246. Disconnect();
  247. handler = handler_;
  248. signal = signal_;
  249. callback = callback_;
  250. param = param_;
  251. signal_handler_connect_ref(handler, signal, callback, param);
  252. }
  253. OBSSignal(const OBSSignal &) = delete;
  254. OBSSignal(OBSSignal &&other) noexcept
  255. : handler(other.handler),
  256. signal(other.signal),
  257. callback(other.callback),
  258. param(other.param)
  259. {
  260. other.handler = nullptr;
  261. other.signal = nullptr;
  262. other.callback = nullptr;
  263. other.param = nullptr;
  264. }
  265. OBSSignal &operator=(const OBSSignal &) = delete;
  266. OBSSignal &operator=(OBSSignal &&other) noexcept
  267. {
  268. Disconnect();
  269. handler = other.handler;
  270. signal = other.signal;
  271. callback = other.callback;
  272. param = other.param;
  273. other.handler = nullptr;
  274. other.signal = nullptr;
  275. other.callback = nullptr;
  276. other.param = nullptr;
  277. return *this;
  278. }
  279. };