// 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 #include #include "base/bind.h" #include "atom/browser/api/atom_api_window.h" #include "atom/browser/native_window.h" #include "atom/browser/ui/file_dialog.h" #include "atom/browser/ui/message_box.h" #include "atom/common/v8_converters/file_path_converter.h" #include "native_mate/dictionary.h" #include "native_mate/scoped_persistent.h" #include "atom/common/v8/node_common.h" namespace mate { template<> struct Converter { static bool FromV8(v8::Isolate* isolate, v8::Handle val, atom::NativeWindow** out) { using atom::api::Window; if (val->IsNull()) { *out = NULL; return true; // NULL is a valid value for NativeWindow*. } else if (val->IsObject()) { Window* window = Window::Unwrap(val->ToObject()); *out = window->window(); return true; } else { return false; } } }; typedef scoped_refptr> RefCountedV8Function; template<> struct Converter { static bool FromV8(v8::Isolate* isolate, v8::Handle val, RefCountedV8Function* out) { if (!val->IsFunction()) return false; v8::Handle function = v8::Handle::Cast(val); *out = new mate::RefCountedPersistent(function); return true; } }; } // namespace mate namespace { template void CallV8Function(const mate::RefCountedV8Function& callback, T arg) { v8::Locker locker(node_isolate); v8::HandleScope handle_scope(node_isolate); v8::Handle value = mate::Converter::ToV8(node_isolate, arg); callback->NewHandle(node_isolate)->Call( v8::Context::GetCurrent()->Global(), 1, &value); } template void CallV8Function2( const mate::RefCountedV8Function& callback, bool result, const T& arg) { if (result) return CallV8Function(callback, arg); else return CallV8Function(callback, NULL); } void ShowMessageBox(int type, const std::vector& buttons, const std::string& title, const std::string& message, const std::string& detail, atom::NativeWindow* window, mate::Arguments* args) { v8::Handle peek = args->PeekNext(); mate::RefCountedV8Function callback; if (mate::Converter::FromV8(node_isolate, peek, &callback)) { atom::ShowMessageBox( window, (atom::MessageBoxType)type, buttons, title, message, detail, base::Bind(&CallV8Function, callback)); } else { int chosen = atom::ShowMessageBox( window, (atom::MessageBoxType)type, buttons, title, message, detail); args->Return(chosen); } } void ShowOpenDialog(const std::string& title, const base::FilePath& default_path, int properties, atom::NativeWindow* window, mate::Arguments* args) { v8::Handle peek = args->PeekNext(); mate::RefCountedV8Function callback; if (mate::Converter::FromV8(node_isolate, peek, &callback)) { file_dialog::ShowOpenDialog( window, title, default_path, properties, base::Bind(&CallV8Function2>, callback)); } else { std::vector paths; if (file_dialog::ShowOpenDialog(window, title, default_path, properties, &paths)) args->Return(paths); } } void ShowSaveDialog(const std::string& title, const base::FilePath& default_path, atom::NativeWindow* window, mate::Arguments* args) { v8::Handle peek = args->PeekNext(); mate::RefCountedV8Function callback; if (mate::Converter::FromV8(node_isolate, peek, &callback)) { file_dialog::ShowSaveDialog( window, title, default_path, base::Bind(&CallV8Function2, callback)); } else { base::FilePath path; if (file_dialog::ShowSaveDialog(window, title, default_path, &path)) args->Return(path); } } void Initialize(v8::Handle exports) { mate::Dictionary dict(v8::Isolate::GetCurrent(), exports); dict.SetMethod("showMessageBox", &ShowMessageBox); dict.SetMethod("showOpenDialog", &ShowOpenDialog); dict.SetMethod("showSaveDialog", &ShowSaveDialog); } } // namespace NODE_MODULE(atom_browser_dialog, Initialize)