obs-scene.c 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. /******************************************************************************
  2. Copyright (C) 2013 by Hugh 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 "graphics/math-defs.h"
  15. #include "obs-scene.h"
  16. static const char *scene_getname(const char *locale)
  17. {
  18. /* TODO: locale lookup of display name */
  19. return "Scene";
  20. }
  21. static void *scene_create(const char *settings, struct obs_source *source)
  22. {
  23. struct obs_scene *scene = bmalloc(sizeof(struct obs_scene));
  24. scene->source = source;
  25. da_init(scene->items);
  26. return scene;
  27. }
  28. static void scene_destroy(void *data)
  29. {
  30. struct obs_scene *scene = data;
  31. size_t i;
  32. for (i = 0; i < scene->items.num; i++) {
  33. struct obs_scene_item *item = scene->items.array[i];
  34. if (item->source)
  35. obs_source_release(item->source);
  36. bfree(item);
  37. }
  38. da_free(scene->items);
  39. bfree(scene);
  40. }
  41. static uint32_t scene_get_output_flags(void *data)
  42. {
  43. return SOURCE_VIDEO;
  44. }
  45. static void scene_video_render(void *data)
  46. {
  47. struct obs_scene *scene = data;
  48. size_t i;
  49. for (i = scene->items.num; i > 0; i--) {
  50. struct obs_scene_item *item = scene->items.array[i-1];
  51. if (obs_source_removed(item->source)) {
  52. obs_source_release(item->source);
  53. da_erase(scene->items, i--);
  54. continue;
  55. }
  56. gs_matrix_push();
  57. gs_matrix_translate3f(item->origin.x, item->origin.y, 0.0f);
  58. gs_matrix_scale3f(item->scale.x, item->scale.y, 1.0f);
  59. gs_matrix_rotaa4f(0.0f, 0.0f, 1.0f, RAD(-item->rot));
  60. gs_matrix_translate3f(-item->pos.x, -item->pos.y, 0.0f);
  61. obs_source_video_render(item->source);
  62. gs_matrix_pop();
  63. }
  64. }
  65. static uint32_t scene_getsize(void *data)
  66. {
  67. return 0;
  68. }
  69. static const struct source_info scene_info =
  70. {
  71. .id = "scene",
  72. .getname = scene_getname,
  73. .create = scene_create,
  74. .destroy = scene_destroy,
  75. .get_output_flags = scene_get_output_flags,
  76. .video_render = scene_video_render,
  77. .getwidth = scene_getsize,
  78. .getheight = scene_getsize,
  79. };
  80. obs_scene_t obs_scene_create(void)
  81. {
  82. struct obs_source *source = bmalloc(sizeof(struct obs_source));
  83. struct obs_scene *scene = scene_create(NULL, source);
  84. memset(source, 0, sizeof(struct obs_source));
  85. source->data = scene;
  86. if (!source->data) {
  87. bfree(source);
  88. return NULL;
  89. }
  90. scene->source = source;
  91. obs_source_init(source, NULL, &scene_info);
  92. memcpy(&source->callbacks, &scene_info, sizeof(struct source_info));
  93. return scene;
  94. }
  95. void obs_scene_destroy(obs_scene_t scene)
  96. {
  97. if (scene)
  98. obs_source_release(scene->source);
  99. }
  100. obs_source_t obs_scene_getsource(obs_scene_t scene)
  101. {
  102. return scene->source;
  103. }
  104. obs_sceneitem_t obs_scene_add(obs_scene_t scene, obs_source_t source)
  105. {
  106. struct obs_scene_item *item = bmalloc(sizeof(struct obs_scene_item));
  107. memset(item, 0, sizeof(struct obs_scene_item));
  108. item->source = source;
  109. item->visible = true;
  110. item->parent = scene;
  111. vec2_set(&item->scale, 1.0f, 1.0f);
  112. if (source)
  113. obs_source_addref(source);
  114. da_push_back(scene->items, &item);
  115. return item;
  116. }
  117. void obs_sceneitem_destroy(obs_sceneitem_t item)
  118. {
  119. if (item) {
  120. if (item->source)
  121. obs_source_release(item->source);
  122. da_erase_item(item->parent->items, item);
  123. bfree(item);
  124. }
  125. }
  126. void obs_sceneitem_setpos(obs_sceneitem_t item, const struct vec2 *pos)
  127. {
  128. vec2_copy(&item->pos, pos);
  129. }
  130. void obs_sceneitem_setrot(obs_sceneitem_t item, float rot)
  131. {
  132. item->rot = rot;
  133. }
  134. void obs_sceneitem_setorigin(obs_sceneitem_t item, const struct vec2 *origin)
  135. {
  136. vec2_copy(&item->origin, origin);
  137. }
  138. void obs_sceneitem_setscale(obs_sceneitem_t item, const struct vec2 *scale)
  139. {
  140. vec2_copy(&item->scale, scale);
  141. }
  142. void obs_sceneitem_setorder(obs_sceneitem_t item, enum order_movement movement)
  143. {
  144. struct obs_scene *scene = item->parent;
  145. if (movement == ORDER_MOVE_UP) {
  146. size_t idx = da_find(scene->items, &item, 0);
  147. if (idx > 0)
  148. da_move_item(scene->items, idx, idx-1);
  149. } else if (movement == ORDER_MOVE_DOWN) {
  150. size_t idx = da_find(scene->items, &item, 0);
  151. if (idx < (scene->items.num-1))
  152. da_move_item(scene->items, idx, idx+1);
  153. } else if (movement == ORDER_MOVE_TOP) {
  154. size_t idx = da_find(scene->items, &item, 0);
  155. if (idx > 0)
  156. da_move_item(scene->items, idx, 0);
  157. } else if (movement == ORDER_MOVE_TOP) {
  158. size_t idx = da_find(scene->items, &item, 0);
  159. if (idx < (scene->items.num-1))
  160. da_move_item(scene->items, idx, scene->items.num-1);
  161. }
  162. }
  163. void obs_sceneitem_getpos(obs_sceneitem_t item, struct vec2 *pos)
  164. {
  165. vec2_copy(pos, &item->pos);
  166. }
  167. float obs_sceneitem_getrot(obs_sceneitem_t item)
  168. {
  169. return item->rot;
  170. }
  171. void obs_sceneitem_getorigin(obs_sceneitem_t item, struct vec2 *origin)
  172. {
  173. vec2_copy(origin, &item->origin);
  174. }
  175. void obs_sceneitem_getscale(obs_sceneitem_t item, struct vec2 *scale)
  176. {
  177. vec2_copy(scale, &item->scale);
  178. }