1 | import { ConcreteBounds, NewElementBuilder } from '@glimmer/runtime';
|
2 | const TEXT_NODE = 3;
|
3 | const NEEDS_EXTRA_CLOSE = new WeakMap();
|
4 |
|
5 | function currentNode(cursor) {
|
6 | let {
|
7 | element,
|
8 | nextSibling
|
9 | } = cursor;
|
10 |
|
11 | if (nextSibling === null) {
|
12 | return element.lastChild;
|
13 | } else {
|
14 | return nextSibling.previousSibling;
|
15 | }
|
16 | }
|
17 |
|
18 | class SerializeBuilder extends NewElementBuilder {
|
19 | constructor() {
|
20 | super(...arguments);
|
21 | this.serializeBlockDepth = 0;
|
22 | }
|
23 |
|
24 | __openBlock() {
|
25 | let {
|
26 | tagName
|
27 | } = this.element;
|
28 |
|
29 | if (tagName !== 'TITLE' && tagName !== 'SCRIPT' && tagName !== 'STYLE') {
|
30 | let depth = this.serializeBlockDepth++;
|
31 |
|
32 | this.__appendComment(`%+b:${depth}%`);
|
33 | }
|
34 |
|
35 | super.__openBlock();
|
36 | }
|
37 |
|
38 | __closeBlock() {
|
39 | let {
|
40 | tagName
|
41 | } = this.element;
|
42 |
|
43 | super.__closeBlock();
|
44 |
|
45 | if (tagName !== 'TITLE' && tagName !== 'SCRIPT' && tagName !== 'STYLE') {
|
46 | let depth = --this.serializeBlockDepth;
|
47 |
|
48 | this.__appendComment(`%-b:${depth}%`);
|
49 | }
|
50 | }
|
51 |
|
52 | __appendHTML(html) {
|
53 | let {
|
54 | tagName
|
55 | } = this.element;
|
56 |
|
57 | if (tagName === 'TITLE' || tagName === 'SCRIPT' || tagName === 'STYLE') {
|
58 | return super.__appendHTML(html);
|
59 | } // Do we need to run the html tokenizer here?
|
60 |
|
61 |
|
62 | let first = this.__appendComment('%glmr%');
|
63 |
|
64 | if (tagName === 'TABLE') {
|
65 | let openIndex = html.indexOf('<');
|
66 |
|
67 | if (openIndex > -1) {
|
68 | let tr = html.slice(openIndex + 1, openIndex + 3);
|
69 |
|
70 | if (tr === 'tr') {
|
71 | html = `<tbody>${html}</tbody>`;
|
72 | }
|
73 | }
|
74 | }
|
75 |
|
76 | if (html === '') {
|
77 | this.__appendComment('% %');
|
78 | } else {
|
79 | super.__appendHTML(html);
|
80 | }
|
81 |
|
82 | let last = this.__appendComment('%glmr%');
|
83 |
|
84 | return new ConcreteBounds(this.element, first, last);
|
85 | }
|
86 |
|
87 | __appendText(string) {
|
88 | let {
|
89 | tagName
|
90 | } = this.element;
|
91 | let current = currentNode(this);
|
92 |
|
93 | if (tagName === 'TITLE' || tagName === 'SCRIPT' || tagName === 'STYLE') {
|
94 | return super.__appendText(string);
|
95 | } else if (string === '') {
|
96 | return this.__appendComment('% %');
|
97 | } else if (current && current.nodeType === TEXT_NODE) {
|
98 | this.__appendComment('%|%');
|
99 | }
|
100 |
|
101 | return super.__appendText(string);
|
102 | }
|
103 |
|
104 | closeElement() {
|
105 | if (NEEDS_EXTRA_CLOSE.has(this.element)) {
|
106 | NEEDS_EXTRA_CLOSE.delete(this.element);
|
107 | super.closeElement();
|
108 | }
|
109 |
|
110 | return super.closeElement();
|
111 | }
|
112 |
|
113 | openElement(tag) {
|
114 | if (tag === 'tr') {
|
115 | if (this.element.tagName !== 'TBODY' && this.element.tagName !== 'THEAD' && this.element.tagName !== 'TFOOT') {
|
116 | this.openElement('tbody'); // This prevents the closeBlock comment from being re-parented
|
117 | // under the auto inserted tbody. Rehydration builder needs to
|
118 | // account for the insertion since it is injected here and not
|
119 | // really in the template.
|
120 |
|
121 | NEEDS_EXTRA_CLOSE.set(this.constructing, true);
|
122 | this.flushElement(null);
|
123 | }
|
124 | }
|
125 |
|
126 | return super.openElement(tag);
|
127 | }
|
128 |
|
129 | pushRemoteElement(element, cursorId, insertBefore = null) {
|
130 | let {
|
131 | dom
|
132 | } = this;
|
133 | let script = dom.createElement('script');
|
134 | script.setAttribute('glmr', cursorId);
|
135 | dom.insertBefore(element, script, insertBefore);
|
136 | return super.pushRemoteElement(element, cursorId, insertBefore);
|
137 | }
|
138 |
|
139 | }
|
140 |
|
141 | export function serializeBuilder(env, cursor) {
|
142 | return SerializeBuilder.forInitialRender(env, cursor);
|
143 | }
|
144 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../packages/@glimmer/node/lib/serialize-builder.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n  Bounds,\n  Environment,\n  Option,\n  ElementBuilder,\n  Maybe,\n  ModifierInstance,\n} from '@glimmer/interfaces';\nimport { ConcreteBounds, NewElementBuilder } from '@glimmer/runtime';\nimport { RemoteLiveBlock } from '@glimmer/runtime';\nimport type { SimpleElement, SimpleNode, SimpleText } from '@simple-dom/interface';\n\nconst TEXT_NODE = 3;\n\nconst NEEDS_EXTRA_CLOSE = new WeakMap<SimpleNode>();\n\nfunction currentNode(\n  cursor: ElementBuilder | { element: SimpleElement; nextSibling: SimpleNode }\n): Option<SimpleNode> {\n  let { element, nextSibling } = cursor;\n\n  if (nextSibling === null) {\n    return element.lastChild;\n  } else {\n    return nextSibling.previousSibling;\n  }\n}\n\nclass SerializeBuilder extends NewElementBuilder implements ElementBuilder {\n  private serializeBlockDepth = 0;\n\n  __openBlock(): void {\n    let { tagName } = this.element;\n\n    if (tagName !== 'TITLE' && tagName !== 'SCRIPT' && tagName !== 'STYLE') {\n      let depth = this.serializeBlockDepth++;\n      this.__appendComment(`%+b:${depth}%`);\n    }\n\n    super.__openBlock();\n  }\n\n  __closeBlock(): void {\n    let { tagName } = this.element;\n\n    super.__closeBlock();\n\n    if (tagName !== 'TITLE' && tagName !== 'SCRIPT' && tagName !== 'STYLE') {\n      let depth = --this.serializeBlockDepth;\n      this.__appendComment(`%-b:${depth}%`);\n    }\n  }\n\n  __appendHTML(html: string): Bounds {\n    let { tagName } = this.element;\n\n    if (tagName === 'TITLE' || tagName === 'SCRIPT' || tagName === 'STYLE') {\n      return super.__appendHTML(html);\n    }\n\n    // Do we need to run the html tokenizer here?\n    let first = this.__appendComment('%glmr%');\n    if (tagName === 'TABLE') {\n      let openIndex = html.indexOf('<');\n      if (openIndex > -1) {\n        let tr = html.slice(openIndex + 1, openIndex + 3);\n        if (tr === 'tr') {\n          html = `<tbody>${html}</tbody>`;\n        }\n      }\n    }\n    if (html === '') {\n      this.__appendComment('% %');\n    } else {\n      super.__appendHTML(html);\n    }\n\n    let last = this.__appendComment('%glmr%');\n    return new ConcreteBounds(this.element, first, last);\n  }\n\n  __appendText(string: string): SimpleText {\n    let { tagName } = this.element;\n    let current = currentNode(this);\n\n    if (tagName === 'TITLE' || tagName === 'SCRIPT' || tagName === 'STYLE') {\n      return super.__appendText(string);\n    } else if (string === '') {\n      return (this.__appendComment('% %') as any) as SimpleText;\n    } else if (current && current.nodeType === TEXT_NODE) {\n      this.__appendComment('%|%');\n    }\n\n    return super.__appendText(string);\n  }\n\n  closeElement(): Option<ModifierInstance[]> {\n    if (NEEDS_EXTRA_CLOSE.has(this.element)) {\n      NEEDS_EXTRA_CLOSE.delete(this.element);\n      super.closeElement();\n    }\n\n    return super.closeElement();\n  }\n\n  openElement(tag: string) {\n    if (tag === 'tr') {\n      if (\n        this.element.tagName !== 'TBODY' &&\n        this.element.tagName !== 'THEAD' &&\n        this.element.tagName !== 'TFOOT'\n      ) {\n        this.openElement('tbody');\n        // This prevents the closeBlock comment from being re-parented\n        // under the auto inserted tbody. Rehydration builder needs to\n        // account for the insertion since it is injected here and not\n        // really in the template.\n        NEEDS_EXTRA_CLOSE.set(this.constructing!, true);\n        this.flushElement(null);\n      }\n    }\n\n    return super.openElement(tag);\n  }\n\n  pushRemoteElement(\n    element: SimpleElement,\n    cursorId: string,\n    insertBefore: Maybe<SimpleNode> = null\n  ): Option<RemoteLiveBlock> {\n    let { dom } = this;\n    let script = dom.createElement('script');\n    script.setAttribute('glmr', cursorId);\n    dom.insertBefore(element, script, insertBefore);\n    return super.pushRemoteElement(element, cursorId, insertBefore);\n  }\n}\n\nexport function serializeBuilder(\n  env: Environment,\n  cursor: { element: SimpleElement; nextSibling: Option<SimpleNode> }\n): ElementBuilder {\n  return SerializeBuilder.forInitialRender(env, cursor);\n}\n"],"sourceRoot":""} |
\ | No newline at end of file |