HTMLVideoElement.cpp 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  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. #include <LibGfx/Bitmap.h>
  8. #include <LibWeb/Bindings/Intrinsics.h>
  9. #include <LibWeb/DOM/Document.h>
  10. #include <LibWeb/Fetch/Fetching/Fetching.h>
  11. #include <LibWeb/Fetch/Infrastructure/FetchAlgorithms.h>
  12. #include <LibWeb/Fetch/Infrastructure/FetchController.h>
  13. #include <LibWeb/Fetch/Infrastructure/HTTP/Requests.h>
  14. #include <LibWeb/Fetch/Infrastructure/HTTP/Responses.h>
  15. #include <LibWeb/HTML/HTMLVideoElement.h>
  16. #include <LibWeb/HTML/VideoTrack.h>
  17. #include <LibWeb/Layout/VideoBox.h>
  18. #include <LibWeb/Platform/ImageCodecPlugin.h>
  19. namespace Web::HTML {
  20. HTMLVideoElement::HTMLVideoElement(DOM::Document& document, DOM::QualifiedName qualified_name)
  21. : HTMLMediaElement(document, move(qualified_name))
  22. {
  23. }
  24. HTMLVideoElement::~HTMLVideoElement() = default;
  25. JS::ThrowCompletionOr<void> HTMLVideoElement::initialize(JS::Realm& realm)
  26. {
  27. MUST_OR_THROW_OOM(Base::initialize(realm));
  28. set_prototype(&Bindings::ensure_web_prototype<Bindings::HTMLVideoElementPrototype>(realm, "HTMLVideoElement"));
  29. return {};
  30. }
  31. void HTMLVideoElement::visit_edges(Cell::Visitor& visitor)
  32. {
  33. Base::visit_edges(visitor);
  34. visitor.visit(m_video_track);
  35. visitor.visit(m_fetch_controller);
  36. }
  37. void HTMLVideoElement::parse_attribute(DeprecatedFlyString const& name, DeprecatedString const& value)
  38. {
  39. Base::parse_attribute(name, value);
  40. if (name == HTML::AttributeNames::poster)
  41. determine_element_poster_frame(value).release_value_but_fixme_should_propagate_errors();
  42. }
  43. void HTMLVideoElement::did_remove_attribute(DeprecatedFlyString const& name)
  44. {
  45. Base::did_remove_attribute(name);
  46. if (name == HTML::AttributeNames::poster)
  47. determine_element_poster_frame({}).release_value_but_fixme_should_propagate_errors();
  48. }
  49. JS::GCPtr<Layout::Node> HTMLVideoElement::create_layout_node(NonnullRefPtr<CSS::StyleProperties> style)
  50. {
  51. return heap().allocate_without_realm<Layout::VideoBox>(document(), *this, move(style));
  52. }
  53. Layout::VideoBox* HTMLVideoElement::layout_node()
  54. {
  55. return static_cast<Layout::VideoBox*>(Node::layout_node());
  56. }
  57. Layout::VideoBox const* HTMLVideoElement::layout_node() const
  58. {
  59. return static_cast<Layout::VideoBox const*>(Node::layout_node());
  60. }
  61. // https://html.spec.whatwg.org/multipage/media.html#dom-video-videowidth
  62. u32 HTMLVideoElement::video_width() const
  63. {
  64. // The videoWidth IDL attribute must return the intrinsic width of the video in CSS pixels. The videoHeight IDL
  65. // attribute must return the intrinsic height of the video in CSS pixels. If the element's readyState attribute
  66. // is HAVE_NOTHING, then the attributes must return 0.
  67. if (ready_state() == ReadyState::HaveNothing)
  68. return 0;
  69. return m_video_width;
  70. }
  71. // https://html.spec.whatwg.org/multipage/media.html#dom-video-videoheight
  72. u32 HTMLVideoElement::video_height() const
  73. {
  74. // The videoWidth IDL attribute must return the intrinsic width of the video in CSS pixels. The videoHeight IDL
  75. // attribute must return the intrinsic height of the video in CSS pixels. If the element's readyState attribute
  76. // is HAVE_NOTHING, then the attributes must return 0.
  77. if (ready_state() == ReadyState::HaveNothing)
  78. return 0;
  79. return m_video_height;
  80. }
  81. void HTMLVideoElement::set_video_track(JS::GCPtr<HTML::VideoTrack> video_track)
  82. {
  83. set_needs_style_update(true);
  84. document().set_needs_layout();
  85. if (m_video_track)
  86. m_video_track->pause_video({});
  87. m_video_track = video_track;
  88. }
  89. void HTMLVideoElement::set_current_frame(Badge<VideoTrack>, RefPtr<Gfx::Bitmap> frame, double position)
  90. {
  91. m_current_frame = { move(frame), position };
  92. if (layout_node())
  93. layout_node()->set_needs_display();
  94. }
  95. void HTMLVideoElement::on_playing()
  96. {
  97. if (m_video_track)
  98. m_video_track->play_video({});
  99. }
  100. void HTMLVideoElement::on_paused()
  101. {
  102. if (m_video_track)
  103. m_video_track->pause_video({});
  104. }
  105. void HTMLVideoElement::on_seek(double position, MediaSeekMode seek_mode)
  106. {
  107. if (m_video_track)
  108. m_video_track->seek(Time::from_milliseconds(position * 1000.0), seek_mode);
  109. }
  110. // https://html.spec.whatwg.org/multipage/media.html#attr-video-poster
  111. WebIDL::ExceptionOr<void> HTMLVideoElement::determine_element_poster_frame(Optional<StringView> const& poster)
  112. {
  113. auto& realm = this->realm();
  114. auto& vm = realm.vm();
  115. m_poster_frame = nullptr;
  116. // 1. If there is an existing instance of this algorithm running for this video element, abort that instance of
  117. // this algorithm without changing the poster frame.
  118. if (m_fetch_controller)
  119. m_fetch_controller->stop_fetch();
  120. // 2. If the poster attribute's value is the empty string or if the attribute is absent, then there is no poster
  121. // frame; return.
  122. if (!poster.has_value() || poster->is_empty())
  123. return {};
  124. // 3. Parse the poster attribute's value relative to the element's node document. If this fails, then there is no
  125. // poster frame; return.
  126. auto url_record = document().parse_url(*poster);
  127. if (!url_record.is_valid())
  128. return {};
  129. // 4. Let request be a new request whose URL is the resulting URL record, client is the element's node document's
  130. // relevant settings object, destination is "image", initiator type is "video", credentials mode is "include",
  131. // and whose use-URL-credentials flag is set.
  132. auto request = Fetch::Infrastructure::Request::create(vm);
  133. request->set_url(move(url_record));
  134. request->set_client(&document().relevant_settings_object());
  135. request->set_destination(Fetch::Infrastructure::Request::Destination::Image);
  136. request->set_initiator_type(Fetch::Infrastructure::Request::InitiatorType::Video);
  137. request->set_credentials_mode(Fetch::Infrastructure::Request::CredentialsMode::Include);
  138. request->set_use_url_credentials(true);
  139. // 5. Fetch request. This must delay the load event of the element's node document.
  140. Fetch::Infrastructure::FetchAlgorithms::Input fetch_algorithms_input {};
  141. m_load_event_delayer.emplace(document());
  142. fetch_algorithms_input.process_response = [this](auto response) mutable {
  143. ScopeGuard guard { [&] { m_load_event_delayer.clear(); } };
  144. auto& realm = this->realm();
  145. auto& global = document().realm().global_object();
  146. if (response->is_network_error())
  147. return;
  148. if (response->type() == Fetch::Infrastructure::Response::Type::Opaque || response->type() == Fetch::Infrastructure::Response::Type::OpaqueRedirect) {
  149. auto& filtered_response = static_cast<Fetch::Infrastructure::FilteredResponse&>(*response);
  150. response = filtered_response.internal_response();
  151. }
  152. auto on_image_data_read = [this](auto image_data) mutable {
  153. m_fetch_controller = nullptr;
  154. // 6. If an image is thus obtained, the poster frame is that image. Otherwise, there is no poster frame.
  155. auto image = Platform::ImageCodecPlugin::the().decode_image(image_data);
  156. if (!image.has_value() || image->frames.is_empty())
  157. return;
  158. m_poster_frame = move(image.release_value().frames[0].bitmap);
  159. };
  160. VERIFY(response->body().has_value());
  161. auto empty_algorithm = [](auto) {};
  162. response->body()->fully_read(realm, move(on_image_data_read), move(empty_algorithm), JS::NonnullGCPtr { global }).release_value_but_fixme_should_propagate_errors();
  163. };
  164. m_fetch_controller = TRY(Fetch::Fetching::fetch(realm, request, Fetch::Infrastructure::FetchAlgorithms::create(vm, move(fetch_algorithms_input))));
  165. return {};
  166. }
  167. }