加载应用

本章将介绍如何使用 createRemoteAppComponent 在宿主应用中加载和集成远程 React 应用。

什么是 createRemoteAppComponent?

createRemoteAppComponent 是 React Bridge 的核心 API,用于在宿主应用中加载远程 React 应用。它具有以下特性:

  • 🚀 自动懒加载 - 远程应用会在需要时才开始加载
  • 🔧 生命周期管理 - 自动处理组件的挂载和卸载
  • 🛣️ 路由集成 - 无缝集成 React Router,支持 basename 注入
  • ⚡ 错误处理 - 内置加载失败和运行时错误处理机制
  • 🎨 样式隔离 - 支持组件级样式和类名配置

安装

npm
yarn
pnpm
npm install @module-federation/bridge-react@latest

基本使用

步骤 1: 配置远程模块

在宿主应用的配置文件中,添加远程应用的配置:

构建工具支持

以下示例使用 Rsbuild 配置,请根据您使用的构建工具进行相应调整:

  • Rsbuild: @module-federation/rsbuild-plugin
  • Rspack: @module-federation/enhanced/rspack
  • Webpack: @module-federation/enhanced/webpack
  • Vite: @module-federation/vite
// rsbuild.config.ts import { pluginModuleFederation } from '@module-federation/rsbuild-plugin'; import { defineConfig } from '@rsbuild/core';  export default defineConfig({  plugins: [  pluginModuleFederation({  name: 'host-app',  remotes: {  'remote1': 'remote1@http://localhost:3001/remoteEntry.js',  },  }),  ], });

步骤 2: 创建远程组件

2.1 定义加载和错误组件

首先,创建加载状态和错误处理组件:

// ./src/components/RemoteComponents.tsx import React from 'react';  // 加载状态组件 export const LoadingComponent = () => (  <div style={{ padding: '20px', textAlign: 'center' }}>  <div>远程应用加载中...</div>  </div> );  // 错误回退组件 export const ErrorFallback = ({ error }: { error: Error }) => (  <div style={{ padding: '20px', border: '1px solid #ff6b6b', borderRadius: '8px' }}>  <h3>远程应用加载失败</h3>  <p>错误详情: {error.message}</p>  <button onClick={() => window.location.reload()}>  重新加载页面  </button>  </div> );

2.2 创建远程应用组件

使用 createRemoteAppComponent 创建远程组件:

// ./src/remotes/Remote1App.tsx import { createRemoteAppComponent } from '@module-federation/bridge-react'; import { loadRemote } from '@module-federation/runtime'; import { LoadingComponent, ErrorFallback } from '../components/RemoteComponents';  export const Remote1App = createRemoteAppComponent({  loader: () => loadRemote('remote1/export-app'),  loading: LoadingComponent,  fallback: ErrorFallback, });

2.3 主应用路由配置

在主应用中配置路由:

// ./src/App.tsx import React from 'react'; import { BrowserRouter, Routes, Route } from 'react-router-dom'; import { Remote1App } from './remotes/Remote1App';  // 宿主应用首页组件 const HomePage = () => (  <div style={{ padding: '20px' }}>  <h1>宿主应用首页</h1>  <p>这是宿主应用的首页内容</p>  </div> );  const App = () => {  return (  <BrowserRouter>  <Routes>  <Route path="/" element={<HomePage />} />  <Route  path="/remote1/*"  // 使用 Remote1App 组件, 将会被懒加载  Component={() => (  <Remote1App  // 可设置 className 和 style 样式,将自动注入到组件上  className={styles.remote1}  style={{ color: 'red' }}  // name 和 age 为远程组件 props, 将自动透传到远程组件  name={'Ming'}  age={12}  // 可设置 ref, 将自动转发到远程组件,可获取 ref 对象操作 dom  ref={ref}  />  )}  />  </Routes>  </BrowserRouter>  ); };  export default App;

远程组件 Props

路由相关属性

  • basename: 设置远程应用的基础路径
  • memoryRoute: 内存路由配置,用于将子应用路由作为 memoryRouter 控制

样式属性

  • style: React.CSSProperties - 设置组件样式
  • className: string - 设置组件类名

引用支持

  • ref: React.Ref<HTMLDivElement> - 转发引用到内部容器元素,可用于 DOM 操作

数据传递

  • props: 传递给远程组件的属性对象
  • 或者直接传递属性,如 userId={'123'}

createRemoteAppComponent API 参考

函数签名

function createRemoteAppComponent<T = Record<string, unknown>, E extends keyof T = keyof T>(  config: RemoteComponentParams<T, E> ): React.ForwardRefExoticComponent<  Omit<RemoteComponentProps<T>, "ref"> & React.RefAttributes<HTMLDivElement> >

RemoteComponentParams<T, E>

配置参数接口:

interface RemoteComponentParams<T = Record<string, unknown>, E extends keyof T = keyof T> {  // 远程模块加载器  loader: () => Promise<T>;    // 加载状态显示内容  loading: React.ReactNode;    // 错误回退组件  fallback: React.ComponentType<{ error: Error }>;    // 导出名称(可选)  export?: E;    // 传递给远程组件的属性(可选)  props?: T; }

RemoteComponentProps<T>

返回组件的属性接口:

interface RemoteComponentProps<T = Record<string, unknown>> {  // 传递给远程组件的属性  props?: T;    // 错误回退组件  fallback?: React.ComponentType<{ error: Error }>;    // 加载状态显示内容  loading?: React.ReactNode;    // 路由基础路径  basename?: string;    // 内存路由配置  memoryRoute?: {  entryPath: string;  initialState?: Record<string, unknown>;  };    // 样式属性  style?: React.CSSProperties;  className?: string;    // 其他自定义属性  [key: string]: unknown; }

参数详解

loader

  • 类型: () => Promise<T>
  • 必需: 是
  • 作用: 用于加载远程模块的函数,返回一个 Promise,该 Promise 解析为远程模块对象
  • 示例:
    loader: () => loadRemote('remote1/export-app') loader: () => import('remote1/export-app')

loading

  • 类型: React.ReactNode
  • 必需: 是
  • 作用: 在远程应用加载期间显示的加载内容,可以是组件、元素或字符串
  • 示例:
    loading: <div>Loading...</div> loading: 'Loading remote app...' loading: <Spinner />

fallback

  • 类型: React.ComponentType<{ error: Error }>
  • 必需: 是
  • 作用: 当远程应用加载失败时显示的错误回退组件,会接收错误对象作为 error 属性
  • 示例:
    fallback: ({ error }) => <div>Error: {error.message}</div> fallback: ErrorBoundaryComponent

export

  • 类型: E extends keyof T (泛型约束,通常是 string)

  • 必需: 否

  • 默认值: 'default'

  • 作用: 指定要使用的远程模块导出名称

  • 示例:

    假设远程模块有以下导出:

    // 远程模块的导出 export default App; // 默认导出 export const provider = App; // 命名导出 provider export const dashboard = Dashboard; // 命名导出 dashboard

    在宿主应用中可以这样使用:

    // 使用默认导出(可以省略 export 参数) createRemoteAppComponent({  loader: () => loadRemote('remote1/export-app'),  // export: 'default' // 可以省略,默认就是 'default' })  // 使用命名导出 provider createRemoteAppComponent({  loader: () => loadRemote('remote1/export-app'),  export: 'provider' })  // 使用命名导出 dashboard createRemoteAppComponent({  loader: () => loadRemote('remote1/export-app'),  export: 'dashboard' })