Skip to content

Commit 9d7021f

Browse files
authoredJun 21, 2024··
fix(runtime): don't register listener before connected to DOM (#5844)
* fix(runtime): don't register listener before connected to DOM * prettier * register host listeners in connectedCallback * consolify * don't register twice * prettier * remove unnecessary import * more explicit tests
1 parent a2fa93b commit 9d7021f

File tree

7 files changed

+55
-8
lines changed

7 files changed

+55
-8
lines changed
 

‎src/client/client-host-ref.ts

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

43
import type * as d from '../declarations';
54

@@ -68,7 +67,6 @@ export const registerHost = (hostElement: d.HostElement, cmpMeta: d.ComponentRun
6867
hostElement['s-p'] = [];
6968
hostElement['s-rc'] = [];
7069
}
71-
addHostEventListeners(hostElement, hostRef, cmpMeta.$listeners$, false);
7270
return hostRefs.set(hostElement, hostRef);
7371
};
7472

‎src/hydrate/platform/hydrate-app.ts

+4-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import { globalScripts } from '@app-globals';
2-
import { doc, getHostRef, loadModule, plt, registerHost } from '@platform';
2+
import { addHostEventListeners, doc, getHostRef, loadModule, plt, registerHost } from '@platform';
33
import { connectedCallback, insertVdomAnnotations } from '@runtime';
44

55
import type * as d from '../../declarations';
@@ -183,6 +183,9 @@ async function hydrateComponent(
183183

184184
if (cmpMeta != null) {
185185
waitingElements.add(elm);
186+
const hostRef = getHostRef(this);
187+
addHostEventListeners(this, hostRef, cmpMeta.$listeners$, false);
188+
186189
try {
187190
connectedCallback(elm);
188191
await elm.componentOnReady();

‎src/hydrate/platform/index.ts

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

43
import type * as d from '../../declarations';
54

@@ -146,7 +145,6 @@ export const registerHost = (elm: d.HostElement, cmpMeta: d.ComponentRuntimeMeta
146145
hostRef.$onReadyPromise$ = new Promise((r) => (hostRef.$onReadyResolve$ = r));
147146
elm['s-p'] = [];
148147
elm['s-rc'] = [];
149-
addHostEventListeners(elm, hostRef, cmpMeta.$listeners$, false);
150148
return hostRefs.set(elm, hostRef);
151149
};
152150

‎src/runtime/bootstrap-custom-element.ts

+4-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import { BUILD } from '@app-data';
2-
import { forceUpdate, getHostRef, registerHost, styles, supportsShadow } from '@platform';
2+
import { addHostEventListeners, forceUpdate, getHostRef, registerHost, styles, supportsShadow } from '@platform';
33
import { CMP_FLAGS } from '@utils';
44

55
import type * as d from '../declarations';
@@ -72,6 +72,9 @@ export const proxyCustomElement = (Cstr: any, compactMeta: d.ComponentRuntimeMet
7272
registerHost(this, cmpMeta);
7373
},
7474
connectedCallback() {
75+
const hostRef = getHostRef(this);
76+
addHostEventListeners(this, hostRef, cmpMeta.$listeners$, false);
77+
7578
connectedCallback(this);
7679
if (BUILD.connectedCallback && originalConnectedCallback) {
7780
originalConnectedCallback.call(this);

‎src/runtime/bootstrap-lazy.ts

+15
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import { BUILD } from '@app-data';
22
import { doc, getHostRef, plt, registerHost, supportsShadow, win } from '@platform';
3+
import { addHostEventListeners } from '@runtime';
34
import { CMP_FLAGS, queryNonceMetaTagContent } from '@utils';
45

56
import type * as d from '../declarations';
@@ -96,6 +97,7 @@ export const bootstrapLazy = (lazyBundles: d.LazyBundlesRuntimeData, options: d.
9697
const HostElement = class extends HTMLElement {
9798
['s-p']: Promise<void>[];
9899
['s-rc']: (() => void)[];
100+
hasRegisteredEventListeners = false;
99101

100102
// StencilLazyHost
101103
constructor(self: HTMLElement) {
@@ -138,6 +140,19 @@ export const bootstrapLazy = (lazyBundles: d.LazyBundlesRuntimeData, options: d.
138140
}
139141

140142
connectedCallback() {
143+
const hostRef = getHostRef(this);
144+
145+
/**
146+
* The `connectedCallback` lifecycle event can potentially be fired multiple times
147+
* if the element is removed from the DOM and re-inserted. This is not a common use case,
148+
* but it can happen in some scenarios. To prevent registering the same event listeners
149+
* multiple times, we will only register them once.
150+
*/
151+
if (!this.hasRegisteredEventListeners) {
152+
this.hasRegisteredEventListeners = true;
153+
addHostEventListeners(this, hostRef, cmpMeta.$listeners$, false);
154+
}
155+
141156
if (appLoadFallback) {
142157
clearTimeout(appLoadFallback);
143158
appLoadFallback = null;

‎src/runtime/test/listen.spec.tsx

+32
Original file line numberDiff line numberDiff line change
@@ -201,4 +201,36 @@ describe('listen', () => {
201201
await waitForChanges();
202202
expect(a).toEqualHtml(`<cmp-a>1 7</cmp-a>`);
203203
});
204+
205+
it('disconnects target listeners when element is not connected to DOM', async () => {
206+
let events = 0;
207+
@Component({ tag: 'cmp-a' })
208+
class CmpA {
209+
@Listen('testEvent', { target: 'document' })
210+
buttonClick() {
211+
events++;
212+
}
213+
214+
render() {
215+
return '';
216+
}
217+
}
218+
219+
const { doc, waitForChanges } = await newSpecPage({
220+
components: [CmpA],
221+
});
222+
223+
jest.spyOn(doc, 'addEventListener');
224+
jest.spyOn(doc, 'removeEventListener');
225+
226+
doc.createElement('cmp-a');
227+
await waitForChanges();
228+
229+
// Event listener will never be called
230+
expect(events).toEqual(0);
231+
232+
// no event listeners have been added as the element is not connected to the DOM
233+
expect(doc.addEventListener.mock.calls.length).toBe(0);
234+
expect(doc.removeEventListener.mock.calls.length).toBe(0);
235+
});
204236
});

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

-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,3 @@
1-
import { addHostEventListeners } from '@runtime';
21
import type * as d from '@stencil/core/internal';
32

43
import { hostRefs } from './testing-constants';
@@ -57,6 +56,5 @@ export const registerHost = (elm: d.HostElement, cmpMeta: d.ComponentRuntimeMeta
5756
hostRef.$onReadyPromise$ = new Promise((r) => (hostRef.$onReadyResolve$ = r));
5857
elm['s-p'] = [];
5958
elm['s-rc'] = [];
60-
addHostEventListeners(elm, hostRef, cmpMeta.$listeners$, false);
6159
hostRefs.set(elm, hostRef);
6260
};

0 commit comments

Comments
 (0)
Please sign in to comment.