71 lines
1.9 KiB
TypeScript
71 lines
1.9 KiB
TypeScript
![]() |
import { clsx } from 'clsx'
|
||
|
import styles from './NotificationsPanel.module.scss'
|
||
|
import { useEscKeyDownHandler } from '../../utils/useEscKeyDownHandler'
|
||
|
import { useOutsideClickHandler } from '../../utils/useOutsideClickHandler'
|
||
|
import { useLocalize } from '../../context/localize'
|
||
|
import { Icon } from '../_shared/Icon'
|
||
|
import { createEffect, For, onCleanup, onMount } from 'solid-js'
|
||
|
import { useNotifications } from '../../context/notifications'
|
||
|
import { NotificationView } from './NotificationView'
|
||
|
|
||
|
type Props = {
|
||
|
isOpen: boolean
|
||
|
onClose: () => void
|
||
|
}
|
||
|
|
||
|
export const NotificationsPanel = (props: Props) => {
|
||
|
const { t } = useLocalize()
|
||
|
const { sortedNotifications } = useNotifications()
|
||
|
const handleHide = () => {
|
||
|
props.onClose()
|
||
|
}
|
||
|
|
||
|
const panelRef: { current: HTMLDivElement } = {
|
||
|
current: null
|
||
|
}
|
||
|
|
||
|
useOutsideClickHandler({
|
||
|
containerRef: panelRef,
|
||
|
predicate: () => props.isOpen,
|
||
|
handler: () => handleHide()
|
||
|
})
|
||
|
|
||
|
createEffect(() => {
|
||
|
document.body.classList.toggle('fixed', props.isOpen)
|
||
|
})
|
||
|
|
||
|
useEscKeyDownHandler(handleHide)
|
||
|
|
||
|
const handleNotificationViewClick = () => {
|
||
|
handleHide()
|
||
|
}
|
||
|
|
||
|
return (
|
||
|
<div
|
||
|
class={clsx(styles.container, {
|
||
|
[styles.isOpened]: props.isOpen
|
||
|
})}
|
||
|
>
|
||
|
<div ref={(el) => (panelRef.current = el)} class={styles.panel}>
|
||
|
<div class={styles.closeButton} onClick={handleHide}>
|
||
|
{/*TODO: check markup (hover)*/}
|
||
|
<Icon name="close" />
|
||
|
</div>
|
||
|
<div class={styles.title}>{t('Notifications')}</div>
|
||
|
<For
|
||
|
each={sortedNotifications()}
|
||
|
fallback={<div class={styles.emptyMessageContainer}>{t('No notifications, yet')}</div>}
|
||
|
>
|
||
|
{(notification) => (
|
||
|
<NotificationView
|
||
|
notification={notification}
|
||
|
class={styles.notificationView}
|
||
|
onClick={handleNotificationViewClick}
|
||
|
/>
|
||
|
)}
|
||
|
</For>
|
||
|
</div>
|
||
|
</div>
|
||
|
)
|
||
|
}
|