aboutsummaryrefslogblamecommitdiffstats
path: root/mail/em-icon-stream.c
blob: 4f7a369d7e699c1aed5de99abb6248bf017c1b6d (plain) (tree)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
















                                                                           
                                                                   






                    
                   


                                         
                                  
 
                                     
 
            
 

                                                         




                          







                                                                                  







                                             



                                                   
 








                                                                                                 
                                                                                                
         
 














                                                                   
                    














                                                                           

                          


















                                                                  
                                                                                             











                                    









                                                                    
                                                        









                                                                         





                                                                                                   
                                                                           



                    


                                                    
                                 
                                      
                       






                                                            
                                                               


                                   
                                                                   
 


                                                                                           
         





                                                                                          
 














                                                                   
                                                                                                             
                          
 
                                                                    
                                
                           
                         
                                                                                                  
                                 

                                  
           
                                                                                         

                                      

                               
                                                 
                                  

                                                                           
                                  

                                                                     







                                                           
                                                              





















                                                                                                         



                  
   
                                                                                          

                                      

                               
 












                                                                                         



                                   
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/*
 * Authors: Jeffrey Stedfast <fejj@ximian.com>
 *      Michael Zucchi <notzed@ximian.com>
 *
 * Copyright 2003 Ximian, Inc. (www.ximian.com)
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of version 2 of the GNU General Public
 * License as published by the Free Software Foundation.
 *
 * 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., 51 Franklin Street, Fifth Floor,
 * Boston, MA 02110-1301, USA.
 *
 */

#ifdef HAVE_CONFIG_H
#include <config.h>
#endif

#include <stdio.h>
#include <string.h>

#include <gdk-pixbuf/gdk-pixbuf.h>
#include <gdk-pixbuf/gdk-pixbuf-loader.h>
#include <gtk/gtkimage.h>
#include "em-icon-stream.h"
#include "e-util/e-icon-factory.h"

#include "libedataserver/e-msgport.h"

#define d(x)

/* fixed-point scale factor for scaled images in cache */
#define EMIS_SCALE (1024)

struct _emis_cache_node {
    EMCacheNode node;

    GdkPixbuf *pixbuf;
};

static void em_icon_stream_class_init (EMIconStreamClass *klass);
static void em_icon_stream_init (CamelObject *object);
static void em_icon_stream_finalize (CamelObject *object);

static ssize_t emis_sync_write(CamelStream *stream, const char *buffer, size_t n);
static int emis_sync_close(CamelStream *stream);
static int emis_sync_flush(CamelStream *stream);

static EMSyncStreamClass *parent_class = NULL;
static EMCache *emis_cache;

static void
emis_cache_free(void *data)
{
    struct _emis_cache_node *node = data;

    g_object_unref(node->pixbuf);
}

CamelType
em_icon_stream_get_type (void)
{
    static CamelType type = CAMEL_INVALID_TYPE;

    if (type == CAMEL_INVALID_TYPE) {
        parent_class = (EMSyncStreamClass *)em_sync_stream_get_type();
        type = camel_type_register (em_sync_stream_get_type(),
                        "EMIconStream",
                        sizeof (EMIconStream),
                        sizeof (EMIconStreamClass),
                        (CamelObjectClassInitFunc) em_icon_stream_class_init,
                        NULL,
                        (CamelObjectInitFunc) em_icon_stream_init,
                        (CamelObjectFinalizeFunc) em_icon_stream_finalize);

        emis_cache = em_cache_new(60, sizeof(struct _emis_cache_node), emis_cache_free);
    }

    return type;
}

static void
em_icon_stream_class_init (EMIconStreamClass *klass)
{
    ((EMSyncStreamClass *)klass)->sync_write = emis_sync_write;
    ((EMSyncStreamClass *)klass)->sync_flush = emis_sync_flush;
    ((EMSyncStreamClass *)klass)->sync_close = emis_sync_close;
}

static void
em_icon_stream_init (CamelObject *object)
{
    EMIconStream *emis = (EMIconStream *)object;

    emis = emis;
}

static void
emis_cleanup(EMIconStream *emis)
{
    if (emis->loader) {
        gdk_pixbuf_loader_close(emis->loader, NULL);
        g_object_unref(emis->loader);
        emis->loader = NULL;
    }

    if (emis->destroy_id) {
        g_signal_handler_disconnect(emis->image, emis->destroy_id);
        emis->destroy_id = 0;
    }

    g_free(emis->key);
    emis->key = NULL;

    emis->image = NULL;
    emis->sync.cancel = TRUE;
}

static void
em_icon_stream_finalize(CamelObject *object)
{
    EMIconStream *emis = (EMIconStream *)object;

    emis_cleanup(emis);
}

static ssize_t
emis_sync_write(CamelStream *stream, const char *buffer, size_t n)
{
    EMIconStream *emis = EM_ICON_STREAM (stream);

    if (emis->loader == NULL)
        return -1;

    if (!gdk_pixbuf_loader_write(emis->loader, (const unsigned char *)buffer, n, NULL)) {
        emis_cleanup(emis);
        return -1;
    }

    return (ssize_t) n;
}

static int
emis_sync_flush(CamelStream *stream)
{
    return 0;
}

static GdkPixbuf *
emis_fit(GdkPixbuf *pixbuf, int maxwidth, int maxheight, int *scale)
{
    GdkPixbuf *mini = NULL;
    int width, height;

    width = gdk_pixbuf_get_width(pixbuf);
    height = gdk_pixbuf_get_height(pixbuf);

    if ((maxwidth && width > maxwidth)
        || (maxheight && height > maxheight)) {
        if (width >= height || maxheight == 0) {
            if (scale)
                *scale = maxwidth * EMIS_SCALE / width;
            height = height * maxwidth / width;
            width = maxwidth;
        } else {
            if (scale)
                *scale = maxheight * EMIS_SCALE / height;
            width = width * maxheight / height;
            height = maxheight;
        }

        /* check if we don't want to scale down too much, if so, do 1 pixel width/height */
        if (width <= 0)
            width = 1;

        if (height <= 0)
            height = 1;

        mini = e_icon_factory_pixbuf_scale (pixbuf, width, height);
    }

    return mini;
}

static int
emis_sync_close(CamelStream *stream)
{
    EMIconStream *emis = (EMIconStream *)stream;
    GdkPixbuf *pixbuf, *mini;
    struct _emis_cache_node *node;
    char *scalekey;
    int scale;

    if (emis->loader == NULL)
        return -1;

    gdk_pixbuf_loader_close(emis->loader, NULL);

    pixbuf = gdk_pixbuf_loader_get_pixbuf(emis->loader);
    if (pixbuf == NULL) {
        d(printf("couldn't get pixbuf from loader\n"));
        emis_cleanup(emis);
        return -1;
    }

    mini = emis_fit(pixbuf, emis->width, emis->height, &scale);
    gtk_image_set_from_pixbuf(emis->image, mini?mini:pixbuf);

    if (emis->keep) {
        node = (struct _emis_cache_node *)em_cache_node_new(emis_cache, emis->key);
        node->pixbuf = g_object_ref(pixbuf);
        em_cache_add(emis_cache, (EMCacheNode *)node);
    }

    if (!emis->keep || mini) {
        scalekey = g_alloca(strlen(emis->key) + 20);
        sprintf(scalekey, "%s.%x", emis->key, scale);
        node = (struct _emis_cache_node *)em_cache_node_new(emis_cache, scalekey);
        node->pixbuf = mini?mini:g_object_ref(pixbuf);
        em_cache_add(emis_cache, (EMCacheNode *)node);
    }

    g_object_unref(emis->loader);
    emis->loader = NULL;

    g_signal_handler_disconnect(emis->image, emis->destroy_id);
    emis->destroy_id = 0;

    return 0;
}

static void
emis_image_destroy(struct _GtkImage *image, EMIconStream *emis)
{
    emis_cleanup(emis);
}

CamelStream *
em_icon_stream_new(GtkImage *image, const char *key, unsigned int maxwidth, unsigned int maxheight, int keep)
{
    EMIconStream *new;

    new = EM_ICON_STREAM(camel_object_new(EM_ICON_STREAM_TYPE));
    new->width = maxwidth;
    new->height = maxheight;
    new->image = image;
    new->keep = keep;
    new->destroy_id = g_signal_connect(image, "destroy", G_CALLBACK(emis_image_destroy), new);
    new->loader = gdk_pixbuf_loader_new();
    new->key = g_strdup(key);

    return (CamelStream *)new;
}

GdkPixbuf *
em_icon_stream_get_image(const char *tkey, unsigned int maxwidth, unsigned int maxheight)
{
    struct _emis_cache_node *node;
    GdkPixbuf *pb = NULL;
    const char *key;

    key = tkey ? tkey : "";

    /* forces the cache to be setup if not */
    em_icon_stream_get_type();

    node = (struct _emis_cache_node *)em_cache_lookup(emis_cache, key);
    if (node) {
        int width, height;

        pb = node->pixbuf;
        g_object_ref(pb);
        em_cache_node_unref(emis_cache, (EMCacheNode *)node);

        width = gdk_pixbuf_get_width(pb);
        height = gdk_pixbuf_get_height(pb);

        if ((maxwidth && width > maxwidth)
            || (maxheight && height > maxheight)) {
            unsigned int scale;
            char *realkey;

            if (maxheight == 0 || width >= height)
                scale = width * EMIS_SCALE / maxwidth;
            else
                scale = height * EMIS_SCALE / maxheight;

            realkey = g_alloca(strlen(key)+20);
            sprintf(realkey, "%s.%x", key, scale);
            node = (struct _emis_cache_node *)em_cache_lookup(emis_cache, realkey);
            if (node) {
                g_object_unref(pb);
                pb = node->pixbuf;
                g_object_ref(pb);
                em_cache_node_unref(emis_cache, (EMCacheNode *)node);
            } else {
                GdkPixbuf *mini = emis_fit(pb, maxwidth, maxheight, NULL);

                g_object_unref(pb);
                pb = mini;
                node = (struct _emis_cache_node *)em_cache_node_new(emis_cache, realkey);
                node->pixbuf = pb;
                g_object_ref(pb);
                em_cache_add(emis_cache, (EMCacheNode *)node);
            }
        }
    }

    return pb;
}

int
em_icon_stream_is_resized(const char *tkey, unsigned int maxwidth, unsigned int maxheight)
{
    int res = FALSE;
    struct _emis_cache_node *node;
    const char *key;

    key = tkey ? tkey : "";

    /* forces the cache to be setup if not */
    em_icon_stream_get_type();

    node = (struct _emis_cache_node *)em_cache_lookup(emis_cache, key);
    if (node) {
        res = (maxwidth && gdk_pixbuf_get_width(node->pixbuf) > maxwidth)
            || (maxheight && gdk_pixbuf_get_width(node->pixbuf) > maxheight);

        em_cache_node_unref(emis_cache, (EMCacheNode *)node);
    }

    return res;
}

void
em_icon_stream_clear_cache(void)
{
    em_cache_clear(emis_cache);
}