重要前提
安装AI Skills的关键前提是:必须科学上网,且开启TUN模式,这一点至关重要,直接决定安装能否顺利完成,在此郑重提醒三遍:科学上网,科学上网,科学上网。查看完整安装教程 →
using-nuqs by andrelandgraf/fullstackrecipes
npx skills add https://github.com/andrelandgraf/fullstackrecipes --skill using-nuqs通过 nuqs 在 URL 查询参数中管理 React 状态。涵盖 Suspense 边界、解析器、清除状态和可深度链接的对话框。
使用 nuqs 在 URL 查询参数中管理 React 状态,以实现可共享的过滤器、搜索和可深度链接的对话框。
参见:
using-nuqsnuqs 在底层使用了 useSearchParams,这需要一个 Suspense 边界。通过包装组件将使用 nuqs 的组件用 Suspense 包裹起来,以保持边界位于同一位置:
import { Suspense } from "react";
type SearchInputProps = {
placeholder?: string;
};
// 带有内置 Suspense 的公共组件
export function SearchInput(props: SearchInputProps) {
return (
<Suspense fallback={<input placeholder={props.placeholder} disabled />}>
<SearchInputClient {...props} />
</Suspense>
);
}
"use client";
import { useQueryState, parseAsString } from "nuqs";
// 使用 nuqs 的内部客户端组件
function SearchInputClient({ placeholder = "Search..." }: SearchInputProps) {
const [search, setSearch] = useQueryState("q", parseAsString.withDefault(""));
return (
<input
value={search}
onChange={(e) => setSearch(e.target.value || null)}
placeholder={placeholder}
/>
);
}
广告位招租
在这里展示您的产品或服务
触达数万 AI 开发者,精准高效
这种模式允许消费组件使用 SearchInput 而无需自己添加 Suspense。
用 useQueryState 替换 useState 以将状态同步到 URL:
"use client";
import {
useQueryState,
parseAsString,
parseAsBoolean,
parseAsArrayOf,
} from "nuqs";
// 字符串状态(搜索、过滤器)
const [search, setSearch] = useQueryState("q", parseAsString.withDefault(""));
// 布尔状态(开关)
const [showArchived, setShowArchived] = useQueryState(
"archived",
parseAsBoolean.withDefault(false),
);
// 数组状态(多选)
const [tags, setTags] = useQueryState(
"tags",
parseAsArrayOf(parseAsString).withDefault([]),
);
设置为 null 以从 URL 中移除:
// 清除单个参数
setSearch(null);
// 清除所有过滤器
function clearFilters() {
setSearch(null);
setTags(null);
setShowArchived(null);
}
当使用 .withDefault() 时,设置为 null 会清除 URL 参数但返回默认值。
使用 URL 参数控制对话框可见性,以实现可共享的链接:
import { Suspense } from "react";
type DeleteDialogProps = {
onDelete: (id: string) => Promise<void>;
};
// 带有内置 Suspense 的公共组件
export function DeleteDialog(props: DeleteDialogProps) {
return (
<Suspense fallback={null}>
<DeleteDialogClient {...props} />
</Suspense>
);
}
"use client";
import { useQueryState, parseAsString } from "nuqs";
import { AlertDialog, AlertDialogContent } from "@/components/ui/alert-dialog";
function DeleteDialogClient({ onDelete }: DeleteDialogProps) {
const [deleteId, setDeleteId] = useQueryState("delete", parseAsString);
async function handleDelete() {
if (!deleteId) return;
await onDelete(deleteId);
setDeleteId(null);
}
return (
<AlertDialog open={!!deleteId} onOpenChange={(open) => !open && setDeleteId(null)}>
<AlertDialogContent>
{/* 确认界面 */}
<Button onClick={handleDelete}>Delete</Button>
</AlertDialogContent>
</AlertDialog>
);
}
以编程方式打开对话框:
// 为特定项目打开删除对话框
setDeleteId("item-123");
// 深度链接:/items?delete=item-123
使用触发按钮打开对话框:
function ItemRow({ item }: { item: Item }) {
const [, setDeleteId] = useQueryState("delete", parseAsString);
return (
<Button variant="ghost" onClick={() => setDeleteId(item.id)}>
Delete
</Button>
);
}
每周安装量
55
代码仓库
GitHub 星标数
8
首次出现
2026年1月20日
安全审计
已安装于
opencode43
codex40
gemini-cli40
cursor40
claude-code39
antigravity33
Manage React state in URL query parameters with nuqs. Covers Suspense boundaries, parsers, clearing state, and deep-linkable dialogs.
Manage React state in URL query parameters with nuqs for shareable filters, search, and deep-linkable dialogs.
See:
using-nuqs in Fullstack Recipesnuqs uses useSearchParams behind the scenes, requiring a Suspense boundary. Wrap nuqs-using components with Suspense via a wrapper component to keep the boundary colocated:
import { Suspense } from "react";
type SearchInputProps = {
placeholder?: string;
};
// Public component with built-in Suspense
export function SearchInput(props: SearchInputProps) {
return (
<Suspense fallback={<input placeholder={props.placeholder} disabled />}>
<SearchInputClient {...props} />
</Suspense>
);
}
"use client";
import { useQueryState, parseAsString } from "nuqs";
// Internal client component that uses nuqs
function SearchInputClient({ placeholder = "Search..." }: SearchInputProps) {
const [search, setSearch] = useQueryState("q", parseAsString.withDefault(""));
return (
<input
value={search}
onChange={(e) => setSearch(e.target.value || null)}
placeholder={placeholder}
/>
);
}
This pattern allows consuming components to use SearchInput without adding Suspense themselves.
Replace useState with useQueryState to sync state to the URL:
"use client";
import {
useQueryState,
parseAsString,
parseAsBoolean,
parseAsArrayOf,
} from "nuqs";
// String state (search, filters)
const [search, setSearch] = useQueryState("q", parseAsString.withDefault(""));
// Boolean state (toggles)
const [showArchived, setShowArchived] = useQueryState(
"archived",
parseAsBoolean.withDefault(false),
);
// Array state (multi-select)
const [tags, setTags] = useQueryState(
"tags",
parseAsArrayOf(parseAsString).withDefault([]),
);
Set to null to remove from URL:
// Clear single param
setSearch(null);
// Clear all filters
function clearFilters() {
setSearch(null);
setTags(null);
setShowArchived(null);
}
When using .withDefault(), setting to null clears the URL param but returns the default value.
Control dialog visibility with URL params for shareable links:
import { Suspense } from "react";
type DeleteDialogProps = {
onDelete: (id: string) => Promise<void>;
};
// Public component with built-in Suspense
export function DeleteDialog(props: DeleteDialogProps) {
return (
<Suspense fallback={null}>
<DeleteDialogClient {...props} />
</Suspense>
);
}
"use client";
import { useQueryState, parseAsString } from "nuqs";
import { AlertDialog, AlertDialogContent } from "@/components/ui/alert-dialog";
function DeleteDialogClient({ onDelete }: DeleteDialogProps) {
const [deleteId, setDeleteId] = useQueryState("delete", parseAsString);
async function handleDelete() {
if (!deleteId) return;
await onDelete(deleteId);
setDeleteId(null);
}
return (
<AlertDialog open={!!deleteId} onOpenChange={(open) => !open && setDeleteId(null)}>
<AlertDialogContent>
{/* Confirmation UI */}
<Button onClick={handleDelete}>Delete</Button>
</AlertDialogContent>
</AlertDialog>
);
}
Open the dialog programmatically:
// Open delete dialog for specific item
setDeleteId("item-123");
// Deep link: /items?delete=item-123
Use a trigger button to open the dialog:
function ItemRow({ item }: { item: Item }) {
const [, setDeleteId] = useQueryState("delete", parseAsString);
return (
<Button variant="ghost" onClick={() => setDeleteId(item.id)}>
Delete
</Button>
);
}
Weekly Installs
55
Repository
GitHub Stars
8
First Seen
Jan 20, 2026
Security Audits
Gen Agent Trust HubPassSocketPassSnykPass
Installed on
opencode43
codex40
gemini-cli40
cursor40
claude-code39
antigravity33
React 组合模式指南:Vercel 组件架构最佳实践,提升代码可维护性
123,700 周安装