| /* |
| * X11DRV clipping functions |
| * |
| * Copyright 1998 Huw Davies |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * This library 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 |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; if not, write to the Free Software |
| * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
| */ |
| |
| #include "config.h" |
| |
| #include "ts_xlib.h" |
| |
| #include <stdio.h> |
| |
| #include "gdi.h" |
| #include "x11drv.h" |
| #include "wine/debug.h" |
| |
| WINE_DEFAULT_DEBUG_CHANNEL(x11drv); |
| |
| /*********************************************************************** |
| * X11DRV_GetRegionData |
| * |
| * Calls GetRegionData on the given region and converts the rectangle |
| * array to XRectangle format. The returned buffer must be freed by |
| * caller using HeapFree(GetProcessHeap(),...). |
| * If hdc_lptodp is not 0, the rectangles are converted through LPtoDP. |
| */ |
| RGNDATA *X11DRV_GetRegionData( HRGN hrgn, HDC hdc_lptodp ) |
| { |
| RGNDATA *data; |
| DWORD size; |
| int i; |
| RECT *rect, tmp; |
| XRectangle *xrect; |
| |
| if (!(size = GetRegionData( hrgn, 0, NULL ))) return NULL; |
| if (sizeof(XRectangle) > sizeof(RECT)) |
| { |
| /* add extra size for XRectangle array */ |
| int count = (size - sizeof(RGNDATAHEADER)) / sizeof(RECT); |
| size += count * (sizeof(XRectangle) - sizeof(RECT)); |
| } |
| if (!(data = HeapAlloc( GetProcessHeap(), 0, size ))) return NULL; |
| if (!GetRegionData( hrgn, size, data )) |
| { |
| HeapFree( GetProcessHeap(), 0, data ); |
| return NULL; |
| } |
| |
| rect = (RECT *)data->Buffer; |
| xrect = (XRectangle *)data->Buffer; |
| if (hdc_lptodp) /* map to device coordinates */ |
| { |
| LPtoDP( hdc_lptodp, (POINT *)rect, data->rdh.nCount * 2 ); |
| for (i = 0; i < data->rdh.nCount; i++) |
| { |
| if (rect[i].right < rect[i].left) |
| { |
| INT tmp = rect[i].right; |
| rect[i].right = rect[i].left; |
| rect[i].left = tmp; |
| } |
| if (rect[i].bottom < rect[i].top) |
| { |
| INT tmp = rect[i].bottom; |
| rect[i].bottom = rect[i].top; |
| rect[i].top = tmp; |
| } |
| } |
| } |
| |
| if (sizeof(XRectangle) > sizeof(RECT)) |
| { |
| /* need to start from the end */ |
| for (i = data->rdh.nCount-1; i >=0; i--) |
| { |
| tmp = rect[i]; |
| xrect[i].x = tmp.left; |
| xrect[i].y = tmp.top; |
| xrect[i].width = tmp.right - tmp.left; |
| xrect[i].height = tmp.bottom - tmp.top; |
| } |
| } |
| else |
| { |
| for (i = 0; i < data->rdh.nCount; i++) |
| { |
| tmp = rect[i]; |
| xrect[i].x = tmp.left; |
| xrect[i].y = tmp.top; |
| xrect[i].width = tmp.right - tmp.left; |
| xrect[i].height = tmp.bottom - tmp.top; |
| } |
| } |
| return data; |
| } |
| |
| |
| /*********************************************************************** |
| * X11DRV_SetDeviceClipping |
| */ |
| void X11DRV_SetDeviceClipping( X11DRV_PDEVICE *physDev, HRGN hrgn ) |
| { |
| RGNDATA *data; |
| |
| if (!(data = X11DRV_GetRegionData( hrgn, 0 ))) return; |
| TSXSetClipRectangles( gdi_display, physDev->gc, physDev->org.x, physDev->org.y, |
| (XRectangle *)data->Buffer, data->rdh.nCount, YXBanded ); |
| HeapFree( GetProcessHeap(), 0, data ); |
| } |
| |
| |
| /*********************************************************************** |
| * X11DRV_SetDrawable |
| * |
| * Set the drawable, clipping mode and origin for a DC. |
| */ |
| void X11DRV_SetDrawable( HDC hdc, Drawable drawable, int mode, const POINT *org, |
| const POINT *drawable_org ) |
| { |
| DC *dc = DC_GetDCPtr( hdc ); |
| if (dc) |
| { |
| X11DRV_PDEVICE *physDev = (X11DRV_PDEVICE *)dc->physDev; |
| |
| physDev->org = *org; |
| physDev->drawable = drawable; |
| physDev->drawable_org = *drawable_org; |
| TSXSetSubwindowMode( gdi_display, physDev->gc, mode ); |
| if(physDev->xrender) |
| X11DRV_XRender_UpdateDrawable( physDev ); |
| GDI_ReleaseObj( hdc ); |
| } |
| } |
| |
| |
| /*********************************************************************** |
| * X11DRV_StartGraphicsExposures |
| * |
| * Set the DC in graphics exposures mode |
| */ |
| void X11DRV_StartGraphicsExposures( HDC hdc ) |
| { |
| DC *dc = DC_GetDCPtr( hdc ); |
| if (dc) |
| { |
| X11DRV_PDEVICE *physDev = (X11DRV_PDEVICE *)dc->physDev; |
| TSXSetGraphicsExposures( gdi_display, physDev->gc, True ); |
| physDev->exposures = 0; |
| GDI_ReleaseObj( hdc ); |
| } |
| } |
| |
| |
| /*********************************************************************** |
| * X11DRV_EndGraphicsExposures |
| * |
| * End the graphics exposures mode and process the events |
| */ |
| void X11DRV_EndGraphicsExposures( HDC hdc, HRGN hrgn ) |
| { |
| HRGN tmp = 0; |
| DC *dc = DC_GetDCPtr( hdc ); |
| |
| if (dc) |
| { |
| XEvent event; |
| X11DRV_PDEVICE *physDev = (X11DRV_PDEVICE *)dc->physDev; |
| |
| SetRectRgn( hrgn, 0, 0, 0, 0 ); |
| wine_tsx11_lock(); |
| XSetGraphicsExposures( gdi_display, physDev->gc, False ); |
| if (physDev->exposures) |
| { |
| XSync( gdi_display, False ); |
| for (;;) |
| { |
| XWindowEvent( gdi_display, physDev->drawable, ~0, &event ); |
| if (event.type == NoExpose) break; |
| if (event.type == GraphicsExpose) |
| { |
| int x = event.xgraphicsexpose.x - physDev->org.x; |
| int y = event.xgraphicsexpose.y - physDev->org.y; |
| |
| TRACE( "got %d,%d %dx%d count %d\n", |
| x, y, event.xgraphicsexpose.width, event.xgraphicsexpose.height, |
| event.xgraphicsexpose.count ); |
| |
| if (!tmp) tmp = CreateRectRgn( 0, 0, 0, 0 ); |
| SetRectRgn( tmp, x, y, |
| x + event.xgraphicsexpose.width, |
| y + event.xgraphicsexpose.height ); |
| CombineRgn( hrgn, hrgn, tmp, RGN_OR ); |
| if (!event.xgraphicsexpose.count) break; |
| } |
| else |
| { |
| ERR( "got unexpected event %d\n", event.type ); |
| break; |
| } |
| if (tmp) DeleteObject( tmp ); |
| } |
| } |
| wine_tsx11_unlock(); |
| GDI_ReleaseObj( hdc ); |
| } |
| } |