mirror of
https://github.com/VincentWei/MiniGUI.git
synced 2026-02-08 11:52:02 +08:00
1182 lines
34 KiB
C
1182 lines
34 KiB
C
///////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// IMPORTANT NOTICE
|
|
//
|
|
// The following open source license statement does not apply to any
|
|
// entity in the Exception List published by FMSoft.
|
|
//
|
|
// For more information, please visit:
|
|
//
|
|
// https://www.fmsoft.cn/exception-list
|
|
//
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
/*
|
|
* This file is part of MiniGUI, a mature cross-platform windowing
|
|
* and Graphics User Interface (GUI) support system for embedded systems
|
|
* and smart IoT devices.
|
|
*
|
|
* Copyright (C) 2002~2020, Beijing FMSoft Technologies Co., Ltd.
|
|
* Copyright (C) 1998~2002, WEI Yongming
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
* Or,
|
|
*
|
|
* As this program is a library, any link to this program must follow
|
|
* GNU General Public License version 3 (GPLv3). If you cannot accept
|
|
* GPLv3, you need to be licensed from FMSoft.
|
|
*
|
|
* If you have got a commercial license of this program, please use it
|
|
* under the terms and conditions of the commercial license.
|
|
*
|
|
* For more information about the commercial license, please refer to
|
|
* <http://www.minigui.com/blog/minigui-licensing-policy/>.
|
|
*/
|
|
/*
|
|
** compsor-fallback.c: the fallback compositor.
|
|
**
|
|
** Create date: 2020-01-19
|
|
**
|
|
** Current maintainer: Wei Yongming.
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
|
|
#include "common.h"
|
|
|
|
#if defined(_MGRM_PROCESSES) && defined(_MGSCHEMA_COMPOSITING)
|
|
|
|
#include "minigui.h"
|
|
#include "gdi.h"
|
|
#include "window.h"
|
|
#include "debug.h"
|
|
|
|
#define SIZE_CLIPRC_HEAP 64
|
|
|
|
#define DIRTY_ZT_PPP 0x01
|
|
#define DIRTY_ZT_WIN 0x02
|
|
#define DIRTY_ZT_WPP 0x04
|
|
|
|
struct _CompositorCtxt {
|
|
RECT rc_screen; // screen rect - avoid duplicated GetScreenRect calls
|
|
BLOCKHEAP cliprc_heap;// heap for clipping rects
|
|
CLIPRGN wins_rgn; // visible region for all windows (subtract popup menus)
|
|
CLIPRGN dirty_rgn; // the dirty region
|
|
CLIPRGN inv_rgn; // the invalid region for a specific znode
|
|
Uint32 dirty_types;// the dirty znode types.
|
|
|
|
/* Since 5.0.6: fields for combining two layers */
|
|
MG_Layer* layer; // the current layer; NULL for topmost layer.
|
|
|
|
int offx; // the offset for the general znodes.
|
|
int offy;
|
|
};
|
|
|
|
static CompositorCtxt* initialize (const char* name)
|
|
{
|
|
CompositorCtxt* ctxt;
|
|
|
|
ctxt = malloc (sizeof (CompositorCtxt));
|
|
if (ctxt) {
|
|
ctxt->rc_screen = GetScreenRect();
|
|
InitFreeClipRectList (&ctxt->cliprc_heap, SIZE_CLIPRC_HEAP);
|
|
InitClipRgn (&ctxt->wins_rgn, &ctxt->cliprc_heap);
|
|
InitClipRgn (&ctxt->dirty_rgn, &ctxt->cliprc_heap);
|
|
InitClipRgn (&ctxt->inv_rgn, &ctxt->cliprc_heap);
|
|
|
|
SetClipRgn (&ctxt->wins_rgn, &ctxt->rc_screen);
|
|
|
|
ctxt->layer = NULL;
|
|
ctxt->offx = 0;
|
|
ctxt->offy = 0;
|
|
}
|
|
|
|
return ctxt;
|
|
}
|
|
|
|
static void terminate (CompositorCtxt* ctxt)
|
|
{
|
|
if (ctxt) {
|
|
EmptyClipRgn (&ctxt->wins_rgn);
|
|
EmptyClipRgn (&ctxt->dirty_rgn);
|
|
EmptyClipRgn (&ctxt->inv_rgn);
|
|
DestroyFreeClipRectList (&ctxt->cliprc_heap);
|
|
free (ctxt);
|
|
}
|
|
}
|
|
|
|
/* return the number of subtracted windows above */
|
|
static int subtract_opaque_win_znodes_above (CompositorCtxt* ctxt, int from)
|
|
{
|
|
const ZNODEHEADER* znode_hdr;
|
|
int nr_subtracted = 0;
|
|
int prev;
|
|
|
|
/* subtract opaque window znodes */
|
|
prev = ServerGetPrevZNode (ctxt->layer, from, NULL);
|
|
while (prev > 0) {
|
|
CLIPRGN* rgn;
|
|
|
|
znode_hdr = ServerGetWinZNodeHeader (ctxt->layer, prev, (void**)&rgn, FALSE);
|
|
assert (znode_hdr);
|
|
|
|
if ((znode_hdr->flags & ZNIF_VISIBLE) &&
|
|
(znode_hdr->ct & CT_SYSTEM_MASK) == CT_OPAQUE) {
|
|
|
|
if (SubtractRegion (&ctxt->dirty_rgn, &ctxt->dirty_rgn, rgn)) {
|
|
nr_subtracted++;
|
|
}
|
|
}
|
|
|
|
prev = ServerGetPrevZNode (ctxt->layer, prev, NULL);
|
|
}
|
|
|
|
return nr_subtracted;
|
|
}
|
|
|
|
static inline void tile_dirty_region_for_wallpaper (CompositorCtxt* ctxt)
|
|
{
|
|
int wp_w = GetGDCapability (HDC_SCREEN, GDCAP_HPIXEL);
|
|
int wp_h = GetGDCapability (HDC_SCREEN, GDCAP_VPIXEL);
|
|
|
|
if (wp_w > 0 && wp_h > 0) {
|
|
|
|
// tile the dirty region of the wallpaper pattern
|
|
int y = 0;
|
|
int left_h = RECTH (ctxt->rc_screen);
|
|
|
|
CopyRegion (&ctxt->inv_rgn, &ctxt->dirty_rgn);
|
|
while (left_h > 0) {
|
|
|
|
int x = 0;
|
|
int left_w = RECTW (ctxt->rc_screen);
|
|
|
|
while (left_w > 0) {
|
|
if (x > 0 || y > 0) {
|
|
OffsetRegion (&ctxt->inv_rgn, x, y);
|
|
UnionRegion (&ctxt->dirty_rgn,
|
|
&ctxt->dirty_rgn, &ctxt->inv_rgn);
|
|
OffsetRegion (&ctxt->inv_rgn, -x, -y);
|
|
}
|
|
|
|
x += wp_w;
|
|
left_w -= wp_w;
|
|
}
|
|
|
|
left_h -= wp_h;
|
|
y += wp_h;
|
|
}
|
|
|
|
EmptyClipRgn (&ctxt->inv_rgn);
|
|
}
|
|
}
|
|
|
|
static void composite_wallpaper_rect (CompositorCtxt* ctxt,
|
|
const RECT* dirty_rc)
|
|
{
|
|
int wp_w = GetGDCapability (HDC_SCREEN, GDCAP_HPIXEL);
|
|
int wp_h = GetGDCapability (HDC_SCREEN, GDCAP_VPIXEL);
|
|
|
|
if (wp_w > 0 && wp_h > 0) {
|
|
|
|
// tile the wallpaper pattern
|
|
int y = 0;
|
|
int left_h = RECTH (ctxt->rc_screen);
|
|
while (left_h > 0) {
|
|
|
|
int x = 0;
|
|
int left_w = RECTW (ctxt->rc_screen);
|
|
while (left_w > 0) {
|
|
RECT rc = { x, y, x + wp_w, y + wp_h };
|
|
RECT eff_rc;
|
|
if (IntersectRect (&eff_rc, dirty_rc, &rc)) {
|
|
SelectClipRect (HDC_SCREEN_SYS, &eff_rc);
|
|
SetMemDCColorKey (HDC_SCREEN, 0, 0);
|
|
SetMemDCAlpha (HDC_SCREEN, 0, 0);
|
|
BitBlt (HDC_SCREEN, 0, 0, RECTW (rc), RECTH (rc),
|
|
HDC_SCREEN_SYS, rc.left, rc.top, 0);
|
|
}
|
|
|
|
x += wp_w;
|
|
left_w -= wp_w;
|
|
}
|
|
|
|
left_h -= wp_h;
|
|
y += wp_h;
|
|
}
|
|
}
|
|
else {
|
|
SetBrushColor (HDC_SCREEN_SYS,
|
|
GetWindowElementPixelEx (HWND_DESKTOP, HDC_SCREEN_SYS,
|
|
WE_BGC_DESKTOP));
|
|
SelectClipRect (HDC_SCREEN_SYS, &ctxt->rc_screen);
|
|
FillBox (HDC_SCREEN_SYS, dirty_rc->left, dirty_rc->top,
|
|
RECTWP(dirty_rc), RECTHP(dirty_rc));
|
|
}
|
|
}
|
|
|
|
static inline void composite_wallpaper (CompositorCtxt* ctxt)
|
|
{
|
|
if (!IsEmptyClipRgn (&ctxt->dirty_rgn)) {
|
|
|
|
CLIPRECT* crc = ctxt->dirty_rgn.head;
|
|
while (crc) {
|
|
composite_wallpaper_rect (ctxt, &crc->rc);
|
|
crc = crc->next;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void composite_ppp_znodes (CompositorCtxt* ctxt)
|
|
{
|
|
int zidx, nr_ppps;
|
|
|
|
nr_ppps = ServerGetPopupMenusCount ();
|
|
/* we start from the topmost popup menu znode */
|
|
for (zidx = (nr_ppps - 1); zidx >= 0; zidx--) {
|
|
const ZNODEHEADER* znode_hdr = NULL;
|
|
CLIPRGN* rgn;
|
|
znode_hdr = ServerGetPopupMenuZNodeHeader (zidx, (void**)&rgn, TRUE);
|
|
assert (znode_hdr);
|
|
assert (rgn);
|
|
|
|
// we use invalid region as the clipping region of the popup menu znode
|
|
ClipRgnIntersect (&ctxt->inv_rgn, &ctxt->dirty_rgn, rgn);
|
|
if (IsEmptyClipRgn (&ctxt->inv_rgn)) {
|
|
ServerReleasePopupMenuZNodeHeader (zidx);
|
|
_DBG_PRINTF ("ppp %d skipped\n", zidx);
|
|
continue;
|
|
}
|
|
else {
|
|
_DBG_PRINTF ("ppp %d composited\n", zidx);
|
|
// blit this popup menu to screen
|
|
SelectClipRegion (HDC_SCREEN_SYS, &ctxt->inv_rgn);
|
|
BitBlt (znode_hdr->mem_dc, 0, 0,
|
|
RECTW (znode_hdr->rc), RECTH (znode_hdr->rc),
|
|
HDC_SCREEN_SYS, znode_hdr->rc.left, znode_hdr->rc.top, 0);
|
|
SelectClipRect (HDC_SCREEN_SYS, NULL);
|
|
|
|
// subtract the clipping region of the popup menu znode
|
|
// from the dirty region, because this compositor
|
|
// handles any popup menu as opaque.
|
|
SubtractRegion (&ctxt->dirty_rgn, &ctxt->dirty_rgn, &ctxt->inv_rgn);
|
|
}
|
|
|
|
ServerReleasePopupMenuZNodeHeader (zidx);
|
|
}
|
|
|
|
EmptyClipRgn (&ctxt->inv_rgn);
|
|
}
|
|
|
|
static void composite_opaque_win_znode (CompositorCtxt* ctxt, int from)
|
|
{
|
|
const ZNODEHEADER* znode_hdr;
|
|
CLIPRGN* rgn;
|
|
|
|
if (IsEmptyClipRgn (&ctxt->dirty_rgn)) {
|
|
return;
|
|
}
|
|
|
|
znode_hdr = ServerGetWinZNodeHeader (ctxt->layer, from, (void**)&rgn, TRUE);
|
|
assert (znode_hdr);
|
|
|
|
if ((znode_hdr->flags & ZNIF_VISIBLE) &&
|
|
((znode_hdr->ct & CT_SYSTEM_MASK) == CT_OPAQUE) &&
|
|
IntersectRegion (&ctxt->inv_rgn, &ctxt->dirty_rgn, rgn)) {
|
|
BOOL offx, offy;
|
|
|
|
if ((ctxt->offx || ctxt->offy) && !(znode_hdr->flags & ZOF_IF_SPECIAL)) {
|
|
offx = ctxt->offx;
|
|
offy = ctxt->offy;
|
|
}
|
|
else {
|
|
offx = 0;
|
|
offy = 0;
|
|
}
|
|
|
|
SetMemDCColorKey (znode_hdr->mem_dc, 0, 0);
|
|
SetMemDCAlpha (znode_hdr->mem_dc, 0, 0);
|
|
SelectClipRegion (HDC_SCREEN_SYS, &ctxt->inv_rgn);
|
|
BitBlt (znode_hdr->mem_dc,
|
|
0, 0, RECTW(znode_hdr->rc), RECTH(znode_hdr->rc),
|
|
HDC_SCREEN_SYS,
|
|
znode_hdr->rc.left + offx, znode_hdr->rc.top + offy, 0);
|
|
SelectClipRect (HDC_SCREEN_SYS, NULL);
|
|
|
|
SubtractRegion (&ctxt->dirty_rgn, &ctxt->dirty_rgn, &ctxt->inv_rgn);
|
|
}
|
|
EmptyClipRgn (&ctxt->inv_rgn);
|
|
|
|
ServerReleaseWinZNodeHeader (ctxt->layer, from);
|
|
}
|
|
|
|
static void composite_all_lucent_win_znodes_above (CompositorCtxt* ctxt,
|
|
const CLIPRGN* dirty_rgn, int zidx)
|
|
{
|
|
int prev;
|
|
|
|
prev = ServerGetPrevZNode (ctxt->layer, zidx, NULL);
|
|
while (prev > 0) {
|
|
const ZNODEHEADER* znode_hdr;
|
|
CLIPRGN* rgn;
|
|
|
|
znode_hdr = ServerGetWinZNodeHeader (ctxt->layer, prev, (void**)&rgn, TRUE);
|
|
assert (znode_hdr);
|
|
|
|
if ((znode_hdr->flags & ZNIF_VISIBLE) &&
|
|
(znode_hdr->ct & CT_SYSTEM_MASK) != CT_OPAQUE &&
|
|
IntersectRegion (&ctxt->inv_rgn, dirty_rgn, rgn)) {
|
|
int offx, offy;
|
|
|
|
if ((ctxt->offx || ctxt->offy) && !(znode_hdr->flags & ZOF_IF_SPECIAL)) {
|
|
offx = ctxt->offx;
|
|
offy = ctxt->offy;
|
|
}
|
|
else {
|
|
offx = 0;
|
|
offy = 0;
|
|
}
|
|
|
|
switch (znode_hdr->ct & CT_SYSTEM_MASK) {
|
|
case CT_COLORKEY:
|
|
SetMemDCAlpha (znode_hdr->mem_dc, 0, 0);
|
|
SetMemDCColorKey (znode_hdr->mem_dc,
|
|
MEMDC_FLAG_SRCCOLORKEY,
|
|
DWORD2Pixel (znode_hdr->mem_dc, znode_hdr->ct_arg));
|
|
break;
|
|
|
|
case CT_LOGICALPIXEL:
|
|
_WRN_PRINTF("CT_LOGICALPIXEL is not implemented\n");
|
|
break;
|
|
|
|
case CT_ALPHACHANNEL:
|
|
SetMemDCColorKey (znode_hdr->mem_dc, 0, 0);
|
|
SetMemDCAlpha (znode_hdr->mem_dc,
|
|
MEMDC_FLAG_SRCALPHA, (BYTE)znode_hdr->ct_arg);
|
|
break;
|
|
|
|
case CT_ALPHAPIXEL:
|
|
SetMemDCColorKey (znode_hdr->mem_dc, 0, 0);
|
|
SetMemDCAlpha (znode_hdr->mem_dc,
|
|
MEMDC_FLAG_SRCPIXELALPHA, 0);
|
|
break;
|
|
|
|
case CT_BLURRED:
|
|
_WRN_PRINTF("CT_BLURRED is not implemented\n");
|
|
break;
|
|
|
|
default:
|
|
assert (0); // never touch here.
|
|
break;
|
|
}
|
|
|
|
SelectClipRegion (HDC_SCREEN_SYS, &ctxt->inv_rgn);
|
|
BitBlt (znode_hdr->mem_dc,
|
|
0, 0, RECTW(znode_hdr->rc), RECTH(znode_hdr->rc),
|
|
HDC_SCREEN_SYS,
|
|
znode_hdr->rc.left + offx, znode_hdr->rc.top + offy, 0);
|
|
SelectClipRect (HDC_SCREEN_SYS, NULL);
|
|
}
|
|
|
|
EmptyClipRgn (&ctxt->inv_rgn);
|
|
|
|
ServerReleaseWinZNodeHeader (ctxt->layer, prev);
|
|
|
|
prev = ServerGetPrevZNode (ctxt->layer, prev, NULL);
|
|
}
|
|
}
|
|
|
|
static void composite_on_dirty_region (CompositorCtxt* ctxt, int from)
|
|
{
|
|
CLIPRGN tmp_rgn;
|
|
int next;
|
|
|
|
/* tmp_rgn will be used to save the dirty region for lucent window znodes
|
|
above the current znode. */
|
|
InitClipRgn (&tmp_rgn, &ctxt->cliprc_heap);
|
|
|
|
/* subtract the opaque window znodes above the current znode from
|
|
the dirty region. */
|
|
if (from > 0 && subtract_opaque_win_znodes_above (ctxt, from) > 0) {
|
|
if (IsEmptyClipRgn (&ctxt->dirty_rgn)) {
|
|
goto done;
|
|
}
|
|
}
|
|
|
|
/* generate the effective dirty region for windows by substracting
|
|
the region of visible popup menus. */
|
|
IntersectRegion (&ctxt->dirty_rgn, &ctxt->dirty_rgn, &ctxt->wins_rgn);
|
|
if (IsEmptyClipRgn (&ctxt->dirty_rgn)) {
|
|
goto done;
|
|
}
|
|
|
|
/* compositing the current window znode and the znodes below it */
|
|
if (from <= 0)
|
|
next = ServerGetNextZNode (ctxt->layer, 0, NULL);
|
|
else
|
|
next = from;
|
|
|
|
while (next > 0) {
|
|
|
|
if (!IsEmptyClipRgn (&ctxt->dirty_rgn)) {
|
|
// save dirty region for compositing the lucent znodes above
|
|
// the current znode.
|
|
CopyRegion (&tmp_rgn, &ctxt->dirty_rgn);
|
|
}
|
|
else
|
|
break;
|
|
|
|
composite_opaque_win_znode (ctxt, next);
|
|
composite_all_lucent_win_znodes_above (ctxt, &tmp_rgn, next);
|
|
|
|
next = ServerGetNextZNode (ctxt->layer, next, NULL);
|
|
}
|
|
|
|
/* compositing the wallpaper */
|
|
if (!IsEmptyClipRgn (&ctxt->dirty_rgn)) {
|
|
composite_wallpaper (ctxt);
|
|
|
|
/* composite all lucent znodes intersected with dirty region. */
|
|
CopyRegion (&tmp_rgn, &ctxt->dirty_rgn);
|
|
composite_all_lucent_win_znodes_above (ctxt, &tmp_rgn, 0);
|
|
}
|
|
|
|
done:
|
|
EmptyClipRgn (&tmp_rgn);
|
|
}
|
|
|
|
static void rebuild_wins_region (CompositorCtxt* ctxt)
|
|
{
|
|
int i, nr_ppps;
|
|
const ZNODEHEADER* znode_hdr = NULL;
|
|
|
|
SetClipRgn (&ctxt->wins_rgn, &ctxt->rc_screen);
|
|
|
|
nr_ppps = ServerGetPopupMenusCount ();
|
|
for (i = 0; i < nr_ppps; i++) {
|
|
CLIPRGN* rgn;
|
|
znode_hdr = ServerGetPopupMenuZNodeHeader (i, (void**)&rgn, FALSE);
|
|
assert (znode_hdr);
|
|
assert (rgn);
|
|
|
|
SubtractRegion (&ctxt->wins_rgn, &ctxt->wins_rgn, rgn);
|
|
}
|
|
}
|
|
|
|
static void calc_mainwin_pos (CompositorCtxt* ctxt, MG_Layer* layer,
|
|
DWORD zt_type, int first_for_type, int cli, CALCPOSINFO* info)
|
|
{
|
|
/* give a default size first */
|
|
if (IsRectEmpty (&info->rc)) {
|
|
|
|
info->rc.left = 0;
|
|
info->rc.top = 0;
|
|
if (info->ex_style & WS_EX_DIALOGBOX) {
|
|
info->rc.right = g_rcScr.right / 2;
|
|
info->rc.bottom = g_rcScr.bottom / 3;
|
|
}
|
|
else {
|
|
info->rc.right = g_rcScr.right / 2;
|
|
info->rc.bottom = g_rcScr.bottom;
|
|
}
|
|
}
|
|
|
|
if (info->ex_style & WS_EX_DIALOGBOX) {
|
|
// center the window vertically and horinzontally
|
|
int width = info->rc.right - info->rc.left;
|
|
int height = info->rc.bottom - info->rc.top;
|
|
|
|
OffsetRect (&info->rc, (g_rcScr.right - width) >> 1,
|
|
(g_rcScr.bottom - height) >> 1);
|
|
}
|
|
else {
|
|
if (first_for_type == 0) {
|
|
info->rc.left = 0;
|
|
info->rc.top = 0;
|
|
}
|
|
else {
|
|
const ZNODEHEADER* znode_hdr;
|
|
znode_hdr =
|
|
ServerGetWinZNodeHeader (layer, first_for_type, NULL, FALSE);
|
|
|
|
info->rc = znode_hdr->rc;
|
|
OffsetRect (&info->rc,
|
|
DEF_OVERLAPPED_OFFSET_X, DEF_OVERLAPPED_OFFSET_Y);
|
|
|
|
/* adjust to a reasonable postion */
|
|
if (info->rc.top > (g_rcScr.bottom * 3 / 4)) {
|
|
OffsetRect (&info->rc, 0, -info->rc.top);
|
|
}
|
|
|
|
if (info->rc.left > (g_rcScr.right * 3 / 4)) {
|
|
OffsetRect (&info->rc, -info->rc.left, 0);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void purge_ppp_data (CompositorCtxt* ctxt, int zidx, void* data)
|
|
{
|
|
EmptyClipRgn ((CLIPRGN*)data);
|
|
mg_slice_delete (CLIPRGN, data);
|
|
}
|
|
|
|
static void purge_win_data (CompositorCtxt* ctxt, MG_Layer* layer,
|
|
int zidx, void* data)
|
|
{
|
|
EmptyClipRgn ((CLIPRGN*)data);
|
|
mg_slice_delete (CLIPRGN, data);
|
|
}
|
|
|
|
static void on_showing_ppp (CompositorCtxt* ctxt, int zidx)
|
|
{
|
|
CLIPRGN* rgn;
|
|
|
|
rgn = mg_slice_new (CLIPRGN);
|
|
InitClipRgn (rgn, &ctxt->cliprc_heap);
|
|
ServerGetPopupMenuZNodeRegion (zidx, RGN_OP_SET | RGN_OP_FLAG_ABS, rgn);
|
|
ServerSetPopupMenuZNodePrivateData (zidx, rgn);
|
|
|
|
/* subtract region of this ppp from wins_rgn */
|
|
SubtractRegion (&ctxt->wins_rgn, &ctxt->wins_rgn, rgn);
|
|
}
|
|
|
|
static BOOL reset_dirty_region (CompositorCtxt* ctxt, MG_Layer* layer)
|
|
{
|
|
if (layer != mgTopmostLayer)
|
|
return FALSE;
|
|
|
|
ctxt->dirty_types = 0;
|
|
EmptyClipRgn (&ctxt->dirty_rgn);
|
|
return TRUE;
|
|
}
|
|
|
|
static BOOL merge_dirty_region (CompositorCtxt* ctxt,
|
|
const ZNODEHEADER* znode_hdr, const CLIPRGN* znode_rgn)
|
|
{
|
|
EmptyClipRgn (&ctxt->inv_rgn);
|
|
if (znode_hdr && znode_hdr->dirty_rcs) {
|
|
int i;
|
|
for (i = 0; i < znode_hdr->nr_dirty_rcs; i++) {
|
|
RECT rc;
|
|
|
|
// device coordinates to screen coordinates
|
|
rc = znode_hdr->dirty_rcs [i];
|
|
OffsetRect (&rc, znode_hdr->rc.left, znode_hdr->rc.top);
|
|
AddClipRect (&ctxt->inv_rgn, &rc);
|
|
}
|
|
|
|
if (znode_rgn)
|
|
IntersectRegion (&ctxt->inv_rgn, &ctxt->inv_rgn, znode_rgn);
|
|
else
|
|
IntersectClipRect (&ctxt->inv_rgn, &ctxt->rc_screen);
|
|
|
|
}
|
|
|
|
if (IsEmptyClipRgn (&ctxt->inv_rgn)) {
|
|
return FALSE;
|
|
}
|
|
|
|
UnionRegion (&ctxt->dirty_rgn, &ctxt->dirty_rgn, &ctxt->inv_rgn);
|
|
EmptyClipRgn (&ctxt->inv_rgn);
|
|
return TRUE;
|
|
}
|
|
|
|
static BOOL merge_dirty_ppp (CompositorCtxt* ctxt, MG_Layer* layer, int zidx)
|
|
{
|
|
BOOL rc = FALSE;
|
|
const ZNODEHEADER* znode_hdr;
|
|
CLIPRGN* rgn;
|
|
|
|
_DBG_PRINTF("called: %d\n", zidx);
|
|
|
|
/* the fallback compositor only manages znodes on the topmost layer. */
|
|
if (layer != mgTopmostLayer)
|
|
return rc;
|
|
|
|
znode_hdr = ServerGetPopupMenuZNodeHeader (zidx, (void**)&rgn, TRUE);
|
|
if (merge_dirty_region (ctxt, znode_hdr, rgn)) {
|
|
ctxt->dirty_types |= DIRTY_ZT_PPP;
|
|
rc = TRUE;
|
|
}
|
|
ServerReleasePopupMenuZNodeHeader (zidx);
|
|
|
|
return rc;
|
|
}
|
|
|
|
static BOOL merge_dirty_win (CompositorCtxt* ctxt, MG_Layer* layer, int zidx)
|
|
{
|
|
BOOL rc = FALSE;
|
|
const ZNODEHEADER* znode_hdr = NULL;
|
|
CLIPRGN *rgn;
|
|
|
|
/* the fallback compositor only manages znodes on the topmost layer. */
|
|
if (layer != mgTopmostLayer)
|
|
return rc;
|
|
|
|
_DBG_PRINTF("called: %d\n", zidx);
|
|
|
|
/* merge the dirty region */
|
|
znode_hdr = ServerGetWinZNodeHeader (NULL, zidx, (void**)&rgn, TRUE);
|
|
if (merge_dirty_region (ctxt, znode_hdr, rgn)) {
|
|
ctxt->dirty_types |= DIRTY_ZT_WIN;
|
|
rc = TRUE;
|
|
}
|
|
ServerReleaseWinZNodeHeader (NULL, zidx);
|
|
return rc;
|
|
}
|
|
|
|
static BOOL merge_dirty_wpp (CompositorCtxt* ctxt, MG_Layer* layer)
|
|
{
|
|
BOOL rc = FALSE;
|
|
const ZNODEHEADER* znode_hdr;
|
|
|
|
znode_hdr = ServerGetWinZNodeHeader (NULL, 0, NULL, TRUE);
|
|
if (merge_dirty_region (ctxt, znode_hdr, NULL)) {
|
|
tile_dirty_region_for_wallpaper (ctxt);
|
|
//subtract_opaque_win_znodes_above (ctxt, 0);
|
|
ctxt->dirty_types |= DIRTY_ZT_WPP;
|
|
rc = TRUE;
|
|
}
|
|
|
|
ServerReleaseWinZNodeHeader (NULL, 0);
|
|
return rc;
|
|
}
|
|
|
|
static BOOL refresh_dirty_region (CompositorCtxt* ctxt, MG_Layer* layer)
|
|
{
|
|
/* the fallback compositor only manages znodes on the topmost layer. */
|
|
if (layer != mgTopmostLayer || IsEmptyClipRgn (&ctxt->dirty_rgn))
|
|
return FALSE;
|
|
|
|
if (ctxt->dirty_types & DIRTY_ZT_PPP)
|
|
composite_ppp_znodes (ctxt);
|
|
composite_on_dirty_region (ctxt, 0);
|
|
EmptyClipRgn (&ctxt->dirty_rgn);
|
|
|
|
SyncUpdateDC (HDC_SCREEN_SYS);
|
|
return TRUE;
|
|
}
|
|
|
|
static void on_hiding_ppp (CompositorCtxt* ctxt, int zidx)
|
|
{
|
|
const ZNODEHEADER* znode_hdr;
|
|
CLIPRGN* rgn = NULL;
|
|
|
|
znode_hdr = ServerGetPopupMenuZNodeHeader (zidx, (void**)&rgn, FALSE);
|
|
assert (znode_hdr);
|
|
|
|
assert (rgn);
|
|
EmptyClipRgn ((CLIPRGN*)rgn);
|
|
mg_slice_delete (CLIPRGN, rgn);
|
|
ServerSetPopupMenuZNodePrivateData (zidx, NULL);
|
|
}
|
|
|
|
static void on_dirty_screen (CompositorCtxt* ctxt,
|
|
MG_Layer* layer, DWORD cause_type, const RECT* rc_dirty)
|
|
{
|
|
/* this compositor only manages znodes on the topmost layer. */
|
|
if (layer == NULL)
|
|
layer = mgTopmostLayer;
|
|
else if (layer != mgTopmostLayer)
|
|
return;
|
|
|
|
_DBG_PRINTF ("called\n");
|
|
/* generate the dirty region */
|
|
if (rc_dirty) {
|
|
SetClipRgn (&ctxt->dirty_rgn, &ctxt->rc_screen);
|
|
IntersectClipRect (&ctxt->dirty_rgn, rc_dirty);
|
|
if (IsEmptyClipRgn (&ctxt->dirty_rgn)) {
|
|
return;
|
|
}
|
|
}
|
|
else {
|
|
// all screen is dirty.
|
|
SetClipRgn (&ctxt->dirty_rgn, &ctxt->rc_screen);
|
|
}
|
|
|
|
/* if the cause type is popup menu:
|
|
* - rebuild the region for windows;
|
|
* - composite all popup menus.
|
|
*
|
|
* note that if any popup menu was composited to the screen,
|
|
* the dirty region will be shrank, because this compositor
|
|
* handles any popup menu as opaque.
|
|
*/
|
|
if (cause_type == ZNIT_POPUPMENU) {
|
|
rebuild_wins_region (ctxt);
|
|
composite_ppp_znodes (ctxt);
|
|
}
|
|
|
|
composite_on_dirty_region (ctxt, 0);
|
|
EmptyClipRgn (&ctxt->dirty_rgn);
|
|
|
|
SyncUpdateDC (HDC_SCREEN_SYS);
|
|
}
|
|
|
|
static void refresh (CompositorCtxt* ctxt)
|
|
{
|
|
rebuild_wins_region (ctxt);
|
|
on_dirty_screen (ctxt, NULL, ZNIT_NULL, NULL);
|
|
}
|
|
|
|
static void on_closed_menu (CompositorCtxt* ctxt, const RECT* rc_bound)
|
|
{
|
|
_DBG_PRINTF ("called\n");
|
|
/* reset the wins_rgn with screen rectangle, because the menu closed */
|
|
SetClipRgn (&ctxt->wins_rgn, &ctxt->rc_screen);
|
|
|
|
/* composite the dirty screen rectangle.
|
|
* we pass cause_type with ZNIT_NULL for no any popup menu currently. */
|
|
on_dirty_screen (ctxt, NULL, ZNIT_NULL, rc_bound);
|
|
}
|
|
|
|
static void on_showing_win (CompositorCtxt* ctxt, MG_Layer* layer, int zidx)
|
|
{
|
|
CLIPRGN* rgn = NULL;
|
|
|
|
_DBG_PRINTF ("called: %d for layer %s\n", zidx, layer->name);
|
|
|
|
ServerGetWinZNodeHeader (layer, zidx, (void**)&rgn, FALSE);
|
|
if (rgn == NULL) {
|
|
rgn = mg_slice_new (CLIPRGN);
|
|
InitClipRgn (rgn, &ctxt->cliprc_heap);
|
|
ServerGetWinZNodeRegion (layer, zidx, RGN_OP_SET | RGN_OP_FLAG_ABS, rgn);
|
|
ServerSetWinZNodePrivateData (layer, zidx, rgn);
|
|
}
|
|
}
|
|
|
|
static void on_hiding_win (CompositorCtxt* ctxt, MG_Layer* layer, int zidx)
|
|
{
|
|
const ZNODEHEADER* znode_hdr;
|
|
CLIPRGN* rgn = NULL;
|
|
|
|
_DBG_PRINTF ("called: %d for layer %s\n", zidx, layer->name);
|
|
|
|
znode_hdr = ServerGetWinZNodeHeader (layer, zidx, (void**)&rgn, FALSE);
|
|
assert (znode_hdr);
|
|
|
|
if (rgn) {
|
|
EmptyClipRgn (rgn);
|
|
mg_slice_delete (CLIPRGN, rgn);
|
|
ServerSetWinZNodePrivateData (layer, zidx, NULL);
|
|
}
|
|
}
|
|
|
|
/* for both on_raised_win and on_changed_ct */
|
|
static void refresh_win (CompositorCtxt* ctxt, MG_Layer* layer, int zidx)
|
|
{
|
|
const ZNODEHEADER* znode_hdr;
|
|
CLIPRGN* rgn = NULL;
|
|
|
|
// the fallback compositor only manages znodes on the topmost layer.
|
|
if (layer != mgTopmostLayer)
|
|
return;
|
|
|
|
_DBG_PRINTF ("called\n");
|
|
znode_hdr = ServerGetWinZNodeHeader (layer, zidx, (void**)&rgn, FALSE);
|
|
assert(znode_hdr);
|
|
|
|
if (!(znode_hdr->flags & ZNIF_VISIBLE)) {
|
|
return;
|
|
}
|
|
|
|
assert (rgn);
|
|
CopyRegion (&ctxt->dirty_rgn, rgn);
|
|
if (IntersectClipRect (&ctxt->dirty_rgn, &ctxt->rc_screen)) {
|
|
composite_on_dirty_region (ctxt, zidx);
|
|
SyncUpdateDC (HDC_SCREEN_SYS);
|
|
}
|
|
EmptyClipRgn (&ctxt->dirty_rgn);
|
|
}
|
|
|
|
static void on_moved_win (CompositorCtxt* ctxt, MG_Layer* layer, int zidx,
|
|
const RECT* rc_org)
|
|
{
|
|
const ZNODEHEADER* znode_hdr;
|
|
CLIPRGN* rgn = NULL;
|
|
|
|
znode_hdr = ServerGetWinZNodeHeader (layer, zidx, (void**)&rgn, FALSE);
|
|
assert (znode_hdr);
|
|
|
|
if (!(znode_hdr->flags & ZNIF_VISIBLE)) {
|
|
return;
|
|
}
|
|
|
|
assert (rgn);
|
|
CopyRegion (&ctxt->dirty_rgn, rgn);
|
|
|
|
// re-generate the region
|
|
ServerGetWinZNodeRegion (layer, zidx, RGN_OP_SET | RGN_OP_FLAG_ABS, rgn);
|
|
ServerSetWinZNodePrivateData (layer, zidx, rgn);
|
|
|
|
// the fallback compositor only manages znodes on the topmost layer.
|
|
if (layer != mgTopmostLayer)
|
|
goto done;
|
|
|
|
UnionRegion (&ctxt->dirty_rgn, &ctxt->dirty_rgn, rgn);
|
|
|
|
/* calculate the dirty region */
|
|
if (IntersectClipRect (&ctxt->dirty_rgn, &ctxt->rc_screen)) {
|
|
composite_on_dirty_region (ctxt, zidx);
|
|
SyncUpdateDC (HDC_SCREEN_SYS);
|
|
}
|
|
|
|
done:
|
|
EmptyClipRgn (&ctxt->dirty_rgn);
|
|
}
|
|
|
|
static void on_changed_rgn (CompositorCtxt* ctxt, MG_Layer* layer,
|
|
int zidx, const RECT* rc_org_bound)
|
|
{
|
|
const ZNODEHEADER* znode_hdr;
|
|
CLIPRGN* rgn = NULL;
|
|
|
|
/* update the region of the current window znode. */
|
|
znode_hdr = ServerGetWinZNodeHeader (layer, zidx, (void**)&rgn, FALSE);
|
|
assert (znode_hdr);
|
|
|
|
if (!(znode_hdr->flags & ZNIF_VISIBLE)) {
|
|
_DBG_PRINTF("not a visible window\n");
|
|
return;
|
|
}
|
|
|
|
assert (rgn);
|
|
CopyRegion (&ctxt->dirty_rgn, rgn);
|
|
|
|
// re-generate the region
|
|
ServerGetWinZNodeRegion (layer, zidx, RGN_OP_SET | RGN_OP_FLAG_ABS, rgn);
|
|
ServerSetWinZNodePrivateData (layer, zidx, rgn);
|
|
|
|
/* the fallback compositor only manages znodes on the topmost layer. */
|
|
if (layer != mgTopmostLayer)
|
|
goto done;
|
|
|
|
/* calculate the dirty region */
|
|
UnionRegion (&ctxt->dirty_rgn, &ctxt->dirty_rgn, rgn);
|
|
if (IntersectClipRect (&ctxt->dirty_rgn, &ctxt->rc_screen)) {
|
|
composite_on_dirty_region (ctxt, zidx);
|
|
SyncUpdateDC (HDC_SCREEN_SYS);
|
|
}
|
|
|
|
done:
|
|
EmptyClipRgn (&ctxt->dirty_rgn);
|
|
}
|
|
|
|
static void on_layer_op (CompositorCtxt* ctxt, int layer_op,
|
|
MG_Layer* layer, MG_Client* client)
|
|
{
|
|
if (layer_op == LCO_TOPMOST_CHANGED) {
|
|
refresh (ctxt);
|
|
}
|
|
}
|
|
|
|
static int offset_regions_of_general_znodes (MG_Layer* layer, int offx, int offy)
|
|
{
|
|
int nr = 0;
|
|
int next;
|
|
|
|
next = ServerGetNextZNode (layer, 0, NULL);
|
|
while (next > 0) {
|
|
const ZNODEHEADER* znode_hdr;
|
|
CLIPRGN* rgn;
|
|
|
|
znode_hdr = ServerGetWinZNodeHeader (layer, next, (void**)&rgn, FALSE);
|
|
assert (znode_hdr);
|
|
|
|
if ((znode_hdr->flags & ZNIF_VISIBLE) &&
|
|
!(znode_hdr->flags & ZOF_IF_SPECIAL)) {
|
|
OffsetRegion (rgn, offx, offy);
|
|
nr++;
|
|
}
|
|
|
|
next = ServerGetNextZNode (layer, next, NULL);
|
|
}
|
|
|
|
return nr;
|
|
}
|
|
|
|
static unsigned int composite_layers (CompositorCtxt* ctxt, MG_Layer* layers[],
|
|
int nr_layers, void* combine_param)
|
|
{
|
|
COMBPARAMS_FALLBACK *cp = (COMBPARAMS_FALLBACK *)combine_param;
|
|
RECT rc0, rc1; /* bounding rectangles for two layers */
|
|
struct timespec ts_start, ts_end;
|
|
|
|
assert (nr_layers == 2);
|
|
|
|
if (cp->percent < 0 || cp->percent > 100) {
|
|
_WRN_PRINTF("Bad percent (%d).\n", cp->percent);
|
|
return 0;
|
|
}
|
|
|
|
clock_gettime (CLOCK_MONOTONIC, &ts_start);
|
|
|
|
rc0 = ctxt->rc_screen;
|
|
rc1 = ctxt->rc_screen;
|
|
if (cp->method == FCM_VERTICAL) {
|
|
rc0.bottom = cp->percent * RECTH (rc0) / 100;
|
|
rc1.top = rc0.bottom;
|
|
}
|
|
else {
|
|
rc0.right = cp->percent * RECTW (rc0) / 100;
|
|
rc1.left = rc0.right;
|
|
}
|
|
|
|
SetClipRgn (&ctxt->wins_rgn, &rc0);
|
|
SetClipRgn (&ctxt->dirty_rgn, &rc0);
|
|
|
|
if (layers[0]) {
|
|
// composite the first layer
|
|
ctxt->layer = layers[0];
|
|
ctxt->offx = rc0.right - ctxt->rc_screen.right;
|
|
ctxt->offy = rc0.bottom - ctxt->rc_screen.bottom;
|
|
|
|
if (ctxt->offx || ctxt->offy)
|
|
offset_regions_of_general_znodes (ctxt->layer, ctxt->offx, ctxt->offy);
|
|
composite_on_dirty_region (ctxt, 0);
|
|
if (ctxt->offx || ctxt->offy)
|
|
offset_regions_of_general_znodes (ctxt->layer, -ctxt->offx, -ctxt->offy);
|
|
}
|
|
else {
|
|
// only composite the wallpaper
|
|
composite_wallpaper (ctxt);
|
|
}
|
|
|
|
SetClipRgn (&ctxt->wins_rgn, &rc1);
|
|
SetClipRgn (&ctxt->dirty_rgn, &rc1);
|
|
|
|
if (layers[1]) {
|
|
// composite the second layer
|
|
ctxt->layer = layers[1];
|
|
ctxt->offx = rc1.left;
|
|
ctxt->offy = rc1.top;
|
|
|
|
if (ctxt->offx || ctxt->offy)
|
|
offset_regions_of_general_znodes (ctxt->layer, ctxt->offx, ctxt->offy);
|
|
composite_on_dirty_region (ctxt, 0);
|
|
if (ctxt->offx || ctxt->offy)
|
|
offset_regions_of_general_znodes (ctxt->layer, -ctxt->offx, -ctxt->offy);
|
|
}
|
|
else {
|
|
// only composite the wallpaper
|
|
composite_wallpaper (ctxt);
|
|
}
|
|
|
|
SyncUpdateDC (HDC_SCREEN_SYS);
|
|
|
|
clock_gettime (CLOCK_MONOTONIC, &ts_end);
|
|
|
|
// reset fields for normal compositing
|
|
EmptyClipRgn (&ctxt->wins_rgn);
|
|
EmptyClipRgn (&ctxt->dirty_rgn);
|
|
|
|
ctxt->layer = NULL;
|
|
ctxt->offx = 0;
|
|
ctxt->offy = 0;
|
|
|
|
return (ts_end.tv_sec - ts_start.tv_sec) * 1000 +
|
|
(ts_end.tv_nsec - ts_start.tv_nsec) / 1000000;
|
|
}
|
|
|
|
/*
|
|
* The fallback implementation is very simple.
|
|
* You can override this operation in your mginit to use mGEff for
|
|
* a better animation timing.
|
|
*/
|
|
static void transit_to_layer (CompositorCtxt* ctxt, MG_Layer* to_layer)
|
|
{
|
|
MG_Layer* layer = mgLayers;
|
|
int idx_layer_0 = -1;
|
|
int idx_layer_1 = -1;
|
|
int i = 0, nr_layers = 0;
|
|
|
|
MG_Layer* layers[2];
|
|
|
|
COMBPARAMS_FALLBACK cp = { FCM_HORIZONTAL, 5 };
|
|
|
|
while (layer) {
|
|
if (layer == mgTopmostLayer)
|
|
idx_layer_0 = i;
|
|
if (layer == to_layer)
|
|
idx_layer_1 = i;
|
|
|
|
layer = layer->next;
|
|
i++;
|
|
nr_layers++;
|
|
}
|
|
|
|
if (idx_layer_0 == idx_layer_1 || idx_layer_0 < 0 || idx_layer_1 < 0) {
|
|
_WRN_PRINTF("Bad layers to transit to (%d, %d).\n",
|
|
idx_layer_0, idx_layer_1);
|
|
return;
|
|
}
|
|
|
|
idx_layer_0 = nr_layers - 1 - idx_layer_0;
|
|
idx_layer_1 = nr_layers - 1 - idx_layer_1;
|
|
|
|
if (idx_layer_1 > idx_layer_0) {
|
|
/* to_layer is to the right of the topmost layer */
|
|
layers[0] = mgTopmostLayer;
|
|
layers[1] = to_layer;
|
|
|
|
cp.percent = 95;
|
|
while (cp.percent > 5) {
|
|
unsigned int t_ms;
|
|
|
|
t_ms = composite_layers (ctxt, layers, 2, &cp);
|
|
if (t_ms > 100)
|
|
_WRN_PRINTF ("Two slow to composite layers: %u\n", t_ms);
|
|
|
|
usleep (20 * 1000);
|
|
cp.percent -= 9;
|
|
}
|
|
}
|
|
else {
|
|
/* to_layer is to the left of the topmost layer */
|
|
layers[0] = to_layer;
|
|
layers[1] = mgTopmostLayer;
|
|
|
|
cp.percent = 5;
|
|
while (cp.percent < 95) {
|
|
unsigned int t_ms;
|
|
|
|
t_ms = composite_layers (ctxt, layers, 2, &cp);
|
|
if (t_ms > 100)
|
|
_WRN_PRINTF ("Two slow to composite layers: %u\n", t_ms);
|
|
|
|
usleep (20 * 1000);
|
|
cp.percent += 9;
|
|
}
|
|
}
|
|
}
|
|
|
|
CompositorOps __mg_fallback_compositor = {
|
|
initialize: initialize,
|
|
terminate: terminate,
|
|
refresh: refresh,
|
|
calc_mainwin_pos: calc_mainwin_pos,
|
|
purge_ppp_data: purge_ppp_data,
|
|
purge_win_data: purge_win_data,
|
|
reset_dirty_region: reset_dirty_region,
|
|
merge_dirty_ppp: merge_dirty_ppp,
|
|
merge_dirty_win: merge_dirty_win,
|
|
merge_dirty_wpp: merge_dirty_wpp,
|
|
refresh_dirty_region: refresh_dirty_region,
|
|
on_dirty_screen: on_dirty_screen,
|
|
on_showing_ppp: on_showing_ppp,
|
|
on_hiding_ppp: on_hiding_ppp,
|
|
on_closed_menu: on_closed_menu,
|
|
on_showing_win: on_showing_win,
|
|
on_hiding_win: on_hiding_win,
|
|
on_raised_win: refresh_win,
|
|
on_changed_ct: refresh_win,
|
|
on_changed_rgn: on_changed_rgn,
|
|
on_moved_win: on_moved_win,
|
|
on_layer_op: on_layer_op,
|
|
composite_layers: composite_layers,
|
|
transit_to_layer: transit_to_layer,
|
|
};
|
|
|
|
#endif /* defined(_MGRM_PROCESSES) && defined(_MGSCHEMA_COMPOSITING) */
|
|
|
|
#if 0 /* deprecated code */
|
|
/*
|
|
* This helper is useful for transparent or blur popup menus
|
|
* The fallback compositor only supports opaque popup menus.
|
|
*/
|
|
static void subtract_opaque_ppp_znodes (CompositorCtxt* ctxt)
|
|
{
|
|
int i, nr_ppp;
|
|
|
|
// subtract opaque menu znodes
|
|
nr_ppp = ServerGetPopupMenusCount();
|
|
for (i = 0; i < nr_ppp; i++) {
|
|
const ZNODEHEADER* znode_hdr;
|
|
CLIPRGN* rgn;
|
|
|
|
znode_hdr = ServerGetPopupMenuZNodeHeader (i, (void**)rgn, FALSE);
|
|
if (znode_hdr && znode_hdr->ct == CT_OPAQUE) {
|
|
SubtractRegion (&ctxt->dirty_rgn, &ctxt->dirty_rgn, rgn);
|
|
}
|
|
}
|
|
}
|
|
|
|
static inline int get_lucent_win_znodes_above (CompositorCtxt* ctxt, int from)
|
|
{
|
|
int nr_lucent = 0;
|
|
int prev;
|
|
|
|
/* subtract opaque window znodes */
|
|
if (from > 0)
|
|
prev = from;
|
|
else
|
|
prev = ServerGetPrevZNode (NULL, 0, NULL);
|
|
|
|
while (prev > 0) {
|
|
const ZNODEHEADER* znode_hdr;
|
|
CLIPRGN* rgn;
|
|
|
|
znode_hdr = ServerGetWinZNodeHeader (NULL, prev, (void**)&rgn, FALSE);
|
|
assert (znode_hdr);
|
|
|
|
if ((znode_hdr->flags & ZNIF_VISIBLE) &&
|
|
(znode_hdr->ct & CT_SYSTEM_MASK) != CT_OPAQUE &&
|
|
AreRegionsIntersected (&ctxt->dirty_rgn, rgn)) {
|
|
nr_lucent++;
|
|
}
|
|
|
|
prev = ServerGetPrevZNode (NULL, prev, NULL);
|
|
}
|
|
|
|
return nr_lucent;
|
|
}
|
|
|
|
static inline void subtract_all_opaque_win_znodes (CompositorCtxt* ctxt,
|
|
CLIPRGN* dirty_rgn)
|
|
{
|
|
int next;
|
|
|
|
if (IsEmptyClipRgn (dirty_rgn)) {
|
|
return;
|
|
}
|
|
|
|
next = ServerGetNextZNode (NULL, 0, NULL);
|
|
while (next > 0) {
|
|
const ZNODEHEADER* znode_hdr;
|
|
CLIPRGN* rgn;
|
|
|
|
znode_hdr = ServerGetWinZNodeHeader (NULL, next, (void**)&rgn, FALSE);
|
|
assert (znode_hdr);
|
|
|
|
if ((znode_hdr->flags & ZNIF_VISIBLE) &&
|
|
(znode_hdr->ct & CT_SYSTEM_MASK) == CT_OPAQUE) {
|
|
|
|
SubtractRegion (dirty_rgn, dirty_rgn, rgn);
|
|
|
|
if (IsEmptyClipRgn (dirty_rgn)) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
next = ServerGetNextZNode (NULL, next, NULL);
|
|
}
|
|
}
|
|
|
|
#endif /* deprecated code */
|
|
|