handle both keyUp and keyDown events, including dealing with devices

that don't generate the former.  Break key-handling code into separate
function.
This commit is contained in:
ehouse 2007-01-06 17:50:43 +00:00
parent 098a2ec5c4
commit 31b6526917
4 changed files with 104 additions and 69 deletions

View file

@ -277,6 +277,7 @@ evt68k2evtARM( EventType* event, const unsigned char* evt68k )
event->data.frmUpdate.updateCode = read_unaligned16( evt68k + 2 ); event->data.frmUpdate.updateCode = read_unaligned16( evt68k + 2 );
break; break;
case keyDownEvent: case keyDownEvent:
case keyUpEvent:
event->data.keyDown.chr = read_unaligned16( evt68k ); event->data.keyDown.chr = read_unaligned16( evt68k );
event->data.keyDown.keyCode = read_unaligned16( evt68k+2 ); event->data.keyDown.keyCode = read_unaligned16( evt68k+2 );
event->data.keyDown.modifiers = read_unaligned16( evt68k+4 ); event->data.keyDown.modifiers = read_unaligned16( evt68k+4 );
@ -371,6 +372,7 @@ evtArm2evt68K( unsigned char* evt68k, const EventType* event )
write_unaligned16( evt68k + 2, event->data.frmUpdate.updateCode ); write_unaligned16( evt68k + 2, event->data.frmUpdate.updateCode );
break; break;
case keyDownEvent: case keyDownEvent:
case keyUpEvent:
write_unaligned16( evt68k, event->data.keyDown.chr ); write_unaligned16( evt68k, event->data.keyDown.chr );
write_unaligned16( evt68k+2, event->data.keyDown.keyCode ); write_unaligned16( evt68k+2, event->data.keyDown.keyCode );
write_unaligned16( evt68k+4, event->data.keyDown.modifiers ); write_unaligned16( evt68k+4, event->data.keyDown.modifiers );

View file

@ -1,6 +1,6 @@
/* -*-mode: C; fill-column: 77; c-basic-offset: 4; compile-command: "make ARCH=ARM_ONLY MEMDEBUG=TRUE"; -*- */ /* -*-mode: C; fill-column: 77; c-basic-offset: 4; compile-command: "make ARCH=ARM_ONLY MEMDEBUG=TRUE"; -*- */
/* /*
* Copyright 1999 - 2004 by Eric House (xwords@eehouse.org). All rights reserved. * Copyright 1999 - 2007 by Eric House (xwords@eehouse.org). All rights reserved.
* *
* This program is free software; you can redistribute it and/or * This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License * modify it under the terms of the GNU General Public License
@ -920,11 +920,13 @@ initHighResGlobals( PalmAppGlobals* globals )
err = FtrGet( sysFtrCreator, sysFtrNumWinVersion, &vers ); err = FtrGet( sysFtrCreator, sysFtrNumWinVersion, &vers );
globals->hasHiRes = ( err == errNone && vers >= 4 ); globals->hasHiRes = ( err == errNone && vers >= 4 );
XP_LOGF( "hasHiRes = %d", (XP_U16)globals->hasHiRes );
globals->oneDotFiveAvail = ( err == errNone && vers >= 5 ); globals->oneDotFiveAvail = ( err == errNone && vers >= 5 );
#ifdef XWFEATURE_FIVEWAY
err = FtrGet( sysFtrCreator, sysFtrNumUIHardwareFlags, &vers ); err = FtrGet( sysFtrCreator, sysFtrNumUIHardwareFlags, &vers );
XP_LOGF( "hasHiRes = %d", globals->hasHiRes ); globals->hasKeyboard = ( (err == errNone)
&& ((vers & sysFtrNumUIHardwareHasKbd) != 0) );
#ifdef XWFEATURE_FIVEWAY
globals->hasFiveWay = ( (err == errNone) globals->hasFiveWay = ( (err == errNone)
&& ((vers & sysFtrNumUIHardwareHas5Way) != 0) ); && ((vers & sysFtrNumUIHardwareHas5Way) != 0) );
#endif #endif
@ -2118,6 +2120,96 @@ handleFocusEvent( PalmAppGlobals* globals, const EventType* event,
} /* handleFocusEvent */ } /* handleFocusEvent */
#endif #endif
static XP_Bool
handleKeyEvent( PalmAppGlobals* globals, const EventType* event,
XP_Bool* handledP )
{
/* keyDownEvent: be very careful here. keyUpEvent is only sent on
devices with a hard keyboard. Do not assume keyUpEvent or all
non-Treos will be broken!!! */
XP_Bool draw = XP_FALSE;
XP_Key xpkey = XP_KEY_NONE;
XP_Bool handled = XP_FALSE;
XP_Bool altOn = (event->data.keyUp.modifiers & shiftKeyMask) != 0;
XP_Bool treatAsUp = !globals->hasKeyboard || (event->eType == keyUpEvent);
Int16 chr;
XP_ASSERT( OFFSET_OF(EventType, data.keyUp.modifiers)
== OFFSET_OF(EventType, data.keyDown.modifiers) );
XP_ASSERT( OFFSET_OF(EventType, data.keyUp.keyCode)
== OFFSET_OF(EventType, data.keyDown.keyCode) );
switch ( event->data.keyDown.keyCode ) {
case pageUpChr:
draw = treatAsUp && scrollBoard( globals, 0, false );
break;
case pageDownChr:
draw = treatAsUp && scrollBoard( globals, 2, false );
break;
case backspaceChr:
xpkey = XP_CURSOR_KEY_DEL;
break;
#ifdef XWFEATURE_FIVEWAY
case vchrRockerCenter:
xpkey = XP_RETURN_KEY;
break;
case vchrRockerLeft:
xpkey = altOn ? XP_CURSOR_KEY_ALTLEFT : XP_CURSOR_KEY_LEFT;
break;
case vchrRockerRight:
xpkey = altOn ? XP_CURSOR_KEY_ALTRIGHT : XP_CURSOR_KEY_RIGHT;
break;
case vchrRockerUp:
xpkey = altOn ? XP_CURSOR_KEY_ALTUP : XP_CURSOR_KEY_UP;
break;
case vchrRockerDown:
xpkey = altOn ? XP_CURSOR_KEY_ALTDOWN : XP_CURSOR_KEY_DOWN;
break;
case chrSpace:
xpkey = XP_RAISEFOCUS_KEY;
break;
#endif
default:
chr = event->data.keyUp.chr;
/* I'm not interested in being dependent on a particular version
of the OS, (can't manage to link against the intl library
anyway) and so don't want to use the 3.5-only text tests. So
let's give the board two shots at each char, one lower case
and another upper. */
if ( chr < 255 && chr > ' ' ) {
draw = treatAsUp && board_handleKeyUp( globals->game.board,
chr, &handled );
if ( !handled && chr >= 'a' ) {
draw = treatAsUp && board_handleKeyUp( globals->game.board,
chr - ('a' - 'A'),
&handled );
}
}
}
if ( xpkey != XP_KEY_NONE ) {
draw = treatAsUp?
board_handleKeyUp( globals->game.board, xpkey, &handled )
: board_handleKeyDown( globals->game.board, xpkey, &handled );
/* If handled comes back false yet something changed (draw),
we'll be getting another event shortly. Put the draw off
until then so we don't flash the tray focussed then not. This
is a hack, but I can't think of a way to integrate it into
board.c logic without making too many palm-centric assumptions
there. */
if ( draw && !handled ) {
draw = XP_FALSE;
}
} else {
/* remove this and break focus drilldown. Why? */
handled = draw;
}
LOG_RETURNF( "%d", draw );
*handledP = handled;
return draw;
} /* handleKeyEvent */
/***************************************************************************** /*****************************************************************************
* *
****************************************************************************/ ****************************************************************************/
@ -2546,72 +2638,9 @@ mainViewHandleEvent( EventPtr event )
break; break;
#endif #endif
case keyDownEvent: { case keyDownEvent:
XP_Key xpkey = XP_KEY_NONE; case keyUpEvent:
Int16 ch = event->data.keyDown.chr; draw = handleKeyEvent( globals, event, &handled );
XP_Bool altOn = (event->data.keyDown.modifiers & shiftKeyMask) != 0;
switch ( ch ) {
case pageUpChr:
draw = scrollBoard( globals, 0, false );
break;
case pageDownChr:
draw = scrollBoard( globals, 2, false );
break;
case backspaceChr:
xpkey = XP_CURSOR_KEY_DEL;
break;
#ifdef XWFEATURE_FIVEWAY
case vchrRockerCenter:
xpkey = XP_RETURN_KEY;
break;
case vchrRockerLeft:
xpkey = altOn ? XP_CURSOR_KEY_ALTLEFT : XP_CURSOR_KEY_LEFT;
break;
case vchrRockerRight:
xpkey = altOn ? XP_CURSOR_KEY_ALTRIGHT : XP_CURSOR_KEY_RIGHT;
break;
case vchrRockerUp:
xpkey = altOn ? XP_CURSOR_KEY_ALTUP : XP_CURSOR_KEY_UP;
break;
case vchrRockerDown:
xpkey = altOn ? XP_CURSOR_KEY_ALTDOWN : XP_CURSOR_KEY_DOWN;
break;
case chrSpace:
xpkey = XP_RAISEFOCUS_KEY;
break;
#endif
default:
/* I'm not interested in being dependent on a particular version
of the OS, (can't manage to link against the intl library
anyway) and so don't want to use the 3.5-only text tests. So
let's give the board two shots at each char, one lower case
and another upper. */
if ( ch < 255 && ch > ' ' ) {
draw = board_handleKey( globals->game.board, ch, &handled );
if ( !handled && ch >= 'a' ) {
draw = board_handleKey( globals->game.board,
ch - ('a' - 'A'), &handled );
}
}
}
if ( xpkey != XP_KEY_NONE ) {
draw = board_handleKey( globals->game.board, xpkey, &handled );
/* If handled comes back false yet something changed (draw),
we'll be getting another event shortly. Put the draw off
until then so we don't flash the tray focussed then not. This
is a hack, but I can't think of a way to integrate it into
board.c logic without making too many palm-centric assumptions
there. */
if ( draw && !handled ) {
draw = XP_FALSE;
}
} else {
/* remove this and break focus drilldown. Why? */
handled = draw;
}
}
break; break;
case sclRepeatEvent: case sclRepeatEvent:

View file

@ -296,6 +296,7 @@ struct PalmAppGlobals {
XP_Bool oneDotFiveAvail; XP_Bool oneDotFiveAvail;
XP_Bool useHiRes; XP_Bool useHiRes;
XP_Bool hasFiveWay; XP_Bool hasFiveWay;
XP_Bool hasKeyboard;
#ifdef XWFEATURE_SEARCHLIMIT #ifdef XWFEATURE_SEARCHLIMIT
XP_Bool askTrayLimits; XP_Bool askTrayLimits;

View file

@ -645,6 +645,9 @@ logEvent( eventsEnum eType )
CASE_STR(firstUserEvent); CASE_STR(firstUserEvent);
CASE_STR(lastUserEvent); CASE_STR(lastUserEvent);
default:
name = "<unknown>";
break;
} }
#undef CASE_STR #undef CASE_STR
if ( !!name ) { if ( !!name ) {