mirror of
https://github.com/zoriya/Kyoo.git
synced 2025-12-09 22:55:12 -05:00
215 lines
5.4 KiB
TypeScript
215 lines
5.4 KiB
TypeScript
/*
|
|
* 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/>.
|
|
*/
|
|
|
|
import { z } from "zod";
|
|
import { zdate } from "../utils";
|
|
import { ResourceP, ImagesP, imageFn } from "../traits";
|
|
import { EpisodeP } from "./episode";
|
|
|
|
/**
|
|
* A video, audio or subtitle track for an episode.
|
|
*/
|
|
export const TrackP = ResourceP.extend({
|
|
/**
|
|
* The title of the stream.
|
|
*/
|
|
title: z.string().nullable(),
|
|
/**
|
|
* The language of this stream (as a ISO-639-2 language code)
|
|
*/
|
|
language: z.string().nullable(),
|
|
/**
|
|
* The codec of this stream.
|
|
*/
|
|
codec: z.string(),
|
|
/**
|
|
* Is this stream the default one of it's type?
|
|
*/
|
|
isDefault: z.boolean(),
|
|
/**
|
|
* Is this stream tagged as forced?
|
|
*/
|
|
isForced: z.boolean(),
|
|
/**
|
|
* Is this track extern to the episode's file?
|
|
*/
|
|
isExternal: z.boolean(),
|
|
/**
|
|
* The index of this track on the episode.
|
|
*/
|
|
trackIndex: z.number(),
|
|
/**
|
|
* A user-friendly name for this track. It does not include the track type.
|
|
*/
|
|
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 const FontP = z.object({
|
|
/*
|
|
* A human-readable identifier, used in the URL.
|
|
*/
|
|
slug: z.string(),
|
|
/*
|
|
* The name of the font file (with the extension).
|
|
*/
|
|
file: z.string(),
|
|
/*
|
|
* The format of this font (the extension).
|
|
*/
|
|
format: z.string(),
|
|
/*
|
|
* The url of the font.
|
|
*/
|
|
link: z.string().transform(imageFn),
|
|
});
|
|
export type Font = z.infer<typeof FontP>;
|
|
|
|
export const ChapterP = z.object({
|
|
/**
|
|
* The start time of the chapter (in second from the start of the episode).
|
|
*/
|
|
startTime: z.number(),
|
|
/**
|
|
* The end time of the chapter (in second from the start of the episode).
|
|
*/
|
|
endTime: z.number(),
|
|
/**
|
|
* The name of this chapter. This should be a human-readable name that could be presented to the
|
|
* user. There should be well-known chapters name for commonly used chapters. For example, use
|
|
* "Opening" for the introduction-song and "Credits" for the end chapter with credits.
|
|
*/
|
|
name: z.string(),
|
|
});
|
|
export type Chapter = z.infer<typeof ChapterP>;
|
|
|
|
const WatchMovieP = z.preprocess(
|
|
(x: any) => {
|
|
if (!x) return x;
|
|
|
|
x.name = x.title;
|
|
return x;
|
|
},
|
|
ImagesP.extend({
|
|
/**
|
|
* The slug of this episode.
|
|
*/
|
|
slug: z.string(),
|
|
/**
|
|
* The title of this episode.
|
|
*/
|
|
name: z.string().nullable(),
|
|
/**
|
|
* The sumarry of this episode.
|
|
*/
|
|
overview: z.string().nullable(),
|
|
/**
|
|
* The release date of this episode. It can be null if unknown.
|
|
*/
|
|
releaseDate: zdate().nullable(),
|
|
/**
|
|
* The container of the video file of this episode. Common containers are mp4, mkv, avi and so
|
|
* on.
|
|
*/
|
|
container: z.string(),
|
|
/**
|
|
* The video track. See Track for more information.
|
|
*/
|
|
video: TrackP,
|
|
/**
|
|
* The list of audio tracks. See Track for more information.
|
|
*/
|
|
audios: z.array(TrackP),
|
|
/**
|
|
* The list of subtitles tracks. See Track for more information.
|
|
*/
|
|
subtitles: z.array(TrackP),
|
|
/**
|
|
* The list of fonts that can be used to display subtitles.
|
|
*/
|
|
fonts: z.array(FontP),
|
|
/**
|
|
* The list of chapters. See Chapter for more information.
|
|
*/
|
|
chapters: z.array(ChapterP),
|
|
/**
|
|
* The links to the videos of this watch item.
|
|
*/
|
|
link: z.object({
|
|
direct: z.string().transform(imageFn),
|
|
hls: z.string().transform(imageFn),
|
|
}),
|
|
}),
|
|
);
|
|
|
|
const WatchEpisodeP = WatchMovieP.and(
|
|
z.object({
|
|
/**
|
|
* The ID of the episode associated with this item.
|
|
*/
|
|
episodeID: z.number(),
|
|
/**
|
|
* The title of the show containing this episode.
|
|
*/
|
|
showTitle: z.string(),
|
|
/**
|
|
* The slug of the show containing this episode
|
|
*/
|
|
showSlug: z.string(),
|
|
/**
|
|
* The season in witch this episode is in.
|
|
*/
|
|
seasonNumber: z.number().nullable(),
|
|
/**
|
|
* The number of this episode is it's season.
|
|
*/
|
|
episodeNumber: z.number().nullable(),
|
|
/**
|
|
* The absolute number of this episode. It's an episode number that is not reset to 1 after a
|
|
* new season.
|
|
*/
|
|
absoluteNumber: z.number().nullable(),
|
|
/**
|
|
* The episode that come before this one if you follow usual watch orders. If this is the first
|
|
* episode or this is a movie, it will be null.
|
|
*/
|
|
previousEpisode: EpisodeP.nullable(),
|
|
/**
|
|
* The episode that come after this one if you follow usual watch orders. If this is the last
|
|
* aired episode or this is a movie, it will be null.
|
|
*/
|
|
nextEpisode: EpisodeP.nullable(),
|
|
}),
|
|
);
|
|
|
|
export const WatchItemP = z.union([
|
|
WatchMovieP.and(z.object({ isMovie: z.literal(true) })),
|
|
WatchEpisodeP.and(z.object({ isMovie: z.literal(false) })),
|
|
]);
|
|
|
|
/**
|
|
* A watch item for a movie or an episode
|
|
*/
|
|
export type WatchItem = z.infer<typeof WatchItemP>;
|