blob: 2f5aa251fdaf7ff81c5b1add5d82e99a85a4999c [file] [log] [blame]
// 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/test/test_renderer_host.h"
#include "content/browser/renderer_host/render_view_host_factory.h"
#include "content/browser/renderer_host/test_render_view_host.h"
#include "content/browser/site_instance_impl.h"
#include "content/browser/web_contents/navigation_entry_impl.h"
#include "content/browser/web_contents/test_web_contents.h"
#include "content/public/browser/web_contents.h"
#include "content/test/mock_render_process_host.h"
#include "content/test/test_browser_context.h"
#if defined(USE_ASH)
#include "ui/aura/test/test_screen.h"
#endif
#if defined(USE_AURA)
#include "ui/aura/env.h"
#include "ui/aura/monitor_manager.h"
#include "ui/aura/root_window.h"
#include "ui/aura/single_monitor_manager.h"
#include "ui/aura/test/test_stacking_client.h"
#endif
namespace content {
// Manages creation of the RenderViewHosts using our special subclass. This
// automatically registers itself when it goes in scope, and unregisters itself
// when it goes out of scope. Since you can't have more than one factory
// registered at a time, you can only have one of these objects at a time.
//
// This is an implementation detail of this file and used only via
// RenderViewHostTestEnabler.
class TestRenderViewHostFactory : public RenderViewHostFactory {
public:
explicit TestRenderViewHostFactory(
content::RenderProcessHostFactory* rph_factory);
virtual ~TestRenderViewHostFactory();
virtual void set_render_process_host_factory(
content::RenderProcessHostFactory* rph_factory);
virtual content::RenderViewHost* CreateRenderViewHost(
content::SiteInstance* instance,
content::RenderViewHostDelegate* delegate,
int routing_id,
content::SessionStorageNamespace* session_storage) OVERRIDE;
private:
// This is a bit of a hack. With the current design of the site instances /
// browsing instances, it's difficult to pass a RenderProcessHostFactory
// around properly.
//
// Instead, we set it right before we create a new RenderViewHost, which
// happens before the RenderProcessHost is created. This way, the instance
// has the correct factory and creates our special RenderProcessHosts.
content::RenderProcessHostFactory* render_process_host_factory_;
DISALLOW_COPY_AND_ASSIGN(TestRenderViewHostFactory);
};
TestRenderViewHostFactory::TestRenderViewHostFactory(
content::RenderProcessHostFactory* rph_factory)
: render_process_host_factory_(rph_factory) {
RenderViewHostFactory::RegisterFactory(this);
}
TestRenderViewHostFactory::~TestRenderViewHostFactory() {
RenderViewHostFactory::UnregisterFactory();
}
void TestRenderViewHostFactory::set_render_process_host_factory(
content::RenderProcessHostFactory* rph_factory) {
render_process_host_factory_ = rph_factory;
}
content::RenderViewHost* TestRenderViewHostFactory::CreateRenderViewHost(
SiteInstance* instance,
RenderViewHostDelegate* delegate,
int routing_id,
SessionStorageNamespace* session_storage) {
// See declaration of render_process_host_factory_ below.
static_cast<SiteInstanceImpl*>(instance)->
set_render_process_host_factory(render_process_host_factory_);
return new TestRenderViewHost(instance, delegate, routing_id);
}
// static
RenderViewHostTester* RenderViewHostTester::For(RenderViewHost* host) {
return static_cast<TestRenderViewHost*>(host);
}
// static
void RenderViewHostTester::EnableAccessibilityUpdatedNotifications(
RenderViewHost* host) {
static_cast<RenderViewHostImpl*>(
host)->set_send_accessibility_updated_notifications(true);
}
// static
RenderViewHost* RenderViewHostTester::GetPendingForController(
NavigationController* controller) {
WebContentsImpl* web_contents = static_cast<WebContentsImpl*>(
controller->GetWebContents());
return web_contents->GetRenderManagerForTesting()->pending_render_view_host();
}
// static
bool RenderViewHostTester::IsRenderViewHostSwappedOut(RenderViewHost* rvh) {
return static_cast<RenderViewHostImpl*>(rvh)->is_swapped_out();
}
// static
bool RenderViewHostTester::TestOnMessageReceived(RenderViewHost* rvh,
const IPC::Message& msg) {
return static_cast<RenderViewHostImpl*>(rvh)->OnMessageReceived(msg);
}
RenderViewHostTestEnabler::RenderViewHostTestEnabler()
: rph_factory_(new MockRenderProcessHostFactory()),
rvh_factory_(new TestRenderViewHostFactory(rph_factory_.get())) {
}
RenderViewHostTestEnabler::~RenderViewHostTestEnabler() {
}
RenderViewHostTestHarness::RenderViewHostTestHarness()
: contents_(NULL) {
}
RenderViewHostTestHarness::~RenderViewHostTestHarness() {
}
NavigationController& RenderViewHostTestHarness::controller() {
return web_contents()->GetController();
}
WebContents* RenderViewHostTestHarness::web_contents() {
return contents_.get();
}
RenderViewHost* RenderViewHostTestHarness::rvh() {
return web_contents()->GetRenderViewHost();
}
RenderViewHost* RenderViewHostTestHarness::pending_rvh() {
return static_cast<TestWebContents*>(web_contents())->
GetRenderManagerForTesting()->pending_render_view_host();
}
RenderViewHost* RenderViewHostTestHarness::active_rvh() {
return pending_rvh() ? pending_rvh() : rvh();
}
BrowserContext* RenderViewHostTestHarness::browser_context() {
return browser_context_.get();
}
MockRenderProcessHost* RenderViewHostTestHarness::process() {
return static_cast<MockRenderProcessHost*>(active_rvh()->GetProcess());
}
void RenderViewHostTestHarness::DeleteContents() {
SetContents(NULL);
}
void RenderViewHostTestHarness::SetContents(WebContents* contents) {
contents_.reset(contents);
}
WebContents* RenderViewHostTestHarness::CreateTestWebContents() {
// See comment above browser_context_ decl for why we check for NULL here.
if (!browser_context_.get())
browser_context_.reset(new TestBrowserContext());
// This will be deleted when the WebContentsImpl goes away.
SiteInstance* instance = SiteInstance::Create(browser_context_.get());
return new TestWebContents(browser_context_.get(), instance);
}
void RenderViewHostTestHarness::NavigateAndCommit(const GURL& url) {
static_cast<TestWebContents*>(web_contents())->NavigateAndCommit(url);
}
void RenderViewHostTestHarness::Reload() {
NavigationEntry* entry = controller().GetLastCommittedEntry();
DCHECK(entry);
controller().Reload(false);
static_cast<TestRenderViewHost*>(
rvh())->SendNavigate(entry->GetPageID(), entry->GetURL());
}
void RenderViewHostTestHarness::SetUp() {
#if defined(USE_AURA)
aura::Env::GetInstance()->SetMonitorManager(new aura::SingleMonitorManager);
root_window_.reset(aura::MonitorManager::CreateRootWindowForPrimaryMonitor());
#if defined(USE_ASH)
gfx::Screen::SetInstance(new aura::TestScreen(root_window_.get()));
#endif // USE_ASH
test_stacking_client_.reset(
new aura::test::TestStackingClient(root_window_.get()));
#endif // USE_AURA
SetContents(CreateTestWebContents());
}
void RenderViewHostTestHarness::TearDown() {
SetContents(NULL);
#if defined(USE_AURA)
test_stacking_client_.reset();
root_window_.reset();
#endif
// Make sure that we flush any messages related to WebContentsImpl destruction
// before we destroy the browser context.
MessageLoop::current()->RunAllPending();
// Release the browser context on the UI thread.
message_loop_.DeleteSoon(FROM_HERE, browser_context_.release());
message_loop_.RunAllPending();
}
void RenderViewHostTestHarness::SetRenderProcessHostFactory(
RenderProcessHostFactory* factory) {
rvh_test_enabler_.rvh_factory_->set_render_process_host_factory(factory);
}
} // namespace content