87 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			87 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import React, { useMemo } from 'react';
 | |
| import styled from 'styled-components/macro';
 | |
| import v4 from 'uuid/v4';
 | |
| import tw from 'twin.macro';
 | |
| import Label from '@/components/elements/Label';
 | |
| import Input from '@/components/elements/Input';
 | |
| 
 | |
| const ToggleContainer = styled.div`
 | |
|     ${tw`relative select-none w-12 leading-normal`};
 | |
| 
 | |
|     & > input[type="checkbox"] {
 | |
|         ${tw`hidden`};
 | |
| 
 | |
|         &:checked + label {
 | |
|             ${tw`bg-primary-500 border-primary-700 shadow-none`};
 | |
|         }
 | |
| 
 | |
|         &:checked + label:before {
 | |
|             right: 0.125rem;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     & > label {
 | |
|         ${tw`mb-0 block overflow-hidden cursor-pointer bg-neutral-400 border border-neutral-700 rounded-full h-6 shadow-inner`};
 | |
|         transition: all 75ms linear;
 | |
| 
 | |
|         &::before {
 | |
|             ${tw`absolute block bg-white border h-5 w-5 rounded-full`};
 | |
|             top: 0.125rem;
 | |
|             right: calc(50% + 0.125rem);
 | |
|             //width: 1.25rem;
 | |
|             //height: 1.25rem;
 | |
|             content: "";
 | |
|             transition: all 75ms ease-in;
 | |
|         }
 | |
|     }
 | |
| `;
 | |
| 
 | |
| export interface SwitchProps {
 | |
|     name: string;
 | |
|     label?: string;
 | |
|     description?: string;
 | |
|     defaultChecked?: boolean;
 | |
|     onChange?: (e: React.ChangeEvent<HTMLInputElement>) => void;
 | |
|     children?: React.ReactNode;
 | |
| }
 | |
| 
 | |
| const Switch = ({ name, label, description, defaultChecked, onChange, children }: SwitchProps) => {
 | |
|     const uuid = useMemo(() => v4(), []);
 | |
| 
 | |
|     return (
 | |
|         <div css={tw`flex items-center`}>
 | |
|             <ToggleContainer css={tw`flex-none`}>
 | |
|                 {children
 | |
|                 || <Input
 | |
|                     id={uuid}
 | |
|                     name={name}
 | |
|                     type={'checkbox'}
 | |
|                     onChange={e => onChange && onChange(e)}
 | |
|                     defaultChecked={defaultChecked}
 | |
|                 />
 | |
|                 }
 | |
|                 <Label htmlFor={uuid}/>
 | |
|             </ToggleContainer>
 | |
|             {(label || description) &&
 | |
|             <div css={tw`ml-4 w-full`}>
 | |
|                 {label &&
 | |
|                 <Label
 | |
|                     css={[ tw`cursor-pointer`, !!description && tw`mb-0` ]}
 | |
|                     htmlFor={uuid}
 | |
|                 >
 | |
|                     {label}
 | |
|                 </Label>
 | |
|                 }
 | |
|                 {description &&
 | |
|                 <p css={tw`text-neutral-400 text-sm mt-2`}>
 | |
|                     {description}
 | |
|                 </p>
 | |
|                 }
 | |
|             </div>
 | |
|             }
 | |
|         </div>
 | |
|     );
 | |
| };
 | |
| 
 | |
| export default Switch;
 | 
