// Copyright (c) 2013 GitHub, Inc. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #include "shell/browser/browser.h" #include #include #include "base/command_line.h" #include "base/environment.h" #include "base/process/launch.h" #include "electron/electron_version.h" #include "shell/browser/native_window.h" #include "shell/browser/window_list.h" #include "shell/common/application_info.h" #if defined(USE_X11) #include "chrome/browser/ui/libgtkui/gtk_util.h" #include "chrome/browser/ui/libgtkui/unity_service.h" #endif namespace atom { const char kXdgSettings[] = "xdg-settings"; const char kXdgSettingsDefaultSchemeHandler[] = "default-url-scheme-handler"; bool LaunchXdgUtility(const std::vector& argv, int* exit_code) { *exit_code = EXIT_FAILURE; int devnull = open("/dev/null", O_RDONLY); if (devnull < 0) return false; base::LaunchOptions options; options.fds_to_remap.push_back(std::make_pair(devnull, STDIN_FILENO)); base::Process process = base::LaunchProcess(argv, options); close(devnull); if (!process.IsValid()) return false; return process.WaitForExit(exit_code); } bool SetDefaultWebClient(const std::string& protocol) { std::unique_ptr env(base::Environment::Create()); std::vector argv; argv.push_back(kXdgSettings); argv.push_back("set"); if (!protocol.empty()) { argv.push_back(kXdgSettingsDefaultSchemeHandler); argv.push_back(protocol); } argv.push_back(libgtkui::GetDesktopName(env.get())); int exit_code; bool ran_ok = LaunchXdgUtility(argv, &exit_code); return ran_ok && exit_code == EXIT_SUCCESS; } void Browser::Focus() { // Focus on the first visible window. for (auto* const window : WindowList::GetWindows()) { if (window->IsVisible()) { window->Focus(true); break; } } } void Browser::AddRecentDocument(const base::FilePath& path) {} void Browser::ClearRecentDocuments() {} void Browser::SetAppUserModelID(const base::string16& name) {} bool Browser::SetAsDefaultProtocolClient(const std::string& protocol, mate::Arguments* args) { return SetDefaultWebClient(protocol); } bool Browser::IsDefaultProtocolClient(const std::string& protocol, mate::Arguments* args) { std::unique_ptr env(base::Environment::Create()); if (protocol.empty()) return false; std::vector argv; argv.push_back(kXdgSettings); argv.push_back("check"); argv.push_back(kXdgSettingsDefaultSchemeHandler); argv.push_back(protocol); argv.push_back(libgtkui::GetDesktopName(env.get())); std::string reply; int success_code; bool ran_ok = base::GetAppOutputWithExitCode(base::CommandLine(argv), &reply, &success_code); if (!ran_ok || success_code != EXIT_SUCCESS) return false; // Allow any reply that starts with "yes". return base::StartsWith(reply, "yes", base::CompareCase::SENSITIVE) ? true : false; } // Todo implement bool Browser::RemoveAsDefaultProtocolClient(const std::string& protocol, mate::Arguments* args) { return false; } bool Browser::SetBadgeCount(int count) { if (IsUnityRunning()) { unity::SetDownloadCount(count); badge_count_ = count; return true; } else { return false; } } void Browser::SetLoginItemSettings(LoginItemSettings settings) {} Browser::LoginItemSettings Browser::GetLoginItemSettings( const LoginItemSettings& options) { return LoginItemSettings(); } std::string Browser::GetExecutableFileVersion() const { return GetApplicationVersion(); } std::string Browser::GetExecutableFileProductName() const { return GetApplicationName(); } bool Browser::IsUnityRunning() { return unity::IsRunning(); } bool Browser::IsEmojiPanelSupported() { return false; } void Browser::ShowAboutPanel() { std::string app_name, version, copyright, icon_path, website; GtkAboutDialog* dialog = GTK_ABOUT_DIALOG(gtk_about_dialog_new()); if (about_panel_options_.GetString("applicationName", &app_name)) gtk_about_dialog_set_program_name(dialog, app_name.c_str()); if (about_panel_options_.GetString("applicationVersion", &version)) gtk_about_dialog_set_version(dialog, version.c_str()); if (about_panel_options_.GetString("copyright", ©right)) gtk_about_dialog_set_copyright(dialog, copyright.c_str()); if (about_panel_options_.GetString("website", &website)) gtk_about_dialog_set_website(dialog, website.c_str()); if (about_panel_options_.GetString("iconPath", &icon_path)) { GError* error = nullptr; GdkPixbuf* icon = gdk_pixbuf_new_from_file(icon_path.c_str(), &error); if (error != nullptr) { g_warning("%s", error->message); g_clear_error(&error); } else { gtk_about_dialog_set_logo(dialog, icon); g_clear_object(&icon); } } gtk_dialog_run(GTK_DIALOG(dialog)); g_clear_object(&dialog); } void Browser::SetAboutPanelOptions(const base::DictionaryValue& options) { about_panel_options_.Clear(); for (const auto& pair : options) { const std::string& key = pair.first; const auto& val = pair.second; if (!key.empty() && val->is_string()) about_panel_options_.SetString(key, val->GetString()); } } } // namespace atom