@expo/metro-config
Version:
A Metro config for running React Native projects with the Metro bundler
110 lines • 4.51 kB
JavaScript
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.getFileName = exports.fileNameFromContents = exports.getCssSerialAssets = void 0;
const js_1 = require("metro/src/DeltaBundler/Serializers/helpers/js");
const path_1 = __importDefault(require("path"));
const css_1 = require("../transform-worker/css");
const filePath_1 = require("../utils/filePath");
const hash_1 = require("../utils/hash");
// s = static
const STATIC_EXPORT_DIRECTORY = '_expo/static/css';
function isTypeJSModule(module) {
return (0, js_1.isJsModule)(module);
}
function getCssSerialAssets(dependencies, { projectRoot, entryFile }) {
const assets = [];
const visited = new Set();
function pushCssModule(module) {
const cssMetadata = getCssMetadata(module);
if (cssMetadata) {
const contents = cssMetadata.code;
// NOTE(cedric): these relative paths are used as URL pathnames when serializing HTML
// Use POSIX-format to avoid urls like `_expo/static/css/some\\file\\name.css`
const originFilename = (0, filePath_1.toPosixPath)(path_1.default.relative(projectRoot, module.path));
const filename = (0, filePath_1.toPosixPath)(path_1.default.join(
// Consistent location
STATIC_EXPORT_DIRECTORY,
// Hashed file contents + name for caching
fileNameFromContents({
// Stable filename for hashing in CI.
filepath: originFilename,
src: contents,
}) + '.css'));
if (cssMetadata.externalImports) {
for (const external of cssMetadata.externalImports) {
let source = `<link rel="stylesheet" href="${external.url}"`;
// TODO: How can we do this for local css imports?
if (external.media) {
source += `media="${external.media}"`;
}
// TODO: supports attribute
source += '>';
assets.push({
type: 'css-external',
originFilename,
filename: external.url,
// Link CSS file
source,
metadata: {
hmrId: (0, css_1.pathToHtmlSafeName)(originFilename),
},
});
}
}
assets.push({
type: 'css',
originFilename,
filename,
source: contents,
metadata: {
hmrId: (0, css_1.pathToHtmlSafeName)(originFilename),
},
});
}
}
function checkDep(absolutePath) {
if (visited.has(absolutePath)) {
return;
}
visited.add(absolutePath);
const next = dependencies.get(absolutePath);
if (!next) {
return;
}
next.dependencies.forEach((dep) => {
// Traverse the deps next to ensure the CSS is pushed in the correct order.
checkDep(dep.absolutePath);
});
// Then push the JS after the siblings.
if (getCssMetadata(next) && isTypeJSModule(next)) {
pushCssModule(next);
}
}
checkDep(entryFile);
return assets;
}
exports.getCssSerialAssets = getCssSerialAssets;
function getCssMetadata(module) {
const data = module.output[0]?.data;
if (data && typeof data === 'object' && 'css' in data) {
if (typeof data.css !== 'object' || !('code' in data.css)) {
throw new Error(`Unexpected CSS metadata in Metro module (${module.path}): ${JSON.stringify(data.css)}`);
}
return data.css;
}
return null;
}
function fileNameFromContents({ filepath, src }) {
// Decode if the path is encoded from the Metro dev server, then normalize paths for Windows support.
const decoded = decodeURIComponent(filepath).replace(/\\/g, '/');
return getFileName(decoded) + '-' + (0, hash_1.hashString)(src);
}
exports.fileNameFromContents = fileNameFromContents;
function getFileName(module) {
return path_1.default.basename(module).replace(/\.[^.]+$/, '');
}
exports.getFileName = getFileName;
//# sourceMappingURL=getCssDeps.js.map
;