You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
2117 lines
53 KiB
2117 lines
53 KiB
/*
|
|
Copyright 2005-2012 Jay Sorg
|
|
|
|
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.
|
|
|
|
Xserver drawing ops and funcs
|
|
|
|
*/
|
|
|
|
#include "rdp.h"
|
|
#include "gcops.h"
|
|
|
|
#if 1
|
|
#define DEBUG_OUT_FUNCS(arg)
|
|
#define DEBUG_OUT_OPS(arg)
|
|
#else
|
|
#define DEBUG_OUT_FUNCS(arg) ErrorF arg
|
|
#define DEBUG_OUT_OPS(arg) ErrorF arg
|
|
#endif
|
|
|
|
extern rdpScreenInfoRec g_rdpScreen; /* from rdpmain.c */
|
|
extern DevPrivateKeyRec g_rdpGCIndex; /* from rdpmain.c */
|
|
extern DevPrivateKeyRec g_rdpWindowIndex; /* from rdpmain.c */
|
|
extern DevPrivateKeyRec g_rdpPixmapIndex; /* from rdpmain.c */
|
|
extern int g_Bpp; /* from rdpmain.c */
|
|
extern ScreenPtr g_pScreen; /* from rdpmain.c */
|
|
|
|
ColormapPtr g_rdpInstalledColormap;
|
|
|
|
static GCFuncs g_rdpGCFuncs =
|
|
{
|
|
rdpValidateGC, rdpChangeGC, rdpCopyGC, rdpDestroyGC, rdpChangeClip,
|
|
rdpDestroyClip, rdpCopyClip
|
|
};
|
|
|
|
static GCOps g_rdpGCOps =
|
|
{
|
|
rdpFillSpans, rdpSetSpans, rdpPutImage, rdpCopyArea, rdpCopyPlane,
|
|
rdpPolyPoint, rdpPolylines, rdpPolySegment, rdpPolyRectangle,
|
|
rdpPolyArc, rdpFillPolygon, rdpPolyFillRect, rdpPolyFillArc,
|
|
rdpPolyText8, rdpPolyText16, rdpImageText8, rdpImageText16,
|
|
rdpImageGlyphBlt, rdpPolyGlyphBlt, rdpPushPixels
|
|
};
|
|
|
|
/******************************************************************************/
|
|
/* return 0, draw nothing */
|
|
/* return 1, draw with no clip */
|
|
/* return 2, draw using clip */
|
|
static int
|
|
rdp_get_clip(RegionPtr pRegion, DrawablePtr pDrawable, GCPtr pGC)
|
|
{
|
|
WindowPtr pWindow;
|
|
RegionPtr temp;
|
|
BoxRec box;
|
|
int rv;
|
|
|
|
rv = 0;
|
|
if (pDrawable->type == DRAWABLE_WINDOW)
|
|
{
|
|
pWindow = (WindowPtr)pDrawable;
|
|
if (pWindow->viewable)
|
|
{
|
|
if (pGC->subWindowMode == IncludeInferiors)
|
|
{
|
|
temp = &pWindow->borderClip;
|
|
}
|
|
else
|
|
{
|
|
temp = &pWindow->clipList;
|
|
}
|
|
if (RegionNotEmpty(temp))
|
|
{
|
|
switch (pGC->clientClipType)
|
|
{
|
|
case CT_NONE:
|
|
rv = 2;
|
|
RegionCopy(pRegion, temp);
|
|
break;
|
|
case CT_REGION:
|
|
rv = 2;
|
|
RegionCopy(pRegion, pGC->clientClip);
|
|
RegionTranslate(pRegion,
|
|
pDrawable->x + pGC->clipOrg.x,
|
|
pDrawable->y + pGC->clipOrg.y);
|
|
RegionIntersect(pRegion, pRegion, temp);
|
|
break;
|
|
default:
|
|
rdpLog("unimp clip type %d\n", pGC->clientClipType);
|
|
break;
|
|
}
|
|
if (rv == 2) /* check if the clip is the entire screen */
|
|
{
|
|
box.x1 = 0;
|
|
box.y1 = 0;
|
|
box.x2 = g_rdpScreen.width;
|
|
box.y2 = g_rdpScreen.height;
|
|
if (RegionContainsRect(pRegion, &box) == rgnIN)
|
|
{
|
|
rv = 1;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
GetTextBoundingBox(DrawablePtr pDrawable, FontPtr font, int x, int y,
|
|
int n, BoxPtr pbox)
|
|
{
|
|
int maxAscent;
|
|
int maxDescent;
|
|
int maxCharWidth;
|
|
|
|
if (FONTASCENT(font) > FONTMAXBOUNDS(font, ascent))
|
|
{
|
|
maxAscent = FONTASCENT(font);
|
|
}
|
|
else
|
|
{
|
|
maxAscent = FONTMAXBOUNDS(font, ascent);
|
|
}
|
|
if (FONTDESCENT(font) > FONTMAXBOUNDS(font, descent))
|
|
{
|
|
maxDescent = FONTDESCENT(font);
|
|
}
|
|
else
|
|
{
|
|
maxDescent = FONTMAXBOUNDS(font, descent);
|
|
}
|
|
if (FONTMAXBOUNDS(font, rightSideBearing) >
|
|
FONTMAXBOUNDS(font, characterWidth))
|
|
{
|
|
maxCharWidth = FONTMAXBOUNDS(font, rightSideBearing);
|
|
}
|
|
else
|
|
{
|
|
maxCharWidth = FONTMAXBOUNDS(font, characterWidth);
|
|
}
|
|
pbox->x1 = pDrawable->x + x;
|
|
pbox->y1 = pDrawable->y + y - maxAscent;
|
|
pbox->x2 = pbox->x1 + maxCharWidth * n;
|
|
pbox->y2 = pbox->y1 + maxAscent + maxDescent;
|
|
if (FONTMINBOUNDS(font, leftSideBearing) < 0)
|
|
{
|
|
pbox->x1 += FONTMINBOUNDS(font, leftSideBearing);
|
|
}
|
|
}
|
|
|
|
/******************************************************************************/
|
|
#define GC_FUNC_PROLOGUE(_pGC) \
|
|
{ \
|
|
priv = (rdpGCPtr)(dixGetPrivateAddr(&(_pGC->devPrivates), &g_rdpGCIndex)); \
|
|
(_pGC)->funcs = priv->funcs; \
|
|
if (priv->ops != 0) \
|
|
{ \
|
|
(_pGC)->ops = priv->ops; \
|
|
} \
|
|
}
|
|
|
|
/******************************************************************************/
|
|
#define GC_FUNC_EPILOGUE(_pGC) \
|
|
{ \
|
|
priv->funcs = (_pGC)->funcs; \
|
|
(_pGC)->funcs = &g_rdpGCFuncs; \
|
|
if (priv->ops != 0) \
|
|
{ \
|
|
priv->ops = (_pGC)->ops; \
|
|
(_pGC)->ops = &g_rdpGCOps; \
|
|
} \
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpValidateGC(GCPtr pGC, unsigned long changes, DrawablePtr d)
|
|
{
|
|
rdpGCRec* priv;
|
|
int viewable;
|
|
RegionPtr pRegion;
|
|
|
|
DEBUG_OUT_FUNCS(("in rdpValidateGC\n"));
|
|
GC_FUNC_PROLOGUE(pGC);
|
|
pGC->funcs->ValidateGC(pGC, changes, d);
|
|
viewable = d->type == DRAWABLE_WINDOW && ((WindowPtr)d)->viewable;
|
|
if (viewable)
|
|
{
|
|
if (pGC->subWindowMode == IncludeInferiors)
|
|
{
|
|
pRegion = &(((WindowPtr)d)->borderClip);
|
|
}
|
|
else
|
|
{
|
|
pRegion = &(((WindowPtr)d)->clipList);
|
|
}
|
|
viewable = RegionNotEmpty(pRegion);
|
|
}
|
|
priv->ops = 0;
|
|
if (viewable)
|
|
{
|
|
priv->ops = pGC->ops;
|
|
}
|
|
GC_FUNC_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpChangeGC(GCPtr pGC, unsigned long mask)
|
|
{
|
|
rdpGCRec* priv;
|
|
|
|
DEBUG_OUT_FUNCS(("in rdpChangeGC\n"));
|
|
GC_FUNC_PROLOGUE(pGC);
|
|
pGC->funcs->ChangeGC(pGC, mask);
|
|
GC_FUNC_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpCopyGC(GCPtr src, unsigned long mask, GCPtr dst)
|
|
{
|
|
rdpGCRec* priv;
|
|
|
|
DEBUG_OUT_FUNCS(("in rdpCopyGC\n"));
|
|
GC_FUNC_PROLOGUE(dst);
|
|
dst->funcs->CopyGC(src, mask, dst);
|
|
GC_FUNC_EPILOGUE(dst);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpDestroyGC(GCPtr pGC)
|
|
{
|
|
rdpGCRec* priv;
|
|
|
|
DEBUG_OUT_FUNCS(("in rdpDestroyGC\n"));
|
|
GC_FUNC_PROLOGUE(pGC);
|
|
pGC->funcs->DestroyGC(pGC);
|
|
GC_FUNC_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpChangeClip(GCPtr pGC, int type, pointer pValue, int nrects)
|
|
{
|
|
rdpGCRec* priv;
|
|
|
|
DEBUG_OUT_FUNCS(("in rdpChangeClip\n"));
|
|
GC_FUNC_PROLOGUE(pGC);
|
|
pGC->funcs->ChangeClip(pGC, type, pValue, nrects);
|
|
GC_FUNC_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpDestroyClip(GCPtr pGC)
|
|
{
|
|
rdpGCRec* priv;
|
|
|
|
DEBUG_OUT_FUNCS(("in rdpDestroyClip\n"));
|
|
GC_FUNC_PROLOGUE(pGC);
|
|
pGC->funcs->DestroyClip(pGC);
|
|
GC_FUNC_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpCopyClip(GCPtr dst, GCPtr src)
|
|
{
|
|
rdpGCRec* priv;
|
|
|
|
DEBUG_OUT_FUNCS(("in rdpCopyClip\n"));
|
|
GC_FUNC_PROLOGUE(dst);
|
|
dst->funcs->CopyClip(dst, src);
|
|
GC_FUNC_EPILOGUE(dst);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
#define GC_OP_PROLOGUE(_pGC) \
|
|
{ \
|
|
priv = (rdpGCPtr)dixGetPrivateAddr(&(pGC->devPrivates), &g_rdpGCIndex); \
|
|
oldFuncs = _pGC->funcs; \
|
|
(_pGC)->funcs = priv->funcs; \
|
|
(_pGC)->ops = priv->ops; \
|
|
}
|
|
|
|
/******************************************************************************/
|
|
#define GC_OP_EPILOGUE(_pGC) \
|
|
{ \
|
|
priv->ops = (_pGC)->ops; \
|
|
(_pGC)->funcs = oldFuncs; \
|
|
(_pGC)->ops = &g_rdpGCOps; \
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpFillSpans(DrawablePtr pDrawable, GCPtr pGC, int nInit,
|
|
DDXPointPtr pptInit, int* pwidthInit, int fSorted)
|
|
{
|
|
rdpGCPtr priv;
|
|
RegionRec clip_reg;
|
|
int cd;
|
|
int j;
|
|
GCFuncs* oldFuncs;
|
|
BoxRec box;
|
|
|
|
DEBUG_OUT_OPS(("in rdpFillSpans\n"));
|
|
GC_OP_PROLOGUE(pGC)
|
|
pGC->ops->FillSpans(pDrawable, pGC, nInit, pptInit, pwidthInit, fSorted);
|
|
RegionInit(&clip_reg, NullBox, 0);
|
|
cd = rdp_get_clip(&clip_reg, pDrawable, pGC);
|
|
if (cd == 1)
|
|
{
|
|
rdpup_begin_update();
|
|
RegionCopy(&clip_reg, &(((WindowPtr)pDrawable)->borderClip));
|
|
for (j = REGION_NUM_RECTS(&clip_reg) - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(&clip_reg)[j];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
else if (cd == 2)
|
|
{
|
|
rdpup_begin_update();
|
|
RegionIntersect(&clip_reg, &clip_reg,
|
|
&(((WindowPtr)pDrawable)->borderClip));
|
|
for (j = REGION_NUM_RECTS(&clip_reg) - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(&clip_reg)[j];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
RegionUninit(&clip_reg);
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpSetSpans(DrawablePtr pDrawable, GCPtr pGC, char* psrc,
|
|
DDXPointPtr ppt, int* pwidth, int nspans, int fSorted)
|
|
{
|
|
rdpGCPtr priv;
|
|
RegionRec clip_reg;
|
|
int cd;
|
|
int j;
|
|
GCFuncs* oldFuncs;
|
|
BoxRec box;
|
|
|
|
DEBUG_OUT_OPS(("in rdpSetSpans\n"));
|
|
GC_OP_PROLOGUE(pGC);
|
|
pGC->ops->SetSpans(pDrawable, pGC, psrc, ppt, pwidth, nspans, fSorted);
|
|
RegionInit(&clip_reg, NullBox, 0);
|
|
cd = rdp_get_clip(&clip_reg, pDrawable, pGC);
|
|
if (cd == 1)
|
|
{
|
|
rdpup_begin_update();
|
|
RegionCopy(&clip_reg, &(((WindowPtr)pDrawable)->borderClip));
|
|
for (j = REGION_NUM_RECTS(&clip_reg) - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(&clip_reg)[j];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
else if (cd == 2)
|
|
{
|
|
rdpup_begin_update();
|
|
RegionIntersect(&clip_reg, &clip_reg,
|
|
&((WindowPtr)pDrawable)->borderClip);
|
|
for (j = REGION_NUM_RECTS(&clip_reg) - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(&clip_reg)[j];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
RegionUninit(&clip_reg);
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpPutImage(DrawablePtr pDrawable, GCPtr pGC, int depth, int x, int y,
|
|
int w, int h, int leftPad, int format, char* pBits)
|
|
{
|
|
rdpGCPtr priv;
|
|
RegionRec clip_reg;
|
|
int cd;
|
|
int j;
|
|
GCFuncs* oldFuncs;
|
|
BoxRec box;
|
|
|
|
DEBUG_OUT_OPS(("in rdpPutImage\n"));
|
|
GC_OP_PROLOGUE(pGC);
|
|
pGC->ops->PutImage(pDrawable, pGC, depth, x, y, w, h, leftPad,
|
|
format, pBits);
|
|
RegionInit(&clip_reg, NullBox, 0);
|
|
cd = rdp_get_clip(&clip_reg, pDrawable, pGC);
|
|
if (cd == 1)
|
|
{
|
|
rdpup_begin_update();
|
|
rdpup_send_area(pDrawable->x + x, pDrawable->y + y, w, h);
|
|
rdpup_end_update();
|
|
}
|
|
else if (cd == 2)
|
|
{
|
|
rdpup_begin_update();
|
|
for (j = REGION_NUM_RECTS(&clip_reg) - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(&clip_reg)[j];
|
|
rdpup_set_clip(box.x1, box.y1, (box.x2 - box.x1), (box.y2 - box.y1));
|
|
rdpup_send_area(pDrawable->x + x, pDrawable->y + y, w, h);
|
|
}
|
|
rdpup_reset_clip();
|
|
rdpup_end_update();
|
|
}
|
|
RegionUninit(&clip_reg);
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static RegionPtr
|
|
rdpCopyArea(DrawablePtr pSrc, DrawablePtr pDst, GCPtr pGC,
|
|
int srcx, int srcy, int w, int h, int dstx, int dsty)
|
|
{
|
|
rdpGCPtr priv;
|
|
RegionPtr rv;
|
|
RegionRec clip_reg;
|
|
RegionRec box_reg;
|
|
int num_clips;
|
|
int cd;
|
|
int j;
|
|
int can_do_screen_blt;
|
|
int dx;
|
|
int dy;
|
|
BoxRec box;
|
|
BoxPtr pbox;
|
|
GCFuncs* oldFuncs;
|
|
|
|
DEBUG_OUT_OPS(("in rdpCopyArea\n"));
|
|
GC_OP_PROLOGUE(pGC);
|
|
rv = pGC->ops->CopyArea(pSrc, pDst, pGC, srcx, srcy, w, h, dstx, dsty);
|
|
RegionInit(&clip_reg, NullBox, 0);
|
|
cd = rdp_get_clip(&clip_reg, pDst, pGC);
|
|
can_do_screen_blt = pSrc->type == DRAWABLE_WINDOW &&
|
|
((WindowPtr)pSrc)->viewable &&
|
|
pGC->alu == GXcopy;
|
|
if (cd == 1)
|
|
{
|
|
rdpup_begin_update();
|
|
if (can_do_screen_blt)
|
|
{
|
|
rdpup_screen_blt(pDst->x + dstx, pDst->y + dsty, w, h,
|
|
pSrc->x + srcx, pSrc->y + srcy);
|
|
}
|
|
else
|
|
{
|
|
rdpup_send_area(pDst->x + dstx, pDst->y + dsty, w, h);
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
else if (cd == 2)
|
|
{
|
|
num_clips = REGION_NUM_RECTS(&clip_reg);
|
|
if (num_clips > 0)
|
|
{
|
|
rdpup_begin_update();
|
|
if (can_do_screen_blt)
|
|
{
|
|
dx = dstx - srcx;
|
|
dy = dsty - srcy;
|
|
if (dy < 0 || (dy == 0 && dx < 0))
|
|
{
|
|
for (j = 0; j < num_clips; j++)
|
|
{
|
|
box = REGION_RECTS(&clip_reg)[j];
|
|
rdpup_set_clip(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
rdpup_screen_blt(pDst->x + dstx, pDst->y + dsty, w, h,
|
|
pSrc->x + srcx, pSrc->y + srcy);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (j = num_clips - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(&clip_reg)[j];
|
|
rdpup_set_clip(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
rdpup_screen_blt(pDst->x + dstx, pDst->y + dsty, w, h,
|
|
pSrc->x + srcx, pSrc->y + srcy);
|
|
}
|
|
}
|
|
rdpup_reset_clip();
|
|
}
|
|
else
|
|
{
|
|
box.x1 = pDst->x + dstx;
|
|
box.y1 = pDst->y + dsty;
|
|
box.x2 = box.x1 + w;
|
|
box.y2 = box.y1 + h;
|
|
RegionInit(&box_reg, &box, 0);
|
|
RegionIntersect(&clip_reg, &clip_reg, &box_reg);
|
|
num_clips = REGION_NUM_RECTS(&clip_reg);
|
|
if (num_clips < 10)
|
|
{
|
|
for (j = num_clips - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(&clip_reg)[j];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pbox = RegionExtents(&clip_reg);
|
|
rdpup_send_area(pbox->x1, pbox->y1, pbox->x2 - pbox->x1,
|
|
pbox->y2 - pbox->y1);
|
|
}
|
|
RegionUninit(&box_reg);
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
}
|
|
RegionUninit(&clip_reg);
|
|
GC_OP_EPILOGUE(pGC);
|
|
return rv;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static RegionPtr
|
|
rdpCopyPlane(DrawablePtr pSrcDrawable, DrawablePtr pDstDrawable,
|
|
GCPtr pGC, int srcx, int srcy, int width, int height,
|
|
int dstx, int dsty, unsigned long bitPlane)
|
|
{
|
|
rdpGCPtr priv;
|
|
RegionPtr rv;
|
|
RegionRec reg;
|
|
int cd;
|
|
GCFuncs* oldFuncs;
|
|
|
|
DEBUG_OUT_OPS(("in rdpCopyPlane\n"));
|
|
GC_OP_PROLOGUE(pGC);
|
|
rv = pGC->ops->CopyPlane(pSrcDrawable, pDstDrawable, pGC, srcx, srcy,
|
|
width, height, dstx, dsty, bitPlane);
|
|
RegionInit(®, NullBox, 0);
|
|
cd = rdp_get_clip(®, pDstDrawable, pGC);
|
|
if (cd == 1)
|
|
{
|
|
}
|
|
else if (cd == 2)
|
|
{
|
|
}
|
|
RegionUninit(®);
|
|
GC_OP_EPILOGUE(pGC);
|
|
return rv;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpPolyPoint(DrawablePtr pDrawable, GCPtr pGC, int mode,
|
|
int npt, DDXPointPtr in_pts)
|
|
{
|
|
rdpGCPtr priv;
|
|
RegionRec clip_reg;
|
|
int num_clips;
|
|
int cd;
|
|
int x;
|
|
int y;
|
|
int i;
|
|
int j;
|
|
GCFuncs* oldFuncs;
|
|
BoxRec box;
|
|
BoxRec total_box;
|
|
DDXPointPtr pts;
|
|
DDXPointRec stack_pts[32];
|
|
|
|
DEBUG_OUT_OPS(("in rdpPolyPoint\n"));
|
|
GC_OP_PROLOGUE(pGC);
|
|
if (npt > 32)
|
|
{
|
|
pts = (DDXPointPtr)g_malloc(sizeof(DDXPointRec) * npt, 0);
|
|
}
|
|
else
|
|
{
|
|
pts = stack_pts;
|
|
}
|
|
for (i = 0; i < npt; i++)
|
|
{
|
|
pts[i].x = pDrawable->x + in_pts[i].x;
|
|
pts[i].y = pDrawable->y + in_pts[i].y;
|
|
if (i == 0)
|
|
{
|
|
total_box.x1 = pts[0].x;
|
|
total_box.y1 = pts[0].y;
|
|
total_box.x2 = pts[0].x;
|
|
total_box.y2 = pts[0].y;
|
|
}
|
|
else
|
|
{
|
|
if (pts[i].x < total_box.x1)
|
|
{
|
|
total_box.x1 = pts[i].x;
|
|
}
|
|
if (pts[i].y < total_box.y1)
|
|
{
|
|
total_box.y1 = pts[i].y;
|
|
}
|
|
if (pts[i].x > total_box.x2)
|
|
{
|
|
total_box.x2 = pts[i].x;
|
|
}
|
|
if (pts[i].y > total_box.y2)
|
|
{
|
|
total_box.y2 = pts[i].y;
|
|
}
|
|
}
|
|
/* todo, use this total_box */
|
|
}
|
|
pGC->ops->PolyPoint(pDrawable, pGC, mode, npt, in_pts);
|
|
RegionInit(&clip_reg, NullBox, 0);
|
|
cd = rdp_get_clip(&clip_reg, pDrawable, pGC);
|
|
if (cd == 1)
|
|
{
|
|
if (npt > 0)
|
|
{
|
|
rdpup_begin_update();
|
|
rdpup_set_fgcolor(pGC->fgPixel);
|
|
for (i = 0; i < npt; i++)
|
|
{
|
|
x = pts[i].x;
|
|
y = pts[i].y;
|
|
rdpup_fill_rect(x, y, 1, 1);
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
}
|
|
else if (cd == 2)
|
|
{
|
|
num_clips = REGION_NUM_RECTS(&clip_reg);
|
|
if (npt > 0 && num_clips > 0)
|
|
{
|
|
rdpup_begin_update();
|
|
rdpup_set_fgcolor(pGC->fgPixel);
|
|
for (j = num_clips - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(&clip_reg)[j];
|
|
rdpup_set_clip(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
for (i = 0; i < npt; i++)
|
|
{
|
|
x = pts[i].x;
|
|
y = pts[i].y;
|
|
rdpup_fill_rect(x, y, 1, 1);
|
|
}
|
|
}
|
|
rdpup_reset_clip();
|
|
rdpup_end_update();
|
|
}
|
|
}
|
|
RegionUninit(&clip_reg);
|
|
if (pts != stack_pts)
|
|
{
|
|
g_free(pts);
|
|
}
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpPolylines(DrawablePtr pDrawable, GCPtr pGC, int mode,
|
|
int npt, DDXPointPtr pptInit)
|
|
{
|
|
rdpGCPtr priv;
|
|
RegionRec clip_reg;
|
|
int num_clips;
|
|
int cd;
|
|
int i;
|
|
int j;
|
|
int x1;
|
|
int y1;
|
|
int x2;
|
|
int y2;
|
|
GCFuncs* oldFuncs;
|
|
BoxRec box;
|
|
DDXPointPtr ppts;
|
|
|
|
DEBUG_OUT_OPS(("in rdpPolylines\n"));
|
|
GC_OP_PROLOGUE(pGC);
|
|
ppts = 0;
|
|
if (npt > 0)
|
|
{
|
|
ppts = (DDXPointPtr)g_malloc(sizeof(DDXPointRec) * npt, 0);
|
|
for (i = 0; i < npt; i++)
|
|
{
|
|
ppts[i] = pptInit[i];
|
|
}
|
|
}
|
|
pGC->ops->Polylines(pDrawable, pGC, mode, npt, pptInit);
|
|
RegionInit(&clip_reg, NullBox, 0);
|
|
cd = rdp_get_clip(&clip_reg, pDrawable, pGC);
|
|
if (cd == 1)
|
|
{
|
|
if (ppts != 0)
|
|
{
|
|
rdpup_begin_update();
|
|
rdpup_set_fgcolor(pGC->fgPixel);
|
|
rdpup_set_opcode(pGC->alu);
|
|
rdpup_set_pen(0, pGC->lineWidth);
|
|
x1 = ppts[0].x + pDrawable->x;
|
|
y1 = ppts[0].y + pDrawable->y;
|
|
for (i = 1; i < npt; i++)
|
|
{
|
|
if (mode == CoordModeOrigin)
|
|
{
|
|
x2 = pDrawable->x + ppts[i].x;
|
|
y2 = pDrawable->y + ppts[i].y;
|
|
}
|
|
else
|
|
{
|
|
x2 = x1 + ppts[i].x;
|
|
y2 = y1 + ppts[i].y;
|
|
}
|
|
rdpup_draw_line(x1, y1, x2, y2);
|
|
x1 = x2;
|
|
y1 = y2;
|
|
}
|
|
rdpup_set_opcode(GXcopy);
|
|
rdpup_end_update();
|
|
}
|
|
}
|
|
else if (cd == 2)
|
|
{
|
|
num_clips = REGION_NUM_RECTS(&clip_reg);
|
|
if (ppts != 0 && num_clips > 0)
|
|
{
|
|
rdpup_begin_update();
|
|
rdpup_set_fgcolor(pGC->fgPixel);
|
|
rdpup_set_opcode(pGC->alu);
|
|
rdpup_set_pen(0, pGC->lineWidth);
|
|
for (j = num_clips - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(&clip_reg)[j];
|
|
rdpup_set_clip(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
x1 = ppts[0].x + pDrawable->x;
|
|
y1 = ppts[0].y + pDrawable->y;
|
|
for (i = 1; i < npt; i++)
|
|
{
|
|
if (mode == CoordModeOrigin)
|
|
{
|
|
x2 = pDrawable->x + ppts[i].x;
|
|
y2 = pDrawable->y + ppts[i].y;
|
|
}
|
|
else
|
|
{
|
|
x2 = x1 + ppts[i].x;
|
|
y2 = y1 + ppts[i].y;
|
|
}
|
|
rdpup_draw_line(x1, y1, x2, y2);
|
|
x1 = x2;
|
|
y1 = y2;
|
|
}
|
|
}
|
|
rdpup_reset_clip();
|
|
rdpup_set_opcode(GXcopy);
|
|
rdpup_end_update();
|
|
}
|
|
}
|
|
RegionUninit(&clip_reg);
|
|
g_free(ppts);
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpPolySegment(DrawablePtr pDrawable, GCPtr pGC, int nseg, xSegment* pSegs)
|
|
{
|
|
rdpGCPtr priv;
|
|
RegionRec clip_reg;
|
|
int cd;
|
|
int i;
|
|
int j;
|
|
GCFuncs* oldFuncs;
|
|
xSegment* segs;
|
|
BoxRec box;
|
|
|
|
DEBUG_OUT_OPS(("in rdpPolySegment\n"));
|
|
GC_OP_PROLOGUE(pGC);
|
|
segs = 0;
|
|
if (nseg) /* get the rects */
|
|
{
|
|
segs = (xSegment*)g_malloc(nseg * sizeof(xSegment), 0);
|
|
for (i = 0; i < nseg; i++)
|
|
{
|
|
segs[i].x1 = pSegs[i].x1 + pDrawable->x;
|
|
segs[i].y1 = pSegs[i].y1 + pDrawable->y;
|
|
segs[i].x2 = pSegs[i].x2 + pDrawable->x;
|
|
segs[i].y2 = pSegs[i].y2 + pDrawable->y;
|
|
}
|
|
}
|
|
pGC->ops->PolySegment(pDrawable, pGC, nseg, pSegs);
|
|
RegionInit(&clip_reg, NullBox, 0);
|
|
cd = rdp_get_clip(&clip_reg, pDrawable, pGC);
|
|
if (cd == 1) /* no clip */
|
|
{
|
|
if (segs != 0)
|
|
{
|
|
rdpup_begin_update();
|
|
rdpup_set_fgcolor(pGC->fgPixel);
|
|
rdpup_set_opcode(pGC->alu);
|
|
rdpup_set_pen(0, pGC->lineWidth);
|
|
for (i = 0; i < nseg; i++)
|
|
{
|
|
rdpup_draw_line(segs[i].x1, segs[i].y1, segs[i].x2, segs[i].y2);
|
|
}
|
|
rdpup_set_opcode(GXcopy);
|
|
rdpup_end_update();
|
|
}
|
|
}
|
|
else if (cd == 2) /* clip */
|
|
{
|
|
if (segs != 0)
|
|
{
|
|
rdpup_begin_update();
|
|
rdpup_set_fgcolor(pGC->fgPixel);
|
|
rdpup_set_opcode(pGC->alu);
|
|
rdpup_set_pen(0, pGC->lineWidth);
|
|
for (j = REGION_NUM_RECTS(&clip_reg) - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(&clip_reg)[j];
|
|
rdpup_set_clip(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
for (i = 0; i < nseg; i++)
|
|
{
|
|
rdpup_draw_line(segs[i].x1, segs[i].y1, segs[i].x2, segs[i].y2);
|
|
}
|
|
}
|
|
rdpup_reset_clip();
|
|
rdpup_set_opcode(GXcopy);
|
|
rdpup_end_update();
|
|
}
|
|
}
|
|
g_free(segs);
|
|
RegionUninit(&clip_reg);
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/* tested with pGC->lineWidth = 0, 1, 2, 4 and opcodes 3 and 6 */
|
|
static void
|
|
rdpPolyRectangle(DrawablePtr pDrawable, GCPtr pGC, int nrects,
|
|
xRectangle* rects)
|
|
{
|
|
rdpGCPtr priv;
|
|
RegionRec clip_reg;
|
|
RegionPtr fill_reg;
|
|
int num_clips;
|
|
int cd;
|
|
int lw;
|
|
int i;
|
|
int j;
|
|
int up;
|
|
int down;
|
|
GCFuncs* oldFuncs;
|
|
xRectangle* regRects;
|
|
xRectangle* r;
|
|
xRectangle* rect1;
|
|
BoxRec box;
|
|
|
|
DEBUG_OUT_OPS(("in rdpPolyRectangle\n"));
|
|
GC_OP_PROLOGUE(pGC);
|
|
/* make a copy of rects */
|
|
rect1 = (xRectangle*)g_malloc(sizeof(xRectangle) * nrects, 0);
|
|
for (i = 0; i < nrects; i++)
|
|
{
|
|
rect1[i] = rects[i];
|
|
}
|
|
pGC->ops->PolyRectangle(pDrawable, pGC, nrects, rects);
|
|
RegionInit(&clip_reg, NullBox, 0);
|
|
cd = rdp_get_clip(&clip_reg, pDrawable, pGC);
|
|
regRects = 0;
|
|
if (cd != 0 && nrects > 0)
|
|
{
|
|
regRects = (xRectangle*)g_malloc(nrects * 4 * sizeof(xRectangle), 0);
|
|
lw = pGC->lineWidth;
|
|
if (lw < 1)
|
|
{
|
|
lw = 1;
|
|
}
|
|
up = lw / 2;
|
|
down = 1 + (lw - 1) / 2;
|
|
for (i = 0; i < nrects; i++)
|
|
{
|
|
r = regRects + i * 4;
|
|
r->x = (rect1[i].x + pDrawable->x) - up;
|
|
r->y = (rect1[i].y + pDrawable->y) - up;
|
|
r->width = rect1[i].width + up + down;
|
|
r->height = lw;
|
|
r++;
|
|
r->x = (rect1[i].x + pDrawable->x) - up;
|
|
r->y = (rect1[i].y + pDrawable->y) + down;
|
|
r->width = lw;
|
|
r->height = MAX(rect1[i].height - (up + down), 0);
|
|
r++;
|
|
r->x = ((rect1[i].x + rect1[i].width) + pDrawable->x) - up;
|
|
r->y = (rect1[i].y + pDrawable->y) + down;
|
|
r->width = lw;
|
|
r->height = MAX(rect1[i].height - (up + down), 0);
|
|
r++;
|
|
r->x = (rect1[i].x + pDrawable->x) - up;
|
|
r->y = ((rect1[i].y + rect1[i].height) + pDrawable->y) - up;
|
|
r->width = rect1[i].width + up + down;
|
|
r->height = lw;
|
|
}
|
|
}
|
|
if (cd == 1)
|
|
{
|
|
if (regRects != 0)
|
|
{
|
|
rdpup_begin_update();
|
|
if (pGC->lineStyle == LineSolid)
|
|
{
|
|
rdpup_set_fgcolor(pGC->fgPixel);
|
|
rdpup_set_opcode(pGC->alu);
|
|
for (i = 0; i < nrects * 4; i++)
|
|
{
|
|
r = regRects + i;
|
|
rdpup_fill_rect(r->x, r->y, r->width, r->height);
|
|
}
|
|
rdpup_set_opcode(GXcopy);
|
|
}
|
|
else
|
|
{
|
|
for (i = 0; i < nrects * 4; i++)
|
|
{
|
|
r = regRects + i;
|
|
rdpup_send_area(r->x, r->y, r->width, r->height);
|
|
}
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
}
|
|
else if (cd == 2)
|
|
{
|
|
if (regRects != 0)
|
|
{
|
|
fill_reg = RegionFromRects(nrects * 4, regRects, CT_NONE);
|
|
RegionIntersect(&clip_reg, &clip_reg, fill_reg);
|
|
num_clips = REGION_NUM_RECTS(&clip_reg);
|
|
if (num_clips > 0)
|
|
{
|
|
rdpup_begin_update();
|
|
if (pGC->lineStyle == LineSolid)
|
|
{
|
|
rdpup_set_fgcolor(pGC->fgPixel);
|
|
rdpup_set_opcode(pGC->alu);
|
|
for (j = num_clips - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(&clip_reg)[j];
|
|
rdpup_fill_rect(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_set_opcode(GXcopy);
|
|
}
|
|
else
|
|
{
|
|
for (j = num_clips - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(&clip_reg)[j];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
RegionDestroy(fill_reg);
|
|
}
|
|
}
|
|
RegionUninit(&clip_reg);
|
|
g_free(regRects);
|
|
g_free(rect1);
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpPolyArc(DrawablePtr pDrawable, GCPtr pGC, int narcs, xArc* parcs)
|
|
{
|
|
rdpGCPtr priv;
|
|
RegionRec clip_reg;
|
|
RegionPtr tmpRegion;
|
|
int cd;
|
|
int lw;
|
|
int extra;
|
|
int i;
|
|
int num_clips;
|
|
GCFuncs* oldFuncs;
|
|
xRectangle* rects;
|
|
BoxRec box;
|
|
|
|
DEBUG_OUT_OPS(("in rdpPolyArc\n"));
|
|
GC_OP_PROLOGUE(pGC);
|
|
rects = 0;
|
|
if (narcs > 0)
|
|
{
|
|
rects = (xRectangle*)g_malloc(narcs * sizeof(xRectangle), 0);
|
|
lw = pGC->lineWidth;
|
|
if (lw == 0)
|
|
{
|
|
lw = 1;
|
|
}
|
|
extra = lw / 2;
|
|
for (i = 0; i < narcs; i++)
|
|
{
|
|
rects[i].x = (parcs[i].x - extra) + pDrawable->x;
|
|
rects[i].y = (parcs[i].y - extra) + pDrawable->y;
|
|
rects[i].width = parcs[i].width + lw;
|
|
rects[i].height = parcs[i].height + lw;
|
|
}
|
|
}
|
|
pGC->ops->PolyArc(pDrawable, pGC, narcs, parcs);
|
|
RegionInit(&clip_reg, NullBox, 0);
|
|
cd = rdp_get_clip(&clip_reg, pDrawable, pGC);
|
|
if (cd == 1)
|
|
{
|
|
if (rects != 0)
|
|
{
|
|
tmpRegion = RegionFromRects(narcs, rects, CT_NONE);
|
|
num_clips = REGION_NUM_RECTS(tmpRegion);
|
|
if (num_clips > 0)
|
|
{
|
|
rdpup_begin_update();
|
|
for (i = num_clips - 1; i >= 0; i--)
|
|
{
|
|
box = REGION_RECTS(tmpRegion)[i];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
RegionDestroy(tmpRegion);
|
|
}
|
|
}
|
|
else if (cd == 2)
|
|
{
|
|
if (rects != 0)
|
|
{
|
|
tmpRegion = RegionFromRects(narcs, rects, CT_NONE);
|
|
RegionIntersect(tmpRegion, tmpRegion, &clip_reg);
|
|
num_clips = REGION_NUM_RECTS(tmpRegion);
|
|
if (num_clips > 0)
|
|
{
|
|
rdpup_begin_update();
|
|
for (i = num_clips - 1; i >= 0; i--)
|
|
{
|
|
box = REGION_RECTS(tmpRegion)[i];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
RegionDestroy(tmpRegion);
|
|
}
|
|
}
|
|
RegionUninit(&clip_reg);
|
|
g_free(rects);
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpFillPolygon(DrawablePtr pDrawable, GCPtr pGC,
|
|
int shape, int mode, int count,
|
|
DDXPointPtr pPts)
|
|
{
|
|
rdpGCPtr priv;
|
|
RegionRec clip_reg;
|
|
RegionRec box_reg;
|
|
int num_clips;
|
|
int cd;
|
|
int maxx;
|
|
int maxy;
|
|
int minx;
|
|
int miny;
|
|
int i;
|
|
int j;
|
|
GCFuncs* oldFuncs;
|
|
BoxRec box;
|
|
|
|
DEBUG_OUT_OPS(("in rdpFillPolygon\n"));
|
|
GC_OP_PROLOGUE(pGC);
|
|
pGC->ops->FillPolygon(pDrawable, pGC, shape, mode, count, pPts);
|
|
RegionInit(&clip_reg, NullBox, 0);
|
|
cd = rdp_get_clip(&clip_reg, pDrawable, pGC);
|
|
if (cd != 0)
|
|
{
|
|
box.x1 = 0;
|
|
box.y1 = 0;
|
|
box.x2 = 0;
|
|
box.y2 = 0;
|
|
if (count > 0)
|
|
{
|
|
maxx = pPts[0].x;
|
|
maxy = pPts[0].y;
|
|
minx = maxx;
|
|
miny = maxy;
|
|
for (i = 1; i < count; i++)
|
|
{
|
|
if (pPts[i].x > maxx)
|
|
{
|
|
maxx = pPts[i].x;
|
|
}
|
|
if (pPts[i].x < minx)
|
|
{
|
|
minx = pPts[i].x;
|
|
}
|
|
if (pPts[i].y > maxy)
|
|
{
|
|
maxy = pPts[i].y;
|
|
}
|
|
if (pPts[i].y < miny)
|
|
{
|
|
miny = pPts[i].y;
|
|
}
|
|
box.x1 = pDrawable->x + minx;
|
|
box.y1 = pDrawable->y + miny;
|
|
box.x2 = pDrawable->x + maxx + 1;
|
|
box.y2 = pDrawable->y + maxy + 1;
|
|
}
|
|
}
|
|
}
|
|
if (cd == 1)
|
|
{
|
|
rdpup_begin_update();
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
rdpup_end_update();
|
|
}
|
|
else if (cd == 2)
|
|
{
|
|
RegionInit(&box_reg, &box, 0);
|
|
RegionIntersect(&clip_reg, &clip_reg, &box_reg);
|
|
num_clips = REGION_NUM_RECTS(&clip_reg);
|
|
if (num_clips > 0)
|
|
{
|
|
rdpup_begin_update();
|
|
for (j = num_clips - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(&clip_reg)[j];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
RegionUninit(&box_reg);
|
|
}
|
|
RegionUninit(&clip_reg);
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpPolyFillRect(DrawablePtr pDrawable, GCPtr pGC, int nrectFill,
|
|
xRectangle* prectInit)
|
|
{
|
|
int i;
|
|
int j;
|
|
int cd;
|
|
int num_clips;
|
|
xRectangle* copy_of_rects;
|
|
rdpGCPtr priv;
|
|
RegionRec clip_reg;
|
|
RegionPtr fill_reg;
|
|
BoxRec box;
|
|
GCFuncs* oldFuncs;
|
|
|
|
DEBUG_OUT_OPS(("in rdpPolyFillRect\n"));
|
|
GC_OP_PROLOGUE(pGC);
|
|
/* make a copy of rects */
|
|
copy_of_rects = (xRectangle*)g_malloc(sizeof(xRectangle) * nrectFill, 0);
|
|
for (i = 0; i < nrectFill; i++)
|
|
{
|
|
copy_of_rects[i] = prectInit[i];
|
|
}
|
|
fill_reg = RegionFromRects(nrectFill, copy_of_rects, CT_NONE);
|
|
g_free(copy_of_rects);
|
|
RegionTranslate(fill_reg, pDrawable->x, pDrawable->y);
|
|
pGC->ops->PolyFillRect(pDrawable, pGC, nrectFill, prectInit);
|
|
RegionInit(&clip_reg, NullBox, 0);
|
|
cd = rdp_get_clip(&clip_reg, pDrawable, pGC);
|
|
if (cd == 1) /* no clip */
|
|
{
|
|
rdpup_begin_update();
|
|
if (pGC->fillStyle == 0 && /* solid fill */
|
|
(pGC->alu == GXclear ||
|
|
pGC->alu == GXset ||
|
|
pGC->alu == GXinvert ||
|
|
pGC->alu == GXnoop ||
|
|
pGC->alu == GXand ||
|
|
pGC->alu == GXcopy /*||
|
|
pGC->alu == GXxor*/)) /* todo, why dosen't xor work? */
|
|
{
|
|
rdpup_set_fgcolor(pGC->fgPixel);
|
|
rdpup_set_opcode(pGC->alu);
|
|
for (j = REGION_NUM_RECTS(fill_reg) - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(fill_reg)[j];
|
|
rdpup_fill_rect(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_set_opcode(GXcopy);
|
|
}
|
|
else /* non solid fill */
|
|
{
|
|
for (j = REGION_NUM_RECTS(fill_reg) - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(fill_reg)[j];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
else if (cd == 2) /* clip */
|
|
{
|
|
RegionIntersect(&clip_reg, &clip_reg, fill_reg);
|
|
num_clips = REGION_NUM_RECTS(&clip_reg);
|
|
if (num_clips > 0)
|
|
{
|
|
rdpup_begin_update();
|
|
if (pGC->fillStyle == 0 && /* solid fill */
|
|
(pGC->alu == GXclear ||
|
|
pGC->alu == GXset ||
|
|
pGC->alu == GXinvert ||
|
|
pGC->alu == GXnoop ||
|
|
pGC->alu == GXand ||
|
|
pGC->alu == GXcopy /*||
|
|
pGC->alu == GXxor*/)) /* todo, why dosen't xor work? */
|
|
{
|
|
rdpup_set_fgcolor(pGC->fgPixel);
|
|
rdpup_set_opcode(pGC->alu);
|
|
for (j = num_clips - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(&clip_reg)[j];
|
|
rdpup_fill_rect(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_set_opcode(GXcopy);
|
|
}
|
|
else /* non solid fill */
|
|
{
|
|
for (j = num_clips - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(&clip_reg)[j];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
}
|
|
RegionUninit(&clip_reg);
|
|
RegionDestroy(fill_reg);
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpPolyFillArc(DrawablePtr pDrawable, GCPtr pGC, int narcs, xArc* parcs)
|
|
{
|
|
rdpGCPtr priv;
|
|
RegionRec clip_reg;
|
|
RegionPtr tmpRegion;
|
|
int cd;
|
|
int lw;
|
|
int extra;
|
|
int i;
|
|
int num_clips;
|
|
GCFuncs* oldFuncs;
|
|
xRectangle* rects;
|
|
BoxRec box;
|
|
|
|
DEBUG_OUT_OPS(("in rdpPolyFillArc\n"));
|
|
GC_OP_PROLOGUE(pGC);
|
|
rects = 0;
|
|
if (narcs > 0)
|
|
{
|
|
rects = (xRectangle*)g_malloc(narcs * sizeof(xRectangle), 0);
|
|
lw = pGC->lineWidth;
|
|
if (lw == 0)
|
|
{
|
|
lw = 1;
|
|
}
|
|
extra = lw / 2;
|
|
for (i = 0; i < narcs; i++)
|
|
{
|
|
rects[i].x = (parcs[i].x - extra) + pDrawable->x;
|
|
rects[i].y = (parcs[i].y - extra) + pDrawable->y;
|
|
rects[i].width = parcs[i].width + lw;
|
|
rects[i].height = parcs[i].height + lw;
|
|
}
|
|
}
|
|
pGC->ops->PolyFillArc(pDrawable, pGC, narcs, parcs);
|
|
RegionInit(&clip_reg, NullBox, 0);
|
|
cd = rdp_get_clip(&clip_reg, pDrawable, pGC);
|
|
if (cd == 1)
|
|
{
|
|
if (rects != 0)
|
|
{
|
|
tmpRegion = RegionFromRects(narcs, rects, CT_NONE);
|
|
num_clips = REGION_NUM_RECTS(tmpRegion);
|
|
if (num_clips > 0)
|
|
{
|
|
rdpup_begin_update();
|
|
for (i = num_clips - 1; i >= 0; i--)
|
|
{
|
|
box = REGION_RECTS(tmpRegion)[i];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
RegionDestroy(tmpRegion);
|
|
}
|
|
}
|
|
else if (cd == 2)
|
|
{
|
|
if (rects != 0)
|
|
{
|
|
tmpRegion = RegionFromRects(narcs, rects, CT_NONE);
|
|
RegionIntersect(tmpRegion, tmpRegion, &clip_reg);
|
|
num_clips = REGION_NUM_RECTS(tmpRegion);
|
|
if (num_clips > 0)
|
|
{
|
|
rdpup_begin_update();
|
|
for (i = num_clips - 1; i >= 0; i--)
|
|
{
|
|
box = REGION_RECTS(tmpRegion)[i];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
RegionDestroy(tmpRegion);
|
|
}
|
|
}
|
|
RegionUninit(&clip_reg);
|
|
g_free(rects);
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static int
|
|
rdpPolyText8(DrawablePtr pDrawable, GCPtr pGC,
|
|
int x, int y, int count, char* chars)
|
|
{
|
|
rdpGCPtr priv;
|
|
RegionRec reg;
|
|
RegionRec reg1;
|
|
int num_clips;
|
|
int cd;
|
|
int j;
|
|
int rv;
|
|
GCFuncs* oldFuncs;
|
|
BoxRec box;
|
|
|
|
DEBUG_OUT_OPS(("in rdpPolyText8\n"));
|
|
GC_OP_PROLOGUE(pGC);
|
|
if (count != 0)
|
|
{
|
|
GetTextBoundingBox(pDrawable, pGC->font, x, y, count, &box);
|
|
}
|
|
rv = pGC->ops->PolyText8(pDrawable, pGC, x, y, count, chars);
|
|
RegionInit(®, NullBox, 0);
|
|
if (count == 0)
|
|
{
|
|
cd = 0;
|
|
}
|
|
else
|
|
{
|
|
cd = rdp_get_clip(®, pDrawable, pGC);
|
|
}
|
|
if (cd == 1)
|
|
{
|
|
rdpup_begin_update();
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
rdpup_end_update();
|
|
}
|
|
else if (cd == 2)
|
|
{
|
|
RegionInit(®1, &box, 0);
|
|
RegionIntersect(®, ®, ®1);
|
|
num_clips = REGION_NUM_RECTS(®);
|
|
if (num_clips > 0)
|
|
{
|
|
rdpup_begin_update();
|
|
for (j = num_clips - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(®)[j];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
RegionUninit(®1);
|
|
}
|
|
RegionUninit(®);
|
|
GC_OP_EPILOGUE(pGC);
|
|
return rv;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static int
|
|
rdpPolyText16(DrawablePtr pDrawable, GCPtr pGC,
|
|
int x, int y, int count, unsigned short* chars)
|
|
{
|
|
rdpGCPtr priv;
|
|
RegionRec reg;
|
|
RegionRec reg1;
|
|
int num_clips;
|
|
int cd;
|
|
int j;
|
|
int rv;
|
|
GCFuncs* oldFuncs;
|
|
BoxRec box;
|
|
|
|
DEBUG_OUT_OPS(("in rdpPolyText16\n"));
|
|
GC_OP_PROLOGUE(pGC);
|
|
if (count != 0)
|
|
{
|
|
GetTextBoundingBox(pDrawable, pGC->font, x, y, count, &box);
|
|
}
|
|
rv = pGC->ops->PolyText16(pDrawable, pGC, x, y, count, chars);
|
|
RegionInit(®, NullBox, 0);
|
|
if (count == 0)
|
|
{
|
|
cd = 0;
|
|
}
|
|
else
|
|
{
|
|
cd = rdp_get_clip(®, pDrawable, pGC);
|
|
}
|
|
if (cd == 1)
|
|
{
|
|
rdpup_begin_update();
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
rdpup_end_update();
|
|
}
|
|
else if (cd == 2)
|
|
{
|
|
RegionInit(®1, &box, 0);
|
|
RegionIntersect(®, ®, ®1);
|
|
num_clips = REGION_NUM_RECTS(®);
|
|
if (num_clips > 0)
|
|
{
|
|
rdpup_begin_update();
|
|
for (j = num_clips - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(®)[j];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
RegionUninit(®1);
|
|
}
|
|
RegionUninit(®);
|
|
GC_OP_EPILOGUE(pGC);
|
|
return rv;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpImageText8(DrawablePtr pDrawable, GCPtr pGC,
|
|
int x, int y, int count, char* chars)
|
|
{
|
|
rdpGCPtr priv;
|
|
RegionRec clip_reg;
|
|
RegionRec box_reg;
|
|
int num_clips;
|
|
int cd;
|
|
int j;
|
|
GCFuncs* oldFuncs;
|
|
BoxRec box;
|
|
|
|
DEBUG_OUT_OPS(("in rdpImageText8\n"));
|
|
GC_OP_PROLOGUE(pGC);
|
|
if (count != 0)
|
|
{
|
|
GetTextBoundingBox(pDrawable, pGC->font, x, y, count, &box);
|
|
}
|
|
pGC->ops->ImageText8(pDrawable, pGC, x, y, count, chars);
|
|
RegionInit(&clip_reg, NullBox, 0);
|
|
if (count == 0)
|
|
{
|
|
cd = 0;
|
|
}
|
|
else
|
|
{
|
|
cd = rdp_get_clip(&clip_reg, pDrawable, pGC);
|
|
}
|
|
if (cd == 1)
|
|
{
|
|
rdpup_begin_update();
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
rdpup_end_update();
|
|
}
|
|
else if (cd == 2)
|
|
{
|
|
RegionInit(&box_reg, &box, 0);
|
|
RegionIntersect(&clip_reg, &clip_reg, &box_reg);
|
|
num_clips = REGION_NUM_RECTS(&clip_reg);
|
|
if (num_clips > 0)
|
|
{
|
|
rdpup_begin_update();
|
|
for (j = num_clips - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(&clip_reg)[j];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
RegionUninit(&box_reg);
|
|
}
|
|
RegionUninit(&clip_reg);
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpImageText16(DrawablePtr pDrawable, GCPtr pGC,
|
|
int x, int y, int count,
|
|
unsigned short* chars)
|
|
{
|
|
rdpGCPtr priv;
|
|
RegionRec clip_reg;
|
|
RegionRec box_reg;
|
|
int num_clips;
|
|
int cd;
|
|
int j;
|
|
GCFuncs* oldFuncs;
|
|
BoxRec box;
|
|
|
|
DEBUG_OUT_OPS(("in rdpImageText16\n"));
|
|
GC_OP_PROLOGUE(pGC);
|
|
if (count != 0)
|
|
{
|
|
GetTextBoundingBox(pDrawable, pGC->font, x, y, count, &box);
|
|
}
|
|
pGC->ops->ImageText16(pDrawable, pGC, x, y, count, chars);
|
|
RegionInit(&clip_reg, NullBox, 0);
|
|
if (count == 0)
|
|
{
|
|
cd = 0;
|
|
}
|
|
else
|
|
{
|
|
cd = rdp_get_clip(&clip_reg, pDrawable, pGC);
|
|
}
|
|
if (cd == 1)
|
|
{
|
|
rdpup_begin_update();
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
rdpup_end_update();
|
|
}
|
|
else if (cd == 2)
|
|
{
|
|
RegionInit(&box_reg, &box, 0);
|
|
RegionIntersect(&clip_reg, &clip_reg, &box_reg);
|
|
num_clips = REGION_NUM_RECTS(&clip_reg);
|
|
if (num_clips > 0)
|
|
{
|
|
rdpup_begin_update();
|
|
for (j = num_clips - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(&clip_reg)[j];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
RegionUninit(&box_reg);
|
|
}
|
|
RegionUninit(&clip_reg);
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpImageGlyphBlt(DrawablePtr pDrawable, GCPtr pGC,
|
|
int x, int y, unsigned int nglyph,
|
|
CharInfoPtr* ppci, pointer pglyphBase)
|
|
{
|
|
rdpGCPtr priv;
|
|
RegionRec reg;
|
|
RegionRec box_reg;
|
|
int num_clips;
|
|
int cd;
|
|
int j;
|
|
GCFuncs* oldFuncs;
|
|
BoxRec box;
|
|
|
|
DEBUG_OUT_OPS(("in rdpImageGlyphBlt\n"));
|
|
GC_OP_PROLOGUE(pGC);
|
|
if (nglyph != 0)
|
|
{
|
|
GetTextBoundingBox(pDrawable, pGC->font, x, y, nglyph, &box);
|
|
}
|
|
pGC->ops->ImageGlyphBlt(pDrawable, pGC, x, y, nglyph, ppci, pglyphBase);
|
|
RegionInit(®, NullBox, 0);
|
|
if (nglyph == 0)
|
|
{
|
|
cd = 0;
|
|
}
|
|
else
|
|
{
|
|
cd = rdp_get_clip(®, pDrawable, pGC);
|
|
}
|
|
if (cd == 1)
|
|
{
|
|
rdpup_begin_update();
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
rdpup_end_update();
|
|
}
|
|
else if (cd == 2)
|
|
{
|
|
RegionInit(&box_reg, &box, 0);
|
|
RegionIntersect(®, ®, &box_reg);
|
|
num_clips = REGION_NUM_RECTS(®);
|
|
if (num_clips > 0)
|
|
{
|
|
rdpup_begin_update();
|
|
for (j = num_clips - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(®)[j];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
RegionUninit(&box_reg);
|
|
}
|
|
RegionUninit(®);
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpPolyGlyphBlt(DrawablePtr pDrawable, GCPtr pGC,
|
|
int x, int y, unsigned int nglyph,
|
|
CharInfoPtr* ppci,
|
|
pointer pglyphBase)
|
|
{
|
|
rdpGCPtr priv;
|
|
RegionRec reg;
|
|
RegionRec box_reg;
|
|
int num_clips;
|
|
int cd;
|
|
int j;
|
|
GCFuncs* oldFuncs;
|
|
BoxRec box;
|
|
|
|
DEBUG_OUT_OPS(("in rdpPolyGlyphBlt\n"));
|
|
GC_OP_PROLOGUE(pGC);
|
|
memset(&box, 0, sizeof(box));
|
|
if (nglyph != 0)
|
|
{
|
|
GetTextBoundingBox(pDrawable, pGC->font, x, y, nglyph, &box);
|
|
}
|
|
pGC->ops->PolyGlyphBlt(pDrawable, pGC, x, y, nglyph, ppci, pglyphBase);
|
|
RegionInit(®, NullBox, 0);
|
|
if (nglyph == 0)
|
|
{
|
|
cd = 0;
|
|
}
|
|
else
|
|
{
|
|
cd = rdp_get_clip(®, pDrawable, pGC);
|
|
}
|
|
if (cd == 1)
|
|
{
|
|
rdpup_begin_update();
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
rdpup_end_update();
|
|
}
|
|
else if (cd == 2)
|
|
{
|
|
RegionInit(&box_reg, &box, 0);
|
|
RegionIntersect(®, ®, &box_reg);
|
|
num_clips = REGION_NUM_RECTS(®);
|
|
if (num_clips > 0)
|
|
{
|
|
rdpup_begin_update();
|
|
for (j = num_clips - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(®)[j];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
RegionUninit(&box_reg);
|
|
}
|
|
RegionUninit(®);
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
static void
|
|
rdpPushPixels(GCPtr pGC, PixmapPtr pBitMap, DrawablePtr pDst,
|
|
int w, int h, int x, int y)
|
|
{
|
|
rdpGCPtr priv;
|
|
RegionRec clip_reg;
|
|
RegionRec box_reg;
|
|
int num_clips;
|
|
int cd;
|
|
int j;
|
|
GCFuncs* oldFuncs;
|
|
BoxRec box;
|
|
|
|
DEBUG_OUT_OPS(("in rdpPushPixels\n"));
|
|
GC_OP_PROLOGUE(pGC);
|
|
memset(&box, 0, sizeof(box));
|
|
pGC->ops->PushPixels(pGC, pBitMap, pDst, w, h, x, y);
|
|
RegionInit(&clip_reg, NullBox, 0);
|
|
cd = rdp_get_clip(&clip_reg, pDst, pGC);
|
|
if (cd == 1)
|
|
{
|
|
rdpup_begin_update();
|
|
rdpup_send_area(x, y, w, h);
|
|
rdpup_end_update();
|
|
}
|
|
else if (cd == 2)
|
|
{
|
|
RegionInit(&box_reg, &box, 0);
|
|
RegionIntersect(&clip_reg, &clip_reg, &box_reg);
|
|
num_clips = REGION_NUM_RECTS(&clip_reg);
|
|
if (num_clips > 0)
|
|
{
|
|
rdpup_begin_update();
|
|
for (j = num_clips - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(&clip_reg)[j];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
RegionUninit(&box_reg);
|
|
}
|
|
RegionUninit(&clip_reg);
|
|
GC_OP_EPILOGUE(pGC);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
Bool
|
|
rdpCloseScreen(int i, ScreenPtr pScreen)
|
|
{
|
|
DEBUG_OUT_OPS(("in rdpCloseScreen\n"));
|
|
pScreen->CloseScreen = g_rdpScreen.CloseScreen;
|
|
pScreen->CreateGC = g_rdpScreen.CreateGC;
|
|
//pScreen->PaintWindowBackground = g_rdpScreen.PaintWindowBackground;
|
|
//pScreen->PaintWindowBorder = g_rdpScreen.PaintWindowBorder;
|
|
pScreen->CopyWindow = g_rdpScreen.CopyWindow;
|
|
pScreen->ClearToBackground = g_rdpScreen.ClearToBackground;
|
|
pScreen->RestoreAreas = g_rdpScreen.RestoreAreas;
|
|
return 1;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
PixmapPtr
|
|
rdpCreatePixmap(ScreenPtr pScreen, int width, int height, int depth,
|
|
unsigned usage_hint)
|
|
{
|
|
PixmapPtr rv;
|
|
rdpPixmapRec* priv;
|
|
|
|
//ErrorF("rdpCreatePixmap:\n");
|
|
//ErrorF(" in width %d height %d depth %d\n", width, height, depth);
|
|
pScreen->CreatePixmap = g_rdpScreen.CreatePixmap;
|
|
rv = pScreen->CreatePixmap(pScreen, width, height, depth, usage_hint);
|
|
priv = GETPIXPRIV(rv);
|
|
pScreen->CreatePixmap = rdpCreatePixmap;
|
|
//ErrorF(" out width %d height %d depth %d\n", rv->drawable.width,
|
|
// rv->drawable.height, rv->drawable.depth);
|
|
return rv;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
Bool
|
|
rdpDestroyPixmap(PixmapPtr pPixmap)
|
|
{
|
|
Bool rv;
|
|
ScreenPtr pScreen;
|
|
rdpPixmapRec* priv;
|
|
|
|
//ErrorF("rdpDestroyPixmap:\n");
|
|
priv = GETPIXPRIV(pPixmap);
|
|
//ErrorF(" refcnt %d\n", pPixmap->refcnt);
|
|
pScreen = pPixmap->drawable.pScreen;
|
|
pScreen->DestroyPixmap = g_rdpScreen.DestroyPixmap;
|
|
rv = pScreen->DestroyPixmap(pPixmap);
|
|
pScreen->DestroyPixmap = rdpDestroyPixmap;
|
|
return rv;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
Bool
|
|
rdpCreateWindow(WindowPtr pWindow)
|
|
{
|
|
ScreenPtr pScreen;
|
|
rdpWindowRec* priv;
|
|
Bool rv;
|
|
|
|
//ErrorF("rdpCreateWindow:\n");
|
|
priv = GETWINPRIV(pWindow);
|
|
//ErrorF(" %p status %d\n", priv, priv->status);
|
|
pScreen = pWindow->drawable.pScreen;
|
|
pScreen->CreateWindow = g_rdpScreen.CreateWindow;
|
|
rv = pScreen->CreateWindow(pWindow);
|
|
pScreen->CreateWindow = rdpCreateWindow;
|
|
return rv;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
Bool
|
|
rdpDestroyWindow(WindowPtr pWindow)
|
|
{
|
|
ScreenPtr pScreen;
|
|
rdpWindowRec* priv;
|
|
Bool rv;
|
|
|
|
//ErrorF("rdpDestroyWindow:\n");
|
|
priv = GETWINPRIV(pWindow);
|
|
pScreen = pWindow->drawable.pScreen;
|
|
pScreen->DestroyWindow = g_rdpScreen.DestroyWindow;
|
|
rv = pScreen->DestroyWindow(pWindow);
|
|
pScreen->DestroyWindow = rdpDestroyWindow;
|
|
return rv;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
Bool
|
|
rdpCreateGC(GCPtr pGC)
|
|
{
|
|
rdpGCRec* priv;
|
|
Bool rv;
|
|
|
|
DEBUG_OUT_OPS(("in rdpCreateGC\n"));
|
|
priv = GETGCPRIV(pGC);
|
|
g_pScreen->CreateGC = g_rdpScreen.CreateGC;
|
|
rv = g_pScreen->CreateGC(pGC);
|
|
if (rv)
|
|
{
|
|
priv->funcs = pGC->funcs;
|
|
priv->ops = 0;
|
|
pGC->funcs = &g_rdpGCFuncs;
|
|
}
|
|
else
|
|
{
|
|
rdpLog("error in rdpCreateGC, CreateGC failed\n");
|
|
}
|
|
g_pScreen->CreateGC = rdpCreateGC;
|
|
return rv;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
void
|
|
rdpCopyWindow(WindowPtr pWin, DDXPointRec ptOldOrg, RegionPtr pOldRegion)
|
|
{
|
|
RegionRec reg;
|
|
RegionRec clip;
|
|
int dx;
|
|
int dy;
|
|
int i;
|
|
int j;
|
|
int num_clip_rects;
|
|
int num_reg_rects;
|
|
BoxRec box1;
|
|
BoxRec box2;
|
|
|
|
DEBUG_OUT_OPS(("in rdpCopyWindow\n"));
|
|
RegionInit(®, NullBox, 0);
|
|
RegionCopy(®, pOldRegion);
|
|
g_pScreen->CopyWindow = g_rdpScreen.CopyWindow;
|
|
g_pScreen->CopyWindow(pWin, ptOldOrg, pOldRegion);
|
|
RegionInit(&clip, NullBox, 0);
|
|
RegionCopy(&clip, &pWin->borderClip);
|
|
dx = pWin->drawable.x - ptOldOrg.x;
|
|
dy = pWin->drawable.y - ptOldOrg.y;
|
|
rdpup_begin_update();
|
|
num_clip_rects = REGION_NUM_RECTS(&clip);
|
|
num_reg_rects = REGION_NUM_RECTS(®);
|
|
/* should maybe sort the rects instead of checking dy < 0 */
|
|
/* If we can depend on the rects going from top to bottom, left
|
|
to right we are ok */
|
|
if (dy < 0 || (dy == 0 && dx < 0))
|
|
{
|
|
for (j = 0; j < num_clip_rects; j++)
|
|
{
|
|
box1 = REGION_RECTS(&clip)[j];
|
|
rdpup_set_clip(box1.x1, box1.y1, box1.x2 - box1.x1, box1.y2 - box1.y1);
|
|
for (i = 0; i < num_reg_rects; i++)
|
|
{
|
|
box2 = REGION_RECTS(®)[i];
|
|
rdpup_screen_blt(box2.x1 + dx, box2.y1 + dy, box2.x2 - box2.x1,
|
|
box2.y2 - box2.y1, box2.x1, box2.y1);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (j = num_clip_rects - 1; j >= 0; j--)
|
|
{
|
|
box1 = REGION_RECTS(&clip)[j];
|
|
rdpup_set_clip(box1.x1, box1.y1, box1.x2 - box1.x1, box1.y2 - box1.y1);
|
|
for (i = num_reg_rects - 1; i >= 0; i--)
|
|
{
|
|
box2 = REGION_RECTS(®)[i];
|
|
rdpup_screen_blt(box2.x1 + dx, box2.y1 + dy, box2.x2 - box2.x1,
|
|
box2.y2 - box2.y1, box2.x1, box2.y1);
|
|
}
|
|
}
|
|
}
|
|
rdpup_reset_clip();
|
|
rdpup_end_update();
|
|
RegionUninit(®);
|
|
RegionUninit(&clip);
|
|
g_pScreen->CopyWindow = rdpCopyWindow;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
void
|
|
rdpClearToBackground(WindowPtr pWin, int x, int y, int w, int h,
|
|
Bool generateExposures)
|
|
{
|
|
int j;
|
|
BoxRec box;
|
|
RegionRec reg;
|
|
|
|
DEBUG_OUT_OPS(("in rdpClearToBackground\n"));
|
|
g_pScreen->ClearToBackground = g_rdpScreen.ClearToBackground;
|
|
g_pScreen->ClearToBackground(pWin, x, y, w, h, generateExposures);
|
|
if (!generateExposures)
|
|
{
|
|
if (w > 0 && h > 0)
|
|
{
|
|
box.x1 = x;
|
|
box.y1 = y;
|
|
box.x2 = box.x1 + w;
|
|
box.y2 = box.y1 + h;
|
|
}
|
|
else
|
|
{
|
|
box.x1 = pWin->drawable.x;
|
|
box.y1 = pWin->drawable.y;
|
|
box.x2 = box.x1 + pWin->drawable.width;
|
|
box.y2 = box.y1 + pWin->drawable.height;
|
|
}
|
|
RegionInit(®, &box, 0);
|
|
RegionIntersect(®, ®, &pWin->clipList);
|
|
rdpup_begin_update();
|
|
for (j = REGION_NUM_RECTS(®) - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(®)[j];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_end_update();
|
|
RegionUninit(®);
|
|
}
|
|
g_pScreen->ClearToBackground = rdpClearToBackground;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
RegionPtr
|
|
rdpRestoreAreas(WindowPtr pWin, RegionPtr prgnExposed)
|
|
{
|
|
RegionRec reg;
|
|
RegionPtr rv;
|
|
int j;
|
|
BoxRec box;
|
|
|
|
DEBUG_OUT_OPS(("in rdpRestoreAreas\n"));
|
|
RegionInit(®, NullBox, 0);
|
|
RegionCopy(®, prgnExposed);
|
|
g_pScreen->RestoreAreas = g_rdpScreen.RestoreAreas;
|
|
rv = g_pScreen->RestoreAreas(pWin, prgnExposed);
|
|
rdpup_begin_update();
|
|
for (j = REGION_NUM_RECTS(®) - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(®)[j];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_end_update();
|
|
RegionUninit(®);
|
|
g_pScreen->RestoreAreas = rdpRestoreAreas;
|
|
return rv;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
void
|
|
rdpInstallColormap(ColormapPtr pmap)
|
|
{
|
|
ColormapPtr oldpmap;
|
|
|
|
oldpmap = g_rdpInstalledColormap;
|
|
if (pmap != oldpmap)
|
|
{
|
|
if (oldpmap != (ColormapPtr)None)
|
|
{
|
|
WalkTree(pmap->pScreen, TellLostMap, (char*)&oldpmap->mid);
|
|
}
|
|
/* Install pmap */
|
|
g_rdpInstalledColormap = pmap;
|
|
WalkTree(pmap->pScreen, TellGainedMap, (char*)&pmap->mid);
|
|
/*rfbSetClientColourMaps(0, 0);*/
|
|
}
|
|
/*g_rdpScreen.InstallColormap(pmap);*/
|
|
}
|
|
|
|
/******************************************************************************/
|
|
void
|
|
rdpUninstallColormap(ColormapPtr pmap)
|
|
{
|
|
ColormapPtr curpmap;
|
|
|
|
curpmap = g_rdpInstalledColormap;
|
|
if (pmap == curpmap)
|
|
{
|
|
if (pmap->mid != pmap->pScreen->defColormap)
|
|
{
|
|
//curpmap = (ColormapPtr)LookupIDByType(pmap->pScreen->defColormap,
|
|
// RT_COLORMAP);
|
|
//pmap->pScreen->InstallColormap(curpmap);
|
|
}
|
|
}
|
|
}
|
|
|
|
/******************************************************************************/
|
|
int
|
|
rdpListInstalledColormaps(ScreenPtr pScreen, Colormap* pmaps)
|
|
{
|
|
*pmaps = g_rdpInstalledColormap->mid;
|
|
return 1;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
void
|
|
rdpStoreColors(ColormapPtr pmap, int ndef, xColorItem* pdefs)
|
|
{
|
|
}
|
|
|
|
/******************************************************************************/
|
|
Bool
|
|
rdpSaveScreen(ScreenPtr pScreen, int on)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/* it looks like all the antialias draws go through here */
|
|
void
|
|
rdpComposite(CARD8 op, PicturePtr pSrc, PicturePtr pMask, PicturePtr pDst,
|
|
INT16 xSrc, INT16 ySrc, INT16 xMask, INT16 yMask, INT16 xDst,
|
|
INT16 yDst, CARD16 width, CARD16 height)
|
|
{
|
|
BoxRec box;
|
|
PictureScreenPtr ps;
|
|
RegionRec reg1;
|
|
RegionRec reg2;
|
|
DrawablePtr p;
|
|
int j;
|
|
int num_clips;
|
|
|
|
DEBUG_OUT_OPS(("in rdpComposite\n"));
|
|
ps = GetPictureScreen(g_pScreen);
|
|
ps->Composite = g_rdpScreen.Composite;
|
|
ps->Composite(op, pSrc, pMask, pDst, xSrc, ySrc,
|
|
xMask, yMask, xDst, yDst, width, height);
|
|
p = pDst->pDrawable;
|
|
if (p->type == DRAWABLE_WINDOW)
|
|
{
|
|
if (pDst->clientClipType == CT_REGION)
|
|
{
|
|
box.x1 = p->x + xDst;
|
|
box.y1 = p->y + yDst;
|
|
box.x2 = box.x1 + width;
|
|
box.y2 = box.y1 + height;
|
|
RegionInit(®1, &box, 0);
|
|
RegionInit(®2, NullBox, 0);
|
|
RegionCopy(®2, pDst->clientClip);
|
|
RegionTranslate(®2, p->x + pDst->clipOrigin.x,
|
|
p->y + pDst->clipOrigin.y);
|
|
RegionIntersect(®1, ®1, ®2);
|
|
num_clips = REGION_NUM_RECTS(®1);
|
|
if (num_clips > 0)
|
|
{
|
|
rdpup_begin_update();
|
|
for (j = num_clips - 1; j >= 0; j--)
|
|
{
|
|
box = REGION_RECTS(®1)[j];
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
}
|
|
rdpup_end_update();
|
|
}
|
|
RegionUninit(®1);
|
|
RegionUninit(®2);
|
|
}
|
|
else
|
|
{
|
|
box.x1 = p->x + xDst;
|
|
box.y1 = p->y + yDst;
|
|
box.x2 = box.x1 + width;
|
|
box.y2 = box.y1 + height;
|
|
rdpup_begin_update();
|
|
rdpup_send_area(box.x1, box.y1, box.x2 - box.x1, box.y2 - box.y1);
|
|
rdpup_end_update();
|
|
}
|
|
}
|
|
ps->Composite = rdpComposite;
|
|
}
|