diff options
Diffstat (limited to 'lib/ephy-gui.c')
-rw-r--r-- | lib/ephy-gui.c | 352 |
1 files changed, 352 insertions, 0 deletions
diff --git a/lib/ephy-gui.c b/lib/ephy-gui.c new file mode 100644 index 000000000..fe4018d38 --- /dev/null +++ b/lib/ephy-gui.c @@ -0,0 +1,352 @@ +/* + * Copyright (C) 2002 Marco Pesenti Gritti + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2, or (at your option) + * any later version. + * + * 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 General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + */ + +#include "ephy-gui.h" +#include "eel-gconf-extensions.h" + +#include <ctype.h> +#include <string.h> +#include <libgnome/gnome-i18n.h> +#include <gtk/gtktreemodel.h> + +/* Styles for tab labels */ +GtkStyle *loading_text_style = NULL; +GtkStyle *new_text_style = NULL; + +/** + * gul_gui_menu_position_under_widget: + */ +void +ephy_gui_menu_position_under_widget (GtkMenu *menu, + gint *x, + gint *y, + gboolean *push_in, + gpointer user_data) +{ + GtkWidget *w = GTK_WIDGET (user_data); + gint width, height; + gint screen_width, screen_height; + GtkRequisition requisition; + + gdk_drawable_get_size (w->window, &width, &height); + gdk_window_get_origin (w->window, x, y); + *y = *y + height; + + gtk_widget_size_request (GTK_WIDGET (menu), &requisition); + + screen_width = gdk_screen_width (); + screen_height = gdk_screen_height (); + + *x = CLAMP (*x, 0, MAX (0, screen_width - requisition.width)); + *y = CLAMP (*y, 0, MAX (0, screen_height - requisition.height)); +} + +/** + * gul_gui_gtk_radio_button_get: get the active member of a radiobutton + * group from one of the buttons in the group. This should be in GTK+! + */ +gint +ephy_gui_gtk_radio_button_get (GtkRadioButton *radio_button) +{ + GtkToggleButton *toggle_button; + gint i, length; + GSList *list; + + /* get group list */ + list = gtk_radio_button_get_group (GTK_RADIO_BUTTON (radio_button)); + length = g_slist_length (list); + + /* iterate over list to find active button */ + for (i = 0; list != NULL; i++, list = g_slist_next (list)) + { + /* get button and text */ + toggle_button = GTK_TOGGLE_BUTTON (list->data); + if (gtk_toggle_button_get_active (toggle_button)) + { + break; + } + } + + /* check we didn't run off end */ + g_assert (list != NULL); + + /* return index (reverse order!) */ + return (length - 1) - i; +} + +/** + * gul_gui_gtk_radio_button_set: set the active member of a radiobutton + * group from one of the buttons in the group. This should be in GTK+! + */ +void +ephy_gui_gtk_radio_button_set (GtkRadioButton *radio_button, gint index) +{ + GtkToggleButton *button; + GSList *list; + gint length; + + /* get the list */ + list = gtk_radio_button_get_group (GTK_RADIO_BUTTON (radio_button)); + + /* check out the length */ + length = g_slist_length (list); + + /* new buttons are *preppended* to the list, so button added as first + * has last position in the list */ + index = (length - 1) - index; + + /* find the right button */ + button = GTK_TOGGLE_BUTTON (g_slist_nth_data (list, index)); + + /* set it... this will de-activate the others in the group */ + if (gtk_toggle_button_get_active (button) == FALSE) + { + gtk_toggle_button_set_active (button, TRUE); + } +} + +GtkWidget * +ephy_gui_append_new_menuitem (GtkWidget *menu, + const char *mnemonic, + GCallback callback, + gpointer data) +{ + GtkWidget *menu_item; + + menu_item = gtk_menu_item_new_with_mnemonic (mnemonic); + gtk_widget_show (menu_item); + gtk_menu_shell_append (GTK_MENU_SHELL (menu), + menu_item); + + if (callback) + { + g_signal_connect (G_OBJECT (menu_item), + "activate", + callback, data); + } + + return menu_item; +} + +GtkWidget * +ephy_gui_append_new_menuitem_stock (GtkWidget *menu, + const char *stock_id, + GCallback callback, + gpointer data) +{ + GtkWidget *menu_item; + + menu_item = gtk_image_menu_item_new_from_stock (stock_id, NULL); + gtk_widget_show (menu_item); + gtk_menu_shell_append (GTK_MENU_SHELL (menu), + menu_item); + + if (callback) + { + g_signal_connect (G_OBJECT (menu_item), + "activate", + callback, data); + } + + return menu_item; +} + +GtkWidget * +ephy_gui_append_new_menuitem_stock_icon (GtkWidget *menu, + const char *stock_id, + const char *mnemonic, + GCallback callback, + gpointer data) +{ + GtkWidget *menu_item; + GtkWidget *image; + + menu_item = gtk_image_menu_item_new_with_mnemonic (mnemonic); + image = gtk_image_new_from_stock (stock_id, GTK_ICON_SIZE_MENU); + gtk_widget_show (image); + gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (menu_item), image); + + gtk_widget_show (menu_item); + gtk_menu_shell_append (GTK_MENU_SHELL (menu), + menu_item); + + if (callback) + { + g_signal_connect (G_OBJECT (menu_item), + "activate", + callback, data); + } + + return menu_item; +} + +GtkWidget * +ephy_gui_append_new_check_menuitem (GtkWidget *menu, + const char *mnemonic, + gboolean value, + GCallback callback, + gpointer data) +{ + GtkWidget *menu_item; + + menu_item = gtk_check_menu_item_new_with_mnemonic (mnemonic); + gtk_widget_show (menu_item); + gtk_menu_shell_append (GTK_MENU_SHELL (menu), + menu_item); + + gtk_check_menu_item_set_active (GTK_CHECK_MENU_ITEM (menu_item), value); + + if (callback) + { + g_signal_connect (G_OBJECT (menu_item), + "activate", + callback, data); + } + + return menu_item; +} + +GtkWidget * +ephy_gui_append_separator (GtkWidget *menu) +{ + GtkWidget *menu_item; + + menu_item = gtk_menu_item_new (); + gtk_widget_show (menu_item); + gtk_menu_shell_append (GTK_MENU_SHELL (menu), + menu_item); + + return menu_item; +} + +gboolean +ephy_gui_confirm_overwrite_file (GtkWidget *parent, const char *filename) +{ + char *question; + GtkWidget *dialog; + gboolean res; + + if (!g_file_test (filename, G_FILE_TEST_EXISTS)) + { + return TRUE; + } + + question = g_strdup_printf (_("File %s will be overwritten.\n" + "If you choose yes, the contents will be lost.\n\n" + "Do you want to continue?"), filename); + dialog = gtk_message_dialog_new (parent ? GTK_WINDOW(parent) : NULL, + GTK_DIALOG_MODAL, + GTK_MESSAGE_QUESTION, + GTK_BUTTONS_YES_NO, + question); + res = (gtk_dialog_run (GTK_DIALOG (dialog)) == GTK_RESPONSE_YES); + gtk_widget_destroy (dialog); + g_free (question); + + return res; +} + +static guint32 +shift_color_component (guchar component, float shift_by) +{ + guint32 result; + if (shift_by > 1.0) { + result = component * (2 - shift_by); + } else { + result = 0xff - shift_by * (0xff - component); + } + + return result & 0xff; +} + +/** + * ephy_gui_rgb_shift_color + * @color: A color. + * @shift_by: darken or lighten factor. + * Returns: An darkened or lightened rgb value. + * + * Darkens (@shift_by > 1) or lightens (@shift_by < 1) + * @color. + */ +guint32 +ephy_gui_rgb_shift_color (guint32 color, float shift_by) +{ + guint32 result; + + /* shift red by shift_by */ + result = shift_color_component((color & 0x00ff0000) >> 16, shift_by); + result <<= 8; + /* shift green by shift_by */ + result |= shift_color_component((color & 0x0000ff00) >> 8, shift_by); + result <<= 8; + /* shift blue by shift_by */ + result |= shift_color_component((color & 0x000000ff), shift_by); + + /* alpha doesn't change */ + result |= (0xff000000 & color); + + return result; +} + +static guint32 +rgb16_to_rgb (gushort r, gushort g, gushort b) +{ + guint32 result; + + result = (0xff0000 | (r & 0xff00)); + result <<= 8; + result |= ((g & 0xff00) | (b >> 8)); + + return result; +} + +/** + * ephy_gui_gdk_color_to_rgb + * @color: A GdkColor style color. + * Returns: An rgb value. + * + * Converts from a GdkColor stlye color to a gdk_rgb one. + * Alpha gets set to fully opaque + */ +guint32 +ephy_gui_gdk_color_to_rgb (const GdkColor *color) +{ + return rgb16_to_rgb (color->red, color->green, color->blue); +} + +/** + * ephy_gui_rgb_to_color + * @color: a gdk_rgb style value. + * + * Converts from a gdk_rgb value style to a GdkColor one. + * The gdk_rgb color alpha channel is ignored. + * + * Return value: A GdkColor structure version of the given RGB color. + */ +GdkColor +ephy_gui_gdk_rgb_to_color (guint32 color) +{ + GdkColor result; + + result.red = ((color >> 16) & 0xFF) * 0x101; + result.green = ((color >> 8) & 0xFF) * 0x101; + result.blue = (color & 0xff) * 0x101; + result.pixel = 0; + + return result; +} |