forked from patternfly/patternfly-react
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathCheckbox.tsx
More file actions
143 lines (134 loc) · 4.76 KB
/
Checkbox.tsx
File metadata and controls
143 lines (134 loc) · 4.76 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
import * as React from 'react';
import styles from '@patternfly/react-styles/css/components/Check/check';
import { css } from '@patternfly/react-styles';
import { PickOptional } from '../../helpers/typeUtils';
import { getDefaultOUIAId, getOUIAProps, OUIAProps } from '../../helpers';
import { ASTERISK } from '../../helpers/htmlConstants';
export interface CheckboxProps
extends Omit<React.HTMLProps<HTMLInputElement>, 'type' | 'onChange' | 'disabled' | 'label'>,
OUIAProps {
/** Additional classes added to the checkbox. */
className?: string;
/** Additional classed added to the radio input */
inputClassName?: string;
/** Flag to show if the checkbox selection is valid or invalid. */
isValid?: boolean;
/** Flag to show if the checkbox is disabled. */
isDisabled?: boolean;
/** Flag to show if the checkbox is required. */
isRequired?: boolean;
/** Flag to show if the checkbox is checked. If null, the checkbox will be indeterminate (partially checked). */
isChecked?: boolean | null;
checked?: boolean;
/** A callback for when the checkbox selection changes. */
onChange?: (event: React.FormEvent<HTMLInputElement>, checked: boolean) => void;
/** Label text of the checkbox. */
label?: React.ReactNode;
/** Id of the checkbox. */
id: string;
/** Aria-label of the checkbox. */
'aria-label'?: string;
/** Description text of the checkbox. */
description?: React.ReactNode;
/** Body text of the checkbox */
body?: React.ReactNode;
/** Sets the input wrapper component to render. Defaults to <div> */
component?: React.ElementType;
/** Value to overwrite the randomly generated data-ouia-component-id.*/
ouiaId?: number | string;
/** Set the value of data-ouia-safe. Only set to true when the component is in a static state, i.e. no animations are occurring. At all other times, this value must be false. */
ouiaSafe?: boolean;
}
// tslint:disable-next-line:no-empty
const defaultOnChange = () => {};
interface CheckboxState {
ouiaStateId: string;
}
class Checkbox extends React.Component<CheckboxProps, CheckboxState> {
static displayName = 'Checkbox';
static defaultProps: PickOptional<CheckboxProps> = {
className: '',
isValid: true,
isDisabled: false,
isRequired: false,
isChecked: false,
onChange: defaultOnChange,
ouiaSafe: true,
component: 'div'
};
constructor(props: any) {
super(props);
this.state = {
ouiaStateId: getDefaultOUIAId(Checkbox.displayName)
};
}
private handleChange = (event: React.FormEvent<HTMLInputElement>): void => {
this.props.onChange(event, event.currentTarget.checked);
};
render() {
const {
'aria-label': ariaLabel,
className,
inputClassName,
onChange,
isValid,
isDisabled,
isRequired,
isChecked,
label,
checked,
defaultChecked,
description,
body,
ouiaId,
ouiaSafe,
component: Component,
...props
} = this.props;
if (!props.id) {
// eslint-disable-next-line no-console
console.error('Checkbox:', 'id is required to make input accessible');
}
const checkedProps: { checked?: boolean; defaultChecked?: boolean } = {};
if ([true, false].includes(checked) || isChecked === true) {
checkedProps.checked = checked || isChecked;
}
if (onChange !== defaultOnChange) {
checkedProps.checked = isChecked;
}
if ([false, true].includes(defaultChecked as boolean)) {
checkedProps.defaultChecked = defaultChecked as boolean;
}
checkedProps.checked = checkedProps.checked === null ? false : checkedProps.checked;
return (
<Component className={css(styles.check, !label && styles.modifiers.standalone, className)}>
<input
{...props}
className={css(styles.checkInput, inputClassName)}
type="checkbox"
onChange={this.handleChange}
aria-invalid={!isValid}
aria-label={ariaLabel}
disabled={isDisabled}
required={isRequired}
ref={(elem) => elem && (elem.indeterminate = isChecked === null)}
{...checkedProps}
{...getOUIAProps(Checkbox.displayName, ouiaId !== undefined ? ouiaId : this.state.ouiaStateId, ouiaSafe)}
/>
{label && (
<label className={css(styles.checkLabel, isDisabled && styles.modifiers.disabled)} htmlFor={props.id}>
{label}
{isRequired && (
<span className={css(styles.checkLabelRequired)} aria-hidden="true">
{ASTERISK}
</span>
)}
</label>
)}
{description && <span className={css(styles.checkDescription)}>{description}</span>}
{body && <span className={css(styles.checkBody)}>{body}</span>}
</Component>
);
}
}
export { Checkbox };