/* * Modern effects for a modern Streamer * Copyright (C) 2017 Michael Fabian Dirks * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA */ #include "filter-shader.hpp" #include "strings.hpp" #include "utility.hpp" #define ST "Filter.Shader" P_INITIALIZER(FilterShaderInit) { initializer_functions.push_back([] { filter::shader::shader_factory::initialize(); }); finalizer_functions.push_back([] { filter::shader::shader_factory::finalize(); }); } static std::shared_ptr factory_instance = nullptr; void filter::shader::shader_factory::initialize() { factory_instance = std::make_shared(); } void filter::shader::shader_factory::finalize() { factory_instance.reset(); } std::shared_ptr filter::shader::shader_factory::get() { return factory_instance; } static void get_defaults(obs_data_t* data) { obs_data_set_default_string(data, S_SHADER_FILE, obs_module_file("shaders/filter/example.effect")); obs_data_set_default_string(data, S_SHADER_TECHNIQUE, "Draw"); } filter::shader::shader_factory::shader_factory() { memset(&_source_info, 0, sizeof(obs_source_info)); _source_info.id = "obs-stream-effects-filter-shader"; _source_info.type = OBS_SOURCE_TYPE_FILTER; _source_info.output_flags = OBS_SOURCE_VIDEO; _source_info.get_name = [](void*) { return D_TRANSLATE(ST); }; _source_info.get_defaults = get_defaults; _source_info.create = [](obs_data_t* data, obs_source_t* self) { try { return static_cast(new filter::shader::shader_instance(data, self)); } catch (std::exception& ex) { P_LOG_ERROR(" Failed to create source, error: %s", ex.what()); } catch (...) { P_LOG_ERROR(" Failed to create source."); } return static_cast(nullptr); }; _source_info.destroy = [](void* ptr) { try { delete reinterpret_cast(ptr); } catch (std::exception& ex) { P_LOG_ERROR(" Failed to delete source, error: %s", ex.what()); } catch (...) { P_LOG_ERROR(" Failed to delete source."); } }; _source_info.get_properties = [](void* ptr) { obs_properties_t* pr = obs_properties_create(); try { if (ptr) reinterpret_cast(ptr)->properties(pr); } catch (std::exception& ex) { P_LOG_ERROR(" Failed to retrieve options, error: %s", ex.what()); } catch (...) { P_LOG_ERROR(" Failed to retrieve options."); } return pr; }; _source_info.get_width = [](void* ptr) { try { if (ptr) return reinterpret_cast(ptr)->width(); } catch (std::exception& ex) { P_LOG_ERROR(" Failed to retrieve width, error: %s", ex.what()); } catch (...) { P_LOG_ERROR(" Failed to retrieve width."); } return uint32_t(0); }; _source_info.get_height = [](void* ptr) { try { if (ptr) return reinterpret_cast(ptr)->height(); } catch (std::exception& ex) { P_LOG_ERROR(" Failed to retrieve height, error: %s", ex.what()); } catch (...) { P_LOG_ERROR(" Failed to retrieve height."); } return uint32_t(0); }; _source_info.update = [](void* ptr, obs_data_t* data) { try { if (ptr) reinterpret_cast(ptr)->update(data); } catch (std::exception& ex) { P_LOG_ERROR(" Failed to update, error: %s", ex.what()); } catch (...) { P_LOG_ERROR(" Failed to update."); } }; _source_info.activate = [](void* ptr) { try { if (ptr) reinterpret_cast(ptr)->activate(); } catch (std::exception& ex) { P_LOG_ERROR(" Failed to activate, error: %s", ex.what()); } catch (...) { P_LOG_ERROR(" Failed to activate."); } }; _source_info.deactivate = [](void* ptr) { try { if (ptr) reinterpret_cast(ptr)->deactivate(); } catch (std::exception& ex) { P_LOG_ERROR(" Failed to deactivate, error: %s", ex.what()); } catch (...) { P_LOG_ERROR(" Failed to deactivate."); } }; _source_info.video_tick = [](void* ptr, float_t time) { try { if (ptr) reinterpret_cast(ptr)->video_tick(time); } catch (std::exception& ex) { P_LOG_ERROR(" Failed to tick, error: %s", ex.what()); } catch (...) { P_LOG_ERROR(" Failed to tick."); } }; _source_info.video_render = [](void* ptr, gs_effect_t* effect) { try { if (ptr) reinterpret_cast(ptr)->video_render(effect); } catch (std::exception& ex) { P_LOG_ERROR(" Failed to render, error: %s", ex.what()); } catch (...) { P_LOG_ERROR(" Failed to render."); } }; obs_register_source(&_source_info); } filter::shader::shader_factory::~shader_factory() {} filter::shader::shader_instance::shader_instance(obs_data_t* data, obs_source_t* self) : _self(self), _active(true), _width(0), _height(0) { _fx = std::make_shared(); _fx->set_valid_property_cb(std::bind(&filter::shader::shader_instance::valid_param, this, std::placeholders::_1)); _fx->set_override_cb(std::bind(&filter::shader::shader_instance::override_param, this, std::placeholders::_1)); _rt = std::make_shared(GS_RGBA, GS_ZS_NONE); _rt2 = std::make_shared(GS_RGBA, GS_ZS_NONE); update(data); } filter::shader::shader_instance::~shader_instance() {} uint32_t filter::shader::shader_instance::width() { return _width; } uint32_t filter::shader::shader_instance::height() { return _height; } void filter::shader::shader_instance::properties(obs_properties_t* props) { _fx->properties(props); } void filter::shader::shader_instance::update(obs_data_t* data) { _fx->update(data); } void filter::shader::shader_instance::activate() { _active = true; } void filter::shader::shader_instance::deactivate() { _active = false; } bool filter::shader::shader_instance::valid_param(std::shared_ptr param) { if (strcmpi(param->get_name().c_str(), "ImageSource")) return false; if (strcmpi(param->get_name().c_str(), "ImageSource_Size")) return false; if (strcmpi(param->get_name().c_str(), "ImageSource_Texel")) return false; return true; } void filter::shader::shader_instance::override_param(std::shared_ptr effect) { auto p_source = effect->get_parameter("ImageSource"); auto p_source_size = effect->get_parameter("ImageSource_Size"); auto p_source_texel = effect->get_parameter("ImageSource_Texel"); if (p_source && (p_source->get_type() == gs::effect_parameter::type::Texture)) { p_source->set_texture(_rt_tex); } if (p_source_size && (p_source_size->get_type() == gs::effect_parameter::type::Float2)) { p_source_size->set_float2(_width, _height); } if (p_source_texel && (p_source_size->get_type() == gs::effect_parameter::type::Float2)) { p_source_texel->set_float2(1.0f / _width, 1.0f / _height); } } void filter::shader::shader_instance::video_tick(float_t sec_since_last) { obs_source_t* target = obs_filter_get_target(_self); { // Update width and height. _width = obs_source_get_base_width(target); _height = obs_source_get_base_height(target); } if (_fx->tick(sec_since_last)) { obs_data_t* data = obs_source_get_settings(_self); update(data); obs_data_release(data); } _rt_updated = false; _rt2_updated = false; } void filter::shader::shader_instance::video_render(gs_effect_t* effect) { // Grab initial values. obs_source_t* parent = obs_filter_get_parent(_self); obs_source_t* target = obs_filter_get_target(_self); gs_effect_t* effect_default = obs_get_base_effect(obs_base_effect::OBS_EFFECT_DEFAULT); // Skip filter if anything is wrong. if (!_active || !parent || !target || !_width || !_height || !effect_default) { obs_source_skip_video_filter(_self); return; } if (!_rt_updated) { if (obs_source_process_filter_begin(_self, GS_RGBA, OBS_ALLOW_DIRECT_RENDERING)) { auto op = _rt->render(_width, _height); gs_blend_state_push(); gs_reset_blend_state(); gs_set_cull_mode(GS_NEITHER); gs_enable_color(true, true, true, true); gs_enable_blending(false); gs_enable_depth_test(false); gs_enable_stencil_test(false); gs_enable_stencil_write(false); gs_ortho(0, static_cast(_width), 0, static_cast(_height), -1., 1.); obs_source_process_filter_end(_self, effect_default, _width, _height); gs_blend_state_pop(); } _rt_tex = _rt->get_texture(); _rt_updated = true; } if (!_rt2_updated) { try { auto op = _rt2->render(_width, _height); _fx->render(); } catch (...) { obs_source_skip_video_filter(_self); } _rt2_tex = _rt2->get_texture(); _rt2_updated = true; } if (!_rt2_tex) return; gs_effect_t* ef = effect ? effect : effect_default; if (gs_eparam_t* prm = gs_effect_get_param_by_name(ef, "image")) gs_effect_set_texture(prm, _rt2_tex->get_object()); while (gs_effect_loop(ef, "Draw")) { gs_draw_sprite(nullptr, 0, _width, _height); } }