script/dom/audio/
audiotracklist.rs1use dom_struct::dom_struct;
6use js::context::JSContext;
7use script_bindings::cell::DomRefCell;
8use script_bindings::reflector::reflect_dom_object_with_cx;
9
10use crate::dom::audio::audiotrack::AudioTrack;
11use crate::dom::bindings::codegen::Bindings::AudioTrackListBinding::AudioTrackListMethods;
12use crate::dom::bindings::inheritance::Castable;
13use crate::dom::bindings::refcounted::Trusted;
14use crate::dom::bindings::reflector::DomGlobal;
15use crate::dom::bindings::root::{Dom, DomRoot};
16use crate::dom::bindings::str::DOMString;
17use crate::dom::eventtarget::EventTarget;
18use crate::dom::html::htmlmediaelement::HTMLMediaElement;
19use crate::dom::window::Window;
20
21#[dom_struct]
22pub(crate) struct AudioTrackList {
23 eventtarget: EventTarget,
24 tracks: DomRefCell<Vec<Dom<AudioTrack>>>,
25 media_element: Option<Dom<HTMLMediaElement>>,
26}
27
28impl AudioTrackList {
29 pub(crate) fn new_inherited(
30 tracks: &[&AudioTrack],
31 media_element: Option<&HTMLMediaElement>,
32 ) -> AudioTrackList {
33 AudioTrackList {
34 eventtarget: EventTarget::new_inherited(),
35 tracks: DomRefCell::new(tracks.iter().map(|track| Dom::from_ref(&**track)).collect()),
36 media_element: media_element.map(Dom::from_ref),
37 }
38 }
39
40 pub(crate) fn new(
41 cx: &mut JSContext,
42 window: &Window,
43 tracks: &[&AudioTrack],
44 media_element: Option<&HTMLMediaElement>,
45 ) -> DomRoot<AudioTrackList> {
46 reflect_dom_object_with_cx(
47 Box::new(AudioTrackList::new_inherited(tracks, media_element)),
48 window,
49 cx,
50 )
51 }
52
53 pub(crate) fn len(&self) -> usize {
54 self.tracks.borrow().len()
55 }
56
57 pub(crate) fn find(&self, track: &AudioTrack) -> Option<usize> {
58 self.tracks.borrow().iter().position(|t| &**t == track)
59 }
60
61 pub(crate) fn item(&self, idx: usize) -> Option<DomRoot<AudioTrack>> {
62 self.tracks
63 .borrow()
64 .get(idx)
65 .map(|track| DomRoot::from_ref(&**track))
66 }
67
68 pub(crate) fn enabled_index(&self) -> Option<usize> {
69 self.tracks
70 .borrow()
71 .iter()
72 .position(|track| track.enabled())
73 }
74
75 pub(crate) fn set_enabled(&self, idx: usize, value: bool) {
76 let track = match self.item(idx) {
77 Some(t) => t,
78 None => return,
79 };
80
81 if track.enabled() == value {
83 return;
84 }
85 track.set_enabled(value);
87 if let Some(media_element) = self.media_element.as_ref() {
88 media_element.set_audio_track(idx, value);
89 }
90
91 let global = &self.global();
93 let this = Trusted::new(self);
94 let task_manager = global.task_manager();
95 let task_source = task_manager.media_element_task_source();
96 task_source.queue(task!(media_track_change: move |cx| {
97 let this = this.root();
98 this.upcast::<EventTarget>().fire_event(cx, atom!("change"));
99 }));
100 }
101
102 pub(crate) fn add(&self, track: &AudioTrack) {
103 self.tracks.borrow_mut().push(Dom::from_ref(track));
104 track.add_track_list(self);
105 }
106
107 pub(crate) fn clear(&self) {
108 self.tracks
109 .borrow()
110 .iter()
111 .for_each(|t| t.remove_track_list());
112 self.tracks.borrow_mut().clear();
113 }
114}
115
116impl AudioTrackListMethods<crate::DomTypeHolder> for AudioTrackList {
117 fn Length(&self) -> u32 {
119 self.len() as u32
120 }
121
122 fn IndexedGetter(&self, idx: u32) -> Option<DomRoot<AudioTrack>> {
124 self.item(idx as usize)
125 }
126
127 fn GetTrackById(&self, id: DOMString) -> Option<DomRoot<AudioTrack>> {
129 self.tracks
130 .borrow()
131 .iter()
132 .find(|track| track.id() == id)
133 .map(|track| DomRoot::from_ref(&**track))
134 }
135
136 event_handler!(change, GetOnchange, SetOnchange);
138
139 event_handler!(addtrack, GetOnaddtrack, SetOnaddtrack);
141
142 event_handler!(removetrack, GetOnremovetrack, SetOnremovetrack);
144}