UNPKG

14.3 kBJavaScriptView Raw
1function _inheritsLoose(subClass, superClass) { subClass.prototype = Object.create(superClass.prototype); subClass.prototype.constructor = subClass; subClass.__proto__ = superClass; }
2
3import { ConcreteBounds, NewElementBuilder } from '@glimmer/runtime';
4var TEXT_NODE = 3;
5var NEEDS_EXTRA_CLOSE = new WeakMap();
6
7function currentNode(cursor) {
8 var element = cursor.element,
9 nextSibling = cursor.nextSibling;
10
11 if (nextSibling === null) {
12 return element.lastChild;
13 } else {
14 return nextSibling.previousSibling;
15 }
16}
17
18var SerializeBuilder = /*#__PURE__*/function (_NewElementBuilder) {
19 _inheritsLoose(SerializeBuilder, _NewElementBuilder);
20
21 function SerializeBuilder() {
22 var _this;
23
24 _this = _NewElementBuilder.apply(this, arguments) || this;
25 _this.serializeBlockDepth = 0;
26 return _this;
27 }
28
29 var _proto = SerializeBuilder.prototype;
30
31 _proto.__openBlock = function __openBlock() {
32 var tagName = this.element.tagName;
33
34 if (tagName !== 'TITLE' && tagName !== 'SCRIPT' && tagName !== 'STYLE') {
35 var depth = this.serializeBlockDepth++;
36
37 this.__appendComment("%+b:" + depth + "%");
38 }
39
40 _NewElementBuilder.prototype.__openBlock.call(this);
41 };
42
43 _proto.__closeBlock = function __closeBlock() {
44 var tagName = this.element.tagName;
45
46 _NewElementBuilder.prototype.__closeBlock.call(this);
47
48 if (tagName !== 'TITLE' && tagName !== 'SCRIPT' && tagName !== 'STYLE') {
49 var depth = --this.serializeBlockDepth;
50
51 this.__appendComment("%-b:" + depth + "%");
52 }
53 };
54
55 _proto.__appendHTML = function __appendHTML(html) {
56 var tagName = this.element.tagName;
57
58 if (tagName === 'TITLE' || tagName === 'SCRIPT' || tagName === 'STYLE') {
59 return _NewElementBuilder.prototype.__appendHTML.call(this, html);
60 } // Do we need to run the html tokenizer here?
61
62
63 var first = this.__appendComment('%glmr%');
64
65 if (tagName === 'TABLE') {
66 var openIndex = html.indexOf('<');
67
68 if (openIndex > -1) {
69 var tr = html.slice(openIndex + 1, openIndex + 3);
70
71 if (tr === 'tr') {
72 html = "<tbody>" + html + "</tbody>";
73 }
74 }
75 }
76
77 if (html === '') {
78 this.__appendComment('% %');
79 } else {
80 _NewElementBuilder.prototype.__appendHTML.call(this, html);
81 }
82
83 var last = this.__appendComment('%glmr%');
84
85 return new ConcreteBounds(this.element, first, last);
86 };
87
88 _proto.__appendText = function __appendText(string) {
89 var tagName = this.element.tagName;
90 var current = currentNode(this);
91
92 if (tagName === 'TITLE' || tagName === 'SCRIPT' || tagName === 'STYLE') {
93 return _NewElementBuilder.prototype.__appendText.call(this, string);
94 } else if (string === '') {
95 return this.__appendComment('% %');
96 } else if (current && current.nodeType === TEXT_NODE) {
97 this.__appendComment('%|%');
98 }
99
100 return _NewElementBuilder.prototype.__appendText.call(this, string);
101 };
102
103 _proto.closeElement = function closeElement() {
104 if (NEEDS_EXTRA_CLOSE.has(this.element)) {
105 NEEDS_EXTRA_CLOSE["delete"](this.element);
106
107 _NewElementBuilder.prototype.closeElement.call(this);
108 }
109
110 return _NewElementBuilder.prototype.closeElement.call(this);
111 };
112
113 _proto.openElement = function 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 _NewElementBuilder.prototype.openElement.call(this, tag);
127 };
128
129 _proto.pushRemoteElement = function pushRemoteElement(element, cursorId, insertBefore) {
130 if (insertBefore === void 0) {
131 insertBefore = null;
132 }
133
134 var dom = this.dom;
135 var script = dom.createElement('script');
136 script.setAttribute('glmr', cursorId);
137 dom.insertBefore(element, script, insertBefore);
138 return _NewElementBuilder.prototype.pushRemoteElement.call(this, element, cursorId, insertBefore);
139 };
140
141 return SerializeBuilder;
142}(NewElementBuilder);
143
144export function serializeBuilder(env, cursor) {
145 return SerializeBuilder.forInitialRender(env, cursor);
146}
147//# 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