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

PageRegistration refactoring #1258 -- part 1

Signed-off-by: Roman <ixrock@gmail.com>
This commit is contained in:
Roman 2020-11-10 17:24:19 +02:00
parent a78bbb5f6c
commit 9ec91c0c9c
26 changed files with 214 additions and 250 deletions

View File

@ -4,16 +4,16 @@ import { supportPageRoute, supportPageURL } from "./src/support.route";
import { Support } from "./src/support"; import { Support } from "./src/support";
export default class SupportPageRendererExtension extends LensRendererExtension { export default class SupportPageRendererExtension extends LensRendererExtension {
globalPages = [ // globalPages = [
{ // {
...supportPageRoute, // ...supportPageRoute,
url: supportPageURL(), // url: supportPageURL(),
hideInMenu: true, // hideInMenu: true,
components: { // components: {
Page: Support, // Page: Support,
} // }
} // }
] // ]
statusBarItems = [ statusBarItems = [
{ {

View File

@ -1,15 +0,0 @@
import React from "react";
import { cssNames } from "../renderer/utils";
import { TabLayout } from "../renderer/components/layout/tab-layout";
import { PageRegistration } from "./registries/page-registry"
export class DynamicPage extends React.Component<{ page: PageRegistration }> {
render() {
const { className, components: { Page }, subPages = [] } = this.props.page;
return (
<TabLayout className={cssNames("ExtensionPage", className)} tabs={subPages}>
<Page/>
</TabLayout>
)
}
}

View File

@ -65,6 +65,7 @@ export class ExtensionLoader {
logger.info('[EXTENSIONS-LOADER]: load on main renderer (cluster manager)') logger.info('[EXTENSIONS-LOADER]: load on main renderer (cluster manager)')
this.autoInitExtensions((extension: LensRendererExtension) => [ this.autoInitExtensions((extension: LensRendererExtension) => [
registries.globalPageRegistry.add(...extension.globalPages), registries.globalPageRegistry.add(...extension.globalPages),
registries.globalPageMenuRegistry.add(...extension.globalPageMenus),
registries.appPreferenceRegistry.add(...extension.appPreferences), registries.appPreferenceRegistry.add(...extension.appPreferences),
registries.clusterFeatureRegistry.add(...extension.clusterFeatures), registries.clusterFeatureRegistry.add(...extension.clusterFeatures),
registries.statusBarRegistry.add(...extension.statusBarItems), registries.statusBarRegistry.add(...extension.statusBarItems),
@ -75,6 +76,7 @@ export class ExtensionLoader {
logger.info('[EXTENSIONS-LOADER]: load on cluster renderer (dashboard)') logger.info('[EXTENSIONS-LOADER]: load on cluster renderer (dashboard)')
this.autoInitExtensions((extension: LensRendererExtension) => [ this.autoInitExtensions((extension: LensRendererExtension) => [
registries.clusterPageRegistry.add(...extension.clusterPages), registries.clusterPageRegistry.add(...extension.clusterPages),
registries.clusterPageMenuRegistry.add(...extension.clusterPageMenus),
registries.kubeObjectMenuRegistry.add(...extension.kubeObjectMenuItems), registries.kubeObjectMenuRegistry.add(...extension.kubeObjectMenuItems),
registries.kubeObjectDetailRegistry.add(...extension.kubeObjectDetailItems), registries.kubeObjectDetailRegistry.add(...extension.kubeObjectDetailItems),
]); ]);

View File

@ -1,14 +1,16 @@
import type { import type {
AppPreferenceRegistration, ClusterFeatureRegistration, AppPreferenceRegistration, ClusterFeatureRegistration,
KubeObjectMenuRegistration, KubeObjectDetailRegistration, KubeObjectMenuRegistration, KubeObjectDetailRegistration, StatusBarRegistration,
PageRegistration, StatusBarRegistration PageRegistration, PageMenuRegistration, PageRegistrationCluster, PageMenuRegistrationCluster,
} from "./registries" } from "./registries"
import { observable } from "mobx"; import { observable } from "mobx";
import { LensExtension } from "./lens-extension" import { LensExtension } from "./lens-extension"
export class LensRendererExtension extends LensExtension { export class LensRendererExtension extends LensExtension {
@observable.shallow globalPages: PageRegistration[] = [] @observable.shallow globalPages: PageRegistration[] = []
@observable.shallow clusterPages: PageRegistration[] = [] @observable.shallow clusterPages: PageRegistrationCluster[] = []
@observable.shallow globalPageMenus: PageMenuRegistration[] = []
@observable.shallow clusterPageMenus: PageMenuRegistrationCluster[] = []
@observable.shallow appPreferences: AppPreferenceRegistration[] = [] @observable.shallow appPreferences: AppPreferenceRegistration[] = []
@observable.shallow clusterFeatures: ClusterFeatureRegistration[] = [] @observable.shallow clusterFeatures: ClusterFeatureRegistration[] = []
@observable.shallow statusBarItems: StatusBarRegistration[] = [] @observable.shallow statusBarItems: StatusBarRegistration[] = []

View File

@ -1,3 +1,4 @@
import type React from "react"
import { BaseRegistry } from "./base-registry"; import { BaseRegistry } from "./base-registry";
import { ClusterFeature } from "../cluster-feature"; import { ClusterFeature } from "../cluster-feature";

View File

@ -1,6 +1,7 @@
// All registries managed by extensions api // All registries managed by extensions api
export * from "./page-registry" export * from "./page-registry"
export * from "./page-menu-registry"
export * from "./menu-registry" export * from "./menu-registry"
export * from "./app-preference-registry" export * from "./app-preference-registry"
export * from "./status-bar-registry" export * from "./status-bar-registry"

View File

@ -0,0 +1,25 @@
// Extensions-api -> Register page menu items
import type React from "react";
import type { IconProps } from "../../renderer/components/icon";
import { BaseRegistry } from "./base-registry";
export interface PageMenuRegistration {
url: string;
title: React.ReactNode;
components: PageMenuComponents;
}
export interface PageMenuRegistrationCluster extends PageMenuRegistration {
subMenus?: Omit<PageMenuRegistration, "components" | "subMenus">[];
}
export interface PageMenuComponents {
Icon: React.ComponentType<IconProps>;
}
export class PageMenuRegistry<T extends PageMenuRegistration> extends BaseRegistry<T> {
}
export const globalPageMenuRegistry = new PageMenuRegistry<PageMenuRegistration>();
export const clusterPageMenuRegistry = new PageMenuRegistry<PageMenuRegistrationCluster>();

View File

@ -1,31 +1,29 @@
// Extensions-api -> Custom page registration // Extensions-api -> Custom page registration
import type React from "react"; import type React from "react";
import type { RouteProps } from "react-router";
import type { IconProps } from "../../renderer/components/icon";
import type { IClassName } from "../../renderer/utils";
import type { TabRoute } from "../../renderer/components/layout/tab-layout";
import { BaseRegistry } from "./base-registry"; import { BaseRegistry } from "./base-registry";
export interface PageRegistration extends RouteProps { export interface PageRegistration {
className?: IClassName; routePath: string; // react-router's path, e.g. "/page/:id"
url?: string; // initial url to be used for building menus and tabs, otherwise "path" applied by default exact?: boolean; // route matching flag, see: https://reactrouter.com/web/api/NavLink/exact-bool
title?: React.ReactNode; // used in sidebar's & tabs-layout if provided
hideInMenu?: boolean; // hide element within app's navigation menu
subPages?: (PageRegistration & TabRoute)[];
components: PageComponents; components: PageComponents;
} }
export interface PageRegistrationCluster extends PageRegistration {
subPages?: Omit<PageRegistration, "subPages">;
}
export interface PageComponents { export interface PageComponents {
Page: React.ComponentType<any>; Page: React.ComponentType<any>;
MenuIcon?: React.ComponentType<IconProps>;
} }
export class GlobalPageRegistry extends BaseRegistry<PageRegistration> { export class PageRegistry<T extends PageRegistration> extends BaseRegistry<T> {
protected routePrefixPath = "/extensions/:name" // todo: figure out how to provide inside extension
getItems() {
return super.getItems();
}
} }
export class ClusterPageRegistry extends BaseRegistry<PageRegistration> { export const globalPageRegistry = new PageRegistry<PageRegistration>();
} export const clusterPageRegistry = new PageRegistry<PageRegistrationCluster>();
export const globalPageRegistry = new GlobalPageRegistry();
export const clusterPageRegistry = new ClusterPageRegistry();

View File

@ -1,3 +1,2 @@
export { navigate, hideDetails, showDetails, getDetailsUrl } from "../../renderer/navigation" export { navigate, hideDetails, showDetails, getDetailsUrl } from "../../renderer/navigation"
export { RouteProps } from "react-router"
export { IURLParams } from "../../common/utils/buildUrl"; export { IURLParams } from "../../common/utils/buildUrl";

View File

@ -1,41 +1,34 @@
import React from "react"; import React from "react";
import { observer } from "mobx-react"; import { observer } from "mobx-react";
import { Redirect, Route, Switch } from "react-router";
import { Trans } from "@lingui/macro"; import { Trans } from "@lingui/macro";
import { TabLayout, TabRoute } from "../layout/tab-layout"; import { TabLayout, TabLayoutRoute } from "../layout/tab-layout";
import { HelmCharts, helmChartsRoute, helmChartsURL } from "../+apps-helm-charts"; import { HelmCharts, helmChartsRoute, helmChartsURL } from "../+apps-helm-charts";
import { HelmReleases, releaseRoute, releaseURL } from "../+apps-releases"; import { HelmReleases, releaseRoute, releaseURL } from "../+apps-releases";
import { namespaceStore } from "../+namespaces/namespace.store"; import { namespaceStore } from "../+namespaces/namespace.store";
@observer @observer
export class Apps extends React.Component { export class Apps extends React.Component {
static get tabRoutes(): TabRoute[] { static get tabRoutes(): TabLayoutRoute[] {
const query = namespaceStore.getContextParams(); const query = namespaceStore.getContextParams();
return [ return [
{ {
title: <Trans>Charts</Trans>, title: <Trans>Charts</Trans>,
component: HelmCharts, component: HelmCharts,
url: helmChartsURL(), url: helmChartsURL(),
path: helmChartsRoute.path, routePath: helmChartsRoute.path.toString(),
}, },
{ {
title: <Trans>Releases</Trans>, title: <Trans>Releases</Trans>,
component: HelmReleases, component: HelmReleases,
url: releaseURL({ query }), url: releaseURL({ query }),
path: releaseRoute.path, routePath: releaseRoute.path.toString(),
}, },
] ]
} }
render() { render() {
const tabRoutes = Apps.tabRoutes;
return ( return (
<TabLayout className="Apps" tabs={tabRoutes}> <TabLayout className="Apps" tabs={Apps.tabRoutes}/>
<Switch>
{tabRoutes.map((route, index) => <Route key={index} {...route}/>)}
<Redirect to={tabRoutes[0].url}/>
</Switch>
</TabLayout>
) )
} }
} }

View File

@ -5,7 +5,7 @@ import { configMapsURL } from "../+config-maps/config-maps.route";
export const configRoute: RouteProps = { export const configRoute: RouteProps = {
get path() { get path() {
return Config.tabRoutes.map(({ path }) => path).flat() return Config.tabRoutes.map(({ routePath }) => routePath).flat()
} }
} }

View File

@ -1,33 +1,26 @@
import React from "react"; import React from "react";
import { observer } from "mobx-react"; import { observer } from "mobx-react";
import { Redirect, Route, Switch } from "react-router";
import { Trans } from "@lingui/macro"; import { Trans } from "@lingui/macro";
import { TabLayout, TabRoute } from "../layout/tab-layout"; import { TabLayout, TabLayoutRoute } from "../layout/tab-layout";
import { ConfigMaps, configMapsRoute, configMapsURL } from "../+config-maps"; import { ConfigMaps, configMapsRoute, configMapsURL } from "../+config-maps";
import { Secrets, secretsRoute, secretsURL } from "../+config-secrets"; import { Secrets, secretsRoute, secretsURL } from "../+config-secrets";
import { namespaceStore } from "../+namespaces/namespace.store"; import { namespaceStore } from "../+namespaces/namespace.store";
import { resourceQuotaRoute, ResourceQuotas, resourceQuotaURL } from "../+config-resource-quotas"; import { resourceQuotaRoute, ResourceQuotas, resourceQuotaURL } from "../+config-resource-quotas";
import { PodDisruptionBudgets, pdbRoute, pdbURL } from "../+config-pod-disruption-budgets"; import { pdbRoute, pdbURL, PodDisruptionBudgets } from "../+config-pod-disruption-budgets";
import { configURL } from "./config.route";
import { HorizontalPodAutoscalers, hpaRoute, hpaURL } from "../+config-autoscalers"; import { HorizontalPodAutoscalers, hpaRoute, hpaURL } from "../+config-autoscalers";
import { isAllowedResource } from "../../../common/rbac" import { isAllowedResource } from "../../../common/rbac"
import { buildURL } from "../../../common/utils/buildUrl";
export const certificatesURL = buildURL("/certificates");
export const issuersURL = buildURL("/issuers");
export const clusterIssuersURL = buildURL("/clusterissuers");
@observer @observer
export class Config extends React.Component { export class Config extends React.Component {
static get tabRoutes(): TabRoute[] { static get tabRoutes(): TabLayoutRoute[] {
const query = namespaceStore.getContextParams() const query = namespaceStore.getContextParams()
const routes: TabRoute[] = [] const routes: TabLayoutRoute[] = []
if (isAllowedResource("configmaps")) { if (isAllowedResource("configmaps")) {
routes.push({ routes.push({
title: <Trans>ConfigMaps</Trans>, title: <Trans>ConfigMaps</Trans>,
component: ConfigMaps, component: ConfigMaps,
url: configMapsURL({ query }), url: configMapsURL({ query }),
path: configMapsRoute.path, routePath: configMapsRoute.path.toString(),
}) })
} }
if (isAllowedResource("secrets")) { if (isAllowedResource("secrets")) {
@ -35,7 +28,7 @@ export class Config extends React.Component {
title: <Trans>Secrets</Trans>, title: <Trans>Secrets</Trans>,
component: Secrets, component: Secrets,
url: secretsURL({ query }), url: secretsURL({ query }),
path: secretsRoute.path, routePath: secretsRoute.path.toString(),
}) })
} }
if (isAllowedResource("resourcequotas")) { if (isAllowedResource("resourcequotas")) {
@ -43,7 +36,7 @@ export class Config extends React.Component {
title: <Trans>Resource Quotas</Trans>, title: <Trans>Resource Quotas</Trans>,
component: ResourceQuotas, component: ResourceQuotas,
url: resourceQuotaURL({ query }), url: resourceQuotaURL({ query }),
path: resourceQuotaRoute.path, routePath: resourceQuotaRoute.path.toString(),
}) })
} }
if (isAllowedResource("horizontalpodautoscalers")) { if (isAllowedResource("horizontalpodautoscalers")) {
@ -51,7 +44,7 @@ export class Config extends React.Component {
title: <Trans>HPA</Trans>, title: <Trans>HPA</Trans>,
component: HorizontalPodAutoscalers, component: HorizontalPodAutoscalers,
url: hpaURL({ query }), url: hpaURL({ query }),
path: hpaRoute.path, routePath: hpaRoute.path.toString(),
}) })
} }
if (isAllowedResource("poddisruptionbudgets")) { if (isAllowedResource("poddisruptionbudgets")) {
@ -59,21 +52,15 @@ export class Config extends React.Component {
title: <Trans>Pod Disruption Budgets</Trans>, title: <Trans>Pod Disruption Budgets</Trans>,
component: PodDisruptionBudgets, component: PodDisruptionBudgets,
url: pdbURL({ query }), url: pdbURL({ query }),
path: pdbRoute.path, routePath: pdbRoute.path.toString(),
}) })
} }
return routes; return routes;
} }
render() { render() {
const tabRoutes = Config.tabRoutes;
return ( return (
<TabLayout className="Config" tabs={tabRoutes}> <TabLayout className="Config" tabs={Config.tabRoutes}/>
<Switch>
{tabRoutes.map((route, index) => <Route key={index} {...route}/>)}
<Redirect to={configURL({ query: namespaceStore.getContextParams() })}/>
</Switch>
</TabLayout>
) )
} }
} }

View File

@ -2,23 +2,20 @@ import React from "react";
import { observer } from "mobx-react"; import { observer } from "mobx-react";
import { Redirect, Route, Switch } from "react-router"; import { Redirect, Route, Switch } from "react-router";
import { Trans } from "@lingui/macro"; import { Trans } from "@lingui/macro";
import { TabLayout, TabRoute } from "../layout/tab-layout"; import { TabLayout, TabLayoutRoute } from "../layout/tab-layout";
import { crdResourcesRoute, crdRoute, crdURL, crdDefinitionsRoute } from "./crd.route"; import { crdResourcesRoute, crdRoute, crdURL, crdDefinitionsRoute } from "./crd.route";
import { CrdList } from "./crd-list"; import { CrdList } from "./crd-list";
import { CrdResources } from "./crd-resources"; import { CrdResources } from "./crd-resources";
// todo: next steps - customization via plugins
// todo: list views (rows content), full details view and if possible chart/prometheus hooks
@observer @observer
export class CustomResources extends React.Component { export class CustomResources extends React.Component {
static get tabRoutes(): TabRoute[] { static get tabRoutes(): TabLayoutRoute[] {
return [ return [
{ {
title: <Trans>Definitions</Trans>, title: <Trans>Definitions</Trans>,
component: CustomResources, component: CustomResources,
url: crdURL(), url: crdURL(),
path: crdRoute.path, routePath: crdRoute.path.toString(),
} }
] ]
} }

View File

@ -5,7 +5,7 @@ import { IURLParams } from "../../../common/utils/buildUrl";
export const networkRoute: RouteProps = { export const networkRoute: RouteProps = {
get path() { get path() {
return Network.tabRoutes.map(({ path }) => path).flat() return Network.tabRoutes.map(({ routePath }) => routePath).flat()
} }
} }

View File

@ -2,32 +2,26 @@ import "./network.scss"
import React from "react"; import React from "react";
import { observer } from "mobx-react"; import { observer } from "mobx-react";
import { Redirect, Route, Switch } from "react-router";
import { RouteComponentProps } from "react-router-dom";
import { Trans } from "@lingui/macro"; import { Trans } from "@lingui/macro";
import { TabLayout, TabRoute } from "../layout/tab-layout"; import { TabLayout, TabLayoutRoute } from "../layout/tab-layout";
import { Services, servicesRoute, servicesURL } from "../+network-services"; import { Services, servicesRoute, servicesURL } from "../+network-services";
import { Endpoints, endpointRoute, endpointURL } from "../+network-endpoints"; import { endpointRoute, Endpoints, endpointURL } from "../+network-endpoints";
import { Ingresses, ingressRoute, ingressURL } from "../+network-ingresses"; import { Ingresses, ingressRoute, ingressURL } from "../+network-ingresses";
import { NetworkPolicies, networkPoliciesRoute, networkPoliciesURL } from "../+network-policies"; import { NetworkPolicies, networkPoliciesRoute, networkPoliciesURL } from "../+network-policies";
import { namespaceStore } from "../+namespaces/namespace.store"; import { namespaceStore } from "../+namespaces/namespace.store";
import { networkURL } from "./network.route";
import { isAllowedResource } from "../../../common/rbac"; import { isAllowedResource } from "../../../common/rbac";
interface Props extends RouteComponentProps<{}> {
}
@observer @observer
export class Network extends React.Component<Props> { export class Network extends React.Component {
static get tabRoutes(): TabRoute[] { static get tabRoutes(): TabLayoutRoute[] {
const query = namespaceStore.getContextParams() const query = namespaceStore.getContextParams()
const routes: TabRoute[] = []; const routes: TabLayoutRoute[] = [];
if (isAllowedResource("services")) { if (isAllowedResource("services")) {
routes.push({ routes.push({
title: <Trans>Services</Trans>, title: <Trans>Services</Trans>,
component: Services, component: Services,
url: servicesURL({ query }), url: servicesURL({ query }),
path: servicesRoute.path, routePath: servicesRoute.path.toString(),
}) })
} }
if (isAllowedResource("endpoints")) { if (isAllowedResource("endpoints")) {
@ -35,7 +29,7 @@ export class Network extends React.Component<Props> {
title: <Trans>Endpoints</Trans>, title: <Trans>Endpoints</Trans>,
component: Endpoints, component: Endpoints,
url: endpointURL({ query }), url: endpointURL({ query }),
path: endpointRoute.path, routePath: endpointRoute.path.toString(),
}) })
} }
if (isAllowedResource("ingresses")) { if (isAllowedResource("ingresses")) {
@ -43,7 +37,7 @@ export class Network extends React.Component<Props> {
title: <Trans>Ingresses</Trans>, title: <Trans>Ingresses</Trans>,
component: Ingresses, component: Ingresses,
url: ingressURL({ query }), url: ingressURL({ query }),
path: ingressRoute.path, routePath: ingressRoute.path.toString(),
}) })
} }
if (isAllowedResource("networkpolicies")) { if (isAllowedResource("networkpolicies")) {
@ -51,21 +45,15 @@ export class Network extends React.Component<Props> {
title: <Trans>Network Policies</Trans>, title: <Trans>Network Policies</Trans>,
component: NetworkPolicies, component: NetworkPolicies,
url: networkPoliciesURL({ query }), url: networkPoliciesURL({ query }),
path: networkPoliciesRoute.path, routePath: networkPoliciesRoute.path.toString(),
}) })
} }
return routes return routes
} }
render() { render() {
const tabRoutes = Network.tabRoutes;
return ( return (
<TabLayout className="Network" tabs={tabRoutes}> <TabLayout className="Network" tabs={Network.tabRoutes}/>
<Switch>
{tabRoutes.map((route, index) => <Route key={index} {...route}/>)}
<Redirect to={networkURL({ query: namespaceStore.getContextParams() })}/>
</Switch>
</TabLayout>
) )
} }
} }

View File

@ -5,7 +5,7 @@ import { IURLParams } from "../../../common/utils/buildUrl";
export const storageRoute: RouteProps = { export const storageRoute: RouteProps = {
get path() { get path() {
return Storage.tabRoutes.map(({ path }) => path).flat() return Storage.tabRoutes.map(({ routePath }) => routePath).flat()
} }
} }

View File

@ -2,31 +2,25 @@ import "./storage.scss"
import React from "react"; import React from "react";
import { observer } from "mobx-react"; import { observer } from "mobx-react";
import { Redirect, Route, Switch } from "react-router";
import { RouteComponentProps } from "react-router-dom";
import { Trans } from "@lingui/macro"; import { Trans } from "@lingui/macro";
import { TabLayout, TabRoute } from "../layout/tab-layout"; import { TabLayout, TabLayoutRoute } from "../layout/tab-layout";
import { PersistentVolumes, volumesRoute, volumesURL } from "../+storage-volumes"; import { PersistentVolumes, volumesRoute, volumesURL } from "../+storage-volumes";
import { StorageClasses, storageClassesRoute, storageClassesURL } from "../+storage-classes"; import { StorageClasses, storageClassesRoute, storageClassesURL } from "../+storage-classes";
import { PersistentVolumeClaims, volumeClaimsRoute, volumeClaimsURL } from "../+storage-volume-claims"; import { PersistentVolumeClaims, volumeClaimsRoute, volumeClaimsURL } from "../+storage-volume-claims";
import { namespaceStore } from "../+namespaces/namespace.store"; import { namespaceStore } from "../+namespaces/namespace.store";
import { storageURL } from "./storage.route";
import { isAllowedResource } from "../../../common/rbac"; import { isAllowedResource } from "../../../common/rbac";
interface Props extends RouteComponentProps<{}> {
}
@observer @observer
export class Storage extends React.Component<Props> { export class Storage extends React.Component {
static get tabRoutes() { static get tabRoutes() {
const tabRoutes: TabRoute[] = []; const tabRoutes: TabLayoutRoute[] = [];
const query = namespaceStore.getContextParams() const query = namespaceStore.getContextParams()
tabRoutes.push({ tabRoutes.push({
title: <Trans>Persistent Volume Claims</Trans>, title: <Trans>Persistent Volume Claims</Trans>,
component: PersistentVolumeClaims, component: PersistentVolumeClaims,
url: volumeClaimsURL({ query }), url: volumeClaimsURL({ query }),
path: volumeClaimsRoute.path, routePath: volumeClaimsRoute.path.toString(),
}) })
if (isAllowedResource('persistentvolumes')) { if (isAllowedResource('persistentvolumes')) {
@ -34,7 +28,7 @@ export class Storage extends React.Component<Props> {
title: <Trans>Persistent Volumes</Trans>, title: <Trans>Persistent Volumes</Trans>,
component: PersistentVolumes, component: PersistentVolumes,
url: volumesURL(), url: volumesURL(),
path: volumesRoute.path, routePath: volumesRoute.path.toString(),
}); });
} }
@ -43,21 +37,15 @@ export class Storage extends React.Component<Props> {
title: <Trans>Storage Classes</Trans>, title: <Trans>Storage Classes</Trans>,
component: StorageClasses, component: StorageClasses,
url: storageClassesURL(), url: storageClassesURL(),
path: storageClassesRoute.path, routePath: storageClassesRoute.path.toString(),
}) })
} }
return tabRoutes; return tabRoutes;
} }
render() { render() {
const tabRoutes = Storage.tabRoutes;
return ( return (
<TabLayout className="Storage" tabs={tabRoutes}> <TabLayout className="Storage" tabs={Storage.tabRoutes}/>
<Switch>
{tabRoutes.map((route, index) => <Route key={index} {...route}/>)}
<Redirect to={storageURL({ query: namespaceStore.getContextParams() })}/>
</Switch>
</TabLayout>
) )
} }
} }

View File

@ -4,7 +4,7 @@ import { UserManagement } from "./user-management"
export const usersManagementRoute: RouteProps = { export const usersManagementRoute: RouteProps = {
get path() { get path() {
return UserManagement.tabRoutes.map(({ path }) => path).flat() return UserManagement.tabRoutes.map(({ routePath }) => routePath).flat()
} }
} }

View File

@ -1,44 +1,39 @@
import "./user-management.scss" import "./user-management.scss"
import React from "react"; import React from "react";
import { observer } from "mobx-react"; import { observer } from "mobx-react";
import { Redirect, Route, Switch } from "react-router";
import { RouteComponentProps } from "react-router-dom";
import { Trans } from "@lingui/macro"; import { Trans } from "@lingui/macro";
import { TabLayout, TabRoute } from "../layout/tab-layout"; import { TabLayout, TabLayoutRoute } from "../layout/tab-layout";
import { Roles } from "../+user-management-roles"; import { Roles } from "../+user-management-roles";
import { RoleBindings } from "../+user-management-roles-bindings"; import { RoleBindings } from "../+user-management-roles-bindings";
import { ServiceAccounts } from "../+user-management-service-accounts"; import { ServiceAccounts } from "../+user-management-service-accounts";
import { roleBindingsRoute, roleBindingsURL, rolesRoute, rolesURL, serviceAccountsRoute, serviceAccountsURL, usersManagementURL } from "./user-management.route"; import { roleBindingsRoute, roleBindingsURL, rolesRoute, rolesURL, serviceAccountsRoute, serviceAccountsURL } from "./user-management.route";
import { namespaceStore } from "../+namespaces/namespace.store"; import { namespaceStore } from "../+namespaces/namespace.store";
import { PodSecurityPolicies, podSecurityPoliciesRoute, podSecurityPoliciesURL } from "../+pod-security-policies"; import { PodSecurityPolicies, podSecurityPoliciesRoute, podSecurityPoliciesURL } from "../+pod-security-policies";
import { isAllowedResource } from "../../../common/rbac"; import { isAllowedResource } from "../../../common/rbac";
interface Props extends RouteComponentProps<{}> {
}
@observer @observer
export class UserManagement extends React.Component<Props> { export class UserManagement extends React.Component {
static get tabRoutes() { static get tabRoutes() {
const tabRoutes: TabRoute[] = []; const tabRoutes: TabLayoutRoute[] = [];
const query = namespaceStore.getContextParams() const query = namespaceStore.getContextParams()
tabRoutes.push( tabRoutes.push(
{ {
title: <Trans>Service Accounts</Trans>, title: <Trans>Service Accounts</Trans>,
component: ServiceAccounts, component: ServiceAccounts,
url: serviceAccountsURL({ query }), url: serviceAccountsURL({ query }),
path: serviceAccountsRoute.path, routePath: serviceAccountsRoute.path.toString(),
}, },
{ {
title: <Trans>Role Bindings</Trans>, title: <Trans>Role Bindings</Trans>,
component: RoleBindings, component: RoleBindings,
url: roleBindingsURL({ query }), url: roleBindingsURL({ query }),
path: roleBindingsRoute.path, routePath: roleBindingsRoute.path.toString(),
}, },
{ {
title: <Trans>Roles</Trans>, title: <Trans>Roles</Trans>,
component: Roles, component: Roles,
url: rolesURL({ query }), url: rolesURL({ query }),
path: rolesRoute.path, routePath: rolesRoute.path.toString(),
}, },
) )
if (isAllowedResource("podsecuritypolicies")) { if (isAllowedResource("podsecuritypolicies")) {
@ -46,21 +41,15 @@ export class UserManagement extends React.Component<Props> {
title: <Trans>Pod Security Policies</Trans>, title: <Trans>Pod Security Policies</Trans>,
component: PodSecurityPolicies, component: PodSecurityPolicies,
url: podSecurityPoliciesURL(), url: podSecurityPoliciesURL(),
path: podSecurityPoliciesRoute.path, routePath: podSecurityPoliciesRoute.path.toString(),
}) })
} }
return tabRoutes; return tabRoutes;
} }
render() { render() {
const tabRoutes = UserManagement.tabRoutes;
return ( return (
<TabLayout className="UserManagement" tabs={tabRoutes}> <TabLayout className="UserManagement" tabs={UserManagement.tabRoutes}/>
<Switch>
{tabRoutes.map((route, index) => <Route key={index} {...route}/>)}
<Redirect to={usersManagementURL({ query: namespaceStore.getContextParams() })}/>
</Switch>
</TabLayout>
) )
} }
} }

View File

@ -5,7 +5,7 @@ import { Workloads } from "./workloads";
export const workloadsRoute: RouteProps = { export const workloadsRoute: RouteProps = {
get path() { get path() {
return Workloads.tabRoutes.map(({ path }) => path).flat() return Workloads.tabRoutes.map(({ routePath }) => routePath).flat()
} }
} }

View File

@ -2,12 +2,10 @@ import "./workloads.scss"
import React from "react"; import React from "react";
import { observer } from "mobx-react"; import { observer } from "mobx-react";
import { Redirect, Route, Switch } from "react-router";
import { RouteComponentProps } from "react-router-dom";
import { Trans } from "@lingui/macro"; import { Trans } from "@lingui/macro";
import { TabLayout, TabRoute } from "../layout/tab-layout"; import { TabLayout, TabLayoutRoute } from "../layout/tab-layout";
import { WorkloadsOverview } from "../+workloads-overview/overview"; import { WorkloadsOverview } from "../+workloads-overview/overview";
import { cronJobsRoute, cronJobsURL, daemonSetsRoute, daemonSetsURL, deploymentsRoute, deploymentsURL, jobsRoute, jobsURL, overviewRoute, overviewURL, podsRoute, podsURL, statefulSetsRoute, statefulSetsURL, workloadsURL } from "./workloads.route"; import { cronJobsRoute, cronJobsURL, daemonSetsRoute, daemonSetsURL, deploymentsRoute, deploymentsURL, jobsRoute, jobsURL, overviewRoute, overviewURL, podsRoute, podsURL, statefulSetsRoute, statefulSetsURL } from "./workloads.route";
import { namespaceStore } from "../+namespaces/namespace.store"; import { namespaceStore } from "../+namespaces/namespace.store";
import { Pods } from "../+workloads-pods"; import { Pods } from "../+workloads-pods";
import { Deployments } from "../+workloads-deployments"; import { Deployments } from "../+workloads-deployments";
@ -17,19 +15,16 @@ import { Jobs } from "../+workloads-jobs";
import { CronJobs } from "../+workloads-cronjobs"; import { CronJobs } from "../+workloads-cronjobs";
import { isAllowedResource } from "../../../common/rbac" import { isAllowedResource } from "../../../common/rbac"
interface Props extends RouteComponentProps {
}
@observer @observer
export class Workloads extends React.Component<Props> { export class Workloads extends React.Component {
static get tabRoutes(): TabRoute[] { static get tabRoutes(): TabLayoutRoute[] {
const query = namespaceStore.getContextParams(); const query = namespaceStore.getContextParams();
const routes: TabRoute[] = [ const routes: TabLayoutRoute[] = [
{ {
title: <Trans>Overview</Trans>, title: <Trans>Overview</Trans>,
component: WorkloadsOverview, component: WorkloadsOverview,
url: overviewURL({ query }), url: overviewURL({ query }),
path: overviewRoute.path routePath: overviewRoute.path.toString()
} }
] ]
if (isAllowedResource("pods")) { if (isAllowedResource("pods")) {
@ -37,7 +32,7 @@ export class Workloads extends React.Component<Props> {
title: <Trans>Pods</Trans>, title: <Trans>Pods</Trans>,
component: Pods, component: Pods,
url: podsURL({ query }), url: podsURL({ query }),
path: podsRoute.path routePath: podsRoute.path.toString()
}) })
} }
if (isAllowedResource("deployments")) { if (isAllowedResource("deployments")) {
@ -45,7 +40,7 @@ export class Workloads extends React.Component<Props> {
title: <Trans>Deployments</Trans>, title: <Trans>Deployments</Trans>,
component: Deployments, component: Deployments,
url: deploymentsURL({ query }), url: deploymentsURL({ query }),
path: deploymentsRoute.path, routePath: deploymentsRoute.path.toString(),
}) })
} }
if (isAllowedResource("daemonsets")) { if (isAllowedResource("daemonsets")) {
@ -53,7 +48,7 @@ export class Workloads extends React.Component<Props> {
title: <Trans>DaemonSets</Trans>, title: <Trans>DaemonSets</Trans>,
component: DaemonSets, component: DaemonSets,
url: daemonSetsURL({ query }), url: daemonSetsURL({ query }),
path: daemonSetsRoute.path, routePath: daemonSetsRoute.path.toString(),
}) })
} }
if (isAllowedResource("statefulsets")) { if (isAllowedResource("statefulsets")) {
@ -61,7 +56,7 @@ export class Workloads extends React.Component<Props> {
title: <Trans>StatefulSets</Trans>, title: <Trans>StatefulSets</Trans>,
component: StatefulSets, component: StatefulSets,
url: statefulSetsURL({ query }), url: statefulSetsURL({ query }),
path: statefulSetsRoute.path, routePath: statefulSetsRoute.path.toString(),
}) })
} }
if (isAllowedResource("jobs")) { if (isAllowedResource("jobs")) {
@ -69,7 +64,7 @@ export class Workloads extends React.Component<Props> {
title: <Trans>Jobs</Trans>, title: <Trans>Jobs</Trans>,
component: Jobs, component: Jobs,
url: jobsURL({ query }), url: jobsURL({ query }),
path: jobsRoute.path, routePath: jobsRoute.path.toString(),
}) })
} }
if (isAllowedResource("cronjobs")) { if (isAllowedResource("cronjobs")) {
@ -77,21 +72,15 @@ export class Workloads extends React.Component<Props> {
title: <Trans>CronJobs</Trans>, title: <Trans>CronJobs</Trans>,
component: CronJobs, component: CronJobs,
url: cronJobsURL({ query }), url: cronJobsURL({ query }),
path: cronJobsRoute.path, routePath: cronJobsRoute.path.toString(),
}) })
} }
return routes; return routes;
} }
render() { render() {
const tabRoutes = Workloads.tabRoutes;
return ( return (
<TabLayout className="Workloads" tabs={tabRoutes}> <TabLayout className="Workloads" tabs={Workloads.tabRoutes}/>
<Switch>
{tabRoutes.map((route, index) => <Route key={index} {...route}/>)}
<Redirect to={workloadsURL({ query: namespaceStore.getContextParams() })}/>
</Switch>
</TabLayout>
) )
} }
} }

View File

@ -29,6 +29,7 @@ import { CustomResources } from "./+custom-resources/custom-resources";
import { crdRoute } from "./+custom-resources"; import { crdRoute } from "./+custom-resources";
import { isAllowedResource } from "../../common/rbac"; import { isAllowedResource } from "../../common/rbac";
import { MainLayout } from "./layout/main-layout"; import { MainLayout } from "./layout/main-layout";
import { TabLayout, TabLayoutRoute } from "./layout/tab-layout";
import { ErrorBoundary } from "./error-boundary"; import { ErrorBoundary } from "./error-boundary";
import { Terminal } from "./dock/terminal"; import { Terminal } from "./dock/terminal";
import { getHostedCluster, getHostedClusterId } from "../../common/cluster-store"; import { getHostedCluster, getHostedClusterId } from "../../common/cluster-store";
@ -36,7 +37,6 @@ import logger from "../../main/logger";
import { clusterIpc } from "../../common/cluster-ipc"; import { clusterIpc } from "../../common/cluster-ipc";
import { webFrame } from "electron"; import { webFrame } from "electron";
import { clusterPageRegistry } from "../../extensions/registries/page-registry"; import { clusterPageRegistry } from "../../extensions/registries/page-registry";
import { DynamicPage } from "../../extensions/dynamic-page";
import { extensionLoader } from "../../extensions/extension-loader"; import { extensionLoader } from "../../extensions/extension-loader";
import { appEventBus } from "../../common/event-bus"; import { appEventBus } from "../../common/event-bus";
import whatInput from 'what-input'; import whatInput from 'what-input';
@ -52,9 +52,13 @@ export class App extends React.Component {
await clusterIpc.setFrameId.invokeFromRenderer(clusterId, frameId); await clusterIpc.setFrameId.invokeFromRenderer(clusterId, frameId);
await getHostedCluster().whenReady; // cluster.activate() is done at this point await getHostedCluster().whenReady; // cluster.activate() is done at this point
extensionLoader.loadOnClusterRenderer(); extensionLoader.loadOnClusterRenderer();
appEventBus.emit({name: "cluster", action: "open", params: { appEventBus.emit({
clusterId: clusterId name: "cluster",
}}) action: "open",
params: {
clusterId: clusterId
}
})
window.addEventListener("online", () => { window.addEventListener("online", () => {
window.location.reload() window.location.reload()
}) })
@ -86,12 +90,19 @@ export class App extends React.Component {
<Route component={CustomResources} {...crdRoute}/> <Route component={CustomResources} {...crdRoute}/>
<Route component={UserManagement} {...usersManagementRoute}/> <Route component={UserManagement} {...usersManagementRoute}/>
<Route component={Apps} {...appsRoute}/> <Route component={Apps} {...appsRoute}/>
{clusterPageRegistry.getItems().map(page => { {clusterPageRegistry.getItems().map(({ components: { Page }, subPages = [], exact, routePath }) => {
return <Route {...page} key={String(page.path)} render={() => <DynamicPage page={page}/>}/> // return (
// <Route key={routePath} path={routePath} exact={exact} render={() => (
// <TabLayout tabs={subPages}>
// <Page/>
// </TabLayout>
// )}/>
// )
})} })}
<Redirect exact from="/" to={this.startURL}/> <Redirect exact from="/" to={this.startURL}/>
<Route component={NotFound}/> <Route component={NotFound}/>
</Switch></MainLayout> </Switch>
</MainLayout>
<Notifications/> <Notifications/>
<ConfirmDialog/> <ConfirmDialog/>
<KubeObjectDetails/> <KubeObjectDetails/>

View File

@ -69,8 +69,8 @@ export class ClusterManager extends React.Component {
<Route component={AddCluster} {...addClusterRoute} /> <Route component={AddCluster} {...addClusterRoute} />
<Route component={ClusterView} {...clusterViewRoute} /> <Route component={ClusterView} {...clusterViewRoute} />
<Route component={ClusterSettings} {...clusterSettingsRoute} /> <Route component={ClusterSettings} {...clusterSettingsRoute} />
{globalPageRegistry.getItems().map(({ path, url = String(path), components: { Page } }) => { {globalPageRegistry.getItems().map(({ routePath, exact, components: { Page } }) => {
return <Route key={url} path={path} component={Page}/> return <Route key={routePath} path={routePath} component={Page} exact={exact}/>
})} })}
<Redirect exact to={this.startUrl}/> <Redirect exact to={this.startUrl}/>
</Switch> </Switch>

View File

@ -5,6 +5,7 @@ import { remote } from "electron"
import type { Cluster } from "../../../main/cluster"; import type { Cluster } from "../../../main/cluster";
import { DragDropContext, Draggable, DraggableProvided, Droppable, DroppableProvided, DropResult } from "react-beautiful-dnd"; import { DragDropContext, Draggable, DraggableProvided, Droppable, DroppableProvided, DropResult } from "react-beautiful-dnd";
import { observer } from "mobx-react"; import { observer } from "mobx-react";
import { matchPath } from "react-router";
import { _i18n } from "../../i18n"; import { _i18n } from "../../i18n";
import { t, Trans } from "@lingui/macro"; import { t, Trans } from "@lingui/macro";
import { userStore } from "../../../common/user-store"; import { userStore } from "../../../common/user-store";
@ -14,7 +15,7 @@ import { ClusterIcon } from "../cluster-icon";
import { Icon } from "../icon"; import { Icon } from "../icon";
import { autobind, cssNames, IClassName } from "../../utils"; import { autobind, cssNames, IClassName } from "../../utils";
import { Badge } from "../badge"; import { Badge } from "../badge";
import { navigate } from "../../navigation"; import { navigate, navigation } from "../../navigation";
import { addClusterURL } from "../+add-cluster"; import { addClusterURL } from "../+add-cluster";
import { clusterSettingsURL } from "../+cluster-settings"; import { clusterSettingsURL } from "../+cluster-settings";
import { landingURL } from "../+landing-page"; import { landingURL } from "../+landing-page";
@ -22,7 +23,7 @@ import { Tooltip } from "../tooltip";
import { ConfirmDialog } from "../confirm-dialog"; import { ConfirmDialog } from "../confirm-dialog";
import { clusterIpc } from "../../../common/cluster-ipc"; import { clusterIpc } from "../../../common/cluster-ipc";
import { clusterViewURL } from "./cluster-view.route"; import { clusterViewURL } from "./cluster-view.route";
import { globalPageRegistry } from "../../../extensions/registries/page-registry"; import { globalPageMenuRegistry } from "../../../extensions/registries";
interface Props { interface Props {
className?: IClassName; className?: IClassName;
@ -138,7 +139,7 @@ export class ClustersMenu extends React.Component<Props> {
</Droppable> </Droppable>
</DragDropContext> </DragDropContext>
</div> </div>
<div className="add-cluster" > <div className="add-cluster">
<Tooltip targetId="add-cluster-icon"> <Tooltip targetId="add-cluster-icon">
<Trans>Add Cluster</Trans> <Trans>Add Cluster</Trans>
</Tooltip> </Tooltip>
@ -148,9 +149,17 @@ export class ClustersMenu extends React.Component<Props> {
)} )}
</div> </div>
<div className="extensions"> <div className="extensions">
{globalPageRegistry.getItems().map(({ path, url = String(path), hideInMenu, components: { MenuIcon } }) => { {globalPageMenuRegistry.getItems().map(({ title, url, components: { Icon } }) => {
if (!MenuIcon || hideInMenu) return; const routePath = "" // todo: find matching route in page-registry + exact
return <MenuIcon key={url} onClick={() => navigate(url)}/> const isActive = !!matchPath(navigation.location.pathname, { path: routePath/*, exact: false*/ });
return (
<Icon
key={routePath}
tooltip={title}
active={isActive}
onClick={() => navigate(url)}
/>
)
})} })}
</div> </div>
</div> </div>

View File

@ -1,4 +1,4 @@
import type { TabRoute } from "./tab-layout"; import type { TabLayoutRoute } from "./tab-layout";
import "./sidebar.scss"; import "./sidebar.scss";
import React from "react"; import React from "react";
@ -27,9 +27,9 @@ import { crdStore } from "../+custom-resources/crd.store";
import { CrdList, crdResourcesRoute, crdRoute, crdURL } from "../+custom-resources"; import { CrdList, crdResourcesRoute, crdRoute, crdURL } from "../+custom-resources";
import { CustomResources } from "../+custom-resources/custom-resources"; import { CustomResources } from "../+custom-resources/custom-resources";
import { navigation } from "../../navigation"; import { navigation } from "../../navigation";
import { clusterPageRegistry } from "../../../extensions/registries/page-registry"; import { isAllowedResource } from "../../../common/rbac"
import { isAllowedResource } from "../../../common/rbac";
import { Spinner } from "../spinner"; import { Spinner } from "../spinner";
import { clusterPageMenuRegistry } from "../../../extensions/registries";
const SidebarContext = React.createContext<SidebarContextValue>({ pinned: false }); const SidebarContext = React.createContext<SidebarContextValue>({ pinned: false });
type SidebarContextValue = { type SidebarContextValue = {
@ -56,18 +56,17 @@ export class Sidebar extends React.Component<Props> {
} }
return Object.entries(crdStore.groups).map(([group, crds]) => { return Object.entries(crdStore.groups).map(([group, crds]) => {
const submenus = crds.map((crd) => { const submenus: TabLayoutRoute[] = crds.map((crd) => {
return { return {
title: crd.getResourceKind(), title: crd.getResourceKind(),
component: CrdList, component: CrdList,
url: crd.getResourceUrl(), url: crd.getResourceUrl(),
path: crdResourcesRoute.path, routePath: String(crdResourcesRoute.path),
}; };
}); });
return ( return (
<SidebarNavItem <SidebarNavItem
key={group} key={group}
id={group}
className="sub-menu-parent" className="sub-menu-parent"
url={crdURL({ query: { groups: group } })} url={crdURL({ query: { groups: group } })}
subMenus={submenus} subMenus={submenus}
@ -98,21 +97,18 @@ export class Sidebar extends React.Component<Props> {
</div> </div>
<div className="sidebar-nav flex column box grow-fixed"> <div className="sidebar-nav flex column box grow-fixed">
<SidebarNavItem <SidebarNavItem
id="cluster"
isHidden={!isAllowedResource("nodes")} isHidden={!isAllowedResource("nodes")}
url={clusterURL()} url={clusterURL()}
text={<Trans>Cluster</Trans>} text={<Trans>Cluster</Trans>}
icon={<Icon svg="kube" />} icon={<Icon svg="kube" />}
/> />
<SidebarNavItem <SidebarNavItem
id="nodes"
isHidden={!isAllowedResource("nodes")} isHidden={!isAllowedResource("nodes")}
url={nodesURL()} url={nodesURL()}
text={<Trans>Nodes</Trans>} text={<Trans>Nodes</Trans>}
icon={<Icon svg="nodes" />} icon={<Icon svg="nodes" />}
/> />
<SidebarNavItem <SidebarNavItem
id="workloads"
isHidden={Workloads.tabRoutes.length == 0} isHidden={Workloads.tabRoutes.length == 0}
url={workloadsURL({ query })} url={workloadsURL({ query })}
routePath={workloadsRoute.path} routePath={workloadsRoute.path}
@ -121,7 +117,6 @@ export class Sidebar extends React.Component<Props> {
icon={<Icon svg="workloads" />} icon={<Icon svg="workloads" />}
/> />
<SidebarNavItem <SidebarNavItem
id="config"
isHidden={Config.tabRoutes.length == 0} isHidden={Config.tabRoutes.length == 0}
url={configURL({ query })} url={configURL({ query })}
routePath={configRoute.path} routePath={configRoute.path}
@ -130,7 +125,6 @@ export class Sidebar extends React.Component<Props> {
icon={<Icon material="list" />} icon={<Icon material="list" />}
/> />
<SidebarNavItem <SidebarNavItem
id="networks"
isHidden={Network.tabRoutes.length == 0} isHidden={Network.tabRoutes.length == 0}
url={networkURL({ query })} url={networkURL({ query })}
routePath={networkRoute.path} routePath={networkRoute.path}
@ -139,7 +133,6 @@ export class Sidebar extends React.Component<Props> {
icon={<Icon material="device_hub" />} icon={<Icon material="device_hub" />}
/> />
<SidebarNavItem <SidebarNavItem
id="storage"
isHidden={Storage.tabRoutes.length == 0} isHidden={Storage.tabRoutes.length == 0}
url={storageURL({ query })} url={storageURL({ query })}
routePath={storageRoute.path} routePath={storageRoute.path}
@ -148,14 +141,12 @@ export class Sidebar extends React.Component<Props> {
text={<Trans>Storage</Trans>} text={<Trans>Storage</Trans>}
/> />
<SidebarNavItem <SidebarNavItem
id="namespaces"
isHidden={!isAllowedResource("namespaces")} isHidden={!isAllowedResource("namespaces")}
url={namespacesURL()} url={namespacesURL()}
icon={<Icon material="layers" />} icon={<Icon material="layers" />}
text={<Trans>Namespaces</Trans>} text={<Trans>Namespaces</Trans>}
/> />
<SidebarNavItem <SidebarNavItem
id="events"
isHidden={!isAllowedResource("events")} isHidden={!isAllowedResource("events")}
url={eventsURL({ query })} url={eventsURL({ query })}
routePath={eventRoute.path} routePath={eventRoute.path}
@ -163,7 +154,6 @@ export class Sidebar extends React.Component<Props> {
text={<Trans>Events</Trans>} text={<Trans>Events</Trans>}
/> />
<SidebarNavItem <SidebarNavItem
id="apps"
url={appsURL({ query })} url={appsURL({ query })}
subMenus={Apps.tabRoutes} subMenus={Apps.tabRoutes}
routePath={appsRoute.path} routePath={appsRoute.path}
@ -171,7 +161,6 @@ export class Sidebar extends React.Component<Props> {
text={<Trans>Apps</Trans>} text={<Trans>Apps</Trans>}
/> />
<SidebarNavItem <SidebarNavItem
id="users"
url={usersManagementURL({ query })} url={usersManagementURL({ query })}
routePath={usersManagementRoute.path} routePath={usersManagementRoute.path}
subMenus={UserManagement.tabRoutes} subMenus={UserManagement.tabRoutes}
@ -179,7 +168,6 @@ export class Sidebar extends React.Component<Props> {
text={<Trans>Access Control</Trans>} text={<Trans>Access Control</Trans>}
/> />
<SidebarNavItem <SidebarNavItem
id="custom-resources"
isHidden={!isAllowedResource("customresourcedefinitions")} isHidden={!isAllowedResource("customresourcedefinitions")}
url={crdURL()} url={crdURL()}
subMenus={CustomResources.tabRoutes} subMenus={CustomResources.tabRoutes}
@ -189,17 +177,13 @@ export class Sidebar extends React.Component<Props> {
> >
{this.renderCustomResources()} {this.renderCustomResources()}
</SidebarNavItem> </SidebarNavItem>
{clusterPageRegistry.getItems().map(({ path, title, url = String(path), hideInMenu, components: { MenuIcon } }) => { {clusterPageMenuRegistry.getItems().map(({ title, url, components: { Icon } }) => {
if (!MenuIcon || hideInMenu) { const routePath = "" // todo: find in page-registry
return;
}
return ( return (
<SidebarNavItem <SidebarNavItem
key={url} id={`sidebar_item_${url}`} key={url} url={url}
url={url} routePath={routePath}
routePath={path} text={title} icon={<Icon />}
text={title}
icon={<MenuIcon />}
/> />
) )
})} })}
@ -211,46 +195,45 @@ export class Sidebar extends React.Component<Props> {
} }
interface SidebarNavItemProps { interface SidebarNavItemProps {
id: string;
url: string; url: string;
text: React.ReactNode | string; text: React.ReactNode | string;
className?: string; className?: string;
icon?: React.ReactNode; icon?: React.ReactNode;
isHidden?: boolean; isHidden?: boolean;
routePath?: string | string[]; routePath?: string | string[];
subMenus?: TabRoute[]; subMenus?: TabLayoutRoute[];
} }
const navItemStorage = createStorage<[string, boolean][]>("sidebar_menu_item", []); const navItemStorage = createStorage<[string, boolean][]>("sidebar_menu_item", []);
const navItemState = observable.map<string, boolean>(navItemStorage.get()); const navItemState = observable.map<string, boolean>(navItemStorage.get());
reaction( reaction(() => [...navItemState], (value) => navItemStorage.set(value));
() => [...navItemState],
(value) => navItemStorage.set(value)
);
@observer @observer
class SidebarNavItem extends React.Component<SidebarNavItemProps> { class SidebarNavItem extends React.Component<SidebarNavItemProps> {
static contextType = SidebarContext; static contextType = SidebarContext;
public context: SidebarContextValue; public context: SidebarContextValue;
get itemId() {
return this.props.url;
}
@computed get isExpanded() { @computed get isExpanded() {
return navItemState.get(this.props.id); return navItemState.get(this.itemId);
} }
toggleSubMenu = () => { toggleSubMenu = () => {
navItemState.set(this.props.id, !this.isExpanded); navItemState.set(this.itemId, !this.isExpanded);
}; };
isActive = () => { isActive = () => {
const { routePath, url } = this.props; const { url, routePath = url } = this.props;
const { pathname } = navigation.location; return !!matchPath(navigation.location.pathname, {
return !!matchPath(pathname, { path: routePath
path: routePath || url,
}); });
}; };
render() { render() {
const { id, isHidden, subMenus = [], icon, text, url, children, className } = this.props; const { isHidden, subMenus = [], icon, text, url, children, className } = this.props;
if (isHidden) { if (isHidden) {
return null; return null;
} }
@ -258,7 +241,7 @@ class SidebarNavItem extends React.Component<SidebarNavItemProps> {
if (extendedView) { if (extendedView) {
const isActive = this.isActive(); const isActive = this.isActive();
return ( return (
<div id={id} className={cssNames("SidebarNavItem", className)}> <div className={cssNames("SidebarNavItem", className)}>
<div className={cssNames("nav-item", { active: isActive })} onClick={this.toggleSubMenu}> <div className={cssNames("nav-item", { active: isActive })} onClick={this.toggleSubMenu}>
{icon} {icon}
<span className="link-text">{text}</span> <span className="link-text">{text}</span>

View File

@ -1,38 +1,55 @@
import "./tab-layout.scss"; import "./tab-layout.scss";
import React, { ReactNode } from "react"; import React, { ReactNode } from "react";
import { matchPath, RouteProps } from "react-router-dom"; import { matchPath, Redirect, Route, Switch } from "react-router";
import { observer } from "mobx-react"; import { observer } from "mobx-react";
import { cssNames } from "../../utils"; import { cssNames, IClassName } from "../../utils";
import { Tab, Tabs } from "../tabs"; import { Tab, Tabs } from "../tabs";
import { ErrorBoundary } from "../error-boundary"; import { ErrorBoundary } from "../error-boundary";
import { navigate, navigation } from "../../navigation"; import { navigate, navigation } from "../../navigation";
export interface TabRoute extends RouteProps {
title: React.ReactNode;
url: string;
}
export interface TabLayoutProps { export interface TabLayoutProps {
children: ReactNode; className?: IClassName;
className?: any; contentClass?: IClassName;
tabs?: TabRoute[]; tabs?: TabLayoutRoute[];
contentClass?: string; children?: ReactNode;
} }
export const TabLayout = observer(({ className, contentClass, tabs, children }: TabLayoutProps) => { export interface TabLayoutRoute {
const routePath = navigation.location.pathname; routePath: string;
title: React.ReactNode;
component: React.ComponentType<any>;
url?: string; // page-url, if not provided `routePath` is used (doesn't work when path has some :placeholder(s))
exact?: boolean; // route-path matching rule
default?: boolean; // initial tab to open with provided `url, by default tabs[0] is used
}
export const TabLayout = observer(({ className, contentClass, tabs = [], children }: TabLayoutProps) => {
const currentLocation = navigation.location.pathname;
const hasTabs = tabs.length > 0;
const startTabUrl = hasTabs ? (tabs.find(tab => tab.default) || tabs[0])?.url : null;
return ( return (
<div className={cssNames("TabLayout", className)}> <div className={cssNames("TabLayout", className)}>
{tabs && ( {hasTabs && (
<Tabs center onChange={(url) => navigate(url)}> <Tabs center onChange={(url) => navigate(url)}>
{tabs.map(({ title, path, url, ...routeProps }) => { {tabs.map(({ title, routePath, url = routePath, exact }) => {
const isActive = !!matchPath(routePath, { path, ...routeProps }); const isActive = !!matchPath(currentLocation, { path: routePath, exact });
return <Tab key={url} label={title} value={url} active={isActive}/>; return <Tab key={url} label={title} value={url} active={isActive}/>;
})} })}
</Tabs> </Tabs>
)} )}
<main className={contentClass}> <main className={cssNames(contentClass)}>
<ErrorBoundary>{children}</ErrorBoundary> <ErrorBoundary>
{hasTabs && (
<Switch>
{tabs.map(({ routePath, exact, component }) => {
return <Route key={routePath} exact={exact} path={routePath} component={component}/>;
})}
<Redirect to={startTabUrl}/>
</Switch>
)}
{children}
</ErrorBoundary>
</main> </main>
</div> </div>
); );