aboutsummaryrefslogblamecommitdiffstats
path: root/calendar/gui/dialogs/meeting-page.c
blob: 37f873b2d40e3fe97a162a22e3f991c1afcf3629 (plain) (tree)
1
2
3
4
5
6
7
8
9
10
                                                                           








                                                           


                                                                   














                                                                            
                 
                         
                          

                                
                          

                                 
                                
                        
                                         

                             
                                  
                                    
                                  

                                  
                                  

                                   
                          
                             
                              
                         
                           
 

                                               
                                
                                     

                                            
                            
        
                                
                               

                              
                       
        




                                         
                            
                                   
                             


                                            
                       
                          
                          
                             
                             
 


                                    

                 
                                               
                          
                          
        
                          



  
                                                    

                                                                 
                                                                  
                                                                                      
                                                                                        
                                                                                                       
                                                                





                                                     
                                   

                                                          
                                              
 
                                                                
                                                                              

                                                                        

                                              
 
                                                       






                                                      
        

                                              
        
                                                     
 
                          
 
                              

                                     

                         
                          


                            
                               
        


                               






























                                                                                                               
           
                                                               
 

                                          
        

                                                                        
                                          
                
                                                                         
                
                                                                      


                                                          
        
                                                                 


                                                        



                                      
                                        
 
              
        
                                            
                                                                  

 

                                       
                                       


                                 
        





                                                    
                               
                                            
        
                                                    
                                                         


                                          
        
                                     
        


                                              
                        
                                           


                                 




                                               



                                        


                           

                                                                                  
















                                              












                                                     




                                           
                   

                           
 
                                                                          


                                                                                      


                                                                                                                 


                                                                        


                                                               
                                                                                                   
 
                                                                             



                                                                
                               
                              

 


                                      
                                   
                                               
                             

                          
        


                                                     




                                                                                         
                                                               



                                                                                                
                                                                            

 
                                               
               
                                                                     


                                 
                                         
        




                                    
        
                                
                               
                                            

                          
                                                    

                                                          

                              
 
                                        
                                                  


                                                                                                
                                                                

                                                                 

                                                                                 
                                      


                                                                         
                                                                          
                                                                 

                                                                                               
                                                                                       
                                                      
                                
                                                                 

                                                                                               
                                                                                       


                                                                               
                                                       

                         
                                                                                                                                                                                                

                                                                   


                                                                                          
 
                                                                                          



                                              
                



                                                
                                                                                            



                                                                                                                 



                                                                                                    
                 
         
        
                               
                                                                 
 

                                  
                    

 
                                                 
               
                                                                       
 

                                 
                                                                    
 


                                    
                              

                                   
 
                                                                         
                                                
                
                                  






                                                                                         
                                                          
                                                                  
                                     


                                                                     
        
                                       
                                           
                                                                 
 
                              
         
 
                                                                       
                                                  
                                                                   

                             




                                                                           
                                                                                         






                                                                                











                                                                                                          








                                                                                  

                    







                                                                               
                                                        
                                 

                             





                                                       


                             

                                         


                                                                            
                                                                          




                                                        
                                    
                                                                              
 
                                               
                                                       
                                           
                                                                                
                                                                                       
 
                                             


                                                                         


                                        
                                              
                                     


                                                 
                                           
 

         
                              
                                  
                               
                            
                               

                                        


                                                 

 
           
                                                 





                                    


                           
        







                                                                                                                 

                                                                                                                          
                        
                                                                                



                                                                   

 











                                                         


                                                      
 
                                                                   
        
                               
                              

 





                                                                                   

                                                                          
                                                                                                                

         


                                                                    
















                                                                             
                                                                           
                                                                       
                                    
                 






                                                       
           



                                                           
                                                                                          


                           

                                                                 











                                                                                                            


                                                                                                      






                                                                        




                                                                      

                                                                                                                  

                                                                  

                        
        



                                  

                                                      


                                    
                                  
                         
                                 





                                                                                  
                                                                                          


                                                                   
                                       
        

















                                                                                                                        
        





                                                                                                 


                                                            
                                                                  

         
                                                                
                            

 





                                             


                                    
 
                                                                   










                                 

                                                                        
 




                                                                                    
 



                                                                                          
                                                                                 
































                                                                                                       

 
           







                                                                          







                                                        
                                                           
 
                                  
 
                                        


      



                                     

  
                                          

                                                                                                                   

  





                                                         
           
                                                                                 
 

                                 
                             


                          
                                  



                             

                           


                                                                  
 
                                                   









                                                                                                                            
                                                                                                          












                                                                                                           
 
                                                                           
 

                                                                                
        
                                                                                
                                                                          
                                                                                  


                    
 

























                                                                                                                        











                                                 









                                                                             
                                                               
                                     

                                 
                                     

                              
                                          
                      
                    
                                    
                        
        

                           





                                                           
                         
                                                                            



                                   
                                                                                     

                            

                                 
                                                                    
                            
 




                                                               
                                                   
                           
                

                                                                               
                                                                       



                                                                      







                                                                                   
         
        


                                         
                           
 



                                                                                             
 



                                                         
                                                              
                           

                          
                                                                 
                                                                                  

                                                                        
                                                                                                                    
                                                                                                        
 

                                                       
                                                                                                              
                                                                                      

                                                                             
                                                                         
        
                                                             
                                                                                                  
 


                                     

                                                                      
                                                                                                  
 











                                                                             
                                                   


                           
                                                       
                                                           
                                       




                            
 







                                
               








                                                             
                                              



                                                            
                                                  
 
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/* Evolution calendar - Main page of the task editor dialog
 *
 * Copyright (C) 2001 Ximian, Inc.
 *
 * Authors: Federico Mena-Quintero <federico@ximian.com>
 *          Miguel de Icaza <miguel@ximian.com>
 *          Seth Alves <alves@hungry.com>
 *          JP Rosevear <jpr@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., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA.
 */

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

#include <glib.h>
#include <gtk/gtkcombo.h>
#include <gtk/gtksignal.h>
#include <gtk/gtktogglebutton.h>
#include <gtk/gtkvbox.h>
#include <gtk/gtkwindow.h>
#include <gtk/gtktreeselection.h>
#include <gdk/gdkkeysyms.h>
#include <libgnome/gnome-i18n.h>
#include <glade/glade.h>
#include <libgnomeui/gnome-stock-icons.h>
#include <misc/e-gui-utils.h>
#include <misc/e-dateedit.h>
#include <e-util/e-dialog-utils.h>
#include <e-util/e-dialog-widgets.h>
#include <e-util/e-util-private.h>

#include "../calendar-component.h"
#include "../e-meeting-attendee.h"
#include "../e-meeting-store.h"
#include "../e-meeting-list-view.h"
#include "../itip-utils.h"
#include "comp-editor-util.h"
#include "e-delegate-dialog.h"
#include "meeting-page.h"
#include "../e-cal-popup.h"

/* Private part of the MeetingPage structure */
struct _MeetingPagePrivate {
    /* Lists of attendees */
    GPtrArray *deleted_attendees;

    /* To use in case of cancellation */
    ECalComponent *comp;
    
    /* List of identities */
    EAccountList *accounts;
    EMeetingAttendee *ia;
    char *default_address;
    char *user_add;
    
    /* Glade XML data */
    GladeXML *xml;

    /* Widgets from the Glade file */
    GtkWidget *main;
    GtkWidget *list_box;
    GtkWidget *organizer_table;
    GtkWidget *organizer;
    GtkWidget *existing_organizer_table;
    GtkWidget *existing_organizer;
    GtkWidget *existing_organizer_btn;
    GtkWidget *add;
    GtkWidget *remove;
    GtkWidget *invite;
    GtkWidget *att_label;
    GtkWidget *org_label;

    /* ListView stuff */
    EMeetingStore *model;
    EMeetingListView *list_view;
    gint row;
    
    /* For handling who the organizer is */
    gboolean user_org;
    gboolean existing;
    
        gboolean updating;
};



static void meeting_page_finalize (GObject *object);

static GtkWidget *meeting_page_get_widget (CompEditorPage *page);
static void meeting_page_focus_main_widget (CompEditorPage *page);
static gboolean meeting_page_fill_widgets (CompEditorPage *page, ECalComponent *comp);
static gboolean meeting_page_fill_component (CompEditorPage *page, ECalComponent *comp);
static void attendee_added_cb (EMeetingListView *emlv, EMeetingAttendee *attendee, gpointer user_data);
G_DEFINE_TYPE (MeetingPage, meeting_page, TYPE_COMP_EDITOR_PAGE)

/* Class initialization function for the task page */
static void
meeting_page_class_init (MeetingPageClass *class)
{
    CompEditorPageClass *editor_page_class;
    GObjectClass *object_class;

    editor_page_class = (CompEditorPageClass *) class;
    object_class = (GObjectClass *) class;

    editor_page_class->get_widget = meeting_page_get_widget;
    editor_page_class->focus_main_widget = meeting_page_focus_main_widget;
    editor_page_class->fill_widgets = meeting_page_fill_widgets;
    editor_page_class->fill_component = meeting_page_fill_component;
    editor_page_class->set_summary = NULL;
    editor_page_class->set_dates = NULL;

    object_class->finalize = meeting_page_finalize;
}

/* Object initialization function for the task page */
static void
meeting_page_init (MeetingPage *mpage)
{
    MeetingPagePrivate *priv;
    
    priv = g_new0 (MeetingPagePrivate, 1);
    mpage->priv = priv;
    
    priv->deleted_attendees = g_ptr_array_new ();

    priv->comp = NULL;

    priv->accounts = NULL;
    priv->ia = NULL;
    priv->default_address = NULL;
    
    priv->xml = NULL;
    priv->main = NULL;
    priv->invite = NULL;
    
    priv->model = NULL;
    priv->list_view = NULL;
    
    priv->updating = FALSE;
}

static EAccount *
get_current_account (MeetingPage *mpage)
{   
    MeetingPagePrivate *priv;
    EIterator *it;
    const char *str;
    
    priv = mpage->priv;

    str = gtk_entry_get_text (GTK_ENTRY (GTK_COMBO (priv->organizer)->entry));
    if (!str)
        return NULL;
    
    for (it = e_list_get_iterator((EList *)priv->accounts); e_iterator_is_valid(it); e_iterator_next(it)) {
        EAccount *a = (EAccount *)e_iterator_get(it);
        char *full = g_strdup_printf("%s <%s>", a->id->name, a->id->address);

        if (!strcmp (full, str)) {
            g_free (full);
            g_object_unref (it);

            return a;
        }
    
        g_free (full);
    }
    g_object_unref (it);
    
    return NULL;    
}

static void
set_attendees (ECalComponent *comp, const GPtrArray *attendees)
{
    GSList *comp_attendees = NULL, *l;
    int i;
    
    for (i = 0; i < attendees->len; i++) {
        EMeetingAttendee *ia = g_ptr_array_index (attendees, i);
        ECalComponentAttendee *ca;
        
        ca = e_meeting_attendee_as_e_cal_component_attendee (ia);
        
        comp_attendees = g_slist_prepend (comp_attendees, ca);
        
    }
    comp_attendees = g_slist_reverse (comp_attendees);
    
    e_cal_component_set_attendee_list (comp, comp_attendees);
    
    for (l = comp_attendees; l != NULL; l = l->next)
        g_free (l->data);   
    g_slist_free (comp_attendees);
}

static void
cleanup_attendees (GPtrArray *attendees)
{
    int i;
    
    for (i = 0; i < attendees->len; i++)
        g_object_unref (g_ptr_array_index (attendees, i));
}

/* Destroy handler for the task page */
static void
meeting_page_finalize (GObject *object)
{
    MeetingPage *mpage;
    MeetingPagePrivate *priv;
    
    g_return_if_fail (object != NULL);
    g_return_if_fail (IS_MEETING_PAGE (object));

    mpage = MEETING_PAGE (object);
    priv = mpage->priv;

    if (priv->comp != NULL)
        g_object_unref (priv->comp);
    
    cleanup_attendees (priv->deleted_attendees);
    g_ptr_array_free (priv->deleted_attendees, TRUE);

    if (priv->ia != NULL)
        g_object_unref (priv->ia);
    
    g_object_unref (priv->model);
    
    if (priv->main)
        gtk_widget_unref (priv->main);

    if (priv->xml) {
        g_object_unref (priv->xml);
        priv->xml = NULL;
    }

    if (priv->default_address) {
        g_free (priv->default_address);
        priv->default_address = NULL;
    }

    if (priv->user_add) {
        g_free (priv->user_add);
        priv->user_add = NULL;
    }
    g_free (priv);
    mpage->priv = NULL;

    if (G_OBJECT_CLASS (meeting_page_parent_class)->finalize)
        (* G_OBJECT_CLASS (meeting_page_parent_class)->finalize) (object);
}



/* get_widget handler for the task page */
static GtkWidget *
meeting_page_get_widget (CompEditorPage *page)
{
    MeetingPage *mpage;
    MeetingPagePrivate *priv;

    mpage = MEETING_PAGE (page);
    priv = mpage->priv;

    return priv->main;
}

/* focus_main_widget handler for the task page */
static void
meeting_page_focus_main_widget (CompEditorPage *page)
{
    MeetingPage *mpage;
    MeetingPagePrivate *priv;

    mpage = MEETING_PAGE (page);
    priv = mpage->priv;

    gtk_widget_grab_focus (priv->organizer);
}

/* Fills the widgets with default values */
static void
clear_widgets (MeetingPage *mpage)
{
    MeetingPagePrivate *priv;
    gchar *str;

    priv = mpage->priv;

    if (COMP_EDITOR_PAGE (mpage)->flags & COMP_EDITOR_PAGE_DELEGATE) {
        str = g_strdup_printf ("<b>%s</b>", _("Dele_gatees"));
        gtk_label_set_markup_with_mnemonic ((GtkLabel *)priv->att_label, str);
        g_free (str);
    }

    if (e_cal_get_static_capability (COMP_EDITOR_PAGE (mpage)->client, CAL_STATIC_CAPABILITY_NO_ORGANIZER)) {
        str = g_strdup_printf ("<b>%s</b>", _("From:"));
        gtk_label_set_markup (GTK_LABEL (priv->org_label), str);
        g_free (str);
        gtk_widget_hide (priv->existing_organizer_btn);
    }
                
    gtk_entry_set_text (GTK_ENTRY (GTK_COMBO (priv->organizer)->entry), priv->default_address);

    gtk_label_set_text (GTK_LABEL (priv->existing_organizer), _("None"));

    gtk_widget_show (priv->organizer_table);
    gtk_widget_hide (priv->existing_organizer_table);   

    priv->existing = FALSE;
    priv->user_org = TRUE;
}

static void
sensitize_widgets (MeetingPage *mpage)
{
    gboolean read_only = FALSE;
    MeetingPagePrivate *priv = mpage->priv;
    GError *error = NULL;
    guint32 flags;
    gboolean delegate;
    
    flags = COMP_EDITOR_PAGE (mpage)->flags;
    delegate = flags & COMP_EDITOR_PAGE_DELEGATE;

    if (!e_cal_is_read_only (COMP_EDITOR_PAGE (mpage)->client, &read_only, &error)) {
        if (error->code != E_CALENDAR_STATUS_BUSY)
            read_only = TRUE;
        g_error_free (error);
    }   
    gtk_widget_set_sensitive (priv->organizer, !read_only);
    gtk_widget_set_sensitive (priv->existing_organizer_btn, (!read_only && priv->user_org));
    gtk_widget_set_sensitive (priv->add, (!read_only &&  priv->user_org) || delegate);
    gtk_widget_set_sensitive (priv->remove, (!read_only &&  priv->user_org) || delegate);
    gtk_widget_set_sensitive (priv->invite, (!read_only &&  priv->user_org) || delegate);
    gtk_widget_set_sensitive (GTK_WIDGET (priv->list_view), !read_only);
}

/* fill_widgets handler for the meeting page */
static gboolean
meeting_page_fill_widgets (CompEditorPage *page, ECalComponent *comp)
{
    MeetingPage *mpage;
    MeetingPagePrivate *priv;
    ECalComponentOrganizer organizer;
    
    mpage = MEETING_PAGE (page);
    priv = mpage->priv;

    priv->updating = TRUE;
    
    
    /* Clean out old data */
    if (priv->comp != NULL)
        g_object_unref (priv->comp);
    priv->comp = NULL;
    
    cleanup_attendees (priv->deleted_attendees);
    g_ptr_array_set_size (priv->deleted_attendees, 0);
    
    /* Clean the screen */
    clear_widgets (mpage);

    /* Component for cancellation */
    priv->comp = e_cal_component_clone (comp);

    priv->user_add = itip_get_comp_attendee (comp, COMP_EDITOR_PAGE (mpage)->client);   

    /* If there is an existing organizer show it properly */
    if (e_cal_component_has_organizer (comp)) {
        e_cal_component_get_organizer (comp, &organizer);
        if (organizer.value != NULL) {
            const gchar *strip = itip_strip_mailto (organizer.value);
            gchar *string;

            gtk_widget_hide (priv->organizer_table);
            gtk_widget_show (priv->existing_organizer_table);
            if (itip_organizer_is_user (comp, page->client)) {
                if (e_cal_get_static_capability (
                        page->client,
                        CAL_STATIC_CAPABILITY_ORGANIZER_NOT_EMAIL_ADDRESS))
                    gtk_widget_hide (priv->existing_organizer_btn);
                priv->user_org = TRUE;
            } else {
                if (e_cal_get_static_capability (
                        page->client,
                        CAL_STATIC_CAPABILITY_ORGANIZER_NOT_EMAIL_ADDRESS))
                    gtk_widget_hide (priv->existing_organizer_btn);
                gtk_widget_set_sensitive (priv->invite, FALSE);
                gtk_widget_set_sensitive (priv->add, FALSE);
                gtk_widget_set_sensitive (priv->remove, FALSE);
                priv->user_org = FALSE;
            }
            
            if (e_cal_get_static_capability (COMP_EDITOR_PAGE (mpage)->client, CAL_STATIC_CAPABILITY_NO_ORGANIZER) && (COMP_EDITOR_PAGE (mpage)->flags & COMP_EDITOR_PAGE_DELEGATE))
                string = g_strdup (priv->user_add);
            else if ( organizer.cn != NULL)
                string = g_strdup_printf ("%s <%s>", organizer.cn, strip);
            else
                string = g_strdup (strip);

            gtk_label_set_text (GTK_LABEL (priv->existing_organizer), string);
            g_free (string);

            priv->existing = TRUE;
        }
    } else {
        EAccount *a;
        
        a = get_current_account (mpage);
        if (a != NULL) {
            priv->ia = e_meeting_store_add_attendee_with_defaults (priv->model);
            g_object_ref (priv->ia);

            e_meeting_attendee_set_address (priv->ia, g_strdup_printf ("MAILTO:%s", a->id->address));
            e_meeting_attendee_set_cn (priv->ia, g_strdup (a->id->name));
            if (page->client && e_cal_get_organizer_must_accept (page->client))
                e_meeting_attendee_set_status (priv->ia, ICAL_PARTSTAT_NEEDSACTION);
            else
                e_meeting_attendee_set_status (priv->ia, ICAL_PARTSTAT_ACCEPTED);
        }
    }
    
    priv->updating = FALSE;
    priv->user_org = page->flags & COMP_EDITOR_PAGE_USER_ORG;

    sensitize_widgets (mpage);

    return TRUE;
}

/* fill_component handler for the meeting page */
static gboolean
meeting_page_fill_component (CompEditorPage *page, ECalComponent *comp)
{
    MeetingPage *mpage;
    MeetingPagePrivate *priv;
    ECalComponentOrganizer organizer = {NULL, NULL, NULL, NULL};

    mpage = MEETING_PAGE (page);
    priv = mpage->priv;

    if (!priv->existing) {
        EAccount *a;
        gchar *addr = NULL;

        /* Find the identity for the organizer or sentby field */
        a = get_current_account (mpage);
        
        /* Sanity Check */
        if (a == NULL) {
            e_notice (page, GTK_MESSAGE_ERROR,
                  _("The organizer selected no longer has an account."));
            return FALSE;           
        }
        
        if (a->id->address == NULL || strlen (a->id->address) == 0) {
            e_notice (page, GTK_MESSAGE_ERROR,
                  _("An organizer is required."));
            return FALSE;
        } 

        addr = g_strdup_printf ("MAILTO:%s", a->id->address);
    
        organizer.value = addr;
        organizer.cn = a->id->name;
        e_cal_component_set_organizer (comp, &organizer);

        g_free (addr);
    }

    if (e_meeting_store_count_actual_attendees (priv->model) < 1) {
        e_notice (page, GTK_MESSAGE_ERROR,
              _("At least one attendee is required."));
        return FALSE;
    }
    

    if (COMP_EDITOR_PAGE (mpage)->flags & COMP_EDITOR_PAGE_DELEGATE ) {
        GSList *attendee_list, *l;
        int i;
        const GPtrArray *attendees = e_meeting_store_get_attendees (priv->model);

        e_cal_component_get_attendee_list (priv->comp, &attendee_list);
        
        for (i = 0; i < attendees->len; i++) {
            EMeetingAttendee *ia = g_ptr_array_index (attendees, i);
            ECalComponentAttendee *ca;
        
            /* Remove the duplicate user from the component if present */
            if (e_meeting_attendee_is_set_delto (ia)) {
                for (l = attendee_list; l; l = l->next) {
                    ECalComponentAttendee *a = l->data;

                    if (g_str_equal (a->value, e_meeting_attendee_get_address (ia))) {
                        attendee_list = g_slist_remove (attendee_list, l->data);
                        break;
                    }
                }
            }
            
            ca = e_meeting_attendee_as_e_cal_component_attendee (ia);

            attendee_list = g_slist_append (attendee_list, ca);
        }
        e_cal_component_set_attendee_list (comp, attendee_list);
        e_cal_component_free_attendee_list (attendee_list);
    } else 
        set_attendees (comp, e_meeting_store_get_attendees (priv->model));

    
    return TRUE;
}



/* Gets the widgets from the XML file and returns if they are all available. */
static gboolean
get_widgets (MeetingPage *mpage)
{
    CompEditorPage *page = COMP_EDITOR_PAGE (mpage);
    MeetingPagePrivate *priv;
    GSList *accel_groups;
    GtkWidget *toplevel;

    priv = mpage->priv;

#define GW(name) glade_xml_get_widget (priv->xml, name)

    priv->main = GW ("meeting-page");
    if (!priv->main)
        return FALSE;

    priv->list_box = GW ("list-box");

    /* Get the GtkAccelGroup from the toplevel window, so we can install
       it when the notebook page is mapped. */
    toplevel = gtk_widget_get_toplevel (priv->main);
    accel_groups = gtk_accel_groups_from_object (G_OBJECT (toplevel));
    if (accel_groups) {
        page->accel_group = accel_groups->data;
        gtk_accel_group_ref (page->accel_group);
    }

    gtk_widget_ref (priv->main);
    gtk_container_remove (GTK_CONTAINER (priv->main->parent), priv->main);

    /* For making the user the organizer */
    priv->organizer_table = GW ("organizer-table");
    priv->organizer = GW ("organizer");
    gtk_combo_set_value_in_list (GTK_COMBO (priv->organizer), FALSE, FALSE);
    gtk_entry_set_editable (GTK_ENTRY (GTK_COMBO (priv->organizer)->entry), FALSE);

    /* For showing existing organizers */
    priv->existing_organizer_table = GW ("existing-organizer-table");
    priv->existing_organizer = GW ("existing-organizer");
    priv->existing_organizer_btn = GW ("existing-organizer-button");

    /* Buttons */
    priv->add = GW ("add-attendee");
    priv->remove = GW ("remove-attendee");
    priv->invite = GW ("invite");

    /* Attendees Label */
    priv->att_label = GW ("attendees-label");
    priv->org_label = GW ("org-label");

#undef GW

    return (priv->list_box
        && priv->att_label
        && priv->invite
        && priv->add
        && priv->remove
        && priv->organizer_table
        && priv->organizer
        && priv->existing_organizer_table
        && priv->existing_organizer
        && priv->existing_organizer_btn);
}

static void
org_changed_cb (GtkWidget *widget, gpointer data)
{
    MeetingPage *mpage;
    MeetingPagePrivate *priv;
    
    mpage = MEETING_PAGE (data);
    priv = mpage->priv;

    if (priv->updating)
        return;
    
    if (!priv->existing && priv->ia != NULL) {
        EAccount *a;
        
        a = get_current_account (mpage);
        if (a != NULL) {
            e_meeting_attendee_set_address (priv->ia, g_strdup_printf ("MAILTO:%s", a->id->address));
            e_meeting_attendee_set_cn (priv->ia, g_strdup (a->id->name));
            
            if (!e_meeting_store_find_attendee (priv->model, e_meeting_attendee_get_address (priv->ia), NULL))
                e_meeting_store_add_attendee (priv->model, priv->ia);
        } else {
            e_meeting_store_remove_attendee (priv->model, priv->ia);
        }
    }
        
    comp_editor_page_notify_changed (COMP_EDITOR_PAGE (mpage));
}

/* Function called to change the organizer */
static void
change_clicked_cb (GtkWidget *widget, gpointer data) 
{
    MeetingPage *mpage;
    MeetingPagePrivate *priv;
    
    mpage = MEETING_PAGE (data);
    priv = mpage->priv;

    gtk_widget_show (priv->organizer_table);
    gtk_widget_hide (priv->existing_organizer_table);
    gtk_widget_set_sensitive (priv->invite, TRUE);
    gtk_widget_set_sensitive (priv->add, TRUE);
    gtk_widget_set_sensitive (priv->remove, TRUE);

    comp_editor_page_notify_changed (COMP_EDITOR_PAGE (mpage));
    
    priv->existing = FALSE;
    priv->user_org = TRUE;
}

static void
add_clicked_cb (GtkButton *btn, MeetingPage *mpage)
{
    EMeetingAttendee *attendee;
    
    attendee = e_meeting_store_add_attendee_with_defaults (mpage->priv->model);

    if (COMP_EDITOR_PAGE (mpage)->flags & COMP_EDITOR_PAGE_DELEGATE) {
        e_meeting_attendee_set_delfrom (attendee, g_strdup_printf ("MAILTO:%s", mpage->priv->user_add));
    }

    e_meeting_list_view_edit (mpage->priv->list_view, attendee);
}

static gboolean
existing_attendee (EMeetingAttendee *ia, ECalComponent *comp) 
{
    GSList *attendees, *l;
    const gchar *ia_address;
    
    ia_address = itip_strip_mailto (e_meeting_attendee_get_address (ia));
    if (!ia_address)
        return FALSE;
    
    e_cal_component_get_attendee_list (comp, &attendees);

    for (l = attendees; l; l = l->next) {
        ECalComponentAttendee *attendee = l->data;
        const char *address;
        
        address = itip_strip_mailto (attendee->value);
        if (address && !g_ascii_strcasecmp (ia_address, address)) {
            e_cal_component_free_attendee_list (attendees);
            return TRUE;
        }
    }
    
    e_cal_component_free_attendee_list (attendees);
    
    return FALSE;
}

static void
remove_attendee (MeetingPage *mpage, EMeetingAttendee *ia) 
{
    MeetingPagePrivate *priv;
    int pos = 0;
    gboolean delegate = (COMP_EDITOR_PAGE (mpage)->flags & COMP_EDITOR_PAGE_DELEGATE);
    
    priv = mpage->priv;

    /* If the user deletes the organizer attendee explicitly,
       assume they no longer want the organizer showing up */
    if (ia == priv->ia) {
        g_object_unref (priv->ia);
        priv->ia = NULL;
    }   
        
    /* If this was a delegatee, no longer delegate */
    if (e_meeting_attendee_is_set_delfrom (ia)) {
        EMeetingAttendee *ib;
        
        ib = e_meeting_store_find_attendee (priv->model, e_meeting_attendee_get_delfrom (ia), &pos);
        if (ib != NULL) {
            e_meeting_attendee_set_delto (ib, NULL);
            
            if (!delegate) 
                e_meeting_attendee_set_edit_level (ib,  E_MEETING_ATTENDEE_EDIT_FULL);
        }       
    }
    
    /* Handle deleting all attendees in the delegation chain */ 
    while (ia != NULL) {
        EMeetingAttendee *ib = NULL;

        if (existing_attendee (ia, priv->comp)) {
            g_object_ref (ia);
            g_ptr_array_add (priv->deleted_attendees, ia);
        }
        
        if (e_meeting_attendee_get_delto (ia) != NULL)
            ib = e_meeting_store_find_attendee (priv->model, e_meeting_attendee_get_delto (ia), NULL);
        e_meeting_store_remove_attendee (priv->model, ia);

        ia = ib;
    }
    
    sensitize_widgets (mpage);
}

static void
remove_clicked_cb (GtkButton *btn, MeetingPage *mpage)
{
    MeetingPagePrivate *priv;
    EMeetingAttendee *ia;
    GtkTreeSelection *selection;
    GList *paths = NULL, *tmp;
    GtkTreeIter iter;
    GtkTreePath *path = NULL;
    gboolean valid_iter;
    char *address;
    
    priv = mpage->priv;

    selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (priv->list_view));
    if (!(paths = gtk_tree_selection_get_selected_rows (selection, &(priv->model)))) {
        g_warning ("Could not get a selection to delete.");
        return;
    }
    paths = g_list_reverse (paths);
    
    for (tmp = paths; tmp; tmp=tmp->next) {
        path = tmp->data;
        
        gtk_tree_model_get_iter (GTK_TREE_MODEL(priv->model), &iter, path);

        gtk_tree_model_get (GTK_TREE_MODEL (priv->model), &iter, E_MEETING_STORE_ADDRESS_COL, &address, -1);
        ia = e_meeting_store_find_attendee (priv->model, address, NULL);
        g_free (address);
        if (!ia) {
            g_warning ("Cannot delete attendee\n");
            continue;
        } else if (e_meeting_attendee_get_edit_level (ia) != E_MEETING_ATTENDEE_EDIT_FULL) {
            g_warning("Not enough rights to delete attendee: %s\n", e_meeting_attendee_get_address(ia));    
            continue;
        }
        
        remove_attendee (mpage, ia);
    }
    
    /* Select closest item after removal */
    valid_iter = gtk_tree_model_get_iter (GTK_TREE_MODEL (priv->model), &iter, path);
    if (!valid_iter) {
        gtk_tree_path_prev (path);
        valid_iter = gtk_tree_model_get_iter (GTK_TREE_MODEL (priv->model), &iter, path);
    }
    
    if (valid_iter) {
        gtk_tree_selection_unselect_all (selection);
        gtk_tree_selection_select_iter (selection, &iter);
    }
    
    g_list_foreach (paths, (GFunc)gtk_tree_path_free, NULL);
    g_list_free (paths);
}

/* Function called to invite more people */
static void
invite_cb (GtkWidget *widget, gpointer data) 
{
    MeetingPage *mpage;
    MeetingPagePrivate *priv;
    
    mpage = MEETING_PAGE (data);
    priv = mpage->priv;

    e_meeting_list_view_invite_others_dialog (priv->list_view);
}

/* Hooks the widget signals */
static void
init_widgets (MeetingPage *mpage)
{
    MeetingPagePrivate *priv;

    priv = mpage->priv;

    /* Organizer */
    g_signal_connect (GTK_COMBO (priv->organizer)->entry, "changed",
              G_CALLBACK (org_changed_cb), mpage);

    g_signal_connect (priv->existing_organizer_btn, "clicked",
              G_CALLBACK (change_clicked_cb), mpage);

    /* Add attendee button */
    g_signal_connect (priv->add, "clicked", G_CALLBACK (add_clicked_cb), mpage);

    /* Remove attendee button */
    g_signal_connect (priv->remove, "clicked", G_CALLBACK (remove_clicked_cb), mpage);

    /* Contacts button */
    g_signal_connect(priv->invite, "clicked", G_CALLBACK (invite_cb), mpage);

    /* Meeting List View */
    g_signal_connect (priv->list_view, "attendee_added", G_CALLBACK (attendee_added_cb), mpage);
}

static void
attendee_added_cb (EMeetingListView *emlv, EMeetingAttendee *ia, gpointer user_data)
{
   MeetingPage *mpage = MEETING_PAGE (user_data);   
   MeetingPagePrivate *priv;
   gboolean delegate = (COMP_EDITOR_PAGE (mpage)->flags & COMP_EDITOR_PAGE_DELEGATE);

   priv = mpage->priv;

   if (delegate) {
       if (existing_attendee (ia, priv->comp))
           e_meeting_store_remove_attendee (priv->model, ia);
       else {
           if (!e_cal_get_static_capability (COMP_EDITOR_PAGE(mpage)->client, 
                       CAL_STATIC_CAPABILITY_DELEGATE_TO_MANY)) {
               const char *delegator_id = e_meeting_attendee_get_delfrom (ia);
               EMeetingAttendee *delegator;

               delegator = e_meeting_store_find_attendee (priv->model, delegator_id, NULL);
               e_meeting_attendee_set_delto (delegator, 
                       g_strdup (e_meeting_attendee_get_address (ia)));

               gtk_widget_set_sensitive (priv->invite, FALSE);
               gtk_widget_set_sensitive (priv->add, FALSE);
           }
       }
   }

}

static void
client_changed_cb (CompEditorPage *page, ECal *client, gpointer user_data)
{
    MeetingPage *mpage = MEETING_PAGE (page);

    sensitize_widgets (mpage);
}

static void
popup_add_cb (EPopup *ep, EPopupItem *pitem, void *data)
{
    MeetingPage *mpage = data;

    add_clicked_cb (NULL, mpage);
}

static void
popup_delete_cb (EPopup *ep, EPopupItem *pitem, void *data)
{
    MeetingPage *mpage = data;

    remove_clicked_cb (NULL, mpage);
}

enum {
    ATTENDEE_CAN_DELEGATE = 1<<1,
    ATTENDEE_CAN_DELETE = 1<<2,
    ATTENDEE_CAN_ADD = 1<<3,
    ATTENDEE_LAST = 1<<4,
};

static EPopupItem context_menu_items[] = {
    { E_POPUP_ITEM, "10.delete", N_("_Remove"), popup_delete_cb, NULL, GTK_STOCK_REMOVE, ATTENDEE_CAN_DELETE },
    { E_POPUP_ITEM, "15.add", N_("_Add "), popup_add_cb, NULL, GTK_STOCK_ADD }, 
};

static void
context_popup_free(EPopup *ep, GSList *items, void *data)
{
    g_slist_free(items);
}

static gint
button_press_event (GtkWidget *widget, GdkEventButton *event, MeetingPage *mpage)
{
    MeetingPagePrivate *priv;
    GtkMenu *menu;
    EMeetingAttendee *ia;
    GtkTreePath *path;
    GtkTreeIter iter;
    char *address;
    guint32 disable_mask = ~0;
    GSList *menus = NULL;
    ECalPopup *ep;
    int i;

    priv = mpage->priv;

    /* only process right-clicks */
    if (event->button != 3 || event->type != GDK_BUTTON_PRESS)
        return FALSE;

    /* only if we right-click on an attendee */
    if (gtk_tree_view_get_path_at_pos (GTK_TREE_VIEW (priv->list_view), event->x, event->y, &path, NULL, NULL, NULL)) {
        GtkTreeSelection *selection;

        if (gtk_tree_model_get_iter (GTK_TREE_MODEL (priv->model), &iter, path)) {
        
            gtk_tree_model_get (GTK_TREE_MODEL (priv->model), &iter, E_MEETING_STORE_ADDRESS_COL, &address, -1);
            ia = e_meeting_store_find_attendee (priv->model, address, &priv->row);
            g_free (address);

            if (ia) {
                selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (priv->list_view));
                gtk_tree_selection_unselect_all (selection);
                gtk_tree_selection_select_path (selection, path);
        
                if (e_meeting_attendee_get_edit_level (ia) == E_MEETING_ATTENDEE_EDIT_FULL)
                    disable_mask &= ~ATTENDEE_CAN_DELETE;
            }
        }
    }

    if (GTK_WIDGET_IS_SENSITIVE(priv->add))
        disable_mask &= ~ATTENDEE_CAN_ADD;
    else if (priv->user_org)
        disable_mask &= ~ATTENDEE_CAN_ADD;

    ep = e_cal_popup_new("org.gnome.evolution.calendar.meeting.popup");

    for (i=0;i<sizeof(context_menu_items)/sizeof(context_menu_items[0]);i++)
        menus = g_slist_prepend(menus, &context_menu_items[i]);
    
    e_popup_add_items((EPopup *)ep, menus, NULL, context_popup_free, mpage);
    menu = e_popup_create_menu_once((EPopup *)ep, NULL, disable_mask);
    gtk_menu_popup (menu, NULL, NULL, NULL, NULL, event->button, event->time);

    return TRUE;
}

static gboolean
list_view_event (EMeetingListView *list_view, GdkEvent *event, MeetingPage *mpage) {
    
    MeetingPagePrivate *priv= mpage->priv;
    
    if (event->type == GDK_2BUTTON_PRESS && mpage->priv->user_org) {
        EMeetingAttendee *attendee;
    
        attendee = e_meeting_store_add_attendee_with_defaults (priv->model);

        if (COMP_EDITOR_PAGE (mpage)->flags & COMP_EDITOR_PAGE_DELEGATE) {
            e_meeting_attendee_set_delfrom (attendee, g_strdup_printf ("MAILTO:%s", mpage->priv->user_add));
        }

        e_meeting_list_view_edit (mpage->priv->list_view, attendee);
        return TRUE;
    }

    return FALSE;   
}


static gboolean
list_key_press (EMeetingListView *list_view, GdkEventKey *event, MeetingPage *mpage)
{
    if (event->keyval == GDK_Delete) {
        remove_clicked_cb (NULL, mpage);

        return TRUE;
    } else if (event->keyval == GDK_Insert) {
        add_clicked_cb (NULL, mpage);

        return TRUE;
    }

    return FALSE;
}

/**
 * meeting_page_construct:
 * @mpage: An task details page.
 * 
 * Constructs an task page by loading its Glade data.
 * 
 * Return value: The same object as @mpage, or NULL if the widgets could not 
 * be created.
 **/
MeetingPage *
meeting_page_construct (MeetingPage *mpage, EMeetingStore *ems,
            ECal *client)
{
    MeetingPagePrivate *priv;
    char *backend_address = NULL;
    EIterator *it;
    EAccount *def_account;
    GList *address_strings = NULL, *l;
    GtkWidget *sw;
    EAccount *a;
    GtkTreeSelection *selection;
    char *gladefile;
    
    priv = mpage->priv;

    gladefile = g_build_filename (EVOLUTION_GLADEDIR, 
                      "meeting-page.glade",
                      NULL);
    priv->xml = glade_xml_new (gladefile, NULL, NULL);
    g_free (gladefile);

    if (!priv->xml) {
        g_message (G_STRLOC ": Could not load the Glade XML file!");
        return NULL;
    }

    if (!get_widgets (mpage)) {
        g_message (G_STRLOC ": Could not find all widgets in the XML file!");
        return NULL;
    }

    /* Address information */
    if (!e_cal_get_cal_address (client, &backend_address, NULL))
        return NULL;

    priv->accounts = itip_addresses_get ();
    def_account = itip_addresses_get_default();
    for (it = e_list_get_iterator((EList *)priv->accounts);
         e_iterator_is_valid(it);
         e_iterator_next(it)) {
        a = (EAccount *)e_iterator_get(it);
        char *full;
        
        full = g_strdup_printf("%s <%s>", a->id->name, a->id->address);

        address_strings = g_list_append(address_strings, full);

        /* Note that the address specified by the backend gets
         * precedence over the default mail address.
         */
        if (backend_address && !strcmp (backend_address, a->id->address)) {
            if (priv->default_address)
                g_free (priv->default_address);
            
            priv->default_address = g_strdup (full);
        } else if (a == def_account && !priv->default_address) {
            priv->default_address = g_strdup (full);
        }
    }
    
    if (backend_address)
        g_free (backend_address);

    g_object_unref(it);

    if (address_strings)
        gtk_combo_set_popdown_strings (GTK_COMBO (priv->organizer), address_strings);
    else
        g_warning ("No potential organizers!");

    for (l = address_strings; l != NULL; l = l->next)
        g_free (l->data);
    g_list_free (address_strings);
    
    /* The etable displaying attendees and their status */
    g_object_ref (ems);
    priv->model = ems;

    priv->list_view = e_meeting_list_view_new (priv->model); 
    selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (priv->list_view));
    gtk_tree_selection_set_mode (selection, GTK_SELECTION_MULTIPLE);
    
    g_signal_connect (G_OBJECT (priv->list_view), "button_press_event", G_CALLBACK (button_press_event), mpage);
    g_signal_connect (G_OBJECT (priv->list_view), "event", G_CALLBACK (list_view_event), mpage);    

    gtk_widget_show (GTK_WIDGET (priv->list_view));
    sw = gtk_scrolled_window_new (NULL, NULL);
    gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (sw), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
    gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (sw), GTK_SHADOW_IN);
    gtk_widget_show (sw);
    gtk_container_add (GTK_CONTAINER (sw), GTK_WIDGET (priv->list_view));
    gtk_box_pack_start (GTK_BOX (priv->list_box), sw, TRUE, TRUE, 0);
    
    /* Set the mnemonic widget for the Attendees label */
    gtk_label_set_mnemonic_widget (GTK_LABEL (priv->att_label), GTK_WIDGET (priv->list_view));

    /* Init the widget signals */
    init_widgets (mpage);

    g_signal_connect_after (G_OBJECT (mpage), "client_changed",
                G_CALLBACK (client_changed_cb), NULL);
    g_signal_connect (priv->list_view, "key_press_event", G_CALLBACK (list_key_press), mpage);

    return mpage;
}

/**
 * meeting_page_new:
 * 
 * Creates a new task details page.
 * 
 * Return value: A newly-created task details page, or NULL if the page could
 * not be created.
 **/
MeetingPage *
meeting_page_new (EMeetingStore *ems, ECal *client)
{
    MeetingPage *mpage;

    mpage = g_object_new (TYPE_MEETING_PAGE, NULL);
    if (!meeting_page_construct (mpage, ems, client)) {
        g_object_unref (mpage);
        return NULL;
    }

    return mpage;
}

/**
 * meeting_page_get_cancel_comp:
 * @mpage: 
 * 
 * 
 * 
 * Return value: 
 **/
ECalComponent *
meeting_page_get_cancel_comp (MeetingPage *mpage)
{
    MeetingPagePrivate *priv;

    g_return_val_if_fail (mpage != NULL, NULL);
    g_return_val_if_fail (IS_MEETING_PAGE (mpage), NULL);

    priv = mpage->priv;

    if (priv->deleted_attendees->len == 0)
        return NULL;
    
    set_attendees (priv->comp, priv->deleted_attendees);
    
    return e_cal_component_clone (priv->comp);
}