diff --git a/src/components/CardBase/CardBase.tsx b/src/components/CardBase/CardBase.tsx index 52d3b0e7e..5af8ae8bc 100644 --- a/src/components/CardBase/CardBase.tsx +++ b/src/components/CardBase/CardBase.tsx @@ -10,7 +10,7 @@ import { WithChildren, } from '../../models'; import {AnalyticsEventsBase, DefaultEventNames} from '../../models/common'; -import {block} from '../../utils'; +import {block, getQaAttrubutes} from '../../utils'; import BackgroundImage from '../BackgroundImage/BackgroundImage'; import RouterLink from '../RouterLink/RouterLink'; @@ -25,6 +25,7 @@ export interface CardBaseProps extends AnalyticsEventsBase, CardBaseParams { target?: HTMLAttributeAnchorTarget; metrikaGoals?: MetrikaGoal; pixelEvents?: ButtonPixel; + qa?: string; } export interface CardHeaderBaseProps { @@ -54,10 +55,12 @@ export const Layout = (props: CardBaseProps) => { url, target, border = 'shadow', + qa, } = props; const handleMetrika = useMetrika(); const handleAnalytics = useAnalytics(DefaultEventNames.CardBase, url); let header, content, footer, image, headerClass, footerClass; + const qaAttributes = getQaAttrubutes(qa, 'header', 'footer', 'body', 'content'); function handleChild(child: ReactElement) { switch (child.type) { @@ -88,13 +91,20 @@ export const Layout = (props: CardBaseProps) => {
{header}
)} -
-
{content}
- {footer &&
{footer}
} +
+
+ {content} +
+ {footer && ( +
+ {footer} +
+ )}
); @@ -116,12 +126,15 @@ export const Layout = (props: CardBaseProps) => { draggable={false} onDragStart={(e) => e.preventDefault()} onClick={onClick} + data-qa={qa} > {cardContent} ) : ( -
{cardContent}
+
+ {cardContent} +
); }; diff --git a/src/components/CardBase/__tests__/CardBase.test.tsx b/src/components/CardBase/__tests__/CardBase.test.tsx new file mode 100644 index 000000000..168bdf7a0 --- /dev/null +++ b/src/components/CardBase/__tests__/CardBase.test.tsx @@ -0,0 +1,245 @@ +import React from 'react'; + +import {render, screen} from '@testing-library/react'; +import userEvent from '@testing-library/user-event'; + +import {TARGETS} from '../../../../test-utils/constants'; +import {testCustomClassName} from '../../../../test-utils/shared/common'; +import {PageConstructorProvider} from '../../../containers/PageConstructor'; +import {AnalyticsContextProps} from '../../../context/analyticsContext'; +import {MetrikaContextProps} from '../../../context/metrikaContext'; +import {CardBorder, PixelEventType} from '../../../models'; +import {getQaAttrubutes} from '../../../utils'; +import CardBase, {CardBaseProps} from '../CardBase'; + +const qaId = 'card-base-component'; +const qaAttributes = getQaAttrubutes(qaId, 'header', 'footer', 'body', 'content'); + +const url = '#'; + +const borders: CardBorder[] = ['shadow', 'line', 'none']; + +describe('CardBase', () => { + test('render CardBase by default', async () => { + render( + + Content + , + ); + const cardBase = screen.getByTestId(qaId); + + expect(cardBase).toBeInTheDocument(); + expect(cardBase).toBeVisible(); + expect(cardBase).not.toBeDisabled(); + + const aTag = screen.queryByRole('link'); + expect(aTag).not.toBeInTheDocument(); + }); + + test('render CardBase by default as a link', async () => { + render( + + Content + , + ); + const cardBase = screen.queryByRole('link'); + + expect(cardBase).toBeInTheDocument(); + expect(cardBase).toBeVisible(); + expect(cardBase).not.toBeDisabled(); + }); + + test.each(new Array(...TARGETS))( + 'render with given "%s" target', + (target) => { + render( + + Content + , + ); + const cardBase = screen.queryByRole('link'); + + expect(cardBase).toHaveAttribute('target', target); + }, + ); + + test('add className', () => { + const children = text; + testCustomClassName({ + component: CardBase, + props: {children, qa: qaId}, + }); + }); + + test('render CardBase with header', async () => { + render( + + Header + , + ); + const cardBaseHeader = screen.queryByTestId(qaAttributes.header); + + expect(cardBaseHeader).toBeInTheDocument(); + expect(cardBaseHeader).toBeVisible(); + expect(cardBaseHeader).not.toBeDisabled(); + }); + + test('render CardBase with content', async () => { + render( + + Content + , + ); + const cardBaseContent = screen.queryByTestId(qaAttributes.content); + + expect(cardBaseContent).toBeInTheDocument(); + expect(cardBaseContent).toBeVisible(); + expect(cardBaseContent).not.toBeDisabled(); + }); + + test('render CardBase with footer', async () => { + render( + + Footer + , + ); + const cardBaseFooter = screen.queryByTestId(qaAttributes.footer); + + expect(cardBaseFooter).toBeInTheDocument(); + expect(cardBaseFooter).toBeVisible(); + expect(cardBaseFooter).not.toBeDisabled(); + }); + + test('add bodyClassName', async () => { + const bodyClassName = 'body-class-name'; + + render( + + Content + , + ); + const cardBaseBody = screen.queryByTestId(qaAttributes.body); + + expect(cardBaseBody).toHaveClass(bodyClassName); + }); + + test('add contentClassName', async () => { + const contentClassName = 'content-class-name'; + + render( + + Content + , + ); + const cardBaseContent = screen.queryByTestId(qaAttributes.content); + + expect(cardBaseContent).toHaveClass(contentClassName); + }); + + test('add className to Header', async () => { + const className = 'body-class-name'; + + render( + + Header + , + ); + + const cardBaseHeader = screen.queryByTestId(qaAttributes.header); + + expect(cardBaseHeader).toHaveClass(className); + }); + + test('add className to Footer', async () => { + const className = 'footer-class-name'; + + render( + + Footer + , + ); + const cardBaseFooter = screen.getByTestId(qaAttributes.footer); + + expect(cardBaseFooter).toHaveClass(className); + }); + + test.each(new Array(...borders))('render with given "%s" border', (border) => { + render( + + Content + , + ); + const button = screen.getByTestId(qaId); + + expect(button).toHaveClass(`pc-card-base-block_border_${border}`); + }); + + test('add metrikaEvent', async () => { + const metrikaContext: MetrikaContextProps = { + metrika: { + reachGoal: jest.fn(), + reachGoals: jest.fn(), + }, + }; + const user = userEvent.setup(); + + render( + + + Content + + , + ); + const cardBase = screen.getByTestId(qaId); + + await user.click(cardBase); + expect(metrikaContext.metrika?.reachGoals).toHaveBeenCalledTimes(1); + }); + + test('add pixelEvent', async () => { + const metrikaContext: MetrikaContextProps = { + pixel: { + trackStandard: jest.fn(), + trackCustom: jest.fn(), + track: jest.fn(), + }, + }; + const user = userEvent.setup(); + + render( + + + Content + + , + ); + const cardBase = screen.getByTestId(qaId); + + await user.click(cardBase); + expect(metrikaContext.pixel?.track).toHaveBeenCalledTimes(1); + }); + + test('add analyticsEvent', async () => { + const analyticsContext: AnalyticsContextProps = { + sendEvents: jest.fn(), + }; + const user = userEvent.setup(); + + render( + + + Content + + , + ); + const cardBase = screen.getByTestId(qaId); + + await user.click(cardBase); + expect(analyticsContext.sendEvents).toHaveBeenCalledTimes(1); + }); +}); diff --git a/test-utils/constants.ts b/test-utils/constants.ts index f8d371ea6..c1bfc8fc5 100644 --- a/test-utils/constants.ts +++ b/test-utils/constants.ts @@ -1 +1,3 @@ +export const TARGETS: React.HTMLAttributeAnchorTarget[] = ['_blank', '_self', '_parent', '_top']; + export const ERROR_INPUT_DATA_MESSAGE = 'There are errors in input test data';