主分支
分支
主分支 (2.15.2)开发
版本
2.15.21.19.3v0.21.0
React Router v7 已发布。 查看文档
Vite
本页内容

Vite

Vite 是一个强大、高性能且可扩展的 JavaScript 项目开发环境。为了改进和扩展 Remix 的打包功能,我们现在支持 Vite 作为替代编译器。未来,Vite 将成为 Remix 的默认编译器。

经典 Remix 编译器 vs. Remix Vite

现有的 Remix 编译器,通过 remix buildremix dev CLI 命令访问,并通过 remix.config.js 配置,现在被称为 “经典 Remix 编译器”。

Remix Vite 插件以及 remix vite:buildremix vite:dev CLI 命令统称为 “Remix Vite”。

未来,除非另有说明,否则文档将假定使用 Remix Vite。

入门

我们提供了一些不同的基于 Vite 的模板供您入门。

# Minimal server:
npx create-remix@latest

# Express:
npx create-remix@latest --template remix-run/remix/templates/express

# Cloudflare:
npx create-remix@latest --template remix-run/remix/templates/cloudflare

# Cloudflare Workers:
npx create-remix@latest --template remix-run/remix/templates/cloudflare-workers

这些模板包含一个 vite.config.ts 文件,其中配置了 Remix Vite 插件。

配置

Remix Vite 插件通过项目根目录下的 vite.config.ts 文件进行配置。有关更多信息,请参阅我们的Vite 配置文档

Cloudflare

要开始使用 Cloudflare,您可以使用 cloudflare 模板

npx create-remix@latest --template remix-run/remix/templates/cloudflare

有两种方式可以在本地运行您的 Cloudflare 应用程序

# Vite
remix vite:dev

# Wrangler
remix vite:build # build app before running wrangler
wrangler pages dev ./build/client

虽然 Vite 提供了更好的开发体验,但 Wrangler 通过在 Cloudflare 的 workerd 运行时 而不是 Node 中运行您的服务器代码,提供了对 Cloudflare 环境更接近的模拟。

Cloudflare 代理

为了在 Vite 中模拟 Cloudflare 环境,Wrangler 提供了 本地 workerd 绑定的 Node 代理。Remix 的 Cloudflare 代理插件为您设置这些代理

import {
  vitePlugin as remix,
  cloudflareDevProxyVitePlugin as remixCloudflareDevProxy,
} from "@remix-run/dev";
import { defineConfig } from "vite";

export default defineConfig({
  plugins: [remixCloudflareDevProxy(), remix()],
});

这些代理在您的 loaderaction 函数中的 context.cloudflare 中可用

export const loader = ({ context }: LoaderFunctionArgs) => {
  const { env, cf, ctx } = context.cloudflare;
  // ... more loader code here...
};

请查看 Cloudflare 的 getPlatformProxy 文档,了解有关每个代理的更多信息。

绑定

要为 Cloudflare 资源配置绑定

每当您更改 wrangler.toml 文件时,您需要运行 wrangler types 以重新生成您的绑定。

然后,您可以通过 context.cloudflare.env 访问您的绑定。例如,对于一个绑定为 MY_KVKV 命名空间

export async function loader({
  context,
}: LoaderFunctionArgs) {
  const { MY_KV } = context.cloudflare.env;
  const value = await MY_KV.get("my-key");
  return json({ value });
}

增强加载上下文

如果您想向加载上下文中添加其他属性,您应该从一个共享模块中导出一个 getLoadContext 函数,以便 Vite、Wrangler 和 Cloudflare Pages 中的加载上下文都以相同的方式增强

import { type AppLoadContext } from "@remix-run/cloudflare";
import { type PlatformProxy } from "wrangler";

// When using `wrangler.toml` to configure bindings,
// `wrangler types` will generate types for those bindings
// into the global `Env` interface.
// Need this empty interface so that typechecking passes
// even if no `wrangler.toml` exists.
interface Env {}

type Cloudflare = Omit<PlatformProxy<Env>, "dispose">;

declare module "@remix-run/cloudflare" {
  interface AppLoadContext {
    cloudflare: Cloudflare;
    extra: string; // augmented
  }
}

type GetLoadContext = (args: {
  request: Request;
  context: { cloudflare: Cloudflare }; // load context _before_ augmentation
}) => AppLoadContext;

// Shared implementation compatible with Vite, Wrangler, and Cloudflare Pages
export const getLoadContext: GetLoadContext = ({
  context,
}) => {
  return {
    ...context,
    extra: "stuff",
  };
};

您必须将 getLoadContext 传递给 Cloudflare 代理插件functions/[[path]].ts 中的请求处理程序,否则您将根据运行应用程序的方式获得不一致的加载上下文增强。

首先,在您的 Vite 配置中将 getLoadContext 传递给 Cloudflare 代理插件,以便在运行 Vite 时增强加载上下文。

import {
  vitePlugin as remix,
  cloudflareDevProxyVitePlugin as remixCloudflareDevProxy,
} from "@remix-run/dev";
import { defineConfig } from "vite";
import tsconfigPaths from "vite-tsconfig-paths";

import { getLoadContext } from "./load-context";

export default defineConfig({
  plugins: [
    remixCloudflareDevProxy({ getLoadContext }),
    remix(),
  ],
});

接下来,在您的 functions/[[path]].ts 文件中将 getLoadContext 传递给请求处理程序,以便在运行 Wrangler 或部署到 Cloudflare Pages 时增强加载上下文。

import { createPagesFunctionHandler } from "@remix-run/cloudflare-pages";

// @ts-ignore - the server build file is generated by `remix vite:build`
import * as build from "../build/server";
import { getLoadContext } from "../load-context";

export const onRequest = createPagesFunctionHandler({
  build,
  getLoadContext,
});

拆分客户端和服务器代码

Vite 处理客户端和服务器代码的混合使用方式与 Classic Remix 编译器不同。有关更多信息,请参阅我们关于拆分客户端和服务器代码的文档。

新的构建输出路径

Vite 管理 public 目录的方式与现有的 Remix 编译器有明显的不同。Vite 将文件从 public 目录复制到客户端构建目录,而 Remix 编译器则保持 public 目录不变,并使用子目录 (public/build) 作为客户端构建目录。

为了使默认的 Remix 项目结构与 Vite 的工作方式保持一致,构建输出路径已更改。现在有一个单一的 buildDirectory 选项,默认为 "build",取代了单独的 assetsBuildDirectoryserverBuildDirectory 选项。这意味着,默认情况下,服务器现在被编译到 build/server 中,客户端现在被编译到 build/client 中。

这也意味着以下配置默认值已更改

  • publicPath 已被 Vite 的 "base" 选项 所取代,该选项的默认值是 "/" 而不是 "/build/"
  • serverBuildPath 已被 serverBuildFile 取代,其默认值为 "index.js"。此文件将写入您配置的 buildDirectory 中的服务器目录中。

Remix 迁移到 Vite 的原因之一是让您在采用 Remix 时学习更少的内容。这意味着,对于您想要使用的任何其他捆绑功能,您应该参考 Vite 文档Vite 插件社区,而不是 Remix 文档。

Vite 有许多 功能插件,这些功能和插件没有内置在现有的 Remix 编译器中。使用任何此类功能都会导致现有的 Remix 编译器无法编译您的应用程序,因此只有当您打算从此以后完全使用 Vite 时才使用它们。

迁移

设置 Vite

👉 将 Vite 作为开发依赖项安装

npm install -D vite

Remix 现在只是一个 Vite 插件,因此您需要将其连接到 Vite。

👉 在您的 Remix 应用程序的根目录中,将 remix.config.js 替换为 vite.config.ts

import { vitePlugin as remix } from "@remix-run/dev";
import { defineConfig } from "vite";

export default defineConfig({
  plugins: [remix()],
});

受支持的 Remix 配置选项的子集应直接传递给插件

export default defineConfig({
  plugins: [
    remix({
      ignoredRouteFiles: ["**/*.css"],
    }),
  ],
});

HMR & HDR

Vite 为 HMR 等开发功能提供了强大的客户端运行时,使 <LiveReload /> 组件过时。在开发中使用 Remix Vite 插件时,<Scripts /> 组件将自动包含 Vite 的客户端运行时和其他仅用于开发的脚本。

👉 删除 <LiveReload/>,保留 <Scripts />

  import {
-   LiveReload,
    Outlet,
    Scripts,
  }

  export default function App() {
    return (
      <html>
        <head>
        </head>
        <body>
          <Outlet />
-         <LiveReload />
          <Scripts />
        </body>
      </html>
    )
  }

TypeScript 集成

Vite 处理各种不同文件类型的导入,有时方式与现有的 Remix 编译器不同,因此让我们从 vite/client 引用 Vite 的类型,而不是从 @remix-run/dev 引用过时的类型。

由于 vite/client 提供的模块类型与 @remix-run/dev 隐式包含的模块类型不兼容,因此您还需要在 TypeScript 配置中启用 skipLibCheck 标志。一旦 Vite 插件成为默认编译器,Remix 将来就不需要此标志。

👉 更新 tsconfig.json

更新 tsconfig.json 中的 types 字段,并确保 skipLibCheckmodulemoduleResolution 都已正确设置。

{
  "compilerOptions": {
    "types": ["@remix-run/node", "vite/client"],
    "skipLibCheck": true,
    "module": "ESNext",
    "moduleResolution": "Bundler"
  }
}

👉 更新/删除 remix.env.d.ts

删除 remix.env.d.ts 中的以下类型声明

- /// <reference types="@remix-run/dev" />
- /// <reference types="@remix-run/node" />

如果 remix.env.d.ts 现在为空,则删除它

rm remix.env.d.ts

从 Remix 应用程序服务器迁移

如果您在开发中使用 remix-serve (或者在没有 -c 标志的情况下使用 remix dev),则需要切换到新的最小开发服务器。它内置于 Remix Vite 插件中,当您运行 remix vite:dev 时,它将接管。

Remix Vite 插件不会安装任何 全局 Node polyfill,因此如果您依赖 remix-serve 来提供它们,则需要自己安装它们。最简单的方法是在您的 Vite 配置顶部调用 installGlobals

Vite 开发服务器的默认端口与 remix-serve 不同,因此如果您想保持相同的端口,则需要通过 Vite 的 server.port 选项配置此端口。

您还需要更新到新的构建输出路径,服务器的路径为 build/server,客户端资产的路径为 build/client

👉 更新您的 devbuildstart 脚本

{
  "scripts": {
    "dev": "remix vite:dev",
    "build": "remix vite:build",
    "start": "remix-serve ./build/server/index.js"
  }
}

👉 在您的 Vite 配置中安装全局 Node polyfill

import { vitePlugin as remix } from "@remix-run/dev";
+import { installGlobals } from "@remix-run/node";
import { defineConfig } from "vite";

+installGlobals();

export default defineConfig({
  plugins: [remix()],
});

👉 配置您的 Vite 开发服务器端口(可选)

export default defineConfig({
  server: {
    port: 3000,
  },
  plugins: [remix()],
});

迁移自定义服务器

如果您在开发中使用自定义服务器,则需要编辑您的自定义服务器以使用 Vite 的 connect 中间件。这将将资产请求和初始渲染请求委托给开发期间的 Vite,让您即使使用自定义服务器也能从 Vite 出色的 DX 中受益。

然后,您可以在开发期间加载名为 "virtual:remix/server-build" 的虚拟模块来创建基于 Vite 的请求处理程序。

您还需要更新您的服务器代码以引用新的构建输出路径,其中服务器构建的路径为 build/server,客户端资产的路径为 build/client

例如,如果您使用的是 Express,这是您可以执行的操作。

👉 更新您的 server.mjs 文件

import { createRequestHandler } from "@remix-run/express";
import { installGlobals } from "@remix-run/node";
import express from "express";

installGlobals();

const viteDevServer =
  process.env.NODE_ENV === "production"
    ? undefined
    : await import("vite").then((vite) =>
        vite.createServer({
          server: { middlewareMode: true },
        })
      );

const app = express();

// handle asset requests
if (viteDevServer) {
  app.use(viteDevServer.middlewares);
} else {
  app.use(
    "/assets",
    express.static("build/client/assets", {
      immutable: true,
      maxAge: "1y",
    })
  );
}
app.use(express.static("build/client", { maxAge: "1h" }));

// handle SSR requests
app.all(
  "*",
  createRequestHandler({
    build: viteDevServer
      ? () =>
          viteDevServer.ssrLoadModule(
            "virtual:remix/server-build"
          )
      : await import("./build/server/index.js"),
  })
);

const port = 3000;
app.listen(port, () =>
  console.log("https://127.0.0.1:" + port)
);

👉 更新您的 builddevstart 脚本

{
  "scripts": {
    "dev": "node ./server.mjs",
    "build": "remix vite:build",
    "start": "cross-env NODE_ENV=production node ./server.mjs"
  }
}

如果您愿意,您也可以使用 TypeScript 编写您的自定义服务器。然后,您可以使用诸如 tsxtsm 之类的工具来运行您的自定义服务器

tsx ./server.ts
node --loader tsm ./server.ts

请记住,如果您这样做,初始服务器启动可能会有一些明显的减速。

迁移 Cloudflare Functions

Remix Vite 插件仅正式支持 Cloudflare Pages,该平台专为全栈应用程序设计,不同于 Cloudflare Workers Sites。如果您当前使用的是 Cloudflare Workers Sites,请参阅 Cloudflare Pages 迁移指南

👉 在 remix 插件 之前 添加 cloudflareDevProxyVitePlugin,以正确覆盖 vite 开发服务器的中间件!

import {
  vitePlugin as remix,
  cloudflareDevProxyVitePlugin,
} from "@remix-run/dev";
import { defineConfig } from "vite";

export default defineConfig({
  plugins: [cloudflareDevProxyVitePlugin(), remix()],
});

您的 Cloudflare 应用程序可能正在设置 Remix Config server 字段以生成一个捕获所有 Cloudflare 函数。使用 Vite,不再需要这种间接性。相反,您可以直接为 Cloudflare 编写捕获所有路由,就像您为 Express 或任何其他自定义服务器编写路由一样。

👉 为 Remix 创建一个捕获所有路由

import { createPagesFunctionHandler } from "@remix-run/cloudflare-pages";

// @ts-ignore - the server build file is generated by `remix vite:build`
import * as build from "../build/server";

export const onRequest = createPagesFunctionHandler({
  build,
});

👉 通过 context.cloudflare.env 而不是 context.env 访问绑定和环境变量

虽然您主要在开发期间使用 Vite,但您也可以使用 Wrangler 来预览和部署您的应用程序。

要了解更多信息,请参阅本文档的Cloudflare部分。

👉 更新您的 package.json 脚本

{
  "scripts": {
    "dev": "remix vite:dev",
    "build": "remix vite:build",
    "preview": "wrangler pages dev ./build/client",
    "deploy": "wrangler pages deploy ./build/client"
  }
}

迁移对构建输出路径的引用

当使用现有 Remix 编译器的默认选项时,服务器被编译到 build 中,客户端被编译到 public/build 中。由于 Vite 通常使用其 public 目录的方式与现有 Remix 编译器相比存在差异,这些输出路径已更改。

👉 更新对构建输出路径的引用

  • 服务器现在默认编译到 build/server 中。
  • 客户端现在默认编译到 build/client 中。

例如,要从 Blues Stack 更新 Dockerfile

-COPY --from=build /myapp/build /myapp/build
-COPY --from=build /myapp/public /myapp/public
+COPY --from=build /myapp/build/server /myapp/build/server
+COPY --from=build /myapp/build/client /myapp/build/client

配置路径别名

Remix 编译器利用 tsconfig.json 中的 paths 选项来解析路径别名。这在 Remix 社区中很常用,用于将 ~ 定义为 app 目录的别名。

Vite 默认不提供任何路径别名。如果您依赖此功能,则可以安装 vite-tsconfig-paths 插件,以自动解析 Vite 中 tsconfig.json 的路径别名,从而匹配 Remix 编译器的行为

👉 安装 vite-tsconfig-paths

npm install -D vite-tsconfig-paths

👉 vite-tsconfig-paths 添加到您的 Vite 配置中

import { vitePlugin as remix } from "@remix-run/dev";
import { defineConfig } from "vite";
import tsconfigPaths from "vite-tsconfig-paths";

export default defineConfig({
  plugins: [remix(), tsconfigPaths()],
});

删除 @remix-run/css-bundle

Vite 内置支持 CSS 副作用导入、PostCSS 和 CSS Modules 以及其他 CSS 捆绑功能。Remix Vite 插件会自动将捆绑的 CSS 附加到相关的路由。

@remix-run/css-bundle@remix-run/css-bundle当使用 Vite 时,该包是多余的,因为它的 cssBundleHref 导出将始终为 undefined

👉 卸载 @remix-run/css-bundle

npm uninstall @remix-run/css-bundle

👉 删除对 cssBundleHref 的引用

- import { cssBundleHref } from "@remix-run/css-bundle";
  import type { LinksFunction } from "@remix-run/node"; // or cloudflare/deno

  export const links: LinksFunction = () => [
-   ...(cssBundleHref
-     ? [{ rel: "stylesheet", href: cssBundleHref }]
-     : []),
    // ...
  ];

如果路由的 links 函数仅用于连接 cssBundleHref,则可以完全删除它。

- import { cssBundleHref } from "@remix-run/css-bundle";
- import type { LinksFunction } from "@remix-run/node"; // or cloudflare/deno

- export const links: LinksFunction = () => [
-   ...(cssBundleHref
-     ? [{ rel: "stylesheet", href: cssBundleHref }]
-     : []),
- ];

这对于其他形式的 CSS 捆绑不是必需的,例如 CSS Modules、CSS 副作用导入、Vanilla Extract 等。

如果你在 links 函数中引用 CSS,你需要更新相应的 CSS 导入,使用 Vite 的显式 ?url 导入语法

👉 links 中使用的 CSS 导入中添加 ?url

.css?url 导入需要 Vite v5.1 或更高版本

-import styles from "~/styles/dashboard.css";
+import styles from "~/styles/dashboard.css?url";

export const links = () => {
  return [
    { rel: "stylesheet", href: styles }
  ];
}

通过 PostCSS 启用 Tailwind

如果你的项目正在使用 Tailwind CSS,你首先需要确保你有一个 PostCSS 配置文件,该文件会被 Vite 自动识别。这是因为当启用 Remix 的 tailwind 选项时,Remix 编译器不需要 PostCSS 配置文件。

👉 如果缺少 PostCSS 配置文件,请添加,包括 tailwindcss 插件

export default {
  plugins: {
    tailwindcss: {},
  },
};

如果你的项目已经有一个 PostCSS 配置文件,你需要添加 tailwindcss 插件(如果它尚未存在)。这是因为当启用 Remix 的 tailwind 配置选项时,Remix 编译器会自动包含此插件。

👉 如果缺少 tailwindcss 插件,请将其添加到你的 PostCSS 配置中

export default {
  plugins: {
    tailwindcss: {},
    autoprefixer: {},
  },
};

👉 迁移 Tailwind CSS 导入

如果你在 links 函数中引用你的 Tailwind CSS 文件,你需要迁移你的 Tailwind CSS 导入语句

添加 Vanilla Extract 插件

如果你正在使用 Vanilla Extract,你需要设置 Vite 插件。

👉 安装官方的 Vite 的 Vanilla Extract 插件

npm install -D @vanilla-extract/vite-plugin

👉 将 Vanilla Extract 插件添加到你的 Vite 配置中

import { vitePlugin as remix } from "@remix-run/dev";
import { vanillaExtractPlugin } from "@vanilla-extract/vite-plugin";
import { defineConfig } from "vite";

export default defineConfig({
  plugins: [remix(), vanillaExtractPlugin()],
});

添加 MDX 插件

如果你正在使用 MDX,由于 Vite 的插件 API 是 Rollup 插件 API 的扩展,你应该使用官方的 MDX Rollup 插件

👉 安装 MDX Rollup 插件

npm install -D @mdx-js/rollup

Remix 插件期望处理 JavaScript 或 TypeScript 文件,因此任何来自其他语言(如 MDX)的转译必须首先完成。在这种情况下,这意味着将 MDX 插件放在 Remix 插件之前

👉 将 MDX Rollup 插件添加到你的 Vite 配置中

import mdx from "@mdx-js/rollup";
import { vitePlugin as remix } from "@remix-run/dev";
import { defineConfig } from "vite";

export default defineConfig({
  plugins: [mdx(), remix()],
});
添加 MDX frontmatter 支持

Remix 编译器允许你在 MDX 中定义 frontmatter。如果你正在使用此功能,你可以在 Vite 中使用 remark-mdx-frontmatter 来实现此目的。

👉 安装所需的 Remark frontmatter 插件

npm install -D remark-frontmatter remark-mdx-frontmatter

👉 将 Remark frontmatter 插件传递给 MDX Rollup 插件

import mdx from "@mdx-js/rollup";
import { vitePlugin as remix } from "@remix-run/dev";
import remarkFrontmatter from "remark-frontmatter";
import remarkMdxFrontmatter from "remark-mdx-frontmatter";
import { defineConfig } from "vite";

export default defineConfig({
  plugins: [
    mdx({
      remarkPlugins: [
        remarkFrontmatter,
        remarkMdxFrontmatter,
      ],
    }),
    remix(),
  ],
});

在 Remix 编译器中,frontmatter 导出名为 attributes。这与 frontmatter 插件的默认导出名称 frontmatter 不同。虽然可以配置 frontmatter 导出名称,但我们建议更新你的应用程序代码以使用默认导出名称。

👉 在 MDX 文件中将 MDX attributes 导出重命名为 frontmatter

  ---
  title: Hello, World!
  ---

- # {attributes.title}
+ # {frontmatter.title}

👉 对于使用者,将 MDX attributes 导出重命名为 frontmatter

  import Component, {
-   attributes,
+   frontmatter,
  } from "./posts/first-post.mdx";
为 MDX 文件定义类型

👉 *.mdx 文件的类型添加到 env.d.ts

/// <reference types="@remix-run/node" />
/// <reference types="vite/client" />

declare module "*.mdx" {
  let MDXComponent: (props: any) => JSX.Element;
  export const frontmatter: any;
  export default MDXComponent;
}
将 MDX frontmatter 映射到路由导出

Remix 编译器允许你在 frontmatter 中定义 headersmetahandle 路由导出。remark-mdx-frontmatter 插件显然不支持这个 Remix 特有的功能。如果你正在使用此功能,你应该手动将 frontmatter 映射到路由导出。

👉 将 MDX 路由的 frontmatter 映射到路由导出

---
meta:
  - title: My First Post
  - name: description
    content: Isn't this awesome?
headers:
  Cache-Control: no-cache
---

export const meta = frontmatter.meta;
export const headers = frontmatter.headers;

# Hello World

请注意,由于你显式地映射 MDX 路由导出,你现在可以自由地使用你喜欢的任何 frontmatter 结构。

---
title: My First Post
description: Isn't this awesome?
---

export const meta = () => {
  return [
    { title: frontmatter.title },
    {
      name: "description",
      content: frontmatter.description,
    },
  ];
};

# Hello World
更新 MDX 文件名用法

Remix 编译器还提供了所有 MDX 文件的 filename 导出。这主要是为了启用链接到 MDX 路由集合。如果你正在使用此功能,你可以通过 glob 导入 在 Vite 中实现此目的,它为你提供了一个方便的数据结构,将文件名映射到模块。这使得维护 MDX 文件列表更容易,因为你不再需要手动导入每个文件。

例如,要导入 posts 目录中的所有 MDX 文件

const posts = import.meta.glob("./posts/*.mdx");

这相当于手动编写

const posts = {
  "./posts/a.mdx": () => import("./posts/a.mdx"),
  "./posts/b.mdx": () => import("./posts/b.mdx"),
  "./posts/c.mdx": () => import("./posts/c.mdx"),
  // etc.
};

如果你愿意,你也可以急切地导入所有 MDX 文件

const posts = import.meta.glob("./posts/*.mdx", {
  eager: true,
});

调试

你可以使用 NODE_OPTIONS 环境变量 来启动调试会话

NODE_OPTIONS="--inspect-brk" npm run dev

然后你可以从浏览器附加调试器。例如,在 Chrome 中,你可以打开 chrome://inspect 或单击开发工具中的 NodeJS 图标以附加调试器。

vite-plugin-inspect

vite-plugin-inspect 会向你展示每个 Vite 插件如何转换你的代码以及每个插件花费的时间。

性能

Remix 包含一个用于性能分析的 --profile 标志。

remix vite:build --profile

当使用 --profile 运行时,将生成一个 .cpuprofile 文件,可以共享或上传到 speedscope.app 进行分析。

你也可以在开发模式下通过按 p + enter 来进行分析,当开发服务器运行时,可以启动新的分析会话或停止当前会话。如果需要分析开发服务器的启动,你也可以使用 --profile 标志在启动时初始化分析会话

remix vite:dev --profile

请记住,你始终可以查看 Vite 性能文档以获取更多提示!

捆绑包分析

要可视化和分析你的捆绑包,你可以使用 rollup-plugin-visualizer 插件

import { vitePlugin as remix } from "@remix-run/dev";
import { visualizer } from "rollup-plugin-visualizer";

export default defineConfig({
  plugins: [
    remix(),
    // `emitFile` is necessary since Remix builds more than one bundle!
    visualizer({ emitFile: true }),
  ],
});

然后,当你运行 remix vite:build 时,它将在你的每个捆绑包中生成一个 stats.html 文件

build
├── client
│   ├── assets/
│   ├── favicon.ico
│   └── stats.html 👈
└── server
    ├── index.js
    └── stats.html 👈

在浏览器中打开 stats.html 以分析你的捆绑包。

故障排除

请查看调试性能部分以获取常规故障排除提示。另外,请通过查看 github 上 remix vite 插件的已知问题,看看是否有人遇到类似的问题。

HMR

如果你期望热更新但却得到完整的页面重新加载,请查看我们的 关于热模块替换的讨论,以了解有关 React Fast Refresh 的限制以及常见问题的解决方法。

ESM / CJS

Vite 支持 ESM 和 CJS 依赖,但有时你仍然可能会遇到 ESM / CJS 互操作问题。通常,这是因为依赖项未正确配置以支持 ESM。我们不会责怪他们,正确支持 ESM 和 CJS 真的很难

有关修复示例错误的演练,请查看 🎥 如何修复 Remix 中的 CJS/ESM 错误

要诊断你的某个依赖项是否配置错误,请查看 publintAre The Types Wrong。此外,你可以使用 vite-plugin-cjs-interop 插件 来平滑外部 CJS 依赖项的 default 导出的问题。

最后,你还可以显式配置哪些依赖项与 Vite 的 ssr.noExternal 选项一起打包到你的服务器中,以模拟 Remix 编译器的 serverDependenciesToBundle 和 Remix Vite 插件。

开发期间浏览器中的服务器代码错误

如果你在开发期间在浏览器控制台中看到指向服务器代码的错误,你可能需要显式隔离仅服务器代码。例如,如果你看到类似

Uncaught ReferenceError: process is not defined

那么你需要找出哪个模块正在拉入依赖项,这些依赖项期望使用仅服务器全局变量(如 process),并使用单独的 .server 模块或使用 vite-env-only隔离代码。由于 Vite 使用 Rollup 在生产环境中 treeshake 你的代码,因此这些错误仅在开发中发生。

与其他基于 Vite 的工具(例如 Vitest、Storybook)一起使用插件

Remix Vite 插件仅用于应用程序的开发服务器和生产构建。虽然还有其他基于 Vite 的工具(如 Vitest 和 Storybook)使用了 Vite 配置文件,但 Remix Vite 插件并未设计用于这些工具。我们目前建议在使用其他基于 Vite 的工具时排除该插件。

对于 Vitest

import { vitePlugin as remix } from "@remix-run/dev";
import { defineConfig, loadEnv } from "vite";

export default defineConfig({
  plugins: [!process.env.VITEST && remix()],
  test: {
    environment: "happy-dom",
    // Additionally, this is to load ".env.test" during vitest
    env: loadEnv("test", process.cwd(), ""),
  },
});

对于 Storybook

import { vitePlugin as remix } from "@remix-run/dev";
import { defineConfig } from "vite";

const isStorybook = process.argv[1]?.includes("storybook");

export default defineConfig({
  plugins: [!isStorybook && remix()],
});

或者,你可以为每个工具使用单独的 Vite 配置文件。例如,要使用专门针对 Remix 的 Vite 配置

remix vite:dev --config vite.config.remix.ts

在不提供 Remix Vite 插件时,你的设置可能还需要提供 Vite Plugin React。例如,当使用 Vitest 时

import { vitePlugin as remix } from "@remix-run/dev";
import react from "@vitejs/plugin-react";
import { defineConfig, loadEnv } from "vite";

export default defineConfig({
  plugins: [!process.env.VITEST ? remix() : react()],
  test: {
    environment: "happy-dom",
    // Additionally, this is to load ".env.test" during vitest
    env: loadEnv("test", process.cwd(), ""),
  },
});

文档重新挂载时,开发中的样式消失

当 React 用于渲染整个文档(如 Remix 所做的那样)时,你可能会在将元素动态注入到 head 元素时遇到问题。如果文档被重新挂载,则现有的 head 元素将被删除并替换为一个全新的元素,从而删除 Vite 在开发期间注入的任何 style 元素。

这是一个已知的 React 问题,已在他们的canary 发布渠道中修复。如果你了解所涉及的风险,你可以将你的应用程序固定到特定的 React 版本,然后使用package overrides 来确保这是整个项目中使用的唯一 React 版本。例如

{
  "dependencies": {
    "react": "18.3.0-canary-...",
    "react-dom": "18.3.0-canary-..."
  },
  "overrides": {
    "react": "18.3.0-canary-...",
    "react-dom": "18.3.0-canary-..."
  }
}

作为参考,这就是 Next.js 代表你内部处理 React 版本控制的方式,因此这种方法比你想象的更广泛地使用,即使它不是 Remix 作为默认提供的方式。

值得强调的是,Vite 注入的样式出现此问题仅在开发中发生。生产构建不会有此问题,因为会生成静态 CSS 文件。

在 Remix 中,当你的 根路由的默认组件导出与其 ErrorBoundary 和/或 HydrateFallback 导出之间交替渲染时,可能会出现此问题,因为这会导致挂载新的文档级组件。

它也可能由于 hydration 错误而发生,因为它会导致 React 从头开始重新渲染整个页面。hydration 错误可能是由你的应用程序代码引起的,但也可能是由操作文档的浏览器扩展引起的。

这对 Vite 来说很重要,因为在开发过程中,Vite 会将 CSS 导入转换为 JS 文件,这些 JS 文件会将样式作为副作用注入到文档中。Vite 这样做是为了支持静态 CSS 文件的延迟加载和 HMR(热模块替换)。

例如,假设你的应用程序有以下 CSS 文件

* { margin: 0 }

在开发过程中,当作为副作用导入时,此 CSS 文件将被转换为以下 JavaScript 代码

import {createHotContext as __vite__createHotContext} from "/@vite/client";
import.meta.hot = __vite__createHotContext("/app/styles.css");
import {updateStyle as __vite__updateStyle, removeStyle as __vite__removeStyle} from "/@vite/client";
const __vite__id = "/path/to/app/styles.css";
const __vite__css = "*{margin:0}"
__vite__updateStyle(__vite__id, __vite__css);
import.meta.hot.accept();
import.meta.hot.prune(()=>__vite__removeStyle(__vite__id));

这种转换不会应用于生产代码,这就是为什么这个样式问题只会影响开发环境。

开发环境中的 Wrangler 错误

当使用 Cloudflare Pages 时,你可能会在 wrangler pages dev 中遇到以下错误

ERROR: Your worker called response.clone(), but did not read the body of both clones.
This is wasteful, as it forces the system to buffer the entire response body
in memory, rather than streaming it through. This may cause your worker to be
unexpectedly terminated for going over the memory limit. If you only meant to
copy the response headers and metadata (e.g. in order to be able to modify
them), use `new Response(response.body, response)` instead.

这是一个Wrangler 已知的问题

致谢

Vite 是一个令人惊叹的项目,我们感谢 Vite 团队所做的工作。特别感谢 来自 Vite 团队的 Matias Capeletto、Arnaud Barré 和 Bjorn Lu 的指导。

Remix 社区很快开始探索对 Vite 的支持,我们感谢他们的贡献

最后,我们受到了其他框架如何实现 Vite 支持的启发

文档和示例在以下许可下发布 MIT