close
  • 简体中文
  • 静态资源

    Rslib 支持在代码中引用图片、字体、音频、视频等类型的静态资源。

    静态资源格式

    以下是 Rslib 默认支持的静态资源格式:

    • 图片:png、jpg、jpeg、gif、svg、bmp、webp、ico、apng、avif、tif、tiff、jfif、pjpeg、pjp、cur。
    • 字体:woff、woff2、eot、ttf、otf、ttc。
    • 音频:mp3、wav、flac、aac、m4a、opus。
    • 视频:mp4、webm、ogg、mov。

    如果你需要引用其他格式的静态资源,请参考 扩展静态资源类型

    在 JavaScript 文件中引用

    在 JavaScript 文件中,可以直接通过 import 的方式引用相对路径下的静态资源:

    // 引用 src/assets 目录下的 logo.png 图片
    import logo from './assets/logo.png';
    
    console.log(logo); // "/static/logo.[hash].png"
    
    export default = () => <img src={logo} />;

    也可以使用路径别名来引用:

    import logo from '@/assets/logo.png';
    
    console.log(logo); // "/static/logo.[hash].png"
    
    export default = () => <img src={logo} />;

    formatcjsesm 时,Rslib 将产物视为会被其他打包工具再次消费的中间产物,默认会在代码转换时将源文件转化为一个 JavaScript 文件和一个根据 output.distPath 输出的静态资源文件,并保留引用静态资源的 importrequire 语句。

    下面是一个使用示例,假设源码如下:

    src/index.ts
    src/assets/logo.svg
    import logo from './assets/logo.svg';
    
    console.log(logo);

    会根据配置文件中的 产物结构 配置,输出如下产物:

    bundle
    bundleless
    dist/index.mjs
    dist/static/svg/logo.svg
    import logo_namespaceObject from './static/svg/logo.svg';
    
    console.log(logo_namespaceObject);

    在 CSS 文件中引用

    在 CSS 文件中,可以引用相对路径下的静态资源:

    src/index.css
    .logo {
      background-image: url('./assets/logo.png');
    }

    也支持使用路径别名来引用:

    src/index.css
    .logo {
      background-image: url('@/assets/logo.png');
    }

    formatcjsesm 时,Rslib 将产物视为会被其他打包工具再次消费的中间产物,默认会将 output.assetPrefix 设置为 "auto" 来使 CSS 产物中保留相对引用路径。

    下面是一个使用示例,假设源码如下:

    src/index.css
    src/assets/logo.png
    .logo {
      background-image: url('./assets/logo.png');
    }

    会输出如下产物:

    dist/index.css
    dist/static/image/logo.png
    .logo {
      background-image: url('./static/image/logo.png');
    }

    在 CSS 中忽略某些文件引用

    如果需要在 CSS 文件中引用绝对路径下的静态资源:

    @font-face {
      font-family: DingTalk;
      src: url('/image/font/foo.ttf');
    }

    默认情况下,Rslib 内置的 css-loader 会解析 url() 中的绝对路径并寻找指定的模块。如果你希望跳过绝对路径的解析,可以配置 tools.cssLoader 来过滤指定的路径,被过滤的路径将被原样保留在代码中。

    export default {
      tools: {
        cssLoader: {
          url: {
            filter: (url) => {
              if (/\/image\/font/.test(url)) {
                return false;
              }
              return true;
            },
          },
        },
      },
    };

    静态资源内联

    formatcjsesm 时,Rslib 将产物视为会被其他打包工具再次消费的中间产物,默认会将 output.dataUriLimit 设置为 0 以不内联任何静态资源。

    构建产物目录

    当静态资源被引用后,会自动被输出到构建产物的目录下,你可以:

    • 通过 output.filename 来修改产物的文件名。例如,在产物的文件名中添加 hash 值,这通常在有同名文件时使用,以避免文件名冲突。
    rslib.config.ts
    export default {
      output: {
        filename: {
          svg: '[name].[contenthash:10].svg',
          font: '[name].[contenthash:10][ext]',
          image: '[name].[contenthash:10][ext]',
          media: '[name].[contenthash:10][ext]',
          assets: '[name].[contenthash:10][ext]',
        },
      },
    };
    • 通过 output.distPath 来修改产物的输出路径。例如,将静态资源产物输出到 dist/resource 目录下。
    rslib.config.ts
    export default {
      output: {
        distPath: {
          svg: 'resource/svg',
          font: 'resource/font',
          image: 'resource/image',
          media: 'resource/media',
          assets: 'resource/assets',
        },
      },
    };

    类型声明

    当你在 TypeScript 代码中引用静态资源时,TypeScript 可能会提示该模块缺少类型定义:

    TS2307: Cannot find module './logo.png' or its corresponding type declarations.

    此时你需要为静态资源添加类型声明文件,请在项目中创建 src/env.d.ts 文件,并添加相应的类型声明。

    • 方法一:如果项目里安装了 @rslib/core 包,你可以直接引用 @rslib/core 提供的 预设类型
    /// <reference types="@rslib/core/types" />
    • 方法二:手动添加需要的类型声明:
    src/env.d.ts
    // 以 png 图片为例
    declare module '*.png' {
      const url: string;
      export default url;
    }

    添加类型声明后,如果依然存在上述错误提示,请尝试重启当前 IDE,或者调整 env.d.ts 所在的目录,使 TypeScript 能够正确识别类型定义。

    扩展静态资源类型

    如果 Rslib 内置的静态资源类型不能满足你的需求,可以通过以下方式扩展额外的静态资源类型。

    使用 source.assetsInclude

    通过 source.assetsInclude 配置项,你可以指定需要被视为静态资源的额外文件类型。

    rslib.config.ts
    export default {
      source: {
        assetsInclude: /\.pdf$/,
      },
    };

    添加以上配置后,你就可以在代码里引用 *.pdf 文件了,比如:

    import myFile from './static/myFile.pdf';
    
    console.log(myFile); // "/static/assets/myFile.[hash].pdf"

    使用 tools.rspack

    可以通过 tools.rspack 来修改内置的 Rspack 配置,并添加自定义的静态资源处理规则。

    比如,把 *.pdf 文件当做静态资源输出到产物目录,可以添加以下配置:

    rslib.config.ts
    export default {
      tools: {
        rspack(config, { addRules }) {
          addRules([
            {
              test: /\.pdf$/,
              // 将资源转换为单独的文件,并且保留 import 语句
              type: 'asset/resource',
              generator: {
                importMode: 'preserve',
              },
            },
          ]);
        },
      },
    };

    关于资源模块的更多介绍,请参考 Rspack - 资源模块