feat: migrate setting modals (#54112)

Co-authored-by: Huyen Nguyen <25715018+huyenltnguyen@users.noreply.github.com>
Co-authored-by: ahmad abdolsaheb <ahmad.abdolsaheb@gmail.com>
This commit is contained in:
Sem Bauke
2024-03-29 13:50:33 +01:00
committed by GitHub
parent f6ae52f6fe
commit 76b4c812cb
19 changed files with 223 additions and 328 deletions

View File

@@ -1,8 +1,7 @@
import { Modal } from '@freecodecamp/react-bootstrap';
import { connect } from 'react-redux';
import React from 'react';
import { useTranslation } from 'react-i18next';
import { Row, Button } from '@freecodecamp/ui';
import { Row, Button, Modal } from '@freecodecamp/ui';
import type { GeneratedExamResults } from '../../redux/prop-types';
import { closeModal } from '../../templates/Challenges/redux/actions';
@@ -51,36 +50,36 @@ const ExamResultsModal = ({
return (
<Modal
aria-labelledby='solution-viewer-modal-title'
bsSize='large'
onHide={() => {
onClose={() => {
closeModal('examResults');
}}
show={isOpen}
size='lg'
open={isOpen}
size='large'
>
<Modal.Header closeButton={true}>
<Modal.Title id='solution-viewer-modal-title'>
{t('settings.labels.results-for', { projectTitle })}
</Modal.Title>
<Modal.Header showCloseButton={true} closeButtonClassNames='close'>
{t('settings.labels.results-for', { projectTitle })}
</Modal.Header>
<Modal.Body style={{ paddingLeft: 30 }}>
<Modal.Body>
<Spacer size='medium' />
<Row>
{t('learn.exam.number-of-questions', {
n: numberOfQuestionsInExam
})}
</Row>{' '}
<Spacer size='medium' />
<Row>
{t('learn.exam.correct-answers', { n: numberOfCorrectAnswers })}
</Row>{' '}
<Spacer size='medium' />
<Row>{t('learn.exam.percent-correct', { n: percentCorrect })}</Row>
<Spacer size='medium' />{' '}
<Row>
{t('learn.exam.time', { t: formatSecondsToTime(examTimeInSeconds) })}
</Row>
<div style={{ paddingLeft: '30px' }}>
<Row>
{t('learn.exam.number-of-questions', {
n: numberOfQuestionsInExam
})}
</Row>{' '}
<Spacer size='medium' />
<Row>
{t('learn.exam.correct-answers', { n: numberOfCorrectAnswers })}
</Row>{' '}
<Spacer size='medium' />
<Row>{t('learn.exam.percent-correct', { n: percentCorrect })}</Row>
<Spacer size='medium' />{' '}
<Row>
{t('learn.exam.time', {
t: formatSecondsToTime(examTimeInSeconds)
})}
</Row>
</div>
<Spacer size='medium' />
</Modal.Body>
<Modal.Footer>

View File

@@ -1,7 +1,6 @@
import React from 'react';
import { useTranslation } from 'react-i18next';
import { Modal } from '@freecodecamp/react-bootstrap';
import { Button } from '@freecodecamp/ui';
import { Button, Modal } from '@freecodecamp/ui';
import { CompletedChallenge } from '../../redux/prop-types';
import SolutionViewer from './solution-viewer';
@@ -23,23 +22,14 @@ const ProjectModal = ({
}: ProjectModalProps): JSX.Element => {
const { t } = useTranslation();
return (
<Modal
data-playwright-test-label='project-solution-viewer-modal'
aria-labelledby='solution-viewer-modal-title'
bsSize='large'
onHide={handleSolutionModalHide}
show={isOpen}
size='lg'
>
<Modal.Header closeButton={true}>
<Modal.Title id='solution-viewer-modal-title'>
{t('settings.labels.solution-for', { projectTitle })}
</Modal.Title>
<Modal onClose={handleSolutionModalHide} open={isOpen} size='large'>
<Modal.Header showCloseButton={true} closeButtonClassNames='close'>
{t('settings.labels.solution-for', { projectTitle })}
</Modal.Header>
<Modal.Body>
<Modal.Body alignment='left'>
<SolutionViewer challengeFiles={challengeFiles} solution={solution} />
</Modal.Body>
<Modal.Footer>
<Modal.Footer alignment='end'>
<Button
data-cy='solution-viewer-close-btn'
onClick={handleSolutionModalHide}

View File

@@ -576,3 +576,14 @@ blockquote .small {
color: white;
background-color: black;
}
/*
* /learn sets some default styles to all `h2`s.
* These rules are to override the defaults and apply danger styles to `danger` modal.
*/
#headlessui-portal-root h2 {
font-weight: normal;
}
#headlessui-portal-root .text-background-danger h2 {
color: var(--background-danger);
}

View File

@@ -5,8 +5,7 @@ import React, { useMemo, useState } from 'react';
import type { TFunction } from 'i18next';
import { withTranslation } from 'react-i18next';
import { connect } from 'react-redux';
import { Modal } from '@freecodecamp/react-bootstrap';
import { Table, Button } from '@freecodecamp/ui';
import { Table, Button, Modal } from '@freecodecamp/ui';
import envData from '../../../../config/env.json';
import { getLangCode } from '../../../../../shared/config/i18n';
@@ -197,17 +196,11 @@ function TimelineInner({
</Table>
)}
{id && (
<Modal
aria-labelledby='contained-modal-title'
onHide={closeSolution}
show={solutionOpen}
>
<Modal.Header closeButton={true}>
<Modal.Title id='contained-modal-title' className='text-center'>
{`${username}'s Solution to ${
idToNameMap.get(id)?.challengeTitle ?? ''
}`}
</Modal.Title>
<Modal onClose={closeSolution} open={solutionOpen}>
<Modal.Header showCloseButton={true} closeButtonClassNames='close'>
{`${username}'s Solution to ${
idToNameMap.get(id)?.challengeTitle ?? ''
}`}
</Modal.Header>
<Modal.Body>
<SolutionViewer

View File

@@ -1,7 +1,6 @@
import { Modal } from '@freecodecamp/react-bootstrap';
import React from 'react';
import { Trans, useTranslation } from 'react-i18next';
import { Button } from '@freecodecamp/ui';
import { Button, Modal } from '@freecodecamp/ui';
import { Spacer } from '../helpers';
@@ -16,19 +15,9 @@ function DeleteModal(props: DeleteModalProps): JSX.Element {
const email = 'support@freecodecamp.org';
const { t } = useTranslation();
return (
<Modal
aria-labelledby='modal-title'
backdrop={true}
bsSize='lg'
className='text-center'
keyboard={true}
onHide={onHide}
show={show}
>
<Modal.Header closeButton={true}>
<Modal.Title id='modal-title'>
{t('settings.danger.delete-title')}
</Modal.Title>
<Modal onClose={onHide} open={show} variant='danger' size='large'>
<Modal.Header showCloseButton={true} closeButtonClassNames='close'>
{t('settings.danger.delete-title')}
</Modal.Header>
<Modal.Body>
<p>{t('settings.danger.delete-p1')}</p>
@@ -40,7 +29,8 @@ function DeleteModal(props: DeleteModalProps): JSX.Element {
</a>
</Trans>
</p>
<hr />
</Modal.Body>
<Modal.Footer>
<Button
block={true}
size='large'
@@ -60,9 +50,6 @@ function DeleteModal(props: DeleteModalProps): JSX.Element {
>
{t('settings.danger.certain')}
</Button>
</Modal.Body>
<Modal.Footer>
<Button onClick={props.onHide}>{t('buttons.close')}</Button>
</Modal.Footer>
</Modal>
);

View File

@@ -1,7 +1,6 @@
import { Modal } from '@freecodecamp/react-bootstrap';
import React from 'react';
import { useTranslation } from 'react-i18next';
import { Button } from '@freecodecamp/ui';
import { Button, Modal } from '@freecodecamp/ui';
import { Spacer } from '../helpers';
@@ -16,24 +15,15 @@ function ResetModal(props: ResetModalProps): JSX.Element {
const { show, onHide } = props;
return (
<Modal
aria-labelledby='modal-title'
backdrop={true}
bsSize='lg'
className='text-center'
keyboard={true}
onHide={onHide}
show={show}
>
<Modal.Header closeButton={true}>
<Modal.Title id='modal-title'>
{t('settings.danger.reset-heading')}
</Modal.Title>
<Modal size='large' onClose={onHide} variant='danger' open={show}>
<Modal.Header showCloseButton={true} closeButtonClassNames='close'>
{t('settings.danger.reset-heading')}
</Modal.Header>
<Modal.Body>
<p>{t('settings.danger.reset-p1')}</p>
<p>{t('settings.danger.reset-p2')}</p>
<hr />
</Modal.Body>
<Modal.Footer>
<Button
block={true}
size='large'
@@ -53,9 +43,6 @@ function ResetModal(props: ResetModalProps): JSX.Element {
>
{t('settings.danger.reset-confirm')}
</Button>
</Modal.Body>
<Modal.Footer>
<Button onClick={props.onHide}>{t('buttons.close')}</Button>
</Modal.Footer>
</Modal>
);

View File

@@ -1,10 +1,9 @@
import React from 'react';
import { Modal } from '@freecodecamp/react-bootstrap';
import { bindActionCreators, Dispatch, AnyAction } from 'redux';
import { createSelector } from 'reselect';
import { connect } from 'react-redux';
import { useTranslation } from 'react-i18next';
import { Button } from '@freecodecamp/ui';
import { Button, Modal } from '@freecodecamp/ui';
import { Spacer } from '../helpers';
import { hardGoTo as navigate, closeSignoutModal } from '../../redux/actions';
@@ -47,20 +46,9 @@ function SignoutModal(props: SignoutModalProps): JSX.Element {
};
return (
<Modal
aria-labelledby='modal-title'
backdrop={true}
bsSize='lg'
className='text-center'
keyboard={true}
onHide={handleModalHide}
onClose={handleModalHide}
show={show}
>
<Modal.Header closeButton={true}>
<Modal.Title id='modal-title' bsSize='lg'>
<span style={{ fontWeight: 'bold' }}>{t('signout.heading')}</span>
</Modal.Title>
<Modal size='large' variant='danger' open={show} onClose={handleModalHide}>
<Modal.Header showCloseButton={true} closeButtonClassNames='close'>
{t('signout.heading')}
</Modal.Header>
<Modal.Body>
<p>

View File

@@ -1,7 +1,6 @@
import React, { useState } from 'react';
import { Modal } from '@freecodecamp/react-bootstrap';
import { Trans, useTranslation } from 'react-i18next';
import { Button } from '@freecodecamp/ui';
import { Button, Modal } from '@freecodecamp/ui';
import store from 'store';
import { Spacer } from '../helpers';
@@ -17,23 +16,11 @@ function StagingWarningModal(): JSX.Element {
setShow(false);
};
return (
<Modal
aria-labelledby='modal-title'
backdrop={true}
bsSize='lg'
className='text-center'
keyboard={true}
onHide={handleModalHide}
onClose={handleModalHide}
show={show}
data-testid={'staging-warning-modal'}
>
<Modal.Header closeButton={true}>
<Modal.Title id='modal-title' bsSize='lg'>
<span style={{ fontWeight: 'bold' }}>
{t('staging-warning.heading')}
</span>
</Modal.Title>
<Modal onClose={handleModalHide} open={show} size='large'>
<Modal.Header showCloseButton={true} closeButtonClassNames='close'>
<span style={{ fontWeight: 'bold' }}>
{t('staging-warning.heading')}
</span>
</Modal.Header>
<Modal.Body>
<p className='text-justify'>{t('staging-warning.p1')}</p>

View File

@@ -2,23 +2,40 @@ import React from 'react';
import { render, screen, fireEvent } from '@testing-library/react';
import store from 'store';
import StagingWarningModal from '.';
describe('StagingWarningModal', () => {
beforeAll(() => {
// The Modal component uses `ResizeObserver` under the hood.
// However, this property is not available in JSDOM, so we need to manually add it to the window object.
// Ref: https://github.com/jsdom/jsdom/issues/3368
Object.defineProperty(window, 'ResizeObserver', {
writable: true,
value: jest.fn(() => ({
observe: jest.fn(),
unobserve: jest.fn(),
disconnect: jest.fn()
}))
});
});
afterEach(() => {
jest.clearAllMocks();
});
it('renders the modal successfully', () => {
render(<StagingWarningModal />);
expect(screen.getByTestId('staging-warning-modal')).toBeInTheDocument();
expect(screen.getByTestId('staging-warning-modal')).toHaveClass('in');
expect(screen.getByRole('dialog')).toBeInTheDocument();
});
it('closes the modal when clicking the close button', () => {
render(<StagingWarningModal />);
fireEvent.click(screen.getByText('Close'));
expect(screen.getByTestId('staging-warning-modal')).not.toHaveClass('in');
expect(screen.queryByRole('dialog')).not.toBeInTheDocument();
});
it('displays the correct modal content', () => {
render(<StagingWarningModal />);
const modalContent = screen.getByTestId('staging-warning-modal');
const modalContent = screen.getByRole('dialog');
expect(modalContent).toHaveTextContent('staging-warning.heading');
expect(modalContent).toHaveTextContent('staging-warning.p1');
expect(modalContent).toHaveTextContent('staging-warning.p2');
@@ -27,8 +44,10 @@ describe('StagingWarningModal', () => {
it('accepts Warning, stores acceptance key in local storage, and closes the modal', () => {
render(<StagingWarningModal />);
fireEvent.click(screen.getByTestId('accepts-warning'));
expect(store.get('acceptedStagingWarning')).toBe(true);
expect(screen.queryByTestId('staging-warning-modal')).not.toHaveClass('in');
expect(screen.queryByRole('dialog')).not.toBeInTheDocument();
});
});

View File

@@ -1,24 +0,0 @@
.reset-modal p {
color: var(--danger-color);
}
.reset-modal-header {
color: var(--danger-color);
background-color: var(--danger-background);
}
.reset-modal-header h4 {
color: var(--danger-color);
}
.reset-modal-header .close {
color: var(--danger-color);
font-size: 28px;
text-shadow: none;
}
@media screen and (max-width: 767px) {
.reset-modal .btn-lg {
font-size: 16px;
}
}

View File

@@ -1,28 +1,20 @@
// Package Utilities
import { Modal } from '@freecodecamp/react-bootstrap';
import React from 'react';
import { useTranslation } from 'react-i18next';
import { connect } from 'react-redux';
import { bindActionCreators, Dispatch } from 'redux';
import { createSelector } from 'reselect';
import { Button } from '@freecodecamp/ui';
import { Button, Modal } from '@freecodecamp/ui';
// Local Utilities
import { closeModal, resetChallenge } from '../redux/actions';
import { isResetModalOpenSelector } from '../redux/selectors';
import callGA from '../../../analytics/call-ga';
// Styles
import './reset-modal.css';
// Types
interface ResetModalProps {
close: () => void;
isOpen: boolean;
reset: () => void;
}
// Redux Setup
const mapStateToProps = createSelector(
isResetModalOpenSelector,
(isOpen: boolean) => ({
@@ -43,25 +35,17 @@ function withActions(...fns: Array<() => void>) {
return () => fns.forEach(fn => fn());
}
// Component
function ResetModal({ reset, close, isOpen }: ResetModalProps): JSX.Element {
const { t } = useTranslation();
if (isOpen) {
callGA({ event: 'pageview', pagePath: '/reset-modal' });
}
return (
<Modal
data-playwright-test-label='reset-modal'
animation={false}
dialogClassName='reset-modal'
keyboard={true}
onHide={close}
show={isOpen}
>
<Modal.Header className='reset-modal-header' closeButton={true}>
<Modal.Title className='text-center'>{t('learn.reset')}</Modal.Title>
<Modal onClose={close} open={isOpen} variant='danger'>
<Modal.Header showCloseButton={true} closeButtonClassNames='close'>
{t('learn.reset')}
</Modal.Header>
<Modal.Body className='reset-modal-body'>
<Modal.Body>
<div className='text-center'>
<p>{t('learn.reset-warn')}</p>
<p>
@@ -69,7 +53,7 @@ function ResetModal({ reset, close, isOpen }: ResetModalProps): JSX.Element {
</p>
</div>
</Modal.Body>
<Modal.Footer className='reset-modal-footer'>
<Modal.Footer>
<Button
data-cy='reset-modal-confirm'
block={true}

View File

@@ -13,20 +13,16 @@ test.describe('Delete Modal component', () => {
.getByRole('button', { name: translations.settings.danger.delete })
.click();
// There are two elements with the `dialog` role in the DOM.
// This appears to be semantically incorrect and should be resolved
// once we have migrated the component to use Dialog from the `ui-components` library.
const dialogs = await page.getByRole('dialog').all();
expect(dialogs).toHaveLength(2);
await expect(
page.getByRole('heading', {
page.getByRole('dialog', {
name: translations.settings.danger['delete-title']
})
).toBeVisible();
await expect(
page.getByText(translations.settings.danger['delete-p1'])
).toBeVisible();
await expect(
page.getByText(translations.settings.danger['delete-p2'])
).toBeVisible();
@@ -47,11 +43,9 @@ test.describe('Delete Modal component', () => {
page.getByRole('button', { name: translations.settings.danger.certain })
).toBeVisible();
// There are 2 close buttons on the modal: one is sr-only on top, and one on the bottom of modal
const closeButtons = await page
.getByRole('button', { name: translations.buttons.close })
.all();
expect(closeButtons).toHaveLength(2);
await expect(
page.getByRole('button', { name: translations.buttons.close })
).toBeVisible();
});
test('should close the modal after the user cancels account deleting', async ({
@@ -61,16 +55,21 @@ test.describe('Delete Modal component', () => {
.getByRole('button', { name: translations.settings.danger.delete })
.click();
const dialogs = await page.getByRole('dialog').all();
expect(dialogs).toHaveLength(2);
await expect(
page.getByRole('dialog', {
name: translations.settings.danger['delete-title']
})
).toBeVisible();
await page
.getByRole('button', { name: translations.settings.danger.nevermind })
.click();
for (const dialog of dialogs) {
await expect(dialog).not.toBeVisible();
}
await expect(
page.getByRole('dialog', {
name: translations.settings.danger['delete-title']
})
).not.toBeVisible();
});
test('should close the modal and redirect to /learn after the user clicks delete', async ({
@@ -87,16 +86,21 @@ test.describe('Delete Modal component', () => {
.getByRole('button', { name: translations.settings.danger.delete })
.click();
const dialogs = await page.getByRole('dialog').all();
expect(dialogs).toHaveLength(2);
await expect(
page.getByRole('dialog', {
name: translations.settings.danger['delete-title']
})
).toBeVisible();
await page
.getByRole('button', { name: translations.settings.danger.certain })
.click();
for (const dialog of dialogs) {
await expect(dialog).not.toBeVisible();
}
await expect(
page.getByRole('dialog', {
name: translations.settings.danger['delete-title']
})
).not.toBeVisible();
await expect(page).toHaveURL(/.*\/learn\/?/);
});

View File

@@ -33,9 +33,9 @@ test('Click on the "Reset" button', async ({ page }) => {
const resetButton = page.getByTestId('lowerJaw-reset-button');
await resetButton.click();
const resetModal = page.getByTestId('reset-modal');
await expect(resetModal).toBeVisible();
await expect(
page.getByRole('dialog', { name: 'Reset this lesson?' })
).toBeVisible();
});
test('Should render UI correctly', async ({ page }) => {

View File

@@ -16,11 +16,9 @@ test('should render the modal content correctly', async ({ page }) => {
await page.getByRole('button', { name: translations.buttons.reset }).click();
// There are two elements with the `dialog` role in the DOM.
// This appears to be semantically incorrect and should be resolved
// once we have migrated the component to use Dialog from the `ui-components` library.
const dialogs = await page.getByRole('dialog').all();
expect(dialogs).toHaveLength(2);
await expect(
page.getByRole('dialog', { name: translations.learn.reset })
).toBeVisible();
await expect(
page.getByRole('button', {
@@ -138,11 +136,9 @@ test('should close when the user clicks the close button', async ({ page }) => {
await page.getByRole('button', { name: translations.buttons.reset }).click();
// There are two elements with the `dialog` role in the DOM.
// This appears to be semantically incorrect and should be resolved
// once we have migrated the component to use Dialog from the `ui-components` library.
const dialogs = await page.getByRole('dialog').all();
expect(dialogs).toHaveLength(2);
await expect(
page.getByRole('dialog', { name: translations.learn.reset })
).toBeVisible();
await page
.getByRole('button', {
@@ -150,5 +146,7 @@ test('should close when the user clicks the close button', async ({ page }) => {
})
.click();
await expect(page.getByRole('dialog')).toBeHidden();
await expect(
page.getByRole('dialog', { name: translations.learn.reset })
).toBeHidden();
});

View File

@@ -14,10 +14,10 @@ test.describe('Signout Modal component', () => {
.getByRole('button', { name: translations.buttons['sign-out'] })
.click();
const dialogs = page.getByRole('dialog');
await expect(dialogs).toHaveCount(2);
await expect(
page.getByRole('dialog', { name: translations.signout.heading })
).toBeVisible();
await expect(page.getByText(translations.signout.heading)).toBeVisible();
await expect(page.getByText(translations.signout.p1)).toBeVisible();
await expect(page.getByText(translations.signout.p2)).toBeVisible();
@@ -40,16 +40,17 @@ test.describe('Signout Modal component', () => {
.getByRole('button', { name: translations.buttons['sign-out'] })
.click();
const dialogs = page.getByRole('dialog');
await expect(dialogs).toHaveCount(2);
await expect(
page.getByRole('dialog', { name: translations.signout.heading })
).toBeVisible();
await page
.getByRole('button', { name: translations.signout.certain })
.click();
for (const dialog of await dialogs.all()) {
await expect(dialog).not.toBeVisible();
}
await expect(
page.getByRole('dialog', { name: translations.signout.heading })
).not.toBeVisible();
await expect(page).toHaveURL(/.*\/learn\/?$/);
});
@@ -59,19 +60,18 @@ test.describe('Signout Modal component', () => {
.getByRole('button', { name: translations.buttons['sign-out'] })
.click();
const dialogs = page.getByRole('dialog');
await expect(dialogs).toHaveCount(2);
await expect(
page.getByRole('dialog', { name: translations.signout.heading })
).toBeVisible();
await page
.getByRole('button', { name: translations.signout.nevermind })
.click();
for (const dialog of await dialogs.all()) {
await expect(dialog).not.toBeVisible();
}
await expect(page).toHaveURL('/');
await expect(
page.getByText(translations.signout.heading)
page.getByRole('dialog', { name: translations.signout.heading })
).not.toBeVisible();
await expect(page).toHaveURL('/');
});
});

View File

@@ -1,71 +0,0 @@
import { test, expect } from '@playwright/test';
test.use({ storageState: 'playwright/.auth/certified-user.json' });
test.beforeEach(async ({ page }) => {
await page.goto('/settings');
});
test('Multifile dropdown testing', async ({ page }) => {
const multifile_dropdown = page.getByTestId('multifile-dropdown');
const isVisible = await multifile_dropdown.isVisible();
if (isVisible) {
await multifile_dropdown.click();
const multifile_dropdown_code = page.getByTestId('multifile-dropdown-code');
const multifile_dropdown_project = page.getByTestId(
'multifile-dropdown-project'
);
await expect(multifile_dropdown_code).toBeVisible();
await expect(multifile_dropdown_project).toBeVisible();
await multifile_dropdown_code.click();
const project_solution_viewer_modal = page.getByTestId(
'project-solution-viewer-modal'
);
await expect(project_solution_viewer_modal).toBeVisible();
const close_button = page.locator('button.close');
await close_button.click();
await multifile_dropdown.click();
await multifile_dropdown_project.click();
const project_preview_modal = page.getByTestId('project-preview-modal');
await expect(project_preview_modal).toBeVisible();
}
});
test('Single Solution testing', async ({ page }) => {
const solution_button = page.getByRole('button', {
name: /Solution for Palindrome Checker/i
});
const isVisible = await solution_button.isVisible();
if (isVisible) {
await solution_button.click();
const solution_viewer_modal = page.getByTestId(
'project-solution-viewer-modal'
);
await expect(solution_viewer_modal).toBeVisible();
}
});
test('External solution testing', async ({ page }) => {
const solution_button = page
.getByRole('button', {
name: /Solution for Build a Random Quote Machine/i
})
.first();
const isVisible = await solution_button.isVisible();
if (isVisible) {
const browserContext = page.context();
const [newPage] = await Promise.all([
browserContext.waitForEvent('page'),
solution_button.click()
]);
await newPage.waitForLoadState();
await expect(newPage).toHaveURL(/^https:\/\/codepen\.io/);
await newPage.close();
}
});

View File

@@ -2,25 +2,19 @@ import { test, expect } from '@playwright/test';
test.use({ storageState: 'playwright/.auth/certified-user.json' });
test.beforeEach(async ({ page }) => {
await page.goto(
'/certification/certifieduser/javascript-algorithms-and-data-structures'
);
});
test.describe('Solution Viewer component', () => {
test('renders the modal correctly', async ({ page }) => {
await page.getByRole('button').filter({ hasText: /view/i }).first().click();
const projectSolutionViewerModal = page.getByTestId(
'project-solution-viewer-modal'
await page.goto(
'/certification/certifieduser/javascript-algorithms-and-data-structures'
);
await expect(projectSolutionViewerModal).toBeVisible();
await page.getByRole('button', { name: /view/i }).first().click();
const projectSolutionViewerModal = page.getByRole('dialog', {
name: 'Solution for'
});
// The modal should show the solution title...
await expect(
page.getByRole('heading').and(page.getByText(/solution for/i))
).toBeVisible();
await expect(projectSolutionViewerModal).toBeVisible();
// ...and relevant code file/s
await expect(page.getByText(/js/i)).toBeVisible();
await expect(page.locator('pre').first()).toBeVisible();
@@ -38,4 +32,47 @@ test.describe('Solution Viewer component', () => {
await bottomRightCloseButton.click();
await expect(projectSolutionViewerModal).toBeHidden();
});
test('renders external project links correctly', async ({ page }) => {
await page.goto(
'/certification/certifieduser/front-end-development-libraries'
);
const projectLink = page.getByRole('link', { name: 'View' }).first();
const browserContext = page.context();
const [newPage] = await Promise.all([
browserContext.waitForEvent('page'),
projectLink.click()
]);
await newPage.waitForLoadState();
await expect(newPage).toHaveURL(/^https:\/\/codepen\.io/);
await newPage.close();
});
test('render projects with multiple solutions correctly', async ({
page
}) => {
await page.goto('/certification/certifieduser/quality-assurance-v7');
const dropdownButton = page.getByRole('button', { name: 'View' }).first();
await dropdownButton.click();
await expect(page.getByRole('menu')).toBeVisible();
const sourceLink = page.getByRole('menuitem', { name: /source/i }).first();
const browserContext = page.context();
const [newPage] = await Promise.all([
browserContext.waitForEvent('page'),
sourceLink.click()
]);
await newPage.waitForLoadState();
await newPage.close();
});
});

View File

@@ -1,13 +1,8 @@
import React, {
type ReactNode,
createContext,
useContext,
Fragment
} from 'react';
import React, { createContext, useContext, Fragment } from 'react';
import { Dialog, Transition } from '@headlessui/react';
import { CloseButton } from '../close-button';
import { type ModalProps, type HeaderProps, BodyProps } from './types';
import type { ModalProps, HeaderProps, BodyProps, FooterProps } from './types';
// There is a close button on the right side of the modal title.
// Some extra padding needs to be added to the left of the title text
@@ -73,8 +68,14 @@ const Body = ({ children, alignment = 'center' }: BodyProps) => {
);
};
const Footer = ({ children }: { children: ReactNode }) => {
return <div className='p-[15px]'>{children}</div>;
const Footer = ({ children, alignment = 'center' }: FooterProps) => {
if (alignment === 'end') {
return <div className='p-[15px] flex justify-end'>{children}</div>;
}
return (
<div className={`p-[15px] flex flex-col justify-center`}>{children}</div>
);
};
const Modal = ({
@@ -101,12 +102,12 @@ const Modal = ({
return (
<ModalContext.Provider value={{ onClose, variant }}>
<Transition.Root show={open} as={Fragment}>
<Dialog onClose={onClose} className='relative z-1050'>
<Dialog onClose={onClose} className='relative z-1050 w-screen h-screen'>
{/* The backdrop, rendered as a fixed sibling to the panel container */}
<div aria-hidden className='fixed inset-0 bg-gray-900 opacity-50' />
{/* Full-screen container of the panel */}
<div className='fixed inset-0 w-screen flex items-start justify-center pt-[30px]'>
<div className='fixed inset-0 flex items-start justify-center pt-[30px] pb-[30px] overflow-scroll'>
<Transition.Child
as={Fragment}
enter='transition-all duration-300 ease-out'

View File

@@ -32,3 +32,8 @@ export interface BodyProps {
*/
alignment?: 'center' | 'left' | 'start';
}
export interface FooterProps {
children: ReactNode;
alignment?: 'center' | 'end';
}