// Copyright 2013 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 "chrome/browser/guestview/guestview.h"
#include "base/lazy_instance.h"
#include "chrome/browser/guestview/adview/adview_guest.h"
#include "chrome/browser/guestview/guestview_constants.h"
#include "chrome/browser/guestview/webview/webview_guest.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/common/content_settings.h"
#include "content/public/browser/render_process_host.h"
#include "content/public/browser/web_contents.h"
#include "content/public/common/url_constants.h"
#include "extensions/browser/event_router.h"
#include "net/base/escape.h"
using content::WebContents;
namespace {
// <embedder_process_id, guest_instance_id> => GuestView*
typedef std::map<std::pair<int, int>, GuestView*> EmbedderGuestViewMap;
static base::LazyInstance<EmbedderGuestViewMap> embedder_guestview_map =
LAZY_INSTANCE_INITIALIZER;
typedef std::map<WebContents*, GuestView*> WebContentsGuestViewMap;
static base::LazyInstance<WebContentsGuestViewMap> webcontents_guestview_map =
LAZY_INSTANCE_INITIALIZER;
} // namespace
GuestView::Event::Event(const std::string& name,
scoped_ptr<DictionaryValue> args)
: name_(name),
args_(args.Pass()) {
}
GuestView::Event::~Event() {
}
scoped_ptr<DictionaryValue> GuestView::Event::GetArguments() {
return args_.Pass();
}
GuestView::GuestView(WebContents* guest_web_contents,
const std::string& extension_id)
: guest_web_contents_(guest_web_contents),
embedder_web_contents_(NULL),
extension_id_(extension_id),
embedder_render_process_id_(0),
browser_context_(guest_web_contents->GetBrowserContext()),
guest_instance_id_(guest_web_contents->GetEmbeddedInstanceID()),
view_instance_id_(guestview::kInstanceIDNone) {
webcontents_guestview_map.Get().insert(
std::make_pair(guest_web_contents, this));
}
// static
GuestView::Type GuestView::GetViewTypeFromString(const std::string& api_type) {
if (api_type == "adview") {
return GuestView::ADVIEW;
} else if (api_type == "webview") {
return GuestView::WEBVIEW;
}
return GuestView::UNKNOWN;
}
// static
GuestView* GuestView::Create(WebContents* guest_web_contents,
const std::string& extension_id,
GuestView::Type view_type) {
switch (view_type) {
case GuestView::WEBVIEW:
return new WebViewGuest(guest_web_contents, extension_id);
case GuestView::ADVIEW:
return new AdViewGuest(guest_web_contents, extension_id);
default:
NOTREACHED();
return NULL;
}
}
// static
GuestView* GuestView::FromWebContents(WebContents* web_contents) {
WebContentsGuestViewMap* guest_map = webcontents_guestview_map.Pointer();
WebContentsGuestViewMap::iterator it = guest_map->find(web_contents);
return it == guest_map->end() ? NULL : it->second;
}
// static
GuestView* GuestView::From(int embedder_process_id, int guest_instance_id) {
EmbedderGuestViewMap* guest_map = embedder_guestview_map.Pointer();
EmbedderGuestViewMap::iterator it = guest_map->find(
std::make_pair(embedder_process_id, guest_instance_id));
return it == guest_map->end() ? NULL : it->second;
}
// static
bool GuestView::GetGuestPartitionConfigForSite(const GURL& site,
std::string* partition_domain,
std::string* partition_name,
bool* in_memory) {
if (!site.SchemeIs(content::kGuestScheme))
return false;
// Since guest URLs are only used for packaged apps, there must be an app
// id in the URL.
CHECK(site.has_host());
*partition_domain = site.host();
// Since persistence is optional, the path must either be empty or the
// literal string.
*in_memory = (site.path() != "/persist");
// The partition name is user supplied value, which we have encoded when the
// URL was created, so it needs to be decoded.
*partition_name = net::UnescapeURLComponent(site.query(),
net::UnescapeRule::NORMAL);
return true;
}
// static
void GuestView::GetDefaultContentSettingRules(
RendererContentSettingRules* rules, bool incognito) {
rules->image_rules.push_back(ContentSettingPatternSource(
ContentSettingsPattern::Wildcard(),
ContentSettingsPattern::Wildcard(),
CONTENT_SETTING_ALLOW,
std::string(),
incognito));
rules->script_rules.push_back(ContentSettingPatternSource(
ContentSettingsPattern::Wildcard(),
ContentSettingsPattern::Wildcard(),
CONTENT_SETTING_ALLOW,
std::string(),
incognito));
}
void GuestView::Attach(content::WebContents* embedder_web_contents,
const base::DictionaryValue& args) {
embedder_web_contents_ = embedder_web_contents;
embedder_render_process_id_ =
embedder_web_contents->GetRenderProcessHost()->GetID();
args.GetInteger(guestview::kParameterInstanceId, &view_instance_id_);
std::pair<int, int> key(embedder_render_process_id_, guest_instance_id_);
embedder_guestview_map.Get().insert(std::make_pair(key, this));
// GuestView::Attach is called prior to initialization (and initial
// navigation) of the guest in the content layer in order to permit mapping
// the necessary associations between the <*view> element and its guest. This
// is needed by the <webview> WebRequest API to allow intercepting resource
// requests during navigation. However, queued events should be fired after
// content layer initialization in order to ensure that load events (such as
// 'loadstop') fire in embedder after the contentWindow is available.
base::MessageLoop::current()->PostTask(
FROM_HERE,
base::Bind(&GuestView::SendQueuedEvents,
base::Unretained(this)));
}
GuestView::Type GuestView::GetViewType() const {
return GuestView::UNKNOWN;
}
WebViewGuest* GuestView::AsWebView() {
return NULL;
}
AdViewGuest* GuestView::AsAdView() {
return NULL;
}
GuestView::~GuestView() {
std::pair<int, int> key(embedder_render_process_id_, guest_instance_id_);
embedder_guestview_map.Get().erase(key);
webcontents_guestview_map.Get().erase(guest_web_contents());
while (!pending_events_.empty()) {
delete pending_events_.front();
pending_events_.pop();
}
}
void GuestView::DispatchEvent(Event* event) {
if (!attached()) {
pending_events_.push(event);
return;
}
Profile* profile = Profile::FromBrowserContext(browser_context_);
extensions::EventFilteringInfo info;
info.SetURL(GURL());
info.SetInstanceID(guest_instance_id_);
scoped_ptr<ListValue> args(new ListValue());
args->Append(event->GetArguments().release());
extensions::EventRouter::DispatchEvent(
embedder_web_contents_, profile, extension_id_,
event->name(), args.Pass(),
extensions::EventRouter::USER_GESTURE_UNKNOWN, info);
delete event;
}
void GuestView::SendQueuedEvents() {
if (!attached())
return;
while (!pending_events_.empty()) {
Event* event = pending_events_.front();
pending_events_.pop();
DispatchEvent(event);
}
}