electron/atom/browser/atom_browser_main_parts.cc

239 lines
7.3 KiB
C++
Raw Normal View History

// Copyright (c) 2013 GitHub, Inc.
2014-04-25 09:49:37 +00:00
// Use of this source code is governed by the MIT license that can be
2013-04-12 01:46:58 +00:00
// found in the LICENSE file.
2014-03-16 00:30:26 +00:00
#include "atom/browser/atom_browser_main_parts.h"
2013-04-12 01:46:58 +00:00
2015-06-24 06:36:05 +00:00
#include "atom/browser/api/trackable_object.h"
2016-11-30 07:30:03 +00:00
#include "atom/browser/atom_access_token_store.h"
2014-03-16 00:30:26 +00:00
#include "atom/browser/atom_browser_client.h"
#include "atom/browser/atom_browser_context.h"
2017-01-18 17:53:54 +00:00
#include "atom/browser/atom_web_ui_controller_factory.h"
#include "atom/browser/bridge_task_runner.h"
2014-03-16 00:30:26 +00:00
#include "atom/browser/browser.h"
#include "atom/browser/javascript_environment.h"
#include "atom/browser/node_debugger.h"
#include "atom/common/api/atom_bindings.h"
2017-03-10 08:33:27 +00:00
#include "atom/common/asar/asar_util.h"
2014-03-16 00:30:26 +00:00
#include "atom/common/node_bindings.h"
#include "atom/common/node_includes.h"
#include "base/command_line.h"
2016-07-04 06:08:55 +00:00
#include "base/threading/thread_task_runner_handle.h"
#include "chrome/browser/browser_process.h"
#include "content/public/browser/child_process_security_policy.h"
2016-11-30 07:30:03 +00:00
#include "device/geolocation/geolocation_delegate.h"
#include "device/geolocation/geolocation_provider.h"
#include "v8/include/v8-debug.h"
2017-04-21 20:45:30 +00:00
#if defined(USE_AURA)
#include "ui/wm/core/wm_state.h"
#endif
#if defined(USE_X11)
2017-01-26 10:55:19 +00:00
#include "chrome/browser/ui/libgtkui/gtk_util.h"
#include "ui/events/devices/x11/touch_factory_x11.h"
#endif
2013-04-12 01:46:58 +00:00
namespace atom {
2016-11-30 07:30:03 +00:00
namespace {
// A provider of Geolocation services to override AccessTokenStore.
class AtomGeolocationDelegate : public device::GeolocationDelegate {
public:
AtomGeolocationDelegate() = default;
scoped_refptr<device::AccessTokenStore> CreateAccessTokenStore() final {
return new AtomAccessTokenStore();
}
private:
DISALLOW_COPY_AND_ASSIGN(AtomGeolocationDelegate);
};
template<typename T>
void Erase(T* container, typename T::iterator iter) {
container->erase(iter);
}
2016-11-30 07:30:03 +00:00
} // namespace
// static
AtomBrowserMainParts* AtomBrowserMainParts::self_ = nullptr;
2017-04-21 20:45:30 +00:00
#if defined(USE_AURA)
wm::WMState* wm_state_ = nullptr;
#endif
2013-04-13 10:39:09 +00:00
AtomBrowserMainParts::AtomBrowserMainParts()
: fake_browser_process_(new BrowserProcess),
exit_code_(nullptr),
browser_(new Browser),
2017-03-08 08:33:44 +00:00
node_bindings_(NodeBindings::Create(NodeBindings::BROWSER)),
atom_bindings_(new AtomBindings(uv_default_loop())),
2014-10-17 04:41:40 +00:00
gc_timer_(true, true) {
DCHECK(!self_) << "Cannot have two AtomBrowserMainParts";
self_ = this;
// Register extension scheme as web safe scheme.
content::ChildProcessSecurityPolicy::GetInstance()->
RegisterWebSafeScheme("chrome-extension");
2013-04-12 01:46:58 +00:00
}
AtomBrowserMainParts::~AtomBrowserMainParts() {
2017-03-10 08:33:27 +00:00
asar::ClearArchives();
// Leak the JavascriptEnvironment on exit.
// This is to work around the bug that V8 would be waiting for background
// tasks to finish on exit, while somehow it waits forever in Electron, more
2016-03-31 23:50:30 +00:00
// about this can be found at https://github.com/electron/electron/issues/4767.
// On the other handle there is actually no need to gracefully shutdown V8
// on exit in the main process, we already ensured all necessary resources get
// cleaned up, and it would make quitting faster.
ignore_result(js_env_.release());
2013-04-12 01:46:58 +00:00
}
// static
AtomBrowserMainParts* AtomBrowserMainParts::Get() {
DCHECK(self_);
return self_;
}
bool AtomBrowserMainParts::SetExitCode(int code) {
if (!exit_code_)
return false;
*exit_code_ = code;
return true;
}
2015-12-10 02:09:59 +00:00
int AtomBrowserMainParts::GetExitCode() {
return exit_code_ != nullptr ? *exit_code_ : 0;
}
base::Closure AtomBrowserMainParts::RegisterDestructionCallback(
2015-06-24 09:58:12 +00:00
const base::Closure& callback) {
auto iter = destructors_.insert(destructors_.end(), callback);
return base::Bind(&Erase<std::list<base::Closure>>, &destructors_, iter);
2015-06-24 09:58:12 +00:00
}
2015-10-04 11:21:36 +00:00
void AtomBrowserMainParts::PreEarlyInitialization() {
2015-10-04 12:08:19 +00:00
brightray::BrowserMainParts::PreEarlyInitialization();
2015-10-04 11:21:36 +00:00
#if defined(OS_POSIX)
HandleSIGCHLD();
#endif
}
2013-04-13 10:39:09 +00:00
void AtomBrowserMainParts::PostEarlyInitialization() {
brightray::BrowserMainParts::PostEarlyInitialization();
// Temporary set the bridge_task_runner_ as current thread's task runner,
// so we can fool gin::PerIsolateData to use it as its task runner, instead
// of getting current message loop's task runner, which is null for now.
bridge_task_runner_ = new BridgeTaskRunner;
base::ThreadTaskRunnerHandle handle(bridge_task_runner_);
// The ProxyResolverV8 has setup a complete V8 environment, in order to
// avoid conflicts we only initialize our V8 environment after that.
js_env_.reset(new JavascriptEnvironment);
2013-04-13 10:39:09 +00:00
node_bindings_->Initialize();
// Support the "--debug" switch.
node_debugger_.reset(new NodeDebugger(js_env_->isolate()));
// Create the global environment.
2016-03-27 10:21:12 +00:00
node::Environment* env =
node_bindings_->CreateEnvironment(js_env_->context());
node_env_.reset(new NodeEnvironment(env));
2013-12-17 13:55:56 +00:00
// Make sure node can get correct environment when debugging.
if (node_debugger_->IsRunning())
2016-11-30 07:30:03 +00:00
env->AssignToContext(v8::Debug::GetDebugContext(js_env_->isolate()));
2016-09-16 22:57:07 +00:00
// Add Electron extended APIs.
2016-03-27 10:21:12 +00:00
atom_bindings_->BindTo(js_env_->isolate(), env->process_object());
// Load everything.
2016-03-27 10:21:12 +00:00
node_bindings_->LoadEnvironment(env);
// Wrap the uv loop with global env.
node_bindings_->set_uv_env(env);
2013-04-13 10:39:09 +00:00
}
void AtomBrowserMainParts::PreMainMessageLoopRun() {
js_env_->OnMessageLoopCreated();
// Run user's main script before most things get initialized, so we can have
// a chance to setup everything.
node_bindings_->PrepareMessageLoop();
node_bindings_->RunMessageLoop();
2017-04-21 20:45:30 +00:00
#if defined(USE_AURA)
wm_state_ = new wm::WMState;
#endif
#if defined(USE_X11)
ui::TouchFactory::SetTouchDeviceListFromCommandLine();
#endif
2014-10-17 04:41:40 +00:00
// Start idle gc.
gc_timer_.Start(
FROM_HERE, base::TimeDelta::FromMinutes(1),
2016-04-13 07:38:39 +00:00
base::Bind(&v8::Isolate::LowMemoryNotification,
base::Unretained(js_env_->isolate())));
2014-10-17 04:41:40 +00:00
content::WebUIControllerFactory::RegisterFactory(
AtomWebUIControllerFactory::GetInstance());
brightray::BrowserMainParts::PreMainMessageLoopRun();
bridge_task_runner_->MessageLoopIsReady();
bridge_task_runner_ = nullptr;
#if defined(USE_X11)
2017-01-26 10:55:19 +00:00
libgtkui::GtkInitFromCommandLine(*base::CommandLine::ForCurrentProcess());
#endif
#if !defined(OS_MACOSX)
2016-06-18 13:26:26 +00:00
// The corresponding call in macOS is in AtomApplicationDelegate.
Browser::Get()->WillFinishLaunching();
2016-09-08 23:23:55 +00:00
std::unique_ptr<base::DictionaryValue> empty_info(new base::DictionaryValue);
2016-09-01 00:26:25 +00:00
Browser::Get()->DidFinishLaunching(*empty_info);
#endif
}
bool AtomBrowserMainParts::MainMessageLoopRun(int* result_code) {
exit_code_ = result_code;
return brightray::BrowserMainParts::MainMessageLoopRun(result_code);
}
2015-10-04 11:21:36 +00:00
void AtomBrowserMainParts::PostMainMessageLoopStart() {
2015-10-04 12:08:19 +00:00
brightray::BrowserMainParts::PostMainMessageLoopStart();
2015-10-04 11:21:36 +00:00
#if defined(OS_POSIX)
HandleShutdownSignals();
#endif
2016-11-30 07:30:03 +00:00
device::GeolocationProvider::SetGeolocationDelegate(
new AtomGeolocationDelegate());
2015-10-04 11:21:36 +00:00
}
void AtomBrowserMainParts::PostMainMessageLoopRun() {
brightray::BrowserMainParts::PostMainMessageLoopRun();
js_env_->OnMessageLoopDestroying();
#if defined(OS_MACOSX)
FreeAppDelegate();
#endif
// Make sure destruction callbacks are called before message loop is
// destroyed, otherwise some objects that need to be deleted on IO thread
// won't be freed.
// We don't use ranged for loop because iterators are getting invalided when
// the callback runs.
for (auto iter = destructors_.begin(); iter != destructors_.end();) {
base::Closure& callback = *iter;
++iter;
callback.Run();
}
}
2013-04-12 01:46:58 +00:00
} // namespace atom