92a3337aeb
CI Pipeline / test (20) (pull_request) Successful in 2m41s
CI Pipeline / test (18) (pull_request) Successful in 3m21s
CI Pipeline / e2e (chromium) (pull_request) Failing after 1m25s
CI Pipeline / e2e (firefox) (pull_request) Failing after 1m24s
CI Pipeline / e2e (webkit) (pull_request) Failing after 1m24s
CI Pipeline / visual-regression (pull_request) Failing after 1m53s
CI Pipeline / performance (pull_request) Failing after 1m31s
CI Pipeline / lint (pull_request) Failing after 1m5s
CI Pipeline / storybook (pull_request) Successful in 1m36s
CI Pipeline / build (pull_request) Failing after 1m19s
84 lines
1.8 KiB
TypeScript
84 lines
1.8 KiB
TypeScript
"use client";
|
|
|
|
import React, { memo, useCallback, useId } from "react";
|
|
import RadioButton from "./RadioButton";
|
|
|
|
interface RadioOption {
|
|
value: string;
|
|
label: string;
|
|
ariaLabel?: string;
|
|
}
|
|
|
|
interface RadioGroupProps {
|
|
name?: string;
|
|
value?: string;
|
|
onChange?: (data: { value: string }) => void;
|
|
mode?: "standard" | "inverse";
|
|
state?: "default" | "hover" | "focus";
|
|
disabled?: boolean;
|
|
options?: RadioOption[];
|
|
className?: string;
|
|
"aria-label"?: string;
|
|
}
|
|
|
|
const RadioGroup = ({
|
|
name,
|
|
value,
|
|
onChange,
|
|
mode = "standard",
|
|
state = "default",
|
|
disabled = false,
|
|
options = [],
|
|
className = "",
|
|
...props
|
|
}: RadioGroupProps) => {
|
|
// Generate unique ID for accessibility if not provided
|
|
const generatedId = useId();
|
|
const groupId = name || `radio-group-${generatedId}`;
|
|
|
|
const handleChange = useCallback(
|
|
(optionValue: string) => {
|
|
if (!disabled && onChange) {
|
|
onChange({ value: optionValue });
|
|
}
|
|
},
|
|
[disabled, onChange],
|
|
);
|
|
|
|
return (
|
|
<div
|
|
className={`space-y-[8px] ${className}`}
|
|
role="radiogroup"
|
|
aria-label={props["aria-label"]}
|
|
{...props}
|
|
>
|
|
{options.map((option) => {
|
|
const isSelected = value === option.value;
|
|
|
|
return (
|
|
<RadioButton
|
|
key={option.value}
|
|
checked={isSelected}
|
|
mode={mode}
|
|
state={state}
|
|
disabled={disabled}
|
|
label={option.label}
|
|
name={groupId}
|
|
value={option.value}
|
|
ariaLabel={option.ariaLabel}
|
|
onChange={({ checked }) => {
|
|
if (checked) {
|
|
handleChange(option.value);
|
|
}
|
|
}}
|
|
/>
|
|
);
|
|
})}
|
|
</div>
|
|
);
|
|
};
|
|
|
|
RadioGroup.displayName = "RadioGroup";
|
|
|
|
export default memo(RadioGroup);
|