scuffle_mp4/boxes/types/
trak.rs1use std::io;
2
3use bytes::{Buf, Bytes};
4
5use super::edts::Edts;
6use super::mdia::Mdia;
7use super::tkhd::Tkhd;
8use crate::boxes::DynBox;
9use crate::boxes::header::BoxHeader;
10use crate::boxes::traits::BoxType;
11
12#[derive(Debug, Clone, PartialEq)]
13pub struct Trak {
16 pub header: BoxHeader,
17 pub tkhd: Tkhd,
18 pub edts: Option<Edts>,
19 pub mdia: Mdia,
20 pub unknown: Vec<DynBox>,
21}
22
23impl Trak {
24 pub fn new(tkhd: Tkhd, edts: Option<Edts>, mdia: Mdia) -> Self {
25 Self {
26 header: BoxHeader::new(Self::NAME),
27 tkhd,
28 edts,
29 mdia,
30 unknown: Vec::new(),
31 }
32 }
33}
34
35impl BoxType for Trak {
36 const NAME: [u8; 4] = *b"trak";
37
38 fn demux(header: BoxHeader, data: Bytes) -> io::Result<Self> {
39 let mut reader = io::Cursor::new(data);
40 let mut tkhd = None;
41 let mut edts = None;
42 let mut mdia = None;
43 let mut unknown = Vec::new();
44
45 while reader.has_remaining() {
46 let dyn_box = DynBox::demux(&mut reader)?;
47
48 match dyn_box {
49 DynBox::Tkhd(b) => {
50 tkhd = Some(*b);
51 }
52 DynBox::Edts(b) => {
53 edts = Some(*b);
54 }
55 DynBox::Mdia(b) => {
56 mdia = Some(*b);
57 }
58 _ => {
59 unknown.push(dyn_box);
60 }
61 }
62 }
63
64 let tkhd = tkhd.ok_or_else(|| io::Error::new(io::ErrorKind::InvalidData, "trak box is missing tkhd box"))?;
65
66 let mdia = mdia.ok_or_else(|| io::Error::new(io::ErrorKind::InvalidData, "trak box is missing mdia box"))?;
67
68 Ok(Self {
69 header,
70 tkhd,
71 edts,
72 mdia,
73 unknown,
74 })
75 }
76
77 fn primitive_size(&self) -> u64 {
78 self.tkhd.size()
79 + self.edts.as_ref().map(|b| b.size()).unwrap_or(0)
80 + self.mdia.size()
81 + self.unknown.iter().map(|b| b.size()).sum::<u64>()
82 }
83
84 fn primitive_mux<T: io::Write>(&self, writer: &mut T) -> io::Result<()> {
85 self.tkhd.mux(writer)?;
86 if let Some(edts) = &self.edts {
87 edts.mux(writer)?;
88 }
89
90 self.mdia.mux(writer)?;
91
92 for box_ in &self.unknown {
93 box_.mux(writer)?;
94 }
95
96 Ok(())
97 }
98}