medit/moo/mooutils/moofileicon.c
2010-12-21 20:15:45 -08:00

807 lines
24 KiB
C

/*
* moofileicon.c
*
* Copyright (C) 2004-2010 by Yevgen Muntyan <emuntyan@users.sourceforge.net>
*
* This file is part of medit. medit 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.
*
* You should have received a copy of the GNU Lesser General Public
* License along with medit. If not, see <http://www.gnu.org/licenses/>.
*/
#include "mooutils/moofileicon.h"
#include "mooutils/moo-mime.h"
#include "mooutils/mooutils-fs.h"
#include "mooutils/mooutils-misc.h"
#include "mooutils/mooutils-debug.h"
#include "mooutils/moostock.h"
#include "moofileicon-symlink.h"
#include <glib/gstdio.h>
#include <gtk/gtk.h>
#include <sys/stat.h>
#include <string.h>
void
moo_file_icon_for_file (MooFileIcon *icon,
G_GNUC_UNUSED const char *path)
{
const char *mime_type = NULL;
g_return_if_fail (icon != NULL);
#ifndef __WIN32__
if (path)
mime_type = moo_get_mime_type_for_file (path, NULL);
#else
MOO_IMPLEMENT_ME
#endif
if (!mime_type || !mime_type[0])
mime_type = MOO_MIME_TYPE_UNKNOWN;
icon->mime_type = mime_type;
icon->type = MOO_ICON_MIME;
icon->emblem = (MooIconEmblem) 0;
}
GdkPixbuf *
moo_get_icon_for_path (const char *path,
GtkWidget *widget,
GtkIconSize size)
{
MooFileIcon icon = {0};
moo_file_icon_for_file (&icon, path);
return moo_file_icon_get_pixbuf (&icon, widget, size);
}
MooFileIcon *
moo_file_icon_new (void)
{
return moo_new (MooFileIcon);
}
MooFileIcon *
moo_file_icon_copy (MooFileIcon *icon)
{
return icon ? moo_obj_dup (MooFileIcon, icon) : NULL;
}
void
moo_file_icon_free (MooFileIcon *icon)
{
if (icon)
moo_free (MooFileIcon, icon);
}
/********************************************************************/
/* Icon cache
*/
#define MOO_ICON_EMBLEM_LEN 2
MOO_DEBUG_INIT(icons, FALSE)
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 = (GdkPixbuf**) 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 = (GdkPixbuf**) 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 = (GHashTable*) 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 = (MooIconCache*) 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;
}
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, GDK_PIXBUF (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, GDK_PIXBUF (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, GDK_PIXBUF (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,
G_GNUC_UNUSED 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))
{
moo_dmsg ("%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)
moo_dmsg ("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)
moo_dmsg ("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 (pixbuf)
moo_dmsg ("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 (pixbuf)
moo_dmsg ("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 (pixbuf)
moo_dmsg ("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 (pixbuf)
moo_dmsg ("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, MOO_MIME_TYPE_UNKNOWN))
{
pixbuf = get_named_icon (icon_theme, "unknown", pixel_size);
if (pixbuf)
moo_dmsg ("got 'unknown' icon for '%s'", mime_type);
if (!pixbuf)
{
pixbuf = create_special_icon (widget, MOO_ICON_FILE, size);
if (pixbuf)
moo_dmsg ("got FILE icon for '%s'", mime_type);
}
if (!pixbuf)
{
moo_dmsg ("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);
if (!pixbuf)
{
const char **parent_types = moo_mime_type_list_parents (mime_type);
if (parent_types && parent_types[0])
{
const char **p;
for (p = parent_types; *p && !pixbuf; ++p)
{
pixbuf = create_mime_icon (widget, *p, size);
if (pixbuf)
moo_dmsg ("used mime type '%s' icon for '%s'", *p, mime_type);
}
g_free ((gpointer) parent_types);
return pixbuf;
}
else
{
pixbuf = create_mime_icon_type (icon_theme, mime_type, size);
}
g_free ((gpointer) parent_types);
}
if (!pixbuf)
pixbuf = create_special_icon (widget, MOO_ICON_FILE, size);
return pixbuf;
}
GdkPixbuf *
moo_file_icon_get_pixbuf (MooFileIcon *icon,
GtkWidget *widget,
GtkIconSize size)
{
GdkScreen *screen;
GdkPixbuf *pixbuf;
GdkPixbuf *original;
MooIconCache *cache;
const char *mime_type;
g_return_val_if_fail (icon != NULL, NULL);
g_return_val_if_fail (!widget || GTK_IS_WIDGET (widget), NULL);
g_return_val_if_fail (icon->type < MOO_ICON_INVALID, NULL);
g_return_val_if_fail (icon->type != MOO_ICON_MIME || icon->mime_type != NULL, NULL);
g_return_val_if_fail (icon->emblem < MOO_ICON_EMBLEM_LEN, NULL);
if (icon->type != MOO_ICON_MIME)
mime_type = NULL;
else
mime_type = icon->mime_type;
if (widget && gtk_widget_has_screen (widget))
screen = gtk_widget_get_screen (widget);
else
screen = gdk_screen_get_default ();
cache = moo_icon_cache_get_for_screen (screen, size);
g_return_val_if_fail (cache != NULL, NULL);
pixbuf = moo_icon_cache_get (cache, icon->type, mime_type, icon->emblem);
if (pixbuf)
return pixbuf;
original = NULL;
if (icon->emblem)
original = moo_icon_cache_get (cache, icon->type, mime_type, (MooIconEmblem) 0);
if (!original)
{
if (mime_type)
original = create_mime_icon (widget, mime_type, size);
else
original = create_special_icon (widget, icon->type, size);
if (!original)
original = create_fallback_icon (widget, size);
g_return_val_if_fail (original != NULL, NULL);
moo_icon_cache_set (cache, icon->type, mime_type, (MooIconEmblem) 0, original);
g_object_unref (original);
}
if (!icon->emblem)
return original;
pixbuf = add_emblem (original, icon->emblem, size);
g_return_val_if_fail (pixbuf != NULL, NULL);
moo_icon_cache_set (cache, icon->type, mime_type, icon->emblem, pixbuf);
g_object_unref (pixbuf);
return pixbuf;
}