|
|
|
#include "ephoto.h"
|
|
|
|
|
|
|
|
typedef enum _Ephoto_Image_Filter Ephoto_Image_Filter;
|
|
|
|
typedef struct _Ephoto_Filter Ephoto_Filter;
|
|
|
|
|
|
|
|
enum _Ephoto_Image_Filter
|
|
|
|
{
|
|
|
|
EPHOTO_IMAGE_FILTER_BLUR,
|
|
|
|
EPHOTO_IMAGE_FILTER_DITHER,
|
|
|
|
EPHOTO_IMAGE_FILTER_EQUALIZE,
|
|
|
|
EPHOTO_IMAGE_FILTER_GRAYSCALE,
|
|
|
|
EPHOTO_IMAGE_FILTER_INVERT,
|
|
|
|
EPHOTO_IMAGE_FILTER_PAINTING,
|
|
|
|
EPHOTO_IMAGE_FILTER_POSTERIZE,
|
|
|
|
EPHOTO_IMAGE_FILTER_SEPIA,
|
|
|
|
EPHOTO_IMAGE_FILTER_SHARPEN,
|
|
|
|
EPHOTO_IMAGE_FILTER_SKETCH,
|
|
|
|
EPHOTO_IMAGE_FILTER_SOBEL
|
|
|
|
};
|
|
|
|
|
|
|
|
struct _Ephoto_Filter
|
|
|
|
{
|
|
|
|
Ephoto_Image_Filter filter;
|
|
|
|
Evas_Object *main;
|
|
|
|
Evas_Object *image;
|
|
|
|
Evas_Object *popup;
|
|
|
|
Ecore_Idler *idler;
|
|
|
|
Eina_List *queue;
|
|
|
|
Evas_Coord w;
|
|
|
|
Evas_Coord h;
|
|
|
|
int pos;
|
|
|
|
int blur_posx;
|
|
|
|
int blur_posy;
|
|
|
|
int rad;
|
|
|
|
int qpos;
|
|
|
|
int qcount;
|
|
|
|
double drad;
|
|
|
|
unsigned int *hist;
|
|
|
|
unsigned int *cdf;
|
|
|
|
unsigned int *im_data;
|
|
|
|
unsigned int *im_data_new;
|
|
|
|
unsigned int *im_data_orig;
|
|
|
|
unsigned int *im_data_two;
|
|
|
|
};
|
|
|
|
|
|
|
|
static Eina_Bool _blur(void *data);
|
|
|
|
static Eina_Bool _sharpen(void *data);
|
|
|
|
static Eina_Bool _dither(void *data);
|
|
|
|
static Eina_Bool _grayscale(void *data);
|
|
|
|
static Eina_Bool _sepia(void *data);
|
|
|
|
static Eina_Bool _negative(void *data);
|
|
|
|
static Eina_Bool _posterize(void *data);
|
|
|
|
static Eina_Bool _dodge(void *data);
|
|
|
|
static Eina_Bool _sobel(void *data);
|
|
|
|
static Eina_Bool _histogram_eq(void *data);
|
|
|
|
|
|
|
|
static Ephoto_Filter *
|
|
|
|
_initialize_filter(Ephoto_Image_Filter filter,
|
|
|
|
Evas_Object *main, Evas_Object *image)
|
|
|
|
{
|
|
|
|
Ephoto_Filter *ef = calloc(1, sizeof(Ephoto_Filter));
|
|
|
|
Evas_Coord w, h;
|
|
|
|
unsigned int *im_data;
|
|
|
|
|
|
|
|
im_data =
|
|
|
|
evas_object_image_data_get(elm_image_object_get(image), EINA_FALSE);
|
|
|
|
evas_object_image_size_get(elm_image_object_get(image), &w, &h);
|
|
|
|
|
|
|
|
ef->filter = filter;
|
|
|
|
ef->main = main;
|
|
|
|
ef->image = image;
|
|
|
|
ef->im_data = malloc(sizeof(unsigned int) * w * h);
|
|
|
|
ef->im_data = memcpy(ef->im_data, im_data, sizeof(unsigned int) * w * h);
|
|
|
|
ef->im_data_new = malloc(sizeof(unsigned int) * w * h);
|
|
|
|
ef->im_data_two = NULL;
|
|
|
|
ef->im_data_orig = NULL;
|
|
|
|
ef->rad = 0;
|
|
|
|
ef->drad = 0.0;
|
|
|
|
ef->pos = 0;
|
|
|
|
ef->blur_posx = 0;
|
|
|
|
ef->blur_posy = 0;
|
|
|
|
ef->w = w;
|
|
|
|
ef->h = h;
|
|
|
|
ef->qpos = 0;
|
|
|
|
ef->qcount = 0;
|
|
|
|
ef->queue = NULL;
|
|
|
|
ef->hist = NULL;
|
|
|
|
ef->cdf = NULL;
|
|
|
|
|
|
|
|
return ef;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
_normalize_color(int color)
|
|
|
|
{
|
|
|
|
if (color < 0)
|
|
|
|
return 0;
|
|
|
|
else if (color > 255)
|
|
|
|
return 255;
|
|
|
|
else
|
|
|
|
return color;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
_mul_color_alpha(int color, int alpha)
|
|
|
|
{
|
|
|
|
if (alpha > 0 && alpha < 255)
|
|
|
|
return color * (255 / alpha);
|
|
|
|
else
|
|
|
|
return color;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
_demul_color_alpha(int color, int alpha)
|
|
|
|
{
|
|
|
|
if (alpha > 0 && alpha < 255)
|
|
|
|
return (color * alpha) / 255;
|
|
|
|
else
|
|
|
|
return color;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
_create_hist(Ephoto_Filter *ef)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < 256; i++)
|
|
|
|
ef->hist[i] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Evas_Object *
|
|
|
|
_processing(Evas_Object *main)
|
|
|
|
{
|
|
|
|
Evas_Object *popup, *box, *label, *pb;
|
|
|
|
|
|
|
|
evas_object_freeze_events_set(main, EINA_TRUE);
|
|
|
|
|
|
|
|
popup = elm_popup_add(main);
|
|
|
|
elm_object_part_text_set(popup, "title,text", _("Applying Filter"));
|
|
|
|
elm_popup_orient_set(popup, ELM_POPUP_ORIENT_CENTER);
|
|
|
|
|
|
|
|
box = elm_box_add(popup);
|
|
|
|
elm_box_horizontal_set(box, EINA_FALSE);
|
|
|
|
evas_object_size_hint_weight_set(box, EVAS_HINT_FILL, EVAS_HINT_FILL);
|
|
|
|
evas_object_size_hint_align_set(box, EVAS_HINT_FILL, EVAS_HINT_FILL);
|
|
|
|
evas_object_show(box);
|
|
|
|
|
|
|
|
label = elm_label_add(box);
|
|
|
|
elm_object_text_set(label,
|
|
|
|
_("Please wait while this filter is applied to your image."));
|
|
|
|
evas_object_size_hint_weight_set(label, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
|
|
|
|
evas_object_size_hint_align_set(label, EVAS_HINT_FILL, EVAS_HINT_FILL);
|
|
|
|
elm_box_pack_end(box, label);
|
|
|
|
evas_object_show(label);
|
|
|
|
|
|
|
|
pb = elm_progressbar_add(box);
|
|
|
|
evas_object_size_hint_weight_set(pb, EVAS_HINT_EXPAND, 0.0);
|
|
|
|
evas_object_size_hint_align_set(pb, EVAS_HINT_FILL, 0.5);
|
|
|
|
elm_object_style_set(pb, "wheel");
|
|
|
|
elm_progressbar_pulse_set(pb, EINA_TRUE);
|
|
|
|
elm_box_pack_end(box, pb);
|
|
|
|
evas_object_show(pb);
|
|
|
|
elm_progressbar_pulse(pb, EINA_TRUE);
|
|
|
|
|
|
|
|
elm_object_part_content_set(popup, "default", box);
|
|
|
|
evas_object_show(popup);
|
|
|
|
return popup;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
_idler_finishing_cb(Ephoto_Filter *ef, Eina_Bool im_data_two)
|
|
|
|
{
|
|
|
|
if (ef->qcount == 0)
|
|
|
|
{
|
|
|
|
ephoto_single_browser_image_data_done(ef->main, ef->im_data_new,
|
|
|
|
ef->w, ef->h);
|
|
|
|
ecore_idler_del(ef->idler);
|
|
|
|
if (ef->popup)
|
|
|
|
{
|
|
|
|
evas_object_del(ef->popup);
|
|
|
|
evas_object_freeze_events_set(ef->main, EINA_FALSE);
|
|
|
|
}
|
|
|
|
if (ef->im_data)
|
|
|
|
free(ef->im_data);
|
|
|
|
if (ef->im_data_two)
|
|
|
|
free(ef->im_data_two);
|
|
|
|
if (ef->im_data_orig)
|
|
|
|
free(ef->im_data_orig);
|
|
|
|
free(ef);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (im_data_two)
|
|
|
|
{
|
|
|
|
if (ef->im_data_two)
|
|
|
|
{
|
|
|
|
free(ef->im_data_two);
|
|
|
|
ef->im_data_two = NULL;
|
|
|
|
}
|
|
|
|
ef->im_data_two = malloc(sizeof(unsigned int) * ef->w * ef->h);
|
|
|
|
ef->im_data_two = memcpy(ef->im_data_two, ef->im_data,
|
|
|
|
sizeof(unsigned int) * ef->w * ef->h);
|
|
|
|
}
|
|
|
|
free(ef->im_data);
|
|
|
|
ef->im_data = NULL;
|
|
|
|
ef->im_data = malloc(sizeof(unsigned int) * ef->w * ef->h);
|
|
|
|
ef->im_data = memcpy(ef->im_data, ef->im_data_new,
|
|
|
|
sizeof(unsigned int) * ef->w * ef->h);
|
|
|
|
if (ef->hist)
|
|
|
|
free(ef->hist);
|
|
|
|
ef->hist = NULL;
|
|
|
|
if (ef->cdf)
|
|
|
|
free(ef->cdf);
|
|
|
|
ef->cdf = NULL;
|
|
|
|
ef->pos = 0;
|
|
|
|
ef->qpos++;
|
|
|
|
ecore_idler_del(ef->idler);
|
|
|
|
if (ef->qpos-1 < ef->qcount)
|
|
|
|
{
|
|
|
|
free(ef->im_data_new);
|
|
|
|
ef->im_data_new = NULL;
|
|
|
|
ef->im_data_new = malloc(sizeof(unsigned int) * ef->w * ef->h);
|
|
|
|
if (ef->idler)
|
|
|
|
{
|
|
|
|
ecore_idler_del(ef->idler);
|
|
|
|
ef->idler = NULL;
|
|
|
|
}
|
|
|
|
ef->idler = ecore_idler_add(eina_list_nth(ef->queue, ef->qpos-1), ef);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ephoto_single_browser_image_data_done(ef->main, ef->im_data_new,
|
|
|
|
ef->w, ef->h);
|
|
|
|
ecore_idler_del(ef->idler);
|
|
|
|
if (ef->popup)
|
|
|
|
{
|
|
|
|
evas_object_del(ef->popup);
|
|
|
|
evas_object_freeze_events_set(ef->main, EINA_FALSE);
|
|
|
|
}
|
|
|
|
if (ef->im_data)
|
|
|
|
free(ef->im_data);
|
|
|
|
if (ef->im_data_two)
|
|
|
|
free(ef->im_data_two);
|
|
|
|
if (ef->im_data_orig)
|
|
|
|
free(ef->im_data_orig);
|
|
|
|
free(ef);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
_blur_vertical(Ephoto_Filter *ef, double rad)
|
|
|
|
{
|
|
|
|
Evas_Coord y, w, h;
|
|
|
|
int i, at, rt, gt, bt, passes = 0;
|
|
|
|
unsigned int *as, *rs, *gs, *bs;
|
|
|
|
double iarr;
|
|
|
|
|
|
|
|
w = ef->w;
|
|
|
|
h = ef->h;
|
|
|
|
|
|
|
|
as = malloc(sizeof(unsigned int) * w * h);
|
|
|
|
rs = malloc(sizeof(unsigned int) * w * h);
|
|
|
|
gs = malloc(sizeof(unsigned int) * w * h);
|
|
|
|
bs = malloc(sizeof(unsigned int) * w * h);
|
|
|
|
|
|
|
|
for (i = 0; i < w * h; i++)
|
|
|
|
{
|
|
|
|
bs[i] = ef->im_data[i] & 0xff;
|
|
|
|
gs[i] = (ef->im_data[i] >> 8) & 0xff;
|
|
|
|
rs[i] = (ef->im_data[i] >> 16) & 0xff;
|
|
|
|
as[i] = (ef->im_data[i] >> 24) & 0xff;
|
|
|
|
}
|
|
|
|
|
|
|
|
iarr = (double) 1 / (rad + rad + 1);
|
|
|
|
for (y = ef->blur_posy; y < h; y++)
|
|
|
|
{
|
|
|
|
int t, l, rr;
|
|
|
|
int fr, fg, fb, fa;
|
|
|
|
int lvr, lvg, lvb, lva;
|
|
|
|
int valr, valg, valb, vala;
|
|
|
|
|
|
|
|
t = y * w;
|
|
|
|
l = t;
|
|
|
|
rr = t + rad;
|
|
|
|
|
|
|
|
fb = bs[t];
|
|
|
|
fg = gs[t];
|
|
|
|
fr = rs[t];
|
|
|
|
fa = as[t];
|
|
|
|
|
|
|
|
lvb = bs[t + w - 1];
|
|
|
|
lvg = gs[t + w - 1];
|
|
|
|
lvr = rs[t + w - 1];
|
|
|
|
lva = as[t + w - 1];
|
|
|
|
|
|
|
|
valb = (rad + 1) * fb;
|
|
|
|
valg = (rad + 1) * fg;
|
|
|
|
valr = (rad + 1) * fr;
|
|
|
|
vala = (rad + 1) * fa;
|
|
|
|
|
|
|
|
for (i = 0; i < rad; i++)
|
|
|
|
{
|
|
|
|
valb += bs[t + i];
|
|
|
|
valg += gs[t + i];
|
|
|
|
valr += rs[t + i];
|
|
|
|
vala += as[t + i];
|
|
|
|
}
|
|
|
|
for (i = 0; i <= rad; i++)
|
|
|
|
{
|
|
|
|
int r = rr++;
|
|
|
|
|
|
|
|
valb += bs[r] - fb;
|
|
|
|
valg += gs[r] - fg;
|
|
|
|
valr += rs[r] - fr;
|
|
|
|
vala += as[r] - fa;
|
|
|
|
|
|
|
|
bt = (int) round(valb * iarr);
|
|
|
|
gt = (int) round(valg * iarr);
|
|
|
|
rt = (int) round(valr * iarr);
|
|
|
|
at = (int) round(vala * iarr);
|
|
|
|
|
|
|
|
bt = _normalize_color(bt);
|
|
|
|
gt = _normalize_color(gt);
|
|
|
|
rt = _normalize_color(rt);
|
|
|
|
at = _normalize_color(at);
|
|
|
|
ef->im_data_new[t++] = (at << 24) | (rt << 16)
|
|
|
|
| (gt << 8) | bt;
|
|
|
|
}
|
|
|
|
for (i = rad + 1; i < w - rad; i++)
|
|
|
|
{
|
|
|
|
int r = rr++;
|
|
|
|
int ll = l++;
|
|
|
|
|
|
|
|
valb += bs[r] - bs[ll];
|
|
|
|
valg += gs[r] - gs[ll];
|
|
|
|
valr += rs[r] - rs[ll];
|
|
|
|
vala += as[r] - as[ll];
|
|
|
|
|
|
|
|
bt = (int) round(valb * iarr);
|
|
|
|
gt = (int) round(valg * iarr);
|
|
|
|
rt = (int) round(valr * iarr);
|
|
|
|
at = (int) round(vala * iarr);
|
|
|
|
bt = _normalize_color(bt);
|
|
|
|
gt = _normalize_color(gt);
|
|
|
|
rt = _normalize_color(rt);
|
|
|
|
at = _normalize_color(at);
|
|
|
|
ef->im_data_new[t++] = (at << 24) | (rt << 16)
|
|
|
|
| (gt << 8) | bt;
|
|
|
|
}
|
|
|
|
for (i = w - rad; i < w; i++)
|
|
|
|
{
|
|
|
|
int ll = l++;
|
|
|
|
|
|
|
|
valb += lvb - bs[ll];
|
|
|
|
valg += lvg - gs[ll];
|
|
|
|
valr += lvr - rs[ll];
|
|
|
|
vala += lva - as[ll];
|
|
|
|
|
|
|
|
bt = (int) round(valb * iarr);
|
|
|
|
gt = (int) round(valg * iarr);
|
|
|
|
rt = (int) round(valr * iarr);
|
|
|
|
at = (int) round(vala * iarr);
|
|
|
|
bt = _normalize_color(bt);
|
|
|
|
gt = _normalize_color(gt);
|
|
|
|
rt = _normalize_color(rt);
|
|
|
|
at = _normalize_color(at);
|
|
|
|
ef->im_data_new[t++] = (at << 24) | (rt << 16)
|
|
|
|
| (gt << 8) | bt;
|
|
|
|
}
|
|
|
|
ef->blur_posy = y;
|
|
|
|
passes++;
|
|
|
|
if (passes == 500)
|
|
|
|
{
|
|
|
|
ef->blur_posy = y++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
free(bs);
|
|
|
|
free(gs);
|
|
|
|
free(rs);
|
|
|
|
free(as);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
_blur_horizontal(Ephoto_Filter *ef, double rad)
|
|
|
|
{
|
|
|
|
Evas_Coord x, w, h;
|
|
|
|
int i, at, rt, gt, bt, passes = 0;
|
|
|
|
unsigned int *as, *rs, *gs, *bs;
|
|
|
|
double iarr;
|
|
|
|
|
|
|
|
w = ef->w;
|
|
|
|
h = ef->h;
|
|
|
|
|
|
|
|
as = malloc(sizeof(unsigned int) * w * h);
|
|
|
|
rs = malloc(sizeof(unsigned int) * w * h);
|
|
|
|
gs = malloc(sizeof(unsigned int) * w * h);
|
|
|
|
bs = malloc(sizeof(unsigned int) * w * h);
|
|
|
|
|
|
|
|
for (i = 0; i < w * h; i++)
|
|
|
|
{
|
|
|
|
bs[i] = ef->im_data_new[i] & 0xff;
|
|
|
|
gs[i] = (ef->im_data_new[i] >> 8) & 0xff;
|
|
|
|
rs[i] = (ef->im_data_new[i] >> 16) & 0xff;
|
|
|
|
as[i] = (ef->im_data_new[i] >> 24) & 0xff;
|
|
|
|
}
|
|
|
|
|
|
|
|
iarr = (double)1 / (rad + rad + 1);
|
|
|
|
for (x = ef->blur_posx; x < w; x++)
|
|
|
|
{
|
|
|
|
int t, l, rr;
|
|
|
|
int fr, fg, fb, fa;
|
|
|
|
int lvr, lvg, lvb, lva;
|
|
|
|
int valr, valg, valb, vala;
|
|
|
|
|
|
|
|
t = x;
|
|
|
|
l = t;
|
|
|
|
rr = t + rad * w;
|
|
|
|
|
|
|
|
fb = bs[t];
|
|
|
|
fg = gs[t];
|
|
|
|
fr = rs[t];
|
|
|
|
fa = as[t];
|
|
|
|
|
|
|
|
lvb = bs[t + w * (h - 1)];
|
|
|
|
lvg = gs[t + w * (h - 1)];
|
|
|
|
lvr = rs[t + w * (h - 1)];
|
|
|
|
lva = as[t + w * (h - 1)];
|
|
|
|
|
|
|
|
valb = (rad + 1) * fb;
|
|
|
|
valg = (rad + 1) * fg;
|
|
|
|
valr = (rad + 1) * fr;
|
|
|
|
vala = (rad + 1) * fa;
|
|
|
|
|
|
|
|
for (i = 0; i < rad; i++)
|
|
|
|
{
|
|
|
|
valb += bs[t + i * w];
|
|
|
|
valg += gs[t + i * w];
|
|
|
|
valr += rs[t + i * w];
|
|
|
|
vala += as[t + i * w];
|
|
|
|
}
|
|
|
|
for (i = 0; i <= rad; i++)
|
|
|
|
{
|
|
|
|
valb += bs[rr] - fb;
|
|
|
|
valg += gs[rr] - fg;
|
|
|
|
valr += rs[rr] - fr;
|
|
|
|
vala += as[rr] - fa;
|
|
|
|
|
|
|
|
bt = (int) round(valb * iarr);
|
|
|
|
gt = (int) round(valg * iarr);
|
|
|
|
rt = (int) round(valr * iarr);
|
|
|
|
at = (int) round(vala * iarr);
|
|
|
|
bt = _normalize_color(bt);
|
|
|
|
gt = _normalize_color(gt);
|
|
|
|
rt = _normalize_color(rt);
|
|
|
|
at = _normalize_color(at);
|
|
|
|
ef->im_data[t] = (at << 24) | (rt << 16)
|
|
|
|
| (gt << 8) | bt;
|
|
|
|
|
|
|
|
rr += w;
|
|
|
|
t += w;
|
|
|
|
}
|
|
|
|
for (i = rad + 1; i < h - rad; i++)
|
|
|
|
{
|
|
|
|
valb += bs[rr] - bs[l];
|
|
|
|
valg += gs[rr] - gs[l];
|
|
|
|
valr += rs[rr] - rs[l];
|
|
|
|
vala += as[rr] - as[l];;
|
|
|
|
|
|
|
|
bt = (int) round(valb * iarr);
|
|
|
|
gt = (int) round(valg * iarr);
|
|
|
|
rt = (int) round(valr * iarr);
|
|
|
|
at = (int) round(vala * iarr);
|
|
|
|
bt = _normalize_color(bt);
|
|
|
|
gt = _normalize_color(gt);
|
|
|
|
rt = _normalize_color(rt);
|
|
|
|
at = _normalize_color(at);
|
|
|
|
ef->im_data[t] = (at << 24) | (rt << 16)
|
|
|
|
| (gt << 8) | bt;
|
|
|
|
|
|
|
|
l += w;
|
|
|
|
rr += w;
|
|
|
|
t += w;
|
|
|
|
}
|
|
|
|
for (i = h - rad; i < h; i++)
|
|
|
|
{
|
|
|
|
valb += lvb - bs[l];
|
|
|
|
valg += lvg - gs[l];
|
|
|
|
valr += lvr - rs[l];
|
|
|
|
vala += lva - as[l];
|
|
|
|
|
|
|
|
bt = (int) round(valb * iarr);
|
|
|
|
gt = (int) round(valg * iarr);
|
|
|
|
rt = (int) round(valr * iarr);
|
|
|
|
at = (int) round(vala * iarr);
|
|
|
|
bt = _normalize_color(bt);
|
|
|
|
gt = _normalize_color(gt);
|
|
|
|
rt = _normalize_color(rt);
|
|
|
|
at = _normalize_color(at);
|
|
|
|
ef->im_data[t] = (at << 24) | (rt << 16)
|
|
|
|
| (gt << 8) | bt;
|
|
|
|
|
|
|
|
l += w;
|
|
|
|
t += w;
|
|
|
|
}
|
|
|
|
ef->blur_posx = x;
|
|
|
|
passes++;
|
|
|
|
if (passes == 500)
|
|
|
|
{
|
|
|
|
ef->blur_posx = x++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
free(bs);
|
|
|
|
free(gs);
|
|
|
|
free(rs);
|
|
|
|
free(as);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Eina_Bool
|
|
|
|
_blur(void *data)
|
|
|
|
{
|
|
|
|
Ephoto_Filter *ef = data;
|
|
|
|
Evas_Coord w, h;
|
|
|
|
int wl, wu, m, i;
|
|
|
|
unsigned int sizes[3];
|
|
|
|
double ideal, rad;
|
|
|
|
|
|
|
|
w = ef->w;
|
|
|
|
h = ef->h;
|
|
|
|
rad = ef->w * .001 + ef->rad;
|
|
|
|
|
|
|
|
ideal = sqrt((12 * rad * rad / 3) + 1);
|
|
|
|
wl = floor(ideal);
|
|
|
|
if (wl % 2 == 0)
|
|
|
|
wl--;
|
|
|
|
wu = wl + 2;
|
|
|
|
ideal = (12 * rad * rad - 3 * wl * wl -
|
|
|
|
4 * 3 * wl - 3 * 3) / (-4 * wl - 4);
|
|
|
|
m = round(ideal);
|
|
|
|
|
|
|
|
for (i = ef->pos; i < 3; i++)
|
|
|
|
{
|
|
|
|
if (i < m)
|
|
|
|
sizes[i] = wl;
|
|
|
|
else
|
|
|
|
sizes[i] = wu;
|
|
|
|
|
|
|
|
rad = (sizes[i] - 1) / 2;
|
|
|
|
|
|
|
|
ef->im_data_new = memcpy(ef->im_data_new, ef->im_data,
|
|
|
|
sizeof(unsigned int) * w * h);
|
|
|
|
if (ef->blur_posy < ef->h - 1)
|
|
|
|
_blur_vertical(ef, rad);
|
|
|
|
if (ef->blur_posy == ef->h -1)
|
|
|
|
_blur_horizontal(ef, rad);
|
|
|
|
if (ef->blur_posx == ef->w - 1 &&
|
|
|
|
ef->blur_posy == ef->h - 1)
|
|
|
|
{
|
|
|
|
ef->pos = i+1;
|
|
|
|
ef->blur_posx = 0;
|
|
|
|
ef->blur_posy = 0;
|
|
|
|
}
|
|
|
|
return EINA_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
_idler_finishing_cb(ef, EINA_FALSE);
|
|
|
|
|
|
|
|
return EINA_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Eina_Bool
|
|
|
|
_sharpen(void *data)
|
|
|
|
{
|
|
|
|
Ephoto_Filter *ef = data;
|
|
|
|
unsigned int *p1, *p2, *p3;
|
|
|
|
int a, r, g, b, passes = 0;
|
|
|
|
int aa, rr, gg, bb;
|
|
|
|
int aaa, rrr, ggg, bbb;
|
|
|
|
Evas_Coord x, y, w, h;
|
|
|
|
|
|
|
|
w = ef->w;
|
|
|
|
h = ef->h;
|
|
|
|
|
|
|
|
for (y = ef->pos; y < h; y++)
|
|
|
|
{
|
|
|
|
p1 = ef->im_data + (y * w);
|
|
|
|
p2 = ef->im_data_orig + (y * w);
|
|
|
|
p3 = ef->im_data_new + (y * w);
|
|
|
|
for (x = 1; x < (w - 1); x++)
|
|
|
|
{
|
|
|
|
b = (int) (*p1 & 0xff);
|
|
|
|
g = (int) ((*p1 >> 8) & 0xff);
|
|
|
|
r = (int) ((*p1 >> 16) & 0xff);
|
|
|
|
a = (int) ((*p1 >> 24) & 0xff);
|
|
|
|
bb = (int) (*p2 & 0xff);
|
|
|
|
gg = (int) ((*p2 >> 8) & 0xff);
|
|
|
|
rr = (int) ((*p2 >> 16) & 0xff);
|
|
|
|
aa = (int) ((*p2 >> 24) & 0xff);
|
|
|
|
|
|
|
|
bbb = (int) ((2 * bb) - b);
|
|
|
|
ggg = (int) ((2 * gg) - g);
|
|
|
|
rrr = (int) ((2 * rr) - r);
|
|
|
|
aaa = (int) ((2 * aa) - a);
|
|
|
|
|
|
|
|
bbb = _normalize_color(bbb);
|
|
|
|
ggg = _normalize_color(ggg);
|
|
|
|
rrr = _normalize_color(rrr);
|
|
|
|
aaa = _normalize_color(aaa);
|
|
|
|
|
|
|
|
*p3 = (aaa << 24) | (rrr << 16) | (ggg << 8) | bbb;
|
|
|
|
p3++;
|
|
|
|
p2++;
|
|
|
|
p1++;
|
|
|
|
}
|
|
|
|
passes++;
|
|
|
|
if (passes == 500)
|
|
|
|
{
|
|
|
|
ef->pos = y++;
|
|
|
|
return EINA_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_idler_finishing_cb(ef, EINA_FALSE);
|
|
|
|
|
|
|
|
return EINA_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Eina_Bool
|
|
|
|
_dither(void *data)
|
|
|
|
{
|
|
|
|
Ephoto_Filter *ef = data;
|
|
|
|
Evas_Coord x, y, w, h;
|
|
|
|
int a, r, g, b, passes = 0;
|
|
|
|
int rr, gg, bb;
|
|
|
|
int errr, errg, errb;
|
|
|
|
|
|
|
|
w = ef->w;
|
|
|
|
h = ef->h;
|
|
|
|
|
|
|
|
for (y = ef->pos; y < h; y++)
|
|
|
|
{
|
|
|
|
for (x = 0; x < w; x++)
|
|
|
|
{
|
|
|
|
int index = y * w + x;
|
|
|
|
|
|
|
|
b = (ef->im_data[index]) & 0xff;
|
|
|
|
g = ((ef->im_data[index] >> 8) & 0xff);
|
|
|
|
r = ((ef->im_data[index] >> 16) & 0xff);
|
|
|
|
a = ((ef->im_data[index] >> 24) & 0xff);
|
|
|
|
b = _mul_color_alpha(b, a);
|
|
|
|
g = _mul_color_alpha(g, a);
|
|
|
|
r = _mul_color_alpha(r, a);
|
|
|
|
bb = (b > 127) ? 255 : 0;
|
|
|
|
gg = (g > 127) ? 255 : 0;
|
|
|
|
rr = (r > 127) ? 255 : 0;
|
|
|
|
rr = _normalize_color(rr);
|
|
|
|
gg = _normalize_color(gg);
|
|
|
|
bb = _normalize_color(bb);
|
|
|
|
bb = _demul_color_alpha(bb, a);
|
|
|
|
gg = _demul_color_alpha(gg, a);
|
|
|
|
rr = _demul_color_alpha(rr, a);
|
|
|
|
ef->im_data_new[index] = (a << 24) | (rr << 16) |
|
|
|
|
(gg << 8) | bb;
|
|
|
|
errb = b - bb;
|
|
|
|
errg = g - gg;
|
|
|
|
errr = r - rr;
|
|
|
|
|
|
|
|
if ((x+1) < w)
|
|
|
|
{
|
|
|
|
index = y * w + x + 1;
|
|
|
|
b = (ef->im_data[index] & 0xff);
|
|
|
|
g = ((ef->im_data[index] >> 8) & 0xff);
|
|
|
|
r = ((ef->im_data[index] >> 16) & 0xff);
|
|
|
|
a = ((ef->im_data[index] >> 24) & 0xff);
|
|
|
|
b = _mul_color_alpha(b, a);
|
|
|
|
g = _mul_color_alpha(g, a);
|
|
|
|
r = _mul_color_alpha(r, a);
|
|
|
|
bb = b + ((7 * errb) >> 4);
|
|
|
|
gg = g + ((7 * errg) >> 4);
|
|
|
|
rr = r + ((7 * errr) >> 4);
|
|
|
|
bb = _normalize_color(bb);
|
|
|
|
gg = _normalize_color(gg);
|
|
|
|
rr = _normalize_color(rr);
|
|
|
|
bb = _demul_color_alpha(bb, a);
|
|
|
|
gg = _demul_color_alpha(gg, a);
|
|
|
|
rr = _demul_color_alpha(rr, a);
|
|
|
|
ef->im_data_new[index] = (a << 24) | (rr << 16) |
|
|
|
|
(gg << 8) | bb;
|
|
|
|
}
|
|
|
|
if (x > 0 && (y+1) < h)
|
|
|
|
{
|
|
|
|
index = (y + 1) * w + (x - 1);
|
|
|
|
b = (ef->im_data[index] & 0xff);
|
|
|
|
g = ((ef->im_data[index] >> 8) & 0xff);
|
|
|
|
r = ((ef->im_data[index] >> 16) & 0xff);
|
|
|
|
a = ((ef->im_data[index] >> 24) & 0xff);
|
|
|
|
b = _mul_color_alpha(b, a);
|
|
|
|
g = _mul_color_alpha(g, a);
|
|
|
|
r = _mul_color_alpha(r, a);
|
|
|
|
bb = b + ((3 * errb) >> 4);
|
|
|
|
gg = g + ((3 * errg) >> 4);
|
|
|
|
rr = r + ((3 * errr) >> 4);
|
|
|
|
bb = _normalize_color(bb);
|
|
|
|
gg = _normalize_color(gg);
|
|
|
|
rr = _normalize_color(rr);
|
|
|
|
bb = _demul_color_alpha(bb, a);
|
|
|
|
gg = _demul_color_alpha(gg, a);
|
|
|
|
rr = _demul_color_alpha(rr, a);
|
|
|
|
ef->im_data_new[index] = (a << 24) | (rr << 16) |
|
|
|
|
(gg << 8) | bb;
|
|
|
|
}
|
|
|
|
if ((y+1) < h)
|
|
|
|
{
|
|
|
|
index = (y + 1) * w + x;
|
|
|
|
b = (ef->im_data[index] & 0xff);
|
|
|
|
g = ((ef->im_data[index] >> 8) & 0xff);
|
|
|
|
r = ((ef->im_data[index] >> 16) & 0xff);
|
|
|
|
a = ((ef->im_data[index] >> 24) & 0xff);
|
|
|
|
b = _mul_color_alpha(b, a);
|
|
|
|
g = _mul_color_alpha(g, a);
|
|
|
|
r = _mul_color_alpha(r, a);
|
|
|
|
bb = b + ((5 * errb) >> 4);
|
|
|
|
gg = g + ((5 * errg) >> 4);
|
|
|
|
rr = r + ((5 * errr) >> 4);
|
|
|
|
bb = _normalize_color(bb);
|
|
|
|
gg = _normalize_color(gg);
|
|
|
|
rr = _normalize_color(rr);
|
|
|
|
bb = _demul_color_alpha(bb, a);
|
|
|
|
gg = _demul_color_alpha(gg, a);
|
|
|
|
rr = _demul_color_alpha(rr, a);
|
|
|
|
ef->im_data_new[index] = (a << 24) | (rr << 16) |
|
|
|
|
(gg << 8) | bb;
|
|
|
|
}
|
|
|
|
if ((y+1) < h && (x+1) < w)
|
|
|
|
{
|
|
|
|
index = (y + 1) * w + (x + 1);
|
|
|
|
b = (ef->im_data[index] & 0xff);
|
|
|
|
g = ((ef->im_data[index] >> 8) & 0xff);
|
|
|
|
r = ((ef->im_data[index] >> 16) & 0xff);
|
|
|
|
a = ((ef->im_data[index] >> 24) & 0xff);
|
|
|
|
b = _mul_color_alpha(b, a);
|
|
|
|
g = _mul_color_alpha(g, a);
|
|
|
|
r = _mul_color_alpha(r, a);
|
|
|
|
bb = b + ((1 * errb) >> 4);
|
|
|
|
gg = g + ((1 * errg) >> 4);
|
|
|
|
rr = r + ((1 * errr) >> 4);
|
|
|
|
bb = _normalize_color(bb);
|
|
|
|
gg = _normalize_color(gg);
|
|
|
|
rr = _normalize_color(rr);
|
|
|
|
bb = _demul_color_alpha(bb, a);
|
|
|
|
gg = _demul_color_alpha(gg, a);
|
|
|
|
rr = _demul_color_alpha(rr, a);
|
|
|
|
ef->im_data_new[index] = (a << 24) | (rr << 16) |
|
|
|
|
(gg << 8) | bb;
|
|
|
|
}
|
|
|
|
/*Extra incrementing the x and y values to get
|
|
|
|
* the desired checkerboard effect is likely a
|
|
|
|
* bad hack*/
|
|
|
|
x++;
|
|
|
|
}
|
|
|
|
y++;
|
|
|
|
passes++;
|
|
|
|
if (passes == 500)
|
|
|
|
{
|
|
|
|
ef->pos = y++;
|
|
|
|
return EINA_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_idler_finishing_cb(ef, EINA_FALSE);
|
|
|
|
|
|
|
|
return EINA_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Eina_Bool
|
|
|
|
_grayscale(void *data)
|
|
|
|
{
|
|
|
|
Ephoto_Filter *ef = data;
|
|
|
|
int gray, i, r, g, b, a, passes = 0;
|
|
|
|
Evas_Coord w, h;
|
|
|
|
|
|
|
|
w = ef->w;
|
|
|
|
h = ef->h;
|
|
|
|
for (i = ef->pos; i < (w * h); i++)
|
|
|
|
{
|
|
|
|
b = (int) ((ef->im_data[i]) & 0xff);
|
|
|
|
g = (int) ((ef->im_data[i] >> 8) & 0xff);
|
|
|
|
r = (int) ((ef->im_data[i] >> 16) & 0xff);
|
|
|
|
a = (int) ((ef->im_data[i] >> 24) & 0xff);
|
|
|
|
b = _mul_color_alpha(b, a);
|
|
|
|
g = _mul_color_alpha(g, a);
|
|
|
|
r = _mul_color_alpha(r, a);
|
|
|
|
gray = (int) ((0.3 * r) + (0.59 * g) + (0.11 * b));
|
|
|
|
if (a >= 0 && a < 255)
|
|
|
|
gray = (gray * a) / 255;
|
|
|
|
ef->im_data_new[i] = (a << 24) | (gray << 16) | (gray << 8) | gray;
|
|
|
|
passes++;
|
|
|
|
if (passes == 500)
|
|
|
|
{
|
|
|
|
ef->pos = i++;
|
|
|
|
return EINA_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_idler_finishing_cb(ef, EINA_FALSE);
|
|
|
|
|
|
|
|
return EINA_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Eina_Bool
|
|
|
|
_sepia(void *data)
|
|
|
|
{
|
|
|
|
Ephoto_Filter *ef = data;
|
|
|
|
int i, r, rr, g, gg, b, bb, a, passes = 0;
|
|
|
|
Evas_Coord w, h;
|
|
|
|
|
|
|
|
w = ef->w;
|
|
|
|
h = ef->h;
|
|
|
|
for (i = ef->pos; i < (w * h); i++)
|
|
|
|
{
|
|
|
|
b = (int) ((ef->im_data[i]) & 0xff);
|
|
|
|
g = (int) ((ef->im_data[i] >> 8) & 0xff);
|
|
|
|
r = (int) ((ef->im_data[i] >> 16) & 0xff);
|
|
|
|
a = (int) ((ef->im_data[i] >> 24) & 0xff);
|
|
|
|
b = _mul_color_alpha(b, a);
|
|
|
|
g = _mul_color_alpha(g, a);
|
|
|
|
r = _mul_color_alpha(r, a);
|
|
|
|
rr = (int) ((r * .393) + (g * .769) + (b * .189));
|
|
|
|
rr = _normalize_color(rr);
|
|
|
|
gg = (int) ((r * .349) + (g * .686) + (b * .168));
|
|
|
|
gg = _normalize_color(gg);
|
|
|
|
bb = (int) ((r * .272) + (g * .534) + (b * .131));
|
|
|
|
bb = _normalize_color(bb);
|
|
|
|
bb = _demul_color_alpha(bb, a);
|
|
|
|
gg = _demul_color_alpha(gg, a);
|
|
|
|
rr = _demul_color_alpha(rr, a);
|
|
|
|
ef->im_data_new[i] = (a << 24) | (rr << 16) | (gg << 8) | bb;
|
|
|
|
passes++;
|
|
|
|
if (passes == 500)
|
|
|
|
{
|
|
|
|
ef->pos = i++;
|
|
|
|
return EINA_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_idler_finishing_cb(ef, EINA_FALSE);
|
|
|
|
|
|
|
|
return EINA_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Eina_Bool
|
|
|
|
_posterize(void *data)
|
|
|
|
{
|
|
|
|
Ephoto_Filter *ef = data;
|
|
|
|
int i, rr, gg, bb, a, passes = 0;
|
|
|
|
double fr, fg, fb, rad;
|
|
|
|
Evas_Coord w, h;
|
|
|
|
|
|
|
|
w = ef->w;
|
|
|
|
h = ef->h;
|
|
|
|
rad = ef->drad;
|
|
|
|
for (i = ef->pos; i < (w * h); i++)
|