mirror of
https://github.com/zhigang1992/react-navigation.git
synced 2026-03-06 17:34:59 +08:00
fix: only use the query params for focused route in path
This commit is contained in:
@@ -18,6 +18,19 @@ type ConfigItem = {
|
||||
screens?: Record<string, ConfigItem>;
|
||||
};
|
||||
|
||||
const getActiveRoute = (state: State): { name: string; params?: object } => {
|
||||
const route =
|
||||
typeof state.index === 'number'
|
||||
? state.routes[state.index]
|
||||
: state.routes[state.routes.length - 1];
|
||||
|
||||
if (route.state) {
|
||||
return getActiveRoute(route.state);
|
||||
}
|
||||
|
||||
return route;
|
||||
};
|
||||
|
||||
/**
|
||||
* Utility to serialize a navigation state object to a path string.
|
||||
*
|
||||
@@ -69,7 +82,8 @@ export default function getPathFromState(
|
||||
|
||||
let pattern: string | undefined;
|
||||
|
||||
let currentParams: Record<string, any> = { ...route.params };
|
||||
let focusedParams: Record<string, any> | undefined;
|
||||
let focusedRoute = getActiveRoute(state);
|
||||
let currentOptions = configs;
|
||||
|
||||
// Keep all the route names that appeared during going deeper in config in case the pattern is resolved to undefined
|
||||
@@ -85,7 +99,7 @@ export default function getPathFromState(
|
||||
if (route.params) {
|
||||
const stringify = currentOptions[route.name]?.stringify;
|
||||
|
||||
currentParams = fromEntries(
|
||||
const currentParams = fromEntries(
|
||||
Object.entries(route.params).map(([key, value]) => [
|
||||
key,
|
||||
stringify?.[key] ? stringify[key](value) : String(value),
|
||||
@@ -95,6 +109,26 @@ export default function getPathFromState(
|
||||
if (pattern) {
|
||||
Object.assign(allParams, currentParams);
|
||||
}
|
||||
|
||||
if (focusedRoute === route) {
|
||||
// If this is the focused route, keep the params for later use
|
||||
// We save it here since it's been stringified already
|
||||
focusedParams = { ...currentParams };
|
||||
|
||||
pattern
|
||||
?.split('/')
|
||||
.filter((p) => p.startsWith(':'))
|
||||
// eslint-disable-next-line no-loop-func
|
||||
.forEach((p) => {
|
||||
const name = getParamName(p);
|
||||
|
||||
// Remove the params present in the pattern since we'll only use the rest for query string
|
||||
if (focusedParams) {
|
||||
// eslint-disable-next-line @typescript-eslint/no-dynamic-delete
|
||||
delete focusedParams[name];
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
// If there is no `screens` property or no nested state, we return pattern
|
||||
@@ -128,18 +162,12 @@ export default function getPathFromState(
|
||||
path += pattern
|
||||
.split('/')
|
||||
.map((p) => {
|
||||
const name = p.replace(/^:/, '').replace(/\?$/, '');
|
||||
const name = getParamName(p);
|
||||
|
||||
// If the path has a pattern for a param, put the param in the path
|
||||
if (p.startsWith(':')) {
|
||||
const value = allParams[name];
|
||||
|
||||
// Remove the used value from the params object since we'll use the rest for query string
|
||||
if (currentParams) {
|
||||
// eslint-disable-next-line @typescript-eslint/no-dynamic-delete
|
||||
delete currentParams[name];
|
||||
}
|
||||
|
||||
if (value === undefined && p.endsWith('?')) {
|
||||
// Optional params without value assigned in route.params should be ignored
|
||||
return '';
|
||||
@@ -155,17 +183,21 @@ export default function getPathFromState(
|
||||
path += encodeURIComponent(route.name);
|
||||
}
|
||||
|
||||
if (!focusedParams) {
|
||||
focusedParams = focusedRoute.params;
|
||||
}
|
||||
|
||||
if (route.state) {
|
||||
path += '/';
|
||||
} else if (currentParams) {
|
||||
for (let param in currentParams) {
|
||||
if (currentParams[param] === 'undefined') {
|
||||
} else if (focusedParams) {
|
||||
for (let param in focusedParams) {
|
||||
if (focusedParams[param] === 'undefined') {
|
||||
// eslint-disable-next-line @typescript-eslint/no-dynamic-delete
|
||||
delete currentParams[param];
|
||||
delete focusedParams[param];
|
||||
}
|
||||
}
|
||||
|
||||
const query = queryString.stringify(currentParams);
|
||||
const query = queryString.stringify(focusedParams);
|
||||
|
||||
if (query) {
|
||||
path += `?${query}`;
|
||||
@@ -189,6 +221,9 @@ const fromEntries = <K extends string, V>(entries: (readonly [K, V])[]) =>
|
||||
return acc;
|
||||
}, {} as Record<K, V>);
|
||||
|
||||
const getParamName = (pattern: string) =>
|
||||
pattern.replace(/^:/, '').replace(/\?$/, '');
|
||||
|
||||
const joinPaths = (...paths: string[]): string =>
|
||||
([] as string[])
|
||||
.concat(...paths.map((p) => p.split('/')))
|
||||
|
||||
Reference in New Issue
Block a user