blob: 60ea6e7d10563ee17940fc5933e833e9a0cfad1a [file] [log] [blame]
/*
* Copyright (C) 2013, 2014 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include "MediaControlsHost.h"
#include "CaptionUserPreferences.h"
#include "HTMLMediaElement.h"
#include "Logging.h"
#include "MediaControlTextTrackContainerElement.h"
#include "Page.h"
#include "PageGroup.h"
#include "RenderTheme.h"
#include "TextTrack.h"
#include "TextTrackList.h"
#include <JavaScriptCore/JSCJSValueInlines.h>
#include <wtf/UUID.h>
namespace WebCore {
const AtomString& MediaControlsHost::automaticKeyword()
{
static NeverDestroyed<const AtomString> automatic("automatic", AtomString::ConstructFromLiteral);
return automatic;
}
const AtomString& MediaControlsHost::forcedOnlyKeyword()
{
static NeverDestroyed<const AtomString> forcedOnly("forced-only", AtomString::ConstructFromLiteral);
return forcedOnly;
}
static const AtomString& alwaysOnKeyword()
{
static NeverDestroyed<const AtomString> alwaysOn("always-on", AtomString::ConstructFromLiteral);
return alwaysOn;
}
static const AtomString& manualKeyword()
{
static NeverDestroyed<const AtomString> alwaysOn("manual", AtomString::ConstructFromLiteral);
return alwaysOn;
}
Ref<MediaControlsHost> MediaControlsHost::create(HTMLMediaElement& mediaElement)
{
return adoptRef(*new MediaControlsHost(mediaElement));
}
MediaControlsHost::MediaControlsHost(HTMLMediaElement& mediaElement)
: m_mediaElement(makeWeakPtr(mediaElement))
{
}
MediaControlsHost::~MediaControlsHost() = default;
Vector<RefPtr<TextTrack>> MediaControlsHost::sortedTrackListForMenu(TextTrackList& trackList)
{
if (!m_mediaElement)
return { };
Page* page = m_mediaElement->document().page();
if (!page)
return { };
return page->group().captionPreferences().sortedTrackListForMenu(&trackList);
}
Vector<RefPtr<AudioTrack>> MediaControlsHost::sortedTrackListForMenu(AudioTrackList& trackList)
{
if (!m_mediaElement)
return { };
Page* page = m_mediaElement->document().page();
if (!page)
return { };
return page->group().captionPreferences().sortedTrackListForMenu(&trackList);
}
String MediaControlsHost::displayNameForTrack(const Optional<TextOrAudioTrack>& track)
{
if (!m_mediaElement || !track)
return emptyString();
Page* page = m_mediaElement->document().page();
if (!page)
return emptyString();
return WTF::visit([page] (auto& track) {
return page->group().captionPreferences().displayNameForTrack(track.get());
}, track.value());
}
TextTrack& MediaControlsHost::captionMenuOffItem()
{
return TextTrack::captionMenuOffItem();
}
TextTrack& MediaControlsHost::captionMenuAutomaticItem()
{
return TextTrack::captionMenuAutomaticItem();
}
AtomString MediaControlsHost::captionDisplayMode() const
{
if (!m_mediaElement)
return emptyAtom();
Page* page = m_mediaElement->document().page();
if (!page)
return emptyAtom();
switch (page->group().captionPreferences().captionDisplayMode()) {
case CaptionUserPreferences::Automatic:
return automaticKeyword();
case CaptionUserPreferences::ForcedOnly:
return forcedOnlyKeyword();
case CaptionUserPreferences::AlwaysOn:
return alwaysOnKeyword();
case CaptionUserPreferences::Manual:
return manualKeyword();
default:
ASSERT_NOT_REACHED();
return emptyAtom();
}
}
void MediaControlsHost::setSelectedTextTrack(TextTrack* track)
{
if (m_mediaElement)
m_mediaElement->setSelectedTextTrack(track);
}
Element* MediaControlsHost::textTrackContainer()
{
if (!m_textTrackContainer && m_mediaElement)
m_textTrackContainer = MediaControlTextTrackContainerElement::create(m_mediaElement->document(), *m_mediaElement);
return m_textTrackContainer.get();
}
void MediaControlsHost::updateTextTrackContainer()
{
if (m_textTrackContainer)
m_textTrackContainer->updateDisplay();
}
void MediaControlsHost::updateTextTrackRepresentationImageIfNeeded()
{
if (m_textTrackContainer)
m_textTrackContainer->updateTextTrackRepresentationImageIfNeeded();
}
void MediaControlsHost::enteredFullscreen()
{
if (m_textTrackContainer)
m_textTrackContainer->enteredFullscreen();
}
void MediaControlsHost::exitedFullscreen()
{
if (m_textTrackContainer)
m_textTrackContainer->exitedFullscreen();
}
void MediaControlsHost::updateCaptionDisplaySizes(ForceUpdate force)
{
if (m_textTrackContainer)
m_textTrackContainer->updateSizes(force == ForceUpdate::Yes ? MediaControlTextTrackContainerElement::ForceUpdate::Yes : MediaControlTextTrackContainerElement::ForceUpdate::No);
}
bool MediaControlsHost::allowsInlineMediaPlayback() const
{
return m_mediaElement && !m_mediaElement->mediaSession().requiresFullscreenForVideoPlayback();
}
bool MediaControlsHost::supportsFullscreen() const
{
return m_mediaElement && m_mediaElement->supportsFullscreen(HTMLMediaElementEnums::VideoFullscreenModeStandard);
}
bool MediaControlsHost::isVideoLayerInline() const
{
return m_mediaElement && m_mediaElement->isVideoLayerInline();
}
bool MediaControlsHost::isInMediaDocument() const
{
return m_mediaElement && m_mediaElement->document().isMediaDocument();
}
void MediaControlsHost::setPreparedToReturnVideoLayerToInline(bool value)
{
if (m_mediaElement)
m_mediaElement->setPreparedToReturnVideoLayerToInline(value);
}
bool MediaControlsHost::userGestureRequired() const
{
return m_mediaElement && !m_mediaElement->mediaSession().playbackPermitted();
}
bool MediaControlsHost::shouldForceControlsDisplay() const
{
return m_mediaElement && m_mediaElement->shouldForceControlsDisplay();
}
String MediaControlsHost::externalDeviceDisplayName() const
{
#if ENABLE(WIRELESS_PLAYBACK_TARGET)
if (!m_mediaElement)
return emptyString();
auto player = m_mediaElement->player();
if (!player) {
LOG(Media, "MediaControlsHost::externalDeviceDisplayName - returning \"\" because player is NULL");
return emptyString();
}
String name = player->wirelessPlaybackTargetName();
LOG(Media, "MediaControlsHost::externalDeviceDisplayName - returning \"%s\"", name.utf8().data());
return name;
#else
return emptyString();
#endif
}
auto MediaControlsHost::externalDeviceType() const -> DeviceType
{
#if !ENABLE(WIRELESS_PLAYBACK_TARGET)
return DeviceType::None;
#else
if (!m_mediaElement)
return DeviceType::None;
auto player = m_mediaElement->player();
if (!player) {
LOG(Media, "MediaControlsHost::externalDeviceType - returning \"none\" because player is NULL");
return DeviceType::None;
}
switch (player->wirelessPlaybackTargetType()) {
case MediaPlayer::WirelessPlaybackTargetType::TargetTypeNone:
return DeviceType::None;
case MediaPlayer::WirelessPlaybackTargetType::TargetTypeAirPlay:
return DeviceType::Airplay;
case MediaPlayer::WirelessPlaybackTargetType::TargetTypeTVOut:
return DeviceType::Tvout;
}
ASSERT_NOT_REACHED();
return DeviceType::None;
#endif
}
bool MediaControlsHost::controlsDependOnPageScaleFactor() const
{
return m_mediaElement && m_mediaElement->mediaControlsDependOnPageScaleFactor();
}
void MediaControlsHost::setControlsDependOnPageScaleFactor(bool value)
{
if (m_mediaElement)
m_mediaElement->setMediaControlsDependOnPageScaleFactor(value);
}
String MediaControlsHost::generateUUID()
{
return createCanonicalUUIDString();
}
String MediaControlsHost::shadowRootCSSText()
{
return RenderTheme::singleton().modernMediaControlsStyleSheet();
}
String MediaControlsHost::base64StringForIconNameAndType(const String& iconName, const String& iconType)
{
return RenderTheme::singleton().mediaControlsBase64StringForIconNameAndType(iconName, iconType);
}
String MediaControlsHost::formattedStringForDuration(double durationInSeconds)
{
return RenderTheme::singleton().mediaControlsFormattedStringForDuration(durationInSeconds);
}
bool MediaControlsHost::compactMode() const
{
#if PLATFORM(WATCHOS)
return true;
#else
return m_simulateCompactMode;
#endif
}
}