medit/moo/moofileview/moofileicons.c
2007-04-11 16:27:26 -05:00

747 lines
22 KiB
C

/*
* moofileicons.c
*
* Copyright (C) 2004-2007 by Yevgen Muntyan <muntyan@math.tamu.edu>
*
* 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 of the License, or
* (at your option) any later version.
*
* See COPYING file that comes with this distribution.
*/
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
#include "moofileview/moofileicons.h"
#include "moofileview/symlink.h"
#include "mooutils/mooutils-misc.h"
#include "mooutils/moostock.h"
#include "mooutils/xdgmime/xdgmime.h"
#include <gtk/gtk.h>
#include <string.h>
#define DEBUG_MESSAGES 0
#define MOO_ICON_EMBLEM_LEN 2
typedef struct {
GdkPixbuf **special_icons[MOO_ICON_INVALID];
GHashTable *mime_icons;
} MooIconCache;
static MooIconCache *moo_icon_cache_new (void);
static void moo_icon_cache_free (MooIconCache *cache);
static MooIconCache *moo_icon_cache_get_for_screen (GdkScreen *screen,
GtkIconSize size);
static GdkPixbuf *create_special_icon (GtkWidget *widget,
MooIconType type,
GtkIconSize size);
static GdkPixbuf *create_mime_icon (GtkWidget *widget,
const char *mime_type,
GtkIconSize size);
static GdkPixbuf *create_fallback_icon (GtkWidget *widget,
GtkIconSize size);
static GdkPixbuf *add_emblem (GdkPixbuf *original,
MooIconEmblem emblem,
GtkIconSize size);
static GdkPixbuf *get_named_icon (GtkIconTheme *icon_theme,
const char *icon_name,
int pixel_size);
static GdkPixbuf *get_stock_icon (GtkWidget *widget,
const char *stock_id,
GtkIconSize size);
static GdkPixbuf **
pixbuf_array_new (void)
{
return g_new0 (GdkPixbuf*, MOO_ICON_EMBLEM_LEN);
}
static void
pixbuf_array_free (GdkPixbuf **pixbufs)
{
if (pixbufs)
{
guint i;
for (i = 0; i < MOO_ICON_EMBLEM_LEN; ++i)
if (pixbufs[i])
g_object_unref (pixbufs[i]);
g_free (pixbufs);
}
}
static void
pixbuf_array_set (GdkPixbuf **array,
MooIconEmblem flags,
GdkPixbuf *pixbuf)
{
GdkPixbuf *tmp;
tmp = array[flags];
array[flags] = pixbuf;
if (pixbuf)
g_object_ref (pixbuf);
if (tmp)
g_object_unref (tmp);
}
static MooIconCache *
moo_icon_cache_new (void)
{
guint i;
MooIconCache *cache = g_new0 (MooIconCache, 1);
for (i = 0; i < MOO_ICON_INVALID; ++i)
cache->special_icons[i] = NULL;
cache->mime_icons = g_hash_table_new_full (g_str_hash, g_str_equal, g_free,
(GDestroyNotify) pixbuf_array_free);
return cache;
}
static void
moo_icon_cache_free (MooIconCache *cache)
{
if (cache)
{
guint i;
for (i = 0; i < MOO_ICON_INVALID; ++i)
pixbuf_array_free (cache->special_icons[i]);
g_hash_table_destroy (cache->mime_icons);
g_free (cache);
}
}
static GdkPixbuf *
moo_icon_cache_get_special (MooIconCache *cache,
MooIconType type,
MooIconEmblem flags)
{
g_return_val_if_fail (type && type < MOO_ICON_INVALID, NULL);
g_return_val_if_fail (flags < MOO_ICON_EMBLEM_LEN, NULL);
return cache->special_icons[type] ? cache->special_icons[type][flags] : NULL;
}
static GdkPixbuf *
moo_icon_cache_get_mime (MooIconCache *cache,
const char *mime,
MooIconEmblem flags)
{
GdkPixbuf **pixbufs;
pixbufs = g_hash_table_lookup (cache->mime_icons, mime);
return pixbufs ? pixbufs[flags] : NULL;
}
static GdkPixbuf *
moo_icon_cache_get (MooIconCache *cache,
MooIconType type,
const char *mime,
MooIconEmblem flags)
{
g_assert (type < MOO_ICON_INVALID);
g_assert ((type == 0 && mime != NULL) || (type != 0 && mime == NULL));
return mime ? moo_icon_cache_get_mime (cache, mime, flags) :
moo_icon_cache_get_special (cache, type, flags);
}
static void
moo_icon_cache_set_special (MooIconCache *cache,
MooIconType type,
MooIconEmblem flags,
GdkPixbuf *pixbuf)
{
GdkPixbuf *tmp;
g_return_if_fail (type && type < MOO_ICON_INVALID);
g_return_if_fail (flags < MOO_ICON_EMBLEM_LEN);
if (!cache->special_icons[type])
cache->special_icons[type] = pixbuf_array_new ();
tmp = cache->special_icons[type][flags];
cache->special_icons[type][flags] = pixbuf;
if (pixbuf)
g_object_ref (pixbuf);
if (tmp)
g_object_unref (pixbuf);
}
static void
moo_icon_cache_set_mime (MooIconCache *cache,
const char *mime,
MooIconEmblem flags,
GdkPixbuf *pixbuf)
{
GdkPixbuf **array;
array = g_hash_table_lookup (cache->mime_icons, mime);
if (!array)
{
array = pixbuf_array_new ();
g_hash_table_insert (cache->mime_icons, g_strdup (mime), array);
}
pixbuf_array_set (array, flags, pixbuf);
}
static void
moo_icon_cache_set (MooIconCache *cache,
MooIconType type,
const char *mime,
MooIconEmblem flags,
GdkPixbuf *pixbuf)
{
g_assert (type < MOO_ICON_INVALID);
g_assert ((type == 0 && mime != NULL) || (type != 0 && mime == NULL));
if (mime)
moo_icon_cache_set_mime (cache, mime, flags, pixbuf);
else
moo_icon_cache_set_special (cache, type, flags, pixbuf);
}
static MooIconCache *
moo_icon_cache_get_for_screen (GdkScreen *screen,
GtkIconSize size)
{
GHashTable *hash;
MooIconCache *cache;
GtkIconTheme *icon_theme;
icon_theme = gtk_icon_theme_get_for_screen (screen);
g_return_val_if_fail (icon_theme != NULL, NULL);
hash = g_object_get_data (G_OBJECT (icon_theme), "moo-icon-cache");
if (!hash)
{
hash = g_hash_table_new_full (g_direct_hash, g_direct_equal, NULL,
(GDestroyNotify) moo_icon_cache_free);
g_object_set_data_full (G_OBJECT (icon_theme), "moo-icon-cache", hash,
(GDestroyNotify) g_hash_table_destroy);
}
cache = g_hash_table_lookup (hash, GINT_TO_POINTER (size));
if (!cache)
{
cache = moo_icon_cache_new ();
g_hash_table_insert (hash, GINT_TO_POINTER (size), cache);
}
return cache;
}
GdkPixbuf *
_moo_get_icon (GtkWidget *widget,
MooIconType type,
const char *mime_type,
MooIconEmblem emblem,
GtkIconSize size)
{
GdkScreen *screen;
GdkPixbuf *pixbuf;
GdkPixbuf *original;
MooIconCache *cache;
g_return_val_if_fail (GTK_IS_WIDGET (widget), NULL);
g_return_val_if_fail (gtk_widget_has_screen (widget), NULL);
g_return_val_if_fail (type < MOO_ICON_INVALID, NULL);
g_return_val_if_fail (type != MOO_ICON_MIME || mime_type != NULL, NULL);
g_return_val_if_fail (emblem < MOO_ICON_EMBLEM_LEN, NULL);
if (type != MOO_ICON_MIME)
mime_type = NULL;
screen = gtk_widget_get_screen (widget);
cache = moo_icon_cache_get_for_screen (screen, size);
g_return_val_if_fail (cache != NULL, NULL);
pixbuf = moo_icon_cache_get (cache, type, mime_type, emblem);
if (pixbuf)
return pixbuf;
original = NULL;
if (emblem)
original = moo_icon_cache_get (cache, type, mime_type, 0);
if (!original)
{
if (mime_type)
original = create_mime_icon (widget, mime_type, size);
else
original = create_special_icon (widget, type, size);
if (!original)
original = create_fallback_icon (widget, size);
g_return_val_if_fail (original != NULL, NULL);
moo_icon_cache_set (cache, type, mime_type, 0, original);
g_object_unref (original);
}
if (!emblem)
return original;
pixbuf = add_emblem (original, emblem, size);
g_return_val_if_fail (pixbuf != NULL, NULL);
moo_icon_cache_set (cache, type, mime_type, emblem, pixbuf);
g_object_unref (pixbuf);
return pixbuf;
}
static GdkPixbuf *
add_arrow (GdkPixbuf *original,
GtkIconSize size)
{
GdkPixbuf *emblem, *pixbuf;
static GdkPixbuf *arrow = NULL;
static GdkPixbuf *small_arrow = NULL;
if (!arrow)
{
arrow = gdk_pixbuf_new_from_inline (-1, SYMLINK_ARROW, TRUE, NULL);
g_return_val_if_fail (arrow != NULL, g_object_ref (original));
}
if (!small_arrow)
{
small_arrow = gdk_pixbuf_new_from_inline (-1, SYMLINK_ARROW_SMALL, TRUE, NULL);
g_return_val_if_fail (arrow != NULL, g_object_ref (original));
}
if (size == GTK_ICON_SIZE_MENU)
emblem = small_arrow;
else
emblem = arrow;
pixbuf = gdk_pixbuf_copy (original);
g_return_val_if_fail (pixbuf != NULL, g_object_ref (original));
gdk_pixbuf_composite (emblem, pixbuf,
0,
gdk_pixbuf_get_height (pixbuf) - gdk_pixbuf_get_height (emblem),
gdk_pixbuf_get_width (emblem),
gdk_pixbuf_get_height (emblem),
0,
gdk_pixbuf_get_height (pixbuf) - gdk_pixbuf_get_height (emblem),
1, 1,
GDK_INTERP_BILINEAR,
255);
return pixbuf;
}
static GdkPixbuf *
add_emblem (GdkPixbuf *original,
MooIconEmblem flags,
GtkIconSize size)
{
g_assert (flags == MOO_ICON_EMBLEM_LINK);
g_assert (GDK_IS_PIXBUF (original));
return add_arrow (original, size);
}
static void
pixels_from_icon_size (GdkScreen *screen,
GtkIconSize size,
int *widthp,
int *heightp)
{
GtkSettings *settings;
int width, height;
settings = gtk_settings_get_for_screen (screen);
if (!gtk_icon_size_lookup_for_settings (settings, size, &width, &height))
{
g_critical ("%s: invalid icon size", G_STRLOC);
size = GTK_ICON_SIZE_MENU;
gtk_icon_size_lookup_for_settings (settings, size, &width, &height);
}
if (widthp)
*widthp = width;
if (heightp)
*heightp = height;
}
static GdkPixbuf *
get_named_icon (GtkIconTheme *icon_theme,
const char *icon_name,
int pixel_size)
{
GdkPixbuf *pixbuf;
pixbuf = gtk_icon_theme_load_icon (icon_theme,
icon_name,
pixel_size,
GTK_ICON_LOOKUP_USE_BUILTIN,
NULL);
/* happens with crystalsvg for some reason */
if (pixbuf && (gdk_pixbuf_get_width (pixbuf) < 5 ||
gdk_pixbuf_get_height (pixbuf) < 5))
{
if (DEBUG_MESSAGES)
_moo_message ("%s: got zero size icon '%s'", G_STRLOC, icon_name);
g_object_unref (pixbuf);
pixbuf = NULL;
}
return pixbuf;
}
static GdkPixbuf *
get_stock_icon (GtkWidget *widget,
const char *stock_id,
GtkIconSize size)
{
return gtk_widget_render_icon (widget, stock_id, size, NULL);
}
static GdkPixbuf *
create_named_icon (GtkIconTheme *icon_theme,
GtkWidget *widget,
GtkIconSize size,
int pixel_size,
const char *fallback_stock,
...)
{
GdkPixbuf *pixbuf = NULL;
va_list args;
char *name;
va_start (args, fallback_stock);
while (!pixbuf && (name = va_arg (args, char *)))
{
pixbuf = get_named_icon (icon_theme, name, pixel_size);
if (!pixbuf && DEBUG_MESSAGES)
_moo_message ("could not load '%s' icon", name);
}
va_end (args);
if (!pixbuf && fallback_stock)
{
pixbuf = gtk_widget_render_icon (widget, fallback_stock, size, NULL);
if (!pixbuf && DEBUG_MESSAGES)
_moo_message ("could not load stock '%s' icon", fallback_stock);
}
if (!pixbuf)
pixbuf = get_stock_icon (widget, MOO_STOCK_FILE, size);
return pixbuf;
}
static GdkPixbuf *
create_fallback_icon (GtkWidget *widget,
GtkIconSize size)
{
int pixel_size;
GdkScreen *screen;
GtkIconTheme *icon_theme;
screen = gtk_widget_get_screen (widget);
icon_theme = gtk_icon_theme_get_for_screen (screen);
pixels_from_icon_size (screen, size, NULL, &pixel_size);
return create_named_icon (icon_theme, widget, size, pixel_size, NULL, NULL);
}
static GdkPixbuf *
create_broken_link_icon (G_GNUC_UNUSED GtkIconTheme *icon_theme,
GtkWidget *widget,
GtkIconSize size)
{
/* XXX */
return gtk_widget_render_icon (widget, GTK_STOCK_MISSING_IMAGE, size, NULL);
}
static GdkPixbuf *
create_broken_icon (G_GNUC_UNUSED GtkIconTheme *icon_theme,
GtkWidget *widget,
GtkIconSize size)
{
/* XXX */
return gtk_widget_render_icon (widget, GTK_STOCK_MISSING_IMAGE, size, NULL);
}
static GdkPixbuf *
create_special_icon (GtkWidget *widget,
MooIconType type,
GtkIconSize size)
{
int pixel_size;
GdkScreen *screen;
GtkIconTheme *icon_theme;
g_assert (type < MOO_ICON_INVALID);
g_assert (type != MOO_ICON_MIME);
screen = gtk_widget_get_screen (widget);
icon_theme = gtk_icon_theme_get_for_screen (screen);
pixels_from_icon_size (screen, size, NULL, &pixel_size);
switch (type)
{
case MOO_ICON_HOME:
return create_named_icon (icon_theme, widget, size, pixel_size, GTK_STOCK_HOME,
"user-home", "gnome-fs-home", "folder_home", NULL);
case MOO_ICON_DESKTOP:
return create_named_icon (icon_theme, widget, size, pixel_size, GTK_STOCK_DIRECTORY,
"user-desktop", "gnome-fs-desktop", "desktop",
"folder", "gnome-fs-directory", NULL);
case MOO_ICON_TRASH:
return create_named_icon (icon_theme, widget, size, pixel_size, GTK_STOCK_DIRECTORY,
"user-trash", "gnome-fs-trash-full", "trashcan_full",
"folder", "gnome-fs-directory", NULL);
case MOO_ICON_DIRECTORY:
return create_named_icon (icon_theme, widget, size, pixel_size, GTK_STOCK_DIRECTORY,
"folder", "gnome-fs-directory", NULL);
case MOO_ICON_BROKEN_LINK:
return create_broken_link_icon (icon_theme, widget, size);
case MOO_ICON_NONEXISTENT:
return create_broken_icon (icon_theme, widget, size);
case MOO_ICON_BLOCK_DEVICE:
return create_named_icon (icon_theme, widget, size, pixel_size, GTK_STOCK_HARDDISK,
"drive-harddisk", "gnome-fs-blockdev", "blockdevice", NULL);
case MOO_ICON_CHARACTER_DEVICE:
return create_named_icon (icon_theme, widget, size, pixel_size, NULL,
"gnome-fs-chardev", "chardevice", "input-keyboard", NULL);
case MOO_ICON_FIFO:
return create_named_icon (icon_theme, widget, size, pixel_size, NULL,
"gnome-fs-fifo", "pipe", NULL);
case MOO_ICON_SOCKET:
return create_named_icon (icon_theme, widget, size, pixel_size, NULL,
"gnome-fs-socket", NULL);
case MOO_ICON_FILE:
return create_named_icon (icon_theme, widget, size, pixel_size, NULL,
"gnome-fs-regular", "unknown", NULL);
case MOO_ICON_BLANK:
return create_fallback_icon (widget, size);
case MOO_ICON_MIME:
case MOO_ICON_INVALID:
g_return_val_if_reached (NULL);
}
g_return_val_if_reached (NULL);
}
static GdkPixbuf *
create_mime_icon_type (GtkIconTheme *icon_theme,
const char *mime_type,
int pixel_size)
{
GString *icon_name;
GdkPixbuf *pixbuf = NULL;
const char *separator;
/* foo/bar-baz */
separator = strchr (mime_type, '/');
if (!separator)
{
g_warning ("%s: mime type '%s' is invalid", G_STRLOC, mime_type);
return NULL;
}
if (!strncmp (mime_type, "text", strlen ("text")))
pixbuf = get_named_icon (icon_theme, "text-x-generic", pixel_size);
else if (!strncmp (mime_type, "audio", strlen ("audio")))
pixbuf = get_named_icon (icon_theme, "audio-x-generic", pixel_size);
else if (!strncmp (mime_type, "image", strlen ("image")))
pixbuf = get_named_icon (icon_theme, "image-x-generic", pixel_size);
if (!pixbuf)
{
/* gnome-mime-foo */
icon_name = g_string_new ("gnome-mime-");
g_string_append_len (icon_name, mime_type, separator - mime_type);
pixbuf = get_named_icon (icon_theme, icon_name->str, pixel_size);
if (DEBUG_MESSAGES && pixbuf)
_moo_message ("got icon '%s' for mime type '%s'", icon_name->str, mime_type);
g_string_free (icon_name, TRUE);
}
if (!pixbuf)
{
if (!strncmp (mime_type, "text", strlen ("text")))
pixbuf = get_named_icon (icon_theme, "txt", pixel_size);
else if (!strncmp (mime_type, "audio", strlen ("audio")))
pixbuf = get_named_icon (icon_theme, "sound", pixel_size);
else if (!strncmp (mime_type, "image", strlen ("image")))
pixbuf = get_named_icon (icon_theme, "image", pixel_size);
}
if (!pixbuf)
{
/* foo */
icon_name = g_string_new_len (mime_type, separator - mime_type);
pixbuf = get_named_icon (icon_theme, icon_name->str, pixel_size);
if (DEBUG_MESSAGES && pixbuf)
_moo_message ("got icon '%s' for mime type '%s'", icon_name->str, mime_type);
g_string_free (icon_name, TRUE);
}
return pixbuf;
}
static GdkPixbuf *
create_mime_icon_exact (GtkIconTheme *icon_theme,
const char *mime_type,
int pixel_size)
{
GString *icon_name;
GdkPixbuf *pixbuf = NULL;
const char *separator;
/* foo/bar-baz */
separator = strchr (mime_type, '/');
if (!separator)
{
g_warning ("%s: mime type '%s' is invalid", G_STRLOC, mime_type);
return NULL;
}
if (!pixbuf)
{
/* foo-bar-baz */
icon_name = g_string_new_len (mime_type, separator - mime_type);
g_string_append_c (icon_name, '-');
g_string_append (icon_name, separator + 1);
pixbuf = get_named_icon (icon_theme, icon_name->str, pixel_size);
if (DEBUG_MESSAGES && pixbuf)
_moo_message ("got icon '%s' for mime type '%s'", icon_name->str, mime_type);
g_string_free (icon_name, TRUE);
}
if (!pixbuf)
{
/* gnome-mime-foo-bar-baz */
icon_name = g_string_new ("gnome-mime-");
g_string_append_len (icon_name, mime_type, separator - mime_type);
g_string_append_c (icon_name, '-');
g_string_append (icon_name, separator + 1);
pixbuf = get_named_icon (icon_theme, icon_name->str, pixel_size);
if (DEBUG_MESSAGES && pixbuf)
_moo_message ("got icon '%s' for mime type '%s'", icon_name->str, mime_type);
g_string_free (icon_name, TRUE);
}
return pixbuf;
}
static GdkPixbuf *
create_mime_icon (GtkWidget *widget,
const char *mime_type,
GtkIconSize size)
{
GdkPixbuf *pixbuf = NULL;
GtkIconTheme *icon_theme;
int pixel_size;
GdkScreen *screen;
screen = gtk_widget_get_screen (widget);
icon_theme = gtk_icon_theme_get_for_screen (screen);
pixels_from_icon_size (screen, size, NULL, &pixel_size);
if (!strcmp (mime_type, XDG_MIME_TYPE_UNKNOWN))
{
pixbuf = get_named_icon (icon_theme, "unknown", pixel_size);
if (DEBUG_MESSAGES && pixbuf)
_moo_message ("got 'unknown' icon for '%s'", mime_type);
if (!pixbuf)
{
pixbuf = create_special_icon (widget, MOO_ICON_FILE, size);
if (DEBUG_MESSAGES && pixbuf)
_moo_message ("got FILE icon for '%s'", mime_type);
}
if (!pixbuf)
{
if (DEBUG_MESSAGES)
_moo_message ("getting fallback icon for mime type '%s'", mime_type);
pixbuf = create_fallback_icon (widget, size);
}
return pixbuf;
}
pixbuf = create_mime_icon_exact (icon_theme, mime_type, pixel_size);
#ifdef MOO_USE_XDGMIME
if (!pixbuf)
{
char **parent_types = xdg_mime_list_mime_parents (mime_type);
if (parent_types && parent_types[0])
{
char **p;
for (p = parent_types; *p && !pixbuf; ++p)
{
pixbuf = create_mime_icon (widget, *p, size);
if (DEBUG_MESSAGES && pixbuf)
_moo_message ("used mime type '%s' icon for '%s'", *p, mime_type);
}
g_free (parent_types);
return pixbuf;
}
else
{
pixbuf = create_mime_icon_type (icon_theme, mime_type, size);
}
g_free (parent_types);
}
#endif
if (!pixbuf)
create_special_icon (widget, MOO_ICON_FILE, size);
return pixbuf;
}