1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107
|
// Copyright 2015 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "third_party/blink/renderer/modules/mediacapturefromelement/html_canvas_element_capture.h"
#include <memory>
#include "media/base/video_frame.h"
#include "third_party/blink/public/platform/platform.h"
#include "third_party/blink/renderer/bindings/core/v8/v8_binding_for_core.h"
#include "third_party/blink/renderer/core/execution_context/execution_context.h"
#include "third_party/blink/renderer/core/frame/local_frame.h"
#include "third_party/blink/renderer/core/html/canvas/html_canvas_element.h"
#include "third_party/blink/renderer/modules/mediacapturefromelement/canvas_capture_handler.h"
#include "third_party/blink/renderer/modules/mediacapturefromelement/canvas_capture_media_stream_track.h"
#include "third_party/blink/renderer/modules/mediastream/media_stream.h"
#include "third_party/blink/renderer/platform/mediastream/media_stream_component.h"
#include "ui/gfx/geometry/size.h"
namespace {
const double kDefaultFrameRate = 60.0;
} // anonymous namespace
namespace blink {
MediaStream* HTMLCanvasElementCapture::captureStream(
ScriptState* script_state,
HTMLCanvasElement& element,
ExceptionState& exception_state) {
return HTMLCanvasElementCapture::captureStream(script_state, element, false,
0, exception_state);
}
MediaStream* HTMLCanvasElementCapture::captureStream(
ScriptState* script_state,
HTMLCanvasElement& element,
double frame_rate,
ExceptionState& exception_state) {
if (frame_rate < 0.0) {
exception_state.ThrowDOMException(DOMExceptionCode::kNotSupportedError,
"Given frame rate is not supported.");
return nullptr;
}
return HTMLCanvasElementCapture::captureStream(script_state, element, true,
frame_rate, exception_state);
}
MediaStream* HTMLCanvasElementCapture::captureStream(
ScriptState* script_state,
HTMLCanvasElement& element,
bool given_frame_rate,
double frame_rate,
ExceptionState& exception_state) {
if (!element.OriginClean()) {
exception_state.ThrowSecurityError("Canvas is not origin-clean.");
return nullptr;
}
ExecutionContext* context = ExecutionContext::From(script_state);
DCHECK(context);
LocalFrame* frame = ToLocalFrameIfNotDetached(script_state->GetContext());
MediaStreamComponent* component = nullptr;
const gfx::Size size(element.width(), element.height());
if (!media::VideoFrame::IsValidSize(size, gfx::Rect(size), size)) {
exception_state.ThrowDOMException(DOMExceptionCode::kNotSupportedError,
"Current canvas size is not supported by "
"CanvasCaptureMediaStreamTrack.");
return nullptr;
}
std::unique_ptr<CanvasCaptureHandler> handler;
if (given_frame_rate) {
handler = CanvasCaptureHandler::CreateCanvasCaptureHandler(
frame, size, frame_rate,
context->GetTaskRunner(TaskType::kInternalMediaRealTime),
Platform::Current()->GetIOTaskRunner(), &component);
} else {
handler = CanvasCaptureHandler::CreateCanvasCaptureHandler(
frame, size, kDefaultFrameRate,
context->GetTaskRunner(TaskType::kInternalMediaRealTime),
Platform::Current()->GetIOTaskRunner(), &component);
}
if (!handler) {
exception_state.ThrowDOMException(
DOMExceptionCode::kNotSupportedError,
"No CanvasCapture handler can be created.");
return nullptr;
}
CanvasCaptureMediaStreamTrack* canvas_track;
if (given_frame_rate) {
canvas_track = MakeGarbageCollected<CanvasCaptureMediaStreamTrack>(
component, &element, context, std::move(handler), frame_rate);
} else {
canvas_track = MakeGarbageCollected<CanvasCaptureMediaStreamTrack>(
component, &element, context, std::move(handler));
}
// We want to capture a frame in the beginning.
canvas_track->requestFrame();
MediaStreamTrackVector tracks;
tracks.push_back(canvas_track);
return MediaStream::Create(context, tracks);
}
} // namespace blink
|