如何修复Remix错误:必须在数据路由器内使用useFetcher?

huangapple go评论107阅读模式
英文:

How do I fix the Remix Error: useFetcher must be used within a data router?

问题

这是您提供的文本的中文翻译:

"我是 Remix 的新手(也是后端编程的新手),在解决问题时感到很迷茫。我正在尝试使用 UseFetcher 允许在“类似待办事项”的应用程序中进行非导航数据的变化。Remix 文档并没有明确说明我需要在数据路由中使用它,示例也没有帮助我搞清楚。

以下是我在 root.tsx 中的 App 组件的样子:

  1. export default function App() {
  2. return (
  3. <html lang="en" className="h-full">
  4. <head>
  5. <Meta />
  6. <Links />
  7. </head>
  8. <body className="h-full">
  9. <Outlet />
  10. <ScrollRestoration />
  11. <Scripts />
  12. <LiveReload />
  13. </body>
  14. </html>
  15. );
  16. }

以及用于显示项目列表的 routes/goods.tsx(其中大部分是从默认的 Indie Stack 改编的):

  1. export async function action({ request }: ActionArgs) {
  2. const formData = await request.formData();
  3. const title = formData.get("title");
  4. const id = formData.get("id");
  5. if (typeof title !== "string" || title.length === 0) {
  6. return json(
  7. { errors: { title: "Title is required" } },
  8. { status: 400 }
  9. );
  10. }
  11. const good = await updateGood({ title, id });
  12. return null;
  13. }
  14. export default function GoodsPage() {
  15. const data = useLoaderData<typeof loader>();
  16. const user = useUser();
  17. return (
  18. <div className="flex h-full min-h-screen flex-col">
  19. <Outlet />
  20. <main className="flex h-full bg-white">
  21. <div className="h-full w-80 border-r bg-gray-50">
  22. {data.completedGoodListItems.length === 0 ? (
  23. <p className="p-4">No goods yet</p>
  24. ) : (
  25. <>
  26. <h2>Incomplete</h2>
  27. <ol>
  28. {data.completedGoodListItems.map((good) => (
  29. <GoodItem key={good.id} good={good}></GoodItem>
  30. ))}
  31. </ol>
  32. </>
  33. )}
  34. <>
  35. <h2>Completed</h2>
  36. <ol>
  37. {data.incompleteGoodListItems.map((good) => (
  38. <GoodItem key={good.id} good={good}></GoodItem>
  39. ))}
  40. </ol>
  41. </>
  42. </div>
  43. <Form action="/logout" method="post">
  44. <button
  45. type="submit"
  46. className="rounded bg-slate-600 py-2 px-4 text-blue-100 hover:bg-blue-500 active:bg-blue-600"
  47. >
  48. Logout
  49. </button>
  50. </Form>
  51. </main>
  52. </div>
  53. );
  54. }
  55. function GoodItem({ good }) {
  56. const fetcher = useFetcher();
  57. return (
  58. <li>
  59. <fetcher.Form method="post">
  60. <input type="text" defaultValue={good.title}></input>
  61. </fetcher.Form>
  62. </li>
  63. );
  64. }

这导致了错误:Error: useFetcher must be used within a data router.

然后我尝试遵循将 App 封装在数据路由内的说明,使用 createBrowserRouter 编写了我的 root.tsx 中的以下代码:

  1. async function loader({ request }: LoaderArgs) {
  2. return json({
  3. user: await getUser(request),
  4. });
  5. }
  6. const router = createBrowserRouter([
  7. {
  8. path: "/",
  9. element: <App />,
  10. // loader: rootLoader,
  11. children: [
  12. {
  13. path: "/goods",
  14. element: <GoodsPage />,
  15. // loader: loaderName,
  16. },
  17. ],
  18. },
  19. ]);
  20. // @ts-ignore
  21. ReactDOM.createRoot(document.getElementById("root")).render(
  22. <RouterProvider router={router} />
  23. );
  24. export default function App() {
  25. return (
  26. <html lang="en" className="h-full">
  27. <head>
  28. <Meta />
  29. <Links />
  30. </head>
  31. <body className="h-full">
  32. <Outlet />
  33. <ScrollRestoration />
  34. <Scripts />
  35. <LiveReload />
  36. </body>
  37. </html>
  38. );
  39. }

我不知道为每个元素添加什么加载程序。我尝试将异步加载程序分配给一个常量,并将其添加到路由的构造函数中,但是收到错误:The expected type comes from property 'loader' which is declared here on type 'RouteObject',所以我将它留空了。

这段代码导致了 ReferenceError: document is not defined 的错误,这显然是因为我没有正确使用此路由的语法或结构。有人可以提供关于我在这种情况下应该如何使用 createBrowserRouter 的指导吗?我知道我需要以某种方式使用 RouterProvider 组件,但我没有足够的经验来看出前进的路径。我在这里漏掉了什么?"

请注意,这是您提供的文本的中文翻译,如果您需要进一步的帮助或有任何问题,请随时提出。

英文:

I'm new to Remix (and backend programming in general) and feeling lost troubleshooting this. I'm trying to UseFetcher to allow for non-navigational data mutations in a "todo-like" application. Remix docs doesn't explicitly say I need to be using it within a data router, and the examples don't clear up my confusion at all.

Here's what my App component looks like in root.tsx:

  1. export default function App() {
  2. return (
  3. &lt;html lang=&quot;en&quot; className=&quot;h-full&quot;&gt;
  4. &lt;head&gt;
  5. &lt;Meta /&gt;
  6. &lt;Links /&gt;
  7. &lt;/head&gt;
  8. &lt;body className=&quot;h-full&quot;&gt;
  9. &lt;Outlet /&gt;
  10. &lt;ScrollRestoration /&gt;
  11. &lt;Scripts /&gt;
  12. &lt;LiveReload /&gt;
  13. &lt;/body&gt;
  14. &lt;/html&gt;
  15. );
  16. }

And my routes/goods.tsx for displaying a list of items (much of this is adapted from the default Indie Stack):

  1. export async function action({ request }: ActionArgs) {
  2. const formData = await request.formData();
  3. const title = formData.get(&quot;title&quot;);
  4. const id = formData.get(&quot;id&quot;);
  5. if (typeof title !== &quot;string&quot; || title.length === 0) {
  6. return json(
  7. { errors: { title: &quot;Title is required&quot; } },
  8. { status: 400 }
  9. );
  10. }
  11. const good = await updateGood({ title, id });
  12. return null;
  13. }
  14. export default function GoodsPage() {
  15. const data = useLoaderData&lt;typeof loader&gt;();
  16. const user = useUser();
  17. return (
  18. &lt;div className=&quot;flex h-full min-h-screen flex-col&quot;&gt;
  19. &lt;Outlet /&gt;
  20. &lt;main className=&quot;flex h-full bg-white&quot;&gt;
  21. &lt;div className=&quot;h-full w-80 border-r bg-gray-50&quot;&gt;
  22. {data.completedGoodListItems.length === 0 ? (
  23. &lt;p className=&quot;p-4&quot;&gt;No goods yet&lt;/p&gt;
  24. ) : (
  25. &lt;&gt;
  26. &lt;h2&gt;Incomplete&lt;/h2&gt;
  27. &lt;ol&gt;
  28. {data.completedGoodListItems.map((good) =&gt; (
  29. &lt;GoodItem key={good.id} good={good}&gt;&lt;/GoodItem&gt;
  30. ))}
  31. &lt;/ol&gt;
  32. &lt;/&gt;
  33. )}
  34. &lt;&gt;
  35. &lt;h2&gt;Completed&lt;/h2&gt;
  36. &lt;ol&gt;
  37. {data.incompleteGoodListItems.map((good) =&gt; (
  38. &lt;GoodItem key={good.id} good={good}&gt;&lt;/GoodItem&gt;
  39. ))}
  40. &lt;/ol&gt;
  41. &lt;/&gt;
  42. &lt;/div&gt;
  43. &lt;Form action=&quot;/logout&quot; method=&quot;post&quot;&gt;
  44. &lt;button
  45. type=&quot;submit&quot;
  46. className=&quot;rounded bg-slate-600 py-2 px-4 text-blue-100 hover:bg-blue-500 active:bg-blue-600&quot;
  47. &gt;
  48. Logout
  49. &lt;/button&gt;
  50. &lt;/Form&gt;
  51. &lt;/main&gt;
  52. &lt;/div&gt;
  53. );
  54. }
  55. function GoodItem ({ good }) {
  56. const fetcher = useFetcher();
  57. return (
  58. &lt;li&gt;
  59. &lt;fetcher.Form method=&quot;post&quot;&gt;
  60. &lt;input type=&quot;text&quot; defaultValue={good.title}&gt;&lt;/input&gt;
  61. &lt;/fetcher.Form&gt;
  62. &lt;/li&gt;
  63. )}

This results in Error: useFetcher must be used within a data router.

So then I try to follow the instructions for encapsulating the App within a data router using createBrowserRouter which leads me to writing this code in my root.tsx:

  1. async function loader({ request }: LoaderArgs) {
  2. return json({
  3. user: await getUser(request),
  4. });
  5. }
  6. const router = createBrowserRouter([
  7. {
  8. path: &quot;/&quot;,
  9. element: &lt;App /&gt;,
  10. // loader: rootLoader,
  11. children: [
  12. {
  13. path: &quot;/goods&quot;,
  14. element: &lt;GoodsPage /&gt;,
  15. // loader: loaderName,
  16. },
  17. ],
  18. },
  19. ]);
  20. // @ts-ignore
  21. ReactDOM.createRoot(document.getElementById(&quot;root&quot;)).render(
  22. &lt;RouterProvider router={router} /&gt;
  23. );
  24. export default function App() {
  25. return (
  26. &lt;html lang=&quot;en&quot; className=&quot;h-full&quot;&gt;
  27. &lt;head&gt;
  28. &lt;Meta /&gt;
  29. &lt;Links /&gt;
  30. &lt;/head&gt;
  31. &lt;body className=&quot;h-full&quot;&gt;
  32. &lt;Outlet /&gt;
  33. &lt;ScrollRestoration /&gt;
  34. &lt;Scripts /&gt;
  35. &lt;LiveReload /&gt;
  36. &lt;/body&gt;
  37. &lt;/html&gt;
  38. );
  39. }

I didn't know what to add for the loaders for each element. I tried assigning the async loader to a const and adding it into the constructor for router, but I received the error: The expected type comes from property 'loader' which is declared here on type 'RouteObject' so I just left it blank.

This code results in ReferenceError: document is not defined certainly because I don't have the syntax or structure correct for this router. Can someone provide some guidance on how I should be using createBrowserRouter in this context? I know I need to use the RouterProvider component in some way, but I don't have enough experience to see the path forward. What am I missing here?

答案1

得分: 1

你很可能从错误的包中导入了 useFetcher。请确保你是从 @remix-run/react 导入它:

  1. import { useFetcher } from "@remix-run/react";
英文:

You are most probably importing useFetcher from an incorrect package. Make sure that you are importing it from @remix-run/react:

  1. import { useFetcher } from &quot;@remix-run/react&quot;;

huangapple
  • 本文由 发表于 2023年1月6日 13:36:21
  • 转载请务必保留本文链接:https://go.coder-hub.com/75027298.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定