transition-luma-wipe.c 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233
  1. #include <obs-module.h>
  2. #include <graphics/image-file.h>
  3. #include <util/dstr.h>
  4. /* clang-format off */
  5. #define S_LUMA_IMG "luma_image"
  6. #define S_LUMA_INV "luma_invert"
  7. #define S_LUMA_SOFT "luma_softness"
  8. #define T_LUMA_IMG obs_module_text("LumaWipe.Image")
  9. #define T_LUMA_INV obs_module_text("LumaWipe.Invert")
  10. #define T_LUMA_SOFT obs_module_text("LumaWipe.Softness")
  11. /* clang-format on */
  12. struct luma_wipe_info {
  13. obs_source_t *source;
  14. gs_effect_t *effect;
  15. gs_eparam_t *ep_a_tex;
  16. gs_eparam_t *ep_b_tex;
  17. gs_eparam_t *ep_l_tex;
  18. gs_eparam_t *ep_progress;
  19. gs_eparam_t *ep_invert;
  20. gs_eparam_t *ep_softness;
  21. gs_image_file_t luma_image;
  22. bool invert_luma;
  23. float softness;
  24. obs_data_t *wipes_list;
  25. };
  26. static const char *luma_wipe_get_name(void *type_data)
  27. {
  28. UNUSED_PARAMETER(type_data);
  29. return obs_module_text("LumaWipeTransition");
  30. }
  31. static void luma_wipe_update(void *data, obs_data_t *settings)
  32. {
  33. struct luma_wipe_info *lwipe = data;
  34. const char *name = obs_data_get_string(settings, S_LUMA_IMG);
  35. lwipe->invert_luma = obs_data_get_bool(settings, S_LUMA_INV);
  36. lwipe->softness = (float)obs_data_get_double(settings, S_LUMA_SOFT);
  37. struct dstr path = {0};
  38. dstr_copy(&path, "luma_wipes/");
  39. dstr_cat(&path, name);
  40. char *file = obs_module_file(path.array);
  41. obs_enter_graphics();
  42. gs_image_file_free(&lwipe->luma_image);
  43. obs_leave_graphics();
  44. gs_image_file_init(&lwipe->luma_image, file);
  45. obs_enter_graphics();
  46. gs_image_file_init_texture(&lwipe->luma_image);
  47. obs_leave_graphics();
  48. bfree(file);
  49. dstr_free(&path);
  50. UNUSED_PARAMETER(settings);
  51. }
  52. static void luma_wipe_get_list(void *data)
  53. {
  54. struct luma_wipe_info *lwipe = data;
  55. char *path = obs_module_file("luma_wipes/wipes.json");
  56. lwipe->wipes_list = obs_data_create_from_json_file(path);
  57. bfree(path);
  58. }
  59. static void *luma_wipe_create(obs_data_t *settings, obs_source_t *source)
  60. {
  61. struct luma_wipe_info *lwipe;
  62. gs_effect_t *effect;
  63. char *file = obs_module_file("luma_wipe_transition.effect");
  64. obs_enter_graphics();
  65. effect = gs_effect_create_from_file(file, NULL);
  66. obs_leave_graphics();
  67. if (!effect) {
  68. blog(LOG_ERROR, "Could not open luma_wipe_transition.effect");
  69. return NULL;
  70. }
  71. bfree(file);
  72. lwipe = bzalloc(sizeof(*lwipe));
  73. lwipe->effect = effect;
  74. lwipe->ep_a_tex = gs_effect_get_param_by_name(effect, "a_tex");
  75. lwipe->ep_b_tex = gs_effect_get_param_by_name(effect, "b_tex");
  76. lwipe->ep_l_tex = gs_effect_get_param_by_name(effect, "l_tex");
  77. lwipe->ep_progress = gs_effect_get_param_by_name(effect, "progress");
  78. lwipe->ep_invert = gs_effect_get_param_by_name(effect, "invert");
  79. lwipe->ep_softness = gs_effect_get_param_by_name(effect, "softness");
  80. lwipe->source = source;
  81. luma_wipe_get_list(lwipe);
  82. luma_wipe_update(lwipe, settings);
  83. return lwipe;
  84. }
  85. static void luma_wipe_destroy(void *data)
  86. {
  87. struct luma_wipe_info *lwipe = data;
  88. obs_enter_graphics();
  89. gs_image_file_free(&lwipe->luma_image);
  90. obs_leave_graphics();
  91. obs_data_release(lwipe->wipes_list);
  92. bfree(lwipe);
  93. }
  94. static obs_properties_t *luma_wipe_properties(void *data)
  95. {
  96. obs_properties_t *props = obs_properties_create();
  97. struct luma_wipe_info *lwipe = data;
  98. obs_property_t *p;
  99. p = obs_properties_add_list(props, S_LUMA_IMG, T_LUMA_IMG,
  100. OBS_COMBO_TYPE_LIST,
  101. OBS_COMBO_FORMAT_STRING);
  102. obs_data_item_t *item = obs_data_first(lwipe->wipes_list);
  103. for (; item != NULL; obs_data_item_next(&item)) {
  104. const char *name = obs_data_item_get_name(item);
  105. const char *path = obs_data_item_get_string(item);
  106. obs_property_list_add_string(p, obs_module_text(name), path);
  107. }
  108. obs_properties_add_float(props, S_LUMA_SOFT, T_LUMA_SOFT, 0.0, 1.0,
  109. 0.05);
  110. obs_properties_add_bool(props, S_LUMA_INV, T_LUMA_INV);
  111. return props;
  112. }
  113. static void luma_wipe_defaults(obs_data_t *settings)
  114. {
  115. obs_data_set_default_string(settings, S_LUMA_IMG, "linear-h.png");
  116. obs_data_set_default_double(settings, S_LUMA_SOFT, 0.03);
  117. obs_data_set_default_bool(settings, S_LUMA_INV, false);
  118. }
  119. static void luma_wipe_callback(void *data, gs_texture_t *a, gs_texture_t *b,
  120. float t, uint32_t cx, uint32_t cy)
  121. {
  122. struct luma_wipe_info *lwipe = data;
  123. const bool previous = gs_framebuffer_srgb_enabled();
  124. gs_enable_framebuffer_srgb(true);
  125. gs_effect_set_texture_srgb(lwipe->ep_a_tex, a);
  126. gs_effect_set_texture_srgb(lwipe->ep_b_tex, b);
  127. gs_effect_set_texture(lwipe->ep_l_tex, lwipe->luma_image.texture);
  128. gs_effect_set_float(lwipe->ep_progress, t);
  129. gs_effect_set_bool(lwipe->ep_invert, lwipe->invert_luma);
  130. gs_effect_set_float(lwipe->ep_softness, lwipe->softness);
  131. while (gs_effect_loop(lwipe->effect, "LumaWipe"))
  132. gs_draw_sprite(NULL, 0, cx, cy);
  133. gs_enable_framebuffer_srgb(previous);
  134. }
  135. void luma_wipe_video_render(void *data, gs_effect_t *effect)
  136. {
  137. struct luma_wipe_info *lwipe = data;
  138. obs_transition_video_render(lwipe->source, luma_wipe_callback);
  139. UNUSED_PARAMETER(effect);
  140. }
  141. static float mix_a(void *data, float t)
  142. {
  143. UNUSED_PARAMETER(data);
  144. return 1.0f - t;
  145. }
  146. static float mix_b(void *data, float t)
  147. {
  148. UNUSED_PARAMETER(data);
  149. return t;
  150. }
  151. bool luma_wipe_audio_render(void *data, uint64_t *ts_out,
  152. struct obs_source_audio_mix *audio, uint32_t mixers,
  153. size_t channels, size_t sample_rate)
  154. {
  155. struct luma_wipe_info *lwipe = data;
  156. return obs_transition_audio_render(lwipe->source, ts_out, audio, mixers,
  157. channels, sample_rate, mix_a, mix_b);
  158. }
  159. static enum gs_color_space
  160. luma_wipe_video_get_color_space(void *data, size_t count,
  161. const enum gs_color_space *preferred_spaces)
  162. {
  163. struct luma_wipe_info *const lwipe = data;
  164. return obs_transition_video_get_color_space(lwipe->source);
  165. }
  166. struct obs_source_info luma_wipe_transition = {
  167. .id = "wipe_transition",
  168. .type = OBS_SOURCE_TYPE_TRANSITION,
  169. .get_name = luma_wipe_get_name,
  170. .create = luma_wipe_create,
  171. .destroy = luma_wipe_destroy,
  172. .update = luma_wipe_update,
  173. .video_render = luma_wipe_video_render,
  174. .audio_render = luma_wipe_audio_render,
  175. .get_properties = luma_wipe_properties,
  176. .get_defaults = luma_wipe_defaults,
  177. .video_get_color_space = luma_wipe_video_get_color_space,
  178. };