aboutsummaryrefslogblamecommitdiffstats
path: root/lib/ephy-gui.c
blob: fe4018d388177cc4b94d9d71aa06f07096168680 (plain) (tree)































































































































































































































































































































































                                                                                       
/*
 *  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;
}