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:
parent
cf3379d0b3
commit
004038e872
@ -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}
|
||||||
|
@ -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 />
|
||||||
|
@ -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>
|
||||||
);
|
);
|
||||||
|
@ -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>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@ -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);
|
||||||
|
@ -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} />;
|
||||||
};
|
};
|
||||||
|
@ -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);
|
||||||
|
@ -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);
|
||||||
|
9
frontend/src/component/project/ProjectId/ProjectId.tsx
Normal file
9
frontend/src/component/project/ProjectId/ProjectId.tsx
Normal 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,
|
||||||
|
}));
|
@ -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 }));
|
||||||
|
@ -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>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user