From ec3b007a5222c4e097d6bdaff59dcb32ea67bac3 Mon Sep 17 00:00:00 2001 From: Grzegorz Ziolkowski Date: Tue, 13 Nov 2018 10:06:22 +0100 Subject: [PATCH 1/6] Add a way to programatically remove editor panels from UI --- docs/data/data-core-edit-post.md | 22 +++++++++ .../options-modal/options/enable-panel.js | 13 ++++-- .../test/__snapshots__/index.js.snap | 8 ++-- .../__snapshots__/meta-boxes-section.js.snap | 8 ++-- packages/edit-post/src/store/actions.js | 14 ++++++ packages/edit-post/src/store/reducer.js | 11 +++++ packages/edit-post/src/store/selectors.js | 18 +++++++- packages/edit-post/src/store/test/actions.js | 10 +++++ packages/edit-post/src/store/test/reducer.js | 15 +++++++ .../edit-post/src/store/test/selectors.js | 45 +++++++++++++++++++ 10 files changed, 151 insertions(+), 13 deletions(-) diff --git a/docs/data/data-core-edit-post.md b/docs/data/data-core-edit-post.md index 93e658e91d479..e94b76aa70124 100644 --- a/docs/data/data-core-edit-post.md +++ b/docs/data/data-core-edit-post.md @@ -89,6 +89,20 @@ Returns true if the publish sidebar is opened. Whether the publish sidebar is open. +### isEditorPanelRemoved + +Returns true if the given panel is removed, or false otherwise. Panels are +not removed by default. + +*Parameters* + + * state: Global application state. + * panelName: A string that identifies the panel. + +*Returns* + +Whether or not the panel is removed. + ### isEditorPanelEnabled Returns true if the given panel is enabled, or false otherwise. Panels are @@ -301,6 +315,14 @@ Returns an action object used to open or close a panel in the editor. * panelName: A string that identifies the panel to open or close. +### removeEditorPanel + +Returns an action object used to remove a panel from the editor. + +*Parameters* + + * panelName: A string that identifies the panel to remove. + ### toggleFeature Returns an action object used to toggle a feature flag. diff --git a/packages/edit-post/src/components/options-modal/options/enable-panel.js b/packages/edit-post/src/components/options-modal/options/enable-panel.js index d21f1051cb039..d47ab67e7e6d9 100644 --- a/packages/edit-post/src/components/options-modal/options/enable-panel.js +++ b/packages/edit-post/src/components/options-modal/options/enable-panel.js @@ -1,7 +1,7 @@ /** * WordPress dependencies */ -import { compose } from '@wordpress/compose'; +import { compose, ifCondition } from '@wordpress/compose'; import { withSelect, withDispatch } from '@wordpress/data'; /** @@ -10,9 +10,14 @@ import { withSelect, withDispatch } from '@wordpress/data'; import BaseOption from './base'; export default compose( - withSelect( ( select, { panelName } ) => ( { - isChecked: select( 'core/edit-post' ).isEditorPanelEnabled( panelName ), - } ) ), + withSelect( ( select, { panelName } ) => { + const { isEditorPanelEnabled, isEditorPanelRemoved } = select( 'core/edit-post' ); + return { + isRemoved: isEditorPanelRemoved( panelName ), + isChecked: isEditorPanelEnabled( panelName ), + }; + } ), + ifCondition( ( { isRemoved } ) => ! isRemoved ), withDispatch( ( dispatch, { panelName } ) => ( { onChange: () => dispatch( 'core/edit-post' ).toggleEditorPanelEnabled( panelName ), } ) ) diff --git a/packages/edit-post/src/components/options-modal/test/__snapshots__/index.js.snap b/packages/edit-post/src/components/options-modal/test/__snapshots__/index.js.snap index 67df8a5bd6089..f5764f8ab4acc 100644 --- a/packages/edit-post/src/components/options-modal/test/__snapshots__/index.js.snap +++ b/packages/edit-post/src/components/options-modal/test/__snapshots__/index.js.snap @@ -28,22 +28,22 @@ exports[`OptionsModal should match snapshot when the modal is active 1`] = ` - - - - diff --git a/packages/edit-post/src/components/options-modal/test/__snapshots__/meta-boxes-section.js.snap b/packages/edit-post/src/components/options-modal/test/__snapshots__/meta-boxes-section.js.snap index 42c22d9030718..b30ba56cfdb8f 100644 --- a/packages/edit-post/src/components/options-modal/test/__snapshots__/meta-boxes-section.js.snap +++ b/packages/edit-post/src/components/options-modal/test/__snapshots__/meta-boxes-section.js.snap @@ -17,12 +17,12 @@ exports[`MetaBoxesSection renders a Custom Fields option and meta box options 1` - - - - { } ); } ); + describe( 'removeEditorPanel', () => { + it( 'should return a REMOVE_PANEL action', () => { + expect( removeEditorPanel( 'post-status' ) ).toEqual( { + type: 'REMOVE_PANEL', + panelName: 'post-status', + } ); + } ); + } ); + describe( 'toggleEditorPanelEnabled', () => { it( 'should return a TOGGLE_PANEL_ENABLED action', () => { expect( toggleEditorPanelEnabled( 'post-status' ) ).toEqual( { diff --git a/packages/edit-post/src/store/test/reducer.js b/packages/edit-post/src/store/test/reducer.js index c64d90f48d155..c970425ab9bdd 100644 --- a/packages/edit-post/src/store/test/reducer.js +++ b/packages/edit-post/src/store/test/reducer.js @@ -169,6 +169,21 @@ describe( 'state', () => { } ); } ); + it( 'should remove panels', () => { + const original = deepFreeze( { + panels: { + 'post-status': {}, + }, + } ); + const state = preferences( original, { + type: 'REMOVE_PANEL', + panelName: 'post-status', + } ); + expect( state.panels ).toEqual( { + 'post-status': { removed: true }, + } ); + } ); + it( 'should return switched mode', () => { const state = preferences( deepFreeze( { editorMode: 'visual' } ), { type: 'SWITCH_MODE', diff --git a/packages/edit-post/src/store/test/selectors.js b/packages/edit-post/src/store/test/selectors.js index 4a13ec8e4392c..d56544b74078c 100644 --- a/packages/edit-post/src/store/test/selectors.js +++ b/packages/edit-post/src/store/test/selectors.js @@ -1,3 +1,8 @@ +/** + * External dependencies + */ +import deepFreeze from 'deep-freeze'; + /** * Internal dependencies */ @@ -16,6 +21,7 @@ import { getActiveMetaBoxLocations, isMetaBoxLocationActive, isEditorPanelEnabled, + isEditorPanelRemoved, } from '../selectors'; describe( 'selectors', () => { @@ -195,6 +201,30 @@ describe( 'selectors', () => { } ); } ); + describe( 'isEditorPanelRemoved', () => { + it( 'should return false by default', () => { + const state = deepFreeze( { + preferences: { + panels: {}, + }, + } ); + + expect( isEditorPanelRemoved( state, 'post-status' ) ).toBe( false ); + } ); + + it( 'should return true when panel was removed', () => { + const state = deepFreeze( { + preferences: { + panels: { + 'post-status': { removed: true }, + }, + }, + } ); + + expect( isEditorPanelRemoved( state, 'post-status' ) ).toBe( true ); + } ); + } ); + describe( 'isEditorPanelEnabled', () => { it( 'should return true by default', () => { const state = { @@ -229,6 +259,21 @@ describe( 'selectors', () => { expect( isEditorPanelEnabled( state, 'post-status' ) ).toBe( false ); } ); + + it( 'should return false when a panel is enabled but removed', () => { + const state = deepFreeze( { + preferences: { + panels: { + 'post-status': { + enabled: true, + removed: true, + }, + }, + }, + } ); + + expect( isEditorPanelEnabled( state, 'post-status' ) ).toBe( false ); + } ); } ); describe( 'isEditorPanelOpened', () => { From 2899681748398e6f98f21413525fabe2bc82fcdf Mon Sep 17 00:00:00 2001 From: Grzegorz Ziolkowski Date: Thu, 15 Nov 2018 15:33:30 +0100 Subject: [PATCH 2/6] Add e2e test to ensure it is possible to remove sidebar panels --- .../specs/new-post-default-content.test.js | 4 +- test/e2e/specs/sidebar.test.js | 42 ++++++++++++++++--- .../utils/find-sidebar-panel-with-title.js | 15 +++++++ test/e2e/support/utils/index.js | 1 + 4 files changed, 54 insertions(+), 8 deletions(-) create mode 100644 test/e2e/support/utils/find-sidebar-panel-with-title.js diff --git a/test/e2e/specs/new-post-default-content.test.js b/test/e2e/specs/new-post-default-content.test.js index 8d5adbe95f9ef..28949da44f1b4 100644 --- a/test/e2e/specs/new-post-default-content.test.js +++ b/test/e2e/specs/new-post-default-content.test.js @@ -1,7 +1,7 @@ /** * Internal dependencies */ -import { newPost, getEditedPostContent, openDocumentSettingsSidebar } from '../support/utils'; +import { findSidebarPanelWithTitle, newPost, getEditedPostContent, openDocumentSettingsSidebar } from '../support/utils'; import { activatePlugin, deactivatePlugin } from '../support/plugins'; describe( 'new editor filtered state', () => { @@ -27,7 +27,7 @@ describe( 'new editor filtered state', () => { // open the sidebar, we want to see the excerpt. await openDocumentSettingsSidebar(); - const [ excerptButton ] = await page.$x( '//div[@class="edit-post-sidebar"]//button[@class="components-button components-panel__body-toggle"][contains(text(),"Excerpt")]' ); + const excerptButton = await findSidebarPanelWithTitle( 'Excerpt' ); if ( excerptButton ) { await excerptButton.click( 'button' ); } diff --git a/test/e2e/specs/sidebar.test.js b/test/e2e/specs/sidebar.test.js index 237c81cdba2d9..69c36fcf6ff29 100644 --- a/test/e2e/specs/sidebar.test.js +++ b/test/e2e/specs/sidebar.test.js @@ -2,8 +2,10 @@ * Internal dependencies */ import { + findSidebarPanelWithTitle, newPost, observeFocusLoss, + openDocumentSettingsSidebar, pressWithModifier, setViewport, } from '../support/utils'; @@ -12,12 +14,12 @@ const SIDEBAR_SELECTOR = '.edit-post-sidebar'; const ACTIVE_SIDEBAR_TAB_SELECTOR = '.edit-post-sidebar__panel-tab.is-active'; const ACTIVE_SIDEBAR_BUTTON_TEXT = 'Document'; -describe( 'Publishing', () => { +describe( 'Sidebar', () => { beforeAll( () => { observeFocusLoss(); } ); - it( 'Should have sidebar visible at the start with document sidebar active on desktop', async () => { + it( 'should have sidebar visible at the start with document sidebar active on desktop', async () => { await setViewport( 'large' ); await newPost(); const { nodesCount, content, height, width } = await page.$$eval( ACTIVE_SIDEBAR_TAB_SELECTOR, ( nodes ) => { @@ -41,14 +43,14 @@ describe( 'Publishing', () => { expect( height ).toBeGreaterThan( 10 ); } ); - it( 'Should have the sidebar closed by default on mobile', async () => { + it( 'should have the sidebar closed by default on mobile', async () => { await setViewport( 'small' ); await newPost(); const sidebar = await page.$( SIDEBAR_SELECTOR ); expect( sidebar ).toBeNull(); } ); - it( 'Should close the sidebar when resizing from desktop to mobile', async () => { + it( 'should close the sidebar when resizing from desktop to mobile', async () => { await setViewport( 'large' ); await newPost(); @@ -62,7 +64,7 @@ describe( 'Publishing', () => { expect( sidebarsMobile ).toHaveLength( 0 ); } ); - it( 'Should reopen sidebar the sidebar when resizing from mobile to desktop if the sidebar was closed automatically', async () => { + it( 'should reopen sidebar the sidebar when resizing from mobile to desktop if the sidebar was closed automatically', async () => { await setViewport( 'large' ); await newPost(); await setViewport( 'small' ); @@ -76,7 +78,7 @@ describe( 'Publishing', () => { expect( sidebarsDesktop ).toHaveLength( 1 ); } ); - it( 'Should preserve tab order while changing active tab', async () => { + it( 'should preserve tab order while changing active tab', async () => { await newPost(); // Region navigate to Sidebar. @@ -102,4 +104,32 @@ describe( 'Publishing', () => { ) ); expect( isActiveBlockTab ).toBe( true ); } ); + + it( 'should be possible to programmatically remove Document Settings panels', async () => { + await newPost(); + + await openDocumentSettingsSidebar(); + + expect( await findSidebarPanelWithTitle( 'Categories' ) ).toBeDefined(); + expect( await findSidebarPanelWithTitle( 'Tags' ) ).toBeDefined(); + expect( await findSidebarPanelWithTitle( 'Featured Image' ) ).toBeDefined(); + expect( await findSidebarPanelWithTitle( 'Excerpt' ) ).toBeDefined(); + expect( await findSidebarPanelWithTitle( 'Discussion' ) ).toBeDefined(); + + await page.evaluate( () => { + const { removeEditorPanel } = wp.data.dispatch( 'core/edit-post' ); + + removeEditorPanel( 'taxonomy-panel-category' ); + removeEditorPanel( 'taxonomy-panel-post_tag' ); + removeEditorPanel( 'featured-image' ); + removeEditorPanel( 'post-excerpt' ); + removeEditorPanel( 'discussion-panel' ); + } ); + + expect( await findSidebarPanelWithTitle( 'Categories' ) ).toBeUndefined(); + expect( await findSidebarPanelWithTitle( 'Tags' ) ).toBeUndefined(); + expect( await findSidebarPanelWithTitle( 'Featured Image' ) ).toBeUndefined(); + expect( await findSidebarPanelWithTitle( 'Excerpt' ) ).toBeUndefined(); + expect( await findSidebarPanelWithTitle( 'Discussion' ) ).toBeUndefined(); + } ); } ); diff --git a/test/e2e/support/utils/find-sidebar-panel-with-title.js b/test/e2e/support/utils/find-sidebar-panel-with-title.js new file mode 100644 index 0000000000000..fae3f33b53910 --- /dev/null +++ b/test/e2e/support/utils/find-sidebar-panel-with-title.js @@ -0,0 +1,15 @@ +/** + * External dependencies + */ +import { first } from 'lodash'; + +/** + * Finds a sidebar panel with the provided title. + * + * @param {string} panelTitle The name of sidebar panel. + * + * @return {?ElementHandle} Object that represents an in-page DOM element. + */ +export async function findSidebarPanelWithTitle( panelTitle ) { + return first( await page.$x( `//div[@class="edit-post-sidebar"]//button[@class="components-button components-panel__body-toggle"][contains(text(),"${ panelTitle }")]` ) ); +} diff --git a/test/e2e/support/utils/index.js b/test/e2e/support/utils/index.js index a8010c897a974..693bf990d74e9 100644 --- a/test/e2e/support/utils/index.js +++ b/test/e2e/support/utils/index.js @@ -8,6 +8,7 @@ export { disablePrePublishChecks } from './disable-pre-publish-checks'; export { enablePageDialogAccept } from './enable-page-dialog-accept'; export { enablePrePublishChecks } from './enable-pre-publish-checks'; export { ensureSidebarOpened } from './ensure-sidebar-opened'; +export { findSidebarPanelWithTitle } from './find-sidebar-panel-with-title'; export { getAllBlocks } from './get-all-blocks'; export { getEditedPostContent } from './get-edited-post-content'; export { getUrl } from './get-url'; From ff9dcae998711af33420eff9a42cf1a4a2009d3d Mon Sep 17 00:00:00 2001 From: Grzegorz Ziolkowski Date: Thu, 15 Nov 2018 15:48:21 +0100 Subject: [PATCH 3/6] Edit post: Remove obsolete panel reducer --- packages/edit-post/src/store/reducer.js | 10 ---------- 1 file changed, 10 deletions(-) diff --git a/packages/edit-post/src/store/reducer.js b/packages/edit-post/src/store/reducer.js index b6210ab7df9a0..4f0c5cd04ae22 100644 --- a/packages/edit-post/src/store/reducer.js +++ b/packages/edit-post/src/store/reducer.js @@ -134,15 +134,6 @@ export function activeGeneralSidebar( state = DEFAULT_ACTIVE_GENERAL_SIDEBAR, ac return state; } -export function panel( state = 'document', action ) { - switch ( action.type ) { - case 'SET_ACTIVE_PANEL': - return action.panel; - } - - return state; -} - /** * Reducer for storing the name of the open modal, or null if no modal is open. * @@ -220,7 +211,6 @@ const metaBoxes = combineReducers( { export default combineReducers( { preferences, activeGeneralSidebar, - panel, activeModal, publishSidebarActive, metaBoxes, From ca80d2ae69de0535f35f178c6ec14a78fc1d59e8 Mon Sep 17 00:00:00 2001 From: Grzegorz Ziolkowski Date: Thu, 15 Nov 2018 16:10:41 +0100 Subject: [PATCH 4/6] Edit post: Refactor state to not persist removed panels --- packages/edit-post/src/store/reducer.js | 29 +++++++++++++-- packages/edit-post/src/store/selectors.js | 7 ++-- packages/edit-post/src/store/test/reducer.js | 36 +++++++++++-------- .../edit-post/src/store/test/selectors.js | 14 +++----- 4 files changed, 55 insertions(+), 31 deletions(-) diff --git a/packages/edit-post/src/store/reducer.js b/packages/edit-post/src/store/reducer.js index 4f0c5cd04ae22..5dedd3f6cd7a2 100644 --- a/packages/edit-post/src/store/reducer.js +++ b/packages/edit-post/src/store/reducer.js @@ -1,7 +1,7 @@ /** * External dependencies */ -import { get } from 'lodash'; +import { get, includes } from 'lodash'; /** * WordPress dependencies @@ -116,6 +116,28 @@ export const preferences = combineReducers( { }, } ); +/** + * Reducer storing the list of all programmatically removed panels. + * + * @param {Array} state Current state. + * @param {Object} action Action object. + * + * @return {Array} Updated state. + */ +export function removedPanels( state = [], action ) { + switch ( action.type ) { + case 'REMOVE_PANEL': + if ( ! includes( state, action.panelName ) ) { + return [ + ...state, + action.panelName, + ]; + } + } + + return state; +} + /** * Reducer returning the next active general sidebar state. The active general * sidebar is a unique name to identify either an editor or plugin sidebar. @@ -209,9 +231,10 @@ const metaBoxes = combineReducers( { } ); export default combineReducers( { - preferences, activeGeneralSidebar, activeModal, - publishSidebarActive, metaBoxes, + preferences, + publishSidebarActive, + removedPanels, } ); diff --git a/packages/edit-post/src/store/selectors.js b/packages/edit-post/src/store/selectors.js index ce046154b5df3..c77a3add99e4b 100644 --- a/packages/edit-post/src/store/selectors.js +++ b/packages/edit-post/src/store/selectors.js @@ -100,8 +100,8 @@ export function isPublishSidebarOpened( state ) { } /** - * Returns true if the given panel is removed, or false otherwise. Panels are - * not removed by default. + * Returns true if the given panel was programmatically removed, or false otherwise. + * All panels are not removed by default. * * @param {Object} state Global application state. * @param {string} panelName A string that identifies the panel. @@ -109,8 +109,7 @@ export function isPublishSidebarOpened( state ) { * @return {boolean} Whether or not the panel is removed. */ export function isEditorPanelRemoved( state, panelName ) { - const panels = getPreference( state, 'panels' ); - return get( panels, [ panelName, 'removed' ], false ); + return includes( state.removedPanels, panelName ); } /** diff --git a/packages/edit-post/src/store/test/reducer.js b/packages/edit-post/src/store/test/reducer.js index c970425ab9bdd..66027cd21dc44 100644 --- a/packages/edit-post/src/store/test/reducer.js +++ b/packages/edit-post/src/store/test/reducer.js @@ -13,6 +13,7 @@ import { activeModal, isSavingMetaBoxes, metaBoxLocations, + removedPanels, } from '../reducer'; describe( 'state', () => { @@ -169,21 +170,6 @@ describe( 'state', () => { } ); } ); - it( 'should remove panels', () => { - const original = deepFreeze( { - panels: { - 'post-status': {}, - }, - } ); - const state = preferences( original, { - type: 'REMOVE_PANEL', - panelName: 'post-status', - } ); - expect( state.panels ).toEqual( { - 'post-status': { removed: true }, - } ); - } ); - it( 'should return switched mode', () => { const state = preferences( deepFreeze( { editorMode: 'visual' } ), { type: 'SWITCH_MODE', @@ -328,4 +314,24 @@ describe( 'state', () => { } ); } ); } ); + + describe( 'removedPanels', () => { + it( 'should remove panel', () => { + const original = deepFreeze( [] ); + const state = removedPanels( original, { + type: 'REMOVE_PANEL', + panelName: 'post-status', + } ); + expect( state ).toEqual( [ 'post-status' ] ); + } ); + + it( 'should not remove already removed panel', () => { + const original = deepFreeze( [ 'post-status' ] ); + const state = removedPanels( original, { + type: 'REMOVE_PANEL', + panelName: 'post-status', + } ); + expect( state ).toBe( original ); + } ); + } ); } ); diff --git a/packages/edit-post/src/store/test/selectors.js b/packages/edit-post/src/store/test/selectors.js index d56544b74078c..b035169b3044d 100644 --- a/packages/edit-post/src/store/test/selectors.js +++ b/packages/edit-post/src/store/test/selectors.js @@ -204,9 +204,7 @@ describe( 'selectors', () => { describe( 'isEditorPanelRemoved', () => { it( 'should return false by default', () => { const state = deepFreeze( { - preferences: { - panels: {}, - }, + removedPanels: [], } ); expect( isEditorPanelRemoved( state, 'post-status' ) ).toBe( false ); @@ -214,11 +212,9 @@ describe( 'selectors', () => { it( 'should return true when panel was removed', () => { const state = deepFreeze( { - preferences: { - panels: { - 'post-status': { removed: true }, - }, - }, + removedPanels: [ + 'post-status', + ], } ); expect( isEditorPanelRemoved( state, 'post-status' ) ).toBe( true ); @@ -266,10 +262,10 @@ describe( 'selectors', () => { panels: { 'post-status': { enabled: true, - removed: true, }, }, }, + removedPanels: [ 'post-status' ], } ); expect( isEditorPanelEnabled( state, 'post-status' ) ).toBe( false ); From 33b4cb3ed8f8b31929cc6c42e5b964a8b5ce89bc Mon Sep 17 00:00:00 2001 From: Grzegorz Ziolkowski Date: Thu, 15 Nov 2018 16:20:03 +0100 Subject: [PATCH 5/6] Edit post: Remove old action handler for removed panels in preferences --- packages/edit-post/src/store/reducer.js | 11 ----------- 1 file changed, 11 deletions(-) diff --git a/packages/edit-post/src/store/reducer.js b/packages/edit-post/src/store/reducer.js index 5dedd3f6cd7a2..a016c701582a9 100644 --- a/packages/edit-post/src/store/reducer.js +++ b/packages/edit-post/src/store/reducer.js @@ -73,17 +73,6 @@ export const preferences = combineReducers( { }, }; } - - case 'REMOVE_PANEL': { - const { panelName } = action; - return { - ...state, - [ panelName ]: { - ...state[ panelName ], - removed: true, - }, - }; - } } return state; From ff2da0d39185c58bbf16301b393b520844c32f01 Mon Sep 17 00:00:00 2001 From: Grzegorz Ziolkowski Date: Thu, 15 Nov 2018 16:47:30 +0100 Subject: [PATCH 6/6] Docs: Update data documentation for edit-post package --- docs/data/data-core-edit-post.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/docs/data/data-core-edit-post.md b/docs/data/data-core-edit-post.md index e94b76aa70124..c12cf62a83d63 100644 --- a/docs/data/data-core-edit-post.md +++ b/docs/data/data-core-edit-post.md @@ -91,8 +91,8 @@ Whether the publish sidebar is open. ### isEditorPanelRemoved -Returns true if the given panel is removed, or false otherwise. Panels are -not removed by default. +Returns true if the given panel was programmatically removed, or false otherwise. +All panels are not removed by default. *Parameters*