Skip to content

Commit c4a9c9e

Browse files
fix(runtime): clear up rootAppliedStyles (#6087)
* fix(runtime): clear up detached hostRefs and rootAppliedStyles * fix build * prettier * fix after rebase * more reverts * more cleanups * remove hostrefs manually instead of cleanups * prettier * fix build * found the leak * found it without errors * delete the right element * clean up after animation frame * prettier * use isConnected to check whether node is in DOM * prettier * solve leak for dist custom elements * prettier * fix: PR comment * more rigurous cleanup * delete the lazy instance after a timeout to ensure that any pending state updates have been processed * prettier * fix unit test * chore(ci): trigger build * disable test * logging * tweak * tweak * clean up * minor comment update * more cleanups
1 parent 22684f3 commit c4a9c9e

10 files changed

+75
-12
lines changed

src/client/client-host-ref.ts

+9
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,15 @@ const hostRefs: WeakMap<d.RuntimeRef, d.HostRef> = /*@__PURE__*/ BUILD.hotModule
2222
? ((window as any).__STENCIL_HOSTREFS__ ||= new WeakMap())
2323
: new WeakMap();
2424

25+
/**
26+
* Given a {@link d.RuntimeRef} remove the corresponding {@link d.HostRef} from
27+
* the {@link hostRefs} WeakMap.
28+
*
29+
* @param ref the runtime ref of interest
30+
* @returns — true if the element was successfully removed, or false if it was not present.
31+
*/
32+
export const deleteHostRef = (ref: d.RuntimeRef) => hostRefs.delete(ref);
33+
2534
/**
2635
* Given a {@link d.RuntimeRef} retrieve the corresponding {@link d.HostRef}
2736
*

src/hydrate/platform/index.ts

+1
Original file line numberDiff line numberDiff line change
@@ -129,6 +129,7 @@ export const supportsConstructableStylesheets = false;
129129
const hostRefs: WeakMap<d.RuntimeRef, d.HostRef> = new WeakMap();
130130

131131
export const getHostRef = (ref: d.RuntimeRef) => hostRefs.get(ref);
132+
export const deleteHostRef = (ref: d.RuntimeRef) => hostRefs.delete(ref);
132133

133134
export const registerInstance = (lazyInstance: any, hostRef: d.HostRef) =>
134135
hostRefs.set((hostRef.$lazyInstance$ = lazyInstance), hostRef);

src/runtime/bootstrap-custom-element.ts

+24-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,14 @@
11
import { BUILD } from '@app-data';
2-
import { addHostEventListeners, forceUpdate, getHostRef, registerHost, styles, supportsShadow } from '@platform';
2+
import {
3+
addHostEventListeners,
4+
deleteHostRef,
5+
forceUpdate,
6+
getHostRef,
7+
plt,
8+
registerHost,
9+
styles,
10+
supportsShadow,
11+
} from '@platform';
312
import { CMP_FLAGS } from '@utils';
413

514
import type * as d from '../declarations';
@@ -89,6 +98,20 @@ export const proxyCustomElement = (Cstr: any, compactMeta: d.ComponentRuntimeMet
8998
if (BUILD.disconnectedCallback && originalDisconnectedCallback) {
9099
originalDisconnectedCallback.call(this);
91100
}
101+
102+
/**
103+
* Clean up Node references lingering around in `hostRef` objects
104+
* to ensure GC can clean up the memory.
105+
*/
106+
plt.raf(() => {
107+
const hostRef = getHostRef(this);
108+
if (hostRef?.$vnode$?.$elm$ instanceof Node && !hostRef.$vnode$.$elm$.isConnected) {
109+
delete hostRef.$vnode$;
110+
}
111+
if (this instanceof Node && !this.isConnected) {
112+
deleteHostRef(this);
113+
}
114+
});
92115
},
93116
__attachShadow() {
94117
if (supportsShadow) {

src/runtime/bootstrap-lazy.ts

+13
Original file line numberDiff line numberDiff line change
@@ -159,6 +159,19 @@ export const bootstrapLazy = (lazyBundles: d.LazyBundlesRuntimeData, options: d.
159159

160160
disconnectedCallback() {
161161
plt.jmp(() => disconnectedCallback(this));
162+
163+
/**
164+
* Clear up references within the `$vnode$` object to the DOM
165+
* node that was removed. This is necessary to ensure that these
166+
* references used as keys in the `hostRef` object can be properly
167+
* garbage collected.
168+
*/
169+
plt.raf(() => {
170+
const hostRef = getHostRef(this);
171+
if (hostRef?.$vnode$?.$elm$ instanceof Node && !hostRef.$vnode$.$elm$.isConnected) {
172+
delete hostRef.$vnode$.$elm$;
173+
}
174+
});
162175
}
163176

164177
componentOnReady() {

src/runtime/disconnected-callback.ts

+15
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@ import { getHostRef, plt } from '@platform';
33

44
import type * as d from '../declarations';
55
import { PLATFORM_FLAGS } from './runtime-constants';
6+
import { rootAppliedStyles } from './styles';
67
import { safeCall } from './update-component';
78

89
const disconnectInstance = (instance: any) => {
@@ -33,4 +34,18 @@ export const disconnectedCallback = async (elm: d.HostElement) => {
3334
hostRef.$onReadyPromise$.then(() => disconnectInstance(hostRef.$lazyInstance$));
3435
}
3536
}
37+
38+
/**
39+
* Remove the element from the `rootAppliedStyles` WeakMap
40+
*/
41+
if (rootAppliedStyles.has(elm)) {
42+
rootAppliedStyles.delete(elm);
43+
}
44+
45+
/**
46+
* Remove the shadow root from the `rootAppliedStyles` WeakMap
47+
*/
48+
if (elm.shadowRoot && rootAppliedStyles.has(elm.shadowRoot as unknown as Element)) {
49+
rootAppliedStyles.delete(elm.shadowRoot as unknown as Element);
50+
}
3651
};

src/runtime/styles.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@ import type * as d from '../declarations';
66
import { createTime } from './profile';
77
import { HYDRATED_STYLE_ID, NODE_TYPE, SLOT_FB_CSS } from './runtime-constants';
88

9-
const rootAppliedStyles: d.RootAppliedStyleMap = /*@__PURE__*/ new WeakMap();
9+
export const rootAppliedStyles: d.RootAppliedStyleMap = /*@__PURE__*/ new WeakMap();
1010

1111
/**
1212
* Register the styles for a component by creating a stylesheet and then

src/runtime/vdom/update-element.ts

+2-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,4 @@
11
import { BUILD } from '@app-data';
2-
import { EMPTY_OBJ } from '@utils';
32

43
import type * as d from '../../declarations';
54
import { NODE_TYPE } from '../runtime-constants';
@@ -24,8 +23,8 @@ export const updateElement = (oldVnode: d.VNode | null, newVnode: d.VNode, isSvg
2423
newVnode.$elm$.nodeType === NODE_TYPE.DocumentFragment && newVnode.$elm$.host
2524
? newVnode.$elm$.host
2625
: (newVnode.$elm$ as any);
27-
const oldVnodeAttrs = (oldVnode && oldVnode.$attrs$) || EMPTY_OBJ;
28-
const newVnodeAttrs = newVnode.$attrs$ || EMPTY_OBJ;
26+
const oldVnodeAttrs = (oldVnode && oldVnode.$attrs$) || {};
27+
const newVnodeAttrs = newVnode.$attrs$ || {};
2928

3029
if (BUILD.updatable) {
3130
// remove attributes no longer present on the vnode by setting them to undefined

src/testing/platform/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
export { Build } from './testing-build';
22
export { modeResolutionChain, styles } from './testing-constants';
3-
export { getHostRef, registerHost, registerInstance } from './testing-host-ref';
3+
export { deleteHostRef, getHostRef, registerHost, registerInstance } from './testing-host-ref';
44
export { consoleDevError, consoleDevInfo, consoleDevWarn, consoleError, setErrorHandler } from './testing-log';
55
export {
66
isMemberInElement,

src/testing/platform/testing-host-ref.ts

+9
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,15 @@ export const getHostRef = (elm: d.RuntimeRef | undefined): d.HostRef | undefined
1111
return hostRefs.get(elm);
1212
};
1313

14+
/**
15+
* Given a {@link d.RuntimeRef} remove the corresponding {@link d.HostRef} from
16+
* the {@link hostRefs} WeakMap.
17+
*
18+
* @param ref the runtime ref of interest
19+
* @returns — true if the element was successfully removed, or false if it was not present.
20+
*/
21+
export const deleteHostRef = (ref: d.RuntimeRef) => hostRefs.delete(ref);
22+
1423
/**
1524
* Add the provided `hostRef` instance to the global {@link hostRefs} map, using the provided `lazyInstance` as a key.
1625
* @param lazyInstance a Stencil component instance

src/utils/constants.ts

-6
Original file line numberDiff line numberDiff line change
@@ -120,12 +120,6 @@ export const enum CMP_FLAGS {
120120
*/
121121
export const DEFAULT_STYLE_MODE = '$';
122122

123-
/**
124-
* Reusable empty obj/array
125-
* Don't add values to these!!
126-
*/
127-
export const EMPTY_OBJ: Record<never, never> = {};
128-
129123
/**
130124
* Namespaces
131125
*/

0 commit comments

Comments
 (0)