@joist/element 4.2.2 → 4.2.3-next.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (74) hide show
  1. package/package.json +4 -1
  2. package/src/lib/query.ts +1 -1
  3. package/src/lib/templating/README.md +292 -0
  4. package/src/lib/templating/bind.ts +40 -0
  5. package/src/lib/templating/define.ts +4 -0
  6. package/src/lib/templating/elements/for.element.test.ts +39 -0
  7. package/src/lib/templating/elements/for.element.ts +151 -0
  8. package/src/lib/templating/elements/if.element.test.ts +55 -0
  9. package/src/lib/templating/elements/if.element.ts +67 -0
  10. package/src/lib/templating/elements/props.element.test.ts +62 -0
  11. package/src/lib/templating/elements/props.element.ts +80 -0
  12. package/src/lib/templating/elements/scope.ts +45 -0
  13. package/src/lib/templating/elements/value.element.test.ts +20 -0
  14. package/src/lib/templating/elements/value.element.ts +41 -0
  15. package/src/lib/templating/events.ts +21 -0
  16. package/src/lib/templating/token.test.ts +74 -0
  17. package/src/lib/templating/token.ts +34 -0
  18. package/src/lib/templating.ts +2 -0
  19. package/src/lib.ts +1 -0
  20. package/target/lib/query.d.ts +1 -1
  21. package/target/lib/templating/bind.d.ts +1 -0
  22. package/target/lib/templating/bind.js +30 -0
  23. package/target/lib/templating/bind.js.map +1 -0
  24. package/target/lib/templating/define.d.ts +4 -0
  25. package/target/lib/templating/define.js +5 -0
  26. package/target/lib/templating/define.js.map +1 -0
  27. package/target/lib/templating/elements/for.element.d.ts +23 -0
  28. package/target/lib/templating/elements/for.element.js +168 -0
  29. package/target/lib/templating/elements/for.element.js.map +1 -0
  30. package/target/lib/templating/elements/for.element.test.d.ts +2 -0
  31. package/target/lib/templating/elements/for.element.test.js +34 -0
  32. package/target/lib/templating/elements/for.element.test.js.map +1 -0
  33. package/target/lib/templating/elements/if.element.d.ts +13 -0
  34. package/target/lib/templating/elements/if.element.js +71 -0
  35. package/target/lib/templating/elements/if.element.js.map +1 -0
  36. package/target/lib/templating/elements/if.element.test.d.ts +1 -0
  37. package/target/lib/templating/elements/if.element.test.js +47 -0
  38. package/target/lib/templating/elements/if.element.test.js.map +1 -0
  39. package/target/lib/templating/elements/props.element.d.ts +11 -0
  40. package/target/lib/templating/elements/props.element.js +92 -0
  41. package/target/lib/templating/elements/props.element.js.map +1 -0
  42. package/target/lib/templating/elements/props.element.test.d.ts +1 -0
  43. package/target/lib/templating/elements/props.element.test.js +53 -0
  44. package/target/lib/templating/elements/props.element.test.js.map +1 -0
  45. package/target/lib/templating/elements/scope.d.ts +13 -0
  46. package/target/lib/templating/elements/scope.js +59 -0
  47. package/target/lib/templating/elements/scope.js.map +1 -0
  48. package/target/lib/templating/elements/value.element.d.ts +9 -0
  49. package/target/lib/templating/elements/value.element.js +56 -0
  50. package/target/lib/templating/elements/value.element.js.map +1 -0
  51. package/target/lib/templating/elements/value.element.test.d.ts +1 -0
  52. package/target/lib/templating/elements/value.element.test.js +16 -0
  53. package/target/lib/templating/elements/value.element.test.js.map +1 -0
  54. package/target/lib/templating/events.d.ts +12 -0
  55. package/target/lib/templating/events.js +10 -0
  56. package/target/lib/templating/events.js.map +1 -0
  57. package/target/lib/templating/token.d.ts +8 -0
  58. package/target/lib/templating/token.js +27 -0
  59. package/target/lib/templating/token.js.map +1 -0
  60. package/target/lib/templating/token.test.js +56 -0
  61. package/target/lib/templating/token.test.js.map +1 -0
  62. package/target/lib/templating.d.ts +2 -0
  63. package/target/lib/templating.js +3 -0
  64. package/target/lib/templating.js.map +1 -0
  65. package/target/lib.d.ts +1 -0
  66. package/target/lib.js.map +1 -1
  67. package/src/lib/template.test.ts +0 -123
  68. package/src/lib/template.ts +0 -130
  69. package/target/lib/template.d.ts +0 -11
  70. package/target/lib/template.js +0 -89
  71. package/target/lib/template.js.map +0 -1
  72. package/target/lib/template.test.js +0 -91
  73. package/target/lib/template.test.js.map +0 -1
  74. /package/target/lib/{template.test.d.ts → templating/token.test.d.ts} +0 -0
@@ -0,0 +1,56 @@
1
+ import { assert } from "chai";
2
+ import { JToken } from "./token.js";
3
+ describe("JToken", () => {
4
+ describe("constructor", () => {
5
+ it("should initialize with a raw token", () => {
6
+ const token = new JToken("example.token");
7
+ assert.equal(token.rawToken, "example.token");
8
+ });
9
+ it("should set isNegated to true if the token starts with '!'", () => {
10
+ const token = new JToken("!example.token");
11
+ assert.isTrue(token.isNegated);
12
+ });
13
+ it("should set isNegated to false if the token does not start with '!'", () => {
14
+ const token = new JToken("example.token");
15
+ assert.isFalse(token.isNegated);
16
+ });
17
+ it("should correctly parse the bindTo property", () => {
18
+ const token = new JToken("example.token");
19
+ assert.equal(token.bindTo, "example");
20
+ });
21
+ it("should correctly parse the path property", () => {
22
+ const token = new JToken("example.token.part");
23
+ assert.deepEqual(token.path, ["token", "part"]);
24
+ });
25
+ it("should remove '!' from bindTo if present", () => {
26
+ const token = new JToken("!example.token");
27
+ assert.equal(token.bindTo, "example");
28
+ });
29
+ });
30
+ describe("readTokenValueFrom", () => {
31
+ it("should read the value from a nested object", () => {
32
+ const token = new JToken("example.token.part");
33
+ const obj = { token: { part: 42 } };
34
+ const value = token.readTokenValueFrom(obj);
35
+ assert.equal(value, 42);
36
+ });
37
+ it("should return undefined if the path does not exist", () => {
38
+ const token = new JToken("example.nonexistent.path");
39
+ const obj = { token: { part: 42 } };
40
+ const value = token.readTokenValueFrom(obj);
41
+ assert.isUndefined(value);
42
+ });
43
+ it("should handle empty paths gracefully", () => {
44
+ const token = new JToken("example");
45
+ const obj = { foo: 42 };
46
+ const value = token.readTokenValueFrom(obj);
47
+ assert.deepEqual(value, { foo: 42 });
48
+ });
49
+ it("should throw an error if the object is null or undefined", () => {
50
+ const token = new JToken("example.token");
51
+ assert.throws(() => token.readTokenValueFrom(null), TypeError);
52
+ assert.throws(() => token.readTokenValueFrom(undefined), TypeError);
53
+ });
54
+ });
55
+ });
56
+ //# sourceMappingURL=token.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"token.test.js","sourceRoot":"","sources":["../../../src/lib/templating/token.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,MAAM,CAAC;AAE9B,OAAO,EAAE,MAAM,EAAE,MAAM,YAAY,CAAC;AAEpC,QAAQ,CAAC,QAAQ,EAAE,GAAG,EAAE;IACtB,QAAQ,CAAC,aAAa,EAAE,GAAG,EAAE;QAC3B,EAAE,CAAC,oCAAoC,EAAE,GAAG,EAAE;YAC5C,MAAM,KAAK,GAAG,IAAI,MAAM,CAAC,eAAe,CAAC,CAAC;YAC1C,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,QAAQ,EAAE,eAAe,CAAC,CAAC;QAChD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,2DAA2D,EAAE,GAAG,EAAE;YACnE,MAAM,KAAK,GAAG,IAAI,MAAM,CAAC,gBAAgB,CAAC,CAAC;YAC3C,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QACjC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,oEAAoE,EAAE,GAAG,EAAE;YAC5E,MAAM,KAAK,GAAG,IAAI,MAAM,CAAC,eAAe,CAAC,CAAC;YAC1C,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAClC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,4CAA4C,EAAE,GAAG,EAAE;YACpD,MAAM,KAAK,GAAG,IAAI,MAAM,CAAC,eAAe,CAAC,CAAC;YAC1C,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,MAAM,EAAE,SAAS,CAAC,CAAC;QACxC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,0CAA0C,EAAE,GAAG,EAAE;YAClD,MAAM,KAAK,GAAG,IAAI,MAAM,CAAC,oBAAoB,CAAC,CAAC;YAC/C,MAAM,CAAC,SAAS,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC,CAAC;QAClD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,0CAA0C,EAAE,GAAG,EAAE;YAClD,MAAM,KAAK,GAAG,IAAI,MAAM,CAAC,gBAAgB,CAAC,CAAC;YAC3C,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,MAAM,EAAE,SAAS,CAAC,CAAC;QACxC,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,QAAQ,CAAC,oBAAoB,EAAE,GAAG,EAAE;QAClC,EAAE,CAAC,4CAA4C,EAAE,GAAG,EAAE;YACpD,MAAM,KAAK,GAAG,IAAI,MAAM,CAAC,oBAAoB,CAAC,CAAC;YAC/C,MAAM,GAAG,GAAG,EAAE,KAAK,EAAE,EAAE,IAAI,EAAE,EAAE,EAAE,EAAE,CAAC;YACpC,MAAM,KAAK,GAAG,KAAK,CAAC,kBAAkB,CAAS,GAAG,CAAC,CAAC;YACpD,MAAM,CAAC,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAC1B,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,oDAAoD,EAAE,GAAG,EAAE;YAC5D,MAAM,KAAK,GAAG,IAAI,MAAM,CAAC,0BAA0B,CAAC,CAAC;YACrD,MAAM,GAAG,GAAG,EAAE,KAAK,EAAE,EAAE,IAAI,EAAE,EAAE,EAAE,EAAE,CAAC;YACpC,MAAM,KAAK,GAAG,KAAK,CAAC,kBAAkB,CAAC,GAAG,CAAC,CAAC;YAC5C,MAAM,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;QAC5B,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,sCAAsC,EAAE,GAAG,EAAE;YAC9C,MAAM,KAAK,GAAG,IAAI,MAAM,CAAC,SAAS,CAAC,CAAC;YACpC,MAAM,GAAG,GAAG,EAAE,GAAG,EAAE,EAAE,EAAE,CAAC;YACxB,MAAM,KAAK,GAAG,KAAK,CAAC,kBAAkB,CAAC,GAAG,CAAC,CAAC;YAE5C,MAAM,CAAC,SAAS,CAAC,KAAK,EAAE,EAAE,GAAG,EAAE,EAAE,EAAE,CAAC,CAAC;QACvC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,0DAA0D,EAAE,GAAG,EAAE;YAClE,MAAM,KAAK,GAAG,IAAI,MAAM,CAAC,eAAe,CAAC,CAAC;YAC1C,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,KAAK,CAAC,kBAAkB,CAAM,IAAW,CAAC,EAChD,SAAS,CACV,CAAC;YAEF,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,KAAK,CAAC,kBAAkB,CAAM,SAAgB,CAAC,EACrD,SAAS,CACV,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -0,0 +1,2 @@
1
+ export { bind } from "./templating/bind.js";
2
+ export { JoistValueEvent } from "./templating/events.js";
@@ -0,0 +1,3 @@
1
+ export { bind } from "./templating/bind.js";
2
+ export { JoistValueEvent } from "./templating/events.js";
3
+ //# sourceMappingURL=templating.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"templating.js","sourceRoot":"","sources":["../../src/lib/templating.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,sBAAsB,CAAC;AAC5C,OAAO,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC"}
package/target/lib.d.ts CHANGED
@@ -4,5 +4,6 @@ export { listen } from "./lib/listen.js";
4
4
  export { element } from "./lib/element.js";
5
5
  export { query } from "./lib/query.js";
6
6
  export { queryAll } from "./lib/query-all.js";
7
+ export { QueryResult } from "./lib/query.js";
7
8
  export { ready } from "./lib/lifecycle.js";
8
9
  export { attrChanged } from "./lib/attr-changed.js";
package/target/lib.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"lib.js","sourceRoot":"","sources":["../src/lib.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AACjE,OAAO,EAAE,IAAI,EAAE,MAAM,eAAe,CAAC;AACrC,OAAO,EAAE,MAAM,EAAE,MAAM,iBAAiB,CAAC;AACzC,OAAO,EAAE,OAAO,EAAE,MAAM,kBAAkB,CAAC;AAC3C,OAAO,EAAE,KAAK,EAAE,MAAM,gBAAgB,CAAC;AACvC,OAAO,EAAE,QAAQ,EAAE,MAAM,oBAAoB,CAAC;AAC9C,OAAO,EAAE,KAAK,EAAE,MAAM,oBAAoB,CAAC;AAC3C,OAAO,EAAE,WAAW,EAAE,MAAM,uBAAuB,CAAC"}
1
+ {"version":3,"file":"lib.js","sourceRoot":"","sources":["../src/lib.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AACjE,OAAO,EAAE,IAAI,EAAE,MAAM,eAAe,CAAC;AACrC,OAAO,EAAE,MAAM,EAAE,MAAM,iBAAiB,CAAC;AACzC,OAAO,EAAE,OAAO,EAAE,MAAM,kBAAkB,CAAC;AAC3C,OAAO,EAAE,KAAK,EAAE,MAAM,gBAAgB,CAAC;AACvC,OAAO,EAAE,QAAQ,EAAE,MAAM,oBAAoB,CAAC;AAE9C,OAAO,EAAE,KAAK,EAAE,MAAM,oBAAoB,CAAC;AAC3C,OAAO,EAAE,WAAW,EAAE,MAAM,uBAAuB,CAAC"}
@@ -1,123 +0,0 @@
1
- import { assert } from "chai";
2
-
3
- import { template } from "./template.js";
4
-
5
- // Run all tests with both shadow and light dom
6
- const TESTS = [
7
- function bindableNodes(el: HTMLElement, root: HTMLElement | ShadowRoot) {
8
- it(`should intialize bindable nodes ${root instanceof ShadowRoot ? "(ShadowDOM)" : "(LightDOM)"}`, () => {
9
- el.title = "Hello World";
10
- el.ariaLabel = "This is the label";
11
- el.ariaDescription = "This is the description";
12
-
13
- root.innerHTML = /*html*/ `
14
- <span #:bind="title"></span>
15
-
16
- <ul>
17
- <li #:bind="ariaLabel"></li>
18
- <li #:bind="ariaDescription"></li>
19
- </ul>
20
- `;
21
-
22
- const render = template().bind(el);
23
-
24
- render();
25
-
26
- assert.equal(
27
- root.innerHTML
28
- .split("\n")
29
- .map((res) => res.trim())
30
- .join(""),
31
- '<span #:bind="title">Hello World</span><ul><li #:bind="ariaLabel">This is the label</li><li #:bind="ariaDescription">This is the description</li></ul>',
32
- );
33
- });
34
- },
35
- function attributeNodes(el: HTMLElement, root: HTMLElement | ShadowRoot) {
36
- it(`should intialize template attributes ${root instanceof ShadowRoot ? "(ShadowDOM)" : "(LightDOM)"}`, () => {
37
- el.ariaLabel = "This is the label";
38
- el.ariaDescription = "This is the description";
39
-
40
- root.innerHTML = /*html*/ `
41
- <ul #:aria-label="ariaLabel" #:aria-description="ariaDescription"></ul>
42
- `;
43
-
44
- const render = template().bind(el);
45
-
46
- render();
47
-
48
- assert.equal(
49
- root.innerHTML
50
- .split("\n")
51
- .map((res) => res.trim())
52
- .join(""),
53
- '<ul #:aria-label="ariaLabel" #:aria-description="ariaDescription" aria-label="This is the label" aria-description="This is the description"></ul>',
54
- );
55
- });
56
- },
57
- function customGetter(el: HTMLElement, root: HTMLElement | ShadowRoot) {
58
- it(`should use custom getter for values ${root instanceof ShadowRoot ? "(ShadowDOM)" : "(LightDOM)"}`, () => {
59
- const data: Record<string, string> = {
60
- title: "Hello World",
61
- ariaLabel: "This is the label",
62
- ariaDescription: "This is the description",
63
- };
64
-
65
- root.innerHTML = /*html*/ `
66
- <span #:bind="title"></span>
67
-
68
- <ul>
69
- <li #:bind="ariaLabel"></li>
70
- <li #:bind="ariaDescription"></li>
71
- </ul>
72
- `;
73
-
74
- const render = template({ value: (key) => data[key] }).bind(el);
75
-
76
- render();
77
-
78
- assert.equal(
79
- root.innerHTML
80
- .split("\n")
81
- .map((res) => res.trim())
82
- .join(""),
83
- '<span #:bind="title">Hello World</span><ul><li #:bind="ariaLabel">This is the label</li><li #:bind="ariaDescription">This is the description</li></ul>',
84
- );
85
- });
86
- },
87
- function customPrefix(el: HTMLElement, root: HTMLElement | ShadowRoot) {
88
- it(`should use custom getter for values ${root instanceof ShadowRoot ? "(ShadowDOM)" : "(LightDOM)"}`, () => {
89
- el.title = "Hello World";
90
- el.ariaLabel = "This is the label";
91
- el.ariaDescription = "This is the description";
92
-
93
- root.innerHTML = /*html*/ `
94
- <span x-bind="title"></span>
95
-
96
- <ul x-aria-label="ariaLabel">
97
- <li x-bind="ariaLabel"></li>
98
- <li x-bind="ariaDescription"></li>
99
- </ul>
100
- `;
101
-
102
- const render = template({ tokenPrefix: "x-" }).bind(el);
103
-
104
- render();
105
-
106
- assert.equal(
107
- root.innerHTML
108
- .split("\n")
109
- .map((res) => res.trim())
110
- .join(""),
111
- '<span x-bind="title">Hello World</span><ul x-aria-label="ariaLabel" aria-label="This is the label"><li x-bind="ariaLabel">This is the label</li><li x-bind="ariaDescription">This is the description</li></ul>',
112
- );
113
- });
114
- },
115
- ];
116
-
117
- for (const test of TESTS) {
118
- const lightEl = document.createElement("div");
119
- test(lightEl, lightEl);
120
-
121
- const shadowEl = document.createElement("div");
122
- test(shadowEl, shadowEl.attachShadow({ mode: "open" }));
123
- }
@@ -1,130 +0,0 @@
1
- type Updater = () => void;
2
- class Updates extends Set<Updater> {}
3
- type TemplateValueGetter = (key: string) => string;
4
-
5
- export interface TemplateOpts {
6
- value?: TemplateValueGetter;
7
- tokenPrefix?: string;
8
- }
9
-
10
- export interface RenderOpts {
11
- refresh?: boolean;
12
- }
13
-
14
- export function template({ tokenPrefix = "#:", value }: TemplateOpts = {}) {
15
- // Track all nodes that can be updated and their associated property
16
- let updates: Updates | null = null;
17
-
18
- return function render<T extends HTMLElement>(
19
- this: T,
20
- opts?: RenderOpts,
21
- ): void {
22
- if (!updates || opts?.refresh) {
23
- updates = findUpdates(this, {
24
- tokenPrefix,
25
- value: value ?? ((key: string) => getTemplateValue(this, key)),
26
- });
27
- } else {
28
- for (const update of updates) {
29
- update();
30
- }
31
- }
32
- };
33
- }
34
-
35
- function findUpdates(el: HTMLElement, opts: Required<TemplateOpts>): Updates {
36
- const iterator = document.createTreeWalker(
37
- el.shadowRoot ?? el,
38
- NodeFilter.SHOW_ELEMENT,
39
- );
40
- const updates = new Updates();
41
-
42
- while (iterator.nextNode()) {
43
- const res = trackElement(iterator.currentNode, updates, opts);
44
-
45
- if (res !== null) {
46
- iterator.currentNode = res;
47
- }
48
- }
49
-
50
- return updates;
51
- }
52
-
53
- /**
54
- * configures and tracks a given Node so that it can be updated in place later.
55
- */
56
- function trackElement(
57
- node: Node,
58
- updates: Updates,
59
- opts: Required<TemplateOpts>,
60
- ): Node | null {
61
- const element = node as Element;
62
- const getter = opts.value;
63
- const tokenPrefix = opts.tokenPrefix;
64
-
65
- for (const attr of element.attributes) {
66
- const nodeValue = attr.value.trim();
67
- const realAttributeName = attr.name.replace(tokenPrefix, "");
68
-
69
- let update: Updater | null = null;
70
-
71
- if (attr.name.startsWith(`${tokenPrefix}bind`)) {
72
- update = () => {
73
- const value = getter(attr.value);
74
-
75
- if (element.textContent !== value) {
76
- element.textContent = getter(attr.value);
77
- }
78
- };
79
- } else if (attr.name.startsWith(tokenPrefix)) {
80
- const isBooleanAttr = nodeValue.startsWith("!");
81
- const isPositive = nodeValue.startsWith("!!");
82
- const propertyKey = nodeValue.replaceAll("!", "");
83
-
84
- if (isBooleanAttr) {
85
- update = () => {
86
- const value = isPositive
87
- ? !!getter(propertyKey)
88
- : !getter(propertyKey);
89
-
90
- if (value) {
91
- element.setAttribute(realAttributeName, "");
92
- } else {
93
- element.removeAttribute(realAttributeName);
94
- }
95
- };
96
- } else {
97
- const realAttribute = document.createAttribute(realAttributeName);
98
- element.setAttributeNode(realAttribute);
99
-
100
- update = () => {
101
- const value = getter(nodeValue);
102
-
103
- if (realAttribute.value !== value) {
104
- realAttribute.value = value;
105
- }
106
- };
107
- }
108
- }
109
-
110
- if (update) {
111
- updates.add(update);
112
-
113
- update();
114
- }
115
- }
116
-
117
- return null;
118
- }
119
-
120
- export function getTemplateValue(obj: object, key: string): any {
121
- const parsed = key.split(".");
122
-
123
- let pointer: any = obj;
124
-
125
- for (const part of parsed) {
126
- pointer = pointer[part];
127
- }
128
-
129
- return pointer;
130
- }
@@ -1,11 +0,0 @@
1
- type TemplateValueGetter = (key: string) => string;
2
- export interface TemplateOpts {
3
- value?: TemplateValueGetter;
4
- tokenPrefix?: string;
5
- }
6
- export interface RenderOpts {
7
- refresh?: boolean;
8
- }
9
- export declare function template({ tokenPrefix, value }?: TemplateOpts): <T extends HTMLElement>(this: T, opts?: RenderOpts) => void;
10
- export declare function getTemplateValue(obj: object, key: string): any;
11
- export {};
@@ -1,89 +0,0 @@
1
- class Updates extends Set {
2
- }
3
- export function template({ tokenPrefix = "#:", value } = {}) {
4
- let updates = null;
5
- return function render(opts) {
6
- if (!updates || opts?.refresh) {
7
- updates = findUpdates(this, {
8
- tokenPrefix,
9
- value: value ?? ((key) => getTemplateValue(this, key)),
10
- });
11
- }
12
- else {
13
- for (const update of updates) {
14
- update();
15
- }
16
- }
17
- };
18
- }
19
- function findUpdates(el, opts) {
20
- const iterator = document.createTreeWalker(el.shadowRoot ?? el, NodeFilter.SHOW_ELEMENT);
21
- const updates = new Updates();
22
- while (iterator.nextNode()) {
23
- const res = trackElement(iterator.currentNode, updates, opts);
24
- if (res !== null) {
25
- iterator.currentNode = res;
26
- }
27
- }
28
- return updates;
29
- }
30
- function trackElement(node, updates, opts) {
31
- const element = node;
32
- const getter = opts.value;
33
- const tokenPrefix = opts.tokenPrefix;
34
- for (const attr of element.attributes) {
35
- const nodeValue = attr.value.trim();
36
- const realAttributeName = attr.name.replace(tokenPrefix, "");
37
- let update = null;
38
- if (attr.name.startsWith(`${tokenPrefix}bind`)) {
39
- update = () => {
40
- const value = getter(attr.value);
41
- if (element.textContent !== value) {
42
- element.textContent = getter(attr.value);
43
- }
44
- };
45
- }
46
- else if (attr.name.startsWith(tokenPrefix)) {
47
- const isBooleanAttr = nodeValue.startsWith("!");
48
- const isPositive = nodeValue.startsWith("!!");
49
- const propertyKey = nodeValue.replaceAll("!", "");
50
- if (isBooleanAttr) {
51
- update = () => {
52
- const value = isPositive
53
- ? !!getter(propertyKey)
54
- : !getter(propertyKey);
55
- if (value) {
56
- element.setAttribute(realAttributeName, "");
57
- }
58
- else {
59
- element.removeAttribute(realAttributeName);
60
- }
61
- };
62
- }
63
- else {
64
- const realAttribute = document.createAttribute(realAttributeName);
65
- element.setAttributeNode(realAttribute);
66
- update = () => {
67
- const value = getter(nodeValue);
68
- if (realAttribute.value !== value) {
69
- realAttribute.value = value;
70
- }
71
- };
72
- }
73
- }
74
- if (update) {
75
- updates.add(update);
76
- update();
77
- }
78
- }
79
- return null;
80
- }
81
- export function getTemplateValue(obj, key) {
82
- const parsed = key.split(".");
83
- let pointer = obj;
84
- for (const part of parsed) {
85
- pointer = pointer[part];
86
- }
87
- return pointer;
88
- }
89
- //# sourceMappingURL=template.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"template.js","sourceRoot":"","sources":["../../src/lib/template.ts"],"names":[],"mappings":"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"}
@@ -1,91 +0,0 @@
1
- import { assert } from "chai";
2
- import { template } from "./template.js";
3
- const TESTS = [
4
- function bindableNodes(el, root) {
5
- it(`should intialize bindable nodes ${root instanceof ShadowRoot ? "(ShadowDOM)" : "(LightDOM)"}`, () => {
6
- el.title = "Hello World";
7
- el.ariaLabel = "This is the label";
8
- el.ariaDescription = "This is the description";
9
- root.innerHTML = `
10
- <span #:bind="title"></span>
11
-
12
- <ul>
13
- <li #:bind="ariaLabel"></li>
14
- <li #:bind="ariaDescription"></li>
15
- </ul>
16
- `;
17
- const render = template().bind(el);
18
- render();
19
- assert.equal(root.innerHTML
20
- .split("\n")
21
- .map((res) => res.trim())
22
- .join(""), '<span #:bind="title">Hello World</span><ul><li #:bind="ariaLabel">This is the label</li><li #:bind="ariaDescription">This is the description</li></ul>');
23
- });
24
- },
25
- function attributeNodes(el, root) {
26
- it(`should intialize template attributes ${root instanceof ShadowRoot ? "(ShadowDOM)" : "(LightDOM)"}`, () => {
27
- el.ariaLabel = "This is the label";
28
- el.ariaDescription = "This is the description";
29
- root.innerHTML = `
30
- <ul #:aria-label="ariaLabel" #:aria-description="ariaDescription"></ul>
31
- `;
32
- const render = template().bind(el);
33
- render();
34
- assert.equal(root.innerHTML
35
- .split("\n")
36
- .map((res) => res.trim())
37
- .join(""), '<ul #:aria-label="ariaLabel" #:aria-description="ariaDescription" aria-label="This is the label" aria-description="This is the description"></ul>');
38
- });
39
- },
40
- function customGetter(el, root) {
41
- it(`should use custom getter for values ${root instanceof ShadowRoot ? "(ShadowDOM)" : "(LightDOM)"}`, () => {
42
- const data = {
43
- title: "Hello World",
44
- ariaLabel: "This is the label",
45
- ariaDescription: "This is the description",
46
- };
47
- root.innerHTML = `
48
- <span #:bind="title"></span>
49
-
50
- <ul>
51
- <li #:bind="ariaLabel"></li>
52
- <li #:bind="ariaDescription"></li>
53
- </ul>
54
- `;
55
- const render = template({ value: (key) => data[key] }).bind(el);
56
- render();
57
- assert.equal(root.innerHTML
58
- .split("\n")
59
- .map((res) => res.trim())
60
- .join(""), '<span #:bind="title">Hello World</span><ul><li #:bind="ariaLabel">This is the label</li><li #:bind="ariaDescription">This is the description</li></ul>');
61
- });
62
- },
63
- function customPrefix(el, root) {
64
- it(`should use custom getter for values ${root instanceof ShadowRoot ? "(ShadowDOM)" : "(LightDOM)"}`, () => {
65
- el.title = "Hello World";
66
- el.ariaLabel = "This is the label";
67
- el.ariaDescription = "This is the description";
68
- root.innerHTML = `
69
- <span x-bind="title"></span>
70
-
71
- <ul x-aria-label="ariaLabel">
72
- <li x-bind="ariaLabel"></li>
73
- <li x-bind="ariaDescription"></li>
74
- </ul>
75
- `;
76
- const render = template({ tokenPrefix: "x-" }).bind(el);
77
- render();
78
- assert.equal(root.innerHTML
79
- .split("\n")
80
- .map((res) => res.trim())
81
- .join(""), '<span x-bind="title">Hello World</span><ul x-aria-label="ariaLabel" aria-label="This is the label"><li x-bind="ariaLabel">This is the label</li><li x-bind="ariaDescription">This is the description</li></ul>');
82
- });
83
- },
84
- ];
85
- for (const test of TESTS) {
86
- const lightEl = document.createElement("div");
87
- test(lightEl, lightEl);
88
- const shadowEl = document.createElement("div");
89
- test(shadowEl, shadowEl.attachShadow({ mode: "open" }));
90
- }
91
- //# sourceMappingURL=template.test.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"template.test.js","sourceRoot":"","sources":["../../src/lib/template.test.ts"],"names":[],"mappings":"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"}