edit_key_translator.c 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331
  1. /*
  2. these #defines are probably the ones most people will be interested in.
  3. You can use these two #defines to hard code the key mappings --- just
  4. uncomment the one you want. But only if you have trouble with learn
  5. keys (which is unlikely).
  6. */
  7. /* KEY_BACKSPACE is the key learned in the learn keys menu : */
  8. #define OUR_BACKSPACE_KEY KEY_BACKSPACE
  9. /* ...otherwise ctrl-h : */
  10. /* #define OUR_BACKSPACE_KEY XCTRL ('h') */
  11. /* ...otherwise 127 or DEL in ascii : */
  12. /* #define OUR_BACKSPACE_KEY 0177 */
  13. /* KEY_DC is the key learned in the learn keys menu */
  14. #define OUR_DELETE_KEY KEY_DC
  15. /* ...otherwise ctrl-d : */
  16. /* #define OUR_DELETE_KEY XCTRL ('d') */
  17. /* ...otherwise 127 or DEL in ascii : */
  18. /* #define OUR_DELETE_KEY 0177 */
  19. /*
  20. This is #include'd into the function edit_translate_key in edit.c.
  21. This sequence of code takes 'x_state' and 'x_key' and translates them
  22. into either 'command' or 'char_for_insertion'. 'x_key' holds one of
  23. KEY_NPAGE, KEY_HOME etc., and 'x_state' holds a bitwise inclusive OR of
  24. CONTROL_PRESSED, ALT_PRESSED or SHIFT_PRESSED, although none may
  25. be supported.
  26. 'command' is one of the editor commands editcmddef.h.
  27. Almost any C code can go into this file. The code below is an example
  28. that may by appended or modified by the user.
  29. */
  30. /* look in this file for the list of commands : */
  31. #include "editcmddef.h"
  32. #define KEY_NUMLOCK ???
  33. /* ordinary translations. (Some of this may be redundant.) Note that keys listed
  34. first take priority when a key is assigned to more than one command */
  35. static const long *key_map;
  36. static const long cooledit_key_map[] =
  37. {OUR_BACKSPACE_KEY, CK_BackSpace, OUR_DELETE_KEY, CK_Delete,
  38. XCTRL ('d'), CK_Delete, '\n', CK_Enter,
  39. KEY_PPAGE, CK_Page_Up, KEY_NPAGE, CK_Page_Down, KEY_LEFT, CK_Left,
  40. KEY_RIGHT, CK_Right, KEY_UP, CK_Up, KEY_DOWN, CK_Down,
  41. ALT ('\t'), CK_Complete_Word, ALT ('\n'), CK_Return,
  42. KEY_HOME, CK_Home, KEY_END, CK_End, '\t', CK_Tab,
  43. XCTRL ('u'), CK_Undo, KEY_IC, CK_Toggle_Insert,
  44. XCTRL ('o'), CK_Shell, KEY_F (3), CK_Mark,
  45. KEY_F (13), CK_Column_Mark, KEY_F (5), CK_Copy,
  46. KEY_F (6), CK_Move, KEY_F (8), CK_Remove, KEY_F (12), CK_Save_As,
  47. KEY_F (2), CK_Save, XCTRL ('n'), CK_New,
  48. XCTRL ('l'), CK_Refresh, ESC_CHAR, CK_Exit, KEY_F (10), CK_Exit,
  49. KEY_F (11), /* edit user menu */ CK_User_Menu,
  50. KEY_F (19), /*C formatter */ CK_Pipe_Block (0),
  51. XCTRL ('p'), /*spell check */ CK_Pipe_Block (1),
  52. KEY_F (15), CK_Insert_File,
  53. XCTRL ('f'), CK_Save_Block, KEY_F (1), CK_Help,
  54. ALT ('t'), CK_Sort, ALT ('m'), CK_Mail,
  55. XCTRL ('z'), CK_Word_Left, XCTRL ('x'), CK_Word_Right,
  56. KEY_F (4), CK_Replace, KEY_F (7), CK_Find, KEY_F (14), CK_Replace_Again,
  57. XCTRL ('h'), CK_BackSpace, ALT ('l'), CK_Goto, ALT ('L'), CK_Goto,
  58. XCTRL ('y'), CK_Delete_Line,
  59. XCTRL ('k'), CK_Delete_To_Line_End,
  60. KEY_F (17), CK_Find_Again, ALT ('p'), CK_Paragraph_Format,
  61. ALT ('b'), CK_Match_Bracket,
  62. 0177, CK_BackSpace,
  63. 0, 0};
  64. static long const emacs_key_map[] =
  65. {OUR_BACKSPACE_KEY, CK_BackSpace, OUR_DELETE_KEY, CK_Delete, '\n', CK_Enter,
  66. KEY_PPAGE, CK_Page_Up, KEY_NPAGE, CK_Page_Down, KEY_LEFT, CK_Left,
  67. KEY_RIGHT, CK_Right, KEY_UP, CK_Up, KEY_DOWN, CK_Down,
  68. ALT ('\t'), CK_Complete_Word, ALT ('\n'), CK_Return,
  69. KEY_HOME, CK_Home, KEY_END, CK_End, '\t', CK_Tab,
  70. XCTRL ('u'), CK_Undo, KEY_IC, CK_Toggle_Insert,
  71. XCTRL ('o'), CK_Shell, KEY_F (3), CK_Mark, KEY_F (13), CK_Column_Mark,
  72. KEY_F (5), CK_Copy,
  73. KEY_F (6), CK_Move, KEY_F (8), CK_Remove, KEY_F (12), CK_Save_As,
  74. KEY_F (2), CK_Save, ALT ('p'), CK_Paragraph_Format,
  75. ALT ('t'), CK_Sort,
  76. XCTRL ('a'), CK_Home, XCTRL ('e'), CK_End,
  77. XCTRL ('b'), CK_Left, XCTRL ('f'), CK_Right,
  78. XCTRL ('n'), CK_Down, XCTRL ('p'), CK_Up,
  79. XCTRL ('d'), CK_Delete,
  80. XCTRL ('v'), CK_Page_Down, ALT ('v'), CK_Page_Up,
  81. XCTRL ('@'), CK_Mark,
  82. XCTRL ('k'), CK_Delete_To_Line_End,
  83. XCTRL ('s'), CK_Find,
  84. ALT ('b'), CK_Word_Left, ALT ('f'), CK_Word_Right,
  85. XCTRL ('w'), CK_XCut,
  86. XCTRL ('y'), CK_XPaste,
  87. ALT ('w'), CK_XStore,
  88. XCTRL ('l'), CK_Refresh, ESC_CHAR, CK_Exit, KEY_F (10), CK_Exit,
  89. KEY_F (11), /* edit user menu */ CK_User_Menu,
  90. KEY_F (19), /*C formatter */ CK_Pipe_Block (0),
  91. ALT ('$'), /*spell check */ CK_Pipe_Block (1),
  92. KEY_F (15), CK_Insert_File,
  93. KEY_F (1), CK_Help,
  94. KEY_F (4), CK_Replace, KEY_F (7), CK_Find, KEY_F (14), CK_Replace_Again,
  95. XCTRL ('h'), CK_BackSpace, ALT ('l'), CK_Goto, ALT ('L'), CK_Goto,
  96. KEY_F (17), CK_Find_Again,
  97. ALT ('<'), CK_Beginning_Of_Text,
  98. ALT ('>'), CK_End_Of_Text,
  99. 0177, CK_BackSpace,
  100. 0, 0};
  101. static long const key_pad_map[10] =
  102. {XCTRL ('o'), KEY_END, KEY_DOWN, KEY_NPAGE, KEY_LEFT,
  103. KEY_DOWN, KEY_RIGHT, KEY_HOME, KEY_UP, KEY_PPAGE};
  104. #define DEFAULT_NUM_LOCK 0
  105. static int num_lock = DEFAULT_NUM_LOCK;
  106. int i = 0;
  107. switch (edit_key_emulation) {
  108. case EDIT_KEY_EMULATION_NORMAL:
  109. key_map = cooledit_key_map;
  110. break;
  111. case EDIT_KEY_EMULATION_EMACS:
  112. key_map = emacs_key_map;
  113. if (x_key == XCTRL ('x')) {
  114. int ext_key;
  115. ext_key = edit_raw_key_query (" Ctrl-X ", _(" Emacs key: "), 0);
  116. switch (ext_key) {
  117. case 's':
  118. command = CK_Save;
  119. goto fin;
  120. case 'x':
  121. command = CK_Exit;
  122. goto fin;
  123. case 'k':
  124. command = CK_New;
  125. goto fin;
  126. case 'e':
  127. command = CK_Macro (edit_raw_key_query (_(" Execute Macro "), _(" Press macro hotkey: "), 1));
  128. if (command == CK_Macro (0))
  129. command = -1;
  130. goto fin;
  131. }
  132. goto fin;
  133. }
  134. break;
  135. }
  136. #ifdef HAVE_CHARSET
  137. if (x_key == XCTRL('t')) {
  138. do_select_codepage();
  139. edit->force = REDRAW_COMPLETELY;
  140. command = CK_Refresh;
  141. goto fin;
  142. }
  143. #endif
  144. if (x_key == XCTRL ('q')) {
  145. char_for_insertion = edit_raw_key_query (_(" Insert Literal "), _(" Press any key: "), 0);
  146. goto fin;
  147. }
  148. if (x_key == XCTRL ('a') && edit_key_emulation != EDIT_KEY_EMULATION_EMACS) {
  149. command = CK_Macro (edit_raw_key_query (_(" Execute Macro "), _(" Press macro hotkey: "), 1));
  150. if (command == CK_Macro (0))
  151. command = -1;
  152. goto fin;
  153. }
  154. /* edit is a pointer to the widget */
  155. if (edit)
  156. if (x_key == XCTRL ('r')) {
  157. command = edit->macro_i < 0 ? CK_Begin_Record_Macro : CK_End_Record_Macro;
  158. goto fin;
  159. }
  160. /* if (x_key == KEY_NUMLOCK) {
  161. num_lock = 1 - num_lock;
  162. return 1;
  163. }
  164. */
  165. /* first translate the key-pad */
  166. if (num_lock) {
  167. if (x_key >= '0' && x_key <= '9') {
  168. x_key = key_pad_map[x_key - '0'];
  169. }
  170. if (x_key == '.') {
  171. x_key = KEY_DC;
  172. }
  173. }
  174. if ((x_state & SHIFT_PRESSED) && (x_state & CONTROL_PRESSED)) {
  175. switch (x_key) {
  176. case KEY_PPAGE:
  177. command = CK_Beginning_Of_Text_Highlight;
  178. goto fin;
  179. case KEY_NPAGE:
  180. command = CK_End_Of_Text_Highlight;
  181. goto fin;
  182. case KEY_LEFT:
  183. command = CK_Word_Left_Highlight;
  184. goto fin;
  185. case KEY_RIGHT:
  186. command = CK_Word_Right_Highlight;
  187. goto fin;
  188. case KEY_UP:
  189. command = CK_Scroll_Up_Highlight;
  190. goto fin;
  191. case KEY_DOWN:
  192. command = CK_Scroll_Down_Highlight;
  193. goto fin;
  194. }
  195. }
  196. if ((x_state & SHIFT_PRESSED) && !(x_state & CONTROL_PRESSED)) {
  197. switch (x_key) {
  198. case KEY_PPAGE:
  199. command = CK_Page_Up_Highlight;
  200. goto fin;
  201. case KEY_NPAGE:
  202. command = CK_Page_Down_Highlight;
  203. goto fin;
  204. case KEY_LEFT:
  205. command = CK_Left_Highlight;
  206. goto fin;
  207. case KEY_RIGHT:
  208. command = CK_Right_Highlight;
  209. goto fin;
  210. case KEY_UP:
  211. command = CK_Up_Highlight;
  212. goto fin;
  213. case KEY_DOWN:
  214. command = CK_Down_Highlight;
  215. goto fin;
  216. case KEY_HOME:
  217. command = CK_Home_Highlight;
  218. goto fin;
  219. case KEY_END:
  220. command = CK_End_Highlight;
  221. goto fin;
  222. case KEY_IC:
  223. command = CK_XPaste;
  224. goto fin;
  225. case KEY_DC:
  226. command = CK_XCut;
  227. goto fin;
  228. }
  229. }
  230. /* things that need a control key */
  231. if (x_state & CONTROL_PRESSED) {
  232. switch (x_key) {
  233. case KEY_F (2):
  234. command = CK_Save_As;
  235. goto fin;
  236. case KEY_F (4):
  237. command = CK_Replace_Again;
  238. goto fin;
  239. case KEY_F (7):
  240. command = CK_Find_Again;
  241. goto fin;
  242. case KEY_BACKSPACE:
  243. command = CK_Undo;
  244. goto fin;
  245. case KEY_PPAGE:
  246. command = CK_Beginning_Of_Text;
  247. goto fin;
  248. case KEY_NPAGE:
  249. command = CK_End_Of_Text;
  250. goto fin;
  251. case KEY_UP:
  252. command = CK_Scroll_Up;
  253. goto fin;
  254. case KEY_DOWN:
  255. command = CK_Scroll_Down;
  256. goto fin;
  257. case KEY_LEFT:
  258. command = CK_Word_Left;
  259. goto fin;
  260. case KEY_RIGHT:
  261. command = CK_Word_Right;
  262. goto fin;
  263. case KEY_IC:
  264. command = CK_XStore;
  265. goto fin;
  266. case KEY_DC:
  267. command = CK_Remove;
  268. goto fin;
  269. }
  270. }
  271. /* an ordinary insertable character */
  272. #ifndef HAVE_CHARSET
  273. if (x_key < 256 && is_printable (x_key)) {
  274. char_for_insertion = x_key;
  275. #else
  276. if (x_key < 256 && is_printable (conv_input [x_key])) {
  277. char_for_insertion = conv_input [x_key];
  278. #endif
  279. goto fin;
  280. }
  281. /* other commands */
  282. i = 0;
  283. while (key_map[i] != x_key && (key_map[i] || key_map[i + 1]))
  284. i += 2;
  285. command = key_map[i + 1];
  286. if (command)
  287. goto fin;
  288. /* Function still not found for this key, so try macro's */
  289. /* This allows the same macro to be
  290. enabled by either eg "ALT('f')" or "XCTRL('f')" or "XCTRL('a'), 'f'" */
  291. /* key.h: #define ALT(x) (0x200 | (x)) */
  292. if (x_key & ALT (0)) { /* is an alt key ? */
  293. command = CK_Macro (x_key - ALT (0));
  294. goto fin;
  295. }
  296. /* key.h: #define XCTRL(x) ((x) & 31) */
  297. if (x_key < ' ') { /* is a ctrl key ? */
  298. command = CK_Macro (x_key);
  299. goto fin;
  300. }
  301. fin: