use std::fmt; use derive_builder::Builder; use crate::tags::{ ExtInf, ExtXByteRange, ExtXDateRange, ExtXDiscontinuity, ExtXKey, ExtXMap, ExtXProgramDateTime, }; use crate::types::ProtocolVersion; use crate::{Encrypted, RequiredVersion}; #[derive(Debug, Clone, Builder)] #[builder(setter(into, strip_option))] /// Media segment. pub struct MediaSegment { #[builder(default)] /// Sets all [`ExtXKey`] tags. keys: Vec, #[builder(default)] /// Sets an [`ExtXMap`] tag. map_tag: Option, #[builder(default)] /// Sets an [`ExtXByteRange`] tag. byte_range_tag: Option, #[builder(default)] /// Sets an [`ExtXDateRange`] tag. date_range_tag: Option, #[builder(default)] /// Sets an [`ExtXDiscontinuity`] tag. discontinuity_tag: Option, #[builder(default)] /// Sets an [`ExtXProgramDateTime`] tag. program_date_time_tag: Option, /// Sets an [`ExtInf`] tag. inf_tag: ExtInf, /// Sets an `URI`. uri: String, } impl MediaSegment { /// Returns a Builder for a [`MasterPlaylist`]. /// /// [`MasterPlaylist`]: crate::MasterPlaylist pub fn builder() -> MediaSegmentBuilder { MediaSegmentBuilder::default() } /// Returns the `URI` of the media segment. pub const fn uri(&self) -> &String { &self.uri } /// Sets the `URI` of the media segment. pub fn set_uri(&mut self, value: T) -> &mut Self where T: Into, { self.uri = value.into(); self } /// Returns the [`ExtInf`] tag associated with the media segment. pub const fn inf_tag(&self) -> &ExtInf { &self.inf_tag } /// Sets the [`ExtInf`] tag associated with the media segment. pub fn set_inf_tag(&mut self, value: T) -> &mut Self where T: Into, { self.inf_tag = value.into(); self } /// Returns the [`ExtXByteRange`] tag associated with the media segment. pub const fn byte_range_tag(&self) -> Option { self.byte_range_tag } /// Sets the [`ExtXByteRange`] tag associated with the media segment. pub fn set_byte_range_tag(&mut self, value: Option) -> &mut Self where T: Into, { self.byte_range_tag = value.map(Into::into); self } /// Returns the [`ExtXDateRange`] tag associated with the media segment. pub const fn date_range_tag(&self) -> &Option { &self.date_range_tag } /// Sets the [`ExtXDateRange`] tag associated with the media segment. pub fn set_date_range_tag(&mut self, value: Option) -> &mut Self where T: Into, { self.date_range_tag = value.map(Into::into); self } /// Returns the [`ExtXDiscontinuity`] tag associated with the media segment. pub const fn discontinuity_tag(&self) -> Option { self.discontinuity_tag } /// Sets the [`ExtXDiscontinuity`] tag associated with the media segment. pub fn set_discontinuity_tag(&mut self, value: Option) -> &mut Self where T: Into, { self.discontinuity_tag = value.map(Into::into); self } /// Returns the [`ExtXProgramDateTime`] tag associated with the media /// segment. pub const fn program_date_time_tag(&self) -> Option { self.program_date_time_tag } /// Sets the [`ExtXProgramDateTime`] tag associated with the media /// segment. pub fn set_program_date_time_tag(&mut self, value: Option) -> &mut Self where T: Into, { self.program_date_time_tag = value.map(Into::into); self } /// Returns the [`ExtXMap`] tag associated with the media segment. pub const fn map_tag(&self) -> &Option { &self.map_tag } /// Sets the [`ExtXMap`] tag associated with the media segment. pub fn set_map_tag(&mut self, value: Option) -> &mut Self where T: Into, { self.map_tag = value.map(Into::into); self } } 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() ); } }