transition-luma-wipe.c 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230
  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. }
  51. static void luma_wipe_get_list(void *data)
  52. {
  53. struct luma_wipe_info *lwipe = data;
  54. char *path = obs_module_file("luma_wipes/wipes.json");
  55. lwipe->wipes_list = obs_data_create_from_json_file(path);
  56. bfree(path);
  57. }
  58. static void *luma_wipe_create(obs_data_t *settings, obs_source_t *source)
  59. {
  60. struct luma_wipe_info *lwipe;
  61. gs_effect_t *effect;
  62. char *file = obs_module_file("luma_wipe_transition.effect");
  63. obs_enter_graphics();
  64. effect = gs_effect_create_from_file(file, NULL);
  65. obs_leave_graphics();
  66. if (!effect) {
  67. blog(LOG_ERROR, "Could not open luma_wipe_transition.effect");
  68. return NULL;
  69. }
  70. bfree(file);
  71. lwipe = bzalloc(sizeof(*lwipe));
  72. lwipe->effect = effect;
  73. lwipe->ep_a_tex = gs_effect_get_param_by_name(effect, "a_tex");
  74. lwipe->ep_b_tex = gs_effect_get_param_by_name(effect, "b_tex");
  75. lwipe->ep_l_tex = gs_effect_get_param_by_name(effect, "l_tex");
  76. lwipe->ep_progress = gs_effect_get_param_by_name(effect, "progress");
  77. lwipe->ep_invert = gs_effect_get_param_by_name(effect, "invert");
  78. lwipe->ep_softness = gs_effect_get_param_by_name(effect, "softness");
  79. lwipe->source = source;
  80. luma_wipe_get_list(lwipe);
  81. luma_wipe_update(lwipe, settings);
  82. return lwipe;
  83. }
  84. static void luma_wipe_destroy(void *data)
  85. {
  86. struct luma_wipe_info *lwipe = data;
  87. obs_enter_graphics();
  88. gs_image_file_free(&lwipe->luma_image);
  89. obs_leave_graphics();
  90. obs_data_release(lwipe->wipes_list);
  91. bfree(lwipe);
  92. }
  93. static obs_properties_t *luma_wipe_properties(void *data)
  94. {
  95. obs_properties_t *props = obs_properties_create();
  96. obs_property_t *p;
  97. p = obs_properties_add_list(props, S_LUMA_IMG, T_LUMA_IMG, OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_STRING);
  98. if (data) {
  99. struct luma_wipe_info *lwipe = data;
  100. obs_data_item_t *item = obs_data_first(lwipe->wipes_list);
  101. for (; item != NULL; obs_data_item_next(&item)) {
  102. const char *name = obs_data_item_get_name(item);
  103. const char *path = obs_data_item_get_string(item);
  104. obs_property_list_add_string(p, obs_module_text(name), path);
  105. }
  106. }
  107. obs_properties_add_float(props, S_LUMA_SOFT, T_LUMA_SOFT, 0.0, 1.0, 0.05);
  108. obs_properties_add_bool(props, S_LUMA_INV, T_LUMA_INV);
  109. return props;
  110. }
  111. static void luma_wipe_defaults(obs_data_t *settings)
  112. {
  113. obs_data_set_default_string(settings, S_LUMA_IMG, "linear-h.png");
  114. obs_data_set_default_double(settings, S_LUMA_SOFT, 0.03);
  115. obs_data_set_default_bool(settings, S_LUMA_INV, false);
  116. }
  117. static void luma_wipe_callback(void *data, gs_texture_t *a, gs_texture_t *b, float t, uint32_t cx, uint32_t cy)
  118. {
  119. struct luma_wipe_info *lwipe = data;
  120. const bool previous = gs_framebuffer_srgb_enabled();
  121. gs_enable_framebuffer_srgb(true);
  122. gs_effect_set_texture_srgb(lwipe->ep_a_tex, a);
  123. gs_effect_set_texture_srgb(lwipe->ep_b_tex, b);
  124. gs_effect_set_texture(lwipe->ep_l_tex, lwipe->luma_image.texture);
  125. gs_effect_set_float(lwipe->ep_progress, t);
  126. gs_effect_set_bool(lwipe->ep_invert, lwipe->invert_luma);
  127. gs_effect_set_float(lwipe->ep_softness, lwipe->softness);
  128. while (gs_effect_loop(lwipe->effect, "LumaWipe"))
  129. gs_draw_sprite(NULL, 0, cx, cy);
  130. gs_enable_framebuffer_srgb(previous);
  131. }
  132. void luma_wipe_video_render(void *data, gs_effect_t *effect)
  133. {
  134. struct luma_wipe_info *lwipe = data;
  135. obs_transition_video_render(lwipe->source, luma_wipe_callback);
  136. UNUSED_PARAMETER(effect);
  137. }
  138. static float mix_a(void *data, float t)
  139. {
  140. UNUSED_PARAMETER(data);
  141. return 1.0f - t;
  142. }
  143. static float mix_b(void *data, float t)
  144. {
  145. UNUSED_PARAMETER(data);
  146. return t;
  147. }
  148. bool luma_wipe_audio_render(void *data, uint64_t *ts_out, struct obs_source_audio_mix *audio, uint32_t mixers,
  149. size_t channels, size_t sample_rate)
  150. {
  151. struct luma_wipe_info *lwipe = data;
  152. return obs_transition_audio_render(lwipe->source, ts_out, audio, mixers, channels, sample_rate, mix_a, mix_b);
  153. }
  154. static enum gs_color_space luma_wipe_video_get_color_space(void *data, size_t count,
  155. const enum gs_color_space *preferred_spaces)
  156. {
  157. UNUSED_PARAMETER(count);
  158. UNUSED_PARAMETER(preferred_spaces);
  159. struct luma_wipe_info *const lwipe = data;
  160. return obs_transition_video_get_color_space(lwipe->source);
  161. }
  162. struct obs_source_info luma_wipe_transition = {
  163. .id = "wipe_transition",
  164. .type = OBS_SOURCE_TYPE_TRANSITION,
  165. .get_name = luma_wipe_get_name,
  166. .create = luma_wipe_create,
  167. .destroy = luma_wipe_destroy,
  168. .update = luma_wipe_update,
  169. .video_render = luma_wipe_video_render,
  170. .audio_render = luma_wipe_audio_render,
  171. .get_properties = luma_wipe_properties,
  172. .get_defaults = luma_wipe_defaults,
  173. .video_get_color_space = luma_wipe_video_get_color_space,
  174. };