2015-07-07 07:02:27 +00:00
|
|
|
// Copyright (c) 2015 GitHub, Inc.
|
|
|
|
// Use of this source code is governed by the MIT license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
|
|
|
#include "atom/browser/ui/message_box.h"
|
|
|
|
|
|
|
|
#include "atom/browser/browser.h"
|
2016-12-20 23:52:58 +00:00
|
|
|
#include "atom/browser/native_window_observer.h"
|
2016-06-21 08:54:55 +00:00
|
|
|
#include "atom/browser/native_window_views.h"
|
2016-07-11 06:29:03 +00:00
|
|
|
#include "atom/browser/unresponsive_suppressor.h"
|
2015-07-07 07:02:27 +00:00
|
|
|
#include "base/callback.h"
|
2015-07-07 07:45:13 +00:00
|
|
|
#include "base/strings/string_util.h"
|
2015-07-07 07:02:27 +00:00
|
|
|
#include "base/strings/utf_string_conversions.h"
|
2017-01-26 10:55:19 +00:00
|
|
|
#include "chrome/browser/ui/libgtkui/gtk_signal.h"
|
|
|
|
#include "chrome/browser/ui/libgtkui/gtk_util.h"
|
|
|
|
#include "chrome/browser/ui/libgtkui/skia_utils_gtk.h"
|
2015-07-07 08:42:03 +00:00
|
|
|
#include "ui/views/widget/desktop_aura/x11_desktop_handler.h"
|
2015-07-07 07:02:27 +00:00
|
|
|
|
|
|
|
#define ANSI_FOREGROUND_RED "\x1b[31m"
|
|
|
|
#define ANSI_FOREGROUND_BLACK "\x1b[30m"
|
|
|
|
#define ANSI_TEXT_BOLD "\x1b[1m"
|
|
|
|
#define ANSI_BACKGROUND_GRAY "\x1b[47m"
|
|
|
|
#define ANSI_RESET "\x1b[0m"
|
|
|
|
|
|
|
|
namespace atom {
|
|
|
|
|
2015-07-07 07:45:13 +00:00
|
|
|
namespace {
|
|
|
|
|
2016-12-20 23:52:58 +00:00
|
|
|
class GtkMessageBox : public NativeWindowObserver {
|
2015-07-07 07:45:13 +00:00
|
|
|
public:
|
2015-07-07 09:03:47 +00:00
|
|
|
GtkMessageBox(NativeWindow* parent_window,
|
2015-07-07 07:45:13 +00:00
|
|
|
MessageBoxType type,
|
|
|
|
const std::vector<std::string>& buttons,
|
2016-01-10 23:33:27 +00:00
|
|
|
int default_id,
|
2015-07-07 10:26:50 +00:00
|
|
|
int cancel_id,
|
2015-07-07 07:45:13 +00:00
|
|
|
const std::string& title,
|
|
|
|
const std::string& message,
|
|
|
|
const std::string& detail,
|
2017-02-06 15:35:36 +00:00
|
|
|
const std::string& checkbox_label,
|
|
|
|
bool checkbox_checked,
|
2015-07-07 07:45:13 +00:00
|
|
|
const gfx::ImageSkia& icon)
|
2016-07-11 06:29:03 +00:00
|
|
|
: cancel_id_(cancel_id),
|
2017-02-06 15:35:36 +00:00
|
|
|
checkbox_checked_(false),
|
2016-06-21 08:54:55 +00:00
|
|
|
parent_(static_cast<NativeWindowViews*>(parent_window)) {
|
2015-07-07 07:45:13 +00:00
|
|
|
// Create dialog.
|
|
|
|
dialog_ = gtk_message_dialog_new(
|
|
|
|
nullptr, // parent
|
2015-07-07 09:03:47 +00:00
|
|
|
static_cast<GtkDialogFlags>(0), // no flags
|
2015-07-07 08:52:53 +00:00
|
|
|
GetMessageType(type), // type
|
2015-07-07 07:45:13 +00:00
|
|
|
GTK_BUTTONS_NONE, // no buttons
|
|
|
|
"%s", message.c_str());
|
2015-07-07 09:08:30 +00:00
|
|
|
if (!detail.empty())
|
|
|
|
gtk_message_dialog_format_secondary_text(
|
|
|
|
GTK_MESSAGE_DIALOG(dialog_), "%s", detail.c_str());
|
2015-07-07 09:09:58 +00:00
|
|
|
if (!title.empty())
|
|
|
|
gtk_window_set_title(GTK_WINDOW(dialog_), title.c_str());
|
2015-07-07 07:45:13 +00:00
|
|
|
|
|
|
|
// Set dialog's icon.
|
|
|
|
if (!icon.isNull()) {
|
2017-01-26 10:55:19 +00:00
|
|
|
GdkPixbuf* pixbuf = libgtkui::GdkPixbufFromSkBitmap(*icon.bitmap());
|
2016-05-11 17:00:10 +00:00
|
|
|
GtkIconSource* iconsource = gtk_icon_source_new();
|
|
|
|
GtkIconSet* iconset = gtk_icon_set_new();
|
|
|
|
gtk_icon_source_set_pixbuf(iconsource, pixbuf);
|
|
|
|
gtk_icon_set_add_source(iconset, iconsource);
|
|
|
|
GtkWidget* image = gtk_image_new_from_icon_set(iconset,
|
|
|
|
GTK_ICON_SIZE_DIALOG);
|
2015-07-07 07:45:13 +00:00
|
|
|
gtk_message_dialog_set_image(GTK_MESSAGE_DIALOG(dialog_), image);
|
|
|
|
gtk_widget_show(image);
|
2016-05-11 17:00:10 +00:00
|
|
|
gtk_icon_source_free(iconsource);
|
|
|
|
gtk_icon_set_unref(iconset);
|
2015-07-07 07:45:13 +00:00
|
|
|
g_object_unref(pixbuf);
|
|
|
|
}
|
|
|
|
|
2017-02-06 15:35:36 +00:00
|
|
|
if (!checkbox_label.empty()) {
|
|
|
|
GtkWidget* message_area =
|
|
|
|
gtk_message_dialog_get_message_area(GTK_MESSAGE_DIALOG(dialog_));
|
|
|
|
GtkWidget* check_button =
|
|
|
|
gtk_check_button_new_with_label(checkbox_label.c_str());
|
|
|
|
g_signal_connect(check_button, "toggled",
|
|
|
|
G_CALLBACK(OnCheckboxToggledThunk), this);
|
|
|
|
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(check_button),
|
|
|
|
checkbox_checked);
|
|
|
|
gtk_container_add(GTK_CONTAINER(message_area), check_button);
|
|
|
|
}
|
|
|
|
|
2015-07-07 07:45:13 +00:00
|
|
|
// Add buttons.
|
|
|
|
for (size_t i = 0; i < buttons.size(); ++i) {
|
2016-01-11 13:12:07 +00:00
|
|
|
GtkWidget* button = gtk_dialog_add_button(
|
|
|
|
GTK_DIALOG(dialog_), TranslateToStock(i, buttons[i]), i);
|
|
|
|
if (static_cast<int>(i) == default_id)
|
2016-01-08 05:23:15 +00:00
|
|
|
gtk_widget_grab_focus(button);
|
2015-07-07 07:45:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Parent window.
|
2016-06-21 08:54:55 +00:00
|
|
|
if (parent_) {
|
2016-12-20 23:52:58 +00:00
|
|
|
parent_->AddObserver(this);
|
2016-06-21 08:54:55 +00:00
|
|
|
parent_->SetEnabled(false);
|
2017-01-26 10:55:19 +00:00
|
|
|
libgtkui::SetGtkTransientForAura(dialog_, parent_->GetNativeWindow());
|
2016-06-21 08:54:55 +00:00
|
|
|
gtk_window_set_modal(GTK_WINDOW(dialog_), TRUE);
|
2015-07-07 07:45:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
~GtkMessageBox() {
|
|
|
|
gtk_widget_destroy(dialog_);
|
2016-12-20 23:52:58 +00:00
|
|
|
if (parent_) {
|
|
|
|
parent_->RemoveObserver(this);
|
2016-06-21 08:54:55 +00:00
|
|
|
parent_->SetEnabled(true);
|
2016-12-20 23:52:58 +00:00
|
|
|
}
|
2015-07-07 07:45:13 +00:00
|
|
|
}
|
|
|
|
|
2015-07-07 08:52:53 +00:00
|
|
|
GtkMessageType GetMessageType(MessageBoxType type) {
|
|
|
|
switch (type) {
|
|
|
|
case MESSAGE_BOX_TYPE_INFORMATION:
|
|
|
|
return GTK_MESSAGE_INFO;
|
|
|
|
case MESSAGE_BOX_TYPE_WARNING:
|
|
|
|
return GTK_MESSAGE_WARNING;
|
|
|
|
case MESSAGE_BOX_TYPE_QUESTION:
|
|
|
|
return GTK_MESSAGE_QUESTION;
|
|
|
|
case MESSAGE_BOX_TYPE_ERROR:
|
|
|
|
return GTK_MESSAGE_ERROR;
|
|
|
|
default:
|
|
|
|
return GTK_MESSAGE_OTHER;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-07 07:45:13 +00:00
|
|
|
const char* TranslateToStock(int id, const std::string& text) {
|
2015-12-07 11:56:23 +00:00
|
|
|
std::string lower = base::ToLowerASCII(text);
|
2015-07-07 10:33:11 +00:00
|
|
|
if (lower == "cancel")
|
2015-07-07 07:45:13 +00:00
|
|
|
return GTK_STOCK_CANCEL;
|
2015-07-07 10:33:11 +00:00
|
|
|
else if (lower == "no")
|
2015-07-07 07:45:13 +00:00
|
|
|
return GTK_STOCK_NO;
|
2015-07-07 10:33:11 +00:00
|
|
|
else if (lower == "ok")
|
2015-07-07 07:45:13 +00:00
|
|
|
return GTK_STOCK_OK;
|
2015-07-07 10:33:11 +00:00
|
|
|
else if (lower == "yes")
|
2015-07-07 07:45:13 +00:00
|
|
|
return GTK_STOCK_YES;
|
2015-07-07 10:33:11 +00:00
|
|
|
else
|
2015-07-07 07:45:13 +00:00
|
|
|
return text.c_str();
|
|
|
|
}
|
|
|
|
|
2015-07-07 08:42:03 +00:00
|
|
|
void Show() {
|
|
|
|
gtk_widget_show_all(dialog_);
|
|
|
|
// We need to call gtk_window_present after making the widgets visible to
|
|
|
|
// make sure window gets correctly raised and gets focus.
|
2017-01-26 10:55:19 +00:00
|
|
|
int time = ui::X11EventSource::GetInstance()->GetTimestamp();
|
2015-07-07 08:42:03 +00:00
|
|
|
gtk_window_present_with_time(GTK_WINDOW(dialog_), time);
|
|
|
|
}
|
|
|
|
|
2015-07-07 09:03:47 +00:00
|
|
|
int RunSynchronous() {
|
|
|
|
Show();
|
|
|
|
int response = gtk_dialog_run(GTK_DIALOG(dialog_));
|
|
|
|
if (response < 0)
|
|
|
|
return cancel_id_;
|
|
|
|
else
|
|
|
|
return response;
|
|
|
|
}
|
|
|
|
|
2015-07-07 07:45:13 +00:00
|
|
|
void RunAsynchronous(const MessageBoxCallback& callback) {
|
|
|
|
callback_ = callback;
|
|
|
|
g_signal_connect(dialog_, "delete-event",
|
|
|
|
G_CALLBACK(gtk_widget_hide_on_delete), nullptr);
|
|
|
|
g_signal_connect(dialog_, "response",
|
|
|
|
G_CALLBACK(OnResponseDialogThunk), this);
|
2015-07-07 08:42:03 +00:00
|
|
|
Show();
|
2015-07-07 07:45:13 +00:00
|
|
|
}
|
|
|
|
|
2016-12-21 00:21:48 +00:00
|
|
|
void OnWindowClosed() override {
|
2016-12-20 23:52:58 +00:00
|
|
|
parent_->RemoveObserver(this);
|
|
|
|
parent_ = nullptr;
|
|
|
|
}
|
|
|
|
|
2015-07-07 07:45:13 +00:00
|
|
|
CHROMEGTK_CALLBACK_1(GtkMessageBox, void, OnResponseDialog, int);
|
2017-02-09 17:49:09 +00:00
|
|
|
CHROMEGTK_CALLBACK_0(GtkMessageBox, void, OnCheckboxToggled);
|
2015-07-07 07:45:13 +00:00
|
|
|
|
|
|
|
private:
|
2016-07-11 06:29:03 +00:00
|
|
|
atom::UnresponsiveSuppressor unresponsive_suppressor_;
|
2015-07-07 07:45:13 +00:00
|
|
|
|
|
|
|
// The id to return when the dialog is closed without pressing buttons.
|
|
|
|
int cancel_id_;
|
|
|
|
|
2017-02-06 15:35:36 +00:00
|
|
|
bool checkbox_checked_;
|
|
|
|
|
2016-06-21 08:54:55 +00:00
|
|
|
NativeWindowViews* parent_;
|
2015-07-07 07:45:13 +00:00
|
|
|
GtkWidget* dialog_;
|
|
|
|
MessageBoxCallback callback_;
|
|
|
|
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(GtkMessageBox);
|
|
|
|
};
|
|
|
|
|
|
|
|
void GtkMessageBox::OnResponseDialog(GtkWidget* widget, int response) {
|
2016-05-11 19:33:43 +00:00
|
|
|
gtk_widget_hide(dialog_);
|
2015-07-07 07:45:13 +00:00
|
|
|
|
|
|
|
if (response < 0)
|
2017-02-06 15:35:36 +00:00
|
|
|
callback_.Run(cancel_id_, checkbox_checked_);
|
2015-07-07 07:45:13 +00:00
|
|
|
else
|
2017-02-06 15:35:36 +00:00
|
|
|
callback_.Run(response, checkbox_checked_);
|
2015-07-07 07:45:13 +00:00
|
|
|
delete this;
|
|
|
|
}
|
|
|
|
|
2017-02-09 17:49:09 +00:00
|
|
|
void GtkMessageBox::OnCheckboxToggled(GtkWidget* widget) {
|
2017-02-06 15:35:36 +00:00
|
|
|
checkbox_checked_ = GTK_TOGGLE_BUTTON(widget)->active;
|
|
|
|
}
|
|
|
|
|
2015-07-07 07:45:13 +00:00
|
|
|
} // namespace
|
|
|
|
|
|
|
|
int ShowMessageBox(NativeWindow* parent,
|
2015-07-07 07:02:27 +00:00
|
|
|
MessageBoxType type,
|
|
|
|
const std::vector<std::string>& buttons,
|
2016-01-10 23:33:27 +00:00
|
|
|
int default_id,
|
2015-07-07 10:26:50 +00:00
|
|
|
int cancel_id,
|
2015-07-23 06:16:43 +00:00
|
|
|
int options,
|
2015-07-07 07:02:27 +00:00
|
|
|
const std::string& title,
|
|
|
|
const std::string& message,
|
|
|
|
const std::string& detail,
|
|
|
|
const gfx::ImageSkia& icon) {
|
2017-02-06 15:35:36 +00:00
|
|
|
return GtkMessageBox(parent, type, buttons, default_id, cancel_id, title,
|
|
|
|
message, detail, "", false, icon)
|
|
|
|
.RunSynchronous();
|
2015-07-07 07:02:27 +00:00
|
|
|
}
|
|
|
|
|
2015-07-07 07:45:13 +00:00
|
|
|
void ShowMessageBox(NativeWindow* parent,
|
2015-07-07 07:02:27 +00:00
|
|
|
MessageBoxType type,
|
|
|
|
const std::vector<std::string>& buttons,
|
2016-01-10 23:33:27 +00:00
|
|
|
int default_id,
|
2015-07-07 10:26:50 +00:00
|
|
|
int cancel_id,
|
2015-07-23 06:16:43 +00:00
|
|
|
int options,
|
2015-07-07 07:02:27 +00:00
|
|
|
const std::string& title,
|
|
|
|
const std::string& message,
|
|
|
|
const std::string& detail,
|
2017-02-06 15:35:36 +00:00
|
|
|
const std::string& checkbox_label,
|
|
|
|
bool checkbox_checked,
|
2015-07-07 07:02:27 +00:00
|
|
|
const gfx::ImageSkia& icon,
|
|
|
|
const MessageBoxCallback& callback) {
|
2017-02-06 15:35:36 +00:00
|
|
|
(new GtkMessageBox(parent, type, buttons, default_id, cancel_id, title,
|
|
|
|
message, detail, checkbox_label, checkbox_checked, icon))
|
|
|
|
->RunAsynchronous(callback);
|
2015-07-07 07:02:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ShowErrorBox(const base::string16& title, const base::string16& content) {
|
|
|
|
if (Browser::Get()->is_ready()) {
|
2017-02-06 15:35:36 +00:00
|
|
|
GtkMessageBox(nullptr, MESSAGE_BOX_TYPE_ERROR, {"OK"}, -1, 0, "Error",
|
2015-07-07 09:03:47 +00:00
|
|
|
base::UTF16ToUTF8(title).c_str(),
|
2017-02-06 15:35:36 +00:00
|
|
|
base::UTF16ToUTF8(content).c_str(), "", false,
|
|
|
|
gfx::ImageSkia())
|
|
|
|
.RunSynchronous();
|
2015-07-07 07:02:27 +00:00
|
|
|
} else {
|
|
|
|
fprintf(stderr,
|
|
|
|
ANSI_TEXT_BOLD ANSI_BACKGROUND_GRAY
|
|
|
|
ANSI_FOREGROUND_RED "%s\n"
|
|
|
|
ANSI_FOREGROUND_BLACK "%s"
|
|
|
|
ANSI_RESET "\n",
|
|
|
|
base::UTF16ToUTF8(title).c_str(),
|
|
|
|
base::UTF16ToUTF8(content).c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace atom
|