summaryrefslogtreecommitdiffstats
path: root/node_modules/sass-loader/lib/importsToResolve.js
blob: 595fc4aa0db2eaeb2163bf9823d6fb625d198a01 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
"use strict";

const path = require("path");

// libsass uses this precedence when importing files without extension
const extPrecedence = [".scss", ".sass", ".css"];

/**
 * When libsass tries to resolve an import, it uses a special algorithm.
 * Since the sass-loader uses webpack to resolve the modules, we need to simulate that algorithm. This function
 * returns an array of import paths to try.
 *
 * @param {string} request
 * @returns {Array<string>}
 */
function importsToResolve(request) {
    // libsass' import algorithm works like this:
    // In case there is no file extension...
    //   - Prefer modules starting with '_'.
    //   - File extension precedence: .scss, .sass, .css.
    // In case there is a file extension...
    //   - If the file is a CSS-file, do not include it all, but just link it via @import url().
    //   - The exact file name must match (no auto-resolving of '_'-modules).

    // Keep in mind: ext can also be something like '.datepicker' when the true extension is omitted and the filename contains a dot.
    // @see https://github.com/webpack-contrib/sass-loader/issues/167
    const ext = path.extname(request);
    const basename = path.basename(request);
    const dirname = path.dirname(request);
    const startsWithUnderscore = basename.charAt(0) === "_";
    const hasCssExt = ext === ".css";
    const hasSassExt = ext === ".scss" || ext === ".sass";

    // a module import is an identifier like 'bootstrap-sass'
    // We also need to check for dirname since it might also be a deep import like 'bootstrap-sass/something'
    let isModuleImport = request.charAt(0) !== "." && dirname === ".";

    if (dirname.charAt(0) === "@") {
        // Check whether it is a deep import from scoped npm package
        // (i.e. @pkg/foo/file), if so, process import as file import;
        // otherwise, if we import from root npm scoped package (i.e. @pkg/foo)
        // process import as a module import.
        isModuleImport = !(dirname.indexOf("/") > -1);
    }

    return (isModuleImport && [request]) || // Do not modify module imports
        (hasCssExt && []) || // Do not import css files
        (hasSassExt && [request]) || // Do not modify imports with explicit extensions
        (startsWithUnderscore ? [] : extPrecedence) // Do not add underscore imports if there is already an underscore
            .map(ext => "_" + basename + ext)
            .concat(
                extPrecedence.map(ext => basename + ext)
            ).map(
                file => dirname + "/" + file // No path.sep required here, because imports inside SASS are usually with /
            );
}

module.exports = importsToResolve;