blob: dbedf4c2fb23f7fbb1f08f3a0e368681ba66a563 [file] [log] [blame]
// Copyright 2015 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 "headless/lib/browser/headless_browser_impl.h"
#include <vector>
#include "base/memory/ptr_util.h"
#include "base/threading/thread_task_runner_handle.h"
#include "content/public/app/content_main.h"
#include "content/public/browser/browser_thread.h"
#include "content/public/browser/web_contents.h"
#include "headless/lib/browser/headless_browser_context_impl.h"
#include "headless/lib/browser/headless_browser_main_parts.h"
#include "headless/lib/browser/headless_web_contents_impl.h"
#include "headless/lib/browser/headless_window_tree_client.h"
#include "headless/lib/headless_content_main_delegate.h"
#include "ui/aura/env.h"
#include "ui/aura/window_tree_host.h"
#include "ui/gfx/geometry/size.h"
namespace headless {
HeadlessBrowserImpl::HeadlessBrowserImpl(
const base::Callback<void(HeadlessBrowser*)>& on_start_callback,
HeadlessBrowser::Options options)
: on_start_callback_(on_start_callback),
options_(std::move(options)),
browser_main_parts_(nullptr) {
DCHECK(!on_start_callback_.is_null());
}
HeadlessBrowserImpl::~HeadlessBrowserImpl() {}
HeadlessWebContents::Builder HeadlessBrowserImpl::CreateWebContentsBuilder() {
DCHECK(BrowserMainThread()->BelongsToCurrentThread());
return HeadlessWebContents::Builder(this);
}
HeadlessBrowserContext::Builder
HeadlessBrowserImpl::CreateBrowserContextBuilder() {
DCHECK(BrowserMainThread()->BelongsToCurrentThread());
return HeadlessBrowserContext::Builder(this);
}
HeadlessWebContents* HeadlessBrowserImpl::CreateWebContents(
HeadlessWebContents::Builder* builder) {
DCHECK(BrowserMainThread()->BelongsToCurrentThread());
std::unique_ptr<HeadlessWebContentsImpl> headless_web_contents =
HeadlessWebContentsImpl::Create(builder, window_tree_host_->window(),
this);
if (!headless_web_contents)
return nullptr;
return RegisterWebContents(std::move(headless_web_contents));
}
HeadlessWebContents* HeadlessBrowserImpl::CreateWebContents(
const GURL& initial_url,
const gfx::Size& size) {
return CreateWebContentsBuilder()
.SetInitialURL(initial_url)
.SetWindowSize(size)
.Build();
}
scoped_refptr<base::SingleThreadTaskRunner>
HeadlessBrowserImpl::BrowserMainThread() const {
return content::BrowserThread::GetMessageLoopProxyForThread(
content::BrowserThread::UI);
}
scoped_refptr<base::SingleThreadTaskRunner>
HeadlessBrowserImpl::BrowserFileThread() const {
return content::BrowserThread::GetMessageLoopProxyForThread(
content::BrowserThread::FILE);
}
void HeadlessBrowserImpl::Shutdown() {
DCHECK(BrowserMainThread()->BelongsToCurrentThread());
BrowserMainThread()->PostTask(FROM_HERE,
base::MessageLoop::QuitWhenIdleClosure());
}
std::vector<HeadlessWebContents*> HeadlessBrowserImpl::GetAllWebContents() {
std::vector<HeadlessWebContents*> result;
result.reserve(web_contents_.size());
for (const auto& web_contents_pair : web_contents_) {
result.push_back(web_contents_pair.first);
}
return result;
}
HeadlessBrowserMainParts* HeadlessBrowserImpl::browser_main_parts() const {
DCHECK(BrowserMainThread()->BelongsToCurrentThread());
return browser_main_parts_;
}
void HeadlessBrowserImpl::set_browser_main_parts(
HeadlessBrowserMainParts* browser_main_parts) {
DCHECK(!browser_main_parts_);
browser_main_parts_ = browser_main_parts;
}
void HeadlessBrowserImpl::RunOnStartCallback() {
DCHECK(aura::Env::GetInstance());
// TODO(eseckler): allow configuration of window (viewport) size by embedder.
const gfx::Size kDefaultSize(800, 600);
window_tree_host_.reset(
aura::WindowTreeHost::Create(gfx::Rect(kDefaultSize)));
window_tree_host_->InitHost();
window_tree_client_.reset(
new HeadlessWindowTreeClient(window_tree_host_->window()));
on_start_callback_.Run(this);
on_start_callback_ = base::Callback<void(HeadlessBrowser*)>();
}
HeadlessWebContentsImpl* HeadlessBrowserImpl::RegisterWebContents(
std::unique_ptr<HeadlessWebContentsImpl> web_contents) {
DCHECK(web_contents);
HeadlessWebContentsImpl* unowned_web_contents = web_contents.get();
web_contents_[unowned_web_contents] = std::move(web_contents);
return unowned_web_contents;
}
void HeadlessBrowserImpl::DestroyWebContents(
HeadlessWebContentsImpl* web_contents) {
auto it = web_contents_.find(web_contents);
DCHECK(it != web_contents_.end());
web_contents_.erase(it);
}
void HeadlessBrowserImpl::SetOptionsForTesting(
HeadlessBrowser::Options options) {
options_ = std::move(options);
browser_main_parts()->default_browser_context()->SetOptionsForTesting(
&options_);
}
int HeadlessBrowserMain(
HeadlessBrowser::Options options,
const base::Callback<void(HeadlessBrowser*)>& on_browser_start_callback) {
content::ContentMainParams params(nullptr);
params.argc = options.argc;
params.argv = options.argv;
// TODO(skyostil): Implement custom message pumps.
DCHECK(!options.message_pump);
std::unique_ptr<HeadlessBrowserImpl> browser(
new HeadlessBrowserImpl(on_browser_start_callback, std::move(options)));
headless::HeadlessContentMainDelegate delegate(std::move(browser));
params.delegate = &delegate;
return content::ContentMain(params);
}
} // namespace headless