2017-07-19 09:32:47 -07:00
|
|
|
#include <obs-module.h>
|
2020-01-25 14:15:06 +11:00
|
|
|
#include <util/dstr.h>
|
2021-08-15 21:51:24 +02:00
|
|
|
#include "util/platform.h"
|
2017-07-19 09:32:47 -07:00
|
|
|
|
2017-08-08 10:08:23 +02:00
|
|
|
#define TIMING_TIME 0
|
|
|
|
#define TIMING_FRAME 1
|
|
|
|
|
2021-08-08 15:42:37 -07:00
|
|
|
enum matte_layout {
|
|
|
|
MATTE_LAYOUT_HORIZONTAL,
|
|
|
|
MATTE_LAYOUT_VERTICAL,
|
|
|
|
MATTE_LAYOUT_SEPARATE_FILE,
|
|
|
|
MATTE_LAYOUT_MASK,
|
|
|
|
};
|
2020-01-28 10:09:51 +01:00
|
|
|
|
2017-09-16 10:11:52 +02:00
|
|
|
enum fade_style { FADE_STYLE_FADE_OUT_FADE_IN, FADE_STYLE_CROSS_FADE };
|
|
|
|
|
2017-07-19 09:32:47 -07:00
|
|
|
struct stinger_info {
|
|
|
|
obs_source_t *source;
|
|
|
|
|
|
|
|
obs_source_t *media_source;
|
2020-01-28 10:09:51 +01:00
|
|
|
obs_source_t *matte_source;
|
2017-07-19 09:32:47 -07:00
|
|
|
|
|
|
|
uint64_t duration_ns;
|
2017-08-08 10:08:23 +02:00
|
|
|
uint64_t duration_frames;
|
2017-07-19 09:32:47 -07:00
|
|
|
uint64_t transition_point_ns;
|
2017-08-08 10:08:23 +02:00
|
|
|
uint64_t transition_point_frame;
|
2017-07-19 09:32:47 -07:00
|
|
|
float transition_point;
|
|
|
|
float transition_a_mul;
|
|
|
|
float transition_b_mul;
|
|
|
|
bool transitioning;
|
2017-08-08 10:08:23 +02:00
|
|
|
bool transition_point_is_frame;
|
2017-08-11 17:14:00 +02:00
|
|
|
int monitoring_type;
|
2017-09-16 10:11:52 +02:00
|
|
|
enum fade_style fade_style;
|
|
|
|
|
2020-01-28 10:09:51 +01:00
|
|
|
bool track_matte_enabled;
|
|
|
|
int matte_layout;
|
|
|
|
float matte_width_factor;
|
|
|
|
float matte_height_factor;
|
|
|
|
bool invert_matte;
|
2021-05-11 00:42:38 -07:00
|
|
|
bool do_texrender;
|
2021-08-10 13:56:50 -07:00
|
|
|
bool matte_rendered;
|
2020-01-28 10:09:51 +01:00
|
|
|
|
|
|
|
gs_effect_t *matte_effect;
|
|
|
|
gs_eparam_t *ep_a_tex;
|
|
|
|
gs_eparam_t *ep_b_tex;
|
|
|
|
gs_eparam_t *ep_matte_tex;
|
|
|
|
gs_eparam_t *ep_invert_matte;
|
|
|
|
|
2021-05-10 00:35:12 -07:00
|
|
|
gs_texrender_t *matte_tex;
|
2021-05-11 00:42:38 -07:00
|
|
|
gs_texrender_t *stinger_tex;
|
2020-01-28 10:09:51 +01:00
|
|
|
|
2017-09-16 10:11:52 +02:00
|
|
|
float (*mix_a)(void *data, float t);
|
|
|
|
float (*mix_b)(void *data, float t);
|
2017-07-19 09:32:47 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
static const char *stinger_get_name(void *type_data)
|
|
|
|
{
|
|
|
|
UNUSED_PARAMETER(type_data);
|
|
|
|
return obs_module_text("StingerTransition");
|
|
|
|
}
|
|
|
|
|
2017-09-16 10:11:52 +02:00
|
|
|
static float mix_a_fade_in_out(void *data, float t);
|
|
|
|
static float mix_b_fade_in_out(void *data, float t);
|
|
|
|
static float mix_a_cross_fade(void *data, float t);
|
|
|
|
static float mix_b_cross_fade(void *data, float t);
|
|
|
|
|
2017-07-19 09:32:47 -07:00
|
|
|
static void stinger_update(void *data, obs_data_t *settings)
|
|
|
|
{
|
|
|
|
struct stinger_info *s = data;
|
|
|
|
const char *path = obs_data_get_string(settings, "path");
|
2020-09-01 22:16:11 +10:00
|
|
|
bool hw_decode = obs_data_get_bool(settings, "hw_decode");
|
2017-07-19 09:32:47 -07:00
|
|
|
|
|
|
|
obs_data_t *media_settings = obs_data_create();
|
|
|
|
obs_data_set_string(media_settings, "local_file", path);
|
2020-09-01 22:16:11 +10:00
|
|
|
obs_data_set_bool(media_settings, "hw_decode", hw_decode);
|
2021-11-10 18:53:07 +01:00
|
|
|
obs_data_set_bool(media_settings, "looping", false);
|
2017-07-19 09:32:47 -07:00
|
|
|
|
|
|
|
obs_source_release(s->media_source);
|
2020-01-25 14:15:06 +11:00
|
|
|
struct dstr name;
|
|
|
|
dstr_init_copy(&name, obs_source_get_name(s->source));
|
|
|
|
dstr_cat(&name, " (Stinger)");
|
|
|
|
s->media_source = obs_source_create_private("ffmpeg_source", name.array,
|
2017-07-19 09:32:47 -07:00
|
|
|
media_settings);
|
2020-01-25 14:15:06 +11:00
|
|
|
dstr_free(&name);
|
2017-07-19 09:32:47 -07:00
|
|
|
obs_data_release(media_settings);
|
|
|
|
|
2017-08-08 10:08:23 +02:00
|
|
|
int64_t point = obs_data_get_int(settings, "transition_point");
|
|
|
|
|
|
|
|
s->transition_point_is_frame = obs_data_get_int(settings, "tp_type") ==
|
|
|
|
TIMING_FRAME;
|
2017-07-19 09:32:47 -07:00
|
|
|
|
2017-08-08 10:08:23 +02:00
|
|
|
if (s->transition_point_is_frame)
|
|
|
|
s->transition_point_frame = (uint64_t)point;
|
|
|
|
else
|
|
|
|
s->transition_point_ns = (uint64_t)(point * 1000000LL);
|
2017-08-11 17:14:00 +02:00
|
|
|
|
2021-05-11 00:41:34 -07:00
|
|
|
bool track_matte_was_enabled = s->track_matte_enabled;
|
|
|
|
|
2020-01-28 10:09:51 +01:00
|
|
|
s->track_matte_enabled =
|
|
|
|
obs_data_get_bool(settings, "track_matte_enabled");
|
2021-03-31 05:09:51 -07:00
|
|
|
s->matte_layout = (int)obs_data_get_int(settings, "track_matte_layout");
|
2020-02-25 20:22:31 +01:00
|
|
|
s->matte_width_factor =
|
|
|
|
(s->matte_layout == MATTE_LAYOUT_HORIZONTAL ? 2.0f : 1.0f);
|
|
|
|
s->matte_height_factor =
|
|
|
|
(s->matte_layout == MATTE_LAYOUT_VERTICAL ? 2.0f : 1.0f);
|
2020-01-28 10:09:51 +01:00
|
|
|
s->invert_matte = obs_data_get_bool(settings, "invert_matte");
|
|
|
|
|
2021-05-11 00:42:38 -07:00
|
|
|
s->do_texrender = s->track_matte_enabled &&
|
2021-08-08 15:42:37 -07:00
|
|
|
s->matte_layout < MATTE_LAYOUT_SEPARATE_FILE;
|
2021-05-11 00:42:38 -07:00
|
|
|
|
2020-01-28 10:09:51 +01:00
|
|
|
if (s->matte_source) {
|
|
|
|
obs_source_release(s->matte_source);
|
|
|
|
s->matte_source = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (s->track_matte_enabled &&
|
|
|
|
s->matte_layout == MATTE_LAYOUT_SEPARATE_FILE) {
|
|
|
|
const char *tm_path =
|
|
|
|
obs_data_get_string(settings, "track_matte_path");
|
|
|
|
|
|
|
|
obs_data_t *tm_media_settings = obs_data_create();
|
|
|
|
obs_data_set_string(tm_media_settings, "local_file", tm_path);
|
2021-11-10 18:53:07 +01:00
|
|
|
obs_data_set_bool(tm_media_settings, "looping", false);
|
2020-01-28 10:09:51 +01:00
|
|
|
|
|
|
|
s->matte_source = obs_source_create_private(
|
|
|
|
"ffmpeg_source", NULL, tm_media_settings);
|
|
|
|
obs_data_release(tm_media_settings);
|
|
|
|
|
|
|
|
// no need to output sound from the matte video
|
|
|
|
obs_source_set_muted(s->matte_source, true);
|
|
|
|
}
|
|
|
|
|
2017-09-16 13:59:53 -07:00
|
|
|
s->monitoring_type =
|
|
|
|
(int)obs_data_get_int(settings, "audio_monitoring");
|
2017-08-11 17:14:00 +02:00
|
|
|
obs_source_set_monitoring_type(s->media_source, s->monitoring_type);
|
2017-09-16 10:11:52 +02:00
|
|
|
|
|
|
|
s->fade_style =
|
|
|
|
(enum fade_style)obs_data_get_int(settings, "audio_fade_style");
|
|
|
|
|
|
|
|
switch (s->fade_style) {
|
|
|
|
default:
|
|
|
|
case FADE_STYLE_FADE_OUT_FADE_IN:
|
|
|
|
s->mix_a = mix_a_fade_in_out;
|
|
|
|
s->mix_b = mix_b_fade_in_out;
|
|
|
|
break;
|
|
|
|
case FADE_STYLE_CROSS_FADE:
|
|
|
|
s->mix_a = mix_a_cross_fade;
|
|
|
|
s->mix_b = mix_b_cross_fade;
|
|
|
|
break;
|
|
|
|
}
|
2021-05-11 00:41:34 -07:00
|
|
|
|
|
|
|
if (s->track_matte_enabled != track_matte_was_enabled) {
|
2021-05-14 03:39:43 -05:00
|
|
|
obs_enter_graphics();
|
|
|
|
|
2021-05-11 00:41:34 -07:00
|
|
|
gs_texrender_destroy(s->matte_tex);
|
2021-05-11 00:42:38 -07:00
|
|
|
gs_texrender_destroy(s->stinger_tex);
|
2021-05-11 00:41:34 -07:00
|
|
|
s->matte_tex = NULL;
|
2021-05-11 00:42:38 -07:00
|
|
|
s->stinger_tex = NULL;
|
2021-05-11 00:41:34 -07:00
|
|
|
|
|
|
|
if (s->track_matte_enabled) {
|
|
|
|
s->matte_tex = gs_texrender_create(GS_RGBA, GS_ZS_NONE);
|
2021-05-11 00:42:38 -07:00
|
|
|
s->stinger_tex =
|
|
|
|
gs_texrender_create(GS_RGBA, GS_ZS_NONE);
|
2021-05-11 00:41:34 -07:00
|
|
|
}
|
2021-05-14 03:39:43 -05:00
|
|
|
|
|
|
|
obs_leave_graphics();
|
2021-05-11 00:41:34 -07:00
|
|
|
}
|
2017-07-19 09:32:47 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void *stinger_create(obs_data_t *settings, obs_source_t *source)
|
|
|
|
{
|
|
|
|
struct stinger_info *s = bzalloc(sizeof(*s));
|
|
|
|
|
|
|
|
s->source = source;
|
2017-09-16 10:11:52 +02:00
|
|
|
s->mix_a = mix_a_fade_in_out;
|
|
|
|
s->mix_b = mix_b_fade_in_out;
|
2017-07-19 09:32:47 -07:00
|
|
|
|
2020-01-28 10:09:51 +01:00
|
|
|
char *effect_file = obs_module_file("stinger_matte_transition.effect");
|
|
|
|
char *error_string = NULL;
|
|
|
|
obs_enter_graphics();
|
|
|
|
s->matte_effect =
|
|
|
|
gs_effect_create_from_file(effect_file, &error_string);
|
|
|
|
obs_leave_graphics();
|
|
|
|
|
|
|
|
if (!s->matte_effect) {
|
|
|
|
blog(LOG_ERROR,
|
|
|
|
"Could not open stinger_matte_transition.effect: %s",
|
|
|
|
error_string);
|
|
|
|
bfree(error_string);
|
|
|
|
bfree(s);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
bfree(effect_file);
|
|
|
|
|
|
|
|
s->ep_a_tex = gs_effect_get_param_by_name(s->matte_effect, "a_tex");
|
|
|
|
s->ep_b_tex = gs_effect_get_param_by_name(s->matte_effect, "b_tex");
|
|
|
|
s->ep_matte_tex =
|
|
|
|
gs_effect_get_param_by_name(s->matte_effect, "matte_tex");
|
|
|
|
s->ep_invert_matte =
|
|
|
|
gs_effect_get_param_by_name(s->matte_effect, "invert_matte");
|
|
|
|
|
2017-07-19 09:32:47 -07:00
|
|
|
obs_transition_enable_fixed(s->source, true, 0);
|
|
|
|
obs_source_update(source, settings);
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void stinger_destroy(void *data)
|
|
|
|
{
|
|
|
|
struct stinger_info *s = data;
|
|
|
|
obs_source_release(s->media_source);
|
2020-01-28 10:09:51 +01:00
|
|
|
obs_source_release(s->matte_source);
|
|
|
|
|
2021-05-13 15:13:07 -07:00
|
|
|
obs_enter_graphics();
|
|
|
|
|
2021-05-10 00:35:12 -07:00
|
|
|
gs_texrender_destroy(s->matte_tex);
|
2021-05-11 00:42:38 -07:00
|
|
|
gs_texrender_destroy(s->stinger_tex);
|
2020-01-28 10:09:51 +01:00
|
|
|
gs_effect_destroy(s->matte_effect);
|
|
|
|
|
2021-05-13 15:13:07 -07:00
|
|
|
obs_leave_graphics();
|
|
|
|
|
2017-07-19 09:32:47 -07:00
|
|
|
bfree(s);
|
|
|
|
}
|
|
|
|
|
2020-09-01 22:16:11 +10:00
|
|
|
static void stinger_defaults(obs_data_t *settings)
|
|
|
|
{
|
|
|
|
obs_data_set_default_bool(settings, "hw_decode", true);
|
|
|
|
}
|
|
|
|
|
2020-01-28 10:09:51 +01:00
|
|
|
static void stinger_matte_render(void *data, gs_texture_t *a, gs_texture_t *b,
|
|
|
|
float t, uint32_t cx, uint32_t cy)
|
|
|
|
{
|
|
|
|
struct stinger_info *s = data;
|
|
|
|
|
|
|
|
struct vec4 background;
|
|
|
|
vec4_zero(&background);
|
|
|
|
|
|
|
|
obs_source_t *matte_source =
|
|
|
|
(s->matte_layout == MATTE_LAYOUT_SEPARATE_FILE
|
|
|
|
? s->matte_source
|
|
|
|
: s->media_source);
|
|
|
|
|
|
|
|
float matte_cx = (float)obs_source_get_width(matte_source) /
|
|
|
|
s->matte_width_factor;
|
|
|
|
float matte_cy = (float)obs_source_get_height(matte_source) /
|
|
|
|
s->matte_height_factor;
|
|
|
|
|
|
|
|
float width_offset = (s->matte_layout == MATTE_LAYOUT_HORIZONTAL
|
|
|
|
? (-matte_cx)
|
|
|
|
: 0.0f);
|
|
|
|
float height_offset =
|
|
|
|
(s->matte_layout == MATTE_LAYOUT_VERTICAL ? (-matte_cy) : 0.0f);
|
|
|
|
|
|
|
|
// Track matte media render
|
|
|
|
if (matte_cx > 0 && matte_cy > 0) {
|
2021-05-10 00:35:12 -07:00
|
|
|
float scale_x = (float)cx / matte_cx;
|
|
|
|
float scale_y = (float)cy / matte_cy;
|
|
|
|
|
|
|
|
if (gs_texrender_begin(s->matte_tex, cx, cy)) {
|
|
|
|
gs_matrix_scale3f(scale_x, scale_y, 1.0f);
|
2020-01-28 10:09:51 +01:00
|
|
|
gs_matrix_translate3f(width_offset, height_offset,
|
|
|
|
0.0f);
|
|
|
|
gs_clear(GS_CLEAR_COLOR, &background, 0.0f, 0);
|
2021-05-11 00:34:50 -07:00
|
|
|
gs_ortho(0.0f, (float)cx, 0.0f, (float)cy, -100.0f,
|
|
|
|
100.0f);
|
|
|
|
|
2020-01-28 10:09:51 +01:00
|
|
|
obs_source_video_render(matte_source);
|
|
|
|
|
2021-05-10 00:35:12 -07:00
|
|
|
gs_texrender_end(s->matte_tex);
|
2020-01-28 10:09:51 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-08 17:50:22 -07:00
|
|
|
const bool previous = gs_framebuffer_srgb_enabled();
|
|
|
|
gs_enable_framebuffer_srgb(true);
|
|
|
|
|
2020-01-28 10:09:51 +01:00
|
|
|
gs_effect_set_texture(s->ep_a_tex, a);
|
|
|
|
gs_effect_set_texture(s->ep_b_tex, b);
|
|
|
|
gs_effect_set_texture(s->ep_matte_tex,
|
2021-05-10 00:35:12 -07:00
|
|
|
gs_texrender_get_texture(s->matte_tex));
|
2020-01-28 10:09:51 +01:00
|
|
|
gs_effect_set_bool(s->ep_invert_matte, s->invert_matte);
|
|
|
|
|
|
|
|
while (gs_effect_loop(s->matte_effect, "StingerMatte"))
|
|
|
|
gs_draw_sprite(NULL, 0, cx, cy);
|
|
|
|
|
2021-05-08 17:50:22 -07:00
|
|
|
gs_enable_framebuffer_srgb(previous);
|
|
|
|
|
2020-01-28 10:09:51 +01:00
|
|
|
UNUSED_PARAMETER(t);
|
|
|
|
}
|
|
|
|
|
2021-05-11 00:42:38 -07:00
|
|
|
static void stinger_texrender(struct stinger_info *s, uint32_t source_cx,
|
|
|
|
uint32_t source_cy, uint32_t media_cx,
|
|
|
|
uint32_t media_cy)
|
|
|
|
{
|
|
|
|
if (gs_texrender_begin(s->stinger_tex, source_cx, source_cy)) {
|
|
|
|
float cx = (float)media_cx / s->matte_width_factor;
|
|
|
|
float cy = (float)media_cy / s->matte_height_factor;
|
|
|
|
|
|
|
|
gs_ortho(0.0f, cx, 0.0f, cy, -100.0f, 100.0f);
|
|
|
|
|
|
|
|
gs_blend_state_push();
|
|
|
|
gs_enable_blending(false);
|
|
|
|
obs_source_video_render(s->media_source);
|
|
|
|
gs_blend_state_pop();
|
|
|
|
|
|
|
|
gs_texrender_end(s->stinger_tex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-19 09:32:47 -07:00
|
|
|
static void stinger_video_render(void *data, gs_effect_t *effect)
|
|
|
|
{
|
|
|
|
struct stinger_info *s = data;
|
|
|
|
|
2021-08-10 13:56:50 -07:00
|
|
|
uint32_t media_cx = obs_source_get_width(s->media_source);
|
|
|
|
uint32_t media_cy = obs_source_get_height(s->media_source);
|
|
|
|
|
2020-01-28 10:09:51 +01:00
|
|
|
if (s->track_matte_enabled) {
|
2021-08-10 13:56:50 -07:00
|
|
|
bool ready = obs_source_active(s->media_source) && !!media_cx &&
|
|
|
|
!!media_cy;
|
|
|
|
if (ready) {
|
|
|
|
if (!s->matte_rendered)
|
|
|
|
s->matte_rendered = true;
|
|
|
|
obs_transition_video_render(s->source,
|
|
|
|
stinger_matte_render);
|
|
|
|
} else {
|
|
|
|
obs_transition_video_render_direct(
|
|
|
|
s->source, s->matte_rendered
|
|
|
|
? OBS_TRANSITION_SOURCE_B
|
|
|
|
: OBS_TRANSITION_SOURCE_A);
|
|
|
|
}
|
2021-08-08 15:42:37 -07:00
|
|
|
if (s->matte_layout == MATTE_LAYOUT_MASK)
|
|
|
|
return;
|
2020-01-28 10:09:51 +01:00
|
|
|
} else {
|
|
|
|
float t = obs_transition_get_time(s->source);
|
|
|
|
bool use_a = t < s->transition_point;
|
2017-07-19 09:32:47 -07:00
|
|
|
|
2020-01-28 10:09:51 +01:00
|
|
|
enum obs_transition_target target =
|
|
|
|
use_a ? OBS_TRANSITION_SOURCE_A
|
|
|
|
: OBS_TRANSITION_SOURCE_B;
|
2017-07-19 09:32:47 -07:00
|
|
|
|
2020-01-28 10:09:51 +01:00
|
|
|
if (!obs_transition_video_render_direct(s->source, target))
|
|
|
|
return;
|
|
|
|
}
|
2017-07-19 09:32:47 -07:00
|
|
|
|
|
|
|
/* --------------------- */
|
|
|
|
|
2021-05-11 00:42:38 -07:00
|
|
|
uint32_t source_cx = obs_source_get_width(s->source);
|
|
|
|
uint32_t source_cy = obs_source_get_height(s->source);
|
|
|
|
|
|
|
|
float source_cxf = (float)source_cx;
|
|
|
|
float source_cyf = (float)source_cy;
|
2020-01-28 10:09:51 +01:00
|
|
|
|
2017-07-19 09:32:47 -07:00
|
|
|
if (!media_cx || !media_cy)
|
|
|
|
return;
|
|
|
|
|
2021-05-11 00:42:38 -07:00
|
|
|
if (s->do_texrender) {
|
|
|
|
stinger_texrender(s, source_cx, source_cy, media_cx, media_cy);
|
|
|
|
|
|
|
|
gs_effect_t *e = obs_get_base_effect(OBS_EFFECT_DEFAULT);
|
|
|
|
gs_eparam_t *p = gs_effect_get_param_by_name(e, "image");
|
|
|
|
gs_texture_t *tex = gs_texrender_get_texture(s->stinger_tex);
|
|
|
|
|
|
|
|
gs_effect_set_texture(p, tex);
|
|
|
|
while (gs_effect_loop(e, "Draw"))
|
|
|
|
gs_draw_sprite(NULL, 0, source_cx, source_cy);
|
2021-05-10 00:35:12 -07:00
|
|
|
} else {
|
2021-05-11 00:42:38 -07:00
|
|
|
gs_matrix_push();
|
|
|
|
gs_matrix_scale3f(source_cxf / (float)media_cx,
|
|
|
|
source_cyf / (float)media_cy, 1.0f);
|
|
|
|
obs_source_video_render(s->media_source);
|
|
|
|
gs_matrix_pop();
|
2021-04-18 04:41:30 +02:00
|
|
|
}
|
|
|
|
|
2021-05-10 00:35:12 -07:00
|
|
|
UNUSED_PARAMETER(effect);
|
2017-07-19 09:32:47 -07:00
|
|
|
}
|
|
|
|
|
2021-05-11 00:30:28 -07:00
|
|
|
static void stinger_video_tick(void *data, float seconds)
|
|
|
|
{
|
|
|
|
struct stinger_info *s = data;
|
|
|
|
|
|
|
|
if (s->track_matte_enabled) {
|
2021-05-11 00:42:38 -07:00
|
|
|
gs_texrender_reset(s->stinger_tex);
|
2021-05-11 00:30:28 -07:00
|
|
|
gs_texrender_reset(s->matte_tex);
|
|
|
|
}
|
|
|
|
|
|
|
|
UNUSED_PARAMETER(seconds);
|
|
|
|
}
|
|
|
|
|
2017-07-19 09:32:47 -07:00
|
|
|
static inline float calc_fade(float t, float mul)
|
|
|
|
{
|
|
|
|
t *= mul;
|
|
|
|
return t > 1.0f ? 1.0f : t;
|
|
|
|
}
|
|
|
|
|
2017-09-16 10:11:52 +02:00
|
|
|
static float mix_a_fade_in_out(void *data, float t)
|
2017-07-19 09:32:47 -07:00
|
|
|
{
|
|
|
|
struct stinger_info *s = data;
|
|
|
|
return 1.0f - calc_fade(t, s->transition_a_mul);
|
|
|
|
}
|
|
|
|
|
2017-09-16 10:11:52 +02:00
|
|
|
static float mix_b_fade_in_out(void *data, float t)
|
2017-07-19 09:32:47 -07:00
|
|
|
{
|
|
|
|
struct stinger_info *s = data;
|
|
|
|
return 1.0f - calc_fade(1.0f - t, s->transition_b_mul);
|
|
|
|
}
|
|
|
|
|
2017-09-16 10:11:52 +02:00
|
|
|
static float mix_a_cross_fade(void *data, float t)
|
|
|
|
{
|
|
|
|
UNUSED_PARAMETER(data);
|
|
|
|
return 1.0f - t;
|
|
|
|
}
|
|
|
|
|
|
|
|
static float mix_b_cross_fade(void *data, float t)
|
|
|
|
{
|
|
|
|
UNUSED_PARAMETER(data);
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
2017-07-19 09:32:47 -07:00
|
|
|
static bool stinger_audio_render(void *data, uint64_t *ts_out,
|
|
|
|
struct obs_source_audio_mix *audio,
|
|
|
|
uint32_t mixers, size_t channels,
|
|
|
|
size_t sample_rate)
|
|
|
|
{
|
|
|
|
struct stinger_info *s = data;
|
|
|
|
uint64_t ts = 0;
|
|
|
|
|
2020-01-28 10:09:51 +01:00
|
|
|
if (!s) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-07-19 09:32:47 -07:00
|
|
|
if (!obs_source_audio_pending(s->media_source)) {
|
|
|
|
ts = obs_source_get_audio_timestamp(s->media_source);
|
|
|
|
if (!ts)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool success = obs_transition_audio_render(s->source, ts_out, audio,
|
2017-09-16 10:11:52 +02:00
|
|
|
mixers, channels,
|
|
|
|
sample_rate, s->mix_a,
|
|
|
|
s->mix_b);
|
2017-07-19 09:32:47 -07:00
|
|
|
if (!ts)
|
|
|
|
return success;
|
|
|
|
|
|
|
|
if (!*ts_out || ts < *ts_out)
|
|
|
|
*ts_out = ts;
|
|
|
|
|
|
|
|
struct obs_source_audio_mix child_audio;
|
|
|
|
obs_source_get_audio_mix(s->media_source, &child_audio);
|
|
|
|
|
|
|
|
for (size_t mix = 0; mix < MAX_AUDIO_MIXES; mix++) {
|
|
|
|
if ((mixers & (1 << mix)) == 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
for (size_t ch = 0; ch < channels; ch++) {
|
|
|
|
register float *out = audio->output[mix].data[ch];
|
|
|
|
register float *in = child_audio.output[mix].data[ch];
|
|
|
|
register float *end = in + AUDIO_OUTPUT_FRAMES;
|
|
|
|
|
|
|
|
while (in < end)
|
|
|
|
*(out++) += *(in++);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void stinger_transition_start(void *data)
|
|
|
|
{
|
|
|
|
struct stinger_info *s = data;
|
|
|
|
|
|
|
|
if (s->media_source) {
|
|
|
|
calldata_t cd = {0};
|
|
|
|
|
|
|
|
proc_handler_t *ph =
|
|
|
|
obs_source_get_proc_handler(s->media_source);
|
2020-01-28 10:09:51 +01:00
|
|
|
proc_handler_t *matte_ph =
|
2021-05-11 00:52:54 -07:00
|
|
|
s->matte_source
|
|
|
|
? obs_source_get_proc_handler(s->matte_source)
|
|
|
|
: NULL;
|
2017-07-19 09:32:47 -07:00
|
|
|
|
|
|
|
if (s->transitioning) {
|
|
|
|
proc_handler_call(ph, "restart", &cd);
|
2020-01-28 10:09:51 +01:00
|
|
|
if (matte_ph) {
|
|
|
|
proc_handler_call(matte_ph, "restart", &cd);
|
|
|
|
}
|
2017-07-19 09:32:47 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-08-10 13:56:50 -07:00
|
|
|
s->matte_rendered = false;
|
|
|
|
|
2017-07-19 09:32:47 -07:00
|
|
|
proc_handler_call(ph, "get_duration", &cd);
|
2017-08-08 10:08:23 +02:00
|
|
|
proc_handler_call(ph, "get_nb_frames", &cd);
|
2019-10-05 12:51:01 -07:00
|
|
|
s->duration_ns =
|
2020-02-01 09:50:57 +11:00
|
|
|
(uint64_t)calldata_int(&cd, "duration") + 250000000ULL;
|
2017-08-08 10:08:23 +02:00
|
|
|
s->duration_frames = (uint64_t)calldata_int(&cd, "num_frames");
|
|
|
|
|
|
|
|
if (s->transition_point_is_frame)
|
|
|
|
s->transition_point =
|
|
|
|
(float)((long double)s->transition_point_frame /
|
|
|
|
(long double)s->duration_frames);
|
|
|
|
else
|
|
|
|
s->transition_point =
|
|
|
|
(float)((long double)s->transition_point_ns /
|
|
|
|
(long double)s->duration_ns);
|
2017-07-19 09:32:47 -07:00
|
|
|
|
2018-06-11 19:28:33 -07:00
|
|
|
if (s->transition_point > 0.999f)
|
|
|
|
s->transition_point = 0.999f;
|
2017-07-19 09:32:47 -07:00
|
|
|
else if (s->transition_point < 0.001f)
|
|
|
|
s->transition_point = 0.001f;
|
|
|
|
|
|
|
|
s->transition_a_mul = (1.0f / s->transition_point);
|
|
|
|
s->transition_b_mul = (1.0f / (1.0f - s->transition_point));
|
|
|
|
|
2021-05-11 00:51:02 -07:00
|
|
|
if (s->track_matte_enabled && s->matte_source) {
|
2020-01-28 10:09:51 +01:00
|
|
|
proc_handler_call(matte_ph, "get_duration", &cd);
|
|
|
|
uint64_t tm_duration_ns =
|
|
|
|
(uint64_t)calldata_int(&cd, "duration");
|
|
|
|
|
|
|
|
s->duration_ns = ((tm_duration_ns > s->duration_ns)
|
|
|
|
? (tm_duration_ns)
|
|
|
|
: (s->duration_ns));
|
|
|
|
|
|
|
|
obs_source_add_active_child(s->source, s->matte_source);
|
|
|
|
}
|
|
|
|
|
2017-07-19 09:32:47 -07:00
|
|
|
obs_transition_enable_fixed(
|
|
|
|
s->source, true, (uint32_t)(s->duration_ns / 1000000));
|
|
|
|
|
|
|
|
calldata_free(&cd);
|
|
|
|
|
|
|
|
obs_source_add_active_child(s->source, s->media_source);
|
|
|
|
}
|
|
|
|
|
|
|
|
s->transitioning = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void stinger_transition_stop(void *data)
|
|
|
|
{
|
|
|
|
struct stinger_info *s = data;
|
|
|
|
|
|
|
|
if (s->media_source)
|
|
|
|
obs_source_remove_active_child(s->source, s->media_source);
|
|
|
|
|
2020-01-28 10:09:51 +01:00
|
|
|
if (s->matte_source)
|
|
|
|
obs_source_remove_active_child(s->source, s->matte_source);
|
|
|
|
|
2017-07-19 09:32:47 -07:00
|
|
|
s->transitioning = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void stinger_enum_active_sources(void *data,
|
|
|
|
obs_source_enum_proc_t enum_callback,
|
|
|
|
void *param)
|
|
|
|
{
|
|
|
|
struct stinger_info *s = data;
|
|
|
|
if (s->media_source && s->transitioning)
|
|
|
|
enum_callback(s->source, s->media_source, param);
|
2020-01-28 10:09:51 +01:00
|
|
|
|
|
|
|
if (s->matte_source && s->transitioning)
|
|
|
|
enum_callback(s->source, s->matte_source, param);
|
2017-07-19 09:32:47 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void stinger_enum_all_sources(void *data,
|
|
|
|
obs_source_enum_proc_t enum_callback,
|
|
|
|
void *param)
|
|
|
|
{
|
|
|
|
struct stinger_info *s = data;
|
|
|
|
if (s->media_source)
|
|
|
|
enum_callback(s->source, s->media_source, param);
|
2020-01-28 10:09:51 +01:00
|
|
|
|
|
|
|
if (s->matte_source)
|
|
|
|
enum_callback(s->source, s->matte_source, param);
|
2017-07-19 09:32:47 -07:00
|
|
|
}
|
|
|
|
|
2022-01-30 12:53:14 +11:00
|
|
|
#define FILE_FILTER " (*.mp4 *.ts *.mov *.wmv *.flv *.mkv *.avi *.gif *.webm);;"
|
2017-07-19 09:32:47 -07:00
|
|
|
|
2017-08-08 08:10:27 -07:00
|
|
|
static bool transition_point_type_modified(obs_properties_t *ppts,
|
|
|
|
obs_property_t *p, obs_data_t *s)
|
|
|
|
{
|
|
|
|
int64_t type = obs_data_get_int(s, "tp_type");
|
2020-01-28 10:09:51 +01:00
|
|
|
|
|
|
|
obs_property_t *prop_transition_point =
|
|
|
|
obs_properties_get(ppts, "transition_point");
|
2017-08-08 08:10:27 -07:00
|
|
|
|
2019-07-22 15:21:17 -05:00
|
|
|
if (type == TIMING_TIME) {
|
2017-08-08 08:10:27 -07:00
|
|
|
obs_property_set_description(
|
2020-01-28 10:09:51 +01:00
|
|
|
prop_transition_point,
|
|
|
|
obs_module_text("TransitionPoint"));
|
|
|
|
} else {
|
|
|
|
obs_property_set_description(
|
|
|
|
prop_transition_point,
|
|
|
|
obs_module_text("TransitionPointFrame"));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool uses_ms_prefix = (type == TIMING_TIME);
|
|
|
|
obs_property_int_set_suffix(p, (uses_ms_prefix ? " ms" : ""));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool track_matte_layout_modified(obs_properties_t *ppts,
|
|
|
|
obs_property_t *p, obs_data_t *s)
|
|
|
|
{
|
2021-03-31 05:09:51 -07:00
|
|
|
int matte_layout = (int)obs_data_get_int(s, "track_matte_layout");
|
2020-01-28 10:09:51 +01:00
|
|
|
obs_property_t *prop_matte_path =
|
|
|
|
obs_properties_get(ppts, "track_matte_path");
|
|
|
|
|
|
|
|
bool uses_separate_file = (matte_layout == MATTE_LAYOUT_SEPARATE_FILE);
|
|
|
|
obs_property_set_visible(prop_matte_path, uses_separate_file);
|
|
|
|
|
|
|
|
UNUSED_PARAMETER(p);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool track_matte_enabled_modified(obs_properties_t *ppts,
|
|
|
|
obs_property_t *p, obs_data_t *s)
|
|
|
|
{
|
|
|
|
bool track_matte_enabled = obs_data_get_bool(s, "track_matte_enabled");
|
|
|
|
obs_property_t *prop_tp_type = obs_properties_get(ppts, "tp_type");
|
|
|
|
|
|
|
|
if (track_matte_enabled) {
|
|
|
|
obs_property_set_description(
|
|
|
|
prop_tp_type,
|
|
|
|
obs_module_text("AudioTransitionPointType"));
|
2019-07-22 15:21:17 -05:00
|
|
|
} else {
|
2017-08-08 08:10:27 -07:00
|
|
|
obs_property_set_description(
|
2020-01-28 10:09:51 +01:00
|
|
|
prop_tp_type, obs_module_text("TransitionPointType"));
|
2019-07-22 15:21:17 -05:00
|
|
|
}
|
2020-01-28 10:09:51 +01:00
|
|
|
|
|
|
|
UNUSED_PARAMETER(p);
|
2017-08-08 08:10:27 -07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-07-19 09:32:47 -07:00
|
|
|
static obs_properties_t *stinger_properties(void *data)
|
|
|
|
{
|
|
|
|
obs_properties_t *ppts = obs_properties_create();
|
2022-01-30 12:53:14 +11:00
|
|
|
struct dstr filter = {0};
|
2017-07-19 09:32:47 -07:00
|
|
|
|
|
|
|
obs_properties_set_flags(ppts, OBS_PROPERTIES_DEFER_UPDATE);
|
|
|
|
|
2022-01-30 12:53:14 +11:00
|
|
|
dstr_copy(&filter, obs_module_text("FileFilter.VideoFiles"));
|
|
|
|
dstr_cat(&filter, FILE_FILTER);
|
|
|
|
dstr_cat(&filter, obs_module_text("FileFilter.AllFiles"));
|
|
|
|
dstr_cat(&filter, " (*.*)");
|
|
|
|
|
2020-01-28 10:09:51 +01:00
|
|
|
// main stinger settings
|
2017-07-19 09:32:47 -07:00
|
|
|
obs_properties_add_path(ppts, "path", obs_module_text("VideoFile"),
|
2022-01-30 12:53:14 +11:00
|
|
|
OBS_PATH_FILE, filter.array, NULL);
|
2020-01-28 10:09:51 +01:00
|
|
|
|
2019-07-18 02:38:50 -05:00
|
|
|
obs_property_t *p = obs_properties_add_list(
|
2017-08-08 10:08:23 +02:00
|
|
|
ppts, "tp_type", obs_module_text("TransitionPointType"),
|
|
|
|
OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT);
|
2020-09-01 22:16:11 +10:00
|
|
|
obs_properties_add_bool(ppts, "hw_decode",
|
|
|
|
obs_module_text("HardwareDecode"));
|
2019-07-18 02:38:50 -05:00
|
|
|
obs_property_list_add_int(p, obs_module_text("TransitionPointTypeTime"),
|
|
|
|
TIMING_TIME);
|
2017-08-08 10:08:23 +02:00
|
|
|
obs_property_list_add_int(
|
2019-07-18 02:38:50 -05:00
|
|
|
p, obs_module_text("TransitionPointTypeFrame"), TIMING_FRAME);
|
2017-08-08 08:10:27 -07:00
|
|
|
|
2019-07-18 02:38:50 -05:00
|
|
|
obs_property_set_modified_callback(p, transition_point_type_modified);
|
2017-08-08 10:08:23 +02:00
|
|
|
|
2019-07-22 15:21:17 -05:00
|
|
|
obs_properties_add_int(ppts, "transition_point",
|
|
|
|
obs_module_text("TransitionPoint"), 0, 120000,
|
|
|
|
1);
|
2017-07-19 09:32:47 -07:00
|
|
|
|
2020-01-28 10:09:51 +01:00
|
|
|
// track matte properties
|
|
|
|
{
|
|
|
|
obs_properties_t *track_matte_group = obs_properties_create();
|
|
|
|
|
|
|
|
p = obs_properties_add_list(track_matte_group,
|
|
|
|
"track_matte_layout",
|
|
|
|
obs_module_text("TrackMatteLayout"),
|
|
|
|
OBS_COMBO_TYPE_LIST,
|
|
|
|
OBS_COMBO_FORMAT_INT);
|
|
|
|
obs_property_list_add_int(
|
|
|
|
p, obs_module_text("TrackMatteLayoutHorizontal"),
|
|
|
|
MATTE_LAYOUT_HORIZONTAL);
|
|
|
|
obs_property_list_add_int(
|
|
|
|
p, obs_module_text("TrackMatteLayoutVertical"),
|
|
|
|
MATTE_LAYOUT_VERTICAL);
|
2021-05-23 03:57:58 -07:00
|
|
|
|
|
|
|
/* TODO: Requires way to synchronize or combine two media files
|
|
|
|
* together */
|
|
|
|
#if 0
|
2020-01-28 10:09:51 +01:00
|
|
|
obs_property_list_add_int(
|
|
|
|
p, obs_module_text("TrackMatteLayoutSeparateFile"),
|
|
|
|
MATTE_LAYOUT_SEPARATE_FILE);
|
2021-05-23 03:57:58 -07:00
|
|
|
#endif
|
2021-08-08 15:42:37 -07:00
|
|
|
obs_property_list_add_int(
|
|
|
|
p, obs_module_text("TrackMatteLayoutMask"),
|
|
|
|
MATTE_LAYOUT_MASK);
|
2020-01-28 10:09:51 +01:00
|
|
|
|
|
|
|
obs_property_set_modified_callback(p,
|
|
|
|
track_matte_layout_modified);
|
|
|
|
|
|
|
|
obs_properties_add_path(track_matte_group, "track_matte_path",
|
|
|
|
obs_module_text("TrackMatteVideoFile"),
|
2022-01-30 12:53:14 +11:00
|
|
|
OBS_PATH_FILE, filter.array, NULL);
|
2020-01-28 10:09:51 +01:00
|
|
|
|
|
|
|
obs_properties_add_bool(track_matte_group, "invert_matte",
|
|
|
|
obs_module_text("InvertTrackMatte"));
|
|
|
|
|
|
|
|
p = obs_properties_add_group(
|
|
|
|
ppts, "track_matte_enabled",
|
|
|
|
obs_module_text("TrackMatteEnabled"),
|
|
|
|
OBS_GROUP_CHECKABLE, track_matte_group);
|
|
|
|
|
|
|
|
obs_property_set_modified_callback(
|
|
|
|
p, track_matte_enabled_modified);
|
|
|
|
}
|
2022-01-30 12:53:14 +11:00
|
|
|
dstr_free(&filter);
|
2020-01-28 10:09:51 +01:00
|
|
|
|
|
|
|
// audio output settings
|
2017-08-11 17:14:00 +02:00
|
|
|
obs_property_t *monitor_list = obs_properties_add_list(
|
|
|
|
ppts, "audio_monitoring", obs_module_text("AudioMonitoring"),
|
|
|
|
OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT);
|
|
|
|
obs_property_list_add_int(monitor_list,
|
|
|
|
obs_module_text("AudioMonitoring.None"),
|
|
|
|
OBS_MONITORING_TYPE_NONE);
|
|
|
|
obs_property_list_add_int(
|
|
|
|
monitor_list, obs_module_text("AudioMonitoring.MonitorOnly"),
|
|
|
|
OBS_MONITORING_TYPE_MONITOR_ONLY);
|
|
|
|
obs_property_list_add_int(monitor_list,
|
|
|
|
obs_module_text("AudioMonitoring.Both"),
|
|
|
|
OBS_MONITORING_TYPE_MONITOR_AND_OUTPUT);
|
2019-06-22 22:13:45 -07:00
|
|
|
|
2020-01-28 10:09:51 +01:00
|
|
|
// audio fade settings
|
2017-09-16 10:11:52 +02:00
|
|
|
obs_property_t *audio_fade_style = obs_properties_add_list(
|
|
|
|
ppts, "audio_fade_style", obs_module_text("AudioFadeStyle"),
|
|
|
|
OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT);
|
|
|
|
obs_property_list_add_int(
|
|
|
|
audio_fade_style,
|
|
|
|
obs_module_text("AudioFadeStyle.FadeOutFadeIn"),
|
|
|
|
FADE_STYLE_FADE_OUT_FADE_IN);
|
|
|
|
obs_property_list_add_int(audio_fade_style,
|
|
|
|
obs_module_text("AudioFadeStyle.CrossFade"),
|
|
|
|
FADE_STYLE_CROSS_FADE);
|
|
|
|
|
2017-07-19 09:32:47 -07:00
|
|
|
UNUSED_PARAMETER(data);
|
|
|
|
return ppts;
|
|
|
|
}
|
|
|
|
|
2021-08-15 21:51:24 +02:00
|
|
|
static void missing_file_callback(void *src, const char *new_path, void *data)
|
|
|
|
{
|
|
|
|
struct stinger_info *s = src;
|
|
|
|
obs_data_t *settings = obs_source_get_settings(s->source);
|
|
|
|
|
|
|
|
const char *type = data;
|
|
|
|
|
|
|
|
if (strcmp(type, "media_source") == 0) {
|
|
|
|
obs_data_set_string(settings, "path", new_path);
|
|
|
|
} else if (strcmp(type, "matte_source") == 0) {
|
|
|
|
obs_data_set_string(settings, "track_matte_path", new_path);
|
|
|
|
}
|
|
|
|
|
|
|
|
obs_source_update(s->source, settings);
|
|
|
|
obs_data_release(settings);
|
|
|
|
}
|
|
|
|
|
|
|
|
static obs_missing_files_t *stinger_missing_files(void *data)
|
|
|
|
{
|
|
|
|
struct stinger_info *s = data;
|
|
|
|
obs_data_t *settings = obs_source_get_settings(s->source);
|
|
|
|
obs_missing_files_t *files = obs_missing_files_create();
|
|
|
|
|
|
|
|
const char *path = obs_data_get_string(settings, "path");
|
|
|
|
|
|
|
|
if (strcmp(path, "") != 0) {
|
|
|
|
if (!os_file_exists(path)) {
|
|
|
|
obs_missing_file_t *file = obs_missing_file_create(
|
|
|
|
path, missing_file_callback,
|
|
|
|
OBS_MISSING_FILE_SOURCE, s->source,
|
|
|
|
(void *)"media_source");
|
|
|
|
|
|
|
|
obs_missing_files_add_file(files, file);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *track_matte_path =
|
|
|
|
obs_data_get_string(settings, "track_matte_path");
|
|
|
|
|
|
|
|
if (strcmp(track_matte_path, "") != 0) {
|
|
|
|
if (!os_file_exists(track_matte_path)) {
|
|
|
|
obs_missing_file_t *file = obs_missing_file_create(
|
|
|
|
track_matte_path, missing_file_callback,
|
|
|
|
OBS_MISSING_FILE_SOURCE, s->source,
|
|
|
|
(void *)"matte_source");
|
|
|
|
|
|
|
|
obs_missing_files_add_file(files, file);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
obs_data_release(settings);
|
|
|
|
return files;
|
|
|
|
}
|
|
|
|
|
2017-07-19 09:32:47 -07:00
|
|
|
struct obs_source_info stinger_transition = {
|
2017-08-10 07:04:56 -07:00
|
|
|
.id = "obs_stinger_transition",
|
2017-07-19 09:32:47 -07:00
|
|
|
.type = OBS_SOURCE_TYPE_TRANSITION,
|
|
|
|
.get_name = stinger_get_name,
|
|
|
|
.create = stinger_create,
|
|
|
|
.destroy = stinger_destroy,
|
|
|
|
.update = stinger_update,
|
2020-09-01 22:16:11 +10:00
|
|
|
.get_defaults = stinger_defaults,
|
2017-07-19 09:32:47 -07:00
|
|
|
.video_render = stinger_video_render,
|
2021-05-11 00:30:28 -07:00
|
|
|
.video_tick = stinger_video_tick,
|
2017-07-19 09:32:47 -07:00
|
|
|
.audio_render = stinger_audio_render,
|
2021-08-15 21:51:24 +02:00
|
|
|
.missing_files = stinger_missing_files,
|
2017-07-19 09:32:47 -07:00
|
|
|
.get_properties = stinger_properties,
|
|
|
|
.enum_active_sources = stinger_enum_active_sources,
|
|
|
|
.enum_all_sources = stinger_enum_all_sources,
|
|
|
|
.transition_start = stinger_transition_start,
|
|
|
|
.transition_stop = stinger_transition_stop,
|
|
|
|
};
|