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

chore: refactor sanitizeOrderable #11838

Closed
wants to merge 64 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
64 commits
Select commit Hold shift + click to select a range
e83dcf8
add order field
GermanJablo Feb 26, 2025
e2e8850
add endpoint and dragHandle to table
GermanJablo Feb 27, 2025
c3c0c13
make table rows draggable
GermanJablo Feb 27, 2025
add4e1d
add onDragEnd handler (call to endpoint)
GermanJablo Feb 27, 2025
bab956a
save WIP
GermanJablo Feb 27, 2025
606ab59
it works
GermanJablo Feb 28, 2025
13577b8
test/sort temporal changes
GermanJablo Feb 28, 2025
5155b62
changes in table.tsx
GermanJablo Feb 28, 2025
2fd71c4
rename field to payload-order to _order
GermanJablo Mar 3, 2025
c51ef57
rename enableCustomOrder to isSortable for consistency
GermanJablo Mar 3, 2025
8bc150d
add experimental notice to isSortable field documentation
GermanJablo Mar 3, 2025
48698f8
add docs
GermanJablo Mar 3, 2025
4ea24c2
show integers in the UI as order keys
GermanJablo Mar 3, 2025
e3eb280
use useListQuery. Revert use integer as order key
GermanJablo Mar 3, 2025
856cec6
save WIP
GermanJablo Mar 3, 2025
497a8ad
Revert "save WIP"
GermanJablo Mar 3, 2025
bb4c398
remove unnecesary logs
GermanJablo Mar 4, 2025
95849b3
remove unnecessary changes
GermanJablo Mar 4, 2025
3f1ea0e
simplify render cells
GermanJablo Mar 4, 2025
5bdb564
revert Cell component
GermanJablo Mar 4, 2025
168d81c
revert changes in ListQuery
GermanJablo Mar 4, 2025
5a1f0b2
fix bug with col actives
GermanJablo Mar 4, 2025
7c6327d
remove onDragEnd prop
GermanJablo Mar 4, 2025
7aa14d6
remove unnecessary code
GermanJablo Mar 4, 2025
895dcd7
simplify map
GermanJablo Mar 4, 2025
de67721
add fetch with optmistic update again + toast
GermanJablo Mar 4, 2025
7e83302
show integers as order keys instead of fractional indexes
GermanJablo Mar 4, 2025
882d508
add required and unique to order field
GermanJablo Mar 4, 2025
364fe87
missing type
GermanJablo Mar 4, 2025
ee0cf01
revert to show fractional indexes. Prevent manual reordering if table…
GermanJablo Mar 5, 2025
fe8f305
last commit
GermanJablo Mar 5, 2025
576cfd9
resolve comments
GermanJablo Mar 6, 2025
5df47a8
improve performance sending keys instead of id (WIP)
GermanJablo Mar 6, 2025
ccb9308
prevent race conditions
GermanJablo Mar 6, 2025
bfbd5a8
The endpoint now receives target and position to operate in any scenario
GermanJablo Mar 6, 2025
946a9b9
setup e2e tests
GermanJablo Mar 6, 2025
b60aee8
add first test
GermanJablo Mar 6, 2025
92939f6
tests done!
GermanJablo Mar 7, 2025
84d59b3
Merge remote-tracking branch 'origin/main' into feat/sortable-collect…
GermanJablo Mar 7, 2025
0ce26e3
fix error
GermanJablo Mar 7, 2025
38189f7
fix type
GermanJablo Mar 7, 2025
271bb42
copy-paste fractional-indexing instead of installing it
GermanJablo Mar 7, 2025
afec8b5
Prevent reordering if user doesn't have editing permissions
GermanJablo Mar 7, 2025
ac3f87d
revert columnsToUse
GermanJablo Mar 8, 2025
c7ca876
fix ids
GermanJablo Mar 8, 2025
374de05
fix ids
GermanJablo Mar 8, 2025
d86a2a6
trying to fix flaky test
GermanJablo Mar 9, 2025
a7dc272
add shimmer effect
GermanJablo Mar 10, 2025
d79f5c6
hide order column from table
GermanJablo Mar 11, 2025
a222a6e
change header
GermanJablo Mar 11, 2025
6150253
add dragHandleVertical icon
GermanJablo Mar 13, 2025
bcf7a55
add sortHeader icon
GermanJablo Mar 13, 2025
e1f0773
rename isSortable to orderable
GermanJablo Mar 13, 2025
5f59407
add sortHeader icon 2
GermanJablo Mar 13, 2025
2f1942e
remove comment
GermanJablo Mar 13, 2025
769833a
add slug prop to table
GermanJablo Mar 13, 2025
c1fde58
style drag handle as disabled when table is sorted by another column
GermanJablo Mar 13, 2025
6aae3ad
create SortableTable component
GermanJablo Mar 13, 2025
5d22f93
Merge remote-tracking branch 'origin/main' into feat/sortable-collect…
GermanJablo Mar 13, 2025
950912a
fix test
GermanJablo Mar 13, 2025
4acace2
Revert "add dragHandleVertical icon"
GermanJablo Mar 17, 2025
743b456
improve icons
GermanJablo Mar 17, 2025
01de4ef
Improve the toggle sort button and extract logic to a custom hook
GermanJablo Mar 17, 2025
099ec38
chore: refactor sanitizeOrderable
DanRibbens Mar 24, 2025
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions docs/configuration/collections.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -70,6 +70,7 @@ The following options are available:
| `fields` * | Array of field types that will determine the structure and functionality of the data stored within this Collection. [More details](../fields/overview). |
| `graphQL` | Manage GraphQL-related properties for this collection. [More](#graphql) |
| `hooks` | Entry point for Hooks. [More details](../hooks/overview#collection-hooks). |
| `orderable` | If true, enables custom ordering for the collection, and documents can be reordered via drag and drop. Uses [fractional indexing](https://observablehq.com/@dgreensp/implementing-fractional-indexing) for efficient reordering. |
| `labels` | Singular and plural labels for use in identifying this Collection throughout Payload. Auto-generated from slug if not defined. |
| `lockDocuments` | Enables or disables document locking. By default, document locking is enabled. Set to an object to configure, or set to `false` to disable locking. [More details](../admin/locked-documents). |
| `slug` * | Unique, URL-friendly string that will act as an identifier for this Collection. |
Expand Down
4 changes: 4 additions & 0 deletions packages/payload/src/collections/config/sanitize.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
// @ts-strict-ignore

import type { Config, SanitizedConfig } from '../../config/types.js'
import type {
CollectionConfig,
Expand Down Expand Up @@ -27,6 +28,7 @@ import {
} from './defaults.js'
import { sanitizeAuthFields, sanitizeUploadFields } from './reservedFieldNames.js'
import { sanitizeCompoundIndexes } from './sanitizeCompoundIndexes.js'
import { sanitizeOrderable } from './sanitizeOrderable.js'
import { validateUseAsTitle } from './useAsTitle.js'

export const sanitizeCollection = async (
Expand Down Expand Up @@ -237,6 +239,8 @@ export const sanitizeCollection = async (

const sanitizedConfig = sanitized as SanitizedCollectionConfig

sanitizeOrderable(sanitizedConfig)

sanitizedConfig.joins = joins
sanitizedConfig.polymorphicJoins = polymorphicJoins

Expand Down
60 changes: 60 additions & 0 deletions packages/payload/src/collections/config/sanitizeOrderable.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,60 @@
import type { Field } from '../../fields/config/types.js'
import type { BeforeChangeHook, SanitizedCollectionConfig } from './types.js'

import { generateKeyBetween } from '../../utilities/fractional-indexing.js'
import { getReorderEndpoint } from '../endpoints/reorder.js'

export const ORDER_FIELD_NAME = '_order'

export const sanitizeOrderable = (collection: SanitizedCollectionConfig) => {
if (!collection.orderable) {
return
}
// 1. Add field
const orderField: Field = {
name: ORDER_FIELD_NAME,
type: 'text',
admin: {
disableBulkEdit: true,
disabled: true,
disableListColumn: true,
disableListFilter: true,
hidden: true,
readOnly: true,
},
index: true,
label: ({ t }) => t('general:order'),
required: true,
unique: true,
}

collection.fields.unshift(orderField)

// 2. Add hook
const orderBeforeChangeHook: BeforeChangeHook = async ({ data, operation, req }) => {
// Only set _order on create, not on update (unless explicitly provided)
if (operation === 'create') {
// Find the last document to place this one after
const lastDoc = await req.payload.find({
collection: collection.slug,
depth: 0,
limit: 1,
pagination: false,
select: { [ORDER_FIELD_NAME]: true },
sort: `-${ORDER_FIELD_NAME}`,
})

const lastOrderValue: null | string = (lastDoc.docs[0]?.[ORDER_FIELD_NAME] as string) || null
data[ORDER_FIELD_NAME] = generateKeyBetween(lastOrderValue, null)
}

return data
}

collection.hooks.beforeChange.push(orderBeforeChangeHook)

// 3. Add endpoint
if (collection.endpoints !== false) {
collection.endpoints.push(getReorderEndpoint(collection))
}
}
11 changes: 11 additions & 0 deletions packages/payload/src/collections/config/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -495,6 +495,17 @@ export type CollectionConfig<TSlug extends CollectionSlug = any> = {
duration: number
}
| false
/**
* If true, enables custom ordering for the collection, and documents in the listView can be reordered via drag and drop.
* New documents are inserted at the end of the list according to this parameter.
*
* Under the hood, a field with {@link https://observablehq.com/@dgreensp/implementing-fractional-indexing|fractional indexing} is used to optimize inserts and reorderings.
*
* @default false
*
* @experimental There may be frequent breaking changes to this API
*/
orderable?: boolean
slug: string
/**
* Add `createdAt` and `updatedAt` fields
Expand Down
131 changes: 131 additions & 0 deletions packages/payload/src/collections/endpoints/reorder.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,131 @@
import type { Endpoint, PayloadHandler } from '../../config/types.js'
import type { PayloadRequest, SanitizedCollectionConfig } from '../../index.js'

import { APIError, executeAccess } from '../../index.js'
import { generateNKeysBetween } from '../../utilities/fractional-indexing.js'
import { ORDER_FIELD_NAME } from '../config/sanitizeOrderable.js'

export const getReorderEndpoint = (
collection: SanitizedCollectionConfig,
): Omit<Endpoint, 'root'> => {
const reorderHandler: PayloadHandler = async (req: PayloadRequest) => {
if (!req.json) {
throw new APIError('Unreachable')
}
const body = await req.json()
type KeyAndID = {
id: string
key: string
}
const { docsToMove, newKeyWillBe, target } = body as {
// array of docs IDs to be moved before or after the target
docsToMove: string[]
// new key relative to the target. We don't use "after" or "before" as
// it can be misleading if the table is sorted in descending order.
newKeyWillBe: 'greater' | 'less'
target: KeyAndID
}

if (!Array.isArray(docsToMove) || docsToMove.length === 0) {
return new Response(JSON.stringify({ error: 'docsToMove must be a non-empty array' }), {
headers: { 'Content-Type': 'application/json' },
status: 400,
})
}
if (
typeof target !== 'object' ||
typeof target.id !== 'string' ||
typeof target.key !== 'string'
) {
return new Response(JSON.stringify({ error: 'target must be an object with id and key' }), {
headers: { 'Content-Type': 'application/json' },
status: 400,
})
}
if (newKeyWillBe !== 'greater' && newKeyWillBe !== 'less') {
return new Response(JSON.stringify({ error: 'newKeyWillBe must be "greater" or "less"' }), {
headers: { 'Content-Type': 'application/json' },
status: 400,
})
}

// Prevent reordering if user doesn't have editing permissions
await executeAccess(
{
// Currently only one doc can be moved at a time. We should review this if we want to allow
// multiple docs to be moved at once in the future.
id: docsToMove[0],
data: {},
req,
},
collection.access.update,
)

const targetId = target.id
let targetKey: null | string = target.key

// If targetKey = pending, we need to find its current key.
// This can only happen if the user reorders rows quickly with a slow connection.
if (targetKey === 'pending') {
const beforeDoc = await req.payload.findByID({
id: targetId,
collection: collection.slug,
depth: 0,
select: { [ORDER_FIELD_NAME]: true },
})
targetKey = (beforeDoc?.[ORDER_FIELD_NAME] as string) || null
}

// The reason the endpoint does not receive this docId as an argument is that there
// are situations where the user may not see or know what the next or previous one is. For
// example, access control restrictions, if docBefore is the last one on the page, etc.
const adjacentDoc = await req.payload.find({
collection: collection.slug,
depth: 0,
limit: 1,
pagination: false,
select: { [ORDER_FIELD_NAME]: true },
sort: newKeyWillBe === 'greater' ? ORDER_FIELD_NAME : `-${ORDER_FIELD_NAME}`,
where: {
[ORDER_FIELD_NAME]: {
[newKeyWillBe === 'greater' ? 'greater_than' : 'less_than']: targetKey,
},
},
})
const adjacentDocKey: null | string =
(adjacentDoc.docs?.[0]?.[ORDER_FIELD_NAME] as string) || null

// Currently N (= docsToMove.length) is always 1. Maybe in the future we will
// allow dragging and reordering multiple documents at once via the UI.
const orderValues =
newKeyWillBe === 'greater'
? generateNKeysBetween(targetKey, adjacentDocKey, docsToMove.length)
: generateNKeysBetween(adjacentDocKey, targetKey, docsToMove.length)

// Update each document with its new order value

for (const id of docsToMove) {
await req.payload.update({
id,
collection: collection.slug,
data: {
[ORDER_FIELD_NAME]: orderValues.shift(),
},
depth: 0,
req,
select: { [ORDER_FIELD_NAME]: true },
})
}

return new Response(JSON.stringify({ orderValues, success: true }), {
headers: { 'Content-Type': 'application/json' },
status: 200,
})
}

return {
handler: reorderHandler,
method: 'post',
path: '/reorder',
}
}
Loading