UNPKG

@atlaskit/editor-plugin-code-block-advanced

Version:

CodeBlockAdvanced plugin for @atlaskit/editor-core

478 lines (469 loc) 24.3 kB
import _toConsumableArray from "@babel/runtime/helpers/toConsumableArray"; import _classCallCheck from "@babel/runtime/helpers/classCallCheck"; import _createClass from "@babel/runtime/helpers/createClass"; import _defineProperty from "@babel/runtime/helpers/defineProperty"; function _createForOfIteratorHelper(r, e) { var t = "undefined" != typeof Symbol && r[Symbol.iterator] || r["@@iterator"]; if (!t) { if (Array.isArray(r) || (t = _unsupportedIterableToArray(r)) || e && r && "number" == typeof r.length) { t && (r = t); var _n = 0, F = function F() {}; return { s: F, n: function n() { return _n >= r.length ? { done: !0 } : { done: !1, value: r[_n++] }; }, e: function e(r) { throw r; }, f: F }; } throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } var o, a = !0, u = !1; return { s: function s() { t = t.call(r); }, n: function n() { var r = t.next(); return a = r.done, r; }, e: function e(r) { u = !0, o = r; }, f: function f() { try { a || null == t.return || t.return(); } finally { if (u) throw o; } } }; } function _unsupportedIterableToArray(r, a) { if (r) { if ("string" == typeof r) return _arrayLikeToArray(r, a); var t = {}.toString.call(r).slice(8, -1); return "Object" === t && r.constructor && (t = r.constructor.name), "Map" === t || "Set" === t ? Array.from(r) : "Arguments" === t || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(t) ? _arrayLikeToArray(r, a) : void 0; } } function _arrayLikeToArray(r, a) { (null == a || a > r.length) && (a = r.length); for (var e = 0, n = Array(a); e < a; e++) n[e] = r[e]; return n; } function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; } function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { _defineProperty(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; } import { closeBrackets } from '@codemirror/autocomplete'; import { syntaxHighlighting, bracketMatching } from '@codemirror/language'; import { Compartment, Facet, EditorState as CodeMirrorState } from '@codemirror/state'; import { EditorView as CodeMirror, lineNumbers, gutters } from '@codemirror/view'; import { getBrowserInfo } from '@atlaskit/editor-common/browser'; import { areCodeBlockLineNumbersHidden, isCodeBlockWordWrapEnabled } from '@atlaskit/editor-common/code-block'; import { messages as floatingToolbarMessages } from '@atlaskit/editor-common/floating-toolbar'; import { blockTypeMessages, codeBlockMessages, roleDescriptionMessages } from '@atlaskit/editor-common/messages'; import { ZERO_WIDTH_SPACE } from '@atlaskit/editor-common/whitespace'; import { NodeSelection } from '@atlaskit/editor-prosemirror/state'; import { DecorationSet } from '@atlaskit/editor-prosemirror/view'; import { fg } from '@atlaskit/platform-feature-flags'; import { expValEquals } from '@atlaskit/tmp-editor-statsig/exp-val-equals'; import { expValEqualsNoExposure } from '@atlaskit/tmp-editor-statsig/exp-val-equals-no-exposure'; import { highlightStyle } from '../ui/syntaxHighlightingTheme'; import { cmTheme, codeFoldingTheme } from '../ui/theme'; import { syncCMWithPM } from './codemirrorSync/syncCMWithPM'; import { getCMSelectionChanges } from './codemirrorSync/updateCMSelection'; import { firstCodeBlockInDocument } from './extensions/firstCodeBlockInDocument'; import { foldGutterExtension, getCodeBlockFoldStateEffects } from './extensions/foldGutter'; import { keymapExtension } from './extensions/keymap'; import { lineSeparatorExtension } from './extensions/lineSeparator'; import { manageSelectionMarker } from './extensions/manageSelectionMarker'; import { prosemirrorDecorationPlugin } from './extensions/prosemirrorDecorations'; import { tripleClickSelectAllExtension } from './extensions/tripleClickExtension'; import getLanguageName from './languages/getLanguageName'; import { LanguageLoader } from './languages/loader'; // Store last observed heights of code blocks var codeBlockHeights = new WeakMap(); // Based on: https://prosemirror.net/examples/codemirror/ var CodeBlockAdvancedNodeView = /*#__PURE__*/function () { function CodeBlockAdvancedNodeView(node, view, getPos, innerDecorations, config) { var _this = this, _config$api, _contentFormatSharedS, _config$api2, _config$api3, _config$api4; _classCallCheck(this, CodeBlockAdvancedNodeView); _defineProperty(this, "lineWrappingCompartment", new Compartment()); _defineProperty(this, "lineNumbersCompartment", new Compartment()); _defineProperty(this, "languageCompartment", new Compartment()); _defineProperty(this, "readOnlyCompartment", new Compartment()); _defineProperty(this, "pmDecorationsCompartment", new Compartment()); _defineProperty(this, "themeCompartment", new Compartment()); _defineProperty(this, "maybeTryingToReachNodeSelection", false); _defineProperty(this, "pmFacet", Facet.define()); _defineProperty(this, "wordWrappingEnabled", false); _defineProperty(this, "lineNumbersHidden", false); _defineProperty(this, "selectCodeBlockNodeAndFocus", function () { _this.selectCodeBlockNode(undefined); _this.view.focus(); }); this.config = config; this.node = node; this.view = view; this.getPos = getPos; var contentFormatSharedState = expValEquals('confluence_compact_text_format', 'isEnabled', true) ? (_config$api = config.api) === null || _config$api === void 0 || (_config$api = _config$api.contentFormat) === null || _config$api === void 0 ? void 0 : _config$api.sharedState : undefined; this.contentMode = expValEquals('confluence_compact_text_format', 'isEnabled', true) ? contentFormatSharedState === null || contentFormatSharedState === void 0 || (_contentFormatSharedS = contentFormatSharedState.currentState) === null || _contentFormatSharedS === void 0 || (_contentFormatSharedS = _contentFormatSharedS.call(contentFormatSharedState)) === null || _contentFormatSharedS === void 0 ? void 0 : _contentFormatSharedS.contentMode : undefined; this.selectionAPI = (_config$api2 = config.api) === null || _config$api2 === void 0 || (_config$api2 = _config$api2.selection) === null || _config$api2 === void 0 ? void 0 : _config$api2.actions; var getNode = function getNode() { return _this.node; }; var onMaybeNodeSelection = function onMaybeNodeSelection() { return _this.maybeTryingToReachNodeSelection = true; }; this.cleanupDisabledState = (_config$api3 = config.api) === null || _config$api3 === void 0 || (_config$api3 = _config$api3.editorDisabled) === null || _config$api3 === void 0 ? void 0 : _config$api3.sharedState.onChange(function () { _this.updateReadonlyState(); }); this.languageLoader = new LanguageLoader(function (lang) { _this.updating = true; _this.cm.dispatch({ effects: _this.languageCompartment.reconfigure(lang) }); _this.updating = false; }); var _config$getIntl = config.getIntl(), formatMessage = _config$getIntl.formatMessage; var formattedAriaLabel = formatMessage(blockTypeMessages.codeblock); var isMacOS = getBrowserInfo().mac; this.cm = new CodeMirror({ doc: this.node.textContent, extensions: [].concat(_toConsumableArray(config.extensions), [this.lineWrappingCompartment.of(isCodeBlockWordWrapEnabled(node) ? CodeMirror.lineWrapping : []), this.languageCompartment.of([]), this.pmDecorationsCompartment.of(this.pmFacet.compute([], function () { return innerDecorations; })), keymapExtension({ view: view, getPos: getPos, getNode: getNode, selectCodeBlockNode: this.selectCodeBlockNode.bind(this), onMaybeNodeSelection: onMaybeNodeSelection, customFindReplace: Boolean((_config$api4 = config.api) === null || _config$api4 === void 0 ? void 0 : _config$api4.findReplace) }), // Goes before cmTheme to override styles config.allowCodeFolding ? [codeFoldingTheme] : [], this.themeCompartment.of(cmTheme({ contentMode: expValEquals('confluence_compact_text_format', 'isEnabled', true) ? this.contentMode : undefined })), syntaxHighlighting(highlightStyle), bracketMatching(), expValEquals('platform_editor_code_block_q4_lovability', 'isEnabled', true) ? this.lineNumbersCompartment.of(this.getLineNumberVisibilityExtensions(node)) : this.getLineNumberExtensions(), // Explicitly disable "sticky" positioning on all gutters to match // Renderer behaviour. gutters({ fixed: false }), CodeMirror.updateListener.of(function (update) { return _this.forwardUpdate(update); }), this.readOnlyCompartment.of([CodeMirrorState.readOnly.of(!this.view.editable), CodeMirror.contentAttributes.of({ contentEditable: "".concat(this.view.editable) })]), closeBrackets(), CodeMirror.editorAttributes.of(_objectSpread({ class: 'code-block' }, fg('platform_editor_adf_with_localid') && { 'data-local-id': this.node.attrs.localId })), manageSelectionMarker(config.api), prosemirrorDecorationPlugin(this.pmFacet, view, getPos), tripleClickSelectAllExtension(), firstCodeBlockInDocument(getPos), CodeMirror.contentAttributes.of(_objectSpread(_objectSpread(_objectSpread({}, !expValEquals('editor_a11y_role_textbox', 'isEnabled', true) && { 'aria-label': "".concat(formattedAriaLabel) }), isMacOS && expValEquals('editor_a11y_role_textbox', 'isEnabled', true) && { role: 'textbox', 'aria-roledescription': formatMessage(roleDescriptionMessages.codeSnippetTextBox), 'aria-describedby': "codesnippet-".concat(this.node.attrs.localId), 'aria-multiline': 'true', 'aria-label': formattedAriaLabel }), !isMacOS && expValEquals('editor_a11y_role_textbox', 'isEnabled', true) && { 'aria-label': formattedAriaLabel, 'aria-describedby': "codesnippet-".concat(this.node.attrs.localId) })), config.allowCodeFolding ? [foldGutterExtension({ selectNode: this.selectCodeBlockNodeAndFocus, getNode: function getNode() { return _this.node; } })] : [], // With platform_editor_fix_advanced_codeblocks_crlf_patch the lineSeparatorExtension is not needed expValEquals('platform_editor_fix_advanced_codeblocks_crlf_patch', 'isEnabled', true) ? [] : [lineSeparatorExtension()]]) }); if (contentFormatSharedState) { this.unsubscribeContentFormat = contentFormatSharedState.onChange(function (_ref) { var nextSharedState = _ref.nextSharedState, prevSharedState = _ref.prevSharedState; var prevMode = prevSharedState === null || prevSharedState === void 0 ? void 0 : prevSharedState.contentMode; var nextMode = nextSharedState === null || nextSharedState === void 0 ? void 0 : nextSharedState.contentMode; if (nextMode === prevMode) { return; } _this.applyContentModeTheme(nextMode); if (_this.updating || _this.cm.hasFocus) { return; } _this.cm.requestMeasure(); }); } // Observe size changes of the CodeMirror DOM and request a measurement pass if (!expValEquals('confluence_compact_text_format', 'isEnabled', true) && expValEquals('cc_editor_ai_content_mode', 'variant', 'test') && fg('platform_editor_content_mode_button_mvp')) { this.ro = new ResizeObserver(function (entries) { // Skip measurements when: // 1. Currently updating (prevents feedback loops) // 2. CodeMirror has focus (user is actively typing/editing) if (_this.updating || _this.cm.hasFocus) { return; } // Only trigger on height changes, not width or other dimension changes var _iterator = _createForOfIteratorHelper(entries), _step; try { for (_iterator.s(); !(_step = _iterator.n()).done;) { var entry = _step.value; var currentHeight = entry.contentRect.height; var lastHeight = codeBlockHeights.get(_this.cm.contentDOM); if (lastHeight !== undefined && lastHeight === currentHeight) { return; } codeBlockHeights.set(_this.cm.contentDOM, currentHeight); } // CodeMirror to re-measure when its content size changes } catch (err) { _iterator.e(err); } finally { _iterator.f(); } _this.cm.requestMeasure(); }); this.ro.observe(this.cm.contentDOM); } // We append an additional element that fixes a selection bug on chrome if the code block // is the first element followed by subsequent code blocks var spaceContainer = document.createElement('span'); spaceContainer.innerText = ZERO_WIDTH_SPACE; spaceContainer.style.height = '0'; // The editor's outer node is our DOM representation this.dom = this.cm.dom; this.dom.appendChild(spaceContainer); if (expValEquals('editor_a11y_role_textbox', 'isEnabled', true) && fg('platform_editor_adf_with_localid')) { this.invisibleAriaDescription = document.createElement('span'); this.invisibleAriaDescription.hidden = true; this.invisibleAriaDescription.id = "codesnippet-".concat(this.node.attrs.localId); this.updateAriaDescription(); this.dom.appendChild(this.invisibleAriaDescription); } // This flag is used to avoid an update loop between the outer and // inner editor this.updating = false; this.updateLanguage(); this.updateLocalIdAttribute(); this.wordWrappingEnabled = isCodeBlockWordWrapEnabled(node); this.lineNumbersHidden = areCodeBlockLineNumbersHidden(node); // Restore fold state after initialization if (config.allowCodeFolding) { this.restoreFoldState(); } } return _createClass(CodeBlockAdvancedNodeView, [{ key: "destroy", value: function destroy() { var _this$cleanupDisabled; // ED-27428: CodeMirror gets into an infinite loop as it detects mutations on removed // decorations. When we change the breakout we destroy the node and cleanup these decorations from // codemirror this.clearProseMirrorDecorations(); (_this$cleanupDisabled = this.cleanupDisabledState) === null || _this$cleanupDisabled === void 0 || _this$cleanupDisabled.call(this); if (expValEquals('confluence_compact_text_format', 'isEnabled', true)) { var _this$unsubscribeCont; (_this$unsubscribeCont = this.unsubscribeContentFormat) === null || _this$unsubscribeCont === void 0 || _this$unsubscribeCont.call(this); } if (!expValEquals('confluence_compact_text_format', 'isEnabled', true) && expValEquals('cc_editor_ai_content_mode', 'variant', 'test') && fg('platform_editor_content_mode_button_mvp')) { var _this$ro; (_this$ro = this.ro) === null || _this$ro === void 0 || _this$ro.disconnect(); } } }, { key: "forwardUpdate", value: function forwardUpdate(update) { var _this$getPos, _this$getPos2; if (this.updating || !this.cm.hasFocus) { return; } var offset = ((_this$getPos = (_this$getPos2 = this.getPos) === null || _this$getPos2 === void 0 ? void 0 : _this$getPos2.call(this)) !== null && _this$getPos !== void 0 ? _this$getPos : 0) + 1; syncCMWithPM({ view: this.view, update: update, offset: offset }); } }, { key: "setSelection", value: function setSelection(anchor, head) { if (!this.maybeTryingToReachNodeSelection) { this.cm.focus(); } this.updating = true; this.cm.dispatch({ selection: { anchor: anchor, head: head } }); this.updating = false; } }, { key: "updateReadonlyState", value: function updateReadonlyState() { this.updating = true; this.cm.dispatch({ effects: this.readOnlyCompartment.reconfigure([CodeMirrorState.readOnly.of(!this.view.editable), CodeMirror.contentAttributes.of({ contentEditable: "".concat(this.view.editable) })]) }); this.updating = false; } }, { key: "updateLanguage", value: function updateLanguage() { this.languageLoader.updateLanguage(this.node.attrs.language); if (expValEquals('editor_a11y_role_textbox', 'isEnabled', true) && fg('platform_editor_adf_with_localid')) { this.updateAriaDescription(); } } }, { key: "updateAriaDescription", value: function updateAriaDescription() { if (!this.invisibleAriaDescription) { return; } var _this$config$getIntl = this.config.getIntl(), formatMessage = _this$config$getIntl.formatMessage; var languageName = getLanguageName(this.node.attrs.language); if (languageName) { this.invisibleAriaDescription.textContent = "".concat(formatMessage(codeBlockMessages.codeblockLanguageAriaDescription, { language: languageName }), " ").concat(formatMessage(floatingToolbarMessages.floatingToolbarAnnouncer)); } else { // If the lanuage is undefined provide a more human readable message this.invisibleAriaDescription.textContent = "".concat(formatMessage(codeBlockMessages.codeBlockLanguageNotSet), " ").concat(formatMessage(floatingToolbarMessages.floatingToolbarAnnouncer)); } } }, { key: "updateLocalIdAttribute", value: function updateLocalIdAttribute() { if (fg('platform_editor_adf_with_localid')) { var localId = this.node.attrs.localId; if (localId) { this.cm.dom.setAttribute('data-local-id', localId); } else { this.cm.dom.removeAttribute('data-local-id'); } } } }, { key: "selectCodeBlockNode", value: function selectCodeBlockNode(relativeSelectionPos) { var _this$selectionAPI, _this$getPos3, _this$getPos4; var tr = (_this$selectionAPI = this.selectionAPI) === null || _this$selectionAPI === void 0 ? void 0 : _this$selectionAPI.selectNearNode({ selectionRelativeToNode: relativeSelectionPos, selection: NodeSelection.create(this.view.state.doc, (_this$getPos3 = (_this$getPos4 = this.getPos) === null || _this$getPos4 === void 0 ? void 0 : _this$getPos4.call(this)) !== null && _this$getPos3 !== void 0 ? _this$getPos3 : 0) })(this.view.state); if (tr) { this.view.dispatch(tr); } } }, { key: "getLineNumberExtensions", value: function getLineNumberExtensions() { var _this2 = this; return [lineNumbers({ domEventHandlers: { click: function click() { _this2.selectCodeBlockNodeAndFocus(); return true; } } })]; } }, { key: "getLineNumberVisibilityExtensions", value: function getLineNumberVisibilityExtensions(node) { if (areCodeBlockLineNumbersHidden(node)) { return []; } return this.getLineNumberExtensions(); } }, { key: "getLineNumbersEffects", value: function getLineNumbersEffects(node) { var lineNumbersHidden = areCodeBlockLineNumbersHidden(node); if (this.lineNumbersHidden !== lineNumbersHidden) { this.lineNumbersHidden = lineNumbersHidden; return this.lineNumbersCompartment.reconfigure(this.getLineNumberVisibilityExtensions(node)); } return undefined; } }, { key: "getWordWrapEffects", value: function getWordWrapEffects(node) { if (this.wordWrappingEnabled !== isCodeBlockWordWrapEnabled(node)) { this.wordWrappingEnabled = !this.wordWrappingEnabled; return this.lineWrappingCompartment.reconfigure(isCodeBlockWordWrapEnabled(node) ? CodeMirror.lineWrapping : []); } return undefined; } }, { key: "restoreFoldState", value: function restoreFoldState() { this.updating = true; var effects = getCodeBlockFoldStateEffects({ node: this.node, cm: this.cm }); if (effects) { this.cm.dispatch({ effects: effects }); } this.updating = false; } }, { key: "applyContentModeTheme", value: function applyContentModeTheme(contentMode) { if (contentMode === this.contentMode) { return; } this.contentMode = contentMode; this.updating = true; this.cm.dispatch({ effects: this.themeCompartment.reconfigure(cmTheme({ contentMode: contentMode })) }); this.updating = false; } }, { key: "update", value: function update(node, _, innerDecorations) { this.maybeTryingToReachNodeSelection = false; if (node.type !== this.node.type) { return false; } this.node = node; if (this.updating) { return true; } this.updateLanguage(); this.updateLocalIdAttribute(); var newText = node.textContent, curText = this.cm.state.doc.toString(); // Updates bundled for performance (to avoid multiple-dispatches) var changes = getCMSelectionChanges(curText, newText); var wordWrapEffect = this.getWordWrapEffects(node); var lineNumbersEffect = expValEquals('platform_editor_code_block_q4_lovability', 'isEnabled', true) ? this.getLineNumbersEffects(node) : undefined; var prosemirrorDecorationsEffect = this.getProseMirrorDecorationEffects(innerDecorations); if (changes || wordWrapEffect || lineNumbersEffect || prosemirrorDecorationsEffect) { this.updating = true; this.cm.dispatch({ effects: [wordWrapEffect, lineNumbersEffect, prosemirrorDecorationsEffect].filter(function (effect) { return !!effect; }), changes: changes }); this.updating = false; } return true; } /** * Updates a facet which stores information on the prosemirror decorations * * This then gets translated to codemirror decorations in `prosemirrorDecorationPlugin` * @param decorationSource * @example */ }, { key: "getProseMirrorDecorationEffects", value: function getProseMirrorDecorationEffects(decorationSource) { var computedFacet = this.pmFacet.compute([], function () { return decorationSource; }); return this.pmDecorationsCompartment.reconfigure(computedFacet); } }, { key: "clearProseMirrorDecorations", value: function clearProseMirrorDecorations() { this.updating = true; var computedFacet = this.pmFacet.compute([], function () { return DecorationSet.empty; }); this.cm.dispatch({ effects: this.pmDecorationsCompartment.reconfigure(computedFacet) }); this.updating = false; } }, { key: "stopEvent", value: function stopEvent(e) { var _this$getPos5; // If we have selected the node we should not stop these events if ((e instanceof KeyboardEvent || e instanceof ClipboardEvent) && this.view.state.selection instanceof NodeSelection && this.view.state.selection.from === ((_this$getPos5 = this.getPos) === null || _this$getPos5 === void 0 ? void 0 : _this$getPos5.call(this))) { return false; } if (e instanceof DragEvent && e.type === 'dragenter' && expValEqualsNoExposure('platform_editor_block_controls_perf_optimization', 'isEnabled', true)) { return false; // Allow dragenter to propagate so that the editor can handle it } return true; } }]); }(); export var getCodeBlockAdvancedNodeView = function getCodeBlockAdvancedNodeView(props) { return function (node, view, getPos, innerDecorations) { return new CodeBlockAdvancedNodeView(node, view, getPos, innerDecorations, props); }; };