// Wrapper for CodeMirror editor // https://github.com/codemirror/codemirror import { ScriptEditor } from "./ScriptEditor"; import 'codemirror/lib/codemirror.css'; import 'codemirror/theme/monokai.css'; import 'codemirror/theme/3024-day.css'; import 'codemirror/theme/3024-night.css'; import 'codemirror/theme/abcdef.css'; import 'codemirror/theme/ambiance-mobile.css'; import 'codemirror/theme/ambiance.css'; import 'codemirror/theme/base16-dark.css'; import 'codemirror/theme/base16-light.css'; import 'codemirror/theme/bespin.css'; import 'codemirror/theme/blackboard.css'; import 'codemirror/theme/cobalt.css'; import 'codemirror/theme/colorforth.css'; import 'codemirror/theme/darcula.css'; import 'codemirror/theme/dracula.css'; import 'codemirror/theme/duotone-dark.css'; import 'codemirror/theme/duotone-light.css'; import 'codemirror/theme/eclipse.css'; import 'codemirror/theme/elegant.css'; import 'codemirror/theme/erlang-dark.css'; import 'codemirror/theme/gruvbox-dark.css'; import 'codemirror/theme/hopscotch.css'; import 'codemirror/theme/icecoder.css'; import 'codemirror/theme/idea.css'; import 'codemirror/theme/isotope.css'; import 'codemirror/theme/lesser-dark.css'; import 'codemirror/theme/liquibyte.css'; import 'codemirror/theme/lucario.css'; import 'codemirror/theme/material.css'; import 'codemirror/theme/mbo.css'; import 'codemirror/theme/mdn-like.css'; import 'codemirror/theme/midnight.css'; import 'codemirror/theme/neat.css'; import 'codemirror/theme/neo.css'; import 'codemirror/theme/night.css'; import 'codemirror/theme/oceanic-next.css'; import 'codemirror/theme/panda-syntax.css'; import 'codemirror/theme/paraiso-dark.css'; import 'codemirror/theme/paraiso-light.css'; import 'codemirror/theme/pastel-on-dark.css'; import 'codemirror/theme/railscasts.css'; import 'codemirror/theme/rubyblue.css'; import 'codemirror/theme/seti.css'; import 'codemirror/theme/shadowfox.css'; import 'codemirror/theme/solarized.css'; import 'codemirror/theme/ssms.css'; import 'codemirror/theme/the-matrix.css'; import 'codemirror/theme/tomorrow-night-bright.css'; import 'codemirror/theme/tomorrow-night-eighties.css'; import 'codemirror/theme/ttcn.css'; import 'codemirror/theme/twilight.css'; import 'codemirror/theme/vibrant-ink.css'; import 'codemirror/theme/xq-dark.css'; import 'codemirror/theme/xq-light.css'; import 'codemirror/theme/yeti.css'; import 'codemirror/theme/zenburn.css'; import "../../css/codemirror-overrides.scss"; import CodeMirror from "codemirror/lib/codemirror.js"; import "codemirror/mode/javascript/javascript.js"; import "./CodeMirrorNetscriptMode"; import 'codemirror/keymap/sublime.js'; import 'codemirror/keymap/vim.js'; import 'codemirror/keymap/emacs.js'; import 'codemirror/addon/comment/continuecomment.js'; import 'codemirror/addon/edit/closebrackets.js'; import 'codemirror/addon/edit/matchbrackets.js'; import 'codemirror/addon/fold/foldcode.js'; import 'codemirror/addon/fold/foldgutter.js'; import 'codemirror/addon/fold/foldgutter.css'; import 'codemirror/addon/fold/brace-fold.js'; import 'codemirror/addon/fold/indent-fold.js'; import 'codemirror/addon/fold/comment-fold.js'; import 'codemirror/addon/hint/show-hint.js'; import 'codemirror/addon/hint/show-hint.css'; import 'codemirror/addon/lint/lint.js'; import 'codemirror/addon/lint/lint.css'; import 'codemirror/addon/search/match-highlighter.js'; import 'codemirror/addon/selection/active-line.js'; window.JSHINT = require('jshint').JSHINT; import './CodeMirrorNetscriptHint.js'; import { NetscriptFunctions } from "../NetscriptFunctions"; import { CodeMirrorThemeSetting } from "../Settings/SettingEnums"; import { Settings } from "../Settings/Settings"; import { clearEventListeners } from "../../utils/uiHelpers/clearEventListeners"; import { createElement } from "../../utils/uiHelpers/createElement"; import { createOptionElement } from "../../utils/uiHelpers/createOptionElement"; import { getSelectText, getSelectValue } from "../../utils/uiHelpers/getSelectData"; import { removeChildrenFromElement } from "../../utils/uiHelpers/removeChildrenFromElement"; // Max number of invisibles to be shown in a group if the "Show Invisibles" option // is marked const MaxInvisibles = 20; function validateInitializationParamters(params) { if (params.saveAndCloseFn == null) { return false; } // Save & close button function if (params.quitFn == null) { return false; } // Quitting editor, aka Engine.loadTerminalContent return true; } class CodeMirrorEditorWrapper extends ScriptEditor { constructor() { super(); this.tabsStyleElement = null; } init(params) { if (this.editor != null) { console.error(`CodeMirrorEditor.init() called when it's already initialized`); return false; } // Validate/Sanitize input if (!validateInitializationParamters(params)) { console.error(`'params' argument passed into CodeMirrorEditor.init() does not have proper properties`); return false; } // Store the filename input this.filenameInput = document.getElementById("script-editor-filename"); if (this.filenameInput == null) { console.error(`Could not get Script Editor filename element (id=script-editor-filename)`); return false; } // Add styling for the "Show Invisibles" option for spaces const classBase = '.CodeMirror .cm-whitespace-'; const spaceChar = 'ยท'; const style = document.createElement('style'); style.setAttribute('data-name', 'js-show-invisibles'); let rules = ''; let spaceChars = ''; for (let i = 1; i <= MaxInvisibles; ++i) { spaceChars += spaceChar; const rule = classBase + i + '::before { content: "' + spaceChars + '";}\n'; rules += rule; } style.textContent = rules; document.head.appendChild(style); // Add an element for the "Show Invisible" option for tabs this.tabsStyleElement = document.createElement('style'); document.head.appendChild(this.tabsStyleElement); // Define a "Save" command for CodeMirror so shortcuts like Ctrl + s // will save in-game CodeMirror.commands.save = function() { params.saveAndCloseFn(); } // Add Netscript Functions to the autocompleter const netscriptFns = []; var fnsObj = NetscriptFunctions(null); for (let name in fnsObj) { if (fnsObj.hasOwnProperty(name)) { netscriptFns.push(name); //Get functions from namespaces const namespaces = ["bladeburner", "hacknet", "codingcontract", "gang"]; if (namespaces.includes(name)) { let namespace = fnsObj[name]; if (typeof namespace !== "object") {continue;} let namespaceFns = Object.keys(namespace); for (let i = 0; i < namespaceFns.length; ++i) { netscriptFns.push(namespaceFns[i]); } } } } CodeMirror.hint.netscript = function(editor) { const origList = CodeMirror.hint.javascript(editor) || {from: editor.getCursor(), to: editor.getCursor(), list: []}; origList.list.push(...netscriptFns); let list = origList.list || []; let cursor = editor.getCursor(); let currentLine = editor.getLine(cursor.line); let start = cursor.ch; let end = start; while (end < currentLine.length && /[\w$]+/.test(currentLine.charAt(end))) ++end; while (start && /[\w$]+/.test(currentLine.charAt(start - 1))) --start; let curWord = start != end && currentLine.slice(start, end); let regex = new RegExp('^' + curWord, 'i'); let result = { list: (!curWord ? list : list.filter(function (item) { return item.match(regex); })).sort(), from: CodeMirror.Pos(cursor.line, start), to: CodeMirror.Pos(cursor.line, end) }; return result; }; } initialized() { return (this.filenameInput != null); } create() { function safeGetElementById(id, whatFor="") { const elem = document.getElementById(id); if (elem == null) { throw new Error(`Could not find ${whatFor} DOM element(id=${id})`); } return elem; } function safeClearEventListeners(id, whatFor="") { const elem = clearEventListeners(id); if (elem == null) { throw new Error(`Could not find ${whatFor} DOM element(id=${id})`); } return elem; } try { if (!this.initialized()) { console.warn(`CodeMirrorEditor.create() called when editor was not initialized`); return; } // Initialize CodeMirror Editor const textAreaElement = safeGetElementById("codemirror-editor", "CodeMirror Textarea"); const formElement = safeGetElementById("codemirror-form-wrapper", "CodeMirror Form Wrapper"); formElement.style.display = "block"; this.editor = CodeMirror.fromTextArea(textAreaElement, { autofocus: true, extraKeys: { "Ctrl-Space": "autocomplete" }, foldGutter: true, gutters: ["CodeMirror-lint-markers", "CodeMirror-linenumbers", "CodeMirror-foldgutter"], highlightSelectionMatches: true, hintOptions: { hint: CodeMirror.hint.netscript }, indentUnit: 4, keyMap: "default", lineNumbers: true, matchBrackets: true, maxInvisibles: 32, mode: "netscript", theme: Settings.EditorTheme, }); // Setup Theme Option const themeDropdown = safeClearEventListeners("script-editor-option-theme", "Theme Selector"); removeChildrenFromElement(themeDropdown); const themeOptions = Object.keys(CodeMirrorThemeSetting); for (let i = 0; i < themeOptions.length; ++i) { const themeKey = themeOptions[i]; const themeValue = CodeMirrorThemeSetting[themeKey]; themeDropdown.add(createOptionElement(themeKey, themeValue)); } if (Settings.EditorTheme) { var initialIndex = 0; for (var i = 0; i < themeDropdown.options.length; ++i) { if (themeDropdown.options[i].value === Settings.EditorTheme) { initialIndex = i; break; } } themeDropdown.selectedIndex = initialIndex; } else { themeDropdown.selectedIndex = 0; } themeDropdown.onchange = () => { const val = themeDropdown.value; Settings.EditorTheme = val; this.editor.setOption("theme", val); }; themeDropdown.onchange(); // Setup Keymap Option const keybindingDropdown = safeClearEventListeners("script-editor-option-keybinding", "Keymap Selector"); if (keybindingDropdown == null) { console.error(`Could not find Script Editor's keybinding selector element (id="script-editor-option-keybinding")`); return false; } removeChildrenFromElement(keybindingDropdown); keybindingDropdown.add(createOptionElement("Default", "default")); keybindingDropdown.add(createOptionElement("Sublime", "sublime")); keybindingDropdown.add(createOptionElement("Vim", "vim")); keybindingDropdown.add(createOptionElement("Emacs", "emacs")); if (Settings.EditorKeybinding) { var initialIndex = 0; for (var i = 0; i < keybindingDropdown.options.length; ++i) { if (keybindingDropdown.options[i].value === Settings.EditorKeybinding) { initialIndex = i; break; } } keybindingDropdown.selectedIndex = initialIndex; } else { keybindingDropdown.selectedIndex = 0; } keybindingDropdown.onchange = () => { const val = keybindingDropdown.value; Settings.EditorKeybinding = val; this.editor.removeKeyMap("sublime"); this.editor.removeKeyMap("emacs"); this.editor.removeKeyMap("vim"); this.editor.addKeyMap(val); this.editor.setOption("keyMap", val); console.log(`Set keymap to ${val} for CodeMirror`); }; keybindingDropdown.onchange(); // Highlight Active line const highlightActiveChkBox = safeClearEventListeners("script-editor-option-highlightactiveline", "Active Line Checkbox"); highlightActiveChkBox.onchange = () => { this.editor.setOption("styleActiveLine", highlightActiveChkBox.checked); }; highlightActiveChkBox.onchange(); // Show Invisibles const showInvisiblesChkBox = safeClearEventListeners("script-editor-option-showinvisibles", "Show Invisible Checkbox"); showInvisiblesChkBox.onchange = () => { const overlayMode = { name: 'invisibles', token: function nextToken(stream) { var ret, spaces = 0, space = stream.peek() === ' '; if (space) { while (space && spaces < MaxInvisibles) { ++spaces; stream.next(); space = stream.peek() === ' '; } ret = 'whitespace whitespace-' + spaces; } else { while (!stream.eol() && !space) { stream.next(); space = stream.peek() === ' '; } ret = 'cm-eol'; } return ret; } }; if (showInvisiblesChkBox.checked) { // Spaces this.editor.addOverlay(overlayMode); // Tabs this.tabsStyleElement.innerHTML = ".cm-tab {background: url();background-position: right;background-repeat: no-repeat;}"; } else { this.editor.removeOverlay("invisibles"); this.tabsStyleElement.innerHTML = ""; } }; showInvisiblesChkBox.onchange(); //Use Soft Tab const softTabChkBox = safeClearEventListeners("script-editor-option-usesofttab", "Soft Tab Checkbox"); softTabChkBox.onchange = () => { this.editor.setOption("indentWithTabs", !softTabChkBox.checked); if (softTabChkBox.checked) { this.editor.addKeyMap({ name: "soft-tabs-keymap", "Tab": function (cm) { if (cm.somethingSelected()) { var sel = cm.getSelection("\n"); // Indent only if there are multiple lines selected, or if the selection spans a full line if (sel.length > 0 && (sel.indexOf("\n") > -1 || sel.length === cm.getLine(cm.getCursor().line).length)) { cm.indentSelection("add"); return; } } if (cm.options.indentWithTabs) cm.execCommand("insertTab"); else cm.execCommand("insertSoftTab"); }, "Shift-Tab": function (cm) { cm.indentSelection("subtract"); } }); } else { this.editor.removeKeyMap("soft-tabs-keymap"); } }; softTabChkBox.onchange(); // Some helper functions for dealing with flexible options function resetFlexibleOption(id) { const fieldset = safeGetElementById(id); removeChildrenFromElement(fieldset); fieldset.style.display = "block"; return fieldset; } function removeFlexibleOption(id) { // This doesn't really remove it, just sets it to invisible const fieldset = resetFlexibleOption(id); fieldset.style.display = "none"; return fieldset; } // Flex 1: Automatically Close Brackets and Quotes const flex1Fieldset = resetFlexibleOption("script-editor-option-flex1-fieldset"); const flex1Id = "script-editor-option-flex1"; flex1Fieldset.appendChild(createElement("label", { for: flex1Id, innerText: "Auto-Close Brackets/Quotes", })); const flex1Checkbox = createElement("input", { checked: true, id: flex1Id, name: flex1Id, type: "checkbox", }); flex1Fieldset.appendChild(flex1Checkbox); flex1Checkbox.onchange = () => { this.editor.setOption("autoCloseBrackets", flex1Checkbox.checked); }; flex1Checkbox.onchange(); // Flex 2: Disable/Enable Linting const flex2Fieldset = resetFlexibleOption("script-editor-option-flex2-fieldset"); const flex2Id = "script-editor-option-flex2"; flex2Fieldset.appendChild(createElement("label", { for: flex2Id, innerText: "Enable Linting", })); const flex2Checkbox = createElement("input", { checked: true, id: flex2Id, name: flex2Id, type: "checkbox", }); flex2Fieldset.appendChild(flex2Checkbox); flex2Checkbox.onchange = () => { if (flex2Checkbox.checked) { this.editor.setOption("lint", CodeMirror.lint.netscript); } else { this.editor.setOption("lint", false); } } flex2Checkbox.onchange(); // Flex 3: Continue Comments const flex3Fieldset = resetFlexibleOption("script-editor-option-flex3-fieldset"); const flex3Id = "script-editor-option-flex3"; flex3Fieldset.appendChild(createElement("label", { for: flex3Id, innerText: "Continue Comments", })); const flex3Checkbox = createElement("input", { checked: true, id: flex3Id, name: flex3Id, type: "checkbox", }); flex3Fieldset.appendChild(flex3Checkbox); flex3Checkbox.onchange = () => { this.editor.setOption("continueComments", flex3Checkbox.checked); } flex3Checkbox.onchange(); removeFlexibleOption("script-editor-option-flex4-fieldset"); this.editor.refresh(); console.log(this.editor.options); } catch(e) { console.error(`Exception caught: ${e}`); return false; } } isFocused() { if (this.editor == null) { return false; } return this.editor.hasFocus(); } // Sets the editor to be invisible setInvisible() { if (!this.initialized()) { console.warn(`CodeMirrorEditor.setInvisible() called when editor was not initialized`); return; } if (this.editor != null) { this.editor.toTextArea(); this.editor = null; } const elem = document.getElementById("codemirror-form-wrapper"); if (elem instanceof HTMLElement) { elem.style.display = "none"; } } } export const CodeMirrorEditor = new CodeMirrorEditorWrapper();