aboutsummaryrefslogtreecommitdiffstats
path: root/components/script/dom/audioparam.rs
diff options
context:
space:
mode:
authoryvt <i@yvt.jp>2021-07-10 17:24:27 +0900
committeryvt <i@yvt.jp>2021-07-10 17:55:42 +0900
commit01a7de50ab1843d85295f9dccad7f4c099e7208c (patch)
treeee53fb6e8889deb7b880ee969e6c662e6128d210 /components/script/dom/audioparam.rs
parentff8d2cdbbfc7a9dc7f38b7dd47cb350fde39388f (diff)
parent94b613fbdaa2b98f2179fc0bbda13c64e6fa0d38 (diff)
downloadservo-01a7de50ab1843d85295f9dccad7f4c099e7208c.tar.gz
servo-01a7de50ab1843d85295f9dccad7f4c099e7208c.zip
Merge remote-tracking branch 'upstream/master' into feat-cow-infra
`tests/wpt/web-platform-tests/html/browsers/origin/cross-origin-objects/cross-origin-objects.html` was reverted to the upstream version.
Diffstat (limited to 'components/script/dom/audioparam.rs')
-rw-r--r--components/script/dom/audioparam.rs308
1 files changed, 308 insertions, 0 deletions
diff --git a/components/script/dom/audioparam.rs b/components/script/dom/audioparam.rs
new file mode 100644
index 00000000000..dd0e72f7bcf
--- /dev/null
+++ b/components/script/dom/audioparam.rs
@@ -0,0 +1,308 @@
+/* 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 crate::dom::baseaudiocontext::BaseAudioContext;
+use crate::dom::bindings::codegen::Bindings::AudioParamBinding::{
+ AudioParamMethods, AutomationRate,
+};
+use crate::dom::bindings::error::{Error, Fallible};
+use crate::dom::bindings::num::Finite;
+use crate::dom::bindings::reflector::{reflect_dom_object, Reflector};
+use crate::dom::bindings::root::{Dom, DomRoot};
+use crate::dom::window::Window;
+use dom_struct::dom_struct;
+use servo_media::audio::graph::NodeId;
+use servo_media::audio::node::AudioNodeMessage;
+use servo_media::audio::param::{ParamRate, ParamType, RampKind, UserAutomationEvent};
+use std::cell::Cell;
+use std::sync::mpsc;
+
+#[dom_struct]
+pub struct AudioParam {
+ reflector_: Reflector,
+ context: Dom<BaseAudioContext>,
+ #[ignore_malloc_size_of = "servo_media"]
+ node: NodeId,
+ #[ignore_malloc_size_of = "servo_media"]
+ param: ParamType,
+ automation_rate: Cell<AutomationRate>,
+ default_value: f32,
+ min_value: f32,
+ max_value: f32,
+}
+
+impl AudioParam {
+ pub fn new_inherited(
+ context: &BaseAudioContext,
+ node: NodeId,
+ param: ParamType,
+ automation_rate: AutomationRate,
+ default_value: f32,
+ min_value: f32,
+ max_value: f32,
+ ) -> AudioParam {
+ AudioParam {
+ reflector_: Reflector::new(),
+ context: Dom::from_ref(context),
+ node,
+ param,
+ automation_rate: Cell::new(automation_rate),
+ default_value,
+ min_value,
+ max_value,
+ }
+ }
+
+ #[allow(unrooted_must_root)]
+ pub fn new(
+ window: &Window,
+ context: &BaseAudioContext,
+ node: NodeId,
+ param: ParamType,
+ automation_rate: AutomationRate,
+ default_value: f32,
+ min_value: f32,
+ max_value: f32,
+ ) -> DomRoot<AudioParam> {
+ let audio_param = AudioParam::new_inherited(
+ context,
+ node,
+ param,
+ automation_rate,
+ default_value,
+ min_value,
+ max_value,
+ );
+ reflect_dom_object(Box::new(audio_param), window)
+ }
+
+ fn message_node(&self, message: AudioNodeMessage) {
+ self.context
+ .audio_context_impl()
+ .lock()
+ .unwrap()
+ .message_node(self.node, message);
+ }
+
+ pub fn context(&self) -> &BaseAudioContext {
+ &self.context
+ }
+
+ pub fn node_id(&self) -> NodeId {
+ self.node
+ }
+
+ pub fn param_type(&self) -> ParamType {
+ self.param
+ }
+}
+
+impl AudioParamMethods for AudioParam {
+ // https://webaudio.github.io/web-audio-api/#dom-audioparam-automationrate
+ fn AutomationRate(&self) -> AutomationRate {
+ self.automation_rate.get()
+ }
+
+ // https://webaudio.github.io/web-audio-api/#dom-audioparam-automationrate
+ fn SetAutomationRate(&self, automation_rate: AutomationRate) {
+ self.automation_rate.set(automation_rate);
+ self.message_node(AudioNodeMessage::SetParamRate(
+ self.param,
+ automation_rate.into(),
+ ));
+ }
+
+ // https://webaudio.github.io/web-audio-api/#dom-audioparam-value
+ fn Value(&self) -> Finite<f32> {
+ let (tx, rx) = mpsc::channel();
+ self.message_node(AudioNodeMessage::GetParamValue(self.param, tx));
+ Finite::wrap(rx.recv().unwrap())
+ }
+
+ // https://webaudio.github.io/web-audio-api/#dom-audioparam-value
+ fn SetValue(&self, value: Finite<f32>) {
+ self.message_node(AudioNodeMessage::SetParam(
+ self.param,
+ UserAutomationEvent::SetValue(*value),
+ ));
+ }
+
+ // https://webaudio.github.io/web-audio-api/#dom-audioparam-defaultvalue
+ fn DefaultValue(&self) -> Finite<f32> {
+ Finite::wrap(self.default_value)
+ }
+
+ // https://webaudio.github.io/web-audio-api/#dom-audioparam-minvalue
+ fn MinValue(&self) -> Finite<f32> {
+ Finite::wrap(self.min_value)
+ }
+
+ // https://webaudio.github.io/web-audio-api/#dom-audioparam-maxvalue
+ fn MaxValue(&self) -> Finite<f32> {
+ Finite::wrap(self.max_value)
+ }
+
+ // https://webaudio.github.io/web-audio-api/#dom-audioparam-setvalueattime
+ fn SetValueAtTime(
+ &self,
+ value: Finite<f32>,
+ start_time: Finite<f64>,
+ ) -> Fallible<DomRoot<AudioParam>> {
+ if *start_time < 0. {
+ return Err(Error::Range(format!(
+ "start time {} should not be negative",
+ *start_time
+ )));
+ }
+ self.message_node(AudioNodeMessage::SetParam(
+ self.param,
+ UserAutomationEvent::SetValueAtTime(*value, *start_time),
+ ));
+ Ok(DomRoot::from_ref(self))
+ }
+
+ // https://webaudio.github.io/web-audio-api/#dom-audioparam-linearramptovalueattime
+ fn LinearRampToValueAtTime(
+ &self,
+ value: Finite<f32>,
+ end_time: Finite<f64>,
+ ) -> Fallible<DomRoot<AudioParam>> {
+ if *end_time < 0. {
+ return Err(Error::Range(format!(
+ "end time {} should not be negative",
+ *end_time
+ )));
+ }
+ self.message_node(AudioNodeMessage::SetParam(
+ self.param,
+ UserAutomationEvent::RampToValueAtTime(RampKind::Linear, *value, *end_time),
+ ));
+ Ok(DomRoot::from_ref(self))
+ }
+
+ // https://webaudio.github.io/web-audio-api/#dom-audioparam-exponentialramptovalueattime
+ fn ExponentialRampToValueAtTime(
+ &self,
+ value: Finite<f32>,
+ end_time: Finite<f64>,
+ ) -> Fallible<DomRoot<AudioParam>> {
+ if *end_time < 0. {
+ return Err(Error::Range(format!(
+ "end time {} should not be negative",
+ *end_time
+ )));
+ }
+ if *value == 0. {
+ return Err(Error::Range(format!(
+ "target value {} should not be 0",
+ *value
+ )));
+ }
+ self.message_node(AudioNodeMessage::SetParam(
+ self.param,
+ UserAutomationEvent::RampToValueAtTime(RampKind::Exponential, *value, *end_time),
+ ));
+ Ok(DomRoot::from_ref(self))
+ }
+
+ // https://webaudio.github.io/web-audio-api/#dom-audioparam-settargetattime
+ fn SetTargetAtTime(
+ &self,
+ target: Finite<f32>,
+ start_time: Finite<f64>,
+ time_constant: Finite<f32>,
+ ) -> Fallible<DomRoot<AudioParam>> {
+ if *start_time < 0. {
+ return Err(Error::Range(format!(
+ "start time {} should not be negative",
+ *start_time
+ )));
+ }
+ if *time_constant < 0. {
+ return Err(Error::Range(format!(
+ "time constant {} should not be negative",
+ *time_constant
+ )));
+ }
+ self.message_node(AudioNodeMessage::SetParam(
+ self.param,
+ UserAutomationEvent::SetTargetAtTime(*target, *start_time, (*time_constant).into()),
+ ));
+ Ok(DomRoot::from_ref(self))
+ }
+
+ // https://webaudio.github.io/web-audio-api/#dom-audioparam-setvaluecurveattime
+ fn SetValueCurveAtTime(
+ &self,
+ values: Vec<Finite<f32>>,
+ start_time: Finite<f64>,
+ end_time: Finite<f64>,
+ ) -> Fallible<DomRoot<AudioParam>> {
+ if *start_time < 0. {
+ return Err(Error::Range(format!(
+ "start time {} should not be negative",
+ *start_time
+ )));
+ }
+ if values.len() < 2. as usize {
+ return Err(Error::InvalidState);
+ }
+
+ if *end_time < 0. {
+ return Err(Error::Range(format!(
+ "end time {} should not be negative",
+ *end_time
+ )));
+ }
+ self.message_node(AudioNodeMessage::SetParam(
+ self.param,
+ UserAutomationEvent::SetValueCurveAtTime(
+ values.into_iter().map(|v| *v).collect(),
+ *start_time,
+ *end_time,
+ ),
+ ));
+ Ok(DomRoot::from_ref(self))
+ }
+
+ // https://webaudio.github.io/web-audio-api/#dom-audioparam-cancelscheduledvalues
+ fn CancelScheduledValues(&self, cancel_time: Finite<f64>) -> Fallible<DomRoot<AudioParam>> {
+ if *cancel_time < 0. {
+ return Err(Error::Range(format!(
+ "cancel time {} should not be negative",
+ *cancel_time
+ )));
+ }
+ self.message_node(AudioNodeMessage::SetParam(
+ self.param,
+ UserAutomationEvent::CancelScheduledValues(*cancel_time),
+ ));
+ Ok(DomRoot::from_ref(self))
+ }
+
+ // https://webaudio.github.io/web-audio-api/#dom-audioparam-cancelandholdattime
+ fn CancelAndHoldAtTime(&self, cancel_time: Finite<f64>) -> Fallible<DomRoot<AudioParam>> {
+ if *cancel_time < 0. {
+ return Err(Error::Range(format!(
+ "cancel time {} should not be negative",
+ *cancel_time
+ )));
+ }
+ self.message_node(AudioNodeMessage::SetParam(
+ self.param,
+ UserAutomationEvent::CancelAndHoldAtTime(*cancel_time),
+ ));
+ Ok(DomRoot::from_ref(self))
+ }
+}
+
+// https://webaudio.github.io/web-audio-api/#enumdef-automationrate
+impl From<AutomationRate> for ParamRate {
+ fn from(rate: AutomationRate) -> Self {
+ match rate {
+ AutomationRate::A_rate => ParamRate::ARate,
+ AutomationRate::K_rate => ParamRate::KRate,
+ }
+ }
+}