-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathindex.js
139 lines (119 loc) · 4.63 KB
/
index.js
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
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
var semver = require("semver");
var sass = require("node-sass");
var DtsCreator = require("typed-css-modules");
var path = require("path");
var fs = require("fs");
var appRoot = require("app-root-path");
var css2rn = require("css-to-react-native-transform").default;
var creator = new DtsCreator();
var upstreamTransformer = null;
var reactNativeVersionString = require("react-native/package.json").version;
var reactNativeMinorVersion = semver(reactNativeVersionString).minor;
if (reactNativeMinorVersion >= 59) {
upstreamTransformer = require("metro-react-native-babel-transformer");
} else if (reactNativeMinorVersion >= 56) {
upstreamTransformer = require("metro/src/reactNativeTransformer");
} else if (reactNativeMinorVersion >= 52) {
upstreamTransformer = require("metro/src/transformer");
} else if (reactNativeMinorVersion >= 47) {
upstreamTransformer = require("metro-bundler/src/transformer");
} else if (reactNativeMinorVersion === 46) {
upstreamTransformer = require("metro-bundler/build/transformer");
} else {
// handle RN <= 0.45
var oldUpstreamTransformer = require("react-native/packager/transformer");
upstreamTransformer = {
transform({ src, filename, options }) {
return oldUpstreamTransformer.transform(src, filename, options);
}
};
}
function isPlatformSpecific(filename) {
var platformSpecific = [".native.", ".ios.", ".android."];
return platformSpecific.some(name => filename.includes(name));
}
// Iterate through the include paths and extensions to find the file variant
function findVariant(name, extensions, includePaths) {
for (let i = 0; i < includePaths.length; i++) {
const includePath = includePaths[i];
// try to find the file iterating through the extensions, in order.
const foundExtention = extensions.find(extension => {
const fname = includePath + "/" + name + extension;
const partialfname = includePath + "/_" + name + extension;
return fs.existsSync(fname) || fs.existsSync(partialfname);
});
if (foundExtention) {
return includePath + "/" + name + foundExtention;
}
}
return false;
}
function renderToCSS({ src, filename, options }) {
const ext = path.extname(filename);
const exts = [
// add the platform specific extension, first in the array to take precedence
options.platform === "android" ? ".android" + ext : ".ios" + ext,
".native" + ext,
ext
];
var defaultOpts = {
includePaths: [path.dirname(filename), appRoot],
indentedSyntax: filename.endsWith(".sass"),
importer: function(url /*, prev, done */) {
// url is the path in import as is, which LibSass encountered.
// prev is the previously resolved path.
// done is an optional callback, either consume it or return value synchronously.
// this.options contains this options hash, this.callback contains the node-style callback
const urlPath = path.parse(url);
const importerOptions = this.options;
const incPaths = importerOptions.includePaths.slice(0).split(":");
if (urlPath.dir.length > 0) {
incPaths.unshift(path.resolve(path.dirname(filename), urlPath.dir)); // add the file's dir to the search array
}
const f = findVariant(urlPath.name, exts, incPaths);
if (f) {
return { file: f };
}
}
};
var opts = options.sassOptions
? Object.assign(defaultOpts, options.sassOptions, { data: src })
: Object.assign(defaultOpts, { data: src });
var result = sass.renderSync(opts);
var css = result.css.toString();
return css;
}
function renderToCSSPromise({ src, filename, options }) {
return Promise.resolve(renderToCSS({ src, filename, options }));
}
function renderCSSToReactNative(css) {
return css2rn(css, { parseMediaQueries: true });
}
module.exports.transform = function(src, filename, options) {
if (typeof src === "object") {
// handle RN >= 0.46
({ src, filename, options } = src);
}
if (filename.endsWith(".scss") || filename.endsWith(".sass")) {
var css = renderToCSS({ src, filename, options });
var cssObject = renderCSSToReactNative(css);
if (isPlatformSpecific(filename)) {
return upstreamTransformer.transform({
src: "module.exports = " + JSON.stringify(cssObject),
filename,
options
});
}
return creator.create(filename, css).then(content => {
return content.writeFile().then(() => {
return upstreamTransformer.transform({
src: "module.exports = " + JSON.stringify(cssObject),
filename,
options
});
});
});
}
return upstreamTransformer.transform({ src, filename, options });
};
module.exports.renderToCSS = renderToCSSPromise;