transition-stinger.c 10.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378
  1. #include <obs-module.h>
  2. #include <util/dstr.h>
  3. #define TIMING_TIME 0
  4. #define TIMING_FRAME 1
  5. enum fade_style { FADE_STYLE_FADE_OUT_FADE_IN, FADE_STYLE_CROSS_FADE };
  6. struct stinger_info {
  7. obs_source_t *source;
  8. obs_source_t *media_source;
  9. uint64_t duration_ns;
  10. uint64_t duration_frames;
  11. uint64_t transition_point_ns;
  12. uint64_t transition_point_frame;
  13. float transition_point;
  14. float transition_a_mul;
  15. float transition_b_mul;
  16. bool transitioning;
  17. bool transition_point_is_frame;
  18. int monitoring_type;
  19. enum fade_style fade_style;
  20. float (*mix_a)(void *data, float t);
  21. float (*mix_b)(void *data, float t);
  22. };
  23. static const char *stinger_get_name(void *type_data)
  24. {
  25. UNUSED_PARAMETER(type_data);
  26. return obs_module_text("StingerTransition");
  27. }
  28. static float mix_a_fade_in_out(void *data, float t);
  29. static float mix_b_fade_in_out(void *data, float t);
  30. static float mix_a_cross_fade(void *data, float t);
  31. static float mix_b_cross_fade(void *data, float t);
  32. static void stinger_update(void *data, obs_data_t *settings)
  33. {
  34. struct stinger_info *s = data;
  35. const char *path = obs_data_get_string(settings, "path");
  36. obs_data_t *media_settings = obs_data_create();
  37. obs_data_set_string(media_settings, "local_file", path);
  38. obs_source_release(s->media_source);
  39. struct dstr name;
  40. dstr_init_copy(&name, obs_source_get_name(s->source));
  41. dstr_cat(&name, " (Stinger)");
  42. s->media_source = obs_source_create_private("ffmpeg_source", name.array,
  43. media_settings);
  44. dstr_free(&name);
  45. obs_data_release(media_settings);
  46. int64_t point = obs_data_get_int(settings, "transition_point");
  47. s->transition_point_is_frame = obs_data_get_int(settings, "tp_type") ==
  48. TIMING_FRAME;
  49. if (s->transition_point_is_frame)
  50. s->transition_point_frame = (uint64_t)point;
  51. else
  52. s->transition_point_ns = (uint64_t)(point * 1000000LL);
  53. s->monitoring_type =
  54. (int)obs_data_get_int(settings, "audio_monitoring");
  55. obs_source_set_monitoring_type(s->media_source, s->monitoring_type);
  56. s->fade_style =
  57. (enum fade_style)obs_data_get_int(settings, "audio_fade_style");
  58. switch (s->fade_style) {
  59. default:
  60. case FADE_STYLE_FADE_OUT_FADE_IN:
  61. s->mix_a = mix_a_fade_in_out;
  62. s->mix_b = mix_b_fade_in_out;
  63. break;
  64. case FADE_STYLE_CROSS_FADE:
  65. s->mix_a = mix_a_cross_fade;
  66. s->mix_b = mix_b_cross_fade;
  67. break;
  68. }
  69. }
  70. static void *stinger_create(obs_data_t *settings, obs_source_t *source)
  71. {
  72. struct stinger_info *s = bzalloc(sizeof(*s));
  73. s->source = source;
  74. s->mix_a = mix_a_fade_in_out;
  75. s->mix_b = mix_b_fade_in_out;
  76. obs_transition_enable_fixed(s->source, true, 0);
  77. obs_source_update(source, settings);
  78. return s;
  79. }
  80. static void stinger_destroy(void *data)
  81. {
  82. struct stinger_info *s = data;
  83. obs_source_release(s->media_source);
  84. bfree(s);
  85. }
  86. static void stinger_video_render(void *data, gs_effect_t *effect)
  87. {
  88. struct stinger_info *s = data;
  89. float t = obs_transition_get_time(s->source);
  90. bool use_a = t < s->transition_point;
  91. enum obs_transition_target target = use_a ? OBS_TRANSITION_SOURCE_A
  92. : OBS_TRANSITION_SOURCE_B;
  93. if (!obs_transition_video_render_direct(s->source, target))
  94. return;
  95. /* --------------------- */
  96. float source_cx = (float)obs_source_get_width(s->source);
  97. float source_cy = (float)obs_source_get_height(s->source);
  98. uint32_t media_cx = obs_source_get_width(s->media_source);
  99. uint32_t media_cy = obs_source_get_height(s->media_source);
  100. if (!media_cx || !media_cy)
  101. return;
  102. float scale_x = source_cx / (float)media_cx;
  103. float scale_y = source_cy / (float)media_cy;
  104. gs_matrix_push();
  105. gs_matrix_scale3f(scale_x, scale_y, 1.0f);
  106. obs_source_video_render(s->media_source);
  107. gs_matrix_pop();
  108. UNUSED_PARAMETER(effect);
  109. }
  110. static inline float calc_fade(float t, float mul)
  111. {
  112. t *= mul;
  113. return t > 1.0f ? 1.0f : t;
  114. }
  115. static float mix_a_fade_in_out(void *data, float t)
  116. {
  117. struct stinger_info *s = data;
  118. return 1.0f - calc_fade(t, s->transition_a_mul);
  119. }
  120. static float mix_b_fade_in_out(void *data, float t)
  121. {
  122. struct stinger_info *s = data;
  123. return 1.0f - calc_fade(1.0f - t, s->transition_b_mul);
  124. }
  125. static float mix_a_cross_fade(void *data, float t)
  126. {
  127. UNUSED_PARAMETER(data);
  128. return 1.0f - t;
  129. }
  130. static float mix_b_cross_fade(void *data, float t)
  131. {
  132. UNUSED_PARAMETER(data);
  133. return t;
  134. }
  135. static bool stinger_audio_render(void *data, uint64_t *ts_out,
  136. struct obs_source_audio_mix *audio,
  137. uint32_t mixers, size_t channels,
  138. size_t sample_rate)
  139. {
  140. struct stinger_info *s = data;
  141. uint64_t ts = 0;
  142. if (!obs_source_audio_pending(s->media_source)) {
  143. ts = obs_source_get_audio_timestamp(s->media_source);
  144. if (!ts)
  145. return false;
  146. }
  147. bool success = obs_transition_audio_render(s->source, ts_out, audio,
  148. mixers, channels,
  149. sample_rate, s->mix_a,
  150. s->mix_b);
  151. if (!ts)
  152. return success;
  153. if (!*ts_out || ts < *ts_out)
  154. *ts_out = ts;
  155. struct obs_source_audio_mix child_audio;
  156. obs_source_get_audio_mix(s->media_source, &child_audio);
  157. for (size_t mix = 0; mix < MAX_AUDIO_MIXES; mix++) {
  158. if ((mixers & (1 << mix)) == 0)
  159. continue;
  160. for (size_t ch = 0; ch < channels; ch++) {
  161. register float *out = audio->output[mix].data[ch];
  162. register float *in = child_audio.output[mix].data[ch];
  163. register float *end = in + AUDIO_OUTPUT_FRAMES;
  164. while (in < end)
  165. *(out++) += *(in++);
  166. }
  167. }
  168. return true;
  169. }
  170. static void stinger_transition_start(void *data)
  171. {
  172. struct stinger_info *s = data;
  173. if (s->media_source) {
  174. calldata_t cd = {0};
  175. proc_handler_t *ph =
  176. obs_source_get_proc_handler(s->media_source);
  177. if (s->transitioning) {
  178. proc_handler_call(ph, "restart", &cd);
  179. return;
  180. }
  181. proc_handler_call(ph, "get_duration", &cd);
  182. proc_handler_call(ph, "get_nb_frames", &cd);
  183. s->duration_ns =
  184. (uint64_t)calldata_int(&cd, "duration") + 250000000ULL;
  185. s->duration_frames = (uint64_t)calldata_int(&cd, "num_frames");
  186. if (s->transition_point_is_frame)
  187. s->transition_point =
  188. (float)((long double)s->transition_point_frame /
  189. (long double)s->duration_frames);
  190. else
  191. s->transition_point =
  192. (float)((long double)s->transition_point_ns /
  193. (long double)s->duration_ns);
  194. if (s->transition_point > 0.999f)
  195. s->transition_point = 0.999f;
  196. else if (s->transition_point < 0.001f)
  197. s->transition_point = 0.001f;
  198. s->transition_a_mul = (1.0f / s->transition_point);
  199. s->transition_b_mul = (1.0f / (1.0f - s->transition_point));
  200. obs_transition_enable_fixed(
  201. s->source, true, (uint32_t)(s->duration_ns / 1000000));
  202. calldata_free(&cd);
  203. obs_source_add_active_child(s->source, s->media_source);
  204. }
  205. s->transitioning = true;
  206. }
  207. static void stinger_transition_stop(void *data)
  208. {
  209. struct stinger_info *s = data;
  210. if (s->media_source)
  211. obs_source_remove_active_child(s->source, s->media_source);
  212. s->transitioning = false;
  213. }
  214. static void stinger_enum_active_sources(void *data,
  215. obs_source_enum_proc_t enum_callback,
  216. void *param)
  217. {
  218. struct stinger_info *s = data;
  219. if (s->media_source && s->transitioning)
  220. enum_callback(s->source, s->media_source, param);
  221. }
  222. static void stinger_enum_all_sources(void *data,
  223. obs_source_enum_proc_t enum_callback,
  224. void *param)
  225. {
  226. struct stinger_info *s = data;
  227. if (s->media_source)
  228. enum_callback(s->source, s->media_source, param);
  229. }
  230. #define FILE_FILTER \
  231. "Video Files (*.mp4 *.ts *.mov *.wmv *.flv *.mkv *.avi *.gif *.webm);;"
  232. static bool transition_point_type_modified(obs_properties_t *ppts,
  233. obs_property_t *p, obs_data_t *s)
  234. {
  235. int64_t type = obs_data_get_int(s, "tp_type");
  236. p = obs_properties_get(ppts, "transition_point");
  237. if (type == TIMING_TIME) {
  238. obs_property_set_description(
  239. p, obs_module_text("TransitionPoint"));
  240. obs_property_int_set_suffix(p, " ms");
  241. } else {
  242. obs_property_set_description(
  243. p, obs_module_text("TransitionPointFrame"));
  244. obs_property_int_set_suffix(p, "");
  245. }
  246. return true;
  247. }
  248. static obs_properties_t *stinger_properties(void *data)
  249. {
  250. obs_properties_t *ppts = obs_properties_create();
  251. obs_properties_set_flags(ppts, OBS_PROPERTIES_DEFER_UPDATE);
  252. obs_properties_add_path(ppts, "path", obs_module_text("VideoFile"),
  253. OBS_PATH_FILE, FILE_FILTER, NULL);
  254. obs_property_t *p = obs_properties_add_list(
  255. ppts, "tp_type", obs_module_text("TransitionPointType"),
  256. OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT);
  257. obs_property_list_add_int(p, obs_module_text("TransitionPointTypeTime"),
  258. TIMING_TIME);
  259. obs_property_list_add_int(
  260. p, obs_module_text("TransitionPointTypeFrame"), TIMING_FRAME);
  261. obs_property_set_modified_callback(p, transition_point_type_modified);
  262. obs_properties_add_int(ppts, "transition_point",
  263. obs_module_text("TransitionPoint"), 0, 120000,
  264. 1);
  265. obs_property_t *monitor_list = obs_properties_add_list(
  266. ppts, "audio_monitoring", obs_module_text("AudioMonitoring"),
  267. OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT);
  268. obs_property_list_add_int(monitor_list,
  269. obs_module_text("AudioMonitoring.None"),
  270. OBS_MONITORING_TYPE_NONE);
  271. obs_property_list_add_int(
  272. monitor_list, obs_module_text("AudioMonitoring.MonitorOnly"),
  273. OBS_MONITORING_TYPE_MONITOR_ONLY);
  274. obs_property_list_add_int(monitor_list,
  275. obs_module_text("AudioMonitoring.Both"),
  276. OBS_MONITORING_TYPE_MONITOR_AND_OUTPUT);
  277. obs_property_t *audio_fade_style = obs_properties_add_list(
  278. ppts, "audio_fade_style", obs_module_text("AudioFadeStyle"),
  279. OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT);
  280. obs_property_list_add_int(
  281. audio_fade_style,
  282. obs_module_text("AudioFadeStyle.FadeOutFadeIn"),
  283. FADE_STYLE_FADE_OUT_FADE_IN);
  284. obs_property_list_add_int(audio_fade_style,
  285. obs_module_text("AudioFadeStyle.CrossFade"),
  286. FADE_STYLE_CROSS_FADE);
  287. UNUSED_PARAMETER(data);
  288. return ppts;
  289. }
  290. struct obs_source_info stinger_transition = {
  291. .id = "obs_stinger_transition",
  292. .type = OBS_SOURCE_TYPE_TRANSITION,
  293. .get_name = stinger_get_name,
  294. .create = stinger_create,
  295. .destroy = stinger_destroy,
  296. .update = stinger_update,
  297. .video_render = stinger_video_render,
  298. .audio_render = stinger_audio_render,
  299. .get_properties = stinger_properties,
  300. .enum_active_sources = stinger_enum_active_sources,
  301. .enum_all_sources = stinger_enum_all_sources,
  302. .transition_start = stinger_transition_start,
  303. .transition_stop = stinger_transition_stop,
  304. };