preset.ts 4.0 KB
Newer Older
1
import { PluginItem } from '@babel/core'
2 3 4 5 6 7
const env = process.env.NODE_ENV
const isProduction = env === 'production'
const isDevelopment = env === 'development'
const isTest = env === 'test'

type StyledJsxPlugin = [string, any] | string
8 9
type StyledJsxBabelOptions =
  | {
J
Joe Haddad 已提交
10
      plugins?: StyledJsxPlugin[]
11
      'babel-test'?: boolean
12 13
    }
  | undefined
14 15

// Resolve styled-jsx plugins
16
function styledJsxOptions(options: StyledJsxBabelOptions) {
17 18 19 20 21 22 23 24
  if (!options) {
    return {}
  }

  if (!Array.isArray(options.plugins)) {
    return options
  }

25 26 27 28 29 30
  options.plugins = options.plugins.map(
    (plugin: StyledJsxPlugin): StyledJsxPlugin => {
      if (Array.isArray(plugin)) {
        const [name, options] = plugin
        return [require.resolve(name), options]
      }
31

32 33 34
      return require.resolve(plugin)
    }
  )
35 36 37 38 39

  return options
}

type NextBabelPresetOptions = {
40 41 42 43
  'preset-env'?: any
  'preset-react'?: any
  'class-properties'?: any
  'transform-runtime'?: any
44 45 46 47
  'styled-jsx'?: StyledJsxBabelOptions
}

type BabelPreset = {
48 49
  presets?: PluginItem[] | null
  plugins?: PluginItem[] | null
50
  sourceType?: 'script' | 'module' | 'unambiguous'
51
  overrides?: any[]
52 53
}

54 55
// Taken from https://github.com/babel/babel/commit/d60c5e1736543a6eac4b549553e107a9ba967051#diff-b4beead8ad9195361b4537601cc22532R158
function supportsStaticESM(caller: any) {
56
  return !!(caller && caller.supportsStaticESM)
57 58
}

59 60 61 62
module.exports = (
  api: any,
  options: NextBabelPresetOptions = {}
): BabelPreset => {
63
  const supportsESM = api.caller(supportsStaticESM)
J
Jason Miller 已提交
64
  const isServer = api.caller((caller: any) => !!caller && caller.isServer)
65 66 67
  const presetEnvConfig = {
    // In the test environment `modules` is often needed to be set to true, babel figures that out by itself using the `'auto'` option
    // In production/development this option is set to `false` so that webpack can handle import/export with tree-shaking
68
    modules: 'auto',
69
    exclude: ['transform-typeof-symbol'],
70
    ...options['preset-env'],
71
  }
J
Jason Miller 已提交
72

73 74
  // When transpiling for the server or tests, target the current Node version
  // if not explicitly specified:
J
Jason Miller 已提交
75
  if (
76
    (isServer || isTest) &&
77 78 79 80 81
    (!presetEnvConfig.targets ||
      !(
        typeof presetEnvConfig.targets === 'object' &&
        'node' in presetEnvConfig.targets
      ))
J
Jason Miller 已提交
82 83 84 85 86 87 88 89
  ) {
    presetEnvConfig.targets = {
      // Targets the current process' version of Node. This requires apps be
      // built and deployed on the same version of Node.
      node: 'current',
    }
  }

90
  return {
91
    sourceType: 'unambiguous',
92
    presets: [
93
      [require('@babel/preset-env').default, presetEnvConfig],
94 95 96 97 98 99 100 101 102 103
      [
        require('@babel/preset-react'),
        {
          // This adds @babel/plugin-transform-react-jsx-source and
          // @babel/plugin-transform-react-jsx-self automatically in development
          development: isDevelopment || isTest,
          ...options['preset-react'],
        },
      ],
      require('@babel/preset-typescript'),
104 105 106 107 108
    ],
    plugins: [
      require('babel-plugin-react-require'),
      require('@babel/plugin-syntax-dynamic-import'),
      require('./plugins/react-loadable-plugin'),
109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128
      [
        require('@babel/plugin-proposal-class-properties'),
        options['class-properties'] || {},
      ],
      [
        require('@babel/plugin-proposal-object-rest-spread'),
        {
          useBuiltIns: true,
        },
      ],
      [
        require('@babel/plugin-transform-runtime'),
        {
          corejs: 2,
          helpers: true,
          regenerator: true,
          useESModules: supportsESM && presetEnvConfig.modules !== 'commonjs',
          ...options['transform-runtime'],
        },
      ],
J
Joe Haddad 已提交
129 130 131 132 133 134
      [
        isTest && options['styled-jsx'] && options['styled-jsx']['babel-test']
          ? require('styled-jsx/babel-test')
          : require('styled-jsx/babel'),
        styledJsxOptions(options['styled-jsx']),
      ],
135
      require('./plugins/amp-attributes'),
136 137 138 139 140 141 142
      isProduction && [
        require('babel-plugin-transform-react-remove-prop-types'),
        {
          removeImport: true,
        },
      ],
    ].filter(Boolean),
143 144
  }
}