mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
174 lines
4.6 KiB
TypeScript
174 lines
4.6 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(): Promise<void>;
|
|
|
|
protected defaultSorting = (item: T): string => item.getName();
|
|
|
|
@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()));
|
|
}
|
|
|
|
getByName(name: string, ...args: any[]): T;
|
|
getByName(name: string): T {
|
|
return this.items.find(item => item.getName() === name);
|
|
}
|
|
|
|
@action
|
|
protected sortItems<E>(items: any[], sorting?: ((item: any) => E)[], order?: "asc" | "desc"): any[] {
|
|
if (!sorting) {
|
|
return orderBy(items, [this.defaultSorting], order);
|
|
}
|
|
|
|
return orderBy(items, sorting, order);
|
|
}
|
|
|
|
protected async createItem(params: { name: string; namespace?: string }, data?: Partial<T>): Promise<T>;
|
|
protected async createItem<D>(...args: D[]): Promise<T>;
|
|
@action
|
|
protected async createItem(request: () => Promise<T>): 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<T[]>;
|
|
@action
|
|
protected async loadItems(request: () => Promise<T[] | any>, sortItems = true): Promise<T[]> {
|
|
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(params: { name: string; namespace?: string }): Promise<T>;
|
|
protected async loadItem<D>(...args: D[]): Promise<T>
|
|
@action
|
|
protected async loadItem(request: () => Promise<T>, sortItems = true): Promise<T> {
|
|
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>): 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>): Promise<void> {
|
|
await request();
|
|
this.items.remove(item);
|
|
this.selectedItemsIds.delete(item.getId());
|
|
}
|
|
|
|
isSelected(item: T): boolean {
|
|
return !!this.selectedItemsIds.get(item.getId());
|
|
}
|
|
|
|
@action
|
|
select(item: T): void {
|
|
this.selectedItemsIds.set(item.getId(), true);
|
|
}
|
|
|
|
@action
|
|
unselect(item: T): void {
|
|
this.selectedItemsIds.delete(item.getId());
|
|
}
|
|
|
|
@action
|
|
toggleSelection(item: T): void {
|
|
if (this.isSelected(item)) {
|
|
this.unselect(item);
|
|
} else {
|
|
this.select(item);
|
|
}
|
|
}
|
|
|
|
@action
|
|
toggleSelectionAll(visibleItems: T[] = this.items): void {
|
|
const allSelected = visibleItems.every(this.isSelected);
|
|
if (allSelected) {
|
|
visibleItems.forEach(this.unselect);
|
|
} else {
|
|
visibleItems.forEach(this.select);
|
|
}
|
|
}
|
|
|
|
isSelectedAll(visibleItems: T[] = this.items): boolean {
|
|
if (!visibleItems.length) {
|
|
return false;
|
|
}
|
|
return visibleItems.every(this.isSelected);
|
|
}
|
|
|
|
@action
|
|
resetSelection(): void {
|
|
this.selectedItemsIds.clear();
|
|
}
|
|
|
|
@action
|
|
reset(): void {
|
|
this.resetSelection();
|
|
this.items.clear();
|
|
this.selectedItemsIds.clear();
|
|
this.isLoaded = false;
|
|
this.isLoading = false;
|
|
}
|
|
|
|
async removeSelectedItems?(): Promise<any>;
|
|
|
|
subscribe(..._args: any[]): () => void {
|
|
return noop;
|
|
}
|
|
|
|
*[Symbol.iterator](): Generator<T, void, undefined> {
|
|
yield* this.items;
|
|
}
|
|
}
|