|
1 | 1 | import { $$ } from 'basic-devtools';
|
2 | 2 |
|
3 |
| -import XWorker from './xworker.js'; |
4 | 3 | import { handle } from './script-handler.js';
|
5 |
| -import { assign } from './utils.js'; |
| 4 | +import { assign, defineProperty } from './utils.js'; |
6 | 5 | import { selectors, prefixes } from './interpreters.js';
|
7 | 6 | import { CUSTOM_SELECTORS, handleCustomType } from './custom.js';
|
8 | 7 | import { listener, addAllListeners } from './listeners.js';
|
9 | 8 |
|
10 |
| -export { define, whenDefined } from './custom.js'; |
11 |
| -export { env } from './listeners.js'; |
| 9 | +import { define as $define, whenDefined as $whenDefined } from './custom.js'; |
| 10 | +import { env as $env } from './listeners.js'; |
| 11 | +import { Hook as $Hook } from './worker/hooks.js'; |
| 12 | +import $XWorker from './xworker.js'; |
| 13 | + |
| 14 | +const polyscript = Symbol.for('polyscript'); |
| 15 | +const alreadyLive = polyscript in globalThis; |
| 16 | + |
| 17 | +// avoid multiple initialization of the same library |
| 18 | +/* c8 ignore start */ |
| 19 | +const { define, whenDefined, env, Hook, XWorker } = ( |
| 20 | + alreadyLive ? |
| 21 | + globalThis[polyscript] : |
| 22 | + defineProperty( |
| 23 | + globalThis, |
| 24 | + polyscript, |
| 25 | + { |
| 26 | + value: { |
| 27 | + define: $define, |
| 28 | + whenDefined: $whenDefined, |
| 29 | + env: $env, |
| 30 | + Hook: $Hook, |
| 31 | + XWorker: $XWorker |
| 32 | + } |
| 33 | + } |
| 34 | + )[polyscript] |
| 35 | +); |
| 36 | +/* c8 ignore stop */ |
| 37 | + |
| 38 | +export { define, whenDefined, env, Hook, XWorker }; |
12 | 39 | export * from './errors.js';
|
13 |
| -export { XWorker }; |
14 | 40 |
|
15 |
| -const mo = new MutationObserver((records) => { |
16 |
| - const selector = selectors.join(','); |
17 |
| - for (const { type, target, attributeName, addedNodes } of records) { |
18 |
| - // attributes are tested via integration / e2e |
19 |
| - /* c8 ignore start */ |
20 |
| - if (type === 'attributes') { |
21 |
| - const i = attributeName.lastIndexOf('-') + 1; |
22 |
| - if (i) { |
23 |
| - const prefix = attributeName.slice(0, i); |
24 |
| - for (const p of prefixes) { |
25 |
| - if (prefix === p) { |
26 |
| - const type = attributeName.slice(i); |
27 |
| - if (type !== 'env') { |
28 |
| - const method = target.hasAttribute(attributeName) |
29 |
| - ? 'add' |
30 |
| - : 'remove'; |
31 |
| - target[`${method}EventListener`](type, listener); |
| 41 | +if (!alreadyLive) { |
| 42 | + const mo = new MutationObserver((records) => { |
| 43 | + const selector = selectors.join(','); |
| 44 | + for (const { type, target, attributeName, addedNodes } of records) { |
| 45 | + // attributes are tested via integration / e2e |
| 46 | + /* c8 ignore start */ |
| 47 | + if (type === 'attributes') { |
| 48 | + const i = attributeName.lastIndexOf('-') + 1; |
| 49 | + if (i) { |
| 50 | + const prefix = attributeName.slice(0, i); |
| 51 | + for (const p of prefixes) { |
| 52 | + if (prefix === p) { |
| 53 | + const type = attributeName.slice(i); |
| 54 | + if (type !== 'env') { |
| 55 | + const method = target.hasAttribute(attributeName) |
| 56 | + ? 'add' |
| 57 | + : 'remove'; |
| 58 | + target[`${method}EventListener`](type, listener); |
| 59 | + } |
| 60 | + break; |
32 | 61 | }
|
33 |
| - break; |
34 | 62 | }
|
35 | 63 | }
|
| 64 | + continue; |
36 | 65 | }
|
37 |
| - continue; |
38 |
| - } |
39 |
| - for (const node of addedNodes) { |
40 |
| - if (node.nodeType === 1) { |
41 |
| - addAllListeners(node); |
42 |
| - if (selector && node.matches(selector)) handle(node); |
43 |
| - else bootstrap(selector, node, true); |
| 66 | + for (const node of addedNodes) { |
| 67 | + if (node.nodeType === 1) { |
| 68 | + addAllListeners(node); |
| 69 | + if (selector && node.matches(selector)) handle(node); |
| 70 | + else bootstrap(selector, node, true); |
| 71 | + } |
44 | 72 | }
|
| 73 | + /* c8 ignore stop */ |
45 | 74 | }
|
46 |
| - /* c8 ignore stop */ |
47 |
| - } |
48 |
| -}); |
| 75 | + }); |
49 | 76 |
|
50 |
| -/* c8 ignore start */ |
51 |
| -const bootstrap = (selector, node, shouldHandle) => { |
52 |
| - if (selector) $$(selector, node).forEach(handle); |
53 |
| - selector = CUSTOM_SELECTORS.join(','); |
54 |
| - if (selector) { |
55 |
| - if (shouldHandle) handleCustomType(node); |
56 |
| - $$(selector, node).forEach(handleCustomType); |
57 |
| - } |
58 |
| -}; |
59 |
| -/* c8 ignore stop */ |
| 77 | + /* c8 ignore start */ |
| 78 | + const bootstrap = (selector, node, shouldHandle) => { |
| 79 | + if (selector) $$(selector, node).forEach(handle); |
| 80 | + selector = CUSTOM_SELECTORS.join(','); |
| 81 | + if (selector) { |
| 82 | + if (shouldHandle) handleCustomType(node); |
| 83 | + $$(selector, node).forEach(handleCustomType); |
| 84 | + } |
| 85 | + }; |
| 86 | + /* c8 ignore stop */ |
| 87 | + |
| 88 | + const observe = (root) => { |
| 89 | + mo.observe(root, { childList: true, subtree: true, attributes: true }); |
| 90 | + return root; |
| 91 | + }; |
60 | 92 |
|
61 |
| -const observe = (root) => { |
62 |
| - mo.observe(root, { childList: true, subtree: true, attributes: true }); |
63 |
| - return root; |
64 |
| -}; |
| 93 | + const { attachShadow } = Element.prototype; |
| 94 | + assign(Element.prototype, { |
| 95 | + attachShadow(init) { |
| 96 | + return observe(attachShadow.call(this, init)); |
| 97 | + }, |
| 98 | + }); |
65 | 99 |
|
66 |
| -const { attachShadow } = Element.prototype; |
67 |
| -assign(Element.prototype, { |
68 |
| - attachShadow(init) { |
69 |
| - return observe(attachShadow.call(this, init)); |
70 |
| - }, |
71 |
| -}); |
| 100 | + // give 3rd party a chance to apply changes before this happens |
| 101 | + queueMicrotask(() => { |
| 102 | + addAllListeners(observe(document)); |
| 103 | + bootstrap(selectors.join(','), document, false); |
| 104 | + }); |
72 | 105 |
|
73 |
| -// give 3rd party a chance to apply changes before this happens |
74 |
| -queueMicrotask(() => { |
75 |
| - addAllListeners(observe(document)); |
76 |
| - bootstrap(selectors.join(','), document, false); |
77 |
| -}); |
| 106 | +} |
0 commit comments