📄 nxkbd.c
字号:
/* * nxkbd.c - Software popup keyboard for Nano-X * * Copyright (C) 2000 by Greg Haerr <greg@censoft.com> * linked-in bitmaps, redrawn keymaps * enhanced shift/control function, fnkeys * supports all ascii values 1-127 * Copyright (C) 2000 by VTech Informations LTD. * Vladimir Cotfas <vladimircotfas@vtech.ca> Aug 31, 2000 * initial port to Nano-X * Copyright (C) 2000 by Jay Carlson * initial soft kbd for W * * This code is licensed with the same license as Microwindows. * * #define KBDPIPE in srvconn.c for named pipe keyboard driver. * Otherwise, the GrInjectKeyboardEvent method is used. */#include <stdio.h>#include <stdlib.h>#define MWINCLUDECOLORS#include "nano-X.h"#define TITLE "Soft Keyboard"#define DISKIMAGES 0 /* =0 use linked-in images*/#define _SOFTKBD_DEBUG 0/* kbd states, each with unique bitmap*/#define NORM 1000#define CNTRL 1001#define SHIFT 1002#define NUM 1003#define INTL 1004/* special chars*/#define BS '\b' /* value of <- on kbd*/#define F1 2000#define NONE 3000/* number of charcodes per bitmap*/#define SCANCODES 41/* size of bitmaps*/#define BM_WIDTH 160#define BM_HEIGHT 61struct keycolumn { short xoffset; short scancode;};struct keyrow { short yoffset; short height; struct keycolumn columns[12];};/* fixed layout for each scancode location*/struct keyrow keyrows[4] = { {0, 15, {{0, 0}, {14, 1}, {28, 2}, {42, 3}, {56, 4}, {70, 5}, {84, 6}, {98, 7}, {112, 8}, {126, 9}, {140, 10}, {999, -1}}}, {15, 15, {{0, 11}, {14, 12}, {28, 13}, {42, 14}, {56, 15}, {70, 16}, {84, 17}, {98, 18}, {112, 19}, {126, 20}, {140, 21}, {999, -1}}}, {30, 15, {{0, 22}, {19, 23}, {33, 24}, {47, 25}, {61, 26}, {75, 27}, {89, 28}, {103, 29}, {117, 30}, {131, 31}, {145, 32}, {999, -1}} }, {45, 15, {{0, 33}, {21, 34}, {36, 35}, {85, 36}, {103, 37}, {117, 38}, {131, 39}, {145, 40}, {999, -1}}}};#define C(x) ((x)&0x1f)/* charcode mappings per kbd state*/static short normal[SCANCODES] = { 'q', 'w', 'e', 'r', 't', 'y', 'u', 'i', 'o', 'p', BS, 'a', 's', 'd', 'f', 'g', 'h', 'j', 'k', 'l', '-', '\r', CNTRL, 'z', 'x', 'c', 'v', 'b', 'n', 'm', ',', '.', ';', SHIFT, INTL, ' ', NUM, '\'', '=', '\\', '/'};static short control[SCANCODES] = { C('q'),C('w'),C('e'),C('r'),C('t'),C('y'),C('u'),C('i'),C('o'),C('p'),'\033', C('a'),C('s'),C('d'),C('f'),C('g'),C('h'),C('j'),C('k'),C('l'),C('_'),'\r', CNTRL,C('z'),C('x'),C('c'),C('v'),C('b'),C('n'),C('m'),C('\\'),C(']'),C('^'), SHIFT,INTL,' ',NUM,NONE,NONE,NONE,'\177'};static short shift[SCANCODES] = { 'Q', 'W', 'E', 'R', 'T', 'Y', 'U', 'I', 'O', 'P', BS, 'A', 'S', 'D', 'F', 'G', 'H', 'J', 'K', 'L', '_', '\r', CNTRL, 'Z', 'X', 'C', 'V', 'B', 'N', 'M', '<', '>', ':', SHIFT, INTL, ' ', NUM, '"', '+', '|', '?'};static short num[SCANCODES] = { '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', BS, '!', '@', '#', '$', '%', '^', '&', '*', '(', ')', '\r', CNTRL, F1, F1+1, F1+2, F1+3, F1+4, F1+5, F1+6, F1+7, '`', '~', SHIFT, INTL, ' ', NUM, '[', ']', '{', '}'};static char *fnkey[] = { "\033OP", "\033OQ", "\033OR", "\033OS", "\03315~", "\03317~", "\03318~", "\03319~", "\03320~", "\03321~"};struct layout_state { char *filename; GR_IMAGE_HDR *imagehdr; short *scancode_translations; short sshift; short ctrl_layout, shift_layout, num_layout;};/* auto-converted .bmp files for internal linking*/extern GR_IMAGE_HDR image_keynorm;extern GR_IMAGE_HDR image_keyctrl;extern GR_IMAGE_HDR image_keyshft;extern GR_IMAGE_HDR image_keynum;static struct layout_state layout_states[] = { { /* 0 */ "keynorm.bmp", &image_keynorm, normal, 0, 2, 1, 3 }, { /* 1 */ "keyshft.bmp", &image_keyshft, shift, 0, 2, 0, 3 }, { /* 2 */ "keyctrl.bmp", &image_keyctrl, control, 1, /* ctrl is single shift*/ 0, 1, 3 }, { /* 3 */ "keynum.bmp", &image_keynum, num, 0, 2, 1, 0 }};#define N_LAYOUT_STATES (sizeof(layout_states) / sizeof(layout_states[0]))static GR_WINDOW_ID w; static GR_GC_ID gc; /* graphics context for text */static int current_layout = 0;#if DISKIMAGESstatic GR_IMAGE_ID layout_images[N_LAYOUT_STATES];static int layout_images_loaded[N_LAYOUT_STATES];#endifextern int KbdWrite(int c);extern int KbdOpen(void);extern void KbdClose(void);static void push_character(int c){#if _SOFTKBD_DEBUG fprintf(stderr, "pushed %d (0x%x) '%c'\n", c, c, c);#endif KbdWrite(c);}static voiddisplay_layout(int layout) {#if DISKIMAGES#define LIBDIR "." /* "/etc/nxkbd.d" */ if (!layout_images_loaded[layout] ) { char buf[128]; /* * OK, load image on the server-side ;-) * DON'T check for errors ;( */ sprintf(buf, "%s/%s", LIBDIR, layout_states[layout].filename); layout_images[layout] = GrLoadImageFromFile(buf, 0); layout_images_loaded[layout] = 1; } GrDrawImageToFit(w, gc, 0, 0, -1, -1, layout_images[layout]);#else GrDrawImageBits(w, gc, 0, 0, layout_states[layout].imagehdr);#endif}static voidprocess_scancode(int scancode){ int c; c = layout_states[current_layout].scancode_translations[scancode];#if _SOFTKBD_DEBUG printf("scancode = %d ", scancode); printf("current_layout = %d ('%s'), scancode (translated) = %d\n", current_layout, layout_states[current_layout].filename, c);#endif switch (c) { default: if (c < 256) /* normal character*/ break; /* handle special functions*/ if (c >= F1 && c < F1+10) { char *p = fnkey[c-F1]; while (*p) push_character(*p++); return; } /* no action for NONE*/ if (c == NONE) return; fprintf(stderr, "nxkbd: key with unknown translation pressed\n"); return; case CNTRL: current_layout = layout_states[current_layout].ctrl_layout; display_layout(current_layout); return; case SHIFT: current_layout = layout_states[current_layout].shift_layout; display_layout(current_layout); return; case NUM: current_layout = layout_states[current_layout].num_layout; display_layout(current_layout); return; case INTL:#if _SOFTKBD_DEBUG printf("INTL not yet implemented\n");#endif return; } if (layout_states[current_layout].sshift) { current_layout = 0; display_layout(current_layout); } push_character(c);}static void mouse_hit(int x, int y) { int row, column; for (row = 0; row < 4; row++) { if (y >= keyrows[row].yoffset && y < keyrows[row].yoffset+keyrows[row].height) { for (column = 0; column < 12; column++) { if (keyrows[row].columns[column].xoffset == 999) { fprintf(stderr, "off end of row\n"); return; } if (x < keyrows[row].columns[column + 1].xoffset) { int scancode = keyrows[row].columns[column].scancode; process_scancode(scancode); return; } } } } fprintf(stderr, "nxkbd: off bottom\n");} intmain(int argc, char* argv[]){ GR_EVENT event; /* current event */ GR_WM_PROPERTIES props; if (GrOpen() < 0) { fprintf(stderr, "nxkbd: cannot open graphics\n"); exit(1); } if (KbdOpen() < 0) { fprintf(stderr, "nxkbd: cannot open kbd named pipe\n");#if 0 exit(1);#endif } #if !DISKIMAGES GrReqShmCmds(4096); /* fast image copy*/#endif w = GrNewWindow(GR_ROOT_WINDOW_ID, 0, 0, BM_WIDTH, BM_HEIGHT, 0, WHITE, BLACK); GrSelectEvents(w, GR_EVENT_MASK_CLOSE_REQ | GR_EVENT_MASK_EXPOSURE | /*GR_EVENT_MASK_FOCUS_IN |*/ /*GR_EVENT_MASK_KEY_DOWN |*/ /* required for focus*/ GR_EVENT_MASK_BUTTON_DOWN);#if 0 /* this code fails when link-app-into-server */ //if (props.title) // can't free with link-into-server //free(props.title); /* title must be alloc'd and copied*/ //props.title = malloc(18); //if (props.title) //strcpy(props.title, TITLE); props.flags = GR_WM_FLAG_NORESIZE | /* don't let user resize window */ GR_WM_FLAG_NOBORDERS | /* don't draw any window borders */ GR_WM_FLAG_NOTITLEBAR | /* don't draw a title bar */ GR_WM_FLAG_NOFOCUS; /* don't set focus to this window*/#endif props.flags = GR_WM_FLAGS_PROPS | GR_WM_FLAGS_TITLE; props.props = GR_WM_PROPS_NOFOCUS; props.props |= /*GR_WM_PROPS_NOMOVE |*/ GR_WM_PROPS_NORAISE | GR_WM_PROPS_BORDER | GR_WM_PROPS_CAPTION | GR_WM_PROPS_CLOSEBOX; props.title = TITLE; GrSetWMProperties(w, &props); GrMapWindow(w); gc = GrNewGC(); current_layout = 0; for (;;) { GrGetNextEvent(&event); switch(event.type) { case GR_EVENT_TYPE_CLOSE_REQ:#if DISKIMAGES { int i; for(i=0; i < N_LAYOUT_STATES; i++) { if( !layout_images_loaded[i] ) continue; GrFreeImage(layout_images[i]); } }#endif GrClose(); exit(0); /* no return*/ case GR_EVENT_TYPE_EXPOSURE: display_layout(current_layout); break; case GR_EVENT_TYPE_BUTTON_DOWN: mouse_hit(event.button.x, event.button.y); break;#if 0 case GR_EVENT_TYPE_FOCUS_IN: if (event.general.otherid != 1) { int lastfocus = event.general.otherid; GrSetFocus(lastfocus); } break;#endif } } /*NOTREACHED*/ return 0; }
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -