| Index: content/renderer/media/media_recorder_handler.cc | 
| diff --git a/content/renderer/media/media_recorder_handler.cc b/content/renderer/media/media_recorder_handler.cc | 
| new file mode 100644 | 
| index 0000000000000000000000000000000000000000..e9892e6bd4e749f79ac3753a8582034d30ddea57 | 
| --- /dev/null | 
| +++ b/content/renderer/media/media_recorder_handler.cc | 
| @@ -0,0 +1,119 @@ | 
| +// Copyright 2015 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 "content/renderer/media/media_recorder_handler.h" | 
| + | 
| +#include "base/bind.h" | 
| +#include "base/location.h" | 
| +#include "base/logging.h" | 
| +#include "content/renderer/media/video_track_recorder.h" | 
| +#include "media/base/bind_to_current_loop.h" | 
| +#include "media/capture/webm_muxer.h" | 
| +#include "third_party/WebKit/public/platform/WebMediaRecorderHandlerClient.h" | 
| +// | 
| +namespace content { | 
| + | 
| +MediaRecorderHandler::MediaRecorderHandler() | 
| +    : recording_(false), client_(nullptr), weak_factory_(this) { | 
| +  DVLOG(3) << __FUNCTION__; | 
| +} | 
| + | 
| +MediaRecorderHandler::~MediaRecorderHandler() { | 
| +  DCHECK(main_render_thread_checker_.CalledOnValidThread()); | 
| +  // Send a |last_in_slice| to our |client_|. | 
| +  if (client_) | 
| +    client_->writeData(nullptr, 0u, true); | 
| +} | 
| + | 
| +bool MediaRecorderHandler::canSupportMimeType( | 
| +    const blink::WebString& mimeType) { | 
| +  DCHECK(main_render_thread_checker_.CalledOnValidThread()); | 
| +  NOTIMPLEMENTED(); | 
| +  return false; | 
| +} | 
| + | 
| +bool MediaRecorderHandler::initialize( | 
| +    blink::WebMediaRecorderHandlerClient* client, | 
| +    const blink::WebMediaStream& media_stream, | 
| +    const blink::WebString& mimeType) { | 
| +  DCHECK(main_render_thread_checker_.CalledOnValidThread()); | 
| +  // TODO(mcasas): check canSupportMimeType(mimeType) is true. | 
| +  media_stream_ = media_stream; | 
| +  DCHECK(client); | 
| +  client_ = client; | 
| + | 
| +  return true; | 
| +} | 
| + | 
| +bool MediaRecorderHandler::start() { | 
| +  DCHECK(main_render_thread_checker_.CalledOnValidThread()); | 
| +  DCHECK(!recording_); | 
| +  return start(0); | 
| +} | 
| + | 
| +bool MediaRecorderHandler::start(int timeslice) { | 
| +  DCHECK(main_render_thread_checker_.CalledOnValidThread()); | 
| +  DCHECK(!recording_); | 
| +  DCHECK(!media_stream_.isNull()); | 
| + | 
| +  webm_muxer_.reset(new media::WebmMuxer(media::BindToCurrentLoop(base::Bind( | 
| +      &MediaRecorderHandler::WriteData, weak_factory_.GetWeakPtr())))); | 
| +  DCHECK(webm_muxer_); | 
| + | 
| +  blink::WebVector<blink::WebMediaStreamTrack> video_tracks; | 
| +  media_stream_.videoTracks(video_tracks); | 
| +  DCHECK(!video_tracks.isEmpty()); | 
| + | 
| +  for (const auto& video_track : video_tracks) { | 
| +    DCHECK(!video_track.isNull()); | 
| + | 
| +    const VideoTrackRecorder::OnEncodedVideoCB on_encoded_video_cb = base::Bind( | 
| +        &media::WebmMuxer::OnEncodedVideo, base::Unretained(webm_muxer_.get())); | 
| + | 
| +    video_recorders_.push_back(new VideoTrackRecorder(video_track, | 
| +                                                      on_encoded_video_cb)); | 
| +  } | 
| + | 
| +  // TODO(mcasas): Add audio_tracks. | 
| + | 
| +  recording_ = true; | 
| +  return true; | 
| +} | 
| + | 
| +void MediaRecorderHandler::stop() { | 
| +  DCHECK(main_render_thread_checker_.CalledOnValidThread()); | 
| +  DCHECK(recording_); | 
| + | 
| +  recording_ = false; | 
| +  video_recorders_.clear(); | 
| +  webm_muxer_.reset(NULL); | 
| +} | 
| + | 
| +void MediaRecorderHandler::pause() { | 
| +  DCHECK(main_render_thread_checker_.CalledOnValidThread()); | 
| +  DCHECK(recording_); | 
| +  recording_ = false; | 
| +  NOTIMPLEMENTED(); | 
| +} | 
| + | 
| +void MediaRecorderHandler::resume() { | 
| +  DCHECK(main_render_thread_checker_.CalledOnValidThread()); | 
| +  DCHECK(!recording_); | 
| +  recording_ = true; | 
| +  NOTIMPLEMENTED(); | 
| +} | 
| + | 
| +void MediaRecorderHandler::WriteData(const base::StringPiece& data) { | 
| +  DCHECK(main_render_thread_checker_.CalledOnValidThread()); | 
| +  client_->writeData(data.data(), data.size(), false  /* lastInslice */); | 
| +} | 
| + | 
| +void MediaRecorderHandler::OnVideoFrameForTesting( | 
| +    const scoped_refptr<media::VideoFrame>& frame, | 
| +    const base::TimeTicks& timestamp) { | 
| +  for (auto* recorder : video_recorders_) | 
| +    recorder->OnVideoFrameForTesting(frame, timestamp); | 
| +} | 
| + | 
| +}  // namespace content | 
|  |