Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Dataform bulk editing support #67344

Open
wants to merge 10 commits into
base: trunk
Choose a base branch
from
26 changes: 21 additions & 5 deletions packages/dataviews/src/components/dataform/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -10,22 +10,38 @@ import type { DataFormProps } from '../../types';
import { DataFormProvider } from '../dataform-context';
import { normalizeFields } from '../../normalize-fields';
import { DataFormLayout } from '../../dataforms-layouts/data-form-layout';
import { MIXED_VALUE } from '../../constants';

/**
* Loops through the list of data items and returns an object with the intersecting ( same ) key and values.
* Skips keys that start with an underscore.
*
* @param data list of items.
*/
function getIntersectingValues< Item extends object >( data: Item[] ): Item {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I just noticed that this function is trying to combine the items into an Item object. Unfortunately, I don't think that's possible really because the Symbol is not a valid Item value and the nesting is not the right thing to do.

I think what we should be doing here is calling field.getValue for all the fields (maybe the fields that are within the form) and compute the result in a merged object (that is not Item) who keys are the field keys and values are the result of getValue calls.

This is necessary because we can't really assume/know the shape of valid Item. this is something we discussed IRL with @oandregal and @gigitux at Rome core days.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmm yeah, I can give this approach a try. I agree it makes more sense, but will need quite a bit more changes I believe.
For example, if we do generate this new merged object using getValue and receive an array of data do we pass the array of data down to each field? or the merged object? or just the first item in the array ( probably the whole array, but currently a lot of fields pass that straight into getValue ).

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I agree it's not very straightforward. Maybe the current PR can be a good interim solution (although not entirely "correct"). Should we try it in a separate PR to see how they compare?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should we try it in a separate PR to see how they compare?

Yeah a different PR is a good idea, otherwise it may be hard to see the pros/cons of each approach.

const intersectingValues = {} as Item;
const keys = Object.keys( data[ 0 ] ) as Array< keyof Item >;
for ( const key of keys ) {
// Skip keys that start with underscore.
if ( key.toString().startsWith( '_' ) ) {
continue;
}
const [ firstRecord, ...remainingRecords ] = data;
const intersects = remainingRecords.every( ( item ) => {
return item[ key ] === firstRecord[ key ];
} );
if ( intersects ) {
intersectingValues[ key ] = firstRecord[ key ];

if ( typeof firstRecord[ key ] === 'object' ) {
// Traverse through nested objects.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why do you think we should traverse nested objects?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Some fields may rely on the nested values only, incase we want to support that for bulk editing. Although we don't have direct use case for that in core right now, aside from maybe metadata fields.
But if we go with your suggestion below, this won't be necessary as we can rely on the passed in fields only.

intersectingValues[ key ] = getIntersectingValues(
data.map( ( item ) => item[ key ] as object )
) as Item[ keyof Item ];
} else {
const intersects = remainingRecords.every( ( item ) => {
return item[ key ] === firstRecord[ key ];
} );
if ( intersects ) {
intersectingValues[ key ] = firstRecord[ key ];
} else {
intersectingValues[ key ] = MIXED_VALUE as Item[ keyof Item ];
}
}
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is an exact replicate of this comment: #67344 (comment) I am not sure if you meant to do that or if GH messed up?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, forget about this one :P

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is actually where I had my initial "Symbol" comment :)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Aaaha that makes a lot of sense! :)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Added in: c27bd65

}
return intersectingValues;
Expand Down
3 changes: 3 additions & 0 deletions packages/dataviews/src/constants.ts
Original file line number Diff line number Diff line change
Expand Up @@ -68,3 +68,6 @@ export const sortIcons = {
export const LAYOUT_TABLE = 'table';
export const LAYOUT_GRID = 'grid';
export const LAYOUT_LIST = 'list';

// Dataform mixed value.
export const MIXED_VALUE = Symbol.for( 'DATAFORM_MIXED_VALUE' );
4 changes: 2 additions & 2 deletions packages/dataviews/src/dataforms-layouts/panel/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ import type {
import DataFormContext from '../../components/dataform-context';
import { DataFormLayout } from '../data-form-layout';
import { isCombinedField } from '../is-combined-field';
import { MIXED_VALUE } from '../../constants';

function DropdownHeader( {
title,
Expand Down Expand Up @@ -114,8 +115,7 @@ function PanelDropdown< Item extends object >( {
);

const fieldValue = fieldDefinition.getValue( { item: data } );
const showMixedValue =
isBulkEditing && ( fieldValue === undefined || fieldValue === '' );
const showMixedValue = isBulkEditing && fieldValue === MIXED_VALUE;

return (
<Dropdown
Expand Down
1 change: 1 addition & 0 deletions packages/dataviews/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,3 +4,4 @@ export { VIEW_LAYOUTS } from './dataviews-layouts';
export { filterSortAndPaginate } from './filter-and-sort-data-view';
export type * from './types';
export { isItemValid } from './validation';
export { MIXED_VALUE as DATAFORM_MIXED_VALUE } from './constants';
9 changes: 7 additions & 2 deletions packages/fields/src/fields/title/index.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
/**
* WordPress dependencies
*/
import type { Field } from '@wordpress/dataviews';
import { type Field } from '@wordpress/dataviews';
import { __ } from '@wordpress/i18n';

/**
Expand All @@ -16,7 +16,12 @@ const titleField: Field< BasePost > = {
id: 'title',
label: __( 'Title' ),
placeholder: __( 'No title' ),
getValue: ( { item } ) => getItemTitle( item ),
getValue: ( { item } ) => {
if ( typeof item.title === 'symbol' ) {
return item.title;
}
return getItemTitle( item );
},
render: TitleView,
enableHiding: false,
};
Expand Down
Loading