Current Path : /var/www/www-root/data/www/info.monolith-realty.ru/bitrix/js/fileman/html_editor/ |
Current File : /var/www/www-root/data/www/info.monolith-realty.ru/bitrix/js/fileman/html_editor/html-views.js |
/** * Bitrix HTML Editor 3.0 * Date: 24.04.13 * Time: 4:23 * * Views class */ (function() { function BXEditorView(editor, element, container) { this.editor = editor; this.element = element; this.container = container; this.config = editor.config || {}; this.isShown = null; this.bbCode = editor.bbCode; BX.addCustomEvent(this.editor, "OnClickBefore", BX.proxy(this.OnClick, this)); } BXEditorView.prototype = { Focus: function() { if (!document.querySelector || this.element.ownerDocument.querySelector(":focus") === this.element) return; try{this.element.focus();}catch(e){} }, Hide: function() { this.isShown = false; this.container.style.display = "none"; }, Show: function() { this.isShown = true; this.container.style.display = ""; }, Disable: function() { this.element.setAttribute("disabled", "disabled"); }, Enable: function() { this.element.removeAttribute("disabled"); }, OnClick: function(params) { }, IsShown: function() { return !!this.isShown; } }; function BXEditorTextareaView(parent, textareaElement, container) { // Call parrent constructor BXEditorIframeView.superclass.constructor.apply(this, arguments); this.name = "textarea"; this.InitEventHandlers(); if (!this.element.value && this.editor.config.content) this.SetValue(this.editor.config.content, false); } BX.extend(BXEditorTextareaView, BXEditorView); BXEditorTextareaView.prototype.Clear = function() { this.element.value = ""; }; BXEditorTextareaView.prototype.GetValue = function(bParse) { var value = this.IsEmpty() ? "" : this.element.value; if (bParse) { value = this.parent.parse(value); } return value; }; BXEditorTextareaView.prototype.SetValue = function(html, bParse, bFormat) { if (bParse) { html = this.editor.Parse(html, true, bFormat); } this.editor.dom.pValueInput.value = this.element.value = html; }; BXEditorTextareaView.prototype.SaveValue = function() { if (this.editor.inited) { this.editor.dom.pValueInput.value = this.element.value; } }; BXEditorTextareaView.prototype.HasPlaceholderSet = function() { var placeholderText = this.element.getAttribute("placeholder") || null, value = this.element.value; return !value || (value === placeholderText); }; BXEditorTextareaView.prototype.IsEmpty = function() { var value = BX.util.trim(this.element.value); return value === '' || this.HasPlaceholderSet(); }; BXEditorTextareaView.prototype.InitEventHandlers = function() { var _this = this; BX.bind(this.element, "focus", function() { _this.editor.On("OnTextareaFocus"); _this.isFocused = true; }); BX.bind(this.element, "blur", function() { _this.editor.On("OnTextareaBlur"); _this.isFocused = false; }); BX.bind(this.element, "keydown", function(e) { _this.editor.textareaKeyDownPreventDefault = false; // Handle Ctrl+Enter if ((e.ctrlKey || e.metaKey) && !e.altKey && e.keyCode === _this.editor.KEY_CODES["enter"]) { _this.editor.On('OnCtrlEnter', [e, _this.editor.GetViewMode()]); return BX.PreventDefault(e); } _this.editor.On('OnTextareaKeydown', [e]); if (_this.editor.textareaKeyDownPreventDefault) return BX.PreventDefault(e); }); BX.bind(this.element, "keyup", function(e) { _this.editor.On('OnTextareaKeyup', [e]); }); }; BXEditorTextareaView.prototype.IsFocused = function() { return this.isFocused; }; BXEditorTextareaView.prototype.ScrollToSelectedText = function(searchText) { // http://blog.blupixelit.eu/scroll-textarea-to-selected-word-using-javascript-jquery/ // var parola_cercata = "parola"; // the searched word // var posi = jQuery('#my_textarea').val().indexOf(parola_cercata); // take the position of the word in the text // if (posi != -1) { // var target = document.getElementById("my_textarea"); // // select the textarea and the word // target.focus(); // if (target.setSelectionRange) // target.setSelectionRange(posi, posi+parola_cercata.length); // else { // var r = target.createTextRange(); // r.collapse(true); // r.moveEnd('character', posi+parola_cercata); // r.moveStart('character', posi); // r.select(); // } // var objDiv = document.getElementById("my_textarea"); // var sh = objDiv.scrollHeight; //height in pixel of the textarea (n_rows*line_height) // var line_ht = jQuery('#my_textarea').css('line-height').replace('px',''); //height in pixel of each row // var n_lines = sh/line_ht; // the total amount of lines // var char_in_line = jQuery('#insert_textarea').val().length / n_lines; // amount of chars for each line // var height = Math.floor(posi/char_in_line); // amount of lines in the textarea // jQuery('#my_textarea').scrollTop(height*line_ht); // scroll to the selected line // } else { // alert('parola '+parola_cercata+' non trovata'); // alert word not found // } }; BXEditorTextareaView.prototype.SelectText = function(searchText) { var value = this.element.value, ind = value.indexOf(searchText); if(ind != -1) { this.element.focus(); this.element.setSelectionRange(ind, ind + searchText.length); } }; BXEditorTextareaView.prototype.GetTextSelection = function() { var res = false; if (this.element.selectionStart != undefined) { res = this.element.value.substr(this.element.selectionStart, this.element.selectionEnd - this.element.selectionStart); } else if (document.selection && document.selection.createRange) { res = document.selection.createRange().text; } else if (window.getSelection) { res = window.getSelection(); res = res.toString(); } return res; }; BXEditorTextareaView.prototype.WrapWith = function (tagBegin, tagEnd, postText) { if (!tagBegin) tagBegin = ""; if (!tagEnd) tagEnd = ""; if (!postText) postText = ""; if (tagBegin.length <= 0 && tagEnd.length <= 0 && postText.length <= 0) return true; var bReplaceText = !!postText, selectedText = this.GetTextSelection(), mode = (selectedText ? 'select' : (bReplaceText ? 'after' : 'in')); //if (!this.bTextareaFocus) // this.pTextarea.focus(); // BUG IN IE if (bReplaceText) { postText = tagBegin + postText + tagEnd; } else if (selectedText) { postText = tagBegin + selectedText + tagEnd; } else { postText = tagBegin + tagEnd; } if (this.element.selectionStart != undefined) { var currentScroll = this.element.scrollTop, start = this.element.selectionStart, end = this.element.selectionEnd; this.element.value = this.element.value.substr(0, start) + postText + this.element.value.substr(end); if (mode == 'select') { this.element.selectionStart = start; this.element.selectionEnd = start + postText.length; } else if (mode == 'in') { this.element.selectionStart = this.element.selectionEnd = start + tagBegin.length; } else { this.element.selectionStart = this.element.selectionEnd = start + postText.length; } this.element.scrollTop = currentScroll; } else if (document.selection && document.selection.createRange) { var sel = document.selection.createRange(); var selection_copy = sel.duplicate(); postText = postText.replace(/\r?\n/g, '\n'); sel.text = postText; sel.setEndPoint('StartToStart', selection_copy); sel.setEndPoint('EndToEnd', selection_copy); if (mode == 'select') { sel.collapse(true); postText = postText.replace(/\r\n/g, '1'); sel.moveEnd('character', postText.length); } else if (mode == 'in') { sel.collapse(false); sel.moveEnd('character', tagBegin.length); sel.collapse(false); } else { sel.collapse(false); sel.moveEnd('character', postText.length); sel.collapse(false); } sel.select(); } else { // failed - just stuff it at the end of the message this.element.value += postText; } return true; }; BXEditorTextareaView.prototype.GetCursorPosition = function() { return this.element.selectionStart; }; class BXEditorIframeCopilot { copilotParams = {}; copilotLoaded = false; selectionTypeCaret = 'Caret'; resultNodeAttr = 'bxhtmled-copilot-result-node'; resultColor = '#8d52ec'; invitationLineModes = { NONE: 'none', LAST_LINE: 'lastLine', EACH_LINE: 'eachLine', }; invitationLineMode = this.invitationLineModes.LAST_LINE; /** * @param iframeView BXEditorIframeView * @param copilotParams {{moduleId, contextId, category, contextParameters, invitationLineMode, isMentionUnavailable}} */ constructor(iframeView, copilotParams = {}) { this.iframeView = iframeView; this.iframeContainer = iframeView.container; this.contentEditable = iframeView.element; this.copilotParams = copilotParams; this.invitationLineMode = copilotParams.invitationLineMode ?? this.invitationLineModes.LAST_LINE; this.invitationLine = this.renderInvitationLine(); this.invitationLineAbsolute = this.renderInvitationLine(); this.insertResultNode = this.renderInsertResultNode(); this.copilot = new BX.AI.Copilot({ moduleId: copilotParams.moduleId, contextId: copilotParams.contextId, category: copilotParams.category, contextParameters: copilotParams.contextParameters, autoHide: true, preventAutoHide: (event) => this.iframeContainer.contains(event.target), }); this.bindHandlers(); this.copilot.init(); } renderInvitationLine() { let placeHolder = BX.message('BXEdCopilotPlaceholder_MSGVER_2'); if (this.copilotParams.isMentionUnavailable) { placeHolder = BX.message('BXEdCopilotPlaceholderWithoutMention_MSGVER_1'); } return BX.Tag.render` <div class="bxhtmled-copilot" placeholder="${placeHolder}"></div> `; } renderInsertResultNode() { return BX.Tag.render` <span class="bxhtmled-insert-result"></span> `; } bindHandlers() { this.copilot.subscribe('finish-init', this.finishInitHandler.bind(this)); this.copilot.subscribe('aiResult', this.aiResultHandler.bind(this)); this.copilot.subscribe('save', this.saveHandler.bind(this)); this.copilot.subscribe('add_below', this.addBelowHandler.bind(this)); this.copilot.subscribe('cancel', this.cancelHandler.bind(this)); this.copilot.subscribe('hide', this.saveResultNodes.bind(this)); document.addEventListener('click', this.documentClickHandler.bind(this)); document.addEventListener('keydown', this.onWindowKeyDownHandler.bind(this)); this.contentEditable.addEventListener('input', this.onContentEditableKeyDown.bind(this)); window.addEventListener('scroll', this.onScrollHandler.bind(this), true); new ResizeObserver(this.onScrollHandler.bind(this)).observe(this.contentEditable); window.addEventListener('resize', this.handleResizeWindow.bind(this)); BX.addCustomEvent(window, "onPullEvent-unicomments", this.startAdjustAnimation.bind(this)); this.hideObserver = new MutationObserver(() => { if (this.iframeContainer.offsetHeight <= 0) { this.hideInvitationLine(); this.hideObserver.disconnect(); } }); this.hideObserver.observe(document.body, {childList: true}); } documentClickHandler(event) { if (!this.iframeContainer.contains(event.target)) { this.hideInvitationLine(); } } startAdjustAnimation() { this.animation?.stop(); this.animation = new BX.easing({ duration: 1000, start: {}, finish: {}, transition : BX.easing.makeEaseOut(BX.easing.transitions.linear), step: () => { if (this.copilot.isShown()) { this.copilot.adjust(this.getAdjustOptionsForRect(this.adjustmentRect)); } if (this.copilotBtnPopup?.isShown()) { this.adjustCopilotButton(this.getAdjustOptionsForRectSpaced(this.adjustmentRect, true)); } }, complete: () => this.animation = null, }); this.animation.animate(); } finishInitHandler() { this.copilotLoaded = true; this.updateInvitationLine(); } aiResultHandler(event) { const resultNodes = this.getResultNodes(); let lastResultSpan; if (resultNodes.length !== 0) { lastResultSpan = [...resultNodes].pop(); lastResultSpan.innerText = event.data.result; } else { lastResultSpan = this.getSpanWithText(event.data.result); BX.Dom.style(lastResultSpan, 'color', this.resultColor); BX.Dom.attr(lastResultSpan, this.resultNodeAttr, 'true'); if (this.invitationLineMode === this.invitationLineModes.LAST_LINE) { this.insertResultNode.after(BX.Tag.render`<br>`); } this.insertResultNode.after(lastResultSpan); this.insertResultNode.remove(); } this.iframeView.ScrollToInsertedText(); this.getSelection().removeAllRanges(); this.copilot.adjust(this.getAdjustOptions(lastResultSpan)); } saveHandler(event) { if (this.getResultNodes().length === 0) { const selection = this.getSelection().getRangeAt(0); const span = this.getSpanWithText(event.data.result); selection.deleteContents(); selection.insertNode(span); this.iframeView.UpdateHeight(); } this.saveResultNodes(); this.iframeView.editor.synchro.FromIframeToTextarea(true, true); this.iframeView.editor.synchro.FromTextareaToIframe(true); this.copilot.hide(); } addBelowHandler(event) { const span = this.getSpanWithText(event.data.result); let lastSelectedElement = null; if (this.getSelectionText() !== '') { lastSelectedElement = this.getSelection().getRangeAt(0).endContainer; } const isSelectionEndOutOfBounds = lastSelectedElement === this.iframeView.element; if (lastSelectedElement && !isSelectionEndOutOfBounds) { lastSelectedElement.after(span); lastSelectedElement.after(BX.Tag.render`<br>`); } else { BX.Dom.append(BX.Tag.render`<br>`, this.contentEditable); BX.Dom.append(span, this.contentEditable); BX.Dom.append(BX.Tag.render`<br>`, this.contentEditable); } this.iframeView.ScrollToInsertedText(); this.getSelection().removeAllRanges(); this.iframeView.editor.synchro.FromIframeToTextarea(true, true); this.iframeView.editor.synchro.FromTextareaToIframe(true); this.copilot.hide(); } cancelHandler() { [...this.getResultNodes()][0]?.before(this.insertResultNode); this.removeResultNodes(); if (this.showRect) { this.copilot.adjust(this.getAdjustOptionsForRect(this.showRect)); } } getSpanWithText(text) { const span = BX.Dom.create('span'); span.innerText = text; return span; } onWindowKeyDownHandler(event) { if (event.key === "Escape") { this.copilot.hide(); } } handleResizeWindow() { this.copilot.adjustWidth(this.getCopilotWidth()); } removeResultNodes() { this.getResultNodes().forEach(resultNode => resultNode.remove()); } saveResultNodes() { for (const resultNode of this.getResultNodes()) { for (const childNode of resultNode.childNodes) { resultNode.before(childNode.cloneNode(true)); } resultNode.remove(); } } onContentEditableMouseDown() { this.insertResultNode.remove(); this.hideInvitationLine(); this.hideCopilotButton(); } onIframeWindowClick() { this.update(); } onContentEditableKeyDown() { this.insertResultNode.remove(); this.updatePopup(); this.hideInvitationLine(); this.hideCopilotButton(); } onContentEditableKeyUp(e) { if (e.key === ' ') { return; } this.update(); } onScrollHandler() { if (this.contentEditable.offsetHeight <= 0) { return; } if (this.invitationLineAbsolute.parentNode) { this.showInvitationLine(); } if (this.copilotBtnPopup?.isShown() && this.getSelectionText() !== '') { const range = this.getSelection().getRangeAt(0); this.adjustCopilotButton(this.getAdjustOptions(range, true)); } if (this.copilot.isShown() && this.getSelectionText() !== '') { const range = this.getSelection().getRangeAt(0); this.copilot.adjust(this.getAdjustOptions(range)); return; } if (this.getResultNodes().pop()) { this.updatePopup(); } else if (this.adjustmentRect) { this.copilot.adjust(this.getAdjustOptionsForRect(this.adjustmentRect)); } } onContentEditableBlur() { setTimeout(() => { this.hideInvitationLine(); }, 0); } onIframeFocus() { setTimeout(() => { this.updateInvitationLine(); }, 0); } shouldBeShown() { return this.invitationLineAbsolute.offsetWidth !== 0 && !this.copilot.isShown(); } show(showFromSpace = false) { this.copilot.setContext(this.contentEditable.innerText); this.insertResultNode.remove(); this.getSelection().getRangeAt(0).insertNode(this.insertResultNode); const bindElement = { top: parseInt(this.invitationLineAbsolute.style.top), left: parseInt(this.invitationLineAbsolute.style.left), }; if (showFromSpace) { bindElement.top += 28; } const containerRect = this.iframeContainer.getBoundingClientRect(); this.adjustmentRect = { bottom: bindElement.top - containerRect.y - window.scrollY, x: bindElement.left - containerRect.x - window.scrollX, width: 0, }; this.showRect = this.adjustmentRect; this.copilot.show({ bindElement, showFromSpace, width: this.getCopilotWidth(), }); this.hideInvitationLine(); } showAtTheBottom() { this.copilot.setContext(this.contentEditable.innerText); this.insertResultNode.remove(); let emptyBr; let emptyLine; if (this.needToAppendEmptyElement()) { emptyBr = BX.Tag.render`<br>`; emptyLine = BX.Tag.render`<div></div>`; const lastNode = this.getFilteredChildNodes(this.contentEditable).pop(); if (this.contentEditable.innerText !== '' && lastNode?.tagName !== 'BR') { this.contentEditable.append(BX.Tag.render`<br>`); } this.contentEditable.append(emptyBr); this.contentEditable.append(emptyLine); } const lastNode = this.getFilteredChildNodes(this.contentEditable).pop(); const bindElement = this.getAdjustOptions(lastNode).position; emptyLine?.remove(); this.showRect = this.adjustmentRect; if (emptyBr) { emptyBr.before(this.insertResultNode); } else { this.contentEditable.append(this.insertResultNode); } this.copilot.show({ bindElement, width: this.getCopilotWidth(), }); this.hideInvitationLine(); } needToAppendEmptyElement() { const innerText = this.contentEditable.innerText; return !innerText || (innerText.at(-1) !== '\n') || (innerText.at(-2) && innerText.at(-2) !== '\n'); } update() { this.updateCopilotButton(); this.updatePopup(); this.updateInvitationLine(); } updateCopilotButton() { const shouldShowCopilotButton = !this.copilot.isShown() && this.getSelectionText() !== ''; if (shouldShowCopilotButton && !this.copilotBtnPopup?.isShown() && this.copilotLoaded) { this.showCopilotButton(); } if (!shouldShowCopilotButton) { this.hideCopilotButton(); } } showCopilotButton() { if (!this.copilotBtnPopup) { this.copilotBtnPopup = new BX.Main.Popup({ bindElement: this.contentEditable, padding: 6, borderRadius: '6px', content: this.renderCopilotButton(), autoHide: true, }); } this.copilotBtnPopup.setMaxWidth(null); this.copilotBtnPopup.setMinWidth(null); this.copilotBtnPopup.setPadding(6); this.adjustCopilotButton(this.getAdjustOptions(this.getSelection().getRangeAt(0), true)); this.copilotBtnPopup.adjustPosition(); this.copilotBtnPopup.show(); setTimeout(() => this.updateCopilotButton(), 0); } renderCopilotButton() { BX.Runtime.loadExtension('ui.icon-set.main'); this.copilotButton = BX.Tag.render` <button class="bxhtmled-copilot-btn" onclick="${this.copilotButtonClickHandler.bind(this)}"> <div class="bxhtmled-copilot-btn-icon ui-icon-set --copilot-ai"></div> ${ BX.message('BXEdCopilotButtonText') } </button> `; return this.copilotButton; } copilotButtonClickHandler() { const adjustOptions = this.getAdjustOptions(this.getSelection().getRangeAt(0)); this.copilot.setSelectedText(this.getSelection().toString()); this.copilot.show({ bindElement: adjustOptions.position, showFromPopup: true, width: this.getCopilotWidth(), }); this.hideCopilotButton(); } adjustCopilotButton(options) { if (options.hide) { this.copilotBtnPopup.setMaxWidth(0); this.copilotBtnPopup.setMinWidth(0); this.copilotBtnPopup.setPadding(0); } else { this.copilotBtnPopup.setMaxWidth(null); this.copilotBtnPopup.setMinWidth(null); this.copilotBtnPopup.setPadding(6); this.copilotBtnPopup.setBindElement(options.position); this.copilotBtnPopup.adjustPosition(); } } getCopilotWidth() { return this.contentEditable.offsetWidth - 30; } hideCopilotButton() { this.copilotBtnPopup?.close(); } updatePopup() { if (!this.copilot.isShown()) { return; } const lastResultSpan = this.getResultNodes().pop(); if (!this.isNotFocusedOrCursorAtResultNode(this.getSelection()) || !lastResultSpan) { this.copilot.hide(); return; } this.copilot.adjust(this.getAdjustOptions(lastResultSpan)); } getAdjustOptions(pivot, isCentered = false) { const pivotRect = pivot.getBoundingClientRect(); return this.getAdjustOptionsForRectSpaced(pivotRect, isCentered); } getAdjustOptionsForRectSpaced(pivotRect, isCentered = false) { const adjustment = this.getAdjustOptionsForRect(pivotRect, isCentered); adjustment.position.top += 10; return adjustment; } getAdjustOptionsForRect(pivotRect, isCentered = false) { this.adjustmentRect = pivotRect; const containerRect = this.iframeContainer.getBoundingClientRect(); return { hide: pivotRect.bottom > this.iframeView.document.documentElement.offsetHeight + 13 || pivotRect.bottom + 13 < 0, position: { top: pivotRect.bottom + containerRect.y + window.scrollY, left: pivotRect.x + containerRect.x + isCentered * (pivotRect.width / 2 - 55) + window.scrollX, }, }; } isNotFocusedOrCursorAtResultNode(selection) { if (!selection.focusNode) { return true; } return this.getResultNodes().filter(node => selection.focusNode === node || selection.focusNode.parentElement === node).length; } getResultNodes() { return [...this.contentEditable.querySelectorAll(`[${this.resultNodeAttr}=true]`)]; } updateInvitationLine() { this.insertResultNode.remove(); if (this.shouldShowInvitationLine() && this.copilotLoaded) { this.showInvitationLine(); } else { this.hideInvitationLine(); } } showInvitationLine() { const selection = this.getSelection(); if (selection.rangeCount === 0) { return false; } const lastNode = this.getFilteredChildNodes(this.contentEditable).pop(); if (this.isZwnbspNode(lastNode)) { lastNode.replaceWith(BX.Tag.render`<br>`); } const range = selection.getRangeAt(0); range.insertNode(this.invitationLine); const container = this.iframeContainer.getBoundingClientRect(); const invitationLineRect = { top: this.invitationLine.offsetTop + container.y - this.contentEditable.parentElement.scrollTop + window.scrollY, left: this.invitationLine.offsetLeft + container.x + window.scrollX, width: this.invitationLine.offsetWidth, }; this.invitationLineAbsolute.style.top = `${invitationLineRect.top}px`; this.invitationLineAbsolute.style.left = `${invitationLineRect.left}px`; this.invitationLineAbsolute.style.width = `${invitationLineRect.width}px`; this.invitationLineAbsolute.style.display = ''; this.hideInvitationLine(); document.body.append(this.invitationLineAbsolute); if (!this.invitationLineAbsolute.registered) { BX.ZIndexManager.register(this.invitationLineAbsolute); BX.ZIndexManager.bringToFront(this.invitationLineAbsolute); this.invitationLineAbsolute.registered = true; } if (!this.shouldDisplayInvitationLine()) { this.invitationLineAbsolute.style.display = 'none'; } } shouldDisplayInvitationLine() { const contentEditableRect = this.iframeContainer.getBoundingClientRect(); const invitationLineRect = this.invitationLineAbsolute.getBoundingClientRect(); return invitationLineRect.bottom + 20 < contentEditableRect.bottom && invitationLineRect.top > contentEditableRect.top; } hideInvitationLine() { this.invitationLine.remove(); this.invitationLineAbsolute.remove(); } getSelectionText() { return this.getSelection().toString().replaceAll('\n', '').trim(); } getSelection() { return this.iframeView.GetSelection(); } shouldShowInvitationLine() { if (!this.iframeContainer.contains(document.activeElement)) { return false; } if (this.invitationLineMode === this.invitationLineModes.NONE) { return false; } if (this.invitationLineMode === this.invitationLineModes.EACH_LINE) { return this.isCursorAtStartOfLine(this.getSelection()); } return this.isCursorAtNewLine(this.getSelection()); } isCursorAtStartOfLine(selection) { if (selection.type !== this.selectionTypeCaret) { return false; } this.removeZwnbspSequence(); const range = selection.getRangeAt(0); const contentEditableOffset = parseInt(getComputedStyle(this.contentEditable).paddingLeft); if (range.getBoundingClientRect().x > contentEditableOffset) { return false; } const tmpNode = BX.Tag.render`<span style="width: 10px; height: 10px;"></span>`; range.insertNode(tmpNode); const previousNode = tmpNode.previousSibling; const nextNode = tmpNode.nextSibling; const offset = tmpNode.getBoundingClientRect().x - contentEditableOffset; if ( selection.focusNode === this.contentEditable && nextNode === null && ( previousNode?.tagName === 'DIV' || this.isZwnbspNode(previousNode) ) ) { if (this.isZwnbspNode(previousNode)) { if (!previousNode.nextSibling) { previousNode.replaceWith(BX.Tag.render`<br>`); } else { previousNode.remove(); } } tmpNode.remove(); return false; } const afterBr = !previousNode || this.doesNodeMakeLine(previousNode) || tmpNode.nodeName !== '#text' && tmpNode.textContent === ''; const beforeBr = !nextNode || this.doesNodeMakeLine(nextNode) && nextNode.tagName !== 'TABLE'; const atStart = offset <= 0; tmpNode.remove(); return ((afterBr && beforeBr) || (this.isZwnbspNode(previousNode) && nextNode?.nodeName === '#text')) && atStart; } isCursorAtNewLine(selection) { if (selection.toString() !== '') { return false; } this.removeZwnbspSequence(); if (selection.focusNode.outerHTML === '<span><br></span>' || selection.focusNode.outerHTML === '<div><br></div>') { selection.focusNode.replaceWith(BX.Tag.render`<br>`); } const nodes = this.getFilteredChildNodes(this.contentEditable); const isEmptyOrOnlyLine = (nodes.length === 0) || (nodes.length === 1 && nodes[0].tagName === 'BR'); const lastNode = nodes.pop(); let doesLastNodeMakeLine = false; if (lastNode?.tagName === 'SPAN') { const spanNodes = this.getFilteredChildNodes(lastNode); const lastSpanNode = spanNodes.pop(); const oneSpanNodeBeforeLast = spanNodes.pop(); doesLastNodeMakeLine = this.doesNodeMakeLine(oneSpanNodeBeforeLast) && this.doesNodeMakeLine(lastSpanNode); } if (!this.isZwnbspNode(selection.focusNode) && selection.focusNode !== this.contentEditable && lastNode?.tagName !== 'SPAN') { return false; } const oneNodeBeforeLast = nodes.pop(); const doLastTwoNodesMakeLine = this.doesNodeMakeLine(oneNodeBeforeLast) && this.doesNodeMakeLine(lastNode) && lastNode?.tagName !== 'BLOCKQUOTE'; return (isEmptyOrOnlyLine || doLastTwoNodesMakeLine || doesLastNodeMakeLine) && this.isCursorAtTheEndOfDocument(selection); } removeZwnbspSequence() { const nodesToRemove = []; this.contentEditable.childNodes.forEach((node) => { if (this.isZwnbspNode(node) && this.isZwnbspNode(node.nextSibling)) { nodesToRemove.push(node); } }); nodesToRemove.forEach((node) => node.remove()); if (this.contentEditable.innerHTML === this.createZwnbspNode().innerHTML) { this.contentEditable.innerHTML = ''; } } getFilteredChildNodes(element) { return [...element.childNodes].filter((node) => { return node.nodeName !== '#text' || this.isZwnbspNode(node) || node.textContent !== '' }); } doesNodeMakeLine(node) { if (!node) { return false; } if (node.outerHTML === '<span><br></span>') { node.replaceWith(BX.Tag.render`<br>`); return true; } if (this.isZwnbspNode(node)) { if (!node.nextSibling) { node.replaceWith(BX.Tag.render`<br>`); } else { node.remove(); } return true; } const makingLineNodes = ['BR', 'UL', 'OL', 'PRE', 'TABLE', 'DIV', 'P', 'BLOCKQUOTE']; if (node.tagName === 'SPAN') { return this.doesNodeMakeLine(this.getFilteredChildNodes(node).pop()); } return makingLineNodes.includes(node.tagName) || makingLineNodes.includes([...node.childNodes].pop()?.tagName); } isZwnbspNode(node) { if (!node) { return false; } return BX.Tag.render`<div>${node.cloneNode()}</div>`.innerHTML === this.createZwnbspNode().innerHTML; } createZwnbspNode() { return BX.Tag.render`<div></div>`; } isCursorAtTheEndOfDocument(selection) { const offset = selection.focusOffset; const node = selection.focusNode; selection.modify("move", "forward", "character"); if (offset === selection.focusOffset && node === selection.focusNode) { return true; } else { selection.modify("move", "backward", "character"); return false; } } } function BXEditorIframeView(editor, textarea, container) { // Call parrent constructor BXEditorIframeView.superclass.constructor.apply(this, arguments); this.name = "wysiwyg"; this.caretNode = "<br>"; } BX.extend(BXEditorIframeView, BXEditorView); BXEditorIframeView.prototype.OnCreateIframe = function() { this.document = this.editor.sandbox.GetDocument(); this.element = this.document.body; this.editor.document = this.document; this.textarea = this.editor.dom.textarea; this.isFocused = false; this.InitEventHandlers(); // Check and init external range library window.rangy.init(); if (this.config.isCopilotEnabled && BX.AI?.Copilot) { this.copilot = new BXEditorIframeCopilot(this, { ...this.config.copilotParams, isMentionUnavailable: this.config.isMentionUnavailable, }); } this.Enable(); }; BXEditorIframeView.prototype.setCopilotContextParameters = function(formData) { this.config.copilotParams.contextParameters = { xmlId: formData[0], entityId: formData[1], }; }; BXEditorIframeView.prototype.ScrollToInsertedText = function() { this.document.documentElement.scrollTop = this.document.documentElement.scrollHeight; this.UpdateHeight(); }; BXEditorIframeView.prototype.UpdateHeight = function() { this.editor.UpdateHeight(); }; BXEditorIframeView.prototype.GetSelection = function() { return this.document.getSelection(); }; BXEditorIframeView.prototype.isCopilotInitialized = function() { return this.copilot && this.copilot.copilotLoaded; }; BXEditorIframeView.prototype.Clear = function() { //this.element.innerHTML = BX.browser.IsFirefox() ? this.caretNode : ""; this.element.innerHTML = this.caretNode; }; BXEditorIframeView.prototype.GetValue = function(bParse, bFormat) { this.iframeValue = this.IsEmpty() ? "" : this.editor.GetInnerHtml(this.element); this.iframeValue = this.iframeValue.replaceAll(/<span style="font-family: var\(--ui-font-family-primary, var\(--ui-font-family-helvetica\)\);">(.*?)<\/span>/g, '$1'); this.iframeValue = this.iframeValue.replace(/<span class="bxhtmled-insert-result"><\/span>/g, ''); this.editor.On('OnIframeBeforeGetValue', [this.iframeValue]); if (bParse) { this.iframeValue = this.editor.Parse(this.iframeValue, false, bFormat); } return this.iframeValue; }; BXEditorIframeView.prototype.SetValue = function(html, bParse) { if (bParse) { html = this.editor.Parse(html); } this.element.innerHTML = html; // Check last child - if it's block node in the end - add <br> tag there this.CheckContentLastChild(this.element); this.editor.On('OnIframeSetValue', [html]); }; BXEditorIframeView.prototype.Show = function() { this.isShown = true; this.container.style.display = ""; this.ReInit(); }; BXEditorIframeView.prototype.ReInit = function() { // Firefox needs this, otherwise contentEditable becomes uneditable this.Disable(); this.Enable(); this.editor.On('OnIframeReInit'); }; BXEditorIframeView.prototype.Hide = function() { this.isShown = false; this.container.style.display = "none"; }; BXEditorIframeView.prototype.Disable = function() { this.element.removeAttribute("contentEditable"); }; BXEditorIframeView.prototype.Enable = function() { this.element.setAttribute("contentEditable", "true"); }; BXEditorIframeView.prototype.Focus = function(setToEnd) { if (BX.browser.IsIE() && this.HasPlaceholderSet()) { this.Clear(); } if (!document.querySelector || this.element.ownerDocument.querySelector(":focus") !== this.element || !this.IsFocused()) { if (BX.browser.IsIOS()) { var _this = this; if (this.focusTimeout) clearTimeout(this.focusTimeout); this.focusTimeout = setTimeout(function() { var orScrollTop = document.documentElement.scrollTop || document.body.scrollTop, orScrollLeft = document.documentElement.scrollLeft || document.body.scrollLeft; BX.focus(_this.element); window.scrollTo(orScrollLeft, orScrollTop); }, 200); } else { BX.focus(this.element); } } if (setToEnd && this.element.lastChild) { if (this.element.lastChild.nodeName === "BR") { this.editor.selection.SetBefore(this.element.lastChild); } else { this.editor.selection.SetAfter(this.element.lastChild); } } }; BXEditorIframeView.prototype.SetFocusedFlag = function(isFocused) { this.isFocused = isFocused; }; BXEditorIframeView.prototype.IsFocused = function() { return this.isFocused; }; BXEditorIframeView.prototype.GetTextContent = function(clearInvisibleSpace) { var txt = this.editor.util.GetTextContent(this.element); return clearInvisibleSpace === true ? txt.replace(/\uFEFF/ig, '') : txt; }; BXEditorIframeView.prototype.HasPlaceholderSet = function() { return this.textarea && this.GetTextContent() == this.textarea.getAttribute("placeholder"); }; BXEditorIframeView.prototype.IsEmpty = function(clearInvisibleSpace) { if (!document.querySelector) return false; var innerHTML = this.element.innerHTML, elementsWithVisualValue = "blockquote, ul, ol, img, embed, object, table, iframe, svg, video, audio, button, input, select, textarea"; return innerHTML === "" || innerHTML === this.caretNode || this.HasPlaceholderSet() || (this.GetTextContent(clearInvisibleSpace) === "" && !this.element.querySelector(elementsWithVisualValue)); }; BXEditorIframeView.prototype._initObjectResizing = function() { var properties = ["width", "height"], propertiesLength = properties.length, element = this.element; this.commands.exec("enableObjectResizing", this.config.allowObjectResizing); if (this.config.allowObjectResizing) { // IE sets inline styles after resizing objects // The following lines make sure _this the width/height css properties // are copied over to the width/height attributes if (browser.supportsEvent("resizeend")) { dom.observe(element, "resizeend", function(event) { var target = event.target || event.srcElement, style = target.style, i = 0, property; for(; i<propertiesLength; i++) { property = properties[i]; if (style[property]) { target.setAttribute(property, parseInt(style[property], 10)); style[property] = ""; } } // After resizing IE sometimes forgets to remove the old resize handles redraw(element); }); } } else { if (browser.supportsEvent("resizestart")) { dom.observe(element, "resizestart", function(event) { event.preventDefault(); }); } } }; /** * With "setActive" IE offers a smart way of focusing elements without scrolling them into view: * http://msdn.microsoft.com/en-us/library/ms536738(v=vs.85).aspx * * Other browsers need a more hacky way: (pssst don't tell my mama) * In order to prevent the element being scrolled into view when focusing it, we simply * move it out of the scrollable area, focus it, and reset it's position */ var focusWithoutScrolling = function(element) { if (element.setActive) { // Following line could cause a js error when the textarea is invisible // See https://github.com/xing/wysihtml5/issues/9 try { element.setActive(); } catch(e) {} } else { var elementStyle = element.style, originalScrollTop = doc.documentElement.scrollTop || doc.body.scrollTop, originalScrollLeft = doc.documentElement.scrollLeft || doc.body.scrollLeft, originalStyles = { position: elementStyle.position, top: elementStyle.top, left: elementStyle.left, WebkitUserSelect: elementStyle.WebkitUserSelect }; dom.setStyles({ position: "absolute", top: "-99999px", left: "-99999px", // Don't ask why but temporarily setting -webkit-user-select to none makes the whole thing performing smoother WebkitUserSelect: "none" }).on(element); element.focus(); dom.setStyles(originalStyles).on(element); if (win.scrollTo) { // Some browser extensions unset this method to prevent annoyances // "Better PopUp Blocker" for Chrome http://code.google.com/p/betterpopupblocker/source/browse/trunk/blockStart.js#100 // Issue: http://code.google.com/p/betterpopupblocker/issues/detail?id=1 win.scrollTo(originalScrollLeft, originalScrollTop); } } }; /** * Taking care of events * - Simulating 'change' event on contentEditable element * - Handling drag & drop logic * - Catch paste events * - Dispatch proprietary newword:composer event * - Keyboard shortcuts */ BXEditorIframeView.prototype.InitEventHandlers = function() { var _this = this, editor = this.editor, value = this.GetValue(), element = this.element, iframeWindow = this.editor.sandbox.GetWindow(), _element = !BX.browser.IsOpera() ? element : this.editor.sandbox.GetWindow(); if (this._eventsInitedObject && this._eventsInitedObject === _element) return; this._eventsInitedObject = _element; BX.bind(_element, "focus", function() { editor.On("OnIframeFocus"); _this.isFocused = true; if (value !== _this.GetValue()) { BX.onCustomEvent(editor, "OnIframeChange"); } _this.copilot?.onIframeFocus(); }); BX.bind(_element, "blur", function() { editor.On("OnIframeBlur"); _this.isFocused = false; setTimeout(function(){value = _this.GetValue();}, 0); _this.copilot?.onContentEditableBlur(); }); BX.bind(_element, "contextmenu", function(e) { if(e && !e.ctrlKey && !e.shiftKey && (BX.getEventButton(e) & BX.MSRIGHT)) { editor.On("OnIframeContextMenu", [e, e.target || e.srcElement, _this.contMenuRangeCollapsed]); } }); BX.bind(_element, "mousedown", function(e) { var range = editor.selection.GetRange(), target = e.target || e.srcElement, bxTag = editor.GetBxTag(target); //mantis: 71174 _this.contMenuRangeCollapsed = range && range.collapsed; if (editor.synchro.IsSyncOn()) { editor.synchro.StopSync(); } if (BX.browser.IsIE10() || BX.browser.IsIE11()) { editor.phpParser.RedrawSurrogates(); } if (target.nodeName == 'BODY' || !editor.phpParser.CheckParentSurrogate(target)) { setTimeout(function() { range = editor.selection.GetRange(); if (range && range.collapsed && range.startContainer && range.startContainer == range.endContainer) { var surr = editor.phpParser.CheckParentSurrogate(range.startContainer); if (surr) { editor.selection.SetInvisibleTextAfterNode(surr); editor.selection.SetInvisibleTextBeforeNode(surr); } } }, 10); } editor.action.actions.quote.checkSpaceAfterQuotes(target); editor.selection.SaveRange(false); setTimeout(function(){editor.selection.SaveRange(false);}, 10); editor.On("OnIframeMouseDown", [e, target, bxTag]); }); BX.bind(_element, "touchend", function(){_this.Focus();}); BX.bind(_element, "touchstart", function(){_this.Focus();}); BX.bind(_element, "click", function(e) { var target = e.target || e.srcElement; editor.On("OnIframeClick", [e, target]); }); BX.bind(_element, "dblclick", function(e) { var target = e.target || e.srcElement; editor.On("OnIframeDblClick", [e, target]); }); BX.bind(_element, "mouseup", function(e) { var target = e.target || e.srcElement; if (!editor.synchro.IsSyncOn()) { editor.synchro.StartSync(); } editor.On("OnIframeMouseUp", [e, target]); }); BX.bind(iframeWindow, 'click', () => this.copilot?.onIframeWindowClick()); // Mantis: 90137 //if (BX.browser.IsIOS()) //{ // // When on iPad/iPhone/IPod after clicking outside of editor, the editor loses focus // // but the UI still acts as if the editor has focus (blinking caret and onscreen keyboard visible) // // We prevent _this by focusing a temporary input element which immediately loses focus // BX.bind(iframeWindow, "blur", function() // { // var // orScrollTop = document.documentElement.scrollTop || document.body.scrollTop, // orScrollLeft = document.documentElement.scrollLeft || document.body.scrollLeft, // input = BX.create('INPUT', { // props:{type: 'text', value: ''} // }, iframeWindow.ownerDocument); // // try // { // editor.selection.InsertNode(input); // } // catch(e) // { // iframeWindow.appendChild(input); // } // // BX.focus(input); // BX.remove(input); // window.scrollTo(orScrollLeft, orScrollTop); // }); //} // --------- Drag & Drop events --------- BX.bind(element, "dragover", function(){editor.On("OnIframeDragOver", arguments);}); BX.bind(element, "dragenter", function(){editor.On("OnIframeDragEnter", arguments);}); BX.bind(element, "dragleave", function(){editor.On("OnIframeDragLeave", arguments);}); BX.bind(element, "dragexit", function(){editor.On("OnIframeDragExit", arguments);}); BX.bind(element, "drop", function(){editor.On("OnIframeDrop", arguments);}); // Chrome & Safari & Firefox only fire the ondrop/ondragend/... events when the ondragover event is cancelled //if (BX.browser.IsChrome() || BX.browser.IsFirefox()) // TODO: Truobles with firefox during selections http://jabber.bx/view.php?id=49370 if (BX.browser.IsFirefox()) { BX.bind(element, "dragover", function(e) { e.preventDefault(); }); BX.bind(element, "dragenter", function(e) { e.preventDefault(); }); } BX.bind(element, 'drop', BX.delegate(this.OnPasteHandler, this)); BX.bind(element, 'paste', (clipboardEvent) => { const event = new BX.Event.BaseEvent({ data: { clipboardEvent: clipboardEvent } }); BX.Event.EventEmitter.emitAsync(this.editor, 'BXEditor:onBeforePasteAsync', event).then(() => { if (!event.isDefaultPrevented()) { this.OnPasteHandler(clipboardEvent); } }); }); BX.bind(element, "keyup", function(e) { var keyCode = e.keyCode, target = editor.selection.GetSelectedNode(true); _this.SetFocusedFlag(true); if (keyCode === editor.KEY_CODES['space'] || keyCode === editor.KEY_CODES['enter']) { if (keyCode === editor.KEY_CODES['enter']) { _this.OnEnterHandlerKeyUp(e, keyCode, target); } editor.On("OnIframeNewWord"); } else { _this.OnKeyUpArrowsHandler(e, keyCode); } editor.selection.SaveRange(); editor.On('OnIframeKeyup', [e, keyCode, target]); // Mantis:#67998 if (keyCode === editor.KEY_CODES['backspace'] && BX.browser.IsChrome() && target && target.nodeType == '3' && target.nextSibling && target.nextSibling.nodeType == '3') { _this.editor.selection.ExecuteAndRestoreSimple(function() { _this.editor.util.SetTextContent(target, _this.editor.util.GetTextContent(target) + _this.editor.util.GetTextContent(target.nextSibling)); target.nextSibling.parentNode.removeChild(target.nextSibling); }); } if (!editor.util.FirstLetterSupported() && _this.editor.parser.firstNodeCheck) { _this.editor.parser.FirstLetterCheckNodes('', '', true); } //mantis:91555, mantis:93629 if (BX.browser.IsChrome()) { if (_this.stopBugusScrollTimeout) clearTimeout(_this.stopBugusScrollTimeout); _this.stopBugusScrollTimeout = setTimeout(function(){_this.stopBugusScroll = false;}, 200); } _this.copilot?.onContentEditableKeyUp(e); }); BX.bind(this.document, 'scroll', () => this.copilot?.onScrollHandler()); BX.bind(element, "mousedown", function(e) { var target = e.target || e.srcElement; if (!editor.util.CheckImageSelectSupport() && target.nodeName === 'IMG') { editor.selection.SelectNode(target); } // Handle mousedown for "code" element in IE if (!editor.util.CheckPreCursorSupport() && target.nodeName === 'PRE') { var selectedNode = editor.selection.GetSelectedNode(true); if (selectedNode && selectedNode != target) { _this.FocusPreElement(target, true); } } _this.copilot?.onContentEditableMouseDown(); }); BX.bind(element, "keydown", BX.proxy(this.KeyDown, this)); // Workaround for chrome bug with bugus scrolling to the top of the page (mantis:91555, mantis:93629) if (BX.browser.IsChrome()) { BX.bind(window, "scroll", BX.proxy(function(e) { if (_this.stopBugusScroll) { if ((!_this.savedScroll || !_this.savedScroll.scrollTop) && _this.lastSavedScroll) _this.savedScroll = _this.lastSavedScroll; if (_this.savedScroll && !_this.lastSavedScroll) _this.lastSavedScroll = _this.savedScroll; _this._RestoreScrollTop(); } }, this)); } // Show urls and srcs in tooltip when hovering links or images var nodeTitles = { IMG: BX.message.SrcTitle + ": ", A: BX.message.UrlTitle + ": " }; BX.bind(element, "mouseover", function(e) { var target = e.target || e.srcElement, value = (target.getAttribute("href") || target.getAttribute("src")), nodeName = target.nodeName; if (nodeTitles[nodeName] && !target.hasAttribute("title") && value && value.indexOf('data:image/') === -1 ) { target.setAttribute("title", nodeTitles[nodeName] + value); target.setAttribute("data-bx-clean-attribute", "title"); } }); this.editor.InitClipboardHandler(); }; BXEditorIframeView.prototype.KeyDown = function(e) { var _this = this, keyCode = e.keyCode, KEY_CODES = this.editor.KEY_CODES; this.SetFocusedFlag(true); this.editor.iframeKeyDownPreventDefault = false; // Workaround for chrome bug with bugus scrollint to the top of the page (mantis:91555, mantis:93629) if (BX.browser.IsChrome()) { this.stopBugusScroll = true; this.savedScroll = BX.GetWindowScrollPos(document); } var command = this.editor.SHORTCUTS[keyCode], selectedNode = this.editor.selection.GetSelectedNode(true), range = this.editor.selection.GetRange(), body = this.document.body, parent; if (e.key === ' ' && this.copilot?.shouldBeShown()) { this.copilot.show(true); e.preventDefault(); return; } this.copilot?.onContentEditableKeyDown(); if ((BX.browser.IsIE() || BX.browser.IsIE10() || BX.browser.IsIE11()) && !BX.util.in_array(keyCode, [16, 17, 18, 20, 65, 144, 37, 38, 39, 40])) { if (selectedNode && selectedNode.nodeName == "BODY" || range.startContainer && range.startContainer.nodeName == "BODY" || (range.startContainer == body.firstChild && range.endContainer == body.lastChild && range.startOffset == 0 && range.endOffset == body.lastChild.length)) { BX.addCustomEvent(this.editor, "OnIframeKeyup", BX.proxy(this._IEBodyClearHandler, this)); } } // Last symbol in iframe and new paragraph in IE if ((BX.browser.IsIE() || BX.browser.IsIE10() || BX.browser.IsIE11()) && keyCode == KEY_CODES['backspace']) { BX.addCustomEvent(this.editor, "OnIframeKeyup", BX.proxy(this._IEBodyClearHandlerEx, this)); } this.isUserTyping = true; if (this.typingTimeout) { this.typingTimeout = clearTimeout(this.typingTimeout); } this.typingTimeout = setTimeout(function() { _this.isUserTyping = false; }, 1000); this.editor.synchro.StartSync(200); this.editor.On('OnIframeKeydown', [e, keyCode, command, selectedNode]); if (this.editor.iframeKeyDownPreventDefault) return BX.PreventDefault(e); // Handle Shortcuts if ((e.ctrlKey || e.metaKey) && !e.altKey && command) { this.editor.action.Exec(command); return BX.PreventDefault(e); } // Bug mantis: #59759 workaround *Chrome only // Begin if ( keyCode === KEY_CODES['backspace'] && range.startOffset == 0 && range.startContainer.nodeType == 3 && range.startContainer.parentNode.firstChild == range.startContainer && range.startContainer.parentNode && range.startContainer.parentNode.nodeName == 'BLOCKQUOTE' && range.startContainer.parentNode.className ) { range.startContainer.parentNode.className = ''; } if ( !BX.browser.IsSafari() && (keyCode === KEY_CODES['backspace'] || keyCode === KEY_CODES['delete']) && range.startContainer.nodeName === '#text' && range.startContainer.previousSibling?.nodeName === 'BLOCKQUOTE' && range.startContainer.nextSibling?.nodeName !== 'BR' && range.startContainer.textContent.length === 1 ) { range.startContainer.before(BX.Tag.render`<span></span>`.innerHTML); } if ( keyCode === KEY_CODES['delete'] && range.collapsed && range.endContainer.nodeType == 3 && range.endOffset == range.endContainer.length ) { var next = this.editor.util.GetNextNotEmptySibling(range.endContainer); if (next) { if(next.nodeName == 'BR') { next = this.editor.util.GetNextNotEmptySibling(next); } if (next && next.nodeName == 'BLOCKQUOTE' && next.className) { next.className = ''; } } } // END: Bug mantis: #59759 // Clear link with image if (selectedNode && selectedNode.nodeName === "IMG" && (keyCode === KEY_CODES['backspace'] || keyCode === KEY_CODES['delete'])) { parent = selectedNode.parentNode; parent.removeChild(selectedNode); // delete image // Parent - is LINK, and it's hasn't got any other childs if (parent.nodeName === "A" && !parent.firstChild) { parent.parentNode.removeChild(parent); } setTimeout(function(){_this.editor.util.Refresh(_this.element);}, 0); BX.PreventDefault(e); } if (range.collapsed && this.OnKeyDownArrowsHandler(e, keyCode, range) === false) { return false; } // Handle Ctrl+Enter if ((e.ctrlKey || e.metaKey) && !e.altKey && keyCode === KEY_CODES["enter"]) { if (this.IsFocused()) this.editor.On("OnIframeBlur"); this.editor.On('OnCtrlEnter', [e, this.editor.GetViewMode()]); return BX.PreventDefault(e); } // Firefox's bug it remove first node for customized lists if (BX.browser.IsFirefox() && selectedNode && (keyCode === KEY_CODES["delete"] || keyCode === KEY_CODES["backspace"])) { var li = selectedNode.nodeName == 'LI' ? selectedNode : BX.findParent(selectedNode, {tag: 'LI'}, body); if (li && li.firstChild && li.firstChild.nodeName == 'I') { var ul = BX.findParent(li, {tag: 'UL'}, body); if (ul) { var customBullitClass = this.editor.action.actions.insertUnorderedList.getCustomBullitClass(ul); if (customBullitClass) { setTimeout(function() { if (ul && li && li.innerHTML !== '') { _this.editor.action.actions.insertUnorderedList.checkCustomBullitList(ul, customBullitClass); } }, 0); } } } } if (!this.editor.util.FirstLetterSupported() && _this.editor.parser.firstNodeCheck && keyCode === this.editor.KEY_CODES['backspace']) { _this.editor.parser.FirstLetterBackspaceHandler(range); } // Handle "Enter" if (!e.shiftKey && (keyCode === KEY_CODES["enter"] || keyCode === KEY_CODES["backspace"])) { return this.OnEnterHandler(e, keyCode, selectedNode, range); } if (keyCode === KEY_CODES["pageUp"] || keyCode === KEY_CODES["pageDown"]) { this.savedScroll = BX.GetWindowScrollPos(document); BX.addCustomEvent(this.editor, "OnIframeKeyup", BX.proxy(this._RestoreScrollTop, this)); setTimeout(BX.proxy(this._RestoreScrollTop, this), 0); } }; BXEditorIframeView.prototype._RestoreScrollTop = function(e) { if (this.savedScroll) { window.scrollTo(this.savedScroll.scrollLeft, this.savedScroll.scrollTop); this.savedScroll = null; } BX.removeCustomEvent(this.editor, "OnIframeKeyup", BX.proxy(this._RestoreScrollTop, this)); }; BXEditorIframeView.prototype._IEBodyClearHandler = function(e) { var p = this.document.body.firstChild; if (e.keyCode == this.editor.KEY_CODES['enter'] && p.nodeName == "P" && p != this.document.body.lastChild) { if (p.innerHTML && p.innerHTML.toLowerCase() == '<br>') { var newPar = p.nextSibling; this.editor.util.ReplaceWithOwnChildren(p); p = newPar; } } if (p && p.nodeName == "P" && p == this.document.body.lastChild) { this.editor.util.ReplaceWithOwnChildren(p); } BX.removeCustomEvent(this.editor, "OnIframeKeyup", BX.proxy(this._IEBodyClearHandler, this)); }; BXEditorIframeView.prototype._IEBodyClearHandlerEx = function(e) { var p = this.document.body.firstChild; if (e.keyCode == this.editor.KEY_CODES['backspace'] && p && p.nodeName == "P" && p == this.document.body.lastChild && (this.editor.util.IsEmptyNode(p, true, true) || p.innerHTML && p.innerHTML.toLowerCase() == '<br>')) { this.editor.util.ReplaceWithOwnChildren(p); } BX.removeCustomEvent(this.editor, "OnIframeKeyup", BX.proxy(this._IEBodyClearHandlerEx, this)); }; BXEditorIframeView.prototype.OnEnterHandler = function(e, keyCode, selectedNode, range) { // TODO: check it again later maybe chrome will fix it // mantis: 55872. Chrome 38 rendering bug workaround if (BX.browser.IsChrome()) { this.document.body.style.minHeight = (parseInt(this.document.body.style.minHeight) + 1) + 'px'; // mantis: 60033 this.document.body.style.minHeight = (parseInt(this.document.body.style.minHeight) - 1) + 'px'; } // Check selectedNode if (!selectedNode) { return; } var _this = this; function unwrap(node) { if (node) { if (node.nodeName !== "P" && node.nodeName !== "DIV") { node = BX.findParent(node, function(n) { return n.nodeName === "P" || n.nodeName === "DIV"; }, _this.document.body); } var emptyNode = _this.editor.util.GetInvisibleTextNode(); if (node) { node.parentNode.insertBefore(emptyNode, node); _this.editor.util.ReplaceWithOwnChildren(node); _this.editor.selection.SelectNode(emptyNode); } } } var list, br, blockElement, blockTags = ["LI", "P", "H1", "H2", "H3", "H4", "H5", "H6"], listTags = ["UL", "OL", "MENU"]; if (BX.util.in_array(selectedNode.nodeName, blockTags)) { blockElement = selectedNode; } else { blockElement = BX.findParent(selectedNode, function(n) { return BX.util.in_array(n.nodeName, blockTags); }, this.document.body); } if (blockElement) { if (blockElement.nodeName === "LI") { if (keyCode === _this.editor.KEY_CODES["enter"] && blockElement && blockElement.parentNode) { var bullitClass = _this.editor.action.actions.insertUnorderedList.getCustomBullitClass(blockElement.parentNode); } // Some browsers create <p> elements after leaving a list // check after keydown of backspace and return whether a <p> got inserted and unwrap it setTimeout(function() { var node = _this.editor.selection.GetSelectedNode(true); if (node) { list = BX.findParent(node, function(n) { return BX.util.in_array(n.nodeName, listTags); }, _this.document.body); // Check if it's list with custom styled bullits - we have to check it all items have same style if (keyCode === _this.editor.KEY_CODES["enter"] && blockElement && blockElement.parentNode) { _this.editor.action.actions.insertUnorderedList.checkCustomBullitList(blockElement.parentNode, bullitClass, true); } // mantis: 82028, 83178 if (list && BX.browser.IsChrome() && keyCode === _this.editor.KEY_CODES["enter"]) { var i, li = list.getElementsByTagName('LI'); for (i = 0; i < li.length; i++) { //&65279; === \uFEFF - invisible space li[i].innerHTML = li[i].innerHTML.replace(/\uFEFF/ig, ''); } } if (!list) { unwrap(node); } } }, 0); } else if (blockElement.nodeName.match(/H[1-6]/) && keyCode === this.editor.KEY_CODES["enter"]) { setTimeout(function() { unwrap(_this.editor.selection.GetSelectedNode()); }, 0); } return true; } if (keyCode === this.editor.KEY_CODES["enter"] && !BX.browser.IsFirefox() && this.editor.action.IsSupported('insertLineBreak')) { if (BX.browser.IsIE10() || BX.browser.IsIE11()) { this.editor.action.Exec('insertHTML', '<br>' + this.editor.INVISIBLE_SPACE); } else if(BX.browser.IsChrome()) { this.editor.action.Exec('insertLineBreak'); // Bug in Chrome - when you press enter but it put carret on the prev string // Chrome 43.0.2357 in Mac puts visible space instead of invisible if (BX.browser.IsMac()) { var tmpId = "bx-editor-temp-" + Math.round(Math.random() * 1000000); this.editor.action.Exec('insertHTML', '<span id="' + tmpId + '">' + this.editor.INVISIBLE_SPACE + '</span>'); var tmpElement = this.editor.GetIframeElement(tmpId); if (tmpElement) BX.remove(tmpElement); } else { this.editor.action.Exec('insertHTML', this.editor.INVISIBLE_SPACE); } } else { this.editor.action.Exec('insertLineBreak'); } return BX.PreventDefault(e); } if ((BX.browser.IsChrome() || BX.browser.IsIE10() || BX.browser.IsIE11()) && keyCode == this.editor.KEY_CODES['backspace'] && range.collapsed) { var checkNode = BX.create('SPAN', false, this.document); this.editor.selection.InsertNode(checkNode); var prev = checkNode.previousSibling; if (prev && prev.nodeType == 3 && this.editor.util.IsEmptyNode(prev, false, false)) { BX.remove(prev); } this.editor.selection.SetBefore(checkNode); BX.remove(checkNode); } }; BXEditorIframeView.prototype.OnEnterHandlerKeyUp = function(e, keyCode, node) { // Clean class of all block nodes when they created after Enter pressing // All new Ps and DIVs should be without classNames if (node) { var _this = this; if (!BX.util.in_array(node.nodeName, this.editor.GetBlockTags())) { node = BX.findParent(node, function(n) { return BX.util.in_array(n.nodeName, _this.editor.GetBlockTags()); }, this.document.body); } if (node && BX.util.in_array(node.nodeName, this.editor.GetBlockTags())) { var html = BX.util.trim(node.innerHTML).toLowerCase(); if (this.editor.util.IsEmptyNode(node, true, true) || html == '' || html == '<br>') { node.removeAttribute("class"); } } } }; BXEditorIframeView.prototype.OnKeyDownArrowsHandler = function(e, keyCode, range) { var node, parentNode, nextNode, prevNode, KC = this.editor.KEY_CODES; this.keyDownRange = range; if (keyCode === KC['right'] || keyCode === KC['down']) { node = range.endContainer; nextNode = node ? node.nextSibling : false; parentNode = node ? node.parentNode : false; if ( node.nodeType == 3 && node.length == range.endOffset && parentNode && parentNode.nodeName !== 'BODY' && (!nextNode || (nextNode && nextNode.nodeName == 'BR' && !nextNode.nextSibling)) && (this.editor.util.IsBlockElement(parentNode) || this.editor.util.IsBlockNode(parentNode)) ) { this.editor.selection.SetInvisibleTextAfterNode(parentNode, true); return BX.PreventDefault(e); } else if( node.nodeType == 3 && this.editor.util.IsEmptyNode(node) && nextNode && (this.editor.util.IsBlockElement(nextNode) || this.editor.util.IsBlockNode(nextNode)) ) { BX.remove(node); if (nextNode.firstChild) { this.editor.selection.SetBefore(nextNode.firstChild); } else { this.editor.selection.SetAfter(nextNode); } return BX.PreventDefault(e); } else if ( node.nodeType == 3 && node.length == range.endOffset && parentNode && parentNode.nodeName !== 'BODY' && nextNode && nextNode.nodeName == 'BR' && !nextNode.nextSibling && (this.editor.util.IsBlockElement(parentNode) || this.editor.util.IsBlockNode(parentNode)) ) { this.editor.selection.SetInvisibleTextAfterNode(parentNode, true); return BX.PreventDefault(e); } } else if (keyCode === KC['left'] || keyCode === KC['up']) { node = range.startContainer; parentNode = node ? node.parentNode : false; prevNode = node ? node.previousSibling : false; if ( node.nodeType == 3 && range.endOffset === 0 && parentNode && parentNode.nodeName !== 'BODY' && !prevNode && (this.editor.util.IsBlockElement(parentNode) || this.editor.util.IsBlockNode(parentNode)) ) { this.editor.selection.SetInvisibleTextBeforeNode(parentNode); return BX.PreventDefault(e); } else if( node.nodeType == 3 && this.editor.util.IsEmptyNode(node) && prevNode && (this.editor.util.IsBlockElement(prevNode) || this.editor.util.IsBlockNode(prevNode)) ) { BX.remove(node); if (prevNode.lastChild) { this.editor.selection.SetAfter(prevNode.lastChild); } else { this.editor.selection.SetBefore(prevNode); } return BX.PreventDefault(e); } } return true; }; BXEditorIframeView.prototype.OnKeyUpArrowsHandler = function(e, keyCode) { var _this = this, pre, prevToSur, nextToSur, keyDownNode, keyDownPre, range = this.editor.selection.GetRange(), node, parentNode, nextNode, prevNode, isEmpty, isSur, sameLastRange, startCont, endCont, startIsSur, endIsSur, KC = this.editor.KEY_CODES; // Arrows right or down if (keyCode === KC['right'] || keyCode === KC['down']) { this.editor.selection.GetStructuralTags(); // Moving cursor by arrows (right & down) if (range.collapsed) { node = range.endContainer; isEmpty = this.editor.util.IsEmptyNode(node); // We check if last range was the same - it means that cursor doesn't // moved when user tried to move it sameLastRange = this.editor.selection.CheckLastRange(range); nextNode = node.nextSibling; if (!this.editor.util.CheckPreCursorSupport()) { if (node.nodeName === 'PRE') { pre = node; } else if (node.nodeType == 3) { pre = BX.findParent(node, {tag: 'PRE'}, this.element); } if(pre) { if (this.keyDownRange) { keyDownNode = this.keyDownRange.endContainer; keyDownPre = keyDownNode == pre ? pre : BX.findParent(keyDownNode, function(n){return n == pre;}, this.element); } _this.FocusPreElement(pre, false, keyDownPre ? null : 'start'); } } // If cursor in the invisible node - we take next node if (node.nodeType == 3 && isEmpty && nextNode) { node = nextNode; isEmpty = this.editor.util.IsEmptyNode(node); } isSur = this.editor.util.CheckSurrogateNode(node); // It's surrogate if (isSur) { nextToSur = node.nextSibling; if (nextToSur && nextToSur.nodeType == 3 && this.editor.util.IsEmptyNode(nextToSur)) this.editor.selection._MoveCursorAfterNode(nextToSur); else this.editor.selection._MoveCursorAfterNode(node); BX.PreventDefault(e); } // If it's element else if (node.nodeType == 1 && node.nodeName != "BODY" && !isEmpty) { if (sameLastRange) { this.editor.selection._MoveCursorAfterNode(node); BX.PreventDefault(e); } } else if (sameLastRange && node.nodeType == 3 && /*node.length == range.endOffset &&*/ !isEmpty) { parentNode = node.parentNode; if (parentNode && node === parentNode.lastChild && parentNode.nodeName != "BODY") { this.editor.selection._MoveCursorAfterNode(parentNode); } } else if (node.nodeType == 3 && node.parentNode) { parentNode = node.parentNode; prevNode = parentNode.previousSibling; // It's empty invisible node before block element which was put there by us. // So we should remove it. if ( (this.editor.util.IsBlockElement(parentNode) || this.editor.util.IsBlockNode(parentNode)) && prevNode && prevNode.nodeType == 3 && this.editor.util.IsEmptyNode(prevNode) ) { BX.remove(prevNode); } } } else // Selection Shift + Right & Shift + down { startCont = range.startContainer; endCont = range.endContainer; startIsSur = this.editor.util.CheckSurrogateNode(startCont); endIsSur = this.editor.util.CheckSurrogateNode(endCont); if (startIsSur) { prevToSur = startCont.previousSibling; if (prevToSur && prevToSur.nodeType == 3 && this.editor.util.IsEmptyNode(prevToSur)) range.setStartBefore(prevToSur); else range.setStartBefore(startCont); this.editor.selection.SetSelection(range); } if (endIsSur) { nextToSur = endCont.nextSibling; if (nextToSur && nextToSur.nodeType == 3 && this.editor.util.IsEmptyNode(nextToSur)) range.setEndAfter(nextToSur); else range.setEndAfter(endCont); this.editor.selection.SetSelection(range); } } } // Arrows left or up else if (keyCode === KC['left'] || keyCode === KC['up']) { this.editor.selection.GetStructuralTags(); // Moving cursor by arrows (left & up) if (range.collapsed) { node = range.startContainer; isEmpty = this.editor.util.IsEmptyNode(node); // We check if last range was the same - it means that cursor doesn't // moved when user tried to move it sameLastRange = this.editor.selection.CheckLastRange(range); // If cursor in the invisible node - we take next node if (node.nodeType == 3 && isEmpty && node.previousSibling) { node = node.previousSibling; isEmpty = this.editor.util.IsEmptyNode(node); } if (!this.editor.util.CheckPreCursorSupport()) { if (node.nodeName === 'PRE') { pre = node; } else if (node.nodeType == 3) { pre = BX.findParent(node, {tag: 'PRE'}, this.element); } if(pre) { if (this.keyDownRange) { keyDownNode = this.keyDownRange.startContainer; keyDownPre = keyDownNode == pre ? pre : BX.findParent(keyDownNode, function(n){return n == pre;}, this.element); } _this.FocusPreElement(pre, false, keyDownPre ? null : 'end'); } } isSur = this.editor.util.CheckSurrogateNode(node); // It's surrogate if (isSur) { prevToSur = node.previousSibling; if (prevToSur && prevToSur.nodeType == 3 && this.editor.util.IsEmptyNode(prevToSur)) this.editor.selection._MoveCursorBeforeNode(prevToSur); else this.editor.selection._MoveCursorBeforeNode(node); BX.PreventDefault(e); } // If it's element else if (node.nodeType == 1 && node.nodeName != "BODY" && !isEmpty) { if (sameLastRange) { this.editor.selection._MoveCursorBeforeNode(node); BX.PreventDefault(e); } } //else if (sameLastRange && node.nodeType == 3 && range.startOffset == 0 && !isEmpty) else if (sameLastRange && node.nodeType == 3 && !isEmpty) { parentNode = node.parentNode; if (parentNode && node === parentNode.firstChild && parentNode.nodeName != "BODY") { this.editor.selection._MoveCursorBeforeNode(parentNode); } } else if (node.nodeType == 3 && node.parentNode) { parentNode = node.parentNode; prevNode = parentNode.nextSibling; // It's empty invisible node after block element which was put there by us. // So we should remove it. if ( (this.editor.util.IsBlockElement(parentNode) || this.editor.util.IsBlockNode(parentNode)) && prevNode && prevNode.nodeType == 3 && this.editor.util.IsEmptyNode(prevNode) ) { BX.remove(prevNode); } } } else // Selection Shift + left & Shift + up { startCont = range.startContainer; endCont = range.endContainer; startIsSur = this.editor.util.CheckSurrogateNode(startCont); endIsSur = this.editor.util.CheckSurrogateNode(endCont); if (startIsSur) { prevToSur = startCont.previousSibling; if (prevToSur && prevToSur.nodeType == 3 && this.editor.util.IsEmptyNode(prevToSur)) range.setStartBefore(prevToSur); else range.setStartBefore(startCont); this.editor.selection.SetSelection(range); } if (endIsSur) { nextToSur = endCont.nextSibling; if (nextToSur && nextToSur.nodeType == 3 && this.editor.util.IsEmptyNode(nextToSur)) range.setEndAfter(nextToSur); else range.setEndAfter(endCont); this.SetSelection(range); } } } this.keyDownRange = null; }; BXEditorIframeView.prototype.FocusPreElement = function(preNode, timeout, mode) { var _this = this; if (this._focusPreElementTimeout) this._focusPreElementTimeout = clearTimeout(this._focusPreElementTimeout); if (timeout) { this._focusPreElementTimeout = setTimeout(function(){ _this.FocusPreElement(preNode, false, mode); }, 100); return; } BX.focus(preNode); if (mode == 'end' && preNode.lastChild) { this.editor.selection.SetAfter(preNode.lastChild); } else if (mode == 'start' && preNode.firstChild) { this.editor.selection.SetBefore(preNode.firstChild); } }; BXEditorIframeView.prototype.OnPasteHandler = function(e) { if (!this.editor.skipPasteHandler) { this.editor.skipPasteHandler = true; this.editor.pasteNodeIndex = {}; var originalScrollTop = document.documentElement.scrollTop || document.body.scrollTop, originalScrollLeft = document.documentElement.scrollLeft || document.body.scrollLeft, _this = this, arNodes = [], curNode, i, node, qnodes; function markGoodNode(n) { if (n && n.setAttribute) { var randValue = Math.round(Math.random() * 1000000); _this.editor.pasteNodeIndex[randValue] = true; n.setAttribute('data-bx-paste-flag', randValue); } } curNode = this.document.body; if (curNode) { qnodes = curNode.querySelectorAll("*"); for (i = 0; i < qnodes.length; i++) { if (qnodes[i].nodeType == 1 && qnodes[i].nodeName != 'BODY' && qnodes[i].nodeName != 'HEAD') { arNodes.push(qnodes[i]); } } for (i = 0; i < curNode.parentNode.childNodes.length; i++) { node = curNode.parentNode.childNodes[i]; if (node.nodeType == 1 && node.nodeName != 'BODY' && node.nodeName != 'HEAD') { arNodes.push(node); } } } for (i = 0; i < arNodes.length; i++) { markGoodNode(arNodes[i]); } var sync = this.editor.synchro.IsSyncOn(); if (sync) { this.editor.synchro.StopSync(); } if (this.editor.iframeView.pasteHandlerTimeout) { clearTimeout(this.editor.iframeView.pasteHandlerTimeout); } this.pasteHandlerTimeout = setTimeout(function() { _this.editor.SetCursorNode(); _this.editor.pasteHandleMode = true; _this.editor.bbParseContentMode = true; _this.editor.synchro.lastIframeValue = false; // Paste control: show menu after pasting content // to let user select weather insert rich content or plain text if (!_this.editor.skipPasteControl) { _this.editor.pasteControl.SaveIframeContent(_this.GetValue()); _this.editor.pasteControl.CheckAndShow(); } _this.editor.synchro.FromIframeToTextarea(true, true); _this.editor.pasteHandleMode = false; _this.editor.bbParseContentMode = false; _this.editor.synchro.lastTextareaValue = false; _this.editor.synchro.FromTextareaToIframe(true); _this.editor.RestoreCursor(); _this.editor.On("OnIframePaste"); _this.editor.On("OnIframeNewWord"); _this.editor.skipPasteHandler = false; if (sync) { _this.editor.synchro.StartSync(); } if (window.scrollTo) { window.scrollTo(originalScrollLeft, originalScrollTop); } }, 0); } }; BXEditorIframeView.prototype.InitAutoLinking = function() { var _this = this, editor = this.editor, nativeAutolinkCanBeDisabled = editor.action.IsSupportedByBrowser("autoUrlDetect"), nativeAutoLink = BX.browser.IsIE() || BX.browser.IsIE9() || BX.browser.IsIE10(); if (nativeAutolinkCanBeDisabled) editor.action.Exec("autoUrlDetect", false); if (editor.config.autoLink === false) return; // Init Autolink system var ignorableParents = {"CODE" : 1, "PRE" : 1, "A" : 1, "SCRIPT" : 1, "HEAD" : 1, "TITLE" : 1, "STYLE" : 1}, urlRegExp = /(((?:https?|ftp):\/\/|www\.)[^\s'"<]{3,500})/gi, emailRegExp = /[\.a-z0-9_\-]+@[\.a-z0-9_\-]+\.[\.a-z0-9_\-]+/gi, MAX_LENGTH = 100, BRACKETS = { ")": "(", "]": "[", "}": "{" }; this.editor.autolinkUrlRegExp = urlRegExp; this.editor.autolinkEmailRegExp = emailRegExp; function autoLinkHandler() { try { if (checkAutoLink()) { editor.selection.ExecuteAndRestore(function(startContainer, endContainer) { if (endContainer && endContainer.parentNode) autoLink(endContainer.parentNode); }); } } catch(e){} } function checkAutoLink() { var node, nodeValue, result = false, doc = editor.GetIframeDoc(), walker = doc.createTreeWalker( doc.body, NodeFilter.SHOW_TEXT, null, false ); while(node = walker.nextNode()) { nodeValue = node.nodeValue || ''; if ((nodeValue.match(emailRegExp) || nodeValue.match(urlRegExp)) && node.parentNode && node.parentNode.nodeName != 'A') { result = true; break; } } return result; } function autoLink(element) { if (element && !ignorableParents[element.nodeName]) { var ignorableParent = BX.findParent(element, function(node) { return !!ignorableParents[node.nodeName]; }, element.ownerDocument.body); if (ignorableParent) return element; if (element === element.ownerDocument.documentElement) element = element.ownerDocument.body; return parseNode(element); } } function convertUrlToLink(str) { str = BX.util.htmlspecialchars(str); return str.replace(urlRegExp, function(match, url) { var punctuation = (url.match(/([^\w\u0430-\u0456\u0451\/\-#](,?))$/i) || [])[1] || "", opening = BRACKETS[punctuation]; url = url.replace(/([^\w\u0430-\u0456\u0451\/\-#](,?))$/i, ""); if (url.split(opening).length > url.split(punctuation).length) { url = url + punctuation; punctuation = ""; } var realUrl = url, displayUrl = BX.util.htmlspecialchars(url); if (url.length > MAX_LENGTH) displayUrl = displayUrl.substr(0, MAX_LENGTH) + "..."; if (realUrl.substr(0, 4) === "www.") realUrl = "http://" + realUrl; BX.onCustomEvent(_this.editor, 'OnAfterUrlConvert', [realUrl]); return '<a href="' + realUrl + '">' + displayUrl + '</a>' + punctuation; }); } function convertEmailToLink(str) { str = BX.util.htmlspecialchars(str); return str.replace(emailRegExp, function(email) { var punctuation = (email.match(/([^\w\/\-](,?))$/i) || [])[1] || "", opening = BRACKETS[punctuation]; email = email.replace(/([^\w\/\-](,?))$/i, ""); if (email.split(opening).length > email.split(punctuation).length) { email = email + punctuation; punctuation = ""; } var realUrl = "mailto:" + email; return '<a href="' + realUrl + '">' + email + '</a>' + punctuation; }); } function getTmpDiv(doc) { var tmp = doc._bx_autolink_temp_div; if (!tmp) tmp = doc._bx_autolink_temp_div = doc.createElement("div"); return tmp; } function parseNode(element) { var res, parentNode, tmpDiv; if (element && !ignorableParents[element.nodeName]) { // Replaces the content of the text node by link if (element.nodeType === 3 && element.data.match(urlRegExp) && element.parentNode) { parentNode = element.parentNode; tmpDiv = getTmpDiv(parentNode.ownerDocument); tmpDiv.innerHTML = "<span></span>" + convertUrlToLink(element.data); tmpDiv.removeChild(tmpDiv.firstChild); while (tmpDiv.firstChild) parentNode.insertBefore(tmpDiv.firstChild, element); parentNode.removeChild(element); } else if (element.nodeType === 3 && element.data.match(emailRegExp) && element.parentNode) { parentNode = element.parentNode; tmpDiv = getTmpDiv(parentNode.ownerDocument); tmpDiv.innerHTML = "<span></span>" + convertEmailToLink(element.data); tmpDiv.removeChild(tmpDiv.firstChild); while (tmpDiv.firstChild) parentNode.insertBefore(tmpDiv.firstChild, element); parentNode.removeChild(element); } else if (element.nodeType === 1) { var childNodes = element.childNodes, i; for (i = 0; i < childNodes.length; i++) parseNode(childNodes[i]); res = element; } } return res; } if (!nativeAutoLink || (nativeAutoLink && nativeAutolinkCanBeDisabled)) { BX.addCustomEvent(editor, "OnIframeNewWord", function() { if (editor.autolinkTimeout) editor.autolinkTimeout = clearTimeout(editor.autolinkTimeout); editor.autolinkTimeout = setTimeout(autoLinkHandler, 500); }); BX.addCustomEvent(editor, "OnSubmit", function() { try { autoLink(editor.GetIframeDoc().body); } catch(e){} }); } var links = editor.sandbox.GetDocument().getElementsByTagName("a"), getTextContent = function(element) { var textContent = BX.util.trim(editor.util.GetTextContent(element)); if (textContent.substr(0, 4) === "www.") textContent = "http://" + textContent; return textContent; }; BX.addCustomEvent(editor, "OnIframeKeydown", function(e, keyCode, command, selectedNode) { if (links.length > 0 && selectedNode) { var link = BX.findParent(selectedNode, {tag: 'A'}, selectedNode.ownerDocument.body); if (link) { var textContent = getTextContent(link); setTimeout(function() { var newTextContent = getTextContent(link); if (newTextContent === textContent) return; // Only set href when new href looks like a valid url if (newTextContent.match(urlRegExp)) link.setAttribute("href", newTextContent); }, 0); } } }); }; BXEditorIframeView.prototype.IsUserTypingNow = function(e) { return this.isFocused && this.isShown && this.isUserTyping; }; BXEditorIframeView.prototype.CheckContentLastChild = function(element) { if (!element) { element = this.element; } var lastChild = element.lastChild; if (lastChild && (this.editor.util.IsEmptyNode(lastChild, true) && this.editor.util.IsBlockNode(lastChild.previousSibling) || this.editor.phpParser.IsSurrogate(lastChild))) { element.appendChild(BX.create('BR', {}, element.ownerDocument)); element.appendChild(this.editor.util.GetInvisibleTextNode()); } }; /** * Class _this takes care that the value of the composer and the textarea is always in sync */ function BXEditorViewsSynchro(editor, textareaView, iframeView) { this.INTERVAL = 500; this.editor = editor; this.textareaView = textareaView; this.iframeView = iframeView; this.lastFocused = 'wysiwyg'; this.InitEventHandlers(); } /** * Sync html from composer to textarea * Takes care of placeholders * @param {Boolean} bParseHtml Whether the html should be sanitized before inserting it into the textarea */ BXEditorViewsSynchro.prototype = { FromIframeToTextarea: function(bParseHtml, bFormat) { var value; if (this.editor.bbCode) { value = this.iframeView.GetValue(this.editor.bbParseContentMode, false); value = BX.util.trim(value); if (value !== this.lastIframeValue) { var bbCodes = this.editor.bbParser.Unparse(value); this.textareaView.SetValue(bbCodes, false, bFormat || this.editor.bbParseContentMode); if (typeof this.lastSavedIframeValue !== 'undefined' && this.lastSavedIframeValue != value) { this.editor.On("OnContentChanged", [bbCodes, value]); } this.lastSavedIframeValue = value; this.lastIframeValue = value; } } else { value = this.iframeView.GetValue(); value = BX.util.trim(value); if (value !== this.lastIframeValue) { this.textareaView.SetValue(value, true, bFormat); if (typeof this.lastSavedIframeValue !== 'undefined' && this.lastSavedIframeValue != value) { this.editor.On("OnContentChanged", [this.textareaView.GetValue() || '', value || '']); } this.lastSavedIframeValue = value; this.lastIframeValue = value; } } }, /** * Sync value of textarea to composer * Takes care of placeholders * @param {Boolean} bParseHtml Whether the html should be sanitized before inserting it into the composer */ FromTextareaToIframe: function(bParseHtml) { var value = this.textareaView.GetValue(); if (value !== this.lastTextareaValue) { if (value) { if (this.editor.bbCode) { var htmlFromBbCode = this.editor.bbParser.Parse(value); // INVISIBLE_CURSOR htmlFromBbCode = htmlFromBbCode.replace(/\u2060/ig, '<span id="bx-cursor-node"> </span>'); this.iframeView.SetValue(htmlFromBbCode, bParseHtml); } else { // INVISIBLE_CURSOR value = value.replace(/\u2060/ig, '<span id="bx-cursor-node"> </span>'); this.iframeView.SetValue(value, bParseHtml); } } else { this.iframeView.Clear(); } this.lastTextareaValue = value; this.editor.On("OnContentChanged", [value || '', this.iframeView.GetValue() || '']); } }, FullSyncFromIframe: function() { this.lastIframeValue = false; this.FromIframeToTextarea(true, true); this.lastTextareaValue = false; this.FromTextareaToIframe(true); }, Sync: function() { var bParseHtml = true; var view = this.editor.currentViewName; if (view === "split") { if (this.GetSplitMode() === "code") { this.FromTextareaToIframe(bParseHtml); } else // wysiwyg { this.FromIframeToTextarea(bParseHtml); } } else if (view === "code") { this.FromTextareaToIframe(bParseHtml); } else // wysiwyg { this.FromIframeToTextarea(bParseHtml); } }, GetSplitMode: function() { var mode = false; if (this.editor.currentViewName == "split") { if (this.editor.iframeView.IsFocused()) { mode = "wysiwyg"; } else if(this.editor.textareaView.IsFocused()) { mode = "code"; } else { mode = this.lastFocused; } } return mode; }, InitEventHandlers: function() { var _this = this; BX.addCustomEvent(this.editor, "OnTextareaFocus", function() { _this.lastFocused = 'code'; _this.StartSync(); }); BX.addCustomEvent(this.editor, "OnIframeFocus", function() { _this.lastFocused = 'wysiwyg'; _this.StartSync(); }); BX.addCustomEvent(this.editor, "OnTextareaBlur", BX.delegate(this.StopSync, this)); BX.addCustomEvent(this.editor, "OnIframeBlur", BX.delegate(this.StopSync, this)); }, StartSync: function(delay) { var _this = this; if (this.interval) { this.interval = clearTimeout(this.interval); } this.delay = delay || this.INTERVAL; // it can reduce or increase initial timeout function sync() { // set delay to normal value _this.delay = _this.INTERVAL; _this.Sync(); _this.interval = setTimeout(sync, _this.delay); } this.interval = setTimeout(sync, _this.delay); }, StopSync: function() { if (this.interval) { this.interval = clearTimeout(this.interval); } }, IsSyncOn: function() { return !!this.interval; }, OnIframeMousedown: function(e, target, bxTag) { }, IsFocusedOnTextarea: function() { return this.editor.currentViewName === "code" || this.editor.currentViewName === "split" && this.GetSplitMode() === "code"; } } // global interface window.BXEditorTextareaView = BXEditorTextareaView; window.BXEditorIframeView = BXEditorIframeView; window.BXEditorViewsSynchro = BXEditorViewsSynchro; })();