gtk: Implement message box APIs
This commit is contained in:
parent
8fca1f52d3
commit
b98cdf71c4
2 changed files with 132 additions and 9 deletions
|
@ -34,7 +34,7 @@ class FileChooserDialog {
|
||||||
const std::string& title,
|
const std::string& title,
|
||||||
const base::FilePath& default_path,
|
const base::FilePath& default_path,
|
||||||
const Filters& filters)
|
const Filters& filters)
|
||||||
: dialog_scope_(new atom::NativeWindow::DialogScope(parent_window)) {
|
: dialog_scope_(parent_window) {
|
||||||
const char* confirm_text = GTK_STOCK_OK;
|
const char* confirm_text = GTK_STOCK_OK;
|
||||||
if (action == GTK_FILE_CHOOSER_ACTION_SAVE)
|
if (action == GTK_FILE_CHOOSER_ACTION_SAVE)
|
||||||
confirm_text = GTK_STOCK_SAVE;
|
confirm_text = GTK_STOCK_SAVE;
|
||||||
|
@ -131,13 +131,13 @@ class FileChooserDialog {
|
||||||
private:
|
private:
|
||||||
void AddFilters(const Filters& filters);
|
void AddFilters(const Filters& filters);
|
||||||
|
|
||||||
|
atom::NativeWindow::DialogScope dialog_scope_;
|
||||||
|
|
||||||
GtkWidget* dialog_;
|
GtkWidget* dialog_;
|
||||||
|
|
||||||
SaveDialogCallback save_callback_;
|
SaveDialogCallback save_callback_;
|
||||||
OpenDialogCallback open_callback_;
|
OpenDialogCallback open_callback_;
|
||||||
|
|
||||||
scoped_ptr<atom::NativeWindow::DialogScope> dialog_scope_;
|
|
||||||
|
|
||||||
DISALLOW_COPY_AND_ASSIGN(FileChooserDialog);
|
DISALLOW_COPY_AND_ASSIGN(FileChooserDialog);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -4,11 +4,14 @@
|
||||||
|
|
||||||
#include "atom/browser/ui/message_box.h"
|
#include "atom/browser/ui/message_box.h"
|
||||||
|
|
||||||
#include <gtk/gtk.h>
|
|
||||||
|
|
||||||
#include "atom/browser/browser.h"
|
#include "atom/browser/browser.h"
|
||||||
|
#include "atom/browser/native_window.h"
|
||||||
#include "base/callback.h"
|
#include "base/callback.h"
|
||||||
|
#include "base/strings/string_util.h"
|
||||||
#include "base/strings/utf_string_conversions.h"
|
#include "base/strings/utf_string_conversions.h"
|
||||||
|
#include "chrome/browser/ui/libgtk2ui/gtk2_signal.h"
|
||||||
|
#include "chrome/browser/ui/libgtk2ui/gtk2_util.h"
|
||||||
|
#include "chrome/browser/ui/libgtk2ui/skia_utils_gtk2.h"
|
||||||
|
|
||||||
#define ANSI_FOREGROUND_RED "\x1b[31m"
|
#define ANSI_FOREGROUND_RED "\x1b[31m"
|
||||||
#define ANSI_FOREGROUND_BLACK "\x1b[30m"
|
#define ANSI_FOREGROUND_BLACK "\x1b[30m"
|
||||||
|
@ -18,17 +21,135 @@
|
||||||
|
|
||||||
namespace atom {
|
namespace atom {
|
||||||
|
|
||||||
int ShowMessageBox(NativeWindow* parent_window,
|
namespace {
|
||||||
|
|
||||||
|
class GtkMessageBox {
|
||||||
|
public:
|
||||||
|
GtkMessageBox(bool synchronous,
|
||||||
|
NativeWindow* parent_window,
|
||||||
|
MessageBoxType type,
|
||||||
|
const std::vector<std::string>& buttons,
|
||||||
|
const std::string& title,
|
||||||
|
const std::string& message,
|
||||||
|
const std::string& detail,
|
||||||
|
const gfx::ImageSkia& icon)
|
||||||
|
: dialog_scope_(parent_window),
|
||||||
|
cancel_id_(0) {
|
||||||
|
// The dialog should be modal when showed synchronous.
|
||||||
|
int flags = GTK_DIALOG_NO_SEPARATOR;
|
||||||
|
if (synchronous)
|
||||||
|
flags |= GTK_DIALOG_MODAL;
|
||||||
|
|
||||||
|
// Create dialog.
|
||||||
|
dialog_ = gtk_message_dialog_new(
|
||||||
|
nullptr, // parent
|
||||||
|
static_cast<GtkDialogFlags>(flags), // flags
|
||||||
|
GTK_MESSAGE_OTHER, // no icon
|
||||||
|
GTK_BUTTONS_NONE, // no buttons
|
||||||
|
"%s", message.c_str());
|
||||||
|
gtk_message_dialog_format_secondary_text(
|
||||||
|
GTK_MESSAGE_DIALOG(dialog_),
|
||||||
|
"%s", detail.empty() ? nullptr : detail.c_str());
|
||||||
|
|
||||||
|
// Set dialog's icon.
|
||||||
|
if (!icon.isNull()) {
|
||||||
|
GdkPixbuf* pixbuf = libgtk2ui::GdkPixbufFromSkBitmap(*icon.bitmap());
|
||||||
|
GtkWidget* image = gtk_image_new_from_pixbuf(pixbuf);
|
||||||
|
gtk_message_dialog_set_image(GTK_MESSAGE_DIALOG(dialog_), image);
|
||||||
|
gtk_widget_show(image);
|
||||||
|
g_object_unref(pixbuf);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add buttons.
|
||||||
|
for (size_t i = 0; i < buttons.size(); ++i) {
|
||||||
|
gtk_dialog_add_button(GTK_DIALOG(dialog_),
|
||||||
|
TranslateToStock(i, buttons[i]),
|
||||||
|
i);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parent window.
|
||||||
|
if (parent_window) {
|
||||||
|
gfx::NativeWindow window = parent_window->GetNativeWindow();
|
||||||
|
libgtk2ui::SetGtkTransientForAura(dialog_, window);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
~GtkMessageBox() {
|
||||||
|
gtk_widget_destroy(dialog_);
|
||||||
|
}
|
||||||
|
|
||||||
|
const char* TranslateToStock(int id, const std::string& text) {
|
||||||
|
std::string lower = base::StringToLowerASCII(text);
|
||||||
|
if (lower == "cancel") {
|
||||||
|
cancel_id_ = id;
|
||||||
|
return GTK_STOCK_CANCEL;
|
||||||
|
} else if (lower == "no") {
|
||||||
|
cancel_id_ = id;
|
||||||
|
return GTK_STOCK_NO;
|
||||||
|
} else if (lower == "ok") {
|
||||||
|
return GTK_STOCK_OK;
|
||||||
|
} else if (lower == "yes") {
|
||||||
|
return GTK_STOCK_YES;
|
||||||
|
} else {
|
||||||
|
return text.c_str();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
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);
|
||||||
|
gtk_widget_show_all(dialog_);
|
||||||
|
}
|
||||||
|
|
||||||
|
CHROMEGTK_CALLBACK_1(GtkMessageBox, void, OnResponseDialog, int);
|
||||||
|
|
||||||
|
GtkWidget* dialog() const { return dialog_; }
|
||||||
|
int cancel_id() const { return cancel_id_; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
atom::NativeWindow::DialogScope dialog_scope_;
|
||||||
|
|
||||||
|
// The id to return when the dialog is closed without pressing buttons.
|
||||||
|
int cancel_id_;
|
||||||
|
|
||||||
|
GtkWidget* dialog_;
|
||||||
|
MessageBoxCallback callback_;
|
||||||
|
|
||||||
|
DISALLOW_COPY_AND_ASSIGN(GtkMessageBox);
|
||||||
|
};
|
||||||
|
|
||||||
|
void GtkMessageBox::OnResponseDialog(GtkWidget* widget, int response) {
|
||||||
|
gtk_widget_hide_all(dialog_);
|
||||||
|
|
||||||
|
if (response < 0)
|
||||||
|
callback_.Run(cancel_id_);
|
||||||
|
else
|
||||||
|
callback_.Run(response);
|
||||||
|
delete this;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
int ShowMessageBox(NativeWindow* parent,
|
||||||
MessageBoxType type,
|
MessageBoxType type,
|
||||||
const std::vector<std::string>& buttons,
|
const std::vector<std::string>& buttons,
|
||||||
const std::string& title,
|
const std::string& title,
|
||||||
const std::string& message,
|
const std::string& message,
|
||||||
const std::string& detail,
|
const std::string& detail,
|
||||||
const gfx::ImageSkia& icon) {
|
const gfx::ImageSkia& icon) {
|
||||||
return 0;
|
GtkMessageBox box(true, parent, type, buttons, title, message, detail, icon);
|
||||||
|
gtk_widget_show_all(box.dialog());
|
||||||
|
int response = gtk_dialog_run(GTK_DIALOG(box.dialog()));
|
||||||
|
if (response < 0)
|
||||||
|
return box.cancel_id();
|
||||||
|
else
|
||||||
|
return response;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ShowMessageBox(NativeWindow* parent_window,
|
void ShowMessageBox(NativeWindow* parent,
|
||||||
MessageBoxType type,
|
MessageBoxType type,
|
||||||
const std::vector<std::string>& buttons,
|
const std::vector<std::string>& buttons,
|
||||||
const std::string& title,
|
const std::string& title,
|
||||||
|
@ -36,7 +157,9 @@ void ShowMessageBox(NativeWindow* parent_window,
|
||||||
const std::string& detail,
|
const std::string& detail,
|
||||||
const gfx::ImageSkia& icon,
|
const gfx::ImageSkia& icon,
|
||||||
const MessageBoxCallback& callback) {
|
const MessageBoxCallback& callback) {
|
||||||
callback.Run(0);
|
auto box = new GtkMessageBox(
|
||||||
|
false, parent, type, buttons, title, message, detail, icon);
|
||||||
|
box->RunAsynchronous(callback);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ShowErrorBox(const base::string16& title, const base::string16& content) {
|
void ShowErrorBox(const base::string16& title, const base::string16& content) {
|
||||||
|
|
Loading…
Reference in a new issue