OpenCores
URL https://opencores.org/ocsvn/or1k/or1k/trunk

Subversion Repositories or1k

[/] [or1k/] [trunk/] [mw/] [src/] [mwin/] [winevent.c] - Rev 1780

Go to most recent revision | Compare with Previous | Blame | View Log

/*
 * Copyright (c) 1999, 2000 Greg Haerr <greg@censoft.com>
 * Copyright (c) 1991 David I. Bell
 * Permission is granted to use, distribute, or modify this source,
 * provided that this copyright notice remains intact.
 *
 * Graphics server event routines for windows.
 */
#include "windows.h"
#include "wintern.h"
#include "device.h"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
 
#if !(DOS_TURBOC | DOS_QUICKC | _MINIX | VXWORKS)
static int
abs(int n)
{
	return n >= 0? n: -n;
}
#endif
 
/*
 * Update mouse status and issue events on it if necessary.
 * This function doesn't block, but is normally only called when
 * there is known to be some data waiting to be read from the mouse.
 */
BOOL
MwCheckMouseEvent(void)
{
	MWCOORD		rootx;		/* latest mouse x position */
	MWCOORD		rooty;		/* latest mouse y position */
	int		newbuttons;	/* latest buttons */
	int		mousestatus;	/* latest mouse status */
 
	/* Read the latest mouse status: */
	mousestatus = GdReadMouse(&rootx, &rooty, &newbuttons);
	if(mousestatus < 0) {
		/*MwError(GR_ERROR_MOUSE_ERROR, 0);*/
		return FALSE;
	} else if(mousestatus) {	/* Deliver events as appropriate: */	
		MwHandleMouseStatus(rootx, rooty, newbuttons);
		return TRUE;
	}
	return FALSE;
}
 
/*
 * Update keyboard status and issue events on it if necessary.
 * This function doesn't block, but is normally only called when
 * there is known to be some data waiting to be read from the keyboard.
 */
BOOL
MwCheckKeyboardEvent(void)
{
	MWKEY	 	mwkey;		/* latest character */
	MWKEYMOD 	modifiers;	/* latest modifiers */
	MWSCANCODE	scancode;
	int	 	keystatus;	/* latest keyboard status */
 
	/* Read the latest keyboard status: */
	keystatus = GdReadKeyboard(&mwkey, &modifiers, &scancode);
	if(keystatus < 0) {
		if(keystatus == -2)	/* special case for ESC pressed*/
			MwTerminate();
		/*MwError(GR_ERROR_KEYBOARD_ERROR, 0);*/
		return FALSE;
	} else if(keystatus) {		/* Deliver events as appropriate: */	
		switch (mwkey) {
		case MWKEY_QUIT:
			MwTerminate();
			/* no return*/
		case MWKEY_REDRAW:
			MwRedrawScreen();
			break;
		case MWKEY_PRINT:
			if (keystatus == 1)
				GdCaptureScreen("screen.bmp");
			break;
		}
		MwDeliverKeyboardEvent(mwkey, modifiers, scancode,
			keystatus==1? TRUE: FALSE);
		return TRUE;
	}
	return FALSE;
}
 
/*
 * Handle all mouse events.  These are mouse enter, mouse exit, mouse
 * motion, mouse position, button down, and button up.  This also moves
 * the cursor to the new mouse position and changes it shape if needed.
 */
void
MwHandleMouseStatus(MWCOORD newx, MWCOORD newy, int newbuttons)
{
	int		changebuttons;	/* buttons that have changed */
	MWKEYMOD	modifiers;	/* latest modifiers */
	static int curbuttons;
 
	GdGetModifierInfo(NULL, &modifiers); /* Read kbd modifiers */
 
	/*
	 * First, if the mouse has moved, then position the cursor to the
	 * new location, which will send mouse enter, mouse exit, focus in,
	 * and focus out events if needed.  Check here for mouse motion and
	 * mouse position events.
	 */
	if (newx != cursorx || newy != cursory) {
		MwMoveCursor(newx, newy);
		MwDeliverMouseEvent(newbuttons, 0, modifiers);
	}
 
	/*
	 * Next, generate a button up event if any buttons have been released.
	 */
	changebuttons = (curbuttons & ~newbuttons);
	if (changebuttons)
		MwDeliverMouseEvent(newbuttons, changebuttons, modifiers);
 
	/*
	 * Finally, generate a button down event if any buttons have been
	 * pressed.
	 */
	changebuttons = (~curbuttons & newbuttons);
	if (changebuttons)
		MwDeliverMouseEvent(newbuttons, changebuttons, modifiers);
 
	curbuttons = newbuttons;
}
 
/*
 * Translate and deliver hardware mouse message to proper window.
 */
void
MwTranslateMouseMessage(HWND hwnd,UINT msg,int hittest)
{
	POINT		pt;
	DWORD		tick;
	static UINT	lastmsg = 0;
	static HWND	lasthwnd;
	static DWORD	lasttick;
	static int	lastx, lasty;
 
	/* determine double click eligibility*/
	if(msg == WM_LBUTTONDOWN || msg == WM_RBUTTONDOWN) {
		tick = GetTickCount();
		if((hwnd->pClass->style & CS_DBLCLKS) &&
		    msg == lastmsg && hwnd == lasthwnd &&
		    tick - lasttick < DBLCLICKSPEED &&
		    abs(cursorx-lastx) < mwSYSMETRICS_CXDOUBLECLK &&
		    abs(cursory-lasty) < mwSYSMETRICS_CYDOUBLECLK)
			msg += (WM_LBUTTONDBLCLK - WM_LBUTTONDOWN);
		lastmsg = msg;
		lasthwnd = hwnd;
		lasttick = tick;
		lastx = cursorx;
		lasty = cursory;
	}
 
	/*
	 * We always send nc mouse message
	 * unlike Windows, for HTCLIENT default processing
	 */
	PostMessage(hwnd, msg + (WM_NCMOUSEMOVE-WM_MOUSEMOVE), hittest,
		MAKELONG(cursorx, cursory));
 
	/* then possibly send user mouse message*/
	if(hittest == HTCLIENT) {
		pt.x = cursorx;
		pt.y = cursory;
		ScreenToClient(hwnd, &pt);
		PostMessage(hwnd, msg, 0, MAKELONG(pt.x, pt.y));
	}
}
 
/*
 * Deliver a mouse button or motion event.
 */
int mwCurrentButtons;
 
void 
MwDeliverMouseEvent(int buttons, int changebuttons, MWKEYMOD modifiers)
{
	HWND	hwnd;
	int	hittest;
	UINT	msg;
 
	mwCurrentButtons = buttons;
 
	hwnd = GetCapture();
	if(!hwnd)
		hwnd = mousewp;
	hittest = SendMessage(hwnd, WM_NCHITTEST, 0, MAKELONG(cursorx,cursory));
 
	if(!changebuttons)
		MwTranslateMouseMessage(hwnd, WM_MOUSEMOVE, hittest);
 
	if(changebuttons & MWBUTTON_L) {
		msg = (buttons&MWBUTTON_L)? WM_LBUTTONDOWN: WM_LBUTTONUP;
		MwTranslateMouseMessage(hwnd, msg, hittest);
	}
 
	if(changebuttons & MWBUTTON_M) {
		msg = (buttons&MWBUTTON_M)? WM_MBUTTONDOWN: WM_MBUTTONUP;
		MwTranslateMouseMessage(hwnd, msg, hittest);
	}
 
	if(changebuttons & MWBUTTON_R) {
		msg = (buttons&MWBUTTON_R)? WM_RBUTTONDOWN: WM_RBUTTONUP;
		MwTranslateMouseMessage(hwnd, msg, hittest);
	}
}
 
/*
 * Deliver a keyboard event.
 */
void
MwDeliverKeyboardEvent(MWKEY keyvalue, MWKEYMOD modifiers, MWSCANCODE scancode,
	BOOL pressed)
{
	WPARAM VK_Code = keyvalue;	/* default no translation*/
 
	/* Keysyms from 1-255 are mapped to ASCII*/
	if (keyvalue < 1 || keyvalue > 255)
	  switch(keyvalue) {
 
	/* arrows + home/end pad*/
	case MWKEY_LEFT:
		VK_Code = VK_LEFT;
		break;
	case MWKEY_RIGHT:
		VK_Code = VK_RIGHT;
		break;
	case MWKEY_UP:
		VK_Code = VK_UP;
		break;
	case MWKEY_DOWN:
		VK_Code =  VK_DOWN;
		break;
	case MWKEY_INSERT:
		VK_Code = VK_INSERT;
		break;
	case MWKEY_DELETE:
		VK_Code = VK_DELETE;
		break;
	case MWKEY_HOME:
		VK_Code = VK_HOME;
		break;
	case MWKEY_END:
		VK_Code = VK_END;
		break;
	case MWKEY_PAGEUP:
		VK_Code = VK_PRIOR;
		break;
	case MWKEY_PAGEDOWN:
		VK_Code = VK_NEXT;
		break;
 
	/* Numeric keypad*/
	case MWKEY_KP0:
		VK_Code = VK_NUMPAD0;
		break;
	case MWKEY_KP1:
		VK_Code = VK_NUMPAD1;
		break;
	case MWKEY_KP2:
		VK_Code = VK_NUMPAD2;
		break;
	case MWKEY_KP3:
		VK_Code = VK_NUMPAD3;
		break;
	case MWKEY_KP4:
		VK_Code = VK_NUMPAD4;
		break;
	case MWKEY_KP5:
		VK_Code = VK_NUMPAD5;
		break;
	case MWKEY_KP6:
		VK_Code = VK_NUMPAD6;
		break;
	case MWKEY_KP7:
		VK_Code = VK_NUMPAD7;
		break;
	case MWKEY_KP8:
		VK_Code = VK_NUMPAD8;
		break;
	case MWKEY_KP9:
		VK_Code = VK_NUMPAD9;
		break;
	case MWKEY_KP_PERIOD:
		VK_Code = VK_DECIMAL;
		break;
	case MWKEY_KP_DIVIDE:
		VK_Code = VK_DIVIDE;
		break;
	case MWKEY_KP_MULTIPLY:
		VK_Code = VK_MULTIPLY;
		break;
	case MWKEY_KP_MINUS:
		VK_Code = VK_SUBTRACT;
		break;
	case MWKEY_KP_PLUS:
		VK_Code = VK_ADD;
		break;
	case MWKEY_KP_ENTER:
		VK_Code = VK_RETURN;
		break;
 
	/* Function keys */
	case MWKEY_F1:
		VK_Code = VK_F1;
		break;
	case MWKEY_F2:
		VK_Code = VK_F2;
		break;
	case MWKEY_F3:
		VK_Code = VK_F3;
		break;
	case MWKEY_F4:
		VK_Code = VK_F4;
		break;
	case MWKEY_F5:
		VK_Code = VK_F5;
		break;
	case MWKEY_F6:
		VK_Code = VK_F6;
		break;
	case MWKEY_F7:
		VK_Code = VK_F7;
		break;
	case MWKEY_F8:
		VK_Code = VK_F8;
		break;
	case MWKEY_F9:
		VK_Code = VK_F9;
		break;
	case MWKEY_F10:
		VK_Code = VK_F10;
		break;
	case MWKEY_F11:
		VK_Code = VK_F11;
		break;
	case MWKEY_F12:
		VK_Code = VK_F12;
		break;
 
	/* Key state modifier keys*/
	case MWKEY_NUMLOCK:
		VK_Code = VK_NUMLOCK;
		break;
	case MWKEY_CAPSLOCK:
		VK_Code = VK_CAPITAL;
		break;
	case MWKEY_SCROLLOCK:
		VK_Code = VK_CAPITAL;
		break;
	case MWKEY_LSHIFT:
		VK_Code = VK_LSHIFT;
		break;
	case MWKEY_RSHIFT:
		VK_Code = VK_RSHIFT;
		break;
	case MWKEY_LCTRL:
		VK_Code = VK_LCONTROL;
		break;
	case MWKEY_RCTRL:
		VK_Code = VK_RCONTROL;
		break;
	case MWKEY_LALT:
		VK_Code = VK_MENU;
		break;
	case MWKEY_RALT:
		VK_Code = VK_MENU;
		break;
 
	/* Misc function keys*/
	case MWKEY_PRINT:
		VK_Code = VK_PRINT;
		break;
	case MWKEY_PAUSE:
		VK_Code = VK_PAUSE;
		break;
	case MWKEY_MENU:
		VK_Code = VK_LMENU;	/* virtual key*/
		break;
 
	/* questionable mappings or no mappings...*/
	case MWKEY_KP_EQUALS:
		VK_Code = '=';	/* FIXME*/
		break;
 
	/* map all non-handled MWKEY values to VK_NONAME*/
#if 0
	case MWKEY_UNKNOWN:
	case MWKEY_SYSREQ:
	case MWKEY_BREAK
	case MWKEY_QUIT:
	case MWKEY_REDRAW:
	case MWKEY_LMETA:
	case MWKEY_RMETA:
	case MWKEY_ALTGR:
	/* Handheld function keys*/
	case MWKEY_RECORD:
	case MWKEY_PLAY:
	case MWKEY_CONTRAST:
	case MWKEY_BRIGHTNESS:
	case MWKEY_SELECTUP:
	case MWKEY_SELECTDOWN:
	case MWKEY_ACCEPT:
	case MWKEY_CANCEL:
	case MWKEY_APP1:
	case MWKEY_APP2:
	case MWKEY_LAST:
#endif
	default:
		VK_Code = VK_NONAME;
		break;
	}
 
	if (pressed)
		SendMessage(focuswp, WM_CHAR, VK_Code, 0L);
}
 
/*
 * Deliver a window expose event.
 * Most of the work is in calculating the update region
 * for better redraw look and feel, and then queuing a
 * WM_PAINT message to the window.
 */
void
MwDeliverExposureEvent(HWND wp, MWCOORD x, MWCOORD y, MWCOORD width,
	MWCOORD height)
{
	if (wp->unmapcount)
		return;
 
	MwUnionUpdateRegion(wp, x, y, width, height, TRUE);
	PostMessage(wp, WM_PAINT, 0, 0L);
}
 
/*
 * Combine the passed rectangle with the update region for the given window.
 * Coordinates are passed relative to window.
 * If bUnion is TRUE, union the rectangle, otherwise subtract it.
 */
void
MwUnionUpdateRegion(HWND wp, MWCOORD x, MWCOORD y, MWCOORD width,
	MWCOORD height, BOOL bUnion)
{
#if UPDATEREGIONS
	MWRECT rc;
 
	if (wp->unmapcount)
		return;
 
	/* convert window relative coords to screen coords*/
	rc.left = x + wp->winrect.left;
	rc.top = y + wp->winrect.top;
	rc.right = rc.left + width;
	rc.bottom = rc.top + height;
 
	if(bUnion)
		GdUnionRectWithRegion(&rc, wp->update);
	else
		GdSubtractRectFromRegion(&rc, wp->update);
#endif
}
 
/*
 * Move the cursor to the specified absolute screen coordinates.
 * The coordinates are that of the defined hot spot of the cursor.
 * The cursor's appearance is changed to that defined for the window
 * in which the cursor is moved to.  In addition, the window the
 * cursor is in is recalculated.
 */
void MwMoveCursor(MWCOORD x, MWCOORD y)
{
	/*
	 * Move the cursor only if necessary, offsetting it to
	 * place the hot spot at the specified coordinates.
	 */
	if (x != cursorx || y != cursory) {
		if(curcursor)
			GdMoveCursor(x - curcursor->cursor.hotx,
				y - curcursor->cursor.hoty);
		cursorx = x;
		cursory = y;
	}
 
	/*
	 * Now check to see which window the mouse is in and whether or
	 * not the cursor shape should be changed.
	 */
	MwCheckMouseWindow();
	MwCheckCursor();
}
 
/*
 * Check to see if the cursor shape is the correct shape for its current
 * location.  If not, its shape is changed.
 */
void MwCheckCursor(void)
{
	HWND 		wp;		/* window cursor is in */
	HCURSOR		cp;		/* cursor definition */
 
	/*
	 * Get the cursor at its current position, and if it is not the
	 * currently defined one, then set the new cursor.  However,
	 * if the window is currently captured, then leave it alone.
	 */
	wp = capturewp;
	if (wp == NULL)
		wp = mousewp;
 
	cp = wp->cursor;
	if (cp == curcursor)
		return;
 
	/*
	 * It needs redefining, so do it.
	 */
	curcursor = cp;
	GdMoveCursor(cursorx - cp->cursor.hotx, cursory - cp->cursor.hoty);
	GdSetCursor(&cp->cursor);
}
 
/*
 * Find the window which is currently visible for the specified coordinates.
 * This just walks down the window tree looking for the deepest mapped
 * window which contains the specified point.  If the coordinates are
 * off the screen, the root window is returned.
 */
HWND
MwFindVisibleWindow(MWCOORD x, MWCOORD y)
{
	HWND	wp;		/* current window */
	HWND	retwp;		/* returned window */
 
	wp = rootwp;
	retwp = wp;
	while (wp) {
		if (!wp->unmapcount &&
		    wp->winrect.left <= x && wp->winrect.top <= y &&
		    wp->winrect.right > x && wp->winrect.bottom > y) {
			retwp = wp;
			wp = wp->children;
			continue;
		}
		wp = wp->siblings;
	}
	return retwp;
}
 
/*
 * Check to see if the window the mouse is currently in has changed.
 */
void MwCheckMouseWindow(void)
{
	HWND	wp;
 
	/* Don't change if window drag or capture in progress*/
	wp = dragwp;
	if(!wp)
		wp = capturewp;
	if(!wp)
		wp = MwFindVisibleWindow(cursorx, cursory);
	mousewp = wp;
}
 
/*
 *  Copy dstsiz bytes, including nul, from src to dst.
 *  Return # bytes, excluding nul, copied.
 */
int
strzcpy(char *dst,const char *src,int dstsiz)
{
	int	cc = dstsiz;
 
	/* return 0 on NULL src*/
	if(!src)
		cc = dstsiz = 1;
 
	while(--dstsiz > 0) {
		if((*dst++ = *src++) == '\0')
			return cc - dstsiz - 1;
	}
	*dst = 0;
	return cc - dstsiz - 1;
}
 

Go to most recent revision | Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2025 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.