pleroma/app/soapbox/features/ui/components/pending-status.tsx

101 lines
3.2 KiB
TypeScript
Raw Normal View History

import classNames from 'clsx';
import React from 'react';
import Account from 'soapbox/components/account';
2022-11-15 08:00:49 -08:00
import StatusContent from 'soapbox/components/status-content';
import StatusReplyMentions from 'soapbox/components/status-reply-mentions';
2022-08-22 09:11:01 -07:00
import { Card, HStack } from 'soapbox/components/ui';
2022-11-15 11:00:40 -08:00
import PlaceholderCard from 'soapbox/features/placeholder/components/placeholder-card';
import PlaceholderMediaGallery from 'soapbox/features/placeholder/components/placeholder-media-gallery';
import QuotedStatus from 'soapbox/features/status/containers/quoted-status-container';
import { useAppSelector } from 'soapbox/hooks';
2022-11-16 05:32:32 -08:00
import { buildStatus } from '../util/pending-status-builder';
2022-11-16 05:32:32 -08:00
import PollPreview from './poll-preview';
2022-06-16 08:47:57 -07:00
import type { Account as AccountEntity, Status as StatusEntity } from 'soapbox/types/entities';
const shouldHaveCard = (pendingStatus: StatusEntity) => {
return Boolean(pendingStatus.content.match(/https?:\/\/\S*/));
};
interface IPendingStatus {
className?: string,
idempotencyKey: string,
muted?: boolean,
2022-08-22 09:11:01 -07:00
thread?: boolean,
}
interface IPendingStatusMedia {
status: StatusEntity,
}
const PendingStatusMedia: React.FC<IPendingStatusMedia> = ({ status }) => {
if (status.media_attachments && !status.media_attachments.isEmpty()) {
return (
<PlaceholderMediaGallery
media={status.media_attachments}
/>
);
} else if (!status.quote && shouldHaveCard(status)) {
return <PlaceholderCard />;
} else {
return null;
}
};
2022-08-22 09:11:01 -07:00
const PendingStatus: React.FC<IPendingStatus> = ({ idempotencyKey, className, muted, thread = false }) => {
const status = useAppSelector((state) => {
const pendingStatus = state.pending_statuses.get(idempotencyKey);
return pendingStatus ? buildStatus(state, pendingStatus, idempotencyKey) : null;
}) as StatusEntity | null;
if (!status) return null;
if (!status.account) return null;
const account = status.account as AccountEntity;
return (
<div className={classNames('opacity-50', className)}>
<div className={classNames('status', { 'status-reply': !!status.in_reply_to_id, muted })} data-id={status.id}>
2022-08-22 09:11:01 -07:00
<Card
className={classNames('py-6 sm:p-5', `status-${status.visibility}`, { 'status-reply': !!status.in_reply_to_id })}
variant={thread ? 'default' : 'rounded'}
>
<div className='mb-4'>
<HStack justifyContent='between' alignItems='start'>
<Account
key={account.id}
account={account}
timestamp={status.created_at}
hideActions
/>
</HStack>
</div>
<div className='status__content-wrapper'>
<StatusReplyMentions status={status} />
<StatusContent
status={status}
collapsable
/>
<PendingStatusMedia status={status} />
2022-06-16 08:08:57 -07:00
{status.poll && <PollPreview pollId={status.poll as string} />}
2022-06-07 12:48:24 -07:00
{status.quote && <QuotedStatus statusId={status.quote as string} />}
</div>
{/* TODO */}
{/* <PlaceholderActionBar /> */}
2022-08-22 09:11:01 -07:00
</Card>
</div>
</div>
);
};
export default PendingStatus;