You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
247 lines
6.9 KiB
247 lines
6.9 KiB
/**
|
|
* This file provides a context and a reducer to manage an election
|
|
*/
|
|
import {createContext, useContext, useReducer, useEffect, Dispatch, SetStateAction} from 'react';
|
|
import {useRouter} from 'next/router';
|
|
import {CandidateItem, GradeItem} from './type';
|
|
import {gradeColors} from '@services/grades';
|
|
|
|
export interface ElectionContextInterface {
|
|
name: string;
|
|
description: string;
|
|
candidates: Array<CandidateItem>;
|
|
grades: Array<GradeItem>;
|
|
hideResults: boolean;
|
|
forceClose: boolean;
|
|
restricted: boolean;
|
|
randomOrder: boolean;
|
|
emails: Array<string>;
|
|
dateEnd: string;
|
|
dateStart?: string;
|
|
ref?: string;
|
|
}
|
|
|
|
const defaultCandidate: CandidateItem = {
|
|
name: '',
|
|
image: '',
|
|
description: '',
|
|
active: false,
|
|
};
|
|
|
|
const defaultElection: ElectionContextInterface = {
|
|
name: '',
|
|
description: '',
|
|
candidates: [{...defaultCandidate}, {...defaultCandidate}],
|
|
grades: [],
|
|
randomOrder: true,
|
|
hideResults: false,
|
|
forceClose: false,
|
|
restricted: false,
|
|
dateEnd: null,
|
|
emails: [],
|
|
};
|
|
|
|
export enum ElectionTypes {
|
|
SET = 'set',
|
|
RESET = 'reset',
|
|
CANDIDATE_PUSH = 'candidate-push',
|
|
CANDIDATE_RM = 'candidate-rm',
|
|
CANDIDATE_SET = 'candidate-set',
|
|
GRADE_PUSH = 'grade-push',
|
|
GRADE_RM = 'grade-rm',
|
|
GRADE_SET = 'grade-set',
|
|
}
|
|
|
|
export type SetAction = {
|
|
type: ElectionTypes.SET;
|
|
field: string;
|
|
value: any;
|
|
}
|
|
export type ResetAction = {
|
|
type: ElectionTypes.RESET;
|
|
value: ElectionContextInterface;
|
|
}
|
|
export type CandidatePushAction = {
|
|
type: ElectionTypes.CANDIDATE_PUSH;
|
|
value: string | CandidateItem;
|
|
}
|
|
export type CandidateRmAction = {
|
|
type: ElectionTypes.CANDIDATE_RM;
|
|
position: number;
|
|
}
|
|
export type CandidateSetAction = {
|
|
type: ElectionTypes.CANDIDATE_SET;
|
|
position: number;
|
|
field: string;
|
|
value: any;
|
|
}
|
|
export type GradePushAction = {
|
|
type: ElectionTypes.GRADE_PUSH;
|
|
value: GradeItem;
|
|
}
|
|
export type GradeRmAction = {
|
|
type: ElectionTypes.GRADE_RM;
|
|
position: number;
|
|
}
|
|
export type GradeSetAction = {
|
|
type: ElectionTypes.GRADE_SET;
|
|
position: number;
|
|
field: string;
|
|
value: any;
|
|
}
|
|
|
|
export type ElectionActionTypes = SetAction | ResetAction | CandidateRmAction | CandidateSetAction | CandidatePushAction | GradeRmAction | GradeSetAction | GradePushAction;
|
|
|
|
type DispatchType = Dispatch<ElectionActionTypes>;
|
|
const ElectionContext = createContext<[ElectionContextInterface, DispatchType]>([defaultElection, () => {}]);
|
|
|
|
export function ElectionProvider({children}) {
|
|
/**
|
|
* Provide the election and the dispatch to all children components
|
|
*/
|
|
const [election, dispatch] = useReducer(electionReducer, defaultElection);
|
|
|
|
// At the initialization, set the name using GET param
|
|
const router = useRouter();
|
|
useEffect(() => {
|
|
if (!router.isReady) return;
|
|
|
|
if (election.name === "" && router.query.name !== "") {
|
|
dispatch({
|
|
type: ElectionTypes.SET,
|
|
field: 'name',
|
|
value: router.query.name,
|
|
});
|
|
}
|
|
|
|
}, [router.isReady]);
|
|
|
|
return (
|
|
<ElectionContext.Provider value={[election, dispatch]}>
|
|
{children}
|
|
</ElectionContext.Provider>
|
|
);
|
|
}
|
|
|
|
export function useElection() {
|
|
/**
|
|
* A simple hook to read the election
|
|
*/
|
|
return useContext(ElectionContext);
|
|
}
|
|
|
|
|
|
function electionReducer(election: ElectionContextInterface, action: ElectionActionTypes): ElectionContextInterface {
|
|
/**
|
|
* Manage all types of action doable on an election
|
|
*/
|
|
switch (action.type) {
|
|
case 'reset': {
|
|
return action.value;
|
|
}
|
|
case 'set': {
|
|
return {...election, [action.field]: action.value};
|
|
}
|
|
case 'candidate-push': {
|
|
if (typeof action.value === "string" && action.value !== "default") {
|
|
throw Error("Unexpected action")
|
|
}
|
|
const candidate =
|
|
action.value === 'default' ? {...defaultCandidate} : action.value;
|
|
const candidates = [...election.candidates, candidate];
|
|
if (candidates.filter(c => !c.active).length === 0) {
|
|
return {
|
|
...election, candidates: [...candidates, {...defaultCandidate}]
|
|
};
|
|
}
|
|
else {
|
|
return {...election, candidates};
|
|
}
|
|
}
|
|
case 'candidate-rm': {
|
|
if (typeof action.position !== 'number') {
|
|
throw Error(`Unexpected candidate position ${action.position}`);
|
|
}
|
|
const candidates = [...election.candidates];
|
|
candidates.splice(action.position, 1);
|
|
return {...election, candidates};
|
|
}
|
|
case 'candidate-set': {
|
|
if (typeof action.position !== 'number') {
|
|
throw Error(`Unexpected candidate position ${action.value}`);
|
|
}
|
|
if (action.field === 'active') {
|
|
throw Error('You are not allowed the set the active flag');
|
|
}
|
|
const candidates = [...election.candidates];
|
|
const candidate = candidates[action.position];
|
|
candidate[action.field] = action.value;
|
|
candidate['active'] = true;
|
|
if (candidates.filter(c => !c.active).length === 0) {
|
|
return {
|
|
...election, candidates: [...candidates, {...defaultCandidate}]
|
|
};
|
|
}
|
|
return {...election, candidates};
|
|
}
|
|
case 'grade-push': {
|
|
const grades = [...election.grades, action.value];
|
|
return {...election, grades};
|
|
}
|
|
case 'grade-rm': {
|
|
if (typeof action.position !== 'number') {
|
|
throw Error(`Unexpected grade position ${action.position}`);
|
|
}
|
|
const grades = [...election.grades];
|
|
grades.splice(action.position);
|
|
return {...election, grades};
|
|
}
|
|
case 'grade-set': {
|
|
if (typeof action.position !== 'number') {
|
|
throw Error(`Unexpected grade position ${action.position}`);
|
|
}
|
|
const grades = [...election.grades];
|
|
const grade = grades[action.position];
|
|
grade[action.field] = action.value;
|
|
return {...election, grades};
|
|
}
|
|
default: {
|
|
throw Error(`Unknown action: ${action.type}`);
|
|
}
|
|
}
|
|
}
|
|
|
|
export const isClosed = (election: ElectionContextInterface) => {
|
|
const dateEnd = new Date(election.dateEnd);
|
|
const now = new Date();
|
|
const isOver = +dateEnd < (+now);
|
|
return election.forceClose || isOver;
|
|
}
|
|
|
|
export const canViewResults = (election: ElectionContextInterface) => {
|
|
const dateEnd = new Date(election.dateEnd);
|
|
const now = new Date();
|
|
const isOver = +dateEnd < (+now);
|
|
return election.forceClose || !election.hideResults || isOver;
|
|
}
|
|
|
|
export const hasEnoughCandidates = (election: ElectionContextInterface) => {
|
|
|
|
const numCandidates = election.candidates.filter(c => c.active && c.name != "").length;
|
|
return numCandidates > 1;
|
|
}
|
|
|
|
export const hasEnoughGrades = (election: ElectionContextInterface) => {
|
|
|
|
const numGrades = election.grades.filter(g => g.active && g.name != "").length;
|
|
return numGrades > 1 && numGrades <= gradeColors.length
|
|
}
|
|
|
|
export const checkName = (election: ElectionContextInterface) => {
|
|
return election.name && election.name !== ""
|
|
}
|
|
|
|
export const canBeFinished = (election: ElectionContextInterface) => {
|
|
return election.restricted || election.forceClose || election.dateEnd || !election.hideResults;
|
|
}
|