chore(eslint): enables recommended eslint rules for testing-library and jest-dom (#42817)
				
					
				
			This commit is contained in:
		| @@ -57,7 +57,7 @@ | ||||
|       "plugins": ["@typescript-eslint"] | ||||
|     }, | ||||
|     { | ||||
|       "files": ["./tools/ui-components/**/*.test.[jt]s?(x)"], | ||||
|       "files": ["./tools/ui-components/**/*.test.[jt]s?(x)", "./client/**/*.test.[jt]s?(x)"], | ||||
|       "extends": ["plugin:testing-library/react", "plugin:jest-dom/recommended"] | ||||
|     } | ||||
|   ] | ||||
|   | ||||
| @@ -25,8 +25,8 @@ describe('<UniversalNav />', () => { | ||||
|   it('renders to the DOM', () => { | ||||
|     const shallow = new ShallowRenderer(); | ||||
|     shallow.render(<UniversalNav {...UniversalNavProps} />); | ||||
|     const result = shallow.getRenderOutput(); | ||||
|     expect(result).toBeTruthy(); | ||||
|     const view = shallow.getRenderOutput(); | ||||
|     expect(view).toBeTruthy(); | ||||
|   }); | ||||
| }); | ||||
|  | ||||
| @@ -48,14 +48,14 @@ describe('<NavLinks />', () => { | ||||
|     }; | ||||
|     const shallow = new ShallowRenderer(); | ||||
|     shallow.render(<NavLinks {...landingPageProps} />); | ||||
|     const result = shallow.getRenderOutput(); | ||||
|     const view = shallow.getRenderOutput(); | ||||
|     expect( | ||||
|       hasDonateNavItem(result) && | ||||
|         hasSignInNavItem(result) && | ||||
|         hasCurriculumNavItem(result) && | ||||
|         hasForumNavItem(result) && | ||||
|         hasNewsNavItem(result) && | ||||
|         hasRadioNavItem(result) | ||||
|       hasDonateNavItem(view) && | ||||
|         hasSignInNavItem(view) && | ||||
|         hasCurriculumNavItem(view) && | ||||
|         hasForumNavItem(view) && | ||||
|         hasNewsNavItem(view) && | ||||
|         hasRadioNavItem(view) | ||||
|     ).toBeTruthy(); | ||||
|   }); | ||||
|  | ||||
| @@ -76,15 +76,15 @@ describe('<NavLinks />', () => { | ||||
|     }; | ||||
|     const shallow = new ShallowRenderer(); | ||||
|     shallow.render(<NavLinks {...landingPageProps} />); | ||||
|     const result = shallow.getRenderOutput(); | ||||
|     const view = shallow.getRenderOutput(); | ||||
|     expect( | ||||
|       hasDonateNavItem(result) && | ||||
|         hasCurriculumNavItem(result) && | ||||
|         hasProfileAndSettingsNavItems(result, landingPageProps.user.username) && | ||||
|         hasForumNavItem(result) && | ||||
|         hasNewsNavItem(result) && | ||||
|         hasRadioNavItem(result) && | ||||
|         hasSignOutNavItem(result) | ||||
|       hasDonateNavItem(view) && | ||||
|         hasCurriculumNavItem(view) && | ||||
|         hasProfileAndSettingsNavItems(view, landingPageProps.user.username) && | ||||
|         hasForumNavItem(view) && | ||||
|         hasNewsNavItem(view) && | ||||
|         hasRadioNavItem(view) && | ||||
|         hasSignOutNavItem(view) | ||||
|     ).toBeTruthy(); | ||||
|   }); | ||||
|  | ||||
| @@ -105,15 +105,15 @@ describe('<NavLinks />', () => { | ||||
|     }; | ||||
|     const shallow = new ShallowRenderer(); | ||||
|     shallow.render(<NavLinks {...landingPageProps} />); | ||||
|     const result = shallow.getRenderOutput(); | ||||
|     const view = shallow.getRenderOutput(); | ||||
|     expect( | ||||
|       hasThanksForDonating(result) && | ||||
|         hasCurriculumNavItem(result) && | ||||
|         hasProfileAndSettingsNavItems(result, landingPageProps.user.username) && | ||||
|         hasForumNavItem(result) && | ||||
|         hasNewsNavItem(result) && | ||||
|         hasRadioNavItem(result) && | ||||
|         hasSignOutNavItem(result) | ||||
|       hasThanksForDonating(view) && | ||||
|         hasCurriculumNavItem(view) && | ||||
|         hasProfileAndSettingsNavItems(view, landingPageProps.user.username) && | ||||
|         hasForumNavItem(view) && | ||||
|         hasNewsNavItem(view) && | ||||
|         hasRadioNavItem(view) && | ||||
|         hasSignOutNavItem(view) | ||||
|     ).toBeTruthy(); | ||||
|   }); | ||||
| }); | ||||
| @@ -131,8 +131,8 @@ describe('<AuthOrProfile />', () => { | ||||
|  | ||||
|     const shallow = new ShallowRenderer(); | ||||
|     shallow.render(<AuthOrProfile {...defaultUserProps} />); | ||||
|     const componentTree = shallow.getRenderOutput(); | ||||
|     expect(avatarHasClass(componentTree, 'default-border')).toBeTruthy(); | ||||
|     const view = shallow.getRenderOutput(); | ||||
|     expect(avatarHasClass(view, 'default-border')).toBeTruthy(); | ||||
|   }); | ||||
|  | ||||
|   it('has avatar with gold border for donating users', () => { | ||||
| @@ -147,9 +147,9 @@ describe('<AuthOrProfile />', () => { | ||||
|     }; | ||||
|     const shallow = new ShallowRenderer(); | ||||
|     shallow.render(<AuthOrProfile {...donatingUserProps} />); | ||||
|     const componentTree = shallow.getRenderOutput(); | ||||
|     const view = shallow.getRenderOutput(); | ||||
|  | ||||
|     expect(avatarHasClass(componentTree, 'gold-border')).toBeTruthy(); | ||||
|     expect(avatarHasClass(view, 'gold-border')).toBeTruthy(); | ||||
|   }); | ||||
|  | ||||
|   it('has avatar with blue border for top contributors', () => { | ||||
| @@ -165,9 +165,9 @@ describe('<AuthOrProfile />', () => { | ||||
|  | ||||
|     const shallow = new ShallowRenderer(); | ||||
|     shallow.render(<AuthOrProfile {...topContributorUserProps} />); | ||||
|     const componentTree = shallow.getRenderOutput(); | ||||
|     const view = shallow.getRenderOutput(); | ||||
|  | ||||
|     expect(avatarHasClass(componentTree, 'blue-border')).toBeTruthy(); | ||||
|     expect(avatarHasClass(view, 'blue-border')).toBeTruthy(); | ||||
|   }); | ||||
|   it('has avatar with purple border for donating top contributors', () => { | ||||
|     const topDonatingContributorUserProps = { | ||||
| @@ -182,8 +182,8 @@ describe('<AuthOrProfile />', () => { | ||||
|     }; | ||||
|     const shallow = new ShallowRenderer(); | ||||
|     shallow.render(<AuthOrProfile {...topDonatingContributorUserProps} />); | ||||
|     const componentTree = shallow.getRenderOutput(); | ||||
|     expect(avatarHasClass(componentTree, 'purple-border')).toBeTruthy(); | ||||
|     const view = shallow.getRenderOutput(); | ||||
|     expect(avatarHasClass(view, 'purple-border')).toBeTruthy(); | ||||
|   }); | ||||
| }); | ||||
|  | ||||
|   | ||||
| @@ -16,7 +16,19 @@ describe('<Intro />', () => { | ||||
|     const container = rendererCreateWithRedux( | ||||
|       <Intro {...loggedOutProps} /> | ||||
|     ).root; | ||||
|  | ||||
|     /** | ||||
|      * This rules had to be disabled because the new lint rules are throwing false positives here. | ||||
|      * They were interpreting react-test-renderer functions as @testing-library/react functions. | ||||
|      */ | ||||
|     // eslint-disable-next-line testing-library/await-async-query | ||||
|     expect(container.findAllByType('blockquote').length === 0).toBeTruthy(); | ||||
|  | ||||
|     /** | ||||
|      * This rules had to be disabled because the new lint rules are throwing false positives here. | ||||
|      * They were interpreting react-test-renderer functions as @testing-library/react functions. | ||||
|      */ | ||||
|     // eslint-disable-next-line testing-library/await-async-query | ||||
|     expect(container.findAllByType('h1').length === 1).toBeTruthy(); | ||||
|   }); | ||||
|  | ||||
| @@ -24,7 +36,19 @@ describe('<Intro />', () => { | ||||
|     const container = rendererCreateWithRedux( | ||||
|       <Intro {...loggedInProps} /> | ||||
|     ).root; | ||||
|  | ||||
|     /** | ||||
|      * This rules had to be disabled because the new lint rules are throwing false positives here. | ||||
|      * They were interpreting react-test-renderer functions as @testing-library/react functions. | ||||
|      */ | ||||
|     // eslint-disable-next-line testing-library/await-async-query | ||||
|     expect(container.findAllByType('blockquote').length === 1).toBeTruthy(); | ||||
|  | ||||
|     /** | ||||
|      * This rules had to be disabled because the new lint rules are throwing false positives here. | ||||
|      * They were interpreting react-test-renderer functions as @testing-library/react functions. | ||||
|      */ | ||||
|     // eslint-disable-next-line testing-library/await-async-query | ||||
|     expect(container.findAllByType('h1').length === 1).toBeTruthy(); | ||||
|   }); | ||||
| }); | ||||
|   | ||||
| @@ -42,6 +42,7 @@ describe('AppMountNotifier', () => { | ||||
|       setup(langCode); | ||||
|  | ||||
|       await waitFor(() => { | ||||
|         /* eslint-disable-next-line testing-library/no-node-access */ | ||||
|         expect(document.querySelector('html')).toHaveAttribute( | ||||
|           'lang', | ||||
|           langCode | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import React from 'react'; | ||||
| import { render, fireEvent } from '@testing-library/react'; | ||||
| import { render, fireEvent, screen } from '@testing-library/react'; | ||||
|  | ||||
| import Form from './Form'; | ||||
|  | ||||
| @@ -21,17 +21,17 @@ const defaultTestProps = { | ||||
| }; | ||||
|  | ||||
| test('should render', () => { | ||||
|   const { getByLabelText, getByText } = render(<Form {...defaultTestProps} />); | ||||
|   render(<Form {...defaultTestProps} />); | ||||
|  | ||||
|   const nameInput = getByLabelText(/name Label/); | ||||
|   const nameInput = screen.getByLabelText(/name Label/); | ||||
|   expect(nameInput).not.toBeRequired(); | ||||
|   expect(nameInput).toHaveAttribute('type', 'text'); | ||||
|  | ||||
|   const websiteInput = getByLabelText(/WebSite label/); | ||||
|   const websiteInput = screen.getByLabelText(/WebSite label/); | ||||
|   expect(websiteInput).toBeRequired(); | ||||
|   expect(websiteInput).toHaveAttribute('type', 'url'); | ||||
|  | ||||
|   const button = getByText(/submit/i); | ||||
|   const button = screen.getByText(/submit/i); | ||||
|   expect(button).toHaveAttribute('type', 'submit'); | ||||
|   expect(button).toBeDisabled(); | ||||
| }); | ||||
| @@ -40,7 +40,7 @@ test('should render with default values', () => { | ||||
|   const websiteValue = 'http://mysite.com'; | ||||
|   const nameValue = 'John'; | ||||
|  | ||||
|   const { getByLabelText, getByText } = render( | ||||
|   render( | ||||
|     <Form | ||||
|       {...defaultTestProps} | ||||
|       enableSubmit={true} | ||||
| @@ -48,13 +48,13 @@ test('should render with default values', () => { | ||||
|     /> | ||||
|   ); | ||||
|  | ||||
|   const nameInput = getByLabelText(/name Label/); | ||||
|   const nameInput = screen.getByLabelText(/name Label/); | ||||
|   expect(nameInput).toHaveValue(nameValue); | ||||
|  | ||||
|   const websiteInput = getByLabelText(/WebSite label/); | ||||
|   const websiteInput = screen.getByLabelText(/WebSite label/); | ||||
|   expect(websiteInput).toHaveValue(websiteValue); | ||||
|  | ||||
|   const button = getByText(/submit/i); | ||||
|   const button = screen.getByText(/submit/i); | ||||
|   expect(button).toBeEnabled(); | ||||
| }); | ||||
|  | ||||
| @@ -66,13 +66,13 @@ test('should submit', () => { | ||||
|   }; | ||||
|   const websiteValue = 'http://mysite.com'; | ||||
|  | ||||
|   const { getByLabelText, getByText } = render(<Form {...props} />); | ||||
|   render(<Form {...props} />); | ||||
|  | ||||
|   const websiteInput = getByLabelText(/WebSite label/); | ||||
|   const websiteInput = screen.getByLabelText(/WebSite label/); | ||||
|   fireEvent.change(websiteInput, { target: { value: websiteValue } }); | ||||
|   expect(websiteInput).toHaveValue(websiteValue); | ||||
|  | ||||
|   const button = getByText(/submit/i); | ||||
|   const button = screen.getByText(/submit/i); | ||||
|   expect(button).toBeEnabled(); | ||||
|  | ||||
|   fireEvent.click(button); | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import React from 'react'; | ||||
| import { render } from '@testing-library/react'; | ||||
| import { render, screen } from '@testing-library/react'; | ||||
|  | ||||
| import BlockSaveButton from './block-save-button'; | ||||
|  | ||||
| @@ -10,14 +10,14 @@ test('<BlockSaveButton /> snapshot', () => { | ||||
| }); | ||||
|  | ||||
| test('Button text should default to the correct translation key', () => { | ||||
|   const { getByRole } = render(<BlockSaveButton />); | ||||
|   render(<BlockSaveButton />); | ||||
|  | ||||
|   expect(getByRole('button')).toHaveTextContent('buttons.save'); | ||||
|   expect(screen.getByRole('button')).toHaveTextContent('buttons.save'); | ||||
| }); | ||||
|  | ||||
| test('Button text should match "children"', () => { | ||||
|   const testText = 'My Text Here'; | ||||
|   const { getByRole } = render(<BlockSaveButton>{testText}</BlockSaveButton>); | ||||
|   render(<BlockSaveButton>{testText}</BlockSaveButton>); | ||||
|  | ||||
|   expect(getByRole('button')).toHaveTextContent(testText); | ||||
|   expect(screen.getByRole('button')).toHaveTextContent(testText); | ||||
| }); | ||||
|   | ||||
| @@ -3,6 +3,7 @@ | ||||
| exports[`<Loader /> matches the fullScreen render snapshot 1`] = ` | ||||
| <div | ||||
|   class="fcc-loader full-screen-wrapper" | ||||
|   data-testid="fcc-loader" | ||||
| > | ||||
|   <div | ||||
|     class="sk-fade-in sk-spinner line-scale-pulse-out" | ||||
| @@ -19,6 +20,7 @@ exports[`<Loader /> matches the fullScreen render snapshot 1`] = ` | ||||
| exports[`<Loader /> matches to the default render snapshot 1`] = ` | ||||
| <div | ||||
|   class="fcc-loader " | ||||
|   data-testid="fcc-loader" | ||||
| > | ||||
|   <div | ||||
|     class="sk-fade-in sk-spinner line-scale-pulse-out" | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import React from 'react'; | ||||
| import { render, cleanup } from '@testing-library/react'; | ||||
| import { render, cleanup, screen } from '@testing-library/react'; | ||||
|  | ||||
| import Loader from './loader'; | ||||
|  | ||||
| @@ -12,8 +12,9 @@ describe('<Loader />', () => { | ||||
|   }); | ||||
|  | ||||
|   it('adds the correct class when given a fullScreen prop', () => { | ||||
|     const { container } = render(<Loader fullScreen={true} />); | ||||
|     expect(container.firstChild).toHaveClass('full-screen-wrapper'); | ||||
|     render(<Loader fullScreen={true} />); | ||||
|     const fccLoader = screen.getByTestId('fcc-loader'); | ||||
|     expect(fccLoader).toHaveClass('full-screen-wrapper'); | ||||
|   }); | ||||
|  | ||||
|   /** | ||||
| @@ -22,12 +23,14 @@ describe('<Loader />', () => { | ||||
|    */ | ||||
|  | ||||
|   it('matches to the default render snapshot', () => { | ||||
|     const { container } = render(<Loader />); | ||||
|     expect(container.firstChild).toMatchSnapshot(); | ||||
|     render(<Loader />); | ||||
|     const fccLoader = screen.getByTestId('fcc-loader'); | ||||
|     expect(fccLoader).toMatchSnapshot(); | ||||
|   }); | ||||
|  | ||||
|   it('matches the fullScreen render snapshot', () => { | ||||
|     const { container } = render(<Loader fullScreen={true} />); | ||||
|     expect(container.firstChild).toMatchSnapshot(); | ||||
|     render(<Loader fullScreen={true} />); | ||||
|     const fccLoader = screen.getByTestId('fcc-loader'); | ||||
|     expect(fccLoader).toMatchSnapshot(); | ||||
|   }); | ||||
| }); | ||||
|   | ||||
| @@ -17,7 +17,10 @@ function Loader({ fullScreen, timeout }: LoaderProps): JSX.Element { | ||||
|     return () => clearTimeout(timerId); | ||||
|   }, [setShowSpinner, showSpinner, timeout]); | ||||
|   return ( | ||||
|     <div className={`fcc-loader ${fullScreen ? 'full-screen-wrapper' : ''}`}> | ||||
|     <div | ||||
|       className={`fcc-loader ${fullScreen ? 'full-screen-wrapper' : ''}`} | ||||
|       data-testid='fcc-loader' | ||||
|     > | ||||
|       {showSpinner && <Spinner name='line-scale-pulse-out' />} | ||||
|     </div> | ||||
|   ); | ||||
|   | ||||
| @@ -10,8 +10,8 @@ describe('<Landing />', () => { | ||||
|   it('renders when visiting index page and logged out', () => { | ||||
|     const shallow = new ShallowRenderer(); | ||||
|     shallow.render(<IndexPage {...loggedOutProps} />); | ||||
|     const result = shallow.getRenderOutput(); | ||||
|     expect(result.type.displayName === 'Landing').toBeTruthy(); | ||||
|     const view = shallow.getRenderOutput(); | ||||
|     expect(view.type.displayName === 'Landing').toBeTruthy(); | ||||
|   }); | ||||
| }); | ||||
|  | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import React from 'react'; | ||||
| import { render } from '@testing-library/react'; | ||||
| import { render, screen } from '@testing-library/react'; | ||||
|  | ||||
| import Profile from './Profile'; | ||||
|  | ||||
| @@ -55,9 +55,9 @@ const notMyProfileProps = { | ||||
|  | ||||
| describe('<Profile/>', () => { | ||||
|   it('renders the report button on another persons profile', () => { | ||||
|     const { getByText } = render(<Profile {...notMyProfileProps} />); | ||||
|     render(<Profile {...notMyProfileProps} />); | ||||
|  | ||||
|     const reportButton: HTMLElement = getByText('buttons.flag-user'); | ||||
|     const reportButton: HTMLElement = screen.getByText('buttons.flag-user'); | ||||
|     expect(reportButton).toHaveAttribute('href', '/user/string/report-user'); | ||||
|   }); | ||||
|  | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import React from 'react'; | ||||
| import { render } from '@testing-library/react'; | ||||
| import { render, screen } from '@testing-library/react'; | ||||
|  | ||||
| import HeatMap from './HeatMap'; | ||||
|  | ||||
| @@ -41,15 +41,15 @@ describe('<HeatMap/>', () => { | ||||
|   */ | ||||
|  | ||||
|   it('calculates the correct longest streak', () => { | ||||
|     const { getByTestId } = render(<HeatMap {...props} />); | ||||
|     expect(getByTestId('longest-streak').textContent).toContain( | ||||
|     render(<HeatMap {...props} />); | ||||
|     expect(screen.getByTestId('longest-streak')).toHaveTextContent( | ||||
|       'profile.longest-streak' | ||||
|     ); | ||||
|   }); | ||||
|  | ||||
|   it('calculates the correct current streak', () => { | ||||
|     const { getByTestId } = render(<HeatMap {...props} />); | ||||
|     expect(getByTestId('current-streak').textContent).toContain( | ||||
|     render(<HeatMap {...props} />); | ||||
|     expect(screen.getByTestId('current-streak')).toHaveTextContent( | ||||
|       'profile.current-streak' | ||||
|     ); | ||||
|   }); | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| /* eslint-disable @typescript-eslint/ban-ts-comment */ | ||||
| /* eslint-disable @typescript-eslint/no-unsafe-call */ | ||||
| import React from 'react'; | ||||
| import { render } from '@testing-library/react'; | ||||
| import { render, screen } from '@testing-library/react'; | ||||
| import TimeLine from './TimeLine'; | ||||
| import { useStaticQuery } from 'gatsby'; | ||||
|  | ||||
| @@ -45,29 +45,25 @@ beforeEach(() => { | ||||
| describe('<TimeLine />', () => { | ||||
|   it('Render button when only solution is present', () => { | ||||
|     // @ts-ignore | ||||
|     const { container } = render(<TimeLine {...propsForOnlySolution} />); | ||||
|  | ||||
|     expect( | ||||
|       container.querySelector('#btn-for-5e46f802ac417301a38fb92b') | ||||
|     ).toHaveAttribute('href', 'https://github.com/freeCodeCamp/freeCodeCamp'); | ||||
|     render(<TimeLine {...propsForOnlySolution} />); | ||||
|     const showViewButton = screen.getByRole('link', { name: 'buttons.view' }); | ||||
|     expect(showViewButton).toHaveAttribute( | ||||
|       'href', | ||||
|       'https://github.com/freeCodeCamp/freeCodeCamp' | ||||
|     ); | ||||
|   }); | ||||
|  | ||||
|   it('Render button when both githubLink and solution is present', () => { | ||||
|     // @ts-ignore | ||||
|     const { container } = render(<TimeLine {...propsForOnlySolution} />); | ||||
|     render(<TimeLine {...propsForOnlySolution} />); | ||||
|  | ||||
|     const linkList = container.querySelector( | ||||
|       '#dropdown-for-5e4f5c4b570f7e3a4949899f + ul' | ||||
|     ); | ||||
|     // @ts-ignore | ||||
|     const links = linkList.querySelectorAll('a'); | ||||
|  | ||||
|     expect(links[0]).toHaveAttribute( | ||||
|     const menuItems = screen.getAllByRole('menuitem'); | ||||
|     expect(menuItems).toHaveLength(2); | ||||
|     expect(menuItems[0]).toHaveAttribute( | ||||
|       'href', | ||||
|       'https://github.com/freeCodeCamp/freeCodeCamp1' | ||||
|     ); | ||||
|  | ||||
|     expect(links[1]).toHaveAttribute( | ||||
|     expect(menuItems[1]).toHaveAttribute( | ||||
|       'href', | ||||
|       'https://github.com/freeCodeCamp/freeCodeCamp2' | ||||
|     ); | ||||
| @@ -75,9 +71,9 @@ describe('<TimeLine />', () => { | ||||
|  | ||||
|   it('rendering the correct button when files is present', () => { | ||||
|     // @ts-ignore | ||||
|     const { getByText } = render(<TimeLine {...propsForOnlySolution} />); | ||||
|     render(<TimeLine {...propsForOnlySolution} />); | ||||
|  | ||||
|     const button = getByText('buttons.show-code'); | ||||
|     const button = screen.getByText('buttons.show-code'); | ||||
|     expect(button).toBeInTheDocument(); | ||||
|   }); | ||||
| }); | ||||
|   | ||||
| @@ -5,10 +5,10 @@ import { SearchBar } from './search-bar'; | ||||
|  | ||||
| describe('<SearchBar />', () => { | ||||
|   it('renders to the DOM', () => { | ||||
|     const shallow = ShallowRenderer.createRenderer(); | ||||
|     shallow.render(<SearchBar {...searchBarProps} />); | ||||
|     const result = shallow.getRenderOutput(); | ||||
|     expect(result).toBeTruthy(); | ||||
|     const utils = ShallowRenderer.createRenderer(); | ||||
|     utils.render(<SearchBar {...searchBarProps} />); | ||||
|     const view = utils.getRenderOutput(); | ||||
|     expect(view).toBeTruthy(); | ||||
|   }); | ||||
|  | ||||
|   /* Todo: When e2e testing is in place, | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import React from 'react'; | ||||
| import { render } from '@testing-library/react'; | ||||
| import { render, screen } from '@testing-library/react'; | ||||
| import { Provider } from 'react-redux'; | ||||
| import { createStore } from '../../redux/createStore'; | ||||
|  | ||||
| @@ -15,75 +15,64 @@ describe('<certification />', () => { | ||||
|   // shallow rendering does not render children component | ||||
|   // form buttons are not included in shallow render | ||||
|   it('Should render show cert button for claimed legacy cert', () => { | ||||
|     const { container } = renderWithRedux( | ||||
|       <CertificationSettings {...defaultTestProps} /> | ||||
|     ); | ||||
|     renderWithRedux(<CertificationSettings {...defaultTestProps} />); | ||||
|  | ||||
|     expect( | ||||
|       container.querySelector( | ||||
|         'a[href="/certification/developmentuser/legacy-data-visualization"]' | ||||
|       ) | ||||
|     ).toHaveTextContent('buttons.show-cert'); | ||||
|   }); | ||||
|  | ||||
|   it('Should link show cert button to the claimed legacy cert', () => { | ||||
|     const { container } = renderWithRedux( | ||||
|       <CertificationSettings {...defaultTestProps} /> | ||||
|       screen.getByRole('link', { | ||||
|         name: 'buttons.show-cert' | ||||
|       }) | ||||
|     ).toHaveAttribute( | ||||
|       'href', | ||||
|       '/certification/developmentuser/legacy-data-visualization' | ||||
|     ); | ||||
|  | ||||
|     expect( | ||||
|       container.querySelector( | ||||
|         'a[href="/certification/developmentuser/legacy-data-visualization"]' | ||||
|       ) | ||||
|     ).toBeInTheDocument(); | ||||
|   }); | ||||
|  | ||||
|   // full forms with unclaimed certs should not shallow render show cert button | ||||
|   it('Should not render show cert button for unclaimed cert with completed projects', () => { | ||||
|     const { container } = renderWithRedux( | ||||
|       <CertificationSettings {...defaultTestProps} /> | ||||
|     ); | ||||
|     renderWithRedux(<CertificationSettings {...defaultTestProps} />); | ||||
|  | ||||
|     expect( | ||||
|       container.querySelector( | ||||
|         'a[href="/certification/developmentuser/legacy-back-end"]' | ||||
|       ) | ||||
|     ).not.toHaveTextContent('buttons.show-cert'); | ||||
|     const allClaimedCerts = screen.getAllByRole('link', { | ||||
|       name: 'buttons.show-cert' | ||||
|     }); | ||||
|  | ||||
|     allClaimedCerts.forEach(cert => { | ||||
|       expect(cert).not.toHaveAttribute( | ||||
|         'href', | ||||
|         '/certification/developmentuser/legacy-back-end' | ||||
|       ); | ||||
|     }); | ||||
|   }); | ||||
|  | ||||
|   // empty forms with unclaimed certs should not shallow render show cert button | ||||
|   it('Should not render show cert button for cert with no completed projects', () => { | ||||
|     const { container } = renderWithRedux( | ||||
|       <CertificationSettings {...defaultTestProps} /> | ||||
|     ); | ||||
|     renderWithRedux(<CertificationSettings {...defaultTestProps} />); | ||||
|  | ||||
|     expect( | ||||
|       container.querySelector( | ||||
|         'a[href="/certification/developmentuser/legacy-front-end"]' | ||||
|       ) | ||||
|     ).not.toHaveTextContent('buttons.show-cert'); | ||||
|     const allClaimedCerts = screen.getAllByRole('link', { | ||||
|       name: 'buttons.show-cert' | ||||
|     }); | ||||
|  | ||||
|     allClaimedCerts.forEach(cert => { | ||||
|       expect(cert).not.toHaveAttribute( | ||||
|         'href', | ||||
|         '/certification/developmentuser/legacy-front-end' | ||||
|       ); | ||||
|     }); | ||||
|   }); | ||||
|  | ||||
|   it('Render button when only solution is present', () => { | ||||
|     const { container } = renderWithRedux( | ||||
|       <CertificationSettings {...propsForOnlySolution} /> | ||||
|     ); | ||||
|     renderWithRedux(<CertificationSettings {...propsForOnlySolution} />); | ||||
|  | ||||
|     expect( | ||||
|       container.querySelector('#btn-for-5e46f802ac417301a38fb92b') | ||||
|       screen.getByRole('link', { | ||||
|         name: 'buttons.show-solution' | ||||
|       }) | ||||
|     ).toHaveAttribute('href', 'https://github.com/freeCodeCamp/freeCodeCamp'); | ||||
|   }); | ||||
|  | ||||
|   it('Render button when both githubLink and solution is present', () => { | ||||
|     const { container } = renderWithRedux( | ||||
|       <CertificationSettings {...propsForOnlySolution} /> | ||||
|     ); | ||||
|  | ||||
|     const linkList = container.querySelector( | ||||
|       '#dropdown-for-5e4f5c4b570f7e3a4949899f + ul' | ||||
|     ); | ||||
|     const links = linkList.querySelectorAll('a'); | ||||
|     renderWithRedux(<CertificationSettings {...propsForOnlySolution} />); | ||||
|  | ||||
|     const links = screen.getAllByRole('menuitem'); | ||||
|     expect(links[0]).toHaveAttribute( | ||||
|       'href', | ||||
|       'https://github.com/freeCodeCamp/freeCodeCamp1' | ||||
| @@ -96,11 +85,9 @@ describe('<certification />', () => { | ||||
|   }); | ||||
|  | ||||
|   it('rendering the correct button when files is present', () => { | ||||
|     const { getByText } = renderWithRedux( | ||||
|       <CertificationSettings {...propsForOnlySolution} /> | ||||
|     ); | ||||
|     renderWithRedux(<CertificationSettings {...propsForOnlySolution} />); | ||||
|  | ||||
|     const button = getByText('buttons.show-code'); | ||||
|     const button = screen.getByText('buttons.show-code'); | ||||
|     expect(button).toBeInTheDocument(); | ||||
|   }); | ||||
| }); | ||||
|   | ||||
| @@ -13,16 +13,16 @@ describe('<Honesty />', () => { | ||||
|     const componentToRender = ( | ||||
|       <Honesty isHonest={false} updateIsHonest={updateIsHonestMock} /> | ||||
|     ); | ||||
|     const component = renderer.render(componentToRender); | ||||
|     expect(component).toMatchSnapshot('Honesty'); | ||||
|     const view = renderer.render(componentToRender); | ||||
|     expect(view).toMatchSnapshot('Honesty'); | ||||
|   }); | ||||
|  | ||||
|   test('<Honesty /> snapshot when isHonest is true', () => { | ||||
|     const componentToRender = ( | ||||
|       <Honesty isHonest={true} updateIsHonest={updateIsHonestMock} /> | ||||
|     ); | ||||
|     const component = renderer.render(componentToRender); | ||||
|     expect(component).toMatchSnapshot('HonestyAccepted'); | ||||
|     const view = renderer.render(componentToRender); | ||||
|     expect(view).toMatchSnapshot('HonestyAccepted'); | ||||
|   }); | ||||
|  | ||||
|   test('should call updateIsHonest method on clicking agree button', () => { | ||||
| @@ -30,6 +30,11 @@ describe('<Honesty />', () => { | ||||
|       <Honesty isHonest={false} updateIsHonest={updateIsHonestMock} /> | ||||
|     ).root; | ||||
|  | ||||
|     /** | ||||
|      * This rules had to be disabled because the new lint rules are throwing false positives here. | ||||
|      * They were interpreting react-test-renderer functions as @testing-library/react functions. | ||||
|      */ | ||||
|     // eslint-disable-next-line testing-library/await-async-query | ||||
|     root.findByType(Button).props.onClick(); | ||||
|     expect(updateIsHonestMock).toHaveBeenCalledWith({ isHonest: true }); | ||||
|   }); | ||||
|   | ||||
| @@ -12,6 +12,7 @@ exports[`<CompletionModalBody /> matches snapshot 1`] = ` | ||||
|     </span> | ||||
|     <svg | ||||
|       class="completion-success-icon" | ||||
|       data-testid="fcc-completion-success-icon" | ||||
|       height="50" | ||||
|       viewBox="0 0 200 200" | ||||
|       width="50" | ||||
| @@ -70,6 +71,7 @@ exports[`<CompletionModalBody /> matches snapshot 1`] = ` | ||||
|       </div> | ||||
|       <div | ||||
|         class="progress-bar-percent" | ||||
|         data-testid="fcc-progress-bar-percent" | ||||
|         style="width: 0%;" | ||||
|       > | ||||
|         <div | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import React from 'react'; | ||||
| import { render, fireEvent } from '@testing-library/react'; | ||||
| import { render, fireEvent, screen } from '@testing-library/react'; | ||||
|  | ||||
| import CompletionModalBody from './completion-modal-body'; | ||||
|  | ||||
| @@ -22,26 +22,22 @@ describe('<CompletionModalBody />', () => { | ||||
|     }); | ||||
|  | ||||
|     test('renders with 0% width initially', () => { | ||||
|       const { container } = render(<CompletionModalBody {...props} />); | ||||
|       expect(container.querySelector('.progress-bar-percent')).toHaveAttribute( | ||||
|         'style', | ||||
|         'width: 0%;' | ||||
|       ); | ||||
|       render(<CompletionModalBody {...props} />); | ||||
|       expect(screen.getByTestId('fcc-progress-bar-percent')).toHaveStyle({ | ||||
|         width: '0%' | ||||
|       }); | ||||
|     }); | ||||
|  | ||||
|     test('has the correct width after animation', () => { | ||||
|       const { container } = render(<CompletionModalBody {...props} />); | ||||
|       render(<CompletionModalBody {...props} />); | ||||
|  | ||||
|       fireEvent.animationEnd( | ||||
|         container.querySelector('.completion-success-icon') as HTMLElement | ||||
|       ); | ||||
|       fireEvent.animationEnd(screen.getByTestId('fcc-completion-success-icon')); | ||||
|  | ||||
|       jest.runAllTimers(); | ||||
|  | ||||
|       expect(container.querySelector('.progress-bar-percent')).toHaveAttribute( | ||||
|         'style', | ||||
|         `width: ${props.completedPercent}%;` | ||||
|       ); | ||||
|       expect(screen.getByTestId('fcc-progress-bar-percent')).toHaveStyle({ | ||||
|         width: `${props.completedPercent}%` | ||||
|       }); | ||||
|     }); | ||||
|   }); | ||||
| }); | ||||
|   | ||||
| @@ -77,6 +77,7 @@ export class CompletionModalBody extends PureComponent< | ||||
|         <div className='completion-challenge-details'> | ||||
|           <GreenPass | ||||
|             className='completion-success-icon' | ||||
|             data-testid='fcc-completion-success-icon' | ||||
|             onAnimationEnd={() => { | ||||
|               setTimeout(() => { | ||||
|                 this.animateProgressBar(completedPercent); | ||||
| @@ -94,6 +95,7 @@ export class CompletionModalBody extends PureComponent< | ||||
|             </div> | ||||
|             <div | ||||
|               className='progress-bar-percent' | ||||
|               data-testid='fcc-progress-bar-percent' | ||||
|               style={{ width: `${this.state.shownPercent}%` }} | ||||
|             > | ||||
|               <div className='progress-bar-foreground'> | ||||
|   | ||||
| @@ -22,10 +22,10 @@ function getComponentNameAndProps(elementType, pathname) { | ||||
|     } | ||||
|   }); | ||||
|   shallow.render(<Provider store={store}>{LayoutReactComponent}</Provider>); | ||||
|   const renderedComponent = shallow.getRenderOutput(); | ||||
|   const view = shallow.getRenderOutput(); | ||||
|   return { | ||||
|     props: renderedComponent.props, | ||||
|     name: renderedComponent.type.WrappedComponent.displayName | ||||
|     props: view.props, | ||||
|     name: view.type.WrappedComponent.displayName | ||||
|   }; | ||||
| } | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user