/* * Copyright (c) 2024, Shannon Booth * Copyright (c) 2025, Tim Ledbetter * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include namespace Web::WebAudio { GC_DEFINE_ALLOCATOR(OscillatorNode); OscillatorNode::~OscillatorNode() = default; WebIDL::ExceptionOr> OscillatorNode::create(JS::Realm& realm, GC::Ref context, OscillatorOptions const& options) { return construct_impl(realm, context, options); } // https://webaudio.github.io/web-audio-api/#dom-oscillatornode-oscillatornode WebIDL::ExceptionOr> OscillatorNode::construct_impl(JS::Realm& realm, GC::Ref context, OscillatorOptions const& options) { if (options.type == Bindings::OscillatorType::Custom && !options.periodic_wave) return WebIDL::InvalidStateError::create(realm, "Oscillator node type 'custom' requires PeriodicWave to be provided"_string); auto node = realm.create(realm, context, options); if (options.type == Bindings::OscillatorType::Custom) node->set_periodic_wave(options.periodic_wave); // Default options for channel count and interpretation // https://webaudio.github.io/web-audio-api/#OscillatorNode AudioNodeDefaultOptions default_options; default_options.channel_count = 2; default_options.channel_count_mode = Bindings::ChannelCountMode::Max; default_options.channel_interpretation = Bindings::ChannelInterpretation::Speakers; // FIXME: Set tail-time to no TRY(node->initialize_audio_node_options(options, default_options)); return node; } OscillatorNode::OscillatorNode(JS::Realm& realm, GC::Ref context, OscillatorOptions const& options) : AudioScheduledSourceNode(realm, context) , m_type(options.type) , m_frequency(AudioParam::create(realm, context, options.frequency, -context->nyquist_frequency(), context->nyquist_frequency(), Bindings::AutomationRate::ARate)) , m_detune(AudioParam::create(realm, context, options.detune, -1200 * AK::log2(NumericLimits::max()), 1200 * AK::log2(NumericLimits::max()), Bindings::AutomationRate::ARate)) { } // https://webaudio.github.io/web-audio-api/#dom-oscillatornode-type Bindings::OscillatorType OscillatorNode::type() const { return m_type; } // https://webaudio.github.io/web-audio-api/#dom-oscillatornode-type WebIDL::ExceptionOr OscillatorNode::set_type(Bindings::OscillatorType type) { if (type == Bindings::OscillatorType::Custom && m_type != Bindings::OscillatorType::Custom) return WebIDL::InvalidStateError::create(realm(), "Oscillator node type cannot be changed to 'custom'"_string); // FIXME: An appropriate PeriodicWave should be set here based on the given type. set_periodic_wave(nullptr); m_type = type; return {}; } // https://webaudio.github.io/web-audio-api/#dom-oscillatornode-setperiodicwave void OscillatorNode::set_periodic_wave(GC::Ptr periodic_wave) { m_periodic_wave = periodic_wave; m_type = Bindings::OscillatorType::Custom; } void OscillatorNode::initialize(JS::Realm& realm) { Base::initialize(realm); WEB_SET_PROTOTYPE_FOR_INTERFACE(OscillatorNode); } void OscillatorNode::visit_edges(Cell::Visitor& visitor) { Base::visit_edges(visitor); visitor.visit(m_frequency); visitor.visit(m_detune); visitor.visit(m_periodic_wave); } }