portainer/app/react/sidebar/EnvironmentSidebar.tsx

160 lines
4.6 KiB
TypeScript

import { useCurrentStateAndParams, useRouter } from '@uirouter/react';
import { useEffect } from 'react';
import { X, Slash } from 'lucide-react';
import clsx from 'clsx';
import { useStore } from 'zustand';
import {
PlatformType,
EnvironmentId,
Environment,
} from '@/react/portainer/environments/types';
import { getPlatformType } from '@/react/portainer/environments/utils';
import { useEnvironment } from '@/react/portainer/environments/queries/useEnvironment';
import { isBE } from '@/react/portainer/feature-flags/feature-flags.service';
import { environmentStore } from '@/react/hooks/current-environment-store';
import { Icon } from '@@/Icon';
import { getPlatformIcon } from '../portainer/environments/utils/get-platform-icon';
import styles from './EnvironmentSidebar.module.css';
import { AzureSidebar } from './AzureSidebar';
import { DockerSidebar } from './DockerSidebar';
import { KubernetesSidebar } from './KubernetesSidebar';
import { SidebarSection, SidebarSectionTitle } from './SidebarSection';
import { useSidebarState } from './useSidebarState';
import { NomadSidebar } from './NomadSidebar';
export function EnvironmentSidebar() {
const { query: currentEnvironmentQuery, clearEnvironment } =
useCurrentEnvironment();
const environment = currentEnvironmentQuery.data;
const { isOpen } = useSidebarState();
if (!isOpen && !environment) {
return null;
}
return (
<div className={clsx(styles.root, 'rounded border border-dotted py-2')}>
{environment ? (
<Content environment={environment} onClear={clearEnvironment} />
) : (
<SidebarSectionTitle>
<div className="flex items-center gap-1">
<span>Environment:</span>
<Icon icon={Slash} className="text-xl !text-gray-6" />
<span className="text-gray-6 text-sm">None selected</span>
</div>
</SidebarSectionTitle>
)}
</div>
);
}
interface ContentProps {
environment: Environment;
onClear: () => void;
}
function Content({ environment, onClear }: ContentProps) {
const platform = getPlatformType(environment.Type);
const Sidebar = getSidebar(platform);
return (
<SidebarSection
title={<Title environment={environment} onClear={onClear} />}
aria-label={environment.Name}
showTitleWhenOpen
>
<div className="mt-2">
{Sidebar && (
<Sidebar environmentId={environment.Id} environment={environment} />
)}
</div>
</SidebarSection>
);
function getSidebar(platform: PlatformType) {
const sidebar: {
[key in PlatformType]: React.ComponentType<{
environmentId: EnvironmentId;
environment: Environment;
}> | null;
} = {
[PlatformType.Azure]: AzureSidebar,
[PlatformType.Docker]: DockerSidebar,
[PlatformType.Kubernetes]: KubernetesSidebar,
[PlatformType.Nomad]: isBE ? NomadSidebar : null,
};
return sidebar[platform];
}
}
function useCurrentEnvironment() {
const { params } = useCurrentStateAndParams();
const router = useRouter();
const envStore = useStore(environmentStore);
const { setEnvironmentId } = envStore;
useEffect(() => {
const environmentId = parseInt(params.endpointId, 10);
if (params.endpointId && !Number.isNaN(environmentId)) {
setEnvironmentId(environmentId);
}
}, [setEnvironmentId, params.endpointId, params.environmentId]);
return { query: useEnvironment(envStore.environmentId), clearEnvironment };
function clearEnvironment() {
if (params.endpointId || params.environmentId) {
router.stateService.go('portainer.home');
}
envStore.clear();
}
}
interface TitleProps {
environment: Environment;
onClear(): void;
}
function Title({ environment, onClear }: TitleProps) {
const { isOpen } = useSidebarState();
const EnvironmentIcon = getPlatformIcon(environment.Type);
if (!isOpen) {
return (
<div className="w-8 flex justify-center -ml-3" title={environment.Name}>
<EnvironmentIcon className="text-2xl" />
</div>
);
}
return (
<div className="flex items-center">
<EnvironmentIcon className="text-2xl mr-3" />
<span className="text-white text-ellipsis overflow-hidden whitespace-nowrap">
{environment.Name}
</span>
<button
title="Clear environment"
type="button"
onClick={onClear}
className={clsx(
styles.closeBtn,
'flex items-center justify-center transition-colors duration-200 rounded border-0 text-sm h-5 w-5 p-1 ml-auto mr-2 text-gray-5 be:text-gray-6 hover:text-white be:hover:text-white'
)}
>
<X />
</button>
</div>
);
}