1 | /* |
2 | * Copyright (C) 2014 Apple Inc. All rights reserved. |
3 | * |
4 | * Redistribution and use in source and binary forms, with or without |
5 | * modification, are permitted provided that the following conditions |
6 | * are met: |
7 | * 1. Redistributions of source code must retain the above copyright |
8 | * notice, this list of conditions and the following disclaimer. |
9 | * 2. Redistributions in binary form must reproduce the above copyright |
10 | * notice, this list of conditions and the following disclaimer in the |
11 | * documentation and/or other materials provided with the distribution. |
12 | * |
13 | * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' |
14 | * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, |
15 | * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
16 | * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS |
17 | * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR |
18 | * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
19 | * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS |
20 | * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN |
21 | * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) |
22 | * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF |
23 | * THE POSSIBILITY OF SUCH DAMAGE. |
24 | */ |
25 | |
26 | #include "config.h" |
27 | |
28 | #if ENABLE(VIDEO) |
29 | |
30 | #include "MediaElementSession.h" |
31 | |
32 | #include "Document.h" |
33 | #include "DocumentLoader.h" |
34 | #include "Frame.h" |
35 | #include "FrameView.h" |
36 | #include "FullscreenManager.h" |
37 | #include "HTMLAudioElement.h" |
38 | #include "HTMLMediaElement.h" |
39 | #include "HTMLNames.h" |
40 | #include "HTMLVideoElement.h" |
41 | #include "HitTestResult.h" |
42 | #include "Logging.h" |
43 | #include "Page.h" |
44 | #include "PlatformMediaSessionManager.h" |
45 | #include "Quirks.h" |
46 | #include "RenderMedia.h" |
47 | #include "RenderView.h" |
48 | #include "ScriptController.h" |
49 | #include "Settings.h" |
50 | #include "SourceBuffer.h" |
51 | #include <wtf/text/StringBuilder.h> |
52 | |
53 | #if PLATFORM(IOS_FAMILY) |
54 | #include "AudioSession.h" |
55 | #include "RuntimeApplicationChecks.h" |
56 | #include <wtf/spi/darwin/dyldSPI.h> |
57 | #endif |
58 | |
59 | namespace WebCore { |
60 | |
61 | static const Seconds clientDataBufferingTimerThrottleDelay { 100_ms }; |
62 | static const Seconds elementMainContentCheckInterval { 250_ms }; |
63 | |
64 | static bool isElementRectMostlyInMainFrame(const HTMLMediaElement&); |
65 | static bool isElementLargeEnoughForMainContent(const HTMLMediaElement&, MediaSessionMainContentPurpose); |
66 | static bool isElementMainContentForPurposesOfAutoplay(const HTMLMediaElement&, bool shouldHitTestMainFrame); |
67 | |
68 | #if !RELEASE_LOG_DISABLED |
69 | static String restrictionNames(MediaElementSession::BehaviorRestrictions restriction) |
70 | { |
71 | StringBuilder restrictionBuilder; |
72 | #define CASE(restrictionType) \ |
73 | if (restriction & MediaElementSession::restrictionType) { \ |
74 | if (!restrictionBuilder.isEmpty()) \ |
75 | restrictionBuilder.appendLiteral(", "); \ |
76 | restrictionBuilder.append(#restrictionType); \ |
77 | } \ |
78 | |
79 | CASE(NoRestrictions) |
80 | CASE(RequireUserGestureForLoad) |
81 | CASE(RequireUserGestureForVideoRateChange) |
82 | CASE(RequireUserGestureForAudioRateChange) |
83 | CASE(RequireUserGestureForFullscreen) |
84 | CASE(RequirePageConsentToLoadMedia) |
85 | CASE(RequirePageConsentToResumeMedia) |
86 | CASE(RequireUserGestureToShowPlaybackTargetPicker) |
87 | CASE(WirelessVideoPlaybackDisabled) |
88 | CASE(RequireUserGestureToAutoplayToExternalDevice) |
89 | CASE(AutoPreloadingNotPermitted) |
90 | CASE(InvisibleAutoplayNotPermitted) |
91 | CASE(OverrideUserGestureRequirementForMainContent) |
92 | CASE(RequireUserGestureToControlControlsManager) |
93 | CASE(RequirePlaybackToControlControlsManager) |
94 | CASE(RequireUserGestureForVideoDueToLowPowerMode) |
95 | |
96 | return restrictionBuilder.toString(); |
97 | } |
98 | #endif |
99 | |
100 | static bool pageExplicitlyAllowsElementToAutoplayInline(const HTMLMediaElement& element) |
101 | { |
102 | Document& document = element.document(); |
103 | Page* page = document.page(); |
104 | return document.isMediaDocument() && !document.ownerElement() && page && page->allowsMediaDocumentInlinePlayback(); |
105 | } |
106 | |
107 | MediaElementSession::MediaElementSession(HTMLMediaElement& element) |
108 | : PlatformMediaSession(element) |
109 | , m_element(element) |
110 | , m_restrictions(NoRestrictions) |
111 | #if ENABLE(WIRELESS_PLAYBACK_TARGET) |
112 | , m_targetAvailabilityChangedTimer(*this, &MediaElementSession::targetAvailabilityChangedTimerFired) |
113 | , m_hasPlaybackTargets(PlatformMediaSessionManager::sharedManager().hasWirelessTargetsAvailable()) |
114 | #endif |
115 | , m_mainContentCheckTimer(*this, &MediaElementSession::mainContentCheckTimerFired) |
116 | , m_clientDataBufferingTimer(*this, &MediaElementSession::clientDataBufferingTimerFired) |
117 | #if !RELEASE_LOG_DISABLED |
118 | , m_logIdentifier(element.logIdentifier()) |
119 | #endif |
120 | { |
121 | } |
122 | |
123 | void MediaElementSession::registerWithDocument(Document& document) |
124 | { |
125 | #if ENABLE(WIRELESS_PLAYBACK_TARGET) |
126 | document.addPlaybackTargetPickerClient(*this); |
127 | #else |
128 | UNUSED_PARAM(document); |
129 | #endif |
130 | } |
131 | |
132 | void MediaElementSession::unregisterWithDocument(Document& document) |
133 | { |
134 | #if ENABLE(WIRELESS_PLAYBACK_TARGET) |
135 | document.removePlaybackTargetPickerClient(*this); |
136 | #else |
137 | UNUSED_PARAM(document); |
138 | #endif |
139 | } |
140 | |
141 | void MediaElementSession::clientWillBeginAutoplaying() |
142 | { |
143 | PlatformMediaSession::clientWillBeginAutoplaying(); |
144 | m_elementIsHiddenBecauseItWasRemovedFromDOM = false; |
145 | updateClientDataBuffering(); |
146 | } |
147 | |
148 | bool MediaElementSession::clientWillBeginPlayback() |
149 | { |
150 | if (!PlatformMediaSession::clientWillBeginPlayback()) |
151 | return false; |
152 | |
153 | m_elementIsHiddenBecauseItWasRemovedFromDOM = false; |
154 | updateClientDataBuffering(); |
155 | return true; |
156 | } |
157 | |
158 | bool MediaElementSession::clientWillPausePlayback() |
159 | { |
160 | if (!PlatformMediaSession::clientWillPausePlayback()) |
161 | return false; |
162 | |
163 | updateClientDataBuffering(); |
164 | return true; |
165 | } |
166 | |
167 | void MediaElementSession::visibilityChanged() |
168 | { |
169 | scheduleClientDataBufferingCheck(); |
170 | |
171 | if (m_element.elementIsHidden() && !m_element.isFullscreen()) |
172 | m_elementIsHiddenUntilVisibleInViewport = true; |
173 | else if (m_element.isVisibleInViewport()) |
174 | m_elementIsHiddenUntilVisibleInViewport = false; |
175 | } |
176 | |
177 | void MediaElementSession::isVisibleInViewportChanged() |
178 | { |
179 | scheduleClientDataBufferingCheck(); |
180 | |
181 | if (m_element.isFullscreen() || m_element.isVisibleInViewport()) |
182 | m_elementIsHiddenUntilVisibleInViewport = false; |
183 | } |
184 | |
185 | void MediaElementSession::inActiveDocumentChanged() |
186 | { |
187 | m_elementIsHiddenBecauseItWasRemovedFromDOM = !m_element.inActiveDocument(); |
188 | scheduleClientDataBufferingCheck(); |
189 | } |
190 | |
191 | void MediaElementSession::scheduleClientDataBufferingCheck() |
192 | { |
193 | if (!m_clientDataBufferingTimer.isActive()) |
194 | m_clientDataBufferingTimer.startOneShot(clientDataBufferingTimerThrottleDelay); |
195 | } |
196 | |
197 | void MediaElementSession::clientDataBufferingTimerFired() |
198 | { |
199 | INFO_LOG(LOGIDENTIFIER, "visible = " , m_element.elementIsHidden()); |
200 | |
201 | updateClientDataBuffering(); |
202 | |
203 | #if PLATFORM(IOS_FAMILY) |
204 | PlatformMediaSessionManager::sharedManager().configureWireLessTargetMonitoring(); |
205 | #endif |
206 | |
207 | if (state() != Playing || !m_element.elementIsHidden()) |
208 | return; |
209 | |
210 | PlatformMediaSessionManager::SessionRestrictions restrictions = PlatformMediaSessionManager::sharedManager().restrictions(mediaType()); |
211 | if ((restrictions & PlatformMediaSessionManager::BackgroundTabPlaybackRestricted) == PlatformMediaSessionManager::BackgroundTabPlaybackRestricted) |
212 | pauseSession(); |
213 | } |
214 | |
215 | void MediaElementSession::updateClientDataBuffering() |
216 | { |
217 | if (m_clientDataBufferingTimer.isActive()) |
218 | m_clientDataBufferingTimer.stop(); |
219 | |
220 | m_element.setBufferingPolicy(preferredBufferingPolicy()); |
221 | } |
222 | |
223 | void MediaElementSession::addBehaviorRestriction(BehaviorRestrictions restrictions) |
224 | { |
225 | if (restrictions & ~m_restrictions) |
226 | INFO_LOG(LOGIDENTIFIER, "adding " , restrictionNames(restrictions & ~m_restrictions)); |
227 | |
228 | m_restrictions |= restrictions; |
229 | |
230 | if (restrictions & OverrideUserGestureRequirementForMainContent) |
231 | m_mainContentCheckTimer.startRepeating(elementMainContentCheckInterval); |
232 | } |
233 | |
234 | void MediaElementSession::removeBehaviorRestriction(BehaviorRestrictions restriction) |
235 | { |
236 | if (restriction & RequireUserGestureToControlControlsManager) { |
237 | m_mostRecentUserInteractionTime = MonotonicTime::now(); |
238 | if (auto page = m_element.document().page()) |
239 | page->setAllowsPlaybackControlsForAutoplayingAudio(true); |
240 | } |
241 | |
242 | if (!(m_restrictions & restriction)) |
243 | return; |
244 | |
245 | INFO_LOG(LOGIDENTIFIER, "removed " , restrictionNames(m_restrictions & restriction)); |
246 | m_restrictions &= ~restriction; |
247 | } |
248 | |
249 | SuccessOr<MediaPlaybackDenialReason> MediaElementSession::playbackPermitted() const |
250 | { |
251 | if (m_element.isSuspended()) { |
252 | ALWAYS_LOG(LOGIDENTIFIER, "Returning FALSE because element is suspended" ); |
253 | return MediaPlaybackDenialReason::InvalidState; |
254 | } |
255 | |
256 | auto& document = m_element.document(); |
257 | auto* page = document.page(); |
258 | if (!page || page->mediaPlaybackIsSuspended()) { |
259 | ALWAYS_LOG(LOGIDENTIFIER, "Returning FALSE because media playback is suspended" ); |
260 | return MediaPlaybackDenialReason::PageConsentRequired; |
261 | } |
262 | |
263 | if (document.isMediaDocument() && !document.ownerElement()) |
264 | return { }; |
265 | |
266 | if (pageExplicitlyAllowsElementToAutoplayInline(m_element)) |
267 | return { }; |
268 | |
269 | if (requiresFullscreenForVideoPlayback() && !fullscreenPermitted()) { |
270 | ALWAYS_LOG(LOGIDENTIFIER, "Returning FALSE because of fullscreen restriction" ); |
271 | return MediaPlaybackDenialReason::FullscreenRequired; |
272 | } |
273 | |
274 | if (m_restrictions & OverrideUserGestureRequirementForMainContent && updateIsMainContent()) |
275 | return { }; |
276 | |
277 | #if ENABLE(MEDIA_STREAM) |
278 | if (m_element.hasMediaStreamSrcObject()) { |
279 | if (document.isCapturing()) |
280 | return { }; |
281 | if (document.mediaState() & MediaProducer::IsPlayingAudio) |
282 | return { }; |
283 | } |
284 | #endif |
285 | |
286 | // FIXME: Why are we checking top-level document only for PerDocumentAutoplayBehavior? |
287 | const auto& topDocument = document.topDocument(); |
288 | if (topDocument.mediaState() & MediaProducer::HasUserInteractedWithMediaElement && topDocument.quirks().needsPerDocumentAutoplayBehavior()) |
289 | return { }; |
290 | |
291 | if (document.hasHadUserInteraction() && document.quirks().shouldAutoplayForArbitraryUserGesture()) |
292 | return { }; |
293 | |
294 | if (m_restrictions & RequireUserGestureForVideoRateChange && m_element.isVideo() && !document.processingUserGestureForMedia()) { |
295 | ALWAYS_LOG(LOGIDENTIFIER, "Returning FALSE because a user gesture is required for video rate change restriction" ); |
296 | return MediaPlaybackDenialReason::UserGestureRequired; |
297 | } |
298 | |
299 | if (m_restrictions & RequireUserGestureForAudioRateChange && (!m_element.isVideo() || m_element.hasAudio()) && !m_element.muted() && m_element.volume() && !document.processingUserGestureForMedia()) { |
300 | ALWAYS_LOG(LOGIDENTIFIER, "Returning FALSE because a user gesture is required for audio rate change restriction" ); |
301 | return MediaPlaybackDenialReason::UserGestureRequired; |
302 | } |
303 | |
304 | if (m_restrictions & RequireUserGestureForVideoDueToLowPowerMode && m_element.isVideo() && !document.processingUserGestureForMedia()) { |
305 | ALWAYS_LOG(LOGIDENTIFIER, "Returning FALSE because of video low power mode restriction" ); |
306 | return MediaPlaybackDenialReason::UserGestureRequired; |
307 | } |
308 | |
309 | return { }; |
310 | } |
311 | |
312 | bool MediaElementSession::autoplayPermitted() const |
313 | { |
314 | const Document& document = m_element.document(); |
315 | if (document.pageCacheState() != Document::NotInPageCache) |
316 | return false; |
317 | if (document.activeDOMObjectsAreSuspended()) |
318 | return false; |
319 | |
320 | if (!hasBehaviorRestriction(MediaElementSession::InvisibleAutoplayNotPermitted)) |
321 | return true; |
322 | |
323 | // If the media element is audible, allow autoplay even when not visible as pausing it would be observable by the user. |
324 | if ((!m_element.isVideo() || m_element.hasAudio()) && !m_element.muted() && m_element.volume()) |
325 | return true; |
326 | |
327 | auto* renderer = m_element.renderer(); |
328 | if (!renderer) { |
329 | ALWAYS_LOG(LOGIDENTIFIER, "Returning FALSE because element has no renderer" ); |
330 | return false; |
331 | } |
332 | if (renderer->style().visibility() != Visibility::Visible) { |
333 | ALWAYS_LOG(LOGIDENTIFIER, "Returning FALSE because element is not visible" ); |
334 | return false; |
335 | } |
336 | if (renderer->view().frameView().isOffscreen()) { |
337 | ALWAYS_LOG(LOGIDENTIFIER, "Returning FALSE because frame is offscreen" ); |
338 | return false; |
339 | } |
340 | if (renderer->visibleInViewportState() != VisibleInViewportState::Yes) { |
341 | ALWAYS_LOG(LOGIDENTIFIER, "Returning FALSE because element is not visible in the viewport" ); |
342 | return false; |
343 | } |
344 | return true; |
345 | } |
346 | |
347 | bool MediaElementSession::dataLoadingPermitted() const |
348 | { |
349 | if (m_restrictions & OverrideUserGestureRequirementForMainContent && updateIsMainContent()) |
350 | return true; |
351 | |
352 | if (m_restrictions & RequireUserGestureForLoad && !m_element.document().processingUserGestureForMedia()) { |
353 | INFO_LOG(LOGIDENTIFIER, "returning FALSE" ); |
354 | return false; |
355 | } |
356 | |
357 | return true; |
358 | } |
359 | |
360 | MediaPlayer::BufferingPolicy MediaElementSession::preferredBufferingPolicy() const |
361 | { |
362 | if (isSuspended()) |
363 | return MediaPlayer::BufferingPolicy::MakeResourcesPurgeable; |
364 | |
365 | if (bufferingSuspended()) |
366 | return MediaPlayer::BufferingPolicy::LimitReadAhead; |
367 | |
368 | if (state() == PlatformMediaSession::Playing) |
369 | return MediaPlayer::BufferingPolicy::Default; |
370 | |
371 | if (shouldOverrideBackgroundLoadingRestriction()) |
372 | return MediaPlayer::BufferingPolicy::Default; |
373 | |
374 | #if ENABLE(WIRELESS_PLAYBACK_TARGET) |
375 | if (m_shouldPlayToPlaybackTarget) |
376 | return MediaPlayer::BufferingPolicy::Default; |
377 | #endif |
378 | |
379 | if (m_elementIsHiddenUntilVisibleInViewport || m_elementIsHiddenBecauseItWasRemovedFromDOM || m_element.elementIsHidden()) |
380 | return MediaPlayer::BufferingPolicy::MakeResourcesPurgeable; |
381 | |
382 | return MediaPlayer::BufferingPolicy::Default; |
383 | } |
384 | |
385 | bool MediaElementSession::fullscreenPermitted() const |
386 | { |
387 | if (m_restrictions & RequireUserGestureForFullscreen && !m_element.document().processingUserGestureForMedia()) { |
388 | INFO_LOG(LOGIDENTIFIER, "returning FALSE" ); |
389 | return false; |
390 | } |
391 | |
392 | return true; |
393 | } |
394 | |
395 | bool MediaElementSession::pageAllowsDataLoading() const |
396 | { |
397 | Page* page = m_element.document().page(); |
398 | if (m_restrictions & RequirePageConsentToLoadMedia && page && !page->canStartMedia()) { |
399 | INFO_LOG(LOGIDENTIFIER, "returning FALSE" ); |
400 | return false; |
401 | } |
402 | |
403 | return true; |
404 | } |
405 | |
406 | bool MediaElementSession::pageAllowsPlaybackAfterResuming() const |
407 | { |
408 | Page* page = m_element.document().page(); |
409 | if (m_restrictions & RequirePageConsentToResumeMedia && page && !page->canStartMedia()) { |
410 | INFO_LOG(LOGIDENTIFIER, "returning FALSE" ); |
411 | return false; |
412 | } |
413 | |
414 | return true; |
415 | } |
416 | |
417 | bool MediaElementSession::canShowControlsManager(PlaybackControlsPurpose purpose) const |
418 | { |
419 | if (m_element.isSuspended() || !m_element.inActiveDocument()) { |
420 | INFO_LOG(LOGIDENTIFIER, "returning FALSE: isSuspended()" ); |
421 | return false; |
422 | } |
423 | |
424 | if (m_element.isFullscreen()) { |
425 | INFO_LOG(LOGIDENTIFIER, "returning TRUE: is fullscreen" ); |
426 | return true; |
427 | } |
428 | |
429 | if (m_element.muted()) { |
430 | INFO_LOG(LOGIDENTIFIER, "returning FALSE: muted" ); |
431 | return false; |
432 | } |
433 | |
434 | if (m_element.document().isMediaDocument() && (m_element.document().frame() && m_element.document().frame()->isMainFrame())) { |
435 | INFO_LOG(LOGIDENTIFIER, "returning TRUE: is media document" ); |
436 | return true; |
437 | } |
438 | |
439 | if (client().presentationType() == Audio) { |
440 | if (!hasBehaviorRestriction(RequireUserGestureToControlControlsManager) || m_element.document().processingUserGestureForMedia()) { |
441 | INFO_LOG(LOGIDENTIFIER, "returning TRUE: audio element with user gesture" ); |
442 | return true; |
443 | } |
444 | |
445 | if (m_element.isPlaying() && allowsPlaybackControlsForAutoplayingAudio()) { |
446 | INFO_LOG(LOGIDENTIFIER, "returning TRUE: user has played media before" ); |
447 | return true; |
448 | } |
449 | |
450 | INFO_LOG(LOGIDENTIFIER, "returning FALSE: audio element is not suitable" ); |
451 | return false; |
452 | } |
453 | |
454 | if (purpose == PlaybackControlsPurpose::ControlsManager && !isElementRectMostlyInMainFrame(m_element)) { |
455 | INFO_LOG(LOGIDENTIFIER, "returning FALSE: not in main frame" ); |
456 | return false; |
457 | } |
458 | |
459 | if (!m_element.hasAudio() && !m_element.hasEverHadAudio()) { |
460 | INFO_LOG(LOGIDENTIFIER, "returning FALSE: no audio" ); |
461 | return false; |
462 | } |
463 | |
464 | if (!playbackPermitted()) { |
465 | INFO_LOG(LOGIDENTIFIER, "returning FALSE: playback not permitted" ); |
466 | return false; |
467 | } |
468 | |
469 | if (!hasBehaviorRestriction(RequireUserGestureToControlControlsManager) || m_element.document().processingUserGestureForMedia()) { |
470 | INFO_LOG(LOGIDENTIFIER, "returning TRUE: no user gesture required" ); |
471 | return true; |
472 | } |
473 | |
474 | if (purpose == PlaybackControlsPurpose::ControlsManager && hasBehaviorRestriction(RequirePlaybackToControlControlsManager) && !m_element.isPlaying()) { |
475 | INFO_LOG(LOGIDENTIFIER, "returning FALSE: needs to be playing" ); |
476 | return false; |
477 | } |
478 | |
479 | if (!m_element.hasEverNotifiedAboutPlaying()) { |
480 | INFO_LOG(LOGIDENTIFIER, "returning FALSE: hasn't fired playing notification" ); |
481 | return false; |
482 | } |
483 | |
484 | #if ENABLE(FULLSCREEN_API) |
485 | // Elements which are not descendents of the current fullscreen element cannot be main content. |
486 | auto* fullscreenElement = m_element.document().fullscreenManager().currentFullscreenElement(); |
487 | if (fullscreenElement && !m_element.isDescendantOf(*fullscreenElement)) { |
488 | INFO_LOG(LOGIDENTIFIER, "returning FALSE: outside of full screen" ); |
489 | return false; |
490 | } |
491 | #endif |
492 | |
493 | // Only allow the main content heuristic to forbid videos from showing up if our purpose is the controls manager. |
494 | if (purpose == PlaybackControlsPurpose::ControlsManager && m_element.isVideo()) { |
495 | if (!m_element.renderer()) { |
496 | INFO_LOG(LOGIDENTIFIER, "returning FALSE: no renderer" ); |
497 | return false; |
498 | } |
499 | |
500 | if (!m_element.hasVideo() && !m_element.hasEverHadVideo()) { |
501 | INFO_LOG(LOGIDENTIFIER, "returning FALSE: no video" ); |
502 | return false; |
503 | } |
504 | |
505 | if (isLargeEnoughForMainContent(MediaSessionMainContentPurpose::MediaControls)) { |
506 | INFO_LOG(LOGIDENTIFIER, "returning TRUE: is main content" ); |
507 | return true; |
508 | } |
509 | } |
510 | |
511 | if (purpose == PlaybackControlsPurpose::NowPlaying) { |
512 | INFO_LOG(LOGIDENTIFIER, "returning TRUE: potentially plays audio" ); |
513 | return true; |
514 | } |
515 | |
516 | INFO_LOG(LOGIDENTIFIER, "returning FALSE: no user gesture" ); |
517 | return false; |
518 | } |
519 | |
520 | bool MediaElementSession::isLargeEnoughForMainContent(MediaSessionMainContentPurpose purpose) const |
521 | { |
522 | return isElementLargeEnoughForMainContent(m_element, purpose); |
523 | } |
524 | |
525 | bool MediaElementSession::isMainContentForPurposesOfAutoplayEvents() const |
526 | { |
527 | return isElementMainContentForPurposesOfAutoplay(m_element, false); |
528 | } |
529 | |
530 | MonotonicTime MediaElementSession::mostRecentUserInteractionTime() const |
531 | { |
532 | return m_mostRecentUserInteractionTime; |
533 | } |
534 | |
535 | bool MediaElementSession::wantsToObserveViewportVisibilityForMediaControls() const |
536 | { |
537 | return isLargeEnoughForMainContent(MediaSessionMainContentPurpose::MediaControls); |
538 | } |
539 | |
540 | bool MediaElementSession::wantsToObserveViewportVisibilityForAutoplay() const |
541 | { |
542 | return m_element.isVideo(); |
543 | } |
544 | |
545 | #if ENABLE(WIRELESS_PLAYBACK_TARGET) |
546 | void MediaElementSession::showPlaybackTargetPicker() |
547 | { |
548 | INFO_LOG(LOGIDENTIFIER); |
549 | |
550 | auto& document = m_element.document(); |
551 | if (m_restrictions & RequireUserGestureToShowPlaybackTargetPicker && !document.processingUserGestureForMedia()) { |
552 | INFO_LOG(LOGIDENTIFIER, "returning early because of permissions" ); |
553 | return; |
554 | } |
555 | |
556 | if (!document.page()) { |
557 | INFO_LOG(LOGIDENTIFIER, "returning early because page is NULL" ); |
558 | return; |
559 | } |
560 | |
561 | #if !PLATFORM(IOS_FAMILY) |
562 | if (m_element.readyState() < HTMLMediaElementEnums::HAVE_METADATA) { |
563 | INFO_LOG(LOGIDENTIFIER, "returning early because element is not playable" ); |
564 | return; |
565 | } |
566 | #endif |
567 | |
568 | auto& audioSession = AudioSession::sharedSession(); |
569 | document.showPlaybackTargetPicker(*this, is<HTMLVideoElement>(m_element), audioSession.routeSharingPolicy(), audioSession.routingContextUID()); |
570 | } |
571 | |
572 | bool MediaElementSession::hasWirelessPlaybackTargets() const |
573 | { |
574 | INFO_LOG(LOGIDENTIFIER, "returning " , m_hasPlaybackTargets); |
575 | |
576 | return m_hasPlaybackTargets; |
577 | } |
578 | |
579 | bool MediaElementSession::wirelessVideoPlaybackDisabled() const |
580 | { |
581 | if (!m_element.document().settings().allowsAirPlayForMediaPlayback()) { |
582 | INFO_LOG(LOGIDENTIFIER, "returning TRUE because of settings" ); |
583 | return true; |
584 | } |
585 | |
586 | if (m_element.hasAttributeWithoutSynchronization(HTMLNames::webkitwirelessvideoplaybackdisabledAttr)) { |
587 | INFO_LOG(LOGIDENTIFIER, "returning TRUE because of attribute" ); |
588 | return true; |
589 | } |
590 | |
591 | #if PLATFORM(IOS_FAMILY) |
592 | auto& legacyAirplayAttributeValue = m_element.attributeWithoutSynchronization(HTMLNames::webkitairplayAttr); |
593 | if (equalLettersIgnoringASCIICase(legacyAirplayAttributeValue, "deny" )) { |
594 | INFO_LOG(LOGIDENTIFIER, "returning TRUE because of legacy attribute" ); |
595 | return true; |
596 | } |
597 | if (equalLettersIgnoringASCIICase(legacyAirplayAttributeValue, "allow" )) { |
598 | INFO_LOG(LOGIDENTIFIER, "returning FALSE because of legacy attribute" ); |
599 | return false; |
600 | } |
601 | #endif |
602 | |
603 | auto player = m_element.player(); |
604 | if (!player) |
605 | return true; |
606 | |
607 | bool disabled = player->wirelessVideoPlaybackDisabled(); |
608 | INFO_LOG(LOGIDENTIFIER, "returning " , disabled, " because media engine says so" ); |
609 | |
610 | return disabled; |
611 | } |
612 | |
613 | void MediaElementSession::setWirelessVideoPlaybackDisabled(bool disabled) |
614 | { |
615 | if (disabled) |
616 | addBehaviorRestriction(WirelessVideoPlaybackDisabled); |
617 | else |
618 | removeBehaviorRestriction(WirelessVideoPlaybackDisabled); |
619 | |
620 | auto player = m_element.player(); |
621 | if (!player) |
622 | return; |
623 | |
624 | INFO_LOG(LOGIDENTIFIER, disabled); |
625 | player->setWirelessVideoPlaybackDisabled(disabled); |
626 | } |
627 | |
628 | void MediaElementSession::setHasPlaybackTargetAvailabilityListeners(bool hasListeners) |
629 | { |
630 | INFO_LOG(LOGIDENTIFIER, hasListeners); |
631 | |
632 | #if PLATFORM(IOS_FAMILY) |
633 | m_hasPlaybackTargetAvailabilityListeners = hasListeners; |
634 | PlatformMediaSessionManager::sharedManager().configureWireLessTargetMonitoring(); |
635 | #else |
636 | UNUSED_PARAM(hasListeners); |
637 | m_element.document().playbackTargetPickerClientStateDidChange(*this, m_element.mediaState()); |
638 | #endif |
639 | } |
640 | |
641 | void MediaElementSession::setPlaybackTarget(Ref<MediaPlaybackTarget>&& device) |
642 | { |
643 | m_playbackTarget = WTFMove(device); |
644 | client().setWirelessPlaybackTarget(*m_playbackTarget.copyRef()); |
645 | } |
646 | |
647 | void MediaElementSession::targetAvailabilityChangedTimerFired() |
648 | { |
649 | client().wirelessRoutesAvailableDidChange(); |
650 | } |
651 | |
652 | void MediaElementSession::externalOutputDeviceAvailableDidChange(bool hasTargets) |
653 | { |
654 | if (m_hasPlaybackTargets == hasTargets) |
655 | return; |
656 | |
657 | INFO_LOG(LOGIDENTIFIER, hasTargets); |
658 | |
659 | m_hasPlaybackTargets = hasTargets; |
660 | m_targetAvailabilityChangedTimer.startOneShot(0_s); |
661 | } |
662 | |
663 | bool MediaElementSession::isPlayingToWirelessPlaybackTarget() const |
664 | { |
665 | #if !PLATFORM(IOS_FAMILY) |
666 | if (!m_playbackTarget || !m_playbackTarget->hasActiveRoute()) |
667 | return false; |
668 | #endif |
669 | |
670 | return client().isPlayingToWirelessPlaybackTarget(); |
671 | } |
672 | |
673 | void MediaElementSession::setShouldPlayToPlaybackTarget(bool shouldPlay) |
674 | { |
675 | INFO_LOG(LOGIDENTIFIER, shouldPlay); |
676 | m_shouldPlayToPlaybackTarget = shouldPlay; |
677 | updateClientDataBuffering(); |
678 | client().setShouldPlayToPlaybackTarget(shouldPlay); |
679 | } |
680 | |
681 | void MediaElementSession::mediaStateDidChange(MediaProducer::MediaStateFlags state) |
682 | { |
683 | m_element.document().playbackTargetPickerClientStateDidChange(*this, state); |
684 | } |
685 | #endif |
686 | |
687 | MediaPlayer::Preload MediaElementSession::effectivePreloadForElement() const |
688 | { |
689 | MediaPlayer::Preload preload = m_element.preloadValue(); |
690 | |
691 | if (pageExplicitlyAllowsElementToAutoplayInline(m_element)) |
692 | return preload; |
693 | |
694 | if (m_restrictions & AutoPreloadingNotPermitted) { |
695 | if (preload > MediaPlayer::MetaData) |
696 | return MediaPlayer::MetaData; |
697 | } |
698 | |
699 | return preload; |
700 | } |
701 | |
702 | bool MediaElementSession::requiresFullscreenForVideoPlayback() const |
703 | { |
704 | if (pageExplicitlyAllowsElementToAutoplayInline(m_element)) |
705 | return false; |
706 | |
707 | if (is<HTMLAudioElement>(m_element)) |
708 | return false; |
709 | |
710 | if (m_element.document().isMediaDocument()) { |
711 | ASSERT(is<HTMLVideoElement>(m_element)); |
712 | const HTMLVideoElement& videoElement = *downcast<const HTMLVideoElement>(&m_element); |
713 | if (m_element.readyState() < HTMLVideoElement::HAVE_METADATA || !videoElement.hasEverHadVideo()) |
714 | return false; |
715 | } |
716 | |
717 | if (m_element.isTemporarilyAllowingInlinePlaybackAfterFullscreen()) |
718 | return false; |
719 | |
720 | if (!m_element.document().settings().allowsInlineMediaPlayback()) |
721 | return true; |
722 | |
723 | if (!m_element.document().settings().inlineMediaPlaybackRequiresPlaysInlineAttribute()) |
724 | return false; |
725 | |
726 | #if PLATFORM(IOS_FAMILY) |
727 | if (IOSApplication::isIBooks()) |
728 | return !m_element.hasAttributeWithoutSynchronization(HTMLNames::webkit_playsinlineAttr) && !m_element.hasAttributeWithoutSynchronization(HTMLNames::playsinlineAttr); |
729 | if (dyld_get_program_sdk_version() < DYLD_IOS_VERSION_10_0) |
730 | return !m_element.hasAttributeWithoutSynchronization(HTMLNames::webkit_playsinlineAttr); |
731 | #endif |
732 | |
733 | if (m_element.document().isMediaDocument() && m_element.document().ownerElement()) |
734 | return false; |
735 | |
736 | return !m_element.hasAttributeWithoutSynchronization(HTMLNames::playsinlineAttr); |
737 | } |
738 | |
739 | bool MediaElementSession::allowsAutomaticMediaDataLoading() const |
740 | { |
741 | if (pageExplicitlyAllowsElementToAutoplayInline(m_element)) |
742 | return true; |
743 | |
744 | if (m_element.document().settings().mediaDataLoadsAutomatically()) |
745 | return true; |
746 | |
747 | return false; |
748 | } |
749 | |
750 | void MediaElementSession::mediaEngineUpdated() |
751 | { |
752 | INFO_LOG(LOGIDENTIFIER); |
753 | |
754 | #if ENABLE(WIRELESS_PLAYBACK_TARGET) |
755 | if (m_restrictions & WirelessVideoPlaybackDisabled) |
756 | setWirelessVideoPlaybackDisabled(true); |
757 | if (m_playbackTarget) |
758 | client().setWirelessPlaybackTarget(*m_playbackTarget.copyRef()); |
759 | if (m_shouldPlayToPlaybackTarget) |
760 | client().setShouldPlayToPlaybackTarget(true); |
761 | #endif |
762 | |
763 | } |
764 | |
765 | void MediaElementSession::resetPlaybackSessionState() |
766 | { |
767 | m_mostRecentUserInteractionTime = MonotonicTime(); |
768 | addBehaviorRestriction(RequireUserGestureToControlControlsManager | RequirePlaybackToControlControlsManager); |
769 | } |
770 | |
771 | void MediaElementSession::suspendBuffering() |
772 | { |
773 | updateClientDataBuffering(); |
774 | } |
775 | |
776 | void MediaElementSession::resumeBuffering() |
777 | { |
778 | updateClientDataBuffering(); |
779 | } |
780 | |
781 | bool MediaElementSession::bufferingSuspended() const |
782 | { |
783 | if (auto* page = m_element.document().page()) |
784 | return page->mediaBufferingIsSuspended(); |
785 | return true; |
786 | } |
787 | |
788 | bool MediaElementSession::allowsPictureInPicture() const |
789 | { |
790 | return m_element.document().settings().allowsPictureInPictureMediaPlayback(); |
791 | } |
792 | |
793 | #if PLATFORM(IOS_FAMILY) |
794 | bool MediaElementSession::requiresPlaybackTargetRouteMonitoring() const |
795 | { |
796 | return m_hasPlaybackTargetAvailabilityListeners && !m_element.elementIsHidden(); |
797 | } |
798 | #endif |
799 | |
800 | #if ENABLE(MEDIA_SOURCE) |
801 | size_t MediaElementSession::maximumMediaSourceBufferSize(const SourceBuffer& buffer) const |
802 | { |
803 | // A good quality 1080p video uses 8,000 kbps and stereo audio uses 384 kbps, so assume 95% for video and 5% for audio. |
804 | const float bufferBudgetPercentageForVideo = .95; |
805 | const float bufferBudgetPercentageForAudio = .05; |
806 | |
807 | size_t maximum = buffer.document().settings().maximumSourceBufferSize(); |
808 | |
809 | // Allow a SourceBuffer to buffer as though it is audio-only even if it doesn't have any active tracks (yet). |
810 | size_t bufferSize = static_cast<size_t>(maximum * bufferBudgetPercentageForAudio); |
811 | if (buffer.hasVideo()) |
812 | bufferSize += static_cast<size_t>(maximum * bufferBudgetPercentageForVideo); |
813 | |
814 | // FIXME: we might want to modify this algorithm to: |
815 | // - decrease the maximum size for background tabs |
816 | // - decrease the maximum size allowed for inactive elements when a process has more than one |
817 | // element, eg. so a page with many elements which are played one at a time doesn't keep |
818 | // everything buffered after an element has finished playing. |
819 | |
820 | return bufferSize; |
821 | } |
822 | #endif |
823 | |
824 | static bool isElementMainContentForPurposesOfAutoplay(const HTMLMediaElement& element, bool shouldHitTestMainFrame) |
825 | { |
826 | Document& document = element.document(); |
827 | if (!document.hasLivingRenderTree() || document.activeDOMObjectsAreStopped() || element.isSuspended() || !element.hasAudio() || !element.hasVideo()) |
828 | return false; |
829 | |
830 | // Elements which have not yet been laid out, or which are not yet in the DOM, cannot be main content. |
831 | auto* renderer = element.renderer(); |
832 | if (!renderer) |
833 | return false; |
834 | |
835 | if (!isElementLargeEnoughForMainContent(element, MediaSessionMainContentPurpose::Autoplay)) |
836 | return false; |
837 | |
838 | // Elements which are hidden by style, or have been scrolled out of view, cannot be main content. |
839 | // But elements which have audio & video and are already playing should not stop playing because |
840 | // they are scrolled off the page. |
841 | if (renderer->style().visibility() != Visibility::Visible) |
842 | return false; |
843 | if (renderer->visibleInViewportState() != VisibleInViewportState::Yes && !element.isPlaying()) |
844 | return false; |
845 | |
846 | // Main content elements must be in the main frame. |
847 | if (!document.frame() || !document.frame()->isMainFrame()) |
848 | return false; |
849 | |
850 | auto& mainFrame = document.frame()->mainFrame(); |
851 | if (!mainFrame.view() || !mainFrame.view()->renderView()) |
852 | return false; |
853 | |
854 | if (!shouldHitTestMainFrame) |
855 | return true; |
856 | |
857 | RenderView& mainRenderView = *mainFrame.view()->renderView(); |
858 | |
859 | // Hit test the area of the main frame where the element appears, to determine if the element is being obscured. |
860 | IntRect rectRelativeToView = element.clientRect(); |
861 | ScrollPosition scrollPosition = mainFrame.view()->documentScrollPositionRelativeToViewOrigin(); |
862 | IntRect rectRelativeToTopDocument(rectRelativeToView.location() + scrollPosition, rectRelativeToView.size()); |
863 | HitTestRequest request(HitTestRequest::ReadOnly | HitTestRequest::Active | HitTestRequest::AllowChildFrameContent | HitTestRequest::IgnoreClipping | HitTestRequest::DisallowUserAgentShadowContent); |
864 | HitTestResult result(rectRelativeToTopDocument.center()); |
865 | |
866 | // Elements which are obscured by other elements cannot be main content. |
867 | mainRenderView.hitTest(request, result); |
868 | result.setToNonUserAgentShadowAncestor(); |
869 | RefPtr<Element> hitElement = result.targetElement(); |
870 | if (hitElement != &element) |
871 | return false; |
872 | |
873 | return true; |
874 | } |
875 | |
876 | static bool isElementRectMostlyInMainFrame(const HTMLMediaElement& element) |
877 | { |
878 | if (!element.renderer()) |
879 | return false; |
880 | |
881 | auto documentFrame = makeRefPtr(element.document().frame()); |
882 | if (!documentFrame) |
883 | return false; |
884 | |
885 | auto mainFrameView = documentFrame->mainFrame().view(); |
886 | if (!mainFrameView) |
887 | return false; |
888 | |
889 | IntRect mainFrameRectAdjustedForScrollPosition = IntRect(-mainFrameView->documentScrollPositionRelativeToViewOrigin(), mainFrameView->contentsSize()); |
890 | IntRect elementRectInMainFrame = element.clientRect(); |
891 | auto totalElementArea = elementRectInMainFrame.area<RecordOverflow>(); |
892 | if (totalElementArea.hasOverflowed()) |
893 | return false; |
894 | |
895 | elementRectInMainFrame.intersect(mainFrameRectAdjustedForScrollPosition); |
896 | |
897 | return elementRectInMainFrame.area().unsafeGet() > totalElementArea.unsafeGet() / 2; |
898 | } |
899 | |
900 | static bool isElementLargeRelativeToMainFrame(const HTMLMediaElement& element) |
901 | { |
902 | static const double minimumPercentageOfMainFrameAreaForMainContent = 0.9; |
903 | auto* renderer = element.renderer(); |
904 | if (!renderer) |
905 | return false; |
906 | |
907 | auto documentFrame = makeRefPtr(element.document().frame()); |
908 | if (!documentFrame) |
909 | return false; |
910 | |
911 | if (!documentFrame->mainFrame().view()) |
912 | return false; |
913 | |
914 | auto& mainFrameView = *documentFrame->mainFrame().view(); |
915 | auto maxVisibleClientWidth = std::min(renderer->clientWidth().toInt(), mainFrameView.visibleWidth()); |
916 | auto maxVisibleClientHeight = std::min(renderer->clientHeight().toInt(), mainFrameView.visibleHeight()); |
917 | |
918 | return maxVisibleClientWidth * maxVisibleClientHeight > minimumPercentageOfMainFrameAreaForMainContent * mainFrameView.visibleWidth() * mainFrameView.visibleHeight(); |
919 | } |
920 | |
921 | static bool isElementLargeEnoughForMainContent(const HTMLMediaElement& element, MediaSessionMainContentPurpose purpose) |
922 | { |
923 | static const double elementMainContentAreaMinimum = 400 * 300; |
924 | static const double maximumAspectRatio = purpose == MediaSessionMainContentPurpose::MediaControls ? 3 : 1.8; |
925 | static const double minimumAspectRatio = .5; // Slightly smaller than 9:16. |
926 | |
927 | // Elements which have not yet been laid out, or which are not yet in the DOM, cannot be main content. |
928 | auto* renderer = element.renderer(); |
929 | if (!renderer) |
930 | return false; |
931 | |
932 | double width = renderer->clientWidth(); |
933 | double height = renderer->clientHeight(); |
934 | double area = width * height; |
935 | double aspectRatio = width / height; |
936 | |
937 | if (area < elementMainContentAreaMinimum) |
938 | return false; |
939 | |
940 | if (aspectRatio >= minimumAspectRatio && aspectRatio <= maximumAspectRatio) |
941 | return true; |
942 | |
943 | return isElementLargeRelativeToMainFrame(element); |
944 | } |
945 | |
946 | void MediaElementSession::mainContentCheckTimerFired() |
947 | { |
948 | if (!hasBehaviorRestriction(OverrideUserGestureRequirementForMainContent)) |
949 | return; |
950 | |
951 | updateIsMainContent(); |
952 | } |
953 | |
954 | bool MediaElementSession::updateIsMainContent() const |
955 | { |
956 | if (m_element.isSuspended()) |
957 | return false; |
958 | |
959 | bool wasMainContent = m_isMainContent; |
960 | m_isMainContent = isElementMainContentForPurposesOfAutoplay(m_element, true); |
961 | |
962 | if (m_isMainContent != wasMainContent) |
963 | m_element.updateShouldPlay(); |
964 | |
965 | return m_isMainContent; |
966 | } |
967 | |
968 | bool MediaElementSession::allowsNowPlayingControlsVisibility() const |
969 | { |
970 | auto page = m_element.document().page(); |
971 | return page && !page->isVisibleAndActive(); |
972 | } |
973 | |
974 | bool MediaElementSession::allowsPlaybackControlsForAutoplayingAudio() const |
975 | { |
976 | auto page = m_element.document().page(); |
977 | return page && page->allowsPlaybackControlsForAutoplayingAudio(); |
978 | } |
979 | |
980 | String convertEnumerationToString(const MediaPlaybackDenialReason enumerationValue) |
981 | { |
982 | static const NeverDestroyed<String> values[] = { |
983 | MAKE_STATIC_STRING_IMPL("UserGestureRequired" ), |
984 | MAKE_STATIC_STRING_IMPL("FullscreenRequired" ), |
985 | MAKE_STATIC_STRING_IMPL("PageConsentRequired" ), |
986 | MAKE_STATIC_STRING_IMPL("InvalidState" ), |
987 | }; |
988 | static_assert(static_cast<size_t>(MediaPlaybackDenialReason::UserGestureRequired) == 0, "MediaPlaybackDenialReason::UserGestureRequired is not 0 as expected" ); |
989 | static_assert(static_cast<size_t>(MediaPlaybackDenialReason::FullscreenRequired) == 1, "MediaPlaybackDenialReason::FullscreenRequired is not 1 as expected" ); |
990 | static_assert(static_cast<size_t>(MediaPlaybackDenialReason::PageConsentRequired) == 2, "MediaPlaybackDenialReason::PageConsentRequired is not 2 as expected" ); |
991 | static_assert(static_cast<size_t>(MediaPlaybackDenialReason::InvalidState) == 3, "MediaPlaybackDenialReason::InvalidState is not 3 as expected" ); |
992 | ASSERT(static_cast<size_t>(enumerationValue) < WTF_ARRAY_LENGTH(values)); |
993 | return values[static_cast<size_t>(enumerationValue)]; |
994 | } |
995 | |
996 | } |
997 | |
998 | #endif // ENABLE(VIDEO) |
999 | |