diff options
author | Cosimo Cecchi <cosimoc@gnome.org> | 2012-01-17 06:07:27 +0800 |
---|---|---|
committer | Cosimo Cecchi <cosimoc@gnome.org> | 2012-01-17 06:25:21 +0800 |
commit | 3e258526ca38fc6dd1c8a8c96f4925531f5031c8 (patch) | |
tree | 3aaabe08ebcbe166fa68d4ad0fc0f9e8fcd4ff7f | |
parent | 6b6d20039c2ce843fbefe0bea34f6dbef785aacd (diff) | |
download | gsoc2013-epiphany-3e258526ca38fc6dd1c8a8c96f4925531f5031c8.tar gsoc2013-epiphany-3e258526ca38fc6dd1c8a8c96f4925531f5031c8.tar.gz gsoc2013-epiphany-3e258526ca38fc6dd1c8a8c96f4925531f5031c8.tar.bz2 gsoc2013-epiphany-3e258526ca38fc6dd1c8a8c96f4925531f5031c8.tar.lz gsoc2013-epiphany-3e258526ca38fc6dd1c8a8c96f4925531f5031c8.tar.xz gsoc2013-epiphany-3e258526ca38fc6dd1c8a8c96f4925531f5031c8.tar.zst gsoc2013-epiphany-3e258526ca38fc6dd1c8a8c96f4925531f5031c8.zip |
floating-bar: import NautilisFloatingBar into Epiphany
Yay for sharing code. It also helps theming.
-rw-r--r-- | embed/ephy-embed.c | 47 | ||||
-rw-r--r-- | src/Makefile.am | 2 | ||||
-rw-r--r-- | src/nautilus-floating-bar.c | 389 | ||||
-rw-r--r-- | src/nautilus-floating-bar.h | 75 |
4 files changed, 476 insertions, 37 deletions
diff --git a/embed/ephy-embed.c b/embed/ephy-embed.c index 62f25581f..c4e9ac5cb 100644 --- a/embed/ephy-embed.c +++ b/embed/ephy-embed.c @@ -74,7 +74,7 @@ struct _EphyEmbedPrivate gboolean inspector_attached; guint is_setting_zoom : 1; GSList *destroy_on_transition_list; - GtkWidget *statusbar_label; + GtkWidget *floating_bar; GtkWidget *progress; GSList *messages; @@ -435,17 +435,13 @@ _ephy_embed_set_statusbar_label (EphyEmbed *embed, const char *label) EphyEmbedPrivate *priv = embed->priv; GtkWidget *parent; - gtk_label_set_label (GTK_LABEL (priv->statusbar_label), label); - - parent = gtk_widget_get_parent (priv->statusbar_label); - if (parent == NULL) - return; + nautilus_floating_bar_set_label (NAUTILUS_FLOATING_BAR (priv->floating_bar), label); if (label == NULL || label[0] == '\0') { - gtk_widget_hide (parent); - gtk_widget_set_halign (gtk_widget_get_parent (parent), GTK_ALIGN_START); + gtk_widget_hide (priv->floating_bar); + gtk_widget_set_halign (priv->floating_bar, GTK_ALIGN_START); } else - gtk_widget_show (parent); + gtk_widget_show (priv->floating_bar); } static gboolean @@ -514,21 +510,6 @@ window_resize_requested (WebKitWebWindowFeatures *features, GParamSpec *pspec, E } static gboolean -frame_enter_notify_cb (GtkWidget *widget, - GdkEventCrossing *event, - gpointer user_data) -{ - if (gtk_widget_get_halign (widget) == GTK_ALIGN_START) - gtk_widget_set_halign (widget, GTK_ALIGN_END); - else - gtk_widget_set_halign (widget, GTK_ALIGN_START); - - gtk_widget_queue_resize (widget); - - return FALSE; -} - -static gboolean clear_progress_cb (EphyEmbed *embed) { gtk_widget_hide (embed->priv->progress); @@ -591,19 +572,11 @@ ephy_embed_constructed (GObject *object) gtk_container_add (GTK_CONTAINER (overlay), scrolled_window); /* statusbar is hidden by default */ - priv->statusbar_label = gtk_label_new (NULL); - eventbox = gtk_event_box_new (); - frame = gtk_frame_new (NULL); - gtk_widget_set_name (frame, "ephy-status-frame"); - gtk_widget_set_halign (eventbox, GTK_ALIGN_START); - gtk_widget_set_valign (eventbox, GTK_ALIGN_END); - gtk_widget_show (eventbox); - - gtk_container_add (GTK_CONTAINER (eventbox), frame); - gtk_container_add (GTK_CONTAINER (frame), priv->statusbar_label); - gtk_overlay_add_overlay (GTK_OVERLAY (overlay), eventbox); - g_signal_connect (eventbox, "enter-notify-event", - G_CALLBACK (frame_enter_notify_cb), object); + priv->floating_bar = nautilus_floating_bar_new (NULL, FALSE); + gtk_widget_set_halign (priv->floating_bar, GTK_ALIGN_START); + gtk_widget_set_valign (priv->floating_bar, GTK_ALIGN_END); + + gtk_overlay_add_overlay (GTK_OVERLAY (overlay), priv->floating_bar); embed->priv->progress = gtk_progress_bar_new (); gtk_widget_set_name (embed->priv->progress, "ephy-progress-bar"); diff --git a/src/Makefile.am b/src/Makefile.am index 24d47ae7d..b6ca3457d 100644 --- a/src/Makefile.am +++ b/src/Makefile.am @@ -31,6 +31,7 @@ NOINST_H_FILES = \ ephy-toolbar.h \ ephy-window-action.h \ languages.h \ + nautilus-floating-bar.h \ pdm-dialog.h \ popup-commands.h \ prefs-dialog.h \ @@ -73,6 +74,7 @@ libephymain_la_SOURCES = \ ephy-toolbar.c \ ephy-window.c \ ephy-window-action.c \ + nautilus-floating-bar.c \ pdm-dialog.c \ popup-commands.c \ prefs-dialog.c \ diff --git a/src/nautilus-floating-bar.c b/src/nautilus-floating-bar.c new file mode 100644 index 000000000..e37a81a27 --- /dev/null +++ b/src/nautilus-floating-bar.c @@ -0,0 +1,389 @@ +/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */ + +/* Nautilus - Floating status bar. + * + * Copyright (C) 2011 Red Hat Inc. + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Library General Public + * License as published by the Free Software Foundation; either + * version 2 of the License, or (at your option) any later version. + * + * This library 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 + * Library General Public License for more details. + * + * You should have received a copy of the GNU Library General Public + * License along with this library; if not, write to the + * Free Software Foundation, Inc., 59 Temple Place - Suite 330, + * Boston, MA 02111-1307, USA. + * + * Authors: Cosimo Cecchi <cosimoc@redhat.com> + * + */ + +#include <config.h> + +#include "nautilus-floating-bar.h" + +struct _NautilusFloatingBarDetails { + gchar *label; + + GtkWidget *label_widget; + GtkWidget *spinner; + gboolean show_spinner; + gboolean is_interactive; +}; + +enum { + PROP_LABEL = 1, + PROP_SHOW_SPINNER, + NUM_PROPERTIES +}; + +enum { + ACTION, + NUM_SIGNALS +}; + +static GParamSpec *properties[NUM_PROPERTIES] = { NULL, }; +static guint signals[NUM_SIGNALS] = { 0, }; + +G_DEFINE_TYPE (NautilusFloatingBar, nautilus_floating_bar, + GTK_TYPE_BOX); + +static void +action_button_clicked_cb (GtkButton *button, + NautilusFloatingBar *self) +{ + gint action_id; + + action_id = GPOINTER_TO_INT + (g_object_get_data (G_OBJECT (button), "action-id")); + + g_signal_emit (self, signals[ACTION], 0, action_id); +} + +static void +nautilus_floating_bar_finalize (GObject *obj) +{ + NautilusFloatingBar *self = NAUTILUS_FLOATING_BAR (obj); + + g_free (self->priv->label); + + G_OBJECT_CLASS (nautilus_floating_bar_parent_class)->finalize (obj); +} + +static void +nautilus_floating_bar_get_property (GObject *object, + guint property_id, + GValue *value, + GParamSpec *pspec) +{ + NautilusFloatingBar *self = NAUTILUS_FLOATING_BAR (object); + + switch (property_id) { + case PROP_LABEL: + g_value_set_string (value, self->priv->label); + break; + case PROP_SHOW_SPINNER: + g_value_set_boolean (value, self->priv->show_spinner); + break; + default: + G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); + break; + } +} + +static void +nautilus_floating_bar_set_property (GObject *object, + guint property_id, + const GValue *value, + GParamSpec *pspec) +{ + NautilusFloatingBar *self = NAUTILUS_FLOATING_BAR (object); + + switch (property_id) { + case PROP_LABEL: + nautilus_floating_bar_set_label (self, g_value_get_string (value)); + break; + case PROP_SHOW_SPINNER: + nautilus_floating_bar_set_show_spinner (self, g_value_get_boolean (value)); + break; + default: + G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); + break; + } +} + +static void +update_label (NautilusFloatingBar *self) +{ + gtk_label_set_text (GTK_LABEL (self->priv->label_widget), self->priv->label); +} + +static gboolean +overlay_enter_notify_cb (GtkWidget *parent, + GdkEventCrossing *event, + gpointer user_data) +{ + GtkWidget *widget = user_data; + + if (event->window != gtk_widget_get_window (widget)) { + return FALSE; + } + + if (NAUTILUS_FLOATING_BAR (widget)->priv->is_interactive) { + return FALSE; + } + + if (gtk_widget_get_halign (widget) == GTK_ALIGN_START) { + gtk_widget_set_halign (widget, GTK_ALIGN_END); + } else { + gtk_widget_set_halign (widget, GTK_ALIGN_START); + } + + gtk_widget_queue_resize (widget); + + return FALSE; +} + +static void +nautilus_floating_bar_parent_set (GtkWidget *widget, + GtkWidget *old_parent) +{ + GtkWidget *parent; + + parent = gtk_widget_get_parent (widget); + + if (old_parent != NULL) { + g_signal_handlers_disconnect_by_func (old_parent, + overlay_enter_notify_cb, widget); + } + + if (parent != NULL) { + g_signal_connect (parent, "enter-notify-event", + G_CALLBACK (overlay_enter_notify_cb), widget); + } +} + +static void +nautilus_floating_bar_show (GtkWidget *widget) +{ + NautilusFloatingBar *self = NAUTILUS_FLOATING_BAR (widget); + + GTK_WIDGET_CLASS (nautilus_floating_bar_parent_class)->show (widget); + + if (self->priv->show_spinner) { + gtk_spinner_start (GTK_SPINNER (self->priv->spinner)); + } +} + +static void +nautilus_floating_bar_hide (GtkWidget *widget) +{ + NautilusFloatingBar *self = NAUTILUS_FLOATING_BAR (widget); + + GTK_WIDGET_CLASS (nautilus_floating_bar_parent_class)->hide (widget); + + gtk_spinner_stop (GTK_SPINNER (self->priv->spinner)); +} + +static gboolean +nautilus_floating_bar_draw (GtkWidget *widget, + cairo_t *cr) +{ + GtkStyleContext *context; + + context = gtk_widget_get_style_context (widget); + + gtk_style_context_save (context); + gtk_style_context_set_state (context, gtk_widget_get_state_flags (widget)); + + gtk_render_background (context, cr, 0, 0, + gtk_widget_get_allocated_width (widget), + gtk_widget_get_allocated_height (widget)); + + gtk_render_frame (context, cr, 0, 0, + gtk_widget_get_allocated_width (widget), + gtk_widget_get_allocated_height (widget)); + + gtk_style_context_restore (context); + + return GTK_WIDGET_CLASS (nautilus_floating_bar_parent_class)->draw (widget, cr);; +} + +static void +nautilus_floating_bar_constructed (GObject *obj) +{ + NautilusFloatingBar *self = NAUTILUS_FLOATING_BAR (obj); + GtkWidget *w, *box; + + G_OBJECT_CLASS (nautilus_floating_bar_parent_class)->constructed (obj); + + box = GTK_WIDGET (obj); + + w = gtk_spinner_new (); + gtk_box_pack_start (GTK_BOX (box), w, FALSE, FALSE, 0); + gtk_widget_set_visible (w, self->priv->show_spinner); + self->priv->spinner = w; + + gtk_widget_set_size_request (w, 16, 16); + gtk_widget_set_margin_left (w, 8); + + w = gtk_label_new (NULL); + gtk_label_set_ellipsize (GTK_LABEL (w), PANGO_ELLIPSIZE_END); + gtk_box_pack_start (GTK_BOX (box), w, FALSE, FALSE, 0); + g_object_set (w, + "margin-top", 2, + "margin-bottom", 2, + "margin-left", 12, + "margin-right", 12, + NULL); + self->priv->label_widget = w; + gtk_widget_show (w); +} + +static void +nautilus_floating_bar_init (NautilusFloatingBar *self) +{ + GtkStyleContext *context; + + self->priv = G_TYPE_INSTANCE_GET_PRIVATE (self, NAUTILUS_TYPE_FLOATING_BAR, + NautilusFloatingBarDetails); + + context = gtk_widget_get_style_context (GTK_WIDGET (self)); + gtk_style_context_add_class (context, "floating-bar"); +} + +static void +nautilus_floating_bar_class_init (NautilusFloatingBarClass *klass) +{ + GObjectClass *oclass = G_OBJECT_CLASS (klass); + GtkWidgetClass *wclass = GTK_WIDGET_CLASS (klass); + + oclass->constructed = nautilus_floating_bar_constructed; + oclass->set_property = nautilus_floating_bar_set_property; + oclass->get_property = nautilus_floating_bar_get_property; + oclass->finalize = nautilus_floating_bar_finalize; + + wclass->draw = nautilus_floating_bar_draw; + wclass->show = nautilus_floating_bar_show; + wclass->hide = nautilus_floating_bar_hide; + wclass->parent_set = nautilus_floating_bar_parent_set; + + properties[PROP_LABEL] = + g_param_spec_string ("label", + "Bar's label", + "Label displayed by the bar", + NULL, + G_PARAM_WRITABLE | G_PARAM_STATIC_STRINGS); + properties[PROP_SHOW_SPINNER] = + g_param_spec_boolean ("show-spinner", + "Show spinner", + "Whether a spinner should be shown in the floating bar", + FALSE, + G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS); + + signals[ACTION] = + g_signal_new ("action", + G_TYPE_FROM_CLASS (klass), + G_SIGNAL_RUN_LAST, + 0, NULL, NULL, + g_cclosure_marshal_VOID__INT, + G_TYPE_NONE, 1, + G_TYPE_INT); + + g_type_class_add_private (klass, sizeof (NautilusFloatingBarDetails)); + g_object_class_install_properties (oclass, NUM_PROPERTIES, properties); +} + +void +nautilus_floating_bar_set_label (NautilusFloatingBar *self, + const gchar *label) +{ + if (g_strcmp0 (self->priv->label, label) != 0) { + g_free (self->priv->label); + self->priv->label = g_strdup (label); + + g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_LABEL]); + + update_label (self); + } +} + +void +nautilus_floating_bar_set_show_spinner (NautilusFloatingBar *self, + gboolean show_spinner) +{ + if (self->priv->show_spinner != show_spinner) { + self->priv->show_spinner = show_spinner; + gtk_widget_set_visible (self->priv->spinner, + show_spinner); + + g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_SHOW_SPINNER]); + } +} + +GtkWidget * +nautilus_floating_bar_new (const gchar *label, + gboolean show_spinner) +{ + return g_object_new (NAUTILUS_TYPE_FLOATING_BAR, + "label", label, + "show-spinner", show_spinner, + "orientation", GTK_ORIENTATION_HORIZONTAL, + "spacing", 8, + NULL); +} + +void +nautilus_floating_bar_add_action (NautilusFloatingBar *self, + const gchar *stock_id, + gint action_id) +{ + GtkWidget *w, *button; + + w = gtk_image_new_from_stock (stock_id, GTK_ICON_SIZE_MENU); + gtk_widget_show (w); + + button = gtk_button_new (); + gtk_button_set_image (GTK_BUTTON (button), w); + gtk_box_pack_end (GTK_BOX (self), button, FALSE, FALSE, 0); + gtk_widget_show (button); + + g_object_set_data (G_OBJECT (button), "action-id", + GINT_TO_POINTER (action_id)); + + g_signal_connect (button, "clicked", + G_CALLBACK (action_button_clicked_cb), self); + + self->priv->is_interactive = TRUE; +} + +void +nautilus_floating_bar_cleanup_actions (NautilusFloatingBar *self) +{ + GtkWidget *widget; + GList *children, *l; + gpointer data; + + children = gtk_container_get_children (GTK_CONTAINER (self)); + l = children; + + while (l != NULL) { + widget = l->data; + data = g_object_get_data (G_OBJECT (widget), "action-id"); + l = l->next; + + if (data != NULL) { + /* destroy this */ + gtk_widget_destroy (widget); + } + } + + g_list_free (children); + + self->priv->is_interactive = FALSE; +} diff --git a/src/nautilus-floating-bar.h b/src/nautilus-floating-bar.h new file mode 100644 index 000000000..03ea0f9e1 --- /dev/null +++ b/src/nautilus-floating-bar.h @@ -0,0 +1,75 @@ +/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */ + +/* Nautilus - Floating status bar. + * + * Copyright (C) 2011 Red Hat Inc. + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Library General Public + * License as published by the Free Software Foundation; either + * version 2 of the License, or (at your option) any later version. + * + * This library 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 + * Library General Public License for more details. + * + * You should have received a copy of the GNU Library General Public + * License along with this library; if not, write to the + * Free Software Foundation, Inc., 59 Temple Place - Suite 330, + * Boston, MA 02111-1307, USA. + * + * Authors: Cosimo Cecchi <cosimoc@redhat.com> + * + */ + +#ifndef __NAUTILUS_FLOATING_BAR_H__ +#define __NAUTILUS_FLOATING_BAR_H__ + +#include <gtk/gtk.h> + +#define NAUTILUS_FLOATING_BAR_ACTION_ID_STOP 1 + +#define NAUTILUS_TYPE_FLOATING_BAR nautilus_floating_bar_get_type() +#define NAUTILUS_FLOATING_BAR(obj) \ + (G_TYPE_CHECK_INSTANCE_CAST ((obj), NAUTILUS_TYPE_FLOATING_BAR, NautilusFloatingBar)) +#define NAUTILUS_FLOATING_BAR_CLASS(klass) \ + (G_TYPE_CHECK_CLASS_CAST ((klass), NAUTILUS_TYPE_FLOATING_BAR, NautilusFloatingBarClass)) +#define NAUTILUS_IS_FLOATING_BAR(obj) \ + (G_TYPE_CHECK_INSTANCE_TYPE ((obj), NAUTILUS_TYPE_FLOATING_BAR)) +#define NAUTILUS_IS_FLOATING_BAR_CLASS(klass) \ + (G_TYPE_CHECK_CLASS_TYPE ((klass), NAUTILUS_TYPE_FLOATING_BAR)) +#define NAUTILUS_FLOATING_BAR_GET_CLASS(obj) \ + (G_TYPE_INSTANCE_GET_CLASS ((obj), NAUTILUS_TYPE_FLOATING_BAR, NautilusFloatingBarClass)) + +typedef struct _NautilusFloatingBar NautilusFloatingBar; +typedef struct _NautilusFloatingBarClass NautilusFloatingBarClass; +typedef struct _NautilusFloatingBarDetails NautilusFloatingBarDetails; + +struct _NautilusFloatingBar { + GtkBox parent; + NautilusFloatingBarDetails *priv; +}; + +struct _NautilusFloatingBarClass { + GtkBoxClass parent_class; +}; + +/* GObject */ +GType nautilus_floating_bar_get_type (void); + +GtkWidget * nautilus_floating_bar_new (const gchar *label, + gboolean show_spinner); + +void nautilus_floating_bar_set_label (NautilusFloatingBar *self, + const gchar *label); +void nautilus_floating_bar_set_show_spinner (NautilusFloatingBar *self, + gboolean show_spinner); + +void nautilus_floating_bar_add_action (NautilusFloatingBar *self, + const gchar *stock_id, + gint action_id); +void nautilus_floating_bar_cleanup_actions (NautilusFloatingBar *self); + +#endif /* __NAUTILUS_FLOATING_BAR_H__ */ + |