1
0
mirror of https://github.com/lensapp/lens.git synced 2025-05-20 05:10:56 +00:00
lens/src/renderer/item.store.ts

191 lines
4.5 KiB
TypeScript

import orderBy from "lodash/orderBy";
import { autobind, noop } from "./utils";
import { action, computed, observable, when } from "mobx";
export interface ItemObject {
getId(): string;
getName(): string;
}
@autobind()
export abstract class ItemStore<T extends ItemObject = ItemObject> {
abstract loadAll(...args: any[]): Promise<void | T[]>;
protected defaultSorting = (item: T) => item.getName();
@observable failedLoading = false;
@observable isLoading = false;
@observable isLoaded = false;
@observable items = observable.array<T>([], { deep: false });
@observable selectedItemsIds = observable.map<string, boolean>();
@computed get selectedItems(): T[] {
return this.items.filter(item => this.selectedItemsIds.get(item.getId()));
}
public getItems(): T[] {
return this.items.toJS();
}
public getTotalCount(): number {
return this.items.length;
}
getByName(name: string, ...args: any[]): T;
getByName(name: string): T {
return this.items.find(item => item.getName() === name);
}
getIndexById(id: string): number {
return this.items.findIndex(item => item.getId() === id);
}
@action
protected sortItems(items: T[] = this.items, sorting?: ((item: T) => any)[], order?: "asc" | "desc"): T[] {
return orderBy(items, sorting || this.defaultSorting, order);
}
protected async createItem(...args: any[]): Promise<any>;
@action
protected async createItem(request: () => Promise<T>) {
const newItem = await request();
const item = this.items.find(item => item.getId() === newItem.getId());
if (item) {
return item;
} else {
const items = this.sortItems([...this.items, newItem]);
this.items.replace(items);
return newItem;
}
}
protected async loadItems(...args: any[]): Promise<any>;
@action
protected async loadItems(request: () => Promise<T[] | any>, sortItems = true) {
if (this.isLoading) {
await when(() => !this.isLoading);
return;
}
this.isLoading = true;
try {
let items = await request();
if (sortItems) items = this.sortItems(items);
this.items.replace(items);
this.isLoaded = true;
} finally {
this.isLoading = false;
}
}
protected async loadItem(...args: any[]): Promise<T>
@action
protected async loadItem(request: () => Promise<T>, sortItems = true) {
const item = await request().catch(() => null);
if (item) {
const existingItem = this.items.find(el => el.getId() === item.getId());
if (existingItem) {
const index = this.items.findIndex(item => item === existingItem);
this.items.splice(index, 1, item);
} else {
let items = [...this.items, item];
if (sortItems) items = this.sortItems(items);
this.items.replace(items);
}
return item;
}
}
@action
protected async updateItem(item: T, request: () => Promise<T>) {
const updatedItem = await request();
const index = this.items.findIndex(i => i.getId() === item.getId());
this.items.splice(index, 1, updatedItem);
return updatedItem;
}
@action
protected async removeItem(item: T, request: () => Promise<any>) {
await request();
this.items.remove(item);
this.selectedItemsIds.delete(item.getId());
}
isSelected(item: T) {
return !!this.selectedItemsIds.get(item.getId());
}
@action
select(item: T) {
this.selectedItemsIds.set(item.getId(), true);
}
@action
unselect(item: T) {
this.selectedItemsIds.delete(item.getId());
}
@action
toggleSelection(item: T) {
if (this.isSelected(item)) {
this.unselect(item);
} else {
this.select(item);
}
}
@action
toggleSelectionAll(visibleItems: T[] = this.items) {
const allSelected = visibleItems.every(this.isSelected);
if (allSelected) {
visibleItems.forEach(this.unselect);
} else {
visibleItems.forEach(this.select);
}
}
isSelectedAll(visibleItems: T[] = this.items) {
if (!visibleItems.length) return false;
return visibleItems.every(this.isSelected);
}
@action
resetSelection() {
this.selectedItemsIds.clear();
}
@action
reset() {
this.resetSelection();
this.items.clear();
this.selectedItemsIds.clear();
this.isLoaded = false;
this.isLoading = false;
}
async removeSelectedItems?(): Promise<any>;
// eslint-disable-next-line unused-imports/no-unused-vars-ts
subscribe(...args: any[]) {
return noop;
}
* [Symbol.iterator]() {
yield* this.items;
}
}