2006-11-26 11:13:41 -07:00
|
|
|
/**************************************************************
|
|
|
|
*
|
|
|
|
* Xplugin cursor support
|
|
|
|
*
|
|
|
|
* Copyright (c) 2001 Torrey T. Lyons and Greg Parker.
|
|
|
|
* Copyright (c) 2002 Apple Computer, 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 ABOVE LISTED COPYRIGHT HOLDER(S) 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(s) of the above copyright
|
|
|
|
* holders shall not be used in advertising or otherwise to promote the sale,
|
|
|
|
* use or other dealings in this Software without prior written authorization.
|
|
|
|
*/
|
2008-11-02 08:26:08 -07:00
|
|
|
|
2009-09-06 13:44:18 -06:00
|
|
|
#include "sanitizedCarbon.h"
|
|
|
|
|
2008-11-02 08:26:08 -07:00
|
|
|
#ifdef HAVE_DIX_CONFIG_H
|
|
|
|
#include <dix-config.h>
|
2007-11-24 10:55:21 -07:00
|
|
|
#endif
|
2008-11-02 08:26:08 -07:00
|
|
|
|
2010-12-05 08:36:02 -07:00
|
|
|
#include "quartz.h"
|
2006-11-26 11:13:41 -07:00
|
|
|
#include "xpr.h"
|
2008-11-02 08:26:08 -07:00
|
|
|
#include "darwinEvents.h"
|
2009-09-06 13:44:18 -06:00
|
|
|
#include <Xplugin.h>
|
2006-11-26 11:13:41 -07:00
|
|
|
|
|
|
|
#include "mi.h"
|
|
|
|
#include "scrnintstr.h"
|
|
|
|
#include "cursorstr.h"
|
|
|
|
#include "mipointrst.h"
|
|
|
|
#include "windowstr.h"
|
|
|
|
#include "globals.h"
|
|
|
|
#include "servermd.h"
|
|
|
|
#include "dixevents.h"
|
2009-09-06 13:44:18 -06:00
|
|
|
#include "x-hash.h"
|
2006-11-26 11:13:41 -07:00
|
|
|
|
|
|
|
typedef struct {
|
2012-06-10 07:21:05 -06:00
|
|
|
int cursorVisible;
|
|
|
|
QueryBestSizeProcPtr QueryBestSize;
|
|
|
|
miPointerSpriteFuncPtr spriteFuncs;
|
2006-11-26 11:13:41 -07:00
|
|
|
} QuartzCursorScreenRec, *QuartzCursorScreenPtr;
|
|
|
|
|
2010-12-05 08:36:02 -07:00
|
|
|
static DevPrivateKeyRec darwinCursorScreenKeyRec;
|
|
|
|
#define darwinCursorScreenKey (&darwinCursorScreenKeyRec)
|
2006-11-26 11:13:41 -07:00
|
|
|
|
2008-11-02 08:26:08 -07:00
|
|
|
#define CURSOR_PRIV(pScreen) ((QuartzCursorScreenPtr) \
|
2012-06-10 07:21:05 -06:00
|
|
|
dixLookupPrivate(&pScreen->devPrivates, \
|
|
|
|
darwinCursorScreenKey))
|
2006-11-26 11:13:41 -07:00
|
|
|
|
|
|
|
static Bool
|
|
|
|
load_cursor(CursorPtr src, int screen)
|
|
|
|
{
|
|
|
|
uint32_t *data;
|
2010-12-05 08:36:02 -07:00
|
|
|
Bool free_data = FALSE;
|
2006-11-26 11:13:41 -07:00
|
|
|
uint32_t rowbytes;
|
|
|
|
int width, height;
|
|
|
|
int hot_x, hot_y;
|
|
|
|
|
|
|
|
uint32_t fg_color, bg_color;
|
|
|
|
uint8_t *srow, *sptr;
|
|
|
|
uint8_t *mrow, *mptr;
|
|
|
|
uint32_t *drow, *dptr;
|
|
|
|
unsigned xcount, ycount;
|
|
|
|
|
|
|
|
xp_error err;
|
|
|
|
|
|
|
|
width = src->bits->width;
|
|
|
|
height = src->bits->height;
|
|
|
|
hot_x = src->bits->xhot;
|
|
|
|
hot_y = src->bits->yhot;
|
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
if (src->bits->argb != NULL) {
|
2008-11-02 08:26:08 -07:00
|
|
|
#if BITMAP_BIT_ORDER == MSBFirst
|
2012-06-10 07:21:05 -06:00
|
|
|
rowbytes = src->bits->width * sizeof(CARD32);
|
|
|
|
data = (uint32_t *)src->bits->argb;
|
2008-11-02 08:26:08 -07:00
|
|
|
#else
|
2012-06-10 07:21:05 -06:00
|
|
|
const uint32_t *be_data = (uint32_t *)src->bits->argb;
|
2008-11-02 08:26:08 -07:00
|
|
|
unsigned i;
|
2012-06-10 07:21:05 -06:00
|
|
|
rowbytes = src->bits->width * sizeof(CARD32);
|
2010-12-05 08:36:02 -07:00
|
|
|
data = malloc(rowbytes * src->bits->height);
|
|
|
|
free_data = TRUE;
|
2012-06-10 07:21:05 -06:00
|
|
|
if (!data) {
|
2010-07-27 13:02:24 -06:00
|
|
|
FatalError("Failed to allocate memory in %s\n", __func__);
|
|
|
|
}
|
2012-06-10 07:21:05 -06:00
|
|
|
for (i = 0; i < (src->bits->width * src->bits->height); i++)
|
|
|
|
data[i] = ntohl(be_data[i]);
|
2008-11-02 08:26:08 -07:00
|
|
|
#endif
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fg_color = 0xFF00 | (src->foreRed >> 8);
|
|
|
|
fg_color <<= 16;
|
|
|
|
fg_color |= src->foreGreen & 0xFF00;
|
|
|
|
fg_color |= src->foreBlue >> 8;
|
|
|
|
|
|
|
|
bg_color = 0xFF00 | (src->backRed >> 8);
|
|
|
|
bg_color <<= 16;
|
|
|
|
bg_color |= src->backGreen & 0xFF00;
|
|
|
|
bg_color |= src->backBlue >> 8;
|
|
|
|
|
|
|
|
fg_color = htonl(fg_color);
|
|
|
|
bg_color = htonl(bg_color);
|
|
|
|
|
|
|
|
/* round up to 8 pixel boundary so we can convert whole bytes */
|
|
|
|
rowbytes = ((src->bits->width * 4) + 31) & ~31;
|
2010-12-05 08:36:02 -07:00
|
|
|
data = malloc(rowbytes * src->bits->height);
|
|
|
|
free_data = TRUE;
|
2012-06-10 07:21:05 -06:00
|
|
|
if (!data) {
|
2010-07-27 13:02:24 -06:00
|
|
|
FatalError("Failed to allocate memory in %s\n", __func__);
|
|
|
|
}
|
2012-06-10 07:21:05 -06:00
|
|
|
|
|
|
|
if (!src->bits->emptyMask) {
|
2006-11-26 11:13:41 -07:00
|
|
|
ycount = src->bits->height;
|
2012-06-10 07:21:05 -06:00
|
|
|
srow = src->bits->source;
|
|
|
|
mrow = src->bits->mask;
|
2006-11-26 11:13:41 -07:00
|
|
|
drow = data;
|
|
|
|
|
|
|
|
while (ycount-- > 0)
|
|
|
|
{
|
2010-07-27 13:02:24 -06:00
|
|
|
xcount = bits_to_bytes(src->bits->width);
|
2012-06-10 07:21:05 -06:00
|
|
|
sptr = srow;
|
|
|
|
mptr = mrow;
|
2006-11-26 11:13:41 -07:00
|
|
|
dptr = drow;
|
|
|
|
|
|
|
|
while (xcount-- > 0)
|
|
|
|
{
|
|
|
|
uint8_t s, m;
|
|
|
|
int i;
|
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
s = *sptr++;
|
|
|
|
m = *mptr++;
|
|
|
|
for (i = 0; i < 8; i++) {
|
2006-11-26 11:13:41 -07:00
|
|
|
#if BITMAP_BIT_ORDER == MSBFirst
|
|
|
|
if (m & 128)
|
|
|
|
*dptr++ = (s & 128) ? fg_color : bg_color;
|
|
|
|
else
|
|
|
|
*dptr++ = 0;
|
2012-06-10 07:21:05 -06:00
|
|
|
s <<= 1;
|
|
|
|
m <<= 1;
|
2006-11-26 11:13:41 -07:00
|
|
|
#else
|
|
|
|
if (m & 1)
|
|
|
|
*dptr++ = (s & 1) ? fg_color : bg_color;
|
|
|
|
else
|
|
|
|
*dptr++ = 0;
|
2012-06-10 07:21:05 -06:00
|
|
|
s >>= 1;
|
|
|
|
m >>= 1;
|
2006-11-26 11:13:41 -07:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
srow += BitmapBytePad(src->bits->width);
|
|
|
|
mrow += BitmapBytePad(src->bits->width);
|
2012-06-10 07:21:05 -06:00
|
|
|
drow = (uint32_t *)((char *)drow + rowbytes);
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
|
|
|
}
|
2012-06-10 07:21:05 -06:00
|
|
|
else {
|
2006-11-26 11:13:41 -07:00
|
|
|
memset(data, 0, src->bits->height * rowbytes);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
err = xp_set_cursor(width, height, hot_x, hot_y, data, rowbytes);
|
2012-06-10 07:21:05 -06:00
|
|
|
if (free_data)
|
2010-12-05 08:36:02 -07:00
|
|
|
free(data);
|
2006-11-26 11:13:41 -07:00
|
|
|
return err == Success;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2012-06-10 07:21:05 -06:00
|
|
|
===========================================================================
|
2006-11-26 11:13:41 -07:00
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
Pointer sprite functions
|
2006-11-26 11:13:41 -07:00
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
===========================================================================
|
|
|
|
*/
|
2006-11-26 11:13:41 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* QuartzRealizeCursor
|
|
|
|
* Convert the X cursor representation to native format if possible.
|
|
|
|
*/
|
|
|
|
static Bool
|
2009-09-06 13:44:18 -06:00
|
|
|
QuartzRealizeCursor(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)
|
2006-11-26 11:13:41 -07:00
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
/* FIXME: cache ARGB8888 representation? */
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* QuartzUnrealizeCursor
|
|
|
|
* Free the storage space associated with a realized cursor.
|
|
|
|
*/
|
|
|
|
static Bool
|
2009-09-06 13:44:18 -06:00
|
|
|
QuartzUnrealizeCursor(DeviceIntPtr pDev, ScreenPtr pScreen, CursorPtr pCursor)
|
2006-11-26 11:13:41 -07:00
|
|
|
{
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* QuartzSetCursor
|
|
|
|
* Set the cursor sprite and position.
|
|
|
|
*/
|
|
|
|
static void
|
2012-06-10 07:21:05 -06:00
|
|
|
QuartzSetCursor(DeviceIntPtr pDev, ScreenPtr pScreen, CursorPtr pCursor,
|
|
|
|
int x,
|
|
|
|
int y)
|
2006-11-26 11:13:41 -07:00
|
|
|
{
|
|
|
|
QuartzCursorScreenPtr ScreenPriv = CURSOR_PRIV(pScreen);
|
|
|
|
|
2010-12-05 08:36:02 -07:00
|
|
|
if (!XQuartzServerVisible)
|
2006-11-26 11:13:41 -07:00
|
|
|
return;
|
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
if (pCursor == NULL) {
|
|
|
|
if (ScreenPriv->cursorVisible) {
|
2006-11-26 11:13:41 -07:00
|
|
|
xp_hide_cursor();
|
|
|
|
ScreenPriv->cursorVisible = FALSE;
|
|
|
|
}
|
|
|
|
}
|
2012-06-10 07:21:05 -06:00
|
|
|
else {
|
2006-11-26 11:13:41 -07:00
|
|
|
load_cursor(pCursor, pScreen->myNum);
|
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
if (!ScreenPriv->cursorVisible) {
|
2006-11-26 11:13:41 -07:00
|
|
|
xp_show_cursor();
|
|
|
|
ScreenPriv->cursorVisible = TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* QuartzMoveCursor
|
|
|
|
* Move the cursor. This is a noop for us.
|
|
|
|
*/
|
|
|
|
static void
|
2009-09-06 13:44:18 -06:00
|
|
|
QuartzMoveCursor(DeviceIntPtr pDev, ScreenPtr pScreen, int x, int y)
|
2012-06-10 07:21:05 -06:00
|
|
|
{}
|
2006-11-26 11:13:41 -07:00
|
|
|
|
|
|
|
/*
|
2012-06-10 07:21:05 -06:00
|
|
|
===========================================================================
|
2006-11-26 11:13:41 -07:00
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
Pointer screen functions
|
2006-11-26 11:13:41 -07:00
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
===========================================================================
|
|
|
|
*/
|
2006-11-26 11:13:41 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* QuartzCursorOffScreen
|
|
|
|
*/
|
|
|
|
static Bool
|
|
|
|
QuartzCursorOffScreen(ScreenPtr *pScreen, int *x, int *y)
|
|
|
|
{
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* QuartzCrossScreen
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
QuartzCrossScreen(ScreenPtr pScreen, Bool entering)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* QuartzWarpCursor
|
|
|
|
* Change the cursor position without generating an event or motion history.
|
|
|
|
* The input coordinates (x,y) are in pScreen-local X11 coordinates.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static void
|
2009-09-06 13:44:18 -06:00
|
|
|
QuartzWarpCursor(DeviceIntPtr pDev, ScreenPtr pScreen, int x, int y)
|
2006-11-26 11:13:41 -07:00
|
|
|
{
|
2012-06-10 07:21:05 -06:00
|
|
|
if (XQuartzServerVisible) {
|
2006-11-26 11:13:41 -07:00
|
|
|
int sx, sy;
|
|
|
|
|
2010-12-05 08:36:02 -07:00
|
|
|
sx = pScreen->x + darwinMainScreenX;
|
|
|
|
sy = pScreen->y + darwinMainScreenY;
|
2006-11-26 11:13:41 -07:00
|
|
|
|
|
|
|
CGWarpMouseCursorPosition(CGPointMake(sx + x, sy + y));
|
|
|
|
}
|
|
|
|
|
2009-09-06 13:44:18 -06:00
|
|
|
miPointerWarpCursor(pDev, pScreen, x, y);
|
|
|
|
miPointerUpdateSprite(pDev);
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static miPointerScreenFuncRec quartzScreenFuncsRec = {
|
|
|
|
QuartzCursorOffScreen,
|
|
|
|
QuartzCrossScreen,
|
|
|
|
QuartzWarpCursor,
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
2012-06-10 07:21:05 -06:00
|
|
|
===========================================================================
|
2006-11-26 11:13:41 -07:00
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
Other screen functions
|
2006-11-26 11:13:41 -07:00
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
===========================================================================
|
|
|
|
*/
|
2006-11-26 11:13:41 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* QuartzCursorQueryBestSize
|
|
|
|
* Handle queries for best cursor size
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
QuartzCursorQueryBestSize(int class, unsigned short *width,
|
|
|
|
unsigned short *height, ScreenPtr pScreen)
|
|
|
|
{
|
|
|
|
QuartzCursorScreenPtr ScreenPriv = CURSOR_PRIV(pScreen);
|
|
|
|
|
2012-06-10 07:21:05 -06:00
|
|
|
if (class == CursorShape) {
|
2006-11-26 11:13:41 -07:00
|
|
|
/* FIXME: query window server? */
|
|
|
|
*width = 32;
|
|
|
|
*height = 32;
|
|
|
|
}
|
2012-06-10 07:21:05 -06:00
|
|
|
else {
|
2006-11-26 11:13:41 -07:00
|
|
|
(*ScreenPriv->QueryBestSize)(class, width, height, pScreen);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* QuartzInitCursor
|
|
|
|
* Initialize cursor support
|
|
|
|
*/
|
|
|
|
Bool
|
|
|
|
QuartzInitCursor(ScreenPtr pScreen)
|
|
|
|
{
|
|
|
|
QuartzCursorScreenPtr ScreenPriv;
|
|
|
|
miPointerScreenPtr PointPriv;
|
|
|
|
|
|
|
|
/* initialize software cursor handling (always needed as backup) */
|
|
|
|
if (!miDCInitialize(pScreen, &quartzScreenFuncsRec))
|
|
|
|
return FALSE;
|
|
|
|
|
2010-12-05 08:36:02 -07:00
|
|
|
if (!dixRegisterPrivateKey(&darwinCursorScreenKeyRec, PRIVATE_SCREEN, 0))
|
2012-06-10 07:21:05 -06:00
|
|
|
return FALSE;
|
2010-12-05 08:36:02 -07:00
|
|
|
|
|
|
|
ScreenPriv = calloc(1, sizeof(QuartzCursorScreenRec));
|
2006-11-26 11:13:41 -07:00
|
|
|
if (ScreenPriv == NULL)
|
|
|
|
return FALSE;
|
|
|
|
|
2008-11-02 08:26:08 -07:00
|
|
|
/* CURSOR_PRIV(pScreen) = ScreenPriv; */
|
|
|
|
dixSetPrivate(&pScreen->devPrivates, darwinCursorScreenKey, ScreenPriv);
|
2006-11-26 11:13:41 -07:00
|
|
|
|
|
|
|
/* override some screen procedures */
|
|
|
|
ScreenPriv->QueryBestSize = pScreen->QueryBestSize;
|
|
|
|
pScreen->QueryBestSize = QuartzCursorQueryBestSize;
|
|
|
|
|
2008-11-02 08:26:08 -07:00
|
|
|
PointPriv = dixLookupPrivate(&pScreen->devPrivates, miPointerScreenKey);
|
2006-11-26 11:13:41 -07:00
|
|
|
|
|
|
|
ScreenPriv->spriteFuncs = PointPriv->spriteFuncs;
|
|
|
|
|
2009-09-06 13:44:18 -06:00
|
|
|
PointPriv->spriteFuncs->RealizeCursor = QuartzRealizeCursor;
|
|
|
|
PointPriv->spriteFuncs->UnrealizeCursor = QuartzUnrealizeCursor;
|
|
|
|
PointPriv->spriteFuncs->SetCursor = QuartzSetCursor;
|
|
|
|
PointPriv->spriteFuncs->MoveCursor = QuartzMoveCursor;
|
2012-06-10 07:21:05 -06:00
|
|
|
|
2006-11-26 11:13:41 -07:00
|
|
|
ScreenPriv->cursorVisible = TRUE;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* QuartzSuspendXCursor
|
|
|
|
* X server is hiding. Restore the Aqua cursor.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
QuartzSuspendXCursor(ScreenPtr pScreen)
|
2012-06-10 07:21:05 -06:00
|
|
|
{}
|
2006-11-26 11:13:41 -07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* QuartzResumeXCursor
|
|
|
|
* X server is showing. Restore the X cursor.
|
|
|
|
*/
|
|
|
|
void
|
2010-07-27 13:02:24 -06:00
|
|
|
QuartzResumeXCursor(ScreenPtr pScreen)
|
2006-11-26 11:13:41 -07:00
|
|
|
{
|
|
|
|
WindowPtr pWin;
|
|
|
|
CursorPtr pCursor;
|
|
|
|
|
2009-09-06 13:44:18 -06:00
|
|
|
/* TODO: Tablet? */
|
2012-06-10 07:21:05 -06:00
|
|
|
|
2009-09-06 13:44:18 -06:00
|
|
|
pWin = GetSpriteWindow(darwinPointer);
|
2006-11-26 11:13:41 -07:00
|
|
|
if (pWin->drawable.pScreen != pScreen)
|
|
|
|
return;
|
|
|
|
|
2009-09-06 13:44:18 -06:00
|
|
|
pCursor = GetSpriteCursor(darwinPointer);
|
2006-11-26 11:13:41 -07:00
|
|
|
if (pCursor == NULL)
|
|
|
|
return;
|
|
|
|
|
2010-07-27 13:02:24 -06:00
|
|
|
QuartzSetCursor(darwinPointer, pScreen, pCursor, /* x */ 0, /* y */ 0);
|
2006-11-26 11:13:41 -07:00
|
|
|
}
|