gradio/js/video/shared/InteractiveVideo.svelte
Hannah bc2cdc1df9
Allow download button for interactive Audio and Video components (#7104)
* allow download button for interactive audio and video components

* add changeset

* document show_download_button in video.py

* ensure show_download_button is by default true for output and false for input components

* fix component test

* tweak default val

* val tweak

* fix test and add story

* pass editable param where needed

* fix another test

---------

Co-authored-by: gradio-pr-bot <gradio-pr-bot@users.noreply.github.com>
Co-authored-by: Dawood Khan <dawoodkhan82@gmail.com>
2024-01-23 11:33:59 +01:00

158 lines
3.5 KiB
Svelte

<script lang="ts">
import { createEventDispatcher } from "svelte";
import { Upload, ModifyUpload } from "@gradio/upload";
import type { FileData } from "@gradio/client";
import { BlockLabel } from "@gradio/atoms";
import { Webcam } from "@gradio/image";
import { Video } from "@gradio/icons";
import { prettyBytes, playable } from "./utils";
import Player from "./Player.svelte";
import type { I18nFormatter } from "@gradio/utils";
import { SelectSource } from "@gradio/atoms";
export let value: FileData | null = null;
export let subtitle: FileData | null = null;
export let sources:
| ["webcam"]
| ["upload"]
| ["webcam", "upload"]
| ["upload", "webcam"] = ["webcam", "upload"];
export let label: string | undefined = undefined;
export let show_download_button = false;
export let show_label = true;
export let mirror_webcam = false;
export let include_audio: boolean;
export let autoplay: boolean;
export let root: string;
export let i18n: I18nFormatter;
export let active_source: "webcam" | "upload" = "webcam";
export let handle_reset_value: () => void = () => {};
const dispatch = createEventDispatcher<{
change: FileData | null;
clear?: never;
play?: never;
pause?: never;
end?: never;
drag: boolean;
error: string;
upload: FileData;
start_recording?: never;
stop_recording?: never;
}>();
function handle_load({ detail }: CustomEvent<FileData | null>): void {
value = detail;
dispatch("change", detail);
dispatch("upload", detail!);
}
function handle_clear(): void {
value = null;
dispatch("change", null);
dispatch("clear");
}
function handle_change(video: FileData): void {
dispatch("change", video);
}
function handle_capture({
detail
}: CustomEvent<FileData | any | null>): void {
dispatch("change", detail);
}
let dragging = false;
$: dispatch("drag", dragging);
</script>
<BlockLabel {show_label} Icon={Video} label={label || "Video"} />
<div data-testid="video" class="video-container">
{#if value === null || value.url === undefined}
<div class="upload-container">
{#if active_source === "upload"}
<Upload
bind:dragging
filetype="video/x-m4v,video/*"
on:load={handle_load}
on:error={({ detail }) => dispatch("error", detail)}
{root}
>
<slot />
</Upload>
{:else if active_source === "webcam"}
<Webcam
{root}
{mirror_webcam}
{include_audio}
mode="video"
on:error
on:capture={handle_capture}
on:start_recording
on:stop_recording
{i18n}
/>
{/if}
</div>
{:else}
<ModifyUpload
{i18n}
on:clear={handle_clear}
download={show_download_button ? value.url : null}
/>
{#if playable()}
{#key value?.url}
<Player
{root}
interactive
{autoplay}
src={value.url}
subtitle={subtitle?.url}
on:play
on:pause
on:stop
on:end
mirror={mirror_webcam && active_source === "webcam"}
{label}
{handle_change}
{handle_reset_value}
/>
{/key}
{:else if value.size}
<div class="file-name">{value.orig_name || value.url}</div>
<div class="file-size">
{prettyBytes(value.size)}
</div>
{/if}
{/if}
<SelectSource {sources} bind:active_source {handle_clear} />
</div>
<style>
.file-name {
padding: var(--size-6);
font-size: var(--text-xxl);
word-break: break-all;
}
.file-size {
padding: var(--size-2);
font-size: var(--text-xl);
}
.upload-container {
height: 100%;
}
.video-container {
display: flex;
height: 100%;
flex-direction: column;
justify-content: center;
align-items: center;
}
</style>