1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
use std::cmp;
use std::time::Duration;

use crate::source::uniform::UniformSourceIterator;
use crate::{Sample, Source};
use cpal::{FromSample, Sample as CpalSample};

/// Internal function that builds a `Mix` object.
pub fn mix<I1, I2>(input1: I1, input2: I2) -> Mix<I1, I2>
where
    I1: Source,
    I1::Item: FromSample<I2::Item> + Sample,
    I2: Source,
    I2::Item: Sample,
{
    let channels = input1.channels();
    let rate = input1.sample_rate();

    Mix {
        input1: UniformSourceIterator::new(input1, channels, rate),
        input2: UniformSourceIterator::new(input2, channels, rate),
    }
}

/// Filter that modifies each sample by a given value.
#[derive(Clone)]
pub struct Mix<I1, I2>
where
    I1: Source,
    I1::Item: FromSample<I2::Item> + Sample,
    I2: Source,
    I2::Item: Sample,
{
    input1: UniformSourceIterator<I1, I1::Item>,
    input2: UniformSourceIterator<I2, I2::Item>,
}

impl<I1, I2> Iterator for Mix<I1, I2>
where
    I1: Source,
    I1::Item: FromSample<I2::Item> + Sample,
    I2: Source,
    I2::Item: Sample,
{
    type Item = I1::Item;

    #[inline]
    fn next(&mut self) -> Option<I1::Item> {
        let s1 = self.input1.next();
        let s2 = self.input2.next();

        match (s1, s2) {
            (Some(s1), Some(s2)) => Some(s1.saturating_add(CpalSample::from_sample(s2))),
            (Some(s1), None) => Some(s1),
            (None, Some(s2)) => Some(CpalSample::from_sample(s2)),
            (None, None) => None,
        }
    }

    #[inline]
    fn size_hint(&self) -> (usize, Option<usize>) {
        let s1 = self.input1.size_hint();
        let s2 = self.input2.size_hint();

        let min = cmp::max(s1.0, s2.0);
        let max = match (s1.1, s2.1) {
            (Some(s1), Some(s2)) => Some(cmp::max(s1, s2)),
            _ => None,
        };

        (min, max)
    }
}

impl<I1, I2> ExactSizeIterator for Mix<I1, I2>
where
    I1: Source + ExactSizeIterator,
    I1::Item: FromSample<I2::Item> + Sample,
    I2: Source + ExactSizeIterator,
    I2::Item: Sample,
{
}

impl<I1, I2> Source for Mix<I1, I2>
where
    I1: Source,
    I1::Item: FromSample<I2::Item> + Sample,
    I2: Source,
    I2::Item: Sample,
{
    #[inline]
    fn current_frame_len(&self) -> Option<usize> {
        let f1 = self.input1.current_frame_len();
        let f2 = self.input2.current_frame_len();

        match (f1, f2) {
            (Some(f1), Some(f2)) => Some(cmp::min(f1, f2)),
            _ => None,
        }
    }

    #[inline]
    fn channels(&self) -> u16 {
        self.input1.channels()
    }

    #[inline]
    fn sample_rate(&self) -> u32 {
        self.input1.sample_rate()
    }

    #[inline]
    fn total_duration(&self) -> Option<Duration> {
        let f1 = self.input1.total_duration();
        let f2 = self.input2.total_duration();

        match (f1, f2) {
            (Some(f1), Some(f2)) => Some(cmp::max(f1, f2)),
            _ => None,
        }
    }
}