diff --git a/packages/dataviews/src/components/dataviews-context/index.ts b/packages/dataviews/src/components/dataviews-context/index.ts index 87acade73bc819..19f6b4178b7b55 100644 --- a/packages/dataviews/src/components/dataviews-context/index.ts +++ b/packages/dataviews/src/components/dataviews-context/index.ts @@ -28,7 +28,6 @@ type DataViewsContextType< Item > = { getItemId: ( item: Item ) => string; onClickItem: ( item: Item ) => void; isItemClickable: ( item: Item ) => boolean; - density: number; }; const DataViewsContext = createContext< DataViewsContextType< any > >( { @@ -47,7 +46,6 @@ const DataViewsContext = createContext< DataViewsContextType< any > >( { getItemId: ( item ) => item.id, onClickItem: () => {}, isItemClickable: () => false, - density: 0, } ); export default DataViewsContext; diff --git a/packages/dataviews/src/components/dataviews-layout/index.tsx b/packages/dataviews/src/components/dataviews-layout/index.tsx index 4ef0125b1f64b5..ebc251eae36a7a 100644 --- a/packages/dataviews/src/components/dataviews-layout/index.tsx +++ b/packages/dataviews/src/components/dataviews-layout/index.tsx @@ -27,7 +27,6 @@ export default function DataViewsLayout() { selection, onChangeSelection, setOpenedFilter, - density, onClickItem, isItemClickable, } = useContext( DataViewsContext ); @@ -49,7 +48,6 @@ export default function DataViewsLayout() { onClickItem={ onClickItem } isItemClickable={ isItemClickable } view={ view } - density={ density } /> ); } diff --git a/packages/dataviews/src/components/dataviews-view-config/index.tsx b/packages/dataviews/src/components/dataviews-view-config/index.tsx index c8b26c51275891..c0f1fe4972a50e 100644 --- a/packages/dataviews/src/components/dataviews-view-config/index.tsx +++ b/packages/dataviews/src/components/dataviews-view-config/index.tsx @@ -35,7 +35,6 @@ import { useInstanceId } from '@wordpress/compose'; */ import { SORTING_DIRECTIONS, - LAYOUT_GRID, LAYOUT_TABLE, sortIcons, sortLabels, @@ -49,7 +48,6 @@ import { import type { SupportedLayouts, View, Field } from '../../types'; import DataViewsContext from '../dataviews-context'; import { unlock } from '../../lock-unlock'; -import DensityPicker from '../../dataviews-layouts/grid/density-picker'; const { Menu } = unlock( componentsPrivateApis ); @@ -512,19 +510,15 @@ function SettingsSection( { ); } -function DataviewsViewConfigDropdown( { - density, - setDensity, -}: { - density: number; - setDensity: React.Dispatch< React.SetStateAction< number > >; -} ) { +function DataviewsViewConfigDropdown() { const { view } = useContext( DataViewsContext ); const popoverId = useInstanceId( _DataViewsViewConfig, 'dataviews-view-config-dropdown' ); - + const usedLayout = VIEW_LAYOUTS.find( + ( layout ) => layout.type === view.type + ); return ( - { view.type === LAYOUT_GRID && ( - + { !! usedLayout?.viewConfigOptions && ( + ) } @@ -570,21 +561,14 @@ function DataviewsViewConfigDropdown( { } function _DataViewsViewConfig( { - density, - setDensity, defaultLayouts = { list: {}, grid: {}, table: {} }, }: { - density: number; - setDensity: React.Dispatch< React.SetStateAction< number > >; defaultLayouts?: SupportedLayouts; } ) { return ( <> - + ); } diff --git a/packages/dataviews/src/components/dataviews/index.tsx b/packages/dataviews/src/components/dataviews/index.tsx index 3e8224e61bc5d5..ee6073f40bf3d8 100644 --- a/packages/dataviews/src/components/dataviews/index.tsx +++ b/packages/dataviews/src/components/dataviews/index.tsx @@ -75,7 +75,6 @@ export default function DataViews< Item >( { header, }: DataViewsProps< Item > ) { const [ selectionState, setSelectionState ] = useState< string[] >( [] ); - const [ density, setDensity ] = useState< number >( 0 ); const isUncontrolled = selectionProperty === undefined || onChangeSelection === undefined; const selection = isUncontrolled ? selectionState : selectionProperty; @@ -119,7 +118,6 @@ export default function DataViews< Item >( { getItemId, isItemClickable, onClickItem, - density, } } >
@@ -151,8 +149,6 @@ export default function DataViews< Item >( { > { header } diff --git a/packages/dataviews/src/dataviews-layouts/grid/index.tsx b/packages/dataviews/src/dataviews-layouts/grid/index.tsx index e218172b7900aa..94977ea6bb1392 100644 --- a/packages/dataviews/src/dataviews-layouts/grid/index.tsx +++ b/packages/dataviews/src/dataviews-layouts/grid/index.tsx @@ -25,6 +25,7 @@ import { useHasAPossibleBulkAction } from '../../components/dataviews-bulk-actio import type { Action, NormalizedField, ViewGridProps } from '../../types'; import type { SetSelection } from '../../private-types'; import getClickableItemProps from '../utils/get-clickable-item-props'; +import { useChangeGridColumnsOnViewportChange } from './preview-size-picker'; interface GridItemProps< Item > { selection: string[]; @@ -192,8 +193,8 @@ export default function ViewGrid< Item >( { isItemClickable, selection, view, - density, }: ViewGridProps< Item > ) { + useChangeGridColumnsOnViewportChange(); const mediaField = fields.find( ( field ) => field.id === view.layout?.mediaField ); @@ -223,8 +224,10 @@ export default function ViewGrid< Item >( { { visibleFields: [], badgeFields: [] } ); const hasData = !! data?.length; - const gridStyle = density - ? { gridTemplateColumns: `repeat(${ density }, minmax(0, 1fr))` } + const gridStyle = view.layout?.gridColumns + ? { + gridTemplateColumns: `repeat(${ view.layout.gridColumns }, minmax(0, 1fr))`, + } : {}; return ( <> diff --git a/packages/dataviews/src/dataviews-layouts/grid/density-picker.tsx b/packages/dataviews/src/dataviews-layouts/grid/preview-size-picker.tsx similarity index 56% rename from packages/dataviews/src/dataviews-layouts/grid/density-picker.tsx rename to packages/dataviews/src/dataviews-layouts/grid/preview-size-picker.tsx index 34ddf6c3fe52f3..4789b36d4539f6 100644 --- a/packages/dataviews/src/dataviews-layouts/grid/density-picker.tsx +++ b/packages/dataviews/src/dataviews-layouts/grid/preview-size-picker.tsx @@ -4,7 +4,13 @@ import { RangeControl } from '@wordpress/components'; import { __ } from '@wordpress/i18n'; import { useViewportMatch } from '@wordpress/compose'; -import { useEffect, useMemo } from '@wordpress/element'; +import { useEffect, useMemo, useContext } from '@wordpress/element'; + +/** + * Internal dependencies + */ +import DataViewsContext from '../../components/dataviews-context'; +import type { ViewGrid } from '../../types'; const viewportBreaks = { xhuge: { min: 3, max: 6, default: 5 }, @@ -39,31 +45,41 @@ function useViewPortBreakpoint() { return null; } -export default function DensityPicker( { - density, - setDensity, -}: { - density: number; - setDensity: React.Dispatch< React.SetStateAction< number > >; -} ) { +export function useChangeGridColumnsOnViewportChange() { const viewport = useViewPortBreakpoint(); + const context = useContext( DataViewsContext ); + const view = context.view as ViewGrid; useEffect( () => { - setDensity( ( _density ) => { - if ( ! viewport || ! _density ) { - return 0; - } - const breakValues = viewportBreaks[ viewport ]; - if ( _density < breakValues.min ) { - return breakValues.min; - } - if ( _density > breakValues.max ) { - return breakValues.max; - } - return _density; - } ); - }, [ setDensity, viewport ] ); + const gridColumns = view.layout?.gridColumns; + let newGridColumns; + if ( ! viewport || ! gridColumns ) { + return; + } + const breakValues = viewportBreaks[ viewport ]; + if ( gridColumns < breakValues.min ) { + newGridColumns = breakValues.min; + } + if ( gridColumns > breakValues.max ) { + newGridColumns = breakValues.max; + } + if ( newGridColumns ) { + context.onChangeView( { + ...view, + layout: { + ...view.layout, + gridColumns: newGridColumns, + }, + } ); + } + }, [ viewport, view, context ] ); +} + +export default function DensityPicker() { + const viewport = useViewPortBreakpoint(); + const context = useContext( DataViewsContext ); + const view = context.view as ViewGrid; const breakValues = viewportBreaks[ viewport || 'mobile' ]; - const densityToUse = density || breakValues.default; + const densityToUse = view.layout?.gridColumns || breakValues.default; const marks = useMemo( () => @@ -94,7 +110,13 @@ export default function DensityPicker( { max={ breakValues.max } withInputField={ false } onChange={ ( value = 0 ) => { - setDensity( breakValues.max + breakValues.min - value ); + context.onChangeView( { + ...view, + layout: { + ...view.layout, + gridColumns: breakValues.max + breakValues.min - value, + }, + } ); } } step={ 1 } /> diff --git a/packages/dataviews/src/dataviews-layouts/index.ts b/packages/dataviews/src/dataviews-layouts/index.ts index eece17d0f4f10c..7abdd1eb459328 100644 --- a/packages/dataviews/src/dataviews-layouts/index.ts +++ b/packages/dataviews/src/dataviews-layouts/index.ts @@ -17,6 +17,8 @@ import ViewGrid from './grid'; import ViewList from './list'; import { LAYOUT_GRID, LAYOUT_LIST, LAYOUT_TABLE } from '../constants'; import type { View, Field } from '../types'; +import PreviewSizePicker from './grid/preview-size-picker'; +import DensityPicker from './table/density-picker'; export const VIEW_LAYOUTS = [ { @@ -24,12 +26,14 @@ export const VIEW_LAYOUTS = [ label: __( 'Table' ), component: ViewTable, icon: blockTable, + viewConfigOptions: DensityPicker, }, { type: LAYOUT_GRID, label: __( 'Grid' ), component: ViewGrid, icon: category, + viewConfigOptions: PreviewSizePicker, }, { type: LAYOUT_LIST, diff --git a/packages/dataviews/src/dataviews-layouts/table/density-picker.tsx b/packages/dataviews/src/dataviews-layouts/table/density-picker.tsx new file mode 100644 index 00000000000000..6d3d31aeb7345e --- /dev/null +++ b/packages/dataviews/src/dataviews-layouts/table/density-picker.tsx @@ -0,0 +1,57 @@ +/** + * WordPress dependencies + */ +import { + __experimentalToggleGroupControl as ToggleGroupControl, + __experimentalToggleGroupControlOption as ToggleGroupControlOption, +} from '@wordpress/components'; +import { __, _x } from '@wordpress/i18n'; +import { useContext } from '@wordpress/element'; + +/** + * Internal dependencies + */ +import DataViewsContext from '../../components/dataviews-context'; +import type { ViewTable, Density } from '../../types'; + +export default function DensityPicker() { + const context = useContext( DataViewsContext ); + const view = context.view as ViewTable; + return ( + { + context.onChangeView( { + ...view, + layout: { + ...view.layout, + density: value as Density, + }, + } ); + } } + isBlock + > + + + + + ); +} diff --git a/packages/dataviews/src/dataviews-layouts/table/index.tsx b/packages/dataviews/src/dataviews-layouts/table/index.tsx index 8ef41db1c38798..042b0237743e9d 100644 --- a/packages/dataviews/src/dataviews-layouts/table/index.tsx +++ b/packages/dataviews/src/dataviews-layouts/table/index.tsx @@ -328,7 +328,13 @@ function ViewTable< Item >( { return ( <> diff --git a/packages/dataviews/src/dataviews-layouts/table/style.scss b/packages/dataviews/src/dataviews-layouts/table/style.scss index ea2c614e4339df..9f7b2a89c8de52 100644 --- a/packages/dataviews/src/dataviews-layouts/table/style.scss +++ b/packages/dataviews/src/dataviews-layouts/table/style.scss @@ -169,6 +169,21 @@ opacity: 1; } } + + // Density style overrides. + &.has-compact-density { + td, + th { + padding: $grid-unit-05; + } + } + + &.has-comfortable-density { + td, + th { + padding: $grid-unit-20; + } + } } /* stylelint-disable-next-line scss/at-rule-no-unknown -- '@container' not globally permitted */ diff --git a/packages/dataviews/src/types.ts b/packages/dataviews/src/types.ts index 8c4276f2541ecc..ea2106015efbc0 100644 --- a/packages/dataviews/src/types.ts +++ b/packages/dataviews/src/types.ts @@ -329,6 +329,8 @@ export interface ColumnStyle { minWidth?: string | number; } +export type Density = 'compact' | 'balanced' | 'comfortable'; + export interface ViewTable extends ViewBase { type: 'table'; @@ -347,6 +349,11 @@ export interface ViewTable extends ViewBase { * The styles for the columns. */ styles?: Record< string, ColumnStyle >; + + /** + * The density of the view. + */ + density?: Density; }; } @@ -389,6 +396,11 @@ export interface ViewGrid extends ViewBase { * The fields to use as badge fields. */ badgeFields?: string[]; + + /** + * The number of grid columns. + */ + gridColumns?: number; }; } @@ -501,7 +513,6 @@ export interface ViewBaseProps< Item > { onClickItem: ( item: Item ) => void; isItemClickable: ( item: Item ) => boolean; view: View; - density: number; } export interface ViewTableProps< Item > extends ViewBaseProps< Item > {