1
0
mirror of https://github.com/Unleash/unleash.git synced 2025-04-01 01:18:10 +02:00

update project archive and revive dialogs (#7918)

This commit is contained in:
Tymoteusz Czech 2024-08-19 15:33:00 +02:00 committed by GitHub
parent cf3379d0b3
commit 004038e872
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
11 changed files with 104 additions and 73 deletions

View File

@ -1,5 +1,5 @@
import { useFeaturesArchive } from 'hooks/api/getters/useFeaturesArchive/useFeaturesArchive'; import { useFeaturesArchive } from 'hooks/api/getters/useFeaturesArchive/useFeaturesArchive';
import type { VFC } from 'react'; import type { FC } from 'react';
import type { SortingRule } from 'react-table'; import type { SortingRule } from 'react-table';
import { createLocalStorage } from 'utils/createLocalStorage'; import { createLocalStorage } from 'utils/createLocalStorage';
import { ArchiveTable } from './ArchiveTable/ArchiveTable'; import { ArchiveTable } from './ArchiveTable/ArchiveTable';
@ -10,7 +10,7 @@ interface IProjectFeaturesTable {
projectId: string; projectId: string;
} }
export const ProjectFeaturesArchiveTable: VFC<IProjectFeaturesTable> = ({ export const ProjectFeaturesArchiveTable: FC<IProjectFeaturesTable> = ({
projectId, projectId,
}) => { }) => {
const { archivedFeatures, loading, refetchArchived } = const { archivedFeatures, loading, refetchArchived } =
@ -23,7 +23,7 @@ export const ProjectFeaturesArchiveTable: VFC<IProjectFeaturesTable> = ({
return ( return (
<ArchiveTable <ArchiveTable
title='Project archive' title='Archived flags'
archivedFeatures={archivedFeatures || []} archivedFeatures={archivedFeatures || []}
loading={loading} loading={loading}
storedParams={value} storedParams={value}

View File

@ -62,11 +62,13 @@ export const ProjectArchiveCard: FC<ProjectArchiveCardProps> = ({
<ProjectIcon color='action' /> <ProjectIcon color='action' />
</StyledIconBox> </StyledIconBox>
<StyledBox data-loading> <StyledBox data-loading>
<StyledCardTitle> <Tooltip title={`id: ${id}`} arrow>
<Highlighter search={searchQuery}> <StyledCardTitle>
{name} <Highlighter search={searchQuery}>
</Highlighter> {name}
</StyledCardTitle> </Highlighter>
</StyledCardTitle>
</Tooltip>
</StyledBox> </StyledBox>
<ProjectModeBadge mode={mode} /> <ProjectModeBadge mode={mode} />
</StyledDivHeader> </StyledDivHeader>
@ -114,7 +116,7 @@ export const ProjectArchiveCard: FC<ProjectArchiveCardProps> = ({
onClick={onRevive} onClick={onRevive}
projectId={id} projectId={id}
permission={UPDATE_PROJECT} permission={UPDATE_PROJECT}
tooltipProps={{ title: 'Restore project' }} tooltipProps={{ title: 'Revive project' }}
data-testid={`revive-feature-flag-button`} data-testid={`revive-feature-flag-button`}
> >
<Undo /> <Undo />

View File

@ -45,11 +45,11 @@ export const ArchiveProjectDialogue = ({
open={open} open={open}
onClick={onClick} onClick={onClick}
onClose={onClose} onClose={onClose}
title='Really archive project' title='Are you sure?'
> >
<Typography> <Typography>
This will archive the project and all feature flags archived in The project will be moved to the projects archive, where it can
it. either be revived or permanently deleted.
</Typography> </Typography>
</Dialogue> </Dialogue>
); );

View File

@ -7,19 +7,26 @@ import useToast from 'hooks/useToast';
import { ConditionallyRender } from 'component/common/ConditionallyRender/ConditionallyRender'; import { ConditionallyRender } from 'component/common/ConditionallyRender/ConditionallyRender';
import { useUiFlag } from 'hooks/useUiFlag'; import { useUiFlag } from 'hooks/useUiFlag';
import useUiConfig from 'hooks/api/getters/useUiConfig/useUiConfig'; import useUiConfig from 'hooks/api/getters/useUiConfig/useUiConfig';
import { Typography } from '@mui/material'; import { styled, Typography } from '@mui/material';
import { ProjectId } from 'component/project/ProjectId/ProjectId';
interface IDeleteProjectDialogueProps { interface IDeleteProjectDialogueProps {
project: string; projectId: string;
projectName?: string;
open: boolean; open: boolean;
onClose: (e: React.SyntheticEvent) => void; onClose: (e: React.SyntheticEvent) => void;
onSuccess?: () => void; onSuccess?: () => void;
} }
const StyledParagraph = styled(Typography)(({ theme }) => ({
marginBottom: theme.spacing(1),
}));
export const DeleteProjectDialogue = ({ export const DeleteProjectDialogue = ({
open, open,
onClose, onClose,
project, projectId,
projectName,
onSuccess, onSuccess,
}: IDeleteProjectDialogueProps) => { }: IDeleteProjectDialogueProps) => {
const { deleteProject } = useProjectApi(); const { deleteProject } = useProjectApi();
@ -32,7 +39,7 @@ export const DeleteProjectDialogue = ({
const onClick = async (e: React.SyntheticEvent) => { const onClick = async (e: React.SyntheticEvent) => {
e.preventDefault(); e.preventDefault();
try { try {
await deleteProject(project); await deleteProject(projectId);
refetchProjects(); refetchProjects();
refetchProjectArchive(); refetchProjectArchive();
setToastData({ setToastData({
@ -52,17 +59,34 @@ export const DeleteProjectDialogue = ({
open={open} open={open}
onClick={onClick} onClick={onClick}
onClose={onClose} onClose={onClose}
title='Really delete project' title='Are you sure?'
> >
<Typography> <StyledParagraph>
This will irreversibly remove the project, all feature flags This will irreversibly remove:
archived in it, all API keys scoped to only this project <ul>
<ConditionallyRender <li>project with all of its settings</li>
condition={isEnterprise() && automatedActionsEnabled} <li>all feature flags archived in it</li>
show=', and all actions configured for it' <li>all API keys scoped to only to this project</li>
/> <ConditionallyRender
. condition={isEnterprise() && automatedActionsEnabled}
</Typography> show={<li>all actions configured for it</li>}
/>
</ul>
</StyledParagraph>
<ConditionallyRender
condition={Boolean(projectName)}
show={
<>
<StyledParagraph>
Are you sure you'd like to permanently delete
project <strong>{projectName}</strong>?
</StyledParagraph>
<StyledParagraph>
Project ID: <ProjectId>{projectId}</ProjectId>
</StyledParagraph>
</>
}
/>
</Dialogue> </Dialogue>
); );
}; };

View File

@ -103,7 +103,7 @@ export const Project = () => {
name: 'health', name: 'health',
}, },
{ {
title: 'Archive', title: 'Archived flags',
path: `${basePath}/archive`, path: `${basePath}/archive`,
name: 'archive', name: 'archive',
}, },
@ -285,7 +285,7 @@ export const Project = () => {
</StyledTabContainer> </StyledTabContainer>
</StyledHeader> </StyledHeader>
<DeleteProjectDialogue <DeleteProjectDialogue
project={projectId} projectId={projectId}
open={showDelDialog} open={showDelDialog}
onClose={() => { onClose={() => {
setShowDelDialog(false); setShowDelDialog(false);

View File

@ -6,7 +6,7 @@ import { useProjectOverviewNameOrId } from 'hooks/api/getters/useProjectOverview
export const ProjectFeaturesArchive = () => { export const ProjectFeaturesArchive = () => {
const projectId = useRequiredPathParam('projectId'); const projectId = useRequiredPathParam('projectId');
const projectName = useProjectOverviewNameOrId(projectId); const projectName = useProjectOverviewNameOrId(projectId);
usePageTitle(`Project archive ${projectName}`); usePageTitle(`Project archived flags ${projectName}`);
return <ProjectFeaturesArchiveTable projectId={projectId} />; return <ProjectFeaturesArchiveTable projectId={projectId} />;
}; };

View File

@ -1,12 +1,10 @@
import { styled } from '@mui/material'; import { Link, styled } from '@mui/material';
import { DELETE_PROJECT } from 'component/providers/AccessProvider/permissions'; import { DELETE_PROJECT } from 'component/providers/AccessProvider/permissions';
import PermissionButton from 'component/common/PermissionButton/PermissionButton'; import PermissionButton from 'component/common/PermissionButton/PermissionButton';
import { useState } from 'react'; import { useState } from 'react';
import { useNavigate } from 'react-router'; import { useNavigate } from 'react-router';
import { useUiFlag } from 'hooks/useUiFlag'; import { Link as RouterLink } from 'react-router-dom';
import { useActions } from 'hooks/api/getters/useActions/useActions';
import { ConditionallyRender } from 'component/common/ConditionallyRender/ConditionallyRender'; import { ConditionallyRender } from 'component/common/ConditionallyRender/ConditionallyRender';
import useUiConfig from 'hooks/api/getters/useUiConfig/useUiConfig';
import { ArchiveProjectDialogue } from '../../ArchiveProject/ArchiveProjectDialogue'; import { ArchiveProjectDialogue } from '../../ArchiveProject/ArchiveProjectDialogue';
const StyledContainer = styled('div')(({ theme }) => ({ const StyledContainer = styled('div')(({ theme }) => ({
@ -32,54 +30,34 @@ export const ArchiveProject = ({
projectId, projectId,
featureCount, featureCount,
}: IDeleteProjectProps) => { }: IDeleteProjectProps) => {
const { isEnterprise } = useUiConfig();
const automatedActionsEnabled = useUiFlag('automatedActions');
const { actions } = useActions(projectId);
const [showArchiveDialog, setShowArchiveDialog] = useState(false); const [showArchiveDialog, setShowArchiveDialog] = useState(false);
const actionsCount = actions.filter(({ enabled }) => enabled).length;
const navigate = useNavigate(); const navigate = useNavigate();
const disabled = featureCount > 0;
return ( return (
<StyledContainer> <StyledContainer>
<p> <p>
Before you can archive a project, you must first archive all the Before you can archive a project, you must first archive all of
feature flags associated with it the feature flags associated with it.
{isEnterprise() && automatedActionsEnabled
? ' and disable all actions that are in it'
: ''}
.
</p> </p>
<ConditionallyRender <ConditionallyRender
condition={featureCount > 0} condition={featureCount > 0}
show={ show={
<p> <p>
Currently there {featureCount <= 1 ? 'is' : 'are'}{' '} Currently there {featureCount <= 1 ? 'is' : 'are'}{' '}
<strong> <Link component={RouterLink} to='../..'>
{featureCount} active feature{' '} <strong>
{featureCount === 1 ? 'flag' : 'flags'}. {featureCount} active feature{' '}
</strong> {featureCount === 1 ? 'flag' : 'flags'}.
</p> </strong>
} </Link>
/>
<ConditionallyRender
condition={
isEnterprise() &&
automatedActionsEnabled &&
actionsCount > 0
}
show={
<p>
Currently there {actionsCount <= 1 ? 'is' : 'are'}{' '}
<strong>
{actionsCount} enabled{' '}
{actionsCount === 1 ? 'action' : 'actions'}.
</strong>
</p> </p>
} }
/> />
<StyledButtonContainer> <StyledButtonContainer>
<PermissionButton <PermissionButton
permission={DELETE_PROJECT} permission={DELETE_PROJECT}
disabled={featureCount > 0} disabled={disabled}
projectId={projectId} projectId={projectId}
onClick={() => { onClick={() => {
setShowArchiveDialog(true); setShowArchiveDialog(true);

View File

@ -25,11 +25,13 @@ const StyledButtonContainer = styled('div')(({ theme }) => ({
interface IDeleteProjectProps { interface IDeleteProjectProps {
projectId: string; projectId: string;
projectName?: string;
featureCount: number; featureCount: number;
} }
export const DeleteProject = ({ export const DeleteProject = ({
projectId, projectId,
projectName,
featureCount, featureCount,
}: IDeleteProjectProps) => { }: IDeleteProjectProps) => {
const { isEnterprise } = useUiConfig(); const { isEnterprise } = useUiConfig();
@ -106,7 +108,8 @@ export const DeleteProject = ({
</PermissionButton> </PermissionButton>
</StyledButtonContainer> </StyledButtonContainer>
<DeleteProjectDialogue <DeleteProjectDialogue
project={projectId} projectId={projectId}
projectName={projectName}
open={showDelDialog} open={showDelDialog}
onClose={() => { onClose={() => {
setShowDelDialog(false); setShowDelDialog(false);

View File

@ -0,0 +1,9 @@
import { styled } from '@mui/material';
export const ProjectId = styled('code')(({ theme }) => ({
backgroundColor: theme.palette.background.elevation2,
padding: theme.spacing(0.5, 1.5),
display: 'inline-block',
borderRadius: `${theme.shape.borderRadius}px`,
fontSize: theme.typography.body2.fontSize,
}));

View File

@ -49,6 +49,7 @@ export const ArchiveProjectList: FC = () => {
const [deleteProject, setDeleteProject] = useState<{ const [deleteProject, setDeleteProject] = useState<{
isOpen: boolean; isOpen: boolean;
id?: string; id?: string;
name?: string;
}>({ isOpen: false }); }>({ isOpen: false });
useEffect(() => { useEffect(() => {
@ -76,6 +77,7 @@ export const ArchiveProjectList: FC = () => {
onDelete={() => onDelete={() =>
setDeleteProject({ setDeleteProject({
id, id,
name: projects?.find((project) => project.id === id)?.name,
isOpen: true, isOpen: true,
}) })
} }
@ -155,7 +157,8 @@ export const ArchiveProjectList: FC = () => {
} }
/> />
<DeleteProjectDialogue <DeleteProjectDialogue
project={deleteProject.id || ''} projectId={deleteProject.id || ''}
projectName={deleteProject.name || ''}
open={deleteProject.isOpen} open={deleteProject.isOpen}
onClose={() => { onClose={() => {
setDeleteProject((state) => ({ ...state, isOpen: false })); setDeleteProject((state) => ({ ...state, isOpen: false }));

View File

@ -1,4 +1,6 @@
import { styled, Typography } from '@mui/material';
import { Dialogue } from 'component/common/Dialogue/Dialogue'; import { Dialogue } from 'component/common/Dialogue/Dialogue';
import { ProjectId } from 'component/project/ProjectId/ProjectId';
import useProjectApi from 'hooks/api/actions/useProjectApi/useProjectApi'; import useProjectApi from 'hooks/api/actions/useProjectApi/useProjectApi';
import useProjects from 'hooks/api/getters/useProjects/useProjects'; import useProjects from 'hooks/api/getters/useProjects/useProjects';
import useToast from 'hooks/useToast'; import useToast from 'hooks/useToast';
@ -11,6 +13,10 @@ type ReviveProjectDialogProps = {
onClose: () => void; onClose: () => void;
}; };
const StyledParagraph = styled(Typography)(({ theme }) => ({
marginBottom: theme.spacing(1),
}));
export const ReviveProjectDialog = ({ export const ReviveProjectDialog = ({
name, name,
id, id,
@ -30,9 +36,9 @@ export const ReviveProjectDialog = ({
refetchProjects(); refetchProjects();
refetchProjectArchive(); refetchProjectArchive();
setToastData({ setToastData({
title: 'Restored project', title: 'Revive project',
type: 'success', type: 'success',
text: 'Successfully restored project', text: 'Successfully revived project',
}); });
} catch (ex: unknown) { } catch (ex: unknown) {
setToastApiError(formatUnknownError(ex)); setToastApiError(formatUnknownError(ex));
@ -43,14 +49,20 @@ export const ReviveProjectDialog = ({
return ( return (
<Dialogue <Dialogue
open={open} open={open}
secondaryButtonText='Close'
onClose={onClose} onClose={onClose}
onClick={onClick} onClick={onClick}
title='Restore archived project' title='Revive an archived project'
> >
Are you sure you'd like to restore project <strong>{name}</strong>{' '} <StyledParagraph>
(id: <code>{id}</code>)? Are you sure you'd like to revive project{' '}
{/* TODO: more explanation */} <strong>{name}</strong>?
</StyledParagraph>
<StyledParagraph>
Project ID: <ProjectId>{id}</ProjectId>
</StyledParagraph>
<StyledParagraph>
All flags in the revived project will remain archived.
</StyledParagraph>
</Dialogue> </Dialogue>
); );
}; };