mirror of
https://github.com/immich-app/immich.git
synced 2025-01-07 00:50:23 -05:00
fix(web): buffering for video player (#520)
* fix(web): buffering for video player * chore(): missing file -_- * refactor(web): using URL builder * chore(): add semicolon * fix(web): video player * remove deadcode Co-authored-by: Alex <alex.tran1502@gmail.com>
This commit is contained in:
parent
3b55cdc0be
commit
fb0fa742f5
4 changed files with 67 additions and 95 deletions
|
@ -1,2 +1,3 @@
|
||||||
export * from './open-api';
|
export * from './open-api';
|
||||||
export * from './api';
|
export * from './api';
|
||||||
|
export * from './utils';
|
||||||
|
|
12
web/src/api/utils.ts
Normal file
12
web/src/api/utils.ts
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
let _basePath = '/api';
|
||||||
|
|
||||||
|
export function getFileUrl(aid: string, did: string, isThumb?: boolean, isWeb?: boolean) {
|
||||||
|
const urlObj = new URL(`${window.location.origin}${_basePath}/asset/file`);
|
||||||
|
|
||||||
|
urlObj.searchParams.append('aid', aid);
|
||||||
|
urlObj.searchParams.append('did', did);
|
||||||
|
if (isThumb !== undefined && isThumb !== null) urlObj.searchParams.append('isThumb', `${isThumb}`);
|
||||||
|
if (isWeb !== undefined && isWeb !== null) urlObj.searchParams.append('isWeb', `${isWeb}`);
|
||||||
|
|
||||||
|
return urlObj.href;
|
||||||
|
}
|
|
@ -3,7 +3,7 @@
|
||||||
|
|
||||||
import { createEventDispatcher, onMount } from 'svelte';
|
import { createEventDispatcher, onMount } from 'svelte';
|
||||||
import LoadingSpinner from '../shared-components/loading-spinner.svelte';
|
import LoadingSpinner from '../shared-components/loading-spinner.svelte';
|
||||||
import { api, AssetResponseDto } from '@api';
|
import { api, AssetResponseDto, getFileUrl } from '@api';
|
||||||
|
|
||||||
export let assetId: string;
|
export let assetId: string;
|
||||||
|
|
||||||
|
@ -13,48 +13,32 @@
|
||||||
|
|
||||||
let videoPlayerNode: HTMLVideoElement;
|
let videoPlayerNode: HTMLVideoElement;
|
||||||
let isVideoLoading = true;
|
let isVideoLoading = true;
|
||||||
|
let videoUrl: string;
|
||||||
|
|
||||||
onMount(async () => {
|
onMount(async () => {
|
||||||
const { data: assetInfo } = await api.assetApi.getAssetById(assetId);
|
const { data: assetInfo } = await api.assetApi.getAssetById(assetId);
|
||||||
|
|
||||||
asset = assetInfo;
|
await loadVideoData(assetInfo);
|
||||||
|
|
||||||
await loadVideoData();
|
asset = assetInfo;
|
||||||
});
|
});
|
||||||
|
|
||||||
const loadVideoData = async () => {
|
const loadVideoData = async (assetInfo: AssetResponseDto) => {
|
||||||
isVideoLoading = true;
|
isVideoLoading = true;
|
||||||
|
|
||||||
try {
|
videoUrl = getFileUrl(assetInfo.deviceAssetId, assetInfo.deviceId, false, true);
|
||||||
const { data } = await api.assetApi.serveFile(
|
|
||||||
asset.deviceAssetId,
|
|
||||||
asset.deviceId,
|
|
||||||
false,
|
|
||||||
true,
|
|
||||||
{
|
|
||||||
responseType: 'blob'
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!(data instanceof Blob)) {
|
return assetInfo;
|
||||||
return;
|
};
|
||||||
}
|
|
||||||
|
|
||||||
const videoData = URL.createObjectURL(data);
|
const handleCanPlay = (ev: Event) => {
|
||||||
videoPlayerNode.src = videoData;
|
const playerNode = ev.target as HTMLVideoElement;
|
||||||
|
|
||||||
videoPlayerNode.load();
|
playerNode.muted = true;
|
||||||
|
playerNode.play();
|
||||||
|
playerNode.muted = false;
|
||||||
|
|
||||||
videoPlayerNode.oncanplay = () => {
|
isVideoLoading = false;
|
||||||
videoPlayerNode.muted = true;
|
|
||||||
videoPlayerNode.play();
|
|
||||||
videoPlayerNode.muted = false;
|
|
||||||
|
|
||||||
isVideoLoading = false;
|
|
||||||
};
|
|
||||||
|
|
||||||
return videoData;
|
|
||||||
} catch (e) {}
|
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
@ -63,7 +47,13 @@
|
||||||
class="flex place-items-center place-content-center h-full select-none"
|
class="flex place-items-center place-content-center h-full select-none"
|
||||||
>
|
>
|
||||||
{#if asset}
|
{#if asset}
|
||||||
<video controls class="h-full object-contain" bind:this={videoPlayerNode}>
|
<video
|
||||||
|
controls
|
||||||
|
class="h-full object-contain"
|
||||||
|
on:canplay={handleCanPlay}
|
||||||
|
bind:this={videoPlayerNode}
|
||||||
|
>
|
||||||
|
<source src={videoUrl} type="video/mp4" />
|
||||||
<track kind="captions" />
|
<track kind="captions" />
|
||||||
</video>
|
</video>
|
||||||
|
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
import PlayCircleOutline from 'svelte-material-icons/PlayCircleOutline.svelte';
|
import PlayCircleOutline from 'svelte-material-icons/PlayCircleOutline.svelte';
|
||||||
import PauseCircleOutline from 'svelte-material-icons/PauseCircleOutline.svelte';
|
import PauseCircleOutline from 'svelte-material-icons/PauseCircleOutline.svelte';
|
||||||
import LoadingSpinner from './loading-spinner.svelte';
|
import LoadingSpinner from './loading-spinner.svelte';
|
||||||
import { api, AssetResponseDto, AssetTypeEnum, ThumbnailFormat } from '@api';
|
import { api, AssetResponseDto, AssetTypeEnum, getFileUrl, ThumbnailFormat } from '@api';
|
||||||
|
|
||||||
const dispatch = createEventDispatcher();
|
const dispatch = createEventDispatcher();
|
||||||
|
|
||||||
|
@ -18,7 +18,7 @@
|
||||||
export let isExisted: boolean = false;
|
export let isExisted: boolean = false;
|
||||||
|
|
||||||
let imageData: string;
|
let imageData: string;
|
||||||
let videoData: string;
|
// let videoData: string;
|
||||||
|
|
||||||
let mouseOver: boolean = false;
|
let mouseOver: boolean = false;
|
||||||
$: dispatch('mouseEvent', { isMouseOver: mouseOver, selectedGroupIndex: groupIndex });
|
$: dispatch('mouseEvent', { isMouseOver: mouseOver, selectedGroupIndex: groupIndex });
|
||||||
|
@ -28,7 +28,8 @@
|
||||||
let isThumbnailVideoPlaying = false;
|
let isThumbnailVideoPlaying = false;
|
||||||
let calculateVideoDurationIntervalHandler: NodeJS.Timer;
|
let calculateVideoDurationIntervalHandler: NodeJS.Timer;
|
||||||
let videoProgress = '00:00';
|
let videoProgress = '00:00';
|
||||||
let videoAbortController: AbortController;
|
// let videoAbortController: AbortController;
|
||||||
|
let videoUrl: string;
|
||||||
|
|
||||||
const loadImageData = async () => {
|
const loadImageData = async () => {
|
||||||
const { data } = await api.assetApi.getAssetThumbnail(asset.id, format, {
|
const { data } = await api.assetApi.getAssetThumbnail(asset.id, format, {
|
||||||
|
@ -42,51 +43,8 @@
|
||||||
|
|
||||||
const loadVideoData = async () => {
|
const loadVideoData = async () => {
|
||||||
isThumbnailVideoPlaying = false;
|
isThumbnailVideoPlaying = false;
|
||||||
videoAbortController = new AbortController();
|
|
||||||
|
|
||||||
try {
|
videoUrl = getFileUrl(asset.deviceAssetId, asset.deviceId, false, true);
|
||||||
const { data } = await api.assetApi.serveFile(
|
|
||||||
asset.deviceAssetId,
|
|
||||||
asset.deviceId,
|
|
||||||
false,
|
|
||||||
true,
|
|
||||||
{
|
|
||||||
responseType: 'blob',
|
|
||||||
signal: videoAbortController.signal
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!(data instanceof Blob)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
videoData = URL.createObjectURL(data);
|
|
||||||
|
|
||||||
videoPlayerNode.src = videoData;
|
|
||||||
|
|
||||||
videoPlayerNode.load();
|
|
||||||
|
|
||||||
videoPlayerNode.onloadeddata = () => {
|
|
||||||
console.log('first frame load');
|
|
||||||
};
|
|
||||||
|
|
||||||
videoPlayerNode.oncanplaythrough = () => {
|
|
||||||
console.log('can play through');
|
|
||||||
};
|
|
||||||
|
|
||||||
videoPlayerNode.oncanplay = () => {
|
|
||||||
console.log('can play');
|
|
||||||
videoPlayerNode.muted = true;
|
|
||||||
videoPlayerNode.play();
|
|
||||||
|
|
||||||
isThumbnailVideoPlaying = true;
|
|
||||||
calculateVideoDurationIntervalHandler = setInterval(() => {
|
|
||||||
videoProgress = getVideoDurationInString(Math.round(videoPlayerNode.currentTime));
|
|
||||||
}, 1000);
|
|
||||||
};
|
|
||||||
|
|
||||||
return videoData;
|
|
||||||
} catch (e) {}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const getVideoDurationInString = (currentTime: number) => {
|
const getVideoDurationInString = (currentTime: number) => {
|
||||||
|
@ -136,12 +94,7 @@
|
||||||
|
|
||||||
const handleMouseLeaveThumbnail = () => {
|
const handleMouseLeaveThumbnail = () => {
|
||||||
mouseOver = false;
|
mouseOver = false;
|
||||||
|
videoUrl = '';
|
||||||
// Stop XHR download of video
|
|
||||||
videoAbortController?.abort();
|
|
||||||
|
|
||||||
// Stop video playback
|
|
||||||
URL.revokeObjectURL(videoData);
|
|
||||||
|
|
||||||
clearInterval(calculateVideoDurationIntervalHandler);
|
clearInterval(calculateVideoDurationIntervalHandler);
|
||||||
|
|
||||||
|
@ -149,6 +102,18 @@
|
||||||
videoProgress = '00:00';
|
videoProgress = '00:00';
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handleCanPlay = (ev: Event) => {
|
||||||
|
const playerNode = ev.target as HTMLVideoElement;
|
||||||
|
|
||||||
|
playerNode.muted = true;
|
||||||
|
playerNode.play();
|
||||||
|
|
||||||
|
isThumbnailVideoPlaying = true;
|
||||||
|
calculateVideoDurationIntervalHandler = setInterval(() => {
|
||||||
|
videoProgress = getVideoDurationInString(Math.round(playerNode.currentTime));
|
||||||
|
}, 1000);
|
||||||
|
};
|
||||||
|
|
||||||
$: getThumbnailBorderStyle = () => {
|
$: getThumbnailBorderStyle = () => {
|
||||||
if (selected) {
|
if (selected) {
|
||||||
return 'border-[20px] border-immich-primary/20';
|
return 'border-[20px] border-immich-primary/20';
|
||||||
|
@ -259,17 +224,21 @@
|
||||||
|
|
||||||
{#if mouseOver && asset.type === AssetTypeEnum.Video}
|
{#if mouseOver && asset.type === AssetTypeEnum.Video}
|
||||||
<div class="absolute w-full h-full top-0" on:mouseenter={loadVideoData}>
|
<div class="absolute w-full h-full top-0" on:mouseenter={loadVideoData}>
|
||||||
<video
|
{#if videoUrl}
|
||||||
muted
|
<video
|
||||||
autoplay
|
muted
|
||||||
preload="none"
|
autoplay
|
||||||
class="h-full object-cover"
|
preload="none"
|
||||||
width="250px"
|
class="h-full object-cover"
|
||||||
style:width={`${thumbnailSize}px`}
|
width="250px"
|
||||||
bind:this={videoPlayerNode}
|
style:width={`${thumbnailSize}px`}
|
||||||
>
|
on:canplay={handleCanPlay}
|
||||||
<track kind="captions" />
|
bind:this={videoPlayerNode}
|
||||||
</video>
|
>
|
||||||
|
<source src={videoUrl} type="video/mp4" />
|
||||||
|
<track kind="captions" />
|
||||||
|
</video>
|
||||||
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
|
|
Loading…
Reference in a new issue