AudioTrack.cpp 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  1. /*
  2. * Copyright (c) 2023, Tim Flynn <trflynn89@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <AK/IDAllocator.h>
  7. #include <LibAudio/Loader.h>
  8. #include <LibJS/Runtime/Realm.h>
  9. #include <LibJS/Runtime/VM.h>
  10. #include <LibWeb/Bindings/AudioTrackPrototype.h>
  11. #include <LibWeb/Bindings/Intrinsics.h>
  12. #include <LibWeb/DOM/Event.h>
  13. #include <LibWeb/HTML/AudioTrack.h>
  14. #include <LibWeb/HTML/AudioTrackList.h>
  15. #include <LibWeb/HTML/EventNames.h>
  16. #include <LibWeb/HTML/HTMLMediaElement.h>
  17. #include <LibWeb/Layout/Node.h>
  18. #include <LibWeb/Painting/Paintable.h>
  19. #include <LibWeb/Platform/AudioCodecPlugin.h>
  20. namespace Web::HTML {
  21. JS_DEFINE_ALLOCATOR(AudioTrack);
  22. static IDAllocator s_audio_track_id_allocator;
  23. AudioTrack::AudioTrack(JS::Realm& realm, JS::NonnullGCPtr<HTMLMediaElement> media_element, NonnullRefPtr<Audio::Loader> loader)
  24. : PlatformObject(realm)
  25. , m_media_element(media_element)
  26. , m_audio_plugin(Platform::AudioCodecPlugin::create(move(loader)).release_value_but_fixme_should_propagate_errors())
  27. {
  28. m_audio_plugin->on_playback_position_updated = [this](auto position) {
  29. if (auto const* paintable = m_media_element->paintable())
  30. paintable->set_needs_display();
  31. auto playback_position = static_cast<double>(position.to_milliseconds()) / 1000.0;
  32. m_media_element->set_current_playback_position(playback_position);
  33. };
  34. }
  35. AudioTrack::~AudioTrack()
  36. {
  37. auto id = m_id.to_number<int>();
  38. VERIFY(id.has_value());
  39. s_audio_track_id_allocator.deallocate(id.value());
  40. }
  41. void AudioTrack::initialize(JS::Realm& realm)
  42. {
  43. Base::initialize(realm);
  44. WEB_SET_PROTOTYPE_FOR_INTERFACE(AudioTrack);
  45. auto id = s_audio_track_id_allocator.allocate();
  46. m_id = MUST(String::number(id));
  47. }
  48. void AudioTrack::play(Badge<HTMLAudioElement>)
  49. {
  50. m_audio_plugin->resume_playback();
  51. }
  52. void AudioTrack::pause(Badge<HTMLAudioElement>)
  53. {
  54. m_audio_plugin->pause_playback();
  55. }
  56. Duration AudioTrack::duration()
  57. {
  58. return m_audio_plugin->duration();
  59. }
  60. void AudioTrack::seek(double position, MediaSeekMode seek_mode)
  61. {
  62. // FIXME: Implement seeking mode.
  63. (void)seek_mode;
  64. m_audio_plugin->seek(position);
  65. }
  66. void AudioTrack::update_volume()
  67. {
  68. m_audio_plugin->set_volume(m_media_element->effective_media_volume());
  69. }
  70. void AudioTrack::visit_edges(Cell::Visitor& visitor)
  71. {
  72. Base::visit_edges(visitor);
  73. visitor.visit(m_media_element);
  74. visitor.visit(m_audio_track_list);
  75. }
  76. // https://html.spec.whatwg.org/multipage/media.html#dom-audiotrack-enabled
  77. void AudioTrack::set_enabled(bool enabled)
  78. {
  79. // On setting, it must enable the track if the new value is true, and disable it otherwise. (If the track is no
  80. // longer in an AudioTrackList object, then the track being enabled or disabled has no effect beyond changing the
  81. // value of the attribute on the AudioTrack object.)
  82. if (m_enabled == enabled)
  83. return;
  84. if (m_audio_track_list) {
  85. // Whenever an audio track in an AudioTrackList that was disabled is enabled, and whenever one that was enabled
  86. // is disabled, the user agent must queue a media element task given the media element to fire an event named
  87. // change at the AudioTrackList object.
  88. m_media_element->queue_a_media_element_task([this]() {
  89. m_audio_track_list->dispatch_event(DOM::Event::create(realm(), HTML::EventNames::change));
  90. });
  91. }
  92. m_enabled = enabled;
  93. }
  94. }