412 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			412 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
"use strict";
 | 
						|
 | 
						|
const path = require("path");
 | 
						|
const schema = require("./loader-options.json");
 | 
						|
const {
 | 
						|
  ABSOLUTE_PUBLIC_PATH,
 | 
						|
  AUTO_PUBLIC_PATH,
 | 
						|
  BASE_URI,
 | 
						|
  SINGLE_DOT_PATH_SEGMENT,
 | 
						|
  evalModuleCode,
 | 
						|
  findModuleById,
 | 
						|
  stringifyLocal,
 | 
						|
  stringifyRequest
 | 
						|
} = require("./utils");
 | 
						|
const MiniCssExtractPlugin = require("./index");
 | 
						|
 | 
						|
/** @typedef {import("schema-utils/declarations/validate").Schema} Schema */
 | 
						|
/** @typedef {import("webpack").Compiler} Compiler */
 | 
						|
/** @typedef {import("webpack").Compilation} Compilation */
 | 
						|
/** @typedef {import("webpack").Chunk} Chunk */
 | 
						|
/** @typedef {import("webpack").Module} Module */
 | 
						|
/** @typedef {import("webpack").sources.Source} Source */
 | 
						|
/** @typedef {import("webpack").AssetInfo} AssetInfo */
 | 
						|
/** @typedef {import("webpack").NormalModule} NormalModule */
 | 
						|
/** @typedef {import("./index.js").LoaderOptions} LoaderOptions */
 | 
						|
 | 
						|
// eslint-disable-next-line jsdoc/no-restricted-syntax
 | 
						|
/** @typedef {{[key: string]: string | Function }} Locals */
 | 
						|
 | 
						|
// eslint-disable-next-line jsdoc/no-restricted-syntax
 | 
						|
/** @typedef {any} EXPECTED_ANY */
 | 
						|
 | 
						|
/**
 | 
						|
 * @typedef {object} Dependency
 | 
						|
 * @property {string} identifier identifier
 | 
						|
 * @property {string | null} context context
 | 
						|
 * @property {Buffer} content content
 | 
						|
 * @property {string=} media media
 | 
						|
 * @property {string=} supports supports
 | 
						|
 * @property {string=} layer layer
 | 
						|
 * @property {Buffer=} sourceMap source map
 | 
						|
 */
 | 
						|
 | 
						|
/**
 | 
						|
 * @param {string} code code
 | 
						|
 * @param {{ loaderContext: import("webpack").LoaderContext<LoaderOptions>, options: LoaderOptions, locals: Locals | undefined }} context context
 | 
						|
 * @returns {string} code and HMR code
 | 
						|
 */
 | 
						|
function hotLoader(code, context) {
 | 
						|
  const localsJsonString = JSON.stringify(JSON.stringify(context.locals));
 | 
						|
  return `${code}
 | 
						|
    if(module.hot) {
 | 
						|
      (function() {
 | 
						|
        var localsJsonString = ${localsJsonString};
 | 
						|
        // ${Date.now()}
 | 
						|
        var cssReload = require(${stringifyRequest(context.loaderContext, path.join(__dirname, "hmr/hotModuleReplacement.js"))})(module.id, ${JSON.stringify(context.options)});
 | 
						|
        // only invalidate when locals change
 | 
						|
        if (
 | 
						|
          module.hot.data &&
 | 
						|
          module.hot.data.value &&
 | 
						|
          module.hot.data.value !== localsJsonString
 | 
						|
        ) {
 | 
						|
          module.hot.invalidate();
 | 
						|
        } else {
 | 
						|
          module.hot.accept();
 | 
						|
        }
 | 
						|
        module.hot.dispose(function(data) {
 | 
						|
          data.value = localsJsonString;
 | 
						|
          cssReload();
 | 
						|
        });
 | 
						|
      })();
 | 
						|
    }
 | 
						|
  `;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * @this {import("webpack").LoaderContext<LoaderOptions>}
 | 
						|
 * @param {string} request request
 | 
						|
 */
 | 
						|
function pitch(request) {
 | 
						|
  if (this._compiler && this._compiler.options && this._compiler.options.experiments && this._compiler.options.experiments.css && this._module && (this._module.type === "css" || this._module.type === "css/auto" || this._module.type === "css/global" || this._module.type === "css/module")) {
 | 
						|
    this.emitWarning(new Error('You can\'t use `experiments.css` (`experiments.futureDefaults` enable built-in CSS support by default) and `mini-css-extract-plugin` together, please set `experiments.css` to `false` or set `{ type: "javascript/auto" }` for rules with `mini-css-extract-plugin` in your webpack config (now `mini-css-extract-plugin` does nothing).'));
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  const options = this.getOptions(/** @type {Schema} */schema);
 | 
						|
  const emit = typeof options.emit !== "undefined" ? options.emit : true;
 | 
						|
  const callback = this.async();
 | 
						|
  const optionsFromPlugin =
 | 
						|
  // @ts-expect-error
 | 
						|
  this[MiniCssExtractPlugin.pluginSymbol];
 | 
						|
  if (!optionsFromPlugin) {
 | 
						|
    callback(new Error("You forgot to add 'mini-css-extract-plugin' plugin (i.e. `{ plugins: [new MiniCssExtractPlugin()] }`), please read https://github.com/webpack-contrib/mini-css-extract-plugin#getting-started"));
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  const {
 | 
						|
    webpack
 | 
						|
  } = /** @type {Compiler} */this._compiler;
 | 
						|
 | 
						|
  /**
 | 
						|
   * @param {EXPECTED_ANY} originalExports original exports
 | 
						|
   * @param {Compilation=} compilation compilation
 | 
						|
   * @param {{ [name: string]: Source }=} assets assets
 | 
						|
   * @param {Map<string, AssetInfo>=} assetsInfo assets info
 | 
						|
   * @returns {void}
 | 
						|
   */
 | 
						|
  const handleExports = (originalExports, compilation, assets, assetsInfo) => {
 | 
						|
    /** @type {Locals | undefined} */
 | 
						|
    let locals;
 | 
						|
    let namedExport;
 | 
						|
    const esModule = typeof options.esModule !== "undefined" ? options.esModule : true;
 | 
						|
 | 
						|
    /**
 | 
						|
     * @param {Dependency[] | [null, object][]} dependencies dependencies
 | 
						|
     */
 | 
						|
    const addDependencies = dependencies => {
 | 
						|
      // eslint-disable-next-line no-eq-null, eqeqeq
 | 
						|
      if (!Array.isArray(dependencies) && dependencies != null) {
 | 
						|
        throw new Error(`Exported value was not extracted as an array: ${JSON.stringify(dependencies)}`);
 | 
						|
      }
 | 
						|
      const identifierCountMap = new Map();
 | 
						|
      let lastDep;
 | 
						|
      for (const dependency of dependencies) {
 | 
						|
        if (!(/** @type {Dependency} */dependency.identifier) || !emit) {
 | 
						|
          continue;
 | 
						|
        }
 | 
						|
        const count = identifierCountMap.get(/** @type {Dependency} */dependency.identifier) || 0;
 | 
						|
        const CssDependency = MiniCssExtractPlugin.getCssDependency(webpack);
 | 
						|
 | 
						|
        /** @type {NormalModule} */
 | 
						|
        this._module.addDependency(lastDep = new CssDependency(/** @type {Dependency} */
 | 
						|
        dependency, /** @type {Dependency} */
 | 
						|
        dependency.context, count));
 | 
						|
        identifierCountMap.set(/** @type {Dependency} */
 | 
						|
        dependency.identifier, count + 1);
 | 
						|
      }
 | 
						|
      if (lastDep && assets) {
 | 
						|
        lastDep.assets = assets;
 | 
						|
        lastDep.assetsInfo = assetsInfo;
 | 
						|
      }
 | 
						|
    };
 | 
						|
    try {
 | 
						|
      const exports = originalExports.__esModule ? originalExports.default : originalExports;
 | 
						|
      namedExport = originalExports.__esModule && (!originalExports.default || !("locals" in originalExports.default));
 | 
						|
      if (namedExport) {
 | 
						|
        for (const key of Object.keys(originalExports)) {
 | 
						|
          if (key !== "default") {
 | 
						|
            if (!locals) {
 | 
						|
              locals = {};
 | 
						|
            }
 | 
						|
 | 
						|
            /** @type {Locals} */
 | 
						|
            locals[key] = originalExports[key];
 | 
						|
          }
 | 
						|
        }
 | 
						|
      } else {
 | 
						|
        locals = exports && exports.locals;
 | 
						|
      }
 | 
						|
 | 
						|
      /** @type {Dependency[] | [null, Record<string, string>][]} */
 | 
						|
      let dependencies;
 | 
						|
      if (!Array.isArray(exports)) {
 | 
						|
        dependencies = [[null, exports]];
 | 
						|
      } else {
 | 
						|
        dependencies = exports.map(([id, content, media, sourceMap, supports, layer]) => {
 | 
						|
          let identifier = id;
 | 
						|
          let context;
 | 
						|
          if (compilation) {
 | 
						|
            const module = /** @type {Module} */
 | 
						|
            findModuleById(compilation, id);
 | 
						|
            identifier = module.identifier();
 | 
						|
            ({
 | 
						|
              context
 | 
						|
            } = module);
 | 
						|
          } else {
 | 
						|
            // TODO check if this context is used somewhere
 | 
						|
            context = this.rootContext;
 | 
						|
          }
 | 
						|
          return {
 | 
						|
            identifier,
 | 
						|
            context,
 | 
						|
            content: Buffer.from(content),
 | 
						|
            media,
 | 
						|
            supports,
 | 
						|
            layer,
 | 
						|
            sourceMap: sourceMap ? Buffer.from(JSON.stringify(sourceMap)) : undefined
 | 
						|
          };
 | 
						|
        });
 | 
						|
      }
 | 
						|
      addDependencies(dependencies);
 | 
						|
    } catch (err) {
 | 
						|
      callback(/** @type {Error} */err);
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    const result = function makeResult() {
 | 
						|
      const defaultExport = typeof options.defaultExport !== "undefined" ? options.defaultExport : false;
 | 
						|
      if (locals) {
 | 
						|
        if (namedExport) {
 | 
						|
          const identifiers = [...function* generateIdentifiers() {
 | 
						|
            let identifierId = 0;
 | 
						|
            for (const key of Object.keys(locals)) {
 | 
						|
              identifierId += 1;
 | 
						|
              yield [`_${identifierId.toString(16)}`, key];
 | 
						|
            }
 | 
						|
          }()];
 | 
						|
          const localsString = identifiers.map(([id, key]) => `\nvar ${id} = ${stringifyLocal(/** @type {Locals} */locals[key])};`).join("");
 | 
						|
          const exportsString = `export { ${identifiers.map(([id, key]) => `${id} as ${JSON.stringify(key)}`).join(", ")} }`;
 | 
						|
          return defaultExport ? `${localsString}\n${exportsString}\nexport default { ${identifiers.map(([id, key]) => `${JSON.stringify(key)}: ${id}`).join(", ")} }\n` : `${localsString}\n${exportsString}\n`;
 | 
						|
        }
 | 
						|
        return `\n${esModule ? "export default" : "module.exports = "} ${JSON.stringify(locals)};`;
 | 
						|
      } else if (esModule) {
 | 
						|
        return defaultExport ? "\nexport {};export default {};" : "\nexport {};";
 | 
						|
      }
 | 
						|
      return "";
 | 
						|
    }();
 | 
						|
    let resultSource = `// extracted by ${MiniCssExtractPlugin.pluginName}`;
 | 
						|
 | 
						|
    // only attempt hotreloading if the css is actually used for something other than hash values
 | 
						|
    resultSource += this.hot && emit ? hotLoader(result, {
 | 
						|
      loaderContext: this,
 | 
						|
      options,
 | 
						|
      locals
 | 
						|
    }) : result;
 | 
						|
    callback(null, resultSource);
 | 
						|
  };
 | 
						|
  let {
 | 
						|
    publicPath
 | 
						|
  } = /** @type {Compilation} */
 | 
						|
  this._compilation.outputOptions;
 | 
						|
  if (typeof options.publicPath === "string") {
 | 
						|
    publicPath = options.publicPath;
 | 
						|
  } else if (typeof options.publicPath === "function") {
 | 
						|
    publicPath = options.publicPath(this.resourcePath, this.rootContext);
 | 
						|
  }
 | 
						|
  if (publicPath === "auto") {
 | 
						|
    publicPath = AUTO_PUBLIC_PATH;
 | 
						|
  }
 | 
						|
  if (typeof optionsFromPlugin.experimentalUseImportModule === "undefined" && typeof this.importModule === "function" || optionsFromPlugin.experimentalUseImportModule) {
 | 
						|
    if (!this.importModule) {
 | 
						|
      callback(new Error("You are using 'experimentalUseImportModule' but 'this.importModule' is not available in loader context. You need to have at least webpack 5.33.2."));
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    let publicPathForExtract;
 | 
						|
    if (typeof publicPath === "string") {
 | 
						|
      const isAbsolutePublicPath = /^[a-zA-Z][a-zA-Z\d+\-.]*?:/.test(publicPath);
 | 
						|
      publicPathForExtract = isAbsolutePublicPath ? publicPath : `${ABSOLUTE_PUBLIC_PATH}${publicPath.replace(/\./g, SINGLE_DOT_PATH_SEGMENT)}`;
 | 
						|
    } else {
 | 
						|
      publicPathForExtract = publicPath;
 | 
						|
    }
 | 
						|
    this.importModule(`${this.resourcePath}.webpack[javascript/auto]!=!!!${request}`, {
 | 
						|
      layer: options.layer,
 | 
						|
      publicPath: (/** @type {string} */publicPathForExtract),
 | 
						|
      baseUri: `${BASE_URI}/`
 | 
						|
    },
 | 
						|
    /**
 | 
						|
     * @param {Error | null | undefined} error error
 | 
						|
     * @param {object} exports exports
 | 
						|
     */
 | 
						|
    (error, exports) => {
 | 
						|
      if (error) {
 | 
						|
        callback(error);
 | 
						|
        return;
 | 
						|
      }
 | 
						|
      handleExports(exports);
 | 
						|
    });
 | 
						|
    return;
 | 
						|
  }
 | 
						|
  const loaders = this.loaders.slice(this.loaderIndex + 1);
 | 
						|
  this.addDependency(this.resourcePath);
 | 
						|
  const childFilename = "*";
 | 
						|
  const outputOptions = {
 | 
						|
    filename: childFilename,
 | 
						|
    publicPath
 | 
						|
  };
 | 
						|
  const childCompiler = /** @type {Compilation} */
 | 
						|
  this._compilation.createChildCompiler(`${MiniCssExtractPlugin.pluginName} ${request}`, outputOptions);
 | 
						|
 | 
						|
  // The templates are compiled and executed by NodeJS - similar to server side rendering
 | 
						|
  // Unfortunately this causes issues as some loaders require an absolute URL to support ES Modules
 | 
						|
  // The following config enables relative URL support for the child compiler
 | 
						|
  childCompiler.options.module = {
 | 
						|
    ...childCompiler.options.module
 | 
						|
  };
 | 
						|
  childCompiler.options.module.parser = {
 | 
						|
    ...childCompiler.options.module.parser
 | 
						|
  };
 | 
						|
  childCompiler.options.module.parser.javascript = {
 | 
						|
    ...childCompiler.options.module.parser.javascript,
 | 
						|
    url: "relative"
 | 
						|
  };
 | 
						|
  const {
 | 
						|
    NodeTemplatePlugin
 | 
						|
  } = webpack.node;
 | 
						|
  const {
 | 
						|
    NodeTargetPlugin
 | 
						|
  } = webpack.node;
 | 
						|
  new NodeTemplatePlugin().apply(childCompiler);
 | 
						|
  new NodeTargetPlugin().apply(childCompiler);
 | 
						|
  const {
 | 
						|
    EntryOptionPlugin
 | 
						|
  } = webpack;
 | 
						|
  const {
 | 
						|
    library: {
 | 
						|
      EnableLibraryPlugin
 | 
						|
    }
 | 
						|
  } = webpack;
 | 
						|
  new EnableLibraryPlugin("commonjs2").apply(childCompiler);
 | 
						|
  EntryOptionPlugin.applyEntryOption(childCompiler, this.context, {
 | 
						|
    child: {
 | 
						|
      library: {
 | 
						|
        type: "commonjs2"
 | 
						|
      },
 | 
						|
      import: [`!!${request}`]
 | 
						|
    }
 | 
						|
  });
 | 
						|
  const {
 | 
						|
    LimitChunkCountPlugin
 | 
						|
  } = webpack.optimize;
 | 
						|
  new LimitChunkCountPlugin({
 | 
						|
    maxChunks: 1
 | 
						|
  }).apply(childCompiler);
 | 
						|
  const {
 | 
						|
    NormalModule
 | 
						|
  } = webpack;
 | 
						|
  childCompiler.hooks.thisCompilation.tap(`${MiniCssExtractPlugin.pluginName} loader`,
 | 
						|
  /**
 | 
						|
   * @param {Compilation} compilation compilation
 | 
						|
   */
 | 
						|
  compilation => {
 | 
						|
    const normalModuleHook = NormalModule.getCompilationHooks(compilation).loader;
 | 
						|
    normalModuleHook.tap(`${MiniCssExtractPlugin.pluginName} loader`, (loaderContext, module) => {
 | 
						|
      if (module.request === request) {
 | 
						|
        module.loaders = loaders.map(loader => ({
 | 
						|
          type: null,
 | 
						|
          loader: loader.path,
 | 
						|
          options: loader.options,
 | 
						|
          ident: loader.ident
 | 
						|
        }));
 | 
						|
      }
 | 
						|
    });
 | 
						|
  });
 | 
						|
 | 
						|
  /** @type {string | Buffer} */
 | 
						|
  let source;
 | 
						|
  childCompiler.hooks.compilation.tap(MiniCssExtractPlugin.pluginName,
 | 
						|
  /**
 | 
						|
   * @param {Compilation} compilation compilation
 | 
						|
   */
 | 
						|
  compilation => {
 | 
						|
    compilation.hooks.processAssets.tap(MiniCssExtractPlugin.pluginName, () => {
 | 
						|
      source = compilation.assets[childFilename] && compilation.assets[childFilename].source();
 | 
						|
 | 
						|
      // Remove all chunk assets
 | 
						|
      for (const chunk of compilation.chunks) {
 | 
						|
        for (const file of chunk.files) {
 | 
						|
          compilation.deleteAsset(file);
 | 
						|
        }
 | 
						|
      }
 | 
						|
    });
 | 
						|
  });
 | 
						|
  childCompiler.runAsChild((error, entries, compilation_) => {
 | 
						|
    if (error) {
 | 
						|
      callback(error);
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    const compilation = /** @type {Compilation} */compilation_;
 | 
						|
    if (compilation.errors.length > 0) {
 | 
						|
      callback(compilation.errors[0]);
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    /** @type {{ [name: string]: Source }} */
 | 
						|
    const assets = Object.create(null);
 | 
						|
    /** @type {Map<string, AssetInfo>} */
 | 
						|
    const assetsInfo = new Map();
 | 
						|
    for (const asset of compilation.getAssets()) {
 | 
						|
      assets[asset.name] = asset.source;
 | 
						|
      assetsInfo.set(asset.name, asset.info);
 | 
						|
    }
 | 
						|
    for (const dep of compilation.fileDependencies) {
 | 
						|
      this.addDependency(dep);
 | 
						|
    }
 | 
						|
    for (const dep of compilation.contextDependencies) {
 | 
						|
      this.addContextDependency(dep);
 | 
						|
    }
 | 
						|
    if (!source) {
 | 
						|
      callback(new Error("Didn't get a result from child compiler"));
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    let originalExports;
 | 
						|
    try {
 | 
						|
      originalExports = evalModuleCode(this, source, request);
 | 
						|
    } catch (err) {
 | 
						|
      callback(/** @type {Error} */err);
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    handleExports(originalExports, compilation, assets, assetsInfo);
 | 
						|
  });
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * @this {import("webpack").LoaderContext<LoaderOptions>}
 | 
						|
 * @param {string} content content
 | 
						|
 * @returns {string | undefined} the original content
 | 
						|
 */
 | 
						|
function loader(content) {
 | 
						|
  if (this._compiler && this._compiler.options && this._compiler.options.experiments && this._compiler.options.experiments.css && this._module && (this._module.type === "css" || this._module.type === "css/auto" || this._module.type === "css/global" || this._module.type === "css/module")) {
 | 
						|
    return content;
 | 
						|
  }
 | 
						|
}
 | 
						|
module.exports = loader;
 | 
						|
module.exports.hotLoader = hotLoader;
 | 
						|
module.exports.pitch = pitch; |