aboutsummaryrefslogblamecommitdiffstats
path: root/addressbook/printing/e-contact-print.c
blob: 002e284366d5acaacb91b2e85578e757468531b2 (plain) (tree)
1
2
3
4
5
6
7
8
9


                                                                           

                                          

                                                                
                                                                   
                                                        











                                                                    
                   
 
                  
                      
                   
                   
 
                 
 


                             
                                
                       
                            
                               
 
                    
                                  






                                       

                                                          
                                                  


                            
                                 

                  
                    

                               

                                  

                                                  
                       

                               
                    
                          
 
                            

  

                                            
 

                                                        
 
 



                                           
 

                            
 

                                        
 
                                                                 
 



                                                         
 
                                                        
 
                                
 
                                             

 




                                                      
 


                            
 
                                                                 
 



                                                                         
 



                                                                           
 
                                                           
 


                                             

                           
                                

 
              


                                                  
 

                            
 
                                                                 
 

                                                         
 
                                                        
 
                                
 
                                              


           
                                                                          
 



                                   
                    
 
                                         
 
                                                                       
 






                                                                            
 




                                                                
 
                                                            
 
                                                                 
 







                                                
 




                                              
 

                                                  
 



                                                     
 
                                                                 
 




                               

           
                                                                       
 
                      
                    

                  
                                                                 

                                                                      
 
                                                             
                                                             















                                                                          
                         
 
                                                                     

                                                                                        
         

                                   
 


                                                             
 

                                                              
 


                                                               
 





                                                                         
         



                                                                         


           
                                                       
 





                                                                    
         

 
          

                                                        


                                                 
                         
 

                                                                   
 

                                                       
 

                                                 
 



                                                               
 
                                       
 










                                                                


         


                                                                   
 
                                       

 

                      
 


                                                                


                             

                                          
 

                                                          

 

                         
 
                                                

 

                       
 
                                                 

 

                                                       
 







                                                                 


         
           
                                                 

                           
                        
 



                                                 
                               
                                       
 

                                                                                  
 















                                       



                                 
                                            
 
                                                                            
 


                                            
 
                                                                            
 


                                             
                                             

                                                                              
                                               

                          
                       
                                                                      
                                
                                                                           



                                                                     
                                                                            
                                                                                 
                                                                                      
                                                                                      
                                                                                      






                                                                                        

























                                                                                 

                                                                                              
























                                                                                      

                                     
 

 

                                          
 























                                                                           

 

                                          
 


















                                                                            
                         
                                   
                                   
                             
 
















                                                                         
 

                                                            
 










































                                                                              
         
 


                                                

 




                                                
 
                                                                        
 




                                                
 









                                                                 
 
 


                                                                     
 






























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

#include <config.h>

#include <ctype.h>
#include <sys/types.h>
#include <stdlib.h>
#include <string.h>

#include <glib.h>

#include <libxml/tree.h>
#include <libxml/parser.h>
#include <libxml/xmlmemory.h>
#include <libgnome/gnome-util.h>
#include <glib/gi18n.h>
#include <libebook/e-book.h>
#include <libebook/e-contact.h>

#include <gtk/gtk.h>
#include <libedataserver/e-flag.h>
#include <libedataserver/e-xml-utils.h>

#include "e-util/e-print.h"
#include "e-util/e-util.h"
#include "e-util/e-util-private.h"

#include "e-contact-print.h"

typedef struct _EContactPrintContext EContactPrintContext;
typedef struct _ContactPrintItem ContactPrintItem;

struct _EContactPrintContext
{
    GtkPrintContext *context;
    gdouble x;
    gdouble y;
    gint column;
    gdouble column_width;
    gdouble column_spacing;
    EContactPrintStyle *style;
    gboolean first_section;

    PangoFontDescription *letter_heading_font;
    gchar *section;
    gboolean first_contact;

    EBook *book;
    EBookQuery *query;

    GList *contact_list;
};

static gdouble
get_font_height (PangoFontDescription *desc)
{
    return pango_units_to_double (
        pango_font_description_get_size (desc));
}

static gdouble
get_font_width (GtkPrintContext *context,
                PangoFontDescription *desc,
                const gchar *text)
{
    PangoLayout *layout;
    gint width, height;

    g_return_val_if_fail (desc, .0);
    g_return_val_if_fail (text, .0);

    layout = gtk_print_context_create_pango_layout (context);

    pango_layout_set_font_description (layout, desc);
    pango_layout_set_text (layout, text, -1);
    pango_layout_set_width (layout, -1);
    pango_layout_set_indent (layout, 0);

    pango_layout_get_size (layout, &width, &height);

    g_object_unref (layout);

    return pango_units_to_double (width);
}

static void
e_contact_output (GtkPrintContext *context,
                  PangoFontDescription *font,
                  gdouble x, gdouble y, gdouble width,
                  const gchar *text)
{
    PangoLayout *layout;
    gdouble indent;
    cairo_t *cr;

    layout = gtk_print_context_create_pango_layout (context);

    if (width == -1 || get_font_width (context, font, text) <= width)
        indent = .0;
    else
        indent = get_font_width (context, font, "     ");

    pango_layout_set_font_description (layout, font);
    pango_layout_set_text (layout, text, -1);
    pango_layout_set_width (layout, pango_units_from_double (width));
    pango_layout_set_indent (layout, pango_units_from_double (indent));

    cr = gtk_print_context_get_cairo_context (context);

    cairo_save (cr);
    cairo_move_to (cr, x, y);
    pango_cairo_show_layout (cr, layout);
    cairo_restore (cr);

    g_object_unref (layout);
}

static gdouble
e_contact_text_height (GtkPrintContext *context,
                       PangoFontDescription *desc,
                       const gchar *text)
{
    PangoLayout *layout;
    gint width, height;

    layout = gtk_print_context_create_pango_layout (context);

    pango_layout_set_font_description (layout, desc);
    pango_layout_set_text (layout, text, -1);

    pango_layout_get_size (layout, &width, &height);

    g_object_unref (layout);

    return pango_units_to_double (height);
}

static void
e_contact_print_letter_heading (EContactPrintContext *ctxt, gchar *letter)
{
    PangoLayout *layout;
    PangoFontDescription *desc;
    PangoFontMetrics *metrics;
    gint width, height;
    cairo_t *cr;

    desc = ctxt->letter_heading_font;

    layout = gtk_print_context_create_pango_layout (ctxt->context);

    /* Make the rectangle thrice the average character width.
     * XXX Works well for English, what about other locales? */
    metrics = pango_context_get_metrics (
        pango_layout_get_context (layout),
        desc, pango_language_get_default ());
    width = pango_font_metrics_get_approximate_char_width (metrics) * 3;
    pango_font_metrics_unref (metrics);

    pango_layout_set_alignment (layout, PANGO_ALIGN_CENTER);
    pango_layout_set_font_description (layout, desc);
    pango_layout_set_text (layout, letter, -1);
    pango_layout_set_width (layout, width);
    pango_layout_get_size (layout, NULL, &height);

    /* Draw white text centered in a black rectangle. */

    cr = gtk_print_context_get_cairo_context (ctxt->context);

    cairo_save (cr);
    cairo_set_source_rgb (cr, .0, .0, .0);
    cairo_rectangle (
        cr, ctxt->x, ctxt->y,
        pango_units_to_double (width),
        pango_units_to_double (height));
    cairo_fill (cr);
    cairo_restore (cr);

    cairo_save (cr);
    cairo_move_to (cr, ctxt->x, ctxt->y);
    cairo_set_source_rgb (cr, 1., 1., 1.);
    pango_cairo_show_layout (cr, layout);
    cairo_restore (cr);

    ctxt->y += pango_units_to_double (height);
}

static void
e_contact_start_new_page (EContactPrintContext *ctxt)
{
    cairo_t *cr;

    cr = gtk_print_context_get_cairo_context (ctxt->context);

    cairo_show_page (cr);

        ctxt->x = ctxt->y = .0;
    ctxt->column = 0;
}

static void
e_contact_print_contact (EContact *contact, EContactPrintContext *ctxt)
{
    char *file_as;
    cairo_t *cr;
    int field;

    cr = gtk_print_context_get_cairo_context (ctxt->context);
    cairo_save(cr);
    ctxt->y += get_font_height (ctxt->style->headings_font) * .2; 

    file_as = e_contact_get (contact, E_CONTACT_FILE_AS);
                                                             
    if (ctxt->style->print_using_grey) {
        cairo_save (cr);
        cairo_set_source_rgb (cr, .85, .85, .85);
        cairo_rectangle (cr, ctxt->x, ctxt->y, ctxt->column_width,
            e_contact_text_height (ctxt->context,
                ctxt->style->headings_font, file_as));
        cairo_fill (cr);
        cairo_restore (cr);
    }

    e_contact_output (
        ctxt->context, ctxt->style->headings_font,
        ctxt->x, ctxt->y, ctxt->column_width + 4, file_as);
    ctxt->y += e_contact_text_height (
        ctxt->context, ctxt->style->headings_font, file_as);

    g_free (file_as);

    ctxt->y += get_font_height (ctxt->style->headings_font) * .2;

    for (field = E_CONTACT_FILE_AS; field != E_CONTACT_LAST_SIMPLE_STRING; field++) 
    {
        const gchar *value;
        gchar *text;

        value = e_contact_get_const (contact, field);
        if (value == NULL || *value == '\0')
            continue;

        text = g_strdup_printf ("%s:  %s",
            e_contact_pretty_name (field), value);

        e_contact_output (
            ctxt->context, ctxt->style->body_font, 
            ctxt->x, ctxt->y, -1, text);

        ctxt->y += e_contact_text_height (
            ctxt->context, ctxt->style->body_font, text);

        ctxt->y += .2 * get_font_height (ctxt->style->body_font);

        g_free (text);
    }

    ctxt->y += get_font_height (ctxt->style->headings_font) * .4 + 8;

    cairo_restore (cr);
}

static void
e_contact_start_new_column (EContactPrintContext *ctxt)
{
    if (++ctxt->column >= ctxt->style->num_columns)
        e_contact_start_new_page (ctxt);
    else {
        ctxt->x = ctxt->column *
            (ctxt->column_width + ctxt->column_spacing);
            ctxt->y = .0;
    }
}

static int
contact_compare (EContact *contact1, EContact *contact2)
{
    const gchar *field1, *field2;

    if (contact1 == NULL || contact2 == NULL)
        return 0;

    field1 = e_contact_get_const (contact1, E_CONTACT_FILE_AS);
    field2 = e_contact_get_const (contact2, E_CONTACT_FILE_AS);

    if (field1 != NULL && field2 != NULL)
        return g_utf8_collate (field1, field2);

    if (field1 != NULL || field2 != NULL)
        return (field1 != NULL) ? -1 : 1;

    field1 = e_contact_get_const (contact1, E_CONTACT_UID);
    field2 = e_contact_get_const (contact2, E_CONTACT_UID);

    g_assert (field1 != NULL && field2 != NULL);

    return strcmp (field1, field2);
}

static void
contacts_added (EBookView *book_view, const GList *contact_list,
                EContactPrintContext *ctxt)
{
    while (contact_list != NULL) {
        ctxt->contact_list = g_list_insert_sorted(
            ctxt->contact_list,
            g_object_ref (contact_list->data),
            (GCompareFunc) contact_compare);
        contact_list = contact_list->next;
    }
}

static void 
sequence_complete (EBookView *book_view, const GList *contact_list,
                   EFlag *book_view_started)
{
    e_flag_set (book_view_started);
}

static gboolean
get_bool (gchar *data)
{
    if (data)
        return (g_ascii_strcasecmp (data, "true") == 0);
    else
        return FALSE;
}

static void
get_string (gchar *data, gchar **variable)
{
    g_free (*variable);
    *variable = g_strdup ((data != NULL) ? data : "");
}

static gint
get_integer (gchar *data)
{
    return (data != NULL) ? atoi (data) : 0;
}

static gdouble
get_float (gchar *data)
{
    return (data != NULL) ? atof (data) : .0;
}

static void
get_font (gchar *data, PangoFontDescription **variable)
{
    PangoFontDescription *desc = NULL;

    if (data != NULL)
        desc = pango_font_description_from_string (data);

    if (desc != NULL) {
        pango_font_description_free (*variable);
        *variable = desc;
    }
}

static void
e_contact_build_style (EContactPrintStyle *style)
{
    xmlDocPtr styledoc;
    gchar *filename;

    style->title = g_strdup("");
    style->type = E_CONTACT_PRINT_TYPE_CARDS;
    style->sections_start_new_page = TRUE;
    style->num_columns = 2;
    style->blank_forms = 2;
    style->letter_headings = FALSE;

    style->headings_font = pango_font_description_from_string ("Sans Bold 8");
    style->body_font = pango_font_description_from_string ("Sans 6");

    style->print_using_grey = TRUE;
    style->paper_type = 0;
    style->paper_width = 8.5;
    style->paper_height = 11;
    style->paper_source = 0;
    style->top_margin = .5;
    style->left_margin = .5;
    style->bottom_margin = .5;
    style->right_margin = .5;
    style->page_size = 0;
    style->page_width = 2.75;
    style->page_height = 4.25;
#if 0
    style->page_width = 4.25;
    style->page_height = 5.5;
#endif
#if 0
    style->page_width = 5.5;
    style->page_height = 8.5;
#endif
    style->orientation_portrait = FALSE;

    style->header_font = pango_font_description_copy (style->body_font);

    style->left_header = g_strdup("");
    style->center_header = g_strdup("");
    style->right_header = g_strdup("");

    style->footer_font = pango_font_description_copy (style->body_font);

    style->left_footer = g_strdup ("");
    style->center_footer = g_strdup ("");
    style->right_footer = g_strdup ("");
    style->reverse_on_even_pages = FALSE;

    filename = g_build_filename (EVOLUTION_ECPSDIR, "medbook.ecps", NULL);
    styledoc = e_xml_parse_file (filename);
    g_free (filename);

    if (styledoc) {
        xmlNodePtr stylenode = xmlDocGetRootElement(styledoc);
        xmlNodePtr node;
        for (node = stylenode->children; node; node = node->next) {
            char *data = xmlNodeGetContent ( node );
            if ( !strcmp( node->name, "title" ) ) {
                get_string(data, &(style->title));
            } else if ( !strcmp( node->name, "type" ) ) {
                if (g_ascii_strcasecmp (data, "cards") == 0)
                    style->type = E_CONTACT_PRINT_TYPE_CARDS;
                else if (g_ascii_strcasecmp (data, "memo_style") == 0)
                    style->type = E_CONTACT_PRINT_TYPE_MEMO_STYLE;
                else if (g_ascii_strcasecmp (data, "phone_list") == 0)
                    style->type = E_CONTACT_PRINT_TYPE_PHONE_LIST;
            } else if ( !strcmp( node->name, "sections_start_new_page" ) ) {
                style->sections_start_new_page = get_bool(data);
            } else if ( !strcmp( node->name, "num_columns" ) ) {
                style->num_columns = get_integer(data);
            } else if ( !strcmp( node->name, "blank_forms" ) ) {
                style->blank_forms = get_integer(data);
            } else if ( !strcmp( node->name, "letter_headings" ) ) {
                style->letter_headings = get_bool(data);
            } else if ( !strcmp( node->name, "headings_font" ) ) {
                get_font( data, &(style->headings_font) );
            } else if ( !strcmp( node->name, "body_font" ) ) {
                get_font( data, &(style->body_font) );
            } else if ( !strcmp( node->name, "print_using_grey" ) ) {
                style->print_using_grey = get_bool(data);
            } else if ( !strcmp( node->name, "paper_width" ) ) {
                style->paper_width = get_float(data);
            } else if ( !strcmp( node->name, "paper_height" ) ) {
                style->paper_height = get_float(data);
            } else if ( !strcmp( node->name, "top_margin" ) ) {
                style->top_margin = get_float(data);
            } else if ( !strcmp( node->name, "left_margin" ) ) {
                style->left_margin = get_float(data);
            } else if ( !strcmp( node->name, "bottom_margin" ) ) {
                style->bottom_margin = get_float(data);
            } else if ( !strcmp( node->name, "right_margin" ) ) {
                style->right_margin = get_float(data);
            } else if ( !strcmp( node->name, "page_width" ) ) {
                style->page_width = get_float(data);
            } else if ( !strcmp( node->name, "page_height" ) ) {
                style->page_height = get_float(data);
            } else if ( !strcmp( node->name, "orientation" ) ) {
                if ( data ) {
                    style->orientation_portrait =
                        (g_ascii_strcasecmp (data, "landscape") != 0);
                } else {
                    style->orientation_portrait = TRUE;
                }
            } else if ( !strcmp( node->name, "header_font" ) ) {
                get_font( data, &(style->header_font) );
            } else if ( !strcmp( node->name, "left_header" ) ) {
                get_string(data, &(style->left_header));
            } else if ( !strcmp( node->name, "center_header" ) ) {
                get_string(data, &(style->center_header));
            } else if ( !strcmp( node->name, "right_header" ) ) {
                get_string(data, &(style->right_header));
            } else if ( !strcmp( node->name, "footer_font" ) ) {
                get_font( data, &(style->footer_font) );
            } else if ( !strcmp( node->name, "left_footer" ) ) {
                get_string(data, &(style->left_footer));
            } else if ( !strcmp( node->name, "center_footer" ) ) {
                get_string(data, &(style->center_footer));
            } else if ( !strcmp( node->name, "right_footer" ) ) {
                get_string(data, &(style->right_footer));
            } else if ( !strcmp( node->name, "reverse_on_even_pages" ) ) {
                style->reverse_on_even_pages = get_bool(data);
            }
            if ( data )
                xmlFree (data);
        }
        xmlFreeDoc(styledoc);
    }

}

static void
load_contacts (EContactPrintContext *ctxt)
{
    /* Load contacts from the EBook.  This is an asynchronous operation
     * but we force it to be synchronous here. */

    EBookView *book_view;
    EFlag *book_view_started;

    book_view_started = e_flag_new ();

    e_book_get_book_view (
        ctxt->book, ctxt->query, NULL, -1, &book_view, NULL);

    g_signal_connect (
        book_view, "contacts_added",
        G_CALLBACK (contacts_added), ctxt);
    g_signal_connect (
        book_view, "sequence_complete",
        G_CALLBACK (sequence_complete), book_view_started);

    e_book_view_start (book_view);

    while (!e_flag_is_set (book_view_started))
        g_main_context_iteration (NULL, TRUE);

    e_flag_free (book_view_started);
}

static void
free_contacts (EContactPrintContext *ctxt)
{
    g_list_foreach (ctxt->contact_list, (GFunc) g_object_unref, NULL);
    g_list_free (ctxt->contact_list);
}

static void
contact_begin_print (GtkPrintOperation *operation,
                     GtkPrintContext *context,
                     EContactPrintContext *ctxt)
{
    GtkPageSetup *setup;
    gdouble page_width;

    e_contact_build_style (ctxt->style);

    setup = gtk_print_context_get_page_setup (context);
    page_width = gtk_page_setup_get_page_width (setup, GTK_UNIT_POINTS);

    ctxt->context = context;
    ctxt->x = ctxt->y = .0;
    ctxt->column = 0;
    ctxt->first_contact = TRUE;
    ctxt->first_section = TRUE;
    ctxt->section = NULL;

    ctxt->column_spacing = gtk_print_context_get_dpi_x (context) / 4;
    ctxt->column_width = (page_width + ctxt->column_spacing) /
        ctxt->style->num_columns - ctxt->column_spacing;

    ctxt->letter_heading_font = pango_font_description_new ();
    pango_font_description_set_family (
        ctxt->letter_heading_font,
        pango_font_description_get_family (
            ctxt->style->headings_font));
    pango_font_description_set_size (
        ctxt->letter_heading_font,
        pango_font_description_get_size (
            ctxt->style->headings_font) * 1.5);

    if (ctxt->book != NULL)
        load_contacts (ctxt);
}

static void
contact_draw (EContact *contact, EContactPrintContext *ctxt)
{
    GtkPageSetup *setup;
    gdouble page_height;
    gchar *file_as;
    gboolean new_section = FALSE;

    setup = gtk_print_context_get_page_setup (ctxt->context);
    page_height = gtk_page_setup_get_page_height (setup, GTK_UNIT_POINTS);

    file_as = e_contact_get (contact, E_CONTACT_FILE_AS);

    if (file_as != NULL) {
        gchar *section;
        gsize width;

        width = g_utf8_next_char (file_as) - file_as;
        section = g_utf8_strup (file_as, width);

        new_section = (ctxt->section == NULL ||
            g_utf8_collate (ctxt->section, section) != 0);

        if (new_section) {
            g_free (ctxt->section);
            ctxt->section = section;
        } else
            g_free (section);
    }

    if (new_section) {
        if (!ctxt->first_contact) {
            if (ctxt->style->sections_start_new_page)
                e_contact_start_new_page (ctxt);
            else if (ctxt->y > page_height)
                e_contact_start_new_column (ctxt);
        }
        if (ctxt->style->letter_headings)
            e_contact_print_letter_heading (ctxt, ctxt->section);
        ctxt->first_section = FALSE;
    } 

    else if (!ctxt->first_contact && (ctxt->y > page_height)) { 
        e_contact_start_new_column (ctxt);
        if (ctxt->style->letter_headings)
            e_contact_print_letter_heading (ctxt, ctxt->section);
    }

    e_contact_print_contact (contact, ctxt);

    ctxt->first_contact = FALSE;
}

static void
contact_draw_page (GtkPrintOperation *operation,
                   GtkPrintContext *context,
                   gint page_nr,
                   EContactPrintContext *ctxt)
{
    g_list_foreach (ctxt->contact_list, (GFunc) contact_draw, ctxt);
}
 
static void
contact_end_print (GtkPrintOperation *operation,
                   GtkPrintContext *context,
                   EContactPrintContext *ctxt)
{
    pango_font_description_free (ctxt->style->headings_font);
    pango_font_description_free (ctxt->style->body_font);
    pango_font_description_free (ctxt->style->header_font);
    pango_font_description_free (ctxt->style->footer_font);
    pango_font_description_free (ctxt->letter_heading_font);

    g_free (ctxt->section);

    if (ctxt->book != NULL)
        free_contacts (ctxt);
}

void
e_contact_print (EBook *book, EBookQuery *query,
                 GList *contact_list, GtkPrintOperationAction action)
{
    GtkPrintOperation *operation;
    EContactPrintContext ctxt;
    EContactPrintStyle style;

    if (book != NULL) {
        ctxt.book = book;
        ctxt.query = query;
        ctxt.contact_list = NULL;
    } else {
        ctxt.book = NULL;
        ctxt.query = NULL;
        ctxt.contact_list = contact_list;
    }
    ctxt.style = &style;

    operation = e_print_operation_new ();
    gtk_print_operation_set_n_pages (operation, 1);

    g_signal_connect (
        operation, "begin-print",
        G_CALLBACK (contact_begin_print), &ctxt);
    g_signal_connect (
        operation, "draw_page",
        G_CALLBACK (contact_draw_page), &ctxt);
    g_signal_connect (
        operation, "end-print",
        G_CALLBACK (contact_end_print), &ctxt);

    gtk_print_operation_run (operation, action, NULL, NULL);

    g_object_unref (operation);
}