mirror of
https://github.com/servo/servo.git
synced 2025-06-04 07:35:36 +00:00
* script: Move Reflector to script_bindings. Signed-off-by: Josh Matthews <josh@joshmatthews.net> * script: Extract global() helper from DomObject into new trait. Move DomObject and related traits to script_bindings. Signed-off-by: Josh Matthews <josh@joshmatthews.net> --------- Signed-off-by: Josh Matthews <josh@joshmatthews.net>
228 lines
8.4 KiB
Rust
228 lines
8.4 KiB
Rust
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at https://mozilla.org/MPL/2.0/. */
|
|
|
|
use std::cell::Cell;
|
|
use std::rc::Rc;
|
|
use std::sync::{mpsc, Arc, Mutex};
|
|
use std::thread::Builder;
|
|
|
|
use base::id::PipelineId;
|
|
use dom_struct::dom_struct;
|
|
use js::rust::HandleObject;
|
|
use servo_media::audio::context::OfflineAudioContextOptions as ServoMediaOfflineAudioContextOptions;
|
|
|
|
use crate::dom::audiobuffer::{AudioBuffer, MAX_SAMPLE_RATE, MIN_SAMPLE_RATE};
|
|
use crate::dom::audionode::MAX_CHANNEL_COUNT;
|
|
use crate::dom::baseaudiocontext::{BaseAudioContext, BaseAudioContextOptions};
|
|
use crate::dom::bindings::cell::DomRefCell;
|
|
use crate::dom::bindings::codegen::Bindings::BaseAudioContextBinding::BaseAudioContext_Binding::BaseAudioContextMethods;
|
|
use crate::dom::bindings::codegen::Bindings::OfflineAudioContextBinding::{
|
|
OfflineAudioContextMethods, OfflineAudioContextOptions,
|
|
};
|
|
use crate::dom::bindings::error::{Error, Fallible};
|
|
use crate::dom::bindings::inheritance::Castable;
|
|
use crate::dom::bindings::num::Finite;
|
|
use crate::dom::bindings::refcounted::Trusted;
|
|
use crate::dom::bindings::reflector::{reflect_dom_object_with_proto, DomGlobal};
|
|
use crate::dom::bindings::root::DomRoot;
|
|
use crate::dom::event::{Event, EventBubbles, EventCancelable};
|
|
use crate::dom::offlineaudiocompletionevent::OfflineAudioCompletionEvent;
|
|
use crate::dom::promise::Promise;
|
|
use crate::dom::window::Window;
|
|
use crate::realms::InRealm;
|
|
use crate::script_runtime::CanGc;
|
|
|
|
#[dom_struct]
|
|
pub(crate) struct OfflineAudioContext {
|
|
context: BaseAudioContext,
|
|
channel_count: u32,
|
|
length: u32,
|
|
rendering_started: Cell<bool>,
|
|
#[ignore_malloc_size_of = "promises are hard"]
|
|
pending_rendering_promise: DomRefCell<Option<Rc<Promise>>>,
|
|
}
|
|
|
|
impl OfflineAudioContext {
|
|
#[cfg_attr(crown, allow(crown::unrooted_must_root))]
|
|
fn new_inherited(
|
|
channel_count: u32,
|
|
length: u32,
|
|
sample_rate: f32,
|
|
pipeline_id: PipelineId,
|
|
) -> Fallible<OfflineAudioContext> {
|
|
let options = ServoMediaOfflineAudioContextOptions {
|
|
channels: channel_count as u8,
|
|
length: length as usize,
|
|
sample_rate,
|
|
};
|
|
let context = BaseAudioContext::new_inherited(
|
|
BaseAudioContextOptions::OfflineAudioContext(options),
|
|
pipeline_id,
|
|
)?;
|
|
Ok(OfflineAudioContext {
|
|
context,
|
|
channel_count,
|
|
length,
|
|
rendering_started: Cell::new(false),
|
|
pending_rendering_promise: Default::default(),
|
|
})
|
|
}
|
|
|
|
#[cfg_attr(crown, allow(crown::unrooted_must_root))]
|
|
fn new(
|
|
window: &Window,
|
|
proto: Option<HandleObject>,
|
|
channel_count: u32,
|
|
length: u32,
|
|
sample_rate: f32,
|
|
can_gc: CanGc,
|
|
) -> Fallible<DomRoot<OfflineAudioContext>> {
|
|
if channel_count > MAX_CHANNEL_COUNT ||
|
|
channel_count == 0 ||
|
|
length == 0 ||
|
|
!(MIN_SAMPLE_RATE..=MAX_SAMPLE_RATE).contains(&sample_rate)
|
|
{
|
|
return Err(Error::NotSupported);
|
|
}
|
|
let pipeline_id = window.pipeline_id();
|
|
let context =
|
|
OfflineAudioContext::new_inherited(channel_count, length, sample_rate, pipeline_id)?;
|
|
Ok(reflect_dom_object_with_proto(
|
|
Box::new(context),
|
|
window,
|
|
proto,
|
|
can_gc,
|
|
))
|
|
}
|
|
}
|
|
|
|
impl OfflineAudioContextMethods<crate::DomTypeHolder> for OfflineAudioContext {
|
|
// https://webaudio.github.io/web-audio-api/#dom-offlineaudiocontext-offlineaudiocontext
|
|
fn Constructor(
|
|
window: &Window,
|
|
proto: Option<HandleObject>,
|
|
can_gc: CanGc,
|
|
options: &OfflineAudioContextOptions,
|
|
) -> Fallible<DomRoot<OfflineAudioContext>> {
|
|
OfflineAudioContext::new(
|
|
window,
|
|
proto,
|
|
options.numberOfChannels,
|
|
options.length,
|
|
*options.sampleRate,
|
|
can_gc,
|
|
)
|
|
}
|
|
|
|
// https://webaudio.github.io/web-audio-api/#dom-offlineaudiocontext-offlineaudiocontext-numberofchannels-length-samplerate
|
|
fn Constructor_(
|
|
window: &Window,
|
|
proto: Option<HandleObject>,
|
|
can_gc: CanGc,
|
|
number_of_channels: u32,
|
|
length: u32,
|
|
sample_rate: Finite<f32>,
|
|
) -> Fallible<DomRoot<OfflineAudioContext>> {
|
|
OfflineAudioContext::new(
|
|
window,
|
|
proto,
|
|
number_of_channels,
|
|
length,
|
|
*sample_rate,
|
|
can_gc,
|
|
)
|
|
}
|
|
|
|
// https://webaudio.github.io/web-audio-api/#dom-offlineaudiocontext-oncomplete
|
|
event_handler!(complete, GetOncomplete, SetOncomplete);
|
|
|
|
// https://webaudio.github.io/web-audio-api/#dom-offlineaudiocontext-length
|
|
fn Length(&self) -> u32 {
|
|
self.length
|
|
}
|
|
|
|
// https://webaudio.github.io/web-audio-api/#dom-offlineaudiocontext-startrendering
|
|
fn StartRendering(&self, comp: InRealm, can_gc: CanGc) -> Rc<Promise> {
|
|
let promise = Promise::new_in_current_realm(comp, can_gc);
|
|
if self.rendering_started.get() {
|
|
promise.reject_error(Error::InvalidState);
|
|
return promise;
|
|
}
|
|
self.rendering_started.set(true);
|
|
|
|
*self.pending_rendering_promise.borrow_mut() = Some(promise.clone());
|
|
|
|
let processed_audio = Arc::new(Mutex::new(Vec::new()));
|
|
let processed_audio_ = processed_audio.clone();
|
|
let (sender, receiver) = mpsc::channel();
|
|
let sender = Mutex::new(sender);
|
|
self.context
|
|
.audio_context_impl()
|
|
.lock()
|
|
.unwrap()
|
|
.set_eos_callback(Box::new(move |buffer| {
|
|
processed_audio_
|
|
.lock()
|
|
.unwrap()
|
|
.extend_from_slice((*buffer).as_ref());
|
|
let _ = sender.lock().unwrap().send(());
|
|
}));
|
|
|
|
let this = Trusted::new(self);
|
|
let task_source = self
|
|
.global()
|
|
.task_manager()
|
|
.dom_manipulation_task_source()
|
|
.to_sendable();
|
|
Builder::new()
|
|
.name("OfflineACResolver".to_owned())
|
|
.spawn(move || {
|
|
let _ = receiver.recv();
|
|
task_source.queue(
|
|
task!(resolve: move || {
|
|
let this = this.root();
|
|
let processed_audio = processed_audio.lock().unwrap();
|
|
let mut processed_audio: Vec<_> = processed_audio
|
|
.chunks(this.length as usize)
|
|
.map(|channel| channel.to_vec())
|
|
.collect();
|
|
// it can end up being empty if the task failed
|
|
if processed_audio.len() != this.length as usize {
|
|
processed_audio.resize(this.length as usize, Vec::new())
|
|
}
|
|
let buffer = AudioBuffer::new(
|
|
this.global().as_window(),
|
|
this.channel_count,
|
|
this.length,
|
|
*this.context.SampleRate(),
|
|
Some(processed_audio.as_slice()),
|
|
CanGc::note());
|
|
(*this.pending_rendering_promise.borrow_mut()).take().unwrap().resolve_native(&buffer);
|
|
let global = &this.global();
|
|
let window = global.as_window();
|
|
let event = OfflineAudioCompletionEvent::new(window,
|
|
atom!("complete"),
|
|
EventBubbles::DoesNotBubble,
|
|
EventCancelable::NotCancelable,
|
|
&buffer, CanGc::note());
|
|
event.upcast::<Event>().fire(this.upcast(), CanGc::note());
|
|
})
|
|
);
|
|
})
|
|
.unwrap();
|
|
|
|
if self
|
|
.context
|
|
.audio_context_impl()
|
|
.lock()
|
|
.unwrap()
|
|
.resume()
|
|
.is_err()
|
|
{
|
|
promise.reject_error(Error::Type("Could not start offline rendering".to_owned()));
|
|
}
|
|
|
|
promise
|
|
}
|
|
}
|