2022-10-19 01:20:10 +08:00
|
|
|
/* global Babel:true Terser:true */
|
2022-09-01 18:50:46 +08:00
|
|
|
|
|
|
|
// This is executed in mini_racer to provide the JS logic for lib/discourse_js_processor.rb
|
|
|
|
|
|
|
|
const makeEmberTemplateCompilerPlugin =
|
|
|
|
require("babel-plugin-ember-template-compilation").default;
|
2022-10-17 21:47:16 +08:00
|
|
|
const colocatedBabelPlugin = require("colocated-babel-plugin").default;
|
2022-09-01 18:50:46 +08:00
|
|
|
const precompile = require("ember-template-compiler").precompile;
|
|
|
|
const Handlebars = require("handlebars").default;
|
|
|
|
|
|
|
|
function manipulateAstNodeForTheme(node, themeId) {
|
|
|
|
// Magically add theme id as the first param for each of these helpers)
|
|
|
|
if (
|
|
|
|
node.path.parts &&
|
|
|
|
["theme-i18n", "theme-prefix", "theme-setting"].includes(node.path.parts[0])
|
|
|
|
) {
|
|
|
|
if (node.params.length === 1) {
|
|
|
|
node.params.unshift({
|
|
|
|
type: "NumberLiteral",
|
|
|
|
value: themeId,
|
|
|
|
original: themeId,
|
|
|
|
loc: { start: {}, end: {} },
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function buildEmberTemplateManipulatorPlugin(themeId) {
|
|
|
|
return function () {
|
|
|
|
return {
|
|
|
|
name: "theme-template-manipulator",
|
|
|
|
visitor: {
|
|
|
|
SubExpression: (node) => manipulateAstNodeForTheme(node, themeId),
|
|
|
|
MustacheStatement: (node) => manipulateAstNodeForTheme(node, themeId),
|
|
|
|
},
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
function buildTemplateCompilerBabelPlugins({ themeId }) {
|
|
|
|
let compileFunction = precompile;
|
|
|
|
if (themeId) {
|
|
|
|
compileFunction = (src, opts) => {
|
|
|
|
return precompile(src, {
|
|
|
|
...opts,
|
|
|
|
plugins: {
|
|
|
|
ast: [buildEmberTemplateManipulatorPlugin(themeId)],
|
|
|
|
},
|
|
|
|
});
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return [
|
2022-10-17 21:47:16 +08:00
|
|
|
colocatedBabelPlugin,
|
2022-09-01 18:50:46 +08:00
|
|
|
require("widget-hbs-compiler").WidgetHbsCompiler,
|
|
|
|
[
|
|
|
|
makeEmberTemplateCompilerPlugin(() => compileFunction),
|
|
|
|
{ enableLegacyModules: ["ember-cli-htmlbars"] },
|
|
|
|
],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
function buildThemeRawHbsTemplateManipulatorPlugin(themeId) {
|
|
|
|
return function (ast) {
|
|
|
|
["SubExpression", "MustacheStatement"].forEach((pass) => {
|
|
|
|
let visitor = new Handlebars.Visitor();
|
|
|
|
visitor.mutating = true;
|
|
|
|
visitor[pass] = (node) => manipulateAstNodeForTheme(node, themeId);
|
|
|
|
visitor.accept(ast);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
exports.compileRawTemplate = function (source, themeId) {
|
|
|
|
try {
|
|
|
|
const RawHandlebars = require("raw-handlebars").default;
|
|
|
|
const plugins = [];
|
|
|
|
if (themeId) {
|
|
|
|
plugins.push(buildThemeRawHbsTemplateManipulatorPlugin(themeId));
|
|
|
|
}
|
|
|
|
return RawHandlebars.precompile(source, false, { plugins }).toString();
|
|
|
|
} catch (error) {
|
|
|
|
// Workaround for https://github.com/rubyjs/mini_racer/issues/262
|
|
|
|
error.message = JSON.stringify(error.message);
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.transpile = function (
|
|
|
|
source,
|
|
|
|
{ moduleId, filename, skipModule, themeId, commonPlugins } = {}
|
|
|
|
) {
|
|
|
|
const plugins = [];
|
|
|
|
plugins.push(...buildTemplateCompilerBabelPlugins({ themeId }));
|
|
|
|
if (moduleId && !skipModule) {
|
|
|
|
plugins.push(["transform-modules-amd", { noInterop: true }]);
|
|
|
|
}
|
|
|
|
plugins.push(...commonPlugins);
|
|
|
|
|
|
|
|
try {
|
|
|
|
return Babel.transform(source, {
|
|
|
|
moduleId,
|
|
|
|
filename,
|
|
|
|
ast: false,
|
|
|
|
plugins,
|
|
|
|
}).code;
|
|
|
|
} catch (error) {
|
|
|
|
// Workaround for https://github.com/rubyjs/mini_racer/issues/262
|
|
|
|
error.message = JSON.stringify(error.message);
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
};
|
2022-10-19 01:20:10 +08:00
|
|
|
|
|
|
|
// mini_racer doesn't have native support for getting the result of an async operation.
|
|
|
|
// To work around that, we provide a getMinifyResult which can be used to fetch the result
|
|
|
|
// in a followup method call.
|
|
|
|
let lastMinifyError, lastMinifyResult;
|
|
|
|
|
|
|
|
exports.minify = async function (sources, options) {
|
|
|
|
lastMinifyError = lastMinifyResult = null;
|
|
|
|
try {
|
|
|
|
lastMinifyResult = await Terser.minify(sources, options);
|
|
|
|
} catch (e) {
|
|
|
|
lastMinifyError = e;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.getMinifyResult = function () {
|
|
|
|
const error = lastMinifyError;
|
|
|
|
const result = lastMinifyResult;
|
|
|
|
|
|
|
|
lastMinifyError = lastMinifyResult = null;
|
|
|
|
|
|
|
|
if (error) {
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
};
|