hyperscript.org
Version:
a small scripting language for the web
86 lines (76 loc) • 2.61 kB
JavaScript
(function (self, factory) {
const plugin = factory(self)
if (typeof exports === 'object' && typeof exports['nodeName'] !== 'string') {
module.exports = plugin
} else {
if ('_hyperscript' in self) self._hyperscript.use(plugin)
}
})(typeof self !== 'undefined' ? self : this, self => {
function compileTemplate(template) {
return template.replace(/(?:^|\n)([^@]*)@?/gm, function (match, p1) {
var templateStr = (" " + p1).replace(/([^\\])\$\{/g, "$1$${escape html ").substring(1);
return "\ncall meta.__ht_template_result.push(`" + templateStr + "`)\n";
});
}
/**
* @param {HyperscriptObject} _hyperscript
*/
return _hyperscript => {
function renderTemplate(template, ctx) {
var buf = [];
const renderCtx = Object.assign({}, ctx);
renderCtx.meta = Object.assign({ __ht_template_result: buf }, ctx.meta);
_hyperscript.evaluate(template, renderCtx);
return buf.join("");
}
_hyperscript.addCommand("render", function (parser, runtime, tokens) {
if (!tokens.matchToken("render")) return;
var template_ = parser.requireElement("expression", tokens);
var templateArgs = {};
if (tokens.matchToken("with")) {
templateArgs = parser.parseElement("namedArgumentList", tokens);
}
return {
args: [template_, templateArgs],
op: function (ctx, template, templateArgs) {
if (!(template instanceof Element)) throw new Error(template_.sourceFor() + " is not an element");
const context = _hyperscript.internals.runtime.makeContext()
context.locals = templateArgs;
ctx.result = renderTemplate(compileTemplate(template.innerHTML), context);
return runtime.findNext(this, ctx);
},
};
});
function escapeHTML(html) {
return String(html)
.replace(/&/g, "&")
.replace(/</g, "<")
.replace(/>/g, ">")
.replace(/\x22/g, """)
.replace(/\x27/g, "'");
}
_hyperscript.addLeafExpression("escape", function (parser, runtime, tokens) {
if (!tokens.matchToken("escape")) return;
var escapeType = tokens.matchTokenType("IDENTIFIER").value;
// hidden! for use in templates
var unescaped = tokens.matchToken("unescaped");
var arg = parser.requireElement("expression", tokens);
return {
args: [arg],
op: function (ctx, arg) {
if (unescaped) return arg;
if (arg === undefined) return "";
switch (escapeType) {
case "html":
return escapeHTML(arg);
default:
throw new Error("Unknown escape: " + escapeType);
}
},
evaluate: function (ctx) {
return runtime.unifiedEval(this, ctx);
},
};
});
}
})