Skip to content

Commit

Permalink
fix(dashboard): Use derived state in DataTable (#11487)
Browse files Browse the repository at this point in the history
**What**
- Uses derived state in DataTable, to prevent the state in the URL and component from going out of sync.
- Introduces a way for RouteModals to restore URL params on close.

Resolves CMRC-936
  • Loading branch information
kasperkristensen authored Mar 3, 2025
1 parent 51b0af1 commit c28ae57
Show file tree
Hide file tree
Showing 7 changed files with 103 additions and 39 deletions.
5 changes: 5 additions & 0 deletions .changeset/breezy-bananas-kick.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
"@medusajs/dashboard": patch
---

fix(dashboard): Use derrived state in DataTable
36 changes: 19 additions & 17 deletions packages/admin/dashboard/src/components/data-table/data-table.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ import {
Text,
useDataTable,
} from "@medusajs/ui"
import React, { ReactNode, useCallback, useState } from "react"
import React, { ReactNode, useCallback, useMemo } from "react"
import { useTranslation } from "react-i18next"
import { Link, useNavigate, useSearchParams } from "react-router-dom"

Expand Down Expand Up @@ -110,7 +110,7 @@ export const DataTable = <TData,>({
const enableCommands = commands && commands.length > 0
const enableSorting = columns.some((column) => column.enableSorting)

const filterIds = filters?.map((f) => f.id) ?? []
const filterIds = useMemo(() => filters?.map((f) => f.id) ?? [], [filters])
const prefixedFilterIds = filterIds.map((id) => getQueryParamKey(id, prefix))

const { offset, order, q, ...filterParams } = useQueryParams(
Expand All @@ -124,9 +124,11 @@ export const DataTable = <TData,>({
)
const [_, setSearchParams] = useSearchParams()

const [search, setSearch] = useState<string>(q ?? "")
const search = useMemo(() => {
return q ?? ""
}, [q])

const handleSearchChange = (value: string) => {
setSearch(value)
setSearchParams((prev) => {
if (value) {
prev.set(getQueryParamKey("q", prefix), value)
Expand All @@ -138,11 +140,13 @@ export const DataTable = <TData,>({
})
}

const [pagination, setPagination] = useState<DataTablePaginationState>(
offset ? parsePaginationState(offset, pageSize) : { pageIndex: 0, pageSize }
)
const pagination: DataTablePaginationState = useMemo(() => {
return offset
? parsePaginationState(offset, pageSize)
: { pageIndex: 0, pageSize }
}, [offset, pageSize])

const handlePaginationChange = (value: DataTablePaginationState) => {
setPagination(value)
setSearchParams((prev) => {
if (value.pageIndex === 0) {
prev.delete(getQueryParamKey("offset", prefix))
Expand All @@ -152,18 +156,16 @@ export const DataTable = <TData,>({
transformPaginationState(value).toString()
)
}

return prev
})
}

const [filtering, setFiltering] = useState<DataTableFilteringState>(
parseFilterState(filterIds, filterParams)
const filtering: DataTableFilteringState = useMemo(
() => parseFilterState(filterIds, filterParams),
[filterIds, filterParams]
)

const handleFilteringChange = (value: DataTableFilteringState) => {
setFiltering(value)

setSearchParams((prev) => {
Array.from(prev.keys()).forEach((key) => {
if (prefixedFilterIds.includes(key) && !(key in value)) {
Expand All @@ -184,11 +186,11 @@ export const DataTable = <TData,>({
})
}

const [sorting, setSorting] = useState<DataTableSortingState | null>(
order ? parseSortingState(order) : null
)
const sorting: DataTableSortingState | null = useMemo(() => {
return order ? parseSortingState(order) : null
}, [order])

const handleSortingChange = (value: DataTableSortingState) => {
setSorting(value)
setSearchParams((prev) => {
if (value) {
const valueToStore = transformSortingState(value)
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
import { useMemo } from "react"
import { Path, useLocation } from "react-router-dom"

/**
* Checks if the current location has a restore_params property.
* If it does, it will return a new path with the params added to it.
* Otherwise, it will return the previous path.
*
* This is useful if the modal needs to return to the original path, with
* the params that were present when the modal was opened.
*/
export const useStateAwareTo = (prev: string | Partial<Path>) => {
const location = useLocation()

const to = useMemo(() => {
const params = location.state?.restore_params

if (!params) {
return prev
}

return `${prev}?${params.toString()}`
}, [location.state, prev])

return to
}
Original file line number Diff line number Diff line change
@@ -1,19 +1,22 @@
import { Drawer, clx } from "@medusajs/ui"
import { PropsWithChildren, useEffect, useState } from "react"
import { useNavigate } from "react-router-dom"
import { Path, useNavigate } from "react-router-dom"
import { useStateAwareTo } from "../hooks/use-state-aware-to"
import { RouteModalForm } from "../route-modal-form"
import { RouteModalProvider } from "../route-modal-provider/route-provider"
import { StackedModalProvider } from "../stacked-modal-provider"

type RouteDrawerProps = PropsWithChildren<{
prev?: string
prev?: string | Partial<Path>
}>

const Root = ({ prev = "..", children }: RouteDrawerProps) => {
const navigate = useNavigate()
const [open, setOpen] = useState(false)
const [stackedModalOpen, onStackedModalOpen] = useState(false)

const to = useStateAwareTo(prev)

/**
* Open the modal when the component mounts. This
* ensures that the entry animation is played.
Expand All @@ -30,7 +33,7 @@ const Root = ({ prev = "..", children }: RouteDrawerProps) => {
const handleOpenChange = (open: boolean) => {
if (!open) {
document.body.style.pointerEvents = "auto"
navigate(prev, { replace: true })
navigate(to, { replace: true })
return
}

Expand All @@ -39,7 +42,7 @@ const Root = ({ prev = "..", children }: RouteDrawerProps) => {

return (
<Drawer open={open} onOpenChange={handleOpenChange}>
<RouteModalProvider prev={prev}>
<RouteModalProvider prev={to}>
<StackedModalProvider onOpenChange={onStackedModalOpen}>
<Drawer.Content
aria-describedby={undefined}
Expand Down
Original file line number Diff line number Diff line change
@@ -1,20 +1,23 @@
import { FocusModal, clx } from "@medusajs/ui"
import { PropsWithChildren, useEffect, useState } from "react"
import { useNavigate } from "react-router-dom"
import { Path, useNavigate } from "react-router-dom"
import { useStateAwareTo } from "../hooks/use-state-aware-to"
import { RouteModalForm } from "../route-modal-form"
import { useRouteModal } from "../route-modal-provider"
import { RouteModalProvider } from "../route-modal-provider/route-provider"
import { StackedModalProvider } from "../stacked-modal-provider"

type RouteFocusModalProps = PropsWithChildren<{
prev?: string
prev?: string | Partial<Path>
}>

const Root = ({ prev = "..", children }: RouteFocusModalProps) => {
const navigate = useNavigate()
const [open, setOpen] = useState(false)
const [stackedModalOpen, onStackedModalOpen] = useState(false)

const to = useStateAwareTo(prev)

/**
* Open the modal when the component mounts. This
* ensures that the entry animation is played.
Expand All @@ -31,7 +34,7 @@ const Root = ({ prev = "..", children }: RouteFocusModalProps) => {
const handleOpenChange = (open: boolean) => {
if (!open) {
document.body.style.pointerEvents = "auto"
navigate(prev, { replace: true })
navigate(to, { replace: true })
return
}

Expand All @@ -40,7 +43,7 @@ const Root = ({ prev = "..", children }: RouteFocusModalProps) => {

return (
<FocusModal open={open} onOpenChange={handleOpenChange}>
<RouteModalProvider prev={prev}>
<RouteModalProvider prev={to}>
<StackedModalProvider onOpenChange={onStackedModalOpen}>
<Content stackedModalOpen={stackedModalOpen}>{children}</Content>
</StackedModalProvider>
Expand Down
Original file line number Diff line number Diff line change
@@ -1,9 +1,9 @@
import { PropsWithChildren, useCallback, useMemo, useState } from "react"
import { useNavigate } from "react-router-dom"
import { Path, useNavigate } from "react-router-dom"
import { RouteModalProviderContext } from "./route-modal-context"

type RouteModalProviderProps = PropsWithChildren<{
prev: string
prev: string | Partial<Path>
}>

export const RouteModalProvider = ({
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ import { useCallback, useMemo } from "react"
import { useTranslation } from "react-i18next"

import { CellContext } from "@tanstack/react-table"
import { useNavigate } from "react-router-dom"
import { useNavigate, useSearchParams } from "react-router-dom"
import { DataTable } from "../../../../../components/data-table"
import { useDataTableDateColumns } from "../../../../../components/data-table/helpers/general/use-data-table-date-columns"
import { useDataTableDateFilters } from "../../../../../components/data-table/helpers/general/use-data-table-date-filters"
Expand All @@ -32,6 +32,7 @@ type ProductVariantSectionProps = {
}

const PAGE_SIZE = 10
const PREFIX = "pv"

export const ProductVariantSection = ({
product,
Expand All @@ -46,15 +47,18 @@ export const ProductVariantSection = ({
manage_inventory,
created_at,
updated_at,
} = useQueryParams([
"q",
"order",
"offset",
"manage_inventory",
"allow_backorder",
"created_at",
"updated_at",
])
} = useQueryParams(
[
"q",
"order",
"offset",
"manage_inventory",
"allow_backorder",
"created_at",
"updated_at",
],
PREFIX
)

const columns = useColumns(product)
const filters = useFilters()
Expand Down Expand Up @@ -132,6 +136,7 @@ export const ProductVariantSection = ({
],
}}
commands={commands}
prefix={PREFIX}
/>
</Container>
)
Expand All @@ -145,6 +150,17 @@ const useColumns = (product: HttpTypes.AdminProduct) => {
const navigate = useNavigate()
const { mutateAsync } = useDeleteVariantLazy(product.id)
const prompt = usePrompt()
const [searchParams] = useSearchParams()

const tableSearchParams = useMemo(() => {
const filtered = new URLSearchParams()
for (const [key, value] of searchParams.entries()) {
if (key.startsWith(`${PREFIX}_`)) {
filtered.append(key, value)
}
}
return filtered
}, [searchParams])

const dateColumns = useDataTableDateColumns<HttpTypes.AdminProductVariant>()

Expand Down Expand Up @@ -215,7 +231,16 @@ const useColumns = (product: HttpTypes.AdminProduct) => {
icon: <PencilSquare />,
label: t("actions.edit"),
onClick: (row) => {
navigate(`edit-variant?variant_id=${row.row.original.id}`)
navigate(
`edit-variant?variant_id=${
row.row.original.id
}&${tableSearchParams.toString()}`,
{
state: {
restore_params: tableSearchParams.toString(),
},
}
)
},
},
]
Expand Down Expand Up @@ -271,7 +296,7 @@ const useColumns = (product: HttpTypes.AdminProduct) => {

return [mainActions, secondaryActions]
},
[handleDelete, navigate, t]
[handleDelete, navigate, t, tableSearchParams]
)

const getInventory = useCallback(
Expand Down

0 comments on commit c28ae57

Please sign in to comment.