Merge pull request #2530 from lbryio/sandbox

Fix render logic of non-playable files
This commit is contained in:
Sean Yesmunt 2019-06-04 14:56:34 -04:00 committed by GitHub
commit a3c1f3a1ee
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23

View file

@ -77,10 +77,12 @@ class MediaPlayer extends React.PureComponent<Props, State> {
// Temp hack to force the video to play if the metadataloaded event was never fired // Temp hack to force the video to play if the metadataloaded event was never fired
// Will be removed with the new video player // Will be removed with the new video player
// Unoptimized MP4s will fail to render. // Unoptimized MP4s will fail to render.
// Note: Don't use this for non-playable files
// @if TARGET='app' // @if TARGET='app'
setTimeout(() => { setTimeout(() => {
const { hasMetadata } = this.state; const { hasMetadata } = this.state;
if (!hasMetadata) { const isPlayableType = this.playableType();
if (!hasMetadata && isPlayableType) {
this.refreshMetadata(); this.refreshMetadata();
this.playMedia(); this.playMedia();
} }
@ -88,6 +90,20 @@ class MediaPlayer extends React.PureComponent<Props, State> {
// @endif // @endif
} }
componentDidUpdate(prevProps: Props) {
const { fileSource } = this.state;
const { downloadCompleted } = this.props;
// Attemp to render a non-playable file once download is completed
if (prevProps.downloadCompleted !== downloadCompleted) {
const isFileType = this.isSupportedFile();
if (isFileType && !fileSource && downloadCompleted) {
this.playMedia();
}
}
}
componentWillUnmount() { componentWillUnmount() {
const mediaElement = this.mediaContainer.current.children[0]; const mediaElement = this.mediaContainer.current.children[0];
@ -139,8 +155,10 @@ class MediaPlayer extends React.PureComponent<Props, State> {
}; };
// Render custom viewer: FileRender // Render custom viewer: FileRender
if (this.isSupportedFile() && downloadCompleted) { if (this.isSupportedFile()) {
this.renderFile(); if (downloadCompleted) {
this.renderFile();
}
} else { } else {
// Render default viewer: render-media (video, audio, img, iframe) // Render default viewer: render-media (video, audio, img, iframe)
const currentMediaContainer = this.mediaContainer.current; const currentMediaContainer = this.mediaContainer.current;
@ -256,9 +274,7 @@ class MediaPlayer extends React.PureComponent<Props, State> {
isRenderMediaSupported() { isRenderMediaSupported() {
// Files supported by render-media // Files supported by render-media
const { contentType } = this.props; const { contentType } = this.props;
return ( return Object.values(player.mime).indexOf(contentType) !== -1;
Object.values(player.mime).indexOf(contentType) !== -1 || MediaPlayer.SANDBOX_TYPES.indexOf(contentType) > -1
);
} }
isSupportedFile() { isSupportedFile() {
@ -274,33 +290,36 @@ class MediaPlayer extends React.PureComponent<Props, State> {
if (MediaPlayer.SANDBOX_TYPES.indexOf(contentType) > -1) { if (MediaPlayer.SANDBOX_TYPES.indexOf(contentType) > -1) {
const outpoint = `${claim.txid}:${claim.nout}`; const outpoint = `${claim.txid}:${claim.nout}`;
// Fetch unpacked url
return fetch(`${MediaPlayer.SANDBOX_SET_BASE_URL}${outpoint}`) fetch(`${MediaPlayer.SANDBOX_SET_BASE_URL}${outpoint}`)
.then(res => res.text()) .then(res => res.text())
.then(url => { .then(url => {
const fileSource = { url: `${MediaPlayer.SANDBOX_CONTENT_BASE_URL}${url}` }; const fileSource = { url: `${MediaPlayer.SANDBOX_CONTENT_BASE_URL}${url}` };
return this.setState({ fileSource }); this.setState({ fileSource });
})
.catch(err => {
console.error(err);
}); });
} else {
// File to render
const fileSource = {
fileName,
contentType,
downloadPath,
fileType: path.extname(fileName).substring(1),
// Readable stream from file
// @if TARGET='app'
stream: opts => fs.createReadStream(downloadPath, opts),
// @endif
};
// Update state
this.setState({ fileSource });
} }
// File to render
const fileSource = {
fileName,
contentType,
downloadPath,
fileType: path.extname(fileName).substring(1),
// Readable stream from file
// @if TARGET='app'
stream: opts => fs.createReadStream(downloadPath, opts),
// @endif
};
// Update state
this.setState({ fileSource });
} }
showLoadingScreen(isFileType: boolean, isPlayableType: boolean) { showLoadingScreen(isFileType: boolean, isPlayableType: boolean) {
const { mediaType, contentType } = this.props; const { mediaType } = this.props;
const { unplayable, fileSource, hasMetadata } = this.state; const { unplayable, fileSource, hasMetadata } = this.state;
if (IS_WEB && ['audio', 'video'].indexOf(mediaType) === -1) { if (IS_WEB && ['audio', 'video'].indexOf(mediaType) === -1) {
@ -325,10 +344,7 @@ class MediaPlayer extends React.PureComponent<Props, State> {
// Files // Files
const isLoadingFile = !fileSource && isFileType; const isLoadingFile = !fileSource && isFileType;
const isLbryPackage = /application\/x(-ext)?-lbry$/.test(contentType); const isUnsupported = !this.isRenderMediaSupported() && !isFileType && !isPlayableType;
const isUnsupported =
(mediaType === 'application' && !isLbryPackage) ||
(!this.isRenderMediaSupported() && !isFileType && !isPlayableType);
// Media (audio, video) // Media (audio, video)
const isUnplayable = isPlayableType && unplayable; const isUnplayable = isPlayableType && unplayable;
const isLoadingMetadata = isPlayableType && (!hasMetadata && !unplayable); const isLoadingMetadata = isPlayableType && (!hasMetadata && !unplayable);
@ -341,8 +357,6 @@ class MediaPlayer extends React.PureComponent<Props, State> {
// Show unsupported error message // Show unsupported error message
} else if (isUnsupported || isUnplayable) { } else if (isUnsupported || isUnplayable) {
loader.loadingStatus = isUnsupported ? unsupportedMessage : unplayableMessage; loader.loadingStatus = isUnsupported ? unsupportedMessage : unplayableMessage;
} else if (isLbryPackage && !isLoadingFile) {
loader.loadingStatus = null;
} }
return loader; return loader;
@ -352,8 +366,7 @@ class MediaPlayer extends React.PureComponent<Props, State> {
const { mediaType, claim } = this.props; const { mediaType, claim } = this.props;
const { fileSource } = this.state; const { fileSource } = this.state;
const isFileType = this.isSupportedFile(); const isFileType = this.isSupportedFile();
const isFileReady = fileSource !== null && isFileType;
const isFileReady = fileSource && isFileType;
const isPlayableType = this.playableType(); const isPlayableType = this.playableType();
const { isLoading, loadingStatus } = this.showLoadingScreen(isFileType, isPlayableType); const { isLoading, loadingStatus } = this.showLoadingScreen(isFileType, isPlayableType);