use std::fmt; use derive_builder::Builder; use shorthand::ShortHand; use crate::tags::{ ExtInf, ExtXByteRange, ExtXDateRange, ExtXDiscontinuity, ExtXKey, ExtXMap, ExtXProgramDateTime, }; use crate::types::ProtocolVersion; use crate::{Encrypted, RequiredVersion}; /// Media segment. #[derive(ShortHand, Debug, Clone, Builder, PartialEq, PartialOrd)] #[builder(setter(into, strip_option))] #[shorthand(enable(must_use, get_mut, collection_magic))] pub struct MediaSegment { /// All [`ExtXKey`] tags. #[builder(default)] keys: Vec, /// The [`ExtXMap`] tag associated with the media segment. #[builder(default)] map_tag: Option, /// The [`ExtXByteRange`] tag associated with the [`MediaSegment`]. #[builder(default)] byte_range_tag: Option, /// The [`ExtXDateRange`] tag associated with the media segment. #[builder(default)] date_range_tag: Option, /// The [`ExtXDiscontinuity`] tag associated with the media segment. #[builder(default)] discontinuity_tag: Option, /// The [`ExtXProgramDateTime`] tag associated with the media /// segment. #[builder(default)] program_date_time_tag: Option, /// The [`ExtInf`] tag associated with the [`MediaSegment`]. inf_tag: ExtInf, /// The `URI` of the [`MediaSegment`]. #[shorthand(enable(into))] uri: String, } impl MediaSegment { /// Returns a builder for a [`MasterPlaylist`]. /// /// [`MasterPlaylist`]: crate::MasterPlaylist pub fn builder() -> MediaSegmentBuilder { MediaSegmentBuilder::default() } } impl MediaSegmentBuilder { /// Pushes an [`ExtXKey`] tag. pub fn push_key_tag>(&mut self, value: VALUE) -> &mut Self { if let Some(key_tags) = &mut self.keys { key_tags.push(value.into()); } else { self.keys = Some(vec![value.into()]); } self } } impl fmt::Display for MediaSegment { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { for value in &self.keys { writeln!(f, "{}", value)?; } if let Some(value) = &self.map_tag { writeln!(f, "{}", value)?; } if let Some(value) = &self.byte_range_tag { writeln!(f, "{}", value)?; } if let Some(value) = &self.date_range_tag { writeln!(f, "{}", value)?; } if let Some(value) = &self.discontinuity_tag { writeln!(f, "{}", value)?; } if let Some(value) = &self.program_date_time_tag { writeln!(f, "{}", value)?; } writeln!(f, "{}", self.inf_tag)?; // TODO: there might be a `,` missing writeln!(f, "{}", self.uri)?; Ok(()) } } impl RequiredVersion for MediaSegment { fn required_version(&self) -> ProtocolVersion { required_version![ self.keys, self.map_tag, self.byte_range_tag, self.date_range_tag, self.discontinuity_tag, self.program_date_time_tag, self.inf_tag ] } } impl Encrypted for MediaSegment { fn keys(&self) -> &Vec { &self.keys } fn keys_mut(&mut self) -> &mut Vec { &mut self.keys } } #[cfg(test)] mod tests { use super::*; use pretty_assertions::assert_eq; use std::time::Duration; #[test] fn test_display() { assert_eq!( MediaSegment::builder() .keys(vec![ExtXKey::empty()]) .map_tag(ExtXMap::new("https://www.example.com/")) .byte_range_tag(ExtXByteRange::new(20, Some(5))) //.date_range_tag() // TODO! .discontinuity_tag(ExtXDiscontinuity) .inf_tag(ExtInf::new(Duration::from_secs(4))) .uri("http://www.uri.com/") .build() .unwrap() .to_string(), "#EXT-X-KEY:METHOD=NONE\n\ #EXT-X-MAP:URI=\"https://www.example.com/\"\n\ #EXT-X-BYTERANGE:20@5\n\ #EXT-X-DISCONTINUITY\n\ #EXTINF:4,\n\ http://www.uri.com/\n" .to_string() ); } }