ctkeyb.c
来自「开放源码的编译器open watcom 1.6.0版的源代码」· C语言 代码 · 共 470 行
C
470 行
/****************************************************************************
*
* Open Watcom Project
*
* Portions Copyright (c) 1983-2002 Sybase, Inc. All Rights Reserved.
*
* ========================================================================
*
* This file contains Original Code and/or Modifications of Original
* Code as defined in and that are subject to the Sybase Open Watcom
* Public License version 1.0 (the 'License'). You may not use this file
* except in compliance with the License. BY USING THIS FILE YOU AGREE TO
* ALL TERMS AND CONDITIONS OF THE LICENSE. A copy of the License is
* provided with the Original Code and Modifications, and is also
* available at www.sybase.com/developer/opensource.
*
* The Original Code and all software distributed under the License are
* distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
* EXPRESS OR IMPLIED, AND SYBASE AND ALL CONTRIBUTORS HEREBY DISCLAIM
* ALL SUCH WARRANTIES, INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR
* NON-INFRINGEMENT. Please see the License for the specific language
* governing rights and limitations under the License.
*
* ========================================================================
*
* Description: Terminal keyboard input processing.
*
****************************************************************************/
#include <term.h>
#include "ctkeyb.h"
unsigned short ShftState;
enum {
EV_STICKY_FUNC = 0xff0,
EV_STICKY_SHIFT,
EV_STICKY_CTRL,
EV_STICKY_ALT,
S_FUNC = S_CAPS
};
#define NUM_ELTS( a ) (sizeof( a ) / sizeof( a[0] ))
struct an_in_term_info {
EVENT ev;
char *str;
};
#define NUM_IN_TERM_INFO_MAPPINGS 74
struct an_in_term_info InTerminfo[NUM_IN_TERM_INFO_MAPPINGS];
bool init_interminfo( void )
/**************************/
{
struct an_in_term_info *entry;
entry = InTerminfo;
#define evmap( code, terminfo_code ) \
entry->ev = EV_##code; \
entry->str = terminfo_code; \
++entry;
evmap( RUB_OUT, key_backspace )
evmap( RUB_OUT, key_clear )
evmap( DELETE, key_dc )
evmap( CURSOR_DOWN, key_down )
evmap( INSERT, key_ic )
evmap( FUNC(1), key_f1 )
evmap( FUNC(2), key_f2 )
evmap( FUNC(3), key_f3 )
evmap( FUNC(4), key_f4 )
evmap( FUNC(5), key_f5 )
evmap( FUNC(6), key_f6 )
evmap( FUNC(7), key_f7 )
evmap( FUNC(8), key_f8 )
evmap( FUNC(9), key_f9 )
evmap( FUNC(10), key_f10 )
evmap( FUNC(11), key_f11 )
evmap( FUNC(12), key_f12 )
evmap( SHIFT_FUNC(1), key_f13 )
evmap( SHIFT_FUNC(2), key_f14 )
evmap( SHIFT_FUNC(3), key_f15 )
evmap( SHIFT_FUNC(4), key_f16 )
evmap( SHIFT_FUNC(5), key_f17 )
evmap( SHIFT_FUNC(6), key_f18 )
evmap( SHIFT_FUNC(7), key_f19 )
evmap( SHIFT_FUNC(8), key_f20 )
evmap( SHIFT_FUNC(9), key_f21 )
evmap( SHIFT_FUNC(10), key_f22 )
evmap( SHIFT_FUNC(11), key_f23 )
evmap( SHIFT_FUNC(12), key_f24 )
evmap( CTRL_FUNC(1), key_f25 )
evmap( CTRL_FUNC(2), key_f26 )
evmap( CTRL_FUNC(3), key_f27 )
evmap( CTRL_FUNC(4), key_f28 )
evmap( CTRL_FUNC(5), key_f29 )
evmap( CTRL_FUNC(6), key_f30 )
evmap( CTRL_FUNC(7), key_f31 )
evmap( CTRL_FUNC(8), key_f32 )
evmap( CTRL_FUNC(9), key_f33 )
evmap( CTRL_FUNC(10), key_f34 )
evmap( CTRL_FUNC(11), key_f35 )
evmap( CTRL_FUNC(12), key_f36 )
evmap( ALT_FUNC(1), key_f37 )
evmap( ALT_FUNC(2), key_f38 )
evmap( ALT_FUNC(3), key_f39 )
evmap( ALT_FUNC(4), key_f40 )
evmap( ALT_FUNC(5), key_f41 )
evmap( ALT_FUNC(6), key_f42 )
evmap( ALT_FUNC(7), key_f43 )
evmap( ALT_FUNC(8), key_f44 )
evmap( ALT_FUNC(9), key_f45 )
evmap( ALT_FUNC(10), key_f46 )
evmap( ALT_FUNC(11), key_f47 )
evmap( ALT_FUNC(12), key_f48 )
evmap( HOME, key_home )
evmap( CURSOR_LEFT, key_left )
evmap( PAGE_DOWN, key_npage )
evmap( PAGE_UP, key_ppage )
evmap( CURSOR_RIGHT, key_right )
evmap( SCROLL_LINE_DOWN, key_sf )
evmap( SCROLL_LINE_UP, key_sr )
evmap( CURSOR_UP, key_up )
evmap( HOME, key_beg )
evmap( ESCAPE, key_cancel )
evmap( END, key_end )
evmap( ENTER, key_enter )
evmap( TAB_FORWARD, key_next )
evmap( TAB_BACKWARD, key_previous )
evmap( SHIFT_HOME, key_sbeg )
evmap( SHIFT_END, key_send )
evmap( SHIFT_CURSOR_LEFT, key_sleft )
evmap( TAB_BACKWARD, key_snext )
evmap( TAB_FORWARD, key_sprevious )
evmap( SHIFT_CURSOR_RIGHT, key_sright )
evmap( TAB_BACKWARD, key_btab )
// Check to see that the correct number of events were added
return( NUM_ELTS( InTerminfo ) == NUM_IN_TERM_INFO_MAPPINGS );
}
static const struct {
EVENT ev;
char ch;
} InStandard[] = {
/*
See ck_keyboard_event function for special handling of these codes.
{ EV_REDRAW_SCREEN, '\x0c' },
{ EV_RUB_OUT, '\x08' },
{ EV_TAB_FORWARD, '\x09' },
*/
{ EV_ENTER, '\r' },
{ EV_ENTER, '\n' },
{ EV_RUB_OUT, '\x7f' },
{ EV_ESCAPE, '\x1b' },
};
typedef struct {
EVENT normal;
EVENT shift;
EVENT ctrl;
EVENT alt;
} event_shift_map;
#define SPECIAL_MAP( name, c ) { c, c, c, EV_ALT_##name }
#define LETTER_MAP( let, c ) { c+' ', c, c-'@', EV_ALT_##let }
#define FUNC_MAP( n ) { EV_FUNC(n),EV_SHIFT_FUNC(n),\
EV_CTRL_FUNC(n),EV_ALT_FUNC(n) }
#define MOTION_MAP( k ) { EV_##k,EV_SHIFT_##k,EV_CTRL_##k,EV_ALT_##k}
#define EV_SHIFT_INSERT EV_INSERT
#define EV_SHIFT_DELETE EV_DELETE
/*
NOTE: this table has to be in increasing value of the first column.
*/
static const event_shift_map ShiftMap[] = {
SPECIAL_MAP( SPACE, ' ' ),
SPECIAL_MAP( MINUS, '-' ),
SPECIAL_MAP( 0, '0' ),
SPECIAL_MAP( 1, '1' ),
SPECIAL_MAP( 2, '2' ),
SPECIAL_MAP( 3, '3' ),
SPECIAL_MAP( 4, '4' ),
SPECIAL_MAP( 5, '5' ),
SPECIAL_MAP( 6, '6' ),
SPECIAL_MAP( 7, '7' ),
SPECIAL_MAP( 8, '8' ),
SPECIAL_MAP( 9, '9' ),
SPECIAL_MAP( EQUAL, '=' ),
LETTER_MAP( A, 'A' ),
LETTER_MAP( B, 'B' ),
LETTER_MAP( C, 'C' ),
LETTER_MAP( D, 'D' ),
LETTER_MAP( E, 'E' ),
LETTER_MAP( F, 'F' ),
LETTER_MAP( G, 'G' ),
LETTER_MAP( H, 'H' ),
LETTER_MAP( I, 'I' ),
LETTER_MAP( J, 'J' ),
LETTER_MAP( K, 'K' ),
LETTER_MAP( L, 'L' ),
LETTER_MAP( M, 'M' ),
LETTER_MAP( N, 'N' ),
LETTER_MAP( O, 'O' ),
LETTER_MAP( P, 'P' ),
LETTER_MAP( Q, 'Q' ),
LETTER_MAP( R, 'R' ),
LETTER_MAP( S, 'S' ),
LETTER_MAP( T, 'T' ),
LETTER_MAP( U, 'U' ),
LETTER_MAP( V, 'V' ),
LETTER_MAP( W, 'W' ),
LETTER_MAP( X, 'X' ),
LETTER_MAP( Y, 'Y' ),
LETTER_MAP( Z, 'Z' ),
FUNC_MAP( 1 ),
FUNC_MAP( 2 ),
FUNC_MAP( 3 ),
FUNC_MAP( 4 ),
FUNC_MAP( 5 ),
FUNC_MAP( 6 ),
FUNC_MAP( 7 ),
FUNC_MAP( 8 ),
FUNC_MAP( 9 ),
FUNC_MAP( 10 ),
MOTION_MAP( HOME ),
MOTION_MAP( CURSOR_UP ),
MOTION_MAP( PAGE_UP ),
MOTION_MAP( CURSOR_LEFT ),
MOTION_MAP( CURSOR_RIGHT ),
MOTION_MAP( END ),
MOTION_MAP( CURSOR_DOWN ),
MOTION_MAP( PAGE_DOWN ),
MOTION_MAP( INSERT ),
MOTION_MAP( DELETE ),
FUNC_MAP( 11 ),
FUNC_MAP( 12 ),
};
extern void tm_saveevent( void );
void intern clear_shift( void )
{
ShftState = 0;
}
void intern ck_arm( void )
/************************/
{
}
#define PUSHBACK_SIZE 32
static char UnreadBuffer[ PUSHBACK_SIZE ];
static int UnreadPos = sizeof( UnreadBuffer );
int nextc(int n) // delay in 0.1 seconds -- not to exceed 9
/**************/
{
unsigned char ch;
int test; // Used to test the return of read()
int fds;
if( UnreadPos < sizeof( UnreadBuffer ) )
return( UnreadBuffer[ UnreadPos++ ] );
n *= 100000;
fds = kb_wait( n/1000000, n%1000000 );
if( fds == 1 ) { // Console has input
test = read( UIConHandle, &ch, 1 );
if( test == -1 ) {
return -1;
}
} else if( fds == 2 ) {
return 256; /* mouse event */
} else {
return -1;
}
return ch;
}
void nextc_unget( char *str, int n )
/**********************************/
{
UnreadPos-=n;
//assert(UnreadPos>=0);
memcpy( &(UnreadBuffer[UnreadPos]), str, n );
}
int find_entry( const void *pkey, const void *pbase )
{
const EVENT *evp = pkey;
const event_shift_map *entry = pbase;
return( *evp - entry->normal );
}
EVENT ck_keyboardevent( void )
/****************************/
{
EVENT ev;
EVENT search_ev;
event_shift_map *entry;
static unsigned short sticky;
static unsigned short real_shift;
ev = TrieRead();
ck_shift_state();
if( ShftState != ( real_shift | sticky ) ) {
/* did it change? */
real_shift = ShftState;
}
switch( ev ) {
case EV_STICKY_FUNC:
sticky ^= S_FUNC;
break;
case EV_STICKY_SHIFT:
sticky ^= S_SHIFT;
break;
case EV_STICKY_CTRL:
sticky ^= S_CTRL;
break;
case EV_STICKY_ALT:
sticky ^= S_ALT;
break;
case EV_SHIFT_PRESS:
real_shift |= S_SHIFT;
break;
case EV_CTRL_PRESS:
real_shift |= S_CTRL;
break;
case EV_ALT_PRESS:
real_shift |= S_ALT;
break;
case EV_SHIFT_RELEASE:
if( !(real_shift & S_SHIFT) ) ev = EV_NO_EVENT;
real_shift &= ~S_SHIFT;
break;
case EV_CTRL_RELEASE:
if( !(real_shift & S_CTRL) ) ev = EV_NO_EVENT;
real_shift &= ~S_CTRL;
break;
case EV_ALT_RELEASE:
if( !(real_shift & S_ALT) ) ev = EV_NO_EVENT;
real_shift &= ~S_ALT;
break;
case EV_NO_EVENT:
break;
default:
if( sticky & S_FUNC ) {
if( ev == '0' ) {
ev = EV_FUNC( 10 );
} else if( ev == 'a' || ev == 'A' ) {
ev = EV_FUNC_11;
} else if( ev == 'b' || ev == 'B' ) {
ev = EV_FUNC_12;
} else if( isdigit( ev ) ) {
ev = EV_FUNC( ev - '0' );
}
sticky &= ~S_FUNC;
}
if( !(real_shift & S_CTRL) ) {
/*
If the ctrl key isn't down (won't ever be on a terminal)
then we want to see certain CTRL-? combinations come back
as some standard UI events.
*/
switch( ev ) {
case '\x08':
/* ctrl-backspace often does the opposite of backspace */
if( strcmp(key_backspace, "\x08" ) == 0 )
ev = EV_RUB_OUT;
else
ev = EV_CTRL_BACKSPACE;
break;
case '\x09':
ev = EV_TAB_FORWARD;
break;
case '\x0c':
ev = EV_REDRAW_SCREEN;
break;
}
}
ShftState = sticky | real_shift;
sticky = 0;
#define S_MASK (S_SHIFT|S_CTRL|S_ALT)
if( ShftState & S_MASK ) {
search_ev = tolower( ev );
entry = bsearch( &search_ev, ShiftMap, NUM_ELTS( ShiftMap ),
sizeof( ShiftMap[0] ), find_entry );
if( entry != NULL ) {
if( ShftState & S_SHIFT ) {
ev = entry->shift;
} else if( ShftState & S_CTRL ) {
ev = entry->ctrl;
} else { /* must be ALT */
ev = entry->alt;
}
}
}
if( ev ) {
UIDebugPrintf1( "UI: Something read: %4.4X", ev );
}
return( ev );
}
ShftState = real_shift;
if( ev ) {
UIDebugPrintf1( "UI: Something read: %4.4X", ev );
}
return( ev );
}
EVENT tk_keyboardevent( void )
{
EVENT ev;
ev = ck_keyboardevent();
if( ev != EV_MOUSE_PRESS ) return( ev );
UIDebugPrintf0( "UI: Mouse event handling" );
tm_saveevent();
return( EV_NO_EVENT ); /* make UI check for mouse events */
}
int init_trie( void )
/*******************/
{
char *str;
char buff[2];
int i;
if( !TrieInit() ) return( FALSE );
if( !init_interminfo() ) return( FALSE );
str = getenv( "TERM" );
/* attempt to adjust backspace with the terminfo definition */
if( str && strncmp(str, "xterm", 5) == 0 ) {
if( strcmp(key_backspace, "\x08" ) == 0 )
write( UIConHandle, "\x1b[?67h", 6 );
else if( strcmp(key_backspace, "\x7f" ) == 0 )
write( UIConHandle, "\x1b[?67l", 6 );
}
buff[1] = '\0';
for( i = 0; i < NUM_ELTS( InStandard ); ++i ) {
buff[0] = InStandard[i].ch;
if( !TrieAdd( InStandard[i].ev, buff ) ) {
TrieFini();
return( FALSE );
}
}
for( i = 0; i < NUM_IN_TERM_INFO_MAPPINGS; ++i ) {
str = InTerminfo[i].str;
if( !TrieAdd( InTerminfo[i].ev, (str?str:"") ) ) {
TrieFini();
return( FALSE );
}
}
return( TRUE );
}
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?