obs-studio/plugins/obs-filters/scroll-filter.c

272 lines
7.1 KiB
C

#include <obs-module.h>
#include <graphics/vec2.h>
struct scroll_filter_data {
obs_source_t *context;
gs_effect_t *effect;
gs_eparam_t *param_add;
gs_eparam_t *param_mul;
gs_eparam_t *param_image;
struct vec2 scroll_speed;
gs_samplerstate_t *sampler;
bool limit_cx;
bool limit_cy;
uint32_t cx;
uint32_t cy;
struct vec2 size_i;
struct vec2 offset;
bool loop;
};
static const char *scroll_filter_get_name(void *unused)
{
UNUSED_PARAMETER(unused);
return obs_module_text("ScrollFilter");
}
static void *scroll_filter_create(obs_data_t *settings, obs_source_t *context)
{
struct scroll_filter_data *filter = bzalloc(sizeof(*filter));
char *effect_path = obs_module_file("crop_filter.effect");
filter->context = context;
obs_enter_graphics();
filter->effect = gs_effect_create_from_file(effect_path, NULL);
obs_leave_graphics();
bfree(effect_path);
if (!filter->effect) {
bfree(filter);
return NULL;
}
filter->param_add =
gs_effect_get_param_by_name(filter->effect, "add_val");
filter->param_mul =
gs_effect_get_param_by_name(filter->effect, "mul_val");
filter->param_image =
gs_effect_get_param_by_name(filter->effect, "image");
obs_source_update(context, settings);
return filter;
}
static void scroll_filter_destroy(void *data)
{
struct scroll_filter_data *filter = data;
obs_enter_graphics();
gs_effect_destroy(filter->effect);
gs_samplerstate_destroy(filter->sampler);
obs_leave_graphics();
bfree(filter);
}
static void scroll_filter_update(void *data, obs_data_t *settings)
{
struct scroll_filter_data *filter = data;
filter->limit_cx = obs_data_get_bool(settings, "limit_cx");
filter->limit_cy = obs_data_get_bool(settings, "limit_cy");
filter->cx = (uint32_t)obs_data_get_int(settings, "cx");
filter->cy = (uint32_t)obs_data_get_int(settings, "cy");
filter->scroll_speed.x =
(float)obs_data_get_double(settings, "speed_x");
filter->scroll_speed.y =
(float)obs_data_get_double(settings, "speed_y");
filter->loop = obs_data_get_bool(settings, "loop");
struct gs_sampler_info sampler_info = {
.filter = GS_FILTER_LINEAR,
.address_u = filter->loop ? GS_ADDRESS_WRAP : GS_ADDRESS_BORDER,
.address_v = filter->loop ? GS_ADDRESS_WRAP : GS_ADDRESS_BORDER,
};
obs_enter_graphics();
gs_samplerstate_destroy(filter->sampler);
filter->sampler = gs_samplerstate_create(&sampler_info);
obs_leave_graphics();
if (filter->scroll_speed.x == 0.0f)
filter->offset.x = 0.0f;
if (filter->scroll_speed.y == 0.0f)
filter->offset.y = 0.0f;
}
static bool limit_cx_clicked(obs_properties_t *props, obs_property_t *p,
obs_data_t *settings)
{
bool limit_size = obs_data_get_bool(settings, "limit_cx");
obs_property_set_visible(obs_properties_get(props, "cx"), limit_size);
UNUSED_PARAMETER(p);
return true;
}
static bool limit_cy_clicked(obs_properties_t *props, obs_property_t *p,
obs_data_t *settings)
{
bool limit_size = obs_data_get_bool(settings, "limit_cy");
obs_property_set_visible(obs_properties_get(props, "cy"), limit_size);
UNUSED_PARAMETER(p);
return true;
}
static obs_properties_t *scroll_filter_properties(void *data)
{
obs_properties_t *props = obs_properties_create();
obs_property_t *p;
obs_properties_add_float_slider(props, "speed_x",
obs_module_text("ScrollFilter.SpeedX"),
-500.0, 500.0, 1.0);
obs_properties_add_float_slider(props, "speed_y",
obs_module_text("ScrollFilter.SpeedY"),
-500.0, 500.0, 1.0);
p = obs_properties_add_bool(props, "limit_cx",
obs_module_text("ScrollFilter.LimitWidth"));
obs_property_set_modified_callback(p, limit_cx_clicked);
obs_properties_add_int(props, "cx", obs_module_text("Crop.Width"), 1,
8192, 1);
p = obs_properties_add_bool(
props, "limit_cy", obs_module_text("ScrollFilter.LimitHeight"));
obs_property_set_modified_callback(p, limit_cy_clicked);
obs_properties_add_int(props, "cy", obs_module_text("Crop.Height"), 1,
8192, 1);
obs_properties_add_bool(props, "loop",
obs_module_text("ScrollFilter.Loop"));
UNUSED_PARAMETER(data);
return props;
}
static void scroll_filter_defaults(obs_data_t *settings)
{
obs_data_set_default_bool(settings, "limit_size", false);
obs_data_set_default_int(settings, "cx", 100);
obs_data_set_default_int(settings, "cy", 100);
obs_data_set_default_bool(settings, "loop", true);
}
static void scroll_filter_tick(void *data, float seconds)
{
struct scroll_filter_data *filter = data;
filter->offset.x += filter->size_i.x * filter->scroll_speed.x * seconds;
filter->offset.y += filter->size_i.y * filter->scroll_speed.y * seconds;
if (filter->loop) {
if (filter->offset.x > 1.0f)
filter->offset.x -= 1.0f;
if (filter->offset.y > 1.0f)
filter->offset.y -= 1.0f;
} else {
if (filter->offset.x > 1.0f)
filter->offset.x = 1.0f;
if (filter->offset.y > 1.0f)
filter->offset.y = 1.0f;
}
}
static void scroll_filter_render(void *data, gs_effect_t *effect)
{
struct scroll_filter_data *filter = data;
struct vec2 mul_val;
uint32_t base_cx;
uint32_t base_cy;
uint32_t cx;
uint32_t cy;
obs_source_t *target = obs_filter_get_target(filter->context);
base_cx = obs_source_get_base_width(target);
base_cy = obs_source_get_base_height(target);
cx = filter->limit_cx ? filter->cx : base_cx;
cy = filter->limit_cy ? filter->cy : base_cy;
if (base_cx && base_cy) {
vec2_set(&filter->size_i, 1.0f / (float)base_cx,
1.0f / (float)base_cy);
} else {
vec2_zero(&filter->size_i);
obs_source_skip_video_filter(filter->context);
return;
}
vec2_set(&mul_val, (float)cx / (float)base_cx,
(float)cy / (float)base_cy);
if (!obs_source_process_filter_begin(filter->context, GS_RGBA,
OBS_NO_DIRECT_RENDERING))
return;
gs_effect_set_vec2(filter->param_add, &filter->offset);
gs_effect_set_vec2(filter->param_mul, &mul_val);
gs_effect_set_next_sampler(filter->param_image, filter->sampler);
gs_blend_state_push();
gs_blend_function(GS_BLEND_ONE, GS_BLEND_INVSRCALPHA);
obs_source_process_filter_end(filter->context, filter->effect, cx, cy);
gs_blend_state_pop();
UNUSED_PARAMETER(effect);
}
static uint32_t scroll_filter_width(void *data)
{
struct scroll_filter_data *filter = data;
obs_source_t *target = obs_filter_get_target(filter->context);
return filter->limit_cx ? filter->cx
: obs_source_get_base_width(target);
}
static uint32_t scroll_filter_height(void *data)
{
struct scroll_filter_data *filter = data;
obs_source_t *target = obs_filter_get_target(filter->context);
return filter->limit_cy ? filter->cy
: obs_source_get_base_height(target);
}
static void scroll_filter_show(void *data)
{
struct scroll_filter_data *filter = data;
filter->offset.x = 0.0f;
filter->offset.y = 0.0f;
}
struct obs_source_info scroll_filter = {
.id = "scroll_filter",
.type = OBS_SOURCE_TYPE_FILTER,
.output_flags = OBS_SOURCE_VIDEO | OBS_SOURCE_SRGB,
.get_name = scroll_filter_get_name,
.create = scroll_filter_create,
.destroy = scroll_filter_destroy,
.update = scroll_filter_update,
.get_properties = scroll_filter_properties,
.get_defaults = scroll_filter_defaults,
.video_tick = scroll_filter_tick,
.video_render = scroll_filter_render,
.get_width = scroll_filter_width,
.get_height = scroll_filter_height,
.show = scroll_filter_show,
};