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