use byteorder::{BigEndian, ReadBytesExt, WriteBytesExt}; use serde::Serialize; use std::io::{Read, Seek, Write}; use crate::mp4box::*; #[derive(Debug, Clone, PartialEq, Default, Serialize)] pub struct FtypBox { pub major_brand: FourCC, pub minor_version: u32, pub compatible_brands: Vec, } impl FtypBox { pub fn get_type(&self) -> BoxType { BoxType::FtypBox } pub fn get_size(&self) -> u64 { HEADER_SIZE + 8 + (4 * self.compatible_brands.len() as u64) } } impl Mp4Box for FtypBox { fn box_type(&self) -> BoxType { self.get_type() } fn box_size(&self) -> u64 { self.get_size() } fn to_json(&self) -> Result { Ok(serde_json::to_string(&self).unwrap()) } fn summary(&self) -> Result { let mut compatible_brands = Vec::new(); for brand in self.compatible_brands.iter() { compatible_brands.push(brand.to_string()); } let s = format!( "major_brand={} minor_version={} compatible_brands={}", self.major_brand, self.minor_version, compatible_brands.join("-") ); Ok(s) } } impl ReadBox<&mut R> for FtypBox { fn read_box(reader: &mut R, size: u64) -> Result { let start = box_start(reader)?; let major = reader.read_u32::()?; let minor = reader.read_u32::()?; if size % 4 != 0 { return Err(Error::InvalidData("invalid ftyp size")); } let brand_count = (size - 16) / 4; // header + major + minor let mut brands = Vec::new(); for _ in 0..brand_count { let b = reader.read_u32::()?; brands.push(From::from(b)); } skip_bytes_to(reader, start + size)?; Ok(FtypBox { major_brand: From::from(major), minor_version: minor, compatible_brands: brands, }) } } impl WriteBox<&mut W> for FtypBox { fn write_box(&self, writer: &mut W) -> Result { let size = self.box_size(); BoxHeader::new(self.box_type(), size).write(writer)?; writer.write_u32::((&self.major_brand).into())?; writer.write_u32::(self.minor_version)?; for b in self.compatible_brands.iter() { writer.write_u32::(b.into())?; } Ok(size) } } #[cfg(test)] mod tests { use super::*; use crate::mp4box::BoxHeader; use std::io::Cursor; #[test] fn test_ftyp() { let src_box = FtypBox { major_brand: str::parse("isom").unwrap(), minor_version: 0, compatible_brands: vec![ str::parse("isom").unwrap(), str::parse("iso2").unwrap(), str::parse("avc1").unwrap(), str::parse("mp41").unwrap(), ], }; let mut buf = Vec::new(); src_box.write_box(&mut buf).unwrap(); assert_eq!(buf.len(), src_box.box_size() as usize); let mut reader = Cursor::new(&buf); let header = BoxHeader::read(&mut reader).unwrap(); assert_eq!(header.name, BoxType::FtypBox); assert_eq!(src_box.box_size(), header.size); let dst_box = FtypBox::read_box(&mut reader, header.size).unwrap(); assert_eq!(src_box, dst_box); } }