diff --git a/packages/base-styles/_mixins.scss b/packages/base-styles/_mixins.scss
index 0333a8bb6adf92..109926ebf411a9 100644
--- a/packages/base-styles/_mixins.scss
+++ b/packages/base-styles/_mixins.scss
@@ -297,6 +297,16 @@
}
}
}
+
+ &[aria-disabled="true"],
+ &:disabled {
+ background: $gray-100;
+ border-color: $gray-300;
+ cursor: default;
+
+ // Override style inherited from wp-admin. Required to avoid degraded appearance on different backgrounds.
+ opacity: 1;
+ }
}
@mixin radio-control {
diff --git a/packages/dataviews/src/bulk-actions.js b/packages/dataviews/src/bulk-actions.js
index 9fd9f628286e09..5e4139fe0622e7 100644
--- a/packages/dataviews/src/bulk-actions.js
+++ b/packages/dataviews/src/bulk-actions.js
@@ -7,7 +7,7 @@ import {
Modal,
} from '@wordpress/components';
import { __, sprintf, _n } from '@wordpress/i18n';
-import { useMemo, useState, useCallback } from '@wordpress/element';
+import { useMemo, useState, useCallback, useEffect } from '@wordpress/element';
/**
* Internal dependencies
@@ -21,6 +21,24 @@ const {
DropdownMenuSeparatorV2: DropdownMenuSeparator,
} = unlock( componentsPrivateApis );
+export function useHasAPossibleBulkAction( actions, item ) {
+ return useMemo( () => {
+ return actions.some( ( action ) => {
+ return action.supportsBulk && action.isEligible( item );
+ } );
+ }, [ actions, item ] );
+}
+
+export function useSomeItemHasAPossibleBulkAction( actions, data ) {
+ return useMemo( () => {
+ return data.some( ( item ) => {
+ return actions.some( ( action ) => {
+ return action.supportsBulk && action.isEligible( item );
+ } );
+ } );
+ }, [ actions, data ] );
+}
+
function ActionWithModal( {
action,
selectedItems,
@@ -107,15 +125,47 @@ export default function BulkActions( {
() => actions.filter( ( action ) => action.supportsBulk ),
[ actions ]
);
- const areAllSelected = selection && selection.length === data.length;
const [ isMenuOpen, onMenuOpenChange ] = useState( false );
const [ actionWithModal, setActionWithModal ] = useState();
+ const selectableItems = useMemo( () => {
+ return data.filter( ( item ) => {
+ return bulkActions.some( ( action ) => action.isEligible( item ) );
+ } );
+ }, [ data, bulkActions ] );
+
+ const numberSelectableItems = selectableItems.length;
+ const areAllSelected =
+ selection && selection.length === numberSelectableItems;
+
const selectedItems = useMemo( () => {
return data.filter( ( item ) =>
selection.includes( getItemId( item ) )
);
}, [ selection, data, getItemId ] );
+ const hasNonSelectableItemSelected = useMemo( () => {
+ return selectedItems.some( ( item ) => {
+ return ! selectableItems.includes( item );
+ } );
+ }, [ selectedItems, selectableItems ] );
+ useEffect( () => {
+ if ( hasNonSelectableItemSelected ) {
+ onSelectionChange(
+ selectedItems.filter( ( selectedItem ) => {
+ return selectableItems.some( ( item ) => {
+ return getItemId( selectedItem ) === getItemId( item );
+ } );
+ } )
+ );
+ }
+ }, [
+ hasNonSelectableItemSelected,
+ selectedItems,
+ selectableItems,
+ getItemId,
+ onSelectionChange,
+ ] );
+
if ( bulkActions.length === 0 ) {
return null;
}
@@ -157,9 +207,9 @@ export default function BulkActions( {
disabled={ areAllSelected }
hideOnClick={ false }
onClick={ () => {
- onSelectionChange( data );
+ onSelectionChange( selectableItems );
} }
- suffix={ data.length }
+ suffix={ numberSelectableItems }
>
{ __( 'Select all' ) }
diff --git a/packages/dataviews/src/dataviews.js b/packages/dataviews/src/dataviews.js
index 49143d0bb0753e..9bcf531006406a 100644
--- a/packages/dataviews/src/dataviews.js
+++ b/packages/dataviews/src/dataviews.js
@@ -20,6 +20,16 @@ import BulkActions from './bulk-actions';
const defaultGetItemId = ( item ) => item.id;
const defaultOnSelectionChange = () => {};
+function useSomeItemHasAPossibleBulkAction( actions, data ) {
+ return useMemo( () => {
+ return data.some( ( item ) => {
+ return actions.some( ( action ) => {
+ return action.supportsBulk && action.isEligible( item );
+ } );
+ } );
+ }, [ actions, data ] );
+}
+
export default function DataViews( {
view,
onChangeView,
@@ -75,6 +85,11 @@ export default function DataViews( {
render: field.render || field.getValue,
} ) );
}, [ fields ] );
+
+ const hasPossibleBulkAction = useSomeItemHasAPossibleBulkAction(
+ actions,
+ data
+ );
return (
@@ -103,15 +118,16 @@ export default function DataViews( {
setOpenedFilter={ setOpenedFilter }
/>
- { [ LAYOUT_TABLE, LAYOUT_GRID ].includes( view.type ) && (
-
- ) }
+ { [ LAYOUT_TABLE, LAYOUT_GRID ].includes( view.type ) &&
+ hasPossibleBulkAction && (
+
+ ) }
{
if ( ! isSelected ) {
onSelectionChange(
diff --git a/packages/dataviews/src/style.scss b/packages/dataviews/src/style.scss
index e865ff8a25fed0..7e45bec25856f7 100644
--- a/packages/dataviews/src/style.scss
+++ b/packages/dataviews/src/style.scss
@@ -119,7 +119,7 @@
background-color: #f8f8f8;
}
- .components-checkbox-control__input {
+ .components-checkbox-control__input.components-checkbox-control__input {
opacity: 0;
&:checked,
diff --git a/packages/dataviews/src/view-grid.js b/packages/dataviews/src/view-grid.js
index 44ab1822a60750..3b46a7424dc1b1 100644
--- a/packages/dataviews/src/view-grid.js
+++ b/packages/dataviews/src/view-grid.js
@@ -22,6 +22,8 @@ import { useState } from '@wordpress/element';
import ItemActions from './item-actions';
import SingleSelectionCheckbox from './single-selection-checkbox';
+import { useHasAPossibleBulkAction } from './bulk-actions';
+
function GridItem( {
selection,
data,
@@ -34,6 +36,7 @@ function GridItem( {
visibleFields,
} ) {
const [ hasNoPointerEvents, setHasNoPointerEvents ] = useState( false );
+ const hasBulkAction = useHasAPossibleBulkAction( actions, item );
const id = getItemId( item );
const isSelected = selection.includes( id );
return (
@@ -41,11 +44,11 @@ function GridItem( {
spacing={ 0 }
key={ id }
className={ classnames( 'dataviews-view-grid__card', {
- 'is-selected': isSelected,
+ 'is-selected': hasBulkAction && isSelected,
'has-no-pointer-events': hasNoPointerEvents,
} ) }
onMouseDown={ ( event ) => {
- if ( event.ctrlKey || event.metaKey ) {
+ if ( hasBulkAction && ( event.ctrlKey || event.metaKey ) ) {
setHasNoPointerEvents( true );
if ( ! isSelected ) {
onSelectionChange(
@@ -91,6 +94,7 @@ function GridItem( {
getItemId={ getItemId }
data={ data }
primaryField={ primaryField }
+ disabled={ ! hasBulkAction }
/>
{ primaryField?.render( { item } ) }
diff --git a/packages/dataviews/src/view-table.js b/packages/dataviews/src/view-table.js
index dc166577034257..9737aa3d462835 100644
--- a/packages/dataviews/src/view-table.js
+++ b/packages/dataviews/src/view-table.js
@@ -23,6 +23,7 @@ import {
useState,
Children,
Fragment,
+ useMemo,
} from '@wordpress/element';
/**
@@ -33,6 +34,10 @@ import { unlock } from './lock-unlock';
import ItemActions from './item-actions';
import { sanitizeOperators } from './utils';
import { ENUMERATION_TYPE, SORTING_DIRECTIONS } from './constants';
+import {
+ useSomeItemHasAPossibleBulkAction,
+ useHasAPossibleBulkAction,
+} from './bulk-actions';
const {
DropdownMenuV2: DropdownMenu,
@@ -186,8 +191,20 @@ const HeaderMenu = forwardRef( function HeaderMenu(
);
} );
-function BulkSelectionCheckbox( { selection, onSelectionChange, data } ) {
- const areAllSelected = selection.length === data.length;
+function BulkSelectionCheckbox( {
+ selection,
+ onSelectionChange,
+ data,
+ actions,
+} ) {
+ const selectableItems = useMemo( () => {
+ return data.filter( ( item ) => {
+ return actions.some(
+ ( action ) => action.supportsBulk && action.isEligible( item )
+ );
+ } );
+ }, [ data, actions ] );
+ const areAllSelected = selection.length === selectableItems.length;
return (
+ { hasBulkActions && (
+
+
+
+
+ |
+ ) }
+ { visibleFields.map( ( field ) => (
+
+
+ { field.render( {
+ item,
+ } ) }
+
+ |
+ ) ) }
+ { !! actions?.length && (
+
+
+ |
+ ) }
+
+ );
+}
+
function ViewTable( {
view,
onChangeView,
@@ -219,10 +311,10 @@ function ViewTable( {
onSelectionChange,
setOpenedFilter,
} ) {
- const hasBulkActions = actions?.some( ( action ) => action.supportsBulk );
const headerMenuRefs = useRef( new Map() );
const headerMenuToFocusRef = useRef();
const [ nextHeaderMenuToFocus, setNextHeaderMenuToFocus ] = useState();
+ const hasBulkActions = useSomeItemHasAPossibleBulkAction( actions, data );
useEffect( () => {
if ( headerMenuToFocusRef.current ) {
@@ -285,6 +377,7 @@ function ViewTable( {
selection={ selection }
onSelectionChange={ onSelectionChange }
data={ data }
+ actions={ actions }
/>
) }
@@ -347,78 +440,19 @@ function ViewTable( {
{ hasData &&
usedData.map( ( item, index ) => (
-
- { hasBulkActions && (
-
-
-
-
- |
- ) }
- { visibleFields.map( ( field ) => (
-
-
- { field.render( {
- item,
- } ) }
-
- |
- ) ) }
- { !! actions?.length && (
-
-
- |
- ) }
-
+ item={ item }
+ hasBulkActions={ hasBulkActions }
+ actions={ actions }
+ id={ getItemId( item ) || index }
+ visibleFields={ visibleFields }
+ primaryField={ primaryField }
+ selection={ selection }
+ getItemId={ getItemId }
+ onSelectionChange={ onSelectionChange }
+ data={ data }
+ />
) ) }