RadioGroup.tsx 1.9 KB
Newer Older
P
Peter Pan 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
/**
 * Copyright 2020 Baidu Inc. All Rights Reserved.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

17
import React, {FunctionComponent, PropsWithChildren, createContext, useCallback, useState} from 'react';
18

19
import type {WithStyled} from '~/utils/style';
20
import styled from 'styled-components';
21 22 23 24 25 26 27 28 29 30 31

const Wrapper = styled.div`
    display: inline-flex;

    > * {
        flex-shrink: 0;
        align-items: flex-start;
    }
`;

// eslint-disable-next-line @typescript-eslint/no-empty-function
32 33
export const EventContext = createContext<(<V extends unknown>(value: V) => unknown) | undefined>(() => {});
export const ValueContext = createContext<unknown>(null);
34

35 36 37
type RadioGroupProps<T> = {
    value?: T;
    onChange?: (value: T) => unknown;
38 39
};

40 41 42 43 44 45
const RadioGroup = <T extends unknown>({
    value,
    onChange,
    children,
    className
}: PropsWithChildren<RadioGroupProps<T>> & WithStyled): ReturnType<FunctionComponent> => {
46 47
    const [selected, setSelected] = useState(value);
    const onSelectedChange = useCallback(
48
        (value: T) => {
49 50 51
            setSelected(value);
            onChange?.(value);
        },
P
Peter Pan 已提交
52
        [onChange]
53 54 55
    );

    return (
56
        <EventContext.Provider value={v => onSelectedChange(v as T)}>
57 58 59 60 61 62 63 64
            <ValueContext.Provider value={selected}>
                <Wrapper className={className}>{children}</Wrapper>
            </ValueContext.Provider>
        </EventContext.Provider>
    );
};

export default RadioGroup;