2006-11-26 11:13:41 -07:00
|
|
|
/*
|
|
|
|
*Copyright (C) 1994-2000 The XFree86 Project, Inc. All Rights Reserved.
|
|
|
|
*
|
|
|
|
*Permission is hereby granted, free of charge, to any person obtaining
|
|
|
|
* a copy of this software and associated documentation files (the
|
|
|
|
*"Software"), to deal in the Software without restriction, including
|
|
|
|
*without limitation the rights to use, copy, modify, merge, publish,
|
|
|
|
*distribute, sublicense, and/or sell copies of the Software, and to
|
|
|
|
*permit persons to whom the Software is furnished to do so, subject to
|
|
|
|
*the following conditions:
|
|
|
|
*
|
|
|
|
*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 XFREE86 PROJECT 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 XFree86 Project
|
|
|
|
*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 XFree86 Project.
|
|
|
|
*
|
|
|
|
* Authors: Dakshinamurthy Karra
|
|
|
|
* Suhaib M Siddiqi
|
|
|
|
* Peter Busch
|
|
|
|
* Harold L Hunt II
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_XWIN_CONFIG_H
|
|
|
|
#include <xwin-config.h>
|
|
|
|
#endif
|
|
|
|
#include "win.h"
|
|
|
|
#include "winmsg.h"
|
|
|
|
#include <cursorstr.h>
|
|
|
|
#include <mipointrst.h>
|
|
|
|
#include <servermd.h>
|
2010-07-27 13:02:24 -06:00
|
|
|
#include "misc.h"
|
2006-11-26 11:13:41 -07:00
|
|
|
|
|
|
|
#define BRIGHTNESS(x) (x##Red * 0.299 + x##Green * 0.587 + x##Blue * 0.114)
|
|
|
|
|
|
|
|
#if 0
|
2012-06-10 07:21:05 -06:00
|
|
|
#define WIN_DEBUG_MSG winDebug
|
2006-11-26 11:13:41 -07:00
|
|
|
#else
|
2012-06-10 07:21:05 -06:00
|
|
|
#define WIN_DEBUG_MSG(...)
|
2006-11-26 11:13:41 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Local function prototypes
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void
|
2012-06-10 07:21:05 -06:00
|
|
|
winPointerWarpCursor(DeviceIntPtr pDev, ScreenPtr pScreen, int x, int y);
|
2006-11-26 11:13:41 -07:00
|
|
|
|
|
|
|
static Bool
|
2012-06-10 07:21:05 -06:00
|
|
|
winCursorOffScreen(ScreenPtr *ppScreen, int *x, int *y);
|
2006-11-26 11:13:41 -07:00
|
|
|
|
|
|
|
static void
|
2012-06-10 07:21:05 -06:00
|
|
|
winCrossScreen(ScreenPtr pScreen, Bool fEntering);
|
2006-11-26 11:13:41 -07:00
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
miPointerScreenFuncRec g_winPointerCursorFuncs = {
|
|
|
|
winCursorOffScreen,
|
|
|
|
winCrossScreen,
|
|
|
|
winPointerWarpCursor
|
2006-11-26 11:13:41 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
2012-06-10 07:21:05 -06:00
|
|
|
winPointerWarpCursor(DeviceIntPtr pDev, ScreenPtr pScreen, int x, int y)
|
2006-11-26 11:13:41 -07:00
|
|
|
{
|
2012-06-10 07:21:05 -06:00
|
|
|
winScreenPriv(pScreen);
|
|
|
|
RECT rcClient;
|
|
|
|
static Bool s_fInitialWarp = TRUE;
|
|
|
|
|
|
|
|
/* Discard first warp call */
|
|
|
|
if (s_fInitialWarp) {
|
|
|
|
/* First warp moves mouse to center of window, just ignore it */
|
|
|
|
|
|
|
|
/* Don't ignore subsequent warps */
|
|
|
|
s_fInitialWarp = FALSE;
|
|
|
|
|
|
|
|
winErrorFVerb(2,
|
|
|
|
"winPointerWarpCursor - Discarding first warp: %d %d\n",
|
|
|
|
x, y);
|
|
|
|
|
|
|
|
return;
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
/*
|
|
|
|
Only update the Windows cursor position if root window is active,
|
|
|
|
or we are in a rootless mode
|
|
|
|
*/
|
|
|
|
if ((pScreenPriv->hwndScreen == GetForegroundWindow())
|
|
|
|
|| pScreenPriv->pScreenInfo->fRootless
|
2010-07-27 13:02:24 -06:00
|
|
|
#ifdef XWIN_MULTIWINDOW
|
2012-06-10 07:21:05 -06:00
|
|
|
|| pScreenPriv->pScreenInfo->fMultiWindow
|
2010-07-27 13:02:24 -06:00
|
|
|
#endif
|
2012-06-10 07:21:05 -06:00
|
|
|
) {
|
|
|
|
/* Get the client area coordinates */
|
|
|
|
GetClientRect(pScreenPriv->hwndScreen, &rcClient);
|
|
|
|
|
|
|
|
/* Translate the client area coords to screen coords */
|
|
|
|
MapWindowPoints(pScreenPriv->hwndScreen,
|
2013-06-07 11:28:45 -06:00
|
|
|
HWND_DESKTOP, (LPPOINT) &rcClient, 2);
|
2012-06-10 07:21:05 -06:00
|
|
|
|
2015-09-16 13:10:19 -06:00
|
|
|
/*
|
2012-06-10 07:21:05 -06:00
|
|
|
* Update the Windows cursor position so that we don't
|
|
|
|
* immediately warp back to the current position.
|
|
|
|
*/
|
|
|
|
SetCursorPos(rcClient.left + x, rcClient.top + y);
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
/* Call the mi warp procedure to do the actual warping in X. */
|
|
|
|
miPointerWarpCursor(pDev, pScreen, x, y);
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static Bool
|
2012-06-10 07:21:05 -06:00
|
|
|
winCursorOffScreen(ScreenPtr *ppScreen, int *x, int *y)
|
2006-11-26 11:13:41 -07:00
|
|
|
{
|
2012-06-10 07:21:05 -06:00
|
|
|
return FALSE;
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2012-06-10 07:21:05 -06:00
|
|
|
winCrossScreen(ScreenPtr pScreen, Bool fEntering)
|
2006-11-26 11:13:41 -07:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned char
|
|
|
|
reverse(unsigned char c)
|
|
|
|
{
|
2012-06-10 07:21:05 -06:00
|
|
|
int i;
|
|
|
|
unsigned char ret = 0;
|
|
|
|
|
|
|
|
for (i = 0; i < 8; ++i) {
|
|
|
|
ret |= ((c >> i) & 1) << (7 - i);
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
2012-06-10 07:21:05 -06:00
|
|
|
return ret;
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Convert X cursor to Windows cursor
|
|
|
|
* FIXME: Perhaps there are more smart code
|
|
|
|
*/
|
|
|
|
static HCURSOR
|
2012-06-10 07:21:05 -06:00
|
|
|
winLoadCursor(ScreenPtr pScreen, CursorPtr pCursor, int screen)
|
2006-11-26 11:13:41 -07:00
|
|
|
{
|
2012-06-10 07:21:05 -06:00
|
|
|
winScreenPriv(pScreen);
|
|
|
|
HCURSOR hCursor = NULL;
|
|
|
|
unsigned char *pAnd;
|
|
|
|
unsigned char *pXor;
|
|
|
|
int nCX, nCY;
|
|
|
|
int nBytes;
|
|
|
|
double dForeY, dBackY;
|
|
|
|
BOOL fReverse;
|
|
|
|
HBITMAP hAnd, hXor;
|
|
|
|
ICONINFO ii;
|
|
|
|
unsigned char *pCur;
|
|
|
|
unsigned char bit;
|
|
|
|
HDC hDC;
|
|
|
|
BITMAPV4HEADER bi;
|
|
|
|
BITMAPINFO *pbmi;
|
2014-05-02 13:27:46 -06:00
|
|
|
uint32_t *lpBits;
|
2012-06-10 07:21:05 -06:00
|
|
|
|
|
|
|
WIN_DEBUG_MSG("winLoadCursor: Win32: %dx%d X11: %dx%d hotspot: %d,%d\n",
|
|
|
|
pScreenPriv->cursor.sm_cx, pScreenPriv->cursor.sm_cy,
|
|
|
|
pCursor->bits->width, pCursor->bits->height,
|
|
|
|
pCursor->bits->xhot, pCursor->bits->yhot);
|
|
|
|
|
2015-09-16 13:10:19 -06:00
|
|
|
/* We can use only White and Black, so calc brightness of color
|
2012-06-10 07:21:05 -06:00
|
|
|
* Also check if the cursor is inverted */
|
|
|
|
dForeY = BRIGHTNESS(pCursor->fore);
|
|
|
|
dBackY = BRIGHTNESS(pCursor->back);
|
|
|
|
fReverse = dForeY < dBackY;
|
|
|
|
|
|
|
|
/* Check wether the X11 cursor is bigger than the win32 cursor */
|
|
|
|
if (pScreenPriv->cursor.sm_cx < pCursor->bits->width ||
|
|
|
|
pScreenPriv->cursor.sm_cy < pCursor->bits->height) {
|
|
|
|
winErrorFVerb(3,
|
|
|
|
"winLoadCursor - Windows requires %dx%d cursor but X requires %dx%d\n",
|
|
|
|
pScreenPriv->cursor.sm_cx, pScreenPriv->cursor.sm_cy,
|
|
|
|
pCursor->bits->width, pCursor->bits->height);
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
|
|
|
|
2015-09-16 13:10:19 -06:00
|
|
|
/* Get the number of bytes required to store the whole cursor image
|
|
|
|
* This is roughly (sm_cx * sm_cy) / 8
|
2012-06-10 07:21:05 -06:00
|
|
|
* round up to 8 pixel boundary so we can convert whole bytes */
|
|
|
|
nBytes =
|
|
|
|
bits_to_bytes(pScreenPriv->cursor.sm_cx) * pScreenPriv->cursor.sm_cy;
|
|
|
|
|
|
|
|
/* Get the effective width and height */
|
|
|
|
nCX = min(pScreenPriv->cursor.sm_cx, pCursor->bits->width);
|
|
|
|
nCY = min(pScreenPriv->cursor.sm_cy, pCursor->bits->height);
|
|
|
|
|
|
|
|
/* Allocate memory for the bitmaps */
|
|
|
|
pAnd = malloc(nBytes);
|
|
|
|
memset(pAnd, 0xFF, nBytes);
|
|
|
|
pXor = calloc(1, nBytes);
|
|
|
|
|
2015-09-16 13:10:19 -06:00
|
|
|
/* Convert the X11 bitmap to a win32 bitmap
|
2012-06-10 07:21:05 -06:00
|
|
|
* The first is for an empty mask */
|
|
|
|
if (pCursor->bits->emptyMask) {
|
|
|
|
int x, y, xmax = bits_to_bytes(nCX);
|
|
|
|
|
|
|
|
for (y = 0; y < nCY; ++y)
|
|
|
|
for (x = 0; x < xmax; ++x) {
|
|
|
|
int nWinPix = bits_to_bytes(pScreenPriv->cursor.sm_cx) * y + x;
|
|
|
|
int nXPix = BitmapBytePad(pCursor->bits->width) * y + x;
|
|
|
|
|
|
|
|
pAnd[nWinPix] = 0;
|
|
|
|
if (fReverse)
|
|
|
|
pXor[nWinPix] = reverse(~pCursor->bits->source[nXPix]);
|
|
|
|
else
|
|
|
|
pXor[nWinPix] = reverse(pCursor->bits->source[nXPix]);
|
|
|
|
}
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
2012-06-10 07:21:05 -06:00
|
|
|
else {
|
|
|
|
int x, y, xmax = bits_to_bytes(nCX);
|
|
|
|
|
|
|
|
for (y = 0; y < nCY; ++y)
|
|
|
|
for (x = 0; x < xmax; ++x) {
|
|
|
|
int nWinPix = bits_to_bytes(pScreenPriv->cursor.sm_cx) * y + x;
|
|
|
|
int nXPix = BitmapBytePad(pCursor->bits->width) * y + x;
|
|
|
|
|
|
|
|
unsigned char mask = pCursor->bits->mask[nXPix];
|
|
|
|
|
|
|
|
pAnd[nWinPix] = reverse(~mask);
|
|
|
|
if (fReverse)
|
|
|
|
pXor[nWinPix] =
|
|
|
|
reverse(~pCursor->bits->source[nXPix] & mask);
|
|
|
|
else
|
|
|
|
pXor[nWinPix] =
|
|
|
|
reverse(pCursor->bits->source[nXPix] & mask);
|
|
|
|
}
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
/* prepare the pointers */
|
|
|
|
hCursor = NULL;
|
|
|
|
lpBits = NULL;
|
|
|
|
|
|
|
|
/* We have a truecolor alpha-blended cursor and can use it! */
|
|
|
|
if (pCursor->bits->argb) {
|
|
|
|
WIN_DEBUG_MSG("winLoadCursor: Trying truecolor alphablended cursor\n");
|
|
|
|
memset(&bi, 0, sizeof(BITMAPV4HEADER));
|
|
|
|
bi.bV4Size = sizeof(BITMAPV4HEADER);
|
|
|
|
bi.bV4Width = pScreenPriv->cursor.sm_cx;
|
|
|
|
bi.bV4Height = -(pScreenPriv->cursor.sm_cy); /* right-side up */
|
|
|
|
bi.bV4Planes = 1;
|
|
|
|
bi.bV4BitCount = 32;
|
|
|
|
bi.bV4V4Compression = BI_BITFIELDS;
|
|
|
|
bi.bV4RedMask = 0x00FF0000;
|
|
|
|
bi.bV4GreenMask = 0x0000FF00;
|
|
|
|
bi.bV4BlueMask = 0x000000FF;
|
|
|
|
bi.bV4AlphaMask = 0xFF000000;
|
|
|
|
|
2014-09-27 11:52:59 -06:00
|
|
|
lpBits = calloc(pScreenPriv->cursor.sm_cx * pScreenPriv->cursor.sm_cy,
|
|
|
|
sizeof(uint32_t));
|
2012-06-10 07:21:05 -06:00
|
|
|
|
|
|
|
if (lpBits) {
|
2013-06-07 11:28:45 -06:00
|
|
|
int y;
|
2012-06-10 07:21:05 -06:00
|
|
|
for (y = 0; y < nCY; y++) {
|
2014-05-02 13:27:46 -06:00
|
|
|
void *src, *dst;
|
2012-06-10 07:21:05 -06:00
|
|
|
src = &(pCursor->bits->argb[y * pCursor->bits->width]);
|
|
|
|
dst = &(lpBits[y * pScreenPriv->cursor.sm_cx]);
|
|
|
|
memcpy(dst, src, 4 * nCX);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} /* End if-truecolor-icon */
|
|
|
|
|
|
|
|
if (!lpBits) {
|
2014-05-02 13:27:46 -06:00
|
|
|
RGBQUAD *pbmiColors;
|
2012-06-10 07:21:05 -06:00
|
|
|
/* Bicolor, use a palettized DIB */
|
|
|
|
WIN_DEBUG_MSG("winLoadCursor: Trying two color cursor\n");
|
2013-06-07 11:28:45 -06:00
|
|
|
pbmi = (BITMAPINFO *) &bi;
|
2014-05-02 13:27:46 -06:00
|
|
|
pbmiColors = &(pbmi->bmiColors[0]);
|
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
memset(pbmi, 0, sizeof(BITMAPINFOHEADER));
|
|
|
|
pbmi->bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
|
|
|
|
pbmi->bmiHeader.biWidth = pScreenPriv->cursor.sm_cx;
|
|
|
|
pbmi->bmiHeader.biHeight = -abs(pScreenPriv->cursor.sm_cy); /* right-side up */
|
|
|
|
pbmi->bmiHeader.biPlanes = 1;
|
|
|
|
pbmi->bmiHeader.biBitCount = 8;
|
|
|
|
pbmi->bmiHeader.biCompression = BI_RGB;
|
|
|
|
pbmi->bmiHeader.biSizeImage = 0;
|
|
|
|
pbmi->bmiHeader.biClrUsed = 3;
|
|
|
|
pbmi->bmiHeader.biClrImportant = 3;
|
2014-05-02 13:27:46 -06:00
|
|
|
|
|
|
|
pbmiColors[0].rgbRed = 0; /* Empty */
|
|
|
|
pbmiColors[0].rgbGreen = 0;
|
|
|
|
pbmiColors[0].rgbBlue = 0;
|
|
|
|
pbmiColors[0].rgbReserved = 0;
|
|
|
|
pbmiColors[1].rgbRed = pCursor->backRed >> 8; /* Background */
|
|
|
|
pbmiColors[1].rgbGreen = pCursor->backGreen >> 8;
|
|
|
|
pbmiColors[1].rgbBlue = pCursor->backBlue >> 8;
|
|
|
|
pbmiColors[1].rgbReserved = 0;
|
|
|
|
pbmiColors[2].rgbRed = pCursor->foreRed >> 8; /* Foreground */
|
|
|
|
pbmiColors[2].rgbGreen = pCursor->foreGreen >> 8;
|
|
|
|
pbmiColors[2].rgbBlue = pCursor->foreBlue >> 8;
|
|
|
|
pbmiColors[2].rgbReserved = 0;
|
2012-06-10 07:21:05 -06:00
|
|
|
|
2014-09-27 11:52:59 -06:00
|
|
|
lpBits = calloc(pScreenPriv->cursor.sm_cx * pScreenPriv->cursor.sm_cy, 1);
|
2012-06-10 07:21:05 -06:00
|
|
|
|
|
|
|
pCur = (unsigned char *) lpBits;
|
|
|
|
if (lpBits) {
|
2013-06-07 11:28:45 -06:00
|
|
|
int x, y;
|
2012-06-10 07:21:05 -06:00
|
|
|
for (y = 0; y < pScreenPriv->cursor.sm_cy; y++) {
|
|
|
|
for (x = 0; x < pScreenPriv->cursor.sm_cx; x++) {
|
|
|
|
if (x >= nCX || y >= nCY) /* Outside of X11 icon bounds */
|
|
|
|
(*pCur++) = 0;
|
|
|
|
else { /* Within X11 icon bounds */
|
|
|
|
|
|
|
|
int nWinPix =
|
|
|
|
bits_to_bytes(pScreenPriv->cursor.sm_cx) * y +
|
|
|
|
(x / 8);
|
|
|
|
|
|
|
|
bit = pAnd[nWinPix];
|
|
|
|
bit = bit & (1 << (7 - (x & 7)));
|
|
|
|
if (!bit) { /* Within the cursor mask? */
|
|
|
|
int nXPix =
|
|
|
|
BitmapBytePad(pCursor->bits->width) * y +
|
|
|
|
(x / 8);
|
|
|
|
bit =
|
|
|
|
~reverse(~pCursor->bits->
|
|
|
|
source[nXPix] & pCursor->bits->
|
|
|
|
mask[nXPix]);
|
|
|
|
bit = bit & (1 << (7 - (x & 7)));
|
|
|
|
if (bit) /* Draw foreground */
|
|
|
|
(*pCur++) = 2;
|
|
|
|
else /* Draw background */
|
|
|
|
(*pCur++) = 1;
|
|
|
|
}
|
|
|
|
else /* Outside the cursor mask */
|
|
|
|
(*pCur++) = 0;
|
|
|
|
}
|
|
|
|
} /* end for (x) */
|
|
|
|
} /* end for (y) */
|
|
|
|
} /* end if (lpbits) */
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
/* If one of the previous two methods gave us the bitmap we need, make a cursor */
|
|
|
|
if (lpBits) {
|
|
|
|
WIN_DEBUG_MSG("winLoadCursor: Creating bitmap cursor: hotspot %d,%d\n",
|
|
|
|
pCursor->bits->xhot, pCursor->bits->yhot);
|
|
|
|
|
|
|
|
hAnd = NULL;
|
|
|
|
hXor = NULL;
|
|
|
|
|
|
|
|
hAnd =
|
|
|
|
CreateBitmap(pScreenPriv->cursor.sm_cx, pScreenPriv->cursor.sm_cy,
|
|
|
|
1, 1, pAnd);
|
|
|
|
|
|
|
|
hDC = GetDC(NULL);
|
|
|
|
if (hDC) {
|
|
|
|
hXor =
|
|
|
|
CreateCompatibleBitmap(hDC, pScreenPriv->cursor.sm_cx,
|
|
|
|
pScreenPriv->cursor.sm_cy);
|
|
|
|
SetDIBits(hDC, hXor, 0, pScreenPriv->cursor.sm_cy, lpBits,
|
2013-06-07 11:28:45 -06:00
|
|
|
(BITMAPINFO *) &bi, DIB_RGB_COLORS);
|
2012-06-10 07:21:05 -06:00
|
|
|
ReleaseDC(NULL, hDC);
|
|
|
|
}
|
|
|
|
free(lpBits);
|
|
|
|
|
|
|
|
if (hAnd && hXor) {
|
|
|
|
ii.fIcon = FALSE;
|
|
|
|
ii.xHotspot = pCursor->bits->xhot;
|
|
|
|
ii.yHotspot = pCursor->bits->yhot;
|
|
|
|
ii.hbmMask = hAnd;
|
|
|
|
ii.hbmColor = hXor;
|
|
|
|
hCursor = (HCURSOR) CreateIconIndirect(&ii);
|
|
|
|
|
|
|
|
if (hCursor == NULL)
|
|
|
|
winW32Error(2, "winLoadCursor - CreateIconIndirect failed:");
|
|
|
|
else {
|
|
|
|
if (GetIconInfo(hCursor, &ii)) {
|
|
|
|
if (ii.fIcon) {
|
|
|
|
WIN_DEBUG_MSG
|
|
|
|
("winLoadCursor: CreateIconIndirect returned no cursor. Trying again.\n");
|
|
|
|
|
|
|
|
DestroyCursor(hCursor);
|
|
|
|
|
|
|
|
ii.fIcon = FALSE;
|
|
|
|
ii.xHotspot = pCursor->bits->xhot;
|
|
|
|
ii.yHotspot = pCursor->bits->yhot;
|
|
|
|
hCursor = (HCURSOR) CreateIconIndirect(&ii);
|
|
|
|
|
|
|
|
if (hCursor == NULL)
|
|
|
|
winW32Error(2,
|
|
|
|
"winLoadCursor - CreateIconIndirect failed:");
|
|
|
|
}
|
|
|
|
/* GetIconInfo creates new bitmaps. Destroy them again */
|
|
|
|
if (ii.hbmMask)
|
|
|
|
DeleteObject(ii.hbmMask);
|
|
|
|
if (ii.hbmColor)
|
|
|
|
DeleteObject(ii.hbmColor);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hAnd)
|
|
|
|
DeleteObject(hAnd);
|
|
|
|
if (hXor)
|
|
|
|
DeleteObject(hXor);
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
if (!hCursor) {
|
|
|
|
/* We couldn't make a color cursor for this screen, use
|
|
|
|
black and white instead */
|
|
|
|
hCursor = CreateCursor(g_hInstance,
|
|
|
|
pCursor->bits->xhot, pCursor->bits->yhot,
|
|
|
|
pScreenPriv->cursor.sm_cx,
|
|
|
|
pScreenPriv->cursor.sm_cy, pAnd, pXor);
|
|
|
|
if (hCursor == NULL)
|
|
|
|
winW32Error(2, "winLoadCursor - CreateCursor failed:");
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
2012-06-10 07:21:05 -06:00
|
|
|
free(pAnd);
|
|
|
|
free(pXor);
|
2006-11-26 11:13:41 -07:00
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
return hCursor;
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
===========================================================================
|
|
|
|
|
|
|
|
Pointer sprite functions
|
|
|
|
|
|
|
|
===========================================================================
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* winRealizeCursor
|
|
|
|
* Convert the X cursor representation to native format if possible.
|
|
|
|
*/
|
|
|
|
static Bool
|
2012-06-10 07:21:05 -06:00
|
|
|
winRealizeCursor(DeviceIntPtr pDev, ScreenPtr pScreen, CursorPtr pCursor)
|
2006-11-26 11:13:41 -07:00
|
|
|
{
|
2012-06-10 07:21:05 -06:00
|
|
|
if (pCursor == NULL || pCursor->bits == NULL)
|
|
|
|
return FALSE;
|
2006-11-26 11:13:41 -07:00
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
/* FIXME: cache ARGB8888 representation? */
|
2006-11-26 11:13:41 -07:00
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
return TRUE;
|
|
|
|
}
|
2006-11-26 11:13:41 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* winUnrealizeCursor
|
|
|
|
* Free the storage space associated with a realized cursor.
|
|
|
|
*/
|
|
|
|
static Bool
|
2010-07-27 13:02:24 -06:00
|
|
|
winUnrealizeCursor(DeviceIntPtr pDev, ScreenPtr pScreen, CursorPtr pCursor)
|
2006-11-26 11:13:41 -07:00
|
|
|
{
|
2012-06-10 07:21:05 -06:00
|
|
|
return TRUE;
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* winSetCursor
|
|
|
|
* Set the cursor sprite and position.
|
|
|
|
*/
|
|
|
|
static void
|
2012-06-10 07:21:05 -06:00
|
|
|
winSetCursor(DeviceIntPtr pDev, ScreenPtr pScreen, CursorPtr pCursor, int x,
|
|
|
|
int y)
|
2006-11-26 11:13:41 -07:00
|
|
|
{
|
2012-06-10 07:21:05 -06:00
|
|
|
POINT ptCurPos, ptTemp;
|
|
|
|
HWND hwnd;
|
|
|
|
RECT rcClient;
|
|
|
|
BOOL bInhibit;
|
|
|
|
|
|
|
|
winScreenPriv(pScreen);
|
|
|
|
WIN_DEBUG_MSG("winSetCursor: cursor=%p\n", pCursor);
|
|
|
|
|
|
|
|
/* Inhibit changing the cursor if the mouse is not in a client area */
|
|
|
|
bInhibit = FALSE;
|
|
|
|
if (GetCursorPos(&ptCurPos)) {
|
|
|
|
hwnd = WindowFromPoint(ptCurPos);
|
|
|
|
if (hwnd) {
|
|
|
|
if (GetClientRect(hwnd, &rcClient)) {
|
|
|
|
ptTemp.x = rcClient.left;
|
|
|
|
ptTemp.y = rcClient.top;
|
|
|
|
if (ClientToScreen(hwnd, &ptTemp)) {
|
|
|
|
rcClient.left = ptTemp.x;
|
|
|
|
rcClient.top = ptTemp.y;
|
|
|
|
ptTemp.x = rcClient.right;
|
|
|
|
ptTemp.y = rcClient.bottom;
|
|
|
|
if (ClientToScreen(hwnd, &ptTemp)) {
|
|
|
|
rcClient.right = ptTemp.x;
|
|
|
|
rcClient.bottom = ptTemp.y;
|
|
|
|
if (!PtInRect(&rcClient, ptCurPos))
|
|
|
|
bInhibit = TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
if (pCursor == NULL) {
|
|
|
|
if (pScreenPriv->cursor.visible) {
|
|
|
|
if (!bInhibit && g_fSoftwareCursor)
|
|
|
|
ShowCursor(FALSE);
|
|
|
|
pScreenPriv->cursor.visible = FALSE;
|
|
|
|
}
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
2012-06-10 07:21:05 -06:00
|
|
|
else {
|
|
|
|
if (pScreenPriv->cursor.handle) {
|
|
|
|
if (!bInhibit)
|
|
|
|
SetCursor(NULL);
|
|
|
|
DestroyCursor(pScreenPriv->cursor.handle);
|
|
|
|
pScreenPriv->cursor.handle = NULL;
|
|
|
|
}
|
|
|
|
pScreenPriv->cursor.handle =
|
|
|
|
winLoadCursor(pScreen, pCursor, pScreen->myNum);
|
|
|
|
WIN_DEBUG_MSG("winSetCursor: handle=%p\n", pScreenPriv->cursor.handle);
|
|
|
|
|
|
|
|
if (!bInhibit)
|
|
|
|
SetCursor(pScreenPriv->cursor.handle);
|
|
|
|
|
|
|
|
if (!pScreenPriv->cursor.visible) {
|
|
|
|
if (!bInhibit && g_fSoftwareCursor)
|
|
|
|
ShowCursor(TRUE);
|
|
|
|
pScreenPriv->cursor.visible = TRUE;
|
|
|
|
}
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2010-07-27 13:02:24 -06:00
|
|
|
* winMoveCursor
|
2006-11-26 11:13:41 -07:00
|
|
|
* Move the cursor. This is a noop for us.
|
|
|
|
*/
|
|
|
|
static void
|
2012-06-10 07:21:05 -06:00
|
|
|
winMoveCursor(DeviceIntPtr pDev, ScreenPtr pScreen, int x, int y)
|
2010-07-27 13:02:24 -06:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static Bool
|
|
|
|
winDeviceCursorInitialize(DeviceIntPtr pDev, ScreenPtr pScr)
|
2006-11-26 11:13:41 -07:00
|
|
|
{
|
2012-06-10 07:21:05 -06:00
|
|
|
winScreenPriv(pScr);
|
|
|
|
return pScreenPriv->cursor.spriteFuncs->DeviceCursorInitialize(pDev, pScr);
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
|
|
|
|
2010-07-27 13:02:24 -06:00
|
|
|
static void
|
|
|
|
winDeviceCursorCleanup(DeviceIntPtr pDev, ScreenPtr pScr)
|
|
|
|
{
|
2012-06-10 07:21:05 -06:00
|
|
|
winScreenPriv(pScr);
|
|
|
|
pScreenPriv->cursor.spriteFuncs->DeviceCursorCleanup(pDev, pScr);
|
2010-07-27 13:02:24 -06:00
|
|
|
}
|
2006-11-26 11:13:41 -07:00
|
|
|
|
|
|
|
static miPointerSpriteFuncRec winSpriteFuncsRec = {
|
2012-06-10 07:21:05 -06:00
|
|
|
winRealizeCursor,
|
|
|
|
winUnrealizeCursor,
|
|
|
|
winSetCursor,
|
|
|
|
winMoveCursor,
|
|
|
|
winDeviceCursorInitialize,
|
|
|
|
winDeviceCursorCleanup
|
2006-11-26 11:13:41 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
===========================================================================
|
|
|
|
|
|
|
|
Other screen functions
|
|
|
|
|
|
|
|
===========================================================================
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* winCursorQueryBestSize
|
|
|
|
* Handle queries for best cursor size
|
|
|
|
*/
|
|
|
|
static void
|
2012-06-10 07:21:05 -06:00
|
|
|
winCursorQueryBestSize(int class, unsigned short *width,
|
|
|
|
unsigned short *height, ScreenPtr pScreen)
|
2006-11-26 11:13:41 -07:00
|
|
|
{
|
2012-06-10 07:21:05 -06:00
|
|
|
winScreenPriv(pScreen);
|
|
|
|
|
|
|
|
if (class == CursorShape) {
|
|
|
|
*width = pScreenPriv->cursor.sm_cx;
|
|
|
|
*height = pScreenPriv->cursor.sm_cy;
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
2012-06-10 07:21:05 -06:00
|
|
|
else {
|
|
|
|
if (pScreenPriv->cursor.QueryBestSize)
|
|
|
|
(*pScreenPriv->cursor.QueryBestSize) (class, width, height,
|
|
|
|
pScreen);
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* winInitCursor
|
|
|
|
* Initialize cursor support
|
|
|
|
*/
|
|
|
|
Bool
|
2012-06-10 07:21:05 -06:00
|
|
|
winInitCursor(ScreenPtr pScreen)
|
2006-11-26 11:13:41 -07:00
|
|
|
{
|
2012-06-10 07:21:05 -06:00
|
|
|
winScreenPriv(pScreen);
|
|
|
|
miPointerScreenPtr pPointPriv;
|
|
|
|
|
|
|
|
/* override some screen procedures */
|
|
|
|
pScreenPriv->cursor.QueryBestSize = pScreen->QueryBestSize;
|
|
|
|
pScreen->QueryBestSize = winCursorQueryBestSize;
|
|
|
|
|
|
|
|
pPointPriv = (miPointerScreenPtr)
|
|
|
|
dixLookupPrivate(&pScreen->devPrivates, miPointerScreenKey);
|
|
|
|
|
|
|
|
pScreenPriv->cursor.spriteFuncs = pPointPriv->spriteFuncs;
|
|
|
|
pPointPriv->spriteFuncs = &winSpriteFuncsRec;
|
|
|
|
|
|
|
|
pScreenPriv->cursor.handle = NULL;
|
|
|
|
pScreenPriv->cursor.visible = FALSE;
|
|
|
|
|
|
|
|
pScreenPriv->cursor.sm_cx = GetSystemMetrics(SM_CXCURSOR);
|
|
|
|
pScreenPriv->cursor.sm_cy = GetSystemMetrics(SM_CYCURSOR);
|
|
|
|
|
|
|
|
return TRUE;
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|