From fd6eafda722a9390cb73e8983468ff8b22c01fca Mon Sep 17 00:00:00 2001 From: Chris Atkin Date: Sat, 8 Aug 2020 11:56:04 +0100 Subject: [PATCH] Provide option for spreading props rather than static assignment This provides an option (`spreadDefaultProps`) to spread the extra props from the top level `svg` element onto the props object, rather than statically assigning them as `defaultProps`. This gives users an optional trade off for `_spread` performance (as raised in the PR that originally setup `defaultProps`: https://github.com/airbnb/babel-plugin-inline-react-svg/pull/8) against being able to tree shake the generated components (which is prevented if they have static assignments). The `README` has been updated, and a sanity test scenario added. This also corrects some bugs which arise when `opts.SVG_DEFAULT_PROPS_CODE` is not assigned. The first is that Babel will throw an error when a substitution key is provided but not used, even if it's value is `undefined`. This happens when `opts.SVG_DEFAULT_PROPS_CODE` is not assigned, and therefore the `SVG_NAME.defaultProps = SVG_DEFAULT_PROPS_CODE` template string is never included - however, the `SVG_DEFAULT_PROPS_CODE` is still passed. The second is using `replaceWith` versus `replaceWithMultiple` when there are multiple nodes to be replaced. This is currently predicated on `opts.SVG_DEFAULT_PROPS_CODE` - which is not accurate. It should be predicated on if there are multiple nodes (`svgReplacement.length > 1`). --- README.md | 2 ++ src/index.js | 26 ++++++++++++++++++++++---- test/sanity.js | 25 ++++++++++++++++++++++++- 3 files changed, 48 insertions(+), 5 deletions(-) diff --git a/README.md b/README.md index a8ba375..cbfe495 100644 --- a/README.md +++ b/README.md @@ -42,8 +42,10 @@ npm install --save-dev babel-plugin-inline-react-svg #### Options + - `ignorePattern` - A pattern that imports will be tested against to selectively ignore imports. - `caseSensitive` - A boolean value that if true will require file paths to match with case-sensitivity. Useful to ensure consistent behavior if working on both a case-sensitive operating system like Linux and a case-insensitive one like OS X or Windows. +- *`spreadDefaultProps`* - A boolean value that if `true` will spread additional props, rather than setting them as `defaultProps` static assignment. This is important for tree shaking, as static assignments prevent efficient dead code removal. - `svgo` - svgo options (`false` to disable). Example: ```json { diff --git a/src/index.js b/src/index.js index 2934b6c..ce83824 100644 --- a/src/index.js +++ b/src/index.js @@ -39,9 +39,26 @@ export default declare(({ `; if (SVG_NAME !== 'default') { - return template(namedTemplate)({ SVG_NAME, SVG_CODE, SVG_DEFAULT_PROPS_CODE }); + const substitutions = { SVG_CODE, SVG_NAME }; + + // If a key is present in the substitutions object, but is unused in the template, even if + // even if it's value is undefined, Babel will throw an error. + if (SVG_DEFAULT_PROPS_CODE) { + substitutions.SVG_DEFAULT_PROPS_CODE = SVG_DEFAULT_PROPS_CODE; + } + + return template(namedTemplate)(substitutions); } - return template(anonymousTemplate)({ SVG_CODE, SVG_DEFAULT_PROPS_CODE, EXPORT_FILENAME }); + + const substitutions = { SVG_CODE, EXPORT_FILENAME }; + + // If a key is present in the substitutions object, but is unused in the template, even if + // even if it's value is undefined, Babel will throw an error. + if (SVG_DEFAULT_PROPS_CODE) { + substitutions.SVG_DEFAULT_PROPS_CODE = SVG_DEFAULT_PROPS_CODE; + } + + return template(anonymousTemplate)(substitutions); }; function applyPlugin(importIdentifier, importPath, path, state, isExport, exportFilename) { @@ -95,8 +112,9 @@ export default declare(({ EXPORT_FILENAME: exportFilename, }; - // Move props off of element and into defaultProps - if (svgCode.openingElement.attributes.length > 1) { + // Move props off of element and into defaultProps, but only if + // the "spreadDefaultProps" argument is not true + if (state.opts.spreadDefaultProps !== true && svgCode.openingElement.attributes.length > 1) { const keepProps = []; const defaultProps = []; diff --git a/test/sanity.js b/test/sanity.js index c397fba..7245b97 100644 --- a/test/sanity.js +++ b/test/sanity.js @@ -14,8 +14,20 @@ function assertReactImport(result) { } } +function assertSpreadProps(result) { + const hasSpreadProps = result.code.match(/_extends\(\{.*"'data-name'": "Livello 1".*\}, props\)/gs); + if (!hasSpreadProps) { + throw new Error('Spread props were not found'); + } + + const hasDefaultProps = result.code.match(/\.defaultProps/g); + if (hasDefaultProps) { + throw new Error('Default props were found, when spread should have been used instead'); + } +} + function validateDefaultProps(result) { - if (!(/'data-name':/g).test(result.code)) { + if (!(/'data-name':?/g).test(result.code)) { throw new Error('data-* props need to be quoted'); } } @@ -226,3 +238,14 @@ transformFile('test/fixtures/test-commented.jsx', { console.log('test/fixtures/test-commented.jsx', result.code); }); */ +transformFile('test/fixtures/test-export-default-as.jsx', { + presets: ['airbnb'], + plugins: [ + [inlineReactSvgPlugin, { spreadDefaultProps: true }], + ], +}, (err, result) => { + if (err) throw err; + console.log('test/fixtures/test-export-default-as.jsx', result.code); + validateDefaultProps(result); + assertSpreadProps(result); +});