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
123
124
125
126
127

use {
    std::sync::{Arc,Mutex},
    self::super::{
        winrt_midi::*,
        wasapi::*,
        media_foundation::*,        
    },
    crate::{
        cx::Cx,
        audio::*,
        midi::*,
        video::*,
        thread::Signal,
        event::Event,
        media_api::CxMediaApi,

    }
};

#[derive(Default)]
pub struct CxWindowsMedia{
    pub (crate) winrt_midi: Option<Arc<Mutex<WinRTMidiAccess >> >,
    pub (crate) wasapi: Option<Arc<Mutex<WasapiAccess >> >,
    pub (crate) media_foundation: Option<Arc<Mutex<MediaFoundationAccess >> >,
    pub (crate) wasapi_change: Signal,
    pub (crate) media_foundation_change: Signal,
    pub (crate) winrt_midi_change: Signal,
}

impl Cx {
    pub (crate) fn handle_media_signals(&mut self) {
        if self.os.media.winrt_midi_change.check_and_clear(){
            let descs = self.os.media.winrt_midi().lock().unwrap().get_updated_descs();
            self.call_event_handler(&Event::MidiPorts(MidiPortsEvent {
                descs,
            }));
        }
        if self.os.media.wasapi_change.check_and_clear(){
            let descs = self.os.media.wasapi().lock().unwrap().get_updated_descs();
            self.call_event_handler(&Event::AudioDevices(AudioDevicesEvent{
                descs
            }));
        }
        if self.os.media.media_foundation_change.check_and_clear(){
            let descs = self.os.media.media_foundation().lock().unwrap().get_updated_descs();
            self.call_event_handler(&Event::VideoInputs(VideoInputsEvent{
                descs
            }));
        }
    }
}

impl CxWindowsMedia {
    
    pub fn winrt_midi(&mut self) -> Arc<Mutex<WinRTMidiAccess >> {
        if self.winrt_midi.is_none() {
            self.winrt_midi = Some(WinRTMidiAccess::new(self.winrt_midi_change.clone()));
        }
        self.winrt_midi.as_ref().unwrap().clone()
    }
    
    pub fn wasapi(&mut self) -> Arc<Mutex<WasapiAccess >> {
        if self.wasapi.is_none() {
            self.wasapi = Some(WasapiAccess::new(self.wasapi_change.clone()));
        }
        self.wasapi.as_ref().unwrap().clone()
    }
    
    pub fn media_foundation(&mut self) -> Arc<Mutex<MediaFoundationAccess >> {
        if self.media_foundation.is_none() {
            self.media_foundation = Some(MediaFoundationAccess::new(self.media_foundation_change.clone()));
        }
        self.media_foundation.as_ref().unwrap().clone()
    }
}

impl CxMediaApi for Cx {
    
    fn midi_input(&mut self) -> MidiInput {
        self.os.media.winrt_midi().lock().unwrap().create_midi_input()
    }
    
    fn midi_output(&mut self)->MidiOutput{
        MidiOutput(Some(OsMidiOutput(self.os.media.winrt_midi())))
    }

    fn midi_reset(&mut self){
        self.os.media.winrt_midi().lock().unwrap().midi_reset();
    }

    fn use_midi_inputs(&mut self, ports: &[MidiPortId]) {
        self.os.media.winrt_midi().lock().unwrap().use_midi_inputs(ports);
    }
    
    fn use_midi_outputs(&mut self, ports: &[MidiPortId]) {
        self.os.media.winrt_midi().lock().unwrap().use_midi_outputs(ports);
    }

    fn use_audio_inputs(&mut self, devices: &[AudioDeviceId]) {
        self.os.media.wasapi().lock().unwrap().use_audio_inputs(devices);
    }
    
    fn use_audio_outputs(&mut self, devices: &[AudioDeviceId]) {
        self.os.media.wasapi().lock().unwrap().use_audio_outputs(devices);
    }
    
    fn audio_output_box(&mut self, index:usize, f: AudioOutputFn) {
        *self.os.media.wasapi().lock().unwrap().audio_output_cb[index].lock().unwrap() = Some(f);
    }
    
    fn audio_input_box(&mut self, index:usize, f: AudioInputFn) {
        *self.os.media.wasapi().lock().unwrap().audio_input_cb[index].lock().unwrap() = Some(f);
    }
    
    fn video_input_box(&mut self, index:usize, f: VideoInputFn){
        *self.os.media.media_foundation().lock().unwrap().video_input_cb[index].lock().unwrap() = Some(f);
    }

    fn use_video_input(&mut self, inputs:&[(VideoInputId, VideoFormatId)]){
        self.os.media.media_foundation().lock().unwrap().use_video_input(inputs);
    }

}