1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-11 16:42:21 -05:00
denoland-deno/js/mixins/dom_iterable.ts

84 lines
2.6 KiB
TypeScript
Raw Normal View History

2019-01-21 14:03:30 -05:00
// Copyright 2018-2019 the Deno authors. All rights reserved. MIT license.
2018-10-23 07:43:43 -04:00
import { DomIterable } from "../dom_types";
import { globalEval } from "../global_eval";
import { requiredArguments } from "../util";
2018-10-23 07:43:43 -04:00
// if we import it directly from "globals" it will break the unit tests so we
// have to grab a reference to the global scope a different way
const window = globalEval("this");
// tslint:disable:no-any
type Constructor<T = {}> = new (...args: any[]) => T;
/** Mixes in a DOM iterable methods into a base class, assumes that there is
* a private data iterable that is part of the base class, located at
* `[dataSymbol]`.
* TODO Don't expose DomIterableMixin from "deno" namespace.
2018-10-23 07:43:43 -04:00
*/
export function DomIterableMixin<K, V, TBase extends Constructor>(
// tslint:disable-next-line:variable-name
Base: TBase,
dataSymbol: symbol
): TBase & Constructor<DomIterable<K, V>> {
// we have to cast `this` as `any` because there is no way to describe the
// Base class in a way where the Symbol `dataSymbol` is defined. So the
// runtime code works, but we do lose a little bit of type safety.
// Additionally, we have to not use .keys() nor .values() since the internal
// slot differs in type - some have a Map, which yields [K, V] in
// Symbol.iterator, and some have an Array, which yields V, in this case
// [K, V] too as they are arrays of tuples.
2018-10-23 07:43:43 -04:00
// tslint:disable-next-line:variable-name
const DomIterable = class extends Base {
*entries(): IterableIterator<[K, V]> {
for (const entry of (this as any)[dataSymbol]) {
2018-10-23 07:43:43 -04:00
yield entry;
}
}
*keys(): IterableIterator<K> {
for (const [key] of (this as any)[dataSymbol]) {
2018-10-23 07:43:43 -04:00
yield key;
}
}
*values(): IterableIterator<V> {
for (const [, value] of (this as any)[dataSymbol]) {
2018-10-23 07:43:43 -04:00
yield value;
}
}
forEach(
callbackfn: (value: V, key: K, parent: this) => void,
// tslint:disable-next-line:no-any
thisArg?: any
): void {
requiredArguments(
`${this.constructor.name}.forEach`,
arguments.length,
1
);
2018-10-23 07:43:43 -04:00
callbackfn = callbackfn.bind(thisArg == null ? window : Object(thisArg));
for (const [key, value] of (this as any)[dataSymbol]) {
2018-10-23 07:43:43 -04:00
callbackfn(value, key, this);
}
}
*[Symbol.iterator](): IterableIterator<[K, V]> {
for (const entry of (this as any)[dataSymbol]) {
yield entry;
}
}
};
// we want the Base class name to be the name of the class.
Object.defineProperty(DomIterable, "name", {
value: Base.name,
configurable: true
});
return DomIterable;
}
// tslint:enable:no-any