2025-03-03 17:56:35 +08:00
|
|
|
|
import { Trans, useLingui } from '@lingui/react/macro';
|
2025-03-05 14:14:12 +08:00
|
|
|
|
import { ControlledMenu, MenuDivider, MenuItem } from '@szhsin/react-menu';
|
|
|
|
|
import { useCallback, useRef, useState } from 'preact/hooks';
|
2024-04-16 00:09:53 +08:00
|
|
|
|
import { useHotkeys } from 'react-hotkeys-hook';
|
2025-03-03 17:56:35 +08:00
|
|
|
|
import { useLongPress } from 'use-long-press';
|
2024-05-24 12:30:20 +08:00
|
|
|
|
import { useSnapshot } from 'valtio';
|
2024-04-16 00:09:53 +08:00
|
|
|
|
|
2025-03-05 14:14:12 +08:00
|
|
|
|
import { api } from '../utils/api';
|
|
|
|
|
import niceDateTime from '../utils/nice-date-time';
|
2023-09-05 18:49:16 +08:00
|
|
|
|
import openCompose from '../utils/open-compose';
|
2024-03-27 21:22:47 +08:00
|
|
|
|
import openOSK from '../utils/open-osk';
|
2025-03-05 14:14:12 +08:00
|
|
|
|
import pmem from '../utils/pmem';
|
2025-03-03 17:56:35 +08:00
|
|
|
|
import safeBoundingBoxPadding from '../utils/safe-bounding-box-padding';
|
2025-03-05 14:14:12 +08:00
|
|
|
|
import showCompose from '../utils/show-compose';
|
2023-09-05 18:49:16 +08:00
|
|
|
|
import states from '../utils/states';
|
2025-03-05 14:14:12 +08:00
|
|
|
|
import statusPeek from '../utils/status-peek';
|
|
|
|
|
import { getCurrentAccountID } from '../utils/store-utils';
|
2023-09-05 18:49:16 +08:00
|
|
|
|
|
|
|
|
|
import Icon from './icon';
|
2025-03-03 17:56:35 +08:00
|
|
|
|
import MenuLink from './menu-link';
|
2025-03-05 14:14:12 +08:00
|
|
|
|
import RelativeTime from './relative-time';
|
|
|
|
|
import SubMenu2 from './submenu2';
|
|
|
|
|
|
|
|
|
|
// Function to fetch the latest posts from the current user
|
|
|
|
|
// Use pmem to memoize fetch results for 1 minute
|
|
|
|
|
const fetchLatestPostsMemoized = pmem(
|
|
|
|
|
async (masto, currentAccountID) => {
|
|
|
|
|
const statusesIterator = masto.v1.accounts
|
|
|
|
|
.$select(currentAccountID)
|
|
|
|
|
.statuses.list({
|
|
|
|
|
limit: 3,
|
|
|
|
|
exclude_replies: true,
|
|
|
|
|
exclude_reblogs: true,
|
|
|
|
|
});
|
|
|
|
|
const { value } = await statusesIterator.next();
|
|
|
|
|
return value || [];
|
|
|
|
|
},
|
|
|
|
|
{ maxAge: 60000 },
|
|
|
|
|
); // 1 minute cache
|
2023-09-05 18:49:16 +08:00
|
|
|
|
|
|
|
|
|
export default function ComposeButton() {
|
2024-12-21 15:03:44 +08:00
|
|
|
|
const { t } = useLingui();
|
2024-05-24 12:30:20 +08:00
|
|
|
|
const snapStates = useSnapshot(states);
|
2025-03-05 14:14:12 +08:00
|
|
|
|
const { masto } = api();
|
2024-05-24 12:30:20 +08:00
|
|
|
|
|
2025-03-03 17:56:35 +08:00
|
|
|
|
// Context menu state
|
|
|
|
|
const [menuOpen, setMenuOpen] = useState(false);
|
2025-03-05 14:14:12 +08:00
|
|
|
|
const [latestPosts, setLatestPosts] = useState([]);
|
|
|
|
|
const [loadingPosts, setLoadingPosts] = useState(false);
|
2025-03-03 17:56:35 +08:00
|
|
|
|
const buttonRef = useRef(null);
|
|
|
|
|
const menuRef = useRef(null);
|
|
|
|
|
|
2023-09-05 21:44:38 +08:00
|
|
|
|
function handleButton(e) {
|
2024-05-24 12:30:20 +08:00
|
|
|
|
if (snapStates.composerState.minimized) {
|
|
|
|
|
states.composerState.minimized = false;
|
|
|
|
|
openOSK();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-05 21:44:38 +08:00
|
|
|
|
if (e.shiftKey) {
|
|
|
|
|
const newWin = openCompose();
|
|
|
|
|
|
|
|
|
|
if (!newWin) {
|
|
|
|
|
states.showCompose = true;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2024-03-27 21:22:47 +08:00
|
|
|
|
openOSK();
|
2023-09-05 21:44:38 +08:00
|
|
|
|
states.showCompose = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-09-05 18:49:16 +08:00
|
|
|
|
|
2023-09-05 21:44:38 +08:00
|
|
|
|
useHotkeys('c, shift+c', handleButton, {
|
|
|
|
|
ignoreEventWhen: (e) => {
|
|
|
|
|
const hasModal = !!document.querySelector('#modal-container > *');
|
|
|
|
|
return hasModal;
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
2025-03-03 17:56:35 +08:00
|
|
|
|
// Setup longpress handler to open context menu
|
|
|
|
|
const bindLongPress = useLongPress(
|
|
|
|
|
() => {
|
|
|
|
|
setMenuOpen(true);
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
threshold: 600,
|
|
|
|
|
},
|
|
|
|
|
);
|
|
|
|
|
|
2025-03-05 14:14:12 +08:00
|
|
|
|
const fetchLatestPosts = useCallback(async () => {
|
|
|
|
|
try {
|
|
|
|
|
setLoadingPosts(true);
|
|
|
|
|
const currentAccountID = getCurrentAccountID();
|
|
|
|
|
if (!currentAccountID) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
const posts = await fetchLatestPostsMemoized(masto, currentAccountID);
|
|
|
|
|
setLatestPosts(posts);
|
|
|
|
|
} catch (error) {
|
|
|
|
|
} finally {
|
|
|
|
|
setLoadingPosts(false);
|
|
|
|
|
}
|
|
|
|
|
}, [masto]);
|
|
|
|
|
|
|
|
|
|
// Function to handle opening the compose window to reply to a post
|
|
|
|
|
const handleReplyToPost = useCallback((post) => {
|
|
|
|
|
showCompose({
|
|
|
|
|
replyToStatus: post,
|
|
|
|
|
});
|
|
|
|
|
setMenuOpen(false);
|
|
|
|
|
}, []);
|
|
|
|
|
|
2023-09-05 21:44:38 +08:00
|
|
|
|
return (
|
2025-03-03 17:56:35 +08:00
|
|
|
|
<>
|
|
|
|
|
<button
|
|
|
|
|
ref={buttonRef}
|
|
|
|
|
type="button"
|
|
|
|
|
id="compose-button"
|
|
|
|
|
onClick={handleButton}
|
|
|
|
|
onContextMenu={(e) => {
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
setMenuOpen(true);
|
|
|
|
|
}}
|
|
|
|
|
{...bindLongPress()}
|
|
|
|
|
class={`${snapStates.composerState.minimized ? 'min' : ''} ${
|
|
|
|
|
snapStates.composerState.publishing ? 'loading' : ''
|
|
|
|
|
} ${snapStates.composerState.publishingError ? 'error' : ''}`}
|
|
|
|
|
>
|
|
|
|
|
<Icon icon="quill" size="xl" alt={t`Compose`} />
|
|
|
|
|
</button>
|
|
|
|
|
<ControlledMenu
|
|
|
|
|
ref={menuRef}
|
|
|
|
|
state={menuOpen ? 'open' : undefined}
|
|
|
|
|
anchorRef={buttonRef}
|
|
|
|
|
onClose={() => setMenuOpen(false)}
|
|
|
|
|
direction="top"
|
|
|
|
|
gap={8} // Add gap between menu and button
|
|
|
|
|
unmountOnClose
|
|
|
|
|
portal={{
|
|
|
|
|
target: document.body,
|
|
|
|
|
}}
|
|
|
|
|
boundingBoxPadding={safeBoundingBoxPadding()}
|
|
|
|
|
containerProps={{
|
|
|
|
|
style: {
|
|
|
|
|
zIndex: 1001,
|
|
|
|
|
},
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
<MenuLink to="/sp">
|
|
|
|
|
<Icon icon="schedule" size="l" />{' '}
|
|
|
|
|
<span>
|
|
|
|
|
<Trans>Scheduled Posts</Trans>
|
|
|
|
|
</span>
|
|
|
|
|
</MenuLink>
|
2025-03-05 14:14:12 +08:00
|
|
|
|
<MenuDivider />
|
|
|
|
|
<SubMenu2
|
|
|
|
|
label={
|
|
|
|
|
<>
|
|
|
|
|
<Icon icon="comment" size="l" />{' '}
|
|
|
|
|
<span className="menu-grow">
|
|
|
|
|
<Trans>Add to thread</Trans>
|
|
|
|
|
</span>
|
|
|
|
|
<Icon icon="chevron-right" />
|
|
|
|
|
</>
|
|
|
|
|
}
|
|
|
|
|
onMenuChange={(e) => {
|
|
|
|
|
if (e.open) {
|
|
|
|
|
fetchLatestPosts();
|
|
|
|
|
}
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
{loadingPosts ? (
|
|
|
|
|
<MenuItem disabled>
|
|
|
|
|
<span>
|
|
|
|
|
<Trans>Loading…</Trans>
|
|
|
|
|
</span>
|
|
|
|
|
</MenuItem>
|
|
|
|
|
) : latestPosts.length === 0 ? (
|
|
|
|
|
<MenuItem disabled>
|
|
|
|
|
<small>
|
|
|
|
|
<Trans>No posts found</Trans>
|
|
|
|
|
</small>
|
|
|
|
|
</MenuItem>
|
|
|
|
|
) : (
|
|
|
|
|
latestPosts.map((post) => {
|
|
|
|
|
const createdDate = new Date(post.createdAt);
|
|
|
|
|
const isWithinDay =
|
|
|
|
|
new Date().getTime() - createdDate.getTime() < 86400000;
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
<MenuItem key={post.id} onClick={() => handleReplyToPost(post)}>
|
|
|
|
|
<small>
|
|
|
|
|
<div class="menu-post-text">{statusPeek(post)}</div>
|
|
|
|
|
<span className="more-insignificant">
|
|
|
|
|
{/* Show relative time if within a day */}
|
|
|
|
|
{isWithinDay && (
|
|
|
|
|
<>
|
|
|
|
|
<RelativeTime datetime={createdDate} format="micro" />{' '}
|
|
|
|
|
‒{' '}
|
|
|
|
|
</>
|
|
|
|
|
)}
|
|
|
|
|
<time
|
|
|
|
|
className="created"
|
|
|
|
|
dateTime={createdDate.toISOString()}
|
|
|
|
|
title={createdDate.toLocaleString()}
|
|
|
|
|
>
|
|
|
|
|
{niceDateTime(post.createdAt)}
|
|
|
|
|
</time>
|
|
|
|
|
</span>
|
|
|
|
|
</small>
|
|
|
|
|
</MenuItem>
|
|
|
|
|
);
|
|
|
|
|
})
|
|
|
|
|
)}
|
|
|
|
|
</SubMenu2>
|
2025-03-03 17:56:35 +08:00
|
|
|
|
</ControlledMenu>
|
|
|
|
|
</>
|
2023-09-05 18:49:16 +08:00
|
|
|
|
);
|
|
|
|
|
}
|