Skip to content

Commit

Permalink
Merge pull request #2302 from avivtur/tree-view-fixes
Browse files Browse the repository at this point in the history
CNV-52004: Vm tree view 4
  • Loading branch information
openshift-merge-bot[bot] authored Dec 9, 2024
2 parents 23798e5 + d27b2c1 commit 0a269cb
Show file tree
Hide file tree
Showing 15 changed files with 443 additions and 166 deletions.
2 changes: 2 additions & 0 deletions locales/en/plugin__kubevirt-plugin.json
Original file line number Diff line number Diff line change
Expand Up @@ -956,6 +956,7 @@
"Project labels": "Project labels",
"Project name to clone the template to": "Project name to clone the template to",
"Project selector": "Project selector",
"Projects": "Projects",
"Provider": "Provider",
"Provisioning": "Provisioning",
"Public SSH key": "Public SSH key",
Expand Down Expand Up @@ -1131,6 +1132,7 @@
"Show all": "Show all",
"Show all alerts": "Show all alerts",
"Show less": "Show less",
"Show OCP default projects": "Show OCP default projects",
"Show top 10": "Show top 10",
"Show top 5": "Show top 5",
"Show uncategorized VirtualMachines": "Show uncategorized VirtualMachines",
Expand Down
44 changes: 33 additions & 11 deletions src/views/virtualmachines/tree/VirtualMachineTreeView.scss
Original file line number Diff line number Diff line change
Expand Up @@ -2,24 +2,46 @@
overflow-x: hidden;
}

.vms-tree-view-body,
.vms-tree-view-toolbar,
.vms-tree-view-toolbar-content {
.pf-v5-c-drawer__body.vms-tree-view-body,
.pf-v5-c-toolbar.vms-tree-view-toolbar,
.pf-v5-c-toolbar__content.vms-tree-view-toolbar-content,
.pf-v5-c-drawer__body.vms-tree-view-actions {
padding: 0;
}

.pf-v5-c-tree-view__node .pf-v5-c-tree-view__node-count,
.vms-tree-view-toolbar-buttons {
.pf-v5-c-drawer__panel-main {
.pf-v5-c-drawer__body {
padding: 0 var(--pf-global--spacer--sm);
height: fit-content;
}
}
.pf-v5-c-toolbar__content.vms-tree-view-toolbar-content,
.vms-tree-view-toolbar-section,
.vms-tree-view-search-input {
width: 100%;
}
.vms-tree-view-search-input {
margin: var(--pf-global--spacer--sm) var(--pf-global--spacer--sm) 0 var(--pf-global--spacer--sm);
}
.pf-v5-c-toolbar.vms-tree-view-toolbar {
margin-bottom: var(--pf-global--spacer--sm);
}
.pf-v5-c-tree-view__node .pf-v5-c-tree-view__node-count {
display: flex;
flex-grow: 1;
flex-direction: row-reverse;
}

.vms-tree-view-toolbar-buttons {
align-self: center;
margin-right: var(--pf-global--spacer--sm);
.pf-v5-c-tree-view {
padding: 0;
}

&__padding {
padding: var(--pf-global--spacer--sm);
}
.vms-tree-view-toolbar-action {
padding: var(--pf-global--spacer--sm);
}

.vms-tree-view-toolbar-default-project-switch {
margin-left: var(--pf-global--spacer--sm);
margin-bottom: 0;
font-weight: var(--pf-global--FontWeight--light);
}
140 changes: 122 additions & 18 deletions src/views/virtualmachines/tree/VirtualMachineTreeView.tsx
Original file line number Diff line number Diff line change
@@ -1,38 +1,58 @@
import React, { FC, MouseEvent, useMemo } from 'react';
import React, { FC, MouseEvent, useEffect, useMemo } from 'react';
import { useLocation, useNavigate } from 'react-router-dom-v5-compat';

import VirtualMachineModel from '@kubevirt-ui/kubevirt-api/console/models/VirtualMachineModel';
import { useQueryParamsMethods } from '@kubevirt-utils/components/ListPageFilter/hooks/useQueryParamsMethods';
import { useKubevirtTranslation } from '@kubevirt-utils/hooks/useKubevirtTranslation';
import useLocalStorage from '@kubevirt-utils/hooks/useLocalStorage';
import { convertResourceArrayToMap, getResourceUrl } from '@kubevirt-utils/resources/shared';
import { getContentScrollableElement } from '@kubevirt-utils/utils/utils';
import { FilterValue, useActiveNamespace } from '@openshift-console/dynamic-plugin-sdk';
import {
Button,
ButtonVariant,
Drawer,
DrawerActions,
DrawerContent,
DrawerContentBody,
DrawerHead,
DrawerPanelBody,
DrawerPanelContent,
Title,
Tooltip,
TreeView,
TreeViewDataItem,
} from '@patternfly/react-core';
// import { PanelCloseIcon, PanelOpenIcon } from '@patternfly/react-icons';
import { TEXT_FILTER_LABELS_ID } from '@virtualmachines/list/hooks/constants';

import TreeViewToolbar from './components/TreeViewToolbar';
import { useHideNamespaceBar } from './hooks/useHideNamespaceBar';
import { useSyncClicksEffects } from './hooks/useSyncClicksEffects';
import { useTreeViewData } from './hooks/useTreeViewData';
import { useTreeViewSearch } from './hooks/useTreeViewSearch';
import ClosePanelIcon from './icons/ClosePanelIcon';
import CollapseAllIcon from './icons/CollapseAllIcon';
import ExpandAllIcon from './icons/ExpandAllIcon';
import OpenPanelIcon from './icons/OpenPanelIcon';
import {
CLOSED_DRAWER_SIZE,
FOLDER_SELECTOR_PREFIX,
HIDE,
OPEN_DRAWER_SIZE,
PANEL_WIDTH_PROPERTY,
PROJECT_SELECTOR_PREFIX,
SHOW,
SHOW_TREE_VIEW,
TREE_VIEW_LAST_WIDTH,
TREE_VIEW_PANEL_ID,
VM_FOLDER_LABEL,
} from './utils/constants';
import {
createTreeViewData,
filterDefaultNamespaceItems,
selectedTreeItem,
setSelectedTreeItem,
treeViewOpen,
} from './utils/utils';

import './VirtualMachineTreeView.scss';
Expand All @@ -42,12 +62,24 @@ type VirtualMachineTreeViewProps = {
};

const VirtualMachineTreeView: FC<VirtualMachineTreeViewProps> = ({ children, onFilterChange }) => {
const { t } = useKubevirtTranslation();
const [activeNamespace] = useActiveNamespace();
const navigate = useNavigate();
const location = useLocation();

const { setOrRemoveQueryArgument } = useQueryParamsMethods();
const { isAdmin, loaded, loadError, projectNames, vms } = useTreeViewData();
const [drawerWidth, setDrawerWidth] = useLocalStorage(TREE_VIEW_LAST_WIDTH, OPEN_DRAWER_SIZE);
const [drawerOpen, setDrawerOpen] = useLocalStorage(SHOW_TREE_VIEW, SHOW);

const {
isAdmin,
loaded,
loadError,
projectNames,
setShowDefaultProjects,
showDefaultProjects,
vms,
} = useTreeViewData();

const vmsMapper = useMemo(() => convertResourceArrayToMap(vms, true), [vms]);

const treeData = useMemo(
Expand All @@ -57,8 +89,27 @@ const VirtualMachineTreeView: FC<VirtualMachineTreeViewProps> = ({ children, onF

const { filteredItems, onSearch, setShowAll, showAll } = useTreeViewSearch(treeData);

const drawerPanel = document.getElementById(TREE_VIEW_PANEL_ID);
const isOpen = drawerOpen === SHOW;

useSyncClicksEffects(activeNamespace, loaded, location);
useHideNamespaceBar();
useEffect(() => {
drawerPanel?.style?.setProperty(PANEL_WIDTH_PROPERTY, drawerWidth);
}, [drawerPanel, drawerWidth]);

const filteredTreeData = useMemo(() => {
if (!isOpen) return [];

const items = filteredItems ?? treeData;
if (showDefaultProjects === HIDE) {
return items
.map((opt) => Object.assign({}, opt))
.filter((item) => filterDefaultNamespaceItems(item));
}

return items;
}, [filteredItems, showDefaultProjects, treeData, isOpen]);

if (loadError) return <>{children}</>;

Expand Down Expand Up @@ -100,6 +151,33 @@ const VirtualMachineTreeView: FC<VirtualMachineTreeViewProps> = ({ children, onF
);
};

const toggleDrawer = () => {
const toggleOpen = !isOpen;
setDrawerOpen(toggleOpen ? SHOW : HIDE);

const size = toggleOpen ? OPEN_DRAWER_SIZE : CLOSED_DRAWER_SIZE;
drawerPanel.style.setProperty(PANEL_WIDTH_PROPERTY, size);
setDrawerWidth(size);
};

const onResize = (
_e: globalThis.MouseEvent | React.KeyboardEvent | TouchEvent,
width: number,
) => {
setDrawerWidth(`${String(width)}px`);
};
const togglePanelButton = (
<Tooltip content={isOpen ? t('Close') : t('Open')}>
<Button
className="vms-tree-view-toolbar-action"
onClick={toggleDrawer}
variant={ButtonVariant.plain}
>
{isOpen ? <ClosePanelIcon /> : <OpenPanelIcon />}
</Button>
</Tooltip>
);

return (
<Drawer isExpanded isInline position="start">
<DrawerContent
Expand All @@ -109,22 +187,48 @@ const VirtualMachineTreeView: FC<VirtualMachineTreeViewProps> = ({ children, onF
height: getContentScrollableElement().offsetHeight || 0,
}}
className="vms-tree-view"
defaultSize={drawerWidth}
id={TREE_VIEW_PANEL_ID}
isResizable={treeViewOpen.value}
isResizable={isOpen}
onResize={onResize}
>
<DrawerPanelBody className="vms-tree-view-body">
<TreeView
toolbar={
<TreeViewToolbar onSearch={onSearch} setShowAll={setShowAll} showAll={showAll} />
}
activeItems={selectedTreeItem.value}
allExpanded={showAll}
data={!treeViewOpen.value ? [] : filteredItems ?? treeData}
hasBadges={loaded}
hasSelectableNodes
onSelect={onSelect}
/>
</DrawerPanelBody>
{!isOpen ? (
togglePanelButton
) : (
<>
<TreeViewToolbar
isOpen={isOpen}
onSearch={onSearch}
setShowDefaultProjects={setShowDefaultProjects}
showDefaultProjects={showDefaultProjects}
/>
<DrawerHead>
<Title headingLevel="h6">{t('Projects')}</Title>
<DrawerActions>
<Tooltip content={showAll ? t('Collapse all') : t('Expand all')}>
<Button
className="vms-tree-view-toolbar-action"
onClick={() => setShowAll((prev) => !prev)}
variant={ButtonVariant.plain}
>
{showAll ? <CollapseAllIcon /> : <ExpandAllIcon />}
</Button>
</Tooltip>
{togglePanelButton}
</DrawerActions>
</DrawerHead>
<DrawerPanelBody className="vms-tree-view-body">
<TreeView
activeItems={selectedTreeItem.value}
allExpanded={showAll}
data={filteredTreeData}
hasBadges={loaded}
hasSelectableNodes
onSelect={onSelect}
/>
</DrawerPanelBody>
</>
)}
</DrawerPanelContent>
}
>
Expand Down
Loading

0 comments on commit 0a269cb

Please sign in to comment.