Webpack + Typescript:获取模块解析失败 - 为什么?

时间:2017-04-05 20:51:14

标签: javascript typescript webpack

我正在使用Typescript和Webpack构建一个knockoutJs项目,而且我遇到了一个非常奇怪的问题。

webpack失败的模块是:

import esPlayerVM from "./components/es-player";
import esPlayerTemplate from "./components/es-player.tmpl.html";
import fsButtonVm from "./components/fullscreen-button";
import fsButtonHtml from "./components/fullscreen-button.tmpl.html";


export default class Registrator {

    static Register() {
        var koComp = (elName: string, vm: any, html: string) => {
            ko.components.register(elName, { viewModel: vm, template: html });
        }

        koComp('es-player', esPlayerVM, esPlayerTemplate);
        koComp('fullscreen-button', fsButtonVm, fsButtonHtml);
        //ko.components.register('es-player', { viewModel: esPlayerVM, template: esPlayerTemplate });
        //this.RegisterComponent('fullscreen-button', fsButtonVm, fsButtonHtml);
    }
}

Webpack输出:

ERROR in ./ComponentRegistration.ts
Module parse failed: C:\Users\Marlon\Documents\Projects\eps-ko\src\ComponentRegistration.ts Unexpected token (10:28)
You may need an appropriate loader to handle this file type.
|
|     static Register() {
|         var koComp = (elName: string, vm: any, html: string) => {
|             ko.components.register(elName, { viewModel: vm, template: html });
|         }
 @ ./Main.ts 16:0-50

如果删除koComp varable,并在Register方法中直接注册该组件,webpack很高兴。最初我有一个私有静态函数,你可以从底部注释中看到,但是webpack也遇到了同样的问题。

我用谷歌搜索,其他人在打字稿输出es2016代码时得到这个,所以他们必须通过babel传递它,但是我的目标是es5,所以我希望我的好吗?我没有使用像.tsx文件这样的任何反应。

我同时使用了ts-loaderawesome-typescript-loader,两者都有相同的错误。

这是我的tsconfig.json

{
    "compilerOptions": {
        "allowSyntheticDefaultImports": true,
        "module": "es2015",
        "target": "es5",
        "moduleResolution": "node",
        "noImplicitAny": true,
        "sourceMap": false,
        "importHelpers": true
    }
}

最初我将modules设置为commonjs,但在尝试ts-loader时我遵循他们的榜样。

这是我的webpack.config

const path = require('path');
const webpack = require('webpack');

const ExtractTextPlugin = require('extract-text-webpack-plugin');
const { CheckerPlugin } = require('awesome-typescript-loader')

// Create multiple instances
const extractCSS = new ExtractTextPlugin('es-player.bundle.css');

module.exports = {
  context: path.resolve(__dirname, './src'),
  entry: {
    app: './Main.ts',
  },
  output: {
    path: path.resolve(__dirname, './dist'),
    filename: 'es-player.bundle.js',
  },
  //devtool: "cheap-eval-source-map",
  resolve: {
    extensions: ['.ts', '.tsx']
  },    
  module: {
    loaders: [
      {
        test: /\.tsx?$/,
        exclude: /node_modules/,
        loader: 'ts-loader'
      }
    ],
    rules: [
      {
        test: /\.css$/,
        use: extractCSS.extract([ 'css-loader', 'postcss-loader' ])
      },
      {
        test: /\.less$/i,
        use: extractCSS.extract([ 'css-loader', 'less-loader', 'postcss-loader' ])
      },
      {
        test: /\.tmpl.html?$/,
        use: 'raw-loader'
      }
    ]
  },
  plugins: [
    extractCSS
    //new CheckerPlugin()
  ],
};

版本:

Typescript: 2.2.2
Webpack: 2.3.3
awesome-typescript-loader: 3.1.2
ts-loader: 2.0.3

我可以直接注册它们,但我想知道发生了什么导致了这个错误。

修改

在挖掘和删除代码之后,如果任何导出的模块在任何方法上都有类型参数等,似乎webpack失败。这真的很奇怪,因为我认为webpack正在处理从加载器生成的转换后的javascript?这推断它正在尝试使用原始的.ts文件。

1 个答案:

答案 0 :(得分:1)

TL:DR WebPack不支持使用loadersrules部分 - 必须使用其中一个。

经过数小时的调试,这是因为在loaders文件的webpack.config.js部分中定义了打字稿加载程序。将其移至rules部分可解决此问题。

修改后的webpack.config.js文件如下所示:

const path = require('path');
const ExtractTextPlugin = require('extract-text-webpack-plugin');
const { CheckerPlugin } = require('awesome-typescript-loader')

const extractCSS = new ExtractTextPlugin('es-player.bundle.css');

module.exports = {
  context: path.resolve(__dirname, './src'),
  entry: {
    app: './Main.ts',
  },
  output: {
    path: path.resolve(__dirname, './dist'),
    filename: 'es-player.bundle.js',
  },
  //devtool: "cheap-eval-source-map",
  resolve: {
    extensions: ['.js', '.ts', '.tsx']
  },    
  module: {
    rules: [
      {
        test: /\.css$/,
        use: extractCSS.extract([ 'css-loader', 'postcss-loader' ])
      },
      {
        test: /\.less$/i,
        use: extractCSS.extract([ 'css-loader', 'less-loader', 'postcss-loader' ])
      },
      {
        test: /\.tmpl.html?$/,
        use: 'raw-loader'
      },
      {
        test: /\.tsx?$/,
        exclude: /node_modules/,
        loader: 'awesome-typescript-loader'
      }
    ]
  },
  plugins: [
    extractCSS,
    new CheckerPlugin()
  ],
};

ts-loaderat-loader状态都在各自的github页面的loaders部分中定义它们。但Webpack 2 docs声明将它们放入rules部分。至少可以说很困惑。