mirror of
https://github.com/ksyasuda/SubMiner.git
synced 2026-03-27 06:12:05 -07:00
* fix: harden preload argv parsing for popup windows * fix: align youtube playback with shared overlay startup * fix: unwrap mpv youtube streams for anki media mining * docs: update docs for youtube subtitle and mining flow * refactor: unify cli and runtime wiring for startup and youtube flow * feat: update subtitle sidebar overlay behavior * chore: add shared log-file source for diagnostics * fix(ci): add changelog fragment for immersion changes * fix: address CodeRabbit review feedback * fix: persist canonical title from youtube metadata * style: format stats library tab * fix: address latest review feedback * style: format stats library files * test: stub launcher youtube deps in CI * test: isolate launcher youtube flow deps * test: stub launcher youtube deps in failing case * test: force x11 backend in launcher ci harness * test: address latest review feedback * fix(launcher): preserve user YouTube ytdl raw options * docs(backlog): update task tracking notes * fix(immersion): special-case youtube media paths in runtime and tracking * feat(stats): improve YouTube media metadata and picker key handling * fix(ci): format stats media library hook * fix: address latest CodeRabbit review items * docs: update youtube release notes and docs * feat: auto-load youtube subtitles before manual picker * fix: restore app-owned youtube subtitle flow * docs: update youtube playback docs and config copy * refactor: remove legacy youtube launcher mode plumbing * fix: refine youtube subtitle startup binding * docs: clarify youtube subtitle startup behavior * fix: address PR #31 latest review follow-ups * fix: address PR #31 follow-up review comments * test: harden youtube picker test harness * udpate backlog * fix: add timeout to youtube metadata probe * docs: refresh youtube and stats docs * update backlog * update backlog * chore: release v0.9.0
85 lines
2.5 KiB
TypeScript
85 lines
2.5 KiB
TypeScript
import { isRemoteMediaPath } from '../jimaku/utils';
|
|
import type { MpvClient } from '../types';
|
|
|
|
export type MediaGenerationKind = 'audio' | 'video';
|
|
|
|
function trimToNonEmptyString(value: unknown): string | null {
|
|
if (typeof value !== 'string') {
|
|
return null;
|
|
}
|
|
const trimmed = value.trim();
|
|
return trimmed.length > 0 ? trimmed : null;
|
|
}
|
|
|
|
function extractUrlsFromMpvEdlSource(source: string): string[] {
|
|
const matches = source.matchAll(/%\d+%(https?:\/\/.*?)(?=;!new_stream|;!global_tags|$)/gms);
|
|
return [...matches]
|
|
.map((match) => trimToNonEmptyString(match[1]))
|
|
.filter((value): value is string => value !== null);
|
|
}
|
|
|
|
function classifyMediaUrl(url: string): MediaGenerationKind | null {
|
|
try {
|
|
const mime = new URL(url).searchParams.get('mime')?.toLowerCase() ?? '';
|
|
if (mime.startsWith('audio/')) {
|
|
return 'audio';
|
|
}
|
|
if (mime.startsWith('video/')) {
|
|
return 'video';
|
|
}
|
|
} catch {
|
|
// Ignore malformed URLs and fall back to stream order.
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
function resolvePreferredUrlFromMpvEdlSource(
|
|
source: string,
|
|
kind: MediaGenerationKind,
|
|
): string | null {
|
|
const urls = extractUrlsFromMpvEdlSource(source);
|
|
if (urls.length === 0) {
|
|
return null;
|
|
}
|
|
|
|
const typedMatch = urls.find((url) => classifyMediaUrl(url) === kind);
|
|
if (typedMatch) {
|
|
return typedMatch;
|
|
}
|
|
|
|
// mpv EDL sources usually list audio streams first and video streams last, so
|
|
// when classifyMediaUrl cannot identify a typed URL we fall back to stream order.
|
|
return kind === 'audio' ? urls[0] ?? null : urls[urls.length - 1] ?? null;
|
|
}
|
|
|
|
export async function resolveMediaGenerationInputPath(
|
|
mpvClient: Pick<MpvClient, 'currentVideoPath' | 'requestProperty'> | null | undefined,
|
|
kind: MediaGenerationKind = 'video',
|
|
): Promise<string | null> {
|
|
const currentVideoPath = trimToNonEmptyString(mpvClient?.currentVideoPath);
|
|
if (!currentVideoPath) {
|
|
return null;
|
|
}
|
|
|
|
if (!isRemoteMediaPath(currentVideoPath) || !mpvClient?.requestProperty) {
|
|
return currentVideoPath;
|
|
}
|
|
|
|
try {
|
|
const streamOpenFilename = trimToNonEmptyString(
|
|
await mpvClient.requestProperty('stream-open-filename'),
|
|
);
|
|
if (streamOpenFilename?.startsWith('edl://')) {
|
|
return resolvePreferredUrlFromMpvEdlSource(streamOpenFilename, kind) ?? streamOpenFilename;
|
|
}
|
|
if (streamOpenFilename) {
|
|
return streamOpenFilename;
|
|
}
|
|
} catch {
|
|
// Fall back to the current path when mpv does not expose a resolved stream URL.
|
|
}
|
|
|
|
return currentVideoPath;
|
|
}
|