obs-scripting-python-frontend.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450
  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. #include <obs-module.h>
  15. #include <obs-frontend-api.h>
  16. #include "obs-scripting-python.h"
  17. #define libobs_to_py(type, obs_obj, ownership, py_obj) \
  18. libobs_to_py_(#type " *", obs_obj, ownership, py_obj, NULL, __func__, __LINE__)
  19. #define py_to_libobs(type, py_obj, libobs_out) py_to_libobs_(#type " *", py_obj, libobs_out, NULL, __func__, __LINE__)
  20. /* ----------------------------------- */
  21. static PyObject *get_scene_names(PyObject *self, PyObject *args)
  22. {
  23. char **names = obs_frontend_get_scene_names();
  24. char **name = names;
  25. PyObject *list = PyList_New(0);
  26. while (name && *name) {
  27. PyObject *py_name = PyUnicode_FromString(*name);
  28. if (py_name) {
  29. PyList_Append(list, py_name);
  30. Py_DECREF(py_name);
  31. }
  32. name++;
  33. }
  34. UNUSED_PARAMETER(self);
  35. UNUSED_PARAMETER(args);
  36. bfree(names);
  37. return list;
  38. }
  39. static PyObject *get_scenes(PyObject *self, PyObject *args)
  40. {
  41. struct obs_frontend_source_list list = {0};
  42. obs_frontend_get_scenes(&list);
  43. PyObject *ret = PyList_New(0);
  44. for (size_t i = 0; i < list.sources.num; i++) {
  45. obs_source_t *source = list.sources.array[i];
  46. PyObject *py_source;
  47. if (libobs_to_py(obs_source_t, source, false, &py_source)) {
  48. PyList_Append(ret, py_source);
  49. Py_DECREF(py_source);
  50. }
  51. }
  52. UNUSED_PARAMETER(self);
  53. UNUSED_PARAMETER(args);
  54. da_free(list.sources);
  55. return ret;
  56. }
  57. static PyObject *get_current_scene(PyObject *self, PyObject *args)
  58. {
  59. obs_source_t *source = obs_frontend_get_current_scene();
  60. PyObject *py_source;
  61. if (!libobs_to_py(obs_source_t, source, false, &py_source)) {
  62. obs_source_release(source);
  63. return python_none();
  64. }
  65. UNUSED_PARAMETER(self);
  66. UNUSED_PARAMETER(args);
  67. return py_source;
  68. }
  69. static PyObject *set_current_scene(PyObject *self, PyObject *args)
  70. {
  71. PyObject *py_source;
  72. obs_source_t *source = NULL;
  73. if (!parse_args(args, "O", &py_source))
  74. return python_none();
  75. if (!py_to_libobs(obs_source_t, py_source, &source))
  76. return python_none();
  77. UNUSED_PARAMETER(self);
  78. obs_frontend_set_current_scene(source);
  79. return python_none();
  80. }
  81. static PyObject *get_transitions(PyObject *self, PyObject *args)
  82. {
  83. struct obs_frontend_source_list list = {0};
  84. obs_frontend_get_transitions(&list);
  85. PyObject *ret = PyList_New(0);
  86. for (size_t i = 0; i < list.sources.num; i++) {
  87. obs_source_t *source = list.sources.array[i];
  88. PyObject *py_source;
  89. if (libobs_to_py(obs_source_t, source, false, &py_source)) {
  90. PyList_Append(ret, py_source);
  91. Py_DECREF(py_source);
  92. }
  93. }
  94. UNUSED_PARAMETER(self);
  95. UNUSED_PARAMETER(args);
  96. da_free(list.sources);
  97. return ret;
  98. }
  99. static PyObject *get_current_transition(PyObject *self, PyObject *args)
  100. {
  101. obs_source_t *source = obs_frontend_get_current_transition();
  102. PyObject *py_source;
  103. if (!libobs_to_py(obs_source_t, source, false, &py_source)) {
  104. obs_source_release(source);
  105. return python_none();
  106. }
  107. UNUSED_PARAMETER(self);
  108. UNUSED_PARAMETER(args);
  109. return py_source;
  110. }
  111. static PyObject *set_current_transition(PyObject *self, PyObject *args)
  112. {
  113. PyObject *py_source;
  114. obs_source_t *source = NULL;
  115. if (!parse_args(args, "O", &py_source))
  116. return python_none();
  117. if (!py_to_libobs(obs_source_t, py_source, &source))
  118. return python_none();
  119. UNUSED_PARAMETER(self);
  120. obs_frontend_set_current_transition(source);
  121. return python_none();
  122. }
  123. static PyObject *get_transition_duration(PyObject *self, PyObject *args)
  124. {
  125. int duration = obs_frontend_get_transition_duration();
  126. PyObject *ret = PyLong_FromLong(duration);
  127. UNUSED_PARAMETER(self);
  128. UNUSED_PARAMETER(args);
  129. return ret;
  130. }
  131. static PyObject *set_transition_duration(PyObject *self, PyObject *args)
  132. {
  133. int duration;
  134. if (!parse_args(args, "i", &duration))
  135. return python_none();
  136. obs_frontend_set_transition_duration(duration);
  137. UNUSED_PARAMETER(self);
  138. return python_none();
  139. }
  140. static PyObject *get_scene_collections(PyObject *self, PyObject *args)
  141. {
  142. char **names = obs_frontend_get_scene_collections();
  143. char **name = names;
  144. PyObject *list = PyList_New(0);
  145. while (name && *name) {
  146. PyObject *py_name = PyUnicode_FromString(*name);
  147. if (py_name) {
  148. PyList_Append(list, py_name);
  149. Py_DECREF(py_name);
  150. }
  151. name++;
  152. }
  153. UNUSED_PARAMETER(self);
  154. UNUSED_PARAMETER(args);
  155. bfree(names);
  156. return list;
  157. }
  158. static PyObject *get_current_scene_collection(PyObject *self, PyObject *args)
  159. {
  160. char *name = obs_frontend_get_current_scene_collection();
  161. PyObject *ret = PyUnicode_FromString(name);
  162. bfree(name);
  163. UNUSED_PARAMETER(self);
  164. UNUSED_PARAMETER(args);
  165. return ret;
  166. }
  167. static PyObject *set_current_scene_collection(PyObject *self, PyObject *args)
  168. {
  169. const char *name;
  170. if (!parse_args(args, "s", &name))
  171. return python_none();
  172. UNUSED_PARAMETER(self);
  173. obs_frontend_set_current_scene_collection(name);
  174. return python_none();
  175. }
  176. static PyObject *get_profiles(PyObject *self, PyObject *args)
  177. {
  178. char **names = obs_frontend_get_profiles();
  179. char **name = names;
  180. PyObject *list = PyList_New(0);
  181. while (name && *name) {
  182. PyObject *py_name = PyUnicode_FromString(*name);
  183. if (py_name) {
  184. PyList_Append(list, py_name);
  185. Py_DECREF(py_name);
  186. }
  187. name++;
  188. }
  189. UNUSED_PARAMETER(self);
  190. UNUSED_PARAMETER(args);
  191. bfree(names);
  192. return list;
  193. }
  194. static PyObject *get_current_profile(PyObject *self, PyObject *args)
  195. {
  196. char *name = obs_frontend_get_current_profile();
  197. PyObject *ret = PyUnicode_FromString(name);
  198. bfree(name);
  199. UNUSED_PARAMETER(self);
  200. UNUSED_PARAMETER(args);
  201. return ret;
  202. }
  203. static PyObject *set_current_profile(PyObject *self, PyObject *args)
  204. {
  205. const char *name;
  206. if (!parse_args(args, "s", &name))
  207. return python_none();
  208. UNUSED_PARAMETER(self);
  209. obs_frontend_set_current_profile(name);
  210. return python_none();
  211. }
  212. /* ----------------------------------- */
  213. static void frontend_save_callback(obs_data_t *save_data, bool saving, void *priv)
  214. {
  215. struct python_obs_callback *cb = priv;
  216. if (script_callback_removed(&cb->base)) {
  217. obs_frontend_remove_save_callback(frontend_save_callback, cb);
  218. return;
  219. }
  220. lock_python();
  221. PyObject *py_save_data;
  222. if (libobs_to_py(obs_data_t, save_data, false, &py_save_data)) {
  223. PyObject *args = Py_BuildValue("(ON)", py_save_data, PyBool_FromLong(saving));
  224. struct python_obs_callback *last_cb = cur_python_cb;
  225. cur_python_cb = cb;
  226. cur_python_script = (struct obs_python_script *)cb->base.script;
  227. PyObject *py_ret = PyObject_CallObject(cb->func, args);
  228. Py_XDECREF(py_ret);
  229. py_error();
  230. cur_python_script = NULL;
  231. cur_python_cb = last_cb;
  232. Py_XDECREF(args);
  233. Py_XDECREF(py_save_data);
  234. }
  235. unlock_python();
  236. }
  237. static PyObject *remove_save_callback(PyObject *self, PyObject *args)
  238. {
  239. struct obs_python_script *script = cur_python_script;
  240. PyObject *py_cb = NULL;
  241. UNUSED_PARAMETER(self);
  242. if (!parse_args(args, "O", &py_cb))
  243. return python_none();
  244. if (!py_cb || !PyFunction_Check(py_cb))
  245. return python_none();
  246. struct python_obs_callback *cb = find_python_obs_callback(script, py_cb);
  247. if (cb)
  248. remove_python_obs_callback(cb);
  249. return python_none();
  250. }
  251. static void add_save_callback_defer(void *cb)
  252. {
  253. obs_frontend_add_save_callback(frontend_save_callback, cb);
  254. }
  255. static PyObject *add_save_callback(PyObject *self, PyObject *args)
  256. {
  257. struct obs_python_script *script = cur_python_script;
  258. PyObject *py_cb = NULL;
  259. UNUSED_PARAMETER(self);
  260. if (!parse_args(args, "O", &py_cb))
  261. return python_none();
  262. if (!py_cb || !PyFunction_Check(py_cb))
  263. return python_none();
  264. struct python_obs_callback *cb = add_python_obs_callback(script, py_cb);
  265. defer_call_post(add_save_callback_defer, cb);
  266. return python_none();
  267. }
  268. static void frontend_event_callback(enum obs_frontend_event event, void *priv)
  269. {
  270. struct python_obs_callback *cb = priv;
  271. if (script_callback_removed(&cb->base)) {
  272. obs_frontend_remove_event_callback(frontend_event_callback, cb);
  273. return;
  274. }
  275. lock_python();
  276. PyObject *args = Py_BuildValue("(i)", event);
  277. struct python_obs_callback *last_cb = cur_python_cb;
  278. cur_python_cb = cb;
  279. cur_python_script = (struct obs_python_script *)cb->base.script;
  280. PyObject *py_ret = PyObject_CallObject(cb->func, args);
  281. Py_XDECREF(py_ret);
  282. py_error();
  283. cur_python_script = NULL;
  284. cur_python_cb = last_cb;
  285. Py_XDECREF(args);
  286. unlock_python();
  287. }
  288. static PyObject *remove_event_callback(PyObject *self, PyObject *args)
  289. {
  290. struct obs_python_script *script = cur_python_script;
  291. PyObject *py_cb = NULL;
  292. UNUSED_PARAMETER(self);
  293. if (!parse_args(args, "O", &py_cb))
  294. return python_none();
  295. if (!py_cb || !PyFunction_Check(py_cb))
  296. return python_none();
  297. struct python_obs_callback *cb = find_python_obs_callback(script, py_cb);
  298. if (cb)
  299. remove_python_obs_callback(cb);
  300. return python_none();
  301. }
  302. static void add_event_callback_defer(void *cb)
  303. {
  304. obs_frontend_add_event_callback(frontend_event_callback, cb);
  305. }
  306. static PyObject *add_event_callback(PyObject *self, PyObject *args)
  307. {
  308. struct obs_python_script *script = cur_python_script;
  309. PyObject *py_cb = NULL;
  310. UNUSED_PARAMETER(self);
  311. if (!parse_args(args, "O", &py_cb))
  312. return python_none();
  313. if (!py_cb || !PyFunction_Check(py_cb))
  314. return python_none();
  315. struct python_obs_callback *cb = add_python_obs_callback(script, py_cb);
  316. defer_call_post(add_event_callback_defer, cb);
  317. return python_none();
  318. }
  319. /* ----------------------------------- */
  320. void add_python_frontend_funcs(PyObject *module)
  321. {
  322. static PyMethodDef funcs[] = {
  323. #define DEF_FUNC(c) {"obs_frontend_" #c, c, METH_VARARGS, NULL}
  324. DEF_FUNC(get_scene_names),
  325. DEF_FUNC(get_scenes),
  326. DEF_FUNC(get_current_scene),
  327. DEF_FUNC(set_current_scene),
  328. DEF_FUNC(get_transitions),
  329. DEF_FUNC(get_current_transition),
  330. DEF_FUNC(set_current_transition),
  331. DEF_FUNC(set_transition_duration),
  332. DEF_FUNC(get_transition_duration),
  333. DEF_FUNC(get_scene_collections),
  334. DEF_FUNC(get_current_scene_collection),
  335. DEF_FUNC(set_current_scene_collection),
  336. DEF_FUNC(get_profiles),
  337. DEF_FUNC(get_current_profile),
  338. DEF_FUNC(set_current_profile),
  339. DEF_FUNC(remove_save_callback),
  340. DEF_FUNC(add_save_callback),
  341. DEF_FUNC(remove_event_callback),
  342. DEF_FUNC(add_event_callback),
  343. #undef DEF_FUNC
  344. {0}};
  345. add_functions_to_py_module(module, funcs);
  346. }