filter-blur: Multiple changes, see description
* Added a 'Bilateral' de-noising blur, currently very resource hungry. It also has two extra sliders called 'Smoothing' and 'Sharpness', the former controls how smooth the output is * Updated effect loading to be global instead of per instance. * Changed default blur size to 5px. * Simplified render loop (see previous effect commit).
This commit is contained in:
		
							parent
							
								
									8f05b348dd
								
							
						
					
					
						commit
						2621d4961f
					
				|  | @ -0,0 +1,79 @@ | ||||||
|  | uniform float4x4 ViewProj; | ||||||
|  | uniform texture2d image; | ||||||
|  | uniform float2 texel; | ||||||
|  | uniform int widthHalf; | ||||||
|  | uniform int width; | ||||||
|  | uniform float bilateralSmoothing; | ||||||
|  | uniform float bilateralSharpness; | ||||||
|  | 
 | ||||||
|  | sampler_state textureSampler { | ||||||
|  | 	Filter    = Point; | ||||||
|  | 	AddressU  = Clamp; | ||||||
|  | 	AddressV  = Clamp; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct VertDataIn { | ||||||
|  | 	float4 pos : POSITION; | ||||||
|  | 	float2 uv  : TEXCOORD0; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct VertDataOut { | ||||||
|  | 	float4 pos : POSITION; | ||||||
|  | 	float2 uv  : TEXCOORD0; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | VertDataOut VSDefault(VertDataIn v_in) | ||||||
|  | { | ||||||
|  | 	VertDataOut vert_out; | ||||||
|  | 	vert_out.pos = mul(float4(v_in.pos.xyz, 1.0), ViewProj); | ||||||
|  | 	vert_out.uv  = v_in.uv; | ||||||
|  | 	return vert_out; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Bilateral Blur | ||||||
|  | float Bilateral(float x, float sigma) | ||||||
|  | { | ||||||
|  | 	return 0.39894 * exp(-0.5 * (x*x) / (sigma*sigma)) / sigma; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | float Bilateral3(float3 v, float sigma) | ||||||
|  | { | ||||||
|  | 	return 0.39894 * exp(-0.5 * dot(v,v) / (sigma*sigma)) / sigma; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | float4 PSBilateral(VertDataOut v_in) : TARGET | ||||||
|  | { | ||||||
|  | 	float4 source = image.Sample(textureSampler, v_in.uv);	 | ||||||
|  | 	 | ||||||
|  | 	float3 color = float3(0, 0, 0); | ||||||
|  | 	float2 rootuv = v_in.uv - (texel * widthHalf); | ||||||
|  | 	float Z = 0.0; | ||||||
|  | 	float bZ = 1.0 / Bilateral(0.0, bilateralSharpness); | ||||||
|  | 	for (int k = -widthHalf; k <= widthHalf; k++) { | ||||||
|  | 		// Sample Color | ||||||
|  | 		float3 sample = image.Sample(textureSampler, rootuv); | ||||||
|  | 		 | ||||||
|  | 		// Bilateral Stuff | ||||||
|  | 		float bKernel = Bilateral(abs(k), bilateralSmoothing); | ||||||
|  | 		bKernel *= bKernel;		 | ||||||
|  | 		float factor = Bilateral3(sample - source.rgb, bilateralSharpness) * bZ * bKernel; | ||||||
|  | 		Z += factor; | ||||||
|  | 		 | ||||||
|  | 		// Store Color | ||||||
|  | 		color += factor * sample; | ||||||
|  | 		 | ||||||
|  | 		// Advance UV | ||||||
|  | 		rootuv += texel; | ||||||
|  | 	} | ||||||
|  | 	 | ||||||
|  | 	return float4(color.rgb / Z, source.a); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | technique Draw | ||||||
|  | { | ||||||
|  | 	pass | ||||||
|  | 	{ | ||||||
|  | 		vertex_shader = VSDefault(v_in); | ||||||
|  | 		pixel_shader  = PSBilateral(v_in); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | @ -37,4 +37,7 @@ Filter.Blur="Blur" | ||||||
| Filter.Blur.Type="Type" | Filter.Blur.Type="Type" | ||||||
| Filter.Blur.Type.Box="Box" | Filter.Blur.Type.Box="Box" | ||||||
| Filter.Blur.Type.Gaussian="Gaussian" | Filter.Blur.Type.Gaussian="Gaussian" | ||||||
|  | Filter.Blur.Type.Bilateral="Bilateral" | ||||||
| Filter.Blur.Size="Size (Pixel)" | Filter.Blur.Size="Size (Pixel)" | ||||||
|  | Filter.Blur.Bilateral.Smoothing="Smoothing" | ||||||
|  | Filter.Blur.Bilateral.Sharpness="Sharpness" | ||||||
|  |  | ||||||
|  | @ -29,6 +29,8 @@ extern "C" { | ||||||
| #pragma warning (pop) | #pragma warning (pop) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static gs_effect_t *g_boxBlurEffect, *g_gaussianBlurEffect, *g_bilateralBlurEffect; | ||||||
|  | 
 | ||||||
| Filter::Blur::Blur() { | Filter::Blur::Blur() { | ||||||
| 	memset(&sourceInfo, 0, sizeof(obs_source_info)); | 	memset(&sourceInfo, 0, sizeof(obs_source_info)); | ||||||
| 	sourceInfo.id = "obs-stream-effects-filter-blur"; | 	sourceInfo.id = "obs-stream-effects-filter-blur"; | ||||||
|  | @ -48,10 +50,56 @@ Filter::Blur::Blur() { | ||||||
| 	sourceInfo.video_tick = video_tick; | 	sourceInfo.video_tick = video_tick; | ||||||
| 	sourceInfo.video_render = video_render; | 	sourceInfo.video_render = video_render; | ||||||
| 
 | 
 | ||||||
| 	obs_register_source(&sourceInfo); | 	obs_enter_graphics(); | ||||||
|  | 	{ | ||||||
|  | 		char* loadError = nullptr; | ||||||
|  | 		char* file = obs_module_file("filter-blur/box.effect"); | ||||||
|  | 		g_boxBlurEffect = gs_effect_create_from_file(file, &loadError); | ||||||
|  | 		bfree(file); | ||||||
|  | 		if (loadError != nullptr) { | ||||||
|  | 			PLOG_ERROR("<filter-blur> Loading effect failed with error(s): %s", loadError); | ||||||
|  | 			bfree(loadError); | ||||||
|  | 		} else if (!g_boxBlurEffect) { | ||||||
|  | 			PLOG_ERROR("<filter-blur> Loading effect failed with unspecified error."); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	{ | ||||||
|  | 		char* loadError = nullptr; | ||||||
|  | 		char* file = obs_module_file("filter-blur/gaussian.effect"); | ||||||
|  | 		g_gaussianBlurEffect = gs_effect_create_from_file(file, &loadError); | ||||||
|  | 		bfree(file); | ||||||
|  | 		if (loadError != nullptr) { | ||||||
|  | 			PLOG_ERROR("<filter-blur> Loading effect failed with error(s): %s", loadError); | ||||||
|  | 			bfree(loadError); | ||||||
|  | 		} else if (!g_gaussianBlurEffect) { | ||||||
|  | 			PLOG_ERROR("<filter-blur> Loading effect failed with unspecified error."); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	{ | ||||||
|  | 		char* loadError = nullptr; | ||||||
|  | 		char* file = obs_module_file("filter-blur/bilateral.effect"); | ||||||
|  | 		g_bilateralBlurEffect = gs_effect_create_from_file(file, &loadError); | ||||||
|  | 		bfree(file); | ||||||
|  | 		if (loadError != nullptr) { | ||||||
|  | 			PLOG_ERROR("<filter-blur> Loading effect failed with error(s): %s", loadError); | ||||||
|  | 			bfree(loadError); | ||||||
|  | 		} else if (!g_bilateralBlurEffect) { | ||||||
|  | 			PLOG_ERROR("<filter-blur> Loading effect failed with unspecified error."); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	obs_leave_graphics(); | ||||||
|  | 
 | ||||||
|  | 	if (g_boxBlurEffect && g_gaussianBlurEffect && g_bilateralBlurEffect) | ||||||
|  | 		obs_register_source(&sourceInfo); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| Filter::Blur::~Blur() {} | Filter::Blur::~Blur() { | ||||||
|  | 	obs_enter_graphics(); | ||||||
|  | 	gs_effect_destroy(g_bilateralBlurEffect); | ||||||
|  | 	gs_effect_destroy(g_gaussianBlurEffect); | ||||||
|  | 	gs_effect_destroy(g_boxBlurEffect); | ||||||
|  | 	obs_leave_graphics(); | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| const char * Filter::Blur::get_name(void *) { | const char * Filter::Blur::get_name(void *) { | ||||||
| 	return P_TRANSLATE(P_FILTER_BLUR); | 	return P_TRANSLATE(P_FILTER_BLUR); | ||||||
|  | @ -59,7 +107,11 @@ const char * Filter::Blur::get_name(void *) { | ||||||
| 
 | 
 | ||||||
| void Filter::Blur::get_defaults(obs_data_t *data) { | void Filter::Blur::get_defaults(obs_data_t *data) { | ||||||
| 	obs_data_set_default_int(data, P_FILTER_BLUR_TYPE, Filter::Blur::Type::Box); | 	obs_data_set_default_int(data, P_FILTER_BLUR_TYPE, Filter::Blur::Type::Box); | ||||||
| 	obs_data_set_default_int(data, P_FILTER_BLUR_SIZE, 1); | 	obs_data_set_default_int(data, P_FILTER_BLUR_SIZE, 5); | ||||||
|  | 
 | ||||||
|  | 	// Bilateral Only
 | ||||||
|  | 	obs_data_set_default_double(data, P_FILTER_BLUR_BILATERAL_SMOOTHING, 50.0); | ||||||
|  | 	obs_data_set_default_double(data, P_FILTER_BLUR_BILATERAL_SHARPNESS, 90.0); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| obs_properties_t * Filter::Blur::get_properties(void *) { | obs_properties_t * Filter::Blur::get_properties(void *) { | ||||||
|  | @ -69,17 +121,50 @@ obs_properties_t * Filter::Blur::get_properties(void *) { | ||||||
| 	p = obs_properties_add_list(pr, P_FILTER_BLUR_TYPE, P_TRANSLATE(P_FILTER_BLUR_TYPE), | 	p = obs_properties_add_list(pr, P_FILTER_BLUR_TYPE, P_TRANSLATE(P_FILTER_BLUR_TYPE), | ||||||
| 		obs_combo_type::OBS_COMBO_TYPE_LIST, obs_combo_format::OBS_COMBO_FORMAT_INT); | 		obs_combo_type::OBS_COMBO_TYPE_LIST, obs_combo_format::OBS_COMBO_FORMAT_INT); | ||||||
| 	obs_property_set_long_description(p, P_TRANSLATE(P_DESC(P_FILTER_BLUR_TYPE))); | 	obs_property_set_long_description(p, P_TRANSLATE(P_DESC(P_FILTER_BLUR_TYPE))); | ||||||
| 	obs_property_list_add_int(p, P_TRANSLATE(P_FILTER_BLUR_TYPE_BOX), Filter::Blur::Type::Box); | 	obs_property_set_modified_callback(p, modified_properties); | ||||||
|  | 	obs_property_list_add_int(p, P_TRANSLATE(P_FILTER_BLUR_TYPE_BOX), | ||||||
|  | 		Filter::Blur::Type::Box); | ||||||
| 	obs_property_list_add_int(p, P_TRANSLATE(P_FILTER_BLUR_TYPE_GAUSSIAN), | 	obs_property_list_add_int(p, P_TRANSLATE(P_FILTER_BLUR_TYPE_GAUSSIAN), | ||||||
| 		Filter::Blur::Type::Gaussian); | 		Filter::Blur::Type::Gaussian); | ||||||
|  | 	obs_property_list_add_int(p, P_TRANSLATE(P_FILTER_BLUR_TYPE_BILATERAL), | ||||||
|  | 		Filter::Blur::Type::Bilateral); | ||||||
| 
 | 
 | ||||||
| 	p = obs_properties_add_int_slider(pr, P_FILTER_BLUR_SIZE, P_TRANSLATE(P_FILTER_BLUR_SIZE), | 	p = obs_properties_add_int_slider(pr, P_FILTER_BLUR_SIZE, | ||||||
| 		1, 100, 1); | 		P_TRANSLATE(P_FILTER_BLUR_SIZE), 1, 25, 1); | ||||||
| 	obs_property_set_long_description(p, P_TRANSLATE(P_DESC(P_FILTER_BLUR_SIZE))); | 	obs_property_set_long_description(p, P_TRANSLATE(P_DESC(P_FILTER_BLUR_SIZE))); | ||||||
| 
 | 
 | ||||||
|  | 	// Bilateral Only
 | ||||||
|  | 	p = obs_properties_add_float_slider(pr, P_FILTER_BLUR_BILATERAL_SMOOTHING, | ||||||
|  | 		P_TRANSLATE(P_FILTER_BLUR_BILATERAL_SMOOTHING), 0.01, 100.0, 0.01); | ||||||
|  | 	obs_property_set_long_description(p, P_TRANSLATE(P_DESC(P_FILTER_BLUR_BILATERAL_SMOOTHING))); | ||||||
|  | 	p = obs_properties_add_float_slider(pr, P_FILTER_BLUR_BILATERAL_SHARPNESS, | ||||||
|  | 		P_TRANSLATE(P_FILTER_BLUR_BILATERAL_SHARPNESS), 0, 99.99, 0.01); | ||||||
|  | 	obs_property_set_long_description(p, P_TRANSLATE(P_DESC(P_FILTER_BLUR_BILATERAL_SHARPNESS))); | ||||||
|  | 	 | ||||||
| 	return pr; | 	return pr; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | bool Filter::Blur::modified_properties(obs_properties_t *pr, obs_property_t *, obs_data_t *d) { | ||||||
|  | 	bool showBilateral = false; | ||||||
|  | 
 | ||||||
|  | 	switch (obs_data_get_int(d, P_FILTER_BLUR_TYPE)) { | ||||||
|  | 		case Filter::Blur::Type::Box: | ||||||
|  | 			break; | ||||||
|  | 		case Filter::Blur::Type::Gaussian: | ||||||
|  | 			break; | ||||||
|  | 		case Filter::Blur::Type::Bilateral: | ||||||
|  | 			showBilateral = true; | ||||||
|  | 			break; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	obs_property_set_visible(obs_properties_get(pr, P_FILTER_BLUR_BILATERAL_SMOOTHING), | ||||||
|  | 		showBilateral); | ||||||
|  | 	obs_property_set_visible(obs_properties_get(pr, P_FILTER_BLUR_BILATERAL_SHARPNESS), | ||||||
|  | 		showBilateral); | ||||||
|  | 
 | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void * Filter::Blur::create(obs_data_t *data, obs_source_t *source) { | void * Filter::Blur::create(obs_data_t *data, obs_source_t *source) { | ||||||
| 	return new Instance(data, source); | 	return new Instance(data, source); | ||||||
| } | } | ||||||
|  | @ -126,16 +211,7 @@ void Filter::Blur::video_render(void *ptr, gs_effect_t *effect) { | ||||||
| 
 | 
 | ||||||
| Filter::Blur::Instance::Instance(obs_data_t *data, obs_source_t *context) : m_source(context) { | Filter::Blur::Instance::Instance(obs_data_t *data, obs_source_t *context) : m_source(context) { | ||||||
| 	obs_enter_graphics(); | 	obs_enter_graphics(); | ||||||
| 	{ | 	m_effect = g_boxBlurEffect; | ||||||
| 		char* loadError = nullptr; |  | ||||||
| 		char* file = obs_module_file("filter-blur/filter.effect"); |  | ||||||
| 		m_effect = gs_effect_create_from_file(file, &loadError); |  | ||||||
| 		bfree(file); |  | ||||||
| 		if (loadError != nullptr) { |  | ||||||
| 			PLOG_ERROR("<filter-blur> Loading effect failed with error(s): %s", loadError); |  | ||||||
| 			bfree(loadError); |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	m_primaryRT = gs_texrender_create(GS_RGBA, GS_ZS_NONE); | 	m_primaryRT = gs_texrender_create(GS_RGBA, GS_ZS_NONE); | ||||||
| 	m_secondaryRT = gs_texrender_create(GS_RGBA, GS_ZS_NONE); | 	m_secondaryRT = gs_texrender_create(GS_RGBA, GS_ZS_NONE); | ||||||
| 	obs_leave_graphics(); | 	obs_leave_graphics(); | ||||||
|  | @ -145,22 +221,29 @@ Filter::Blur::Instance::Instance(obs_data_t *data, obs_source_t *context) : m_so | ||||||
| 
 | 
 | ||||||
| Filter::Blur::Instance::~Instance() { | Filter::Blur::Instance::~Instance() { | ||||||
| 	obs_enter_graphics(); | 	obs_enter_graphics(); | ||||||
| 	gs_effect_destroy(m_effect); |  | ||||||
| 	gs_texrender_destroy(m_primaryRT); | 	gs_texrender_destroy(m_primaryRT); | ||||||
| 	gs_texrender_destroy(m_secondaryRT); | 	gs_texrender_destroy(m_secondaryRT); | ||||||
| 	obs_leave_graphics(); | 	obs_leave_graphics(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void Filter::Blur::Instance::update(obs_data_t *data) { | void Filter::Blur::Instance::update(obs_data_t *data) { | ||||||
| 	switch (obs_data_get_int(data, P_FILTER_BLUR_TYPE)) { | 	m_type = (Type)obs_data_get_int(data, P_FILTER_BLUR_TYPE); | ||||||
|  | 	switch (m_type) { | ||||||
| 		case Filter::Blur::Type::Box: | 		case Filter::Blur::Type::Box: | ||||||
| 			m_technique = gs_effect_get_technique(m_effect, "Box"); | 			m_effect = g_boxBlurEffect; | ||||||
| 			break; | 			break; | ||||||
| 		case Filter::Blur::Type::Gaussian: | 		case Filter::Blur::Type::Gaussian: | ||||||
| 			m_technique = gs_effect_get_technique(m_effect, "Gaussian"); | 			m_effect = g_gaussianBlurEffect; | ||||||
|  | 			break; | ||||||
|  | 		case Filter::Blur::Type::Bilateral: | ||||||
|  | 			m_effect = g_bilateralBlurEffect; | ||||||
| 			break; | 			break; | ||||||
| 	} | 	} | ||||||
| 	m_filterWidth = (int)obs_data_get_int(data, P_FILTER_BLUR_SIZE); | 	m_size = (uint64_t)obs_data_get_int(data, P_FILTER_BLUR_SIZE); | ||||||
|  | 
 | ||||||
|  | 	// Bilateral Blur
 | ||||||
|  | 	m_bilateralSmoothing = obs_data_get_double(data, P_FILTER_BLUR_BILATERAL_SMOOTHING) / 100.0; | ||||||
|  | 	m_bilateralSharpness = obs_data_get_double(data, P_FILTER_BLUR_BILATERAL_SHARPNESS) / 100.0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| uint32_t Filter::Blur::Instance::get_width() { | uint32_t Filter::Blur::Instance::get_width() { | ||||||
|  | @ -182,6 +265,7 @@ void Filter::Blur::Instance::hide() {} | ||||||
| void Filter::Blur::Instance::video_tick(float) {} | void Filter::Blur::Instance::video_tick(float) {} | ||||||
| 
 | 
 | ||||||
| void Filter::Blur::Instance::video_render(gs_effect_t *effect) { | void Filter::Blur::Instance::video_render(gs_effect_t *effect) { | ||||||
|  | 	bool failed = false; | ||||||
| 	obs_source_t | 	obs_source_t | ||||||
| 		*parent = obs_filter_get_parent(m_source), | 		*parent = obs_filter_get_parent(m_source), | ||||||
| 		*target = obs_filter_get_target(m_source); | 		*target = obs_filter_get_target(m_source); | ||||||
|  | @ -197,8 +281,7 @@ void Filter::Blur::Instance::video_render(gs_effect_t *effect) { | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	gs_effect_t* defaultEffect = obs_get_base_effect(obs_base_effect::OBS_EFFECT_DEFAULT); | 	gs_effect_t* defaultEffect = obs_get_base_effect(obs_base_effect::OBS_EFFECT_DEFAULT); | ||||||
| 	gs_texture_t *sourceTexture = nullptr, | 	gs_texture_t *sourceTexture = nullptr, *horizontalTexture = nullptr; | ||||||
| 		*horizontalTexture = nullptr; |  | ||||||
| 
 | 
 | ||||||
| #pragma region Source To Texture | #pragma region Source To Texture | ||||||
| 	gs_texrender_reset(m_primaryRT); | 	gs_texrender_reset(m_primaryRT); | ||||||
|  | @ -207,7 +290,7 @@ void Filter::Blur::Instance::video_render(gs_effect_t *effect) { | ||||||
| 		obs_source_skip_video_filter(m_source); | 		obs_source_skip_video_filter(m_source); | ||||||
| 		return; | 		return; | ||||||
| 	} else { | 	} else { | ||||||
| 		gs_ortho(0, baseW, 0, baseH, -1, 1); | 		gs_ortho(0, (float)baseW, 0, (float)baseH, -1, 1); | ||||||
| 
 | 
 | ||||||
| 		// Clear to Black
 | 		// Clear to Black
 | ||||||
| 		vec4 black; | 		vec4 black; | ||||||
|  | @ -232,12 +315,16 @@ void Filter::Blur::Instance::video_render(gs_effect_t *effect) { | ||||||
| 			obs_source_process_filter_end(m_source, effect ? effect : defaultEffect, baseW, baseH); | 			obs_source_process_filter_end(m_source, effect ? effect : defaultEffect, baseW, baseH); | ||||||
| 		} else { | 		} else { | ||||||
| 			PLOG_ERROR("<filter-blur> Unable to render source."); | 			PLOG_ERROR("<filter-blur> Unable to render source."); | ||||||
| 			obs_source_skip_video_filter(m_source); | 			failed = true; | ||||||
| 			return; |  | ||||||
| 		} | 		} | ||||||
| 		gs_texrender_end(m_primaryRT); | 		gs_texrender_end(m_primaryRT); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	if (failed) { | ||||||
|  | 		obs_source_skip_video_filter(m_source); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	sourceTexture = gs_texrender_get_texture(m_primaryRT); | 	sourceTexture = gs_texrender_get_texture(m_primaryRT); | ||||||
| 	if (!sourceTexture) { | 	if (!sourceTexture) { | ||||||
| 		PLOG_ERROR("<filter-blur> Failed to get source texture."); | 		PLOG_ERROR("<filter-blur> Failed to get source texture."); | ||||||
|  | @ -253,7 +340,7 @@ void Filter::Blur::Instance::video_render(gs_effect_t *effect) { | ||||||
| 		obs_source_skip_video_filter(m_source); | 		obs_source_skip_video_filter(m_source); | ||||||
| 		return; | 		return; | ||||||
| 	} else { | 	} else { | ||||||
| 		gs_ortho(0, baseW, 0, baseH, -1, 1); | 		gs_ortho(0, (float)baseW, 0, (float)baseH, -1, 1); | ||||||
| 
 | 
 | ||||||
| 		// Clear to Black
 | 		// Clear to Black
 | ||||||
| 		vec4 black; | 		vec4 black; | ||||||
|  | @ -274,44 +361,35 @@ void Filter::Blur::Instance::video_render(gs_effect_t *effect) { | ||||||
| 		gs_enable_color(true, true, true, true); | 		gs_enable_color(true, true, true, true); | ||||||
| 
 | 
 | ||||||
| 		// Prepare Effect
 | 		// Prepare Effect
 | ||||||
| 		gs_eparam_t *param = gs_effect_get_param_by_name(m_effect, "texel"); | 		if (!apply_effect_param(sourceTexture, (float)(1.0 / baseW), 0)) { | ||||||
| 		if (param) { | 			PLOG_ERROR("<filter-blur> Failed to apply effect parameters."); | ||||||
| 			vec2 texel; | 			failed = true; | ||||||
| 			vec2_set(&texel, (float)(1.0 / baseW), 0); |  | ||||||
| 			gs_effect_set_vec2(param, &texel); |  | ||||||
| 		} else { | 		} else { | ||||||
| 			PLOG_ERROR("<filter-blur> Failed to set texel param."); | 			while (gs_effect_loop(m_effect, "Draw")) { | ||||||
| 			obs_source_skip_video_filter(m_source); |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
| 		param = gs_effect_get_param_by_name(m_effect, "size"); |  | ||||||
| 		if (param) { |  | ||||||
| 			gs_effect_set_int(param, m_filterWidth); |  | ||||||
| 		} else { |  | ||||||
| 			PLOG_ERROR("<filter-blur> Failed to set size param."); |  | ||||||
| 			obs_source_skip_video_filter(m_source); |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
| 		param = gs_effect_get_param_by_name(m_effect, "image"); |  | ||||||
| 		if (param) { |  | ||||||
| 			gs_effect_set_texture(param, sourceTexture); |  | ||||||
| 		} else { |  | ||||||
| 			PLOG_ERROR("<filter-blur> Failed to set image param."); |  | ||||||
| 			obs_source_skip_video_filter(m_source); |  | ||||||
| 			return; |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		// Render
 |  | ||||||
| 		if (gs_technique_begin(m_technique)) { |  | ||||||
| 			if (gs_technique_begin_pass_by_name(m_technique, "Horizontal")) { |  | ||||||
| 				gs_draw_sprite(sourceTexture, 0, baseW, baseH); | 				gs_draw_sprite(sourceTexture, 0, baseW, baseH); | ||||||
| 				gs_technique_end_pass(m_technique); |  | ||||||
| 			} | 			} | ||||||
| 			gs_technique_end(m_technique); | 			//// Render
 | ||||||
|  | 			//if (gs_technique_begin(m_technique)) {
 | ||||||
|  | 			//	PLOG_ERROR("<filter-blur> Failed to render effect.");
 | ||||||
|  | 			//	failed = true;
 | ||||||
|  | 			//} else {
 | ||||||
|  | 			//	if (!gs_technique_begin_pass_by_name(m_technique, "Draw")) {
 | ||||||
|  | 			//		PLOG_ERROR("<filter-blur> Failed to render horizontal pass.");
 | ||||||
|  | 			//		failed = true;
 | ||||||
|  | 			//	} else {
 | ||||||
|  | 			//		gs_technique_end_pass(m_technique);
 | ||||||
|  | 			//	}
 | ||||||
|  | 			//	gs_technique_end(m_technique);
 | ||||||
|  | 			//}
 | ||||||
| 		} | 		} | ||||||
| 		gs_texrender_end(m_secondaryRT); | 		gs_texrender_end(m_secondaryRT); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	if (failed) { | ||||||
|  | 		obs_source_skip_video_filter(m_source); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	horizontalTexture = gs_texrender_get_texture(m_secondaryRT); | 	horizontalTexture = gs_texrender_get_texture(m_secondaryRT); | ||||||
| 	if (!horizontalTexture) { | 	if (!horizontalTexture) { | ||||||
| 		PLOG_ERROR("<filter-blur> Failed to get horizontal pass texture."); | 		PLOG_ERROR("<filter-blur> Failed to get horizontal pass texture."); | ||||||
|  | @ -320,53 +398,102 @@ void Filter::Blur::Instance::video_render(gs_effect_t *effect) { | ||||||
| 	} | 	} | ||||||
| #pragma endregion Horizontal Pass | #pragma endregion Horizontal Pass | ||||||
| 
 | 
 | ||||||
| #pragma region Horizontal Pass | #pragma region Vertical Pass | ||||||
| 	// Prepare Effect
 | 	// Prepare Effect
 | ||||||
| 	gs_eparam_t *param = gs_effect_get_param_by_name(m_effect, "texel"); | 	if (!apply_effect_param(horizontalTexture, 0, (float)(1.0 / baseH))) { | ||||||
| 	if (param) { | 		PLOG_ERROR("<filter-blur> Failed to apply effect parameters."); | ||||||
| 		vec2 texel; | 		failed = true; | ||||||
| 		vec2_set(&texel, 0, (float)(1.0 / baseH)); |  | ||||||
| 		gs_effect_set_vec2(param, &texel); |  | ||||||
| 	} else { | 	} else { | ||||||
| 		PLOG_ERROR("<filter-blur> Failed to set texel param."); | 		while (gs_effect_loop(m_effect, "Draw")) { | ||||||
| 		obs_source_skip_video_filter(m_source); |  | ||||||
| 		return; |  | ||||||
| 	} |  | ||||||
| 	param = gs_effect_get_param_by_name(m_effect, "size"); |  | ||||||
| 	if (param) { |  | ||||||
| 		gs_effect_set_int(param, m_filterWidth); |  | ||||||
| 	} else { |  | ||||||
| 		PLOG_ERROR("<filter-blur> Failed to set size param."); |  | ||||||
| 		obs_source_skip_video_filter(m_source); |  | ||||||
| 		return; |  | ||||||
| 	} |  | ||||||
| 	param = gs_effect_get_param_by_name(m_effect, "image"); |  | ||||||
| 	if (param) { |  | ||||||
| 		gs_effect_set_texture(param, horizontalTexture); |  | ||||||
| 	} else { |  | ||||||
| 		PLOG_ERROR("<filter-blur> Failed to set image param."); |  | ||||||
| 		obs_source_skip_video_filter(m_source); |  | ||||||
| 		return; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	// Render
 |  | ||||||
| 	if (gs_technique_begin(m_technique)) { |  | ||||||
| 		if (gs_technique_begin_pass_by_name(m_technique, "Vertical")) { |  | ||||||
| 			gs_draw_sprite(horizontalTexture, 0, baseW, baseH); | 			gs_draw_sprite(horizontalTexture, 0, baseW, baseH); | ||||||
| 			gs_technique_end_pass(m_technique); |  | ||||||
| 		} | 		} | ||||||
| 		gs_technique_end(m_technique); | 		// Render
 | ||||||
|  | 		//if (!gs_technique_begin(m_technique)) {
 | ||||||
|  | 		//	PLOG_ERROR("<filter-blur> Failed to render effect.");
 | ||||||
|  | 		//	failed = true;
 | ||||||
|  | 		//} else {
 | ||||||
|  | 		//	if (!gs_technique_begin_pass_by_name(m_technique, "Draw")) {
 | ||||||
|  | 		//		PLOG_ERROR("<filter-blur> Failed to render vertical pass.");
 | ||||||
|  | 		//		failed = true;
 | ||||||
|  | 		//	} else {
 | ||||||
|  | 		//		gs_draw_sprite(sourceTexture, 0, baseW, baseH);
 | ||||||
|  | 		//		gs_technique_end_pass(m_technique);
 | ||||||
|  | 		//	}
 | ||||||
|  | 		//	gs_technique_end(m_technique);
 | ||||||
|  | 		//}
 | ||||||
| 	} | 	} | ||||||
| #pragma endregion Horizontal Pass |  | ||||||
| 
 | 
 | ||||||
| 
 | 	if (failed) { | ||||||
| 
 | 		obs_source_skip_video_filter(m_source); | ||||||
| 
 | 		return; | ||||||
| 	// Draw final shape
 | 	} | ||||||
| 	//gs_reset_blend_state();
 | #pragma endregion Vertical Pass | ||||||
| 	//gs_enable_depth_test(false);
 | } | ||||||
| 	//while (gs_effect_loop(defaultEffect, "Draw")) {
 | 
 | ||||||
| 	//	gs_effect_set_texture(gs_effect_get_param_by_name(defaultEffect, "image"), horizontalTexture);
 | bool Filter::Blur::Instance::apply_effect_param(gs_texture_t* texture, float uvTexelX, float uvTexelY) { | ||||||
| 	//	gs_draw_sprite(horizontalTexture, 0, 0, 0);
 | 	gs_eparam_t *param; | ||||||
| 	//}
 | 
 | ||||||
|  | 	// UV Stepping
 | ||||||
|  | 	param = gs_effect_get_param_by_name(m_effect, "texel"); | ||||||
|  | 	if (!param) { | ||||||
|  | 		PLOG_ERROR("<filter-blur> Failed to set texel param."); | ||||||
|  | 		return false; | ||||||
|  | 	} else { | ||||||
|  | 		PLOG_DEBUG("<filter-blur> Applying texel parameter."); | ||||||
|  | 		vec2 texel; | ||||||
|  | 		vec2_set(&texel, uvTexelX, uvTexelY); | ||||||
|  | 		gs_effect_set_vec2(param, &texel); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Filter Width
 | ||||||
|  | 	param = gs_effect_get_param_by_name(m_effect, "widthHalf"); | ||||||
|  | 	if (!param) { | ||||||
|  | 		PLOG_ERROR("<filter-blur> Failed to set widthHalf param."); | ||||||
|  | 		return false; | ||||||
|  | 	} else { | ||||||
|  | 		PLOG_DEBUG("<filter-blur> Applying widthHalf parameter."); | ||||||
|  | 		gs_effect_set_int(param, (int)m_size); | ||||||
|  | 	} | ||||||
|  | 	param = gs_effect_get_param_by_name(m_effect, "width"); | ||||||
|  | 	if (!param) { | ||||||
|  | 		PLOG_ERROR("<filter-blur> Failed to set width param."); | ||||||
|  | 		return false; | ||||||
|  | 	} else { | ||||||
|  | 		PLOG_DEBUG("<filter-blur> Applying width parameter."); | ||||||
|  | 		gs_effect_set_int(param, (int)(1 + m_size * 2)); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Texture
 | ||||||
|  | 	param = gs_effect_get_param_by_name(m_effect, "image"); | ||||||
|  | 	if (!param) { | ||||||
|  | 		PLOG_ERROR("<filter-blur> Failed to set image param."); | ||||||
|  | 		return false; | ||||||
|  | 	} else { | ||||||
|  | 		PLOG_DEBUG("<filter-blur> Applying image parameter."); | ||||||
|  | 		gs_effect_set_texture(param, texture); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Bilateral Blur
 | ||||||
|  | 	if (m_type == Type::Bilateral) { | ||||||
|  | 		param = gs_effect_get_param_by_name(m_effect, "bilateralSmoothing"); | ||||||
|  | 		if (!param) { | ||||||
|  | 			PLOG_ERROR("<filter-blur> Failed to set bilateralSmoothing param."); | ||||||
|  | 			return false; | ||||||
|  | 		} else { | ||||||
|  | 			PLOG_DEBUG("<filter-blur> Applying bilateralSmoothing parameter."); | ||||||
|  | 			gs_effect_set_float(param, | ||||||
|  | 				(float)(m_bilateralSmoothing * (1 + m_size * 2))); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		param = gs_effect_get_param_by_name(m_effect, "bilateralSharpness"); | ||||||
|  | 		if (!param) { | ||||||
|  | 			PLOG_ERROR("<filter-blur> Failed to set bilateralSmoothing param."); | ||||||
|  | 			return false; | ||||||
|  | 		} else { | ||||||
|  | 			PLOG_DEBUG("<filter-blur> Applying bilateralSharpness parameter."); | ||||||
|  | 			gs_effect_set_float(param, (float)(1.0 - m_bilateralSharpness)); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return true; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -25,7 +25,10 @@ | ||||||
| #define P_FILTER_BLUR_TYPE				"Filter.Blur.Type" | #define P_FILTER_BLUR_TYPE				"Filter.Blur.Type" | ||||||
| #define P_FILTER_BLUR_TYPE_BOX				"Filter.Blur.Type.Box" | #define P_FILTER_BLUR_TYPE_BOX				"Filter.Blur.Type.Box" | ||||||
| #define P_FILTER_BLUR_TYPE_GAUSSIAN			"Filter.Blur.Type.Gaussian" | #define P_FILTER_BLUR_TYPE_GAUSSIAN			"Filter.Blur.Type.Gaussian" | ||||||
|  | #define P_FILTER_BLUR_TYPE_BILATERAL			"Filter.Blur.Type.Bilateral" | ||||||
| #define P_FILTER_BLUR_SIZE				"Filter.Blur.Size" | #define P_FILTER_BLUR_SIZE				"Filter.Blur.Size" | ||||||
|  | #define P_FILTER_BLUR_BILATERAL_SMOOTHING		"Filter.Blur.Bilateral.Smoothing" | ||||||
|  | #define P_FILTER_BLUR_BILATERAL_SHARPNESS		"Filter.Blur.Bilateral.Sharpness" | ||||||
| 
 | 
 | ||||||
| namespace Filter { | namespace Filter { | ||||||
| 	class Blur { | 	class Blur { | ||||||
|  | @ -36,6 +39,7 @@ namespace Filter { | ||||||
| 		static const char *get_name(void *); | 		static const char *get_name(void *); | ||||||
| 		static void get_defaults(obs_data_t *); | 		static void get_defaults(obs_data_t *); | ||||||
| 		static obs_properties_t *get_properties(void *); | 		static obs_properties_t *get_properties(void *); | ||||||
|  | 		static bool modified_properties(obs_properties_t *, obs_property_t *, obs_data_t *); | ||||||
| 
 | 
 | ||||||
| 		static void *create(obs_data_t *, obs_source_t *); | 		static void *create(obs_data_t *, obs_source_t *); | ||||||
| 		static void destroy(void *); | 		static void destroy(void *); | ||||||
|  | @ -52,6 +56,7 @@ namespace Filter { | ||||||
| 		enum Type : int64_t { | 		enum Type : int64_t { | ||||||
| 			Box, | 			Box, | ||||||
| 			Gaussian, | 			Gaussian, | ||||||
|  | 			Bilateral, | ||||||
| 		}; | 		}; | ||||||
| 
 | 
 | ||||||
| 		private: | 		private: | ||||||
|  | @ -73,17 +78,22 @@ namespace Filter { | ||||||
| 			void video_tick(float); | 			void video_tick(float); | ||||||
| 			void video_render(gs_effect_t*); | 			void video_render(gs_effect_t*); | ||||||
| 
 | 
 | ||||||
|  | 			bool apply_effect_param(gs_texture_t* texture,  | ||||||
|  | 				float uvTexelX, float uvTexelY); | ||||||
|  | 
 | ||||||
| 			private: | 			private: | ||||||
| 			obs_source_t | 			obs_source_t *m_source; | ||||||
| 				*m_source; | 			gs_effect_t *m_effect; | ||||||
| 			gs_effect_t | 			gs_technique_t *m_technique; | ||||||
| 				*m_effect; | 			gs_texrender_t *m_primaryRT, *m_secondaryRT; | ||||||
| 			gs_technique_t | 
 | ||||||
| 				*m_technique; | 			// Blur
 | ||||||
| 			gs_texrender_t  | 			Type m_type; | ||||||
| 				*m_primaryRT, | 			uint64_t m_size; | ||||||
| 				*m_secondaryRT; | 
 | ||||||
| 			int m_filterWidth; | 			// Bilateral
 | ||||||
|  | 			double_t m_bilateralSmoothing; | ||||||
|  | 			double_t m_bilateralSharpness; | ||||||
| 		}; | 		}; | ||||||
| 	}; | 	}; | ||||||
| } | } | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue