#[cfg(feature = "with-tremor")] extern crate librespot_tremor as vorbis; #[cfg(not(feature = "with-tremor"))] extern crate vorbis; use super::{AudioDecoder, AudioError, AudioPacket}; use std::error; use std::fmt; use std::io::{Read, Seek}; pub struct VorbisDecoder(vorbis::Decoder); pub struct VorbisError(vorbis::VorbisError); impl VorbisDecoder where R: Read + Seek, { pub fn new(input: R) -> Result, VorbisError> { Ok(VorbisDecoder(vorbis::Decoder::new(input)?)) } } impl AudioDecoder for VorbisDecoder where R: Read + Seek, { #[cfg(not(feature = "with-tremor"))] fn seek(&mut self, ms: i64) -> Result<(), AudioError> { self.0.time_seek(ms as f64 / 1000f64)?; Ok(()) } #[cfg(feature = "with-tremor")] fn seek(&mut self, ms: i64) -> Result<(), AudioError> { self.0.time_seek(ms)?; Ok(()) } fn next_packet(&mut self) -> Result, AudioError> { loop { match self.0.packets().next() { Some(Ok(packet)) => { // Losslessly represent [-32768, 32767] to [-1.0, 1.0] while maintaining DC linearity. return Ok(Some(AudioPacket::Samples( packet .data .iter() .map(|sample| { ((*sample as f64 + 0.5) / (std::i16::MAX as f64 + 0.5)) as f32 }) .collect(), ))); } None => return Ok(None), Some(Err(vorbis::VorbisError::Hole)) => (), Some(Err(err)) => return Err(err.into()), } } } } impl From for VorbisError { fn from(err: vorbis::VorbisError) -> VorbisError { VorbisError(err) } } impl fmt::Debug for VorbisError { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { fmt::Debug::fmt(&self.0, f) } } impl fmt::Display for VorbisError { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { fmt::Display::fmt(&self.0, f) } } impl error::Error for VorbisError { fn source(&self) -> Option<&(dyn error::Error + 'static)> { error::Error::source(&self.0) } } impl From for AudioError { fn from(err: vorbis::VorbisError) -> AudioError { AudioError::VorbisError(VorbisError(err)) } }