-
Notifications
You must be signed in to change notification settings - Fork 92
/
virtual.ts
100 lines (81 loc) 路 2.8 KB
/
virtual.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
import { directive, DirectiveParameters, ChildPart, PartInfo } from 'lit/directive.js';
import { noChange } from 'lit';
import { AsyncDirective } from 'lit/async-directive.js';
import { GenericRenderer } from './core';
import { BaseScheduler } from './scheduler';
const includes = Array.prototype.includes;
interface Renderer extends GenericRenderer<ChildPart> {
(this: ChildPart, ...args: unknown[]): unknown | void;
}
const partToScheduler: WeakMap<ChildPart, Scheduler> = new WeakMap();
const schedulerToPart: WeakMap<Scheduler, ChildPart> = new WeakMap();
class Scheduler extends BaseScheduler<object, ChildPart, Renderer, ChildPart> {
args!: unknown[];
setValue: Function;
constructor(renderer: Renderer, part: ChildPart, setValue: Function) {
super(renderer, part);
this.state.virtual = true;
this.setValue = setValue;
}
render(): unknown {
return this.state.run(() => this.renderer.apply(this.host, this.args));
}
commit(result: unknown): void {
this.setValue(result);
}
teardown(): void {
super.teardown();
let part = schedulerToPart.get(this);
partToScheduler.delete(part!);
}
}
function makeVirtual() : any {
function virtual(renderer: Renderer) {
class VirtualDirective extends AsyncDirective {
cont: Scheduler | undefined;
constructor(partInfo: PartInfo) {
super(partInfo);
this.cont = undefined;
}
update(part: ChildPart, args: DirectiveParameters<this>) {
this.cont = partToScheduler.get(part);
if (!this.cont || this.cont.renderer !== renderer) {
this.cont = new Scheduler(renderer, part, (r: unknown) => {this.setValue(r)});
partToScheduler.set(part, this.cont);
schedulerToPart.set(this.cont, part);
teardownOnRemove(this.cont, part);
}
this.cont.args = args;
this.cont.update();
return this.render(args);
}
render(args: unknown) {
return noChange;
}
}
return directive(VirtualDirective);
}
return virtual;
}
function teardownOnRemove(cont: BaseScheduler<object, ChildPart, Renderer, ChildPart>, part: ChildPart, node = part.startNode): void {
let frag = node!.parentNode!;
let mo = new MutationObserver(mutations => {
for(let mutation of mutations) {
if(includes.call(mutation.removedNodes, node)) {
mo.disconnect();
if(node!.parentNode instanceof ShadowRoot) {
teardownOnRemove(cont, part);
} else {
cont.teardown();
}
break;
} else if(includes.call(mutation.addedNodes, node!.nextSibling)) {
mo.disconnect();
teardownOnRemove(cont, part, node!.nextSibling || undefined);
break;
}
}
});
mo.observe(frag, { childList: true });
}
export { makeVirtual, Renderer as VirtualRenderer };