transition-slide.c 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. #include <obs-module.h>
  2. #include <graphics/vec2.h>
  3. #include "easings.h"
  4. #define S_DIRECTION "direction"
  5. struct slide_info {
  6. obs_source_t *source;
  7. gs_effect_t *effect;
  8. gs_eparam_t *a_param;
  9. gs_eparam_t *b_param;
  10. gs_eparam_t *tex_a_dir_param;
  11. gs_eparam_t *tex_b_dir_param;
  12. struct vec2 dir;
  13. bool slide_in;
  14. };
  15. static const char *slide_get_name(void *type_data)
  16. {
  17. UNUSED_PARAMETER(type_data);
  18. return obs_module_text("SlideTransition");
  19. }
  20. static void slide_update(void *data, obs_data_t *settings)
  21. {
  22. struct slide_info *slide = data;
  23. const char *dir = obs_data_get_string(settings, S_DIRECTION);
  24. if (strcmp(dir, "right") == 0)
  25. slide->dir = (struct vec2){-1.0f, 0.0f};
  26. else if (strcmp(dir, "up") == 0)
  27. slide->dir = (struct vec2){0.0f, 1.0f};
  28. else if (strcmp(dir, "down") == 0)
  29. slide->dir = (struct vec2){0.0f, -1.0f};
  30. else /* left */
  31. slide->dir = (struct vec2){1.0f, 0.0f};
  32. }
  33. void *slide_create(obs_data_t *settings, obs_source_t *source)
  34. {
  35. struct slide_info *slide;
  36. gs_effect_t *effect;
  37. char *file = obs_module_file("slide_transition.effect");
  38. obs_enter_graphics();
  39. effect = gs_effect_create_from_file(file, NULL);
  40. obs_leave_graphics();
  41. bfree(file);
  42. if (!effect) {
  43. blog(LOG_ERROR, "Could not find slide_transition.effect");
  44. return NULL;
  45. }
  46. slide = bzalloc(sizeof(*slide));
  47. slide->source = source;
  48. slide->effect = effect;
  49. slide->a_param = gs_effect_get_param_by_name(effect, "tex_a");
  50. slide->b_param = gs_effect_get_param_by_name(effect, "tex_b");
  51. slide->tex_a_dir_param =
  52. gs_effect_get_param_by_name(effect, "tex_a_dir");
  53. slide->tex_b_dir_param =
  54. gs_effect_get_param_by_name(effect, "tex_b_dir");
  55. obs_source_update(source, settings);
  56. return slide;
  57. }
  58. void slide_destroy(void *data)
  59. {
  60. struct slide_info *slide = data;
  61. bfree(slide);
  62. }
  63. static void slide_callback(void *data, gs_texture_t *a, gs_texture_t *b,
  64. float t, uint32_t cx, uint32_t cy)
  65. {
  66. struct slide_info *slide = data;
  67. struct vec2 tex_a_dir = slide->dir;
  68. struct vec2 tex_b_dir = slide->dir;
  69. t = cubic_ease_in_out(t);
  70. vec2_mulf(&tex_a_dir, &tex_a_dir, t);
  71. vec2_mulf(&tex_b_dir, &tex_b_dir, 1.0f - t);
  72. const bool previous = gs_framebuffer_srgb_enabled();
  73. gs_enable_framebuffer_srgb(true);
  74. gs_effect_set_texture_srgb(slide->a_param, a);
  75. gs_effect_set_texture_srgb(slide->b_param, b);
  76. gs_effect_set_vec2(slide->tex_a_dir_param, &tex_a_dir);
  77. gs_effect_set_vec2(slide->tex_b_dir_param, &tex_b_dir);
  78. while (gs_effect_loop(slide->effect, "Slide"))
  79. gs_draw_sprite(NULL, 0, cx, cy);
  80. gs_enable_framebuffer_srgb(previous);
  81. }
  82. void slide_video_render(void *data, gs_effect_t *effect)
  83. {
  84. struct slide_info *slide = data;
  85. obs_transition_video_render(slide->source, slide_callback);
  86. UNUSED_PARAMETER(effect);
  87. }
  88. static float mix_a(void *data, float t)
  89. {
  90. UNUSED_PARAMETER(data);
  91. return 1.0f - cubic_ease_in_out(t);
  92. }
  93. static float mix_b(void *data, float t)
  94. {
  95. UNUSED_PARAMETER(data);
  96. return cubic_ease_in_out(t);
  97. }
  98. bool slide_audio_render(void *data, uint64_t *ts_out,
  99. struct obs_source_audio_mix *audio, uint32_t mixers,
  100. size_t channels, size_t sample_rate)
  101. {
  102. struct slide_info *slide = data;
  103. return obs_transition_audio_render(slide->source, ts_out, audio, mixers,
  104. channels, sample_rate, mix_a, mix_b);
  105. }
  106. static obs_properties_t *slide_properties(void *data)
  107. {
  108. obs_properties_t *ppts = obs_properties_create();
  109. obs_property_t *p;
  110. p = obs_properties_add_list(ppts, S_DIRECTION,
  111. obs_module_text("Direction"),
  112. OBS_COMBO_TYPE_LIST,
  113. OBS_COMBO_FORMAT_STRING);
  114. obs_property_list_add_string(p, obs_module_text("Direction.Left"),
  115. "left");
  116. obs_property_list_add_string(p, obs_module_text("Direction.Right"),
  117. "right");
  118. obs_property_list_add_string(p, obs_module_text("Direction.Up"), "up");
  119. obs_property_list_add_string(p, obs_module_text("Direction.Down"),
  120. "down");
  121. UNUSED_PARAMETER(data);
  122. return ppts;
  123. }
  124. static enum gs_color_space
  125. slide_video_get_color_space(void *data, size_t count,
  126. const enum gs_color_space *preferred_spaces)
  127. {
  128. UNUSED_PARAMETER(count);
  129. UNUSED_PARAMETER(preferred_spaces);
  130. struct slide_info *const slide = data;
  131. return obs_transition_video_get_color_space(slide->source);
  132. }
  133. struct obs_source_info slide_transition = {
  134. .id = "slide_transition",
  135. .type = OBS_SOURCE_TYPE_TRANSITION,
  136. .get_name = slide_get_name,
  137. .create = slide_create,
  138. .destroy = slide_destroy,
  139. .update = slide_update,
  140. .video_render = slide_video_render,
  141. .audio_render = slide_audio_render,
  142. .get_properties = slide_properties,
  143. .video_get_color_space = slide_video_get_color_space,
  144. };