// Copyright (c) 2013 GitHub, Inc. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "browser/api/atom_api_window.h" #include "base/process_util.h" #include "base/values.h" #include "browser/native_window.h" #include "common/v8_conversions.h" #include "common/v8_value_converter_impl.h" #include "content/public/browser/navigation_entry.h" #include "content/public/browser/web_contents.h" #include "content/public/browser/render_process_host.h" #include "ui/gfx/point.h" #include "ui/gfx/rect.h" #include "ui/gfx/size.h" #include "vendor/node/src/node_buffer.h" using content::V8ValueConverter; using content::NavigationController; using node::ObjectWrap; #define UNWRAP_WINDOW_AND_CHECK \ Window* self = ObjectWrap::Unwrap(args.This()); \ if (self == NULL) \ return node::ThrowError("Window is already destroyed") namespace atom { namespace api { Window::Window(v8::Handle wrapper, base::DictionaryValue* options) : EventEmitter(wrapper), window_(NativeWindow::Create(options)) { window_->InitFromOptions(options); window_->AddObserver(this); } Window::~Window() { Emit("destroyed"); window_->RemoveObserver(this); } void Window::OnPageTitleUpdated(bool* prevent_default, const std::string& title) { base::ListValue args; args.AppendString(title); *prevent_default = Emit("page-title-updated", &args); } void Window::OnLoadingStateChanged(bool is_loading) { base::ListValue args; args.AppendBoolean(is_loading); Emit("loading-state-changed", &args); } void Window::WillCloseWindow(bool* prevent_default) { *prevent_default = Emit("close"); } void Window::OnWindowClosed() { Emit("closed"); // Free memory when native window is closed, the delete is delayed so other // observers would not get a invalid pointer of NativeWindow. base::MessageLoop::current()->DeleteSoon(FROM_HERE, this); } void Window::OnWindowBlur() { Emit("blur"); } void Window::OnRendererUnresponsive() { Emit("unresponsive"); } void Window::OnRendererResponsive() { Emit("responsive"); } void Window::OnRendererCrashed() { Emit("crashed"); } void Window::OnCapturePageDone(v8::Persistent callback, const std::vector& data) { v8::HandleScope scope; // TODO(zcbenz): Use new Buffer API when we updated to node v0.12.x. node::Buffer* buffer = node::Buffer::New( reinterpret_cast(data.data()), data.size()); v8::Handle arg = buffer->handle_; callback->Call(v8::Context::GetCurrent()->Global(), 1, &arg); callback.Dispose(v8::Isolate::GetCurrent()); } // static v8::Handle Window::New(const v8::Arguments &args) { v8::HandleScope scope; if (!args.IsConstructCall()) return node::ThrowError("Require constructor call"); if (!args[0]->IsObject()) return node::ThrowTypeError("Need options creating Window"); scoped_ptr converter(new V8ValueConverterImpl()); scoped_ptr options( converter->FromV8Value(args[0], v8::Context::GetCurrent())); if (!options || !options->IsType(base::Value::TYPE_DICTIONARY)) return node::ThrowTypeError("Invalid options"); new Window(args.This(), static_cast(options.get())); // Give js code a chance to do initialization. node::MakeCallback(args.This(), "_init", 0, NULL); return args.This(); } // static v8::Handle Window::Destroy(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; base::ProcessHandle handle = self->window_->GetRenderProcessHandle(); delete self; // Check if the render process is terminated, it could happen that the render // became a zombie. if (!base::WaitForSingleProcess(handle, base::TimeDelta::FromMilliseconds(500))) base::KillProcess(handle, 0, true); return v8::Undefined(); } // static v8::Handle Window::Close(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; self->window_->Close(); return v8::Undefined(); } // static v8::Handle Window::Focus(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; self->window_->Focus(args[0]->IsBoolean() ? args[0]->BooleanValue(): true); return v8::Undefined(); } // static v8::Handle Window::IsFocused(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; return ToV8Value(self->window_->IsFocused()); } // static v8::Handle Window::Show(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; self->window_->Show(); return v8::Undefined(); } // static v8::Handle Window::Hide(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; self->window_->Hide(); return v8::Undefined(); } // static v8::Handle Window::IsVisible(const v8::Arguments& args) { UNWRAP_WINDOW_AND_CHECK; return ToV8Value(self->window_->IsVisible()); } // static v8::Handle Window::Maximize(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; self->window_->Maximize(); return v8::Undefined(); } // static v8::Handle Window::Unmaximize(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; self->window_->Unmaximize(); return v8::Undefined(); } // static v8::Handle Window::Minimize(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; self->window_->Minimize(); return v8::Undefined(); } // static v8::Handle Window::Restore(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; self->window_->Restore(); return v8::Undefined(); } // static v8::Handle Window::SetFullscreen(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; bool fs; if (!FromV8Arguments(args, &fs)) return node::ThrowTypeError("Bad argument"); self->window_->SetFullscreen(fs); return v8::Undefined(); } // static v8::Handle Window::IsFullscreen(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; return ToV8Value(self->window_->IsFullscreen()); } // static v8::Handle Window::SetSize(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; int width, height; if (!FromV8Arguments(args, &width, &height)) return node::ThrowTypeError("Bad argument"); self->window_->SetSize(gfx::Size(width, height)); return v8::Undefined(); } // static v8::Handle Window::GetSize(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; gfx::Size size = self->window_->GetSize(); v8::Handle ret = v8::Array::New(2); ret->Set(0, ToV8Value(size.width())); ret->Set(1, ToV8Value(size.height())); return ret; } // static v8::Handle Window::SetMinimumSize(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; int width, height; if (!FromV8Arguments(args, &width, &height)) return node::ThrowTypeError("Bad argument"); self->window_->SetMinimumSize(gfx::Size(width, height)); return v8::Undefined(); } // static v8::Handle Window::GetMinimumSize(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; gfx::Size size = self->window_->GetMinimumSize(); v8::Handle ret = v8::Array::New(2); ret->Set(0, ToV8Value(size.width())); ret->Set(1, ToV8Value(size.height())); return ret; } // static v8::Handle Window::SetMaximumSize(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; int width, height; if (!FromV8Arguments(args, &width, &height)) return node::ThrowTypeError("Bad argument"); self->window_->SetMaximumSize(gfx::Size(width, height)); return v8::Undefined(); } // static v8::Handle Window::GetMaximumSize(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; gfx::Size size = self->window_->GetMaximumSize(); v8::Handle ret = v8::Array::New(2); ret->Set(0, ToV8Value(size.width())); ret->Set(1, ToV8Value(size.height())); return ret; } // static v8::Handle Window::SetResizable(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; bool resizable; if (!FromV8Arguments(args, &resizable)) return node::ThrowTypeError("Bad argument"); self->window_->SetResizable(resizable); return v8::Undefined(); } // static v8::Handle Window::IsResizable(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; return ToV8Value(self->window_->IsResizable()); } // static v8::Handle Window::SetAlwaysOnTop(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; bool top; if (!FromV8Arguments(args, &top)) return node::ThrowTypeError("Bad argument"); self->window_->SetAlwaysOnTop(top); return v8::Undefined(); } // static v8::Handle Window::IsAlwaysOnTop(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; return ToV8Value(self->window_->IsAlwaysOnTop()); } // static v8::Handle Window::Center(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; self->window_->Center(); return v8::Undefined(); } // static v8::Handle Window::SetPosition(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; int x, y; if (!FromV8Arguments(args, &x, &y)) return node::ThrowTypeError("Bad argument"); self->window_->SetPosition(gfx::Point(x, y)); return v8::Undefined(); } // static v8::Handle Window::GetPosition(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; gfx::Point pos = self->window_->GetPosition(); v8::Handle ret = v8::Array::New(2); ret->Set(0, ToV8Value(pos.x())); ret->Set(1, ToV8Value(pos.y())); return ret; } // static v8::Handle Window::SetTitle(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; std::string title; if (!FromV8Arguments(args, &title)) return node::ThrowTypeError("Bad argument"); self->window_->SetTitle(title); return v8::Undefined(); } // static v8::Handle Window::GetTitle(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; return ToV8Value(self->window_->GetTitle()); } // static v8::Handle Window::FlashFrame(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; self->window_->FlashFrame( args[0]->IsBoolean() ? args[0]->BooleanValue(): true); return v8::Undefined(); } // static v8::Handle Window::SetKiosk(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; bool kiosk; if (!FromV8Arguments(args, &kiosk)) return node::ThrowTypeError("Bad argument"); self->window_->SetKiosk(kiosk); return v8::Undefined(); } // static v8::Handle Window::IsKiosk(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; return ToV8Value(self->window_->IsKiosk()); } // static v8::Handle Window::OpenDevTools(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; self->window_->OpenDevTools(); return v8::Undefined(); } // static v8::Handle Window::CloseDevTools(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; self->window_->CloseDevTools(); return v8::Undefined(); } // static v8::Handle Window::IsDevToolsOpened(const v8::Arguments& args) { UNWRAP_WINDOW_AND_CHECK; return ToV8Value(self->window_->IsDevToolsOpened()); } // static v8::Handle Window::InspectElement(const v8::Arguments& args) { UNWRAP_WINDOW_AND_CHECK; int x, y; if (!FromV8Arguments(args, &x, &y)) return node::ThrowTypeError("Bad argument"); self->window_->InspectElement(x, y); return v8::Undefined(); } // static v8::Handle Window::FocusOnWebView(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; self->window_->FocusOnWebView(); return v8::Undefined(); } // static v8::Handle Window::BlurWebView(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; self->window_->BlurWebView(); return v8::Undefined(); } // static v8::Handle Window::IsWebViewFocused(const v8::Arguments& args) { UNWRAP_WINDOW_AND_CHECK; return ToV8Value(self->window_->IsWebViewFocused()); } // static v8::Handle Window::RestartHangMonitorTimeout( const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; self->window_->RestartHangMonitorTimeout(); return v8::Undefined(); } // static v8::Handle Window::CapturePage(const v8::Arguments& args) { UNWRAP_WINDOW_AND_CHECK; gfx::Rect rect; v8::Persistent callback; if (!FromV8Arguments(args, &rect, &callback) && !FromV8Arguments(args, &callback)) return node::ThrowTypeError("Bad argument"); self->window_->CapturePage(rect, base::Bind(&Window::OnCapturePageDone, base::Unretained(self), callback)); return v8::Undefined(); } // static v8::Handle Window::GetPageTitle(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; return ToV8Value(self->window_->GetWebContents()->GetTitle()); } // static v8::Handle Window::IsLoading(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; return ToV8Value(self->window_->GetWebContents()->IsLoading()); } // static v8::Handle Window::IsWaitingForResponse(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; return ToV8Value(self->window_->GetWebContents()->IsWaitingForResponse()); } // static v8::Handle Window::Stop(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; self->window_->GetWebContents()->Stop(); return v8::Undefined(); } // static v8::Handle Window::GetRoutingID(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; return ToV8Value(self->window_->GetWebContents()->GetRoutingID()); } // static v8::Handle Window::GetProcessID(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; return ToV8Value( self->window_->GetWebContents()->GetRenderProcessHost()->GetID()); } // static v8::Handle Window::IsCrashed(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; return ToV8Value(self->window_->GetWebContents()->IsCrashed()); } // static v8::Handle Window::LoadURL(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; GURL url; if (!FromV8Arguments(args, &url)) return node::ThrowTypeError("Bad argument"); NavigationController& controller = self->window_->GetWebContents()->GetController(); content::NavigationController::LoadURLParams params(url); params.transition_type = content::PAGE_TRANSITION_TYPED; params.override_user_agent = content::NavigationController::UA_OVERRIDE_TRUE; controller.LoadURLWithParams(params); return v8::Undefined(); } // static v8::Handle Window::GetURL(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; NavigationController& controller = self->window_->GetWebContents()->GetController(); GURL url; if (controller.GetActiveEntry()) url = controller.GetActiveEntry()->GetVirtualURL(); return ToV8Value(url); } // static v8::Handle Window::CanGoBack(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; NavigationController& controller = self->window_->GetWebContents()->GetController(); return ToV8Value(controller.CanGoBack()); } // static v8::Handle Window::CanGoForward(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; NavigationController& controller = self->window_->GetWebContents()->GetController(); return ToV8Value(controller.CanGoForward()); } // static v8::Handle Window::CanGoToOffset(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; int offset; if (!FromV8Arguments(args, &offset)) return node::ThrowTypeError("Bad argument"); NavigationController& controller = self->window_->GetWebContents()->GetController(); return ToV8Value(controller.CanGoToOffset(offset)); } // static v8::Handle Window::GoBack(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; NavigationController& controller = self->window_->GetWebContents()->GetController(); controller.GoBack(); return v8::Undefined(); } // static v8::Handle Window::GoForward(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; NavigationController& controller = self->window_->GetWebContents()->GetController(); controller.GoForward(); return v8::Undefined(); } // static v8::Handle Window::GoToIndex(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; int index; if (!FromV8Arguments(args, &index)) return node::ThrowTypeError("Bad argument"); NavigationController& controller = self->window_->GetWebContents()->GetController(); controller.GoToIndex(index); return v8::Undefined(); } // static v8::Handle Window::GoToOffset(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; int offset; if (!FromV8Arguments(args, &offset)) return node::ThrowTypeError("Bad argument"); NavigationController& controller = self->window_->GetWebContents()->GetController(); controller.GoToOffset(offset); return v8::Undefined(); } // static v8::Handle Window::Reload(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; NavigationController& controller = self->window_->GetWebContents()->GetController(); controller.Reload(args[0]->IsBoolean() ? args[0]->BooleanValue(): false); return v8::Undefined(); } // static v8::Handle Window::ReloadIgnoringCache(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; NavigationController& controller = self->window_->GetWebContents()->GetController(); controller.ReloadIgnoringCache( args[0]->IsBoolean() ? args[0]->BooleanValue(): false); return v8::Undefined(); } // static void Window::Initialize(v8::Handle target) { v8::HandleScope scope; v8::Local t = v8::FunctionTemplate::New(Window::New); t->InstanceTemplate()->SetInternalFieldCount(1); t->SetClassName(v8::String::NewSymbol("BrowserWindow")); NODE_SET_PROTOTYPE_METHOD(t, "destroy", Destroy); NODE_SET_PROTOTYPE_METHOD(t, "close", Close); NODE_SET_PROTOTYPE_METHOD(t, "focus", Focus); NODE_SET_PROTOTYPE_METHOD(t, "isFocused", IsFocused); NODE_SET_PROTOTYPE_METHOD(t, "show", Show); NODE_SET_PROTOTYPE_METHOD(t, "hide", Hide); NODE_SET_PROTOTYPE_METHOD(t, "isVisible", IsVisible); NODE_SET_PROTOTYPE_METHOD(t, "maximize", Maximize); NODE_SET_PROTOTYPE_METHOD(t, "unmaximize", Unmaximize); NODE_SET_PROTOTYPE_METHOD(t, "minimize", Minimize); NODE_SET_PROTOTYPE_METHOD(t, "restore", Restore); NODE_SET_PROTOTYPE_METHOD(t, "setFullScreen", SetFullscreen); NODE_SET_PROTOTYPE_METHOD(t, "isFullScreen", IsFullscreen); NODE_SET_PROTOTYPE_METHOD(t, "setSize", SetSize); NODE_SET_PROTOTYPE_METHOD(t, "getSize", GetSize); NODE_SET_PROTOTYPE_METHOD(t, "setMinimumSize", SetMinimumSize); NODE_SET_PROTOTYPE_METHOD(t, "getMinimumSize", GetMinimumSize); NODE_SET_PROTOTYPE_METHOD(t, "setMaximumSize", SetMaximumSize); NODE_SET_PROTOTYPE_METHOD(t, "getMaximumSize", GetMaximumSize); NODE_SET_PROTOTYPE_METHOD(t, "setResizable", SetResizable); NODE_SET_PROTOTYPE_METHOD(t, "isResizable", IsResizable); NODE_SET_PROTOTYPE_METHOD(t, "setAlwaysOnTop", SetAlwaysOnTop); NODE_SET_PROTOTYPE_METHOD(t, "isAlwaysOnTop", IsAlwaysOnTop); NODE_SET_PROTOTYPE_METHOD(t, "center", Center); NODE_SET_PROTOTYPE_METHOD(t, "setPosition", SetPosition); NODE_SET_PROTOTYPE_METHOD(t, "getPosition", GetPosition); NODE_SET_PROTOTYPE_METHOD(t, "setTitle", SetTitle); NODE_SET_PROTOTYPE_METHOD(t, "getTitle", GetTitle); NODE_SET_PROTOTYPE_METHOD(t, "flashFrame", FlashFrame); NODE_SET_PROTOTYPE_METHOD(t, "setKiosk", SetKiosk); NODE_SET_PROTOTYPE_METHOD(t, "isKiosk", IsKiosk); NODE_SET_PROTOTYPE_METHOD(t, "openDevTools", OpenDevTools); NODE_SET_PROTOTYPE_METHOD(t, "closeDevTools", CloseDevTools); NODE_SET_PROTOTYPE_METHOD(t, "isDevToolsOpened", IsDevToolsOpened); NODE_SET_PROTOTYPE_METHOD(t, "inspectElement", InspectElement); NODE_SET_PROTOTYPE_METHOD(t, "focusOnWebView", FocusOnWebView); NODE_SET_PROTOTYPE_METHOD(t, "blurWebView", BlurWebView); NODE_SET_PROTOTYPE_METHOD(t, "isWebViewFocused", IsWebViewFocused); NODE_SET_PROTOTYPE_METHOD(t, "restartHangMonitorTimeout", RestartHangMonitorTimeout); NODE_SET_PROTOTYPE_METHOD(t, "capturePage", CapturePage); NODE_SET_PROTOTYPE_METHOD(t, "getPageTitle", GetPageTitle); NODE_SET_PROTOTYPE_METHOD(t, "isLoading", IsLoading); NODE_SET_PROTOTYPE_METHOD(t, "isWaitingForResponse", IsWaitingForResponse); NODE_SET_PROTOTYPE_METHOD(t, "stop", Stop); NODE_SET_PROTOTYPE_METHOD(t, "getRoutingId", GetRoutingID); NODE_SET_PROTOTYPE_METHOD(t, "getProcessId", GetProcessID); NODE_SET_PROTOTYPE_METHOD(t, "isCrashed", IsCrashed); NODE_SET_PROTOTYPE_METHOD(t, "loadUrl", LoadURL); NODE_SET_PROTOTYPE_METHOD(t, "getUrl", GetURL); NODE_SET_PROTOTYPE_METHOD(t, "canGoBack", CanGoBack); NODE_SET_PROTOTYPE_METHOD(t, "canGoForward", CanGoForward); NODE_SET_PROTOTYPE_METHOD(t, "canGoToOffset", CanGoToOffset); NODE_SET_PROTOTYPE_METHOD(t, "goBack", GoBack); NODE_SET_PROTOTYPE_METHOD(t, "goForward", GoForward); NODE_SET_PROTOTYPE_METHOD(t, "goToIndex", GoToIndex); NODE_SET_PROTOTYPE_METHOD(t, "goToOffset", GoToOffset); NODE_SET_PROTOTYPE_METHOD(t, "reload", Reload); NODE_SET_PROTOTYPE_METHOD(t, "reloadIgnoringCache", ReloadIgnoringCache); target->Set(v8::String::NewSymbol("BrowserWindow"), t->GetFunction()); } } // namespace api } // namespace atom NODE_MODULE(atom_browser_window, atom::api::Window::Initialize)