|
| 1 | +import type { UrlKeys } from './defs' |
| 2 | +import type { inferParserType, ParserMap } from './parsers' |
| 3 | + |
| 4 | +export type LoaderInput = |
| 5 | + | URL |
| 6 | + | Request |
| 7 | + | URLSearchParams |
| 8 | + | Record<string, string | string[] | undefined> |
| 9 | + | string |
| 10 | + |
| 11 | +export type LoaderOptions<Parsers extends ParserMap> = { |
| 12 | + urlKeys?: UrlKeys<Parsers> |
| 13 | +} |
| 14 | + |
| 15 | +export type LoaderFunction<Parsers extends ParserMap> = ReturnType< |
| 16 | + typeof createLoader<Parsers> |
| 17 | +> |
| 18 | + |
| 19 | +export function createLoader<Parsers extends ParserMap>( |
| 20 | + parsers: Parsers, |
| 21 | + { urlKeys = {} }: LoaderOptions<Parsers> = {} |
| 22 | +) { |
| 23 | + type ParsedSearchParams = inferParserType<Parsers> |
| 24 | + |
| 25 | + /** |
| 26 | + * Load & parse search params from (almost) any input. |
| 27 | + * |
| 28 | + * While loaders are typically used in the context of a React Router / Remix |
| 29 | + * loader function, it can also be used in Next.js API routes or |
| 30 | + * getServerSideProps functions, or even with the app router `searchParams` |
| 31 | + * page prop (sync or async), if you don't need the cache behaviours. |
| 32 | + */ |
| 33 | + function loadSearchParams( |
| 34 | + input: LoaderInput, |
| 35 | + options?: LoaderOptions<Parsers> |
| 36 | + ): ParsedSearchParams |
| 37 | + |
| 38 | + /** |
| 39 | + * Load & parse search params from (almost) any input. |
| 40 | + * |
| 41 | + * While loaders are typically used in the context of a React Router / Remix |
| 42 | + * loader function, it can also be used in Next.js API routes or |
| 43 | + * getServerSideProps functions, or even with the app router `searchParams` |
| 44 | + * page prop (sync or async), if you don't need the cache behaviours. |
| 45 | + * |
| 46 | + * Note: this async overload makes it easier to use against the `searchParams` |
| 47 | + * page prop in Next.js 15 app router: |
| 48 | + * |
| 49 | + * ```tsx |
| 50 | + * export default async function Page({ searchParams }) { |
| 51 | + * const parsedSearchParamsPromise = loadSearchParams(searchParams) |
| 52 | + * return ( |
| 53 | + * // Pre-render & stream the shell immediately |
| 54 | + * <StaticShell> |
| 55 | + * <Suspense> |
| 56 | + * // Stream the Promise down |
| 57 | + * <DynamicComponent searchParams={parsedSearchParamsPromise} /> |
| 58 | + * </Suspense> |
| 59 | + * </StaticShell> |
| 60 | + * ) |
| 61 | + * } |
| 62 | + * ``` |
| 63 | + */ |
| 64 | + function loadSearchParams( |
| 65 | + input: Promise<LoaderInput>, |
| 66 | + options?: LoaderOptions<Parsers> |
| 67 | + ): Promise<ParsedSearchParams> |
| 68 | + |
| 69 | + function loadSearchParams(input: LoaderInput | Promise<LoaderInput>) { |
| 70 | + if (input instanceof Promise) { |
| 71 | + return input.then(i => loadSearchParams(i)) |
| 72 | + } |
| 73 | + const searchParams = extractSearchParams(input) |
| 74 | + const result = {} as any |
| 75 | + for (const [key, parser] of Object.entries(parsers)) { |
| 76 | + const urlKey = urlKeys[key] ?? key |
| 77 | + const value = searchParams.get(urlKey) |
| 78 | + result[key] = parser.parseServerSide(value ?? undefined) |
| 79 | + } |
| 80 | + return result |
| 81 | + } |
| 82 | + return loadSearchParams |
| 83 | +} |
| 84 | + |
| 85 | +function extractSearchParams(input: LoaderInput): URLSearchParams { |
| 86 | + try { |
| 87 | + if (input instanceof Request) { |
| 88 | + if (input.url) { |
| 89 | + return new URL(input.url).searchParams |
| 90 | + } else { |
| 91 | + return new URLSearchParams() |
| 92 | + } |
| 93 | + } |
| 94 | + if (input instanceof URL) { |
| 95 | + return input.searchParams |
| 96 | + } |
| 97 | + if (input instanceof URLSearchParams) { |
| 98 | + return input |
| 99 | + } |
| 100 | + if (typeof input === 'object') { |
| 101 | + const entries = Object.entries(input) |
| 102 | + const searchParams = new URLSearchParams() |
| 103 | + for (const [key, value] of entries) { |
| 104 | + if (Array.isArray(value)) { |
| 105 | + for (const v of value) { |
| 106 | + searchParams.append(key, v) |
| 107 | + } |
| 108 | + } else if (value !== undefined) { |
| 109 | + searchParams.set(key, value) |
| 110 | + } |
| 111 | + } |
| 112 | + return searchParams |
| 113 | + } |
| 114 | + if (typeof input === 'string') { |
| 115 | + if ('canParse' in URL && URL.canParse(input)) { |
| 116 | + return new URL(input).searchParams |
| 117 | + } |
| 118 | + return new URLSearchParams(input) |
| 119 | + } |
| 120 | + } catch (e) { |
| 121 | + return new URLSearchParams() |
| 122 | + } |
| 123 | + return new URLSearchParams() |
| 124 | +} |
0 commit comments