mirror of https://github.com/webpack/webpack.git
				
				
				
			
		
			
				
	
	
		
			104 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
			
		
		
	
	
			104 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
| /*
 | |
| 	MIT License http://www.opensource.org/licenses/mit-license.php
 | |
| 	Author Tobias Koppers @sokra
 | |
| */
 | |
| "use strict";
 | |
| 
 | |
| const validateOptions = require("schema-utils");
 | |
| const schema = require("../../schemas/plugins/optimize/OccurrenceOrderModuleIdsPlugin.json");
 | |
| 
 | |
| class OccurrenceOrderModuleIdsPlugin {
 | |
| 	constructor(options = {}) {
 | |
| 		validateOptions(schema, options, "Occurrence Order Module Ids Plugin");
 | |
| 		this.options = options;
 | |
| 	}
 | |
| 
 | |
| 	apply(compiler) {
 | |
| 		const prioritiseInitial = this.options.prioritiseInitial;
 | |
| 		compiler.hooks.compilation.tap(
 | |
| 			"OccurrenceOrderModuleIdsPlugin",
 | |
| 			compilation => {
 | |
| 				compilation.hooks.optimizeModuleOrder.tap(
 | |
| 					"OccurrenceOrderModuleIdsPlugin",
 | |
| 					modules => {
 | |
| 						const occursInInitialChunksMap = new Map();
 | |
| 						const occursInAllChunksMap = new Map();
 | |
| 
 | |
| 						const initialChunkChunkMap = new Map();
 | |
| 						const entryCountMap = new Map();
 | |
| 						for (const m of modules) {
 | |
| 							let initial = 0;
 | |
| 							let entry = 0;
 | |
| 							for (const c of m.chunksIterable) {
 | |
| 								if (c.canBeInitial()) initial++;
 | |
| 								if (c.entryModule === m) entry++;
 | |
| 							}
 | |
| 							initialChunkChunkMap.set(m, initial);
 | |
| 							entryCountMap.set(m, entry);
 | |
| 						}
 | |
| 
 | |
| 						const countOccursInEntry = (sum, r) => {
 | |
| 							if (!r.module) {
 | |
| 								return sum;
 | |
| 							}
 | |
| 							return sum + initialChunkChunkMap.get(r.module);
 | |
| 						};
 | |
| 						const countOccurs = (sum, r) => {
 | |
| 							if (!r.module) {
 | |
| 								return sum;
 | |
| 							}
 | |
| 							let factor = 1;
 | |
| 							if (typeof r.dependency.getNumberOfIdOccurrences === "function") {
 | |
| 								factor = r.dependency.getNumberOfIdOccurrences();
 | |
| 							}
 | |
| 							if (factor === 0) {
 | |
| 								return sum;
 | |
| 							}
 | |
| 							return sum + factor * r.module.getNumberOfChunks();
 | |
| 						};
 | |
| 
 | |
| 						if (prioritiseInitial) {
 | |
| 							for (const m of modules) {
 | |
| 								const result =
 | |
| 									m.reasons.reduce(countOccursInEntry, 0) +
 | |
| 									initialChunkChunkMap.get(m) +
 | |
| 									entryCountMap.get(m);
 | |
| 								occursInInitialChunksMap.set(m, result);
 | |
| 							}
 | |
| 						}
 | |
| 
 | |
| 						const originalOrder = new Map();
 | |
| 						let i = 0;
 | |
| 						for (const m of modules) {
 | |
| 							const result =
 | |
| 								m.reasons.reduce(countOccurs, 0) +
 | |
| 								m.getNumberOfChunks() +
 | |
| 								entryCountMap.get(m);
 | |
| 							occursInAllChunksMap.set(m, result);
 | |
| 							originalOrder.set(m, i++);
 | |
| 						}
 | |
| 
 | |
| 						modules.sort((a, b) => {
 | |
| 							if (prioritiseInitial) {
 | |
| 								const aEntryOccurs = occursInInitialChunksMap.get(a);
 | |
| 								const bEntryOccurs = occursInInitialChunksMap.get(b);
 | |
| 								if (aEntryOccurs > bEntryOccurs) return -1;
 | |
| 								if (aEntryOccurs < bEntryOccurs) return 1;
 | |
| 							}
 | |
| 							const aOccurs = occursInAllChunksMap.get(a);
 | |
| 							const bOccurs = occursInAllChunksMap.get(b);
 | |
| 							if (aOccurs > bOccurs) return -1;
 | |
| 							if (aOccurs < bOccurs) return 1;
 | |
| 							const orgA = originalOrder.get(a);
 | |
| 							const orgB = originalOrder.get(b);
 | |
| 							return orgA - orgB;
 | |
| 						});
 | |
| 					}
 | |
| 				);
 | |
| 			}
 | |
| 		);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| module.exports = OccurrenceOrderModuleIdsPlugin;
 |