Skip to content

Commit

Permalink
added playwright testing
Browse files Browse the repository at this point in the history
  • Loading branch information
csamuele committed Oct 23, 2024
1 parent 95267a5 commit 677cc77
Show file tree
Hide file tree
Showing 26 changed files with 933 additions and 81 deletions.
6 changes: 4 additions & 2 deletions packages/frontend/src/components/DragAndDrop/SortableItem.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -62,12 +62,14 @@ export function SortableItem({ children, id }: PropsWithChildren<Props>) {
interface DragHandleProps {
style?: CSSProperties;
disabled?: boolean;
ariaLabel?: string;
testId?: string;
}
export function DragHandle({ style, disabled }: DragHandleProps) {
export function DragHandle({ style, disabled, ariaLabel }: DragHandleProps) {
const { attributes, listeners, ref } = useContext(SortableItemContext);

return (
<IconButton {...attributes} {...listeners} ref={ref} style={style} disabled={disabled}>
<IconButton {...attributes} {...listeners} ref={ref} style={style} disabled={disabled} aria-label={ariaLabel}>
<MuiDragHandle />
</IconButton>
);
Expand Down
4 changes: 2 additions & 2 deletions packages/frontend/src/components/Election/ElectionHome.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -71,7 +71,7 @@ const ElectionHome = () => {
voterAuth.has_voted == false && voterAuth.authorized_voter && !voterAuth.required &&

<Box display='flex' flexDirection='column' alignItems='center' gap={5} sx={{ p: 1}}>
<Button fullWidth variant='contained' href={`/${String(election?.election_id)}/vote`} >
<Button role='button' name='vote' fullWidth variant='contained' href={`/${String(election?.election_id)}/vote`} >
<Typography align='center' variant="h3" component="h3" fontWeight='bold' sx={{ p: 2 }}>
{t('election_home.vote')}
</Typography>
Expand All @@ -87,7 +87,7 @@ const ElectionHome = () => {

{election.state === 'draft' && <>
<Box display='flex' flexDirection='column' alignItems='center' gap={5} sx={{ p: 1}}>
<Button fullWidth variant='contained' href={`/${String(election?.election_id)}/vote`} >
<Button role='button' name='vote' fullWidth variant='contained' href={`/${String(election?.election_id)}/vote`} >
<Typography align='center' variant="h3" component="h3" fontWeight='bold' sx={{ p: 2 }}>
{t('election_home.vote')}
</Typography>
Expand Down
Original file line number Diff line number Diff line change
@@ -1,7 +1,8 @@
import React, { useCallback, useMemo } from 'react';
import { Box, Typography } from '@mui/material';
import { Box, BoxProps, Button, Typography } from '@mui/material';
import { Candidate } from '@equal-vote/star-vote-shared/domain_model/Candidate';
import { IBallotContext } from '../VotePage';
import styled from '@emotion/styled';


interface BubbleGridProps {
Expand All @@ -14,14 +15,21 @@ interface BubbleGridProps {
fontSX: object;
}

const NoStyleButton = styled(Button)({
all: 'unset',
display: 'inline-block',
cursor: 'pointer',
});


const BubbleGrid: React.FC<BubbleGridProps> = ({ ballotContext, columnValues, columns, numHeaderRows, onClick, makeArea, fontSX }) => {
const { candidates, instructionsRead, alertBubbles } = ballotContext;
// Step 1: Create a triplet of candidateIndex, columnIndex, and columnValue for each candidate

const candidateColumnPairsNested = useMemo(() => {
return candidates.map((_, candidateIndex) =>
return candidates.map((candidate, candidateIndex) =>
columnValues.map((columnValue, columnIndex) =>
[candidateIndex, columnIndex, columnValue]
[candidateIndex, columnIndex, columnValue, candidate.candidate_name] as [number, number, number, string] // Add candidate name for accessibility
)
);
}, [candidates, columnValues]);
Expand All @@ -46,13 +54,14 @@ const BubbleGrid: React.FC<BubbleGridProps> = ({ ballotContext, columnValues, co
// Step 3: Map over the flattened array to render the Box components
return (
<>
{candidateColumnPairsFlat.map(([candidateIndex, columnIndex, columnValue]) => (
<Box
{candidateColumnPairsFlat.map(([candidateIndex, columnIndex, columnValue, candidate_name]) => (
<button
key={`${candidateIndex}-${columnIndex}`}
role='button'
name={`${candidate_name}_rank-${columnValue}`}
className={className(candidateIndex, columnValue)}
onClick={() => onClick(candidateIndex, columnValue)}
sx={{
margin: 'auto',
style={{
//For Rows:
//numHeaderRows offsets grid to account for header rows
//+1 offsets grid to account for candidateIndex starting at 0
Expand All @@ -66,7 +75,7 @@ const BubbleGrid: React.FC<BubbleGridProps> = ({ ballotContext, columnValues, co
<Typography variant='body1' sx={{ ...fontSX }}>
{columns.length === 1 ? ' ' : columnValue}
</Typography>
</Box>
</button>
))
}
</>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ interface ColumnHeadingsProps {
}
export default function ColumnHeadings({ columnIndex, columnTitle, gridArea, starHeadings, fontSX }: ColumnHeadingsProps) {
return (
<Box key={columnIndex} sx={{ gridArea: gridArea }}>
<Box key={columnIndex} sx={{ gridArea: gridArea }}className='column-headings'>
{starHeadings &&
<ScoreIcon
// NOTE: I tried doing this in CSS with :first-child but it didn't work
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -117,7 +117,7 @@ export default function GenericBallotGrid({
{rowBackgrounds}
{/* Column Warnings */}
{ballotContext.warningColumns && ballotContext.warningColumns.map((columnValue, columnIndex) =>
<Box key={columnIndex}
<Box key={columnIndex} aria-label={`warning-skipped-column-${columnValue}`}
sx={{
gridArea: makeArea(1, 1 + columnValue, 1, numHeaderRows + 1 + ballotContext.candidates.length * 2),
height: '100%',
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -186,6 +186,10 @@ const VotePage = () => {
if(pages.length == 0){
return <Container disableGutters={true} maxWidth="sm"><h3>No races created for election</h3></Container>
}
const isOnLastPage = currentPage === pages.length - 1
const noScores = pages.every(page => page.candidates.every(candidate => candidate.score === null))
const thereAreWarnings = pages.some(page => page.warnings)
const submitButtonDisabled = !isOnLastPage || (isPending || noScores || thereAreWarnings)

return (
<Container disableGutters={true} maxWidth="sm">
Expand Down Expand Up @@ -255,8 +259,9 @@ const VotePage = () => {
<Box sx={{ display: 'flex', justifyContent: "space-between" }}>
<Button
variant='contained'
name='submit-ballot'
onClick={() => setIsOpen(true)}
disabled={isPending || currentPage !== pages.length - 1 || pages[currentPage].candidates.every(candidate => candidate.score === null || pages.some(page => page.warnings))}//disable unless on last page and at least one candidate scored
disabled={submitButtonDisabled}//disable unless on last page and at least one candidate scored
style={{ margin: "auto", minWidth: "150px", marginTop: "40px" }}>
<Typography variant="h6">{t('ballot.submit_ballot')}</Typography>
</Button>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -307,27 +307,36 @@ const CandidateDialog = ({ onEditCandidate, candidate, index, onSave, open, hand
</Dialog>
)
}

export const CandidateForm = ({ onEditCandidate, candidate, index, onDeleteCandidate, disabled, inputRef, onKeyDown}) => {
interface CandidateFormProps {
onEditCandidate: (newCandidate: Candidate) => void,
candidate: Candidate,
index: number,
onDeleteCandidate: () => void,
disabled: boolean,
inputRef: (el: React.MutableRefObject<any[]>) => React.MutableRefObject<any[]>,
onKeyDown: (e: React.KeyboardEvent<HTMLInputElement>) => void,
electionState: string
}
export const CandidateForm = ({ onEditCandidate, candidate, index, onDeleteCandidate, disabled, inputRef, onKeyDown, electionState}: CandidateFormProps) => {

const [open, setOpen] = React.useState(false);
const handleOpen = () => setOpen(true);
const handleClose = () => setOpen(false);
const flags = useFeatureFlags();
const onSave = () => { handleClose() }
return (
<Paper elevation={4} sx={{ width: '100%' }}>
<Paper elevation={4} sx={{ width: '100%' }} aria-label={`candidate-form-${index + 1}`}>
<Box
sx={{ display: 'flex', justifyContent: 'space-between', bgcolor: 'background.paper', borderRadius: 10 }}
alignItems={'center'}
>
<DragHandle style={{marginLeft: 5}} disabled={disabled}/>
<DragHandle style={{marginLeft: 5}} disabled={disabled} ariaLabel={`drag-candidate-number-${index + 1}`}/>

<Box sx={{ overflow: 'hidden', textOverflow: 'ellipsis', width: '100%', pl: 2 }}>
<TextField
id={'candidate-name'}
name="new-candidate-name"
// label={"Candidate Name"}
id={`candidate-name-${index + 1}`}
name={`candidate-name-${index + 1}`}
data-testid={`candidate-name-${index + 1}`}
type="text"
value={candidate.candidate_name}
fullWidth
Expand All @@ -336,6 +345,7 @@ export const CandidateForm = ({ onEditCandidate, candidate, index, onDeleteCandi
onChange={(e) => onEditCandidate({ ...candidate, candidate_name: e.target.value })}
inputRef={inputRef}
onKeyDown={onKeyDown}
disabled={electionState !== 'draft'}
/>
</Box>

Expand All @@ -348,7 +358,8 @@ export const CandidateForm = ({ onEditCandidate, candidate, index, onDeleteCandi
</IconButton>
}
<IconButton
aria-label="delete"
aria-label={`delete-candidate-number-${index + 1}`}
name={`delete-${candidate.candidate_name}`}
color="error"
onClick={onDeleteCandidate}
disabled={disabled}>
Expand All @@ -360,7 +371,7 @@ export const CandidateForm = ({ onEditCandidate, candidate, index, onDeleteCandi
)
}

const AddCandidate = ({ onAddNewCandidate }) => {
const AddCandidate = ({ onAddNewCandidate, index }) => {

const handleEnter = (e) => {
saveNewCandidate()
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -282,7 +282,7 @@ export default () => {
setErrors={setErrors}
showLabel={false}
/>
<StepButtons activeStep={activeStep} setActiveStep={setActiveStep} canContinue={election.title != '' && errors.title == ''}/>
<StepButtons activeStep={activeStep} setActiveStep={setActiveStep} canContinue={/^[^\s][a-zA-Z0-9\s]{3,49}$/.test(election.title) && errors.title == ''}/>
</StepContent>
</Step>
<Step>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -17,8 +17,8 @@ export const ElectionTitleField = ({termType, value, onUpdateValue, errors, setE
inputProps={{ pattern: "[a-z]{3,15}" }}
error={errors.title !== ''}
required
id="election-name"
name="name"
id="election-title"
name="election-title"
// TODO: This bolding method only works for the text fields, if we like it we should figure out a way to add it to other fields as well
// inputProps={getStyle('title')}
label={showLabel? t('election_details.title') : ""}
Expand Down Expand Up @@ -71,7 +71,7 @@ export default function ElectionDetailsForm({editedElection, applyUpdate, errors
<Grid item xs={12} sx={{ m: 0, p: 1 }}>
<TextField
id="election-description"
name="description"
name="election-description"
label={t('election_details.description')}
multiline
fullWidth
Expand Down Expand Up @@ -125,10 +125,11 @@ export default function ElectionDetailsForm({editedElection, applyUpdate, errors
<>
<Grid item xs={4} sx={{ m: 0, p: 1 }} justifyContent='center'>
<FormControl fullWidth>
<InputLabel id="demo-simple-select-label">{t('election_details.time_zone')}</InputLabel>
<InputLabel id="time-zone-label">{t('election_details.time_zone')}</InputLabel>
<Select
labelId="demo-simple-select-label"
id="demo-simple-select"
labelId="time-zone-label"
id="time-zone-select"
name='edit-time-zone'
value={timeZone}
label={t('election_details.time_zone')}
onChange={(e) => {
Expand All @@ -152,6 +153,8 @@ export default function ElectionDetailsForm({editedElection, applyUpdate, errors
{/* datetime-local is formatted according to the OS locale, I don't think there's a way to override it*/}
<Input
type='datetime-local'
name='edit-start-time'
data-testid='start-time'
error={errors.startTime !== ''}
value={dateToLocalLuxonDate(editedElection.start_time, timeZone)}
onChange={(e) => {
Expand All @@ -176,6 +179,8 @@ export default function ElectionDetailsForm({editedElection, applyUpdate, errors
{/* datetime-local is formatted according to the OS locale, I don't think there's a way to override it*/}
<Input
type='datetime-local'
name='edit-end-time'
data-testid='end-time'
error={errors.endTime !== ''}
value={dateToLocalLuxonDate(editedElection.end_time, timeZone)}
onChange={(e) => {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -65,7 +65,8 @@ export default function ElectionDetailsInlineForm() {

<Box sx={{}}>
<IconButton
aria-label="edit"
aria-label="edit-election-details"
name="edit-election-details"
disabled={election.state!=='draft'}
onClick={handleOpen}>
<EditIcon />
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -82,13 +82,14 @@ export default function ElectionSettings() {
</Box>
<Box sx={{ flexShrink: 1, p: 1 }}>
<IconButton
aria-label="edit"
aria-label="edit-settings"
name="edit-settings"
onClick={handleOpen}>
<EditIcon />
</IconButton>
</Box>
</Box>
<Dialog
{open && <Dialog
open={open}
onClose={handleClose}
>
Expand All @@ -111,6 +112,7 @@ export default function ElectionSettings() {

<TextField
id="rank-limit"
name="rank-limit"
type="number"
value={editedElectionSettings.max_rankings ? editedElectionSettings.max_rankings : default_rankings}
onChange={(e) => applySettingsUpdate((settings) => { settings.max_rankings = Number(e.target.value) })}
Expand Down Expand Up @@ -142,7 +144,7 @@ export default function ElectionSettings() {
{t('keyword.save')}
</StyledButton>
</DialogActions>
</Dialog>
</Dialog>}
</Paper>
)
}
Original file line number Diff line number Diff line change
Expand Up @@ -28,22 +28,24 @@ export default function AddRace() {
}}>
<StyledButton
type='button'
name='add-race'
aria-label='add-race'
variant="contained"
fullWidth={false}
sx={{ borderRadius: 28, backgroundColor: 'brand.green' }}
onClick={handleOpen}
disabled={election.state!=='draft'}>
Add
</StyledButton>
<RaceDialog onSaveRace={onAdd} open={open} handleClose={handleClose} editedRace={editedRace}>
{open && <RaceDialog onSaveRace={onAdd} open={open} handleClose={handleClose} editedRace={editedRace}>
<RaceForm
race_index={election.races.length}
editedRace={editedRace}
errors={errors}
setErrors={setErrors}
applyRaceUpdate={applyRaceUpdate}
/>
</RaceDialog>
</RaceDialog>}
</Box>
)
}
4 changes: 3 additions & 1 deletion packages/frontend/src/components/ElectionForm/Races/Race.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,9 @@ export default function Race({ race, race_index }) {
<Box sx={{ flexShrink: 1, p: 1 }}>
<Tooltip title='Edit'>
<IconButton
aria-label="edit"
aria-label={`edit-${race.title}`}
role='button'
name={`edit-${race.title}`}
onClick={handleOpen}>
{election.state === 'draft' ? <EditIcon /> : <VisibilityIcon />}
</IconButton>
Expand Down
Loading

0 comments on commit 677cc77

Please sign in to comment.