From e4033542edbff3b6a91d8f24b257dcb6be1b52fb Mon Sep 17 00:00:00 2001 From: Wei-Hsin Yeh Date: Tue, 12 Nov 2024 09:49:36 +0800 Subject: [PATCH] Implement the drop shadow Add the `twin_stack_blur()` function to implement Mario's Stack Blur algorithm, which blurs the target pixel map. Additionally, create a test window to evaluate the effect of `twin_stack_blur()`. Implement a shadow pixel map beneath the test window with a shadow effect, enabling the shadow pixel map to blur the background and create a frosted glass effect. The shadow effect of the test window only becomes visible when the test window is on the top layer. Ref: https://melatonin.dev/blog/implementing-marios-stack-blur-15-times-in-cpp/ See #34 Signed-off-by: Wei-Hsin Yeh --- apps/multi.c | 56 ++++++++++++++++--- include/twin.h | 15 ++++- src/draw.c | 147 +++++++++++++++++++++++++++++++++++++++++++++++++ src/pixmap.c | 1 + src/screen.c | 82 +++++++++++++++++++++++++-- src/toplevel.c | 2 +- src/window.c | 61 +++++++++++++++++++- 7 files changed, 348 insertions(+), 16 deletions(-) diff --git a/apps/multi.c b/apps/multi.c index c33fd30..7b96c60 100644 --- a/apps/multi.c +++ b/apps/multi.c @@ -7,11 +7,12 @@ #include "apps_multi.h" #define D(x) twin_double_to_fixed(x) +#define ASSET_PATH "assets/" static void apps_line_start(twin_screen_t *screen, int x, int y, int w, int h) { twin_window_t *window = twin_window_create( - screen, TWIN_ARGB32, TwinWindowApplication, x, y, w, h); + screen, TWIN_ARGB32, TwinWindowApplication, x, y, w, h, false); twin_pixmap_t *pixmap = window->pixmap; twin_path_t *stroke = twin_path_create(); twin_fixed_t fy; @@ -38,7 +39,7 @@ static void apps_circletext_start(twin_screen_t *screen, int h) { twin_window_t *window = twin_window_create( - screen, TWIN_ARGB32, TwinWindowApplication, x, y, w, h); + screen, TWIN_ARGB32, TwinWindowApplication, x, y, w, h, false); int wid = window->client.right - window->client.left; int hei = window->client.bottom - window->client.top; twin_pixmap_t *pixmap = window->pixmap; @@ -83,7 +84,7 @@ static void apps_quickbrown_start(twin_screen_t *screen, int h) { twin_window_t *window = twin_window_create( - screen, TWIN_ARGB32, TwinWindowApplication, x, y, w, h); + screen, TWIN_ARGB32, TwinWindowApplication, x, y, w, h, false); int wid = window->client.right - window->client.left; int hei = window->client.bottom - window->client.top; twin_pixmap_t *pixmap = window->pixmap; @@ -126,7 +127,7 @@ static void apps_quickbrown_start(twin_screen_t *screen, static void apps_ascii_start(twin_screen_t *screen, int x, int y, int w, int h) { twin_window_t *window = twin_window_create( - screen, TWIN_ARGB32, TwinWindowApplication, x, y, w, h); + screen, TWIN_ARGB32, TwinWindowApplication, x, y, w, h, false); int wid = window->client.right - window->client.left; int hei = window->client.bottom - window->client.top; twin_pixmap_t *pixmap = window->pixmap; @@ -174,7 +175,7 @@ static void apps_ascii_start(twin_screen_t *screen, int x, int y, int w, int h) static void apps_jelly_start(twin_screen_t *screen, int x, int y, int w, int h) { twin_window_t *window = twin_window_create( - screen, TWIN_ARGB32, TwinWindowApplication, x, y, w, h); + screen, TWIN_ARGB32, TwinWindowApplication, x, y, w, h, false); int wid = window->client.right - window->client.left; int hei = window->client.bottom - window->client.top; twin_pixmap_t *pixmap = window->pixmap; @@ -250,7 +251,7 @@ static void draw_flower(twin_path_t *path, static void apps_flower_start(twin_screen_t *screen, int x, int y, int w, int h) { twin_window_t *window = twin_window_create( - screen, TWIN_ARGB32, TwinWindowApplication, x, y, w, h); + screen, TWIN_ARGB32, TwinWindowApplication, x, y, w, h, false); twin_pixmap_t *pixmap = window->pixmap; twin_path_t *stroke = twin_path_create(); twin_path_t *path = twin_path_create(); @@ -272,6 +273,47 @@ static void apps_flower_start(twin_screen_t *screen, int x, int y, int w, int h) twin_window_show(window); } +static void apps_test(twin_screen_t *screen, int x, int y, int w, int h) +{ + twin_pixmap_t *raw_background = + twin_pixmap_from_file(ASSET_PATH "tux.png", TWIN_ARGB32); + twin_window_t *window = twin_window_create( + screen, TWIN_ARGB32, TwinWindowApplication, x, y, w, h, true); + twin_window_set_name(window, "Test"); + twin_pixmap_t *scaled_background = twin_pixmap_create( + TWIN_ARGB32, window->pixmap->width, window->pixmap->height); + twin_fixed_t sx, sy; + sx = twin_fixed_div( + twin_int_to_fixed(raw_background->width), + twin_int_to_fixed(window->client.right - window->client.left)); + sy = twin_fixed_div( + twin_int_to_fixed(raw_background->height), + twin_int_to_fixed(window->client.bottom - window->client.top)); + + twin_matrix_scale(&raw_background->transform, sx, sy); + twin_operand_t srcop = { + .source_kind = TWIN_PIXMAP, + .u.pixmap = raw_background, + }; + + twin_composite(scaled_background, 0, 0, &srcop, 0, 0, 0, 0, 0, TWIN_SOURCE, + screen->width, screen->height); + + twin_pointer_t src, dst; + for (int y = window->client.top; y < window->client.bottom; y++) + for (int x = window->client.left; x < window->client.right; x++) { + src = + twin_pixmap_pointer(scaled_background, x - window->client.left, + y - window->client.top); + dst = twin_pixmap_pointer(window->pixmap, x, y); + *dst.argb32 = *src.argb32 | 0xff000000; + } + + twin_pixmap_destroy(scaled_background); + twin_pixmap_destroy(raw_background); + twin_window_show(window); +} + void apps_multi_start(twin_screen_t *screen, const char *name, int x, @@ -285,5 +327,5 @@ void apps_multi_start(twin_screen_t *screen, apps_quickbrown_start(screen, x += 20, y += 20, w, h); apps_ascii_start(screen, x += 20, y += 20, w, h); apps_jelly_start(screen, x += 20, y += 20, w / 2, h); - apps_flower_start(screen, x += 20, y += 20, w, h); + apps_test(screen, x += 20, y += 20, w, h); } diff --git a/include/twin.h b/include/twin.h index e2622f4..ff692f9 100644 --- a/include/twin.h +++ b/include/twin.h @@ -194,6 +194,7 @@ typedef struct _twin_pixmap { * Pixels */ twin_animation_t *animation; + bool shadow; twin_pointer_t p; /* * When representing a window, this point @@ -422,6 +423,10 @@ typedef void (*twin_destroy_func_t)(twin_window_t *window); struct _twin_window { twin_screen_t *screen; twin_pixmap_t *pixmap; + bool shadow; + twin_pixmap_t *shadow_pixmap; + twin_coord_t shadow_offset_x; + twin_coord_t shadow_offset_y; twin_window_style_t style; twin_rect_t client; twin_rect_t damage; @@ -648,6 +653,13 @@ void twin_fill(twin_pixmap_t *dst, void twin_premultiply_alpha(twin_pixmap_t *px); +void twin_stack_blur(twin_pixmap_t *px, + int radius, + twin_coord_t left, + twin_coord_t right, + twin_coord_t top, + twin_coord_t bottom); + /* * event.c */ @@ -1142,7 +1154,8 @@ twin_window_t *twin_window_create(twin_screen_t *screen, twin_coord_t x, twin_coord_t y, twin_coord_t width, - twin_coord_t height); + twin_coord_t height, + bool shadow); void twin_window_destroy(twin_window_t *window); diff --git a/src/draw.c b/src/draw.c index e3f1655..b5dbd2f 100644 --- a/src/draw.c +++ b/src/draw.c @@ -302,6 +302,153 @@ static const twin_src_msk_op comp3[2][4][4][3] = { #define operand_index(o) \ ((o)->source_kind == TWIN_SOLID ? 3 : o->u.pixmap->format) +#define _twin_add_ARGB(s, d, i, t) (((t) = (s) + twin_get_8(d, i))) +#define _twin_add(s, d, t) (((t) = (s) + (d))) +#define _twin_div(d, den, i, t) \ + (((t) = (d) / (den)), (t) = twin_get_8((t), 0), \ + (twin_argb32_t) twin_sat(t) << (i)) +#define _twin_sub_ARGB(s, d, i, t) (((t) = (s) - twin_get_8(d, i))) +#define _twin_sub(s, d, t) (((t) = (s) - (d))) +#define twin_put_8(d, i, t) (((t) = (d) << (i))) + +#define min(x, y) \ + ({ \ + typeof(x) _x = (x); \ + typeof(y) _y = (y); \ + (void) (&_x == &_y); \ + _x < _y ? _x : _y; \ + }) +#define max(x, y) \ + ({ \ + typeof(x) _x = (x); \ + typeof(y) _y = (y); \ + (void) (&_x == &_y); \ + _x > _y ? _x : _y; \ + }) + +void twin_stack(twin_pixmap_t *trg_px, + twin_pixmap_t *src_px, + int radius, + int first_str, + int first_end, + int second_str, + int second_end, + bool horiz_span) +{ + int den = (radius + 1) * (radius + 1); + twin_pointer_t src_ptr, trg_ptr, old_ptr, new_ptr; + twin_argb32_t sumInR, sumOutR, sumR, sumInG, sumOutG, sumG, sumInB, sumOutB, + sumB, _cur, _old, _new, _src; + uint16_t t1, t2, t3; + for (int first = first_str; first < first_end; first++) { + sumInR = sumOutR = sumR = sumInG = sumOutG = sumG = sumInB = sumOutB = + sumB = 0x00000000; + + /* Initialize SumOut by padding */ + if (horiz_span) + src_ptr = twin_pixmap_pointer(src_px, second_str, first); + else + src_ptr = twin_pixmap_pointer(src_px, first, second_str); + _src = *src_ptr.argb32; + + for (int i = second_str; i < second_str + radius; i++) { + sumOutR = _twin_add_ARGB(sumOutR, _src, 0, t1); + sumOutG = _twin_add_ARGB(sumOutG, _src, 8, t2); + sumOutB = _twin_add_ARGB(sumOutB, _src, 16, t3); + for (int j = 0; j < (i - second_str) + 1; j++) { + sumR = _twin_add_ARGB(sumR, _src, 0, t1); + sumG = _twin_add_ARGB(sumG, _src, 8, t2); + sumB = _twin_add_ARGB(sumB, _src, 16, t3); + } + } + + /* Initialize SumIn */ + for (int i = second_str; i < second_str + radius; i++) { + if (horiz_span) + src_ptr = twin_pixmap_pointer(src_px, i, first); + else + src_ptr = twin_pixmap_pointer(src_px, first, i); + _src = *src_ptr.argb32; + sumInR = _twin_add_ARGB(sumInR, _src, 0, t1); + sumInG = _twin_add_ARGB(sumInG, _src, 8, t2); + sumInB = _twin_add_ARGB(sumInB, _src, 16, t3); + for (int j = 0; j < radius - (i - second_str); j++) { + sumR = _twin_add_ARGB(sumR, _src, 0, t1); + sumG = _twin_add_ARGB(sumG, _src, 8, t2); + sumB = _twin_add_ARGB(sumB, _src, 16, t3); + } + } + + for (int cur = second_str; cur < second_end; cur++) { + if (horiz_span) { + src_ptr = twin_pixmap_pointer(src_px, cur, first); + trg_ptr = twin_pixmap_pointer(trg_px, cur, first); + old_ptr = twin_pixmap_pointer( + src_px, max(cur - radius, second_str), first); + new_ptr = twin_pixmap_pointer( + src_px, min(cur + radius, second_end - 1), first); + } else { + src_ptr = twin_pixmap_pointer(src_px, first, cur); + trg_ptr = twin_pixmap_pointer(trg_px, first, cur); + old_ptr = twin_pixmap_pointer(src_px, first, + max(cur - radius, second_str)); + new_ptr = twin_pixmap_pointer( + src_px, first, min(cur + radius, second_end - 1)); + } + _cur = *src_ptr.argb32; + _old = *old_ptr.argb32; + _new = *new_ptr.argb32; + /* STEP 1 : sum_out + current */ + sumOutR = _twin_add_ARGB(sumOutR, _cur, 0, t1); + sumOutG = _twin_add_ARGB(sumOutG, _cur, 8, t2); + sumOutB = _twin_add_ARGB(sumOutB, _cur, 16, t3); + /* STEP 2 : sum_in + new */ + sumInR = _twin_add_ARGB(sumInR, _new, 0, t1); + sumInG = _twin_add_ARGB(sumInG, _new, 8, t2); + sumInB = _twin_add_ARGB(sumInB, _new, 16, t3); + /* STEP 3 : sum + sum_in */ + sumR = _twin_add(sumR, sumInR, t1); + sumG = _twin_add(sumG, sumInG, t2); + sumB = _twin_add(sumB, sumInB, t3); + /* STEP 4 : sum / denominator */ + *trg_ptr.argb32 = + (_twin_div(sumR, den, 0, t1) | _twin_div(sumG, den, 8, t2) | + _twin_div(sumB, den, 16, t3) | (*src_ptr.argb32 & 0xff000000)); + /* STEP 5 : sum - sum_out */ + sumR = _twin_sub(sumR, sumOutR, t1); + sumG = _twin_sub(sumG, sumOutG, t2); + sumB = _twin_sub(sumB, sumOutB, t3); + /* STEP 6 : sum_out - old */ + sumOutR = _twin_sub_ARGB(sumOutR, _old, 0, t1); + sumOutG = _twin_sub_ARGB(sumOutG, _old, 8, t2); + sumOutB = _twin_sub_ARGB(sumOutB, _old, 16, t3); + /* STEP 7 : sum_in - current */ + sumInR = _twin_sub_ARGB(sumInR, _cur, 0, t1); + sumInG = _twin_sub_ARGB(sumInG, _cur, 8, t2); + sumInB = _twin_sub_ARGB(sumInB, _cur, 16, t3); + } + } +} + +void twin_stack_blur(twin_pixmap_t *px, + int radius, + twin_coord_t left, + twin_coord_t right, + twin_coord_t top, + twin_coord_t bottom) +{ + if (px->format != TWIN_ARGB32) + return; + twin_pixmap_t *tmp_px = + twin_pixmap_create(px->format, px->width, px->height); + memcpy(tmp_px->p.v, px->p.v, + px->width * px->height * twin_bytes_per_pixel(px->format)); + twin_stack(tmp_px, px, radius, top, bottom, left, right, true); + twin_stack(px, tmp_px, radius, left, right, top, bottom, false); + twin_pixmap_destroy(tmp_px); + return; +} + /* FIXME: source clipping is busted */ static void _twin_composite_simple(twin_pixmap_t *dst, twin_coord_t dst_x, diff --git a/src/pixmap.c b/src/pixmap.c index f620411..7a08edf 100644 --- a/src/pixmap.c +++ b/src/pixmap.c @@ -43,6 +43,7 @@ twin_pixmap_t *twin_pixmap_create(twin_format_t format, pixmap->stride = stride; pixmap->disable = 0; pixmap->animation = NULL; + pixmap->shadow = false; pixmap->p.v = pixmap + 1; memset(pixmap->p.v, '\0', space); return pixmap; diff --git a/src/screen.c b/src/screen.c index 76bcfe2..9b99547 100644 --- a/src/screen.c +++ b/src/screen.c @@ -134,7 +134,7 @@ static void twin_screen_span_pixmap(twin_screen_t maybe_unused *screen, return; if (p->y + p->height <= y) return; - /* bounds check in x*/ + /* bounds check in x */ p_left = left; if (p_left < p->x) p_left = p->x; @@ -183,6 +183,78 @@ void twin_screen_update(twin_screen_t *screen) if (screen->put_begin) (*screen->put_begin)(left, top, right, bottom, screen->closure); + /* Handle drop shadow. */ + + /* Find the drop shadow. */ + twin_pixmap_t *shadow = NULL; + twin_pointer_t dst; + + /* The shadow effect of the test window only becomes visible when it is + * on the top layer. */ + bool blur = true; + for (p = screen->bottom; p; p = p->up) { + if (p->shadow) { + shadow = p; + continue; + } + if (shadow && shadow->window != p->window) + blur = false; + } + if (shadow) { + twin_fill(shadow, 0x00000000, TWIN_SOURCE, 0, 0, shadow->width, + shadow->height); + for (y = 0; y < (*shadow).height; y++) { + twin_pointer_t dst; + twin_source_u src; + twin_coord_t p_start, shadow_start, real_start, real_end, + overlapped_width, src_y; + int i = 0; + bool is_p_greater; + /* Take the screen's background as the bottom layer of the + * shadow. */ + if (screen->background) { + p = screen->background; + real_start = (*shadow).x; + real_end = (*shadow).x + (*shadow).width; + overlapped_width = real_end - real_start; + src_y = (*shadow).y + y; + if (src_y > 0 && src_y < (*p).height - 1) { + dst = twin_pixmap_pointer(shadow, 0, y); + src.p = twin_pixmap_pointer(p, (*shadow).x, src_y); + pop32(dst, src, overlapped_width); + } + } + /* Render each layer of the pixmap under the shadow pixel map + * onto the shadow. */ + for (p = screen->bottom; p; p = p->up) { + if (p->shadow) + break; + + src_y = (*shadow).y + y - (*p).y; + if (src_y < 0 || src_y >= (*p).height) + continue; + + is_p_greater = (*p).x > (*shadow).x; + p_start = is_p_greater ? 0 : (*shadow).x - (*p).x; + shadow_start = is_p_greater ? (*p).x - (*shadow).x : 0; + real_start = is_p_greater ? (*p).x : (*shadow).x; + real_end = + ((*p).x + (*p).width > (*shadow).x + (*shadow).width) + ? (*shadow).x + (*shadow).width + : (*p).x + (*p).width; + if (real_start >= real_end) + continue; + overlapped_width = real_end - real_start; + dst = twin_pixmap_pointer(shadow, shadow_start, y); + src.p = twin_pixmap_pointer(p, p_start, src_y); + pop32(dst, src, overlapped_width); + } + } + /* Add a frosted glass effect to the shadow pixmap. */ + if (blur) + twin_stack_blur(shadow, 2, 0, shadow->width, 0, shadow->height); + } + for (y = top; y < bottom; y++) { if (screen->background) { twin_pointer_t dst; @@ -356,7 +428,8 @@ bool twin_screen_dispatch(twin_screen_t *screen, twin_event_t *event) evt = *event; evt.kind = TwinEventLeave; _twin_adj_mouse_evt(&evt, pixmap); - twin_pixmap_dispatch(pixmap, &evt); + if (!pixmap->shadow) + twin_pixmap_dispatch(pixmap, &evt); } pixmap = screen->target = ntarget; @@ -365,7 +438,8 @@ bool twin_screen_dispatch(twin_screen_t *screen, twin_event_t *event) evt = *event; _twin_adj_mouse_evt(&evt, pixmap); evt.kind = TwinEventEnter; - twin_pixmap_dispatch(pixmap, &evt); + if (!pixmap->shadow) + twin_pixmap_dispatch(pixmap, &evt); } } @@ -389,7 +463,7 @@ bool twin_screen_dispatch(twin_screen_t *screen, twin_event_t *event) pixmap = NULL; break; } - if (pixmap) + if (pixmap && !pixmap->shadow) return twin_pixmap_dispatch(pixmap, event); return false; } diff --git a/src/toplevel.c b/src/toplevel.c index 53d6e68..8216086 100644 --- a/src/toplevel.c +++ b/src/toplevel.c @@ -81,7 +81,7 @@ twin_toplevel_t *twin_toplevel_create(twin_screen_t *screen, { twin_toplevel_t *toplevel; twin_window_t *window = - twin_window_create(screen, format, style, x, y, width, height); + twin_window_create(screen, format, style, x, y, width, height, false); if (!window) return NULL; diff --git a/src/window.c b/src/window.c index 72fa65a..7d11255 100644 --- a/src/window.c +++ b/src/window.c @@ -23,7 +23,8 @@ twin_window_t *twin_window_create(twin_screen_t *screen, twin_coord_t x, twin_coord_t y, twin_coord_t width, - twin_coord_t height) + twin_coord_t height, + bool shadow) { twin_window_t *window = malloc(sizeof(twin_window_t)); twin_coord_t left, top, right, bottom; @@ -59,6 +60,21 @@ twin_window_t *twin_window_create(twin_screen_t *screen, twin_pixmap_origin_to_clip(window->pixmap); window->pixmap->window = window; twin_pixmap_move(window->pixmap, x, y); + + window->shadow = shadow; + if (window->shadow) { + /* Add a shadow pixel map under the window with a shadow effect. */ + window->shadow_offset_x = 50, window->shadow_offset_y = 50; + window->shadow_pixmap = twin_pixmap_create(format, width, height); + if (!window->shadow_pixmap) + return NULL; + window->shadow_pixmap->shadow = true; + window->shadow_pixmap->window = window; + twin_pixmap_move(window->shadow_pixmap, x + window->shadow_offset_x, + y + window->shadow_offset_y); + } else + window->shadow_pixmap = NULL; + window->damage = window->client; window->client_grab = false; window->want_focus = false; @@ -75,6 +91,8 @@ twin_window_t *twin_window_create(twin_screen_t *screen, void twin_window_destroy(twin_window_t *window) { twin_window_hide(window); + if (window->shadow) + twin_pixmap_destroy(window->shadow_pixmap); twin_pixmap_destroy(window->pixmap); free(window->name); free(window); @@ -82,13 +100,19 @@ void twin_window_destroy(twin_window_t *window) void twin_window_show(twin_window_t *window) { - if (window->pixmap != window->screen->top) + if (window->shadow) { + twin_pixmap_show(window->shadow_pixmap, window->screen, + window->screen->top); + twin_pixmap_show(window->pixmap, window->screen, window->shadow_pixmap); + } else if (window->pixmap != window->screen->top) twin_pixmap_show(window->pixmap, window->screen, window->screen->top); } void twin_window_hide(twin_window_t *window) { twin_pixmap_hide(window->pixmap); + if (window->shadow) + twin_pixmap_hide(window->shadow_pixmap); } void twin_window_configure(twin_window_t *window, @@ -101,6 +125,9 @@ void twin_window_configure(twin_window_t *window, bool need_repaint = false; twin_pixmap_disable_update(window->pixmap); + if (window->shadow) + twin_pixmap_disable_update(window->shadow_pixmap); + if (style != window->style) { window->style = style; need_repaint = true; @@ -112,22 +139,42 @@ void twin_window_configure(twin_window_t *window, window->pixmap = twin_pixmap_create(old->format, width, height); window->pixmap->window = window; twin_pixmap_move(window->pixmap, x, y); + if (window->shadow) + twin_pixmap_move(window->shadow_pixmap, x + window->shadow_offset_x, + y + window->shadow_offset_y); if (old->screen) twin_pixmap_show(window->pixmap, window->screen, old); for (i = 0; i < old->disable; i++) twin_pixmap_disable_update(window->pixmap); twin_pixmap_destroy(old); + twin_pixmap_destroy(window->shadow_pixmap); twin_pixmap_reset_clip(window->pixmap); twin_pixmap_clip(window->pixmap, window->client.left, window->client.top, window->client.right, window->client.bottom); twin_pixmap_origin_to_clip(window->pixmap); + + if (window->shadow) { + window->shadow_pixmap = + twin_pixmap_create(old->format, width, height); + if (!window->shadow_pixmap) + return; + window->shadow_pixmap->shadow = true; + twin_pixmap_move(window->shadow_pixmap, x + window->shadow_offset_x, + y + window->shadow_offset_y); + } } - if (x != window->pixmap->x || y != window->pixmap->y) + if (x != window->pixmap->x || y != window->pixmap->y) { twin_pixmap_move(window->pixmap, x, y); + if (window->shadow) + twin_pixmap_move(window->shadow_pixmap, x + window->shadow_offset_x, + y + window->shadow_offset_y); + } if (need_repaint) twin_window_draw(window); twin_pixmap_enable_update(window->pixmap); + if (window->shadow) + twin_pixmap_enable_update(window->shadow_pixmap); } void twin_window_style_size(twin_window_style_t style, twin_rect_t *size) @@ -307,12 +354,18 @@ void twin_window_draw(twin_window_t *window) window->damage.right, window->damage.bottom); twin_screen_disable_update(window->screen); + if (window->shadow) + twin_pixmap_disable_update(window->shadow_pixmap); (*window->draw)(window); /* damage matching screen area */ twin_pixmap_damage(pixmap, window->damage.left, window->damage.top, window->damage.right, window->damage.bottom); + if (window->shadow) + twin_pixmap_damage(window->shadow_pixmap, window->damage.left, + window->damage.top, window->damage.right, + window->damage.bottom); twin_screen_enable_update(window->screen); /* clear damage and restore clip */ @@ -321,6 +374,8 @@ void twin_window_draw(twin_window_t *window) twin_pixmap_reset_clip(pixmap); twin_pixmap_clip(pixmap, window->client.left, window->client.top, window->client.right, window->client.bottom); + if (window->shadow) + twin_pixmap_enable_update(window->shadow_pixmap); } /* window keep track of local damage */