1
0

transition-slide.c 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171
  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 = gs_effect_get_param_by_name(effect, "tex_a_dir");
  52. slide->tex_b_dir_param = gs_effect_get_param_by_name(effect, "tex_b_dir");
  53. obs_source_update(source, settings);
  54. return slide;
  55. }
  56. void slide_destroy(void *data)
  57. {
  58. struct slide_info *slide = data;
  59. bfree(slide);
  60. }
  61. static void slide_callback(void *data, gs_texture_t *a, gs_texture_t *b, float t, uint32_t cx, uint32_t cy)
  62. {
  63. struct slide_info *slide = data;
  64. struct vec2 tex_a_dir = slide->dir;
  65. struct vec2 tex_b_dir = slide->dir;
  66. t = cubic_ease_in_out(t);
  67. vec2_mulf(&tex_a_dir, &tex_a_dir, t);
  68. vec2_mulf(&tex_b_dir, &tex_b_dir, 1.0f - t);
  69. const bool previous = gs_framebuffer_srgb_enabled();
  70. gs_enable_framebuffer_srgb(true);
  71. gs_effect_set_texture_srgb(slide->a_param, a);
  72. gs_effect_set_texture_srgb(slide->b_param, b);
  73. gs_effect_set_vec2(slide->tex_a_dir_param, &tex_a_dir);
  74. gs_effect_set_vec2(slide->tex_b_dir_param, &tex_b_dir);
  75. while (gs_effect_loop(slide->effect, "Slide"))
  76. gs_draw_sprite(NULL, 0, cx, cy);
  77. gs_enable_framebuffer_srgb(previous);
  78. }
  79. void slide_video_render(void *data, gs_effect_t *effect)
  80. {
  81. struct slide_info *slide = data;
  82. obs_transition_video_render(slide->source, slide_callback);
  83. UNUSED_PARAMETER(effect);
  84. }
  85. static float mix_a(void *data, float t)
  86. {
  87. UNUSED_PARAMETER(data);
  88. return 1.0f - cubic_ease_in_out(t);
  89. }
  90. static float mix_b(void *data, float t)
  91. {
  92. UNUSED_PARAMETER(data);
  93. return cubic_ease_in_out(t);
  94. }
  95. bool slide_audio_render(void *data, uint64_t *ts_out, struct obs_source_audio_mix *audio, uint32_t mixers,
  96. size_t channels, size_t sample_rate)
  97. {
  98. struct slide_info *slide = data;
  99. return obs_transition_audio_render(slide->source, ts_out, audio, mixers, channels, sample_rate, mix_a, mix_b);
  100. }
  101. static obs_properties_t *slide_properties(void *data)
  102. {
  103. obs_properties_t *ppts = obs_properties_create();
  104. obs_property_t *p;
  105. p = obs_properties_add_list(ppts, S_DIRECTION, obs_module_text("Direction"), OBS_COMBO_TYPE_LIST,
  106. OBS_COMBO_FORMAT_STRING);
  107. obs_property_list_add_string(p, obs_module_text("Direction.Left"), "left");
  108. obs_property_list_add_string(p, obs_module_text("Direction.Right"), "right");
  109. obs_property_list_add_string(p, obs_module_text("Direction.Up"), "up");
  110. obs_property_list_add_string(p, obs_module_text("Direction.Down"), "down");
  111. UNUSED_PARAMETER(data);
  112. return ppts;
  113. }
  114. static enum gs_color_space slide_video_get_color_space(void *data, size_t count,
  115. const enum gs_color_space *preferred_spaces)
  116. {
  117. UNUSED_PARAMETER(count);
  118. UNUSED_PARAMETER(preferred_spaces);
  119. struct slide_info *const slide = data;
  120. return obs_transition_video_get_color_space(slide->source);
  121. }
  122. struct obs_source_info slide_transition = {
  123. .id = "slide_transition",
  124. .type = OBS_SOURCE_TYPE_TRANSITION,
  125. .get_name = slide_get_name,
  126. .create = slide_create,
  127. .destroy = slide_destroy,
  128. .update = slide_update,
  129. .video_render = slide_video_render,
  130. .audio_render = slide_audio_render,
  131. .get_properties = slide_properties,
  132. .video_get_color_space = slide_video_get_color_space,
  133. };