1
0
mirror of https://github.com/Unleash/unleash.git synced 2025-01-01 00:08:27 +01:00
unleash.unleash/frontend/src/component/segments/SegmentFormStepOne.tsx
Gastón Fournier ab4a6f5c92
chore: GA project specific segments (#3583)
## About the changes
General availability of project-specific segments 

Now new permissions can be assigned to non-admin users to allow them to
create segments specifically bounded to a project (note that unique
naming restriction applies across projects, this limitation can be
lifted later).

![image](https://user-images.githubusercontent.com/455064/233579161-40ab25c5-a306-4fbc-82e5-8d09b3cb6a5f.png)

This enables a section in the project configuration to add
project-specific segments

![image](https://user-images.githubusercontent.com/455064/233578924-7056c626-ff1f-4dad-b00c-7fbd851158a6.png)
2023-04-21 10:25:40 +02:00

181 lines
6.3 KiB
TypeScript

import { Autocomplete, Button, styled, TextField } from '@mui/material';
import Input from 'component/common/Input/Input';
import React, { useEffect } from 'react';
import { useNavigate } from 'react-router-dom';
import { SegmentFormStep } from './SegmentForm';
import {
SEGMENT_NAME_ID,
SEGMENT_DESC_ID,
SEGMENT_NEXT_BTN_ID,
} from 'utils/testIds';
import { ConditionallyRender } from 'component/common/ConditionallyRender/ConditionallyRender';
import useUiConfig from 'hooks/api/getters/useUiConfig/useUiConfig';
import useProjects from 'hooks/api/getters/useProjects/useProjects';
import { useOptionalPathParam } from 'hooks/useOptionalPathParam';
import { GO_BACK } from 'constants/navigate';
import { useStrategiesBySegment } from 'hooks/api/getters/useStrategiesBySegment/useStrategiesBySegment';
import { SegmentProjectAlert } from './SegmentProjectAlert';
interface ISegmentFormPartOneProps {
name: string;
description: string;
project?: string;
setName: React.Dispatch<React.SetStateAction<string>>;
setDescription: React.Dispatch<React.SetStateAction<string>>;
setProject: React.Dispatch<React.SetStateAction<string | undefined>>;
errors: { [key: string]: string };
clearErrors: () => void;
setCurrentStep: React.Dispatch<React.SetStateAction<SegmentFormStep>>;
}
const StyledForm = styled('div')(({ theme }) => ({
display: 'flex',
flexDirection: 'column',
height: '100%',
}));
const StyledContainer = styled('div')(({ theme }) => ({
maxWidth: '400px',
}));
const StyledInputDescription = styled('p')(({ theme }) => ({
marginBottom: theme.spacing(1),
}));
const StyledInput = styled(Input)(({ theme }) => ({
width: '100%',
marginBottom: theme.spacing(2),
}));
const StyledButtonContainer = styled('div')(({ theme }) => ({
marginTop: 'auto',
display: 'flex',
justifyContent: 'flex-end',
}));
const StyledCancelButton = styled(Button)(({ theme }) => ({
marginLeft: theme.spacing(3),
}));
export const SegmentFormStepOne: React.FC<ISegmentFormPartOneProps> = ({
children,
name,
description,
project,
setName,
setDescription,
setProject,
errors,
clearErrors,
setCurrentStep,
}) => {
const segmentId = useOptionalPathParam('segmentId');
const projectId = useOptionalPathParam('projectId');
const { uiConfig } = useUiConfig();
const navigate = useNavigate();
const { projects, loading: loadingProjects } = useProjects();
const { strategies, loading: loadingStrategies } =
useStrategiesBySegment(segmentId);
const projectsUsed = new Set<string>(
strategies.map(({ projectId }) => projectId!).filter(Boolean)
);
const availableProjects = projects.filter(
({ id }) =>
!projectsUsed.size ||
(projectsUsed.size === 1 && projectsUsed.has(id))
);
const [selectedProject, setSelectedProject] = React.useState(
projects.find(({ id }) => id === project) ?? null
);
useEffect(() => {
setSelectedProject(projects.find(({ id }) => id === project) ?? null);
}, [project, projects]);
const loading = loadingProjects && loadingStrategies;
return (
<StyledForm>
<StyledContainer>
<StyledInputDescription>
What is the segment name?
</StyledInputDescription>
<StyledInput
label="Segment name"
value={name}
onChange={e => setName(e.target.value)}
error={Boolean(errors.name)}
errorText={errors.name}
autoFocus
required
data-testid={SEGMENT_NAME_ID}
/>
<StyledInputDescription>
What is the segment description?
</StyledInputDescription>
<StyledInput
label="Description (optional)"
value={description}
onChange={e => setDescription(e.target.value)}
error={Boolean(errors.description)}
errorText={errors.description}
data-testid={SEGMENT_DESC_ID}
/>
<ConditionallyRender
condition={!projectId && !loading}
show={
<>
<StyledInputDescription>
Is this segment tied to a specific project?
</StyledInputDescription>
<Autocomplete
size="small"
value={selectedProject}
onChange={(_, newValue) => {
setProject(newValue?.id);
}}
options={availableProjects}
getOptionLabel={option => option.name}
renderInput={params => (
<TextField {...params} label="Project" />
)}
disabled={projectsUsed.size > 1}
/>
<SegmentProjectAlert
projects={projects}
strategies={strategies}
projectsUsed={Array.from(projectsUsed)}
availableProjects={availableProjects}
/>
</>
}
/>
</StyledContainer>
<StyledButtonContainer>
<Button
type="button"
variant="contained"
color="primary"
onClick={() => setCurrentStep(2)}
disabled={name.length === 0 || Boolean(errors.name)}
data-testid={SEGMENT_NEXT_BTN_ID}
>
Next
</Button>
<StyledCancelButton
type="button"
onClick={() => {
navigate(GO_BACK);
}}
>
Cancel
</StyledCancelButton>
</StyledButtonContainer>
</StyledForm>
);
};