747 lines
22 KiB
C
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;
|
|
}
|