2019-08-13 01:35:13 -04:00
// @flow
// This component is entirely for triggering the start of a file view
2020-04-13 19:48:11 -04:00
// The actual viewer for a file exists in TextViewer and FileRenderFloating
2019-08-13 01:35:13 -04:00
// They can't exist in one component because we need to handle/listen for the start of a new file view
// while a file is currently being viewed
2022-02-01 17:30:57 -03:00
import { useIsMobile } from 'effects/use-screensize' ;
2022-02-01 17:29:00 -03:00
import React from 'react' ;
2019-08-13 01:35:13 -04:00
import classnames from 'classnames' ;
2020-04-01 14:43:50 -04:00
import * as PAGES from 'constants/pages' ;
import * as RENDER _MODES from 'constants/file_render_modes' ;
2019-08-13 01:35:13 -04:00
import Button from 'component/button' ;
2021-06-25 12:15:17 +08:00
import { getThumbnailCdnUrl } from 'util/thumbnail' ;
2020-04-01 14:43:50 -04:00
import Nag from 'component/common/nag' ;
2021-06-25 12:15:17 +08:00
// $FlowFixMe cannot resolve ...
import FileRenderPlaceholder from 'static/img/fileRenderPlaceholder.png' ;
2022-02-01 17:29:00 -03:00
import * as COLLECTIONS _CONSTS from 'constants/collections' ;
2019-08-13 01:35:13 -04:00
type Props = {
isPlaying : boolean ,
fileInfo : FileListItem ,
uri : string ,
2021-03-11 12:08:11 -05:00
history : { push : ( string ) => void } ,
2021-09-10 14:27:21 -03:00
location : { search : ? string , pathname : string , href : string , state : { forceAutoplay : boolean } } ,
2019-08-13 01:35:13 -04:00
obscurePreview : boolean ,
insufficientCredits : boolean ,
2021-06-25 12:15:17 +08:00
claimThumbnail ? : string ,
2019-08-13 01:35:13 -04:00
autoplay : boolean ,
2019-08-28 19:33:38 -04:00
costInfo : any ,
2020-01-06 13:32:35 -05:00
inline : boolean ,
2020-04-01 14:43:50 -04:00
renderMode : string ,
2020-05-21 11:38:28 -04:00
claimWasPurchased : boolean ,
authenticated : boolean ,
2021-01-08 10:21:27 -05:00
videoTheaterMode : boolean ,
2022-02-01 17:30:57 -03:00
activeLivestreamForChannel ? : any ,
claimId ? : string ,
2022-02-01 17:29:00 -03:00
doUriInitiatePlay : ( uri : string , collectionId : ? string , isPlayable : boolean ) => void ,
2022-02-01 17:30:57 -03:00
doSetPlayingUri : ( { uri : ? string } ) => void ,
2019-08-13 01:35:13 -04:00
} ;
2020-04-01 14:43:50 -04:00
export default function FileRenderInitiator ( props : Props ) {
2019-08-13 23:09:25 -04:00
const {
isPlaying ,
fileInfo ,
uri ,
obscurePreview ,
insufficientCredits ,
2020-04-01 14:43:50 -04:00
history ,
2020-05-21 11:38:28 -04:00
location ,
2021-06-25 12:15:17 +08:00
claimThumbnail ,
2022-02-01 17:29:00 -03:00
autoplay ,
2020-04-01 14:43:50 -04:00
renderMode ,
2019-08-28 19:33:38 -04:00
costInfo ,
2020-05-21 11:38:28 -04:00
claimWasPurchased ,
authenticated ,
2021-01-08 10:21:27 -05:00
videoTheaterMode ,
2022-02-01 17:30:57 -03:00
activeLivestreamForChannel ,
claimId ,
2022-02-01 17:29:00 -03:00
doUriInitiatePlay ,
2022-02-01 17:30:57 -03:00
doSetPlayingUri ,
2019-08-13 23:09:25 -04:00
} = props ;
2021-05-15 07:59:21 +02:00
2022-02-01 17:29:00 -03:00
const containerRef = React . useRef < any > ( ) ;
2022-02-01 17:30:57 -03:00
const isMobile = useIsMobile ( ) ;
2022-02-01 17:29:00 -03:00
const [ thumbnail , setThumbnail ] = React . useState ( FileRenderPlaceholder ) ;
const { search , href , state : locationState } = location ;
const urlParams = search && new URLSearchParams ( search ) ;
const collectionId = urlParams && urlParams . get ( COLLECTIONS _CONSTS . COLLECTION _ID ) ;
2021-09-10 14:27:21 -03:00
// check if there is a time or autoplay parameter, if so force autoplay
2022-02-01 17:29:00 -03:00
const urlTimeParam = href && href . indexOf ( 't=' ) > - 1 ;
const forceAutoplayParam = locationState && locationState . forceAutoplay ;
const shouldAutoplay = forceAutoplayParam || urlTimeParam || autoplay ;
2021-05-15 07:59:21 +02:00
2021-09-10 14:27:21 -03:00
const isFree = costInfo && costInfo . cost === 0 ;
const canViewFile = isFree || claimWasPurchased ;
2020-04-01 14:43:50 -04:00
const isPlayable = RENDER _MODES . FLOATING _MODES . includes ( renderMode ) ;
2021-03-11 12:08:11 -05:00
const isText = RENDER _MODES . TEXT _MODES . includes ( renderMode ) ;
2022-02-01 17:30:57 -03:00
const isCurrentClaimLive = activeLivestreamForChannel && claimId && activeLivestreamForChannel . claimId === claimId ;
const isMobileClaimLive = isMobile && isCurrentClaimLive ;
const foundCover = thumbnail !== FileRenderPlaceholder ;
2021-06-25 12:15:17 +08:00
2022-02-01 17:29:00 -03:00
const renderUnsupported = RENDER _MODES . UNSUPPORTED _IN _THIS _APP . includes ( renderMode ) ;
const disabled = renderUnsupported || ( ! fileInfo && insufficientCredits && ! claimWasPurchased ) ;
const shouldRedirect = ! authenticated && ! isFree ;
2019-08-13 01:35:13 -04:00
2022-02-01 17:30:57 -03:00
React . useEffect ( ( ) => {
// Set livestream as playing uri so it can be rendered by <FileRenderMobile />
// instead of showing an empty cover image. Needs cover to fill the space with the player.
if ( isMobileClaimLive && foundCover ) {
doSetPlayingUri ( { uri } ) ;
}
2022-02-07 13:51:07 -03:00
} , [ doSetPlayingUri , foundCover , isMobileClaimLive , uri ] ) ;
2022-02-01 17:30:57 -03:00
2020-05-21 11:38:28 -04:00
function doAuthRedirect ( ) {
history . push ( ` / $ / ${ PAGES . AUTH } ?redirect= ${ encodeURIComponent ( location . pathname ) } ` ) ;
}
2022-02-01 17:29:00 -03:00
React . useEffect ( ( ) => {
if ( ! claimThumbnail ) return ;
2019-08-13 01:35:13 -04:00
2022-02-01 17:29:00 -03:00
setTimeout ( ( ) => {
let newThumbnail = claimThumbnail ;
2019-08-13 01:35:13 -04:00
2022-02-01 17:29:00 -03:00
if (
containerRef . current &&
containerRef . current . parentElement &&
containerRef . current . parentElement . offsetWidth
) {
const w = containerRef . current . parentElement . offsetWidth ;
newThumbnail = getThumbnailCdnUrl ( { thumbnail : newThumbnail , width : w , height : w } ) ;
}
2019-08-13 01:35:13 -04:00
2022-02-01 17:29:00 -03:00
if ( newThumbnail !== thumbnail ) {
setThumbnail ( newThumbnail ) ;
2019-08-13 01:35:13 -04:00
}
2022-02-01 17:29:00 -03:00
} , 200 ) ;
} , [ claimThumbnail , thumbnail ] ) ;
2019-08-13 01:35:13 -04:00
2022-02-01 17:29:00 -03:00
// Wrap this in useCallback because we need to use it to the view effect
// If we don't a new instance will be created for every render and react will think the dependencies have changed, which will add/remove the listener for every render
const viewFile = React . useCallback ( ( ) => {
doUriInitiatePlay ( uri , collectionId , isPlayable ) ;
} , [ collectionId , doUriInitiatePlay , isPlayable , uri ] ) ;
2019-08-13 01:35:13 -04:00
2022-02-01 17:29:00 -03:00
React . useEffect ( ( ) => {
2019-08-13 01:35:13 -04:00
const videoOnPage = document . querySelector ( 'video' ) ;
2022-02-01 17:29:00 -03:00
2021-05-24 16:46:47 +08:00
if (
2021-09-10 14:27:21 -03:00
( canViewFile || forceAutoplayParam ) &&
2022-02-01 17:29:00 -03:00
( ( shouldAutoplay && ( ! videoOnPage || forceAutoplayParam ) && isPlayable ) ||
2021-09-10 14:27:21 -03:00
RENDER _MODES . AUTO _RENDER _MODES . includes ( renderMode ) )
2021-05-24 16:46:47 +08:00
) {
2019-08-13 01:35:13 -04:00
viewFile ( ) ;
}
2022-02-01 17:29:00 -03:00
} , [ canViewFile , forceAutoplayParam , isPlayable , renderMode , shouldAutoplay , viewFile ] ) ;
2020-04-01 14:43:50 -04:00
/ *
once content is playing , let the appropriate < FileRender > take care of it ...
but for playables , always render so area can be used to fill with floating player
* /
2022-02-01 17:29:00 -03:00
if ( isPlaying && ! isPlayable && canViewFile && ! collectionId ) {
return null ;
2020-04-01 14:43:50 -04:00
}
2019-08-13 01:35:13 -04:00
return (
< div
2021-06-25 12:15:17 +08:00
ref = { containerRef }
2022-02-01 17:30:57 -03:00
onClick = { disabled || isMobileClaimLive ? undefined : shouldRedirect ? doAuthRedirect : viewFile }
2021-06-25 12:15:17 +08:00
style = { thumbnail && ! obscurePreview ? { backgroundImage : ` url(" ${ thumbnail } ") ` } : { } }
2020-04-01 14:43:50 -04:00
className = { classnames ( 'content__cover' , {
'content__cover--disabled' : disabled ,
2021-01-08 10:21:27 -05:00
'content__cover--theater-mode' : videoTheaterMode ,
2021-03-11 12:08:11 -05:00
'content__cover--text' : isText ,
2019-08-13 01:35:13 -04:00
'card__media--nsfw' : obscurePreview ,
} ) }
>
2022-02-01 17:29:00 -03:00
{ renderUnsupported ? (
2020-04-01 14:43:50 -04:00
< Nag
type = "helpful"
inline
message = { _ _ ( 'This content requires LBRY Desktop to display.' ) }
actionText = { _ _ ( 'Get the App' ) }
href = "https://lbry.com/get"
2019-09-02 09:24:00 -04:00
/ >
2022-02-01 17:29:00 -03:00
) : (
! claimWasPurchased &&
insufficientCredits && (
< Nag
type = "helpful"
inline
message = { _ _ ( 'You need more Credits to purchase this.' ) }
actionText = { _ _ ( 'Open Rewards' ) }
onClick = { ( ) => history . push ( ` / $ / ${ PAGES . REWARDS } ` ) }
/ >
)
2019-09-02 09:24:00 -04:00
) }
2022-02-01 17:29:00 -03:00
2022-02-01 17:30:57 -03:00
{ ! disabled && ! isMobileClaimLive && (
2019-08-13 01:35:13 -04:00
< Button
2020-05-21 17:25:38 -04:00
requiresAuth = { shouldRedirect }
2019-08-13 01:35:13 -04:00
onClick = { viewFile }
iconSize = { 30 }
title = { isPlayable ? _ _ ( 'Play' ) : _ _ ( 'View' ) }
className = { classnames ( 'button--icon' , {
'button--play' : isPlayable ,
'button--view' : ! isPlayable ,
} ) }
/ >
) }
< / div >
) ;
}