61 lines
1.6 KiB
TypeScript
61 lines
1.6 KiB
TypeScript
import React, { useEffect, useState } from 'react';
|
|
import { useHistory } from 'react-router-dom';
|
|
|
|
import { fetchStatus } from 'soapbox/actions/statuses';
|
|
import MissingIndicator from 'soapbox/components/missing_indicator';
|
|
import Status from 'soapbox/components/status';
|
|
import { Spinner } from 'soapbox/components/ui';
|
|
import { useAppDispatch, useAppSelector } from 'soapbox/hooks';
|
|
import { makeGetStatus } from 'soapbox/selectors';
|
|
|
|
interface IEmbeddedStatus {
|
|
params: {
|
|
statusId: string,
|
|
},
|
|
}
|
|
|
|
const getStatus = makeGetStatus();
|
|
|
|
/** Status to be presented in an iframe for embeds on external websites. */
|
|
const EmbeddedStatus: React.FC<IEmbeddedStatus> = ({ params }) => {
|
|
const dispatch = useAppDispatch();
|
|
const history = useHistory();
|
|
const status = useAppSelector(state => getStatus(state, { id: params.statusId }));
|
|
|
|
const [loading, setLoading] = useState(true);
|
|
|
|
// Prevent navigation for UX and security.
|
|
// https://stackoverflow.com/a/71531211
|
|
history.block();
|
|
|
|
useEffect(() => {
|
|
dispatch(fetchStatus(params.statusId))
|
|
.then(() => setLoading(false))
|
|
.catch(() => setLoading(false));
|
|
}, []);
|
|
|
|
const renderInner = () => {
|
|
if (loading) {
|
|
return <Spinner />;
|
|
} else if (status) {
|
|
return <Status status={status} variant='default' />;
|
|
} else {
|
|
return <MissingIndicator nested />;
|
|
}
|
|
};
|
|
|
|
return (
|
|
<a
|
|
className='bg-white dark:bg-gray-800'
|
|
href={status?.url || '#'}
|
|
onClick={e => e.stopPropagation()}
|
|
target='_blank'
|
|
>
|
|
<div className='p-4 sm:p-6 max-w-3xl pointer-events-none'>
|
|
{renderInner()}
|
|
</div>
|
|
</a>
|
|
);
|
|
};
|
|
|
|
export default EmbeddedStatus;
|