forked from enlightenment/efl
Evas filters: Remove complicated displacement flags
The displacement effect is way too complicated. Let's keep it simple and have only one displacement map format (RG + Alpha). Here's what's missing now: - Alpha support, to blend in the input with a variable intensity - Extra padding (see below) Also, the intensity VS. map values are not perfectly defined yet. Problems: How to create a complete mirror effect (map needs to go over boundaries... add extra padding to the buffers).
This commit is contained in:
parent
b54eb40006
commit
e7a67dd13b
|
@ -938,24 +938,10 @@ evas_filter_command_displacement_map_add(Evas_Filter_Context *ctx,
|
|||
if (!cmd) goto end;
|
||||
|
||||
DRAW_FILL_SET(fillmode);
|
||||
cmd->displacement.flags = flags & EVAS_FILTER_DISPLACE_BITMASK;
|
||||
cmd->displacement.flags = flags & EVAS_FILTER_DISPLACE_BITMASK;
|
||||
cmd->displacement.intensity = intensity;
|
||||
cmdid = cmd->id;
|
||||
|
||||
if (!cmd->displacement.flags)
|
||||
{
|
||||
INF("No flags specified for displacement. Assuming XY.");
|
||||
if (map->alpha_only)
|
||||
cmd->displacement.flags = EVAS_FILTER_DISPLACE_XY_ALPHA;
|
||||
else
|
||||
cmd->displacement.flags = EVAS_FILTER_DISPLACE_XY_RG;
|
||||
}
|
||||
if (map->alpha_only && (cmd->displacement.flags & EVAS_FILTER_DISPLACE_RG))
|
||||
{
|
||||
WRN("Incompatible displacement flags specified (RG map but provided Alpha image)");
|
||||
cmd->displacement.flags &= ~EVAS_FILTER_DISPLACE_RG;
|
||||
}
|
||||
|
||||
if (tmp)
|
||||
{
|
||||
if (evas_filter_command_blend_add(ctx, draw_context, disp_out->id,
|
||||
|
|
|
@ -1,15 +1,17 @@
|
|||
#include "evas_filter.h"
|
||||
#include "evas_filter_private.h"
|
||||
|
||||
#warning TODO: Add alpha support
|
||||
|
||||
static void
|
||||
_filter_displace_cpu_alpha_do(int w, int h, int map_w, int map_h, int map_step,
|
||||
int dx, int dy, int intensity,
|
||||
_filter_displace_cpu_alpha_do(int w, int h, int map_w, int map_h, int intensity,
|
||||
DATA8 *dst, DATA8 *src, DATA8 *map_start,
|
||||
Eina_Bool displace_x, Eina_Bool displace_y,
|
||||
Eina_Bool stretch, Eina_Bool smooth)
|
||||
{
|
||||
int x, y, map_x, map_y;
|
||||
const int map_stride = map_w * map_step;
|
||||
const int map_stride = map_w * sizeof(DATA32);
|
||||
const int dx = RED;
|
||||
const int dy = GREEN;
|
||||
DATA8 *map;
|
||||
|
||||
for (y = 0, map_y = 0; y < h; y++, map_y++)
|
||||
|
@ -27,24 +29,19 @@ _filter_displace_cpu_alpha_do(int w, int h, int map_w, int map_h, int map_step,
|
|||
map_x = 0;
|
||||
map = map_start + (map_y * map_stride);
|
||||
}
|
||||
else map += map_step;
|
||||
else map += sizeof(DATA32);
|
||||
|
||||
if (displace_x)
|
||||
{
|
||||
val = ((int) map[dx] - 128) * intensity;
|
||||
offx = val >> 7;
|
||||
offx_dec = val & 0x7f;
|
||||
if ((x + offx) < 0) { offx = -x; out = 1; }
|
||||
if ((x + offx + 1) >= w) { offx = w - x - 2; out = 1; }
|
||||
}
|
||||
if (displace_y)
|
||||
{
|
||||
val = ((int) map[dy] - 128) * intensity;
|
||||
offy = val >> 7;
|
||||
offy_dec = val & 0x7f;
|
||||
if ((y + offy) < 0) { offy = -y; out = 1; }
|
||||
if ((y + offy + 1) >= h) { offy = h - y - 2; out = 1; }
|
||||
}
|
||||
val = ((int) map[dx] - 128) * intensity;
|
||||
offx = val >> 7;
|
||||
offx_dec = val & 0x7f;
|
||||
if ((x + offx) < 0) { offx = -x; out = 1; }
|
||||
if ((x + offx + 1) >= w) { offx = w - x - 2; out = 1; }
|
||||
|
||||
val = ((int) map[dy] - 128) * intensity;
|
||||
offy = val >> 7;
|
||||
offy_dec = val & 0x7f;
|
||||
if ((y + offy) < 0) { offy = -y; out = 1; }
|
||||
if ((y + offy + 1) >= h) { offy = h - y - 2; out = 1; }
|
||||
|
||||
if (out && !stretch)
|
||||
*dst = 0;
|
||||
|
@ -52,7 +49,7 @@ _filter_displace_cpu_alpha_do(int w, int h, int map_w, int map_h, int map_step,
|
|||
{
|
||||
if (!smooth)
|
||||
*dst = src[offx + offy * w];
|
||||
else if (displace_x && displace_y)
|
||||
else
|
||||
{
|
||||
val = src[offx + offy * w] * (128 - offx_dec) * (128 - offy_dec);
|
||||
val += src[offx + 1 + offy * w] * offx_dec * (128 - offy_dec);
|
||||
|
@ -60,34 +57,20 @@ _filter_displace_cpu_alpha_do(int w, int h, int map_w, int map_h, int map_step,
|
|||
val += src[offx + 1 + (offy + 1) * w] * offx_dec * offy_dec;
|
||||
*dst = val >> 14; // <=> *dst = val / (128 * 128)
|
||||
}
|
||||
else
|
||||
{
|
||||
if (displace_x)
|
||||
{
|
||||
val = (int) src[offx + offy * w] * (128 - offx_dec);
|
||||
val += (int) src[offx + 1 + offy * w] * offx_dec;
|
||||
}
|
||||
else
|
||||
{
|
||||
val = (int) src[offx + offy * w] * (128 - offy_dec);
|
||||
val += (int) src[offx + (offy + 1) * w] * offy_dec;
|
||||
}
|
||||
*dst = val >> 7; // <=> *dst = val / 128
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
_filter_displace_cpu_rgba_do(int w, int h, int map_w, int map_h, int map_step,
|
||||
int dx, int dy, int intensity,
|
||||
_filter_displace_cpu_rgba_do(int w, int h, int map_w, int map_h, int intensity,
|
||||
DATA8 *map_start, DATA32 *src, DATA32 *dst,
|
||||
Eina_Bool displace_x, Eina_Bool displace_y,
|
||||
Eina_Bool stretch, Eina_Bool smooth)
|
||||
{
|
||||
int x, y, map_x, map_y;
|
||||
const int map_stride = map_step * map_w;
|
||||
const int map_stride = sizeof(DATA32) * map_w;
|
||||
const int dx = RED;
|
||||
const int dy = GREEN;
|
||||
DATA8 *map;
|
||||
|
||||
for (y = 0, map_y = 0; y < h; y++, map_y++)
|
||||
|
@ -105,30 +88,25 @@ _filter_displace_cpu_rgba_do(int w, int h, int map_w, int map_h, int map_step,
|
|||
map_x = 0;
|
||||
map = map_start + (map_y * map_stride);
|
||||
}
|
||||
else map += map_step;
|
||||
else map += sizeof(DATA32);
|
||||
|
||||
if (displace_x)
|
||||
{
|
||||
val = ((int) map[dx] - 128) * intensity;
|
||||
offx = val >> 7;
|
||||
offx_dec = val & 0x7f;
|
||||
if ((x + offx) < 0) { offx = -x; out = 1; }
|
||||
if ((x + offx + 1) >= w) { offx = w - x - 2; out = 1; }
|
||||
}
|
||||
if (displace_y)
|
||||
{
|
||||
val = ((int) map[dy] - 128) * intensity;
|
||||
offy = val >> 7;
|
||||
offy_dec = val & 0x7f;
|
||||
if ((y + offy) < 0) { offy = -y; out = 1; }
|
||||
if ((y + offy + 1) >= h) { offy = h - y - 2; out = 1; }
|
||||
}
|
||||
val = ((int) map[dx] - 128) * intensity;
|
||||
offx = val >> 7;
|
||||
offx_dec = val & 0x7f;
|
||||
if ((x + offx) < 0) { offx = -x; out = 1; }
|
||||
if ((x + offx + 1) >= w) { offx = w - x - 2; out = 1; }
|
||||
|
||||
val = ((int) map[dy] - 128) * intensity;
|
||||
offy = val >> 7;
|
||||
offy_dec = val & 0x7f;
|
||||
if ((y + offy) < 0) { offy = -y; out = 1; }
|
||||
if ((y + offy + 1) >= h) { offy = h - y - 2; out = 1; }
|
||||
|
||||
if (out && !stretch)
|
||||
*dst = A_VAL(src + offx + offy * w) << (ALPHA * 8);
|
||||
else if (!smooth)
|
||||
*dst = src[offx + offy * w];
|
||||
else if (displace_x && displace_y)
|
||||
else
|
||||
{
|
||||
int R, G, B, A;
|
||||
DATA32 s00, s01, s10, s11; // indexes represent x,y
|
||||
|
@ -160,54 +138,6 @@ _filter_displace_cpu_rgba_do(int w, int h, int map_w, int map_h, int map_step,
|
|||
G >>= 14;
|
||||
B >>= 14;
|
||||
|
||||
*dst = ARGB_JOIN(A, R, G, B);
|
||||
}
|
||||
else if (displace_x)
|
||||
{
|
||||
int R, G, B, A;
|
||||
DATA32 s00, s10;
|
||||
int mul00, mul10;
|
||||
|
||||
mul00 = (128 - offx_dec);
|
||||
mul10 = offx_dec;
|
||||
|
||||
s00 = src[offx + offy * w];
|
||||
s10 = src[offx + 1 + offy * w];
|
||||
|
||||
A = (ALPHA_OF(s00) * mul00) + (ALPHA_OF(s10) * mul10);
|
||||
R = (RED_OF(s00) * mul00) + (RED_OF(s10) * mul10);
|
||||
G = (GREEN_OF(s00) * mul00) + (GREEN_OF(s10) * mul10);
|
||||
B = (BLUE_OF(s00) * mul00) + (BLUE_OF(s10) * mul10);
|
||||
|
||||
A >>= 7;
|
||||
R >>= 7;
|
||||
G >>= 7;
|
||||
B >>= 7;
|
||||
|
||||
*dst = ARGB_JOIN(A, R, G, B);
|
||||
}
|
||||
else
|
||||
{
|
||||
int R, G, B, A;
|
||||
DATA32 s00, s01;
|
||||
int mul00, mul01;
|
||||
|
||||
mul00 = (128 * offy_dec);
|
||||
mul01 = offy_dec;
|
||||
|
||||
s00 = src[offx + offy * w];
|
||||
s01 = src[offx + (offy + 1)* w];
|
||||
|
||||
A = (ALPHA_OF(s00) * mul00) + (ALPHA_OF(s01) * mul01);
|
||||
R = (RED_OF(s00) * mul00) + (RED_OF(s01) * mul01);
|
||||
G = (GREEN_OF(s00) * mul00) + (GREEN_OF(s01) * mul01);
|
||||
B = (BLUE_OF(s00) * mul00) + (BLUE_OF(s01) * mul01);
|
||||
|
||||
A >>= 7;
|
||||
R >>= 7;
|
||||
G >>= 7;
|
||||
B >>= 7;
|
||||
|
||||
*dst = ARGB_JOIN(A, R, G, B);
|
||||
}
|
||||
}
|
||||
|
@ -224,9 +154,9 @@ _filter_displace_cpu_rgba_do(int w, int h, int map_w, int map_h, int map_step,
|
|||
static Eina_Bool
|
||||
_filter_displace_cpu_alpha(Evas_Filter_Command *cmd)
|
||||
{
|
||||
int w, h, map_w, map_h, intensity, map_step, dx = 0, dy = 0;
|
||||
int w, h, map_w, map_h, intensity;
|
||||
DATA8 *dst, *src, *map_start;
|
||||
Eina_Bool displace_x, displace_y, stretch, smooth;
|
||||
Eina_Bool stretch, smooth;
|
||||
|
||||
w = cmd->input->w;
|
||||
h = cmd->input->h;
|
||||
|
@ -243,30 +173,14 @@ _filter_displace_cpu_alpha(Evas_Filter_Command *cmd)
|
|||
EINA_SAFETY_ON_NULL_RETURN_VAL(map_start, EINA_FALSE);
|
||||
EINA_SAFETY_ON_NULL_RETURN_VAL(dst, EINA_FALSE);
|
||||
|
||||
displace_x = cmd->displacement.flags & EVAS_FILTER_DISPLACE_X;
|
||||
displace_y = cmd->displacement.flags & EVAS_FILTER_DISPLACE_Y;
|
||||
stretch = cmd->displacement.flags & EVAS_FILTER_DISPLACE_STRETCH;
|
||||
smooth = cmd->displacement.flags & EVAS_FILTER_DISPLACE_LINEAR;
|
||||
map_w = cmd->mask->w;
|
||||
map_h = cmd->mask->h;
|
||||
intensity = cmd->displacement.intensity;
|
||||
|
||||
if (cmd->mask->alpha_only)
|
||||
map_step = sizeof(DATA8);
|
||||
else
|
||||
{
|
||||
map_step = sizeof(DATA32);
|
||||
if (cmd->displacement.flags & EVAS_FILTER_DISPLACE_RG)
|
||||
{
|
||||
dx = RED;
|
||||
dy = GREEN;
|
||||
}
|
||||
else dx = dy = ALPHA;
|
||||
}
|
||||
|
||||
_filter_displace_cpu_alpha_do(w, h, map_w, map_h, map_step, dx, dy,
|
||||
intensity, dst, src, map_start,
|
||||
displace_x, displace_y, stretch, smooth);
|
||||
_filter_displace_cpu_alpha_do(w, h, map_w, map_h, intensity,
|
||||
dst, src, map_start, stretch, smooth);
|
||||
|
||||
return EINA_TRUE;
|
||||
}
|
||||
|
@ -281,10 +195,10 @@ _filter_displace_cpu_alpha(Evas_Filter_Command *cmd)
|
|||
static Eina_Bool
|
||||
_filter_displace_cpu_rgba(Evas_Filter_Command *cmd)
|
||||
{
|
||||
int w, h, map_w, map_h, intensity, map_step, dx, dy;
|
||||
int w, h, map_w, map_h, intensity;
|
||||
DATA32 *dst, *src;
|
||||
DATA8 *map_start;
|
||||
Eina_Bool displace_x, displace_y, stretch, smooth;
|
||||
Eina_Bool stretch, smooth;
|
||||
|
||||
w = cmd->input->w;
|
||||
h = cmd->input->h;
|
||||
|
@ -301,39 +215,14 @@ _filter_displace_cpu_rgba(Evas_Filter_Command *cmd)
|
|||
EINA_SAFETY_ON_NULL_RETURN_VAL(map_start, EINA_FALSE);
|
||||
EINA_SAFETY_ON_NULL_RETURN_VAL(dst, EINA_FALSE);
|
||||
|
||||
displace_x = cmd->displacement.flags & EVAS_FILTER_DISPLACE_X;
|
||||
displace_y = cmd->displacement.flags & EVAS_FILTER_DISPLACE_Y;
|
||||
stretch = cmd->displacement.flags & EVAS_FILTER_DISPLACE_STRETCH;
|
||||
smooth = cmd->displacement.flags & EVAS_FILTER_DISPLACE_LINEAR;
|
||||
map_w = cmd->mask->w;
|
||||
map_h = cmd->mask->h;
|
||||
intensity = cmd->displacement.intensity;
|
||||
|
||||
if (!displace_x && !displace_y)
|
||||
{
|
||||
WRN("Invalid displacement flags! Defaulting to XY displacement.");
|
||||
displace_x = displace_y = EINA_TRUE;
|
||||
}
|
||||
|
||||
if (cmd->mask->alpha_only)
|
||||
{
|
||||
map_step = sizeof(DATA8);
|
||||
dx = dy = 0;
|
||||
}
|
||||
else
|
||||
{
|
||||
map_step = sizeof(DATA32);
|
||||
if (cmd->displacement.flags & EVAS_FILTER_DISPLACE_RG)
|
||||
{
|
||||
dx = RED;
|
||||
dy = GREEN;
|
||||
}
|
||||
else dx = dy = ALPHA;
|
||||
}
|
||||
|
||||
_filter_displace_cpu_rgba_do(w, h, map_w, map_h, map_step, dx, dy,
|
||||
intensity, map_start, src, dst,
|
||||
displace_x, displace_y, stretch, smooth);
|
||||
_filter_displace_cpu_rgba_do(w, h, map_w, map_h, intensity, map_start,
|
||||
src, dst, stretch, smooth);
|
||||
|
||||
return EINA_TRUE;
|
||||
}
|
||||
|
@ -345,6 +234,7 @@ evas_filter_displace_cpu_func_get(Evas_Filter_Command *cmd)
|
|||
EINA_SAFETY_ON_NULL_RETURN_VAL(cmd->input, NULL);
|
||||
EINA_SAFETY_ON_NULL_RETURN_VAL(cmd->output, NULL);
|
||||
EINA_SAFETY_ON_NULL_RETURN_VAL(cmd->mask, NULL);
|
||||
EINA_SAFETY_ON_FALSE_RETURN_VAL(!cmd->mask->alpha_only, NULL);
|
||||
|
||||
if (cmd->input->alpha_only != cmd->output->alpha_only)
|
||||
{
|
||||
|
@ -353,29 +243,7 @@ evas_filter_displace_cpu_func_get(Evas_Filter_Command *cmd)
|
|||
}
|
||||
|
||||
if (cmd->input->alpha_only)
|
||||
{
|
||||
if ((cmd->displacement.flags & EVAS_FILTER_DISPLACE_RG)
|
||||
&& cmd->mask->alpha_only)
|
||||
{
|
||||
goto invalid_flags;
|
||||
}
|
||||
return _filter_displace_cpu_alpha;
|
||||
}
|
||||
return _filter_displace_cpu_alpha;
|
||||
else
|
||||
{
|
||||
if ((cmd->displacement.flags & EVAS_FILTER_DISPLACE_RG)
|
||||
&& cmd->mask->alpha_only)
|
||||
{
|
||||
goto invalid_flags;
|
||||
}
|
||||
return _filter_displace_cpu_rgba;
|
||||
}
|
||||
|
||||
invalid_flags:
|
||||
ERR("Incompatible flags (0x%02x) and data (input %s, output %s, map %s)",
|
||||
(cmd->displacement.flags & EVAS_FILTER_DISPLACE_BITMASK),
|
||||
cmd->input->alpha_only ? "alpha" : "rgba",
|
||||
cmd->output->alpha_only ? "alpha" : "rgba",
|
||||
cmd->mask->alpha_only ? "alpha" : "rgba");
|
||||
return NULL;
|
||||
return _filter_displace_cpu_rgba;
|
||||
}
|
||||
|
|
|
@ -83,7 +83,7 @@ struct _Evas_Filter_Command
|
|||
|
||||
struct
|
||||
{
|
||||
// mask is an Alpha or RG(BA) texture, see flags. Must be of the same size as the input & output buffers (for now, FIXME)
|
||||
// mask contains the map data
|
||||
Evas_Filter_Displacement_Flags flags;
|
||||
int intensity; // Max displacement in pixels
|
||||
} displacement;
|
||||
|
|
|
@ -53,20 +53,11 @@ enum _Evas_Filter_Channel
|
|||
|
||||
enum _Evas_Filter_Displacement_Flags
|
||||
{
|
||||
EVAS_FILTER_DISPLACE_ALPHA = 0x0, /**< Displace based on Alpha values */
|
||||
EVAS_FILTER_DISPLACE_X = 0x1, /**< X displacement */
|
||||
EVAS_FILTER_DISPLACE_X_ALPHA = 0x1, /**< X displacement */
|
||||
EVAS_FILTER_DISPLACE_Y = 0x2, /**< Y displacement */
|
||||
EVAS_FILTER_DISPLACE_Y_ALPHA = 0x2, /**< Y displacement */
|
||||
EVAS_FILTER_DISPLACE_RG = 0x4, /**< Displace based on R(G) values */
|
||||
EVAS_FILTER_DISPLACE_XY_ALPHA = 0x3, /**< XY displacement based on alpha */
|
||||
EVAS_FILTER_DISPLACE_XY_RG = 0x7, /**< Full 2-D displacement based on RG values */
|
||||
EVAS_FILTER_DISPLACE_NEAREST = 0x0, /**< Interpolate between pixels (linear interpolation) */
|
||||
EVAS_FILTER_DISPLACE_LINEAR = 0x8, /**< Interpolate between pixels (linear interpolation) */
|
||||
EVAS_FILTER_DISPLACE_LINEAR = 0x1, /**< Interpolate between pixels (linear interpolation) */
|
||||
EVAS_FILTER_DISPLACE_BLACK = 0x0, /**< Use black (or transparent) when going out of bounds) */
|
||||
EVAS_FILTER_DISPLACE_STRETCH = 0x8, /**< Stretch border pixels when going out of bounds */
|
||||
//EVAS_FILTER_DISPLACE_WRAP = 0x10, /**< Wrap around input when displacement goes out of bounds */
|
||||
EVAS_FILTER_DISPLACE_BITMASK = 0xf
|
||||
EVAS_FILTER_DISPLACE_STRETCH = 0x2, /**< Stretch border pixels when going out of bounds */
|
||||
EVAS_FILTER_DISPLACE_BITMASK = 0x3
|
||||
};
|
||||
|
||||
enum _Evas_Filter_Bump_Flags
|
||||
|
|
Loading…
Reference in New Issue