You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
162 lines
4.0 KiB
TypeScript
162 lines
4.0 KiB
TypeScript
import React, { ChangeEvent } from 'react';
|
|
import styled from 'styled-components';
|
|
import { black } from '../../state/ducks/SessionTheme';
|
|
import { Flex } from '../basic/Flex';
|
|
// tslint:disable: react-unused-props-and-state
|
|
|
|
type Props = {
|
|
label: string;
|
|
value: string;
|
|
active: boolean;
|
|
inputName?: string;
|
|
beforeMargins?: string;
|
|
onClick?: (value: string) => void;
|
|
};
|
|
|
|
const StyledInput = styled.input<{
|
|
filledSize: number;
|
|
outlineOffset: number;
|
|
selectedColor: string;
|
|
}>`
|
|
opacity: 0;
|
|
position: absolute;
|
|
cursor: pointer;
|
|
width: ${props => props.filledSize + props.outlineOffset}px;
|
|
height: ${props => props.filledSize + props.outlineOffset}px;
|
|
|
|
:checked + label:before {
|
|
background: ${props => props.selectedColor};
|
|
}
|
|
`;
|
|
// tslint:disable: use-simple-attributes
|
|
|
|
const StyledLabel = styled.label<{
|
|
selectedColor: string;
|
|
filledSize: number;
|
|
outlineOffset: number;
|
|
beforeMargins?: string;
|
|
}>`
|
|
cursor: pointer;
|
|
|
|
:before {
|
|
content: '';
|
|
display: inline-block;
|
|
border-radius: 100%;
|
|
|
|
transition: var(--default-duration);
|
|
padding: ${props => props.filledSize}px;
|
|
outline: var(--color-text) solid 1px;
|
|
border: none;
|
|
outline-offset: ${props => props.outlineOffset}px;
|
|
${props => props.beforeMargins && `margin: ${props.beforeMargins};`};
|
|
}
|
|
`;
|
|
|
|
export const SessionRadio = (props: Props) => {
|
|
const { label, inputName, value, active, onClick, beforeMargins } = props;
|
|
|
|
function clickHandler(e: ChangeEvent<any>) {
|
|
if (onClick) {
|
|
// let something else catch the event if our click handler is not set
|
|
e.stopPropagation();
|
|
onClick(value);
|
|
}
|
|
}
|
|
|
|
const selectedColor = 'var(--color-accent)';
|
|
const filledSize = 15 / 2;
|
|
const outlineOffset = 2;
|
|
|
|
return (
|
|
<Flex container={true} padding="0 0 0 var(--margins-lg)">
|
|
<StyledInput
|
|
type="radio"
|
|
name={inputName || ''}
|
|
value={value}
|
|
aria-checked={active}
|
|
checked={active}
|
|
onChange={clickHandler}
|
|
filledSize={filledSize * 2}
|
|
outlineOffset={outlineOffset}
|
|
selectedColor={selectedColor}
|
|
/>
|
|
<StyledLabel
|
|
role="button"
|
|
onClick={clickHandler}
|
|
selectedColor={selectedColor}
|
|
filledSize={filledSize - 1}
|
|
outlineOffset={outlineOffset}
|
|
beforeMargins={beforeMargins}
|
|
aria-label={label}
|
|
>
|
|
{label}
|
|
</StyledLabel>
|
|
</Flex>
|
|
);
|
|
};
|
|
|
|
const StyledInputOutlineSelected = styled(StyledInput)`
|
|
label:before,
|
|
label:before {
|
|
outline: none;
|
|
}
|
|
:checked + label:before {
|
|
outline: var(--color-text) solid 1px;
|
|
}
|
|
`;
|
|
const StyledLabelOutlineSelected = styled(StyledLabel)<{ selectedColor: string }>`
|
|
:before {
|
|
background: ${props => props.selectedColor};
|
|
outline: ${black} solid 1px;
|
|
}
|
|
`;
|
|
|
|
/**
|
|
* Keeping this component here so we can reuse the `StyledInput` and `StyledLabel` defined locally rather than exporting them
|
|
*/
|
|
export const SessionRadioPrimaryColors = (props: {
|
|
value: string;
|
|
active: boolean;
|
|
inputName?: string;
|
|
onClick: (value: string) => void;
|
|
ariaLabel: string;
|
|
color: string; // by default, we use the theme accent color but for the settings screen we need to be able to force it
|
|
}) => {
|
|
const { inputName, value, active, onClick, color, ariaLabel } = props;
|
|
|
|
function clickHandler(e: ChangeEvent<any>) {
|
|
e.stopPropagation();
|
|
onClick(value);
|
|
}
|
|
|
|
const filledSize = 31 / 2;
|
|
const outlineOffset = 5;
|
|
|
|
return (
|
|
<Flex container={true} padding="0 0 5px 0">
|
|
<StyledInputOutlineSelected
|
|
type="radio"
|
|
name={inputName || ''}
|
|
value={value}
|
|
aria-checked={active}
|
|
checked={active}
|
|
onChange={clickHandler}
|
|
filledSize={filledSize}
|
|
outlineOffset={outlineOffset}
|
|
selectedColor={color}
|
|
aria-label={ariaLabel}
|
|
/>
|
|
|
|
<StyledLabelOutlineSelected
|
|
role="button"
|
|
onClick={clickHandler}
|
|
selectedColor={color}
|
|
filledSize={filledSize}
|
|
outlineOffset={outlineOffset}
|
|
>
|
|
{''}
|
|
</StyledLabelOutlineSelected>
|
|
</Flex>
|
|
);
|
|
};
|