⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 xkeymap.c

📁 rdesktop是一个开放源码的Window NT中断服务器的客户端
💻 C
📖 第 1 页 / 共 2 页
字号:
/*   rdesktop: A Remote Desktop Protocol client.   User interface services - X keyboard mapping   Copyright (C) Matthew Chapman 1999-2005   Copyright (C) Peter Astrand <peter@cendio.se> 2003      This program is free software; you can redistribute it and/or modify   it under the terms of the GNU General Public License as published by   the Free Software Foundation; either version 2 of the License, or   (at your option) any later version.      This program is distributed in the hope that it will be useful,   but WITHOUT ANY WARRANTY; without even the implied warranty of   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the   GNU General Public License for more details.   You should have received a copy of the GNU General Public License   along with this program; if not, write to the Free Software   Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.*/#ifdef RDP2VNC#include "vnc/x11stubs.h"#else#include <X11/Xlib.h>#include <X11/keysym.h>#endif#include <ctype.h>#include <limits.h>#include <time.h>#include "rdesktop.h"#include "scancodes.h"#define KEYMAP_SIZE 0xffff+1#define KEYMAP_MASK 0xffff#define KEYMAP_MAX_LINE_LENGTH 80extern Display *g_display;extern Window g_wnd;extern char keymapname[16];extern int g_keylayout;extern int g_win_button_size;extern BOOL g_enable_compose;extern BOOL g_use_rdp5;extern BOOL g_numlock_sync;static BOOL keymap_loaded;static key_translation keymap[KEYMAP_SIZE];static int min_keycode;static uint16 remote_modifier_state = 0;static uint16 saved_remote_modifier_state = 0;static void update_modifier_state(uint8 scancode, BOOL pressed);static voidadd_to_keymap(char *keyname, uint8 scancode, uint16 modifiers, char *mapname){	KeySym keysym;	keysym = XStringToKeysym(keyname);	if (keysym == NoSymbol)	{		DEBUG_KBD(("Bad keysym \"%s\" in keymap %s (ignoring)\n", keyname, mapname));		return;	}	DEBUG_KBD(("Adding translation, keysym=0x%x, scancode=0x%x, "		   "modifiers=0x%x\n", (unsigned int) keysym, scancode, modifiers));	keymap[keysym & KEYMAP_MASK].scancode = scancode;	keymap[keysym & KEYMAP_MASK].modifiers = modifiers;	return;}static BOOLxkeymap_read(char *mapname){	FILE *fp;	char line[KEYMAP_MAX_LINE_LENGTH];	char path[PATH_MAX], inplace_path[PATH_MAX];	unsigned int line_num = 0;	unsigned int line_length = 0;	char *keyname, *p;	char *line_rest;	uint8 scancode;	uint16 modifiers;	strcpy(path, KEYMAP_PATH);	strncat(path, mapname, sizeof(path) - sizeof(KEYMAP_PATH));	fp = fopen(path, "r");	if (fp == NULL)	{		/* in case we are running from the source tree */		strcpy(inplace_path, "keymaps/");		strncat(inplace_path, mapname, sizeof(inplace_path) - sizeof("keymaps/"));		fp = fopen(inplace_path, "r");		if (fp == NULL)		{			error("Failed to open keymap %s\n", path);			return False;		}	}	/* FIXME: More tolerant on white space */	while (fgets(line, sizeof(line), fp) != NULL)	{		line_num++;		/* Replace the \n with \0 */		p = strchr(line, '\n');		if (p != NULL)			*p = 0;		line_length = strlen(line);		/* Completely empty line */		if (strspn(line, " \t\n\r\f\v") == line_length)		{			continue;		}		/* Include */		if (strncmp(line, "include ", 8) == 0)		{			if (!xkeymap_read(line + 8))				return False;			continue;		}		/* map */		if (strncmp(line, "map ", 4) == 0)		{			g_keylayout = strtol(line + 4, NULL, 16);			DEBUG_KBD(("Keylayout 0x%x\n", g_keylayout));			continue;		}		/* compose */		if (strncmp(line, "enable_compose", 15) == 0)		{			DEBUG_KBD(("Enabling compose handling\n"));			g_enable_compose = True;			continue;		}		/* Comment */		if (line[0] == '#')		{			continue;		}		/* Normal line */		keyname = line;		p = strchr(line, ' ');		if (p == NULL)		{			error("Bad line %d in keymap %s\n", line_num, mapname);			continue;		}		else		{			*p = 0;		}		/* scancode */		p++;		scancode = strtol(p, &line_rest, 16);		/* flags */		/* FIXME: Should allow case-insensitive flag names. 		   Fix by using lex+yacc... */		modifiers = 0;		if (strstr(line_rest, "altgr"))		{			MASK_ADD_BITS(modifiers, MapAltGrMask);		}		if (strstr(line_rest, "shift"))		{			MASK_ADD_BITS(modifiers, MapLeftShiftMask);		}		if (strstr(line_rest, "numlock"))		{			MASK_ADD_BITS(modifiers, MapNumLockMask);		}		if (strstr(line_rest, "localstate"))		{			MASK_ADD_BITS(modifiers, MapLocalStateMask);		}		if (strstr(line_rest, "inhibit"))		{			MASK_ADD_BITS(modifiers, MapInhibitMask);		}		add_to_keymap(keyname, scancode, modifiers, mapname);		if (strstr(line_rest, "addupper"))		{			/* Automatically add uppercase key, with same modifiers 			   plus shift */			for (p = keyname; *p; p++)				*p = toupper((int) *p);			MASK_ADD_BITS(modifiers, MapLeftShiftMask);			add_to_keymap(keyname, scancode, modifiers, mapname);		}	}	fclose(fp);	return True;}/* Before connecting and creating UI */voidxkeymap_init(void){	unsigned int max_keycode;	char *mapname_ptr;	/* Make keymapname lowercase */	mapname_ptr = keymapname;	while (*mapname_ptr)	{		*mapname_ptr = tolower((int) *mapname_ptr);		mapname_ptr++;	}	if (strcmp(keymapname, "none"))	{		if (xkeymap_read(keymapname))			keymap_loaded = True;	}	XDisplayKeycodes(g_display, &min_keycode, (int *) &max_keycode);}static voidsend_winkey(uint32 ev_time, BOOL pressed, BOOL leftkey){	uint8 winkey;	if (leftkey)		winkey = SCANCODE_CHAR_LWIN;	else		winkey = SCANCODE_CHAR_RWIN;	if (pressed)	{		if (g_use_rdp5)		{			rdp_send_scancode(ev_time, RDP_KEYPRESS, winkey);		}		else		{			/* RDP4 doesn't support winkey. Fake with Ctrl-Esc */			rdp_send_scancode(ev_time, RDP_KEYPRESS, SCANCODE_CHAR_LCTRL);			rdp_send_scancode(ev_time, RDP_KEYPRESS, SCANCODE_CHAR_ESC);		}	}	else	{		/* key released */		if (g_use_rdp5)		{			rdp_send_scancode(ev_time, RDP_KEYRELEASE, winkey);		}		else		{			rdp_send_scancode(ev_time, RDP_KEYRELEASE, SCANCODE_CHAR_ESC);			rdp_send_scancode(ev_time, RDP_KEYRELEASE, SCANCODE_CHAR_LCTRL);		}	}}static voidreset_winkey(uint32 ev_time){	if (g_use_rdp5)	{		/* For some reason, it seems to suffice to release		 *either* the left or right winkey. */		rdp_send_scancode(ev_time, RDP_KEYRELEASE, SCANCODE_CHAR_LWIN);	}}/* Handles, for example, multi-scancode keypresses (which is not   possible via keymap-files) */BOOLhandle_special_keys(uint32 keysym, unsigned int state, uint32 ev_time, BOOL pressed){	switch (keysym)	{		case XK_Return:			if ((get_key_state(state, XK_Alt_L) || get_key_state(state, XK_Alt_R))			    && (get_key_state(state, XK_Control_L)				|| get_key_state(state, XK_Control_R)))			{				/* Ctrl-Alt-Enter: toggle full screen */				if (pressed)					xwin_toggle_fullscreen();				return True;			}			break;		case XK_Break:			/* Send Break sequence E0 46 E0 C6 */			if (pressed)			{				rdp_send_scancode(ev_time, RDP_KEYPRESS,						  (SCANCODE_EXTENDED | 0x46));				rdp_send_scancode(ev_time, RDP_KEYPRESS,						  (SCANCODE_EXTENDED | 0xc6));			}			/* No release sequence */			return True;			break;		case XK_Pause:			/* According to MS Keyboard Scan Code			   Specification, pressing Pause should result			   in E1 1D 45 E1 9D C5. I'm not exactly sure			   of how this is supposed to be sent via			   RDP. The code below seems to work, but with			   the side effect that Left Ctrl stays			   down. Therefore, we release it when Pause			   is released. */			if (pressed)			{				rdp_send_input(ev_time, RDP_INPUT_SCANCODE, RDP_KEYPRESS, 0xe1, 0);				rdp_send_input(ev_time, RDP_INPUT_SCANCODE, RDP_KEYPRESS, 0x1d, 0);				rdp_send_input(ev_time, RDP_INPUT_SCANCODE, RDP_KEYPRESS, 0x45, 0);				rdp_send_input(ev_time, RDP_INPUT_SCANCODE, RDP_KEYPRESS, 0xe1, 0);				rdp_send_input(ev_time, RDP_INPUT_SCANCODE, RDP_KEYPRESS, 0x9d, 0);				rdp_send_input(ev_time, RDP_INPUT_SCANCODE, RDP_KEYPRESS, 0xc5, 0);			}			else			{				/* Release Left Ctrl */				rdp_send_input(ev_time, RDP_INPUT_SCANCODE, RDP_KEYRELEASE,					       0x1d, 0);			}			return True;			break;		case XK_Meta_L:	/* Windows keys */		case XK_Super_L:		case XK_Hyper_L:			send_winkey(ev_time, pressed, True);			return True;			break;		case XK_Meta_R:		case XK_Super_R:		case XK_Hyper_R:			send_winkey(ev_time, pressed, False);			return True;			break;		case XK_space:			/* Prevent access to the Windows system menu in single app mode */			if (g_win_button_size			    && (get_key_state(state, XK_Alt_L) || get_key_state(state, XK_Alt_R)))				return True;			break;		case XK_Num_Lock:			/* FIXME: We might want to do RDP_INPUT_SYNCHRONIZE here, if g_numlock_sync */			if (!g_numlock_sync)				/* Inhibit */				return True;

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -