256 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			256 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			C
		
	
	
	
/*
 | 
						|
 | 
						|
Copyright 1993, 1998  The Open Group
 | 
						|
 | 
						|
Permission to use, copy, modify, distribute, and sell this software and its
 | 
						|
documentation for any purpose is hereby granted without fee, provided that
 | 
						|
the above copyright notice appear in all copies and that both that
 | 
						|
copyright notice and this permission notice appear in supporting
 | 
						|
documentation.
 | 
						|
 | 
						|
The above copyright notice and this permission notice shall be included
 | 
						|
in all copies or substantial portions of the Software.
 | 
						|
 | 
						|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
 | 
						|
OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
 | 
						|
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
 | 
						|
IN NO EVENT SHALL THE OPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR
 | 
						|
OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
 | 
						|
ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
 | 
						|
OTHER DEALINGS IN THE SOFTWARE.
 | 
						|
 | 
						|
Except as contained in this notice, the name of The Open Group shall
 | 
						|
not be used in advertising or otherwise to promote the sale, use or
 | 
						|
other dealings in this Software without prior written authorization
 | 
						|
from The Open Group.
 | 
						|
 | 
						|
*/
 | 
						|
 | 
						|
 | 
						|
#ifdef HAVE_DIX_CONFIG_H
 | 
						|
#include <dix-config.h>
 | 
						|
#endif
 | 
						|
 | 
						|
#include "scrnintstr.h"
 | 
						|
#include "gcstruct.h"
 | 
						|
#include "pixmapstr.h"
 | 
						|
#include "windowstr.h"
 | 
						|
#include "migc.h"
 | 
						|
 | 
						|
/* ARGSUSED */
 | 
						|
void
 | 
						|
miChangeGC(GCPtr pGC, unsigned long mask)
 | 
						|
{
 | 
						|
    return;
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
miDestroyGC(GCPtr pGC)
 | 
						|
{
 | 
						|
    if (pGC->pRotatedPixmap)
 | 
						|
	(*pGC->pScreen->DestroyPixmap) (pGC->pRotatedPixmap);
 | 
						|
    if (pGC->freeCompClip)
 | 
						|
	REGION_DESTROY(pGC->pScreen, pGC->pCompositeClip);
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
miDestroyClip(GCPtr pGC)
 | 
						|
{
 | 
						|
    if (pGC->clientClipType == CT_NONE)
 | 
						|
	return;
 | 
						|
    else if (pGC->clientClipType == CT_PIXMAP)
 | 
						|
    {
 | 
						|
	(*pGC->pScreen->DestroyPixmap) ((PixmapPtr) (pGC->clientClip));
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
	/*
 | 
						|
	 * we know we'll never have a list of rectangles, since ChangeClip
 | 
						|
	 * immediately turns them into a region
 | 
						|
	 */
 | 
						|
	REGION_DESTROY(pGC->pScreen, pGC->clientClip);
 | 
						|
    }
 | 
						|
    pGC->clientClip = NULL;
 | 
						|
    pGC->clientClipType = CT_NONE;
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
miChangeClip( GCPtr pGC, int type, pointer pvalue, int nrects)
 | 
						|
{
 | 
						|
    (*pGC->funcs->DestroyClip) (pGC);
 | 
						|
    if (type == CT_PIXMAP)
 | 
						|
    {
 | 
						|
	/* convert the pixmap to a region */
 | 
						|
	pGC->clientClip = (pointer) BITMAP_TO_REGION(pGC->pScreen,
 | 
						|
							(PixmapPtr) pvalue);
 | 
						|
	(*pGC->pScreen->DestroyPixmap) (pvalue);
 | 
						|
    }
 | 
						|
    else if (type == CT_REGION)
 | 
						|
    {
 | 
						|
	/* stuff the region in the GC */
 | 
						|
	pGC->clientClip = pvalue;
 | 
						|
    }
 | 
						|
    else if (type != CT_NONE)
 | 
						|
    {
 | 
						|
	pGC->clientClip = (pointer) RECTS_TO_REGION(pGC->pScreen, nrects,
 | 
						|
						      (xRectangle *) pvalue,
 | 
						|
								    type);
 | 
						|
	xfree(pvalue);
 | 
						|
    }
 | 
						|
    pGC->clientClipType = (type != CT_NONE && pGC->clientClip) ? CT_REGION : CT_NONE;
 | 
						|
    pGC->stateChanges |= GCClipMask;
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
miCopyClip(GCPtr pgcDst, GCPtr pgcSrc)
 | 
						|
{
 | 
						|
    RegionPtr       prgnNew;
 | 
						|
 | 
						|
    switch (pgcSrc->clientClipType)
 | 
						|
    {
 | 
						|
      case CT_PIXMAP:
 | 
						|
	((PixmapPtr) pgcSrc->clientClip)->refcnt++;
 | 
						|
	/* Fall through !! */
 | 
						|
      case CT_NONE:
 | 
						|
	(*pgcDst->funcs->ChangeClip) (pgcDst, (int) pgcSrc->clientClipType,
 | 
						|
				   pgcSrc->clientClip, 0);
 | 
						|
	break;
 | 
						|
      case CT_REGION:
 | 
						|
	prgnNew = REGION_CREATE(pgcSrc->pScreen, NULL, 1);
 | 
						|
	REGION_COPY(pgcDst->pScreen, prgnNew,
 | 
						|
					(RegionPtr) (pgcSrc->clientClip));
 | 
						|
	(*pgcDst->funcs->ChangeClip) (pgcDst, CT_REGION, (pointer) prgnNew, 0);
 | 
						|
	break;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
/* ARGSUSED */
 | 
						|
void
 | 
						|
miCopyGC(GCPtr pGCSrc, unsigned long changes, GCPtr pGCDst)
 | 
						|
{
 | 
						|
    return;
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
miComputeCompositeClip( GCPtr pGC, DrawablePtr pDrawable)
 | 
						|
{
 | 
						|
    ScreenPtr       pScreen;
 | 
						|
 | 
						|
    /* This prevents warnings about pScreen not being used. */
 | 
						|
    pGC->pScreen = pScreen = pGC->pScreen;
 | 
						|
 | 
						|
    if (pDrawable->type == DRAWABLE_WINDOW)
 | 
						|
    {
 | 
						|
	WindowPtr       pWin = (WindowPtr) pDrawable;
 | 
						|
	RegionPtr       pregWin;
 | 
						|
	Bool            freeTmpClip, freeCompClip;
 | 
						|
 | 
						|
	if (pGC->subWindowMode == IncludeInferiors)
 | 
						|
	{
 | 
						|
	    pregWin = NotClippedByChildren(pWin);
 | 
						|
	    freeTmpClip = TRUE;
 | 
						|
	}
 | 
						|
	else
 | 
						|
	{
 | 
						|
	    pregWin = &pWin->clipList;
 | 
						|
	    freeTmpClip = FALSE;
 | 
						|
	}
 | 
						|
	freeCompClip = pGC->freeCompClip;
 | 
						|
 | 
						|
	/*
 | 
						|
	 * if there is no client clip, we can get by with just keeping the
 | 
						|
	 * pointer we got, and remembering whether or not should destroy (or
 | 
						|
	 * maybe re-use) it later.  this way, we avoid unnecessary copying of
 | 
						|
	 * regions.  (this wins especially if many clients clip by children
 | 
						|
	 * and have no client clip.)
 | 
						|
	 */
 | 
						|
	if (pGC->clientClipType == CT_NONE)
 | 
						|
	{
 | 
						|
	    if (freeCompClip)
 | 
						|
		REGION_DESTROY(pScreen, pGC->pCompositeClip);
 | 
						|
	    pGC->pCompositeClip = pregWin;
 | 
						|
	    pGC->freeCompClip = freeTmpClip;
 | 
						|
	}
 | 
						|
	else
 | 
						|
	{
 | 
						|
	    /*
 | 
						|
	     * we need one 'real' region to put into the composite clip. if
 | 
						|
	     * pregWin the current composite clip are real, we can get rid of
 | 
						|
	     * one. if pregWin is real and the current composite clip isn't,
 | 
						|
	     * use pregWin for the composite clip. if the current composite
 | 
						|
	     * clip is real and pregWin isn't, use the current composite
 | 
						|
	     * clip. if neither is real, create a new region.
 | 
						|
	     */
 | 
						|
 | 
						|
	    REGION_TRANSLATE(pScreen, pGC->clientClip,
 | 
						|
					 pDrawable->x + pGC->clipOrg.x,
 | 
						|
					 pDrawable->y + pGC->clipOrg.y);
 | 
						|
 | 
						|
	    if (freeCompClip)
 | 
						|
	    {
 | 
						|
		REGION_INTERSECT(pGC->pScreen, pGC->pCompositeClip,
 | 
						|
					    pregWin, pGC->clientClip);
 | 
						|
		if (freeTmpClip)
 | 
						|
		    REGION_DESTROY(pScreen, pregWin);
 | 
						|
	    }
 | 
						|
	    else if (freeTmpClip)
 | 
						|
	    {
 | 
						|
		REGION_INTERSECT(pScreen, pregWin, pregWin, pGC->clientClip);
 | 
						|
		pGC->pCompositeClip = pregWin;
 | 
						|
	    }
 | 
						|
	    else
 | 
						|
	    {
 | 
						|
		pGC->pCompositeClip = REGION_CREATE(pScreen, NullBox, 0);
 | 
						|
		REGION_INTERSECT(pScreen, pGC->pCompositeClip,
 | 
						|
				       pregWin, pGC->clientClip);
 | 
						|
	    }
 | 
						|
	    pGC->freeCompClip = TRUE;
 | 
						|
	    REGION_TRANSLATE(pScreen, pGC->clientClip,
 | 
						|
					 -(pDrawable->x + pGC->clipOrg.x),
 | 
						|
					 -(pDrawable->y + pGC->clipOrg.y));
 | 
						|
	}
 | 
						|
    }	/* end of composite clip for a window */
 | 
						|
    else
 | 
						|
    {
 | 
						|
	BoxRec          pixbounds;
 | 
						|
 | 
						|
	/* XXX should we translate by drawable.x/y here ? */
 | 
						|
	/* If you want pixmaps in offscreen memory, yes */
 | 
						|
	pixbounds.x1 = pDrawable->x;
 | 
						|
	pixbounds.y1 = pDrawable->y;
 | 
						|
	pixbounds.x2 = pDrawable->x + pDrawable->width;
 | 
						|
	pixbounds.y2 = pDrawable->y + pDrawable->height;
 | 
						|
 | 
						|
	if (pGC->freeCompClip)
 | 
						|
	{
 | 
						|
	    REGION_RESET(pScreen, pGC->pCompositeClip, &pixbounds);
 | 
						|
	}
 | 
						|
	else
 | 
						|
	{
 | 
						|
	    pGC->freeCompClip = TRUE;
 | 
						|
	    pGC->pCompositeClip = REGION_CREATE(pScreen, &pixbounds, 1);
 | 
						|
	}
 | 
						|
 | 
						|
	if (pGC->clientClipType == CT_REGION)
 | 
						|
	{
 | 
						|
	    if(pDrawable->x || pDrawable->y) {
 | 
						|
	        REGION_TRANSLATE(pScreen, pGC->clientClip,
 | 
						|
					  pDrawable->x + pGC->clipOrg.x, 
 | 
						|
					  pDrawable->y + pGC->clipOrg.y);
 | 
						|
	        REGION_INTERSECT(pScreen, pGC->pCompositeClip,
 | 
						|
				pGC->pCompositeClip, pGC->clientClip);
 | 
						|
	        REGION_TRANSLATE(pScreen, pGC->clientClip,
 | 
						|
					  -(pDrawable->x + pGC->clipOrg.x), 
 | 
						|
					  -(pDrawable->y + pGC->clipOrg.y));
 | 
						|
	    } else {
 | 
						|
	        REGION_TRANSLATE(pScreen, pGC->pCompositeClip,
 | 
						|
					 -pGC->clipOrg.x, -pGC->clipOrg.y);
 | 
						|
	        REGION_INTERSECT(pScreen, pGC->pCompositeClip,
 | 
						|
				pGC->pCompositeClip, pGC->clientClip);
 | 
						|
	        REGION_TRANSLATE(pScreen, pGC->pCompositeClip,
 | 
						|
					 pGC->clipOrg.x, pGC->clipOrg.y);
 | 
						|
	    }
 | 
						|
	}
 | 
						|
    }	/* end of composite clip for pixmap */
 | 
						|
} /* end miComputeCompositeClip */
 |