mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
Using id prop to preserve expanding state
Signed-off-by: Alex Andreev <alex.andreev.email@gmail.com>
This commit is contained in:
parent
57e0370aa7
commit
41385c324e
@ -13,6 +13,7 @@ import type { TabLayoutRoute } from "./tab-layout";
|
|||||||
import type { SidebarContextValue } from "./sidebar-context";
|
import type { SidebarContextValue } from "./sidebar-context";
|
||||||
|
|
||||||
interface SidebarNavItemProps {
|
interface SidebarNavItemProps {
|
||||||
|
id: string; // Used to save nav item collapse/expand state in local storage
|
||||||
url: string;
|
url: string;
|
||||||
text: React.ReactNode | string;
|
text: React.ReactNode | string;
|
||||||
className?: string;
|
className?: string;
|
||||||
@ -20,7 +21,6 @@ interface SidebarNavItemProps {
|
|||||||
isHidden?: boolean;
|
isHidden?: boolean;
|
||||||
isActive?: boolean;
|
isActive?: boolean;
|
||||||
subMenus?: TabLayoutRoute[];
|
subMenus?: TabLayoutRoute[];
|
||||||
testId?: string; // data-test-id="" property for integration tests
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const navItemStorage = createStorage<[string, boolean][]>("sidebar_menu_item", []);
|
const navItemStorage = createStorage<[string, boolean][]>("sidebar_menu_item", []);
|
||||||
@ -33,22 +33,16 @@ export class SidebarNavItem extends React.Component<SidebarNavItemProps> {
|
|||||||
static contextType = SidebarContext;
|
static contextType = SidebarContext;
|
||||||
public context: SidebarContextValue;
|
public context: SidebarContextValue;
|
||||||
|
|
||||||
get itemId() {
|
|
||||||
const url = new URL(this.props.url, `${window.location.protocol}//${window.location.host}`);
|
|
||||||
|
|
||||||
return url.pathname; // pathname without get params
|
|
||||||
}
|
|
||||||
|
|
||||||
@computed get isExpanded() {
|
@computed get isExpanded() {
|
||||||
return navItemState.get(this.itemId);
|
return navItemState.get(this.props.id);
|
||||||
}
|
}
|
||||||
|
|
||||||
toggleSubMenu = () => {
|
toggleSubMenu = () => {
|
||||||
navItemState.set(this.itemId, !this.isExpanded);
|
navItemState.set(this.props.id, !this.isExpanded);
|
||||||
};
|
};
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
const { isHidden, isActive, subMenus = [], icon, text, url, children, className, testId } = this.props;
|
const { isHidden, isActive, subMenus = [], icon, text, url, children, className, id } = this.props;
|
||||||
|
|
||||||
if (isHidden) {
|
if (isHidden) {
|
||||||
return null;
|
return null;
|
||||||
@ -57,7 +51,7 @@ export class SidebarNavItem extends React.Component<SidebarNavItemProps> {
|
|||||||
|
|
||||||
if (extendedView) {
|
if (extendedView) {
|
||||||
return (
|
return (
|
||||||
<div className={cssNames("SidebarNavItem", className)} data-test-id={testId}>
|
<div className={cssNames("SidebarNavItem", className)} data-test-id={id}>
|
||||||
<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>
|
||||||
|
|||||||
@ -64,6 +64,7 @@ export class Sidebar extends React.Component<Props> {
|
|||||||
return (
|
return (
|
||||||
<SidebarNavItem
|
<SidebarNavItem
|
||||||
key={group}
|
key={group}
|
||||||
|
id={`crd-${group}`}
|
||||||
className="sub-menu-parent"
|
className="sub-menu-parent"
|
||||||
url={crdURL({ query: { groups: group } })}
|
url={crdURL({ query: { groups: group } })}
|
||||||
subMenus={submenus}
|
subMenus={submenus}
|
||||||
@ -100,6 +101,7 @@ export class Sidebar extends React.Component<Props> {
|
|||||||
return clusterPageMenuRegistry.getRootItems().map((menuItem, index) => {
|
return clusterPageMenuRegistry.getRootItems().map((menuItem, index) => {
|
||||||
const registeredPage = clusterPageRegistry.getByPageMenuTarget(menuItem.target);
|
const registeredPage = clusterPageRegistry.getByPageMenuTarget(menuItem.target);
|
||||||
const tabRoutes = this.getTabLayoutRoutes(menuItem);
|
const tabRoutes = this.getTabLayoutRoutes(menuItem);
|
||||||
|
const id = `registered-item-${index}`;
|
||||||
let pageUrl: string;
|
let pageUrl: string;
|
||||||
let isActive = false;
|
let isActive = false;
|
||||||
|
|
||||||
@ -117,7 +119,8 @@ export class Sidebar extends React.Component<Props> {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<SidebarNavItem
|
<SidebarNavItem
|
||||||
key={`registered-item-${index}`}
|
key={id}
|
||||||
|
id={id}
|
||||||
url={pageUrl}
|
url={pageUrl}
|
||||||
text={menuItem.title}
|
text={menuItem.title}
|
||||||
icon={<menuItem.components.Icon/>}
|
icon={<menuItem.components.Icon/>}
|
||||||
@ -150,7 +153,7 @@ 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
|
||||||
testId="cluster"
|
id="cluster"
|
||||||
isActive={isActiveRoute(clusterRoute)}
|
isActive={isActiveRoute(clusterRoute)}
|
||||||
isHidden={!isAllowedResource("nodes")}
|
isHidden={!isAllowedResource("nodes")}
|
||||||
url={clusterURL()}
|
url={clusterURL()}
|
||||||
@ -158,7 +161,7 @@ export class Sidebar extends React.Component<Props> {
|
|||||||
icon={<Icon svg="kube"/>}
|
icon={<Icon svg="kube"/>}
|
||||||
/>
|
/>
|
||||||
<SidebarNavItem
|
<SidebarNavItem
|
||||||
testId="nodes"
|
id="nodes"
|
||||||
isActive={isActiveRoute(nodesRoute)}
|
isActive={isActiveRoute(nodesRoute)}
|
||||||
isHidden={!isAllowedResource("nodes")}
|
isHidden={!isAllowedResource("nodes")}
|
||||||
url={nodesURL()}
|
url={nodesURL()}
|
||||||
@ -166,7 +169,7 @@ export class Sidebar extends React.Component<Props> {
|
|||||||
icon={<Icon svg="nodes"/>}
|
icon={<Icon svg="nodes"/>}
|
||||||
/>
|
/>
|
||||||
<SidebarNavItem
|
<SidebarNavItem
|
||||||
testId="workloads"
|
id="workloads"
|
||||||
isActive={isActiveRoute(workloadsRoute)}
|
isActive={isActiveRoute(workloadsRoute)}
|
||||||
isHidden={Workloads.tabRoutes.length == 0}
|
isHidden={Workloads.tabRoutes.length == 0}
|
||||||
url={workloadsURL({ query })}
|
url={workloadsURL({ query })}
|
||||||
@ -175,7 +178,7 @@ export class Sidebar extends React.Component<Props> {
|
|||||||
icon={<Icon svg="workloads"/>}
|
icon={<Icon svg="workloads"/>}
|
||||||
/>
|
/>
|
||||||
<SidebarNavItem
|
<SidebarNavItem
|
||||||
testId="config"
|
id="config"
|
||||||
isActive={isActiveRoute(configRoute)}
|
isActive={isActiveRoute(configRoute)}
|
||||||
isHidden={Config.tabRoutes.length == 0}
|
isHidden={Config.tabRoutes.length == 0}
|
||||||
url={configURL({ query })}
|
url={configURL({ query })}
|
||||||
@ -184,7 +187,7 @@ export class Sidebar extends React.Component<Props> {
|
|||||||
icon={<Icon material="list"/>}
|
icon={<Icon material="list"/>}
|
||||||
/>
|
/>
|
||||||
<SidebarNavItem
|
<SidebarNavItem
|
||||||
testId="networks"
|
id="networks"
|
||||||
isActive={isActiveRoute(networkRoute)}
|
isActive={isActiveRoute(networkRoute)}
|
||||||
isHidden={Network.tabRoutes.length == 0}
|
isHidden={Network.tabRoutes.length == 0}
|
||||||
url={networkURL({ query })}
|
url={networkURL({ query })}
|
||||||
@ -193,7 +196,7 @@ export class Sidebar extends React.Component<Props> {
|
|||||||
icon={<Icon material="device_hub"/>}
|
icon={<Icon material="device_hub"/>}
|
||||||
/>
|
/>
|
||||||
<SidebarNavItem
|
<SidebarNavItem
|
||||||
testId="storage"
|
id="storage"
|
||||||
isActive={isActiveRoute(storageRoute)}
|
isActive={isActiveRoute(storageRoute)}
|
||||||
isHidden={Storage.tabRoutes.length == 0}
|
isHidden={Storage.tabRoutes.length == 0}
|
||||||
url={storageURL({ query })}
|
url={storageURL({ query })}
|
||||||
@ -202,7 +205,7 @@ export class Sidebar extends React.Component<Props> {
|
|||||||
text={<Trans>Storage</Trans>}
|
text={<Trans>Storage</Trans>}
|
||||||
/>
|
/>
|
||||||
<SidebarNavItem
|
<SidebarNavItem
|
||||||
testId="namespaces"
|
id="namespaces"
|
||||||
isActive={isActiveRoute(namespacesRoute)}
|
isActive={isActiveRoute(namespacesRoute)}
|
||||||
isHidden={!isAllowedResource("namespaces")}
|
isHidden={!isAllowedResource("namespaces")}
|
||||||
url={namespacesURL()}
|
url={namespacesURL()}
|
||||||
@ -210,7 +213,7 @@ export class Sidebar extends React.Component<Props> {
|
|||||||
text={<Trans>Namespaces</Trans>}
|
text={<Trans>Namespaces</Trans>}
|
||||||
/>
|
/>
|
||||||
<SidebarNavItem
|
<SidebarNavItem
|
||||||
testId="events"
|
id="events"
|
||||||
isActive={isActiveRoute(eventRoute)}
|
isActive={isActiveRoute(eventRoute)}
|
||||||
isHidden={!isAllowedResource("events")}
|
isHidden={!isAllowedResource("events")}
|
||||||
url={eventsURL({ query })}
|
url={eventsURL({ query })}
|
||||||
@ -218,7 +221,7 @@ export class Sidebar extends React.Component<Props> {
|
|||||||
text={<Trans>Events</Trans>}
|
text={<Trans>Events</Trans>}
|
||||||
/>
|
/>
|
||||||
<SidebarNavItem
|
<SidebarNavItem
|
||||||
testId="apps"
|
id="apps"
|
||||||
isActive={isActiveRoute(appsRoute)}
|
isActive={isActiveRoute(appsRoute)}
|
||||||
url={appsURL({ query })}
|
url={appsURL({ query })}
|
||||||
subMenus={Apps.tabRoutes}
|
subMenus={Apps.tabRoutes}
|
||||||
@ -226,7 +229,7 @@ export class Sidebar extends React.Component<Props> {
|
|||||||
text={<Trans>Apps</Trans>}
|
text={<Trans>Apps</Trans>}
|
||||||
/>
|
/>
|
||||||
<SidebarNavItem
|
<SidebarNavItem
|
||||||
testId="users"
|
id="users"
|
||||||
isActive={isActiveRoute(usersManagementRoute)}
|
isActive={isActiveRoute(usersManagementRoute)}
|
||||||
url={usersManagementURL({ query })}
|
url={usersManagementURL({ query })}
|
||||||
subMenus={UserManagement.tabRoutes}
|
subMenus={UserManagement.tabRoutes}
|
||||||
@ -234,7 +237,7 @@ export class Sidebar extends React.Component<Props> {
|
|||||||
text={<Trans>Access Control</Trans>}
|
text={<Trans>Access Control</Trans>}
|
||||||
/>
|
/>
|
||||||
<SidebarNavItem
|
<SidebarNavItem
|
||||||
testId="custom-resources"
|
id="custom-resources"
|
||||||
isActive={isActiveRoute(crdRoute)}
|
isActive={isActiveRoute(crdRoute)}
|
||||||
isHidden={!isAllowedResource("customresourcedefinitions")}
|
isHidden={!isAllowedResource("customresourcedefinitions")}
|
||||||
url={crdURL()}
|
url={crdURL()}
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user