use std::io::{Read, Seek, SeekFrom}; use std::cmp; use std::cmp::Ordering; use super::Decoder; use super::conversions; use cpal::{self, Endpoint, Voice}; use hound::WavReader; use hound::WavSpec; pub struct WavDecoder { reader: conversions::AmplifierIterator + Send>>, voice: Voice, } impl WavDecoder { pub fn new(endpoint: &Endpoint, mut data: R) -> Result where R: Read + Seek + Send + 'static { if !is_wave(data.by_ref()) { return Err(data); } let reader = WavReader::new(data).unwrap(); let spec = reader.spec(); // choosing a format amongst the ones available let voice_format = { let mut formats_list = endpoint.get_supported_formats_list().unwrap().collect::>(); formats_list.sort_by(|f1, f2| { if (f1.samples_rate.0 % spec.sample_rate) == 0 && (f2.samples_rate.0 % spec.sample_rate) != 0 { return Ordering::Greater; } else if (f2.samples_rate.0 % spec.sample_rate) == 0 && (f1.samples_rate.0 % spec.sample_rate) != 0 { return Ordering::Less; } if f1.channels.len() >= spec.channels as usize && f1.channels.len() < f2.channels.len() { return Ordering::Greater; } else if f2.channels.len() >= spec.channels as usize && f2.channels.len() < f1.channels.len() { return Ordering::Less; } if f1.data_type == cpal::SampleFormat::I16 && f2.data_type != cpal::SampleFormat::I16 { return Ordering::Greater; } else if f2.data_type == cpal::SampleFormat::I16 && f1.data_type != cpal::SampleFormat::I16 { return Ordering::Less; } Ordering::Equal }); formats_list.into_iter().next().unwrap() }; let voice = Voice::new(endpoint, &voice_format).unwrap(); let reader = SamplesIterator { reader: reader, samples_read: 0 }; let reader = conversions::ChannelsCountConverter::new(reader, spec.channels, voice.get_channels()); let reader = conversions::SamplesRateConverter::new(reader, cpal::SamplesRate(spec.sample_rate), voice.get_samples_rate()); Ok(WavDecoder { reader: conversions::AmplifierIterator::new(Box::new(reader), 1.0), voice: voice, }) } } struct SamplesIterator where R: Read + Seek { reader: WavReader, samples_read: u32, } impl Iterator for SamplesIterator where R: Read + Seek { type Item = i16; #[inline] fn next(&mut self) -> Option { if let Some(value) = self.reader.samples().next() { self.samples_read += 1; Some(value.unwrap_or(0)) } else { None } } #[inline] fn size_hint(&self) -> (usize, Option) { let len = (self.reader.len() - self.samples_read) as usize; (len, Some(len)) } } impl ExactSizeIterator for SamplesIterator where R: Read + Seek {} /// Returns true if the stream contains WAV data, then resets it to where it was. fn is_wave(mut data: R) -> bool where R: Read + Seek { let stream_pos = data.seek(SeekFrom::Current(0)).unwrap(); if WavReader::new(data.by_ref()).is_err() { data.seek(SeekFrom::Start(stream_pos)).unwrap(); return false; } data.seek(SeekFrom::Start(stream_pos)).unwrap(); true } impl Decoder for WavDecoder { fn write(&mut self) -> u64 { let (min, _) = self.reader.size_hint(); let min = cmp::min(min, 10240); // using a minimal value so that filters get applied // quickly if min == 0 { // finished return 1000000000; } let len = { let mut buffer = self.voice.append_data(min); let len = buffer.len(); conversions::convert_and_write(self.reader.by_ref(), &mut buffer); len }; let duration = len as u64 * 1000000000 / self.voice.get_samples_rate().0 as u64; self.voice.play(); duration } fn set_volume(&mut self, value: f32) { self.reader.set_amplification(value); } }