// yt - A fully featured command line YouTube client // // Copyright (C) 2024 Benedikt Peetz <benedikt.peetz@b-peetz.de> // SPDX-License-Identifier: GPL-3.0-or-later // // This file is part of Yt. // // You should have received a copy of the License along with this program. // If not, see <https://www.gnu.org/licenses/gpl-3.0.txt>. use std::{fmt::Write, path::PathBuf}; use url::Url; use crate::{app::App, storage::video_database::extractor_hash::ExtractorHash}; pub mod downloader; pub mod extractor_hash; pub mod getters; pub mod setters; #[derive(Debug)] pub struct Video { pub cache_path: Option<PathBuf>, pub description: Option<String>, pub duration: Option<f64>, pub extractor_hash: ExtractorHash, pub last_status_change: i64, /// The associated subscription this video was fetched from (null, when the video was `add`ed) pub parent_subscription_name: Option<String>, pub priority: i64, pub publish_date: Option<i64>, pub status: VideoStatus, /// The video is currently changing its state (for example from being `SELECT` to being `CACHE`) pub status_change: bool, pub thumbnail_url: Option<Url>, pub title: String, pub url: Url, } #[derive(Debug)] pub struct VideoOptions { pub yt_dlp: YtDlpOptions, pub mpv: MpvOptions, } impl VideoOptions { pub(crate) fn new(subtitle_langs: String, playback_speed: f64) -> Self { let yt_dlp = YtDlpOptions { subtitle_langs }; let mpv = MpvOptions { playback_speed }; Self { yt_dlp, mpv } } /// This will write out the options that are different from the defaults. /// Beware, that this does not set the priority. pub fn to_cli_flags(self, app: &App) -> String { let mut f = String::new(); if self.mpv.playback_speed != app.config.select.playback_speed { write!(f, " --speed '{}'", self.mpv.playback_speed).expect("Works"); } if self.yt_dlp.subtitle_langs != app.config.select.subtitle_langs { write!(f, " --subtitle-langs '{}'", self.yt_dlp.subtitle_langs).expect("Works"); } f.trim().to_owned() } } #[derive(Debug)] /// Additionally settings passed to mpv on watch pub struct MpvOptions { /// The playback speed. (1 is 100%, 2.7 is 270%, and so on) pub playback_speed: f64, } #[derive(Debug)] /// Additionally configuration options, passed to yt-dlp on download pub struct YtDlpOptions { /// In the form of `lang1,lang2,lang3` (e.g. `en,de,sv`) pub subtitle_langs: String, } /// # Video Lifetime (words in <brackets> are commands): /// <Pick> /// / \ /// <Watch> <Drop> -> Dropped // yt select /// | /// Cache // yt cache /// | /// Watched // yt watch #[derive(Default, Debug, PartialEq, Eq, PartialOrd, Ord)] pub enum VideoStatus { #[default] Pick, /// The video has been select to be watched Watch, /// The video has been cached and is ready to be watched Cached, /// The video has been watched Watched, /// The video has been select to be dropped Drop, /// The video has been dropped Dropped, } impl VideoStatus { pub fn as_command(&self) -> &str { // NOTE: Keep the serialize able variants synced with the main `select` function <2024-06-14> match self { VideoStatus::Pick => "pick", VideoStatus::Watch => "watch", VideoStatus::Cached => "watch", VideoStatus::Watched => "watch", VideoStatus::Drop => "drop", VideoStatus::Dropped => "drop", } } pub fn as_db_integer(&self) -> i64 { // These numbers should not change their mapping! // Oh, and keep them in sync with the SQLite check constraint. match self { VideoStatus::Pick => 0, VideoStatus::Watch => 1, VideoStatus::Cached => 2, VideoStatus::Watched => 3, VideoStatus::Drop => 4, VideoStatus::Dropped => 5, } } pub fn from_db_integer(num: i64) -> Self { match num { 0 => Self::Pick, 1 => Self::Watch, 2 => Self::Cached, 3 => Self::Watched, 4 => Self::Drop, 5 => Self::Dropped, other => unreachable!( "The database returned a enum discriminator, unknown to us: '{}'", other ), } } pub fn as_str(&self) -> &'static str { match self { VideoStatus::Pick => "Pick", VideoStatus::Watch => "Watch", VideoStatus::Cached => "Cache", VideoStatus::Watched => "Watched", VideoStatus::Drop => "Drop", VideoStatus::Dropped => "Dropped", } } }