Writing is not easy, without the permission of the author forbid to reprint in any form! If you think the article is good, welcome to follow, like and share!

Webpack modular

  • Webpack packages code that allows us to use a variety of modularity, but the most common ones are CommonJS, ES Module.

  • Including the following principles:

    • CommonJS modular implementation principle;
    • ES Module implementation principle;
    • CommonJS load ES Module principle;
    • ES Module load CommonJS principle;

CommonJS:

Before packaging

const { dateFormat, priceFormat } = require('./js/format');

console.log(dateFormat("abc"));
console.log(priceFormat("abc"));
Copy the code

After packaging

// Define an object
// Module path (key): function (value)
var __webpack_modules__ = {
  "./src/js/format.js":
    (function (module) {
      const dateFormat = (date) = > {
        return "2020-12-12";
      }
      const priceFormat = (price) = > {
        return "100.00";
      }

      // Put the exported variable into the Exports object in the Module object
      module.exports = {
        dateFormat,
        priceFormat
      }
    })
}

// Define an object to be used as a cache for loading modules
var __webpack_module_cache__ = {};

// is a function that is used when a module is loaded
function __webpack_require__(moduleId) {
  // 1. Check whether the cache has been loaded
  if (__webpack_module_cache__[moduleId]) {
    return __webpack_module_cache__[moduleId].exports;
  }

  // 2. Assign the same object to the module variable and __webpack_module_cache__[moduleId]
  var module = __webpack_module_cache__[moduleId] = { exports: {}};// 3. Load the execution module
  __webpack_modules__[moduleId](module.module.exports, __webpack_require__);

  Module. exports {dateFormat: function, priceForamt: function}
  return module.exports;
}


// Start executing the code logic
!function () {
  // load./ SRC /js/ form.js
  const { dateFormat, priceFormat } = __webpack_require__("./src/js/format.js");
  console.log(dateFormat("abc"));
  console.log(priceFormat("abc")); } ();Copy the code

ES Module

Before packaging

import { sum, mul } from "./js/math";

console.log(mul(20.30));
console.log(sum(20.30));
Copy the code

After packaging

// 1. Define an object that contains our module mapping
var __webpack_modules__ = {
  "./src/es_index.js":
    (function (__unused_webpack_module, __webpack_exports__, __webpack_require__) {
      // The purpose of calling r is to record an __esModule -> true
      __webpack_require__.r(__webpack_exports__);

      // _js_math__WEBPACK_IMPORTED_MODULE_0__ == exports
      var _js_math__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__("./src/js/math.js");

      console.log(_js_math__WEBPACK_IMPORTED_MODULE_0__.mul(20.30));
      console.log(_js_math__WEBPACK_IMPORTED_MODULE_0__.sum(20.30));
    }),
  "./src/js/math.js":
    (function (__unused_webpack_module, __webpack_exports__, __webpack_require__) {
      __webpack_require__.r(__webpack_exports__);

      // Calls the d function: sets a proxy definition for exports
      // Exports objects have no corresponding functions themselves
      __webpack_require__.d(__webpack_exports__, {
        "sum": function () { return sum; },
        "mul": function () { returnmul; }});const sum = (num1, num2) = > {
        return num1 + num2;
      }
      const mul = (num1, num2) = > {
        returnnum1 * num2; }})};// 2. Module cache
var __webpack_module_cache__ = {};

// 3. Implement the require function (load module)
function __webpack_require__(moduleId) {
  if (__webpack_module_cache__[moduleId]) {
    return __webpack_module_cache__[moduleId].exports;
  }
  var module = __webpack_module_cache__[moduleId] = {
    exports: {}}; __webpack_modules__[moduleId](module.module.exports, __webpack_require__);
  return module.exports; }!function () {
  // __webpack_require__ Adds a property: d -> value function
  __webpack_require__.d = function (exports, definition) {
    for (var key in definition) {
      if(__webpack_require__.o(definition, key) && ! __webpack_require__.o(exports, key)) {
        Object.defineProperty(exports, key, { enumerable: true.get: definition[key] }); }}}; } (); !function () {
  // the __webpack_require__ function object adds a property: o -> function
  __webpack_require__.o = function (obj, prop) { return Object.prototype.hasOwnProperty.call(obj, prop); }
}();

!function () {
  // the __webpack_require__ function object adds a property: r -> value function
  __webpack_require__.r = function (exports) {
    if (typeof Symbol! = ='undefined' && Symbol.toStringTag) {
      Object.defineProperty(exports.Symbol.toStringTag, { value: 'Module' });
    }
    Object.defineProperty(exports.'__esModule', { value: true}); }; } (); __webpack_require__("./src/es_index.js");
Copy the code

ES Module and CommonJS mix import and export

Before packaging

// Es module exports content, CommonJS imports content
const { sum, mul } = require("./js/math");

// CommonJS exports content, es Module imports content
import { dateFormat, priceFormat } from "./js/format";

console.log(sum(20.30));
console.log(mul(20.30));

console.log(dateFormat("aaa"));
console.log(priceFormat("bbb"));

console.log(abc);
Copy the code

After packaging

var __webpack_modules__ = ({
  "./src/index.js":
    (function (__unused_webpack_module, __webpack_exports__, __webpack_require__) {
      "use strict";
      __webpack_require__.r(__webpack_exports__);
      var _js_format__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__("./src/js/format.js");
      var _js_format__WEBPACK_IMPORTED_MODULE_0___default = __webpack_require__.n(_js_format__WEBPACK_IMPORTED_MODULE_0__);
      
      // Es module exports content, CommonJS imports content
      const math = __webpack_require__("./src/js/math.js");

      // CommonJS exports content, es Module imports content
      console.log(math.sum(20.30));
      console.log(math.mul(20.30));
      console.log(_js_format__WEBPACK_IMPORTED_MODULE_0___default().dateFormat("aaa"));
      console.log(_js_format__WEBPACK_IMPORTED_MODULE_0___default().priceFormat("bbb"));
    }),
  "./src/js/format.js":
    (function (module) {
      const dateFormat = (date) = > {
        return "2020-12-12";
      }
      const priceFormat = (price) = > {
        return "100.00";
      }
      module.exports = {
        dateFormat,
        priceFormat
      }
    }),

  "./src/js/math.js":
    (function (__unused_webpack_module, __webpack_exports__, __webpack_require__) {
      
      __webpack_require__.r(__webpack_exports__);
      __webpack_require__.d(__webpack_exports__, {
        "sum": function () { return sum; },
        "mul": function () { returnmul; }});const sum = (num1, num2) = > {
        return num1 + num2;
      }

      const mul = (num1, num2) = > {
        returnnum1 * num2; }})});var __webpack_module_cache__ = {};

// The require function
function __webpack_require__(moduleId) {
  // Check if module is in cache
  if (__webpack_module_cache__[moduleId]) {
    return __webpack_module_cache__[moduleId].exports;
  }
  // Create a new module (and put it into the cache)
  var module = __webpack_module_cache__[moduleId] = {
    // no module.id needed
    // no module.loaded needed
    exports: {}};// Execute the module function
  __webpack_modules__[moduleId](module.module.exports, __webpack_require__);

  // Return the exports of the module
  return module.exports; }!function () {
  // getDefaultExport function for compatibility with non-harmony modules
  __webpack_require__.n = function (module) {
    var getter = module && module.__esModule ?
      function () { return module['default']; } :
      function () { return module; };
    __webpack_require__.d(getter, { a: getter });
    returngetter; }; } ();/* webpack/runtime/define property getters */
!function () {
  // define getter functions for harmony exports
  __webpack_require__.d = function (exports, definition) {
    for (var key in definition) {
      if(__webpack_require__.o(definition, key) && ! __webpack_require__.o(exports, key)) {
        Object.defineProperty(exports, key, { enumerable: true.get: definition[key] }); }}}; } ();/* webpack/runtime/hasOwnProperty shorthand */
!function () {
  __webpack_require__.o = function (obj, prop) { return Object.prototype.hasOwnProperty.call(obj, prop); }
}();

/* webpack/runtime/make namespace object */
!function () {
  // define __esModule on exports
  __webpack_require__.r = function (exports) {
    if (typeof Symbol! = ='undefined' && Symbol.toStringTag) {
      Object.defineProperty(exports.Symbol.toStringTag, { value: 'Module' });
    }
    Object.defineProperty(exports.'__esModule', { value: true}); }; } (); __webpack_require__("./src/index.js");
Copy the code

conclusion

The principle is similar

  1. Modularity code is stored through a Webpack_module object
    • Key indicates the file name
    • Value is the file code
  2. Modularization code is cached via webpack_module_cache
  3. Use webpack_require to read and import code from webpack_module_cache or webpack_module

Understanding the Source – the Map

Reference documentation

  • Our code is typically packaged when running in a browser:

    • The actual code running on the browser is different from the code we write;
    • For example, ES6 code might be converted to ES5;
    • For example, the corresponding code line number and column number will be inconsistent after compilation.
    • For example, when the code is uglified, the code name will be modified;
    • For example, we write code in TypeScript and so on, which is converted to JavaScript.
  • SourceMap enables debugging of such converted inconsistent code

    • Source-map is a mapping from converted code to the original source file;
    • Enable the browser to refactor the source and display the reconstructed source in the debugger;
    • Easier to locate source file errors

How do I use SourceMap

  • Step 1: Generate source-map file according to the source file. When webpack is packaged, it can be configured to generate source-map.

    devtool:'source-map'

  • Step 2: At the end of the transformed code, add a comment pointing to sourcemap;

    //# sourceMappingURL=common.bundle.js.map

  • The browser will find the corresponding source-map according to our annotations, and restore our code according to the source-map, which is convenient for debugging.

  • In Chrome, we can open source-map as follows:

    • Tick the Enable JavaScript source maps
    • Tick the Enable CSS source maps
  • The original source-Map generated file was 10 times the size of the original file, reduced by about 50% in the second version, and reduced by another 50% in the third version, so now a 133KB file, the final source-Map size is about 300KB.

  • What does the current source-Map look like?

    • Version: The version currently in use, that is, the latest third version;

    • Sources: which files are converted from source-Map and packaged code (the most original file);

    • Names: the names of variables and attributes before the transformation (you do not need to keep the names before the transformation because you are currently using development mode);

    • Description :source-map Specifies information, such as location information, used to map the source file, a set of Base64 VLQ(Unavailable) encoding;

    • File: packaged file (browser loaded file);

    • SourceContent: The specific code information before the conversion (as opposed to sources);

    • SourceRoot: The root directory relative to all sources;

Generate SourceMap

How to generate the corresponding source-map when using WebPack packaging?

  • Webpack gives us a huge number of options (currently 26) to work with source-Map;

  • Reference: webpack.docschina.org/configurati…

  • Select different values, the generated source-map will be slightly different, the packaging process will also have different performance, you can choose according to different circumstances;

The following values do not generate a source-map

  • False: do not use source-map, that is, there is no source-map content.

  • None: Default value in production mode (no value is written), source-map is not generated.

  • Eval: the default value in development mode. Source-map is not generated

    • But it will add //# sourceURL=;
    • It will be parsed by the browser at execution time, and generate some corresponding file directories in the debug panel, so that we can debug the code;

Use source-map values

  • source-map

    • Generate a separate source-map file with a comment pointing to the source-map file in the packaged JS file;
      • //# sourceMappingURL=bundle.js.map
  • eval-source-map

    • Source-map is added to the eval function as a DataUrl
  • inline-source-map

    • Source-map is added to the end of the packaged JS file as a DataUrl (comment)
  • cheap-source-map:

    • A source-map is generated, but it is cheaper because it does not generate Column Mapping.
    • Because in development, we can usually locate errors with only line information
  • cheap-module-source-map:

    • Source-map is generated, similar to cheap-source-map, but better for source-map from loader.
    • Source-map processing from loader is better than source-map processing. If loader does something special to our source code, such as Babel, it will revert to the source code module format. For example, custom blank lines will not be removed.
  • hidden-source-map:

    • Sourcemap is generated, but the source-map file is not referenced;
    • This is equivalent to removing the sourcemap reference comment from the package file
    • # sourceMappingURL=bundle.js.map
    • If added manually, source-map will take effect
  • nosources-source-map:

    • Sourcemap is generated, but the generated Sourcemap only has error messages, not source files;
    • You can’t view the source code by clicking the error message

A combination of multiple values

  • In fact, webPack provides us with 26 values that can be combined.

  • The rules of combination are as follows:

  • The inline – | hidden – | eval: choose three values;

  • Nosources: Optional value.

  • Cheap is optional and can follow the value of module;

So what are the best practices in development?

  • Development phase: Source-map or cheap-module-source-map is recommended

    • This value is used by vue and React respectively to obtain debugging information for rapid development.
  • Testing phase: Source-map or cheap-module-source-map is recommended

    • During the test phase we also want to see the correct error message in the browser;
  • Release phase :false, default (no write)


  • Nuggets: Front-end LeBron

  • Zhihu: Front-end LeBron

  • Continue to share technical blog posts, follow the wechat public account 👉🏻 front-end LeBron