Skip to content

fatfisz/babel-plugin-jsx-svg-inject

Repository files navigation

babel-plugin-jsx-svg-inject

Add SVG file imports for React components

This plugin can transform this:

<Icon svgName="eye" />

into this:

var _svgContents = <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 200 200"><circle cx="100" cy="100" r="100"/></svg>;
<Icon svgName="eye" svgContents={_svgContents} />;

The main benefit here is that explicit imports for the images are not needed and instead a single component can be used.

The plugin includes the contents of the file as a JSX element, so there is no need for additional plugins or loaders. There is also an option to have the imports added instead (useImports).

How does it work?

The plugin searches for a specific prop on JSX Elements (by default it's 'svgName') and adds the contents of the file as a JSX element (or a file import) for each one. The exported value is put in an additional prop (by default it's 'svgContents').

Upgrading to v.5

The plugin now outputs JSX elements instead of text. This has the benefit of not needing to use dangerouslySetInnerHTML, which in turn solves a problem React has with swapping SVGs in IE 11. The plugin encourages a use of very few (in most cases one) image components, so the changes should be minimal.

For example, if you previously had a component like this:

export default function Image({ svgName, svgContents, ...props }) {
  return <svg {...props} dangerouslySetInnerHTML={{ __html: svgContents }} />;
}

Image.propTypes = {
  svgContents: PropTypes.string.isRequired,
  svgName: PropTypes.string.isRequired,
};

it could be rewritten like so:

export default function Image({ svgName, svgContents, ...props }) {
  return <svg {...props}>{svgContents}</svg>;  // svgContents is used the same as React children
}

Image.propTypes = {
  svgContents: PropTypes.node.isRequired,  // The type has changed
  svgName: PropTypes.string.isRequired,
};

Upgrading to v.4

The plugin now inlines the source of the image by default. Previously another plugin or loader was needed for that (an import was added instead), which could be quite bothersome.

The upgrade should be painless if no complicated logic was used for importing the files. In case the imports are preferred, please use the useImports option.

Upgrading to v.3

In version 2.x and below the plugin looked for a JSX element with a specific name. This was less flexible - there was no simple way of having SVG injected into different tags. This was changed in version 3.0.0 and the JSX property now has the role that the tag previously had.

Usage

Install from the npm and then add this to .babelrc:

{
  "plugins": [
    "jsx-svg-inject"
  ]
}

Options

root

Default: '.'

The path to the icons.

In case root is a relative path (e.g. ./some-path), it will be resolved using the current working directory.

nameProp

Default: 'svgName'

The value of the prop with this name will be used for getting the SVG path.

contentsProp

Default: 'svgContents'

The value of the prop with this name will be used for passing the SVG contents.

unwrap

Default: false

If set to true, the <svg> element will be stripped and its props will be passed to the resulting element.

For example, with this option enabled and a file like this:

<SVG svgName="some-path/foo" />

the output might be something similar to this:

var _svgContents = <rect x=\"10\" y=\"10\" width=\"100\" height=\"100\" />;
<SVG svgName="some-path/foo" svgContents={_svgContents} viewBox="0 0 120 120" height="120" width="120" xmlns="http://www.w3.org/2000/svg" />;

When the <svg> element has multiple children, they have keys provided automatically.

useImports

Default: false

If true, a relative import will be added instead of an inlined content. This was the default behavior before v.4.

License

Copyright (c) 2017 Rafał Ruciński. Licensed under the MIT license.

About

Add SVG file imports for React components

Topics

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published