| name | search-params |
| description | URL search param and hash state management. Use when adding or modifying URL search params, working with useSearchParams, setSearchParams, useSearchParamState, or navigate() with query strings or hash fragments, or fixing browser back/forward button issues. |
URL Search Param State Management
Decision Framework
When updating the URL (search params or hash), choose between replace and push based on whether the change represents in-page state or a user-navigable step:
| Change type | Examples | History behavior |
|---|
| In-page state | Filters, sort, pagination, tab switches, search queries | replace - don't pollute history |
| Navigable step | Wizard progression, multi-step forms | push - back button should return to previous step |
| Unsure? | | Ask the developer before choosing |
Why this matters: Pushing in-page state changes clutters the browser history. Users clicking "back" expect to leave the page, not undo a filter toggle. This is the #1 cause of "back button is broken" bugs.
Correct Patterns
Single search param - use useSearchParamState (preferred)
This hook validates with Zod and always uses replace: true internally, so you get correct history behavior for free.
import { useSearchParamState } from '@app/hooks/useSearchParamState';
import { z } from 'zod';
const TabSchema = z.enum(['overview', 'details', 'settings']);
const [activeTab, setActiveTab] = useSearchParamState('tab', TabSchema, 'overview');
Key file: src/app/src/hooks/useSearchParamState.ts
Multiple search params - use setSearchParams with replace: true
When updating multiple params at once, use setSearchParams directly but always pass { replace: true } for in-page state:
const [searchParams, setSearchParams] = useSearchParams();
setSearchParams(
(params) => {
params.set('status', 'active');
params.set('sort', 'name');
return params;
},
{ replace: true },
);
Hash-based navigation - navigate() with replace or push
For wizard/multi-step flows where back button should traverse steps, use push (the default):
const updateHash = (newStep: string) => {
navigate(`#${newStep}`);
};
For hash changes that represent in-page state, use replace:
navigate(`#${section}`, { replace: true });
URL normalization after save
When the URL needs to be updated to include a new ID after a create/save operation (not a user action), use replace:
navigate(`/evals/${newConfigId}`, { replace: true });
Anti-Patterns
Pushing in-page state changes (breaks back button)
setSearchParams((params) => {
params.set('filter', value);
return params;
});
navigate(`?tab=${newTab}`);
Using raw useSearchParams for a single param without validation
const [searchParams, setSearchParams] = useSearchParams();
const tab = searchParams.get('tab');
const setTab = (v: string) => {
setSearchParams((p) => {
p.set('tab', v);
return p;
});
};
const [tab, setTab] = useSearchParamState('tab', TabSchema, 'overview');
Using empty strings instead of null
setTab('');
setTab(null);
Key Files
src/app/src/hooks/useSearchParamState.ts - primary hook (uses replace internally)
src/app/src/pages/eval/components/ResultsView.tsx - example of correct { replace: true } usage
src/app/src/pages/redteam/setup/page.tsx - example of intentional push for wizard steps