2022-12-10 12:14:48 +03:00
|
|
|
|
import './status.css';
|
|
|
|
|
|
2023-01-24 15:56:43 +03:00
|
|
|
|
import { Menu, MenuItem } from '@szhsin/react-menu';
|
2022-12-10 12:14:48 +03:00
|
|
|
|
import { getBlurHashAverageColor } from 'fast-blurhash';
|
|
|
|
|
import mem from 'mem';
|
2023-01-07 15:26:23 +03:00
|
|
|
|
import { memo } from 'preact/compat';
|
2022-12-18 17:56:00 +03:00
|
|
|
|
import {
|
|
|
|
|
useEffect,
|
|
|
|
|
useLayoutEffect,
|
|
|
|
|
useMemo,
|
|
|
|
|
useRef,
|
|
|
|
|
useState,
|
|
|
|
|
} from 'preact/hooks';
|
2022-12-30 16:36:14 +03:00
|
|
|
|
import { useHotkeys } from 'react-hotkeys-hook';
|
2022-12-28 14:43:02 +03:00
|
|
|
|
import 'swiped-events';
|
2022-12-17 16:06:51 +03:00
|
|
|
|
import useResizeObserver from 'use-resize-observer';
|
2022-12-10 12:14:48 +03:00
|
|
|
|
import { useSnapshot } from 'valtio';
|
|
|
|
|
|
2022-12-11 16:22:22 +03:00
|
|
|
|
import Loader from '../components/loader';
|
2022-12-10 12:14:48 +03:00
|
|
|
|
import Modal from '../components/modal';
|
|
|
|
|
import NameText from '../components/name-text';
|
|
|
|
|
import enhanceContent from '../utils/enhance-content';
|
2023-01-22 15:29:48 +03:00
|
|
|
|
import handleAccountLinks from '../utils/handle-account-links';
|
2022-12-14 19:41:48 +03:00
|
|
|
|
import htmlContentLength from '../utils/html-content-length';
|
2022-12-10 12:14:48 +03:00
|
|
|
|
import shortenNumber from '../utils/shorten-number';
|
2023-01-09 14:11:34 +03:00
|
|
|
|
import states, { saveStatus } from '../utils/states';
|
2022-12-12 16:54:31 +03:00
|
|
|
|
import store from '../utils/store';
|
2022-12-10 12:14:48 +03:00
|
|
|
|
import visibilityIconsMap from '../utils/visibility-icons-map';
|
|
|
|
|
|
|
|
|
|
import Avatar from './avatar';
|
|
|
|
|
import Icon from './icon';
|
2023-01-20 19:23:59 +03:00
|
|
|
|
import Link from './link';
|
2023-01-05 05:50:27 +03:00
|
|
|
|
import RelativeTime from './relative-time';
|
2022-12-10 12:14:48 +03:00
|
|
|
|
|
2022-12-18 16:10:05 +03:00
|
|
|
|
function fetchAccount(id) {
|
2023-01-28 13:52:18 +03:00
|
|
|
|
try {
|
|
|
|
|
return masto.v1.accounts.fetch(id);
|
|
|
|
|
} catch (e) {
|
|
|
|
|
return Promise.reject(e);
|
|
|
|
|
}
|
2022-12-18 16:10:05 +03:00
|
|
|
|
}
|
|
|
|
|
const memFetchAccount = mem(fetchAccount);
|
2022-12-10 12:14:48 +03:00
|
|
|
|
|
2022-12-18 16:10:05 +03:00
|
|
|
|
function Status({
|
|
|
|
|
statusID,
|
|
|
|
|
status,
|
|
|
|
|
withinContext,
|
|
|
|
|
size = 'm',
|
|
|
|
|
skeleton,
|
|
|
|
|
readOnly,
|
|
|
|
|
}) {
|
|
|
|
|
if (skeleton) {
|
2022-12-10 12:14:48 +03:00
|
|
|
|
return (
|
2022-12-18 16:10:05 +03:00
|
|
|
|
<div class="status skeleton">
|
|
|
|
|
<Avatar size="xxl" />
|
|
|
|
|
<div class="container">
|
|
|
|
|
<div class="meta">███ ████████████</div>
|
|
|
|
|
<div class="content-container">
|
|
|
|
|
<div class="content">
|
|
|
|
|
<p>████ ████████████</p>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
2022-12-10 12:14:48 +03:00
|
|
|
|
</div>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-18 16:10:05 +03:00
|
|
|
|
const snapStates = useSnapshot(states);
|
|
|
|
|
if (!status) {
|
2023-01-07 15:26:23 +03:00
|
|
|
|
status = snapStates.statuses[statusID];
|
2022-12-18 16:10:05 +03:00
|
|
|
|
}
|
|
|
|
|
if (!status) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-10 12:14:48 +03:00
|
|
|
|
const {
|
2022-12-18 16:10:05 +03:00
|
|
|
|
account: {
|
|
|
|
|
acct,
|
|
|
|
|
avatar,
|
|
|
|
|
avatarStatic,
|
|
|
|
|
id: accountId,
|
|
|
|
|
url,
|
|
|
|
|
displayName,
|
|
|
|
|
username,
|
|
|
|
|
emojis: accountEmojis,
|
|
|
|
|
},
|
|
|
|
|
id,
|
|
|
|
|
repliesCount,
|
|
|
|
|
reblogged,
|
|
|
|
|
reblogsCount,
|
|
|
|
|
favourited,
|
|
|
|
|
favouritesCount,
|
|
|
|
|
bookmarked,
|
|
|
|
|
poll,
|
|
|
|
|
muted,
|
|
|
|
|
sensitive,
|
|
|
|
|
spoilerText,
|
|
|
|
|
visibility, // public, unlisted, private, direct
|
|
|
|
|
language,
|
|
|
|
|
editedAt,
|
|
|
|
|
filtered,
|
|
|
|
|
card,
|
|
|
|
|
createdAt,
|
2023-01-04 12:27:43 +03:00
|
|
|
|
inReplyToId,
|
2022-12-18 16:10:05 +03:00
|
|
|
|
inReplyToAccountId,
|
|
|
|
|
content,
|
|
|
|
|
mentions,
|
|
|
|
|
mediaAttachments,
|
|
|
|
|
reblog,
|
|
|
|
|
uri,
|
|
|
|
|
emojis,
|
2022-12-22 17:43:04 +03:00
|
|
|
|
_deleted,
|
2022-12-18 16:10:05 +03:00
|
|
|
|
} = status;
|
2022-12-10 12:14:48 +03:00
|
|
|
|
|
2023-01-07 15:26:23 +03:00
|
|
|
|
console.debug('RENDER Status', id, status?.account.displayName);
|
|
|
|
|
|
2022-12-18 16:10:05 +03:00
|
|
|
|
const createdAtDate = new Date(createdAt);
|
|
|
|
|
const editedAtDate = new Date(editedAt);
|
2022-12-10 12:14:48 +03:00
|
|
|
|
|
2022-12-18 16:10:05 +03:00
|
|
|
|
const isSelf = useMemo(() => {
|
|
|
|
|
const currentAccount = store.session.get('currentAccount');
|
|
|
|
|
return currentAccount && currentAccount === accountId;
|
|
|
|
|
}, [accountId]);
|
2022-12-10 12:14:48 +03:00
|
|
|
|
|
2022-12-18 16:10:05 +03:00
|
|
|
|
let inReplyToAccountRef = mentions?.find(
|
|
|
|
|
(mention) => mention.id === inReplyToAccountId,
|
|
|
|
|
);
|
|
|
|
|
if (!inReplyToAccountRef && inReplyToAccountId === id) {
|
|
|
|
|
inReplyToAccountRef = { url, username, displayName };
|
|
|
|
|
}
|
|
|
|
|
const [inReplyToAccount, setInReplyToAccount] = useState(inReplyToAccountRef);
|
|
|
|
|
if (!withinContext && !inReplyToAccount && inReplyToAccountId) {
|
2023-01-07 15:26:23 +03:00
|
|
|
|
const account = states.accounts[inReplyToAccountId];
|
2022-12-18 16:10:05 +03:00
|
|
|
|
if (account) {
|
|
|
|
|
setInReplyToAccount(account);
|
|
|
|
|
} else {
|
|
|
|
|
memFetchAccount(inReplyToAccountId)
|
|
|
|
|
.then((account) => {
|
|
|
|
|
setInReplyToAccount(account);
|
2023-01-07 15:26:23 +03:00
|
|
|
|
states.accounts[account.id] = account;
|
2022-12-18 16:10:05 +03:00
|
|
|
|
})
|
|
|
|
|
.catch((e) => {});
|
|
|
|
|
}
|
2022-12-10 12:14:48 +03:00
|
|
|
|
}
|
|
|
|
|
|
2023-01-07 15:26:23 +03:00
|
|
|
|
const showSpoiler = !!snapStates.spoilers[id] || false;
|
2022-12-10 12:14:48 +03:00
|
|
|
|
|
2022-12-18 16:10:05 +03:00
|
|
|
|
const debugHover = (e) => {
|
|
|
|
|
if (e.shiftKey) {
|
|
|
|
|
console.log(status);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const [showMediaModal, setShowMediaModal] = useState(false);
|
2022-12-14 16:48:17 +03:00
|
|
|
|
|
2022-12-18 16:10:05 +03:00
|
|
|
|
if (reblog) {
|
|
|
|
|
return (
|
|
|
|
|
<div class="status-reblog" onMouseEnter={debugHover}>
|
|
|
|
|
<div class="status-pre-meta">
|
|
|
|
|
<Icon icon="rocket" size="l" />{' '}
|
|
|
|
|
<NameText account={status.account} showAvatar /> boosted
|
|
|
|
|
</div>
|
|
|
|
|
<Status status={reblog} size={size} />
|
|
|
|
|
</div>
|
|
|
|
|
);
|
|
|
|
|
}
|
2022-12-10 12:14:48 +03:00
|
|
|
|
|
2022-12-18 16:10:05 +03:00
|
|
|
|
const [showEdited, setShowEdited] = useState(false);
|
|
|
|
|
|
|
|
|
|
const currentYear = new Date().getFullYear();
|
|
|
|
|
|
|
|
|
|
const spoilerContentRef = useRef(null);
|
|
|
|
|
useResizeObserver({
|
|
|
|
|
ref: spoilerContentRef,
|
|
|
|
|
onResize: () => {
|
|
|
|
|
if (spoilerContentRef.current) {
|
|
|
|
|
const { scrollHeight, clientHeight } = spoilerContentRef.current;
|
|
|
|
|
spoilerContentRef.current.classList.toggle(
|
|
|
|
|
'truncated',
|
|
|
|
|
scrollHeight > clientHeight,
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
const contentRef = useRef(null);
|
|
|
|
|
useResizeObserver({
|
|
|
|
|
ref: contentRef,
|
|
|
|
|
onResize: () => {
|
|
|
|
|
if (contentRef.current) {
|
|
|
|
|
const { scrollHeight, clientHeight } = contentRef.current;
|
|
|
|
|
contentRef.current.classList.toggle(
|
|
|
|
|
'truncated',
|
|
|
|
|
scrollHeight > clientHeight,
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
const readMoreText = 'Read more →';
|
2022-12-10 12:14:48 +03:00
|
|
|
|
|
2022-12-30 15:37:57 +03:00
|
|
|
|
const statusRef = useRef(null);
|
|
|
|
|
|
2022-12-10 12:14:48 +03:00
|
|
|
|
return (
|
2022-12-29 11:12:09 +03:00
|
|
|
|
<article
|
2022-12-30 15:37:57 +03:00
|
|
|
|
ref={statusRef}
|
|
|
|
|
tabindex="-1"
|
2022-12-18 16:10:05 +03:00
|
|
|
|
class={`status ${
|
|
|
|
|
!withinContext && inReplyToAccount ? 'status-reply-to' : ''
|
|
|
|
|
} visibility-${visibility} ${
|
|
|
|
|
{
|
|
|
|
|
s: 'small',
|
|
|
|
|
m: 'medium',
|
|
|
|
|
l: 'large',
|
|
|
|
|
}[size]
|
|
|
|
|
}`}
|
|
|
|
|
onMouseEnter={debugHover}
|
|
|
|
|
>
|
2022-12-20 15:17:38 +03:00
|
|
|
|
{size !== 'l' && (
|
|
|
|
|
<div class="status-badge">
|
|
|
|
|
{reblogged && <Icon class="reblog" icon="rocket" size="s" />}
|
|
|
|
|
{favourited && <Icon class="favourite" icon="heart" size="s" />}
|
|
|
|
|
{bookmarked && <Icon class="bookmark" icon="bookmark" size="s" />}
|
|
|
|
|
</div>
|
|
|
|
|
)}
|
2022-12-18 16:10:05 +03:00
|
|
|
|
{size !== 's' && (
|
|
|
|
|
<a
|
|
|
|
|
href={url}
|
2022-12-29 11:11:58 +03:00
|
|
|
|
tabindex="-1"
|
2022-12-18 16:10:05 +03:00
|
|
|
|
// target="_blank"
|
|
|
|
|
title={`@${acct}`}
|
|
|
|
|
onClick={(e) => {
|
2022-12-10 12:14:48 +03:00
|
|
|
|
e.preventDefault();
|
2022-12-18 16:10:05 +03:00
|
|
|
|
e.stopPropagation();
|
|
|
|
|
states.showAccount = status.account;
|
2022-12-10 12:14:48 +03:00
|
|
|
|
}}
|
|
|
|
|
>
|
2022-12-18 16:10:05 +03:00
|
|
|
|
<Avatar url={avatarStatic} size="xxl" />
|
|
|
|
|
</a>
|
2022-12-10 12:14:48 +03:00
|
|
|
|
)}
|
2022-12-18 16:10:05 +03:00
|
|
|
|
<div class="container">
|
|
|
|
|
<div class="meta">
|
2022-12-22 05:35:39 +03:00
|
|
|
|
{/* <span> */}
|
|
|
|
|
<NameText
|
|
|
|
|
account={status.account}
|
|
|
|
|
showAvatar={size === 's'}
|
|
|
|
|
showAcct={size === 'l'}
|
|
|
|
|
/>
|
|
|
|
|
{/* {inReplyToAccount && !withinContext && size !== 's' && (
|
2022-12-18 16:10:05 +03:00
|
|
|
|
<>
|
|
|
|
|
{' '}
|
|
|
|
|
<span class="ib">
|
|
|
|
|
<Icon icon="arrow-right" class="arrow" />{' '}
|
|
|
|
|
<NameText account={inReplyToAccount} short />
|
|
|
|
|
</span>
|
|
|
|
|
</>
|
2022-12-22 05:35:39 +03:00
|
|
|
|
)} */}
|
|
|
|
|
{/* </span> */}{' '}
|
2022-12-18 16:10:05 +03:00
|
|
|
|
{size !== 'l' &&
|
|
|
|
|
(uri ? (
|
2023-01-20 19:23:59 +03:00
|
|
|
|
<Link to={`/s/${id}`} class="time">
|
2022-12-18 16:10:05 +03:00
|
|
|
|
<Icon
|
|
|
|
|
icon={visibilityIconsMap[visibility]}
|
|
|
|
|
alt={visibility}
|
|
|
|
|
size="s"
|
|
|
|
|
/>{' '}
|
2023-01-05 05:50:27 +03:00
|
|
|
|
<RelativeTime datetime={createdAtDate} format="micro" />
|
2023-01-20 19:23:59 +03:00
|
|
|
|
</Link>
|
2022-12-18 16:10:05 +03:00
|
|
|
|
) : (
|
|
|
|
|
<span class="time">
|
|
|
|
|
<Icon
|
|
|
|
|
icon={visibilityIconsMap[visibility]}
|
|
|
|
|
alt={visibility}
|
|
|
|
|
size="s"
|
|
|
|
|
/>{' '}
|
2023-01-05 05:50:27 +03:00
|
|
|
|
<RelativeTime datetime={createdAtDate} format="micro" />
|
2022-12-18 16:10:05 +03:00
|
|
|
|
</span>
|
|
|
|
|
))}
|
|
|
|
|
</div>
|
2023-01-23 15:34:53 +03:00
|
|
|
|
{!withinContext && (
|
2023-01-10 14:59:02 +03:00
|
|
|
|
<>
|
|
|
|
|
{inReplyToAccountId === status.account?.id ||
|
|
|
|
|
!!snapStates.statusThreadNumber[id] ? (
|
|
|
|
|
<div class="status-thread-badge">
|
|
|
|
|
<Icon icon="thread" size="s" />
|
|
|
|
|
Thread
|
|
|
|
|
{snapStates.statusThreadNumber[id]
|
|
|
|
|
? ` ${snapStates.statusThreadNumber[id]}/X`
|
|
|
|
|
: ''}
|
|
|
|
|
</div>
|
|
|
|
|
) : (
|
|
|
|
|
!!inReplyToId &&
|
|
|
|
|
!!inReplyToAccount &&
|
|
|
|
|
(!!spoilerText ||
|
|
|
|
|
!mentions.find((mention) => {
|
|
|
|
|
return mention.id === inReplyToAccountId;
|
|
|
|
|
})) && (
|
|
|
|
|
<div class="status-reply-badge">
|
|
|
|
|
<Icon icon="reply" />{' '}
|
|
|
|
|
<NameText account={inReplyToAccount} short />
|
2022-12-23 20:22:25 +03:00
|
|
|
|
</div>
|
2023-01-10 14:59:02 +03:00
|
|
|
|
)
|
|
|
|
|
)}
|
|
|
|
|
</>
|
|
|
|
|
)}
|
2022-12-18 16:10:05 +03:00
|
|
|
|
<div
|
|
|
|
|
class={`content-container ${
|
|
|
|
|
sensitive || spoilerText ? 'has-spoiler' : ''
|
|
|
|
|
} ${showSpoiler ? 'show-spoiler' : ''}`}
|
|
|
|
|
style={
|
|
|
|
|
size === 'l' && {
|
|
|
|
|
'--content-text-weight':
|
|
|
|
|
Math.round(
|
|
|
|
|
(spoilerText.length + htmlContentLength(content)) / 140,
|
|
|
|
|
) || 1,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
>
|
|
|
|
|
{!!spoilerText && sensitive && (
|
2022-12-14 16:48:17 +03:00
|
|
|
|
<>
|
2022-12-18 16:10:05 +03:00
|
|
|
|
<div
|
|
|
|
|
class="content"
|
|
|
|
|
lang={language}
|
|
|
|
|
ref={spoilerContentRef}
|
|
|
|
|
data-read-more={readMoreText}
|
|
|
|
|
>
|
|
|
|
|
<p>{spoilerText}</p>
|
|
|
|
|
</div>
|
|
|
|
|
<button
|
2023-01-28 17:34:36 +03:00
|
|
|
|
class={`light spoiler ${showSpoiler ? 'spoiling' : ''}`}
|
2022-12-18 16:10:05 +03:00
|
|
|
|
type="button"
|
|
|
|
|
onClick={(e) => {
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
e.stopPropagation();
|
2022-12-20 14:14:50 +03:00
|
|
|
|
if (showSpoiler) {
|
2023-01-07 15:26:23 +03:00
|
|
|
|
delete states.spoilers[id];
|
2022-12-20 14:14:50 +03:00
|
|
|
|
} else {
|
2023-01-07 15:26:23 +03:00
|
|
|
|
states.spoilers[id] = true;
|
2022-12-20 14:14:50 +03:00
|
|
|
|
}
|
2022-12-18 16:10:05 +03:00
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
<Icon icon={showSpoiler ? 'eye-open' : 'eye-close'} />{' '}
|
|
|
|
|
{showSpoiler ? 'Show less' : 'Show more'}
|
|
|
|
|
</button>
|
2022-12-10 12:14:48 +03:00
|
|
|
|
</>
|
|
|
|
|
)}
|
|
|
|
|
<div
|
|
|
|
|
class="content"
|
2022-12-17 16:06:51 +03:00
|
|
|
|
lang={language}
|
|
|
|
|
ref={contentRef}
|
|
|
|
|
data-read-more={readMoreText}
|
2023-01-22 15:29:48 +03:00
|
|
|
|
onClick={handleAccountLinks({ mentions })}
|
2022-12-10 12:14:48 +03:00
|
|
|
|
dangerouslySetInnerHTML={{
|
|
|
|
|
__html: enhanceContent(content, {
|
|
|
|
|
emojis,
|
2022-12-11 04:28:02 +03:00
|
|
|
|
postEnhanceDOM: (dom) => {
|
|
|
|
|
dom
|
|
|
|
|
.querySelectorAll('a.u-url[target="_blank"]')
|
|
|
|
|
.forEach((a) => {
|
|
|
|
|
// Remove target="_blank" from links
|
2023-01-19 10:51:54 +03:00
|
|
|
|
if (!/http/i.test(a.innerText.trim())) {
|
|
|
|
|
a.removeAttribute('target');
|
|
|
|
|
}
|
2022-12-11 04:28:02 +03:00
|
|
|
|
});
|
|
|
|
|
},
|
2022-12-10 12:14:48 +03:00
|
|
|
|
}),
|
|
|
|
|
}}
|
|
|
|
|
/>
|
2022-12-21 14:29:37 +03:00
|
|
|
|
{!!poll && (
|
|
|
|
|
<Poll
|
2022-12-27 13:09:23 +03:00
|
|
|
|
lang={language}
|
2022-12-21 14:29:37 +03:00
|
|
|
|
poll={poll}
|
|
|
|
|
readOnly={readOnly}
|
|
|
|
|
onUpdate={(newPoll) => {
|
2023-01-07 15:26:23 +03:00
|
|
|
|
states.statuses[id].poll = newPoll;
|
2022-12-21 14:29:37 +03:00
|
|
|
|
}}
|
|
|
|
|
/>
|
|
|
|
|
)}
|
2022-12-10 12:14:48 +03:00
|
|
|
|
{!spoilerText && sensitive && !!mediaAttachments.length && (
|
|
|
|
|
<button
|
2023-01-28 17:34:36 +03:00
|
|
|
|
class={`plain spoiler ${showSpoiler ? 'spoiling' : ''}`}
|
2022-12-10 12:14:48 +03:00
|
|
|
|
type="button"
|
|
|
|
|
onClick={(e) => {
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
e.stopPropagation();
|
2022-12-20 14:14:50 +03:00
|
|
|
|
if (showSpoiler) {
|
2023-01-07 15:26:23 +03:00
|
|
|
|
delete states.spoilers[id];
|
2022-12-20 14:14:50 +03:00
|
|
|
|
} else {
|
2023-01-07 15:26:23 +03:00
|
|
|
|
states.spoilers[id] = true;
|
2022-12-20 14:14:50 +03:00
|
|
|
|
}
|
2022-12-10 12:14:48 +03:00
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
<Icon icon={showSpoiler ? 'eye-open' : 'eye-close'} /> Sensitive
|
|
|
|
|
content
|
|
|
|
|
</button>
|
|
|
|
|
)}
|
2022-12-12 11:25:55 +03:00
|
|
|
|
{!!mediaAttachments.length && (
|
2022-12-22 05:47:45 +03:00
|
|
|
|
<div
|
2023-01-23 15:35:15 +03:00
|
|
|
|
class={`media-container media-eq${mediaAttachments.length} ${
|
2022-12-22 05:47:45 +03:00
|
|
|
|
mediaAttachments.length > 2 ? 'media-gt2' : ''
|
|
|
|
|
} ${mediaAttachments.length > 4 ? 'media-gt4' : ''}`}
|
|
|
|
|
>
|
2022-12-28 05:51:57 +03:00
|
|
|
|
{mediaAttachments
|
|
|
|
|
.slice(0, size === 'l' ? undefined : 4)
|
|
|
|
|
.map((media, i) => (
|
|
|
|
|
<Media
|
|
|
|
|
key={media.id}
|
|
|
|
|
media={media}
|
2023-01-06 13:25:47 +03:00
|
|
|
|
autoAnimate={size === 'l'}
|
2022-12-28 05:51:57 +03:00
|
|
|
|
onClick={(e) => {
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
e.stopPropagation();
|
|
|
|
|
setShowMediaModal(i);
|
|
|
|
|
}}
|
|
|
|
|
/>
|
|
|
|
|
))}
|
2022-12-10 12:14:48 +03:00
|
|
|
|
</div>
|
|
|
|
|
)}
|
|
|
|
|
{!!card &&
|
2023-01-05 08:29:44 +03:00
|
|
|
|
!sensitive &&
|
|
|
|
|
!spoilerText &&
|
2023-01-05 10:28:04 +03:00
|
|
|
|
!poll &&
|
2023-01-07 06:52:23 +03:00
|
|
|
|
!mediaAttachments.length && <Card card={card} />}
|
2022-12-10 12:14:48 +03:00
|
|
|
|
</div>
|
|
|
|
|
{size === 'l' && (
|
2022-12-11 16:22:22 +03:00
|
|
|
|
<>
|
|
|
|
|
<div class="extra-meta">
|
|
|
|
|
<Icon icon={visibilityIconsMap[visibility]} alt={visibility} />{' '}
|
|
|
|
|
<a href={uri} target="_blank">
|
|
|
|
|
<time class="created" datetime={createdAtDate.toISOString()}>
|
|
|
|
|
{Intl.DateTimeFormat('en', {
|
|
|
|
|
// Show year if not current year
|
|
|
|
|
year:
|
|
|
|
|
createdAtDate.getFullYear() === currentYear
|
|
|
|
|
? undefined
|
|
|
|
|
: 'numeric',
|
|
|
|
|
month: 'short',
|
|
|
|
|
day: 'numeric',
|
|
|
|
|
hour: 'numeric',
|
|
|
|
|
minute: '2-digit',
|
|
|
|
|
}).format(createdAtDate)}
|
|
|
|
|
</time>
|
|
|
|
|
</a>
|
|
|
|
|
{editedAt && (
|
2022-12-10 12:14:48 +03:00
|
|
|
|
<>
|
|
|
|
|
{' '}
|
2022-12-11 16:22:22 +03:00
|
|
|
|
• <Icon icon="pencil" alt="Edited" />{' '}
|
|
|
|
|
<time
|
|
|
|
|
class="edited"
|
|
|
|
|
datetime={editedAtDate.toISOString()}
|
|
|
|
|
onClick={() => {
|
|
|
|
|
setShowEdited(id);
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
{Intl.DateTimeFormat('en', {
|
|
|
|
|
// Show year if not this year
|
|
|
|
|
year:
|
|
|
|
|
editedAtDate.getFullYear() === currentYear
|
|
|
|
|
? undefined
|
|
|
|
|
: 'numeric',
|
|
|
|
|
month: 'short',
|
|
|
|
|
day: 'numeric',
|
|
|
|
|
hour: 'numeric',
|
|
|
|
|
minute: '2-digit',
|
|
|
|
|
}).format(editedAtDate)}
|
|
|
|
|
</time>
|
2022-12-10 12:14:48 +03:00
|
|
|
|
</>
|
|
|
|
|
)}
|
2022-12-11 16:22:22 +03:00
|
|
|
|
</div>
|
|
|
|
|
<div class="actions">
|
2022-12-19 08:38:16 +03:00
|
|
|
|
<div class="action has-count">
|
|
|
|
|
<StatusButton
|
|
|
|
|
title="Reply"
|
|
|
|
|
alt="Comments"
|
|
|
|
|
class="reply-button"
|
|
|
|
|
icon="comment"
|
|
|
|
|
count={repliesCount}
|
|
|
|
|
onClick={() => {
|
|
|
|
|
states.showCompose = {
|
|
|
|
|
replyToStatus: status,
|
|
|
|
|
};
|
|
|
|
|
}}
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
2022-12-11 16:22:22 +03:00
|
|
|
|
{/* TODO: if visibility = private, only can reblog own statuses */}
|
|
|
|
|
{visibility !== 'direct' && (
|
2022-12-19 08:38:16 +03:00
|
|
|
|
<div class="action has-count">
|
|
|
|
|
<StatusButton
|
|
|
|
|
checked={reblogged}
|
|
|
|
|
title={['Boost', 'Unboost']}
|
|
|
|
|
alt={['Boost', 'Boosted']}
|
|
|
|
|
class="reblog-button"
|
|
|
|
|
icon="rocket"
|
|
|
|
|
count={reblogsCount}
|
|
|
|
|
onClick={async () => {
|
|
|
|
|
try {
|
2022-12-20 05:09:05 +03:00
|
|
|
|
if (!reblogged) {
|
|
|
|
|
const yes = confirm(
|
|
|
|
|
'Are you sure that you want to boost this post?',
|
|
|
|
|
);
|
|
|
|
|
if (!yes) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-12-19 08:38:16 +03:00
|
|
|
|
// Optimistic
|
2023-01-07 15:26:23 +03:00
|
|
|
|
states.statuses[id] = {
|
2022-12-19 08:38:16 +03:00
|
|
|
|
...status,
|
|
|
|
|
reblogged: !reblogged,
|
|
|
|
|
reblogsCount: reblogsCount + (reblogged ? -1 : 1),
|
2023-01-07 15:26:23 +03:00
|
|
|
|
};
|
2022-12-19 08:38:16 +03:00
|
|
|
|
if (reblogged) {
|
2022-12-25 18:28:55 +03:00
|
|
|
|
const newStatus = await masto.v1.statuses.unreblog(
|
|
|
|
|
id,
|
|
|
|
|
);
|
2023-01-09 14:11:34 +03:00
|
|
|
|
saveStatus(newStatus);
|
2022-12-19 08:38:16 +03:00
|
|
|
|
} else {
|
2022-12-25 18:28:55 +03:00
|
|
|
|
const newStatus = await masto.v1.statuses.reblog(id);
|
2023-01-09 14:11:34 +03:00
|
|
|
|
saveStatus(newStatus);
|
2022-12-19 08:38:16 +03:00
|
|
|
|
}
|
|
|
|
|
} catch (e) {
|
|
|
|
|
console.error(e);
|
2022-12-25 08:22:41 +03:00
|
|
|
|
// Revert optimistism
|
2023-01-07 15:26:23 +03:00
|
|
|
|
states.statuses[id] = status;
|
2022-12-19 08:38:16 +03:00
|
|
|
|
}
|
|
|
|
|
}}
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
|
|
|
|
)}
|
|
|
|
|
<div class="action has-count">
|
2022-12-17 12:26:41 +03:00
|
|
|
|
<StatusButton
|
2022-12-19 08:38:16 +03:00
|
|
|
|
checked={favourited}
|
|
|
|
|
title={['Favourite', 'Unfavourite']}
|
|
|
|
|
alt={['Favourite', 'Favourited']}
|
|
|
|
|
class="favourite-button"
|
|
|
|
|
icon="heart"
|
|
|
|
|
count={favouritesCount}
|
2022-12-17 12:26:41 +03:00
|
|
|
|
onClick={async () => {
|
2022-12-11 16:22:22 +03:00
|
|
|
|
try {
|
2022-12-17 12:26:41 +03:00
|
|
|
|
// Optimistic
|
2023-01-07 15:26:23 +03:00
|
|
|
|
states.statuses[statusID] = {
|
2022-12-17 12:26:41 +03:00
|
|
|
|
...status,
|
2022-12-19 08:38:16 +03:00
|
|
|
|
favourited: !favourited,
|
|
|
|
|
favouritesCount:
|
|
|
|
|
favouritesCount + (favourited ? -1 : 1),
|
2023-01-07 15:26:23 +03:00
|
|
|
|
};
|
2022-12-19 08:38:16 +03:00
|
|
|
|
if (favourited) {
|
2022-12-25 18:28:55 +03:00
|
|
|
|
const newStatus = await masto.v1.statuses.unfavourite(
|
|
|
|
|
id,
|
|
|
|
|
);
|
2023-01-09 14:11:34 +03:00
|
|
|
|
saveStatus(newStatus);
|
2022-12-11 16:22:22 +03:00
|
|
|
|
} else {
|
2022-12-25 18:28:55 +03:00
|
|
|
|
const newStatus = await masto.v1.statuses.favourite(id);
|
2023-01-09 14:11:34 +03:00
|
|
|
|
saveStatus(newStatus);
|
2022-12-11 16:22:22 +03:00
|
|
|
|
}
|
|
|
|
|
} catch (e) {
|
|
|
|
|
console.error(e);
|
2022-12-25 08:22:41 +03:00
|
|
|
|
// Revert optimistism
|
2023-01-07 15:26:23 +03:00
|
|
|
|
states.statuses[statusID] = status;
|
2022-12-11 16:22:22 +03:00
|
|
|
|
}
|
|
|
|
|
}}
|
2022-12-17 12:26:41 +03:00
|
|
|
|
/>
|
2022-12-19 08:38:16 +03:00
|
|
|
|
</div>
|
|
|
|
|
<div class="action">
|
|
|
|
|
<StatusButton
|
|
|
|
|
checked={bookmarked}
|
|
|
|
|
title={['Bookmark', 'Unbookmark']}
|
|
|
|
|
alt={['Bookmark', 'Bookmarked']}
|
|
|
|
|
class="bookmark-button"
|
|
|
|
|
icon="bookmark"
|
|
|
|
|
onClick={async () => {
|
|
|
|
|
try {
|
|
|
|
|
// Optimistic
|
2023-01-07 15:26:23 +03:00
|
|
|
|
states.statuses[statusID] = {
|
2022-12-19 08:38:16 +03:00
|
|
|
|
...status,
|
|
|
|
|
bookmarked: !bookmarked,
|
2023-01-07 15:26:23 +03:00
|
|
|
|
};
|
2022-12-19 08:38:16 +03:00
|
|
|
|
if (bookmarked) {
|
2022-12-25 18:28:55 +03:00
|
|
|
|
const newStatus = await masto.v1.statuses.unbookmark(
|
|
|
|
|
id,
|
|
|
|
|
);
|
2023-01-09 14:11:34 +03:00
|
|
|
|
saveStatus(newStatus);
|
2022-12-19 08:38:16 +03:00
|
|
|
|
} else {
|
2022-12-25 18:28:55 +03:00
|
|
|
|
const newStatus = await masto.v1.statuses.bookmark(id);
|
2023-01-09 14:11:34 +03:00
|
|
|
|
saveStatus(newStatus);
|
2022-12-19 08:38:16 +03:00
|
|
|
|
}
|
|
|
|
|
} catch (e) {
|
|
|
|
|
console.error(e);
|
2022-12-25 08:22:41 +03:00
|
|
|
|
// Revert optimistism
|
2023-01-07 15:26:23 +03:00
|
|
|
|
states.statuses[statusID] = status;
|
2022-12-11 16:22:22 +03:00
|
|
|
|
}
|
2022-12-19 08:38:16 +03:00
|
|
|
|
}}
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
2022-12-12 16:54:31 +03:00
|
|
|
|
{isSelf && (
|
2023-01-24 15:56:43 +03:00
|
|
|
|
<Menu
|
|
|
|
|
align="end"
|
|
|
|
|
menuButton={
|
|
|
|
|
<div class="action">
|
|
|
|
|
<button
|
|
|
|
|
type="button"
|
|
|
|
|
title="More"
|
|
|
|
|
class="plain more-button"
|
|
|
|
|
>
|
|
|
|
|
<Icon icon="more" size="l" alt="More" />
|
|
|
|
|
</button>
|
|
|
|
|
</div>
|
|
|
|
|
}
|
|
|
|
|
>
|
|
|
|
|
{isSelf && (
|
|
|
|
|
<MenuItem
|
|
|
|
|
onClick={() => {
|
|
|
|
|
states.showCompose = {
|
|
|
|
|
editStatus: status,
|
|
|
|
|
};
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
Edit…
|
|
|
|
|
</MenuItem>
|
|
|
|
|
)}
|
|
|
|
|
</Menu>
|
2022-12-12 16:54:31 +03:00
|
|
|
|
)}
|
2022-12-11 16:22:22 +03:00
|
|
|
|
</div>
|
|
|
|
|
</>
|
2022-12-10 12:14:48 +03:00
|
|
|
|
)}
|
|
|
|
|
</div>
|
|
|
|
|
{showMediaModal !== false && (
|
|
|
|
|
<Modal>
|
2022-12-18 17:56:00 +03:00
|
|
|
|
<Carousel
|
|
|
|
|
mediaAttachments={mediaAttachments}
|
|
|
|
|
index={showMediaModal}
|
|
|
|
|
onClose={() => {
|
|
|
|
|
setShowMediaModal(false);
|
2022-12-18 16:10:05 +03:00
|
|
|
|
}}
|
2022-12-18 17:56:00 +03:00
|
|
|
|
/>
|
2022-12-18 16:10:05 +03:00
|
|
|
|
</Modal>
|
|
|
|
|
)}
|
|
|
|
|
{!!showEdited && (
|
|
|
|
|
<Modal
|
|
|
|
|
onClick={(e) => {
|
|
|
|
|
if (e.target === e.currentTarget) {
|
|
|
|
|
setShowEdited(false);
|
2022-12-30 15:37:57 +03:00
|
|
|
|
statusRef.current?.focus();
|
2022-12-18 16:10:05 +03:00
|
|
|
|
}
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
<EditedAtModal
|
|
|
|
|
statusID={showEdited}
|
|
|
|
|
onClose={() => {
|
|
|
|
|
setShowEdited(false);
|
2022-12-30 15:37:57 +03:00
|
|
|
|
statusRef.current?.focus();
|
2022-12-18 16:10:05 +03:00
|
|
|
|
}}
|
|
|
|
|
/>
|
|
|
|
|
</Modal>
|
|
|
|
|
)}
|
2022-12-29 11:12:09 +03:00
|
|
|
|
</article>
|
2022-12-18 16:10:05 +03:00
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
Media type
|
|
|
|
|
===
|
|
|
|
|
unknown = unsupported or unrecognized file type
|
|
|
|
|
image = Static image
|
|
|
|
|
gifv = Looping, soundless animation
|
|
|
|
|
video = Video clip
|
|
|
|
|
audio = Audio track
|
|
|
|
|
*/
|
|
|
|
|
|
2023-01-06 13:25:47 +03:00
|
|
|
|
function Media({ media, showOriginal, autoAnimate, onClick = () => {} }) {
|
2022-12-18 16:10:05 +03:00
|
|
|
|
const { blurhash, description, meta, previewUrl, remoteUrl, url, type } =
|
|
|
|
|
media;
|
|
|
|
|
const { original, small, focus } = meta || {};
|
|
|
|
|
|
|
|
|
|
const width = showOriginal ? original?.width : small?.width;
|
|
|
|
|
const height = showOriginal ? original?.height : small?.height;
|
|
|
|
|
const mediaURL = showOriginal ? url : previewUrl;
|
|
|
|
|
|
|
|
|
|
const rgbAverageColor = blurhash ? getBlurHashAverageColor(blurhash) : null;
|
|
|
|
|
|
|
|
|
|
const videoRef = useRef();
|
|
|
|
|
|
|
|
|
|
let focalBackgroundPosition;
|
|
|
|
|
if (focus) {
|
|
|
|
|
// Convert focal point to CSS background position
|
|
|
|
|
// Formula from jquery-focuspoint
|
|
|
|
|
// x = -1, y = 1 => 0% 0%
|
|
|
|
|
// x = 0, y = 0 => 50% 50%
|
|
|
|
|
// x = 1, y = -1 => 100% 100%
|
|
|
|
|
const x = ((focus.x + 1) / 2) * 100;
|
|
|
|
|
const y = ((1 - focus.y) / 2) * 100;
|
|
|
|
|
focalBackgroundPosition = `${x.toFixed(0)}% ${y.toFixed(0)}%`;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (type === 'image' || (type === 'unknown' && previewUrl && url)) {
|
|
|
|
|
// Note: type: unknown might not have width/height
|
|
|
|
|
return (
|
|
|
|
|
<div
|
|
|
|
|
class={`media media-image`}
|
|
|
|
|
onClick={onClick}
|
|
|
|
|
style={
|
|
|
|
|
showOriginal && {
|
|
|
|
|
backgroundImage: `url(${previewUrl})`,
|
|
|
|
|
backgroundSize: 'contain',
|
|
|
|
|
backgroundRepeat: 'no-repeat',
|
|
|
|
|
backgroundPosition: 'center',
|
|
|
|
|
aspectRatio: `${width}/${height}`,
|
|
|
|
|
width,
|
|
|
|
|
height,
|
|
|
|
|
maxWidth: '100%',
|
|
|
|
|
maxHeight: '100%',
|
|
|
|
|
display: 'flex',
|
|
|
|
|
justifyContent: 'center',
|
|
|
|
|
alignItems: 'center',
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
>
|
|
|
|
|
<img
|
|
|
|
|
src={mediaURL}
|
|
|
|
|
alt={description}
|
|
|
|
|
width={width}
|
|
|
|
|
height={height}
|
2023-01-06 17:08:22 +03:00
|
|
|
|
loading={showOriginal ? 'eager' : 'lazy'}
|
2022-12-18 16:10:05 +03:00
|
|
|
|
style={
|
|
|
|
|
!showOriginal && {
|
|
|
|
|
backgroundColor:
|
|
|
|
|
rgbAverageColor && `rgb(${rgbAverageColor.join(',')})`,
|
|
|
|
|
backgroundPosition: focalBackgroundPosition || 'center',
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
|
|
|
|
);
|
|
|
|
|
} else if (type === 'gifv' || type === 'video') {
|
2023-01-07 09:45:04 +03:00
|
|
|
|
const shortDuration = original.duration < 31;
|
|
|
|
|
const isGIF = type === 'gifv' && shortDuration;
|
|
|
|
|
// If GIF is too long, treat it as a video
|
|
|
|
|
const loopable = original.duration < 61;
|
2023-01-02 09:21:38 +03:00
|
|
|
|
const formattedDuration = formatDuration(original.duration);
|
2023-01-06 13:25:47 +03:00
|
|
|
|
const hoverAnimate = !showOriginal && !autoAnimate && isGIF;
|
2023-01-07 09:45:04 +03:00
|
|
|
|
const autoGIFAnimate = !showOriginal && autoAnimate && isGIF;
|
2022-12-18 16:10:05 +03:00
|
|
|
|
return (
|
|
|
|
|
<div
|
2023-01-06 13:25:47 +03:00
|
|
|
|
class={`media media-${isGIF ? 'gif' : 'video'} ${
|
2023-01-07 09:45:04 +03:00
|
|
|
|
autoGIFAnimate ? 'media-contain' : ''
|
2023-01-06 13:25:47 +03:00
|
|
|
|
}`}
|
2023-01-02 09:21:38 +03:00
|
|
|
|
data-formatted-duration={formattedDuration}
|
2023-01-07 09:45:04 +03:00
|
|
|
|
data-label={isGIF && !showOriginal && !autoGIFAnimate ? 'GIF' : ''}
|
2022-12-18 16:10:05 +03:00
|
|
|
|
style={{
|
|
|
|
|
backgroundColor:
|
|
|
|
|
rgbAverageColor && `rgb(${rgbAverageColor.join(',')})`,
|
|
|
|
|
}}
|
|
|
|
|
onClick={(e) => {
|
2023-01-06 13:25:47 +03:00
|
|
|
|
if (hoverAnimate) {
|
2022-12-28 09:47:39 +03:00
|
|
|
|
try {
|
2022-12-25 20:09:19 +03:00
|
|
|
|
videoRef.current.pause();
|
2022-12-28 09:47:39 +03:00
|
|
|
|
} catch (e) {}
|
2022-12-18 16:10:05 +03:00
|
|
|
|
}
|
|
|
|
|
onClick(e);
|
|
|
|
|
}}
|
|
|
|
|
onMouseEnter={() => {
|
2023-01-06 13:25:47 +03:00
|
|
|
|
if (hoverAnimate) {
|
2022-12-18 16:10:05 +03:00
|
|
|
|
try {
|
2022-12-24 09:15:34 +03:00
|
|
|
|
videoRef.current.play();
|
2022-12-18 16:10:05 +03:00
|
|
|
|
} catch (e) {}
|
|
|
|
|
}
|
|
|
|
|
}}
|
|
|
|
|
onMouseLeave={() => {
|
2023-01-06 13:25:47 +03:00
|
|
|
|
if (hoverAnimate) {
|
2022-12-18 16:10:05 +03:00
|
|
|
|
try {
|
2022-12-24 09:15:34 +03:00
|
|
|
|
videoRef.current.pause();
|
2022-12-18 16:10:05 +03:00
|
|
|
|
} catch (e) {}
|
|
|
|
|
}
|
|
|
|
|
}}
|
|
|
|
|
>
|
2023-01-07 09:45:04 +03:00
|
|
|
|
{showOriginal || autoGIFAnimate ? (
|
2023-01-06 14:07:04 +03:00
|
|
|
|
<div
|
|
|
|
|
style={{
|
|
|
|
|
width: '100%',
|
|
|
|
|
height: '100%',
|
|
|
|
|
}}
|
|
|
|
|
dangerouslySetInnerHTML={{
|
|
|
|
|
__html: `
|
|
|
|
|
<video
|
|
|
|
|
src="${url}"
|
|
|
|
|
poster="${previewUrl}"
|
|
|
|
|
width="${width}"
|
|
|
|
|
height="${height}"
|
|
|
|
|
preload="auto"
|
|
|
|
|
autoplay
|
|
|
|
|
muted="${isGIF}"
|
2023-01-07 09:45:04 +03:00
|
|
|
|
${isGIF ? '' : 'controls'}
|
2023-01-06 14:07:04 +03:00
|
|
|
|
playsinline
|
|
|
|
|
loop="${loopable}"
|
2023-01-27 12:51:31 +03:00
|
|
|
|
${
|
|
|
|
|
isGIF
|
|
|
|
|
? 'ondblclick="this.paused ? this.play() : this.pause()"'
|
|
|
|
|
: ''
|
|
|
|
|
}
|
2023-01-06 14:07:04 +03:00
|
|
|
|
></video>
|
|
|
|
|
`,
|
|
|
|
|
}}
|
2022-12-28 09:47:39 +03:00
|
|
|
|
/>
|
2022-12-18 16:10:05 +03:00
|
|
|
|
) : isGIF ? (
|
|
|
|
|
<video
|
|
|
|
|
ref={videoRef}
|
|
|
|
|
src={url}
|
|
|
|
|
poster={previewUrl}
|
|
|
|
|
width={width}
|
|
|
|
|
height={height}
|
|
|
|
|
preload="auto"
|
|
|
|
|
// controls
|
|
|
|
|
playsinline
|
|
|
|
|
loop
|
|
|
|
|
muted
|
|
|
|
|
/>
|
|
|
|
|
) : (
|
|
|
|
|
<img
|
|
|
|
|
src={previewUrl}
|
|
|
|
|
alt={description}
|
|
|
|
|
width={width}
|
|
|
|
|
height={height}
|
|
|
|
|
loading="lazy"
|
|
|
|
|
/>
|
|
|
|
|
)}
|
|
|
|
|
</div>
|
|
|
|
|
);
|
|
|
|
|
} else if (type === 'audio') {
|
2023-01-08 20:17:16 +03:00
|
|
|
|
const formattedDuration = formatDuration(original.duration);
|
2022-12-18 16:10:05 +03:00
|
|
|
|
return (
|
2023-01-08 20:17:16 +03:00
|
|
|
|
<div
|
|
|
|
|
class="media media-audio"
|
|
|
|
|
data-formatted-duration={formattedDuration}
|
|
|
|
|
onClick={onClick}
|
|
|
|
|
>
|
|
|
|
|
{showOriginal ? (
|
|
|
|
|
<audio src={remoteUrl || url} preload="none" controls autoplay />
|
|
|
|
|
) : previewUrl ? (
|
|
|
|
|
<img
|
|
|
|
|
src={previewUrl}
|
|
|
|
|
alt={description}
|
|
|
|
|
width={width}
|
|
|
|
|
height={height}
|
|
|
|
|
loading="lazy"
|
|
|
|
|
/>
|
|
|
|
|
) : null}
|
2022-12-18 16:10:05 +03:00
|
|
|
|
</div>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-07 06:52:23 +03:00
|
|
|
|
function Card({ card }) {
|
2022-12-18 16:10:05 +03:00
|
|
|
|
const {
|
|
|
|
|
blurhash,
|
|
|
|
|
title,
|
|
|
|
|
description,
|
|
|
|
|
html,
|
|
|
|
|
providerName,
|
|
|
|
|
authorName,
|
|
|
|
|
width,
|
|
|
|
|
height,
|
|
|
|
|
image,
|
|
|
|
|
url,
|
|
|
|
|
type,
|
|
|
|
|
embedUrl,
|
|
|
|
|
} = card;
|
|
|
|
|
|
|
|
|
|
/* type
|
|
|
|
|
link = Link OEmbed
|
|
|
|
|
photo = Photo OEmbed
|
|
|
|
|
video = Video OEmbed
|
|
|
|
|
rich = iframe OEmbed. Not currently accepted, so won’t show up in practice.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
const hasText = title || providerName || authorName;
|
2022-12-29 06:34:29 +03:00
|
|
|
|
const isLandscape = width / height >= 1.2;
|
2023-01-07 06:52:23 +03:00
|
|
|
|
const size = isLandscape ? 'large' : '';
|
2022-12-18 16:10:05 +03:00
|
|
|
|
|
|
|
|
|
if (hasText && image) {
|
|
|
|
|
const domain = new URL(url).hostname.replace(/^www\./, '');
|
|
|
|
|
return (
|
|
|
|
|
<a
|
|
|
|
|
href={url}
|
|
|
|
|
target="_blank"
|
|
|
|
|
rel="nofollow noopener noreferrer"
|
2022-12-29 03:57:01 +03:00
|
|
|
|
class={`card link ${size}`}
|
2022-12-18 16:10:05 +03:00
|
|
|
|
>
|
2023-01-07 15:25:13 +03:00
|
|
|
|
<div class="card-image">
|
|
|
|
|
<img
|
|
|
|
|
src={image}
|
|
|
|
|
width={width}
|
|
|
|
|
height={height}
|
|
|
|
|
loading="lazy"
|
|
|
|
|
alt=""
|
|
|
|
|
onError={(e) => {
|
|
|
|
|
try {
|
|
|
|
|
e.target.style.display = 'none';
|
|
|
|
|
} catch (e) {}
|
|
|
|
|
}}
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
2022-12-18 16:10:05 +03:00
|
|
|
|
<div class="meta-container">
|
|
|
|
|
<p class="meta domain">{domain}</p>
|
|
|
|
|
<p
|
|
|
|
|
class="title"
|
|
|
|
|
dangerouslySetInnerHTML={{
|
|
|
|
|
__html: title,
|
|
|
|
|
}}
|
|
|
|
|
/>
|
|
|
|
|
<p class="meta">{description || providerName || authorName}</p>
|
|
|
|
|
</div>
|
|
|
|
|
</a>
|
|
|
|
|
);
|
|
|
|
|
} else if (type === 'photo') {
|
|
|
|
|
return (
|
|
|
|
|
<a
|
|
|
|
|
href={url}
|
|
|
|
|
target="_blank"
|
|
|
|
|
rel="nofollow noopener noreferrer"
|
|
|
|
|
class="card photo"
|
|
|
|
|
>
|
|
|
|
|
<img
|
|
|
|
|
src={embedUrl}
|
|
|
|
|
width={width}
|
|
|
|
|
height={height}
|
|
|
|
|
alt={title || description}
|
|
|
|
|
loading="lazy"
|
|
|
|
|
style={{
|
|
|
|
|
height: 'auto',
|
|
|
|
|
aspectRatio: `${width}/${height}`,
|
|
|
|
|
}}
|
|
|
|
|
/>
|
|
|
|
|
</a>
|
|
|
|
|
);
|
|
|
|
|
} else if (type === 'video') {
|
|
|
|
|
return (
|
|
|
|
|
<div
|
|
|
|
|
class="card video"
|
|
|
|
|
style={{
|
|
|
|
|
aspectRatio: `${width}/${height}`,
|
|
|
|
|
}}
|
|
|
|
|
dangerouslySetInnerHTML={{ __html: html }}
|
|
|
|
|
/>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-27 13:09:23 +03:00
|
|
|
|
function Poll({ poll, lang, readOnly, onUpdate = () => {} }) {
|
2022-12-18 16:10:05 +03:00
|
|
|
|
const [uiState, setUIState] = useState('default');
|
|
|
|
|
|
|
|
|
|
const {
|
|
|
|
|
expired,
|
|
|
|
|
expiresAt,
|
|
|
|
|
id,
|
|
|
|
|
multiple,
|
|
|
|
|
options,
|
|
|
|
|
ownVotes,
|
|
|
|
|
voted,
|
|
|
|
|
votersCount,
|
|
|
|
|
votesCount,
|
2022-12-21 14:29:37 +03:00
|
|
|
|
} = poll;
|
2022-12-18 16:10:05 +03:00
|
|
|
|
|
|
|
|
|
const expiresAtDate = !!expiresAt && new Date(expiresAt);
|
|
|
|
|
|
2022-12-22 16:52:59 +03:00
|
|
|
|
// Update poll at point of expiry
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
let timeout;
|
|
|
|
|
if (!expired && expiresAtDate) {
|
|
|
|
|
const ms = expiresAtDate.getTime() - Date.now() + 1; // +1 to give it a little buffer
|
|
|
|
|
if (ms > 0) {
|
|
|
|
|
timeout = setTimeout(() => {
|
|
|
|
|
setUIState('loading');
|
|
|
|
|
(async () => {
|
|
|
|
|
try {
|
2022-12-26 20:17:04 +03:00
|
|
|
|
const pollResponse = await masto.v1.polls.fetch(id);
|
2022-12-22 16:52:59 +03:00
|
|
|
|
onUpdate(pollResponse);
|
|
|
|
|
} catch (e) {
|
|
|
|
|
// Silent fail
|
|
|
|
|
}
|
|
|
|
|
setUIState('default');
|
|
|
|
|
})();
|
|
|
|
|
}, ms);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return () => {
|
|
|
|
|
clearTimeout(timeout);
|
|
|
|
|
};
|
|
|
|
|
}, [expired, expiresAtDate]);
|
|
|
|
|
|
2022-12-22 09:59:36 +03:00
|
|
|
|
const pollVotesCount = votersCount || votesCount;
|
|
|
|
|
let roundPrecision = 0;
|
|
|
|
|
if (pollVotesCount <= 1000) {
|
|
|
|
|
roundPrecision = 0;
|
|
|
|
|
} else if (pollVotesCount <= 10000) {
|
|
|
|
|
roundPrecision = 1;
|
|
|
|
|
} else if (pollVotesCount <= 100000) {
|
|
|
|
|
roundPrecision = 2;
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-18 16:10:05 +03:00
|
|
|
|
return (
|
2022-12-21 14:46:38 +03:00
|
|
|
|
<div
|
2022-12-27 13:09:23 +03:00
|
|
|
|
lang={lang}
|
2022-12-21 14:46:38 +03:00
|
|
|
|
class={`poll ${readOnly ? 'read-only' : ''} ${
|
|
|
|
|
uiState === 'loading' ? 'loading' : ''
|
|
|
|
|
}`}
|
|
|
|
|
>
|
2022-12-18 16:10:05 +03:00
|
|
|
|
{voted || expired ? (
|
|
|
|
|
options.map((option, i) => {
|
|
|
|
|
const { title, votesCount: optionVotesCount } = option;
|
2022-12-25 18:29:25 +03:00
|
|
|
|
const percentage = pollVotesCount
|
|
|
|
|
? ((optionVotesCount / pollVotesCount) * 100).toFixed(
|
|
|
|
|
roundPrecision,
|
|
|
|
|
)
|
|
|
|
|
: 0;
|
2022-12-18 16:10:05 +03:00
|
|
|
|
// check if current poll choice is the leading one
|
|
|
|
|
const isLeading =
|
|
|
|
|
optionVotesCount > 0 &&
|
|
|
|
|
optionVotesCount === Math.max(...options.map((o) => o.votesCount));
|
|
|
|
|
return (
|
|
|
|
|
<div
|
|
|
|
|
key={`${i}-${title}-${optionVotesCount}`}
|
|
|
|
|
class={`poll-option ${isLeading ? 'poll-option-leading' : ''}`}
|
|
|
|
|
style={{
|
|
|
|
|
'--percentage': `${percentage}%`,
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
<div class="poll-option-title">
|
|
|
|
|
{title}
|
|
|
|
|
{voted && ownVotes.includes(i) && (
|
|
|
|
|
<>
|
|
|
|
|
{' '}
|
2022-12-18 18:06:05 +03:00
|
|
|
|
<Icon icon="check-circle" />
|
2022-12-18 16:10:05 +03:00
|
|
|
|
</>
|
|
|
|
|
)}
|
|
|
|
|
</div>
|
|
|
|
|
<div
|
|
|
|
|
class="poll-option-votes"
|
|
|
|
|
title={`${optionVotesCount} vote${
|
|
|
|
|
optionVotesCount === 1 ? '' : 's'
|
|
|
|
|
}`}
|
|
|
|
|
>
|
|
|
|
|
{percentage}%
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
);
|
|
|
|
|
})
|
|
|
|
|
) : (
|
|
|
|
|
<form
|
|
|
|
|
onSubmit={async (e) => {
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
const form = e.target;
|
|
|
|
|
const formData = new FormData(form);
|
|
|
|
|
const votes = [];
|
|
|
|
|
formData.forEach((value, key) => {
|
|
|
|
|
if (key === 'poll') {
|
|
|
|
|
votes.push(value);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
console.log(votes);
|
|
|
|
|
setUIState('loading');
|
2022-12-26 20:17:04 +03:00
|
|
|
|
const pollResponse = await masto.v1.polls.vote(id, {
|
2022-12-18 16:10:05 +03:00
|
|
|
|
choices: votes,
|
|
|
|
|
});
|
|
|
|
|
console.log(pollResponse);
|
2022-12-21 14:29:37 +03:00
|
|
|
|
onUpdate(pollResponse);
|
2022-12-18 16:10:05 +03:00
|
|
|
|
setUIState('default');
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
{options.map((option, i) => {
|
|
|
|
|
const { title } = option;
|
|
|
|
|
return (
|
|
|
|
|
<div class="poll-option">
|
|
|
|
|
<label class="poll-label">
|
|
|
|
|
<input
|
|
|
|
|
type={multiple ? 'checkbox' : 'radio'}
|
|
|
|
|
name="poll"
|
|
|
|
|
value={i}
|
|
|
|
|
disabled={uiState === 'loading'}
|
|
|
|
|
readOnly={readOnly}
|
|
|
|
|
/>
|
|
|
|
|
<span class="poll-option-title">{title}</span>
|
|
|
|
|
</label>
|
|
|
|
|
</div>
|
|
|
|
|
);
|
|
|
|
|
})}
|
|
|
|
|
{!readOnly && (
|
2022-12-10 18:39:12 +03:00
|
|
|
|
<button
|
2022-12-18 16:10:05 +03:00
|
|
|
|
class="poll-vote-button"
|
|
|
|
|
type="submit"
|
|
|
|
|
disabled={uiState === 'loading'}
|
2022-12-10 18:39:12 +03:00
|
|
|
|
>
|
2022-12-18 16:10:05 +03:00
|
|
|
|
Vote
|
2022-12-10 18:39:12 +03:00
|
|
|
|
</button>
|
2022-12-10 12:14:48 +03:00
|
|
|
|
)}
|
2022-12-18 16:10:05 +03:00
|
|
|
|
</form>
|
2022-12-10 12:14:48 +03:00
|
|
|
|
)}
|
2022-12-18 16:10:05 +03:00
|
|
|
|
{!readOnly && (
|
|
|
|
|
<p class="poll-meta">
|
2022-12-21 14:46:38 +03:00
|
|
|
|
{!expired && (
|
|
|
|
|
<>
|
|
|
|
|
<button
|
|
|
|
|
type="button"
|
|
|
|
|
class="textual"
|
|
|
|
|
disabled={uiState === 'loading'}
|
|
|
|
|
onClick={(e) => {
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
setUIState('loading');
|
|
|
|
|
(async () => {
|
|
|
|
|
try {
|
2022-12-26 20:17:04 +03:00
|
|
|
|
const pollResponse = await masto.v1.polls.fetch(id);
|
2022-12-21 14:46:38 +03:00
|
|
|
|
onUpdate(pollResponse);
|
|
|
|
|
} catch (e) {
|
|
|
|
|
// Silent fail
|
|
|
|
|
}
|
|
|
|
|
setUIState('default');
|
|
|
|
|
})();
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
Refresh
|
|
|
|
|
</button>{' '}
|
|
|
|
|
•{' '}
|
|
|
|
|
</>
|
|
|
|
|
)}
|
2023-01-14 19:17:07 +03:00
|
|
|
|
<span title={votesCount}>{shortenNumber(votesCount)}</span> vote
|
|
|
|
|
{votesCount === 1 ? '' : 's'}
|
|
|
|
|
{!!votersCount && votersCount !== votesCount && (
|
2022-12-18 16:10:05 +03:00
|
|
|
|
<>
|
|
|
|
|
{' '}
|
2023-01-14 19:17:07 +03:00
|
|
|
|
•{' '}
|
|
|
|
|
<span title={votersCount}>{shortenNumber(votersCount)}</span>{' '}
|
|
|
|
|
voter
|
|
|
|
|
{votersCount === 1 ? '' : 's'}
|
2022-12-18 16:10:05 +03:00
|
|
|
|
</>
|
|
|
|
|
)}{' '}
|
|
|
|
|
• {expired ? 'Ended' : 'Ending'}{' '}
|
2023-01-05 05:50:27 +03:00
|
|
|
|
{!!expiresAtDate && <RelativeTime datetime={expiresAtDate} />}
|
2022-12-18 16:10:05 +03:00
|
|
|
|
</p>
|
|
|
|
|
)}
|
|
|
|
|
</div>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function EditedAtModal({ statusID, onClose = () => {} }) {
|
|
|
|
|
const [uiState, setUIState] = useState('default');
|
|
|
|
|
const [editHistory, setEditHistory] = useState([]);
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
setUIState('loading');
|
|
|
|
|
(async () => {
|
|
|
|
|
try {
|
2022-12-25 18:28:55 +03:00
|
|
|
|
const editHistory = await masto.v1.statuses.listHistory(statusID);
|
2022-12-18 16:10:05 +03:00
|
|
|
|
console.log(editHistory);
|
|
|
|
|
setEditHistory(editHistory);
|
|
|
|
|
setUIState('default');
|
|
|
|
|
} catch (e) {
|
|
|
|
|
console.error(e);
|
|
|
|
|
setUIState('error');
|
|
|
|
|
}
|
|
|
|
|
})();
|
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
|
|
const currentYear = new Date().getFullYear();
|
|
|
|
|
|
|
|
|
|
return (
|
2022-12-30 15:37:57 +03:00
|
|
|
|
<div id="edit-history" class="sheet">
|
2022-12-25 13:01:01 +03:00
|
|
|
|
<header>
|
|
|
|
|
{/* <button type="button" class="close-button plain large" onClick={onClose}>
|
2022-12-18 16:10:05 +03:00
|
|
|
|
<Icon icon="x" alt="Close" />
|
2022-12-21 15:00:45 +03:00
|
|
|
|
</button> */}
|
2022-12-25 13:01:01 +03:00
|
|
|
|
<h2>Edit History</h2>
|
|
|
|
|
{uiState === 'error' && <p>Failed to load history</p>}
|
|
|
|
|
{uiState === 'loading' && (
|
|
|
|
|
<p>
|
|
|
|
|
<Loader abrupt /> Loading…
|
|
|
|
|
</p>
|
|
|
|
|
)}
|
|
|
|
|
</header>
|
2022-12-30 15:37:57 +03:00
|
|
|
|
<main tabIndex="-1">
|
2022-12-25 13:01:01 +03:00
|
|
|
|
{editHistory.length > 0 && (
|
|
|
|
|
<ol>
|
|
|
|
|
{editHistory.map((status) => {
|
|
|
|
|
const { createdAt } = status;
|
|
|
|
|
const createdAtDate = new Date(createdAt);
|
|
|
|
|
return (
|
|
|
|
|
<li key={createdAt} class="history-item">
|
|
|
|
|
<h3>
|
|
|
|
|
<time>
|
|
|
|
|
{Intl.DateTimeFormat('en', {
|
|
|
|
|
// Show year if not current year
|
|
|
|
|
year:
|
|
|
|
|
createdAtDate.getFullYear() === currentYear
|
|
|
|
|
? undefined
|
|
|
|
|
: 'numeric',
|
|
|
|
|
month: 'short',
|
|
|
|
|
day: 'numeric',
|
|
|
|
|
weekday: 'short',
|
|
|
|
|
hour: 'numeric',
|
|
|
|
|
minute: '2-digit',
|
|
|
|
|
second: '2-digit',
|
|
|
|
|
}).format(createdAtDate)}
|
|
|
|
|
</time>
|
|
|
|
|
</h3>
|
|
|
|
|
<Status status={status} size="s" withinContext readOnly />
|
|
|
|
|
</li>
|
|
|
|
|
);
|
|
|
|
|
})}
|
|
|
|
|
</ol>
|
|
|
|
|
)}
|
|
|
|
|
</main>
|
2022-12-10 12:14:48 +03:00
|
|
|
|
</div>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-17 12:26:41 +03:00
|
|
|
|
function StatusButton({
|
|
|
|
|
checked,
|
|
|
|
|
count,
|
|
|
|
|
class: className,
|
|
|
|
|
title,
|
|
|
|
|
alt,
|
|
|
|
|
icon,
|
|
|
|
|
onClick,
|
|
|
|
|
...props
|
|
|
|
|
}) {
|
|
|
|
|
if (typeof title === 'string') {
|
|
|
|
|
title = [title, title];
|
|
|
|
|
}
|
|
|
|
|
if (typeof alt === 'string') {
|
|
|
|
|
alt = [alt, alt];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const [buttonTitle, setButtonTitle] = useState(title[0] || '');
|
|
|
|
|
const [iconAlt, setIconAlt] = useState(alt[0] || '');
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
if (checked) {
|
|
|
|
|
setButtonTitle(title[1] || '');
|
|
|
|
|
setIconAlt(alt[1] || '');
|
|
|
|
|
} else {
|
|
|
|
|
setButtonTitle(title[0] || '');
|
|
|
|
|
setIconAlt(alt[0] || '');
|
|
|
|
|
}
|
|
|
|
|
}, [checked, title, alt]);
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
<button
|
|
|
|
|
type="button"
|
|
|
|
|
title={buttonTitle}
|
|
|
|
|
class={`plain ${className} ${checked ? 'checked' : ''}`}
|
|
|
|
|
onClick={(e) => {
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
e.stopPropagation();
|
|
|
|
|
onClick(e);
|
|
|
|
|
}}
|
|
|
|
|
{...props}
|
|
|
|
|
>
|
|
|
|
|
<Icon icon={icon} size="l" alt={iconAlt} />
|
|
|
|
|
{!!count && (
|
|
|
|
|
<>
|
|
|
|
|
{' '}
|
2022-12-17 19:13:56 +03:00
|
|
|
|
<small title={count}>{shortenNumber(count)}</small>
|
2022-12-17 12:26:41 +03:00
|
|
|
|
</>
|
|
|
|
|
)}
|
|
|
|
|
</button>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-18 17:56:00 +03:00
|
|
|
|
function Carousel({ mediaAttachments, index = 0, onClose = () => {} }) {
|
|
|
|
|
const carouselRef = useRef(null);
|
|
|
|
|
|
|
|
|
|
const [currentIndex, setCurrentIndex] = useState(index);
|
|
|
|
|
const carouselFocusItem = useRef(null);
|
|
|
|
|
useLayoutEffect(() => {
|
2023-01-25 15:37:48 +03:00
|
|
|
|
carouselFocusItem.current?.scrollIntoView();
|
2022-12-18 17:56:00 +03:00
|
|
|
|
}, []);
|
|
|
|
|
|
2022-12-28 15:46:38 +03:00
|
|
|
|
const [showControls, setShowControls] = useState(true);
|
2022-12-28 14:31:08 +03:00
|
|
|
|
|
2022-12-28 14:43:02 +03:00
|
|
|
|
useEffect(() => {
|
|
|
|
|
let handleSwipe = () => {
|
2022-12-28 15:35:49 +03:00
|
|
|
|
onClose();
|
2022-12-28 14:43:02 +03:00
|
|
|
|
};
|
|
|
|
|
if (carouselRef.current) {
|
|
|
|
|
carouselRef.current.addEventListener('swiped-down', handleSwipe);
|
|
|
|
|
}
|
|
|
|
|
return () => {
|
|
|
|
|
if (carouselRef.current) {
|
|
|
|
|
carouselRef.current.removeEventListener('swiped-down', handleSwipe);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
}, []);
|
|
|
|
|
|
2022-12-30 16:36:14 +03:00
|
|
|
|
useHotkeys('esc', onClose, [onClose]);
|
|
|
|
|
|
2023-01-22 11:27:00 +03:00
|
|
|
|
const [showMediaAlt, setShowMediaAlt] = useState(false);
|
|
|
|
|
|
2023-01-25 15:37:48 +03:00
|
|
|
|
useEffect(() => {
|
|
|
|
|
let handleScroll = () => {
|
|
|
|
|
const { clientWidth, scrollLeft } = carouselRef.current;
|
|
|
|
|
const index = Math.round(scrollLeft / clientWidth);
|
|
|
|
|
setCurrentIndex(index);
|
|
|
|
|
};
|
|
|
|
|
if (carouselRef.current) {
|
|
|
|
|
carouselRef.current.addEventListener('scroll', handleScroll, {
|
|
|
|
|
passive: true,
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
return () => {
|
|
|
|
|
if (carouselRef.current) {
|
|
|
|
|
carouselRef.current.removeEventListener('scroll', handleScroll);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
}, []);
|
|
|
|
|
|
2022-12-18 17:56:00 +03:00
|
|
|
|
return (
|
|
|
|
|
<>
|
|
|
|
|
<div
|
|
|
|
|
ref={carouselRef}
|
2022-12-29 11:11:58 +03:00
|
|
|
|
tabIndex="-1"
|
2022-12-28 14:43:02 +03:00
|
|
|
|
data-swipe-threshold="44"
|
2022-12-18 17:56:00 +03:00
|
|
|
|
class="carousel"
|
|
|
|
|
onClick={(e) => {
|
|
|
|
|
if (
|
|
|
|
|
e.target.classList.contains('carousel-item') ||
|
|
|
|
|
e.target.classList.contains('media')
|
|
|
|
|
) {
|
|
|
|
|
onClose();
|
|
|
|
|
}
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
{mediaAttachments?.map((media, i) => {
|
|
|
|
|
const { blurhash } = media;
|
|
|
|
|
const rgbAverageColor = blurhash
|
|
|
|
|
? getBlurHashAverageColor(blurhash)
|
|
|
|
|
: null;
|
|
|
|
|
return (
|
2023-01-25 15:37:48 +03:00
|
|
|
|
<div
|
2022-12-18 17:56:00 +03:00
|
|
|
|
class="carousel-item"
|
|
|
|
|
style={{
|
2023-01-07 15:38:05 +03:00
|
|
|
|
'--average-color': `rgb(${rgbAverageColor?.join(',')})`,
|
|
|
|
|
'--average-color-alpha': `rgba(${rgbAverageColor?.join(
|
|
|
|
|
',',
|
|
|
|
|
)}, .5)`,
|
2022-12-18 17:56:00 +03:00
|
|
|
|
}}
|
|
|
|
|
tabindex="0"
|
|
|
|
|
key={media.id}
|
2023-01-25 15:37:48 +03:00
|
|
|
|
ref={i === currentIndex ? carouselFocusItem : null}
|
2022-12-28 14:31:08 +03:00
|
|
|
|
onClick={(e) => {
|
|
|
|
|
if (e.target !== e.currentTarget) {
|
|
|
|
|
setShowControls(!showControls);
|
|
|
|
|
}
|
|
|
|
|
}}
|
2022-12-18 17:56:00 +03:00
|
|
|
|
>
|
2023-01-22 11:27:00 +03:00
|
|
|
|
{!!media.description && (
|
|
|
|
|
<button
|
|
|
|
|
type="button"
|
|
|
|
|
class="plain2 media-alt"
|
|
|
|
|
hidden={!showControls}
|
|
|
|
|
onClick={() => {
|
|
|
|
|
setShowMediaAlt(media.description);
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
<span class="tag">ALT</span>{' '}
|
|
|
|
|
<span class="media-alt-desc">{media.description}</span>
|
|
|
|
|
</button>
|
|
|
|
|
)}
|
2022-12-18 17:56:00 +03:00
|
|
|
|
<Media media={media} showOriginal />
|
2023-01-25 15:37:48 +03:00
|
|
|
|
</div>
|
2022-12-18 17:56:00 +03:00
|
|
|
|
);
|
|
|
|
|
})}
|
|
|
|
|
</div>
|
2022-12-28 14:31:08 +03:00
|
|
|
|
<div class="carousel-top-controls" hidden={!showControls}>
|
2023-01-22 11:27:00 +03:00
|
|
|
|
<span>
|
|
|
|
|
<button
|
|
|
|
|
type="button"
|
2023-01-24 06:32:33 +03:00
|
|
|
|
class="carousel-button plain3"
|
2023-01-22 11:27:00 +03:00
|
|
|
|
onClick={() => onClose()}
|
|
|
|
|
>
|
|
|
|
|
<Icon icon="x" />
|
|
|
|
|
</button>
|
|
|
|
|
</span>
|
|
|
|
|
{mediaAttachments?.length > 1 ? (
|
|
|
|
|
<span class="carousel-dots">
|
|
|
|
|
{mediaAttachments?.map((media, i) => (
|
|
|
|
|
<button
|
|
|
|
|
key={media.id}
|
|
|
|
|
type="button"
|
|
|
|
|
disabled={i === currentIndex}
|
|
|
|
|
class={`plain carousel-dot ${
|
|
|
|
|
i === currentIndex ? 'active' : ''
|
|
|
|
|
}`}
|
|
|
|
|
onClick={(e) => {
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
e.stopPropagation();
|
2023-01-25 15:37:48 +03:00
|
|
|
|
carouselRef.current.scrollTo({
|
|
|
|
|
left: carouselRef.current.clientWidth * i,
|
|
|
|
|
behavior: 'smooth',
|
|
|
|
|
});
|
2023-01-22 11:27:00 +03:00
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
•
|
|
|
|
|
</button>
|
|
|
|
|
))}
|
|
|
|
|
</span>
|
|
|
|
|
) : (
|
|
|
|
|
<span />
|
|
|
|
|
)}
|
2023-01-01 07:28:54 +03:00
|
|
|
|
<span>
|
|
|
|
|
<a
|
|
|
|
|
href={
|
|
|
|
|
mediaAttachments[currentIndex]?.remoteUrl ||
|
|
|
|
|
mediaAttachments[currentIndex]?.url
|
|
|
|
|
}
|
|
|
|
|
target="_blank"
|
2023-01-24 06:32:33 +03:00
|
|
|
|
class="button carousel-button plain3"
|
2023-01-01 07:28:54 +03:00
|
|
|
|
title="Open original media in new window"
|
|
|
|
|
>
|
|
|
|
|
<Icon icon="popout" alt="Open original media in new window" />
|
|
|
|
|
</a>{' '}
|
|
|
|
|
</span>
|
2022-12-18 17:56:00 +03:00
|
|
|
|
</div>
|
|
|
|
|
{mediaAttachments?.length > 1 && (
|
2022-12-28 14:31:08 +03:00
|
|
|
|
<div class="carousel-controls" hidden={!showControls}>
|
2022-12-18 17:56:00 +03:00
|
|
|
|
<button
|
|
|
|
|
type="button"
|
2023-01-24 06:32:33 +03:00
|
|
|
|
class="carousel-button plain3"
|
2022-12-18 17:56:00 +03:00
|
|
|
|
hidden={currentIndex === 0}
|
|
|
|
|
onClick={(e) => {
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
e.stopPropagation();
|
2023-01-25 15:37:48 +03:00
|
|
|
|
carouselRef.current.scrollTo({
|
|
|
|
|
left: carouselRef.current.clientWidth * (currentIndex - 1),
|
|
|
|
|
behavior: 'smooth',
|
|
|
|
|
});
|
2022-12-18 17:56:00 +03:00
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
<Icon icon="arrow-left" />
|
|
|
|
|
</button>
|
|
|
|
|
<button
|
|
|
|
|
type="button"
|
2023-01-24 06:32:33 +03:00
|
|
|
|
class="carousel-button plain3"
|
2022-12-18 17:56:00 +03:00
|
|
|
|
hidden={currentIndex === mediaAttachments.length - 1}
|
|
|
|
|
onClick={(e) => {
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
e.stopPropagation();
|
2023-01-25 15:37:48 +03:00
|
|
|
|
carouselRef.current.scrollTo({
|
|
|
|
|
left: carouselRef.current.clientWidth * (currentIndex + 1),
|
|
|
|
|
behavior: 'smooth',
|
|
|
|
|
});
|
2022-12-18 17:56:00 +03:00
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
<Icon icon="arrow-right" />
|
|
|
|
|
</button>
|
|
|
|
|
</div>
|
|
|
|
|
)}
|
2023-01-22 11:27:00 +03:00
|
|
|
|
{!!showMediaAlt && (
|
|
|
|
|
<Modal
|
|
|
|
|
class="light"
|
|
|
|
|
onClick={(e) => {
|
|
|
|
|
if (e.target === e.currentTarget) {
|
|
|
|
|
setShowMediaAlt(false);
|
|
|
|
|
}
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
<div class="sheet">
|
|
|
|
|
<header>
|
|
|
|
|
<h2>Media description</h2>
|
|
|
|
|
</header>
|
|
|
|
|
<main>
|
2023-01-24 19:40:05 +03:00
|
|
|
|
<p
|
|
|
|
|
style={{
|
|
|
|
|
whiteSpace: 'pre-wrap',
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
{showMediaAlt}
|
|
|
|
|
</p>
|
2023-01-22 11:27:00 +03:00
|
|
|
|
</main>
|
|
|
|
|
</div>
|
|
|
|
|
</Modal>
|
|
|
|
|
)}
|
2022-12-18 17:56:00 +03:00
|
|
|
|
</>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-02 09:21:38 +03:00
|
|
|
|
function formatDuration(time) {
|
|
|
|
|
if (!time) return;
|
|
|
|
|
let hours = Math.floor(time / 3600);
|
|
|
|
|
let minutes = Math.floor((time % 3600) / 60);
|
|
|
|
|
let seconds = Math.round(time % 60);
|
|
|
|
|
|
|
|
|
|
if (hours === 0) {
|
|
|
|
|
return `${minutes}:${seconds.toString().padStart(2, '0')}`;
|
|
|
|
|
} else {
|
|
|
|
|
return `${hours}:${minutes.toString().padStart(2, '0')}:${seconds
|
|
|
|
|
.toString()
|
|
|
|
|
.padStart(2, '0')}`;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-07 15:26:23 +03:00
|
|
|
|
export default memo(Status);
|