Press n or j to go to the next uncovered block, b, p or k for the previous block.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 | /* eslint-disable jsx-a11y/click-events-have-key-events */ import React, { useEffect } from 'react'; import { css, keyframes } from '@emotion/react'; interface ToastRenderData { text: string; } export function useToast() { const [isOpen, setIsOpen] = React.useState(false); const [renderData, setRenderData] = React.useState<ToastRenderData>(); const onClose = () => setIsOpen(false); const openToast = (data: ToastRenderData) => { setRenderData(data); setIsOpen(true); }; const toastProps = React.useMemo( () => ({ ...renderData, onClose, isOpen }), [renderData, isOpen, setIsOpen] ); return { props: toastProps, openToast }; } interface ToastProps { isOpen: boolean; onClose: () => void; text: string; } export function Toast({ isOpen, onClose, text }: ToastProps) { useEffect(() => { const timer = setTimeout(() => { onClose(); }, 3000); return () => { clearTimeout(timer); }; }, [isOpen]); return isOpen ? ( <div css={toastWrapper}> <div css={toastBody}> <div css={toastText}>{text}</div> </div> </div> ) : null; } const toastAnimation = keyframes` 0% { opacity: 0; transform: translateY(10px); } 20% { opacity: 1; transform: translateY(0); } 80% { opacity: 1; transform: translateY(0); } 100% { opacity: 0; transform: translateY(10px); } `; const toastWrapper = css` position: fixed; display: flex; justify-content: center; align-items: flex-end; top: 0; right: 0; bottom: 20px; left: 0; padding: 0 16px; z-index: 10000; animation: ${toastAnimation} 3s ease-in-out; `; const toastBody = css` display: flex; flex-direction: column; justify-content: center; width: 168px; height: 36px; border-radius: 16px; background: #363636; box-shadow: 0 10px 130px 0 rgba(0, 0, 0, 0.15), 0 4px 20px 0 rgba(0, 0, 0, 0.15); `; const toastText = css` color: #ffffff; text-align: center; font-family: Noto Sans KR sans-serif; font-size: 14px; font-style: normal; font-weight: 400; line-height: 20px; `; |