From 259bc4bd6a82846e8e1aa05050da237e3c64ffd9 Mon Sep 17 00:00:00 2001 From: Ihor Korenets Date: Mon, 1 Apr 2024 17:52:22 +0300 Subject: [PATCH 1/6] [Checkbox]: rework from class component into FC --- uui-components/src/inputs/Checkbox.tsx | 110 ++++++++---------- .../src/inputs/__tests__/Checkbox.test.tsx | 41 +++++-- .../inputs/__tests__/Checkbox.test.tsx | 14 +-- .../__snapshots__/Checkbox.test.tsx.snap | 22 ++-- .../layout/__tests__/CheckboxGroup.test.tsx | 43 ++++--- 5 files changed, 127 insertions(+), 103 deletions(-) diff --git a/uui-components/src/inputs/Checkbox.tsx b/uui-components/src/inputs/Checkbox.tsx index 3498dd947b..d65cac16b9 100644 --- a/uui-components/src/inputs/Checkbox.tsx +++ b/uui-components/src/inputs/Checkbox.tsx @@ -1,10 +1,8 @@ import * as React from 'react'; -import { cx, IHasTabIndex, uuiMarkers } from '@epam/uui-core'; +import { cx, IHasTabIndex, useUuiContext, uuiMarkers } from '@epam/uui-core'; +import { Icon, uuiMod, uuiElement, isEventTargetInsideClickable, CheckboxCoreProps } from '@epam/uui-core'; +import { IconContainer } from '../layout'; import css from './Checkbox.module.scss'; -import { - Icon, uuiMod, uuiElement, isEventTargetInsideClickable, CheckboxCoreProps, UuiContexts, UuiContext, -} from '@epam/uui-core'; -import { IconContainer } from '../layout/IconContainer'; export interface CheckboxProps extends CheckboxCoreProps, IHasTabIndex { /** Render callback for checkbox label. @@ -26,20 +24,19 @@ export interface CheckboxProps extends CheckboxCoreProps, IHasTabIndex { indeterminateIcon?: Icon; } -export class Checkbox extends React.Component { - static contextType = UuiContext; - context: UuiContexts; +export function Checkbox(props: CheckboxProps) { + const context = useUuiContext(); - handleChange = (e: React.SyntheticEvent) => { - !isEventTargetInsideClickable(e) && this.props.onValueChange(!this.props.value); + const handleChange = (e: React.SyntheticEvent) => { + !isEventTargetInsideClickable(e) && props.onValueChange(!props.value); - if (this.props.getValueChangeAnalyticsEvent) { - const event = this.props.getValueChangeAnalyticsEvent(!this.props.value, this.props.value); - this.context.uuiAnalytics.sendEvent(event); + if (props.getValueChangeAnalyticsEvent) { + const event = props.getValueChangeAnalyticsEvent(!props.value, props.value); + context.uuiAnalytics.sendEvent(event); } }; - handleAriaCheckedValue = (indeterminate: boolean, value: boolean): boolean | 'mixed' => { + const handleAriaCheckedValue = (indeterminate: boolean, value: boolean): boolean | 'mixed' => { if (indeterminate) { return 'mixed'; } @@ -47,51 +44,46 @@ export class Checkbox extends React.Component { return value == null ? false : value; }; - render() { - let label = this.props.label; - if (this.props.renderLabel) { - label = this.props.renderLabel(); - } - const ariaCheckedValue = this.handleAriaCheckedValue(this.props.indeterminate, this.props.value); + const label = props.renderLabel ? props.renderLabel() : props.label; + const ariaCheckedValue = handleAriaCheckedValue(props.indeterminate, props.value); - return ( - + ); } diff --git a/uui-components/src/inputs/__tests__/Checkbox.test.tsx b/uui-components/src/inputs/__tests__/Checkbox.test.tsx index f45995b7da..eb7a20190c 100644 --- a/uui-components/src/inputs/__tests__/Checkbox.test.tsx +++ b/uui-components/src/inputs/__tests__/Checkbox.test.tsx @@ -1,6 +1,6 @@ import React from 'react'; import { Checkbox, CheckboxProps } from '../Checkbox'; -import { render, screen, fireEvent, setupComponentForTest, userEvent } from '@epam/uui-test-utils'; +import { screen, fireEvent, setupComponentForTest, userEvent } from '@epam/uui-test-utils'; async function setupCheckbox(params: Partial) { const { mocks, setProps } = await setupComponentForTest( @@ -71,18 +71,29 @@ describe('Checkbox', () => { expect(getValueChangeAnalyticsEvent).toHaveBeenCalled(); }); - it('should not handle change event when readonly', () => { + it('should not handle change event when readonly', async () => { const onValueChange = jest.fn(); - render(); + await setupCheckbox({ + value: false, + onValueChange, + isReadonly: true, + label: 'Label', + }); const input = screen.getByRole('checkbox'); fireEvent.click(input); expect(onValueChange).not.toHaveBeenCalled(); }); - it('should handle focus event', () => { + it('should handle focus event', async () => { const onFocus = jest.fn(); - render(); + const onValueChange = jest.fn(); + await setupCheckbox({ + value: false, + onValueChange, + label: 'Label', + onFocus, + }); const input = screen.getByRole('checkbox'); input.focus(); @@ -91,9 +102,15 @@ describe('Checkbox', () => { expect(input).toHaveFocus(); }); - it('should handle blur event', () => { + it('should handle blur event', async () => { const onBlur = jest.fn(); - render(); + const onValueChange = jest.fn(); + await setupCheckbox({ + value: false, + onValueChange, + label: 'Label', + onBlur, + }); const input = screen.getByRole('checkbox'); input.focus(); @@ -103,4 +120,14 @@ describe('Checkbox', () => { expect(onBlur).toHaveBeenCalled(); expect(input).not.toHaveFocus(); }); + + it('when state equals isInvalid: true, Checkbox_container must have a \'uui-invalid\' class', async () => { + await setupCheckbox({ + value: false, + isInvalid: true, + }); + const input = screen.getByRole('checkbox'); + + expect(input.parentElement.parentElement).toHaveClass('uui-invalid'); + }); }); diff --git a/uui/components/inputs/__tests__/Checkbox.test.tsx b/uui/components/inputs/__tests__/Checkbox.test.tsx index 10a0483bc4..2cf36ce681 100644 --- a/uui/components/inputs/__tests__/Checkbox.test.tsx +++ b/uui/components/inputs/__tests__/Checkbox.test.tsx @@ -1,15 +1,15 @@ import React from 'react'; -import { Checkbox } from '../Checkbox'; -import { renderer } from '@epam/uui-test-utils'; +import { renderSnapshotWithContextAsync } from '@epam/uui-test-utils'; +import { Checkbox } from '@epam/uui'; -describe('Checkbox', () => { - it('should be rendered correctly', () => { - const tree = renderer.create().toJSON(); +describe('TestComponent', () => { + it('should render with minimum props', async () => { + const tree = await renderSnapshotWithContextAsync(); expect(tree).toMatchSnapshot(); }); - it('should be rendered correctly', () => { - const tree = renderer.create().toJSON(); + it('should render with maximum props', async () => { + const tree = await renderSnapshotWithContextAsync(); expect(tree).toMatchSnapshot(); }); }); diff --git a/uui/components/inputs/__tests__/__snapshots__/Checkbox.test.tsx.snap b/uui/components/inputs/__tests__/__snapshots__/Checkbox.test.tsx.snap index 77a3dcd8ab..9124420d43 100644 --- a/uui/components/inputs/__tests__/__snapshots__/Checkbox.test.tsx.snap +++ b/uui/components/inputs/__tests__/__snapshots__/Checkbox.test.tsx.snap @@ -1,8 +1,8 @@ // Jest Snapshot v1, https://goo.gl/fbAQLP -exports[`Checkbox should be rendered correctly 1`] = ` +exports[`TestComponent should render with maximum props 1`] = `