export class Collection { private database: Promise; constructor(public readonly db: string, public readonly collection: string, public readonly version?: number, private setup?: (db: IDBDatabase, event: IDBVersionChangeEvent) => void) { this.database = new Promise((resolve, reject) => { const req = indexedDB.open(this.db, this.version); req.onerror = () => reject(req.error); req.onsuccess = () => resolve(req.result); req.onupgradeneeded = (event) => { const db = req.result; if(!db.objectStoreNames.contains(collection)) db.createObjectStore(collection, {keyPath: undefined}); if (this.setup) this.setup(db, event as IDBVersionChangeEvent); }; }); } async tx(collection: string, fn: ((store: IDBObjectStore) => IDBRequest), readonly?: boolean): Promise { const db = await this.database; const tx = db.transaction(collection, readonly ? 'readonly' : 'readwrite'); const store = tx.objectStore(collection); return new Promise((resolve, reject) => { const request = fn(store); request.onsuccess = () => resolve(request.result); request.onerror = () => reject(request.error); }); } add(value: T, key?: K): Promise { return this.tx(this.collection, store => store.add(value, key)); } count(): Promise { return this.tx(this.collection, store => store.count(), true); } put(key: K, value: T): Promise { return this.tx(this.collection, store => store.put(value, key)); } getAll(): Promise { return this.tx(this.collection, store => store.getAll(), true); } getAllKeys(): Promise { return this.tx(this.collection, store => store.getAllKeys(), true); } get(key: K): Promise { return this.tx(this.collection, store => store.get(key), true); } delete(key: K): Promise { return this.tx(this.collection, store => store.delete(key)); } clear(): Promise { return this.tx(this.collection, store => store.clear()); } }