2019-08-13 07:35:13 +02:00
// @flow
// This component is entirely for triggering the start of a file view
2020-04-14 01:48:11 +02:00
// The actual viewer for a file exists in TextViewer and FileRenderFloating
2019-08-13 07:35:13 +02: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
2020-01-13 19:33:07 +01:00
import React , { useEffect , useCallback } from 'react' ;
2019-08-13 07:35:13 +02:00
import classnames from 'classnames' ;
2020-04-01 20:43:50 +02:00
import * as PAGES from 'constants/pages' ;
import * as RENDER _MODES from 'constants/file_render_modes' ;
2019-08-13 07:35:13 +02:00
import Button from 'component/button' ;
import isUserTyping from 'util/detect-typing' ;
2020-04-01 20:43:50 +02:00
import Nag from 'component/common/nag' ;
2019-08-13 07:35:13 +02:00
const SPACE _BAR _KEYCODE = 32 ;
type Props = {
play : string => void ,
isLoading : boolean ,
isPlaying : boolean ,
fileInfo : FileListItem ,
uri : string ,
2020-04-01 20:43:50 +02:00
history : { push : string => void } ,
2020-05-21 17:38:28 +02:00
location : { search : ? string , pathname : string } ,
2019-08-13 07:35:13 +02:00
obscurePreview : boolean ,
insufficientCredits : boolean ,
thumbnail ? : string ,
autoplay : boolean ,
2019-08-19 20:15:54 +02:00
hasCostInfo : boolean ,
2019-08-29 01:33:38 +02:00
costInfo : any ,
2020-01-06 19:32:35 +01:00
inline : boolean ,
2020-04-01 20:43:50 +02:00
renderMode : string ,
2020-01-13 19:33:07 +01:00
claim : StreamClaim ,
2020-05-21 17:38:28 +02:00
claimWasPurchased : boolean ,
authenticated : boolean ,
2019-08-13 07:35:13 +02:00
} ;
2020-04-01 20:43:50 +02:00
export default function FileRenderInitiator ( props : Props ) {
2019-08-14 05:09:25 +02:00
const {
play ,
isPlaying ,
fileInfo ,
uri ,
obscurePreview ,
insufficientCredits ,
2020-04-01 20:43:50 +02:00
history ,
2020-05-21 17:38:28 +02:00
location ,
2019-08-14 05:09:25 +02:00
thumbnail ,
autoplay ,
2020-04-01 20:43:50 +02:00
renderMode ,
2019-08-19 20:15:54 +02:00
hasCostInfo ,
2019-08-29 01:33:38 +02:00
costInfo ,
2020-05-21 17:38:28 +02:00
claimWasPurchased ,
authenticated ,
2019-08-14 05:09:25 +02:00
} = props ;
2020-04-01 20:43:50 +02:00
2019-08-29 05:24:04 +02:00
const cost = costInfo && costInfo . cost ;
2020-04-01 20:43:50 +02:00
const isFree = hasCostInfo && cost === 0 ;
2019-08-13 07:35:13 +02:00
const fileStatus = fileInfo && fileInfo . status ;
2020-04-01 20:43:50 +02:00
const isPlayable = RENDER _MODES . FLOATING _MODES . includes ( renderMode ) ;
2019-08-13 07:35:13 +02:00
2020-05-21 17:38:28 +02:00
function doAuthRedirect ( ) {
history . push ( ` / $ / ${ PAGES . AUTH } ?redirect= ${ encodeURIComponent ( location . pathname ) } ` ) ;
}
2019-08-13 07:35:13 +02:00
// Wrap this in useCallback because we need to use it to the keyboard effect
2019-10-10 07:26:56 +02:00
// 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
2019-08-13 07:35:13 +02:00
const viewFile = useCallback (
( e ? : SyntheticInputEvent < * > | KeyboardEvent ) => {
if ( e ) {
e . stopPropagation ( ) ;
}
play ( uri ) ;
} ,
[ play , uri ]
) ;
useEffect ( ( ) => {
// This is just for beginning to download a file
// Play/Pause/Fullscreen will be handled by the respective viewers because not every file type should behave the same
function handleKeyDown ( e : KeyboardEvent ) {
if ( ! isUserTyping ( ) && e . keyCode === SPACE _BAR _KEYCODE ) {
e . preventDefault ( ) ;
if ( ! isPlaying || fileStatus === 'stopped' ) {
viewFile ( e ) ;
}
}
}
window . addEventListener ( 'keydown' , handleKeyDown ) ;
return ( ) => {
window . removeEventListener ( 'keydown' , handleKeyDown ) ;
} ;
} , [ isPlaying , fileStatus , viewFile ] ) ;
useEffect ( ( ) => {
const videoOnPage = document . querySelector ( 'video' ) ;
2020-04-01 20:43:50 +02:00
if ( isFree && ( ( autoplay && ! videoOnPage && isPlayable ) || RENDER _MODES . AUTO _RENDER _MODES . includes ( renderMode ) ) ) {
2019-08-13 07:35:13 +02:00
viewFile ( ) ;
}
2020-04-14 01:48:11 +02:00
} , [ autoplay , viewFile , isFree , renderMode , isPlayable ] ) ;
2020-04-01 20:43:50 +02: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
* /
if ( isPlaying && ! isPlayable ) {
return null ;
}
2020-05-21 17:38:28 +02:00
const showAppNag = IS _WEB && RENDER _MODES . UNSUPPORTED _IN _THIS _APP . includes ( renderMode ) ;
const disabled = showAppNag || ( ! fileInfo && insufficientCredits && ! claimWasPurchased ) ;
const shouldRedirect = IS _WEB && ! authenticated && ! isFree ;
2019-08-13 07:35:13 +02:00
return (
< div
2020-05-21 17:38:28 +02:00
onClick = { disabled ? undefined : shouldRedirect ? doAuthRedirect : viewFile }
2020-04-01 20:43:50 +02:00
style = { thumbnail && ! obscurePreview ? { backgroundImage : ` url(" ${ thumbnail } ") ` } : { } }
className = { classnames ( 'content__cover' , {
'content__cover--disabled' : disabled ,
2019-08-13 07:35:13 +02:00
'card__media--nsfw' : obscurePreview ,
} ) }
>
2020-04-01 20:43:50 +02:00
{ showAppNag && (
< Nag
type = "helpful"
inline
message = { _ _ ( 'This content requires LBRY Desktop to display.' ) }
actionText = { _ _ ( 'Get the App' ) }
href = "https://lbry.com/get"
2019-09-02 15:24:00 +02:00
/ >
) }
2020-05-21 17:38:28 +02:00
{ ! claimWasPurchased && insufficientCredits && ! showAppNag && (
2020-04-01 20:43:50 +02:00
< Nag
type = "helpful"
inline
message = { _ _ ( 'You need more credits to purchase this.' ) }
actionText = { _ _ ( 'Open Rewards' ) }
onClick = { ( ) => history . push ( ` / $ / ${ PAGES . REWARDS } ` ) }
/ >
) }
{ ! disabled && (
2019-08-13 07:35:13 +02:00
< Button
2020-05-21 17:38:28 +02:00
requiresAuth = { IS _WEB }
2019-08-13 07:35:13 +02:00
onClick = { viewFile }
iconSize = { 30 }
title = { isPlayable ? _ _ ( 'Play' ) : _ _ ( 'View' ) }
className = { classnames ( 'button--icon' , {
'button--play' : isPlayable ,
'button--view' : ! isPlayable ,
} ) }
/ >
) }
< / div >
) ;
}