16747b715Smrg/* 26747b715Smrg * Copyright 2007-2008 Peter Hutterer 36747b715Smrg * 46747b715Smrg * Permission is hereby granted, free of charge, to any person obtaining a 56747b715Smrg * copy of this software and associated documentation files (the "Software"), 66747b715Smrg * to deal in the Software without restriction, including without limitation 76747b715Smrg * the rights to use, copy, modify, merge, publish, distribute, sublicense, 86747b715Smrg * and/or sell copies of the Software, and to permit persons to whom the 96747b715Smrg * Software is furnished to do so, subject to the following conditions: 106747b715Smrg * 116747b715Smrg * The above copyright notice and this permission notice (including the next 126747b715Smrg * paragraph) shall be included in all copies or substantial portions of the 136747b715Smrg * Software. 146747b715Smrg * 156747b715Smrg * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 166747b715Smrg * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 176747b715Smrg * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL 186747b715Smrg * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 196747b715Smrg * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 206747b715Smrg * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER 216747b715Smrg * DEALINGS IN THE SOFTWARE. 226747b715Smrg * 236747b715Smrg * Author: Peter Hutterer, University of South Australia, NICTA 246747b715Smrg */ 256747b715Smrg 266747b715Smrg/*********************************************************************** 276747b715Smrg * 286747b715Smrg * Request to Warp the pointer location of an extension input device. 296747b715Smrg * 306747b715Smrg */ 316747b715Smrg 326747b715Smrg#ifdef HAVE_DIX_CONFIG_H 336747b715Smrg#include <dix-config.h> 346747b715Smrg#endif 356747b715Smrg 36f7df2e56Smrg#include <X11/X.h> /* for inputstr.h */ 37f7df2e56Smrg#include <X11/Xproto.h> /* Request macro */ 38f7df2e56Smrg#include "inputstr.h" /* DeviceIntPtr */ 39f7df2e56Smrg#include "windowstr.h" /* window structure */ 40f7df2e56Smrg#include "scrnintstr.h" /* screen structure */ 416747b715Smrg#include <X11/extensions/XI.h> 426747b715Smrg#include <X11/extensions/XI2proto.h> 436747b715Smrg#include "extnsionst.h" 446747b715Smrg#include "exevents.h" 456747b715Smrg#include "exglobals.h" 46f7df2e56Smrg#include "mipointer.h" /* for miPointerUpdateSprite */ 476747b715Smrg 486747b715Smrg#include "xiwarppointer.h" 496747b715Smrg/*********************************************************************** 506747b715Smrg * 516747b715Smrg * This procedure allows a client to warp the pointer of a device. 526747b715Smrg * 536747b715Smrg */ 546747b715Smrg 557e31ba66Smrgint _X_COLD 566747b715SmrgSProcXIWarpPointer(ClientPtr client) 576747b715Smrg{ 586747b715Smrg REQUEST(xXIWarpPointerReq); 590b0d8713Smrg REQUEST_SIZE_MATCH(xXIWarpPointerReq); 600b0d8713Smrg 61f7df2e56Smrg swaps(&stuff->length); 62f7df2e56Smrg swapl(&stuff->src_win); 63f7df2e56Smrg swapl(&stuff->dst_win); 64f7df2e56Smrg swapl(&stuff->src_x); 65f7df2e56Smrg swapl(&stuff->src_y); 66f7df2e56Smrg swaps(&stuff->src_width); 67f7df2e56Smrg swaps(&stuff->src_height); 68f7df2e56Smrg swapl(&stuff->dst_x); 69f7df2e56Smrg swapl(&stuff->dst_y); 70f7df2e56Smrg swaps(&stuff->deviceid); 716747b715Smrg return (ProcXIWarpPointer(client)); 726747b715Smrg} 736747b715Smrg 746747b715Smrgint 756747b715SmrgProcXIWarpPointer(ClientPtr client) 766747b715Smrg{ 776747b715Smrg int rc; 786747b715Smrg int x, y; 796747b715Smrg WindowPtr dest = NULL; 806747b715Smrg DeviceIntPtr pDev; 816747b715Smrg SpritePtr pSprite; 826747b715Smrg ScreenPtr newScreen; 836747b715Smrg int src_x, src_y; 846747b715Smrg int dst_x, dst_y; 856747b715Smrg 866747b715Smrg REQUEST(xXIWarpPointerReq); 876747b715Smrg REQUEST_SIZE_MATCH(xXIWarpPointerReq); 886747b715Smrg 896747b715Smrg /* FIXME: panoramix stuff is missing, look at ProcWarpPointer */ 906747b715Smrg 916747b715Smrg rc = dixLookupDevice(&pDev, stuff->deviceid, client, DixWriteAccess); 926747b715Smrg 93f7df2e56Smrg if (rc != Success) { 946747b715Smrg client->errorValue = stuff->deviceid; 956747b715Smrg return rc; 966747b715Smrg } 976747b715Smrg 98f7df2e56Smrg if ((!IsMaster(pDev) && !IsFloating(pDev)) || 99f7df2e56Smrg (IsMaster(pDev) && !IsPointerDevice(pDev))) { 1006747b715Smrg client->errorValue = stuff->deviceid; 1016747b715Smrg return BadDevice; 1026747b715Smrg } 1036747b715Smrg 104f7df2e56Smrg if (stuff->dst_win != None) { 1056747b715Smrg rc = dixLookupWindow(&dest, stuff->dst_win, client, DixGetAttrAccess); 106f7df2e56Smrg if (rc != Success) { 1076747b715Smrg client->errorValue = stuff->dst_win; 1086747b715Smrg return rc; 1096747b715Smrg } 1106747b715Smrg } 1116747b715Smrg 1126747b715Smrg pSprite = pDev->spriteInfo->sprite; 1136747b715Smrg x = pSprite->hotPhys.x; 1146747b715Smrg y = pSprite->hotPhys.y; 1156747b715Smrg 116f7df2e56Smrg src_x = stuff->src_x / (double) (1 << 16); 117f7df2e56Smrg src_y = stuff->src_y / (double) (1 << 16); 118f7df2e56Smrg dst_x = stuff->dst_x / (double) (1 << 16); 119f7df2e56Smrg dst_y = stuff->dst_y / (double) (1 << 16); 1206747b715Smrg 121f7df2e56Smrg if (stuff->src_win != None) { 1226747b715Smrg int winX, winY; 1236747b715Smrg WindowPtr src; 1246747b715Smrg 1256747b715Smrg rc = dixLookupWindow(&src, stuff->src_win, client, DixGetAttrAccess); 126f7df2e56Smrg if (rc != Success) { 1276747b715Smrg client->errorValue = stuff->src_win; 1286747b715Smrg return rc; 1296747b715Smrg } 1306747b715Smrg 1316747b715Smrg winX = src->drawable.x; 1326747b715Smrg winY = src->drawable.y; 1336747b715Smrg if (src->drawable.pScreen != pSprite->hotPhys.pScreen || 134f7df2e56Smrg x < winX + src_x || 135f7df2e56Smrg y < winY + src_y || 136f7df2e56Smrg (stuff->src_width != 0 && 137f7df2e56Smrg winX + src_x + (int) stuff->src_width < 0) || 138f7df2e56Smrg (stuff->src_height != 0 && 139f7df2e56Smrg winY + src_y + (int) stuff->src_height < y) || 140f7df2e56Smrg !PointInWindowIsVisible(src, x, y)) 1416747b715Smrg return Success; 1426747b715Smrg } 1436747b715Smrg 144f7df2e56Smrg if (dest) { 1456747b715Smrg x = dest->drawable.x; 1466747b715Smrg y = dest->drawable.y; 1476747b715Smrg newScreen = dest->drawable.pScreen; 148f7df2e56Smrg } 149f7df2e56Smrg else 1506747b715Smrg newScreen = pSprite->hotPhys.pScreen; 1516747b715Smrg 1526747b715Smrg x += dst_x; 1536747b715Smrg y += dst_y; 1546747b715Smrg 1556747b715Smrg if (x < 0) 1566747b715Smrg x = 0; 1576747b715Smrg else if (x > newScreen->width) 1586747b715Smrg x = newScreen->width - 1; 1596747b715Smrg 1606747b715Smrg if (y < 0) 1616747b715Smrg y = 0; 1626747b715Smrg else if (y > newScreen->height) 1636747b715Smrg y = newScreen->height - 1; 1646747b715Smrg 165f7df2e56Smrg if (newScreen == pSprite->hotPhys.pScreen) { 1666747b715Smrg if (x < pSprite->physLimits.x1) 1676747b715Smrg x = pSprite->physLimits.x1; 1686747b715Smrg else if (x >= pSprite->physLimits.x2) 1696747b715Smrg x = pSprite->physLimits.x2 - 1; 1706747b715Smrg 1716747b715Smrg if (y < pSprite->physLimits.y1) 1726747b715Smrg y = pSprite->physLimits.y1; 1736747b715Smrg else if (y >= pSprite->physLimits.y2) 1746747b715Smrg y = pSprite->physLimits.y2 - 1; 1756747b715Smrg 1766747b715Smrg if (pSprite->hotShape) 1776747b715Smrg ConfineToShape(pDev, pSprite->hotShape, &x, &y); 178f7df2e56Smrg (*newScreen->SetCursorPosition) (pDev, newScreen, x, y, TRUE); 179f7df2e56Smrg } 180f7df2e56Smrg else if (!PointerConfinedToScreen(pDev)) { 1816747b715Smrg NewCurrentScreen(pDev, newScreen, x, y); 1826747b715Smrg } 1836747b715Smrg 1846747b715Smrg /* if we don't update the device, we get a jump next time it moves */ 1856747b715Smrg pDev->last.valuators[0] = x; 1866747b715Smrg pDev->last.valuators[1] = y; 1876747b715Smrg miPointerUpdateSprite(pDev); 1886747b715Smrg 1897e31ba66Smrg if (*newScreen->CursorWarpedTo) 1907e31ba66Smrg (*newScreen->CursorWarpedTo) (pDev, newScreen, client, 1917e31ba66Smrg dest, pSprite, x, y); 1927e31ba66Smrg 1936747b715Smrg /* FIXME: XWarpPointer is supposed to generate an event. It doesn't do it 1946747b715Smrg here though. */ 1956747b715Smrg return Success; 1966747b715Smrg} 197