/*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) version 3.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with the program; if not, see <http://www.gnu.org/licenses/>
*
*
* Authors:
* Michael Zucchi <notzed@ximian.com>
* Jonathon Jongsma <jonathon.jongsma@collabora.co.uk>
*
* Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
* Copyright (C) 2009 Intel Corporation
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <glib/gi18n.h>
#include "e-alert-dialog.h"
#define E_ALERT_DIALOG_GET_PRIVATE(obj) \
(G_TYPE_INSTANCE_GET_PRIVATE \
((obj), E_TYPE_ALERT_DIALOG, EAlertDialogPrivate))
struct _EAlertDialogPrivate {
GtkWidget *content_area; /* not referenced */
EAlert *alert;
};
enum {
PROP_0,
PROP_ALERT
};
G_DEFINE_TYPE (
EAlertDialog,
e_alert_dialog,
GTK_TYPE_DIALOG)
static void
alert_dialog_set_alert (EAlertDialog *dialog,
EAlert *alert)
{
g_return_if_fail (E_IS_ALERT (alert));
g_return_if_fail (dialog->priv->alert == NULL);
dialog->priv->alert = g_object_ref (alert);
}
static void
alert_dialog_set_property (GObject *object,
guint property_id,
const GValue *value,
GParamSpec *pspec)
{
switch (property_id) {
case PROP_ALERT:
alert_dialog_set_alert (
E_ALERT_DIALOG (object),
g_value_get_object (value));
return;
}
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}
static void
alert_dialog_get_property (GObject *object,
guint property_id,
GValue *value,
GParamSpec *pspec)
{
switch (property_id) {
case PROP_ALERT:
g_value_set_object (
value, e_alert_dialog_get_alert (
E_ALERT_DIALOG (object)));
return;
}
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}
static void
alert_dialog_dispose (GObject *object)
{
EAlertDialogPrivate *priv;
priv = E_ALERT_DIALOG_GET_PRIVATE (object);
if (priv->alert) {
g_signal_handlers_disconnect_matched (
priv->alert, G_SIGNAL_MATCH_DATA,
0, 0, NULL, NULL, object);
g_object_unref (priv->alert);
priv->alert = NULL;
}
/* Chain up to parent's dispose() method. */
G_OBJECT_CLASS (e_alert_dialog_parent_class)->dispose (object);
}
static void
alert_dialog_constructed (GObject *object)
{
EAlert *alert;
EAlertDialog *dialog;
GtkWidget *action_area;
GtkWidget *content_area;
GtkWidget *container;
GtkWidget *widget;
PangoAttribute *attr;
PangoAttrList *list;
GList *actions;
const gchar *primary, *secondary;
gint default_response;
/* Chain up to parent's constructed() method. */
G_OBJECT_CLASS (e_alert_dialog_parent_class)->constructed (object);
dialog = E_ALERT_DIALOG (object);
alert = e_alert_dialog_get_alert (dialog);
default_response = e_alert_get_default_response (alert);
gtk_window_set_title (GTK_WINDOW (dialog), " ");
/* XXX Making the window non-resizable is the only way at
* present for GTK+ to pick a reasonable default size.
* See https://bugzilla.gnome.org/681937 for details. */
gtk_window_set_resizable (GTK_WINDOW (dialog), FALSE);
action_area = gtk_dialog_get_action_area (GTK_DIALOG (dialog));
content_area = gtk_dialog_get_content_area (GTK_DIALOG (dialog));
gtk_widget_ensure_style (GTK_WIDGET (dialog));
gtk_container_set_border_width (GTK_CONTAINER (action_area), 12);
gtk_container_set_border_width (GTK_CONTAINER (content_area), 0);
gtk_window_set_destroy_with_parent (GTK_WINDOW (dialog), TRUE);
/* Forward EAlert::response signals to GtkDialog::response. */
g_signal_connect_swapped (
alert, "response",
G_CALLBACK (gtk_dialog_response), dialog);
/* Add buttons from actions. */
actions = e_alert_peek_actions (alert);
if (!actions) {
GtkAction *action;
/* Make sure there is at least one action,
* thus the dialog can be closed. */
action = gtk_action_new (
"alert-response-0", _("_Dismiss"), NULL, NULL);
e_alert_add_action (alert, action, GTK_RESPONSE_CLOSE);
g_object_unref (action);
actions = e_alert_peek_actions (alert);
}
while (actions != NULL) {
GtkWidget *button;
gpointer data;
/* These actions are already wired to trigger an
* EAlert::response signal when activated, which
* will in turn call to gtk_dialog_response(),
* so we can add buttons directly to the action
* area without knowing their response IDs.
* (XXX Well, kind of. See below.) */
button = gtk_button_new ();
gtk_widget_set_can_default (button, TRUE);
gtk_activatable_set_related_action (
GTK_ACTIVATABLE (button),
GTK_ACTION (actions->data));
gtk_box_pack_end (
GTK_BOX (action_area),
button, FALSE, FALSE, 0);
/* This is set in e_alert_add_action(). */
data = g_object_get_data (
actions->data, "e-alert-response-id");
/* Normally GtkDialog sets the initial focus widget to
* the button corresponding to the default response, but
* because the buttons are not directly tied to response
* IDs, we have set both the default widget and the
* initial focus widget ourselves. */
if (GPOINTER_TO_INT (data) == default_response) {
gtk_widget_grab_default (button);
gtk_widget_grab_focus (button);
}
actions = g_list_next (actions);
}
widget = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 12);
gtk_container_set_border_width (GTK_CONTAINER (widget), 12);
gtk_box_pack_start (GTK_BOX (content_area), widget, FALSE, FALSE, 0);
gtk_widget_show (widget);
container = widget;
widget = e_alert_create_image (alert, GTK_ICON_SIZE_DIALOG);
gtk_misc_set_alignment (GTK_MISC (widget), 0.0, 0.0);
gtk_box_pack_start (GTK_BOX (container), widget, FALSE, FALSE, 0);
gtk_widget_show (widget);
widget = gtk_box_new (GTK_ORIENTATION_VERTICAL, 12);
gtk_box_pack_start (GTK_BOX (container), widget, FALSE, FALSE, 0);
dialog->priv->content_area = widget;
gtk_widget_show (widget);
container = widget;
primary = e_alert_get_primary_text (alert);
secondary = e_alert_get_secondary_text (alert);
list = pango_attr_list_new ();
attr = pango_attr_scale_new (PANGO_SCALE_LARGE);
pango_attr_list_insert (list, attr);
attr = pango_attr_weight_new (PANGO_WEIGHT_BOLD);
pango_attr_list_insert (list, attr);
widget = gtk_label_new (primary);
gtk_label_set_attributes (GTK_LABEL (widget), list);
gtk_label_set_line_wrap (GTK_LABEL (widget), TRUE);
gtk_label_set_selectable (GTK_LABEL (widget), TRUE);
gtk_misc_set_alignment (GTK_MISC (widget), 0.0, 0.0);
gtk_box_pack_start (GTK_BOX (container), widget, FALSE, FALSE, 0);
gtk_widget_set_can_focus (widget, FALSE);
gtk_widget_show (widget);
widget = gtk_label_new (secondary);
gtk_label_set_line_wrap (GTK_LABEL (widget), TRUE);
gtk_label_set_selectable (GTK_LABEL (widget), TRUE);
gtk_misc_set_alignment (GTK_MISC (widget), 0.0, 0.0);
gtk_box_pack_start (GTK_BOX (container), widget, FALSE, FALSE, 0);
gtk_widget_set_can_focus (widget, FALSE);
gtk_widget_show (widget);
pango_attr_list_unref (list);
}
static void
e_alert_dialog_class_init (EAlertDialogClass *class)
{
GObjectClass *object_class;
g_type_class_add_private (class, sizeof (EAlertDialogPrivate));
object_class = G_OBJECT_CLASS (class);
object_class->set_property = alert_dialog_set_property;
object_class->get_property = alert_dialog_get_property;
object_class->dispose = alert_dialog_dispose;
object_class->constructed = alert_dialog_constructed;
g_object_class_install_property (
object_class,
PROP_ALERT,
g_param_spec_object (
"alert",
"Alert",
"Alert to be displayed",
E_TYPE_ALERT,
G_PARAM_READWRITE |
G_PARAM_CONSTRUCT_ONLY |
G_PARAM_STATIC_STRINGS));
}
static void
e_alert_dialog_init (EAlertDialog *dialog)
{
dialog->priv = E_ALERT_DIALOG_GET_PRIVATE (dialog);
}
GtkWidget *
e_alert_dialog_new (GtkWindow *parent,
EAlert *alert)
{
g_return_val_if_fail (E_IS_ALERT (alert), NULL);
return g_object_new (
E_TYPE_ALERT_DIALOG,
"alert", alert, "transient-for", parent, NULL);
}
GtkWidget *
e_alert_dialog_new_for_args (GtkWindow *parent,
const gchar *tag,
...)
{
GtkWidget *dialog;
EAlert *alert;
va_list ap;
g_return_val_if_fail (tag != NULL, NULL);
va_start (ap, tag);
alert = e_alert_new_valist (tag, ap);
va_end (ap);
dialog = e_alert_dialog_new (parent, alert);
g_object_unref (alert);
return dialog;
}
static gboolean
dialog_focus_in_event_cb (GtkWindow *dialog,
GdkEvent *event,
GtkWindow *parent)
{
gtk_window_set_urgency_hint (parent, FALSE);
return FALSE;
}
gint
e_alert_run_dialog (GtkWindow *parent,
EAlert *alert)
{
GtkWidget *dialog;
gint response;
gulong signal_id = 0;
g_return_val_if_fail (E_IS_ALERT (alert), 0);
dialog = e_alert_dialog_new (parent, alert);
if (parent != NULL) {
gtk_window_set_urgency_hint (parent, TRUE);
signal_id = g_signal_connect (
dialog, "focus-in-event",
G_CALLBACK (dialog_focus_in_event_cb), parent);
} else {
gtk_window_set_urgency_hint (GTK_WINDOW (dialog), TRUE);
}
response = gtk_dialog_run (GTK_DIALOG (dialog));
if (signal_id > 0) {
gtk_window_set_urgency_hint (parent, FALSE);
g_signal_handler_disconnect (dialog, signal_id);
}
gtk_widget_destroy (dialog);
return response;
}
gint
e_alert_run_dialog_for_args (GtkWindow *parent,
const gchar *tag,
...)
{
EAlert *alert;
gint response;
va_list ap;
g_return_val_if_fail (tag != NULL, 0);
va_start (ap, tag);
alert = e_alert_new_valist (tag, ap);
va_end (ap);
response = e_alert_run_dialog (parent, alert);
g_object_unref (alert);
return response;
}
/**
* e_alert_dialog_get_alert:
* @dialog: an #EAlertDialog
*
* Returns the #EAlert associated with @dialog.
*
* Returns: the #EAlert associated with @dialog
**/
EAlert *
e_alert_dialog_get_alert (EAlertDialog *dialog)
{
g_return_val_if_fail (E_IS_ALERT_DIALOG (dialog), NULL);
return dialog->priv->alert;
}
/**
* e_alert_dialog_get_content_area:
* @dialog: an #EAlertDialog
*
* Returns the vertical box containing the primary and secondary labels.
* Use this to pack additional widgets into the dialog with the proper
* horizontal alignment (maintaining the left margin below the image).
*
* Returns: the content area #GtkBox
**/
GtkWidget *
e_alert_dialog_get_content_area (EAlertDialog *dialog)
{
g_return_val_if_fail (E_IS_ALERT_DIALOG (dialog), NULL);
return dialog->priv->content_area;
}