mirror of
https://github.com/zoriya/Kyoo.git
synced 2025-10-24 15:29:06 -04:00
Add liks in the API for videos, fonts and subtitle
This commit is contained in:
parent
0652ffef68
commit
edc33f3c37
@ -25,7 +25,7 @@ namespace Kyoo.Abstractions.Models
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// A font of an <see cref="Episode"/>.
|
/// A font of an <see cref="Episode"/>.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class Font
|
public class Font : ILink
|
||||||
{
|
{
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// A human-readable identifier, used in the URL.
|
/// A human-readable identifier, used in the URL.
|
||||||
@ -47,6 +47,9 @@ namespace Kyoo.Abstractions.Models
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
[SerializeIgnore] public string Path { get; set; }
|
[SerializeIgnore] public string Path { get; set; }
|
||||||
|
|
||||||
|
/// <inheritdoc/>
|
||||||
|
public object Link { get; init; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Create a new empty <see cref="Font"/>.
|
/// Create a new empty <see cref="Font"/>.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@ -56,12 +59,14 @@ namespace Kyoo.Abstractions.Models
|
|||||||
/// Create a new <see cref="Font"/> from a path.
|
/// Create a new <see cref="Font"/> from a path.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="path">The path of the font.</param>
|
/// <param name="path">The path of the font.</param>
|
||||||
public Font(string path)
|
/// <param name="episodeSlug">The slug of the episode that contains this font.</param>
|
||||||
|
public Font(string path, string episodeSlug)
|
||||||
{
|
{
|
||||||
Slug = Utility.ToSlug(PathIO.GetFileNameWithoutExtension(path));
|
Slug = Utility.ToSlug(PathIO.GetFileNameWithoutExtension(path));
|
||||||
Path = path;
|
Path = path;
|
||||||
File = PathIO.GetFileName(path);
|
File = PathIO.GetFileName(path);
|
||||||
Format = PathIO.GetExtension(path).Replace(".", string.Empty);
|
Format = PathIO.GetExtension(path).Replace(".", string.Empty);
|
||||||
|
Link = $"/watch/{episodeSlug}/font/{Slug}.{Format}";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,31 @@
|
|||||||
|
// Kyoo - A portable and vast media library solution.
|
||||||
|
// Copyright (c) Kyoo.
|
||||||
|
//
|
||||||
|
// See AUTHORS.md and LICENSE file in the project root for full license information.
|
||||||
|
//
|
||||||
|
// Kyoo is free software: you can redistribute it and/or modify
|
||||||
|
// it under the terms of the GNU General Public License as published by
|
||||||
|
// the Free Software Foundation, either version 3 of the License, or
|
||||||
|
// any later version.
|
||||||
|
//
|
||||||
|
// Kyoo is distributed in the hope that it will be useful,
|
||||||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
// GNU General Public License for more details.
|
||||||
|
//
|
||||||
|
// You should have received a copy of the GNU General Public License
|
||||||
|
// along with Kyoo. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
namespace Kyoo.Abstractions.Models
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// An interface to represent resources that should have a link field in their return values (like videos).
|
||||||
|
/// </summary>
|
||||||
|
public interface ILink
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// The link to return, in most cases this should be a string.
|
||||||
|
/// </summary>
|
||||||
|
public object Link { get; }
|
||||||
|
}
|
||||||
|
}
|
@ -55,7 +55,7 @@ namespace Kyoo.Abstractions.Models
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// A video, audio or subtitle track for an episode.
|
/// A video, audio or subtitle track for an episode.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class Track : IResource
|
public class Track : IResource, ILink
|
||||||
{
|
{
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
public int ID { get; set; }
|
public int ID { get; set; }
|
||||||
@ -66,9 +66,9 @@ namespace Kyoo.Abstractions.Models
|
|||||||
{
|
{
|
||||||
get
|
get
|
||||||
{
|
{
|
||||||
string type = Type.ToString().ToLower();
|
string type = Type.ToString().ToLowerInvariant();
|
||||||
string index = TrackIndex != 0 ? $"-{TrackIndex}" : string.Empty;
|
string index = TrackIndex != 0 ? $"-{TrackIndex}" : string.Empty;
|
||||||
string episode = _episodeSlug ?? Episode?.Slug ?? EpisodeID.ToString();
|
string episode = _episodeSlug ?? Episode?.Slug ?? EpisodeID.ToString(CultureInfo.InvariantCulture);
|
||||||
return $"{episode}.{Language ?? "und"}{index}{(IsForced ? ".forced" : string.Empty)}.{type}";
|
return $"{episode}.{Language ?? "und"}{index}{(IsForced ? ".forced" : string.Empty)}.{type}";
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -92,7 +92,7 @@ namespace Kyoo.Abstractions.Models
|
|||||||
Language = match.Groups["lang"].Value;
|
Language = match.Groups["lang"].Value;
|
||||||
if (Language == "und")
|
if (Language == "und")
|
||||||
Language = null;
|
Language = null;
|
||||||
TrackIndex = match.Groups["index"].Success ? int.Parse(match.Groups["index"].Value) : 0;
|
TrackIndex = match.Groups["index"].Success ? int.Parse(match.Groups["index"].Value, CultureInfo.InvariantCulture) : 0;
|
||||||
IsForced = match.Groups["forced"].Success;
|
IsForced = match.Groups["forced"].Success;
|
||||||
Type = Enum.Parse<StreamType>(match.Groups["type"].Value, true);
|
Type = Enum.Parse<StreamType>(match.Groups["type"].Value, true);
|
||||||
}
|
}
|
||||||
@ -198,6 +198,9 @@ namespace Kyoo.Abstractions.Models
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
[SerializeIgnore] private Episode _episode;
|
[SerializeIgnore] private Episode _episode;
|
||||||
|
|
||||||
|
/// <inheritdoc/>
|
||||||
|
public object Link => Type == StreamType.Subtitle ? $"/subtitle/{Slug}" : null;
|
||||||
|
|
||||||
// Converting mkv track language to c# system language tag.
|
// Converting mkv track language to c# system language tag.
|
||||||
private static string _GetLanguage(string mkvLanguage)
|
private static string _GetLanguage(string mkvLanguage)
|
||||||
{
|
{
|
||||||
|
@ -33,7 +33,7 @@ namespace Kyoo.Abstractions.Models
|
|||||||
/// Information about tracks and display information that could be used by the player.
|
/// Information about tracks and display information that could be used by the player.
|
||||||
/// This contains mostly data from an <see cref="Episode"/> with another form.
|
/// This contains mostly data from an <see cref="Episode"/> with another form.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class WatchItem : CustomTypeDescriptor, IThumbnails
|
public class WatchItem : CustomTypeDescriptor, IThumbnails, ILink
|
||||||
{
|
{
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// The ID of the episode associated with this item.
|
/// The ID of the episode associated with this item.
|
||||||
@ -136,6 +136,13 @@ namespace Kyoo.Abstractions.Models
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public ICollection<Chapter> Chapters { get; set; }
|
public ICollection<Chapter> Chapters { get; set; }
|
||||||
|
|
||||||
|
/// <inheritdoc/>
|
||||||
|
public object Link => new
|
||||||
|
{
|
||||||
|
Direct = $"/video/direct/{Slug}",
|
||||||
|
Transmux = $"/video/transmux/{Slug}/master.m3u8",
|
||||||
|
};
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Create a <see cref="WatchItem"/> from an <see cref="Episode"/>.
|
/// Create a <see cref="WatchItem"/> from an <see cref="Episode"/>.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
@ -233,7 +233,7 @@ namespace Kyoo.Core.Controllers
|
|||||||
{
|
{
|
||||||
string path = _files.Combine(await _files.GetExtraDirectory(episode), "Attachments");
|
string path = _files.Combine(await _files.GetExtraDirectory(episode), "Attachments");
|
||||||
return (await _files.ListFiles(path))
|
return (await _files.ListFiles(path))
|
||||||
.Select(x => new Font(x))
|
.Select(x => new Font(x, episode.Slug))
|
||||||
.ToArray();
|
.ToArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -245,7 +245,7 @@ namespace Kyoo.Core.Controllers
|
|||||||
.FirstOrDefault(x => Utility.ToSlug(Path.GetFileNameWithoutExtension(x)) == slug);
|
.FirstOrDefault(x => Utility.ToSlug(Path.GetFileNameWithoutExtension(x)) == slug);
|
||||||
if (font == null)
|
if (font == null)
|
||||||
return null;
|
return null;
|
||||||
return new Font(font);
|
return new Font(font, episode.Slug);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
|
@ -20,7 +20,7 @@
|
|||||||
|
|
||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
import { zdate } from "~/utils/zod";
|
import { zdate } from "~/utils/zod";
|
||||||
import { ResourceP, ImagesP } from "../traits";
|
import { ResourceP, ImagesP, imageFn } from "../traits";
|
||||||
import { EpisodeP } from "./episode";
|
import { EpisodeP } from "./episode";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -59,6 +59,10 @@ export const TrackP = ResourceP.extend({
|
|||||||
* A user-friendly name for this track. It does not include the track type.
|
* A user-friendly name for this track. It does not include the track type.
|
||||||
*/
|
*/
|
||||||
displayName: z.string(),
|
displayName: z.string(),
|
||||||
|
/*
|
||||||
|
* The url of this track (only if this is a subtitle)..
|
||||||
|
*/
|
||||||
|
link: z.string().transform(imageFn).nullable(),
|
||||||
});
|
});
|
||||||
export type Track = z.infer<typeof TrackP>;
|
export type Track = z.infer<typeof TrackP>;
|
||||||
|
|
||||||
@ -78,7 +82,7 @@ export const FontP = z.object({
|
|||||||
/*
|
/*
|
||||||
* The url of the font.
|
* The url of the font.
|
||||||
*/
|
*/
|
||||||
link: z.string(),
|
link: z.string().transform(imageFn),
|
||||||
});
|
});
|
||||||
export type Font = z.infer<typeof FontP>;
|
export type Font = z.infer<typeof FontP>;
|
||||||
|
|
||||||
@ -105,13 +109,6 @@ const WatchMovieP = z.preprocess(
|
|||||||
if (!x) return x;
|
if (!x) return x;
|
||||||
|
|
||||||
x.name = x.title;
|
x.name = x.title;
|
||||||
x.link = {
|
|
||||||
direct: `/api/video/${x.slug}`,
|
|
||||||
};
|
|
||||||
x.fonts = x.fonts?.map((y: Font) => {
|
|
||||||
y.link = `/api/watch/${x.slug}/font/${y.slug}.${y.format}`;
|
|
||||||
return y;
|
|
||||||
})
|
|
||||||
return x;
|
return x;
|
||||||
},
|
},
|
||||||
ImagesP.extend({
|
ImagesP.extend({
|
||||||
@ -155,7 +152,8 @@ const WatchMovieP = z.preprocess(
|
|||||||
* The links to the videos of this watch item.
|
* The links to the videos of this watch item.
|
||||||
*/
|
*/
|
||||||
link: z.object({
|
link: z.object({
|
||||||
direct: z.string(),
|
direct: z.string().transform(imageFn),
|
||||||
|
transmux: z.string().transform(imageFn),
|
||||||
}),
|
}),
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
|
@ -20,7 +20,7 @@
|
|||||||
|
|
||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
|
|
||||||
const imageFn = (url: string) => (url.startsWith("/api") ? url : `/api${url}`);
|
export const imageFn = (url: string) => (url.startsWith("/api") ? url : `/api${url}`);
|
||||||
|
|
||||||
export const ImagesP = z.object({
|
export const ImagesP = z.object({
|
||||||
/**
|
/**
|
||||||
|
@ -21,7 +21,7 @@
|
|||||||
import { BoxProps } from "@mui/material";
|
import { BoxProps } from "@mui/material";
|
||||||
import { atom, useSetAtom } from "jotai";
|
import { atom, useSetAtom } from "jotai";
|
||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
import { RefObject, useCallback, useEffect, useRef, useState } from "react";
|
import { RefObject, useEffect, useRef } from "react";
|
||||||
import { Font, Track } from "~/models/resources/watch-item";
|
import { Font, Track } from "~/models/resources/watch-item";
|
||||||
import { bakedAtom } from "~/utils/jotai-utils";
|
import { bakedAtom } from "~/utils/jotai-utils";
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
@ -169,7 +169,7 @@ export const [_subtitleAtom, subtitleAtom] = bakedAtom<Track | null, { track: Tr
|
|||||||
track.kind = "subtitles";
|
track.kind = "subtitles";
|
||||||
track.label = value.track.displayName;
|
track.label = value.track.displayName;
|
||||||
if (value.track.language) track.srclang = value.track.language;
|
if (value.track.language) track.srclang = value.track.language;
|
||||||
track.src = `subtitle/${value.track.slug}.vtt`;
|
track.src = value.track.link!;
|
||||||
track.className = "subtitle_container";
|
track.className = "subtitle_container";
|
||||||
track.default = true;
|
track.default = true;
|
||||||
track.onload = () => {
|
track.onload = () => {
|
||||||
@ -184,7 +184,7 @@ export const [_subtitleAtom, subtitleAtom] = bakedAtom<Track | null, { track: Tr
|
|||||||
suboctoAtom,
|
suboctoAtom,
|
||||||
new SubtitleOctopus({
|
new SubtitleOctopus({
|
||||||
video: player.current,
|
video: player.current,
|
||||||
subUrl: `/api/subtitle/${value.track.slug}`,
|
subUrl: value.track.link!,
|
||||||
workerUrl: "/_next/static/chunks/subtitles-octopus-worker.js",
|
workerUrl: "/_next/static/chunks/subtitles-octopus-worker.js",
|
||||||
legacyWorkerUrl: "/_next/static/chunks/subtitles-octopus-worker-legacy.js",
|
legacyWorkerUrl: "/_next/static/chunks/subtitles-octopus-worker-legacy.js",
|
||||||
fonts: value.fonts?.map((x) => x.link),
|
fonts: value.fonts?.map((x) => x.link),
|
||||||
|
Loading…
x
Reference in New Issue
Block a user