midori/src/sokoke.c

499 lines
16 KiB
C
Raw Normal View History

2007-12-16 22:20:24 +00:00
/*
Copyright (C) 2007-2008 Christian Dywan <christian@twotoasts.de>
2007-12-16 22:20:24 +00:00
This library 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.1 of the License, or (at your option) any later version.
See the file COPYING for the full license text.
*/
#include "sokoke.h"
#include "search.h"
2008-04-20 04:41:26 +00:00
#include "config.h"
#include "main.h"
2007-12-16 22:20:24 +00:00
2008-06-01 13:15:38 +00:00
#ifdef HAVE_UNISTD_H
#include <unistd.h>
#endif
2007-12-16 22:20:24 +00:00
#include <string.h>
#include <gdk/gdkkeysyms.h>
#include <glib/gi18n.h>
2008-04-20 04:41:26 +00:00
#include <glib/gprintf.h>
gchar*
sokoke_magic_uri (const gchar* uri, const gchar* default_search_uri)
{
// Add file:// if we have a local path
if (g_path_is_absolute (uri))
return g_strconcat ("file://", uri, NULL);
// Construct an absolute path if the file is relative
if (g_file_test (uri, G_FILE_TEST_EXISTS) && g_file_test (uri, G_FILE_TEST_IS_REGULAR))
{
gchar* current_dir = g_get_current_dir ();
gchar* result = g_strconcat ("file://", current_dir, G_DIR_SEPARATOR_S, uri, NULL);
g_free (current_dir);
return result;
}
// Do we need to add a protocol?
if (!strstr (uri, "://"))
{
// Do we have a domain, ip address or localhost?
if (strchr (uri, '.') != NULL || !strcmp (uri, "localhost"))
return g_strconcat ("http://", uri, NULL);
// We don't want to search? So return early.
if (!default_search_uri)
return g_strdup (uri);
gchar* search;
const gchar* search_uri = NULL;
// Do we have a keyword and a string?
gchar** parts = g_strsplit (uri, " ", 2);
if (parts[0] && parts[1])
{
guint n = g_list_length (searchEngines);
guint i;
for (i = 0; i < n; i++)
{
SearchEngine* search_engine = (SearchEngine*)g_list_nth_data (
searchEngines, i);
if (!strcmp (search_engine_get_keyword (search_engine),
parts[0]))
search_uri = search_engine->url;
}
if (search_uri)
search = g_strdup_printf (search_uri, parts[1]);
}
// We only have a word or there is no matching keyword, so search for it
if (!search_uri)
search = g_strdup_printf (default_search_uri, uri);
return search;
}
return g_strdup (uri);
}
void
sokoke_entry_setup_completion (GtkEntry* entry)
{
/* TODO: The current behavior works only with the beginning of strings
But we want to match "localhost" with "loc" and "hos" */
GtkEntryCompletion* completion = gtk_entry_completion_new ();
gtk_entry_completion_set_model (completion,
GTK_TREE_MODEL (gtk_list_store_new (1, G_TYPE_STRING)));
gtk_entry_completion_set_text_column (completion, 0);
gtk_entry_completion_set_minimum_key_length (completion, 3);
gtk_entry_set_completion (entry, completion);
gtk_entry_completion_set_popup_completion (completion, FALSE); //...
}
void
sokoke_entry_append_completion (GtkEntry* entry, const gchar* text)
{
GtkEntryCompletion* completion = gtk_entry_get_completion (entry);
GtkTreeModel* completion_store = gtk_entry_completion_get_model (completion);
GtkTreeIter iter;
gtk_list_store_insert (GTK_LIST_STORE (completion_store), &iter, 0);
gtk_list_store_set (GTK_LIST_STORE (completion_store), &iter, 0, text, -1);
}
2008-06-01 15:39:57 +00:00
void
sokoke_combo_box_add_strings (GtkComboBox* combobox,
const gchar* label_first, ...)
2007-12-16 22:20:24 +00:00
{
// Add a number of strings to a combobox, terminated with NULL
// This works only for text comboboxes
va_list args;
2008-06-01 15:39:57 +00:00
va_start (args, label_first);
2007-12-16 22:20:24 +00:00
const gchar* label;
2008-06-01 15:39:57 +00:00
for (label = label_first; label; label = va_arg (args, const gchar*))
gtk_combo_box_append_text (combobox, label);
2007-12-16 22:20:24 +00:00
2008-06-01 15:39:57 +00:00
va_end (args);
2007-12-16 22:20:24 +00:00
}
2008-06-01 15:39:57 +00:00
void sokoke_widget_set_visible (GtkWidget* widget, gboolean visible)
2007-12-16 22:20:24 +00:00
{
// Show or hide the widget
2008-06-01 15:39:57 +00:00
if (visible)
gtk_widget_show (widget);
2007-12-16 22:20:24 +00:00
else
2008-06-01 15:39:57 +00:00
gtk_widget_hide (widget);
2007-12-16 22:20:24 +00:00
}
2008-06-01 15:39:57 +00:00
void
sokoke_container_show_children (GtkContainer* container)
2007-12-16 22:20:24 +00:00
{
// Show every child but not the container itself
2008-06-01 15:39:57 +00:00
gtk_container_foreach (container, (GtkCallback)(gtk_widget_show_all), NULL);
2007-12-16 22:20:24 +00:00
}
2008-06-01 15:39:57 +00:00
void
sokoke_widget_set_tooltip_text (GtkWidget* widget, const gchar* text)
2007-12-16 22:20:24 +00:00
{
#if GTK_CHECK_VERSION(2, 12, 0)
2008-06-01 15:39:57 +00:00
gtk_widget_set_tooltip_text (widget, text);
2007-12-16 22:20:24 +00:00
#else
static GtkTooltips* tooltips;
2008-06-01 15:39:57 +00:00
if (!tooltips)
tooltips = gtk_tooltips_new ();
gtk_tooltips_set_tip (tooltips, widget, text, NULL);
2007-12-16 22:20:24 +00:00
#endif
}
void
sokoke_tool_item_set_tooltip_text (GtkToolItem* toolitem, const gchar* text)
2007-12-16 22:20:24 +00:00
{
if (text && *text)
2008-06-01 15:39:57 +00:00
{
#if GTK_CHECK_VERSION(2, 12, 0)
gtk_tool_item_set_tooltip_text (toolitem, text);
#else
static GtkTooltips* tooltips = NULL;
if (G_UNLIKELY (!tooltips))
tooltips = gtk_tooltips_new();
gtk_tool_item_set_tooltip (toolitem, tooltips, text, NULL);
2008-06-01 15:39:57 +00:00
#endif
}
2007-12-16 22:20:24 +00:00
}
typedef struct
{
GtkWidget* widget;
SokokeMenuPos position;
} SokokePopupInfo;
static void
sokoke_widget_popup_position_menu (GtkMenu* menu,
gint* x,
gint* y,
gboolean* push_in,
gpointer user_data)
{
gint wx, wy;
gint menu_width;
GtkRequisition menu_req;
GtkRequisition widget_req;
SokokePopupInfo* info = user_data;
GtkWidget* widget = info->widget;
// Retrieve size and position of both widget and menu
if (GTK_WIDGET_NO_WINDOW (widget))
{
gdk_window_get_position (widget->window, &wx, &wy);
wx += widget->allocation.x;
wy += widget->allocation.y;
}
else
gdk_window_get_origin (widget->window, &wx, &wy);
gtk_widget_size_request (GTK_WIDGET (menu), &menu_req);
gtk_widget_size_request (widget, &widget_req);
menu_width = menu_req.width;
gint widget_height = widget_req.height; // Better than allocation.height
// Calculate menu position
if (info->position == SOKOKE_MENU_POSITION_CURSOR)
; // Do nothing?
else if (info->position == SOKOKE_MENU_POSITION_RIGHT)
{
*x = wx + widget->allocation.width - menu_width;
*y = wy + widget_height;
} else if (info->position == SOKOKE_MENU_POSITION_LEFT)
{
*x = wx;
*y = wy + widget_height;
}
*push_in = TRUE;
}
void
sokoke_widget_popup (GtkWidget* widget,
GtkMenu* menu,
GdkEventButton* event,
SokokeMenuPos pos)
2007-12-16 22:20:24 +00:00
{
int button, event_time;
if (event)
2007-12-16 22:20:24 +00:00
{
button = event->button;
event_time = event->time;
}
else
{
button = 0;
event_time = gtk_get_current_event_time ();
2007-12-16 22:20:24 +00:00
}
if (!gtk_menu_get_attach_widget(menu))
gtk_menu_attach_to_widget (menu, widget, NULL);
if (widget)
{
SokokePopupInfo info = { widget, pos };
gtk_menu_popup (menu, NULL, NULL,
sokoke_widget_popup_position_menu, &info,
button, event_time);
}
else
gtk_menu_popup (menu, NULL, NULL, NULL, NULL, button, event_time);
2007-12-16 22:20:24 +00:00
}
typedef enum
2007-12-16 22:20:24 +00:00
{
SOKOKE_DESKTOP_UNTESTED,
SOKOKE_DESKTOP_XFCE,
SOKOKE_DESKTOP_UNKNOWN
} SokokeDesktop;
2007-12-16 22:20:24 +00:00
2008-06-01 15:39:57 +00:00
static SokokeDesktop
sokoke_get_desktop (void)
2007-12-16 22:20:24 +00:00
{
static SokokeDesktop desktop = SOKOKE_DESKTOP_UNTESTED;
2008-06-01 15:39:57 +00:00
if (G_UNLIKELY (desktop == SOKOKE_DESKTOP_UNTESTED))
{
// Are we running in Xfce?
gint result; gchar* out; gchar* err;
2008-06-01 15:39:57 +00:00
gboolean success = g_spawn_command_line_sync (
"xprop -root _DT_SAVE_MODE | grep -q xfce4",
&out, &err, &result, NULL);
if (success && !result)
desktop = SOKOKE_DESKTOP_XFCE;
else
desktop = SOKOKE_DESKTOP_UNKNOWN;
}
return desktop;
2007-12-16 22:20:24 +00:00
}
2008-06-01 15:39:57 +00:00
GtkWidget*
sokoke_xfce_header_new (const gchar* icon,
const gchar* title)
2007-12-16 22:20:24 +00:00
{
// Create an xfce header with icon and title
// This returns NULL if the desktop is not xfce
2008-06-01 15:39:57 +00:00
if (sokoke_get_desktop () == SOKOKE_DESKTOP_XFCE)
2007-12-16 22:20:24 +00:00
{
2008-06-01 15:39:57 +00:00
GtkWidget* entry = gtk_entry_new ();
gchar* markup;
2008-06-01 15:39:57 +00:00
GtkWidget* xfce_heading = gtk_event_box_new ();
gtk_widget_modify_bg (xfce_heading, GTK_STATE_NORMAL,
&entry->style->base[GTK_STATE_NORMAL]);
GtkWidget* hbox = gtk_hbox_new (FALSE, 12);
gtk_container_set_border_width (GTK_CONTAINER (hbox), 6);
GtkWidget* image = gtk_image_new_from_icon_name (icon,
GTK_ICON_SIZE_DIALOG);
gtk_box_pack_start (GTK_BOX (hbox), image, FALSE, FALSE, 0);
GtkWidget* label = gtk_label_new (NULL);
gtk_widget_modify_fg (label, GTK_STATE_NORMAL
2007-12-16 22:20:24 +00:00
, &entry->style->text[GTK_STATE_NORMAL]);
2008-06-01 15:39:57 +00:00
markup = g_strdup_printf ("<span size='large' weight='bold'>%s</span>",
title);
gtk_label_set_markup (GTK_LABEL (label), markup);
gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, FALSE, 0);
gtk_container_add (GTK_CONTAINER (xfce_heading), hbox);
g_free (markup);
2007-12-16 22:20:24 +00:00
return xfce_heading;
}
return NULL;
}
2008-06-01 15:39:57 +00:00
GtkWidget*
2008-06-01 13:15:38 +00:00
sokoke_superuser_warning_new (void)
2007-12-16 22:20:24 +00:00
{
// Create a horizontal bar with a security warning
// This returns NULL if the user is no superuser
#ifdef HAVE_UNISTD_H
2008-06-01 13:15:38 +00:00
if (G_UNLIKELY (!geteuid ())) // effective superuser?
2007-12-16 22:20:24 +00:00
{
2008-06-01 13:15:38 +00:00
GtkWidget* hbox = gtk_event_box_new ();
gtk_widget_modify_bg (hbox, GTK_STATE_NORMAL,
&hbox->style->bg[GTK_STATE_SELECTED]);
GtkWidget* label = gtk_label_new (
_("Warning: You are using a superuser account!"));
gtk_misc_set_padding (GTK_MISC (label), 0, 2);
gtk_widget_modify_fg (GTK_WIDGET (label), GTK_STATE_NORMAL,
&GTK_WIDGET (label)->style->fg[GTK_STATE_SELECTED]);
gtk_widget_show (label);
gtk_container_add (GTK_CONTAINER(hbox), GTK_WIDGET (label));
gtk_widget_show (hbox);
2007-12-16 22:20:24 +00:00
return hbox;
}
#endif
return NULL;
}
2008-06-01 15:39:57 +00:00
GtkWidget*
sokoke_hig_frame_new (const gchar* title)
2007-12-16 22:20:24 +00:00
{
// Create a frame with no actual frame but a bold label and indentation
2008-06-01 15:39:57 +00:00
GtkWidget* frame = gtk_frame_new (NULL);
gchar* title_bold = g_strdup_printf ("<b>%s</b>", title);
GtkWidget* label = gtk_label_new (NULL);
gtk_label_set_markup (GTK_LABEL (label), title_bold);
g_free (title_bold);
gtk_frame_set_label_widget (GTK_FRAME (frame), label);
gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_NONE);
2007-12-16 22:20:24 +00:00
return frame;
}
2008-06-01 15:39:57 +00:00
void
sokoke_widget_set_pango_font_style (GtkWidget* widget,
PangoStyle style)
2007-12-16 22:20:24 +00:00
{
// Conveniently change the pango font style
// For some reason we need to reset if we actually want the normal style
2008-06-01 15:39:57 +00:00
if (style == PANGO_STYLE_NORMAL)
gtk_widget_modify_font (widget, NULL);
2007-12-16 22:20:24 +00:00
else
{
2008-06-01 15:39:57 +00:00
PangoFontDescription* font_description = pango_font_description_new ();
pango_font_description_set_style (font_description, PANGO_STYLE_ITALIC);
gtk_widget_modify_font (widget, font_description);
pango_font_description_free (font_description);
2007-12-16 22:20:24 +00:00
}
}
static gboolean
sokoke_on_entry_focus_in_event (GtkEntry* entry,
GdkEventFocus* event,
gpointer userdata)
2007-12-16 22:20:24 +00:00
{
gint default_text = GPOINTER_TO_INT (
g_object_get_data (G_OBJECT (entry), "sokoke_has_default"));
if (default_text)
2007-12-16 22:20:24 +00:00
{
gtk_entry_set_text (entry, "");
g_object_set_data (G_OBJECT(entry), "sokoke_has_default",
GINT_TO_POINTER (0));
sokoke_widget_set_pango_font_style (GTK_WIDGET (entry),
PANGO_STYLE_NORMAL);
2007-12-16 22:20:24 +00:00
}
return FALSE;
}
static gboolean
sokoke_on_entry_focus_out_event (GtkEntry* entry,
GdkEventFocus* event,
gpointer userdata)
2007-12-16 22:20:24 +00:00
{
const gchar* text = gtk_entry_get_text (entry);
if (text && !*text)
2007-12-16 22:20:24 +00:00
{
const gchar* defaultText = (const gchar*)g_object_get_data (
G_OBJECT (entry), "sokoke_default_text");
gtk_entry_set_text (entry, defaultText);
g_object_set_data (G_OBJECT(entry),
"sokoke_has_default", GINT_TO_POINTER (1));
2008-06-01 15:39:57 +00:00
sokoke_widget_set_pango_font_style (GTK_WIDGET (entry),
PANGO_STYLE_ITALIC);
2007-12-16 22:20:24 +00:00
}
return FALSE;
}
void
sokoke_entry_set_default_text (GtkEntry* entry,
const gchar* default_text)
2007-12-16 22:20:24 +00:00
{
// Note: The default text initially overwrites any previous text
gchar* old_value = g_object_get_data (G_OBJECT (entry),
"sokoke_default_text");
if (!old_value)
2007-12-16 22:20:24 +00:00
{
g_object_set_data (G_OBJECT (entry), "sokoke_has_default",
GINT_TO_POINTER (1));
sokoke_widget_set_pango_font_style (GTK_WIDGET (entry),
PANGO_STYLE_ITALIC);
gtk_entry_set_text (entry, default_text);
2007-12-16 22:20:24 +00:00
}
g_object_set_data (G_OBJECT (entry), "sokoke_default_text",
(gpointer)default_text);
g_signal_connect (entry, "focus-in-event",
G_CALLBACK (sokoke_on_entry_focus_in_event), NULL);
g_signal_connect (entry, "focus-out-event",
G_CALLBACK (sokoke_on_entry_focus_out_event), NULL);
2007-12-16 22:20:24 +00:00
}
2008-06-01 15:39:57 +00:00
gchar*
sokoke_key_file_get_string_default (GKeyFile* key_file,
const gchar* group,
const gchar* key,
const gchar* default_value,
GError** error)
2007-12-16 22:20:24 +00:00
{
2008-06-01 15:39:57 +00:00
gchar* value = g_key_file_get_string (key_file, group, key, error);
return value == NULL ? g_strdup (default_value) : value;
2007-12-16 22:20:24 +00:00
}
2008-06-01 15:39:57 +00:00
gint
sokoke_key_file_get_integer_default (GKeyFile* key_file,
const gchar* group,
const gchar* key,
const gint default_value,
GError** error)
2007-12-16 22:20:24 +00:00
{
2008-06-01 15:39:57 +00:00
if (!g_key_file_has_key (key_file, group, key, NULL))
return default_value;
return g_key_file_get_integer (key_file, group, key, error);
2007-12-16 22:20:24 +00:00
}
gdouble
sokoke_key_file_get_double_default (GKeyFile* key_file,
const gchar* group,
const gchar* key,
const gdouble default_value,
GError** error)
{
if (!g_key_file_has_key (key_file, group, key, NULL))
return default_value;
return g_key_file_get_double (key_file, group, key, error);
}
gboolean
sokoke_key_file_get_boolean_default (GKeyFile* key_file,
const gchar* group,
const gchar* key,
const gboolean default_value,
GError** error)
{
if (!g_key_file_has_key (key_file, group, key, NULL))
return default_value;
return g_key_file_get_boolean (key_file, group, key, error);
}
2008-06-01 15:39:57 +00:00
gboolean
sokoke_key_file_save_to_file (GKeyFile* key_file,
const gchar* filename,
GError** error)
2007-12-16 22:20:24 +00:00
{
2008-06-01 15:39:57 +00:00
gchar* data = g_key_file_to_data (key_file, NULL, error);
if (!data)
2007-12-16 22:20:24 +00:00
return FALSE;
FILE* fp;
2008-06-01 15:39:57 +00:00
if (!(fp = fopen (filename, "w")))
2007-12-16 22:20:24 +00:00
{
2008-06-01 15:39:57 +00:00
*error = g_error_new (G_FILE_ERROR, G_FILE_ERROR_ACCES,
_("Writing failed."));
2007-12-16 22:20:24 +00:00
return FALSE;
}
2008-06-01 15:39:57 +00:00
fputs (data, fp);
fclose (fp);
g_free (data);
2007-12-16 22:20:24 +00:00
return TRUE;
}
2008-06-01 15:39:57 +00:00
void
sokoke_widget_get_text_size (GtkWidget* widget,
const gchar* text,
gint* width,
gint* height)
2007-12-16 22:20:24 +00:00
{
2008-06-01 15:39:57 +00:00
PangoLayout* layout = gtk_widget_create_pango_layout (widget, text);
pango_layout_get_pixel_size (layout, width, height);
g_object_unref (layout);
2007-12-16 22:20:24 +00:00
}