// Copyright (c) 2011 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/chromeos/native_dialog_window.h"
#include <gtk/gtk.h>
#include "base/logging.h"
#include "base/utf_string_conversions.h"
#include "chrome/browser/chromeos/frame/bubble_window.h"
#include "chrome/browser/ui/views/window.h"
#include "ui/base/gtk/gtk_signal.h"
#include "views/controls/native/native_view_host.h"
#include "views/window/dialog_delegate.h"
#include "views/window/non_client_view.h"
#include "views/window/window.h"
namespace {
const int kDialogPadding = 3;
const char kNativeDialogHost[] = "_chromeos_native_dialog_host_";
// TODO(xiyuan): Use gtk_window_get_default_widget with GTK 2.14+.
// Gets the default widget of given dialog.
GtkWidget* GetDialogDefaultWidget(GtkDialog* dialog) {
GtkWidget* default_widget = NULL;
GList* children = gtk_container_get_children(
GTK_CONTAINER(dialog->action_area));
GList* current = children;
while (current) {
GtkWidget* widget = reinterpret_cast<GtkWidget*>(current->data);
if (GTK_WIDGET_HAS_DEFAULT(widget)) {
default_widget = widget;
break;
}
current = g_list_next(current);
}
g_list_free(children);
return default_widget;
}
} // namespace
namespace chromeos {
class NativeDialogHost : public views::View,
public views::DialogDelegate {
public:
NativeDialogHost(gfx::NativeView native_dialog,
int flags,
const gfx::Size& size,
const gfx::Size& min_size);
~NativeDialogHost();
// views::DialogDelegate implementation:
virtual bool CanResize() const { return flags_ & DIALOG_FLAG_RESIZEABLE; }
virtual int GetDialogButtons() const { return 0; }
virtual std::wstring GetWindowTitle() const { return title_; }
virtual views::View* GetContentsView() { return this; }
virtual bool IsModal() const { return flags_ & DIALOG_FLAG_MODAL; }
virtual void WindowClosing();
protected:
CHROMEGTK_CALLBACK_0(NativeDialogHost, void, OnCheckResize);
CHROMEGTK_CALLBACK_0(NativeDialogHost, void, OnDialogDestroy);
// views::View implementation:
virtual gfx::Size GetPreferredSize();
virtual void Layout();
virtual void ViewHierarchyChanged(bool is_add,
views::View* parent,
views::View* child);
private:
// Init and attach to native dialog.
void Init();
// Check and apply minimum size restriction.
void CheckSize();
// The GtkDialog whose vbox will be displayed in this view.
gfx::NativeView dialog_;
// NativeViewHost for the dialog's contents.
views::NativeViewHost* contents_view_;
std::wstring title_;
int flags_;
gfx::Size size_;
gfx::Size preferred_size_;
gfx::Size min_size_;
int destroy_signal_id_;
DISALLOW_IMPLICIT_CONSTRUCTORS(NativeDialogHost);
};
///////////////////////////////////////////////////////////////////////////////
// NativeDialogHost, public:
NativeDialogHost::NativeDialogHost(gfx::NativeView native_dialog,
int flags,
const gfx::Size& size,
const gfx::Size& min_size)
: dialog_(native_dialog),
contents_view_(NULL),
flags_(flags),
size_(size),
preferred_size_(size),
min_size_(min_size),
destroy_signal_id_(0) {
const char* title = gtk_window_get_title(GTK_WINDOW(dialog_));
if (title)
UTF8ToWide(title, strlen(title), &title_);
destroy_signal_id_ = g_signal_connect(dialog_, "destroy",
G_CALLBACK(&OnDialogDestroyThunk), this);
}
NativeDialogHost::~NativeDialogHost() {
}
void NativeDialogHost::OnCheckResize(GtkWidget* widget) {
// Do auto height resize only when we are asked to do so.
if (size_.height() == 0) {
gfx::NativeView contents = contents_view_->native_view();
// Check whether preferred height has changed. We keep the current width
// unchanged and pass "-1" as height to let gtk calculate a proper height.
gtk_widget_set_size_request(contents, width(), -1);
GtkRequisition requsition = { 0 };
gtk_widget_size_request(contents, &requsition);
if (preferred_size_.height() != requsition.height) {
preferred_size_.set_width(requsition.width);
preferred_size_.set_height(requsition.height);
CheckSize();
SizeToPreferredSize();
gfx::Size window_size = window()->non_client_view()->GetPreferredSize();
gfx::Rect window_bounds = window()->GetBounds();
window_bounds.set_width(window_size.width());
window_bounds.set_height(window_size.height());
window()->SetWindowBounds(window_bounds, NULL);
}
}
}
void NativeDialogHost::OnDialogDestroy(GtkWidget* widget) {
dialog_ = NULL;
destroy_signal_id_ = 0;
window()->CloseWindow();
}
///////////////////////////////////////////////////////////////////////////////
// NativeDialogHost, views::DialogDelegate implementation:
void NativeDialogHost::WindowClosing() {
if (dialog_) {
// Disconnect the "destroy" signal because we are about to destroy
// the dialog ourselves and no longer interested in it.
g_signal_handler_disconnect(G_OBJECT(dialog_), destroy_signal_id_);
gtk_dialog_response(GTK_DIALOG(dialog_), GTK_RESPONSE_DELETE_EVENT);
}
}
///////////////////////////////////////////////////////////////////////////////
// NativeDialogHost, views::View implementation:
gfx::Size NativeDialogHost::GetPreferredSize() {
return preferred_size_;
}
void NativeDialogHost::Layout() {
contents_view_->SetBounds(0, 0, width(), height());
}
void NativeDialogHost::ViewHierarchyChanged(bool is_add,
views::View* parent,
views::View* child) {
if (is_add && child == this)
Init();
}
///////////////////////////////////////////////////////////////////////////////
// NativeDialogHost, private:
void NativeDialogHost::Init() {
if (contents_view_)
return;
// Get default widget of the dialog.
GtkWidget* default_widget = GetDialogDefaultWidget(GTK_DIALOG(dialog_));
// Get focus widget of the dialog.
GtkWidget* focus_widget = gtk_window_get_focus(GTK_WINDOW(dialog_));
// Create a GtkAlignment as dialog contents container.
GtkWidget* contents = gtk_alignment_new(0.5, 0.5, 1.0, 1.0);
gtk_alignment_set_padding(GTK_ALIGNMENT(contents),
kDialogPadding, kDialogPadding,
kDialogPadding, kDialogPadding);
// Move dialog contents into our container.
GtkWidget* dialog_contents = GTK_DIALOG(dialog_)->vbox;
g_object_ref(dialog_contents);
gtk_container_remove(GTK_CONTAINER(dialog_), dialog_contents);
gtk_container_add(GTK_CONTAINER(contents), dialog_contents);
g_object_unref(dialog_contents);
gtk_widget_hide(dialog_);
g_object_set_data(G_OBJECT(dialog_), kNativeDialogHost,
reinterpret_cast<gpointer>(this));
gtk_widget_show_all(contents);
contents_view_ = new views::NativeViewHost();
// TODO(xiyuan): Find a better way to get proper background.
contents_view_->set_background(views::Background::CreateSolidBackground(
BubbleWindow::kBackgroundColor));
AddChildView(contents_view_);
contents_view_->Attach(contents);
g_signal_connect(window()->GetNativeWindow(), "check-resize",
G_CALLBACK(&OnCheckResizeThunk), this);
const int padding = 2 * kDialogPadding;
// Use gtk's default size if size is not specified.
if (size_.IsEmpty()) {
// Use given width or height if given.
if (size_.width() || size_.height()) {
int width = size_.width() == 0 ? -1 : size_.width() + padding;
int height = size_.height() == 0 ? -1 : size_.height() + padding;
gtk_widget_set_size_request(contents, width, height);
}
GtkRequisition requsition = { 0 };
gtk_widget_size_request(contents, &requsition);
preferred_size_.set_width(requsition.width);
preferred_size_.set_height(requsition.height);
} else {
preferred_size_.set_width(size_.width() + padding);
preferred_size_.set_height(size_.height() + padding);
}
CheckSize();
if (default_widget)
gtk_widget_grab_default(default_widget);
if (focus_widget)
gtk_widget_grab_focus(focus_widget);
}
void NativeDialogHost::CheckSize() {
// Apply the minimum size.
if (preferred_size_.width() < min_size_.width())
preferred_size_.set_width(min_size_.width());
if (preferred_size_.height() < min_size_.height())
preferred_size_.set_height(min_size_.height());
}
void ShowNativeDialog(gfx::NativeWindow parent,
gfx::NativeView native_dialog,
int flags,
const gfx::Size& size,
const gfx::Size& min_size) {
NativeDialogHost* native_dialog_host =
new NativeDialogHost(native_dialog, flags, size, min_size);
browser::CreateViewsWindow(parent, gfx::Rect(), native_dialog_host);
native_dialog_host->window()->Show();
}
gfx::NativeWindow GetNativeDialogWindow(gfx::NativeView native_dialog) {
NativeDialogHost* host = reinterpret_cast<NativeDialogHost*>(
g_object_get_data(G_OBJECT(native_dialog), kNativeDialogHost));
return host ? host->window()->GetNativeWindow() : NULL;
}
gfx::Rect GetNativeDialogContentsBounds(gfx::NativeView native_dialog) {
NativeDialogHost* host = reinterpret_cast<NativeDialogHost*>(
g_object_get_data(G_OBJECT(native_dialog), kNativeDialogHost));
return host ? host->bounds() : gfx::Rect();
}
} // namespace chromeos