signal-desktop/ts/components/CircleCheckbox.tsx

78 lines
1.8 KiB
TypeScript
Raw Normal View History

// Copyright 2021 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
2023-03-09 21:46:01 +00:00
import React, { useCallback } from 'react';
2023-02-23 21:32:19 +00:00
import classNames from 'classnames';
import { getClassNamesFor } from '../util/getClassNamesFor';
2023-02-23 21:32:19 +00:00
import { missingCaseError } from '../util/missingCaseError';
export enum Variant {
Normal = 'Normal',
Small = 'Small',
}
export type Props = {
id?: string;
2023-02-23 21:32:19 +00:00
variant?: Variant;
checked?: boolean;
disabled?: boolean;
isRadio?: boolean;
name?: string;
2023-02-23 21:32:19 +00:00
moduleClassName?: string;
onChange?: (value: boolean) => unknown;
onClick?: () => unknown;
};
/**
* A fancy checkbox
*
* It's only the checkbox, it does NOT produce a label.
* It is a controlled component, you must provide a value and
* update it yourself onClick/onChange.
*/
export function CircleCheckbox({
id,
2023-02-23 21:32:19 +00:00
variant = Variant.Normal,
checked,
disabled,
isRadio,
2023-02-23 21:32:19 +00:00
moduleClassName,
name,
onChange,
onClick,
}: Props): JSX.Element {
2023-02-23 21:32:19 +00:00
const getClassName = getClassNamesFor('CircleCheckbox', moduleClassName);
let variantModifier: string;
if (variant === Variant.Normal) {
variantModifier = getClassName('__checkbox--normal');
} else if (variant === Variant.Small) {
variantModifier = getClassName('__checkbox--small');
} else {
throw missingCaseError(variant);
}
2023-03-09 21:46:01 +00:00
const onChangeWrapped = useCallback(
(ev: React.ChangeEvent<HTMLInputElement>) => {
onChange?.(ev.target.checked);
},
[onChange]
);
return (
2023-02-23 21:32:19 +00:00
<div className={classNames(getClassName('__checkbox'), variantModifier)}>
<input
checked={Boolean(checked)}
disabled={disabled}
aria-disabled={disabled}
id={id}
name={name}
2023-03-09 21:46:01 +00:00
onChange={onChangeWrapped}
onClick={onClick}
type={isRadio ? 'radio' : 'checkbox'}
/>
</div>
);
}