Add support for different states of proposal

This commit is contained in:
Kacper Donat 2020-08-02 23:00:18 +02:00 committed by Gitea
parent 1d08617d34
commit a3475c5f30
9 changed files with 126 additions and 22 deletions

View File

@ -17,6 +17,15 @@ const plugins = [
},
'core'
],
[
'babel-plugin-import',
{
'libraryName': 'mdi-material-ui',
'libraryDirectory': '.',
'camel2DashComponentName': false
},
'mdi-material-ui'
],
[
'babel-plugin-import',
{

View File

@ -2,6 +2,7 @@ import moment, { Moment } from "moment";
import { Box, Step as StepperStep, StepContent, StepLabel, StepProps as StepperStepProps, Typography } from "@material-ui/core";
import { useTranslation } from "react-i18next";
import React, { ReactChild, useMemo } from "react";
import { StepIcon } from "@/components/stepIcon";
type StepProps = StepperStepProps & {
until?: Moment;
@ -9,24 +10,21 @@ type StepProps = StepperStepProps & {
label: string;
state?: ReactChild | null;
/** this roughly translates into completed */
accepted?: boolean;
/** this roughly translates into error */
waiting?: boolean;
declined?: boolean;
sent?: boolean;
}
const now = moment();
export const Step = ({ until, label, completedOn, children, accepted = false, declined = false, completed = false, state = null, ...props }: StepProps) => {
export const Step = (props: StepProps) => {
const { until, label, completedOn, children, completed = false, declined = false, waiting = false, state = null, ...rest } = props;
const { t } = useTranslation();
const isLate = useMemo(() => until?.isBefore(completedOn || now), [completedOn, until]);
const left = useMemo(() => moment.duration(now.diff(until)), [until]);
return <StepperStep { ...props } completed={ completed }>
<StepLabel error={ declined }>
return <StepperStep { ...rest } completed={ completed }>
<StepLabel error={ declined } StepIconComponent={ StepIcon } StepIconProps={{ ...props, waiting } as any}>
{ label }
{ until && <Box>
{ state && <>

View File

@ -0,0 +1,22 @@
import { StepIcon as MuiStepIcon, StepIconProps as MuiStepIconProps, Theme } from "@material-ui/core";
import React from "react";
import { TimerSand } from "mdi-material-ui";
import { createStyles, makeStyles } from "@material-ui/core/styles";
type StepIconProps = MuiStepIconProps & {
waiting: boolean
}
const useStyles = makeStyles((theme: Theme) => createStyles({
root: {
color: theme.palette.primary.main
}
}))
export const StepIcon = ({ waiting, ...props }: StepIconProps) => {
const classes = useStyles();
return waiting
? <TimerSand className={ classes.root }/>
: <MuiStepIcon { ...props } />;
}

View File

@ -4,6 +4,7 @@ import { Link as RouterLink } from "react-router-dom";
import { route } from "@/routing";
import { InternshipForm } from "@/forms/internship";
import React from "react";
import { ProposalComment } from "@/pages";
export const InternshipProposalPage = () => {
return <Page title="Zgłoszenie praktyki">
@ -15,6 +16,7 @@ export const InternshipProposalPage = () => {
<Page.Title>Zgłoszenie praktyki</Page.Title>
</Page.Header>
<Container maxWidth={ "md" }>
<ProposalComment />
<InternshipForm/>
</Container>
</Page>

View File

@ -1,6 +1,6 @@
import React, { useMemo } from "react";
import React, { HTMLProps, useMemo } from "react";
import { Page } from "@/pages/base";
import { Button, Container, Stepper, StepProps, Theme, Typography } from "@material-ui/core";
import { Box, Button, ButtonProps, Container, Stepper, StepProps, Theme, Typography } from "@material-ui/core";
import { Link as RouterLink } from "react-router-dom";
import { route } from "@/routing";
import { useTranslation } from "react-i18next";
@ -12,6 +12,8 @@ import { Description as DescriptionIcon } from "@material-ui/icons"
import { Actions, Step } from "@/components";
import { getInternshipProposalStatus, InternshipProposalState, InternshipProposalStatus } from "@/state/reducer/proposal";
import { createStyles, makeStyles } from "@material-ui/core/styles";
import { ClipboardEditOutline, CommentQuestion, FileFind } from "mdi-material-ui";
import { Alert, AlertTitle } from "@material-ui/lab";
const getColorByStatus = (status: InternshipProposalStatus, theme: Theme) => {
@ -51,18 +53,70 @@ const ProposalStatus = () => {
return <span className={ classes.foreground }>{ t(`proposal.status.${status}`) }</span>;
}
const ProposalActions = () => {
const status = useSelector<AppState, InternshipProposalStatus>(state => getInternshipProposalStatus(state.proposal));
const { t } = useTranslation();
const ReviewAction = (props: ButtonProps) =>
<Button startIcon={ <FileFind /> } { ...props }>{ t('review') }</Button>
const FormAction = ({ children = t('steps.internship-proposal.form'), ...props }: ButtonProps) =>
<Button to={ route("internship_proposal") } variant="contained" color="primary" component={ RouterLink } startIcon={ <ClipboardEditOutline /> } { ...props as any }>
{ children }
</Button>
const ContactAction = (props: ButtonProps) =>
<Button startIcon={ <CommentQuestion /> } variant="outlined" color="primary" { ...props }>{ t('contact') }</Button>
switch (status) {
case "awaiting":
return <Actions>
<ReviewAction />
</Actions>
case "accepted":
return <Actions>
<ReviewAction />
<FormAction variant="outlined" color="secondary">{ t('make-changes') }</FormAction>
</Actions>
case "declined":
return <Actions>
<FormAction>{ t('fix-errors') }</FormAction>
<ContactAction/>
</Actions>
case "draft":
return <Actions>
<FormAction color="primary">{ t('steps.internship-proposal.action') }</FormAction>
</Actions>
default:
return <Actions />
}
}
export const ProposalComment = (props: HTMLProps<HTMLDivElement>) => {
const { comment, declined } = useSelector<AppState, InternshipProposalState>(state => state.proposal);
const { t } = useTranslation();
return <Alert severity={declined ? "error" : "warning"} {...props as any}>
<AlertTitle>{ t('comments') }</AlertTitle>
{ comment }
</Alert>
}
const ProposalStep = (props: StepProps) => {
const { t } = useTranslation();
const { sent } = useSelector<AppState, InternshipProposalState>(state => state.proposal);
const { sent, comment, declined } = useSelector<AppState, InternshipProposalState>(state => state.proposal);
const status = useSelector<AppState, InternshipProposalStatus>(state => getInternshipProposalStatus(state.proposal));
const deadlines = useSelector<AppState, Deadlines>(state => getEditionDeadlines(state.edition as Edition)); // edition cannot be null at this point
return <Step { ...props } label={ t('steps.internship-proposal.header') } active={ true } completed={ sent } until={ deadlines.proposal } state={ <ProposalStatus /> }>
<p>{ t('steps.internship-proposal.info') }</p>
<Button to={ route("internship_proposal") } variant="contained" color="primary" component={ RouterLink }>
{ t('steps.internship-proposal.form') }
</Button>
return <Step { ...props }
label={ t('steps.internship-proposal.header') }
active={ true } completed={ sent } declined={ declined } waiting={ status == "awaiting" }
until={ deadlines.proposal }
state={ <ProposalStatus /> }>
<p>{ t(`steps.internship-proposal.info.${status}`) }</p>
{ comment && <Box pb={2}><ProposalComment /></Box> }
<ProposalActions />
</Step>;
}

View File

@ -1,5 +1,5 @@
import { Nullable } from "@/helpers";
import { emptyCompany, Internship, Mentor } from "@/data";
import { emptyBranchOffice, emptyCompany, Internship, Mentor } from "@/data";
export const emptyMentor: Mentor = {
phone: null,
@ -18,4 +18,6 @@ export const emptyInternship: Nullable<Internship> = {
lengthInWeeks: 0,
mentor: emptyMentor,
company: emptyCompany,
hours: 0,
office: emptyBranchOffice,
}

View File

@ -14,7 +14,7 @@ export interface SendProposalAction extends Action<InternshipProposalActions.Sen
}
export interface ReceiveProposalApproveAction extends Action<InternshipProposalActions.Approve> {
comment: string | null;
}
export interface ReceiveProposalDeclineAction extends Action<InternshipProposalActions.Decline> {

View File

@ -52,7 +52,7 @@ const internshipProposalReducer = (state: InternshipProposalState = defaultInter
...state,
accepted: true,
declined: false,
comment: ""
comment: action.comment,
}
case InternshipProposalActions.Decline:
return {
@ -74,6 +74,7 @@ const internshipProposalReducer = (state: InternshipProposalState = defaultInter
sentOn: momentSerializationTransformer.transform(moment()),
accepted: false,
declined: false,
comment: null,
}
default:
return state;

View File

@ -12,6 +12,12 @@ left: jeszcze {{ left, humanize }}
confirm: zatwierdź
go-back: wstecz
make-changes: wprowadź zmiany
review: podgląd
fix-errors: popraw uwagi
contact: skontaktuj się z pełnomocnikiem
comments: Zgłoszone uwagi
dropzone: "Przeciągnij i upuść plik bądź kliknij, aby wybrać"
sections:
@ -48,9 +54,19 @@ steps:
form: "Uzupełnij dane"
internship-proposal:
header: "Zgłoszenie praktyki"
info: >
Przed podjęciem praktyki należy ją zgłosić.
info:
draft: >
Przed podjęciem praktyki należy ją zgłosić.
awaiting: >
Twoje zgłoszenie musi zostać zweryfikowane i zatwierdzone. Po weryfikacji zostaniesz poinformowany o
akceptacji bądź konieczności wprowadzenia zmian.
accepted: >
Twoje zgłoszenie zostało zweryfikowane i zaakceptowane.
declined: >
Twoje zgłoszenie zostało zweryfikowane i odrzucone. Popraw zgłoszone uwagi i wyślij zgłoszenie ponownie. W razie
pytań możesz również skontaktować się z pełnomocnikiem ds. praktyk Twojego kierunku.
form: "Formularz zgłaszania praktyki"
action: "zgłoś praktykę"
plan:
header: "Indywidualny Program Praktyki"
info: ""