From 5f0d8c63140d61d602a188f1bbae04a3c6226c44 Mon Sep 17 00:00:00 2001 From: Vaadin Bot Date: Thu, 3 Oct 2024 14:34:28 +0200 Subject: [PATCH] fix: delay parsing htmlValue when RTE is attached but not rendered (#7890) (#7930) Co-authored-by: Diego Cardoso --- .../src/vaadin-rich-text-editor-mixin.js | 32 ++++++- .../rich-text-editor/test/attach-lit.test.js | 3 + .../test/attach-polymer.test.js | 3 + .../rich-text-editor/test/attach.common.js | 90 +++++++++++++++++++ .../rich-text-editor/test/basic.common.js | 66 -------------- 5 files changed, 124 insertions(+), 70 deletions(-) create mode 100644 packages/rich-text-editor/test/attach-lit.test.js create mode 100644 packages/rich-text-editor/test/attach-polymer.test.js create mode 100644 packages/rich-text-editor/test/attach.common.js diff --git a/packages/rich-text-editor/src/vaadin-rich-text-editor-mixin.js b/packages/rich-text-editor/src/vaadin-rich-text-editor-mixin.js index afef5e6b9b..7adf61157c 100644 --- a/packages/rich-text-editor/src/vaadin-rich-text-editor-mixin.js +++ b/packages/rich-text-editor/src/vaadin-rich-text-editor-mixin.js @@ -793,10 +793,26 @@ export const RichTextEditorMixin = (superClass) => */ dangerouslySetHtmlValue(htmlValue) { if (!this._editor) { - // The editor isn't ready yet, store the value for later - this.__pendingHtmlValue = htmlValue; - // Clear a possible value to prevent it from clearing the pending htmlValue once the editor property is set - this.value = ''; + this.__savePendingHtmlValue(htmlValue); + + return; + } + + // In Firefox, the styles are not properly computed when the element is placed + // in a Lit component, as the element is first attached to the DOM and then + // the shadowRoot is initialized. This causes the `hmlValue` to not be correctly + // parsed into the delta format used by Quill. To work around this, we check + // if the display property is set and if not, we wait for the element to intersect + // with the viewport before trying to set the value again. + if (!getComputedStyle(this).display) { + this.__savePendingHtmlValue(htmlValue); + const observer = new IntersectionObserver(() => { + if (getComputedStyle(this).display) { + this.__flushPendingHtmlValue(); + observer.disconnect(); + } + }); + observer.observe(this); return; } @@ -823,6 +839,14 @@ export const RichTextEditorMixin = (superClass) => this._editor.setContents(deltaFromHtml, SOURCE.API); } + /** @private */ + __savePendingHtmlValue(htmlValue) { + // The editor isn't ready yet, store the value for later + this.__pendingHtmlValue = htmlValue; + // Clear a possible value to prevent it from clearing the pending htmlValue once the editor property is set + this.value = ''; + } + /** @private */ __flushPendingHtmlValue() { if (this.__pendingHtmlValue) { diff --git a/packages/rich-text-editor/test/attach-lit.test.js b/packages/rich-text-editor/test/attach-lit.test.js new file mode 100644 index 0000000000..f1265383d6 --- /dev/null +++ b/packages/rich-text-editor/test/attach-lit.test.js @@ -0,0 +1,3 @@ +import '../theme/lumo/vaadin-rich-text-editor-styles.js'; +import '../src/vaadin-lit-rich-text-editor.js'; +import './attach.common.js'; diff --git a/packages/rich-text-editor/test/attach-polymer.test.js b/packages/rich-text-editor/test/attach-polymer.test.js new file mode 100644 index 0000000000..1cd4e8e513 --- /dev/null +++ b/packages/rich-text-editor/test/attach-polymer.test.js @@ -0,0 +1,3 @@ +import '../theme/lumo/vaadin-rich-text-editor-styles.js'; +import '../src/vaadin-rich-text-editor.js'; +import './attach.common.js'; diff --git a/packages/rich-text-editor/test/attach.common.js b/packages/rich-text-editor/test/attach.common.js new file mode 100644 index 0000000000..b39f210d82 --- /dev/null +++ b/packages/rich-text-editor/test/attach.common.js @@ -0,0 +1,90 @@ +import { expect } from '@vaadin/chai-plugins'; +import { fixtureSync, nextRender, nextUpdate } from '@vaadin/testing-helpers'; +import sinon from 'sinon'; + +describe('attach/detach', () => { + let rte, editor; + + const flushValueDebouncer = () => rte.__debounceSetValue && rte.__debounceSetValue.flush(); + + async function attach(shadow = false) { + const parent = fixtureSync('
'); + if (shadow) { + parent.attachShadow({ mode: 'open' }); + } + parent.appendChild(rte); + await nextRender(); + flushValueDebouncer(); + } + + beforeEach(async () => { + rte = fixtureSync(''); + await nextRender(); + flushValueDebouncer(); + editor = rte._editor; + }); + + describe('detach and re-attach', () => { + it('should disconnect the emitter when detached', () => { + const spy = sinon.spy(editor.emitter, 'disconnect'); + + rte.parentNode.removeChild(rte); + + expect(spy).to.be.calledOnce; + }); + + it('should re-connect the emitter when detached and re-attached', async () => { + const parent = rte.parentNode; + parent.removeChild(rte); + + const spy = sinon.spy(editor.emitter, 'connect'); + + parent.appendChild(rte); + await nextUpdate(rte); + + expect(spy).to.be.calledOnce; + }); + + it('should parse htmlValue correctly when element is attached but not rendered', async () => { + await attach(true); + rte.dangerouslySetHtmlValue('

Foo

  • Bar
  • Baz
'); + rte.parentNode.shadowRoot.innerHTML = ''; + await nextRender(); + flushValueDebouncer(); + expect(rte.htmlValue).to.equal('

Foo

  • Bar
  • Baz
'); + }); + }); + + describe('unattached rich text editor', () => { + beforeEach(() => { + rte = document.createElement('vaadin-rich-text-editor'); + }); + + it('should not throw when setting html value', () => { + expect(() => rte.dangerouslySetHtmlValue('

Foo

')).to.not.throw(Error); + }); + + it('should have the html value once attached', async () => { + rte.dangerouslySetHtmlValue('

Foo

'); + await attach(); + + expect(rte.htmlValue).to.equal('

Foo

'); + }); + + it('should override the htmlValue', async () => { + rte.dangerouslySetHtmlValue('

Foo

'); + rte.value = JSON.stringify([{ insert: 'Vaadin' }]); + await attach(); + + expect(rte.htmlValue).to.equal('

Vaadin

'); + }); + + it('should override the value', async () => { + rte.value = JSON.stringify([{ insert: 'Vaadin' }]); + rte.dangerouslySetHtmlValue('

Foo

'); + await attach(); + + expect(rte.htmlValue).to.equal('

Foo

'); + }); + }); +}); diff --git a/packages/rich-text-editor/test/basic.common.js b/packages/rich-text-editor/test/basic.common.js index 3a370841fa..9eeaf27eeb 100644 --- a/packages/rich-text-editor/test/basic.common.js +++ b/packages/rich-text-editor/test/basic.common.js @@ -573,70 +573,4 @@ describe('rich text editor', () => { ); }); }); - - describe('detach and re-attach', () => { - it('should disconnect the emitter when detached', () => { - const spy = sinon.spy(editor.emitter, 'disconnect'); - - rte.parentNode.removeChild(rte); - - expect(spy).to.be.calledOnce; - }); - - it('should re-connect the emitter when detached and re-attached', async () => { - const parent = rte.parentNode; - parent.removeChild(rte); - - const spy = sinon.spy(editor.emitter, 'connect'); - - parent.appendChild(rte); - await nextUpdate(rte); - - expect(spy).to.be.calledOnce; - }); - }); -}); - -describe('unattached rich text editor', () => { - let rte; - - beforeEach(() => { - rte = document.createElement('vaadin-rich-text-editor'); - }); - - const flushValueDebouncer = () => rte.__debounceSetValue && rte.__debounceSetValue.flush(); - - async function attach() { - const parent = fixtureSync('
'); - parent.appendChild(rte); - await nextRender(); - flushValueDebouncer(); - } - - it('should not throw when setting html value', () => { - expect(() => rte.dangerouslySetHtmlValue('

Foo

')).to.not.throw(Error); - }); - - it('should have the html value once attached', async () => { - rte.dangerouslySetHtmlValue('

Foo

'); - await attach(); - - expect(rte.htmlValue).to.equal('

Foo

'); - }); - - it('should override the htmlValue', async () => { - rte.dangerouslySetHtmlValue('

Foo

'); - rte.value = JSON.stringify([{ insert: 'Vaadin' }]); - await attach(); - - expect(rte.htmlValue).to.equal('

Vaadin

'); - }); - - it('should override the value', async () => { - rte.value = JSON.stringify([{ insert: 'Vaadin' }]); - rte.dangerouslySetHtmlValue('

Foo

'); - await attach(); - - expect(rte.htmlValue).to.equal('

Foo

'); - }); });