From 2d4ee6371285831365161bed471d25a8399aa93f Mon Sep 17 00:00:00 2001 From: Daniel Richards Date: Tue, 9 Apr 2019 15:42:46 +0800 Subject: [PATCH 1/3] Remove logic that causes a metabox save before a preview --- packages/edit-post/src/store/effects.js | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/packages/edit-post/src/store/effects.js b/packages/edit-post/src/store/effects.js index 673025c3fae51..4897b75b2060f 100644 --- a/packages/edit-post/src/store/effects.js +++ b/packages/edit-post/src/store/effects.js @@ -45,26 +45,22 @@ const effects = { let wasSavingPost = select( 'core/editor' ).isSavingPost(); let wasAutosavingPost = select( 'core/editor' ).isAutosavingPost(); - let wasPreviewingPost = select( 'core/editor' ).isPreviewingPost(); // Save metaboxes when performing a full save on the post. subscribe( () => { const isSavingPost = select( 'core/editor' ).isSavingPost(); const isAutosavingPost = select( 'core/editor' ).isAutosavingPost(); - const isPreviewingPost = select( 'core/editor' ).isPreviewingPost(); const hasActiveMetaBoxes = select( 'core/edit-post' ).hasMetaBoxes(); // Save metaboxes on save completion, except for autosaves that are not a post preview. const shouldTriggerMetaboxesSave = ( hasActiveMetaBoxes && ( - ( wasSavingPost && ! isSavingPost && ! wasAutosavingPost ) || - ( wasAutosavingPost && wasPreviewingPost && ! isPreviewingPost ) + ( wasSavingPost && ! isSavingPost && ! wasAutosavingPost ) ) ); // Save current state for next inspection. wasSavingPost = isSavingPost; wasAutosavingPost = isAutosavingPost; - wasPreviewingPost = isPreviewingPost; if ( shouldTriggerMetaboxesSave ) { store.dispatch( requestMetaBoxUpdates() ); From f10ea1dec934ff082f9a9b6a69d7594b58248a20 Mon Sep 17 00:00:00 2001 From: Daniel Richards Date: Mon, 11 Feb 2019 15:41:37 +0800 Subject: [PATCH 2/3] Add e2e test for preview with custom fields active. --- packages/e2e-tests/specs/preview.test.js | 149 +++++++++++++++++------ 1 file changed, 115 insertions(+), 34 deletions(-) diff --git a/packages/e2e-tests/specs/preview.test.js b/packages/e2e-tests/specs/preview.test.js index 8488ffe1ec1e3..31b827f9bbbb1 100644 --- a/packages/e2e-tests/specs/preview.test.js +++ b/packages/e2e-tests/specs/preview.test.js @@ -12,47 +12,75 @@ import { createURL, publishPost, saveDraft, + clickOnMoreMenuItem, + pressKeyWithModifier, } from '@wordpress/e2e-test-utils'; -describe( 'Preview', () => { - beforeEach( async () => { - await createNewPost(); - } ); +async function openPreviewPage( editorPage ) { + let openTabs = await browser.pages(); + const expectedTabsCount = openTabs.length + 1; + await editorPage.click( '.editor-post-preview' ); - async function openPreviewPage( editorPage ) { - let openTabs = await browser.pages(); - const expectedTabsCount = openTabs.length + 1; - await editorPage.click( '.editor-post-preview' ); - - // Wait for the new tab to open. - while ( openTabs.length < expectedTabsCount ) { - await editorPage.waitFor( 1 ); - openTabs = await browser.pages(); - } - - const previewPage = last( openTabs ); - // Wait for the preview to load. We can't do interstitial detection here, - // because it might load too quickly for us to pick up, so we wait for - // the preview to load by waiting for the title to appear. - await previewPage.waitForSelector( '.entry-title' ); - return previewPage; + // Wait for the new tab to open. + while ( openTabs.length < expectedTabsCount ) { + await editorPage.waitFor( 1 ); + openTabs = await browser.pages(); } - /** - * Given a Puppeteer Page instance for a preview window, clicks Preview, and - * awaits the window navigation. - * - * @param {puppeteer.Page} previewPage Page on which to await navigation. - * - * @return {Promise} Promise resolving once navigation completes. - */ - async function waitForPreviewNavigation( previewPage ) { - const navigationCompleted = previewPage.waitForNavigation(); - await page.click( '.editor-post-preview' ); + const previewPage = last( openTabs ); + // Wait for the preview to load. We can't do interstitial detection here, + // because it might load too quickly for us to pick up, so we wait for + // the preview to load by waiting for the title to appear. + await previewPage.waitForSelector( '.entry-title' ); + return previewPage; +} + +/** + * Given a Puppeteer Page instance for a preview window, clicks Preview, and + * awaits the window navigation. + * + * @param {puppeteer.Page} previewPage Page on which to await navigation. + * + * @return {Promise} Promise resolving once navigation completes. + */ +async function waitForPreviewNavigation( previewPage ) { + const navigationCompleted = previewPage.waitForNavigation(); + await page.click( '.editor-post-preview' ); + return navigationCompleted; +} + +/** + * Enables or disables the custom fields option. + * + * Note that this is implemented separately from the `toggleScreenOptions` + * utility, since the custom fields option triggers a page reload and requires + * extra async logic to wait for navigation to complete. + * + * @param {boolean} shouldBeChecked If true, turns the option on. If false, off. + */ +async function toggleCustomFieldsOption( shouldBeChecked ) { + await clickOnMoreMenuItem( 'Options' ); + const [ handle ] = await page.$x( `//label[contains(text(), "Custom Fields")]` ); + + const isChecked = await page.evaluate( + ( element ) => element.control.checked, + handle + ); + if ( isChecked !== shouldBeChecked ) { + const navigationCompleted = page.waitForNavigation(); + await handle.click(); return navigationCompleted; } - it( 'Should open a preview window for a new post', async () => { + await page.click( 'button[aria-label="Close dialog"]' ); +} + +describe( 'Preview', () => { + beforeEach( async () => { + await createNewPost(); + } ); + + it( 'should open a preview window for a new post', async () => { const editorPage = page; // Disabled until content present. @@ -129,7 +157,7 @@ describe( 'Preview', () => { await previewPage.close(); } ); - it( 'Should not revert title during a preview right after a save draft', async () => { + it( 'should not revert title during a preview right after a save draft', async () => { const editorPage = page; // Type aaaaa in the title filed. @@ -166,3 +194,56 @@ describe( 'Preview', () => { await previewPage.close(); } ); } ); + +describe( 'Preview + Custom Fields', async () => { + beforeEach( async () => { + await createNewPost(); + await toggleCustomFieldsOption( true ); + } ); + + afterEach( async () => { + await toggleCustomFieldsOption( false ); + } ); + + // Catch regressions of https://github.com/WordPress/gutenberg/issues/12617 + it( 'displays edits to the post title and content in the preview', async () => { + const editorPage = page; + + // Add an initial title and content. + await editorPage.type( '.editor-post-title__input', 'title 1' ); + await editorPage.keyboard.press( 'Tab' ); + await editorPage.keyboard.type( 'content 1' ); + + // Open the preview page. + const previewPage = await openPreviewPage( editorPage ); + + // Check the title and preview match. + let previewTitle = await previewPage.$eval( '.entry-title', ( node ) => node.textContent ); + expect( previewTitle ).toBe( 'title 1' ); + let previewContent = await previewPage.$eval( '.entry-content p', ( node ) => node.textContent ); + expect( previewContent ).toBe( 'content 1' ); + + // Return to editor and modify the title and content. + await editorPage.bringToFront(); + await editorPage.click( '.editor-post-title__input' ); + await pressKeyWithModifier( 'shift', 'Home' ); + await editorPage.keyboard.press( 'Delete' ); + await editorPage.keyboard.type( 'title 2' ); + await editorPage.keyboard.press( 'Tab' ); + await pressKeyWithModifier( 'shift', 'Home' ); + await editorPage.keyboard.press( 'Delete' ); + await editorPage.keyboard.type( 'content 2' ); + + // Open the preview page. + await waitForPreviewNavigation( previewPage ); + + // Title in preview should match input. + previewTitle = await previewPage.$eval( '.entry-title', ( node ) => node.textContent ); + expect( previewTitle ).toBe( 'title 2' ); + previewContent = await previewPage.$eval( '.entry-content p', ( node ) => node.textContent ); + expect( previewContent ).toBe( 'content 2' ); + + // Make sure the editor is active for the afterEach function. + await editorPage.bringToFront(); + } ); +} ); From a4412564bed6c3678fb1a0bf369267bf3e7b4e03 Mon Sep 17 00:00:00 2001 From: Daniel Richards Date: Tue, 9 Apr 2019 17:42:51 +0800 Subject: [PATCH 3/3] Improve test robustness --- packages/e2e-tests/specs/preview.test.js | 16 ++++++++++------ 1 file changed, 10 insertions(+), 6 deletions(-) diff --git a/packages/e2e-tests/specs/preview.test.js b/packages/e2e-tests/specs/preview.test.js index 31b827f9bbbb1..da77d0690aebf 100644 --- a/packages/e2e-tests/specs/preview.test.js +++ b/packages/e2e-tests/specs/preview.test.js @@ -59,20 +59,24 @@ async function waitForPreviewNavigation( previewPage ) { * @param {boolean} shouldBeChecked If true, turns the option on. If false, off. */ async function toggleCustomFieldsOption( shouldBeChecked ) { + const checkboxXPath = '//*[contains(@class, "edit-post-options-modal")]//label[contains(text(), "Custom Fields")]'; await clickOnMoreMenuItem( 'Options' ); - const [ handle ] = await page.$x( `//label[contains(text(), "Custom Fields")]` ); + await page.waitForXPath( checkboxXPath ); + const [ checkboxHandle ] = await page.$x( checkboxXPath ); const isChecked = await page.evaluate( ( element ) => element.control.checked, - handle + checkboxHandle ); + if ( isChecked !== shouldBeChecked ) { const navigationCompleted = page.waitForNavigation(); - await handle.click(); - return navigationCompleted; + await checkboxHandle.click(); + await navigationCompleted; + return; } - await page.click( 'button[aria-label="Close dialog"]' ); + await page.click( '.edit-post-options-modal button[aria-label="Close dialog"]' ); } describe( 'Preview', () => { @@ -195,7 +199,7 @@ describe( 'Preview', () => { } ); } ); -describe( 'Preview + Custom Fields', async () => { +describe( 'Preview with Custom Fields enabled', async () => { beforeEach( async () => { await createNewPost(); await toggleCustomFieldsOption( true );