From 40a004573606518ea13e8cea634a2fc2c7699816 Mon Sep 17 00:00:00 2001 From: Josemi Date: Fri, 12 Jul 2024 19:02:24 +0200 Subject: [PATCH 1/4] Fix api for helpers --- index.js | 14 ++++++-------- 1 file changed, 6 insertions(+), 8 deletions(-) diff --git a/index.js b/index.js index 70cdc7a..ab49078 100644 --- a/index.js +++ b/index.js @@ -12,14 +12,14 @@ const escape = s => s.toString().replace(/[&<>\"']/g, m => escapedChars[m]); const get = (c, p) => (p === "." ? c : p.split(".").reduce((x, k) => x?.[k], c)) ?? ""; const defaultHelpers = { - "each": ({value, fn}) => { + "each": (value, opt) => { const items = (typeof value === "object" ? Object.entries(value || {}) : []); return items - .map((item, index) => fn(item[1], {index: index, key: item[0], value: item[1], first: index === 0, last: index === items.length - 1})) + .map((item, index) => opt.fn(item[1], {index: index, key: item[0], value: item[1], first: index === 0, last: index === items.length - 1})) .join(""); }, - "if": ({value, fn, context}) => !!value ? fn(context) : "", - "unless": ({value, fn, context}) => !!!value ? fn(context) : "", + "if": (value, opt) => !!value ? opt.fn(opt.context) : "", + "unless": (value, opt) => !!!value ? opt.fn(opt.context) : "", }; const compile = (tokens, output, context, partials, helpers, vars, fn = {}, index = 0, section = "") => { @@ -35,12 +35,10 @@ const compile = (tokens, output, context, partials, helpers, vars, fn = {}, inde output.push(get(context, tokens[i].slice(1).trim())); } else if (tokens[i].startsWith("#") && typeof helpers[tokens[i].slice(1).trim().split(" ")[0]] === "function") { - const [t, v] = tokens[i].slice(1).trim().split(" "); + const [t, ...args] = tokens[i].slice(1).trim().split(" "); const j = i + 1; - output.push(helpers[t]({ + output.push(helpers[t](...args.map(v => (v || "").startsWith("@") ? get(vars, v.slice(1)) : get(context, v || ".")), { context: context, - key: v || ".", - value: (v || "").startsWith("@") ? get(vars, v.slice(1)) : get(context, v || "."), fn: (blockContext = {}, blockVars = {}, blockOutput = []) => { i = compile(tokens, blockOutput, blockContext, partials, helpers, {...vars, ...blockVars, root: vars.root}, fn, j, t); return blockOutput.join(""); From ad4d83e46702bfea71a2970689f4e024da2f8c11 Mon Sep 17 00:00:00 2001 From: Josemi Date: Fri, 12 Jul 2024 19:02:30 +0200 Subject: [PATCH 2/4] Update tests --- test.js | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/test.js b/test.js index d46491f..22899b4 100644 --- a/test.js +++ b/test.js @@ -189,11 +189,20 @@ describe("{{#customHelper }}", () => { it("should allow to execute a simple custom helper", () => { const options = { helpers: { - hello: ({value}) => `Hello ${value}!!`, + hello: value => `Hello ${value}!!`, }, }; assert.equal(m("{{#hello name}}{{/hello}}", {name: "Bob"}, options), "Hello Bob!!"); }); + + it("should allow to provide multiple values to custom helper", () => { + const options = { + helpers: { + concat: (a, b) => [a, b].join(" "), + }, + }; + assert.equal(m("{{#concat a b}}{{/concat}}!", {a: "hello", b: "world"}, options), "hello world!"); + }); }); describe("{{@root}}", () => { From 85fc9ca9eb88135186f6ad066b323f2a1bd82771 Mon Sep 17 00:00:00 2001 From: Josemi Date: Fri, 12 Jul 2024 19:03:12 +0200 Subject: [PATCH 3/4] Minor code improvements in each helper --- index.js | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/index.js b/index.js index ab49078..36cbefb 100644 --- a/index.js +++ b/index.js @@ -13,9 +13,8 @@ const get = (c, p) => (p === "." ? c : p.split(".").reduce((x, k) => x?.[k], c)) const defaultHelpers = { "each": (value, opt) => { - const items = (typeof value === "object" ? Object.entries(value || {}) : []); - return items - .map((item, index) => opt.fn(item[1], {index: index, key: item[0], value: item[1], first: index === 0, last: index === items.length - 1})) + return (typeof value === "object" ? Object.entries(value || {}) : []) + .map((item, index, items) => opt.fn(item[1], {index: index, key: item[0], value: item[1], first: index === 0, last: index === items.length - 1})) .join(""); }, "if": (value, opt) => !!value ? opt.fn(opt.context) : "", From 0ad98a06d79be0dd325be0ae17a9f362adc3ba9a Mon Sep 17 00:00:00 2001 From: Josemi Date: Fri, 12 Jul 2024 19:12:35 +0200 Subject: [PATCH 4/4] Update API for helpers --- README.md | 33 +++++++++++++++++++++++++-------- 1 file changed, 25 insertions(+), 8 deletions(-) diff --git a/README.md b/README.md index 1e7a4e3..0925fd2 100644 --- a/README.md +++ b/README.md @@ -214,7 +214,7 @@ const data = { }; const options = { helpers: { - customHelper: ({context, value, key, options, fn}) => { + customHelper: value => { return `Hello, ${value}!`; }, }, @@ -224,15 +224,32 @@ const result = m(template, data, options); console.log(result); // Output: "Hello, World!" ``` -Custom helper functions receive a single object parameter containing the following fields: +Custom helper functions receive multiple arguments, where the first N arguments are the variables with the helper is called in the template, and the last argument is an options object containing the following keys: -- `context`: The current context (data) where the helper has been executed. -- `value`: The current value passed to the helper. -- `key`: The field used to extract the value from the current context. -- `options`: The global options object. -- `fn`: A function that executes the template provided in the helper block and returns a string with the evaluated template in the provided context. +- `context`: the current context (data) where the helper has been executed. +- `fn`: a function that executes the template provided in the helper block and returns a string with the evaluated template in the provided context. -The helper function must return a string, which will be injected into the result string. +The helper function must return a string, which will be injected into the result string. Example: + +```javascript +const data = { + items: [ + { name: "John" }, + { name: "Alice" }, + { name: "Bob" }, + ], +}; +const options = { + helpers: { + customEach: (items, opt) => { + return items.map((item, index) => opt.fn({ ...item, index: index})).join(""); + }, + }, +}; + +const result = m("{{#customEach items}}{{index}}: {{name}}, {{/customEach}}", data, options); +console.log(result); // --> "0: John, 1: Alice, 2: Bob," +``` ### Runtime Variables