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

Extract CatalogEntityDetails to seperate root component

Signed-off-by: Sebastian Malton <sebastian@malton.name>
This commit is contained in:
Sebastian Malton 2023-01-13 13:40:18 -05:00
parent f92da1796e
commit c0f439d887
14 changed files with 330 additions and 156 deletions

View File

@ -34,6 +34,10 @@ export class CatalogCategoryRegistry {
}; };
} }
getById(id: string) {
return iter.find(this.categories.values(), (category) => category.getId() === id);
}
@computed get items() { @computed get items() {
return Array.from(this.categories); return Array.from(this.categories);
} }

View File

@ -0,0 +1,20 @@
/**
* Copyright (c) OpenLens Authors. All rights reserved.
* Licensed under MIT License. See LICENSE in root directory for more information.
*/
import { getInjectable } from "@ogre-tools/injectable";
import type { CatalogEntity } from "../../catalog-entity";
import catalogEntityRegistryInjectable from "./registry.injectable";
export type GetEntityById = (id: string) => CatalogEntity | undefined;
const getEntityByIdInjectable = getInjectable({
id: "get-entity-by-id",
instantiate: (di): GetEntityById => {
const catalogEntityRegistry = di.inject(catalogEntityRegistryInjectable);
return (id) => catalogEntityRegistry.getById(id);
},
});
export default getEntityByIdInjectable;

View File

@ -3,12 +3,16 @@
* Licensed under MIT License. See LICENSE in root directory for more information. * Licensed under MIT License. See LICENSE in root directory for more information.
*/ */
import type { DiContainer } from "@ogre-tools/injectable";
import type { CatalogCategoryMetadata, CatalogCategorySpec } from "../../../../common/catalog"; import type { CatalogCategoryMetadata, CatalogCategorySpec } from "../../../../common/catalog";
import { CatalogEntity, categoryVersion } from "../../../../common/catalog"; import { CatalogEntity, categoryVersion } from "../../../../common/catalog";
import catalogCategoryRegistryInjectable from "../../../../common/catalog/category-registry.injectable";
import { CatalogCategory } from "../../../api/catalog-entity"; import { CatalogCategory } from "../../../api/catalog-entity";
import catalogEntityRegistryInjectable from "../../../api/catalog/entity/registry.injectable";
import { getDiForUnitTesting } from "../../../getDiForUnitTesting";
import { noop } from "../../../utils"; import { noop } from "../../../utils";
import type { CatalogEntityStore } from "../catalog-entity-store/catalog-entity.store"; import type { CatalogEntityStore } from "../catalog-entity-store.injectable";
import { catalogEntityStore } from "../catalog-entity-store/catalog-entity.store"; import catalogEntityStoreInjectable from "../catalog-entity-store.injectable";
class TestEntityOne extends CatalogEntity { class TestEntityOne extends CatalogEntity {
public static readonly apiVersion: string = "entity.k8slens.dev/v1alpha1"; public static readonly apiVersion: string = "entity.k8slens.dev/v1alpha1";
@ -63,6 +67,12 @@ class TestCategoryTwo extends CatalogCategory {
} }
describe("CatalogEntityStore", () => { describe("CatalogEntityStore", () => {
let di: DiContainer;
beforeEach(() => {
di = getDiForUnitTesting({ doGeneralOverrides: true });
});
describe("getTotalCount", () => { describe("getTotalCount", () => {
let store: CatalogEntityStore; let store: CatalogEntityStore;
let testCategoryOne: TestCategoryOne; let testCategoryOne: TestCategoryOne;
@ -129,21 +139,22 @@ describe("CatalogEntityStore", () => {
testCategoryOne = new TestCategoryOne(); testCategoryOne = new TestCategoryOne();
testCategoryTwo = new TestCategoryTwo(); testCategoryTwo = new TestCategoryTwo();
store = catalogEntityStore({
catalogRegistry: { di.override(catalogCategoryRegistryInjectable, () => ({
items: [ items: [
testCategoryOne, testCategoryOne,
testCategoryTwo, testCategoryTwo,
], ],
}, }));
entityRegistry: { di.override(catalogEntityRegistryInjectable, () => ({
onRun: noop, onRun: noop,
filteredItems: entityItems, filteredItems: entityItems,
getItemsForCategory: <T extends CatalogEntity>(category: CatalogCategory): T[] => { getItemsForCategory: <T extends CatalogEntity>(category: CatalogCategory): T[] => {
return entityItems.filter(item => category.spec.versions.some(version => item instanceof version.entityClass)) as T[]; return entityItems.filter(item => category.spec.versions.some(version => item instanceof version.entityClass)) as T[];
}, },
}, } as any));
});
store = di.inject(catalogEntityStoreInjectable);
}); });
it("given no active category, returns count of all kinds", () => { it("given no active category, returns count of all kinds", () => {

View File

@ -0,0 +1,74 @@
/**
* Copyright (c) OpenLens Authors. All rights reserved.
* Licensed under MIT License. See LICENSE in root directory for more information.
*/
import { getInjectable } from "@ogre-tools/injectable";
import catalogEntityRegistryInjectable from "../../api/catalog/entity/registry.injectable";
import type { IComputedValue, IObservableValue } from "mobx";
import { computed, observable, reaction } from "mobx";
import type { CatalogEntity } from "../../api/catalog-entity";
import type { CatalogCategory } from "../../../common/catalog";
import type { Disposer } from "../../../common/utils";
import { disposer } from "../../../common/utils";
import type { ItemListStore } from "../item-object-list";
import catalogCategoryRegistryInjectable from "../../../common/catalog/category-registry.injectable";
import selectedCatalogEntityParamInjectable from "./entity-details/selected-uid.injectable";
export type CatalogEntityStore = ItemListStore<CatalogEntity, false> & {
readonly entities: IComputedValue<CatalogEntity[]>;
readonly activeCategory: IObservableValue<CatalogCategory | undefined>;
watch(): Disposer;
onRun(entity: CatalogEntity): void;
};
const catalogEntityStoreInjectable = getInjectable({
id: "catalog-entity-store",
instantiate: (di): CatalogEntityStore => {
const catalogEntityRegistry = di.inject(catalogEntityRegistryInjectable);
const catalogCategoryRegistry = di.inject(catalogCategoryRegistryInjectable);
const selectedCatalogEntityParam = di.inject(selectedCatalogEntityParamInjectable);
const activeCategory = observable.box<CatalogCategory | undefined>(undefined);
const entities = computed(() => {
const category = activeCategory.get();
return category
? catalogEntityRegistry.getItemsForCategory(category, { filtered: true })
: catalogEntityRegistry.filteredItems;
});
const loadAll = () => {
const category = activeCategory.get();
if (category) {
category.emit("load");
} else {
for (const category of catalogCategoryRegistry.items) {
category.emit("load");
}
}
};
return {
entities,
activeCategory,
watch: () => disposer(
reaction(() => entities.get(), loadAll),
reaction(() => activeCategory.get(), loadAll, { delay: 100 }),
),
onRun: entity => catalogEntityRegistry.onRun(entity),
failedLoading: false,
getTotalCount: () => entities.get().length,
isLoaded: true,
isSelected: (item) => item.getId() === selectedCatalogEntityParam.get(),
isSelectedAll: () => false,
pickOnlySelected: () => [],
toggleSelection: () => {},
toggleSelectionAll: () => {},
removeSelectedItems: async () => {},
};
},
});
export default catalogEntityStoreInjectable;

View File

@ -1,19 +0,0 @@
/**
* Copyright (c) OpenLens Authors. All rights reserved.
* Licensed under MIT License. See LICENSE in root directory for more information.
*/
import { getInjectable } from "@ogre-tools/injectable";
import { catalogEntityStore } from "./catalog-entity.store";
import catalogEntityRegistryInjectable from "../../../api/catalog/entity/registry.injectable";
import catalogCategoryRegistryInjectable from "../../../../common/catalog/category-registry.injectable";
const catalogEntityStoreInjectable = getInjectable({
id: "catalog-entity-store",
instantiate: (di) => catalogEntityStore({
entityRegistry: di.inject(catalogEntityRegistryInjectable),
catalogRegistry: di.inject(catalogCategoryRegistryInjectable),
}),
});
export default catalogEntityStoreInjectable;

View File

@ -1,86 +0,0 @@
/**
* Copyright (c) OpenLens Authors. All rights reserved.
* Licensed under MIT License. See LICENSE in root directory for more information.
*/
import type { IComputedValue, IObservableValue } from "mobx";
import { computed, observable, reaction } from "mobx";
import type { CatalogEntityRegistry } from "../../../api/catalog/entity/registry";
import type { CatalogEntity } from "../../../api/catalog-entity";
import type { CatalogCategory, CatalogCategoryRegistry } from "../../../../common/catalog";
import type { Disposer } from "../../../../common/utils";
import { disposer } from "../../../../common/utils";
import type { ItemListStore } from "../../item-object-list";
type EntityRegistry = Pick<CatalogEntityRegistry, "getItemsForCategory" | "filteredItems" | "onRun">;
type CatalogRegistry = Pick<CatalogCategoryRegistry, "items">;
interface Dependencies {
entityRegistry: EntityRegistry;
catalogRegistry: CatalogRegistry;
}
export type CatalogEntityStore = ItemListStore<CatalogEntity, false> & {
readonly entities: IComputedValue<CatalogEntity[]>;
readonly activeCategory: IObservableValue<CatalogCategory | undefined>;
readonly selectedItemId: IObservableValue<string | undefined>;
readonly selectedItem: IComputedValue<CatalogEntity | undefined>;
watch(): Disposer;
onRun(entity: CatalogEntity): void;
};
export function catalogEntityStore({
entityRegistry,
catalogRegistry,
}: Dependencies): CatalogEntityStore {
const activeCategory = observable.box<CatalogCategory | undefined>(undefined);
const selectedItemId = observable.box<string | undefined>(undefined);
const entities = computed(() => {
const category = activeCategory.get();
return category
? entityRegistry.getItemsForCategory(category, { filtered: true })
: entityRegistry.filteredItems;
});
const selectedItem = computed(() => {
const id = selectedItemId.get();
if (!id) {
return undefined;
}
return entities.get().find(entity => entity.getId() === id);
});
const loadAll = () => {
const category = activeCategory.get();
if (category) {
category.emit("load");
} else {
for (const category of catalogRegistry.items) {
category.emit("load");
}
}
};
return {
entities,
selectedItem,
activeCategory,
selectedItemId,
watch: () => disposer(
reaction(() => entities.get(), loadAll),
reaction(() => activeCategory.get(), loadAll, { delay: 100 }),
),
onRun: entity => entityRegistry.onRun(entity),
failedLoading: false,
getTotalCount: () => entities.get().length,
isLoaded: true,
isSelected: (item) => item.getId() === selectedItemId.get(),
isSelectedAll: () => false,
pickOnlySelected: () => [],
toggleSelection: () => {},
toggleSelectionAll: () => {},
removeSelectedItems: async () => {},
};
}

View File

@ -10,10 +10,10 @@ import { Catalog } from "./catalog";
import type { CatalogEntityActionContext, CatalogEntityData } from "../../../common/catalog"; import type { CatalogEntityActionContext, CatalogEntityData } from "../../../common/catalog";
import { CatalogEntity } from "../../../common/catalog"; import { CatalogEntity } from "../../../common/catalog";
import type { CatalogEntityOnBeforeRun, CatalogEntityRegistry } from "../../api/catalog/entity/registry"; import type { CatalogEntityOnBeforeRun, CatalogEntityRegistry } from "../../api/catalog/entity/registry";
import type { CatalogEntityStore } from "./catalog-entity-store/catalog-entity.store"; import type { CatalogEntityStore } from "./catalog-entity-store.injectable";
import { getDiForUnitTesting } from "../../getDiForUnitTesting"; import { getDiForUnitTesting } from "../../getDiForUnitTesting";
import type { DiContainer } from "@ogre-tools/injectable"; import type { DiContainer } from "@ogre-tools/injectable";
import catalogEntityStoreInjectable from "./catalog-entity-store/catalog-entity-store.injectable"; import catalogEntityStoreInjectable from "./catalog-entity-store.injectable";
import catalogEntityRegistryInjectable from "../../api/catalog/entity/registry.injectable"; import catalogEntityRegistryInjectable from "../../api/catalog/entity/registry.injectable";
import type { DiRender } from "../test-utils/renderFor"; import type { DiRender } from "../test-utils/renderFor";
import { renderFor } from "../test-utils/renderFor"; import { renderFor } from "../test-utils/renderFor";

View File

@ -10,7 +10,7 @@ import { disposeOnUnmount, observer } from "mobx-react";
import { ItemListLayout } from "../item-object-list"; import { ItemListLayout } from "../item-object-list";
import type { IComputedValue } from "mobx"; import type { IComputedValue } from "mobx";
import { action, computed, makeObservable, observable, reaction, runInAction, when } from "mobx"; import { action, computed, makeObservable, observable, reaction, runInAction, when } from "mobx";
import type { CatalogEntityStore } from "./catalog-entity-store/catalog-entity.store"; import type { CatalogEntityStore } from "./catalog-entity-store.injectable";
import { MenuItem, MenuActions } from "../menu"; import { MenuItem, MenuActions } from "../menu";
import type { CatalogEntityContextMenu } from "../../api/catalog-entity"; import type { CatalogEntityContextMenu } from "../../api/catalog-entity";
import type { CatalogCategory, CatalogCategoryRegistry, CatalogEntity } from "../../../common/catalog"; import type { CatalogCategory, CatalogCategoryRegistry, CatalogEntity } from "../../../common/catalog";
@ -19,7 +19,6 @@ import type { ShowNotification } from "../notifications";
import { MainLayout } from "../layout/main-layout"; import { MainLayout } from "../layout/main-layout";
import type { StorageLayer } from "../../utils"; import type { StorageLayer } from "../../utils";
import { prevDefault } from "../../utils"; import { prevDefault } from "../../utils";
import { CatalogEntityDetails } from "./entity-details/view";
import { CatalogMenu } from "./catalog-menu"; import { CatalogMenu } from "./catalog-menu";
import { RenderDelay } from "../render-delay/render-delay"; import { RenderDelay } from "../render-delay/render-delay";
import { Icon } from "../icon"; import { Icon } from "../icon";
@ -27,7 +26,7 @@ import { HotbarToggleMenuItem } from "./hotbar-toggle-menu-item";
import { Avatar } from "../avatar"; import { Avatar } from "../avatar";
import { withInjectables } from "@ogre-tools/injectable-react"; import { withInjectables } from "@ogre-tools/injectable-react";
import catalogPreviousActiveTabStorageInjectable from "./catalog-previous-active-tab-storage/catalog-previous-active-tab-storage.injectable"; import catalogPreviousActiveTabStorageInjectable from "./catalog-previous-active-tab-storage/catalog-previous-active-tab-storage.injectable";
import catalogEntityStoreInjectable from "./catalog-entity-store/catalog-entity-store.injectable"; import catalogEntityStoreInjectable from "./catalog-entity-store.injectable";
import type { GetCategoryColumnsParams, CategoryColumns } from "./columns/get.injectable"; import type { GetCategoryColumnsParams, CategoryColumns } from "./columns/get.injectable";
import getCategoryColumnsInjectable from "./columns/get.injectable"; import getCategoryColumnsInjectable from "./columns/get.injectable";
import type { RegisteredCustomCategoryViewDecl } from "./custom-views.injectable"; import type { RegisteredCustomCategoryViewDecl } from "./custom-views.injectable";
@ -51,6 +50,10 @@ import emitAppEventInjectable from "../../../common/app-event-bus/emit-event.inj
import type { Logger } from "../../../common/logger"; import type { Logger } from "../../../common/logger";
import loggerInjectable from "../../../common/logger.injectable"; import loggerInjectable from "../../../common/logger.injectable";
import showErrorNotificationInjectable from "../notifications/show-error-notification.injectable"; import showErrorNotificationInjectable from "../notifications/show-error-notification.injectable";
import type { ShowEntityDetails } from "./entity-details/show.injectable";
import showEntityDetailsInjectable from "./entity-details/show.injectable";
import type { OnCatalogEntityListClick } from "./entity-details/on-catalog-click.injectable";
import onCatalogEntityListClickInjectable from "./entity-details/on-catalog-click.injectable";
interface Dependencies { interface Dependencies {
catalogPreviousActiveTabStorage: StorageLayer<string | null>; catalogPreviousActiveTabStorage: StorageLayer<string | null>;
@ -58,6 +61,8 @@ interface Dependencies {
getCategoryColumns: (params: GetCategoryColumnsParams) => CategoryColumns; getCategoryColumns: (params: GetCategoryColumnsParams) => CategoryColumns;
customCategoryViews: IComputedValue<Map<string, Map<string, RegisteredCustomCategoryViewDecl>>>; customCategoryViews: IComputedValue<Map<string, Map<string, RegisteredCustomCategoryViewDecl>>>;
emitEvent: EmitAppEvent; emitEvent: EmitAppEvent;
showEntityDetails: ShowEntityDetails;
onCatalogEntityListClick: OnCatalogEntityListClick;
routeParameters: { routeParameters: {
group: IComputedValue<string>; group: IComputedValue<string>;
kind: IComputedValue<string>; kind: IComputedValue<string>;
@ -161,26 +166,16 @@ class NonInjectedCatalog extends React.Component<Dependencies> {
this.props.hotbarStore.removeFromHotbar(entity.getId()); this.props.hotbarStore.removeFromHotbar(entity.getId());
} }
onDetails = (entity: CatalogEntity) => {
if (this.props.catalogEntityStore.selectedItemId.get()) {
this.props.catalogEntityStore.selectedItemId.set(undefined);
} else {
this.props.catalogEntityStore.onRun(entity);
}
};
get categories() {
return this.props.catalogCategoryRegistry.items;
}
onTabChange = action((tabId: string | null) => { onTabChange = action((tabId: string | null) => {
const activeCategory = this.categories.find(category => category.getId() === tabId); const activeCategory = tabId
? this.props.catalogCategoryRegistry.getById(tabId)
: undefined;
this.props.emitEvent({ this.props.emitEvent({
name: "catalog", name: "catalog",
action: "change-category", action: "change-category",
params: { params: {
category: activeCategory ? activeCategory.getName() : "Browse", category: activeCategory?.getName() ?? "Browse",
}, },
}); });
@ -211,7 +206,7 @@ class NonInjectedCatalog extends React.Component<Dependencies> {
<MenuItem <MenuItem
key="open-details" key="open-details"
data-testid={`open-details-menu-item-for-${entity.getId()}`} data-testid={`open-details-menu-item-for-${entity.getId()}`}
onClick={() => this.props.catalogEntityStore.selectedItemId.set(entity.getId())} onClick={() => this.props.showEntityDetails(entity.getId())}
> >
View Details View Details
</MenuItem> </MenuItem>
@ -309,7 +304,7 @@ class NonInjectedCatalog extends React.Component<Dependencies> {
disabled: !entity.isEnabled(), disabled: !entity.isEnabled(),
})} })}
{...getCategoryColumns({ activeCategory })} {...getCategoryColumns({ activeCategory })}
onDetails={this.onDetails} onDetails={this.props.onCatalogEntityListClick}
renderItemMenu={this.renderItemMenu} renderItemMenu={this.renderItemMenu}
data-testid={`catalog-list-for-${activeCategory?.metadata.name ?? "browse-all"}`} data-testid={`catalog-list-for-${activeCategory?.metadata.name ?? "browse-all"}`}
/> />
@ -318,7 +313,6 @@ class NonInjectedCatalog extends React.Component<Dependencies> {
render() { render() {
const activeCategory = this.props.catalogEntityStore.activeCategory.get(); const activeCategory = this.props.catalogEntityStore.activeCategory.get();
const selectedItem = this.props.catalogEntityStore.selectedItem.get();
return ( return (
<MainLayout <MainLayout
@ -333,15 +327,7 @@ class NonInjectedCatalog extends React.Component<Dependencies> {
{this.renderViews(activeCategory)} {this.renderViews(activeCategory)}
</div> </div>
{ {
selectedItem activeCategory
? (
<CatalogEntityDetails
entity={selectedItem}
hideDetails={() => this.props.catalogEntityStore.selectedItemId.set(undefined)}
onRun={() => this.props.catalogEntityStore.onRun(selectedItem)}
/>
)
: activeCategory
? ( ? (
<RenderDelay> <RenderDelay>
<CatalogAddButton category={activeCategory} /> <CatalogAddButton category={activeCategory} />
@ -371,5 +357,7 @@ export const Catalog = withInjectables<Dependencies>(NonInjectedCatalog, {
normalizeMenuItem: di.inject(normalizeCatalogEntityContextMenuInjectable), normalizeMenuItem: di.inject(normalizeCatalogEntityContextMenuInjectable),
logger: di.inject(loggerInjectable), logger: di.inject(loggerInjectable),
showErrorNotification: di.inject(showErrorNotificationInjectable), showErrorNotification: di.inject(showErrorNotificationInjectable),
showEntityDetails: di.inject(showEntityDetailsInjectable),
onCatalogEntityListClick: di.inject(onCatalogEntityListClickInjectable),
}), }),
}); });

View File

@ -0,0 +1,65 @@
/**
* Copyright (c) OpenLens Authors. All rights reserved.
* Licensed under MIT License. See LICENSE in root directory for more information.
*/
import { getInjectable } from "@ogre-tools/injectable";
import { withInjectables } from "@ogre-tools/injectable-react";
import type { IComputedValue } from "mobx";
import { computed } from "mobx";
import { observer } from "mobx-react";
import React from "react";
import type { CatalogEntity } from "../../../api/catalog-entity";
import type { CatalogEntityRegistry } from "../../../api/catalog/entity/registry";
import catalogEntityRegistryInjectable from "../../../api/catalog/entity/registry.injectable";
import { rootFrameChildComponentInjectionToken } from "../../../frames/root-frame/root-frame-child-component-injection-token";
import type { HideEntityDetails } from "./hide.injectable";
import hideEntityDetailsInjectable from "./hide.injectable";
import selectedCatalogEntityInjectable from "./selected-entity.injectable";
import { CatalogEntityDetails } from "./view";
interface Dependencies {
selectedCatalogEntity: IComputedValue<CatalogEntity | undefined>;
hideEntityDetails: HideEntityDetails;
catalogEntityRegistry: CatalogEntityRegistry;
}
const NonInjectedCatalogEntityDetailsComponent = observer(({
selectedCatalogEntity,
hideEntityDetails,
catalogEntityRegistry,
}: Dependencies) => {
const entity = selectedCatalogEntity.get();
if (!entity) {
return null;
}
return (
<CatalogEntityDetails
entity={entity}
hideDetails={hideEntityDetails}
onRun={() => catalogEntityRegistry.onRun(entity)}
/>
);
});
const CatalogEntityDetailsComponent = withInjectables<Dependencies>(NonInjectedCatalogEntityDetailsComponent, {
getProps: (di, props) => ({
...props,
selectedCatalogEntity: di.inject(selectedCatalogEntityInjectable),
catalogEntityRegistry: di.inject(catalogEntityRegistryInjectable),
hideEntityDetails: di.inject(hideEntityDetailsInjectable),
}),
});
const catalogEntityDetailsComponentInjectable = getInjectable({
id: "catalog-entity-details-component",
instantiate: () => ({
id: "catalog-entity-details-component",
Component: CatalogEntityDetailsComponent,
shouldRender: computed(() => true),
}),
injectionToken: rootFrameChildComponentInjectionToken,
});
export default catalogEntityDetailsComponentInjectable;

View File

@ -0,0 +1,20 @@
/**
* Copyright (c) OpenLens Authors. All rights reserved.
* Licensed under MIT License. See LICENSE in root directory for more information.
*/
import { getInjectable } from "@ogre-tools/injectable";
import selectedCatalogEntityParamInjectable from "./selected-uid.injectable";
export type HideEntityDetails = () => void;
const hideEntityDetailsInjectable = getInjectable({
id: "hide-entity-details",
instantiate: (di) => {
const selectedCatalogEntityParam = di.inject(selectedCatalogEntityParamInjectable);
return () => selectedCatalogEntityParam.clear();
},
});
export default hideEntityDetailsInjectable;

View File

@ -0,0 +1,31 @@
/**
* Copyright (c) OpenLens Authors. All rights reserved.
* Licensed under MIT License. See LICENSE in root directory for more information.
*/
import { getInjectable } from "@ogre-tools/injectable";
import { action } from "mobx";
import type { CatalogEntity } from "../../../api/catalog-entity";
import catalogEntityRegistryInjectable from "../../../api/catalog/entity/registry.injectable";
import selectedCatalogEntityParamInjectable from "./selected-uid.injectable";
export type OnCatalogEntityListClick = (entity: CatalogEntity) => void;
const onCatalogEntityListClickInjectable = getInjectable({
id: "on-catalog-entity-list-click",
instantiate: (di): OnCatalogEntityListClick => {
const selectedCatalogEntityParam = di.inject(selectedCatalogEntityParamInjectable);
const catalogEntityRegistry = di.inject(catalogEntityRegistryInjectable);
return action(entity => {
if (selectedCatalogEntityParam.get() === entity.getId()) {
selectedCatalogEntityParam.clear();
} else if (selectedCatalogEntityParam.get() === undefined) {
catalogEntityRegistry.onRun(entity);
} else {
selectedCatalogEntityParam.set(entity.getId());
}
});
},
});
export default onCatalogEntityListClickInjectable;

View File

@ -0,0 +1,28 @@
/**
* Copyright (c) OpenLens Authors. All rights reserved.
* Licensed under MIT License. See LICENSE in root directory for more information.
*/
import { getInjectable } from "@ogre-tools/injectable";
import { computed } from "mobx";
import getEntityByIdInjectable from "../../../api/catalog/entity/get-by-id.injectable";
import selectedCatalogEntityParamInjectable from "./selected-uid.injectable";
const selectedCatalogEntityInjectable = getInjectable({
id: "selected-catalog-entity",
instantiate: (di) => {
const getEntityById = di.inject(getEntityByIdInjectable);
const selectedCatalogEntityParam = di.inject(selectedCatalogEntityParamInjectable);
return computed(() => {
const id = selectedCatalogEntityParam.get();
if (!id) {
return undefined;
}
return getEntityById(id);
});
},
});
export default selectedCatalogEntityInjectable;

View File

@ -0,0 +1,19 @@
/**
* Copyright (c) OpenLens Authors. All rights reserved.
* Licensed under MIT License. See LICENSE in root directory for more information.
*/
import { getInjectable } from "@ogre-tools/injectable";
import createPageParamInjectable from "../../../navigation/create-page-param.injectable";
const selectedCatalogEntityParamInjectable = getInjectable({
id: "selected-catalog-entity-param",
instantiate: (di) => {
const createPageParam = di.inject(createPageParamInjectable);
return createPageParam({
name: "catalog-entity-details",
});
},
});
export default selectedCatalogEntityParamInjectable;

View File

@ -0,0 +1,19 @@
/**
* Copyright (c) OpenLens Authors. All rights reserved.
* Licensed under MIT License. See LICENSE in root directory for more information.
*/
import { getInjectable } from "@ogre-tools/injectable";
import selectedCatalogEntityParamInjectable from "./selected-uid.injectable";
export type ShowEntityDetails = (id: string) => void;
const showEntityDetailsInjectable = getInjectable({
id: "show-entity-details",
instantiate: (di): ShowEntityDetails => {
const selectedCatalogEntityParam = di.inject(selectedCatalogEntityParamInjectable);
return (id) => selectedCatalogEntityParam.set(id);
},
});
export default showEntityDetailsInjectable;