/*
* 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.
*
* $Id$
*/
#include "config.h"
#include "ephy-gui.h"
#include "eel-gconf-extensions.h"
#include "ephy-debug.h"
#include <ctype.h>
#include <string.h>
#include <glib/gi18n.h>
#include <gdk/gdkx.h>
#include <libgnome/gnome-help.h>
#include <gtk/gtktreemodel.h>
#include <gtk/gtkmessagedialog.h>
#include <gtk/gtkhbox.h>
#include <gtk/gtkimage.h>
#include <gtk/gtklabel.h>
#include <gtk/gtkstock.h>
#include <gtk/gtkmain.h>
#include <gtk/gtktreeselection.h>
#include <gtk/gtktoolbar.h>
#include <unistd.h>
void
ephy_gui_sanitise_popup_position (GtkMenu *menu,
GtkWidget *widget,
gint *x,
gint *y)
{
GdkScreen *screen = gtk_widget_get_screen (widget);
gint monitor_num;
GdkRectangle monitor;
GtkRequisition req;
g_return_if_fail (widget != NULL);
gtk_widget_size_request (GTK_WIDGET (menu), &req);
monitor_num = gdk_screen_get_monitor_at_point (screen, *x, *y);
gtk_menu_set_monitor (menu, monitor_num);
gdk_screen_get_monitor_geometry (screen, monitor_num, &monitor);
*x = CLAMP (*x, monitor.x, monitor.x + MAX (0, monitor.width - req.width));
*y = CLAMP (*y, monitor.y, monitor.y + MAX (0, monitor.height - req.height));
}
void
ephy_gui_menu_position_tree_selection (GtkMenu *menu,
gint *x,
gint *y,
gboolean *push_in,
gpointer user_data)
{
GtkTreeSelection *selection;
GList *selected_rows;
GtkTreeModel *model;
GtkTreeView *tree_view = GTK_TREE_VIEW (user_data);
GtkWidget *widget = GTK_WIDGET (user_data);
GtkRequisition req;
GdkRectangle visible;
gtk_widget_size_request (GTK_WIDGET (menu), &req);
gdk_window_get_origin (widget->window, x, y);
*x += (widget->allocation.width - req.width) / 2;
/* Add on height for the treeview title */
gtk_tree_view_get_visible_rect (tree_view, &visible);
*y += widget->allocation.height - visible.height;
selection = gtk_tree_view_get_selection (tree_view);
selected_rows = gtk_tree_selection_get_selected_rows (selection, &model);
if (selected_rows)
{
GdkRectangle cell_rect;
gtk_tree_view_get_cell_area (tree_view, selected_rows->data,
NULL, &cell_rect);
*y += CLAMP (cell_rect.y + cell_rect.height, 0, visible.height);
g_list_foreach (selected_rows, (GFunc)gtk_tree_path_free, NULL);
g_list_free (selected_rows);
}
ephy_gui_sanitise_popup_position (menu, widget, x, y);
}
/**
* ephy_gui_menu_position_under_widget:
* @menu:
* @x:
* @y:
* @push_in:
* @user_data: a #GtkWidget
*
* Positions @menu under (or over, depending on space available) the widget
* @user_data.
*/
void
ephy_gui_menu_position_under_widget (GtkMenu *menu,
gint *x,
gint *y,
gboolean *push_in,
gpointer user_data)
{
/* Adapted from gtktoolbar.c */
GtkWidget *widget = GTK_WIDGET (user_data);
GtkWidget *container;
GtkRequisition req;
GtkRequisition menu_req;
GdkRectangle monitor;
int monitor_num;
GdkScreen *screen;
g_return_if_fail (GTK_IS_WIDGET (widget));
container = gtk_widget_get_ancestor (widget, GTK_TYPE_CONTAINER);
g_return_if_fail (container != NULL);
gtk_widget_size_request (widget, &req);
gtk_widget_size_request (GTK_WIDGET (menu), &menu_req);
screen = gtk_widget_get_screen (GTK_WIDGET (menu));
monitor_num = gdk_screen_get_monitor_at_window (screen, widget->window);
if (monitor_num < 0)
monitor_num = 0;
gdk_screen_get_monitor_geometry (screen, monitor_num, &monitor);
gdk_window_get_origin (widget->window, x, y);
if (GTK_WIDGET_NO_WINDOW (widget))
{
*x += widget->allocation.x;
*y += widget->allocation.y;
}
if (gtk_widget_get_direction (container) == GTK_TEXT_DIR_LTR)
*x += widget->allocation.width - req.width;
else
*x += req.width - menu_req.width;
if ((*y + widget->allocation.height + menu_req.height) <= monitor.y + monitor.height)
*y += widget->allocation.height;
else if ((*y - menu_req.height) >= monitor.y)
*y -= menu_req.height;
else if (monitor.y + monitor.height - (*y + widget->allocation.height) > *y)
*y += widget->allocation.height;
else
*y -= menu_req.height;
*push_in = FALSE;
}
/**
* ephy_gui_menu_position_under_widget:
* @menu:
* @x:
* @y:
* @push_in:
* @user_data: a #GtkWidget which has to be contained in (a widget on) a #GtkToolbar
*
* Positions @menu under (or over, depending on space available) the
* @user_data.
*/
void
ephy_gui_menu_position_on_toolbar (GtkMenu *menu,
gint *x,
gint *y,
gboolean *push_in,
gpointer user_data)
{
/* Adapted from gtktoolbar.c */
GtkWidget *widget = GTK_WIDGET (user_data);
GtkToolbar *toolbar;
GtkRequisition req;
GtkRequisition menu_req;
GdkRectangle monitor;
int monitor_num;
GdkScreen *screen;
g_return_if_fail (GTK_IS_WIDGET (widget));
toolbar = GTK_TOOLBAR (gtk_widget_get_ancestor (widget, GTK_TYPE_TOOLBAR));
g_return_if_fail (toolbar != NULL);
gtk_widget_size_request (widget, &req);
gtk_widget_size_request (GTK_WIDGET (menu), &menu_req);
screen = gtk_widget_get_screen (GTK_WIDGET (menu));
monitor_num = gdk_screen_get_monitor_at_window (screen, widget->window);
if (monitor_num < 0)
monitor_num = 0;
gdk_screen_get_monitor_geometry (screen, monitor_num, &monitor);
gdk_window_get_origin (widget->window, x, y);
if (GTK_WIDGET_NO_WINDOW (widget))
{
*x += widget->allocation.x;
*y += widget->allocation.y;
}
if (toolbar->orientation == GTK_ORIENTATION_HORIZONTAL)
{
if (gtk_widget_get_direction (GTK_WIDGET (toolbar)) == GTK_TEXT_DIR_LTR)
*x += widget->allocation.width - req.width;
else
*x += req.width - menu_req.width;
if ((*y + widget->allocation.height + menu_req.height) <= monitor.y + monitor.height)
*y += widget->allocation.height;
else if ((*y - menu_req.height) >= monitor.y)
*y -= menu_req.height;
else if (monitor.y + monitor.height - (*y + widget->allocation.height) > *y)
*y += widget->allocation.height;
else
*y -= menu_req.height;
}
else
{
if (gtk_widget_get_direction (GTK_WIDGET (toolbar)) == GTK_TEXT_DIR_LTR)
*x += widget->allocation.width;
else
*x -= menu_req.width;
if (*y + menu_req.height > monitor.y + monitor.height &&
*y + widget->allocation.height - monitor.y > monitor.y + monitor.height - *y)
*y += widget->allocation.height - menu_req.height;
}
*push_in = FALSE;
}
void
ephy_gui_menu_position_on_panel (GtkMenu *menu,
gint *x,
gint *y,
gboolean *push_in,
gpointer user_data)
{
GtkWidget *widget = GTK_WIDGET (user_data);
GtkRequisition requisition;
GdkScreen *screen;
screen = gtk_widget_get_screen (widget);
gdk_window_get_origin (widget->window, x, y);
gtk_widget_size_request (GTK_WIDGET (menu), &requisition);
if (GTK_WIDGET_NO_WINDOW (widget))
{
*x += widget->allocation.x;
*y += widget->allocation.y;
}
/* FIXME: Adapt to vertical panels, but egg_tray_icon_get_orientation doesn't seem to work */
if (*y > gdk_screen_get_height (screen) / 2)
{
*y -= requisition.height;
}
else
{
*y += widget->allocation.height;
}
*push_in = FALSE;
ephy_gui_sanitise_popup_position (menu, widget, x, y);
}
GtkWindowGroup *
ephy_gui_ensure_window_group (GtkWindow *window)
{
GtkWindowGroup *group;
group = window->group;
if (group == NULL)
{
group = gtk_window_group_new ();
gtk_window_group_add_window (group, window);
g_object_unref (group);
}
return group;
}
gboolean
ephy_gui_check_location_writable (GtkWidget *parent,
const char *filename)
{
GtkWidget *dialog;
char *display_name;
if (filename == NULL) return FALSE;
if (!g_file_test (filename, G_FILE_TEST_EXISTS))
{
char *path = g_path_get_dirname (filename);
gboolean writable = access (path, W_OK) == 0;
/* check if path is writable to */
if (!writable)
{
dialog = gtk_message_dialog_new (
parent ? GTK_WINDOW(parent) : NULL,
GTK_DIALOG_DESTROY_WITH_PARENT,
GTK_MESSAGE_ERROR,
GTK_BUTTONS_CLOSE,
_("Directory “%s” is not writable"), path);
gtk_message_dialog_format_secondary_text (
GTK_MESSAGE_DIALOG (dialog),
_("You do not have permission to "
"create files in this directory."));
gtk_window_set_title (GTK_WINDOW (dialog), _("Directory not Writable"));
gtk_window_set_icon_name (GTK_WINDOW (dialog), "web-browser");
if (parent != NULL)
{
gtk_window_group_add_window (
ephy_gui_ensure_window_group (GTK_WINDOW (parent)),
GTK_WINDOW (dialog));
}
gtk_dialog_run (GTK_DIALOG (dialog));
gtk_widget_destroy (dialog);
}
g_free (path);
return writable;
}
display_name = g_filename_display_basename (filename);
/* check if file is writable */
if (access (filename, W_OK) != 0)
{
dialog = gtk_message_dialog_new (
parent ? GTK_WINDOW(parent) : NULL,
GTK_DIALOG_DESTROY_WITH_PARENT,
GTK_MESSAGE_ERROR,
GTK_BUTTONS_CLOSE,
_("Cannot overwrite existing file “%s”"), display_name);
gtk_message_dialog_format_secondary_text (
GTK_MESSAGE_DIALOG (dialog),
_("A file with this name already exists and "
"you don't have permission to overwrite it."));
gtk_window_set_title (GTK_WINDOW (dialog), _("Cannot Overwrite File"));
gtk_window_set_icon_name (GTK_WINDOW (dialog), "web-browser");
if (parent != NULL)
{
gtk_window_group_add_window (
ephy_gui_ensure_window_group (GTK_WINDOW (parent)),
GTK_WINDOW (dialog));
}
gtk_dialog_run (GTK_DIALOG (dialog));
gtk_widget_destroy (dialog);
return FALSE;
}
return TRUE;
}
void
ephy_gui_help (GtkWindow *parent,
const char *file_name,
const char *link_id)
{
GError *err = NULL;
gnome_help_display (file_name, link_id, &err);
if (err != NULL)
{
GtkWidget *dialog;
dialog = gtk_message_dialog_new (parent,
GTK_DIALOG_DESTROY_WITH_PARENT,
GTK_MESSAGE_ERROR,
GTK_BUTTONS_OK,
_("Could not display help: %s"), err->message);
g_signal_connect (G_OBJECT (dialog), "response",
G_CALLBACK (gtk_widget_destroy),
NULL);
gtk_dialog_set_has_separator (GTK_DIALOG (dialog), FALSE);
gtk_widget_show (dialog);
g_error_free (err);
}
}
void
ephy_gui_get_current_event (GdkEventType *otype,
guint *ostate,
guint *obutton)
{
GdkEvent *event;
GdkEventType type = GDK_NOTHING;
guint state = 0, button = (guint) -1;
event = gtk_get_current_event ();
if (event != NULL)
{
type = event->type;
if (type == GDK_KEY_PRESS ||
type == GDK_KEY_RELEASE)
{
state = event->key.state;
}
else if (type == GDK_BUTTON_PRESS ||
type == GDK_BUTTON_RELEASE ||
type == GDK_2BUTTON_PRESS ||
type == GDK_3BUTTON_PRESS)
{
button = event->button.button;
state = event->button.state;
}
gdk_event_free (event);
}
if (otype) *otype = type;
if (ostate) *ostate = state & gtk_accelerator_get_default_mod_mask ();
if (obutton) *obutton = button;
}
gboolean
ephy_gui_is_middle_click (void)
{
gboolean is_middle_click = FALSE;
GdkEvent *event;
event = gtk_get_current_event ();
if (event != NULL)
{
if (event->type == GDK_BUTTON_RELEASE)
{
guint modifiers, button, state;
modifiers = gtk_accelerator_get_default_mod_mask ();
button = event->button.button;
state = event->button.state;
/* middle-click or control-click */
if ((button == 1 && ((state & modifiers) == GDK_CONTROL_MASK)) ||
(button == 2 && ((state & modifiers) == 0)))
{
is_middle_click = TRUE;
}
}
gdk_event_free (event);
}
return is_middle_click;
}
void
ephy_gui_window_update_user_time (GtkWidget *window,
guint32 user_time)
{
LOG ("updating user time on window %p to %d", window, user_time);
if (user_time != 0)
{
gtk_widget_realize (window);
gdk_x11_window_set_user_time (window->window,
user_time);
}
}
/* gtk+ bug 166379 */
/* adapted from gtk+/gtk/gtkwindow.c */
void
ephy_gui_window_present (GtkWindow *window,
guint32 user_time)
{
GtkWidget *widget;
g_return_if_fail (GTK_IS_WINDOW (window));
widget = GTK_WIDGET (window);
if (GTK_WIDGET_VISIBLE (window))
{
g_assert (widget->window != NULL);
gdk_window_show (widget->window);
/* note that gdk_window_focus() will also move the window to
* the current desktop, for WM spec compliant window managers.
*/
gdk_window_focus (widget->window, user_time);
}
else
{
gtk_widget_show (widget);
}
}
/* Pending gtk+ bug http://bugzilla.gnome.org/show_bug.cgi?id=328069 */
GtkWidget *
ephy_gui_message_dialog_get_content_box (GtkWidget *dialog)
{
GtkWidget *container;
GList *children;
/* Get the hbox which is the first child of the main vbox */
children = gtk_container_get_children (GTK_CONTAINER (GTK_DIALOG (dialog)->vbox));
g_return_val_if_fail (children != NULL, NULL);
container = GTK_WIDGET (children->data);
g_list_free (children);
/* Get the vbox which is the second child of the hbox */
children = gtk_container_get_children (GTK_CONTAINER (container));
g_return_val_if_fail (children != NULL && children->next != NULL, NULL);
container = GTK_WIDGET (children->next->data);
g_list_free (children);
return container;
}