使用 webpack 5 将 SVG 导入为 React 组件

import SVG as React Components with webpack 5

我想在我的应用程序中使用 SVG 作为 React 组件。 我正在使用:react 17.0.2、Webpack 5.57.1​​、@svgr/webpack 6.2.1.

我遵循了在 webpack.config 文件中添加 svgr 的步骤,就像在 svgr 文档中一样 svgr-doc 但是控制台开发工具中出现错误:

react-dom.development.js:67 Warning: < /> is using incorrect casing. Use PascalCase for React components, or lowercase for HTML elements.

SVG 没有显示。 我无法将其作为 React 组件导入。

我的应用程序:

import React from 'react';
import ReactDOM from 'react-dom';

import CalendarIcon from './Calendar.svg'

const App = () => (
   <div>
       Test
       <CalendarIcon/>
   </div>
);

ReactDOM.render(<App />, document.getElementById('app'));

webpack 文件:

const path = require('path');
const HtmlWebPackPlugin = require('html-webpack-plugin');
const ModuleFederationPlugin = require('webpack/lib/container/ModuleFederationPlugin');

const deps = require('./package.json').dependencies;

module.exports = {
  output: {
    filename: '[name].[contenthash:20].esm.js',
    chunkFilename: '[name].[chunkhash:20].esm.js',
    hashFunction: 'xxhash64',
    pathinfo: false,
    crossOriginLoading: false,
    clean: true,
    publicPath: 'auto',
  },

  resolve: {
    extensions: ['.tsx', '.ts', '.jsx', '.js', '.json'],
  },

  devServer: {
    port: 3011,
    historyApiFallback: true,
  },

  module: {
    rules: [
      {
        test: /\.m?js/,
        type: 'javascript/auto',
        resolve: {
          fullySpecified: false,
        },
      },
      {
        test: /\.(css|s[ac]ss)$/i,
        use: ['style-loader', 'css-loader', 'postcss-loader'],
      },
      {
        test: /\.(ts|tsx|js|jsx)$/,
        exclude: /node_modules/,
        use: {
          loader: 'babel-loader',
        },
      },
      {
        test: /\.svg$/i,
        issuer: /\.[jt]sx?$/,
        use: ['@svgr/webpack'],
      },
      {
        test: /\.(?:ico|gif|png|jpg|jpeg)$/i,
        type: 'asset/resource',
      },
      {
        test: /\.(woff(2)?|eot|ttf|otf|svg|)$/,
        type: 'asset/inline',
      },
    ],
  },
  plugins: [
    new ModuleFederationPlugin({
      name: 'sidebar',
      filename: 'remoteEntry.js',
      remotes: {},
      exposes: {
        './SideBar': './src/components/Sidebar/index.tsx',
      },
      shared: {
        ...deps,
        react: {
          singleton: true,
          requiredVersion: deps.react,
        },
        'react-dom': {
          singleton: true,
          requiredVersion: deps['react-dom'],
        },
      },
    }),
    new HtmlWebPackPlugin({
      template: './src/index.html',
    }),
  ],
};

babelrc 文件:

{
  "presets": ["@babel/preset-typescript", "@babel/preset-react", "@babel/preset-env"],
  "plugins": [
    ["@babel/transform-runtime"],
    "babel-plugin-styled-components"
  ]
}

从您的 webpack rules 配置来看,您似乎与 type: "asset/inline" 的最后一条规则发生了名称冲突,后者也根据您的 test 处理 svg案例.

要解决此问题,您可以删除最后一条规则中的 svg,使其变为

test: /\.(woff(2)?|eot|ttf|otf|)$/,

或者您可以将 webpack 配置为使用带有附加约束的两个规则,如 in the SVGR docs. 所述,如果您在应用程序的其他位置需要文件内容,建议这样做。为此,添加

resourceQuery: /url/, // *.svg?url

最后一条规则和

resourceQuery: { not: [/url/] }, // exclude react component if *.svg?url

根据您的 @svgr/webpack 规则。