glass-easel 0.1.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 (237) hide show
  1. package/README.md +40 -0
  2. package/dist/glass_easel.all.d.ts +1 -0
  3. package/dist/glass_easel.all.js +2 -0
  4. package/dist/glass_easel.all.js.map +1 -0
  5. package/dist/glass_easel.domlike.global.d.ts +1 -0
  6. package/dist/glass_easel.domlike.global.js +2 -0
  7. package/dist/glass_easel.domlike.global.js.map +1 -0
  8. package/dist/index.d.ts +1 -0
  9. package/dist/index.js +1 -0
  10. package/dist/types/src/backend/backend_protocol.d.ts +119 -0
  11. package/dist/types/src/backend/backend_protocol.d.ts.map +1 -0
  12. package/dist/types/src/backend/composed_backend_protocol.d.ts +90 -0
  13. package/dist/types/src/backend/composed_backend_protocol.d.ts.map +1 -0
  14. package/dist/types/src/backend/domlike_backend_protocol.d.ts +76 -0
  15. package/dist/types/src/backend/domlike_backend_protocol.d.ts.map +1 -0
  16. package/dist/types/src/backend/mode.d.ts +46 -0
  17. package/dist/types/src/backend/mode.d.ts.map +1 -0
  18. package/dist/types/src/backend/suggested_backend_protocol.d.ts +30 -0
  19. package/dist/types/src/backend/suggested_backend_protocol.d.ts.map +1 -0
  20. package/dist/types/src/behavior.d.ts +428 -0
  21. package/dist/types/src/behavior.d.ts.map +1 -0
  22. package/dist/types/src/class_list.d.ts +79 -0
  23. package/dist/types/src/class_list.d.ts.map +1 -0
  24. package/dist/types/src/component.d.ts +291 -0
  25. package/dist/types/src/component.d.ts.map +1 -0
  26. package/dist/types/src/component_params.d.ts +239 -0
  27. package/dist/types/src/component_params.d.ts.map +1 -0
  28. package/dist/types/src/component_space.d.ts +164 -0
  29. package/dist/types/src/component_space.d.ts.map +1 -0
  30. package/dist/types/src/data_path.d.ts +5 -0
  31. package/dist/types/src/data_path.d.ts.map +1 -0
  32. package/dist/types/src/data_proxy.d.ts +107 -0
  33. package/dist/types/src/data_proxy.d.ts.map +1 -0
  34. package/dist/types/src/data_utils.d.ts +3 -0
  35. package/dist/types/src/data_utils.d.ts.map +1 -0
  36. package/dist/types/src/element.d.ts +275 -0
  37. package/dist/types/src/element.d.ts.map +1 -0
  38. package/dist/types/src/element_iterator.d.ts +43 -0
  39. package/dist/types/src/element_iterator.d.ts.map +1 -0
  40. package/dist/types/src/event.d.ts +104 -0
  41. package/dist/types/src/event.d.ts.map +1 -0
  42. package/dist/types/src/external_shadow_tree.d.ts +20 -0
  43. package/dist/types/src/external_shadow_tree.d.ts.map +1 -0
  44. package/dist/types/src/func_arr.d.ts +39 -0
  45. package/dist/types/src/func_arr.d.ts.map +1 -0
  46. package/dist/types/src/global_options.d.ts +111 -0
  47. package/dist/types/src/global_options.d.ts.map +1 -0
  48. package/dist/types/src/index.d.ts +43 -0
  49. package/dist/types/src/index.d.ts.map +1 -0
  50. package/dist/types/src/mutation_observer.d.ts +79 -0
  51. package/dist/types/src/mutation_observer.d.ts.map +1 -0
  52. package/dist/types/src/native_node.d.ts +8 -0
  53. package/dist/types/src/native_node.d.ts.map +1 -0
  54. package/dist/types/src/node.d.ts +49 -0
  55. package/dist/types/src/node.d.ts.map +1 -0
  56. package/dist/types/src/relation.d.ts +47 -0
  57. package/dist/types/src/relation.d.ts.map +1 -0
  58. package/dist/types/src/render.d.ts +3 -0
  59. package/dist/types/src/render.d.ts.map +1 -0
  60. package/dist/types/src/selector.d.ts +32 -0
  61. package/dist/types/src/selector.d.ts.map +1 -0
  62. package/dist/types/src/shadow_root.d.ts +136 -0
  63. package/dist/types/src/shadow_root.d.ts.map +1 -0
  64. package/dist/types/src/template_engine.d.ts +18 -0
  65. package/dist/types/src/template_engine.d.ts.map +1 -0
  66. package/dist/types/src/text_node.d.ts +32 -0
  67. package/dist/types/src/text_node.d.ts.map +1 -0
  68. package/dist/types/src/tmpl/index.d.ts +18 -0
  69. package/dist/types/src/tmpl/index.d.ts.map +1 -0
  70. package/dist/types/src/tmpl/native_rendering.d.ts +45 -0
  71. package/dist/types/src/tmpl/native_rendering.d.ts.map +1 -0
  72. package/dist/types/src/tmpl/proc_gen_wrapper.d.ts +80 -0
  73. package/dist/types/src/tmpl/proc_gen_wrapper.d.ts.map +1 -0
  74. package/dist/types/src/tmpl/proc_gen_wrapper_dom.d.ts +50 -0
  75. package/dist/types/src/tmpl/proc_gen_wrapper_dom.d.ts.map +1 -0
  76. package/dist/types/src/tmpl/range_list_diff.d.ts +19 -0
  77. package/dist/types/src/tmpl/range_list_diff.d.ts.map +1 -0
  78. package/dist/types/src/trait_behaviors.d.ts +38 -0
  79. package/dist/types/src/trait_behaviors.d.ts.map +1 -0
  80. package/dist/types/src/virtual_node.d.ts +10 -0
  81. package/dist/types/src/virtual_node.d.ts.map +1 -0
  82. package/dist/types/tests/backend/domlike.test.d.ts +2 -0
  83. package/dist/types/tests/backend/domlike.test.d.ts.map +1 -0
  84. package/dist/types/tests/base/env.d.ts +29 -0
  85. package/dist/types/tests/base/env.d.ts.map +1 -0
  86. package/dist/types/tests/base/match.d.ts +9 -0
  87. package/dist/types/tests/base/match.d.ts.map +1 -0
  88. package/dist/types/tests/core/backend.test.d.ts +2 -0
  89. package/dist/types/tests/core/backend.test.d.ts.map +1 -0
  90. package/dist/types/tests/core/behavior.test.d.ts +2 -0
  91. package/dist/types/tests/core/behavior.test.d.ts.map +1 -0
  92. package/dist/types/tests/core/component_space.test.d.ts +2 -0
  93. package/dist/types/tests/core/component_space.test.d.ts.map +1 -0
  94. package/dist/types/tests/core/data_update.test.d.ts +2 -0
  95. package/dist/types/tests/core/data_update.test.d.ts.map +1 -0
  96. package/dist/types/tests/core/misc.test.d.ts +2 -0
  97. package/dist/types/tests/core/misc.test.d.ts.map +1 -0
  98. package/dist/types/tests/core/placeholder.test.d.ts +2 -0
  99. package/dist/types/tests/core/placeholder.test.d.ts.map +1 -0
  100. package/dist/types/tests/core/slot.test.d.ts +2 -0
  101. package/dist/types/tests/core/slot.test.d.ts.map +1 -0
  102. package/dist/types/tests/core/trait_behaviors.test.d.ts +2 -0
  103. package/dist/types/tests/core/trait_behaviors.test.d.ts.map +1 -0
  104. package/dist/types/tests/tmpl/binding_map.test.d.ts +2 -0
  105. package/dist/types/tests/tmpl/binding_map.test.d.ts.map +1 -0
  106. package/dist/types/tests/tmpl/event.test.d.ts +2 -0
  107. package/dist/types/tests/tmpl/event.test.d.ts.map +1 -0
  108. package/dist/types/tests/tmpl/expression.test.d.ts +2 -0
  109. package/dist/types/tests/tmpl/expression.test.d.ts.map +1 -0
  110. package/dist/types/tests/tmpl/lvalue.test.d.ts +2 -0
  111. package/dist/types/tests/tmpl/lvalue.test.d.ts.map +1 -0
  112. package/dist/types/tests/tmpl/native_rendering.test.d.ts +2 -0
  113. package/dist/types/tests/tmpl/native_rendering.test.d.ts.map +1 -0
  114. package/dist/types/tests/tmpl/structure.test.d.ts +2 -0
  115. package/dist/types/tests/tmpl/structure.test.d.ts.map +1 -0
  116. package/dist/types/tests/types/chaining.test.d.ts +2 -0
  117. package/dist/types/tests/types/chaining.test.d.ts.map +1 -0
  118. package/dist/types/tests/types/createElement.test.d.ts +2 -0
  119. package/dist/types/tests/types/createElement.test.d.ts.map +1 -0
  120. package/dist/types/tests/types/definition.test.d.ts +2 -0
  121. package/dist/types/tests/types/definition.test.d.ts.map +1 -0
  122. package/guide/zh_CN/advanced/binding_map_update.md +32 -0
  123. package/guide/zh_CN/advanced/build_args.md +28 -0
  124. package/guide/zh_CN/advanced/component_filter.md +70 -0
  125. package/guide/zh_CN/advanced/component_space.md +124 -0
  126. package/guide/zh_CN/advanced/custom_backend.md +53 -0
  127. package/guide/zh_CN/advanced/error_listener.md +32 -0
  128. package/guide/zh_CN/advanced/external_component.md +73 -0
  129. package/guide/zh_CN/advanced/template_engine.md +61 -0
  130. package/guide/zh_CN/appendix/backend_protocol.md +501 -0
  131. package/guide/zh_CN/appendix/list_diff_algorithm.md +406 -0
  132. package/guide/zh_CN/basic/beginning.md +94 -0
  133. package/guide/zh_CN/basic/component.md +156 -0
  134. package/guide/zh_CN/basic/event.md +169 -0
  135. package/guide/zh_CN/basic/lifetime.md +66 -0
  136. package/guide/zh_CN/basic/method.md +62 -0
  137. package/guide/zh_CN/basic/template.md +135 -0
  138. package/guide/zh_CN/data_management/advanced_update.md +170 -0
  139. package/guide/zh_CN/data_management/data_deep_copy.md +157 -0
  140. package/guide/zh_CN/data_management/data_observer.md +154 -0
  141. package/guide/zh_CN/data_management/property_early_init.md +31 -0
  142. package/guide/zh_CN/data_management/pure_data_pattern.md +21 -0
  143. package/guide/zh_CN/index.md +93 -0
  144. package/guide/zh_CN/interaction/behavior.md +52 -0
  145. package/guide/zh_CN/interaction/component_path.md +37 -0
  146. package/guide/zh_CN/interaction/generic.md +73 -0
  147. package/guide/zh_CN/interaction/placeholder.md +40 -0
  148. package/guide/zh_CN/interaction/relation.md +151 -0
  149. package/guide/zh_CN/interaction/slot.md +137 -0
  150. package/guide/zh_CN/interaction/template_import.md +94 -0
  151. package/guide/zh_CN/interaction/trait_behavior.md +117 -0
  152. package/guide/zh_CN/styling/external_class.md +46 -0
  153. package/guide/zh_CN/styling/style_isolation.md +54 -0
  154. package/guide/zh_CN/styling/virtual_host.md +52 -0
  155. package/guide/zh_CN/tree/element_iterator.md +54 -0
  156. package/guide/zh_CN/tree/mutation_observer.md +52 -0
  157. package/guide/zh_CN/tree/node_tree.md +142 -0
  158. package/guide/zh_CN/tree/node_tree_modification.md +78 -0
  159. package/guide/zh_CN/tree/selector.md +66 -0
  160. package/jest.config.js +6 -0
  161. package/jest.dts.config.js +9 -0
  162. package/jest.unit.config.js +14 -0
  163. package/package.json +28 -0
  164. package/src/backend/backend_protocol.ts +313 -0
  165. package/src/backend/composed_backend_protocol.ts +252 -0
  166. package/src/backend/domlike_backend_protocol.ts +370 -0
  167. package/src/backend/mode.ts +51 -0
  168. package/src/backend/suggested_backend_protocol.ts +83 -0
  169. package/src/behavior.ts +1655 -0
  170. package/src/bootstrap_dom_dev.js +22 -0
  171. package/src/class_list.ts +376 -0
  172. package/src/component.ts +1309 -0
  173. package/src/component_params.ts +461 -0
  174. package/src/component_space.ts +547 -0
  175. package/src/data_path.ts +225 -0
  176. package/src/data_proxy.ts +670 -0
  177. package/src/data_utils.ts +50 -0
  178. package/src/element.ts +1966 -0
  179. package/src/element_iterator.ts +158 -0
  180. package/src/event.ts +401 -0
  181. package/src/external_shadow_tree.ts +27 -0
  182. package/src/func_arr.ts +198 -0
  183. package/src/global_options.ts +242 -0
  184. package/src/index.ts +187 -0
  185. package/src/mutation_observer.ts +252 -0
  186. package/src/native_node.ts +74 -0
  187. package/src/node.ts +174 -0
  188. package/src/relation.ts +380 -0
  189. package/src/render.ts +25 -0
  190. package/src/selector.ts +218 -0
  191. package/src/shadow_root.ts +766 -0
  192. package/src/template_engine.ts +45 -0
  193. package/src/text_node.ts +149 -0
  194. package/src/tmpl/index.ts +199 -0
  195. package/src/tmpl/native_rendering.ts +175 -0
  196. package/src/tmpl/proc_gen_wrapper.ts +954 -0
  197. package/src/tmpl/proc_gen_wrapper_dom.ts +230 -0
  198. package/src/tmpl/range_list_diff.ts +443 -0
  199. package/src/trait_behaviors.ts +51 -0
  200. package/src/virtual_node.ts +51 -0
  201. package/tests/backend/domlike.test.ts +254 -0
  202. package/tests/base/env.ts +78 -0
  203. package/tests/base/match.ts +185 -0
  204. package/tests/core/backend.test.ts +144 -0
  205. package/tests/core/behavior.test.ts +546 -0
  206. package/tests/core/component_space.test.ts +212 -0
  207. package/tests/core/data_update.test.ts +461 -0
  208. package/tests/core/misc.test.ts +339 -0
  209. package/tests/core/placeholder.test.ts +180 -0
  210. package/tests/core/slot.test.ts +1495 -0
  211. package/tests/core/trait_behaviors.test.ts +153 -0
  212. package/tests/legacy/README.md +3 -0
  213. package/tests/legacy/behavior.test.js +293 -0
  214. package/tests/legacy/component.test.js +1247 -0
  215. package/tests/legacy/data_path.test.js +149 -0
  216. package/tests/legacy/data_proxy.test.js +759 -0
  217. package/tests/legacy/element_iterator.test.js +148 -0
  218. package/tests/legacy/event.test.js +849 -0
  219. package/tests/legacy/external.test.js +510 -0
  220. package/tests/legacy/extra_info.test.js +109 -0
  221. package/tests/legacy/generics.test.js +176 -0
  222. package/tests/legacy/mutation_observer.test.js +210 -0
  223. package/tests/legacy/relation.test.js +517 -0
  224. package/tests/legacy/selector.test.js +263 -0
  225. package/tests/legacy/slot.test.js +915 -0
  226. package/tests/legacy/virtual.test.js +394 -0
  227. package/tests/tmpl/binding_map.test.ts +208 -0
  228. package/tests/tmpl/event.test.ts +206 -0
  229. package/tests/tmpl/expression.test.ts +429 -0
  230. package/tests/tmpl/lvalue.test.ts +160 -0
  231. package/tests/tmpl/native_rendering.test.ts +155 -0
  232. package/tests/tmpl/structure.test.ts +998 -0
  233. package/tests/types/chaining.test.ts +614 -0
  234. package/tests/types/createElement.test.ts +82 -0
  235. package/tests/types/definition.test.ts +442 -0
  236. package/tsconfig.json +11 -0
  237. package/webpack.config.js +270 -0
@@ -0,0 +1,339 @@
1
+ import {
2
+ tmpl,
3
+ domBackend,
4
+ } from '../base/env'
5
+ import * as glassEasel from '../../src'
6
+
7
+ const componentSpace = new glassEasel.ComponentSpace()
8
+ componentSpace.updateComponentOptions({
9
+ writeFieldsToNode: true,
10
+ writeIdToDOM: true,
11
+ })
12
+ componentSpace.defineComponent({
13
+ is: '',
14
+ })
15
+
16
+ describe('dump element', () => {
17
+ test('dump node structure (shadow)', () => {
18
+ const compDef = componentSpace.defineComponent({
19
+ template: tmpl('<block wx:if="1"><div id="a" class="a" style="color: red">A</div></block>'),
20
+ })
21
+ const elem = glassEasel.Component.createWithContext('root', compDef.general(), domBackend)
22
+ const shadowDump = glassEasel.dumpElementToString(elem.shadowRoot, false)
23
+ expect(shadowDump).toBe([
24
+ '<(virtual):shadow>',
25
+ ' <(virtual):wx:if>',
26
+ ' <div id="a" class="a" style="color: red">',
27
+ ' A',
28
+ ].join('\n'))
29
+ })
30
+
31
+ test('dump node structure (composed)', () => {
32
+ componentSpace.defineComponent({
33
+ is: 'comp-a',
34
+ options: { multipleSlots: true },
35
+ properties: {
36
+ prop: String,
37
+ },
38
+ template: tmpl('<block wx:if="1"> <slot name="a" /> </block>'),
39
+ })
40
+ const compDef = componentSpace.defineComponent({
41
+ is: 'comp-b',
42
+ template: tmpl('<comp-a prop="p"><div attr="c" slot="a">A</div></comp-a>'),
43
+ })
44
+ const elem = glassEasel.Component.createWithContext('root', compDef.general(), domBackend)
45
+ const composedDump = glassEasel.dumpElementToString(elem, true)
46
+ expect(composedDump).toBe([
47
+ '<root:comp-b>',
48
+ ' <(virtual):shadow>',
49
+ ' <comp-a:comp-a prop="p">',
50
+ ' <(virtual):shadow>',
51
+ ' <(virtual):wx:if>',
52
+ ' <(virtual):slot (slot) name="a">',
53
+ ' <div slot="a" attr="c">',
54
+ ' A',
55
+ ].join('\n'))
56
+ })
57
+
58
+ test('dump node structure (external component)', () => {
59
+ const def = componentSpace.defineComponent({
60
+ options: { externalComponent: true },
61
+ properties: {
62
+ prop: String,
63
+ },
64
+ template: tmpl('<div />'),
65
+ })
66
+ const elem = glassEasel.Component.createWithContext('root', def.general(), domBackend)
67
+ const composedDump = glassEasel.dumpElementToString(elem, true)
68
+ expect(composedDump).toBe([
69
+ '<root: prop="">',
70
+ ' <(external)>',
71
+ ].join('\n'))
72
+ })
73
+
74
+ test('dump invalid node', () => {
75
+ expect(glassEasel.dumpElementToString(null, false)).toBe('<(null)>')
76
+ expect(glassEasel.dumpElementToString(undefined, false)).toBe('<(undefined)>')
77
+ expect(glassEasel.dumpElementToString(NaN, false)).toBe('<(unknown)>')
78
+ })
79
+ })
80
+
81
+ describe('deep copy', () => {
82
+ test('simple deep copy', () => {
83
+ const data = {
84
+ a: [1, 'a'],
85
+ b: {
86
+ b1: true,
87
+ },
88
+ s: Symbol('syn'),
89
+ }
90
+ const copied = glassEasel.dataUtils.deepCopy(data, false)
91
+ expect(copied.a).toStrictEqual(data.a)
92
+ expect(copied.a !== data.a).toBe(true)
93
+ expect(copied.b).toStrictEqual(data.b)
94
+ expect(copied.b !== data.b).toBe(true)
95
+ expect(copied.s.description).toBe(data.s.description)
96
+ expect(copied.s !== data.s).toBe(true)
97
+ })
98
+
99
+ test('deep copy with object recursion', () => {
100
+ const data = {
101
+ a: {
102
+ b: null,
103
+ c: 123,
104
+ },
105
+ }
106
+ data.a.b = data.a as unknown as null
107
+ const copied = glassEasel.dataUtils.deepCopy(data, true)
108
+ expect(copied.a.c).toBe(data.a.c)
109
+ expect(copied.a !== data.a).toBe(true)
110
+ expect(copied.a.b !== data.a.b).toBe(true)
111
+ expect(copied.a.b).toBe(copied.a)
112
+ })
113
+
114
+ test('deep copy with array recursion', () => {
115
+ const data = {
116
+ a: [null, Symbol('syn')],
117
+ }
118
+ data.a[0] = data.a as unknown as null
119
+ const copied = glassEasel.dataUtils.deepCopy(data, true)
120
+ expect(copied.a[1]!.description).toBe(data.a[1]!.description)
121
+ expect(copied.a !== data.a).toBe(true)
122
+ expect(copied.a[0] !== data.a[0]).toBe(true)
123
+ expect(copied.a[0]).toBe(copied.a)
124
+ })
125
+ })
126
+
127
+ describe('event', () => {
128
+ test('read event status', () => {
129
+ const event = new glassEasel.Event('test', {})
130
+ expect(event.defaultPrevented()).toBe(false)
131
+ expect(event.getEventBubbleStatus()).toBe(glassEasel.EventBubbleStatus.Normal)
132
+ event.preventDefault()
133
+ expect(event.defaultPrevented()).toBe(true)
134
+ expect(event.getEventBubbleStatus()).toBe(glassEasel.EventBubbleStatus.NoDefault)
135
+ })
136
+
137
+ test('legacy event binding syntax', () => {
138
+ const eventArr: number[] = []
139
+ const childCompDef = glassEasel.Component.register({
140
+ lifetimes: {
141
+ attached() {
142
+ this.triggerEvent('abc', 123, { capturePhase: true })
143
+ this.triggerEvent('abc', 456)
144
+ },
145
+ },
146
+ }, componentSpace)
147
+ const compDef = glassEasel.Component.register({
148
+ using: {
149
+ child: childCompDef,
150
+ },
151
+ template: tmpl(`
152
+ <child bindabc="handler" capture-bindabc="handler" catchabc="handler" capture-catchabc="handler" onabc="handler" />
153
+ `),
154
+ methods: {
155
+ handler(ev: glassEasel.ShadowedEvent<number>) {
156
+ eventArr.push(ev.detail)
157
+ },
158
+ },
159
+ }, componentSpace)
160
+ const comp = glassEasel.createElement('root', compDef)
161
+ glassEasel.Element.pretendAttached(comp)
162
+ expect(eventArr).toEqual([123, 123, 456, 456, 456])
163
+ })
164
+
165
+ test('listener change lifetimes', () => {
166
+ const eventArr: [boolean, string, any][] = []
167
+ const childCompDef = glassEasel.Component.register({
168
+ options: {
169
+ listenerChangeLifetimes: true,
170
+ },
171
+ lifetimes: {
172
+ listenerChanged: (
173
+ isAdd: boolean,
174
+ eventName: string,
175
+ listener: any,
176
+ options: EventListenerOptions,
177
+ ) => {
178
+ eventArr.push([isAdd, eventName, listener])
179
+ expect(options.capture).toBe(true)
180
+ },
181
+ },
182
+ }, componentSpace)
183
+ const compDef = glassEasel.Component.register({
184
+ using: {
185
+ child: childCompDef,
186
+ },
187
+ template: tmpl(`
188
+ <child id="a" />
189
+ `),
190
+ }, componentSpace)
191
+ const comp = glassEasel.createElement('root', compDef)
192
+ const child = comp.getShadowRoot()!.getElementById('a')!.asInstanceOf(childCompDef)!
193
+ const listener = () => { /* empty */ }
194
+ child.addListener('testEv', listener, { capture: true })
195
+ child.removeListener('testEv', listener, { capture: true })
196
+ child.removeListener('testEv', listener, { capture: true })
197
+ child.removeListener('testEv2', listener, { capture: true })
198
+ expect(eventArr).toEqual([[true, 'testEv', listener], [false, 'testEv', listener]])
199
+ })
200
+ })
201
+
202
+ describe('component utils', () => {
203
+ test('#getMethodsFromDef #getMethod', () => {
204
+ const compDef = glassEasel.Component.register({
205
+ methods: {
206
+ abc() {
207
+ return 'abc'
208
+ },
209
+ },
210
+ }, componentSpace)
211
+ expect(compDef.isPrepared()).toBe(false)
212
+ compDef.prepare()
213
+ expect(compDef.isPrepared()).toBe(true)
214
+ expect(glassEasel.Component.getMethodsFromDef(compDef.general()).abc!())
215
+ .toBe('abc')
216
+ const comp = glassEasel.createElement('root', compDef.general())
217
+ expect(glassEasel.Component.getMethod(comp.general(), 'abc')!())
218
+ .toBe('abc')
219
+ })
220
+
221
+ test('#isInnerDataExcluded', () => {
222
+ const compDef = glassEasel.Component.register({
223
+ options: {
224
+ pureDataPattern: /^_/,
225
+ },
226
+ }, componentSpace)
227
+ const comp = glassEasel.createElement('root', compDef.general())
228
+ expect(comp.isInnerDataExcluded('_a')).toBe(true)
229
+ expect(comp.isInnerDataExcluded('a')).toBe(false)
230
+ })
231
+
232
+ test('#getInnerData', () => {
233
+ const compDef = glassEasel.Component.register({
234
+ options: {
235
+ dataDeepCopy: glassEasel.DeepCopyKind.Simple,
236
+ },
237
+ data: {
238
+ a: 123,
239
+ },
240
+ }, componentSpace)
241
+ const comp = glassEasel.createElement('root', compDef.general())
242
+ expect(glassEasel.Component.getInnerData(comp.general())).toStrictEqual(comp.data)
243
+ })
244
+
245
+ test('#getInnerData', () => {
246
+ const compDef = glassEasel.Component.register({
247
+ options: {
248
+ dataDeepCopy: glassEasel.DeepCopyKind.Simple,
249
+ },
250
+ data: {
251
+ a: 123,
252
+ },
253
+ }, componentSpace)
254
+ const comp = glassEasel.createElement('root', compDef.general())
255
+ const oldData = comp.data
256
+ glassEasel.Component.replaceWholeData(comp.general(), { b: 456 })
257
+ expect(oldData).toStrictEqual({ a: 123 })
258
+ expect(comp.data).toStrictEqual({ b: 456 })
259
+ })
260
+
261
+ test('getComponentDependencies', () => {
262
+ const def0 = componentSpace.defineComponent({ is: 'common/def0', using: { p: '/common/def1' } })
263
+ const def1 = componentSpace.defineComponent({ is: 'common/def1', using: { p: '/common/def0' } })
264
+ const def2 = componentSpace.defineComponent({ is: 'common/def2', using: { p: './none' }, placeholders: { p: './def1' } })
265
+ const def3 = componentSpace.defineComponent({ is: 'def3', using: { p: 'common/def1' } })
266
+ const def4 = componentSpace.defineComponent({ is: 'def4', using: { 'p-a': '/common/def2', 'p-b': def3 } })
267
+ def3.prepare()
268
+ const ret = def4.getComponentDependencies()
269
+ expect(ret).toContain(def0)
270
+ expect(ret).toContain(def1)
271
+ expect(ret).toContain(def2)
272
+ expect(ret).toContain(def3)
273
+ expect(ret.size).toBe(4)
274
+ })
275
+
276
+ test('disallowNativeNode', () => {
277
+ const template = Object.assign(tmpl(`
278
+ <div />
279
+ `), { disallowNativeNode: true })
280
+ const compDef = componentSpace.defineComponent({ template })
281
+ const elem = glassEasel.createElement('root', compDef.general())
282
+ expect(elem.getShadowRoot()!.childNodes[0]).toBeInstanceOf(glassEasel.Component)
283
+ })
284
+
285
+ test('propertyEarlyInit', () => {
286
+ const callOrder: number[] = []
287
+ const lateInit = componentSpace.define()
288
+ .options({ propertyEarlyInit: false })
289
+ .property('a', Boolean)
290
+ .template(tmpl(`
291
+ <div wx:if="{{a}}" id="a">{{a}}</div>
292
+ `))
293
+ .observer('a', function () {
294
+ callOrder.push(1)
295
+ expect(this.getShadowRoot()!.getElementById('a')).toBe(undefined)
296
+ })
297
+ .lifetime('created', function () {
298
+ callOrder.push(2)
299
+ expect(this.getShadowRoot()!.getElementById('a')).toBe(undefined)
300
+ })
301
+ .lifetime('attached', function () {
302
+ callOrder.push(3)
303
+ expect(this.getShadowRoot()!.getElementById('a')).toBeInstanceOf(glassEasel.NativeNode)
304
+ })
305
+ .registerComponent()
306
+ const earlyInit = componentSpace.define()
307
+ .options({ propertyEarlyInit: true })
308
+ .property('a', Boolean)
309
+ .template(tmpl(`
310
+ <div wx:if="{{a}}" id="a">{{a}}</div>
311
+ `))
312
+ .observer('a', function () {
313
+ callOrder.push(4)
314
+ expect(this.getShadowRoot()!.getElementById('a')).toBe(undefined)
315
+ })
316
+ .lifetime('created', function () {
317
+ callOrder.push(5)
318
+ expect(this.getShadowRoot()!.getElementById('a')).toBeInstanceOf(glassEasel.NativeNode)
319
+ })
320
+ .lifetime('attached', function () {
321
+ callOrder.push(6)
322
+ expect(this.getShadowRoot()!.getElementById('a')).toBeInstanceOf(glassEasel.NativeNode)
323
+ })
324
+ .registerComponent()
325
+ const compDef = componentSpace.define()
326
+ .usingComponents({
327
+ 'early-init': earlyInit.general(),
328
+ 'late-init': lateInit.general(),
329
+ })
330
+ .template(tmpl(`
331
+ <late-init a="{{true}}" />
332
+ <early-init a="{{true}}" />
333
+ `))
334
+ .registerComponent()
335
+ const elem = glassEasel.createElement('root', compDef.general())
336
+ glassEasel.Element.pretendAttached(elem)
337
+ expect(callOrder).toStrictEqual([2, 1, 4, 5, 3, 6])
338
+ })
339
+ })
@@ -0,0 +1,180 @@
1
+ import { tmpl, domBackend } from '../base/env'
2
+ import * as glassEasel from '../../src'
3
+ import { virtual as matchElementWithDom } from '../base/match'
4
+
5
+ const componentSpace = new glassEasel.ComponentSpace()
6
+ componentSpace.updateComponentOptions({
7
+ writeFieldsToNode: true,
8
+ writeIdToDOM: true,
9
+ })
10
+ componentSpace.defineComponent({
11
+ is: '',
12
+ })
13
+
14
+ const domHtml = (elem: glassEasel.Element): string => {
15
+ const domElem = elem.getBackendElement() as unknown as Element
16
+ return domElem.innerHTML
17
+ }
18
+
19
+ describe('placeholder', () => {
20
+ test('using simple placeholder and waiting', () => {
21
+ const def = componentSpace.define()
22
+ .placeholders({
23
+ child: '',
24
+ })
25
+ .definition({
26
+ using: {
27
+ child: 'placeholder/simple/child',
28
+ 'child-another': 'placeholder/simple/child',
29
+ },
30
+ template: tmpl(`
31
+ <div>
32
+ <child>
33
+ <span />
34
+ </child>
35
+ <child-another wx:if="{{b}}" id="b" />
36
+ </div>
37
+ `),
38
+ })
39
+ .registerComponent()
40
+ const elem = glassEasel.Component.createWithContext('root', def.general(), domBackend)
41
+ expect(domHtml(elem)).toBe('<div><child><span></span></child></div>')
42
+ matchElementWithDom(elem)
43
+
44
+ componentSpace.defineComponent({
45
+ is: 'placeholder/simple/child',
46
+ template: tmpl('child<div><slot/></div>'),
47
+ })
48
+ expect(domHtml(elem)).toBe('<div><child>child<div><span></span></div></child></div>')
49
+ matchElementWithDom(elem)
50
+
51
+ elem.setData({
52
+ b: true,
53
+ })
54
+ expect(domHtml(elem)).toBe('<div><child>child<div><span></span></div></child><child-another id="b"></child-another></div>')
55
+ matchElementWithDom(elem)
56
+ })
57
+
58
+ test('using placeholder across component spaces and waiting', () => {
59
+ const mainCs = new glassEasel.ComponentSpace()
60
+ mainCs.defineComponent({ is: '' })
61
+ const extraCs = new glassEasel.ComponentSpace()
62
+ mainCs.importSpace('space://extra', extraCs, false)
63
+ mainCs.importSpace('space-private://extra', extraCs, true)
64
+
65
+ const def = mainCs.defineComponent({
66
+ using: {
67
+ child: 'space://extra/child',
68
+ 'child-private': 'space-private://extra/child',
69
+ },
70
+ placeholders: {
71
+ child: '',
72
+ 'child-private': '',
73
+ },
74
+ template: tmpl(`
75
+ <child />
76
+ <child-private />
77
+ `),
78
+ })
79
+ const elem = glassEasel.Component.createWithContext('root', def.general(), domBackend)
80
+ expect(domHtml(elem)).toBe('<child></child><child-private></child-private>')
81
+ matchElementWithDom(elem)
82
+
83
+ extraCs.defineComponent({
84
+ is: 'child',
85
+ template: tmpl('A'),
86
+ })
87
+ expect(domHtml(elem)).toBe('<child></child><child-private>A</child-private>')
88
+ matchElementWithDom(elem)
89
+
90
+ extraCs.exportComponent('child', 'child')
91
+ expect(domHtml(elem)).toBe('<child>A</child><child-private>A</child-private>')
92
+ matchElementWithDom(elem)
93
+ })
94
+
95
+ test('trigger lifetimes when replacing', () => {
96
+ const callOrder: number[] = []
97
+ const placeholder = componentSpace.defineComponent({
98
+ properties: {
99
+ n: Number,
100
+ },
101
+ template: tmpl('<span>{{n + 1}}<slot/></span>'),
102
+ lifetimes: {
103
+ created() {
104
+ callOrder.push(1)
105
+ },
106
+ attached() {
107
+ callOrder.push(2)
108
+ },
109
+ detached() {
110
+ callOrder.push(3)
111
+ },
112
+ moved() {
113
+ callOrder.push(0)
114
+ },
115
+ },
116
+ })
117
+ componentSpace.defineComponent({
118
+ is: 'placeholder/lifetime/a',
119
+ lifetimes: {
120
+ attached() {
121
+ callOrder.push(7)
122
+ },
123
+ detached() {
124
+ callOrder.push(0)
125
+ },
126
+ moved() {
127
+ callOrder.push(8)
128
+ },
129
+ },
130
+ })
131
+ const def = componentSpace.defineComponent({
132
+ is: 'placeholder/lifetime/parent',
133
+ using: {
134
+ child: 'child',
135
+ a: '../lifetime/a',
136
+ placeholder: placeholder.general(),
137
+ },
138
+ placeholders: {
139
+ child: 'placeholder',
140
+ },
141
+ template: tmpl(`
142
+ <child n="2">
143
+ <a />
144
+ </child>
145
+ `),
146
+ })
147
+ const elem = glassEasel.Component.createWithContext('root', def.general(), domBackend)
148
+ expect(callOrder).toStrictEqual([1])
149
+ glassEasel.Element.pretendAttached(elem)
150
+ expect(callOrder).toStrictEqual([1, 2, 7])
151
+ expect(domHtml(elem)).toBe('<child><span>3<a></a></span></child>')
152
+ matchElementWithDom(elem)
153
+ callOrder.splice(0, 99)
154
+
155
+ componentSpace.defineComponent({
156
+ is: 'placeholder/lifetime/child',
157
+ properties: {
158
+ n: String,
159
+ },
160
+ template: tmpl('<div>{{n + 1}}</div>'),
161
+ lifetimes: {
162
+ created() {
163
+ callOrder.push(4)
164
+ },
165
+ attached() {
166
+ callOrder.push(5)
167
+ },
168
+ detached() {
169
+ callOrder.push(0)
170
+ },
171
+ moved() {
172
+ callOrder.push(6)
173
+ },
174
+ },
175
+ })
176
+ expect(callOrder).toStrictEqual([4, 3, 5, 8])
177
+ expect(domHtml(elem)).toBe('<child><div>21</div></child>')
178
+ matchElementWithDom(elem)
179
+ })
180
+ })