aboutsummaryrefslogblamecommitdiffstats
path: root/addressbook/gui/widgets/e-minicard.c
blob: 6ff1d94aebb5c90605341fba1e602e91f9ab0c96 (plain) (tree)





















                                                                           
                   


                             
                   



                                  

                             
 



                                                                              
                                                   


                                                                          
                                                                   
 
                                                                
                                             







                                                  
                      
                









































                                                                                       
                                                                 
                                                             




                                                               
                                             














                                                 
                              
  
                        

                                                                                    












                                                            

                                                                    
                                                               

                                                                                          
                 
                
                           











                                                                                                               
                    
                                                       
                      
                      




                                                                        


                                                   
















                                                                 
                           
                                                                                               
                      
                      
                                                                       
                      






                                       








                                                  

                                                                





                                                                      

                                          































                                                                          
                                                         
                                                                              



                                                              
                                            
                                        
                                                                   
 
                            
                                           
 

                                
















                                                                





                                                                           


                                                         
















































































































                                                                                                                                                     
            
                         


           
















                                                                                         
                                                               
 
                                  



                                                 















                                                                          
                              
                          
                          
                            
                           









                                                                                     
                 



                                                            
                                                    
                                                      



                                                           
                                                  
                                                    
                                                   
                                     
 



                                                                              
                                                            
                            


                                                                              



                                                              
 


                                                               


                                                               


                                                                   


                                                                                                                                                     
                                                                        
                                                                                              
                                                                               























                                                                                                                                                   
                 
 

                                                                



           
                                                     
 
                                                 



                                                                            
                





                                                                      
                                                        




                                                               



                                                                                
                                                                                                          












                                                                               
                                                                              
                                              

                                                     

                                                                  
 









                                                            
                          











                                                               
                                          
                                                           
                                                    

                                                           
                                                    
                                     


                                                          
                                  
                             
                                 
                                                                                                    



                         
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/* 
 * e-minicard.c
 * Copyright (C) 2000  Helix Code, Inc.
 * Author: Chris Lahey <clahey@helixcode.com>
 *
 * This library 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.
 *
 * 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 library; if not, write to the
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
 * Boston, MA 02111-1307, USA.
 */

#include <config.h>
#include <gnome.h>
#include "e-minicard.h"
#include "e-minicard-label.h"
#include "e-text.h"
#include <ebook/e-book.h>
#include <e-util/e-canvas.h>
#include <e-util/e-util.h>
#include <e-util/e-canvas-utils.h>
#include "e-contact-editor.h"
#include "e-minicard-view.h"

static void e_minicard_init     (EMinicard       *card);
static void e_minicard_class_init   (EMinicardClass  *klass);
static void e_minicard_set_arg (GtkObject *o, GtkArg *arg, guint arg_id);
static void e_minicard_get_arg (GtkObject *object, GtkArg *arg, guint arg_id);
static void e_minicard_destroy (GtkObject *object);
static gboolean e_minicard_event (GnomeCanvasItem *item, GdkEvent *event);
static void e_minicard_realize (GnomeCanvasItem *item);
static void e_minicard_unrealize (GnomeCanvasItem *item);
static void e_minicard_reflow ( GnomeCanvasItem *item, int flags );

static void e_minicard_resize_children( EMinicard *e_minicard );
static void remodel( EMinicard *e_minicard );

static GnomeCanvasGroupClass *parent_class = NULL;

/* The arguments we take */
enum {
    ARG_0,
    ARG_WIDTH,
    ARG_HEIGHT,
    ARG_HAS_FOCUS,
    ARG_CARD
};

GtkType
e_minicard_get_type (void)
{
  static GtkType minicard_type = 0;

  if (!minicard_type)
    {
      static const GtkTypeInfo minicard_info =
      {
        "EMinicard",
        sizeof (EMinicard),
        sizeof (EMinicardClass),
        (GtkClassInitFunc) e_minicard_class_init,
        (GtkObjectInitFunc) e_minicard_init,
        /* reserved_1 */ NULL,
        /* reserved_2 */ NULL,
        (GtkClassInitFunc) NULL,
      };

      minicard_type = gtk_type_unique (gnome_canvas_group_get_type (), &minicard_info);
    }

  return minicard_type;
}

static void
e_minicard_class_init (EMinicardClass *klass)
{
  GtkObjectClass *object_class;
  GnomeCanvasItemClass *item_class;

  object_class = (GtkObjectClass*) klass;
  item_class = (GnomeCanvasItemClass *) klass;

  parent_class = gtk_type_class (gnome_canvas_group_get_type ());
  
  gtk_object_add_arg_type ("EMinicard::width", GTK_TYPE_DOUBLE, 
               GTK_ARG_READWRITE, ARG_WIDTH); 
  gtk_object_add_arg_type ("EMinicard::height", GTK_TYPE_DOUBLE, 
               GTK_ARG_READABLE, ARG_HEIGHT);
  gtk_object_add_arg_type ("EMinicard::has_focus", GTK_TYPE_ENUM,
               GTK_ARG_READWRITE, ARG_HAS_FOCUS);
  gtk_object_add_arg_type ("EMinicard::card", GTK_TYPE_OBJECT, 
               GTK_ARG_READWRITE, ARG_CARD);
 
  object_class->set_arg = e_minicard_set_arg;
  object_class->get_arg = e_minicard_get_arg;
  object_class->destroy = e_minicard_destroy;
  
  /* GnomeCanvasItem method overrides */
  item_class->realize     = e_minicard_realize;
  item_class->unrealize   = e_minicard_unrealize;
  item_class->event       = e_minicard_event;
}

static void
e_minicard_init (EMinicard *minicard)
{
  /*   minicard->card = NULL;*/
  minicard->rect = NULL;
  minicard->fields = NULL;
  minicard->width = 10;
  minicard->height = 10;
  minicard->has_focus = FALSE;
  
  minicard->card = NULL;

  e_canvas_item_set_reflow_callback(GNOME_CANVAS_ITEM(minicard), e_minicard_reflow);
}

static void
e_minicard_set_arg (GtkObject *o, GtkArg *arg, guint arg_id)
{
    GnomeCanvasItem *item;
    EMinicard *e_minicard;

    item = GNOME_CANVAS_ITEM (o);
    e_minicard = E_MINICARD (o);
    
    switch (arg_id){
    case ARG_WIDTH:
        if (e_minicard->width != GTK_VALUE_DOUBLE (*arg)) {
            e_minicard->width = GTK_VALUE_DOUBLE (*arg);
            e_minicard_resize_children(e_minicard);
            if ( GTK_OBJECT_FLAGS( e_minicard ) & GNOME_CANVAS_ITEM_REALIZED )
                e_canvas_item_request_reflow(item);
        }
      break;
    case ARG_HAS_FOCUS:
        if (e_minicard->fields) {
            if ( GTK_VALUE_ENUM(*arg) == E_FOCUS_START ||
                 GTK_VALUE_ENUM(*arg) == E_FOCUS_CURRENT) {
                gnome_canvas_item_set(GNOME_CANVAS_ITEM(e_minicard->fields->data),
                              "has_focus", GTK_VALUE_ENUM(*arg),
                              NULL);
            } else if ( GTK_VALUE_ENUM(*arg) == E_FOCUS_END ) {
                gnome_canvas_item_set(GNOME_CANVAS_ITEM(g_list_last(e_minicard->fields)->data),
                              "has_focus", GTK_VALUE_ENUM(*arg),
                              NULL);
            }
        }
        else
            e_canvas_item_grab_focus(item);
        break;
    case ARG_CARD:
        if (e_minicard->card)
            gtk_object_unref (GTK_OBJECT(e_minicard->card));
        e_minicard->card = E_CARD(GTK_VALUE_OBJECT (*arg));
        if (e_minicard->card)
            gtk_object_ref (GTK_OBJECT(e_minicard->card));
        remodel(e_minicard);
        e_canvas_item_request_reflow(item);
        break;
    }
}

static void
e_minicard_get_arg (GtkObject *object, GtkArg *arg, guint arg_id)
{
    EMinicard *e_minicard;

    e_minicard = E_MINICARD (object);

    switch (arg_id) {
    case ARG_WIDTH:
      GTK_VALUE_DOUBLE (*arg) = e_minicard->width;
      break;
    case ARG_HEIGHT:
      GTK_VALUE_DOUBLE (*arg) = e_minicard->height;
      break;
    case ARG_HAS_FOCUS:
        GTK_VALUE_ENUM (*arg) = e_minicard->has_focus ? E_FOCUS_CURRENT : E_FOCUS_NONE;
        break;
    case ARG_CARD:
        GTK_VALUE_OBJECT (*arg) = GTK_OBJECT(e_minicard->card);
        break;
    default:
      arg->type = GTK_TYPE_INVALID;
      break;
    }
}

static void
e_minicard_destroy (GtkObject *object)
{
    EMinicard *e_minicard;

    g_return_if_fail (object != NULL);
    g_return_if_fail (E_IS_MINICARD (object));

    e_minicard = E_MINICARD (object);

    if (e_minicard->card)
        gtk_object_unref (GTK_OBJECT(e_minicard->card));

    if (GTK_OBJECT_CLASS (parent_class)->destroy)
        (* GTK_OBJECT_CLASS (parent_class)->destroy) (object);
}

static void
e_minicard_realize (GnomeCanvasItem *item)
{
    EMinicard *e_minicard;
    GnomeCanvasGroup *group;

    e_minicard = E_MINICARD (item);
    group = GNOME_CANVAS_GROUP( item );

    if (GNOME_CANVAS_ITEM_CLASS(parent_class)->realize)
        (* GNOME_CANVAS_ITEM_CLASS(parent_class)->realize) (item);
    
    e_minicard->rect =
      gnome_canvas_item_new( group,
                 gnome_canvas_rect_get_type(),
                 "x1", (double) 0,
                 "y1", (double) 0,
                 "x2", (double) e_minicard->width - 1,
                 "y2", (double) e_minicard->height - 1,
                 "outline_color", NULL,
                 NULL );

    e_minicard->header_rect =
      gnome_canvas_item_new( group,
                 gnome_canvas_rect_get_type(),
                 "x1", (double) 2,
                 "y1", (double) 2,
                 "x2", (double) e_minicard->width - 3,
                 "y2", (double) e_minicard->height - 3,
                 "fill_color", "grey70",
                 NULL );

    e_minicard->header_text =
      gnome_canvas_item_new( group,
                 e_text_get_type(),
                 "anchor", GTK_ANCHOR_NW,
                 "width", (double) ( e_minicard->width - 12 ),
                 "clip", TRUE,
                 "use_ellipsis", TRUE,
                 "font", "lucidasans-bold-10",
                 "fill_color", "black",
                 "text", "",
                 NULL );
    e_canvas_item_move_absolute(e_minicard->header_text, 6, 6);

    remodel(e_minicard);
    e_canvas_item_request_reflow(item);

    if (!item->canvas->aa) {
    }
}

static void
e_minicard_unrealize (GnomeCanvasItem *item)
{
  EMinicard *e_minicard;

  e_minicard = E_MINICARD (item);

  if (!item->canvas->aa)
    {
    }

  if (GNOME_CANVAS_ITEM_CLASS(parent_class)->unrealize)
    (* GNOME_CANVAS_ITEM_CLASS(parent_class)->unrealize) (item);
}

static void
card_changed_cb (EBook* book, EBookStatus status, gpointer user_data)
{
    g_print ("%s: %s(): a card was changed\n", __FILE__, __FUNCTION__);
}

static gboolean
e_minicard_event (GnomeCanvasItem *item, GdkEvent *event)
{
    EMinicard *e_minicard;
    
    e_minicard = E_MINICARD (item);
    
    switch( event->type ) {
    case GDK_FOCUS_CHANGE:
        {
            GdkEventFocus *focus_event = (GdkEventFocus *) event;
            if ( focus_event->in ) {
                gnome_canvas_item_set( e_minicard->rect, 
                               "outline_color", "grey50", 
                               NULL );
                gnome_canvas_item_set( e_minicard->header_rect, 
                               "fill_color", "darkblue",
                               NULL );
                gnome_canvas_item_set( e_minicard->header_text, 
                               "fill_color", "white",
                               NULL );
                e_minicard->has_focus = TRUE;
            } else {
                gnome_canvas_item_set( e_minicard->rect, 
                               "outline_color", NULL, 
                               NULL );
                gnome_canvas_item_set( e_minicard->header_rect, 
                               "fill_color", "grey70",
                               NULL );
                gnome_canvas_item_set( e_minicard->header_text, 
                               "fill_color", "black",
                               NULL );
                e_minicard->has_focus = FALSE;
            }
        }
        break;
    case GDK_BUTTON_PRESS:
        if (event->button.button == 1) {
            e_canvas_item_grab_focus(item);
        }
        break;
    case GDK_2BUTTON_PRESS:
        if (E_IS_MINICARD_VIEW(item->parent)) {
            gint result;
            GtkWidget* contact_editor =
                e_contact_editor_new(e_minicard->card);
            EBook *book;
            GtkWidget *dlg;
            gtk_object_get(GTK_OBJECT(item->parent),
                       "book", &book,
                       NULL);
            
            dlg = gnome_dialog_new ("Contact Editor", "Save", "Cancel", NULL);
            
            g_assert (E_IS_BOOK (book));
            
            gtk_box_pack_start (GTK_BOX (GNOME_DIALOG (dlg)->vbox),
                        contact_editor, TRUE, TRUE, 0);
            
            gtk_widget_show_all (dlg);
            
            gnome_dialog_close_hides (GNOME_DIALOG (dlg), TRUE);
            result = gnome_dialog_run_and_close (GNOME_DIALOG (dlg));
            
            
            /* If the user clicks "okay"...*/
            if (result == 0) {
                ECard *card;
                g_assert (contact_editor);
                g_assert (GTK_IS_OBJECT (contact_editor));
                gtk_object_get(GTK_OBJECT(contact_editor),
                           "card", &card,
                           NULL);
                
                /* Add the card in the contact editor to our ebook */
                e_book_commit_card (book,
                            card,
                            card_changed_cb,
                            NULL);
            }
        }
        break;
    case GDK_KEY_PRESS:
        if (event->key.keyval == GDK_Tab || 
            event->key.keyval == GDK_KP_Tab || 
            event->key.keyval == GDK_ISO_Left_Tab) {
            GList *list;
            for (list = e_minicard->fields; list; list = list->next) {
                GnomeCanvasItem *item = GNOME_CANVAS_ITEM (list->data);
                EFocus has_focus;
                gtk_object_get(GTK_OBJECT(item),
                           "has_focus", &has_focus,
                           NULL);
                if (has_focus != E_FOCUS_NONE) {
                    if (event->key.state & GDK_SHIFT_MASK)
                        list = list->prev;
                    else
                        list = list->next;
                    if (list) {
                        item = GNOME_CANVAS_ITEM (list->data);
                        gnome_canvas_item_set(item,
                                      "has_focus", (event->key.state & GDK_SHIFT_MASK) ? E_FOCUS_END : E_FOCUS_START,
                                      NULL);
                        return 1;
                    } else {
                        return 0;
                    }
                }
            }
        }
    default:
        break;
    }
    
    if (GNOME_CANVAS_ITEM_CLASS( parent_class )->event)
        return (* GNOME_CANVAS_ITEM_CLASS( parent_class )->event) (item, event);
    else
        return 0;
}

static void
e_minicard_resize_children( EMinicard *e_minicard )
{
    if ( GTK_OBJECT_FLAGS( e_minicard ) & GNOME_CANVAS_ITEM_REALIZED ) {
        GList *list;
        
        gnome_canvas_item_set( e_minicard->header_text,
                       "width", (double) e_minicard->width - 12,
                       NULL );
        for ( list = e_minicard->fields; list; list = g_list_next( list ) ) {
            gnome_canvas_item_set( GNOME_CANVAS_ITEM( list->data ),
                           "width", (double) e_minicard->width - 4.0,
                           NULL );
        }
    }
}

static void
add_field (EMinicard *e_minicard, char *fieldname, char* field)
{
    GnomeCanvasItem *new_item;
    GnomeCanvasGroup *group;

    group = GNOME_CANVAS_GROUP( e_minicard );

    new_item = e_minicard_label_new(group);
    gnome_canvas_item_set( new_item,
                   "width", e_minicard->width - 4.0,
                   "fieldname", fieldname,
                   "field", field,
                   NULL );
    e_minicard->fields = g_list_append( e_minicard->fields, new_item);
    e_canvas_item_move_absolute(new_item, 2, e_minicard->height);
}
        

static void
remodel( EMinicard *e_minicard )
{
    if (e_minicard->card) {
        char *fname;
        char *file_as;
        char *url;
        char *org;
        char *title;
        char *role;
        ECardList *address_list;
        ECardList *phone_list;
        ECardList *email_list;

        ECardIterator *iterator;

        GList *list;

        for ( list = e_minicard->fields; list; list = g_list_next( list ) ) {
            gtk_object_destroy( GTK_OBJECT( list->data ) );
        }
        g_list_free(e_minicard->fields);
        e_minicard->fields = NULL;

        gtk_object_get(GTK_OBJECT(e_minicard->card),
                   "full_name",  &fname,
                   "file_as",    &file_as,
                   "address",    &address_list,
                   "phone",      &phone_list,
                   "email",      &email_list,
                   "url",        &url,
                   "org",        &org,
                   "title",      &title,
                   "role",       &role,
                   NULL);

        if (e_minicard->header_text) {
            if (file_as)
                gnome_canvas_item_set(e_minicard->header_text,
                              "text", file_as,
                              NULL);
            else
                gnome_canvas_item_set(e_minicard->header_text,
                              "text", "",
                              NULL);
        }

        if (fname)
            add_field(e_minicard, "Name:", fname);

        if (org)
            add_field(e_minicard, "Company:", org);

        if (title)
            add_field(e_minicard, "Title:", title);

        if (role)
            add_field(e_minicard, "Profession:", role);

        if (address_list) {
            for (iterator = e_card_list_get_iterator(address_list); e_card_iterator_is_valid(iterator); e_card_iterator_next(iterator)) {
                const ECardDeliveryAddress *address = e_card_iterator_get(iterator);
                if (address->flags & E_CARD_ADDR_WORK) {
                    add_field(e_minicard, "Work Address:", address->city);
                } else if (address->flags & E_CARD_ADDR_HOME) {
                    add_field(e_minicard, "Home Address:", address->city);
                } else {
                    add_field(e_minicard, "Address:", address->city);
                }
            }
        }
        if (phone_list) {
            for (iterator = e_card_list_get_iterator(phone_list); e_card_iterator_is_valid(iterator); e_card_iterator_next(iterator)) {
                const ECardPhone *phone = e_card_iterator_get(iterator);
                if (phone->flags & E_CARD_PHONE_WORK) {
                    add_field(e_minicard, "Work Phone:", phone->number);
                } else if (phone->flags & E_CARD_PHONE_HOME) {
                    add_field(e_minicard, "Home Phone:", phone->number);
                } else if (phone->flags & E_CARD_PHONE_CELL) {
                    add_field(e_minicard, "Mobile Phone:", phone->number);
                } else {
                    add_field(e_minicard, "Phone:", phone->number);
                }
            }
        }
        if (email_list) {
            for (iterator = e_card_list_get_iterator(email_list); e_card_iterator_is_valid(iterator); e_card_iterator_next(iterator)) {
                add_field(e_minicard, "Email:", (char *) e_card_iterator_get(iterator));
            }
        }

        if (url)
            add_field(e_minicard, "Web page:", url);
    }
}

static void
e_minicard_reflow( GnomeCanvasItem *item, int flags )
{
    EMinicard *e_minicard = E_MINICARD(item);
    if ( GTK_OBJECT_FLAGS( e_minicard ) & GNOME_CANVAS_ITEM_REALIZED ) {
        GList *list;
        gdouble text_height;
        gint old_height;
        
        old_height = e_minicard->height;

        gtk_object_get( GTK_OBJECT( e_minicard->header_text ),
                "text_height", &text_height,
                NULL );
        
        e_minicard->height = text_height + 10.0;
        
        gnome_canvas_item_set( e_minicard->header_rect,
                       "y2", text_height + 9.0,
                       NULL );
        
        for(list = e_minicard->fields; list; list = g_list_next(list)) {
            gtk_object_get (GTK_OBJECT(list->data),
                    "height", &text_height,
                    NULL);
            e_canvas_item_move_absolute(GNOME_CANVAS_ITEM(list->data), 2, e_minicard->height);
            e_minicard->height += text_height;
        }
        e_minicard->height += 2;
        
        gnome_canvas_item_set( e_minicard->rect,
                       "y2", (double) e_minicard->height - 1,
                       NULL );
        
        gnome_canvas_item_set( e_minicard->rect,
                       "x2", (double) e_minicard->width - 1.0,
                       "y2", (double) e_minicard->height - 1.0,
                       NULL );
        gnome_canvas_item_set( e_minicard->header_rect,
                       "x2", (double) e_minicard->width - 3.0,
                       NULL );

        if (old_height != e_minicard->height)
            e_canvas_item_request_parent_reflow(item);
    }
}

char *
e_minicard_get_card_id (EMinicard *minicard)
{
    g_return_val_if_fail(minicard != NULL, NULL);
    g_return_val_if_fail(E_IS_MINICARD(minicard), NULL);

    if (minicard->card) {
        return e_card_get_id(minicard->card);
    } else {
        return "";
    }
}

int
e_minicard_compare (EMinicard *minicard1, EMinicard *minicard2)
{
    g_return_val_if_fail(minicard1 != NULL, 0);
    g_return_val_if_fail(E_IS_MINICARD(minicard1), 0);
    g_return_val_if_fail(minicard2 != NULL, 0);
    g_return_val_if_fail(E_IS_MINICARD(minicard2), 0);

    if (minicard1->card && minicard2->card) {
        char *file_as1, *file_as2;
        gtk_object_get(GTK_OBJECT(minicard1->card),
                   "file_as", &file_as1,
                   NULL);
        gtk_object_get(GTK_OBJECT(minicard2->card),
                   "file_as", &file_as2,
                   NULL);
        if (file_as1 && file_as2)
            return strcmp(file_as1, file_as2);
        if (file_as1)
            return -1;
        if (file_as2)
            return 1;
        return strcmp(e_minicard_get_card_id(minicard1), e_minicard_get_card_id(minicard2));
    } else {
        return 0;
    }
}