// 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/values.h" #include "browser/native_window.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" 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 { namespace { // Converts string16 to V8 String. v8::Handle UTF16ToV8String(const string16& s) { return v8::String::New(reinterpret_cast(s.data()), s.size()); } } // namespace 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::WillCloseWindow(bool* prevent_default) { *prevent_default = Emit("close"); } void Window::OnWindowClosed() { Emit("closed"); // Free memory immediately when window is closed. delete this; } void Window::OnWindowBlur() { Emit("blur"); } void Window::OnRendererUnresponsive() { Emit("unresponsive"); } void Window::OnRendererResponsive() { Emit("responsive"); } void Window::OnRendererCrashed() { Emit("crashed"); } // 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())); return args.This(); } // static v8::Handle Window::Destroy(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; delete self; 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 v8::Boolean::New(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::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; if (args.Length() < 1 || !args[0]->IsBoolean()) return node::ThrowTypeError("Bad argument"); self->window_->SetFullscreen(args[0]->BooleanValue()); return v8::Undefined(); } // static v8::Handle Window::IsFullscreen(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; return v8::Boolean::New(self->window_->IsFullscreen()); } // static v8::Handle Window::SetSize(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; if (args.Length() < 2) return node::ThrowTypeError("Bad argument"); self->window_->SetSize( gfx::Size(args[0]->IntegerValue(), args[1]->IntegerValue())); 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, v8::Integer::New(size.width())); ret->Set(1, v8::Integer::New(size.height())); return ret; } // static v8::Handle Window::SetMinimumSize(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; if (args.Length() < 2) return node::ThrowTypeError("Bad argument"); self->window_->SetMinimumSize( gfx::Size(args[0]->IntegerValue(), args[1]->IntegerValue())); 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, v8::Integer::New(size.width())); ret->Set(1, v8::Integer::New(size.height())); return ret; } // static v8::Handle Window::SetMaximumSize(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; if (args.Length() < 2) return node::ThrowTypeError("Bad argument"); self->window_->SetMaximumSize( gfx::Size(args[0]->IntegerValue(), args[1]->IntegerValue())); 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, v8::Integer::New(size.width())); ret->Set(1, v8::Integer::New(size.height())); return ret; } // static v8::Handle Window::SetResizable(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; if (args.Length() < 1 || !args[0]->IsBoolean()) return node::ThrowTypeError("Bad argument"); self->window_->SetResizable(args[0]->BooleanValue()); return v8::Undefined(); } // static v8::Handle Window::IsResizable(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; return v8::Boolean::New(self->window_->IsResizable()); } // static v8::Handle Window::SetAlwaysOnTop(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; if (args.Length() < 1 || !args[0]->IsBoolean()) return node::ThrowTypeError("Bad argument"); self->window_->SetAlwaysOnTop(args[0]->BooleanValue()); return v8::Undefined(); } // static v8::Handle Window::IsAlwaysOnTop(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; return v8::Boolean::New(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; if (args.Length() < 2) return node::ThrowTypeError("Bad argument"); self->window_->SetPosition( gfx::Point(args[0]->IntegerValue(), args[1]->IntegerValue())); 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, v8::Integer::New(pos.x())); ret->Set(1, v8::Integer::New(pos.y())); return ret; } // static v8::Handle Window::SetTitle(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; if (args.Length() < 1 || !args[0]->IsString()) return node::ThrowTypeError("Bad argument"); self->window_->SetTitle(*v8::String::Utf8Value(args[0])); return v8::Undefined(); } // static v8::Handle Window::GetTitle(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; std::string title = self->window_->GetTitle(); return v8::String::New(title.c_str(), title.size()); } // 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; if (args.Length() < 1 || !args[0]->IsBoolean()) return node::ThrowTypeError("Bad argument"); self->window_->SetKiosk(args[0]->BooleanValue()); return v8::Undefined(); } // static v8::Handle Window::IsKiosk(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; return v8::Boolean::New(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::InspectElement(const v8::Arguments& args) { UNWRAP_WINDOW_AND_CHECK; self->window_->InspectElement(args[0]->IntegerValue(), args[1]->IntegerValue()); 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::RestartHangMonitorTimeout( const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; self->window_->RestartHangMonitorTimeout(); return v8::Undefined(); } // static v8::Handle Window::GetPageTitle(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; string16 title = self->window_->GetWebContents()->GetTitle(); return UTF16ToV8String(title); } // static v8::Handle Window::IsLoading(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; return v8::Boolean::New(self->window_->GetWebContents()->IsLoading()); } // static v8::Handle Window::IsWaitingForResponse(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; return v8::Boolean::New( 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 v8::Integer::New(self->window_->GetWebContents()->GetRoutingID()); } // static v8::Handle Window::GetProcessID(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; return v8::Integer::New( self->window_->GetWebContents()->GetRenderProcessHost()->GetID()); } // static v8::Handle Window::IsCrashed(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; return v8::Boolean::New(self->window_->GetWebContents()->IsCrashed()); } // static v8::Handle Window::LoadURL(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; if (args.Length() < 1 || !args[0]->IsString()) return node::ThrowTypeError("Bad argument"); NavigationController& controller = self->window_->GetWebContents()->GetController(); controller.LoadURL(GURL(*v8::String::Utf8Value(args[0])), content::Referrer(), content::PAGE_TRANSITION_AUTO_TOPLEVEL, std::string()); return v8::Undefined(); } // static v8::Handle Window::GetURL(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; NavigationController& controller = self->window_->GetWebContents()->GetController(); std::string url; if (controller.GetActiveEntry()) url = controller.GetActiveEntry()->GetVirtualURL().spec(); return v8::String::New(url.c_str(), url.size()); } // static v8::Handle Window::CanGoBack(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; NavigationController& controller = self->window_->GetWebContents()->GetController(); return v8::Boolean::New(controller.CanGoBack()); } // static v8::Handle Window::CanGoForward(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; NavigationController& controller = self->window_->GetWebContents()->GetController(); return v8::Boolean::New(controller.CanGoForward()); } // static v8::Handle Window::CanGoToOffset(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; if (args.Length() < 1) return node::ThrowTypeError("Bad argument"); NavigationController& controller = self->window_->GetWebContents()->GetController(); int offset = args[0]->IntegerValue(); return v8::Boolean::New(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; if (args.Length() < 1) return node::ThrowTypeError("Bad argument"); NavigationController& controller = self->window_->GetWebContents()->GetController(); controller.GoToIndex(args[0]->IntegerValue()); return v8::Undefined(); } // static v8::Handle Window::GoToOffset(const v8::Arguments &args) { UNWRAP_WINDOW_AND_CHECK; if (args.Length() < 1) return node::ThrowTypeError("Bad argument"); NavigationController& controller = self->window_->GetWebContents()->GetController(); controller.GoToOffset(args[0]->IntegerValue()); 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, "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, "inspectElement", InspectElement); NODE_SET_PROTOTYPE_METHOD(t, "focusOnWebView", FocusOnWebView); NODE_SET_PROTOTYPE_METHOD(t, "blurWebView", BlurWebView); NODE_SET_PROTOTYPE_METHOD(t, "restartHangMonitorTimeout", RestartHangMonitorTimeout); 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)