diff options
| author | metamuffin <metamuffin@disroot.org> | 2023-10-24 07:40:51 +0200 | 
|---|---|---|
| committer | metamuffin <metamuffin@disroot.org> | 2023-10-24 07:40:51 +0200 | 
| commit | b462195f2dcfe457eae7791c14e4b834b2d5ab29 (patch) | |
| tree | 492ba43952fa6798320f2b2bb4d4bd5484e2e4f2 /web/script/player/mediacaps.ts | |
| parent | 6e9ccad881a7f887599bc8f3f6b9ca2424a2cc5e (diff) | |
| parent | 55f7f06cecd5b6f5661f6f22e8bb3e0448b9713a (diff) | |
| download | jellything-b462195f2dcfe457eae7791c14e4b834b2d5ab29.tar jellything-b462195f2dcfe457eae7791c14e4b834b2d5ab29.tar.bz2 jellything-b462195f2dcfe457eae7791c14e4b834b2d5ab29.tar.zst | |
Merge branch 'master' of codeberg.org:metamuffin/jellything
Diffstat (limited to 'web/script/player/mediacaps.ts')
| -rw-r--r-- | web/script/player/mediacaps.ts | 66 | 
1 files changed, 66 insertions, 0 deletions
| diff --git a/web/script/player/mediacaps.ts b/web/script/player/mediacaps.ts new file mode 100644 index 0000000..ff143c0 --- /dev/null +++ b/web/script/player/mediacaps.ts @@ -0,0 +1,66 @@ +import { SourceTrack, SourceTrackKind } from "./jhls.d.ts"; + +const cache = new Map<string, boolean>() + +// TODO this testing method makes the assumption, that if the codec is supported on its own, it can be +// TODO     arbitrarly combined with others that are supported. in reality this is true but the spec does not gurantee it. + +export async function test_media_capability(track: SourceTrack): Promise<boolean> { +    const cache_key = `${get_track_kind(track.kind)};${track.codec}` +    const cached = cache.get(cache_key); +    if (cached !== undefined) return cached +    const r = await test_media_capability_inner(track) +    console.log(`${r ? "positive" : "negative"} media capability test finished for codec=${track.codec}`); +    cache.set(cache_key, r) +    return r +} +async function test_media_capability_inner(track: SourceTrack) { +    if (track.kind.subtitles) { +        return track.codec == "V_TEXT/WEBVTT" // TODO: actually implement it +    } +    let res; +    const codec = MASTROSKA_CODEC_MAP[track.codec] +    if (!codec) return console.warn(`unknown codec: ${track.codec}`), false +    if (track.kind.audio) { +        res = await navigator.mediaCapabilities.decodingInfo({ +            type: "media-source", +            audio: { +                contentType: `audio/webm^; codecs=${codec}`, +                samplerate: track.kind.audio.sample_rate, +                channels: "" + track.kind.audio.channels, +                bitrate: 128 * 1000, +            } +        }) +    } +    if (track.kind.video) { +        res = await navigator.mediaCapabilities.decodingInfo({ +            type: "media-source", +            video: { +                contentType: `video/webm; codecs=${codec}`, +                framerate: track.kind.video.fps || 30, +                width: track.kind.video.width, +                height: track.kind.video.height, +                bitrate: 5 * 1000 * 1000 // TODO we dont know this but we should in the future +            } +        }) +    } +    return res?.supported ?? false +} + +const MASTROSKA_CODEC_MAP: { [key: string]: string } = { +    "V_VP9": "vp9", +    "V_VP8": "vp8", +    "V_AV1": "av1", +    "V_MPEG4/ISO/AVC": "h264", +    "V_MPEGH/ISO/HEVC": "h265", +    "A_OPUS": "opus", +    "A_VORBIS": "vorbis", +    "S_TEXT/WEBVTT": "webvtt", +} + +export function get_track_kind(track: SourceTrackKind): "audio" | "video" | "subtitles" { +    if (track.audio) return "audio" +    if (track.video) return "video" +    if (track.subtitles) return "subtitles" +    throw new Error("invalid track"); +} | 
