From 07b68c98b3bb61e6a6efb369a76f44a16b278517 Mon Sep 17 00:00:00 2001 From: Jason Chen Date: Thu, 28 Nov 2024 22:09:57 -0800 Subject: [PATCH 1/4] Handle consecutive spaces when copying and pasting Co-authored-by: Zihua Li <635902+luin@users.noreply.github.com> --- packages/quill/src/core/editor.ts | 3 +- packages/quill/src/modules/clipboard.ts | 20 +++++++------ packages/quill/test/unit/core/editor.spec.ts | 28 +++++++++++++++++-- .../quill/test/unit/modules/clipboard.spec.ts | 24 +++++++++++----- packages/quill/tsconfig.json | 2 +- tsconfig.json | 2 +- 6 files changed, 58 insertions(+), 21 deletions(-) diff --git a/packages/quill/src/core/editor.ts b/packages/quill/src/core/editor.ts index a19485840d..b6391a7cd4 100644 --- a/packages/quill/src/core/editor.ts +++ b/packages/quill/src/core/editor.ts @@ -370,7 +370,8 @@ function convertHTML( return blot.html(index, length); } if (blot instanceof TextBlot) { - return escapeText(blot.value().slice(index, index + length)); + const escapedText = escapeText(blot.value().slice(index, index + length)); + return escapedText.replaceAll(' ', ' '); } if (blot instanceof ParentBlot) { // TODO fix API diff --git a/packages/quill/src/modules/clipboard.ts b/packages/quill/src/modules/clipboard.ts index e4c3f755b5..78ea670839 100644 --- a/packages/quill/src/modules/clipboard.ts +++ b/packages/quill/src/modules/clipboard.ts @@ -626,7 +626,7 @@ function matchTable( function matchText(node: HTMLElement, delta: Delta, scroll: ScrollBlot) { // @ts-expect-error - let text = node.data; + let text = node.data as string; // Word represents empty line with   if (node.parentElement?.tagName === 'O:P') { return delta.insert(text.trim()); @@ -639,12 +639,10 @@ function matchText(node: HTMLElement, delta: Delta, scroll: ScrollBlot) { ) { return delta; } - const replacer = (collapse: unknown, match: string) => { - const replaced = match.replace(/[^\u00a0]/g, ''); // \u00a0 is nbsp; - return replaced.length < 1 && collapse ? ' ' : replaced; - }; - text = text.replace(/\r\n/g, ' ').replace(/\n/g, ' '); - text = text.replace(/\s\s+/g, replacer.bind(replacer, true)); // collapse whitespace + // convert all non-nbsp whitespace into regular space + text = text.replace(/[^\S\u00a0]/g, ' '); + // collapse consecutive spaces into one + text = text.replace(/ {2,}/g, ' '); if ( (node.previousSibling == null && node.parentElement != null && @@ -652,7 +650,8 @@ function matchText(node: HTMLElement, delta: Delta, scroll: ScrollBlot) { (node.previousSibling instanceof Element && isLine(node.previousSibling, scroll)) ) { - text = text.replace(/^\s+/, replacer.bind(replacer, false)); + // block structure means we don't need leading space + text = text.replace(/^ /, ''); } if ( (node.nextSibling == null && @@ -660,8 +659,11 @@ function matchText(node: HTMLElement, delta: Delta, scroll: ScrollBlot) { isLine(node.parentElement, scroll)) || (node.nextSibling instanceof Element && isLine(node.nextSibling, scroll)) ) { - text = text.replace(/\s+$/, replacer.bind(replacer, false)); + // block structure means we don't need trailing space + text = text.replace(/ $/, ''); } + // done removing whitespace and can normalize all to regular space + text = text.replaceAll('\u00a0', ' '); } return delta.insert(text); } diff --git a/packages/quill/test/unit/core/editor.spec.ts b/packages/quill/test/unit/core/editor.spec.ts index 0c595332bb..2d47a1dec9 100644 --- a/packages/quill/test/unit/core/editor.spec.ts +++ b/packages/quill/test/unit/core/editor.spec.ts @@ -28,9 +28,11 @@ import { ColorClass } from '../../../src/formats/color.js'; import Quill from '../../../src/core.js'; import { normalizeHTML } from '../__helpers__/utils.js'; -const createEditor = (html: string) => { +const createEditor = (htmlOrContents: string | Delta) => { const container = document.createElement('div'); - container.innerHTML = normalizeHTML(html); + if (typeof htmlOrContents === 'string') { + container.innerHTML = normalizeHTML(htmlOrContents); + } document.body.appendChild(container); const quill = new Quill(container, { registry: createRegistry([ @@ -54,6 +56,9 @@ const createEditor = (html: string) => { SizeClass, ]), }); + if (typeof htmlOrContents !== 'string') { + quill.setContents(htmlOrContents); + } return quill.editor; }; @@ -1246,6 +1251,25 @@ describe('Editor', () => { ); }); + test('collapsible spaces', () => { + expect( + createEditor('

123 123 123

').getHTML( + 0, + 11, + ), + ).toEqual('123 123 123'); + + expect(createEditor(new Delta().insert('1 2\n')).getHTML(0, 5)).toEqual( + '1   2', + ); + + expect( + createEditor( + new Delta().insert(' 123', { bold: true }).insert('\n'), + ).getHTML(0, 5), + ).toEqual('  123'); + }); + test('mixed list', () => { const editor = createEditor( ` diff --git a/packages/quill/test/unit/modules/clipboard.spec.ts b/packages/quill/test/unit/modules/clipboard.spec.ts index 0ba7c159ed..69142690a2 100644 --- a/packages/quill/test/unit/modules/clipboard.spec.ts +++ b/packages/quill/test/unit/modules/clipboard.spec.ts @@ -244,6 +244,12 @@ describe('Clipboard', () => { expect(delta).toEqual(new Delta().insert('0\n1 2 3 4\n5 6 7 8')); }); + test('multiple whitespaces', () => { + const html = '
1 2 3
'; + const delta = createClipboard().convert({ html }); + expect(delta).toEqual(new Delta().insert('1 2 3')); + }); + test('inline whitespace', () => { const html = '

0 1 2

'; const delta = createClipboard().convert({ html }); @@ -256,19 +262,23 @@ describe('Clipboard', () => { const html = '1 2'; const delta = createClipboard().convert({ html }); expect(delta).toEqual( - new Delta() - .insert('0\u00a0') - .insert('1', { bold: true }) - .insert('\u00a02'), + new Delta().insert('0 ').insert('1', { bold: true }).insert(' 2'), ); }); test('consecutive intentional whitespace', () => { const html = '  1  '; const delta = createClipboard().convert({ html }); - expect(delta).toEqual( - new Delta().insert('\u00a0\u00a01\u00a0\u00a0', { bold: true }), - ); + expect(delta).toEqual(new Delta().insert(' 1 ', { bold: true })); + }); + + test('intentional whitespace at line start/end', () => { + expect( + createClipboard().convert({ html: '

0  

  2

' }), + ).toEqual(new Delta().insert('0 \n 2')); + expect( + createClipboard().convert({ html: '

 2

' }), + ).toEqual(new Delta().insert('0 \n 2')); }); test('newlines between inline elements', () => { diff --git a/packages/quill/tsconfig.json b/packages/quill/tsconfig.json index ad105b7c1e..062af2c011 100644 --- a/packages/quill/tsconfig.json +++ b/packages/quill/tsconfig.json @@ -7,7 +7,7 @@ "compilerOptions": { "outDir": "./dist", "allowSyntheticDefaultImports": true, - "target": "ES2020", + "target": "ES2021", "sourceMap": true, "resolveJsonModule": true, "declaration": false, diff --git a/tsconfig.json b/tsconfig.json index 87860003ec..146b141a1f 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -7,7 +7,7 @@ }, "compilerOptions": { "allowSyntheticDefaultImports": true, - "target": "ES2020", + "target": "ES2021", "sourceMap": true, "declaration": true, "module": "ES2020", From 910ab4ed63ecff295559247b39dd69537a7e816a Mon Sep 17 00:00:00 2001 From: Zihua Li Date: Fri, 29 Nov 2024 21:59:37 +0800 Subject: [PATCH 2/4] Config Git correctly for releasing (#4506) Co-authored-by: Zihua Li <635902+luin@users.noreply.github.com> --- scripts/release.js | 1 + 1 file changed, 1 insertion(+) diff --git a/scripts/release.js b/scripts/release.js index d53057b7d8..ad4cc9fb73 100755 --- a/scripts/release.js +++ b/scripts/release.js @@ -31,6 +31,7 @@ if (!process.env.CI) { exec('echo "//registry.npmjs.org/:_authToken=${NPM_TOKEN}" > ~/.npmrc'); async function main() { + const configGit = (await import("./utils/configGit.mjs")).default; await configGit(); /* From 620ca42ee242f030462ea16553b9ed2b17e34b6e Mon Sep 17 00:00:00 2001 From: Zihua Li <635902+luin@users.noreply.github.com> Date: Sat, 30 Nov 2024 20:10:21 +0800 Subject: [PATCH 3/4] Add writable permission for GitHub Actions --- .github/workflows/release.yml | 3 +++ 1 file changed, 3 insertions(+) diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml index e3379fc50f..c08b91f799 100644 --- a/.github/workflows/release.yml +++ b/.github/workflows/release.yml @@ -13,6 +13,9 @@ on: default: true required: true +permissions: + contents: write + jobs: test: uses: ./.github/workflows/_test.yml From ebe16ca24724ac4f52505628ac2c4934f0a98b85 Mon Sep 17 00:00:00 2001 From: Zihua Li <635902+luin@users.noreply.github.com> Date: Sat, 30 Nov 2024 12:19:20 +0000 Subject: [PATCH 4/4] 2.0.3 --- package-lock.json | 8 ++++---- package.json | 2 +- packages/quill/package.json | 2 +- packages/website/package.json | 2 +- 4 files changed, 7 insertions(+), 7 deletions(-) diff --git a/package-lock.json b/package-lock.json index 237dfd0ea1..23085e2961 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,12 +1,12 @@ { "name": "quill-monorepo", - "version": "2.0.2", + "version": "2.0.3", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "quill-monorepo", - "version": "2.0.2", + "version": "2.0.3", "license": "BSD-3-Clause", "workspaces": [ "packages/*" @@ -19236,7 +19236,7 @@ } }, "packages/quill": { - "version": "2.0.2", + "version": "2.0.3", "license": "BSD-3-Clause", "dependencies": { "eventemitter3": "^5.0.1", @@ -19361,7 +19361,7 @@ } }, "packages/website": { - "version": "2.0.2", + "version": "2.0.3", "license": "BSD-3-Clause", "dependencies": { "@codesandbox/sandpack-react": "^2.11.3", diff --git a/package.json b/package.json index 13aec42cde..275b2260ac 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "quill-monorepo", - "version": "2.0.2", + "version": "2.0.3", "description": "Quill development environment", "private": true, "author": "Jason Chen ", diff --git a/packages/quill/package.json b/packages/quill/package.json index 1ce5f24dc5..4dd6a2e662 100644 --- a/packages/quill/package.json +++ b/packages/quill/package.json @@ -1,6 +1,6 @@ { "name": "quill", - "version": "2.0.2", + "version": "2.0.3", "description": "Your powerful, rich text editor", "author": "Jason Chen ", "homepage": "https://quilljs.com", diff --git a/packages/website/package.json b/packages/website/package.json index c089961461..59e235b6b9 100644 --- a/packages/website/package.json +++ b/packages/website/package.json @@ -1,6 +1,6 @@ { "name": "website", - "version": "2.0.2", + "version": "2.0.3", "description": "Quill official website", "private": true, "homepage": "https://quilljs.com",