// Copyright 2014 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 "components/pdf/renderer/pepper_pdf_host.h"
#include "components/pdf/common/pdf_messages.h"
#include "components/pdf/renderer/pdf_resource_util.h"
#include "components/pdf/renderer/ppb_pdf_impl.h"
#include "content/public/common/referrer.h"
#include "content/public/renderer/pepper_plugin_instance.h"
#include "content/public/renderer/render_thread.h"
#include "content/public/renderer/render_view.h"
#include "content/public/renderer/renderer_ppapi_host.h"
#include "ppapi/host/dispatch_host_message.h"
#include "ppapi/host/host_message_context.h"
#include "ppapi/host/ppapi_host.h"
#include "ppapi/proxy/host_dispatcher.h"
#include "ppapi/proxy/ppapi_messages.h"
#include "ppapi/proxy/ppb_image_data_proxy.h"
#include "ppapi/shared_impl/ppb_image_data_shared.h"
#include "ppapi/shared_impl/scoped_pp_resource.h"
#include "ppapi/thunk/enter.h"
#include "ppapi/thunk/ppb_image_data_api.h"
#include "skia/ext/platform_canvas.h"
#include "third_party/WebKit/public/web/WebDocument.h"
#include "third_party/WebKit/public/web/WebElement.h"
#include "third_party/WebKit/public/web/WebLocalFrame.h"
#include "third_party/WebKit/public/web/WebPluginContainer.h"
#include "third_party/WebKit/public/web/WebView.h"
#include "third_party/skia/include/core/SkBitmap.h"
#include "ui/base/layout.h"
#include "ui/gfx/image/image_skia.h"
#include "ui/gfx/image/image_skia_rep.h"
#include "ui/gfx/point.h"
namespace pdf {
PepperPDFHost::PepperPDFHost(content::RendererPpapiHost* host,
PP_Instance instance,
PP_Resource resource)
: ppapi::host::ResourceHost(host->GetPpapiHost(), instance, resource),
host_(host) {}
PepperPDFHost::~PepperPDFHost() {}
int32_t PepperPDFHost::OnResourceMessageReceived(
const IPC::Message& msg,
ppapi::host::HostMessageContext* context) {
PPAPI_BEGIN_MESSAGE_MAP(PepperPDFHost, msg)
PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_PDF_GetLocalizedString,
OnHostMsgGetLocalizedString)
PPAPI_DISPATCH_HOST_RESOURCE_CALL_0(PpapiHostMsg_PDF_DidStartLoading,
OnHostMsgDidStartLoading)
PPAPI_DISPATCH_HOST_RESOURCE_CALL_0(PpapiHostMsg_PDF_DidStopLoading,
OnHostMsgDidStopLoading)
PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_PDF_UserMetricsRecordAction,
OnHostMsgUserMetricsRecordAction)
PPAPI_DISPATCH_HOST_RESOURCE_CALL_0(PpapiHostMsg_PDF_HasUnsupportedFeature,
OnHostMsgHasUnsupportedFeature)
PPAPI_DISPATCH_HOST_RESOURCE_CALL_0(PpapiHostMsg_PDF_Print, OnHostMsgPrint)
PPAPI_DISPATCH_HOST_RESOURCE_CALL_0(PpapiHostMsg_PDF_SaveAs,
OnHostMsgSaveAs)
PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_PDF_GetResourceImage,
OnHostMsgGetResourceImage)
PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_PDF_SetSelectedText,
OnHostMsgSetSelectedText)
PPAPI_DISPATCH_HOST_RESOURCE_CALL(PpapiHostMsg_PDF_SetLinkUnderCursor,
OnHostMsgSetLinkUnderCursor)
PPAPI_END_MESSAGE_MAP()
return PP_ERROR_FAILED;
}
int32_t PepperPDFHost::OnHostMsgGetLocalizedString(
ppapi::host::HostMessageContext* context,
PP_ResourceString string_id) {
std::string rv = GetStringResource(string_id);
context->reply_msg = PpapiPluginMsg_PDF_GetLocalizedStringReply(rv);
return PP_OK;
}
int32_t PepperPDFHost::OnHostMsgDidStartLoading(
ppapi::host::HostMessageContext* context) {
content::PepperPluginInstance* instance =
host_->GetPluginInstance(pp_instance());
if (!instance)
return PP_ERROR_FAILED;
instance->GetRenderView()->DidStartLoading();
return PP_OK;
}
int32_t PepperPDFHost::OnHostMsgDidStopLoading(
ppapi::host::HostMessageContext* context) {
content::PepperPluginInstance* instance =
host_->GetPluginInstance(pp_instance());
if (!instance)
return PP_ERROR_FAILED;
instance->GetRenderView()->DidStopLoading();
return PP_OK;
}
int32_t PepperPDFHost::OnHostMsgSetContentRestriction(
ppapi::host::HostMessageContext* context,
int restrictions) {
content::PepperPluginInstance* instance =
host_->GetPluginInstance(pp_instance());
if (!instance)
return PP_ERROR_FAILED;
instance->GetRenderView()->Send(new PDFHostMsg_PDFUpdateContentRestrictions(
instance->GetRenderView()->GetRoutingID(), restrictions));
return PP_OK;
}
int32_t PepperPDFHost::OnHostMsgUserMetricsRecordAction(
ppapi::host::HostMessageContext* context,
const std::string& action) {
if (action.empty())
return PP_ERROR_FAILED;
content::RenderThread::Get()->RecordComputedAction(action);
return PP_OK;
}
int32_t PepperPDFHost::OnHostMsgHasUnsupportedFeature(
ppapi::host::HostMessageContext* context) {
content::PepperPluginInstance* instance =
host_->GetPluginInstance(pp_instance());
if (!instance)
return PP_ERROR_FAILED;
blink::WebView* view =
instance->GetContainer()->element().document().frame()->view();
content::RenderView* render_view = content::RenderView::FromWebView(view);
render_view->Send(
new PDFHostMsg_PDFHasUnsupportedFeature(render_view->GetRoutingID()));
return PP_OK;
}
int32_t PepperPDFHost::OnHostMsgPrint(
ppapi::host::HostMessageContext* context) {
return PPB_PDF_Impl::InvokePrintingForInstance(pp_instance()) ? PP_OK :
PP_ERROR_FAILED;
}
int32_t PepperPDFHost::OnHostMsgSaveAs(
ppapi::host::HostMessageContext* context) {
content::PepperPluginInstance* instance =
host_->GetPluginInstance(pp_instance());
if (!instance)
return PP_ERROR_FAILED;
GURL url = instance->GetPluginURL();
content::RenderView* render_view = instance->GetRenderView();
blink::WebLocalFrame* frame =
render_view->GetWebView()->mainFrame()->toWebLocalFrame();
content::Referrer referrer(frame->document().url(),
frame->document().referrerPolicy());
render_view->Send(
new PDFHostMsg_PDFSaveURLAs(render_view->GetRoutingID(), url, referrer));
return PP_OK;
}
int32_t PepperPDFHost::OnHostMsgGetResourceImage(
ppapi::host::HostMessageContext* context,
PP_ResourceImage image_id,
float scale) {
gfx::ImageSkia* res_image_skia = GetImageResource(image_id);
if (!res_image_skia)
return PP_ERROR_FAILED;
gfx::ImageSkiaRep image_skia_rep = res_image_skia->GetRepresentation(scale);
if (image_skia_rep.is_null() || image_skia_rep.scale() != scale)
return PP_ERROR_FAILED;
PP_Size pp_size;
pp_size.width = image_skia_rep.pixel_width();
pp_size.height = image_skia_rep.pixel_height();
ppapi::HostResource host_resource;
PP_ImageDataDesc image_data_desc;
IPC::PlatformFileForTransit image_handle;
uint32_t byte_count = 0;
bool success =
CreateImageData(pp_instance(),
ppapi::PPB_ImageData_Shared::GetNativeImageDataFormat(),
pp_size,
image_skia_rep.sk_bitmap(),
&host_resource,
&image_data_desc,
&image_handle,
&byte_count);
ppapi::ScopedPPResource image_data_resource(
ppapi::ScopedPPResource::PassRef(), host_resource.host_resource());
if (!success)
return PP_ERROR_FAILED;
ppapi::host::ReplyMessageContext reply_context =
context->MakeReplyMessageContext();
ppapi::proxy::SerializedHandle serialized_handle;
serialized_handle.set_shmem(image_handle, byte_count);
reply_context.params.AppendHandle(serialized_handle);
SendReply(
reply_context,
PpapiPluginMsg_PDF_GetResourceImageReply(host_resource, image_data_desc));
// Keep a reference to the resource only if the function succeeds.
image_data_resource.Release();
return PP_OK_COMPLETIONPENDING;
}
int32_t PepperPDFHost::OnHostMsgSetSelectedText(
ppapi::host::HostMessageContext* context,
const base::string16& selected_text) {
content::PepperPluginInstance* instance =
host_->GetPluginInstance(pp_instance());
if (!instance)
return PP_ERROR_FAILED;
instance->SetSelectedText(selected_text);
return PP_OK;
}
int32_t PepperPDFHost::OnHostMsgSetLinkUnderCursor(
ppapi::host::HostMessageContext* context,
const std::string& url) {
content::PepperPluginInstance* instance =
host_->GetPluginInstance(pp_instance());
if (!instance)
return PP_ERROR_FAILED;
instance->SetLinkUnderCursor(url);
return PP_OK;
}
// TODO(raymes): This function is mainly copied from ppb_image_data_proxy.cc.
// It's a mess and needs to be fixed in several ways but this is better done
// when we refactor PPB_ImageData. On success, the image handle will be
// non-null.
bool PepperPDFHost::CreateImageData(
PP_Instance instance,
PP_ImageDataFormat format,
const PP_Size& size,
const SkBitmap& pixels_to_write,
ppapi::HostResource* result,
PP_ImageDataDesc* out_image_data_desc,
IPC::PlatformFileForTransit* out_image_handle,
uint32_t* out_byte_count) {
PP_Resource resource = ppapi::proxy::PPB_ImageData_Proxy::CreateImageData(
instance,
ppapi::PPB_ImageData_Shared::SIMPLE,
format,
size,
false /* init_to_zero */,
out_image_data_desc,
out_image_handle,
out_byte_count);
if (!resource)
return false;
result->SetHostResource(instance, resource);
// Write the image to the resource shared memory.
ppapi::thunk::EnterResourceNoLock<ppapi::thunk::PPB_ImageData_API>
enter_resource(resource, false);
if (enter_resource.failed())
return false;
ppapi::thunk::PPB_ImageData_API* image_data =
static_cast<ppapi::thunk::PPB_ImageData_API*>(enter_resource.object());
SkCanvas* canvas = image_data->GetCanvas();
bool needs_unmapping = false;
if (!canvas) {
needs_unmapping = true;
image_data->Map();
canvas = image_data->GetCanvas();
if (!canvas)
return false; // Failure mapping.
}
const SkBitmap* bitmap = &skia::GetTopDevice(*canvas)->accessBitmap(false);
pixels_to_write.copyPixelsTo(
bitmap->getPixels(), bitmap->getSize(), bitmap->rowBytes());
if (needs_unmapping)
image_data->Unmap();
return true;
}
} // namespace pdf