From 2228cb3ae967da571814f7617ebbb63fc05f0b3f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Greg=20Berg=C3=A9?= Date: Sun, 22 Mar 2020 12:19:25 +0100 Subject: [PATCH] feat: add typescript option --- .../src/__snapshots__/index.test.js.snap | 280 ++++++++++++++++++ .../src/index.js | 15 +- .../src/index.test.js | 270 +++++++++-------- .../src/util.js | 122 +++++++- website/src/pages/docs/custom-templates.mdx | 93 ++++++ website/src/pages/docs/typescript.mdx | 57 ---- 6 files changed, 636 insertions(+), 201 deletions(-) create mode 100644 packages/babel-plugin-transform-svg-component/src/__snapshots__/index.test.js.snap create mode 100644 website/src/pages/docs/custom-templates.mdx delete mode 100644 website/src/pages/docs/typescript.mdx diff --git a/packages/babel-plugin-transform-svg-component/src/__snapshots__/index.test.js.snap b/packages/babel-plugin-transform-svg-component/src/__snapshots__/index.test.js.snap new file mode 100644 index 00000000..7ed69aaa --- /dev/null +++ b/packages/babel-plugin-transform-svg-component/src/__snapshots__/index.test.js.snap @@ -0,0 +1,280 @@ +// Jest Snapshot v1, https://goo.gl/fbAQLP + +exports[` 1`] = ` +"import * as React from 'react'; + +const MyComponent = () =>
{}
; + +export default MyComponent;" +`; + +exports[` 2`] = ` +"import * as React from 'react'; + +const MyComponent = () =>
{}
; + +export default MyComponent;" +`; + +exports[`plugin javascript custom templates support basic template 1`] = ` +"import * as React from 'react'; + +const MyComponent = () => ; + +export default MyComponent;" +`; + +exports[`plugin javascript custom templates supports TypeScript template 1`] = ` +"import * as React from 'react'; + +const MyComponent = (props: React.SVGProps) => ; + +export default MyComponent;" +`; + +exports[`plugin javascript custom templates supports template that does not return an array 1`] = `";"`; + +exports[`plugin javascript transforms whole program 1`] = ` +"import * as React from \\"react\\"; + +function SvgComponent() { + return ; +} + +export default SvgComponent;" +`; + +exports[`plugin javascript with "expandProps" add props 1`] = ` +"import * as React from \\"react\\"; + +function SvgComponent(props) { + return ; +} + +export default SvgComponent;" +`; + +exports[`plugin javascript with "memo" option wrap component in "React.memo" 1`] = ` +"import * as React from \\"react\\"; + +function SvgComponent() { + return ; +} + +const MemoSvgComponent = React.memo(SvgComponent); +export default MemoSvgComponent;" +`; + +exports[`plugin javascript with "native" option adds import from "react-native-svg" 1`] = ` +"import * as React from \\"react\\"; +import Svg from \\"react-native-svg\\"; + +function SvgComponent() { + return ; +} + +export default SvgComponent;" +`; + +exports[`plugin javascript with "native.expo" option adds import from "react-native-svg" & from "expo" 1`] = ` +"import * as React from \\"react\\"; +import \\"expo\\"; + +function SvgComponent() { + return ; +} + +export default SvgComponent;" +`; + +exports[`plugin javascript with "ref" and "expandProps" option expands props 1`] = ` +"import * as React from \\"react\\"; + +function SvgComponent({ + svgRef, + ...props +}) { + return ; +} + +const ForwardRef = React.forwardRef((props, ref) => ); +export default ForwardRef;" +`; + +exports[`plugin javascript with "ref" option adds ForwardRef component 1`] = ` +"import * as React from \\"react\\"; + +function SvgComponent({ + svgRef +}) { + return ; +} + +const ForwardRef = React.forwardRef((props, ref) => ); +export default ForwardRef;" +`; + +exports[`plugin javascript with "titleProp" adds "titleProp" and "titleId" prop 1`] = ` +"import * as React from \\"react\\"; + +function SvgComponent({ + title, + titleId +}) { + return ; +} + +export default SvgComponent;" +`; + +exports[`plugin javascript with both "memo" and "ref" option wrap component in "React.memo" and "React.forwardRef" 1`] = ` +"import * as React from \\"react\\"; + +function SvgComponent({ + svgRef +}) { + return ; +} + +const MemoSvgComponent = React.memo(SvgComponent); +const ForwardRef = React.forwardRef((props, ref) => ); +export default ForwardRef;" +`; + +exports[`plugin typescript custom templates support basic template 1`] = ` +"import * as React from 'react'; + +const MyComponent = () => ; + +export default MyComponent;" +`; + +exports[`plugin typescript custom templates supports TypeScript template 1`] = ` +"import * as React from 'react'; + +const MyComponent = (props: React.SVGProps) => ; + +export default MyComponent;" +`; + +exports[`plugin typescript custom templates supports template that does not return an array 1`] = `";"`; + +exports[`plugin typescript transforms whole program 1`] = ` +"import * as React from \\"react\\"; + +function SvgComponent() { + return ; +} + +export default SvgComponent;" +`; + +exports[`plugin typescript with "expandProps" add props 1`] = ` +"import * as React from \\"react\\"; + +function SvgComponent(props: React.SVGProps) { + return ; +} + +export default SvgComponent;" +`; + +exports[`plugin typescript with "memo" option wrap component in "React.memo" 1`] = ` +"import * as React from \\"react\\"; + +function SvgComponent() { + return ; +} + +const MemoSvgComponent = React.memo(SvgComponent); +export default MemoSvgComponent;" +`; + +exports[`plugin typescript with "native" option adds import from "react-native-svg" 1`] = ` +"import * as React from \\"react\\"; +import Svg from \\"react-native-svg\\"; + +function SvgComponent() { + return ; +} + +export default SvgComponent;" +`; + +exports[`plugin typescript with "native.expo" option adds import from "react-native-svg" & from "expo" 1`] = ` +"import * as React from \\"react\\"; +import \\"expo\\"; + +function SvgComponent() { + return ; +} + +export default SvgComponent;" +`; + +exports[`plugin typescript with "ref" and "expandProps" option expands props 1`] = ` +"import * as React from \\"react\\"; +interface SVGRProps { + svgRef?: React.Ref +} + +function SvgComponent({ + svgRef, + ...props +}: React.SVGProps & SVGRProps) { + return ; +} + +const ForwardRef = React.forwardRef((props, ref) => ); +export default ForwardRef;" +`; + +exports[`plugin typescript with "ref" option adds ForwardRef component 1`] = ` +"import * as React from \\"react\\"; +interface SVGRProps { + svgRef?: React.Ref +} + +function SvgComponent({ + svgRef +}: SVGRProps) { + return ; +} + +const ForwardRef = React.forwardRef((props, ref) => ); +export default ForwardRef;" +`; + +exports[`plugin typescript with "titleProp" adds "titleProp" and "titleId" prop 1`] = ` +"import * as React from \\"react\\"; +interface SVGRProps { + title?: String, + titleId?: String, +} + +function SvgComponent({ + title, + titleId +}: SVGRProps) { + return ; +} + +export default SvgComponent;" +`; + +exports[`plugin typescript with both "memo" and "ref" option wrap component in "React.memo" and "React.forwardRef" 1`] = ` +"import * as React from \\"react\\"; +interface SVGRProps { + svgRef?: React.Ref +} + +function SvgComponent({ + svgRef +}: SVGRProps) { + return ; +} + +const MemoSvgComponent = React.memo(SvgComponent); +const ForwardRef = React.forwardRef((props, ref) => ); +export default ForwardRef;" +`; diff --git a/packages/babel-plugin-transform-svg-component/src/index.js b/packages/babel-plugin-transform-svg-component/src/index.js index 408b9cc3..ed202cd9 100644 --- a/packages/babel-plugin-transform-svg-component/src/index.js +++ b/packages/babel-plugin-transform-svg-component/src/index.js @@ -1,16 +1,22 @@ -import { getProps, getImport, getExport } from './util' +import { getProps, getImport, getExport, getInterface } from './util' function defaultTemplate( { template }, opts, - { imports, componentName, props, jsx, exports }, + { imports, interfaces, componentName, props, jsx, exports }, ) { - return template.ast`${imports} + const plugins = ['jsx'] + if (opts.typescript) { + plugins.push('typescript') + } + const typeScriptTpl = template.smart({ plugins }) + return typeScriptTpl.ast`${imports} +${interfaces} function ${componentName}(${props}) { return ${jsx}; } ${exports} -` + ` } const plugin = (api, opts) => ({ @@ -20,6 +26,7 @@ const plugin = (api, opts) => ({ const template = opts.template || defaultTemplate const body = template(api, opts, { componentName: t.identifier(opts.state.componentName), + interfaces: getInterface(api, opts), props: getProps(api, opts), imports: getImport(api, opts), exports: getExport(api, opts), diff --git a/packages/babel-plugin-transform-svg-component/src/index.test.js b/packages/babel-plugin-transform-svg-component/src/index.test.js index e9f126a2..311007eb 100644 --- a/packages/babel-plugin-transform-svg-component/src/index.test.js +++ b/packages/babel-plugin-transform-svg-component/src/index.test.js @@ -1,9 +1,12 @@ import { transform } from '@babel/core' import plugin from '.' -const testPlugin = (code, options) => { +const testPlugin = language => (code, options) => { const result = transform(code, { - plugins: ['@babel/plugin-syntax-jsx', [plugin, options]], + plugins: [ + '@babel/plugin-syntax-jsx', + [plugin, { ...options, typescript: language === 'typescript' }], + ], configFile: false, }) @@ -11,159 +14,150 @@ const testPlugin = (code, options) => { } describe('plugin', () => { - it('should transform whole program', () => { - const { code } = testPlugin('
', { - state: { componentName: 'SvgComponent' }, + describe.each(['javascript', 'typescript'])('%s', language => { + it('transforms whole program', () => { + const { code } = testPlugin(language)('', { + state: { componentName: 'SvgComponent' }, + }) + expect(code).toMatchSnapshot() }) - expect(code).toMatchInlineSnapshot(` - "import * as React from \\"react\\"; - function SvgComponent() { - return
; - } - - export default SvgComponent;" - `) - }) - - it('should add import for react-native-svg', () => { - const { code } = testPlugin('
', { - state: { componentName: 'SvgComponent' }, - native: true, + describe('with "native" option', () => { + it('adds import from "react-native-svg"', () => { + const { code } = testPlugin(language)('', { + state: { componentName: 'SvgComponent' }, + native: true, + }) + expect(code).toMatchSnapshot() + }) }) - expect(code).toMatchInlineSnapshot(` - "import * as React from \\"react\\"; - import Svg from \\"react-native-svg\\"; - - function SvgComponent() { - return
; - } - - export default SvgComponent;" - `) - }) - it('should import for expo', () => { - const { code } = testPlugin('
', { - state: { componentName: 'SvgComponent' }, - native: { expo: true }, + describe('with "native.expo" option', () => { + it('adds import from "react-native-svg" & from "expo"', () => { + const { code } = testPlugin(language)('', { + state: { componentName: 'SvgComponent' }, + native: { expo: true }, + }) + expect(code).toMatchSnapshot() + }) }) - expect(code).toMatchInlineSnapshot(` - "import * as React from \\"react\\"; - import \\"expo\\"; - function SvgComponent() { - return
; - } - - export default SvgComponent;" - `) - }) - - it('should support custom template', () => { - const { code } = testPlugin('
', { - template: ( - { template }, - opts, - { jsx }, - ) => template.ast`import * as React from 'react'; - const MyComponent = () => ${jsx} - export default MyComponent -`, - state: { componentName: 'SvgComponent' }, + describe('with "ref" option', () => { + it('adds ForwardRef component', () => { + const { code } = testPlugin(language)('', { + state: { componentName: 'SvgComponent' }, + ref: true, + }) + expect(code).toMatchSnapshot() + }) }) - expect(code).toMatchInlineSnapshot(` - "import * as React from 'react'; - - const MyComponent = () =>
; - - export default MyComponent;" - `) - }) - it('should support custom typescript template', () => { - const { code } = testPlugin('
', { - template: ({ template }, opts, { jsx }) => { - const typescriptTemplate = template.smart({ plugins: ['typescript'] }) - return typescriptTemplate.ast` - import * as React from 'react'; - const MyComponent = (props: React.SVGProps) => ${jsx}; - export default MyComponent; - ` - }, - state: { componentName: 'SvgComponent' }, + describe('with "titleProp"', () => { + it('adds "titleProp" and "titleId" prop', () => { + const { code } = testPlugin(language)('', { + state: { componentName: 'SvgComponent' }, + titleProp: true, + }) + expect(code).toMatchSnapshot() + }) }) - expect(code).toMatchInlineSnapshot(` - "import * as React from 'react'; - const MyComponent = (props: React.SVGProps) =>
; - - export default MyComponent;" - `) - }) - - it('should handle template that does not return an array', () => { - const { code } = testPlugin('
', { - template: ({ template }, opts, { jsx }) => template.ast`${jsx}`, - state: { componentName: 'SvgComponent' }, + describe('with "expandProps"', () => { + it('add props', () => { + const { code } = testPlugin(language)('', { + state: { componentName: 'SvgComponent' }, + expandProps: true, + }) + expect(code).toMatchSnapshot() + }) }) - expect(code).toMatchInlineSnapshot(`"
;"`) - }) - it('should work with ref', () => { - const { code } = testPlugin('
', { - state: { componentName: 'SvgComponent' }, - ref: true, + describe('with "ref" and "expandProps" option', () => { + it('expands props', () => { + const { code } = testPlugin(language)('', { + state: { componentName: 'SvgComponent' }, + expandProps: true, + ref: true, + }) + expect(code).toMatchSnapshot() + }) }) - expect(code).toMatchInlineSnapshot(` - "import * as React from \\"react\\"; - - function SvgComponent({ - svgRef - }) { - return
; - } - - const ForwardRef = React.forwardRef((props, ref) => ); - export default ForwardRef;" - `) - }) - it('should work with memo', () => { - const { code } = testPlugin('
', { - state: { componentName: 'SvgComponent' }, - memo: true, + describe('with "memo" option', () => { + it('wrap component in "React.memo"', () => { + const { code } = testPlugin(language)('', { + state: { componentName: 'SvgComponent' }, + memo: true, + }) + expect(code).toMatchSnapshot() + }) }) - expect(code).toMatchInlineSnapshot(` - "import * as React from \\"react\\"; - - function SvgComponent() { - return
; - } - - const MemoSvgComponent = React.memo(SvgComponent); - export default MemoSvgComponent;" - `) - }) - it('should work with memo + ref', () => { - const { code } = testPlugin('
', { - state: { componentName: 'SvgComponent' }, - memo: true, - ref: true, + describe('with both "memo" and "ref" option', () => { + it('wrap component in "React.memo" and "React.forwardRef"', () => { + const { code } = testPlugin(language)('', { + state: { componentName: 'SvgComponent' }, + memo: true, + ref: true, + }) + expect(code).toMatchSnapshot() + }) }) - expect(code).toMatchInlineSnapshot(` - "import * as React from \\"react\\"; - - function SvgComponent({ - svgRef - }) { - return
; - } - const MemoSvgComponent = React.memo(SvgComponent); - const ForwardRef = React.forwardRef((props, ref) => ); - export default ForwardRef;" - `) + describe('custom templates', () => { + it('support basic template', () => { + const { code } = testPlugin(language)('', { + template: ( + { template }, + opts, + { jsx }, + ) => template.ast`import * as React from 'react'; + const MyComponent = () => ${jsx} + export default MyComponent + `, + state: { componentName: 'SvgComponent' }, + }) + expect(code).toMatchSnapshot() + }) + + describe('it supports JSX template', () => { + const { code } = testPlugin(language)('', { + template: ({ template }, opts, { jsx }) => { + const jsxTemplate = template.smart({ plugins: ['jsx'] }) + return jsxTemplate.ast`import * as React from 'react'; + const MyComponent = () =>
{${jsx}}
+ export default MyComponent + ` + }, + state: { componentName: 'SvgComponent' }, + }) + expect(code).toMatchSnapshot() + }) + + it('supports TypeScript template', () => { + const { code } = testPlugin(language)('', { + template: ({ template }, opts, { jsx }) => { + const typescriptTemplate = template.smart({ + plugins: ['typescript'], + }) + return typescriptTemplate.ast` + import * as React from 'react'; + const MyComponent = (props: React.SVGProps) => ${jsx}; + export default MyComponent; + ` + }, + state: { componentName: 'SvgComponent' }, + }) + expect(code).toMatchSnapshot() + }) + + it('supports template that does not return an array', () => { + const { code } = testPlugin(language)('', { + template: ({ template }, opts, { jsx }) => template.ast`${jsx}`, + state: { componentName: 'SvgComponent' }, + }) + expect(code).toMatchSnapshot() + }) + }) }) }) diff --git a/packages/babel-plugin-transform-svg-component/src/util.js b/packages/babel-plugin-transform-svg-component/src/util.js index 6a69a556..375bc333 100644 --- a/packages/babel-plugin-transform-svg-component/src/util.js +++ b/packages/babel-plugin-transform-svg-component/src/util.js @@ -1,3 +1,74 @@ +function typeAnnotation(typeAnnotation) { + return { + type: 'TypeAnnotation', + typeAnnotation, + } +} + +function genericTypeAnnotation(id, typeParameters = null) { + return { type: 'GenericTypeAnnotation', id, typeParameters } +} + +function typeParameters(params) { + return { + type: 'TypeParameterInstantiation', + params, + } +} + +function qualifiedTypeIdentifier(qualification, id) { + return { type: 'QualifiedTypeIdentifier', qualification, id } +} + +function intersectionTypeAnnotation(types) { + return { type: 'IntersectionTypeAnnotation', types } +} + +function interfaceDeclaration(id, body) { + return { + type: 'InterfaceDeclaration', + id, + typeParameters: null, + extends: [], + implements: [], + mixins: [], + body, + } +} + +function objectTypeAnnotation(properties) { + return { + type: 'ObjectTypeAnnotation', + properties, + } +} + +function objectTypeProperty(key, value, optional = false) { + return { + type: 'ObjectTypeProperty', + key, + static: false, + proto: false, + kind: 'init', + method: false, + value, + variance: null, + optional, + } +} + +function addTypeAnotation(obj, typeAnnotation, opts) { + if (!opts.typescript) return obj + return { ...obj, typeAnnotation } +} + +function getSvgPropsTypeAnnotation(t) { + return genericTypeAnnotation( + qualifiedTypeIdentifier(t.identifier('React'), t.identifier('SVGProps')), + typeParameters([genericTypeAnnotation(t.identifier('SVGSVGElement'))]), + ) +} + export const getProps = ({ types: t }, opts) => { const props = [] @@ -41,10 +112,57 @@ export const getProps = ({ types: t }, opts) => { } if (props.length === 1 && opts.expandProps) { - return t.identifier('props') + return addTypeAnotation( + t.identifier('props'), + typeAnnotation(getSvgPropsTypeAnnotation(t)), + opts, + ) } - return t.objectPattern(props) + return addTypeAnotation( + t.objectPattern(props), + typeAnnotation( + opts.expandProps + ? intersectionTypeAnnotation([ + getSvgPropsTypeAnnotation(t), + genericTypeAnnotation(t.identifier('SVGRProps')), + ]) + : genericTypeAnnotation(t.identifier('SVGRProps')), + ), + opts, + ) +} + +export const getInterface = ({ types: t }, opts) => { + if (!opts.typescript) return null + const properties = [] + if (opts.ref) { + properties.push( + objectTypeProperty( + t.identifier('svgRef'), + genericTypeAnnotation( + qualifiedTypeIdentifier(t.identifier('React'), t.identifier('Ref')), + typeParameters([ + genericTypeAnnotation(t.identifier('SVGSVGElement')), + ]), + ), + true, + ), + ) + } + if (opts.titleProp) { + properties.push( + objectTypeProperty(t.identifier('title'), t.identifier('String'), true), + ) + properties.push( + objectTypeProperty(t.identifier('titleId'), t.identifier('String'), true), + ) + } + if (properties.length === 0) return null + return interfaceDeclaration( + t.identifier('SVGRProps'), + objectTypeAnnotation(properties), + ) } export const getImport = ({ types: t }, opts) => { diff --git a/website/src/pages/docs/custom-templates.mdx b/website/src/pages/docs/custom-templates.mdx new file mode 100644 index 00000000..20d43098 --- /dev/null +++ b/website/src/pages/docs/custom-templates.mdx @@ -0,0 +1,93 @@ +--- +menu: Advanced +title: Custom Templates +order: 6 +--- + +# Custom Templates + +Custom templates give you the opportunity to personalize the final generated component by SVGR. In most of case you don't need it, only advanced use-cases require templates. + +## Create a custom template + +A custom template takes place in a file that exports a "template function". + +This function is called in a babel plugin: `babel-plugin-transform-svg-component` and must returns a Babel AST. If you are not familiar with all Babel stuff, you should read [this guide](https://github.com/jamiebuilds/babel-handbook). + +- `api`: The API object returned by Babel +- `opts`: Options passed to `babel-plugin-transform-svg-component` +- `astParts`: All pre-compiled parts by SVGR + - `componentName`: The component name + - `props`: The properties + - `interfaces`: All necessary interfaces (typescript) + - `imports`: All necessary imports + - `exports`: The export of the component + - `jsx`: The JSX part of the component + +The following template is the default template used by SVGR. It is a good idea to start with it: + +```js +function defaultTemplate( + { template }, + opts, + { imports, componentName, props, jsx, exports }, +) { + return template.ast`${imports} +function ${componentName}(${props}) { + return ${jsx}; +} +${exports} +` +} + +module.exports = defaultTemplate +``` + +As you can see, we use [the `template.ast` helper of Babel](https://babeljs.io/docs/en/babel-template#template). This function have to return an AST. + +Let's try something very simple. You want to add some PropTypes to your component: + +```js +function propTypesTemplate( + { template }, + opts, + { imports, componentName, props, jsx, exports }, +) { + return template.ast`${imports} +import PropTypes from 'prop-types'; + +function ${componentName}(${props}) { + return ${jsx}; +} + +${componentName}.propTypes = { + title: PropTypes.string, +}; + +${exports} +` +} + +module.exports = propTypesTemplate +``` + +As you can see it is very natural, we just add code and use AST parts in the template. + +## Use template with CLI + +You can use this template in the CLI, along the `--ext` argument: + +```sh +$ npx @svgr/cli --template my/template.js --ext .ts my-icon.svg +``` + +## Use template in config + +Specify `.svgrrc.js`: + +```js +// .svgrrc.js +module.exports = { + template: require('./my-template'), +} +``` diff --git a/website/src/pages/docs/typescript.mdx b/website/src/pages/docs/typescript.mdx deleted file mode 100644 index c8ba4e93..00000000 --- a/website/src/pages/docs/typescript.mdx +++ /dev/null @@ -1,57 +0,0 @@ ---- -menu: Advanced -title: TypeScript -order: 5 ---- - -# TypeScript - -Every languages supported by Babel are also supported by SVGR including TypeScript and Flow. - -## Create a custom template - -To target TypeScript, you need to create a custom template: - -```js -function template( - { template }, - opts, - { imports, componentName, props, jsx, exports } -) { - const typeScriptTpl = template.smart({ plugins: ['typescript'] }) - return typeScriptTpl.ast` - import * as React from 'react'; - const ${componentName} = (props: React.SVGProps) => ${jsx}; - export default ${componentName}; - ` -} - -module.exports = template -``` - -You can use this template in the CLI, along the `--ext` argument: - -```sh -$ npx @svgr/cli --template path/to/template.js --ext .ts my-icon.svg -``` - -Or specify it in your `.svgrrc.js`: - -```js -module.exports = { - template( - { template }, - opts, - { imports, componentName, props, jsx, exports } - ) { - const typeScriptTpl = template.smart({ plugins: ['typescript'] }) - return typeScriptTpl.ast` - import * as React from 'react'; - const ${componentName} = (props: React.SVGProps) => ${jsx}; - export default ${componentName}; - ` - }, -} -``` - -> To target flow, follow the same steps and replace `plugins: ['typescript']` by `plugins: ['flow']`