Bug Report
π Search Terms
- liskov
- addEventListener
- overload
π Version & Regression Information
- This is the behavior in every version I tried, and I reviewed the FAQ for entries about everything. I've checked the full FAQ.
β― Playground Link
Playground link with relevant code
π» Code
interface FooBarEventMap {
click: CustomEvent;
}
class FooBarElement extends HTMLElement {
}
interface FooBarElement extends HTMLElement {
addEventListener<T extends keyof FooBarEventMap>(
type: T,
listener: (this: FooBarElement, ev: FooBarEventMap[T]) => any,
options?: boolean | AddEventListenerOptions,
): void;
}
interface HTMLElementTagNameMap {
"foo-bar": FooBarElement;
}
let one: NodeListOf<Node> = document.querySelectorAll('foo-bar')!;
one.forEach(el => el.addEventListener('click', null));
let two: NodeListOf<Node> = document.querySelectorAll('div')!;
two.forEach(el => el.addEventListener('click', null));
let three: NodeListOf<FooBarElement> = document.querySelectorAll('foo-bar')!;
three.forEach(el => el.addEventListener('click', null));
let four: NodeListOf<HTMLDivElement> = document.querySelectorAll('div')!;
four.forEach(el => el.addEventListener('click', null));
π Actual behavior
An error is reported for the assignment to one.
This is technically correct: FooBarElement is a subtype of HTMLElement which in turn is a subtype of Node. FooBarElement.addEventListener() does not accept all parameters that are accepted by Node.addEventListener(), making this a LSP violation.
However the same is true for HTMLDivElement. The assignment to two is accepted, but as evidenced by the assignment to four, an HTMLDivElement does not actually accept null.
When I duplicate the addEventListener overload to:
interface FooBarElement extends HTMLElement {
addEventListener<T extends keyof FooBarEventMap>(
type: T,
listener: (this: FooBarElement, ev: FooBarEventMap[T]) => any,
options?: boolean | AddEventListenerOptions,
): void;
addEventListener<T extends keyof FooBarEventMap>(
type: T,
listener: (this: FooBarElement, ev: FooBarEventMap[T]) => any,
options?: boolean | AddEventListenerOptions,
): void;
}
β¦ both overloads are 100% identical. Then the assignment to one is no longer reported as an error.
π Expected behavior
I would not expect this inconsistent behavior:
- Either both HTMLDivElement and FooBarElement should report an error when assigning to a
NodeListOf<Node>, or neither should.
- Adding an additional identical overload should not make the error go away.
Additional context
The real world use case is consuming the WoltLab/d.ts repository with a tsconfig.json that has strict: true configured.
Within that repository there is an element with an incompatible overload in:
https://github.com/WoltLab/d.ts/blob/57f923f03e37885885326bbd622d15b554feb9b5/WoltLabSuite/Core/Element/woltlab-core-dialog.d.ts#L40
and it is registered in global.d.ts in:
https://github.com/WoltLab/d.ts/blob/57f923f03e37885885326bbd622d15b554feb9b5/global.d.ts#L122
Now when attempting to compile a project that has the repository as a dependency, the following error is reported:
node_modules/typescript/lib/lib.dom.d.ts:10535:87 - error TS2344: Type 'HTMLElementTagNameMap[K]' does not satisfy the constraint 'Node'.
Type 'HTMLInputElement | HTMLElement | WoltlabCoreDialogElement | HTMLDialogElement | WoltlabCoreDialogControlElement | ... 66 more ... | HTMLVideoElement' is not assignable to type 'Node'.
Type 'WoltlabCoreDialogElement' is not assignable to type 'Node'.
Types of property 'addEventListener' are incompatible.
Type '<T extends keyof WoltlabCoreDialogEventMap>(type: T, listener: (this: WoltlabCoreDialogElement, ev: WoltlabCoreDialogEventMap[T]) => any, options?: boolean | ... 1 more ... | undefined) => void' is not assignable to type '(type: string, callback: EventListenerOrEventListenerObject | null, options?: boolean | AddEventListenerOptions | undefined) => void'.
Types of parameters 'listener' and 'callback' are incompatible.
Type 'EventListenerOrEventListenerObject | null' is not assignable to type '(this: WoltlabCoreDialogElement, ev: CustomEvent<any> | CustomEvent<ValidateCallback[]>) => any'.
Type 'null' is not assignable to type '(this: WoltlabCoreDialogElement, ev: CustomEvent<any> | CustomEvent<ValidateCallback[]>) => any'.
10535 querySelectorAll<K extends keyof HTMLElementTagNameMap>(selectors: K): NodeListOf<HTMLElementTagNameMap[K]>;
Bug Report
π Search Terms
π Version & Regression Information
β― Playground Link
Playground link with relevant code
π» Code
π Actual behavior
An error is reported for the assignment to
one.This is technically correct:
FooBarElementis a subtype ofHTMLElementwhich in turn is a subtype ofNode.FooBarElement.addEventListener()does not accept all parameters that are accepted byNode.addEventListener(), making this a LSP violation.However the same is true for
HTMLDivElement. The assignment totwois accepted, but as evidenced by the assignment tofour, anHTMLDivElementdoes not actually acceptnull.When I duplicate the
addEventListeneroverload to:β¦ both overloads are 100% identical. Then the assignment to
oneis no longer reported as an error.π Expected behavior
I would not expect this inconsistent behavior:
NodeListOf<Node>, or neither should.Additional context
The real world use case is consuming the WoltLab/d.ts repository with a tsconfig.json that has
strict: trueconfigured.Within that repository there is an element with an incompatible overload in:
https://github.com/WoltLab/d.ts/blob/57f923f03e37885885326bbd622d15b554feb9b5/WoltLabSuite/Core/Element/woltlab-core-dialog.d.ts#L40
and it is registered in global.d.ts in:
https://github.com/WoltLab/d.ts/blob/57f923f03e37885885326bbd622d15b554feb9b5/global.d.ts#L122
Now when attempting to compile a project that has the repository as a dependency, the following error is reported: