plugin.ts 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369
  1. // Copyright (c) Jupyter Development Team.
  2. // Distributed under the terms of the Modified BSD License.
  3. import {
  4. JupyterLab, JupyterLabPlugin
  5. } from '../application';
  6. import {
  7. CommandIDs as CommandPaletteCommandIDs
  8. } from '../commandpalette';
  9. import {
  10. CommandIDs as ConsoleCommandIDs
  11. } from '../console';
  12. import {
  13. CommandIDs as EditorWidgetCommandIDs
  14. } from '../editorwidget';
  15. import {
  16. CommandIDs as FileBrowserCommandIDs
  17. } from '../filebrowser';
  18. import {
  19. CommandIDs as HelpCommandIDs
  20. } from '../help';
  21. import {
  22. CommandIDs as ImageWidgetCommandIDs
  23. } from '../imagewidget';
  24. import {
  25. CommandIDs as InspectorCommandIDs
  26. } from '../inspector';
  27. import {
  28. CommandIDs as NotebookCommandIDs
  29. } from '../notebook';
  30. import {
  31. CommandIDs as TooltipCommandIDs
  32. } from '../tooltip';
  33. /**
  34. * The list of default application shortcuts.
  35. *
  36. * #### Notes
  37. * When setting shortcut selectors, there are two concepts to consider:
  38. * specificity and matchability. These two interact in sometimes
  39. * counterintuitive ways. Keyboard events are triggered from an element and
  40. * they propagate up the DOM until they reach the `documentElement` (`<body>`).
  41. *
  42. * When a registered shortcut sequence is fired, the shortcut manager checks
  43. * the node that fired the event and each of its ancestors until a node matches
  44. * one or more registered selectors. The *first* matching selector in the
  45. * chain of ancestors will invoke the shortcut handler and the traversal will
  46. * end at that point. If a node matches more than one selector, the handler for
  47. * whichever selector is more *specific* fires.
  48. * @see https://www.w3.org/TR/css3-selectors/#specificity
  49. *
  50. * The practical consequence of this is that a very broadly matching selector,
  51. * e.g. `'*'` or `'div'` may match and therefore invoke a handler *before* a
  52. * more specific selector. The most common pitfall is to use the universal
  53. * (`'*'`) selector. For almost any use case where a global keyboard shortcut is
  54. * required, using the `'body'` selector is more appropriate.
  55. */
  56. const SHORTCUTS = [
  57. {
  58. command: CommandPaletteCommandIDs.activate,
  59. selector: 'body',
  60. keys: ['Accel Shift P']
  61. },
  62. {
  63. command: ConsoleCommandIDs.run,
  64. selector: '.jp-CodeConsole-prompt',
  65. keys: ['Enter']
  66. },
  67. {
  68. command: ConsoleCommandIDs.runForced,
  69. selector: '.jp-CodeConsole-prompt',
  70. keys: ['Shift Enter']
  71. },
  72. {
  73. command: ConsoleCommandIDs.linebreak,
  74. selector: '.jp-CodeConsole-prompt',
  75. keys: ['Ctrl Enter']
  76. },
  77. {
  78. command: EditorWidgetCommandIDs.runCode,
  79. selector: '.jp-EditorWidget',
  80. keys: ['Shift Enter']
  81. },
  82. {
  83. command: FileBrowserCommandIDs.toggleBrowser,
  84. selector: 'body',
  85. keys: ['Accel Shift F']
  86. },
  87. {
  88. command: FileBrowserCommandIDs.newTextFile,
  89. selector: 'body',
  90. keys: ['Ctrl O']
  91. },
  92. {
  93. command: FileBrowserCommandIDs.newNotebook,
  94. selector: 'body',
  95. keys: ['Ctrl Shift N']
  96. },
  97. {
  98. command: FileBrowserCommandIDs.save,
  99. selector: '.jp-Document',
  100. keys: ['Accel S']
  101. },
  102. {
  103. command: FileBrowserCommandIDs.close,
  104. selector: '.jp-Document',
  105. keys: ['Ctrl Q']
  106. },
  107. {
  108. command: FileBrowserCommandIDs.closeAllFiles,
  109. selector: '.jp-Document',
  110. keys: ['Ctrl Shift Q']
  111. },
  112. {
  113. command: HelpCommandIDs.toggle,
  114. selector: 'body',
  115. keys: ['Accel Shift H']
  116. },
  117. {
  118. command: ImageWidgetCommandIDs.zoomIn,
  119. selector: '.jp-ImageWidget',
  120. keys: ['=']
  121. },
  122. {
  123. command: ImageWidgetCommandIDs.zoomOut,
  124. selector: '.jp-ImageWidget',
  125. keys: ['-']
  126. },
  127. {
  128. command: ImageWidgetCommandIDs.resetZoom,
  129. selector: '.jp-ImageWidget',
  130. keys: ['0']
  131. },
  132. {
  133. command: InspectorCommandIDs.open,
  134. selector: '.jp-CodeConsole-prompt',
  135. keys: ['Accel I']
  136. },
  137. {
  138. command: NotebookCommandIDs.runAndAdvance,
  139. selector: '.jp-Notebook',
  140. keys: ['Shift Enter']
  141. },
  142. {
  143. command: NotebookCommandIDs.runAndInsert,
  144. selector: '.jp-Notebook',
  145. keys: ['Alt Enter']
  146. },
  147. {
  148. command: NotebookCommandIDs.run,
  149. selector: '.jp-Notebook',
  150. keys: ['Ctrl Enter']
  151. },
  152. {
  153. command: NotebookCommandIDs.interrupt,
  154. selector: '.jp-Notebook.jp-mod-commandMode',
  155. keys: ['I', 'I']
  156. },
  157. {
  158. command: NotebookCommandIDs.restart,
  159. selector: '.jp-Notebook.jp-mod-commandMode',
  160. keys: ['0', '0']
  161. },
  162. {
  163. command: NotebookCommandIDs.toCode,
  164. selector: '.jp-Notebook.jp-mod-commandMode',
  165. keys: ['Y']
  166. },
  167. {
  168. command: NotebookCommandIDs.toMarkdown,
  169. selector: '.jp-Notebook.jp-mod-commandMode',
  170. keys: ['M']
  171. },
  172. {
  173. command: NotebookCommandIDs.toRaw,
  174. selector: '.jp-Notebook.jp-mod-commandMode',
  175. keys: ['R']
  176. },
  177. {
  178. command: NotebookCommandIDs.deleteCell,
  179. selector: '.jp-Notebook.jp-mod-commandMode',
  180. keys: ['D', 'D'],
  181. },
  182. {
  183. command: NotebookCommandIDs.split,
  184. selector: '.jp-Notebook.jp-mod-editMode',
  185. keys: ['Ctrl Shift -'],
  186. },
  187. {
  188. command: NotebookCommandIDs.merge,
  189. selector: '.jp-Notebook.jp-mod-commandMode',
  190. keys: ['Shift M'],
  191. },
  192. {
  193. command: NotebookCommandIDs.selectAbove,
  194. selector: '.jp-Notebook.jp-mod-commandMode',
  195. keys: ['ArrowUp'],
  196. },
  197. {
  198. command: NotebookCommandIDs.selectAbove,
  199. selector: '.jp-Notebook.jp-mod-commandMode',
  200. keys: ['K'],
  201. },
  202. {
  203. command: NotebookCommandIDs.selectBelow,
  204. selector: '.jp-Notebook.jp-mod-commandMode',
  205. keys: ['ArrowDown'],
  206. },
  207. {
  208. command: NotebookCommandIDs.selectBelow,
  209. selector: '.jp-Notebook.jp-mod-commandMode',
  210. keys: ['J'],
  211. },
  212. {
  213. command: NotebookCommandIDs.extendAbove,
  214. selector: '.jp-Notebook.jp-mod-commandMode',
  215. keys: ['Shift ArrowUp'],
  216. },
  217. {
  218. command: NotebookCommandIDs.extendAbove,
  219. selector: '.jp-Notebook.jp-mod-commandMode',
  220. keys: ['Shift K'],
  221. },
  222. {
  223. command: NotebookCommandIDs.extendBelow,
  224. selector: '.jp-Notebook.jp-mod-commandMode',
  225. keys: ['Shift ArrowDown'],
  226. },
  227. {
  228. command: NotebookCommandIDs.extendBelow,
  229. selector: '.jp-Notebook.jp-mod-commandMode',
  230. keys: ['Shift J'],
  231. },
  232. {
  233. command: NotebookCommandIDs.undo,
  234. selector: '.jp-Notebook.jp-mod-commandMode',
  235. keys: ['Z'],
  236. },
  237. {
  238. command: NotebookCommandIDs.redo,
  239. selector: '.jp-Notebook.jp-mod-commandMode',
  240. keys: ['Shift Z'],
  241. },
  242. {
  243. command: NotebookCommandIDs.cut,
  244. selector: '.jp-Notebook.jp-mod-commandMode',
  245. keys: ['X']
  246. },
  247. {
  248. command: NotebookCommandIDs.copy,
  249. selector: '.jp-Notebook.jp-mod-commandMode',
  250. keys: ['C']
  251. },
  252. {
  253. command: NotebookCommandIDs.paste,
  254. selector: '.jp-Notebook.jp-mod-commandMode',
  255. keys: ['V']
  256. },
  257. {
  258. command: NotebookCommandIDs.insertAbove,
  259. selector: '.jp-Notebook.jp-mod-commandMode',
  260. keys: ['A']
  261. },
  262. {
  263. command: NotebookCommandIDs.insertBelow,
  264. selector: '.jp-Notebook.jp-mod-commandMode',
  265. keys: ['B']
  266. },
  267. {
  268. command: NotebookCommandIDs.toggleLines,
  269. selector: '.jp-Notebook.jp-mod-commandMode',
  270. keys: ['L']
  271. },
  272. {
  273. command: NotebookCommandIDs.markdown1,
  274. selector: '.jp-Notebook.jp-mod-commandMode',
  275. keys: ['1']
  276. },
  277. {
  278. command: NotebookCommandIDs.markdown2,
  279. selector: '.jp-Notebook.jp-mod-commandMode',
  280. keys: ['2']
  281. },
  282. {
  283. command: NotebookCommandIDs.markdown3,
  284. selector: '.jp-Notebook.jp-mod-commandMode',
  285. keys: ['3']
  286. },
  287. {
  288. command: NotebookCommandIDs.markdown4,
  289. selector: '.jp-Notebook.jp-mod-commandMode',
  290. keys: ['4']
  291. },
  292. {
  293. command: NotebookCommandIDs.markdown5,
  294. selector: '.jp-Notebook.jp-mod-commandMode',
  295. keys: ['5']
  296. },
  297. {
  298. command: NotebookCommandIDs.markdown6,
  299. selector: '.jp-Notebook.jp-mod-commandMode',
  300. keys: ['6']
  301. },
  302. {
  303. command: NotebookCommandIDs.editMode,
  304. selector: '.jp-Notebook.jp-mod-commandMode',
  305. keys: ['Enter']
  306. },
  307. {
  308. command: NotebookCommandIDs.commandMode,
  309. selector: '.jp-Notebook.jp-mod-editMode',
  310. keys: ['Escape']
  311. },
  312. {
  313. command: NotebookCommandIDs.commandMode,
  314. selector: '.jp-Notebook.jp-mod-editMode',
  315. keys: ['Ctrl M']
  316. },
  317. {
  318. command: TooltipCommandIDs.launch,
  319. selector: '.jp-Notebook',
  320. keys: ['Shift Tab'],
  321. args: { notebook: true }
  322. },
  323. {
  324. command: TooltipCommandIDs.launch,
  325. selector: '.jp-ConsolePanel',
  326. keys: ['Shift Tab'],
  327. args: { notebook: false }
  328. },
  329. {
  330. command: TooltipCommandIDs.remove,
  331. selector: '.jp-Notebook.jp-Tooltip-anchor',
  332. keys: ['Escape']
  333. },
  334. {
  335. command: TooltipCommandIDs.remove,
  336. selector: '.jp-CodeConsole.jp-Tooltip-anchor',
  337. keys: ['Escape']
  338. }
  339. ];
  340. /**
  341. * The default shortcuts extension.
  342. */
  343. const plugin: JupyterLabPlugin<void> = {
  344. id: 'jupyter.extensions.shortcuts',
  345. activate: (app: JupyterLab): void => {
  346. SHORTCUTS.forEach(shortcut => { app.keymap.addBinding(shortcut); });
  347. },
  348. autoStart: true
  349. };
  350. /**
  351. * Export the plugin as default.
  352. */
  353. export default plugin;