1
0
Fork 0
mirror of https://github.com/Orama-Interactive/Pixelorama.git synced 2025-01-18 17:19:50 +00:00

Add alpha erase blend mode (#1117)

* Add alpha erase blend mode

* implemented suggestions
This commit is contained in:
Variable 2024-10-13 03:57:10 +05:00 committed by GitHub
parent 1ed52903b3
commit c21e089e58
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
4 changed files with 31 additions and 20 deletions

View file

@ -12,6 +12,7 @@ signal visibility_changed ## Emits when [member visible] is changed.
enum BlendModes { enum BlendModes {
PASS_THROUGH = -2, ## Only for group layers. Ignores group blending, like it doesn't exist. PASS_THROUGH = -2, ## Only for group layers. Ignores group blending, like it doesn't exist.
NORMAL = 0, ## The blend layer colors are simply placed on top of the base colors. NORMAL = 0, ## The blend layer colors are simply placed on top of the base colors.
ERASE, ## Subtracts the numerical value of alpha from the base alpha.
DARKEN, ## Keeps the darker colors between the blend and the base layers. DARKEN, ## Keeps the darker colors between the blend and the base layers.
MULTIPLY, ## Multiplies the numerical values of the two colors, giving a darker result. MULTIPLY, ## Multiplies the numerical values of the two colors, giving a darker result.
COLOR_BURN, ## Darkens by increasing the contrast between the blend and base colors. COLOR_BURN, ## Darkens by increasing the contrast between the blend and base colors.

View file

@ -61,68 +61,74 @@ vec4 blend(int blend_type, vec4 current_color, vec4 prev_color, float opacity) {
return prev_color; return prev_color;
} }
vec4 result; vec4 result;
bool should_blend_alpha = true;
switch(blend_type) { switch(blend_type) {
case 1: // Darken case 1: // Erase
result = prev_color;
result.a -= current_color.a; // clamping will be done at the end so not doing it here.
should_blend_alpha = false;
break;
case 2: // Darken
result.rgb = min(prev_color.rgb, current_color.rgb); result.rgb = min(prev_color.rgb, current_color.rgb);
break; break;
case 2: // Multiply case 3: // Multiply
result.rgb = prev_color.rgb * current_color.rgb; result.rgb = prev_color.rgb * current_color.rgb;
break; break;
case 3: // Color burn case 4: // Color burn
result.rgb = 1.0 - (1.0 - prev_color.rgb) / current_color.rgb; result.rgb = 1.0 - (1.0 - prev_color.rgb) / current_color.rgb;
break; break;
case 4: // Linear burn case 5: // Linear burn
result.rgb = prev_color.rgb + current_color.rgb - 1.0; result.rgb = prev_color.rgb + current_color.rgb - 1.0;
break; break;
case 5: // Lighten case 6: // Lighten
result.rgb = max(prev_color.rgb, current_color.rgb); result.rgb = max(prev_color.rgb, current_color.rgb);
break; break;
case 6: // Screen case 7: // Screen
result.rgb = 1.0 - (1.0 - prev_color.rgb) * (1.0 - current_color.rgb); result.rgb = 1.0 - (1.0 - prev_color.rgb) * (1.0 - current_color.rgb);
break; break;
case 7: // Color dodge case 8: // Color dodge
result.rgb = prev_color.rgb / (1.0 - current_color.rgb); result.rgb = prev_color.rgb / (1.0 - current_color.rgb);
break; break;
case 8: // Add (linear dodge) case 9: // Add (linear dodge)
result.rgb = prev_color.rgb + current_color.rgb; result.rgb = prev_color.rgb + current_color.rgb;
break; break;
case 9: // Overlay case 10: // Overlay
result.rgb = mix(2.0 * prev_color.rgb * current_color.rgb, 1.0 - 2.0 * (1.0 - current_color.rgb) * (1.0 - prev_color.rgb), round(prev_color.rgb)); result.rgb = mix(2.0 * prev_color.rgb * current_color.rgb, 1.0 - 2.0 * (1.0 - current_color.rgb) * (1.0 - prev_color.rgb), round(prev_color.rgb));
break; break;
case 10: // Soft light case 11: // Soft light
result.rgb = mix(2.0 * prev_color.rgb * current_color.rgb + prev_color.rgb * prev_color.rgb * (1.0 - 2.0 * current_color.rgb), sqrt(prev_color.rgb) * (2.0 * current_color.rgb - 1.0) + (2.0 * prev_color.rgb) * (1.0 - current_color.rgb), round(prev_color.rgb)); result.rgb = mix(2.0 * prev_color.rgb * current_color.rgb + prev_color.rgb * prev_color.rgb * (1.0 - 2.0 * current_color.rgb), sqrt(prev_color.rgb) * (2.0 * current_color.rgb - 1.0) + (2.0 * prev_color.rgb) * (1.0 - current_color.rgb), round(prev_color.rgb));
break; break;
case 11: // Hard light case 12: // Hard light
result.rgb = mix(2.0 * prev_color.rgb * current_color.rgb, 1.0 - 2.0 * (1.0 - current_color.rgb) * (1.0 - prev_color.rgb), round(current_color.rgb)); result.rgb = mix(2.0 * prev_color.rgb * current_color.rgb, 1.0 - 2.0 * (1.0 - current_color.rgb) * (1.0 - prev_color.rgb), round(current_color.rgb));
break; break;
case 12: // Difference case 13: // Difference
result.rgb = abs(prev_color.rgb - current_color.rgb); result.rgb = abs(prev_color.rgb - current_color.rgb);
break; break;
case 13: // Exclusion case 14: // Exclusion
result.rgb = prev_color.rgb + current_color.rgb - 2.0 * prev_color.rgb * current_color.rgb; result.rgb = prev_color.rgb + current_color.rgb - 2.0 * prev_color.rgb * current_color.rgb;
break; break;
case 14: // Subtract case 15: // Subtract
result.rgb = prev_color.rgb - current_color.rgb; result.rgb = prev_color.rgb - current_color.rgb;
break; break;
case 15: // Divide case 16: // Divide
result.rgb = prev_color.rgb / current_color.rgb; result.rgb = prev_color.rgb / current_color.rgb;
break; break;
case 16: // Hue case 17: // Hue
vec3 current_hsl = rgb_to_hsl(current_color.rgb); vec3 current_hsl = rgb_to_hsl(current_color.rgb);
vec3 prev_hsl = rgb_to_hsl(prev_color.rgb); vec3 prev_hsl = rgb_to_hsl(prev_color.rgb);
result.rgb = hsl_to_rgb(vec3(current_hsl.r, prev_hsl.g, prev_hsl.b)); result.rgb = hsl_to_rgb(vec3(current_hsl.r, prev_hsl.g, prev_hsl.b));
break; break;
case 17: // Saturation case 18: // Saturation
vec3 current_hsl = rgb_to_hsl(current_color.rgb); vec3 current_hsl = rgb_to_hsl(current_color.rgb);
vec3 prev_hsl = rgb_to_hsl(prev_color.rgb); vec3 prev_hsl = rgb_to_hsl(prev_color.rgb);
result.rgb = hsl_to_rgb(vec3(prev_hsl.r, current_hsl.g, prev_hsl.b)); result.rgb = hsl_to_rgb(vec3(prev_hsl.r, current_hsl.g, prev_hsl.b));
break; break;
case 18: // Color case 19: // Color
vec3 current_hsl = rgb_to_hsl(current_color.rgb); vec3 current_hsl = rgb_to_hsl(current_color.rgb);
vec3 prev_hsl = rgb_to_hsl(prev_color.rgb); vec3 prev_hsl = rgb_to_hsl(prev_color.rgb);
result.rgb = hsl_to_rgb(vec3(current_hsl.r, current_hsl.g, prev_hsl.b)); result.rgb = hsl_to_rgb(vec3(current_hsl.r, current_hsl.g, prev_hsl.b));
break; break;
case 19: // Luminosity case 20: // Luminosity
vec3 current_hsl = rgb_to_hsl(current_color.rgb); vec3 current_hsl = rgb_to_hsl(current_color.rgb);
vec3 prev_hsl = rgb_to_hsl(prev_color.rgb); vec3 prev_hsl = rgb_to_hsl(prev_color.rgb);
result.rgb = hsl_to_rgb(vec3(prev_hsl.r, prev_hsl.g, current_hsl.b)); result.rgb = hsl_to_rgb(vec3(prev_hsl.r, prev_hsl.g, current_hsl.b));
@ -132,7 +138,9 @@ vec4 blend(int blend_type, vec4 current_color, vec4 prev_color, float opacity) {
break; break;
} }
result.rgb = mix(prev_color.rgb, result.rgb, current_color.a); result.rgb = mix(prev_color.rgb, result.rgb, current_color.a);
if (should_blend_alpha){
result.a = prev_color.a * (1.0 - current_color.a) + current_color.a; result.a = prev_color.a * (1.0 - current_color.a) + current_color.a;
}
result = clamp(result, 0.0, 1.0); result = clamp(result, 0.0, 1.0);
return mix(current_color, result, prev_color.a); return mix(current_color, result, prev_color.a);
} }

View file

@ -239,6 +239,7 @@ func _fill_blend_modes_option_button() -> void:
# Special blend mode that appears only when group layers are selected # Special blend mode that appears only when group layers are selected
blend_modes_button.add_item("Pass through", BaseLayer.BlendModes.PASS_THROUGH) blend_modes_button.add_item("Pass through", BaseLayer.BlendModes.PASS_THROUGH)
blend_modes_button.add_item("Normal", BaseLayer.BlendModes.NORMAL) blend_modes_button.add_item("Normal", BaseLayer.BlendModes.NORMAL)
blend_modes_button.add_item("Erase", BaseLayer.BlendModes.ERASE)
blend_modes_button.add_separator("Darken") blend_modes_button.add_separator("Darken")
blend_modes_button.add_item("Darken", BaseLayer.BlendModes.DARKEN) blend_modes_button.add_item("Darken", BaseLayer.BlendModes.DARKEN)
blend_modes_button.add_item("Multiply", BaseLayer.BlendModes.MULTIPLY) blend_modes_button.add_item("Multiply", BaseLayer.BlendModes.MULTIPLY)

View file

@ -39,6 +39,7 @@ func _fill_blend_modes_option_button() -> void:
# Special blend mode that appears only when group layers are selected # Special blend mode that appears only when group layers are selected
blend_modes_button.add_item("Pass through", BaseLayer.BlendModes.PASS_THROUGH) blend_modes_button.add_item("Pass through", BaseLayer.BlendModes.PASS_THROUGH)
blend_modes_button.add_item("Normal", BaseLayer.BlendModes.NORMAL) blend_modes_button.add_item("Normal", BaseLayer.BlendModes.NORMAL)
blend_modes_button.add_item("Erase", BaseLayer.BlendModes.ERASE)
blend_modes_button.add_item("Darken", BaseLayer.BlendModes.DARKEN) blend_modes_button.add_item("Darken", BaseLayer.BlendModes.DARKEN)
blend_modes_button.add_item("Multiply", BaseLayer.BlendModes.MULTIPLY) blend_modes_button.add_item("Multiply", BaseLayer.BlendModes.MULTIPLY)
blend_modes_button.add_item("Color burn", BaseLayer.BlendModes.COLOR_BURN) blend_modes_button.add_item("Color burn", BaseLayer.BlendModes.COLOR_BURN)