Skip to content

Commit

Permalink
chore: support for sticky params in URL and intent ops (#7429)
Browse files Browse the repository at this point in the history
* chore: support for sticky params in URL and intent ops

* chore: support for sticky params in URL and intent ops

* feat(sanity): preserve current sticky parameters when resolving intent links

* chore(sanity): use `noop` from `lodash` in test

* tests(router): reworking sticky params to support mocked testing

---------

Co-authored-by: Ash <[email protected]>
  • Loading branch information
jordanl17 and juice49 authored Sep 2, 2024
1 parent 5ebff0d commit 8bc721b
Show file tree
Hide file tree
Showing 6 changed files with 173 additions and 12 deletions.
77 changes: 73 additions & 4 deletions packages/sanity/src/router/IntentLink.test.tsx
Original file line number Diff line number Diff line change
@@ -1,10 +1,15 @@
import {describe, expect, it} from '@jest/globals'
import {describe, expect, it, jest} from '@jest/globals'
import {render} from '@testing-library/react'
import {noop} from 'lodash'

import {IntentLink} from './IntentLink'
import {route} from './route'
import {RouterProvider} from './RouterProvider'

jest.mock('./stickyParams', () => ({
STICKY_PARAMS: ['aTestStickyParam'],
}))

describe('IntentLink', () => {
it('should resolve intent link with query params', () => {
const router = route.create('/test', [route.intents('/intent')])
Expand All @@ -15,19 +20,83 @@ describe('IntentLink', () => {
id: 'document-id-123',
type: 'document-type',
}}
searchParams={[['perspective', `bundle.summer-drop`]]}
searchParams={[['aTestStickyParam', `aStickyParam.value`]]}
/>,
{
wrapper: ({children}) => (
<RouterProvider onNavigate={noop} router={router} state={{}}>
{children}
</RouterProvider>
),
},
)
// Component should render the query param in the href
expect(component.container.querySelector('a')?.href).toContain(
'/test/intent/edit/id=document-id-123;type=document-type/?aTestStickyParam=aStickyParam.value',
)
})

it('should preserve sticky parameters when resolving intent link', () => {
const router = route.create('/test', [route.intents('/intent')])
const component = render(
<IntentLink
intent="edit"
params={{
id: 'document-id-123',
type: 'document-type',
}}
/>,
{
wrapper: ({children}) => (
<RouterProvider onNavigate={() => null} router={router} state={{}}>
<RouterProvider
onNavigate={noop}
router={router}
state={{
_searchParams: [['aTestStickyParam', 'aStickyParam.value']],
}}
>
{children}
</RouterProvider>
),
},
)
// Component should render the query param in the href
expect(component.container.querySelector('a')?.href).toContain(
'/test/intent/edit/id=document-id-123;type=document-type/?perspective=bundle.summer-drop',
'/test/intent/edit/id=document-id-123;type=document-type/?aTestStickyParam=aStickyParam.value',
)
})

it('should allow sticky parameters to be overridden when resolving intent link', () => {
const router = route.create('/test', [route.intents('/intent')])
const component = render(
<IntentLink
intent="edit"
params={{
id: 'document-id-123',
type: 'document-type',
}}
searchParams={[['aTestStickyParam', `aStickyParam.value.to-be-defined`]]}
/>,
{
wrapper: ({children}) => (
<RouterProvider
onNavigate={noop}
router={router}
state={{
_searchParams: [['aTestStickyParam', 'aStickyParam.value.to-be-overridden']],
}}
>
{children}
</RouterProvider>
),
},
)
// Component should render the query param in the href
expect(component.container.querySelector('a')?.href).toContain(
'/test/intent/edit/id=document-id-123;type=document-type/?aTestStickyParam=aStickyParam.value.to-be-defined',
)
expect(component.container.querySelector('a')?.href).not.toContain(
'aTestStickyParam=aStickyParam.value.to-be-overridden',
)
})
})
85 changes: 78 additions & 7 deletions packages/sanity/src/router/RouterProvider.tsx
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
import {fromPairs, partition, toPairs} from 'lodash'
import {type ReactElement, type ReactNode, useCallback, useMemo} from 'react'
import {RouterContext} from 'sanity/_singletons'

import {STICKY_PARAMS} from './stickyParams'
import {
type IntentParameters,
type NavigateOptions,
Expand Down Expand Up @@ -87,17 +89,49 @@ export function RouterProvider(props: RouterProviderProps): ReactElement {
intent: intentName,
params,
payload,
_searchParams,
_searchParams: toPairs({
...fromPairs((state._searchParams ?? []).filter(([key]) => STICKY_PARAMS.includes(key))),
...fromPairs(_searchParams ?? []),
}),
})
},
[routerProp],
[routerProp, state._searchParams],
)

const resolvePathFromState = useCallback(
(nextState: Record<string, unknown>): string => {
return routerProp.encode(nextState)
(nextState: RouterState): string => {
const currentStateParams = state._searchParams || []
const nextStateParams = nextState._searchParams || []
const nextParams = STICKY_PARAMS.reduce((acc, param) => {
return replaceStickyParam(
acc,
param,
findParam(nextStateParams, param) ?? findParam(currentStateParams, param),
)
}, nextStateParams || [])

return routerProp.encode({
...nextState,
_searchParams: nextParams,
})
},
[routerProp, state],
)

const handleNavigateStickyParam = useCallback(
(param: string, value: string | undefined, options: NavigateOptions = {}) => {
if (!STICKY_PARAMS.includes(param)) {
throw new Error('Parameter is not sticky')
}
onNavigate({
path: resolvePathFromState({
...state,
_searchParams: [[param, value || '']],
}),
replace: options.replace,
})
},
[routerProp],
[onNavigate, resolvePathFromState, state],
)

const navigate = useCallback(
Expand All @@ -114,17 +148,54 @@ export function RouterProvider(props: RouterProviderProps): ReactElement {
[onNavigate, resolveIntentLink],
)

const [routerState, stickyParams] = useMemo(() => {
if (!state._searchParams) {
return [state, null]
}
const {_searchParams, ...rest} = state
const [sticky, restParams] = partition(_searchParams, ([key]) => STICKY_PARAMS.includes(key))
if (sticky.length === 0) {
return [state, null]
}
return [{...rest, _searchParams: restParams}, sticky]
}, [state])

const router: RouterContextValue = useMemo(
() => ({
navigate,
navigateIntent,
navigateStickyParam: handleNavigateStickyParam,
navigateUrl: onNavigate,
resolveIntentLink,
resolvePathFromState,
state,
state: routerState,
stickyParams: Object.fromEntries(stickyParams || []),
}),
[navigate, navigateIntent, onNavigate, resolveIntentLink, resolvePathFromState, state],
[
handleNavigateStickyParam,
navigate,
navigateIntent,
onNavigate,
resolveIntentLink,
resolvePathFromState,
routerState,
stickyParams,
],
)

return <RouterContext.Provider value={router}>{props.children}</RouterContext.Provider>
}

function replaceStickyParam(
current: SearchParam[],
param: string,
value: string | undefined,
): SearchParam[] {
const filtered = current.filter(([key]) => key !== param)
return value === undefined || value == '' ? filtered : [...filtered, [param, value]]
}

function findParam(searchParams: SearchParam[], key: string): string | undefined {
const entry = searchParams.find(([k]) => k === key)
return entry ? entry[1] : undefined
}
1 change: 1 addition & 0 deletions packages/sanity/src/router/stickyParams.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
export const STICKY_PARAMS: string[] = []
10 changes: 10 additions & 0 deletions packages/sanity/src/router/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -264,6 +264,11 @@ export interface RouterContextValue {
*/
navigateUrl: (opts: {path: string; replace?: boolean}) => void

/**
* Navigates to the current URL with the sticky url search param set to the given value
*/
navigateStickyParam: (param: string, value: string, options?: NavigateOptions) => void

/**
* Navigates to the given router state.
* See {@link RouterState} and {@link NavigateOptions}
Expand All @@ -280,4 +285,9 @@ export interface RouterContextValue {
* The current router state. See {@link RouterState}
*/
state: RouterState

/**
* The current router state. See {@link RouterState}
*/
stickyParams: Record<string, string | undefined>
}
8 changes: 7 additions & 1 deletion packages/sanity/src/structure/components/IntentButton.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,13 @@ export const IntentButton = forwardRef(function IntentButton(
linkRef: ForwardedRef<HTMLAnchorElement>,
) {
return (
<IntentLink {...linkProps} intent={intent.type} params={intent.params} ref={linkRef} />
<IntentLink
{...linkProps}
intent={intent.type}
params={intent.params}
ref={linkRef}
searchParams={intent.searchParams}
/>
)
}),
[intent],
Expand Down
4 changes: 4 additions & 0 deletions packages/sanity/src/structure/structureBuilder/Intent.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
import {type SearchParam} from 'sanity/router'

import {getTypeNamesFromFilter, type PartialDocumentList} from './DocumentList'
import {type StructureNode} from './StructureNodes'

Expand Down Expand Up @@ -75,6 +77,8 @@ export interface Intent {
/** Intent parameters. See {@link IntentParams}
*/
params?: IntentParams

searchParams?: SearchParam[]
}

/**
Expand Down

0 comments on commit 8bc721b

Please sign in to comment.