次のTypeScriptプログラムをES5に変換しました。
ファイル1:
class BaseElement extends HTMLElement {
constructor() {
super();
}
}
ファイル2:
import {BaseElement} from './BaseElement';
class MyElement extends BaseElement {
constructor() {
super();
}
}
var el = new MyElement();
ファイル内にすべてを手動で配置すると、コードは正常に機能し、ブラウザーで実行され、HTMLElementは問題なく構築されます。ただし、webpackでパックするとすぐに、次のエラーメッセージが表示されます。
Uncaught TypeError: Failed to construct 'HTMLElement': Please use the 'new' operator, this DOM object constructor cannot be called as a function.
Webpackを使用しない場合、次のJSコードが作成されます。
var __extends = (this && this.__extends) || function (d, b) {
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
function __() { this.constructor = d; }
d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
};
var BaseElement = (function (_super) {
__extends(BaseElement, _super);
function BaseElement() {
_super.call(this);
}
return BaseElement;
}(HTMLElement));
var MyElement = (function (_super) {
__extends(MyElement, _super);
function MyElement() {
_super.call(this);
}
MyElement.prototype.createdCallback = function () {
this.innerHTML = "lol";
};
return MyElement;
}(BaseElement));
var el = new MyElement();
Webpackを使用して、次のコードが作成されます。
var __extends = (this && this.__extends) || function (d, b) {
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
function __() { this.constructor = d; }
d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
};
/******/ (function(modules) { // webpackBootstrap
/******/ // The module cache
/******/ var installedModules = {};
/******/ // The require function
/******/ function __webpack_require__(moduleId) {
/******/ // Check if module is in cache
/******/ if(installedModules[moduleId])
/******/ return installedModules[moduleId].exports;
/******/ // Create a new module (and put it into the cache)
/******/ var module = installedModules[moduleId] = {
/******/ exports: {},
/******/ id: moduleId,
/******/ loaded: false
/******/ };
/******/ // Execute the module function
/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
/******/ // Flag the module as loaded
/******/ module.loaded = true;
/******/ // Return the exports of the module
/******/ return module.exports;
/******/ }
/******/ // expose the modules object (__webpack_modules__)
/******/ __webpack_require__.m = modules;
/******/ // expose the module cache
/******/ __webpack_require__.c = installedModules;
/******/ // __webpack_public_path__
/******/ __webpack_require__.p = "";
/******/ // Load entry module and return exports
/******/ return __webpack_require__(0);
/******/ })
/************************************************************************/
/******/ ([
/* 0 */
/***/ function(module, exports, __webpack_require__) {
__webpack_require__(1);
__webpack_require__(2);
/***/ },
/* 1 */
/***/ function(module, exports) {
"use strict";
var BaseElement = (function (_super) {
__extends(BaseElement, _super);
function BaseElement() {
_super.call(this);
}
return BaseElement;
}(HTMLElement));
exports.BaseElement = BaseElement;
/***/ },
/* 2 */
/***/ function(module, exports, __webpack_require__) {
"use strict";
var BaseElement_1 = __webpack_require__(1);
var MyElement = (function (_super) {
__extends(MyElement, _super);
function MyElement() {
_super.call(this);
}
MyElement.prototype.createdCallback = function () {
this.innerHTML = "lol";
};
return MyElement;
}(BaseElement_1.BaseElement));
exports.MyElement = MyElement;
// TODO: inject
var p = new MyElement();
/***/ }
/******/ ]);
基本的に、webpackはモジュールを関数に入れ、それらの間でエクスポート変数を維持しますが、HTMLElementの構築は失敗します。 webpack(上記のコード)がなければ、正常に動作します。
何か案は?
蒸散問題です。 ES5をトランスパイルまたは使用している場合は、ネイティブWebコンポーネントをサポートするブラウザー用のネイティブシムをバンドルする必要があります( https://github.com/webcomponents/custom-elements/blob/master/src/native- shim.js )
HTMLElementコンストラクターは_
new.target
_の値を使用して現在呼び出されているコンストラクターのカスタム要素定義を検索するため、ES5スタイルのクラスはネイティブカスタム要素では機能しません。 _new.target
_は、new
が呼び出されたときにのみ設定され、super()呼び出しによってのみ伝達されます。 super()はES5ではエミュレートできません。SuperClass.call(this)`` only works when extending other ES5-style classes, and does not propagate
new.target`のパターン。
問題のディスカッションを確認してください https://github.com/webcomponents/custom-elements/issues/29
ES5スタイルのクラスは、ネイティブカスタム要素では機能しません
回避策を実行するには、tsconfig.jsonファイルのターゲットをes6に変更するだけです。
この問題を解決してこの問題を修正しました- https://github.com/facebook/create-react-app/issues/3225
基本的に私はnpmを介してこれら2つのプラグインをインストールし、私のWebPack構成にそれらの2つのプラグインをBabelに追加しました:
use: [
{
loader: 'babel-loader',
options: {
presets: ['es2015'],
// https://github.com/facebook/create-react-app/issues/3225
plugins: ['transform-custom-element-classes', 'transform-es2015-classes']
},
}
],
1)Babel 7.6.0
個人的には、エラーはこれらの特定のdevDependenciesでなくなっているようです:
"devDependencies": {
"@babel/core": "^7.6.0",
"@babel/preset-env": "^7.6.0",
"babel-loader": "^8.0.6",
"webpack": "^4.39.3",
"webpack-cli": "^3.3.8"
}
そして、このwebpack.config.js:
var glob = require('glob');
var path = require('path');
module.exports = {
entry: glob.sync('./app/scripts/**.js').reduce(function(obj, el){
obj[path.parse(el).name] = el;
return obj
},{}),
output: {
path: path.resolve(__dirname, './dist/scripts'),
filename: "[name].js"
},
module: {
rules: [
{
test: /\.js$/,
loader: 'babel-loader',
include: [
path.resolve(__dirname, 'app/scripts')
],
options: {
presets: ['@babel/env']
}
}
]
}
};
つまり、基本的には/ app/scriptsフォルダー内の.jsファイルをトランスパイルし、@babel/preset-env
パッケージを使用してbabel-loaderを使用して/ dist/scriptsに保存するようにwebpackに指示しています。
2)バベル6。*。0
ただし、@babel/core
6.*.*
を使用している場合は、これを確認することをお勧めします https://medium.com/@allenhwkim/chrome-browser-custom-element-error-e86db5ae3b8c 。それは非常に単純で、すべてのbabelパッケージを更新しようとする前に、私はすでにそれをうまく使用していました。
あなたがする必要がある「すべて」はnpm install babel-plugin-transform-es2015-classes babel-plugin-transform-custom-element-classes --save-dev
であり、それらをwebpack.config.jsに追加します(npm install --save-dev babel-preset-es2015
も忘れないでください):
module: {
rules: [
{
test: /\.js$/,
loader: 'babel-loader',
include: [
path.resolve(__dirname, 'app/scripts')
],
options: {
presets: ['es2015'],
plubins: ["transform-custom-element-classes", "transform-es2015-classes"]
}
}
]
}
Webpackがなくても動作しますか? playground を介して実行すると、(実行時に)説明したのと同じエラーが発生します。
とにかく、HTMLElement
を拡張するべきではありません。HTMLElement
は、実際にはTypeScriptのインターフェイスであるため、何らかの形で実装する必要があります。
ブラウザではオブジェクトタイプとして存在しますが、TypeScriptクラスとして宣言されていないため、TypeScriptはそれを適切に拡張できません。
この問題を回避する方法については、この answer を参照してください。
Webコンポーネント用のBabel 7 + Webpack 4構成:
package.json:
"devDependencies": {
"@babel/core": "^7.3.4",
"@babel/plugin-proposal-class-properties": "^7.3.4",
"@babel/preset-env": "^7.3.4",
"babel-loader": "^8.0.5",
"babel-plugin-transform-custom-element-classes": "^0.1.0",
"webpack": "^4.29.6",
"webpack-cli": "^3.2.3",
"webpack-dev-server": "^3.2.1"
}
webpack.config.js:
module: {
rules: [
{
test: /\.js$/,
use:{
loader: 'babel-loader',
options: {
presets: ['@babel/preset-env'],
plugins: [
"transform-custom-element-classes",
"@babel/plugin-proposal-class-properties",
]
},
},
exclude: /node_modules/
}
]
}
transform-es2015-classes
プラグインを使用すると、Babel 7を使用するときにbabel-preset-env
にすでに含まれているため、ビルドプロセスが中断されます。 @babel/plugin-proposal-class-properties
は、ライフサイクルコールバックに必要です。 es2015
などの毎年のプリセットの使用はBabel 7で廃止されました。代わりに@babel/preset-env
を使用してください。