| // Copyright (c) 2009 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "config.h" |
| #include "WebMediaPlayerClientImpl.h" |
| |
| #if ENABLE(VIDEO) |
| |
| #include "Frame.h" |
| #include "GraphicsContext.h" |
| #include "HTMLMediaElement.h" |
| #include "IntSize.h" |
| #include "KURL.h" |
| #include "MediaPlayer.h" |
| #include "NotImplemented.h" |
| #include "RenderView.h" |
| #include "TimeRanges.h" |
| #include "VideoLayerChromium.h" |
| |
| #if USE(ACCELERATED_COMPOSITING) |
| #include "RenderLayerCompositor.h" |
| #endif |
| |
| #include "VideoFrameChromium.h" |
| #include "VideoFrameChromiumImpl.h" |
| #include "WebCanvas.h" |
| #include "WebCString.h" |
| #include "WebFrameClient.h" |
| #include "WebFrameImpl.h" |
| #include "WebKit.h" |
| #include "WebKitClient.h" |
| #include "WebMediaElement.h" |
| #include "WebMediaPlayer.h" |
| #include "WebMimeRegistry.h" |
| #include "WebRect.h" |
| #include "WebSize.h" |
| #include "WebString.h" |
| #include "WebURL.h" |
| #include "WebViewImpl.h" |
| |
| // WebCommon.h defines WEBKIT_USING_SKIA so this has to be included last. |
| #if WEBKIT_USING_SKIA |
| #include "PlatformContextSkia.h" |
| #endif |
| |
| #include <wtf/Assertions.h> |
| #include <wtf/text/CString.h> |
| |
| using namespace WebCore; |
| |
| namespace WebKit { |
| |
| static WebMediaPlayer* createWebMediaPlayer( |
| WebMediaPlayerClient* client, Frame* frame) |
| { |
| WebFrameImpl* webFrame = WebFrameImpl::fromFrame(frame); |
| |
| if (!webFrame->client()) |
| return 0; |
| return webFrame->client()->createMediaPlayer(webFrame, client); |
| } |
| |
| bool WebMediaPlayerClientImpl::m_isEnabled = false; |
| |
| bool WebMediaPlayerClientImpl::isEnabled() |
| { |
| return m_isEnabled; |
| } |
| |
| void WebMediaPlayerClientImpl::setIsEnabled(bool isEnabled) |
| { |
| m_isEnabled = isEnabled; |
| } |
| |
| void WebMediaPlayerClientImpl::registerSelf(MediaEngineRegistrar registrar) |
| { |
| if (m_isEnabled) { |
| registrar(WebMediaPlayerClientImpl::create, |
| WebMediaPlayerClientImpl::getSupportedTypes, |
| WebMediaPlayerClientImpl::supportsType, |
| 0, |
| 0, |
| 0); |
| } |
| } |
| |
| WebMediaPlayerClientImpl* WebMediaPlayerClientImpl::fromMediaElement(const WebMediaElement* element) |
| { |
| PlatformMedia pm = element->constUnwrap<HTMLMediaElement>()->platformMedia(); |
| return static_cast<WebMediaPlayerClientImpl*>(pm.media.chromiumMediaPlayer); |
| } |
| |
| WebMediaPlayer* WebMediaPlayerClientImpl::mediaPlayer() const |
| { |
| return m_webMediaPlayer.get(); |
| } |
| |
| // WebMediaPlayerClient -------------------------------------------------------- |
| |
| WebMediaPlayerClientImpl::~WebMediaPlayerClientImpl() |
| { |
| // VideoLayerChromium may outlive this object so make sure all frames are |
| // released. |
| #if USE(ACCELERATED_COMPOSITING) |
| if (m_videoLayer.get()) |
| m_videoLayer->releaseCurrentFrame(); |
| #endif |
| } |
| |
| void WebMediaPlayerClientImpl::networkStateChanged() |
| { |
| ASSERT(m_mediaPlayer); |
| m_mediaPlayer->networkStateChanged(); |
| } |
| |
| void WebMediaPlayerClientImpl::readyStateChanged() |
| { |
| ASSERT(m_mediaPlayer); |
| m_mediaPlayer->readyStateChanged(); |
| #if USE(ACCELERATED_COMPOSITING) |
| if (hasVideo() && supportsAcceleratedRendering() && !m_videoLayer.get()) |
| m_videoLayer = VideoLayerChromium::create(0, this); |
| #endif |
| } |
| |
| void WebMediaPlayerClientImpl::volumeChanged(float newVolume) |
| { |
| ASSERT(m_mediaPlayer); |
| m_mediaPlayer->volumeChanged(newVolume); |
| } |
| |
| void WebMediaPlayerClientImpl::muteChanged(bool newMute) |
| { |
| ASSERT(m_mediaPlayer); |
| m_mediaPlayer->muteChanged(newMute); |
| } |
| |
| void WebMediaPlayerClientImpl::timeChanged() |
| { |
| ASSERT(m_mediaPlayer); |
| m_mediaPlayer->timeChanged(); |
| } |
| |
| void WebMediaPlayerClientImpl::repaint() |
| { |
| ASSERT(m_mediaPlayer); |
| #if USE(ACCELERATED_COMPOSITING) |
| if (m_videoLayer.get() && supportsAcceleratedRendering()) |
| m_videoLayer->setNeedsDisplay(IntRect(0, 0, m_videoLayer->bounds().width(), m_videoLayer->bounds().height())); |
| #endif |
| m_mediaPlayer->repaint(); |
| } |
| |
| void WebMediaPlayerClientImpl::durationChanged() |
| { |
| ASSERT(m_mediaPlayer); |
| m_mediaPlayer->durationChanged(); |
| } |
| |
| void WebMediaPlayerClientImpl::rateChanged() |
| { |
| ASSERT(m_mediaPlayer); |
| m_mediaPlayer->rateChanged(); |
| } |
| |
| void WebMediaPlayerClientImpl::sizeChanged() |
| { |
| ASSERT(m_mediaPlayer); |
| m_mediaPlayer->sizeChanged(); |
| } |
| |
| void WebMediaPlayerClientImpl::sawUnsupportedTracks() |
| { |
| ASSERT(m_mediaPlayer); |
| m_mediaPlayer->mediaPlayerClient()->mediaPlayerSawUnsupportedTracks(m_mediaPlayer); |
| } |
| |
| float WebMediaPlayerClientImpl::volume() const |
| { |
| if (m_mediaPlayer) |
| return m_mediaPlayer->volume(); |
| return 0.0f; |
| } |
| |
| void WebMediaPlayerClientImpl::playbackStateChanged() |
| { |
| ASSERT(m_mediaPlayer); |
| m_mediaPlayer->playbackStateChanged(); |
| } |
| |
| WebMediaPlayer::Preload WebMediaPlayerClientImpl::preload() const |
| { |
| if (m_mediaPlayer) |
| return static_cast<WebMediaPlayer::Preload>(m_mediaPlayer->preload()); |
| return static_cast<WebMediaPlayer::Preload>(m_preload); |
| } |
| |
| // MediaPlayerPrivateInterface ------------------------------------------------- |
| |
| void WebMediaPlayerClientImpl::load(const String& url) |
| { |
| m_url = url; |
| |
| // Video frame object is owned by WebMediaPlayer. Before destroying |
| // WebMediaPlayer all frames need to be released. |
| #if USE(ACCELERATED_COMPOSITING) |
| if (m_videoLayer.get()) |
| m_videoLayer->releaseCurrentFrame(); |
| #endif |
| |
| if (m_preload == MediaPlayer::None) { |
| m_webMediaPlayer.clear(); |
| m_delayingLoad = true; |
| } else |
| loadInternal(); |
| } |
| |
| void WebMediaPlayerClientImpl::loadInternal() |
| { |
| Frame* frame = static_cast<HTMLMediaElement*>(m_mediaPlayer->mediaPlayerClient())->document()->frame(); |
| m_webMediaPlayer.set(createWebMediaPlayer(this, frame)); |
| if (m_webMediaPlayer.get()) |
| m_webMediaPlayer->load(KURL(ParsedURLString, m_url)); |
| } |
| |
| void WebMediaPlayerClientImpl::cancelLoad() |
| { |
| if (m_webMediaPlayer.get()) |
| m_webMediaPlayer->cancelLoad(); |
| } |
| |
| #if USE(ACCELERATED_COMPOSITING) |
| PlatformLayer* WebMediaPlayerClientImpl::platformLayer() const |
| { |
| ASSERT(m_supportsAcceleratedCompositing); |
| return m_videoLayer.get(); |
| } |
| #endif |
| |
| PlatformMedia WebMediaPlayerClientImpl::platformMedia() const |
| { |
| PlatformMedia pm; |
| pm.type = PlatformMedia::ChromiumMediaPlayerType; |
| pm.media.chromiumMediaPlayer = const_cast<WebMediaPlayerClientImpl*>(this); |
| return pm; |
| } |
| |
| void WebMediaPlayerClientImpl::play() |
| { |
| if (m_webMediaPlayer.get()) |
| m_webMediaPlayer->play(); |
| } |
| |
| void WebMediaPlayerClientImpl::pause() |
| { |
| if (m_webMediaPlayer.get()) |
| m_webMediaPlayer->pause(); |
| } |
| |
| void WebMediaPlayerClientImpl::prepareToPlay() |
| { |
| if (m_delayingLoad) |
| startDelayedLoad(); |
| } |
| |
| IntSize WebMediaPlayerClientImpl::naturalSize() const |
| { |
| if (m_webMediaPlayer.get()) |
| return m_webMediaPlayer->naturalSize(); |
| return IntSize(); |
| } |
| |
| bool WebMediaPlayerClientImpl::hasVideo() const |
| { |
| if (m_webMediaPlayer.get()) |
| return m_webMediaPlayer->hasVideo(); |
| return false; |
| } |
| |
| bool WebMediaPlayerClientImpl::hasAudio() const |
| { |
| if (m_webMediaPlayer.get()) |
| return m_webMediaPlayer->hasAudio(); |
| return false; |
| } |
| |
| void WebMediaPlayerClientImpl::setVisible(bool visible) |
| { |
| if (m_webMediaPlayer.get()) |
| m_webMediaPlayer->setVisible(visible); |
| } |
| |
| float WebMediaPlayerClientImpl::duration() const |
| { |
| if (m_webMediaPlayer.get()) |
| return m_webMediaPlayer->duration(); |
| return 0.0f; |
| } |
| |
| float WebMediaPlayerClientImpl::currentTime() const |
| { |
| if (m_webMediaPlayer.get()) |
| return m_webMediaPlayer->currentTime(); |
| return 0.0f; |
| } |
| |
| void WebMediaPlayerClientImpl::seek(float time) |
| { |
| if (m_webMediaPlayer.get()) |
| m_webMediaPlayer->seek(time); |
| } |
| |
| bool WebMediaPlayerClientImpl::seeking() const |
| { |
| if (m_webMediaPlayer.get()) |
| return m_webMediaPlayer->seeking(); |
| return false; |
| } |
| |
| void WebMediaPlayerClientImpl::setEndTime(float time) |
| { |
| if (m_webMediaPlayer.get()) |
| m_webMediaPlayer->setEndTime(time); |
| } |
| |
| void WebMediaPlayerClientImpl::setRate(float rate) |
| { |
| if (m_webMediaPlayer.get()) |
| m_webMediaPlayer->setRate(rate); |
| } |
| |
| bool WebMediaPlayerClientImpl::paused() const |
| { |
| if (m_webMediaPlayer.get()) |
| return m_webMediaPlayer->paused(); |
| return false; |
| } |
| |
| bool WebMediaPlayerClientImpl::supportsFullscreen() const |
| { |
| if (m_webMediaPlayer.get()) |
| return m_webMediaPlayer->supportsFullscreen(); |
| return false; |
| } |
| |
| bool WebMediaPlayerClientImpl::supportsSave() const |
| { |
| if (m_webMediaPlayer.get()) |
| return m_webMediaPlayer->supportsSave(); |
| return false; |
| } |
| |
| void WebMediaPlayerClientImpl::setVolume(float volume) |
| { |
| if (m_webMediaPlayer.get()) |
| m_webMediaPlayer->setVolume(volume); |
| } |
| |
| MediaPlayer::NetworkState WebMediaPlayerClientImpl::networkState() const |
| { |
| if (m_webMediaPlayer.get()) |
| return static_cast<MediaPlayer::NetworkState>(m_webMediaPlayer->networkState()); |
| return MediaPlayer::Empty; |
| } |
| |
| MediaPlayer::ReadyState WebMediaPlayerClientImpl::readyState() const |
| { |
| if (m_webMediaPlayer.get()) |
| return static_cast<MediaPlayer::ReadyState>(m_webMediaPlayer->readyState()); |
| return MediaPlayer::HaveNothing; |
| } |
| |
| float WebMediaPlayerClientImpl::maxTimeSeekable() const |
| { |
| if (m_webMediaPlayer.get()) |
| return m_webMediaPlayer->maxTimeSeekable(); |
| return 0.0f; |
| } |
| |
| PassRefPtr<TimeRanges> WebMediaPlayerClientImpl::buffered() const |
| { |
| if (m_webMediaPlayer.get()) { |
| const WebTimeRanges& webRanges = m_webMediaPlayer->buffered(); |
| |
| // FIXME: Save the time ranges in a member variable and update it when needed. |
| RefPtr<TimeRanges> ranges = TimeRanges::create(); |
| for (size_t i = 0; i < webRanges.size(); ++i) |
| ranges->add(webRanges[i].start, webRanges[i].end); |
| return ranges.release(); |
| } |
| return TimeRanges::create(); |
| } |
| |
| int WebMediaPlayerClientImpl::dataRate() const |
| { |
| if (m_webMediaPlayer.get()) |
| return m_webMediaPlayer->dataRate(); |
| return 0; |
| } |
| |
| bool WebMediaPlayerClientImpl::totalBytesKnown() const |
| { |
| if (m_webMediaPlayer.get()) |
| return m_webMediaPlayer->totalBytesKnown(); |
| return false; |
| } |
| |
| unsigned WebMediaPlayerClientImpl::totalBytes() const |
| { |
| if (m_webMediaPlayer.get()) |
| return static_cast<unsigned>(m_webMediaPlayer->totalBytes()); |
| return 0; |
| } |
| |
| unsigned WebMediaPlayerClientImpl::bytesLoaded() const |
| { |
| if (m_webMediaPlayer.get()) |
| return static_cast<unsigned>(m_webMediaPlayer->bytesLoaded()); |
| return 0; |
| } |
| |
| void WebMediaPlayerClientImpl::setSize(const IntSize& size) |
| { |
| if (m_webMediaPlayer.get()) |
| m_webMediaPlayer->setSize(WebSize(size.width(), size.height())); |
| } |
| |
| void WebMediaPlayerClientImpl::paint(GraphicsContext* context, const IntRect& rect) |
| { |
| #if USE(ACCELERATED_COMPOSITING) |
| // If we are using GPU to render video, ignore requests to paint frames into |
| // canvas because it will be taken care of by VideoLayerChromium. |
| if (acceleratedRenderingInUse()) |
| return; |
| #endif |
| paintCurrentFrameInContext(context, rect); |
| } |
| |
| void WebMediaPlayerClientImpl::paintCurrentFrameInContext(GraphicsContext* context, const IntRect& rect) |
| { |
| // Normally GraphicsContext operations do nothing when painting is disabled. |
| // Since we're accessing platformContext() directly we have to manually |
| // check. |
| if (m_webMediaPlayer.get() && !context->paintingDisabled()) { |
| #if WEBKIT_USING_SKIA |
| PlatformGraphicsContext* platformContext = context->platformContext(); |
| WebCanvas* canvas = platformContext->canvas(); |
| |
| canvas->saveLayerAlpha(0, platformContext->getNormalizedAlpha()); |
| |
| m_webMediaPlayer->paint(canvas, rect); |
| |
| canvas->restore(); |
| #elif WEBKIT_USING_CG |
| m_webMediaPlayer->paint(context->platformContext(), rect); |
| #else |
| notImplemented(); |
| #endif |
| } |
| } |
| |
| void WebMediaPlayerClientImpl::setPreload(MediaPlayer::Preload preload) |
| { |
| m_preload = preload; |
| |
| if (m_webMediaPlayer.get()) |
| m_webMediaPlayer->setPreload(static_cast<WebMediaPlayer::Preload>(preload)); |
| |
| if (m_delayingLoad && m_preload != MediaPlayer::None) |
| startDelayedLoad(); |
| } |
| |
| bool WebMediaPlayerClientImpl::hasSingleSecurityOrigin() const |
| { |
| if (m_webMediaPlayer.get()) |
| return m_webMediaPlayer->hasSingleSecurityOrigin(); |
| return false; |
| } |
| |
| MediaPlayer::MovieLoadType WebMediaPlayerClientImpl::movieLoadType() const |
| { |
| if (m_webMediaPlayer.get()) |
| return static_cast<MediaPlayer::MovieLoadType>( |
| m_webMediaPlayer->movieLoadType()); |
| return MediaPlayer::Unknown; |
| } |
| |
| unsigned WebMediaPlayerClientImpl::decodedFrameCount() const |
| { |
| if (m_webMediaPlayer.get()) |
| return m_webMediaPlayer->decodedFrameCount(); |
| return 0; |
| } |
| |
| unsigned WebMediaPlayerClientImpl::droppedFrameCount() const |
| { |
| if (m_webMediaPlayer.get()) |
| return m_webMediaPlayer->droppedFrameCount(); |
| return 0; |
| } |
| |
| unsigned WebMediaPlayerClientImpl::audioDecodedByteCount() const |
| { |
| if (m_webMediaPlayer.get()) |
| return m_webMediaPlayer->audioDecodedByteCount(); |
| return 0; |
| } |
| |
| unsigned WebMediaPlayerClientImpl::videoDecodedByteCount() const |
| { |
| if (m_webMediaPlayer.get()) |
| return m_webMediaPlayer->videoDecodedByteCount(); |
| return 0; |
| } |
| |
| #if USE(ACCELERATED_COMPOSITING) |
| bool WebMediaPlayerClientImpl::supportsAcceleratedRendering() const |
| { |
| return m_supportsAcceleratedCompositing; |
| } |
| |
| bool WebMediaPlayerClientImpl::acceleratedRenderingInUse() |
| { |
| return m_videoLayer.get() && m_videoLayer->layerRenderer(); |
| } |
| |
| VideoFrameChromium* WebMediaPlayerClientImpl::getCurrentFrame() |
| { |
| VideoFrameChromium* videoFrame = 0; |
| if (m_webMediaPlayer.get()) { |
| WebVideoFrame* webkitVideoFrame = m_webMediaPlayer->getCurrentFrame(); |
| if (webkitVideoFrame) |
| videoFrame = new VideoFrameChromiumImpl(webkitVideoFrame); |
| } |
| return videoFrame; |
| } |
| |
| void WebMediaPlayerClientImpl::putCurrentFrame(VideoFrameChromium* videoFrame) |
| { |
| if (videoFrame) { |
| if (m_webMediaPlayer.get()) { |
| m_webMediaPlayer->putCurrentFrame( |
| VideoFrameChromiumImpl::toWebVideoFrame(videoFrame)); |
| } |
| delete videoFrame; |
| } |
| } |
| #endif |
| |
| MediaPlayerPrivateInterface* WebMediaPlayerClientImpl::create(MediaPlayer* player) |
| { |
| WebMediaPlayerClientImpl* client = new WebMediaPlayerClientImpl(); |
| client->m_mediaPlayer = player; |
| |
| #if USE(ACCELERATED_COMPOSITING) |
| Frame* frame = static_cast<HTMLMediaElement*>( |
| client->m_mediaPlayer->mediaPlayerClient())->document()->frame(); |
| |
| // This does not actually check whether the hardware can support accelerated |
| // compositing, but only if the flag is set. However, this is checked lazily |
| // in WebViewImpl::setIsAcceleratedCompositingActive() and will fail there |
| // if necessary. |
| client->m_supportsAcceleratedCompositing = |
| frame->contentRenderer()->compositor()->hasAcceleratedCompositing(); |
| #endif |
| |
| return client; |
| } |
| |
| void WebMediaPlayerClientImpl::getSupportedTypes(HashSet<String>& supportedTypes) |
| { |
| // FIXME: integrate this list with WebMediaPlayerClientImpl::supportsType. |
| notImplemented(); |
| } |
| |
| MediaPlayer::SupportsType WebMediaPlayerClientImpl::supportsType(const String& type, |
| const String& codecs) |
| { |
| WebMimeRegistry::SupportsType supportsType = |
| webKitClient()->mimeRegistry()->supportsMediaMIMEType(type, codecs); |
| |
| switch (supportsType) { |
| default: |
| ASSERT_NOT_REACHED(); |
| case WebMimeRegistry::IsNotSupported: |
| return MediaPlayer::IsNotSupported; |
| case WebMimeRegistry::IsSupported: |
| return MediaPlayer::IsSupported; |
| case WebMimeRegistry::MayBeSupported: |
| return MediaPlayer::MayBeSupported; |
| } |
| return MediaPlayer::IsNotSupported; |
| } |
| |
| void WebMediaPlayerClientImpl::startDelayedLoad() |
| { |
| ASSERT(m_delayingLoad); |
| ASSERT(!m_webMediaPlayer.get()); |
| |
| m_delayingLoad = false; |
| |
| loadInternal(); |
| } |
| |
| WebMediaPlayerClientImpl::WebMediaPlayerClientImpl() |
| : m_mediaPlayer(0) |
| , m_delayingLoad(false) |
| , m_preload(MediaPlayer::MetaData) |
| #if USE(ACCELERATED_COMPOSITING) |
| , m_videoLayer(0) |
| , m_supportsAcceleratedCompositing(false) |
| #endif |
| { |
| } |
| |
| } // namespace WebKit |
| |
| #endif // ENABLE(VIDEO) |