lbry-desktop/ui/component/claimPreview/view.jsx

356 lines
12 KiB
React
Raw Normal View History

2019-06-11 14:10:58 -04:00
// @flow
import type { Node } from 'react';
import React, { useEffect, forwardRef } from 'react';
import { NavLink, withRouter } from 'react-router-dom';
2019-06-11 14:10:58 -04:00
import classnames from 'classnames';
2021-02-17 15:28:58 +08:00
import { parseURI } from 'lbry-redux';
import { formatLbryUrlForWeb } from 'util/url';
2019-08-25 12:42:25 +01:00
import { isEmpty } from 'util/object';
2020-01-06 13:32:35 -05:00
import FileThumbnail from 'component/fileThumbnail';
2019-06-11 14:10:58 -04:00
import UriIndicator from 'component/uriIndicator';
import FileProperties from 'component/fileProperties';
2019-06-19 01:05:43 -04:00
import ClaimTags from 'component/claimTags';
2019-06-11 14:10:58 -04:00
import SubscribeButton from 'component/subscribeButton';
import ChannelThumbnail from 'component/channelThumbnail';
2020-02-12 13:59:48 -05:00
import ClaimSupportButton from 'component/claimSupportButton';
import useGetThumbnail from 'effects/use-get-thumbnail';
2020-01-30 17:25:15 -05:00
import ClaimPreviewTitle from 'component/claimPreviewTitle';
2020-01-31 11:12:47 -05:00
import ClaimPreviewSubtitle from 'component/claimPreviewSubtitle';
2020-01-31 11:43:14 -05:00
import ClaimRepostAuthor from 'component/claimRepostAuthor';
2020-02-04 21:54:19 -05:00
import FileDownloadLink from 'component/fileDownloadLink';
import AbandonedChannelPreview from 'component/abandonedChannelPreview';
2020-05-07 08:22:55 -04:00
import PublishPending from 'component/publishPending';
import ClaimMenuList from 'component/claimMenuList';
import ClaimPreviewLoading from './claim-preview-loading';
import ClaimPreviewHidden from './claim-preview-no-mature';
import ClaimPreviewNoContent from './claim-preview-no-content';
import { ENABLE_NO_SOURCE_CLAIMS } from 'config';
2019-06-11 14:10:58 -04:00
type Props = {
uri: string,
claim: ?Claim,
obscureNsfw: boolean,
showUserBlocked: boolean,
2019-06-11 14:10:58 -04:00
claimIsMine: boolean,
pending?: boolean,
reflectingProgress?: any, // fxme
resolveUri: (string) => void,
2019-06-11 14:10:58 -04:00
isResolvingUri: boolean,
history: { push: (string) => void },
2019-06-11 14:10:58 -04:00
title: string,
nsfw: boolean,
2019-07-21 17:31:22 -04:00
placeholder: string,
2019-06-19 01:05:43 -04:00
type: string,
2019-07-11 14:06:25 -04:00
hasVisitedUri: boolean,
blackListedOutpoints: Array<{
txid: string,
nout: number,
}>,
filteredOutpoints: Array<{
txid: string,
nout: number,
}>,
mutedUris: Array<string>,
blockedUris: Array<string>,
channelIsBlocked: boolean,
actions: boolean | Node | string | number,
properties: boolean | Node | string | number | ((Claim) => Node),
empty?: Node,
onClick?: (any) => any,
streamingUrl: ?string,
getFile: (string) => void,
customShouldHide?: (Claim) => boolean,
2020-02-05 13:01:07 -05:00
showUnresolvedClaim?: boolean,
showNullPlaceholder?: boolean,
2020-02-12 13:59:48 -05:00
includeSupportAction?: boolean,
hideActions?: boolean,
renderActions?: (Claim) => ?Node,
wrapperElement?: string,
hideRepostLabel?: boolean,
repostUrl?: string,
hideMenu?: boolean,
isLivestream?: boolean,
2019-06-11 14:10:58 -04:00
};
2019-08-02 02:28:14 -04:00
const ClaimPreview = forwardRef<any, {}>((props: Props, ref: any) => {
2019-06-11 14:10:58 -04:00
const {
// core
2019-06-11 14:10:58 -04:00
uri,
claim,
2019-06-11 14:10:58 -04:00
isResolvingUri,
// core actions
getFile,
2019-06-11 14:10:58 -04:00
resolveUri,
// claim properties
// is the claim consider nsfw?
nsfw,
claimIsMine,
streamingUrl,
// user properties
channelIsBlocked,
2019-07-11 14:06:25 -04:00
hasVisitedUri,
// component
history,
wrapperElement,
type,
placeholder,
// pending
reflectingProgress,
pending,
empty,
// modifiers
customShouldHide,
showNullPlaceholder,
// value from show mature content user setting
// true if the user doesn't wanna see nsfw content
obscureNsfw,
showUserBlocked,
showUnresolvedClaim,
hideRepostLabel = false,
hideActions = false,
properties,
onClick,
actions,
mutedUris,
blockedUris,
blackListedOutpoints,
filteredOutpoints,
2020-02-12 13:59:48 -05:00
includeSupportAction,
renderActions,
hideMenu = false,
// repostUrl,
isLivestream,
2019-06-11 14:10:58 -04:00
} = props;
const WrapperElement = wrapperElement || 'li';
2019-10-03 17:40:54 -04:00
const shouldFetch =
claim === undefined || (claim !== null && claim.value_type === 'channel' && isEmpty(claim.meta) && !pending);
2019-07-21 17:31:22 -04:00
const abandoned = !isResolvingUri && !claim;
const shouldHideActions = hideActions || type === 'small' || type === 'tooltip';
const canonicalUrl = claim && claim.canonical_url;
let isValid = false;
if (uri) {
try {
parseURI(uri);
isValid = true;
} catch (e) {
isValid = false;
}
2019-07-11 15:03:31 -04:00
}
2021-01-13 10:44:44 -05:00
const isRepost = claim && claim.repost_url;
2019-07-05 14:11:55 -04:00
2021-01-13 10:44:44 -05:00
const contentUri = claim && isRepost ? claim.canonical_url || claim.permanent_url : uri;
const isChannelUri = isValid ? parseURI(contentUri).isChannel : false;
2019-07-29 10:12:53 -04:00
const signingChannel = claim && claim.signing_channel;
2020-02-05 01:54:49 +02:00
const navigateUrl = formatLbryUrlForWeb((claim && claim.canonical_url) || uri || '/');
const navLinkProps = {
to: navigateUrl,
onClick: (e) => e.stopPropagation(),
};
// do not block abandoned and nsfw claims if showUserBlocked is passed
2019-10-17 12:58:28 -04:00
let shouldHide =
placeholder !== 'loading' &&
!showUserBlocked &&
((abandoned && !showUnresolvedClaim) || (!claimIsMine && obscureNsfw && nsfw));
2019-07-05 14:11:55 -04:00
// This will be replaced once blocking is done at the wallet server level
if (claim && !claimIsMine && !shouldHide && blackListedOutpoints) {
shouldHide = blackListedOutpoints.some(
(outpoint) =>
(signingChannel && outpoint.txid === signingChannel.txid && outpoint.nout === signingChannel.nout) ||
(outpoint.txid === claim.txid && outpoint.nout === claim.nout)
);
}
// We're checking to see if the stream outpoint
// or signing channel outpoint is in the filter list
if (claim && !claimIsMine && !shouldHide && filteredOutpoints) {
shouldHide = filteredOutpoints.some(
(outpoint) =>
(signingChannel && outpoint.txid === signingChannel.txid && outpoint.nout === signingChannel.nout) ||
(outpoint.txid === claim.txid && outpoint.nout === claim.nout)
);
2019-07-05 14:11:55 -04:00
}
// block stream claims
if (claim && !shouldHide && !showUserBlocked && mutedUris.length && signingChannel) {
shouldHide = mutedUris.some((blockedUri) => blockedUri === signingChannel.permanent_url);
}
if (claim && !shouldHide && !showUserBlocked && blockedUris.length && signingChannel) {
shouldHide = blockedUris.some((blockedUri) => blockedUri === signingChannel.permanent_url);
}
2019-06-11 14:10:58 -04:00
2020-01-08 11:36:49 -05:00
if (!shouldHide && customShouldHide && claim) {
if (customShouldHide(claim)) {
shouldHide = true;
}
}
2020-01-13 14:12:27 -05:00
// Weird placement warning
// Make sure this happens after we figure out if this claim needs to be hidden
const thumbnailUrl = useGetThumbnail(contentUri, claim, streamingUrl, getFile, shouldHide);
2020-01-13 14:12:27 -05:00
function handleOnClick(e) {
if (onClick) {
onClick(e);
}
if (claim && !pending) {
2020-02-05 01:54:49 +02:00
history.push(navigateUrl);
2019-06-11 14:10:58 -04:00
}
}
useEffect(() => {
2020-12-28 14:26:55 -05:00
if (isValid && !isResolvingUri && shouldFetch && uri) {
2019-06-11 14:10:58 -04:00
resolveUri(uri);
}
}, [isValid, uri, isResolvingUri, shouldFetch, resolveUri]);
2019-06-11 14:10:58 -04:00
if ((shouldHide && !showNullPlaceholder) || (isLivestream && !ENABLE_NO_SOURCE_CLAIMS)) {
2019-06-11 14:10:58 -04:00
return null;
}
if (placeholder === 'loading' || (uri && !claim && isResolvingUri)) {
2021-01-13 10:44:44 -05:00
return <ClaimPreviewLoading isChannel={isChannelUri} type={type} />;
}
if (claim && showNullPlaceholder && shouldHide && nsfw && obscureNsfw) {
return (
2021-01-13 10:44:44 -05:00
<ClaimPreviewHidden
message={__('Mature content hidden by your preferences')}
isChannel={isChannelUri}
type={type}
/>
);
}
if (claim && showNullPlaceholder && shouldHide) {
2021-01-13 10:44:44 -05:00
return <ClaimPreviewHidden message={__('This content is hidden')} isChannel={isChannelUri} type={type} />;
2019-06-11 14:10:58 -04:00
}
if (!claim && (showNullPlaceholder || empty)) {
2021-01-13 10:44:44 -05:00
return empty || <ClaimPreviewNoContent isChannel={isChannelUri} type={type} />;
}
2020-02-05 13:01:07 -05:00
if (!shouldFetch && showUnresolvedClaim && !isResolvingUri && claim === null) {
return <AbandonedChannelPreview uri={contentUri} type />;
}
if (placeholder === 'publish' && !claim && contentUri.startsWith('lbry://@')) {
return null;
}
2019-06-11 14:10:58 -04:00
return (
<WrapperElement
2019-08-02 02:28:14 -04:00
ref={ref}
2019-06-11 14:10:58 -04:00
role="link"
onClick={pending || type === 'inline' ? undefined : handleOnClick}
2020-01-31 11:43:14 -05:00
className={classnames('claim-preview__wrapper', {
2021-01-13 10:44:44 -05:00
'claim-preview__wrapper--channel': isChannelUri && type !== 'inline',
2020-01-31 11:43:14 -05:00
'claim-preview__wrapper--inline': type === 'inline',
'claim-preview__wrapper--small': type === 'small',
2019-06-11 14:10:58 -04:00
})}
>
2021-03-29 19:05:18 -04:00
<>
{!hideRepostLabel && <ClaimRepostAuthor uri={uri} />}
2019-06-11 14:10:58 -04:00
2021-03-29 19:05:18 -04:00
<div
className={classnames('claim-preview', {
'claim-preview--small': type === 'small' || type === 'tooltip',
'claim-preview--large': type === 'large',
'claim-preview--inline': type === 'inline',
'claim-preview--tooltip': type === 'tooltip',
'claim-preview--channel': isChannelUri,
'claim-preview--visited': !isChannelUri && !claimIsMine && hasVisitedUri,
'claim-preview--pending': pending,
})}
>
{isChannelUri && claim ? (
<UriIndicator uri={contentUri} link>
<ChannelThumbnail uri={contentUri} />
</UriIndicator>
) : (
<>
{!pending ? (
2020-05-07 08:22:55 -04:00
<NavLink {...navLinkProps}>
2021-03-29 19:05:18 -04:00
<FileThumbnail thumbnail={thumbnailUrl}>
{/* @if TARGET='app' */}
{claim && (
<div className="claim-preview__hover-actions">
<FileDownloadLink uri={canonicalUrl} hideOpenButton hideDownloadStatus />
</div>
)}
{/* @endif */}
{!isRepost && !isChannelUri && !isLivestream && (
<div className="claim-preview__file-property-overlay">
<FileProperties uri={contentUri} small />
</div>
)}
</FileThumbnail>
2020-05-07 08:22:55 -04:00
</NavLink>
2021-03-29 19:05:18 -04:00
) : (
<FileThumbnail thumbnail={thumbnailUrl} />
2020-05-07 08:22:55 -04:00
)}
2021-03-29 19:05:18 -04:00
</>
)}
<div className="claim-preview__text">
<div className="claim-preview-metadata">
<div className="claim-preview-info">
{pending ? (
<ClaimPreviewTitle uri={contentUri} />
) : (
<NavLink {...navLinkProps}>
<ClaimPreviewTitle uri={uri} />
</NavLink>
)}
</div>
<ClaimPreviewSubtitle uri={uri} type={type} />
{(pending || !!reflectingProgress) && <PublishPending uri={uri} />}
2020-01-31 11:43:14 -05:00
</div>
2021-03-29 19:05:18 -04:00
{type !== 'small' && (
<div className="claim-preview__actions">
{!pending && (
<>
{renderActions && claim && renderActions(claim)}
{shouldHideActions || renderActions ? null : actions !== undefined ? (
actions
) : (
<div className="claim-preview__primary-actions">
{!isChannelUri && signingChannel && (
<div className="claim-preview__channel-staked">
<ChannelThumbnail uri={signingChannel.permanent_url} />
</div>
)}
2021-03-29 19:05:18 -04:00
{isChannelUri && !channelIsBlocked && !claimIsMine && (
<SubscribeButton
uri={contentUri.startsWith('lbry://') ? contentUri : `lbry://${contentUri}`}
/>
)}
2021-03-29 19:05:18 -04:00
{includeSupportAction && <ClaimSupportButton uri={uri} />}
</div>
)}
</>
)}
{claim && (
<React.Fragment>
{typeof properties === 'function' ? (
properties(claim)
) : properties !== undefined ? (
properties
) : (
<ClaimTags uri={uri} type={type} />
)}
</React.Fragment>
)}
</div>
)}
</div>
2019-06-11 14:10:58 -04:00
</div>
2021-03-29 19:05:18 -04:00
{!hideMenu && <ClaimMenuList uri={uri} />}
</>
</WrapperElement>
2019-06-11 14:10:58 -04:00
);
2019-08-02 02:28:14 -04:00
});
2019-06-11 14:10:58 -04:00
export default withRouter(ClaimPreview);