From d7f5a22a5bfba184e191a6703454750f80299063 Mon Sep 17 00:00:00 2001 From: silverwind Date: Sun, 3 Mar 2024 21:29:19 +0100 Subject: [PATCH 1/4] Support pasting URLs over markdown text --- .../js/features/comp/ComboMarkdownEditor.js | 13 ++++- .../features/comp/{ImagePaste.js => Paste.js} | 57 +++++++++++-------- web_src/js/utils/dom.js | 39 +++++++++++++ web_src/js/utils/url.js | 12 ++++ web_src/js/utils/url.test.js | 9 ++- 5 files changed, 103 insertions(+), 27 deletions(-) rename web_src/js/features/comp/{ImagePaste.js => Paste.js} (74%) diff --git a/web_src/js/features/comp/ComboMarkdownEditor.js b/web_src/js/features/comp/ComboMarkdownEditor.js index 4c973358e34b3..46ccc09836017 100644 --- a/web_src/js/features/comp/ComboMarkdownEditor.js +++ b/web_src/js/features/comp/ComboMarkdownEditor.js @@ -3,7 +3,7 @@ import '@github/text-expander-element'; import $ from 'jquery'; import {attachTribute} from '../tribute.js'; import {hideElem, showElem, autosize, isElemVisible} from '../../utils/dom.js'; -import {initEasyMDEImagePaste, initTextareaImagePaste} from './ImagePaste.js'; +import {initEasyMDEImagePaste, initTextareaImagePaste} from './Paste.js'; import {handleGlobalEnterQuickSubmit} from './QuickSubmit.js'; import {renderPreviewPanelContent} from '../repo-editor.js'; import {easyMDEToolbarActions} from './EasyMDEToolbarActions.js'; @@ -84,6 +84,17 @@ class ComboMarkdownEditor { if (el.nodeName === 'BUTTON' && !el.getAttribute('type')) el.setAttribute('type', 'button'); } + this.textarea.addEventListener('keydown', (e) => { + if (e.shiftKey) { + e.target._giteaShiftDown = true; + } + }); + this.textarea.addEventListener('keyup', (e) => { + if (!e.shiftKey) { + e.target._giteaShiftDown = false; + } + }); + const monospaceButton = this.container.querySelector('.markdown-switch-monospace'); const monospaceEnabled = localStorage?.getItem('markdown-editor-monospace') === 'true'; const monospaceText = monospaceButton.getAttribute(monospaceEnabled ? 'data-disable-text' : 'data-enable-text'); diff --git a/web_src/js/features/comp/ImagePaste.js b/web_src/js/features/comp/Paste.js similarity index 74% rename from web_src/js/features/comp/ImagePaste.js rename to web_src/js/features/comp/Paste.js index b727880bc851c..f6359b4c62f42 100644 --- a/web_src/js/features/comp/ImagePaste.js +++ b/web_src/js/features/comp/Paste.js @@ -1,6 +1,8 @@ import {htmlEscape} from 'escape-goat'; import {POST} from '../../modules/fetch.js'; import {imageInfo} from '../../utils/image.js'; +import {getPastedContent, replaceTextareaSelection} from '../../utils/dom.js'; +import {isUrl} from '../../utils/url.js'; async function uploadFile(file, uploadUrl) { const formData = new FormData(); @@ -10,17 +12,6 @@ async function uploadFile(file, uploadUrl) { return await res.json(); } -function clipboardPastedImages(e) { - if (!e.clipboardData) return []; - - const files = []; - for (const item of e.clipboardData.items || []) { - if (!item.type || !item.type.startsWith('image/')) continue; - files.push(item.getAsFile()); - } - return files; -} - function triggerEditorContentChanged(target) { target.dispatchEvent(new CustomEvent('ce-editor-content-changed', {bubbles: true})); } @@ -91,20 +82,16 @@ class CodeMirrorEditor { } } -const uploadClipboardImage = async (editor, dropzone, e) => { +async function handleClipboardImages(editor, dropzone, images, e) { const uploadUrl = dropzone.getAttribute('data-upload-url'); const filesContainer = dropzone.querySelector('.files'); - if (!uploadUrl || !filesContainer) return; + if (!dropzone || !uploadUrl || !filesContainer || !images.length) return; - const pastedImages = clipboardPastedImages(e); - if (!pastedImages || pastedImages.length === 0) { - return; - } e.preventDefault(); e.stopPropagation(); - for (const img of pastedImages) { + for (const img of images) { const name = img.name.slice(0, img.name.lastIndexOf('.')); const placeholder = `![${name}](uploading ...)`; @@ -131,18 +118,38 @@ const uploadClipboardImage = async (editor, dropzone, e) => { input.value = uuid; filesContainer.append(input); } -}; +} + +function handleClipboardText(textarea, text, e) { + // when pasting links over selected text, turn it into [text](link), except when shift key is held + const {value, selectionStart, selectionEnd, _giteaShiftDown} = textarea; + if (_giteaShiftDown) return; + const selectedText = value.substring(selectionStart, selectionEnd); + const trimmedText = text.trim(); + if (selectedText && isUrl(trimmedText)) { + e.stopPropagation(); + e.preventDefault(); + replaceTextareaSelection(textarea, `[${selectedText}](${trimmedText})`); + } +} export function initEasyMDEImagePaste(easyMDE, dropzone) { - if (!dropzone) return; - easyMDE.codemirror.on('paste', async (_, e) => { - return uploadClipboardImage(new CodeMirrorEditor(easyMDE.codemirror), dropzone, e); + easyMDE.codemirror.on('paste', (_, e) => { + const {images} = getPastedContent(e); + if (images.length) { + handleClipboardImages(new CodeMirrorEditor(easyMDE.codemirror), dropzone, images, e); + } }); } export function initTextareaImagePaste(textarea, dropzone) { - if (!dropzone) return; - textarea.addEventListener('paste', async (e) => { - return uploadClipboardImage(new TextareaEditor(textarea), dropzone, e); + textarea.addEventListener('paste', (e) => { + const {images, text} = getPastedContent(e); + if (text) { + handleClipboardText(textarea, text, e); + } + if (images.length) { + handleClipboardImages(new TextareaEditor(textarea), dropzone, images, e); + } }); } diff --git a/web_src/js/utils/dom.js b/web_src/js/utils/dom.js index 91535dc187595..b242463cab6fa 100644 --- a/web_src/js/utils/dom.js +++ b/web_src/js/utils/dom.js @@ -243,3 +243,42 @@ export function isElemVisible(element) { return Boolean(element.offsetWidth || element.offsetHeight || element.getClientRects().length); } + +// extract text and images from "paste" event +export function getPastedContent(e) { + if (!e.clipboardData) return {text: '', images: []}; + + const images = []; + for (const item of e.clipboardData.items || []) { + if (item.type?.startsWith('image/')) { + images.push(item.getAsFile()); + } + } + + const text = e.clipboardData.getData('text'); + return {text, images}; +} + +// replace selected text in a textarea while preserving editor history, e.g. CTRL-Z works after this +export function replaceTextareaSelection(textarea, text) { + const before = textarea.value.slice(0, textarea.selectionStart ?? undefined); + const after = textarea.value.slice(textarea.selectionEnd ?? undefined); + let success = true; + + textarea.contentEditable = 'true'; + try { + success = document.execCommand('insertText', false, text); + } catch { + success = false; + } + textarea.contentEditable = 'false'; + + if (success && !textarea.value.slice(0, textarea.selectionStart ?? undefined).endsWith(text)) { + success = false; + } + + if (!success) { + textarea.value = `${before}${text}${after}`; + textarea.dispatchEvent(new CustomEvent('change', {bubbles: true, cancelable: true})); + } +} diff --git a/web_src/js/utils/url.js b/web_src/js/utils/url.js index a40737ca6f5a2..470ece31b0ed8 100644 --- a/web_src/js/utils/url.js +++ b/web_src/js/utils/url.js @@ -1,3 +1,15 @@ export function pathEscapeSegments(s) { return s.split('/').map(encodeURIComponent).join('/'); } + +function stripSlash(url) { + return url.endsWith('/') ? url.slice(0, -1) : url; +} + +export function isUrl(url) { + try { + return stripSlash((new URL(url).href)).trim() === stripSlash(url).trim(); + } catch { + return false; + } +} diff --git a/web_src/js/utils/url.test.js b/web_src/js/utils/url.test.js index 3dbedec94f1e0..08c6373ffb002 100644 --- a/web_src/js/utils/url.test.js +++ b/web_src/js/utils/url.test.js @@ -1,6 +1,13 @@ -import {pathEscapeSegments} from './url.js'; +import {pathEscapeSegments, isUrl} from './url.js'; test('pathEscapeSegments', () => { expect(pathEscapeSegments('a/b/c')).toEqual('a/b/c'); expect(pathEscapeSegments('a/b/ c')).toEqual('a/b/%20c'); }); + +test('isUrl', () => { + expect(isUrl('https://example.com')).toEqual(true); + expect(isUrl('https://example.com/')).toEqual(true); + expect(isUrl('https://example.com/index.html')).toEqual(true); + expect(isUrl('/index.html')).toEqual(false); +}); From 8230f6e53ec88018a20a27b455caf1b3b3f13f13 Mon Sep 17 00:00:00 2001 From: silverwind Date: Mon, 4 Mar 2024 00:20:45 +0100 Subject: [PATCH 2/4] accept images or text, tweak function --- web_src/js/features/comp/Paste.js | 5 ++--- web_src/js/utils/dom.js | 7 ++----- 2 files changed, 4 insertions(+), 8 deletions(-) diff --git a/web_src/js/features/comp/Paste.js b/web_src/js/features/comp/Paste.js index f6359b4c62f42..be7c7670a1e35 100644 --- a/web_src/js/features/comp/Paste.js +++ b/web_src/js/features/comp/Paste.js @@ -145,11 +145,10 @@ export function initEasyMDEImagePaste(easyMDE, dropzone) { export function initTextareaImagePaste(textarea, dropzone) { textarea.addEventListener('paste', (e) => { const {images, text} = getPastedContent(e); - if (text) { - handleClipboardText(textarea, text, e); - } if (images.length) { handleClipboardImages(new TextareaEditor(textarea), dropzone, images, e); + } else if (text) { + handleClipboardText(textarea, text, e); } }); } diff --git a/web_src/js/utils/dom.js b/web_src/js/utils/dom.js index b242463cab6fa..aa7c2604aa7a0 100644 --- a/web_src/js/utils/dom.js +++ b/web_src/js/utils/dom.js @@ -246,16 +246,13 @@ export function isElemVisible(element) { // extract text and images from "paste" event export function getPastedContent(e) { - if (!e.clipboardData) return {text: '', images: []}; - const images = []; - for (const item of e.clipboardData.items || []) { + for (const item of e.clipboardData?.items ?? []) { if (item.type?.startsWith('image/')) { images.push(item.getAsFile()); } } - - const text = e.clipboardData.getData('text'); + const text = e.clipboardData?.getData?.('text') ?? ''; return {text, images}; } From 0fe75ec3a24176c462425711da18007f90d56061 Mon Sep 17 00:00:00 2001 From: silverwind Date: Mon, 4 Mar 2024 02:23:36 +0100 Subject: [PATCH 3/4] rename shift property --- web_src/js/features/comp/ComboMarkdownEditor.js | 4 ++-- web_src/js/features/comp/Paste.js | 4 ++-- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/web_src/js/features/comp/ComboMarkdownEditor.js b/web_src/js/features/comp/ComboMarkdownEditor.js index 46ccc09836017..01ef95156261d 100644 --- a/web_src/js/features/comp/ComboMarkdownEditor.js +++ b/web_src/js/features/comp/ComboMarkdownEditor.js @@ -86,12 +86,12 @@ class ComboMarkdownEditor { this.textarea.addEventListener('keydown', (e) => { if (e.shiftKey) { - e.target._giteaShiftDown = true; + e.target._shiftDown = true; } }); this.textarea.addEventListener('keyup', (e) => { if (!e.shiftKey) { - e.target._giteaShiftDown = false; + e.target._shiftDown = false; } }); diff --git a/web_src/js/features/comp/Paste.js b/web_src/js/features/comp/Paste.js index be7c7670a1e35..4ee13be87e591 100644 --- a/web_src/js/features/comp/Paste.js +++ b/web_src/js/features/comp/Paste.js @@ -122,8 +122,8 @@ async function handleClipboardImages(editor, dropzone, images, e) { function handleClipboardText(textarea, text, e) { // when pasting links over selected text, turn it into [text](link), except when shift key is held - const {value, selectionStart, selectionEnd, _giteaShiftDown} = textarea; - if (_giteaShiftDown) return; + const {value, selectionStart, selectionEnd, _shiftDown} = textarea; + if (_shiftDown) return; const selectedText = value.substring(selectionStart, selectionEnd); const trimmedText = text.trim(); if (selectedText && isUrl(trimmedText)) { From 4a41430b68cabda0c262bf7ff616c10d1e1a25f3 Mon Sep 17 00:00:00 2001 From: silverwind Date: Fri, 8 Mar 2024 00:25:37 +0100 Subject: [PATCH 4/4] rename functions --- web_src/js/features/comp/ComboMarkdownEditor.js | 6 +++--- web_src/js/features/comp/Paste.js | 4 ++-- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/web_src/js/features/comp/ComboMarkdownEditor.js b/web_src/js/features/comp/ComboMarkdownEditor.js index 01ef95156261d..1e7b554b98a21 100644 --- a/web_src/js/features/comp/ComboMarkdownEditor.js +++ b/web_src/js/features/comp/ComboMarkdownEditor.js @@ -3,7 +3,7 @@ import '@github/text-expander-element'; import $ from 'jquery'; import {attachTribute} from '../tribute.js'; import {hideElem, showElem, autosize, isElemVisible} from '../../utils/dom.js'; -import {initEasyMDEImagePaste, initTextareaImagePaste} from './Paste.js'; +import {initEasyMDEPaste, initTextareaPaste} from './Paste.js'; import {handleGlobalEnterQuickSubmit} from './QuickSubmit.js'; import {renderPreviewPanelContent} from '../repo-editor.js'; import {easyMDEToolbarActions} from './EasyMDEToolbarActions.js'; @@ -119,7 +119,7 @@ class ComboMarkdownEditor { }); if (this.dropzone) { - initTextareaImagePaste(this.textarea, this.dropzone); + initTextareaPaste(this.textarea, this.dropzone); } } @@ -252,7 +252,7 @@ class ComboMarkdownEditor { }); this.applyEditorHeights(this.container.querySelector('.CodeMirror-scroll'), this.options.editorHeights); await attachTribute(this.easyMDE.codemirror.getInputField(), {mentions: true, emoji: true}); - initEasyMDEImagePaste(this.easyMDE, this.dropzone); + initEasyMDEPaste(this.easyMDE, this.dropzone); hideElem(this.textareaMarkdownToolbar); } diff --git a/web_src/js/features/comp/Paste.js b/web_src/js/features/comp/Paste.js index 4ee13be87e591..b26296d1fc967 100644 --- a/web_src/js/features/comp/Paste.js +++ b/web_src/js/features/comp/Paste.js @@ -133,7 +133,7 @@ function handleClipboardText(textarea, text, e) { } } -export function initEasyMDEImagePaste(easyMDE, dropzone) { +export function initEasyMDEPaste(easyMDE, dropzone) { easyMDE.codemirror.on('paste', (_, e) => { const {images} = getPastedContent(e); if (images.length) { @@ -142,7 +142,7 @@ export function initEasyMDEImagePaste(easyMDE, dropzone) { }); } -export function initTextareaImagePaste(textarea, dropzone) { +export function initTextareaPaste(textarea, dropzone) { textarea.addEventListener('paste', (e) => { const {images, text} = getPastedContent(e); if (images.length) {