Created
May 4, 2015 18:29
-
-
Save bryanforbes/af292900328412373bab to your computer and use it in GitHub Desktop.
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import has from './has'; | |
export function hasClass(feature: string, trueClass: Function, falseClass: Function): ClassDecorator { | |
return function <T extends Function>(target: T): T { | |
return <any>(has(feature) ? trueClass : falseClass); | |
}; | |
} |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import { hasClass } from './decorators'; | |
import global from './global'; | |
module Native { | |
interface NativeWeakMap<K, V> { | |
delete(key: K): boolean; | |
get(key: K): V; | |
has(key: K): boolean; | |
set(key: K, value?: V): NativeWeakMap<K, V>; | |
} | |
// Since ES6 classes aren't inheritable, create a thin wrapper around | |
// the native WeakMap with an inheritable TS class | |
export class WeakMap<K, V> { | |
private _wm: NativeWeakMap<K, V>; | |
constructor(iterable?: any) { | |
Object.defineProperty(this, '_wm', { | |
value: new global.WeakMap(iterable) | |
}); | |
} | |
set(key: any, value?: any): Native.WeakMap<K, V> { | |
this._wm.set(key, value); | |
return this; | |
} | |
get(key: any): V { | |
return this._wm.get(key); | |
} | |
has(key: any): boolean { | |
return this._wm.has(key); | |
} | |
delete(key: any): boolean { | |
return this._wm.delete(key); | |
} | |
} | |
} | |
module Shim { | |
function getUID(): number { | |
return Math.floor(Math.random() * 100000000); | |
} | |
let generateName = (function () { | |
let startId = Math.floor(Date.now() % 100000000); | |
return function generateName(): string { | |
return '__wm' + getUID() + (startId++ + '__'); | |
}; | |
})(); | |
let deleted = {}; | |
export class WeakMap<K, V> { | |
private _name: string; | |
constructor(iterable?: any) { | |
Object.defineProperty(this, '_name', { | |
value: generateName() | |
}); | |
// TODO: | |
for (let [ key, value ] of iterable) { | |
this.set(key, value); | |
} | |
} | |
set(key: any, value?: any): Shim.WeakMap<K, V> { | |
let entry: [ K, V ] = key[this._name]; | |
if (entry && entry[0] === key) { | |
entry[1] = value; | |
} | |
else { | |
entry = [ key, value ]; | |
Object.defineProperty(entry, '0', { | |
value: key | |
}); | |
Object.defineProperty(key, this._name, { | |
value: entry | |
}); | |
} | |
return this; | |
} | |
get(key: any): V { | |
let entry: [ K, V ] = key[this._name]; | |
if (entry && entry[0] === key && entry[1] !== deleted) { | |
return entry[1]; | |
} | |
} | |
has(key: any): boolean { | |
let entry: [ K, V ] = key[this._name]; | |
return Boolean(entry && entry[0] === key && entry[1] !== deleted); | |
} | |
delete(key: any): boolean { | |
let entry: [ K, V ] = key[this._name]; | |
if (entry && entry[0] === key) { | |
this.set(key, <any>deleted); | |
} | |
return false; | |
} | |
} | |
} | |
@hasClass('weakmap', Native.WeakMap, Shim.WeakMap) | |
export default class WeakMap<K, V> { | |
constructor(iterable?: any) {} | |
delete(key: K): boolean { throw new Error(); } | |
get(key: K): V { throw new Error(); } | |
has(key: K): boolean { throw new Error(); } | |
set(key: K, value?: V): WeakMap<K, V> { throw new Error(); } | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment