mirror of
https://github.com/hcengineering/platform.git
synced 2025-04-30 04:05:40 +00:00
fix: extract video player to separate component (#8086)
Some checks are pending
CI / test (push) Blocked by required conditions
CI / build (push) Waiting to run
CI / svelte-check (push) Blocked by required conditions
CI / formatting (push) Blocked by required conditions
CI / uitest (push) Waiting to run
CI / uitest-pg (push) Waiting to run
CI / uitest-qms (push) Waiting to run
CI / uitest-workspaces (push) Waiting to run
CI / docker-build (push) Blocked by required conditions
CI / dist-build (push) Blocked by required conditions
Some checks are pending
CI / test (push) Blocked by required conditions
CI / build (push) Waiting to run
CI / svelte-check (push) Blocked by required conditions
CI / formatting (push) Blocked by required conditions
CI / uitest (push) Waiting to run
CI / uitest-pg (push) Waiting to run
CI / uitest-qms (push) Waiting to run
CI / uitest-workspaces (push) Waiting to run
CI / docker-build (push) Blocked by required conditions
CI / dist-build (push) Blocked by required conditions
Signed-off-by: Alexander Onnikov <Alexander.Onnikov@xored.com>
This commit is contained in:
parent
815caef82a
commit
0a2aad3650
@ -26158,6 +26158,7 @@ snapshots:
|
|||||||
eslint-plugin-promise: 6.1.1(eslint@8.56.0)
|
eslint-plugin-promise: 6.1.1(eslint@8.56.0)
|
||||||
eslint-plugin-svelte: 2.35.1(eslint@8.56.0)(svelte@4.2.19)(ts-node@10.9.2(@types/node@20.11.19)(typescript@5.3.3))
|
eslint-plugin-svelte: 2.35.1(eslint@8.56.0)(svelte@4.2.19)(ts-node@10.9.2(@types/node@20.11.19)(typescript@5.3.3))
|
||||||
fast-equals: 5.2.2
|
fast-equals: 5.2.2
|
||||||
|
hls.js: 1.5.20
|
||||||
jest: 29.7.0(@types/node@20.11.19)(ts-node@10.9.2(@types/node@20.11.19)(typescript@5.3.3))
|
jest: 29.7.0(@types/node@20.11.19)(ts-node@10.9.2(@types/node@20.11.19)(typescript@5.3.3))
|
||||||
prettier: 3.2.5
|
prettier: 3.2.5
|
||||||
prettier-plugin-svelte: 3.2.2(prettier@3.2.5)(svelte@4.2.19)
|
prettier-plugin-svelte: 3.2.2(prettier@3.2.5)(svelte@4.2.19)
|
||||||
|
@ -49,7 +49,8 @@
|
|||||||
"date-fns-tz": "^2.0.0",
|
"date-fns-tz": "^2.0.0",
|
||||||
"dompurify": "^3.1.6",
|
"dompurify": "^3.1.6",
|
||||||
"@hcengineering/analytics": "^0.6.0",
|
"@hcengineering/analytics": "^0.6.0",
|
||||||
"emojibase": "^16.0.0"
|
"emojibase": "^16.0.0",
|
||||||
|
"hls.js": "^1.5.20"
|
||||||
},
|
},
|
||||||
"repository": "https://github.com/hcengineering/platform",
|
"repository": "https://github.com/hcengineering/platform",
|
||||||
"publishConfig": {
|
"publishConfig": {
|
||||||
|
68
packages/ui/src/components/HlsVideo.svelte
Normal file
68
packages/ui/src/components/HlsVideo.svelte
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
<!--
|
||||||
|
// Copyright © 2025 Hardcore Engineering Inc.
|
||||||
|
//
|
||||||
|
// Licensed under the Eclipse Public License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License. You may
|
||||||
|
// obtain a copy of the License at https://www.eclipse.org/legal/epl-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
//
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
-->
|
||||||
|
<script lang="ts">
|
||||||
|
import HLS from 'hls.js'
|
||||||
|
import { onDestroy, onMount } from 'svelte'
|
||||||
|
|
||||||
|
export let src: string
|
||||||
|
export let hlsSrc: string
|
||||||
|
export let hlsThumbnail: string
|
||||||
|
export let name: string = ''
|
||||||
|
export let preload = true
|
||||||
|
|
||||||
|
let video: HTMLVideoElement
|
||||||
|
let hls: HLS
|
||||||
|
|
||||||
|
onMount(() => {
|
||||||
|
if (HLS.isSupported()) {
|
||||||
|
hls?.destroy()
|
||||||
|
hls = new HLS({ autoStartLoad: false })
|
||||||
|
hls.loadSource(hlsSrc)
|
||||||
|
hls.attachMedia(video)
|
||||||
|
|
||||||
|
video.poster = hlsThumbnail
|
||||||
|
video.onplay = () => {
|
||||||
|
// autoStartLoad disables autoplay, so we need to enable it manually
|
||||||
|
video.onplay = null
|
||||||
|
hls.startLoad()
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
video.src = src
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
onDestroy(() => {
|
||||||
|
hls?.destroy()
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<video bind:this={video} width="100%" height="100%" preload={preload ? 'auto' : 'none'} controls>
|
||||||
|
<track kind="captions" label={name} />
|
||||||
|
</video>
|
||||||
|
|
||||||
|
<style lang="scss">
|
||||||
|
video {
|
||||||
|
border-radius: inherit;
|
||||||
|
object-fit: contain;
|
||||||
|
}
|
||||||
|
|
||||||
|
video::-webkit-media-controls {
|
||||||
|
visibility: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
video::-webkit-media-controls-enclosure {
|
||||||
|
visibility: visible;
|
||||||
|
}
|
||||||
|
</style>
|
40
packages/ui/src/components/Video.svelte
Normal file
40
packages/ui/src/components/Video.svelte
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
<!--
|
||||||
|
// Copyright © 2025 Hardcore Engineering Inc.
|
||||||
|
//
|
||||||
|
// Licensed under the Eclipse Public License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License. You may
|
||||||
|
// obtain a copy of the License at https://www.eclipse.org/legal/epl-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
//
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
-->
|
||||||
|
<script lang="ts">
|
||||||
|
export let src: string
|
||||||
|
export let name: string = ''
|
||||||
|
export let preload = true
|
||||||
|
|
||||||
|
let video: HTMLVideoElement
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<video bind:this={video} {src} width="100%" height="100%" preload={preload ? 'auto' : 'none'} controls>
|
||||||
|
<track kind="captions" label={name} />
|
||||||
|
</video>
|
||||||
|
|
||||||
|
<style lang="scss">
|
||||||
|
video {
|
||||||
|
border-radius: inherit;
|
||||||
|
object-fit: contain;
|
||||||
|
}
|
||||||
|
|
||||||
|
video::-webkit-media-controls {
|
||||||
|
visibility: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
video::-webkit-media-controls-enclosure {
|
||||||
|
visibility: visible;
|
||||||
|
}
|
||||||
|
</style>
|
@ -284,6 +284,8 @@ export { default as NestedDropdown } from './components/NestedDropdown.svelte'
|
|||||||
|
|
||||||
export { default as Dock } from './components/Dock.svelte'
|
export { default as Dock } from './components/Dock.svelte'
|
||||||
export { default as Image } from './components/Image.svelte'
|
export { default as Image } from './components/Image.svelte'
|
||||||
|
export { default as Video } from './components/Video.svelte'
|
||||||
|
export { default as HlsVideo } from './components/HlsVideo.svelte'
|
||||||
|
|
||||||
export * from './types'
|
export * from './types'
|
||||||
export * from './location'
|
export * from './location'
|
||||||
|
@ -14,10 +14,9 @@
|
|||||||
-->
|
-->
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import type { Attachment } from '@hcengineering/attachment'
|
import type { Attachment } from '@hcengineering/attachment'
|
||||||
|
|
||||||
import type { WithLookup } from '@hcengineering/core'
|
import type { WithLookup } from '@hcengineering/core'
|
||||||
import { getFileUrl } from '@hcengineering/presentation'
|
import { getFileUrl, getVideoMeta } from '@hcengineering/presentation'
|
||||||
import AttachmentPresenter from './AttachmentPresenter.svelte'
|
import { HlsVideo, Video } from '@hcengineering/ui'
|
||||||
|
|
||||||
export let value: WithLookup<Attachment>
|
export let value: WithLookup<Attachment>
|
||||||
export let preload = true
|
export let preload = true
|
||||||
@ -55,18 +54,26 @@
|
|||||||
|
|
||||||
return { width, height }
|
return { width, height }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function toStyle (size: 'auto' | number): string {
|
||||||
|
return size === 'auto' ? 'auto' : `${size}px`
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<video controls width={dimensions.width} height={dimensions.height} preload={preload ? 'auto' : 'none'}>
|
<div class="container" style="width:{toStyle(dimensions.width)}; height:{toStyle(dimensions.height)}">
|
||||||
<source src={getFileUrl(value.file, value.name)} />
|
{#await getVideoMeta(value.file, value.name) then meta}
|
||||||
<track kind="captions" label={value.name} />
|
{@const src = getFileUrl(value.file, value.name)}
|
||||||
<div class="container">
|
|
||||||
<AttachmentPresenter {value} />
|
{#if meta && meta.status === 'ready'}
|
||||||
|
<HlsVideo {src} {preload} hlsSrc={meta.hls} hlsThumbnail={meta.thumbnail} name={value.name} />
|
||||||
|
{:else}
|
||||||
|
<Video {src} {preload} name={value.name} />
|
||||||
|
{/if}
|
||||||
|
{/await}
|
||||||
</div>
|
</div>
|
||||||
</video>
|
|
||||||
|
|
||||||
<style lang="scss">
|
<style lang="scss">
|
||||||
video {
|
.container {
|
||||||
max-width: 20rem;
|
max-width: 20rem;
|
||||||
max-height: 20rem;
|
max-height: 20rem;
|
||||||
min-width: 4rem;
|
min-width: 4rem;
|
||||||
|
@ -58,7 +58,6 @@
|
|||||||
"@hcengineering/text-editor-resources": "^0.6.0",
|
"@hcengineering/text-editor-resources": "^0.6.0",
|
||||||
"@hcengineering/analytics": "^0.6.0",
|
"@hcengineering/analytics": "^0.6.0",
|
||||||
"@hcengineering/query": "^0.6.12",
|
"@hcengineering/query": "^0.6.12",
|
||||||
"fast-equals": "^5.2.2",
|
"fast-equals": "^5.2.2"
|
||||||
"hls.js": "^1.5.20"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -15,68 +15,33 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { type Blob, type Ref } from '@hcengineering/core'
|
import { type Blob, type Ref } from '@hcengineering/core'
|
||||||
import { getFileUrl, getVideoMeta, type BlobMetadata } from '@hcengineering/presentation'
|
import { getFileUrl, getVideoMeta, type BlobMetadata } from '@hcengineering/presentation'
|
||||||
import { onDestroy } from 'svelte'
|
import { HlsVideo, Video } from '@hcengineering/ui'
|
||||||
import HLS from 'hls.js'
|
|
||||||
|
|
||||||
export let value: Ref<Blob>
|
export let value: Ref<Blob>
|
||||||
export let name: string
|
export let name: string
|
||||||
export let metadata: BlobMetadata | undefined
|
export let metadata: BlobMetadata | undefined
|
||||||
export let fit: boolean = false
|
export let fit: boolean = false
|
||||||
|
|
||||||
let video: HTMLVideoElement
|
|
||||||
let hls: HLS
|
|
||||||
|
|
||||||
async function fetchVideoMeta (value: Ref<Blob>, name: string): Promise<void> {
|
|
||||||
const src = getFileUrl(value, name)
|
|
||||||
const meta = await getVideoMeta(value, name)
|
|
||||||
if (meta != null && meta.status === 'ready' && HLS.isSupported()) {
|
|
||||||
hls?.destroy()
|
|
||||||
hls = new HLS({ autoStartLoad: false })
|
|
||||||
hls.loadSource(meta.hls)
|
|
||||||
hls.attachMedia(video)
|
|
||||||
|
|
||||||
video.poster = meta.thumbnail
|
|
||||||
video.onplay = () => {
|
|
||||||
// autoStartLoad disables autoplay, so we need to enable it manually
|
|
||||||
video.onplay = null
|
|
||||||
hls.startLoad()
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
video.src = src
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
onDestroy(() => {
|
|
||||||
hls?.destroy()
|
|
||||||
})
|
|
||||||
|
|
||||||
$: aspectRatio =
|
$: aspectRatio =
|
||||||
metadata?.originalWidth && metadata?.originalHeight
|
metadata?.originalWidth && metadata?.originalHeight
|
||||||
? `${metadata.originalWidth} / ${metadata.originalHeight}`
|
? `${metadata.originalWidth} / ${metadata.originalHeight}`
|
||||||
: '16 / 9'
|
: '16 / 9'
|
||||||
$: maxWidth = metadata?.originalWidth ? `min(${metadata.originalWidth}px, 100%)` : undefined
|
$: maxWidth = metadata?.originalWidth ? `min(${metadata.originalWidth}px, 100%)` : undefined
|
||||||
$: maxHeight = metadata?.originalHeight ? `min(${metadata.originalHeight}px, 80vh)` : undefined
|
$: maxHeight = metadata?.originalHeight ? `min(${metadata.originalHeight}px, 80vh)` : undefined
|
||||||
$: void fetchVideoMeta(value, name)
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<video
|
<div
|
||||||
bind:this={video}
|
|
||||||
width="100%"
|
|
||||||
style:aspect-ratio={aspectRatio}
|
style:aspect-ratio={aspectRatio}
|
||||||
style:max-width={fit ? '100%' : maxWidth}
|
style:max-width={fit ? '100%' : maxWidth}
|
||||||
style:max-height={fit ? '100%' : maxHeight}
|
style:max-height={fit ? '100%' : maxHeight}
|
||||||
controls
|
|
||||||
preload={'auto'}
|
|
||||||
>
|
>
|
||||||
<track kind="captions" label={name} />
|
{#await getVideoMeta(value, name) then meta}
|
||||||
</video>
|
{@const src = getFileUrl(value, name)}
|
||||||
|
|
||||||
<style lang="scss">
|
{#if meta && meta.status === 'ready'}
|
||||||
video::-webkit-media-controls {
|
<HlsVideo {src} {name} hlsSrc={meta.hls} hlsThumbnail={meta.thumbnail} />
|
||||||
visibility: hidden;
|
{:else}
|
||||||
}
|
<Video {src} {name} />
|
||||||
|
{/if}
|
||||||
video::-webkit-media-controls-enclosure {
|
{/await}
|
||||||
visibility: visible;
|
</div>
|
||||||
}
|
|
||||||
</style>
|
|
||||||
|
Loading…
Reference in New Issue
Block a user