2023-04-30 21:03:09 +08:00
|
|
|
import './notifications-menu.css';
|
|
|
|
|
|
|
|
import { ControlledMenu } from '@szhsin/react-menu';
|
2023-04-14 15:30:04 +08:00
|
|
|
import { memo } from 'preact/compat';
|
2023-04-30 21:03:09 +08:00
|
|
|
import { useEffect, useRef, useState } from 'preact/hooks';
|
2023-02-15 10:49:36 +08:00
|
|
|
import { useSnapshot } from 'valtio';
|
2022-12-10 17:14:48 +08:00
|
|
|
|
2023-02-18 21:14:24 +08:00
|
|
|
import Columns from '../components/columns';
|
2022-12-10 17:14:48 +08:00
|
|
|
import Icon from '../components/icon';
|
2023-02-15 10:49:36 +08:00
|
|
|
import Link from '../components/link';
|
2023-04-30 21:03:09 +08:00
|
|
|
import Loader from '../components/loader';
|
|
|
|
import Notification from '../components/notification';
|
|
|
|
import { api } from '../utils/api';
|
2023-01-13 15:30:09 +08:00
|
|
|
import db from '../utils/db';
|
2023-04-30 21:03:09 +08:00
|
|
|
import groupNotifications from '../utils/group-notifications';
|
2023-02-09 22:27:49 +08:00
|
|
|
import openCompose from '../utils/open-compose';
|
2023-04-30 21:03:09 +08:00
|
|
|
import states, { saveStatus } from '../utils/states';
|
2023-01-13 15:30:09 +08:00
|
|
|
import { getCurrentAccountNS } from '../utils/store-utils';
|
2022-12-10 17:14:48 +08:00
|
|
|
|
2023-02-09 22:27:49 +08:00
|
|
|
import Following from './following';
|
2023-01-02 21:36:24 +08:00
|
|
|
|
2023-02-09 22:27:49 +08:00
|
|
|
function Home() {
|
2023-02-15 10:49:36 +08:00
|
|
|
const snapStates = useSnapshot(states);
|
2023-01-13 15:30:09 +08:00
|
|
|
useEffect(() => {
|
|
|
|
(async () => {
|
|
|
|
const keys = await db.drafts.keys();
|
|
|
|
if (keys.length) {
|
|
|
|
const ns = getCurrentAccountNS();
|
|
|
|
const ownKeys = keys.filter((key) => key.startsWith(ns));
|
|
|
|
if (ownKeys.length) {
|
|
|
|
states.showDrafts = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})();
|
|
|
|
}, []);
|
|
|
|
|
2022-12-10 17:14:48 +08:00
|
|
|
return (
|
2023-01-21 00:23:59 +08:00
|
|
|
<>
|
2023-02-27 23:59:41 +08:00
|
|
|
{(snapStates.settings.shortcutsColumnsMode ||
|
|
|
|
snapStates.settings.shortcutsViewMode === 'multi-column') &&
|
2023-02-20 00:22:10 +08:00
|
|
|
!!snapStates.shortcuts?.length ? (
|
2023-02-18 21:14:24 +08:00
|
|
|
<Columns />
|
2023-02-18 20:48:24 +08:00
|
|
|
) : (
|
|
|
|
<Following
|
|
|
|
title="Home"
|
|
|
|
path="/"
|
|
|
|
id="home"
|
|
|
|
headerStart={false}
|
2023-04-30 21:59:14 +08:00
|
|
|
headerEnd={<NotificationsLink />}
|
2023-02-18 20:48:24 +08:00
|
|
|
/>
|
|
|
|
)}
|
2023-01-21 00:23:59 +08:00
|
|
|
<button
|
2023-02-09 22:27:49 +08:00
|
|
|
// hidden={scrollDirection === 'end' && !nearReachStart}
|
2023-01-21 00:23:59 +08:00
|
|
|
type="button"
|
|
|
|
id="compose-button"
|
|
|
|
onClick={(e) => {
|
|
|
|
if (e.shiftKey) {
|
|
|
|
const newWin = openCompose();
|
|
|
|
if (!newWin) {
|
|
|
|
alert('Looks like your browser is blocking popups.');
|
|
|
|
states.showCompose = true;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
states.showCompose = true;
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
>
|
2023-02-16 17:51:54 +08:00
|
|
|
<Icon icon="quill" size="xl" alt="Compose" />
|
2023-01-21 00:23:59 +08:00
|
|
|
</button>
|
|
|
|
</>
|
2022-12-10 17:14:48 +08:00
|
|
|
);
|
2022-12-16 13:27:04 +08:00
|
|
|
}
|
|
|
|
|
2023-04-30 21:59:14 +08:00
|
|
|
function NotificationsLink() {
|
|
|
|
const snapStates = useSnapshot(states);
|
|
|
|
const notificationLinkRef = useRef();
|
|
|
|
const [menuState, setMenuState] = useState(undefined);
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<Link
|
|
|
|
ref={notificationLinkRef}
|
|
|
|
to="/notifications"
|
|
|
|
class={`button plain notifications-button ${
|
|
|
|
snapStates.notificationsShowNew ? 'has-badge' : ''
|
|
|
|
} ${menuState}`}
|
|
|
|
onClick={(e) => {
|
|
|
|
e.stopPropagation();
|
|
|
|
if (window.matchMedia('(min-width: calc(40em))').matches) {
|
|
|
|
e.preventDefault();
|
|
|
|
setMenuState((state) => (!state ? 'open' : undefined));
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<Icon icon="notification" size="l" alt="Notifications" />
|
|
|
|
</Link>
|
|
|
|
<NotificationsMenu
|
|
|
|
state={menuState}
|
|
|
|
anchorRef={notificationLinkRef}
|
|
|
|
onClose={() => setMenuState(undefined)}
|
|
|
|
/>
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-04-30 21:03:09 +08:00
|
|
|
const NOTIFICATIONS_LIMIT = 30;
|
|
|
|
const NOTIFICATIONS_DISPLAY_LIMIT = 5;
|
|
|
|
function NotificationsMenu({ anchorRef, state, onClose }) {
|
|
|
|
const { masto, instance } = api();
|
|
|
|
const snapStates = useSnapshot(states);
|
|
|
|
const [uiState, setUIState] = useState('default');
|
|
|
|
|
|
|
|
const notificationsIterator = masto.v1.notifications.list({
|
|
|
|
limit: NOTIFICATIONS_LIMIT,
|
|
|
|
});
|
|
|
|
|
|
|
|
async function fetchNotifications() {
|
|
|
|
const allNotifications = await notificationsIterator.next();
|
|
|
|
const notifications = allNotifications.value;
|
|
|
|
|
|
|
|
if (notifications?.length) {
|
|
|
|
notifications.forEach((notification) => {
|
|
|
|
saveStatus(notification.status, instance, {
|
|
|
|
skipThreading: true,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
const groupedNotifications = groupNotifications(notifications);
|
|
|
|
|
|
|
|
states.notificationsLast = notifications[0];
|
|
|
|
states.notifications = groupedNotifications;
|
|
|
|
}
|
|
|
|
|
|
|
|
states.notificationsShowNew = false;
|
|
|
|
states.notificationsLastFetchTime = Date.now();
|
|
|
|
return allNotifications;
|
|
|
|
}
|
|
|
|
|
|
|
|
function loadNotifications() {
|
|
|
|
setUIState('loading');
|
|
|
|
(async () => {
|
|
|
|
try {
|
|
|
|
await fetchNotifications();
|
|
|
|
setUIState('default');
|
|
|
|
} catch (e) {
|
|
|
|
setUIState('error');
|
|
|
|
}
|
|
|
|
})();
|
|
|
|
}
|
|
|
|
|
|
|
|
useEffect(() => {
|
2023-04-30 21:59:14 +08:00
|
|
|
if (state === 'open') loadNotifications();
|
|
|
|
}, [state]);
|
2023-04-30 21:03:09 +08:00
|
|
|
|
|
|
|
return (
|
|
|
|
<ControlledMenu
|
|
|
|
menuClassName="notifications-menu"
|
|
|
|
state={state}
|
|
|
|
anchorRef={anchorRef}
|
|
|
|
onClose={onClose}
|
|
|
|
portal={{
|
|
|
|
target: document.body,
|
|
|
|
}}
|
|
|
|
overflow="auto"
|
|
|
|
viewScroll="close"
|
|
|
|
position="anchor"
|
|
|
|
align="center"
|
|
|
|
boundingBoxPadding="8 8 8 8"
|
|
|
|
>
|
|
|
|
<header>
|
|
|
|
<h2>Notifications</h2>
|
|
|
|
</header>
|
2023-05-01 00:53:49 +08:00
|
|
|
<main>
|
|
|
|
{snapStates.notifications.length ? (
|
|
|
|
<>
|
|
|
|
{snapStates.notifications
|
|
|
|
.slice(0, NOTIFICATIONS_DISPLAY_LIMIT)
|
|
|
|
.map((notification) => (
|
|
|
|
<Notification
|
|
|
|
key={notification.id}
|
|
|
|
instance={instance}
|
|
|
|
notification={notification}
|
|
|
|
/>
|
|
|
|
))}
|
|
|
|
</>
|
|
|
|
) : uiState === 'loading' ? (
|
2023-04-30 21:03:09 +08:00
|
|
|
<div class="ui-state">
|
2023-05-01 00:53:49 +08:00
|
|
|
<Loader abrupt />
|
2023-04-30 21:03:09 +08:00
|
|
|
</div>
|
2023-05-01 00:53:49 +08:00
|
|
|
) : (
|
|
|
|
uiState === 'error' && (
|
|
|
|
<div class="ui-state">
|
|
|
|
<p>Unable to fetch notifications.</p>
|
|
|
|
<p>
|
|
|
|
<button type="button" onClick={loadNotifications}>
|
|
|
|
Try again
|
|
|
|
</button>
|
|
|
|
</p>
|
|
|
|
</div>
|
|
|
|
)
|
|
|
|
)}
|
|
|
|
</main>
|
2023-04-30 21:03:09 +08:00
|
|
|
<footer>
|
|
|
|
<Link to="/mentions" class="button plain">
|
|
|
|
<Icon icon="at" /> <span>Mentions</span>
|
|
|
|
</Link>
|
|
|
|
<Link to="/notifications" class="button plain2">
|
|
|
|
<b>See all</b> <Icon icon="arrow-right" />
|
|
|
|
</Link>
|
|
|
|
</footer>
|
|
|
|
</ControlledMenu>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-04-14 15:30:04 +08:00
|
|
|
export default memo(Home);
|