summaryrefslogtreecommitdiff
path: root/js/mixins/dom_iterable.ts
diff options
context:
space:
mode:
Diffstat (limited to 'js/mixins/dom_iterable.ts')
-rw-r--r--js/mixins/dom_iterable.ts70
1 files changed, 70 insertions, 0 deletions
diff --git a/js/mixins/dom_iterable.ts b/js/mixins/dom_iterable.ts
new file mode 100644
index 000000000..015226589
--- /dev/null
+++ b/js/mixins/dom_iterable.ts
@@ -0,0 +1,70 @@
+import { DomIterable } from "../dom_types";
+import { globalEval } from "../global_eval";
+
+// 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]`.
+ */
+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.
+
+ // tslint:disable-next-line:variable-name
+ const DomIterable = class extends Base {
+ *entries(): IterableIterator<[K, V]> {
+ for (const entry of (this as any)[dataSymbol].entries()) {
+ yield entry;
+ }
+ }
+
+ *keys(): IterableIterator<K> {
+ for (const key of (this as any)[dataSymbol].keys()) {
+ yield key;
+ }
+ }
+
+ *values(): IterableIterator<V> {
+ for (const value of (this as any)[dataSymbol].values()) {
+ yield value;
+ }
+ }
+
+ forEach(
+ callbackfn: (value: V, key: K, parent: this) => void,
+ // tslint:disable-next-line:no-any
+ thisArg?: any
+ ): void {
+ callbackfn = callbackfn.bind(thisArg == null ? window : Object(thisArg));
+ for (const [key, value] of (this as any)[dataSymbol].entries()) {
+ 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