BabelJS - wtyczki Babel
BabelJS to kompilator javascript, który zmienia składnię podanego kodu w oparciu o dostępne ustawienia i wtyczki. Przepływ kompilacji Babel obejmuje następujące 3 części -
- parsing
- transforming
- printing
Kod przekazany babel jest zwracany w takiej postaci, w jakiej został zmieniony. Widzieliśmy już ustawienia wstępne dodawane do pliku .babelrc w celu kompilacji kodu od es6 do es5 lub odwrotnie. Presety to nic innego jak zestaw wtyczek. Babel niczego nie zmieni, jeśli ustawienia wstępne lub szczegóły wtyczek nie zostaną podane podczas kompilacji.
Omówmy teraz następujące wtyczki -
- transform-class-properties
- Transform-exponentiation-operator
- For-of
- obiekt odpoczywa i rozprzestrzenia się
- async/await
Teraz stworzymy konfigurację projektu i popracujemy nad kilkoma wtyczkami, co da jasne zrozumienie wymagań wtyczek w babel.
Komenda
npm init
Musimy zainstalować wymagane pakiety dla babel - babel cli, babel core, babel-preset itp.
Pakiety do Babel 6
npm install babel-cli babel-core babel-preset-es2015 --save-dev
Pakiety do Babel 7
npm install @babel/cli @babel/core @babel/preset-env --save-dev
Utwórz plik js w swoim projekcie i napisz kod js.
Klasy - Właściwości klasy-transformacji
W tym celu należy przestrzegać poniższych kodów -
Przykład
main.js
class Person {
constructor(fname, lname, age, address) {
this.fname = fname;
this.lname = lname;
this.age = age;
this.address = address;
}
get fullname() {
return this.fname + "-" + this.lname;
}
}
var a = new Person("Siya", "Kapoor", "15", "Mumbai");
var persondet = a.fullname;
W tej chwili nie przekazaliśmy babel żadnych szczegółów ustawień wstępnych ani wtyczek. Jeśli zdarzy nam się przetransponować kod za pomocą polecenia -
npx babel main.js --out-file main_out.js
main_out.js
class Person {
constructor(fname, lname, age, address) {
this.fname = fname;
this.lname = lname;
this.age = age;
this.address = address;
}
get fullname() {
return this.fname + "-" + this.lname;
}
}
var a = new Person("Siya", "Kapoor", "15", "Mumbai");
var persondet = a.fullname;
Otrzymamy kod tak, jak jest. Dodajmy teraz preset do.babelrc plik.
Note - Utwórz .babelrc plik w folderze głównym projektu.
.babelrc for babel 6
.babelrc for babel 7
{
"presets":["@babel/env"]
}
Zainstalowaliśmy już ustawienia wstępne; teraz uruchommy ponownie polecenie -
npx babel main.js --out-file main_out.js
main_out.js
"use strict";
var _createClass = function () {
function defineProperties(target, props) {
for (var i = 0; i < props.length; i++) {
var descriptor = props[i];
descriptor.enumerable = descriptor.enumerable || false;
descriptor.configurable = true;
if ("value" in descriptor) descriptor.writable = true;
Object.defineProperty(target, descriptor.key, descriptor);
}
}
return function (Constructor, protoProps, staticProps) {
if (protoProps) defineProperties(Constructor.prototype, protoProps);
if (staticProps) defineProperties(Constructor, staticProps);
return Constructor;
};
}();
function _classCallCheck(instance, Constructor) {
if (!(instance instanceof Constructor)) {
throw new TypeError("Cannot call a class as a function");
}
}
var Person = function () {
function Person(fname, lname, age, address) {
_classCallCheck(this, Person);
this.fname = fname;
this.lname = lname;
this.age = age;
this.address = address;
}
_createClass(Person, [{
key: "fullname",
get: function get() {
return this.fname + "-" + this.lname;
}
}]);
return Person;
}();
var a = new Person("Siya", "Kapoor", "15", "Mumbai");
var persondet = a.fullname;
W ES6 składnia klas jest następująca
class Person {
constructor(fname, lname, age, address) {
this.fname = fname;
this.lname = lname;
this.age = age;
this.address = address;
}
get fullname() {
return this.fname + "-" + this.lname;
}
}
Istnieje konstruktor, w którym zdefiniowane są wszystkie właściwości klasy. Okazało się, że musimy zdefiniować właściwości klasy poza klasą, której nie możemy zrobić.
Przykład
class Person {
name = "Siya Kapoor";
fullname = () => {
return this.name;
}
}
var a = new Person();
var persondet = a.fullname();
console.log("%c"+persondet, "font-size:25px;color:red;");
Jeśli zdarzy nam się skompilować powyższy kod, spowoduje to błąd w babel. Powoduje to, że kod nie jest kompilowany.
Aby to działało tak, jak chcemy, możemy skorzystać z wtyczki babel o nazwie babel-plugin-transform-class-properties. Aby to działało, musimy go najpierw zainstalować w następujący sposób -
Pakiety do Babel 6
npm install --save-dev babel-plugin-transform-class-properties
Opakowanie na babel 7
npm install --save-dev @babel/plugin-proposal-class-properties
Add the plugin to .babelrc file for babel 6 -
.babelrc for babel 7
{
"plugins": ["@babel/plugin-proposal-class-properties"]
}
Teraz ponownie uruchomimy polecenie.
Komenda
npx babel main.js --out-file main_out.js
main.js
class Person {
name = "Siya Kapoor";
fullname = () => {
return this.name;
}
}
var a = new Person();
var persondet = a.fullname();
console.log("%c"+persondet, "font-size:25px;color:red;");
Compiled to main_out.js
class Person {
constructor() {
this.name = "Siya Kapoor";
this.fullname = () => {
return this.name;
};
}
}
var a = new Person();
var persondet = a.fullname();
console.log("%c"+persondet, "font-size:25px;color:red;");
Output
Poniżej przedstawiono dane wyjściowe, które otrzymujemy podczas korzystania z przeglądarki -
Operator potęgowania - operator potęgowania transformacji
** jest operatorem używanym do potęgowania w ES7. Poniższy przykład pokazuje działanie tego samego w ES7. Pokazuje również, jak przetransponować kod za pomocą babeljs.
Przykład
let sqr = 9 ** 2;
console.log("%c"+sqr, "font-size:25px;color:red;");
Aby przetransponować operator potęgowania, potrzebujemy wtyczki do zainstalowania w następujący sposób -
Packages for babel 6
npm install --save-dev babel-plugin-transform-exponentiation-operator
Packages for babel 7
npm install --save-dev @babel/plugin-transform-exponentiation-operator
Dodaj szczegóły wtyczki do .babelrc plik w następujący sposób dla babel 6 -
{
"plugins": ["transform-exponentiation-operator"]
}
.babelrc for babel 7
{
"plugins": ["@babel/plugin-transform-exponentiation-operator"]
}
command
npx babel exponeniation.js --out-file exponeniation_out.js
exponeniation_out.js
let sqr = Math.pow(9, 2);
console.log("%c" + sqr, "font-size:25px;color:red;");
Output
For-of
Pakiety wymagane dla wtyczek w babel6 i 7 są następujące -
Babel6
npm install --save-dev babel-plugin-transform-es2015-for-of
Babel 7
npm install --save-dev @babel/plugin-transform-for-of
.babelrc for babel6
{
"plugins": ["transform-es2015-for-of"]
}
.babelrc for babel7
{
"plugins": ["@babel/plugin-transform-for-of"]
}
forof.js
let foo = ["PHP", "C++", "Mysql", "JAVA"];
for (var i of foo) {
console.log(i);
}
Komenda
npx babel forof.js --out-file forof_es5.js
Forof_es5.js
let foo = ["PHP", "C++", "Mysql", "JAVA"];
var _iteratorNormalCompletion = true;
var _didIteratorError = false;
var _iteratorError = undefined;
try {
for (var _iterator = foo[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {
var i = _step.value;
console.log(i);
}
} catch (err) {
_didIteratorError = true;
_iteratorError = err;
} finally {
try {
if (!_iteratorNormalCompletion && _iterator.return) {
_iterator.return();
}
} finally {
if (_didIteratorError) {
throw _iteratorError;
}
}
}
Output
rozłożenie reszty obiektu
Pakiety wymagane dla wtyczek w babel6 i 7 są następujące -
Babel 6
npm install --save-dev babel-plugin-transform-object-rest-spread
Babel 7
npm install --save-dev @babel/plugin-proposal-object-rest-spread
.babelrc for babel6
{
"plugins": ["transform-object-rest-spread"]
}
.babelrc for babel7
{
"plugins": ["@babel/plugin-proposal-object-rest-spread"]
}
o.js
let { x1, y1, ...z1 } = { x1: 11, y1: 12, a: 23, b: 24 };
console.log(x1);
console.log(y1);
console.log(z1);
let n = { x1, y1, ...z1};
console.log(n);
Komenda
npx babel o.js --out-file o_es5.js
o_es5.js
var _extends = Object.assign || function (target) {
for (var i = 1; i < arguments.length; i++) {
var source = arguments[i]; for (var key in source) {
if (Object.prototype.hasOwnProperty.call(source, key)) {
target[key] = source[key];
}
}
}
return target;
};
function _objectWithoutProperties(obj, keys) {
var target = {};
for (var i in obj) {
if (keys.indexOf(i) >= 0) continue;
if (!Object.prototype.hasOwnProperty.call(obj, i)) continue;
target[i] = obj[i];
}
return target;
}
let _x1$y1$a$b = { x1: 11, y1: 12, a: 23, b: 24 },
{ x1, y1 } = _x1$y1$a$b,
z1 = _objectWithoutProperties(_x1$y1$a$b, ["x1", "y1"]);
console.log(x1);
console.log(y1);
console.log(z1);
let n = _extends({ x1, y1 }, z1);
console.log(n);
Output
async / await
Potrzebujemy następujących pakietów do zainstalowania dla babel 6 -
npm install --save-dev babel-plugin-transform-async-to-generator
Pakiety do Babel 7
npm install --save-dev @babel/plugin-transform-async-to-generator
.babelrc for babel 6
{
"plugins": ["transform-async-to-generator"]
}
.babelrc for babel 7
{
"plugins": ["@babel/plugin-transform-async-to-generator"]
}
async.js
let timer = () => {
return new Promise(resolve => {
setTimeout(() => {
resolve("Promise resolved after 5 seconds");
}, 5000);
});
};
let out = async () => {
let msg = await timer();
console.log(msg);
console.log("hello after await");
};
out();
Komenda
npx babel async.js --out-file async_es5.js
async_es5.js
function _asyncToGenerator(fn) {
return function () {
var gen = fn.apply(this, arguments);
return new Promise(function (resolve, reject) {
function step(key, arg) {
try {
var info = gen[key](arg);
var value = info.value;
} catch (error) {
reject(error);
return;
} if (info.done) {
resolve(value);
} else {
return Promise.resolve(value).then(function (value) {
step("next", value);
},
function (err) {
step("throw", err); });
}
} return step("next");
});
};
}
let timer = () => {
return new Promise(resolve => {
setTimeout(() => {
resolve("Promise resolved after 5 seconds");
}, 5000);
});
};
let out = (() => {
var _ref = _asyncToGenerator(function* () {
let msg = yield timer();
console.log(msg);
console.log("hello after await");
});
return function out() {
return _ref.apply(this, arguments);
};
})();
out();
Musimy użyć polyfill, ponieważ nie będzie działać w przeglądarkach, w których obietnice nie są obsługiwane.
Output