| 
									
										
										
										
											2020-05-14 18:06:01 +02:00
										 |  |  | import {basename, extname, isObject, isDarkTheme} from '../utils.js'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const languagesByFilename = {}; | 
					
						
							|  |  |  | const languagesByExt = {}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-08 11:53:00 +02:00
										 |  |  | const baseOptions = { | 
					
						
							|  |  |  |   fontFamily: 'var(--fonts-monospace)', | 
					
						
							|  |  |  |   fontSize: 14, // https://github.com/microsoft/monaco-editor/issues/2242
 | 
					
						
							|  |  |  |   links: false, | 
					
						
							|  |  |  |   minimap: {enabled: false}, | 
					
						
							|  |  |  |   occurrencesHighlight: false, | 
					
						
							|  |  |  |   overviewRulerLanes: 0, | 
					
						
							|  |  |  |   renderIndentGuides: false, | 
					
						
							|  |  |  |   renderLineHighlight: 'all', | 
					
						
							|  |  |  |   renderLineHighlightOnlyWhenFocus: true, | 
					
						
							|  |  |  |   renderWhitespace: 'none', | 
					
						
							|  |  |  |   rulers: false, | 
					
						
							|  |  |  |   scrollbar: {horizontalScrollbarSize: 6, verticalScrollbarSize: 6}, | 
					
						
							|  |  |  |   scrollBeyondLastLine: false, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-14 18:06:01 +02:00
										 |  |  | function getEditorconfig(input) { | 
					
						
							|  |  |  |   try { | 
					
						
							|  |  |  |     return JSON.parse(input.dataset.editorconfig); | 
					
						
							| 
									
										
										
										
											2020-06-09 23:31:15 +02:00
										 |  |  |   } catch { | 
					
						
							| 
									
										
										
										
											2020-05-14 18:06:01 +02:00
										 |  |  |     return null; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function initLanguages(monaco) { | 
					
						
							|  |  |  |   for (const {filenames, extensions, id} of monaco.languages.getLanguages()) { | 
					
						
							|  |  |  |     for (const filename of filenames || []) { | 
					
						
							|  |  |  |       languagesByFilename[filename] = id; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     for (const extension of extensions || []) { | 
					
						
							|  |  |  |       languagesByExt[extension] = id; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function getLanguage(filename) { | 
					
						
							|  |  |  |   return languagesByFilename[filename] || languagesByExt[extname(filename)] || 'plaintext'; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-14 04:57:34 +01:00
										 |  |  | function updateEditor(monaco, editor, filename, lineWrapExts) { | 
					
						
							| 
									
										
										
										
											2021-04-08 11:53:00 +02:00
										 |  |  |   editor.updateOptions(getFileBasedOptions(filename, lineWrapExts)); | 
					
						
							| 
									
										
										
										
											2020-05-14 18:06:01 +02:00
										 |  |  |   const model = editor.getModel(); | 
					
						
							|  |  |  |   const language = model.getModeId(); | 
					
						
							| 
									
										
										
										
											2020-11-14 04:57:34 +01:00
										 |  |  |   const newLanguage = getLanguage(filename); | 
					
						
							| 
									
										
										
										
											2020-05-14 18:06:01 +02:00
										 |  |  |   if (language !== newLanguage) monaco.editor.setModelLanguage(model, newLanguage); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-03 13:19:32 +02:00
										 |  |  | // export editor for customization - https://github.com/go-gitea/gitea/issues/10409
 | 
					
						
							|  |  |  | function exportEditor(editor) { | 
					
						
							|  |  |  |   if (!window.codeEditors) window.codeEditors = []; | 
					
						
							|  |  |  |   if (!window.codeEditors.includes(editor)) window.codeEditors.push(editor); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-14 04:57:34 +01:00
										 |  |  | export async function createMonaco(textarea, filename, editorOpts) { | 
					
						
							| 
									
										
										
										
											2020-05-14 18:06:01 +02:00
										 |  |  |   const monaco = await import(/* webpackChunkName: "monaco" */'monaco-editor'); | 
					
						
							| 
									
										
										
										
											2020-11-14 04:57:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-14 18:06:01 +02:00
										 |  |  |   initLanguages(monaco); | 
					
						
							| 
									
										
										
										
											2020-11-14 04:57:34 +01:00
										 |  |  |   let {language, ...other} = editorOpts; | 
					
						
							|  |  |  |   if (!language) language = getLanguage(filename); | 
					
						
							| 
									
										
										
										
											2020-05-14 18:06:01 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const container = document.createElement('div'); | 
					
						
							|  |  |  |   container.className = 'monaco-editor-container'; | 
					
						
							|  |  |  |   textarea.parentNode.appendChild(container); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-08 11:53:00 +02:00
										 |  |  |   // https://github.com/microsoft/monaco-editor/issues/2427
 | 
					
						
							|  |  |  |   const styles = window.getComputedStyle(document.documentElement); | 
					
						
							|  |  |  |   const getProp = (name) => styles.getPropertyValue(name).trim(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   monaco.editor.defineTheme('gitea', { | 
					
						
							|  |  |  |     base: isDarkTheme() ? 'vs-dark' : 'vs', | 
					
						
							|  |  |  |     inherit: true, | 
					
						
							|  |  |  |     rules: [ | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         background: getProp('--color-code-bg'), | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     ], | 
					
						
							|  |  |  |     colors: { | 
					
						
							|  |  |  |       'editor.background': getProp('--color-code-bg'), | 
					
						
							|  |  |  |       'editor.foreground': getProp('--color-text'), | 
					
						
							|  |  |  |       'editor.inactiveSelectionBackground': getProp('--color-primary-light-4'), | 
					
						
							|  |  |  |       'editor.lineHighlightBackground': getProp('--color-editor-line-highlight'), | 
					
						
							|  |  |  |       'editor.selectionBackground': getProp('--color-primary-light-3'), | 
					
						
							|  |  |  |       'editor.selectionForeground': getProp('--color-primary-light-3'), | 
					
						
							|  |  |  |       'editorLineNumber.background': getProp('--color-code-bg'), | 
					
						
							|  |  |  |       'editorLineNumber.foreground': getProp('--color-secondary-dark-6'), | 
					
						
							|  |  |  |       'editorWidget.background': getProp('--color-body'), | 
					
						
							|  |  |  |       'editorWidget.border': getProp('--color-secondary'), | 
					
						
							|  |  |  |       'input.background': getProp('--color-input-background'), | 
					
						
							|  |  |  |       'input.border': getProp('--color-input-border'), | 
					
						
							|  |  |  |       'input.foreground': getProp('--color-input-text'), | 
					
						
							|  |  |  |       'scrollbar.shadow': getProp('--color-shadow'), | 
					
						
							|  |  |  |       'progressBar.background': getProp('--color-primary'), | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-14 18:06:01 +02:00
										 |  |  |   const editor = monaco.editor.create(container, { | 
					
						
							|  |  |  |     value: textarea.value, | 
					
						
							| 
									
										
										
										
											2021-04-08 11:53:00 +02:00
										 |  |  |     theme: 'gitea', | 
					
						
							| 
									
										
										
										
											2020-11-14 04:57:34 +01:00
										 |  |  |     language, | 
					
						
							|  |  |  |     ...other, | 
					
						
							| 
									
										
										
										
											2020-05-14 18:06:01 +02:00
										 |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const model = editor.getModel(); | 
					
						
							|  |  |  |   model.onDidChangeContent(() => { | 
					
						
							|  |  |  |     textarea.value = editor.getValue(); | 
					
						
							|  |  |  |     textarea.dispatchEvent(new Event('change')); // seems to be needed for jquery-are-you-sure
 | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   window.addEventListener('resize', () => { | 
					
						
							|  |  |  |     editor.layout(); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-14 04:57:34 +01:00
										 |  |  |   exportEditor(editor); | 
					
						
							| 
									
										
										
										
											2020-05-14 18:06:01 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const loading = document.querySelector('.editor-loading'); | 
					
						
							|  |  |  |   if (loading) loading.remove(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-14 04:57:34 +01:00
										 |  |  |   return {monaco, editor}; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-06-03 13:19:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-14 04:57:34 +01:00
										 |  |  | function getFileBasedOptions(filename, lineWrapExts) { | 
					
						
							|  |  |  |   return { | 
					
						
							|  |  |  |     wordWrap: (lineWrapExts || []).includes(extname(filename)) ? 'on' : 'off', | 
					
						
							|  |  |  |   }; | 
					
						
							| 
									
										
										
										
											2020-05-14 18:06:01 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-14 04:57:34 +01:00
										 |  |  | export async function createCodeEditor(textarea, filenameInput, previewFileModes) { | 
					
						
							|  |  |  |   const filename = basename(filenameInput.value); | 
					
						
							|  |  |  |   const previewLink = document.querySelector('a[data-tab=preview]'); | 
					
						
							|  |  |  |   const markdownExts = (textarea.dataset.markdownFileExts || '').split(','); | 
					
						
							|  |  |  |   const lineWrapExts = (textarea.dataset.lineWrapExtensions || '').split(','); | 
					
						
							|  |  |  |   const isMarkdown = markdownExts.includes(extname(filename)); | 
					
						
							|  |  |  |   const editorConfig = getEditorconfig(filenameInput); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (previewLink) { | 
					
						
							|  |  |  |     if (isMarkdown && (previewFileModes || []).includes('markdown')) { | 
					
						
							|  |  |  |       previewLink.dataset.url = previewLink.dataset.url.replace(/(.*)\/.*/i, `$1/markdown`); | 
					
						
							|  |  |  |       previewLink.style.display = ''; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       previewLink.style.display = 'none'; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-05-14 18:06:01 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-14 04:57:34 +01:00
										 |  |  |   const {monaco, editor} = await createMonaco(textarea, filename, { | 
					
						
							| 
									
										
										
										
											2021-04-08 11:53:00 +02:00
										 |  |  |     ...baseOptions, | 
					
						
							| 
									
										
										
										
											2020-11-14 04:57:34 +01:00
										 |  |  |     ...getFileBasedOptions(filenameInput.value, lineWrapExts), | 
					
						
							|  |  |  |     ...getEditorConfigOptions(editorConfig), | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   filenameInput.addEventListener('keyup', () => { | 
					
						
							|  |  |  |     const filename = filenameInput.value; | 
					
						
							|  |  |  |     updateEditor(monaco, editor, filename, lineWrapExts); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return editor; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function getEditorConfigOptions(ec) { | 
					
						
							|  |  |  |   if (!isObject(ec)) return {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const opts = {}; | 
					
						
							|  |  |  |   opts.detectIndentation = !('indent_style' in ec) || !('indent_size' in ec); | 
					
						
							|  |  |  |   if ('indent_size' in ec) opts.indentSize = Number(ec.indent_size); | 
					
						
							|  |  |  |   if ('tab_width' in ec) opts.tabSize = Number(ec.tab_width) || opts.indentSize; | 
					
						
							|  |  |  |   if ('max_line_length' in ec) opts.rulers = [Number(ec.max_line_length)]; | 
					
						
							|  |  |  |   opts.trimAutoWhitespace = ec.trim_trailing_whitespace === true; | 
					
						
							|  |  |  |   opts.insertSpaces = ec.indent_style === 'space'; | 
					
						
							|  |  |  |   opts.useTabStops = ec.indent_style === 'tab'; | 
					
						
							| 
									
										
										
										
											2020-05-14 18:06:01 +02:00
										 |  |  |   return opts; | 
					
						
							|  |  |  | } |