HTMLMediaElement.h 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. /*
  2. * Copyright (c) 2020, the SerenityOS developers.
  3. * Copyright (c) 2023, Tim Flynn <trflynn89@serenityos.org>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #pragma once
  8. #include <AK/ByteBuffer.h>
  9. #include <AK/Time.h>
  10. #include <AK/Variant.h>
  11. #include <LibJS/Heap/MarkedVector.h>
  12. #include <LibJS/SafeFunction.h>
  13. #include <LibWeb/HTML/EventLoop/Task.h>
  14. #include <LibWeb/HTML/HTMLElement.h>
  15. #include <LibWeb/WebIDL/DOMException.h>
  16. #include <math.h>
  17. namespace Web::HTML {
  18. class HTMLMediaElement : public HTMLElement {
  19. WEB_PLATFORM_OBJECT(HTMLMediaElement, HTMLElement);
  20. public:
  21. virtual ~HTMLMediaElement() override;
  22. void queue_a_media_element_task(JS::SafeFunction<void()> steps);
  23. enum class NetworkState : u16 {
  24. Empty,
  25. Idle,
  26. Loading,
  27. NoSource,
  28. };
  29. NetworkState network_state() const { return m_network_state; }
  30. WebIDL::ExceptionOr<Bindings::CanPlayTypeResult> can_play_type(DeprecatedString const& type) const;
  31. enum class ReadyState : u16 {
  32. HaveNothing,
  33. HaveMetadata,
  34. HaveCurrentData,
  35. HaveFutureData,
  36. HaveEnoughData,
  37. };
  38. ReadyState ready_state() const { return m_ready_state; }
  39. WebIDL::ExceptionOr<void> load();
  40. double duration() const;
  41. bool paused() const { return m_paused; }
  42. WebIDL::ExceptionOr<JS::NonnullGCPtr<JS::Promise>> play();
  43. WebIDL::ExceptionOr<void> pause();
  44. JS::NonnullGCPtr<VideoTrackList> video_tracks() const { return *m_video_tracks; }
  45. protected:
  46. HTMLMediaElement(DOM::Document&, DOM::QualifiedName);
  47. virtual JS::ThrowCompletionOr<void> initialize(JS::Realm&) override;
  48. virtual void visit_edges(Cell::Visitor&) override;
  49. // Override in subclasses to handle implementation-specific behavior when the element state changes
  50. // to playing or paused, e.g. to start/stop play timers.
  51. virtual void on_playing() { }
  52. virtual void on_paused() { }
  53. private:
  54. struct EntireResource { };
  55. using ByteRange = Variant<EntireResource>; // FIXME: This will need to include "until end" and an actual byte range.
  56. virtual void parse_attribute(DeprecatedFlyString const& name, DeprecatedString const& value) override;
  57. Task::Source media_element_event_task_source() const { return m_media_element_event_task_source.source; }
  58. WebIDL::ExceptionOr<void> load_element();
  59. WebIDL::ExceptionOr<void> select_resource();
  60. WebIDL::ExceptionOr<void> fetch_resource(AK::URL const&, Function<void()> failure_callback);
  61. static bool verify_response(JS::NonnullGCPtr<Fetch::Infrastructure::Response>, ByteRange const&);
  62. WebIDL::ExceptionOr<void> process_media_data(Function<void()> failure_callback);
  63. WebIDL::ExceptionOr<void> handle_media_source_failure(Span<JS::NonnullGCPtr<WebIDL::Promise>> promises);
  64. void forget_media_resource_specific_tracks();
  65. void set_ready_state(ReadyState);
  66. WebIDL::ExceptionOr<void> play_element();
  67. WebIDL::ExceptionOr<void> pause_element();
  68. void notify_about_playing();
  69. void set_paused(bool);
  70. void set_duration(double);
  71. WebIDL::ExceptionOr<void> dispatch_time_update_event();
  72. JS::MarkedVector<JS::NonnullGCPtr<WebIDL::Promise>> take_pending_play_promises();
  73. void resolve_pending_play_promises(ReadonlySpan<JS::NonnullGCPtr<WebIDL::Promise>> promises);
  74. void reject_pending_play_promises(ReadonlySpan<JS::NonnullGCPtr<WebIDL::Promise>> promises, JS::NonnullGCPtr<WebIDL::DOMException> error);
  75. // https://html.spec.whatwg.org/multipage/media.html#reject-pending-play-promises
  76. template<typename ErrorType>
  77. void reject_pending_play_promises(ReadonlySpan<JS::NonnullGCPtr<WebIDL::Promise>> promises, FlyString const& message)
  78. {
  79. auto& realm = this->realm();
  80. auto error = ErrorType::create(realm, message.to_deprecated_fly_string());
  81. reject_pending_play_promises(promises, error);
  82. }
  83. // https://html.spec.whatwg.org/multipage/media.html#media-element-event-task-source
  84. UniqueTaskSource m_media_element_event_task_source {};
  85. // https://html.spec.whatwg.org/multipage/media.html#dom-media-networkstate
  86. NetworkState m_network_state { NetworkState::Empty };
  87. // https://html.spec.whatwg.org/multipage/media.html#dom-media-readystate
  88. ReadyState m_ready_state { ReadyState::HaveNothing };
  89. bool m_first_data_load_event_since_load_start { false };
  90. // https://html.spec.whatwg.org/multipage/media.html#dom-media-duration
  91. double m_duration { NAN };
  92. // https://html.spec.whatwg.org/multipage/media.html#list-of-pending-play-promises
  93. JS::MarkedVector<JS::NonnullGCPtr<WebIDL::Promise>> m_pending_play_promises;
  94. // https://html.spec.whatwg.org/multipage/media.html#dom-media-paused
  95. bool m_paused { true };
  96. // https://html.spec.whatwg.org/multipage/media.html#dom-media-videotracks
  97. JS::GCPtr<VideoTrackList> m_video_tracks;
  98. // https://html.spec.whatwg.org/multipage/media.html#media-data
  99. ByteBuffer m_media_data;
  100. bool m_running_time_update_event_handler { false };
  101. Optional<Time> m_last_time_update_event_time;
  102. JS::GCPtr<Fetch::Infrastructure::FetchController> m_fetch_controller;
  103. };
  104. }