// Copyright (c) 2012 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/browser/intents/intent_injector.h" #include "base/bind.h" #include "base/command_line.h" #include "base/file_path.h" #include "base/logging.h" #include "base/string16.h" #include "content/browser/child_process_security_policy_impl.h" #include "content/browser/renderer_host/render_view_host_impl.h" #include "content/browser/web_contents/web_contents_impl.h" #include "content/common/intents_messages.h" #include "content/public/browser/web_intents_dispatcher.h" #include "content/public/common/content_switches.h" #include "webkit/glue/web_intent_data.h" #include "webkit/glue/web_intent_reply_data.h" using content::RenderViewHost; using content::WebContents; IntentInjector::IntentInjector(WebContents* web_contents) : content::WebContentsObserver(web_contents), intents_dispatcher_(NULL) { DCHECK(web_contents); } IntentInjector::~IntentInjector() { } void IntentInjector::WebContentsDestroyed(content::WebContents* contents) { if (intents_dispatcher_) { intents_dispatcher_->SendReplyMessage( webkit_glue::WEB_INTENT_SERVICE_CONTENTS_CLOSED, string16()); } delete this; } void IntentInjector::SourceWebContentsDestroyed(WebContents* contents) { intents_dispatcher_ = NULL; } void IntentInjector::SetIntent( content::WebIntentsDispatcher* intents_dispatcher, const webkit_glue::WebIntentData& intent) { intents_dispatcher_ = intents_dispatcher; intents_dispatcher_->RegisterReplyNotification( base::Bind(&IntentInjector::OnSendReturnMessage, base::Unretained(this))); source_intent_.reset(new webkit_glue::WebIntentData(intent)); initial_url_ = web_contents()->GetPendingSiteInstance()->GetSite(); } void IntentInjector::OnSendReturnMessage( webkit_glue::WebIntentReplyType reply_type) { intents_dispatcher_ = NULL; } void IntentInjector::RenderViewCreated(RenderViewHost* render_view_host) { if (source_intent_.get() == NULL || CommandLine::ForCurrentProcess()->HasSwitch( switches::kDisableWebIntents) || web_contents()->GetRenderViewHost() == NULL) { return; } // Only deliver the intent to the renderer if it has the same origin // as the initial delivery target. if (initial_url_.GetOrigin() != render_view_host->GetSiteInstance()->GetSite().GetOrigin()) { return; } if (source_intent_->data_type == webkit_glue::WebIntentData::BLOB) { // Grant read permission on the blob file to the delivered context. int child_id = render_view_host->GetProcess()->GetID(); ChildProcessSecurityPolicyImpl::GetInstance()->GrantReadFile( child_id, source_intent_->blob_file); } render_view_host->Send(new IntentsMsg_SetWebIntentData( render_view_host->GetRoutingID(), *(source_intent_.get()))); } bool IntentInjector::OnMessageReceived(const IPC::Message& message) { bool handled = true; IPC_BEGIN_MESSAGE_MAP(IntentInjector, message) IPC_MESSAGE_HANDLER(IntentsHostMsg_WebIntentReply, OnReply); IPC_MESSAGE_UNHANDLED(handled = false) IPC_END_MESSAGE_MAP() return handled; } void IntentInjector::OnReply(webkit_glue::WebIntentReplyType reply_type, const string16& data) { if (CommandLine::ForCurrentProcess()->HasSwitch(switches::kDisableWebIntents)) NOTREACHED(); if (intents_dispatcher_) { // Ensure we only call SendReplyMessage once. content::WebIntentsDispatcher* intents_dispatcher = intents_dispatcher_; intents_dispatcher_ = NULL; intents_dispatcher->SendReplyMessage(reply_type, data); } }