mirror of
				https://github.com/livebook-dev/livebook.git
				synced 2025-10-25 21:06:08 +08:00 
			
		
		
		
	vim/emacs modes with settings (#2173)
This commit is contained in:
		
							parent
							
								
									c768681e23
								
							
						
					
					
						commit
						9f96bd2f1a
					
				
					 7 changed files with 102 additions and 1 deletions
				
			
		|  | @ -8,6 +8,8 @@ import RemoteUser from "./live_editor/remote_user"; | ||||||
| import { replacedSuffixLength } from "../../lib/text_utils"; | import { replacedSuffixLength } from "../../lib/text_utils"; | ||||||
| import { settingsStore } from "../../lib/settings"; | import { settingsStore } from "../../lib/settings"; | ||||||
| import Doctest from "./live_editor/doctest"; | import Doctest from "./live_editor/doctest"; | ||||||
|  | import { initVimMode } from "monaco-vim"; | ||||||
|  | import { EmacsExtension, unregisterKey } from "monaco-emacs"; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * Mounts cell source editor with real-time collaboration mechanism. |  * Mounts cell source editor with real-time collaboration mechanism. | ||||||
|  | @ -367,6 +369,18 @@ class LiveEditor { | ||||||
| 
 | 
 | ||||||
|     // Add the widgets that the editor was initialized with
 |     // Add the widgets that the editor was initialized with
 | ||||||
|     this._initializeWidgets(); |     this._initializeWidgets(); | ||||||
|  | 
 | ||||||
|  |     // Set the editor mode
 | ||||||
|  |     if (settings.editor_mode == "emacs") { | ||||||
|  |       this.emacsMode = new EmacsExtension(this.editor); | ||||||
|  |       this.emacsMode.start(); | ||||||
|  |       unregisterKey("Tab"); | ||||||
|  |     } else if (settings.editor_mode == "vim") { | ||||||
|  |       this.vimMode = initVimMode(this.editor); | ||||||
|  |       this.vimMode.on("vim-mode-change", ({ mode: mode }) => { | ||||||
|  |         this.editor.getDomNode().setAttribute("data-vim-mode", mode); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   /** |   /** | ||||||
|  |  | ||||||
|  | @ -26,6 +26,7 @@ const EditorSettings = { | ||||||
|     const editorMarkdownWordWrapCheckbox = this.el.querySelector( |     const editorMarkdownWordWrapCheckbox = this.el.querySelector( | ||||||
|       `[name="editor_markdown_word_wrap"][value="true"]` |       `[name="editor_markdown_word_wrap"][value="true"]` | ||||||
|     ); |     ); | ||||||
|  |     const editorMode = this.el.querySelector(`select[name="editor_mode"]`); | ||||||
| 
 | 
 | ||||||
|     editorAutoCompletionCheckbox.checked = settings.editor_auto_completion; |     editorAutoCompletionCheckbox.checked = settings.editor_auto_completion; | ||||||
|     editorAutoSignatureCheckbox.checked = settings.editor_auto_signature; |     editorAutoSignatureCheckbox.checked = settings.editor_auto_signature; | ||||||
|  | @ -34,6 +35,7 @@ const EditorSettings = { | ||||||
|     editorLightThemeCheckbox.checked = |     editorLightThemeCheckbox.checked = | ||||||
|       settings.editor_theme === EDITOR_THEME.light ? true : false; |       settings.editor_theme === EDITOR_THEME.light ? true : false; | ||||||
|     editorMarkdownWordWrapCheckbox.checked = settings.editor_markdown_word_wrap; |     editorMarkdownWordWrapCheckbox.checked = settings.editor_markdown_word_wrap; | ||||||
|  |     editorMode.value = settings.editor_mode; | ||||||
| 
 | 
 | ||||||
|     editorAutoCompletionCheckbox.addEventListener("change", (event) => { |     editorAutoCompletionCheckbox.addEventListener("change", (event) => { | ||||||
|       settingsStore.update({ editor_auto_completion: event.target.checked }); |       settingsStore.update({ editor_auto_completion: event.target.checked }); | ||||||
|  | @ -62,6 +64,10 @@ const EditorSettings = { | ||||||
|     editorMarkdownWordWrapCheckbox.addEventListener("change", (event) => { |     editorMarkdownWordWrapCheckbox.addEventListener("change", (event) => { | ||||||
|       settingsStore.update({ editor_markdown_word_wrap: event.target.checked }); |       settingsStore.update({ editor_markdown_word_wrap: event.target.checked }); | ||||||
|     }); |     }); | ||||||
|  | 
 | ||||||
|  |     editorMode.addEventListener("change", (event) => { | ||||||
|  |       settingsStore.update({ editor_mode: event.target.value }); | ||||||
|  |     }); | ||||||
|   }, |   }, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -469,6 +469,11 @@ const Session = { | ||||||
|       return true; |       return true; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     // Vim insert mode
 | ||||||
|  |     if (editor.dataset.vimMode == "insert") { | ||||||
|  |       return true; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     return false; |     return false; | ||||||
|   }, |   }, | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -7,6 +7,12 @@ export const EDITOR_FONT_SIZE = { | ||||||
|   large: 16, |   large: 16, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | export const EDITOR_MODE = { | ||||||
|  |   default: "default", | ||||||
|  |   emacs: "emacs", | ||||||
|  |   vim: "vim", | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| export const EDITOR_THEME = { | export const EDITOR_THEME = { | ||||||
|   default: "default", |   default: "default", | ||||||
|   light: "light", |   light: "light", | ||||||
|  | @ -18,6 +24,7 @@ const DEFAULT_SETTINGS = { | ||||||
|   editor_font_size: EDITOR_FONT_SIZE.normal, |   editor_font_size: EDITOR_FONT_SIZE.normal, | ||||||
|   editor_theme: EDITOR_THEME.default, |   editor_theme: EDITOR_THEME.default, | ||||||
|   editor_markdown_word_wrap: true, |   editor_markdown_word_wrap: true, | ||||||
|  |   editor_mode: EDITOR_MODE.default, | ||||||
|   custom_view_show_section: true, |   custom_view_show_section: true, | ||||||
|   custom_view_show_markdown: true, |   custom_view_show_markdown: true, | ||||||
|   custom_view_show_output: true, |   custom_view_show_output: true, | ||||||
|  |  | ||||||
							
								
								
									
										59
									
								
								assets/package-lock.json
									
										
									
										generated
									
									
									
								
							
							
						
						
									
										59
									
								
								assets/package-lock.json
									
										
									
										generated
									
									
									
								
							|  | @ -15,6 +15,8 @@ | ||||||
|         "jest": "^29.1.2", |         "jest": "^29.1.2", | ||||||
|         "mermaid": "^10.0.2", |         "mermaid": "^10.0.2", | ||||||
|         "monaco-editor": "^0.39.0", |         "monaco-editor": "^0.39.0", | ||||||
|  |         "monaco-emacs": "^0.3.0", | ||||||
|  |         "monaco-vim": "^0.4.0", | ||||||
|         "morphdom": "^2.6.1", |         "morphdom": "^2.6.1", | ||||||
|         "phoenix": "file:../deps/phoenix", |         "phoenix": "file:../deps/phoenix", | ||||||
|         "phoenix_html": "file:../deps/phoenix_html", |         "phoenix_html": "file:../deps/phoenix_html", | ||||||
|  | @ -56,7 +58,7 @@ | ||||||
|       "version": "3.3.1" |       "version": "3.3.1" | ||||||
|     }, |     }, | ||||||
|     "../deps/phoenix_live_view": { |     "../deps/phoenix_live_view": { | ||||||
|       "version": "0.19.3", |       "version": "0.19.5", | ||||||
|       "license": "MIT" |       "license": "MIT" | ||||||
|     }, |     }, | ||||||
|     "node_modules/@alloc/quick-lru": { |     "node_modules/@alloc/quick-lru": { | ||||||
|  | @ -7016,6 +7018,16 @@ | ||||||
|       "integrity": "sha512-FT1yDzDYEoYWhnSGnpE/4Kj1fLZkDFyqRb7fNt6FdYOSxlUWAtp42Eh6Wb0rGIv/m9Bgo7x4GhQbm5Ys4SG5ow==", |       "integrity": "sha512-FT1yDzDYEoYWhnSGnpE/4Kj1fLZkDFyqRb7fNt6FdYOSxlUWAtp42Eh6Wb0rGIv/m9Bgo7x4GhQbm5Ys4SG5ow==", | ||||||
|       "dev": true |       "dev": true | ||||||
|     }, |     }, | ||||||
|  |     "node_modules/lodash.kebabcase": { | ||||||
|  |       "version": "4.1.1", | ||||||
|  |       "resolved": "https://registry.npmjs.org/lodash.kebabcase/-/lodash.kebabcase-4.1.1.tgz", | ||||||
|  |       "integrity": "sha512-N8XRTIMMqqDgSy4VLKPnJ/+hpGZN+PHQiJnSenYqPaVV/NCqEogTnAdZLQiGKhxX+JCs8waWq2t1XHWKOmlY8g==" | ||||||
|  |     }, | ||||||
|  |     "node_modules/lodash.throttle": { | ||||||
|  |       "version": "4.1.1", | ||||||
|  |       "resolved": "https://registry.npmjs.org/lodash.throttle/-/lodash.throttle-4.1.1.tgz", | ||||||
|  |       "integrity": "sha512-wIkUCfVKpVsWo3JSZlc+8MB5it+2AN5W8J7YVMST30UrvcQNZ1Okbj+rbVniijTWE6FGYy4XJq/rHkas8qJMLQ==" | ||||||
|  |     }, | ||||||
|     "node_modules/longest-streak": { |     "node_modules/longest-streak": { | ||||||
|       "version": "3.0.1", |       "version": "3.0.1", | ||||||
|       "resolved": "https://registry.npmjs.org/longest-streak/-/longest-streak-3.0.1.tgz", |       "resolved": "https://registry.npmjs.org/longest-streak/-/longest-streak-3.0.1.tgz", | ||||||
|  | @ -7946,6 +7958,26 @@ | ||||||
|       "resolved": "https://registry.npmjs.org/monaco-editor/-/monaco-editor-0.39.0.tgz", |       "resolved": "https://registry.npmjs.org/monaco-editor/-/monaco-editor-0.39.0.tgz", | ||||||
|       "integrity": "sha512-zhbZ2Nx93tLR8aJmL2zI1mhJpsl87HMebNBM6R8z4pLfs8pj604pIVIVwyF1TivcfNtIPpMXL+nb3DsBmE/x6Q==" |       "integrity": "sha512-zhbZ2Nx93tLR8aJmL2zI1mhJpsl87HMebNBM6R8z4pLfs8pj604pIVIVwyF1TivcfNtIPpMXL+nb3DsBmE/x6Q==" | ||||||
|     }, |     }, | ||||||
|  |     "node_modules/monaco-emacs": { | ||||||
|  |       "version": "0.3.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/monaco-emacs/-/monaco-emacs-0.3.0.tgz", | ||||||
|  |       "integrity": "sha512-T7uyCIqpLBmU2dO9pzQ3KTUt2RNOFtqLVQZA0lF8oJ2nphnomSNl29WpkTCJKJVFMmGKauPVB9fknEMFibEwCA==", | ||||||
|  |       "dependencies": { | ||||||
|  |         "lodash.kebabcase": "^4.1.1", | ||||||
|  |         "lodash.throttle": "^4.1.1" | ||||||
|  |       }, | ||||||
|  |       "peerDependencies": { | ||||||
|  |         "monaco-editor": ">=0.31" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "node_modules/monaco-vim": { | ||||||
|  |       "version": "0.4.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/monaco-vim/-/monaco-vim-0.4.0.tgz", | ||||||
|  |       "integrity": "sha512-+CsW0+Mvx2+eitkXS7OpUXIu57qXlqAL8oVkYhkPCEZ/c6+6gOp/IcG7w+Lb33YiZuTyvJ891+czkeJRPIEwVA==", | ||||||
|  |       "peerDependencies": { | ||||||
|  |         "monaco-editor": "*" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "node_modules/morphdom": { |     "node_modules/morphdom": { | ||||||
|       "version": "2.7.0", |       "version": "2.7.0", | ||||||
|       "resolved": "https://registry.npmjs.org/morphdom/-/morphdom-2.7.0.tgz", |       "resolved": "https://registry.npmjs.org/morphdom/-/morphdom-2.7.0.tgz", | ||||||
|  | @ -14778,6 +14810,16 @@ | ||||||
|       "integrity": "sha512-FT1yDzDYEoYWhnSGnpE/4Kj1fLZkDFyqRb7fNt6FdYOSxlUWAtp42Eh6Wb0rGIv/m9Bgo7x4GhQbm5Ys4SG5ow==", |       "integrity": "sha512-FT1yDzDYEoYWhnSGnpE/4Kj1fLZkDFyqRb7fNt6FdYOSxlUWAtp42Eh6Wb0rGIv/m9Bgo7x4GhQbm5Ys4SG5ow==", | ||||||
|       "dev": true |       "dev": true | ||||||
|     }, |     }, | ||||||
|  |     "lodash.kebabcase": { | ||||||
|  |       "version": "4.1.1", | ||||||
|  |       "resolved": "https://registry.npmjs.org/lodash.kebabcase/-/lodash.kebabcase-4.1.1.tgz", | ||||||
|  |       "integrity": "sha512-N8XRTIMMqqDgSy4VLKPnJ/+hpGZN+PHQiJnSenYqPaVV/NCqEogTnAdZLQiGKhxX+JCs8waWq2t1XHWKOmlY8g==" | ||||||
|  |     }, | ||||||
|  |     "lodash.throttle": { | ||||||
|  |       "version": "4.1.1", | ||||||
|  |       "resolved": "https://registry.npmjs.org/lodash.throttle/-/lodash.throttle-4.1.1.tgz", | ||||||
|  |       "integrity": "sha512-wIkUCfVKpVsWo3JSZlc+8MB5it+2AN5W8J7YVMST30UrvcQNZ1Okbj+rbVniijTWE6FGYy4XJq/rHkas8qJMLQ==" | ||||||
|  |     }, | ||||||
|     "longest-streak": { |     "longest-streak": { | ||||||
|       "version": "3.0.1", |       "version": "3.0.1", | ||||||
|       "resolved": "https://registry.npmjs.org/longest-streak/-/longest-streak-3.0.1.tgz", |       "resolved": "https://registry.npmjs.org/longest-streak/-/longest-streak-3.0.1.tgz", | ||||||
|  | @ -15374,6 +15416,21 @@ | ||||||
|       "resolved": "https://registry.npmjs.org/monaco-editor/-/monaco-editor-0.39.0.tgz", |       "resolved": "https://registry.npmjs.org/monaco-editor/-/monaco-editor-0.39.0.tgz", | ||||||
|       "integrity": "sha512-zhbZ2Nx93tLR8aJmL2zI1mhJpsl87HMebNBM6R8z4pLfs8pj604pIVIVwyF1TivcfNtIPpMXL+nb3DsBmE/x6Q==" |       "integrity": "sha512-zhbZ2Nx93tLR8aJmL2zI1mhJpsl87HMebNBM6R8z4pLfs8pj604pIVIVwyF1TivcfNtIPpMXL+nb3DsBmE/x6Q==" | ||||||
|     }, |     }, | ||||||
|  |     "monaco-emacs": { | ||||||
|  |       "version": "0.3.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/monaco-emacs/-/monaco-emacs-0.3.0.tgz", | ||||||
|  |       "integrity": "sha512-T7uyCIqpLBmU2dO9pzQ3KTUt2RNOFtqLVQZA0lF8oJ2nphnomSNl29WpkTCJKJVFMmGKauPVB9fknEMFibEwCA==", | ||||||
|  |       "requires": { | ||||||
|  |         "lodash.kebabcase": "^4.1.1", | ||||||
|  |         "lodash.throttle": "^4.1.1" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "monaco-vim": { | ||||||
|  |       "version": "0.4.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/monaco-vim/-/monaco-vim-0.4.0.tgz", | ||||||
|  |       "integrity": "sha512-+CsW0+Mvx2+eitkXS7OpUXIu57qXlqAL8oVkYhkPCEZ/c6+6gOp/IcG7w+Lb33YiZuTyvJ891+czkeJRPIEwVA==", | ||||||
|  |       "requires": {} | ||||||
|  |     }, | ||||||
|     "morphdom": { |     "morphdom": { | ||||||
|       "version": "2.7.0", |       "version": "2.7.0", | ||||||
|       "resolved": "https://registry.npmjs.org/morphdom/-/morphdom-2.7.0.tgz", |       "resolved": "https://registry.npmjs.org/morphdom/-/morphdom-2.7.0.tgz", | ||||||
|  |  | ||||||
|  | @ -19,6 +19,8 @@ | ||||||
|     "jest": "^29.1.2", |     "jest": "^29.1.2", | ||||||
|     "mermaid": "^10.0.2", |     "mermaid": "^10.0.2", | ||||||
|     "monaco-editor": "^0.39.0", |     "monaco-editor": "^0.39.0", | ||||||
|  |     "monaco-emacs": "^0.3.0", | ||||||
|  |     "monaco-vim": "^0.4.0", | ||||||
|     "morphdom": "^2.6.1", |     "morphdom": "^2.6.1", | ||||||
|     "phoenix": "file:../deps/phoenix", |     "phoenix": "file:../deps/phoenix", | ||||||
|     "phoenix_html": "file:../deps/phoenix_html", |     "phoenix_html": "file:../deps/phoenix_html", | ||||||
|  |  | ||||||
|  | @ -168,6 +168,16 @@ defmodule LivebookWeb.SettingsLive do | ||||||
|                 label="Wrap words in Markdown" |                 label="Wrap words in Markdown" | ||||||
|                 value={false} |                 value={false} | ||||||
|               /> |               /> | ||||||
|  |               <.select_field | ||||||
|  |                 name="editor_mode" | ||||||
|  |                 label="Key bindings" | ||||||
|  |                 value="normal" | ||||||
|  |                 options={[ | ||||||
|  |                   {"Default", "default"}, | ||||||
|  |                   {"Emacs", "emacs"}, | ||||||
|  |                   {"Vim", "vim"} | ||||||
|  |                 ]} | ||||||
|  |               /> | ||||||
|             </div> |             </div> | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		
		Reference in a new issue