From 2bde9b41fcb70d65c5f4dbc0987943fcef109343 Mon Sep 17 00:00:00 2001 From: Lourens Schep Date: Mon, 9 Dec 2024 13:23:22 +0100 Subject: [PATCH 1/4] Update PostCardPanel for multiple posts --- .../src/components/post-edit/header.js | 88 ------------------- .../src/components/post-edit/index.js | 11 ++- .../src/components/post-edit/style.scss | 10 --- .../src/components/post-actions/index.js | 66 +++++++++----- .../src/components/post-card-panel/index.js | 78 ++++++++++------ .../src/components/post-card-panel/style.scss | 4 + 6 files changed, 110 insertions(+), 147 deletions(-) delete mode 100644 packages/edit-site/src/components/post-edit/header.js diff --git a/packages/edit-site/src/components/post-edit/header.js b/packages/edit-site/src/components/post-edit/header.js deleted file mode 100644 index 305589d0cc22bf..00000000000000 --- a/packages/edit-site/src/components/post-edit/header.js +++ /dev/null @@ -1,88 +0,0 @@ -/** - * WordPress dependencies - */ -import { - __experimentalHStack as HStack, - __experimentalVStack as VStack, - Icon, - __experimentalText as Text, -} from '@wordpress/components'; -import { useMemo } from '@wordpress/element'; -import { - privateApis as editorPrivateApis, - store as editorStore, -} from '@wordpress/editor'; -import { store as coreStore } from '@wordpress/core-data'; -import { useSelect } from '@wordpress/data'; -import { sprintf, __ } from '@wordpress/i18n'; - -/** - * Internal dependencies - */ -import { unlock } from '../../lock-unlock'; - -const { PostCardPanel } = unlock( editorPrivateApis ); - -export default function PostEditHeader( { postType, postId } ) { - const ids = useMemo( () => postId.split( ',' ), [ postId ] ); - const { icon, labels } = useSelect( - ( select ) => { - const { getEditedEntityRecord, getPostType } = select( coreStore ); - const { getPostIcon } = unlock( select( editorStore ) ); - const _record = getEditedEntityRecord( - 'postType', - postType, - ids[ 0 ] - ); - - return { - icon: getPostIcon( postType, { - area: _record?.area, - } ), - labels: getPostType( postType )?.labels, - }; - }, - [ ids, postType ] - ); - - if ( ids.length === 1 ) { - return ( - - ); - } - - return ( - - - - - { labels?.name && - sprintf( - // translators: %i number of selected items %s: Name of the plural post type e.g: "Posts". - __( '%i %s' ), - ids.length, - labels?.name - ) } - - - - { sprintf( - // translators: %s: Name of the plural post type e.g: "Posts". - __( 'Changes will be applied to all selected %s.' ), - labels?.name.toLowerCase() - ) } - - - ); -} diff --git a/packages/edit-site/src/components/post-edit/index.js b/packages/edit-site/src/components/post-edit/index.js index b3954e4ddbdeaf..aa4e901f7abb04 100644 --- a/packages/edit-site/src/components/post-edit/index.js +++ b/packages/edit-site/src/components/post-edit/index.js @@ -18,12 +18,11 @@ import { privateApis as editorPrivateApis } from '@wordpress/editor'; * Internal dependencies */ import Page from '../page'; -import PostEditHeader from '../post-edit/header'; import { unlock } from '../../lock-unlock'; import usePatternSettings from '../page-patterns/use-pattern-settings'; import { privateApis as blockEditorPrivateApis } from '@wordpress/block-editor'; -const { usePostFields } = unlock( editorPrivateApis ); +const { usePostFields, PostCardPanel } = unlock( editorPrivateApis ); const fieldsWithBulkEditSupport = [ 'title', @@ -159,7 +158,13 @@ function PostEditForm( { postType, postId } ) { return ( - + 1 ? ids : undefined } + /> { const { getEditedEntityRecord, getEntityRecordPermissions } = unlock( select( coreStore ) ); return { - item: getEditedEntityRecord( 'postType', postType, postId ), - permissions: getEntityRecordPermissions( - 'postType', - postType, - postId + items: postIds.map( ( postId ) => + getEditedEntityRecord( 'postType', postType, postId ) + ), + permissions: postIds.map( ( postId ) => + getEntityRecordPermissions( 'postType', postType, postId ) ), }; }, - [ postId, postType ] + [ postIds, postType ] ); - const itemWithPermissions = useMemo( () => { - return { + + return useMemo( () => { + return items.map( ( item, index ) => ( { ...item, - permissions, - }; - }, [ item, permissions ] ); + permissions: permissions[ index ], + } ) ); + }, [ items, permissions ] ); +} + +export default function PostActions( { + postType, + postId, + postIds, + onActionPerformed, +} ) { + const [ activeModalAction, setActiveModalAction ] = useState( null ); + const _postIds = useMemo( () => { + if ( postIds && postIds.length ) { + return postIds; + } + return postId ? [ postId ] : []; + }, [ postId, postIds ] ); + + const itemsWithPermissions = useEditedEntityRecordsWithPermissions( + postType, + _postIds + ); const allActions = usePostActions( { postType, onActionPerformed } ); const actions = useMemo( () => { return allActions.filter( ( action ) => { return ( - ! action.isEligible || action.isEligible( itemWithPermissions ) + ( ! action.isEligible || + itemsWithPermissions.some( ( itemWithPermissions ) => + action.isEligible( itemWithPermissions ) + ) ) && + ( itemsWithPermissions.length < 2 || action.supportsBulk ) ); } ); - }, [ allActions, itemWithPermissions ] ); + }, [ allActions, itemsWithPermissions ] ); return ( <> @@ -70,14 +94,14 @@ export default function PostActions( { postType, postId, onActionPerformed } ) { > { !! activeModalAction && ( setActiveModalAction( null ) } /> ) } @@ -119,7 +143,7 @@ export function ActionModal( { action, items, closeModal } ) { ); } -function ActionsDropdownMenuGroup( { actions, item, setActiveModalAction } ) { +function ActionsDropdownMenuGroup( { actions, items, setActiveModalAction } ) { const registry = useRegistry(); return ( @@ -133,9 +157,9 @@ function ActionsDropdownMenuGroup( { actions, item, setActiveModalAction } ) { setActiveModalAction( action ); return; } - action.callback( [ item ], { registry } ); + action.callback( items, { registry } ); } } - items={ [ item ] } + items={ items } /> ); } ) } diff --git a/packages/editor/src/components/post-card-panel/index.js b/packages/editor/src/components/post-card-panel/index.js index e347d3c79fc521..70a81e24855dcd 100644 --- a/packages/editor/src/components/post-card-panel/index.js +++ b/packages/editor/src/components/post-card-panel/index.js @@ -4,11 +4,12 @@ import { Icon, __experimentalHStack as HStack, + __experimentalVStack as VStack, __experimentalText as Text, } from '@wordpress/components'; import { store as coreStore } from '@wordpress/core-data'; import { useSelect } from '@wordpress/data'; -import { __ } from '@wordpress/i18n'; +import { __, sprintf } from '@wordpress/i18n'; import { decodeEntities } from '@wordpress/html-entities'; /** @@ -24,50 +25,67 @@ import PostActions from '../post-actions'; import usePageTypeBadge from '../../utils/pageTypeBadge'; import { getTemplateInfo } from '../../utils/get-template-info'; +const EMPTY_ARRAY = []; + export default function PostCardPanel( { postType, postId, + postIds = EMPTY_ARRAY, onActionPerformed, } ) { - const { title, icon } = useSelect( + const { postTitle, icon, labels } = useSelect( ( select ) => { - const { getEditedEntityRecord } = select( coreStore ); + const { getEditedEntityRecord, getEntityRecord, getPostType } = + select( coreStore ); + const { getPostIcon } = unlock( select( editorStore ) ); + let _title = ''; const _record = getEditedEntityRecord( 'postType', postType, - postId + postIds.length ? postIds[ 0 ] : postId ); + if ( postId || postIds.length === 1 ) { + const { default_template_types: templateTypes = [] } = + getEntityRecord( 'root', '__unstableBase' ) ?? {}; - const { default_template_types: templateTypes = [] } = - select( coreStore ).getEntityRecord( - 'root', - '__unstableBase' - ) ?? {}; - - const _templateInfo = [ - TEMPLATE_POST_TYPE, - TEMPLATE_PART_POST_TYPE, - ].includes( postType ) - ? getTemplateInfo( { - template: _record, - templateTypes, - } ) - : {}; + const _templateInfo = [ + TEMPLATE_POST_TYPE, + TEMPLATE_PART_POST_TYPE, + ].includes( postType ) + ? getTemplateInfo( { + template: _record, + templateTypes, + } ) + : {}; + _title = _templateInfo?.title || _record?.title; + } return { - title: _templateInfo?.title || _record?.title, - icon: unlock( select( editorStore ) ).getPostIcon( postType, { + postTitle: _title, + icon: getPostIcon( postType, { area: _record?.area, } ), + labels: getPostType( postType )?.labels, }; }, - [ postId, postType ] + [ postId, postType, postIds ] ); const pageTypeBadge = usePageTypeBadge( postId ); + let title = __( 'No title' ); + if ( labels?.name && postIds.length > 1 ) { + title = sprintf( + // translators: %i number of selected items %s: Name of the plural post type e.g: "Posts". + __( '%i %s' ), + postIds.length, + labels?.name + ); + } else if ( postTitle ) { + title = decodeEntities( postTitle ); + } return ( -
+ - { title ? decodeEntities( title ) : __( 'No title' ) } + { title } { pageTypeBadge && ( { pageTypeBadge } @@ -90,9 +108,19 @@ export default function PostCardPanel( { -
+ { postIds.length > 1 && ( + + { sprintf( + // translators: %s: Name of the plural post type e.g: "Posts". + __( 'Changes will be applied to all selected %s.' ), + labels?.name.toLowerCase() + ) } + + ) } +
); } diff --git a/packages/editor/src/components/post-card-panel/style.scss b/packages/editor/src/components/post-card-panel/style.scss index d9ecfc58304e85..c3638b313a8285 100644 --- a/packages/editor/src/components/post-card-panel/style.scss +++ b/packages/editor/src/components/post-card-panel/style.scss @@ -33,6 +33,10 @@ &.has-description &__header { margin-bottom: $grid-unit-10; } + + .editor-post-card-panel__description { + color: $gray-700; + } } .editor-post-card-panel__title-badge { From f1797ab6cf376ae264eeec8c6369e42f32f9398e Mon Sep 17 00:00:00 2001 From: Lourens Schep Date: Mon, 9 Dec 2024 13:35:57 +0100 Subject: [PATCH 2/4] Fix rename missed during rebase --- packages/editor/src/components/post-actions/index.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/editor/src/components/post-actions/index.js b/packages/editor/src/components/post-actions/index.js index 5d308a53f19c51..dd3502c8235f91 100644 --- a/packages/editor/src/components/post-actions/index.js +++ b/packages/editor/src/components/post-actions/index.js @@ -94,7 +94,7 @@ export default function PostActions( { > From df934ac376220417421c6cec4a8c9e66130159b9 Mon Sep 17 00:00:00 2001 From: Lourens Schep Date: Mon, 9 Dec 2024 13:47:49 +0100 Subject: [PATCH 3/4] Add inline comment --- packages/editor/src/components/post-card-panel/index.js | 1 + 1 file changed, 1 insertion(+) diff --git a/packages/editor/src/components/post-card-panel/index.js b/packages/editor/src/components/post-card-panel/index.js index 70a81e24855dcd..5e302ecf4d5ab3 100644 --- a/packages/editor/src/components/post-card-panel/index.js +++ b/packages/editor/src/components/post-card-panel/index.js @@ -42,6 +42,7 @@ export default function PostCardPanel( { const _record = getEditedEntityRecord( 'postType', postType, + // Use first post if multiple postIds, as we only use it for the icon. postIds.length ? postIds[ 0 ] : postId ); if ( postId || postIds.length === 1 ) { From 80cb8cce3046b9a0d8c679edc2d939b2a7d00db4 Mon Sep 17 00:00:00 2001 From: Lourens Schep Date: Mon, 9 Dec 2024 14:54:55 +0100 Subject: [PATCH 4/4] Stick with single postId prop --- .../src/components/post-edit/index.js | 8 +---- .../src/components/post-actions/index.js | 13 +++------ .../src/components/post-card-panel/index.js | 29 ++++++++++++------- packages/editor/src/utils/pageTypeBadge.js | 8 +++-- 4 files changed, 29 insertions(+), 29 deletions(-) diff --git a/packages/edit-site/src/components/post-edit/index.js b/packages/edit-site/src/components/post-edit/index.js index aa4e901f7abb04..d32666bcf3aa1b 100644 --- a/packages/edit-site/src/components/post-edit/index.js +++ b/packages/edit-site/src/components/post-edit/index.js @@ -158,13 +158,7 @@ function PostEditForm( { postType, postId } ) { return ( - 1 ? ids : undefined } - /> + { - if ( postIds && postIds.length ) { - return postIds; + if ( Array.isArray( postId ) ) { + return postId; } return postId ? [ postId ] : []; - }, [ postId, postIds ] ); + }, [ postId ] ); const itemsWithPermissions = useEditedEntityRecordsWithPermissions( postType, diff --git a/packages/editor/src/components/post-card-panel/index.js b/packages/editor/src/components/post-card-panel/index.js index 5e302ecf4d5ab3..7849f014ab49c8 100644 --- a/packages/editor/src/components/post-card-panel/index.js +++ b/packages/editor/src/components/post-card-panel/index.js @@ -9,6 +9,7 @@ import { } from '@wordpress/components'; import { store as coreStore } from '@wordpress/core-data'; import { useSelect } from '@wordpress/data'; +import { useMemo } from '@wordpress/element'; import { __, sprintf } from '@wordpress/i18n'; import { decodeEntities } from '@wordpress/html-entities'; @@ -25,14 +26,24 @@ import PostActions from '../post-actions'; import usePageTypeBadge from '../../utils/pageTypeBadge'; import { getTemplateInfo } from '../../utils/get-template-info'; -const EMPTY_ARRAY = []; - +/** + * Renders a title of the post type and the available quick actions available within a 3-dot dropdown. + * + * @param {Object} props - Component props. + * @param {string} [props.postType] - The post type string. + * @param {string|string[]} [props.postId] - The post id or list of post ids. + * @param {Function} [props.onActionPerformed] - A callback function for when a quick action is performed. + * @return {React.ReactNode} The rendered component. + */ export default function PostCardPanel( { postType, postId, - postIds = EMPTY_ARRAY, onActionPerformed, } ) { + const postIds = useMemo( + () => ( Array.isArray( postId ) ? postId : [ postId ] ), + [ postId ] + ); const { postTitle, icon, labels } = useSelect( ( select ) => { const { getEditedEntityRecord, getEntityRecord, getPostType } = @@ -42,10 +53,9 @@ export default function PostCardPanel( { const _record = getEditedEntityRecord( 'postType', postType, - // Use first post if multiple postIds, as we only use it for the icon. - postIds.length ? postIds[ 0 ] : postId + postIds[ 0 ] ); - if ( postId || postIds.length === 1 ) { + if ( postIds.length === 1 ) { const { default_template_types: templateTypes = [] } = getEntityRecord( 'root', '__unstableBase' ) ?? {}; @@ -69,7 +79,7 @@ export default function PostCardPanel( { labels: getPostType( postType )?.labels, }; }, - [ postId, postType, postIds ] + [ postIds, postType ] ); const pageTypeBadge = usePageTypeBadge( postId ); @@ -78,7 +88,7 @@ export default function PostCardPanel( { title = sprintf( // translators: %i number of selected items %s: Name of the plural post type e.g: "Posts". __( '%i %s' ), - postIds.length, + postId.length, labels?.name ); } else if ( postTitle ) { @@ -100,7 +110,7 @@ export default function PostCardPanel( { as="h2" > { title } - { pageTypeBadge && ( + { pageTypeBadge && postIds.length === 1 && ( { pageTypeBadge } @@ -109,7 +119,6 @@ export default function PostCardPanel( { diff --git a/packages/editor/src/utils/pageTypeBadge.js b/packages/editor/src/utils/pageTypeBadge.js index bc787b12284222..3dc7d4750be739 100644 --- a/packages/editor/src/utils/pageTypeBadge.js +++ b/packages/editor/src/utils/pageTypeBadge.js @@ -8,7 +8,7 @@ import { store as coreStore } from '@wordpress/core-data'; /** * Custom hook to get the page type badge for the current post on edit site view. * - * @param {number} postId postId of the current post being edited. + * @param {number|string} postId postId of the current post being edited. */ export default function usePageTypeBadge( postId ) { const { isFrontPage, isPostsPage } = useSelect( ( select ) => { @@ -20,9 +20,11 @@ export default function usePageTypeBadge( postId ) { ? getEditedEntityRecord( 'root', 'site' ) : undefined; + const _postId = parseInt( postId, 10 ); + return { - isFrontPage: siteSettings?.page_on_front === postId, - isPostsPage: siteSettings?.page_for_posts === postId, + isFrontPage: siteSettings?.page_on_front === _postId, + isPostsPage: siteSettings?.page_for_posts === _postId, }; } );