aboutsummaryrefslogblamecommitdiffstats
path: root/addressbook/gui/contact-editor/e-contact-editor.c
blob: 8a7c92e37109dedd7a60b9b5dbe070300f7f56fc (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-editor.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 "e-contact-editor.h"

#include <string.h>
#include <time.h>
#include <gtk/gtkcheckbutton.h>
#include <gtk/gtkcheckmenuitem.h>
#include <gtk/gtkcombo.h>
#include <gtk/gtktextview.h>
#include <gtk/gtkmessagedialog.h>
#include <gtk/gtkstock.h>
#include <gtk/gtkentry.h>
#include <gtk/gtklabel.h>
#include <libgnomeui/gnome-popup-menu.h>
#include <libgnomeui/gnome-window-icon.h>
#include <libgnome/gnome-util.h>
#include <libgnome/gnome-i18n.h>

#include <bonobo/bonobo-ui-container.h>
#include <bonobo/bonobo-ui-util.h>
#include <bonobo/bonobo-window.h>
#include <gdk-pixbuf/gdk-pixbuf.h>
#include <gal/widgets/e-categories.h>
#include <gal/widgets/e-gui-utils.h>
#include <gal/e-text/e-entry.h>

#include <libebook/e-address-western.h>

#include <e-util/e-categories-master-list-wombat.h>

#include <camel/camel.h>

#include "addressbook/gui/component/addressbook.h"
#include "addressbook/printing/e-contact-print.h"
#include "addressbook/printing/e-contact-print-envelope.h"
#include "addressbook/gui/widgets/eab-gui-util.h"
#include "e-util/e-gui-utils.h"
#include "widgets/misc/e-dateedit.h"
#include "widgets/misc/e-image-chooser.h"
#include "widgets/misc/e-url-entry.h"
#include "widgets/misc/e-source-option-menu.h"
#include "shell/evolution-shell-component-utils.h"

#include "eab-contact-merging.h"

#include "e-contact-editor-address.h"
#include "e-contact-editor-im.h"
#include "e-contact-editor-fullname.h"
#include "e-contact-editor-marshal.h"

/* Signal IDs */
enum {
    CONTACT_ADDED,
    CONTACT_MODIFIED,
    CONTACT_DELETED,
    EDITOR_CLOSED,
    LAST_SIGNAL
};

/* IM columns */
enum {
    COLUMN_IM_ICON,
    COLUMN_IM_SERVICE,
    COLUMN_IM_SCREENNAME,
    COLUMN_IM_LOCATION,
    COLUMN_IM_LOCATION_TYPE,
    COLUMN_IM_SERVICE_FIELD,
    NUM_IM_COLUMNS
};


static void e_contact_editor_init       (EContactEditor      *editor);
static void e_contact_editor_class_init (EContactEditorClass     *klass);
static void e_contact_editor_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec);
static void e_contact_editor_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec);
static void e_contact_editor_dispose (GObject *object);

static void _email_arrow_pressed (GtkWidget *widget, GdkEventButton *button, EContactEditor *editor);
static void _phone_arrow_pressed (GtkWidget *widget, GdkEventButton *button, EContactEditor *editor);
static void _address_arrow_pressed (GtkWidget *widget, GdkEventButton *button, EContactEditor *editor);
static void set_im_fields(EContactEditor *editor);
#if 0
static void find_address_mailing (EContactEditor *editor);
#endif
static void enable_writable_fields(EContactEditor *editor);
static void set_editable(EContactEditor *editor);
static void fill_in_info(EContactEditor *editor);
static void extract_info(EContactEditor *editor);
static void set_field(EContactEditor *editor, GtkEntry *entry, const char *string);
static void set_address_field(EContactEditor *editor, int result);
static void set_phone_field(EContactEditor *editor, GtkWidget *entry, const char *phone_number);
static void set_fields(EContactEditor *editor);
static void command_state_changed (EContactEditor *ce);
static void widget_changed (GtkWidget *widget, EContactEditor *editor);
static void close_dialog (EContactEditor *ce);
static void enable_widget (GtkWidget *widget, gboolean enabled);

static GtkObjectClass *parent_class = NULL;

static guint contact_editor_signals[LAST_SIGNAL];

/* The arguments we take */
enum {
    PROP_0,
    PROP_SOURCE_BOOK,
    PROP_TARGET_BOOK,
    PROP_CONTACT,
    PROP_IS_NEW_CONTACT,
    PROP_EDITABLE,
    PROP_CHANGED,
    PROP_WRITABLE_FIELDS
};

enum {
    DYNAMIC_LIST_EMAIL,
    DYNAMIC_LIST_PHONE,
    DYNAMIC_LIST_ADDRESS
};

static EContactField phones[] = {
    E_CONTACT_PHONE_ASSISTANT,
    E_CONTACT_PHONE_BUSINESS,
    E_CONTACT_PHONE_BUSINESS_2,
    E_CONTACT_PHONE_BUSINESS_FAX,
    E_CONTACT_PHONE_CALLBACK,
    E_CONTACT_PHONE_CAR,
    E_CONTACT_PHONE_COMPANY,
    E_CONTACT_PHONE_HOME,
    E_CONTACT_PHONE_HOME_2,
    E_CONTACT_PHONE_HOME_FAX,
    E_CONTACT_PHONE_ISDN,
    E_CONTACT_PHONE_MOBILE,
    E_CONTACT_PHONE_OTHER,
    E_CONTACT_PHONE_OTHER_FAX,
    E_CONTACT_PHONE_PAGER,
    E_CONTACT_PHONE_PRIMARY,
    E_CONTACT_PHONE_RADIO,
    E_CONTACT_PHONE_TELEX,
    E_CONTACT_PHONE_TTYTDD,
};

static EContactField emails[] = {
    E_CONTACT_EMAIL_1,
    E_CONTACT_EMAIL_2,
    E_CONTACT_EMAIL_3
};

static GSList *all_contact_editors = NULL;

GType
e_contact_editor_get_type (void)
{
    static GType contact_editor_type = 0;

    if (!contact_editor_type) {
        static const GTypeInfo contact_editor_info =  {
            sizeof (EContactEditorClass),
            NULL,           /* base_init */
            NULL,           /* base_finalize */
            (GClassInitFunc) e_contact_editor_class_init,
            NULL,           /* class_finalize */
            NULL,           /* class_data */
            sizeof (EContactEditor),
            0,             /* n_preallocs */
            (GInstanceInitFunc) e_contact_editor_init,
        };

        contact_editor_type = g_type_register_static (GTK_TYPE_OBJECT, "EContactEditor", &contact_editor_info, 0);
    }

    return contact_editor_type;
}

static void
e_contact_editor_class_init (EContactEditorClass *klass)
{
  GObjectClass *object_class = G_OBJECT_CLASS (klass);

  parent_class = g_type_class_ref (GTK_TYPE_OBJECT);

  object_class->set_property = e_contact_editor_set_property;
  object_class->get_property = e_contact_editor_get_property;
  object_class->dispose = e_contact_editor_dispose;

  g_object_class_install_property (object_class, PROP_SOURCE_BOOK, 
                   g_param_spec_object ("source_book",
                            _("Source Book"),
                            /*_( */"XXX blurb" /*)*/,
                            E_TYPE_BOOK,
                            G_PARAM_READWRITE));

  g_object_class_install_property (object_class, PROP_TARGET_BOOK, 
                   g_param_spec_object ("target_book",
                            _("Target Book"),
                            /*_( */"XXX blurb" /*)*/,
                            E_TYPE_BOOK,
                            G_PARAM_READWRITE));

  g_object_class_install_property (object_class, PROP_CONTACT, 
                   g_param_spec_object ("contact",
                            _("Contact"),
                            /*_( */"XXX blurb" /*)*/,
                            E_TYPE_CONTACT,
                            G_PARAM_READWRITE));

  g_object_class_install_property (object_class, PROP_IS_NEW_CONTACT, 
                   g_param_spec_boolean ("is_new_contact",
                             _("Is New Contact"),
                             /*_( */"XXX blurb" /*)*/,
                             FALSE,
                             G_PARAM_READWRITE));

  g_object_class_install_property (object_class, PROP_WRITABLE_FIELDS, 
                   g_param_spec_object ("writable_fields",
                            _("Writable Fields"),
                            /*_( */"XXX blurb" /*)*/,
                            E_TYPE_LIST,
                            G_PARAM_READWRITE));

  g_object_class_install_property (object_class, PROP_EDITABLE, 
                   g_param_spec_boolean ("editable",
                             _("Editable"),
                             /*_( */"XXX blurb" /*)*/,
                             FALSE,
                             G_PARAM_READWRITE));

  g_object_class_install_property (object_class, PROP_CHANGED, 
                   g_param_spec_boolean ("changed",
                             _("Changed"),
                             /*_( */"XXX blurb" /*)*/,
                             FALSE,
                             G_PARAM_READWRITE));

  contact_editor_signals[CONTACT_ADDED] =
      g_signal_new ("contact_added",
            G_OBJECT_CLASS_TYPE (object_class),
            G_SIGNAL_RUN_FIRST,
            G_STRUCT_OFFSET (EContactEditorClass, contact_added),
            NULL, NULL,
            e_contact_editor_marshal_NONE__INT_OBJECT,
            G_TYPE_NONE, 2,
            G_TYPE_INT, G_TYPE_OBJECT);

  contact_editor_signals[CONTACT_MODIFIED] =
      g_signal_new ("contact_modified",
            G_OBJECT_CLASS_TYPE (object_class),
            G_SIGNAL_RUN_FIRST,
            G_STRUCT_OFFSET (EContactEditorClass, contact_modified),
            NULL, NULL,
            e_contact_editor_marshal_NONE__INT_OBJECT,
            G_TYPE_NONE, 2,
            G_TYPE_INT, G_TYPE_OBJECT);

  contact_editor_signals[CONTACT_DELETED] =
      g_signal_new ("contact_deleted",
            G_OBJECT_CLASS_TYPE (object_class),
            G_SIGNAL_RUN_FIRST,
            G_STRUCT_OFFSET (EContactEditorClass, contact_deleted),
            NULL, NULL,
            e_contact_editor_marshal_NONE__INT_OBJECT,
            G_TYPE_NONE, 2,
            G_TYPE_INT, G_TYPE_OBJECT);

  contact_editor_signals[EDITOR_CLOSED] =
      g_signal_new ("editor_closed",
            G_OBJECT_CLASS_TYPE (object_class),
            G_SIGNAL_RUN_FIRST,
            G_STRUCT_OFFSET (EContactEditorClass, editor_closed),
            NULL, NULL,
            e_contact_editor_marshal_NONE__NONE,
            G_TYPE_NONE, 0);
}

static void
connect_arrow_button_signal (EContactEditor *editor, gchar *button_xml, GCallback func)
{
    GladeXML *gui = editor->gui;
    GtkWidget *button = glade_xml_get_widget(gui, button_xml);
    if (button && GTK_IS_BUTTON(button)) {
        g_signal_connect(button, "button_press_event", func, editor);
    }
}

static void
connect_arrow_button_signals (EContactEditor *editor)
{
    connect_arrow_button_signal(editor, "button-phone1", G_CALLBACK (_phone_arrow_pressed));
    connect_arrow_button_signal(editor, "button-phone2", G_CALLBACK (_phone_arrow_pressed));
    connect_arrow_button_signal(editor, "button-phone3", G_CALLBACK (_phone_arrow_pressed));
    connect_arrow_button_signal(editor, "button-phone4", G_CALLBACK (_phone_arrow_pressed));
    connect_arrow_button_signal(editor, "button-address", G_CALLBACK (_address_arrow_pressed));
    connect_arrow_button_signal(editor, "button-email1", G_CALLBACK (_email_arrow_pressed));
}

static void
add_im_clicked(GtkWidget *widget, EContactEditor *editor)
{
    GtkDialog *dialog;
    int result;
    EVCardAttribute *new_attr;

    dialog = GTK_DIALOG(e_contact_editor_im_new(E_CONTACT_IM_AIM, "HOME", NULL));

    gtk_widget_show(GTK_WIDGET(dialog));
    result = gtk_dialog_run(dialog);
    gtk_widget_hide(GTK_WIDGET(dialog));

    if (result == GTK_RESPONSE_OK) {
        GList *new_list = NULL;
        EContactField service;
        const char *screenname;
        const char *location;

        g_object_get(dialog,
                 "service", &service,
                 "location", &location,
                 "username", &screenname,
                 NULL);

        new_list = e_contact_get_attributes (editor->contact, service);

        new_attr = e_vcard_attribute_new ("", e_contact_vcard_attribute (service));
        if (location)
            e_vcard_attribute_add_param_with_value (new_attr,
                                e_vcard_attribute_param_new (EVC_TYPE),
                                location);
        e_vcard_attribute_add_value (new_attr, screenname);

        new_list = g_list_append(new_list, new_attr);

        e_contact_set_attributes (editor->contact, service, new_list);

        e_vcard_attribute_free (new_attr);
        g_list_free(new_list);

        set_im_fields(editor);

        widget_changed(NULL, editor);
    }

    gtk_widget_destroy(GTK_WIDGET(dialog));
}

static void
edit_im_clicked(GtkWidget *widget, EContactEditor *editor)
{
    GtkWidget *treeview;
    GtkTreeSelection *selection;
    GtkDialog *dialog;
    GtkTreeIter iter;
    EContactField old_service, service;
    const char *old_location, *location;
    const char *old_screenname, *screenname;
    int result;
    EVCardAttribute *new_attr;

    treeview = glade_xml_get_widget(editor->gui, "treeview-im");

    selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(treeview));

    if (! gtk_tree_selection_get_selected(selection, NULL, &iter) ) {
        return;
    } 
    
    gtk_tree_model_get(GTK_TREE_MODEL(editor->im_model), &iter,
               COLUMN_IM_SERVICE_FIELD, &old_service,
               COLUMN_IM_LOCATION_TYPE, &old_location,
               COLUMN_IM_SCREENNAME, &old_screenname,
               -1);

    dialog = GTK_DIALOG(e_contact_editor_im_new(old_service, old_location, old_screenname));

    gtk_widget_show(GTK_WIDGET(dialog));
    result = gtk_dialog_run(dialog);
    gtk_widget_hide(GTK_WIDGET(dialog));

    if (result == GTK_RESPONSE_OK) {
        GList *old_list, *new_list = NULL, *l;
        gboolean found;

        g_object_get(dialog,
                 "service", &service,
                 "location", &location,
                 "username", &screenname,
                 NULL);

        if (service == old_service &&
            ((!location && !old_location) ||
             (location && old_location &&
              !strcmp(old_location, location))) &&
            !strcmp(screenname, old_screenname)) {

            gtk_widget_destroy(GTK_WIDGET(dialog));
            return;
        }

        /* Remove the old. */
        old_list = e_contact_get_attributes (editor->contact, old_service);
        found = FALSE;

        for (l = old_list; l != NULL; l = l->next) {
            EVCardAttribute *attr = l->data;
            
            if (!found
                && !strcmp(e_vcard_attribute_get_value (attr), old_screenname)
                && ((old_location && e_vcard_attribute_has_type (attr, old_location))
                || (!old_location && !e_vcard_attribute_has_type (attr, "HOME") && !e_vcard_attribute_has_type (attr, "WORK")))) {
                e_vcard_attribute_free (attr);
                found = TRUE;
            }
            else {
                new_list = g_list_append(new_list, attr);
            }
        }
        g_list_free (old_list);

        /* create a new attribute based on the new screenname */
        new_attr = e_vcard_attribute_new ("", e_contact_vcard_attribute (service));
        if (location)
            e_vcard_attribute_add_param_with_value (new_attr,
                                e_vcard_attribute_param_new (EVC_TYPE),
                                location);
        e_vcard_attribute_add_value (new_attr, screenname);

        if (service == old_service) {
            /* we're just appending it to the same list again */
            new_list = g_list_append(new_list, new_attr);

            e_contact_set_attributes (editor->contact, old_service, new_list);
        }
        else {
            /* we're appending it to a different list, so
               we need to write out the old service's list
               and then get the new service's list, then
               add the new attribute to that.  so
               confusing! */
            e_contact_set_attributes (editor->contact, old_service, new_list);
        
            /* We have to add this elsewhere. */
            new_list = e_contact_get_attributes (editor->contact, service);

            new_list = g_list_append(new_list, new_attr);

            e_contact_set_attributes (editor->contact, service, new_list);
        }

        set_im_fields(editor);

        widget_changed(NULL, editor);
    }

    gtk_widget_destroy(GTK_WIDGET(dialog));
}

static void
remove_im_clicked(GtkWidget *widget, EContactEditor *editor)
{
    GtkWidget *treeview;
    GtkTreeSelection *selection;
    GtkTreeIter iter;
    EContactField old_service;
    const char *old_screenname;
    GList *old_list, *new_list = NULL, *l;

    treeview = glade_xml_get_widget(editor->gui, "treeview-im");

    selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(treeview));

    gtk_tree_selection_get_selected(selection, NULL, &iter);

    gtk_tree_model_get(GTK_TREE_MODEL(editor->im_model), &iter,
                       COLUMN_IM_SERVICE_FIELD, &old_service,
                       COLUMN_IM_SCREENNAME, &old_screenname,
                       -1);

    old_list = e_contact_get(editor->contact, old_service);

    for (l = old_list; l != NULL; l = l->next) {
        const char *temp_screenname = (const char *)l->data;

        if (strcmp(temp_screenname, old_screenname))
            new_list = g_list_append(new_list, g_strdup(temp_screenname));
    }

    e_contact_set(editor->contact, old_service, new_list);

    if (new_list != NULL) {
        g_list_foreach(new_list, (GFunc)g_free, NULL);
        g_list_free(new_list);
    }

    gtk_list_store_remove(editor->im_model, &iter);

    widget_changed(NULL, editor);
}


static gboolean
im_button_press_cb(GtkWidget *treeview, GdkEventButton *event,
                   EContactEditor *editor)
{
    if (event->button == 1 && event->type == GDK_2BUTTON_PRESS)
        edit_im_clicked(NULL, editor);

    return FALSE;
}

static void
im_selected_cb(GtkTreeSelection *selection, EContactEditor *editor)
{
    gboolean sensitive = gtk_tree_selection_get_selected(selection, NULL, NULL);

    gtk_widget_set_sensitive(glade_xml_get_widget(editor->gui, "button-im-remove"), sensitive);
    gtk_widget_set_sensitive(glade_xml_get_widget(editor->gui, "button-im-edit"),   sensitive);
}


static void
im_treeview_drag_data_get_cb(GtkWidget *widget, GdkDragContext *dc,
                             GtkSelectionData *data, guint info,
                             guint time, EContactEditor *editor)
{
    if (data->target == gdk_atom_intern("application/x-im-contact", FALSE)) {
        GtkTreeRowReference *ref;
        GtkTreePath *sourcerow;
        GtkTreeIter iter;
        const char *protocol;
        const char *screenname;
        const char *alias;
        GString *str;
        char *mime_str;
        EContactField service_field;
        static char *protocols[] = { "aim", "nov", "jabber", "yahoo", "msn", "icq" };

        ref = g_object_get_data(G_OBJECT(dc), "gtk-tree-view-source-row");
        sourcerow = gtk_tree_row_reference_get_path(ref);

        if (!sourcerow)
            return;

        gtk_tree_model_get_iter(GTK_TREE_MODEL(editor->im_model), &iter,
                                sourcerow);

        gtk_tree_model_get(GTK_TREE_MODEL(editor->im_model), &iter,
                           COLUMN_IM_SERVICE_FIELD, &service_field,
                           COLUMN_IM_SCREENNAME, &screenname,
                           -1);

        alias = e_contact_get_const(editor->contact, E_CONTACT_FULL_NAME);

        protocol = protocols[service_field - E_CONTACT_IM_AIM];

        str = g_string_new(NULL);

        g_string_printf(str,
                "MIME-Version: 1.0\r\n"
                "Content-Type: application/x-im-contact\r\n"
                "X-IM-Protocol: %s\r\n"
                "X-IM-Username: %s\r\n",
                protocol,
                screenname);

        if (alias && *alias)
            g_string_append_printf(str, "X-IM-Alias: %s\r\n", alias);

        str = g_string_append(str, "\r\n");

        mime_str = g_string_free(str, FALSE);

        gtk_selection_data_set(data,
                       gdk_atom_intern("application/x-im-contact", FALSE),
                       8,
                       mime_str,
                       strlen(mime_str) + 1);

        g_free(mime_str);
        gtk_tree_path_free(sourcerow);
    }
}

static void
im_treeview_drag_data_rcv_cb(GtkWidget *widget, GdkDragContext *dc,
                 guint x, guint y, GtkSelectionData *sd,
                 guint info, guint t, EContactEditor *editor)
{
    if (sd->target == gdk_atom_intern("application/x-im-contact", FALSE) && sd->data) {
        CamelMimeParser *parser;
        CamelStream *stream;
        char *buffer;
        char *username = NULL;
        char *protocol = NULL;
        int len;
        int state;

        parser = camel_mime_parser_new();

        stream = camel_stream_mem_new_with_buffer(sd->data, sd->length);

        if (camel_mime_parser_init_with_stream(parser, stream) == -1) {
            g_warning("Unable to create parser for stream");
            return;
        }

        while ((state = camel_mime_parser_step(parser, &buffer, &len)) != CAMEL_MIME_PARSER_STATE_EOF) {
            if (state == CAMEL_MIME_PARSER_STATE_HEADER) {
                const char *temp;
                char *temp2;

                if ((temp = camel_mime_parser_header(parser, "X-IM-Username", NULL)) != NULL) {
                    temp2 = g_strdup(temp);
                    username = g_strdup(g_strstrip(temp2));
                    g_free(temp2);
                }

                if ((temp = camel_mime_parser_header(parser, "X-IM-Protocol", NULL)) != NULL) {
                    temp2 = g_strdup(temp);
                    protocol = g_strdup(g_strstrip(temp2));
                    g_free(temp2);
                }

                break;
            }
        }

        camel_object_unref(parser);

        if (username != NULL && protocol != NULL) {
            GList *old_list, *new_list = NULL, *l;
            EContactField field;
            gboolean found = FALSE;

            if (!strcmp(protocol, "aim"))
                field = E_CONTACT_IM_AIM;
            else if (!strcmp(protocol, "nov"))
                field = E_CONTACT_IM_GROUPWISE;
            else if (!strcmp(protocol, "icq"))
                field = E_CONTACT_IM_ICQ;
            else if (!strcmp(protocol, "yahoo"))
                field = E_CONTACT_IM_YAHOO;
            else if (!strcmp(protocol, "msn"))
                field = E_CONTACT_IM_MSN;
            else if (!strcmp(protocol, "jabber"))
                field = E_CONTACT_IM_JABBER;
            else {
                g_free(username);
                g_free(protocol);
                gtk_drag_finish(dc, FALSE, (dc->action == GDK_ACTION_MOVE), t);
                return;
            }

            old_list = e_contact_get(editor->contact, field);

            for (l = old_list; l != NULL; l = l->next) {
                const char *name = (const char *)l->data;

                if (!strcmp(name, username)) {
                    found = TRUE;
                    break;
                }

                new_list = g_list_append(new_list, g_strdup(l->data));
            }

            if (!found) {
                new_list = g_list_append(new_list, g_strdup(username));

                e_contact_set(editor->contact, field, new_list);
            }

            if (new_list != NULL) {
                g_list_foreach(new_list, (GFunc)g_free, NULL);
                g_list_free(new_list);
            }

            set_im_fields(editor);
        }

        if (username != NULL)
            g_free(username);

        if (protocol != NULL)
            g_free(protocol);

        gtk_drag_finish(dc, TRUE, (dc->action == GDK_ACTION_MOVE), t);
    }
}

static void
setup_im_treeview(EContactEditor *editor)
{
    GtkWidget *treeview;
    GtkTreeSelection *selection;
    GtkTreeViewColumn *column;
    GtkCellRenderer *renderer;
    GtkTargetEntry gte[] = {{"application/x-im-contact", 0, 0}};

    treeview = glade_xml_get_widget(editor->gui, "treeview-im");

    if (!treeview || !GTK_IS_TREE_VIEW(treeview))
        return;

    editor->im_model = gtk_list_store_new(NUM_IM_COLUMNS,
                          GDK_TYPE_PIXBUF, G_TYPE_STRING,
                          G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_INT);

    gtk_tree_view_set_model(GTK_TREE_VIEW(treeview),
                GTK_TREE_MODEL(editor->im_model));

    g_signal_connect(G_OBJECT(treeview), "button-press-event",
             G_CALLBACK(im_button_press_cb), editor);

    column = gtk_tree_view_column_new();
    gtk_tree_view_column_set_title(column, _("Service"));
    gtk_tree_view_insert_column(GTK_TREE_VIEW(treeview), column, -1);

    renderer = gtk_cell_renderer_pixbuf_new();
    gtk_tree_view_column_pack_start(column, renderer, FALSE);
    gtk_tree_view_column_add_attribute(column, renderer,
                       "pixbuf", COLUMN_IM_ICON);

    renderer = gtk_cell_renderer_text_new();
    gtk_tree_view_column_pack_start(column, renderer, TRUE);
    gtk_tree_view_column_add_attribute(column, renderer,
                       "text", COLUMN_IM_SERVICE);

    column = gtk_tree_view_column_new();
    gtk_tree_view_column_set_title(column, _("Account Name"));
    gtk_tree_view_insert_column(GTK_TREE_VIEW(treeview), column, -1);
    
    renderer = gtk_cell_renderer_text_new();
    gtk_tree_view_column_pack_start(column, renderer, TRUE);
    gtk_tree_view_column_add_attribute(column, renderer,
                       "text", COLUMN_IM_SCREENNAME);

    column = gtk_tree_view_column_new();
    gtk_tree_view_column_set_title(column, _("Location"));
    gtk_tree_view_insert_column(GTK_TREE_VIEW(treeview), column, -1);
    
    renderer = gtk_cell_renderer_text_new();
    gtk_tree_view_column_pack_start(column, renderer, TRUE);
    gtk_tree_view_column_add_attribute(column, renderer,
                       "text", COLUMN_IM_LOCATION);

    selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(treeview));
    g_signal_connect(G_OBJECT(selection), "changed",
             G_CALLBACK(im_selected_cb), editor);

    /* Setup drag-and-drop */
    gtk_tree_view_enable_model_drag_source(GTK_TREE_VIEW(treeview),
                           GDK_BUTTON1_MASK, gte, 1,
                           GDK_ACTION_COPY);
    gtk_tree_view_enable_model_drag_dest(GTK_TREE_VIEW(treeview),
                         gte, 1,
                         GDK_ACTION_COPY | GDK_ACTION_MOVE);

    g_signal_connect(G_OBJECT(treeview), "drag-data-get",
             G_CALLBACK(im_treeview_drag_data_get_cb), editor);
    g_signal_connect(G_OBJECT(treeview), "drag-data-received",
             G_CALLBACK(im_treeview_drag_data_rcv_cb), editor);
}

static void
wants_html_changed (GtkWidget *widget, EContactEditor *editor)
{
    gboolean wants_html;
    g_object_get (widget,
              "active", &wants_html,
              NULL);

    e_contact_set (editor->contact, E_CONTACT_WANTS_HTML, GINT_TO_POINTER (wants_html));

    widget_changed (widget, editor);
}

static void
phone_entry_changed (GtkWidget *widget, EContactEditor *editor)
{
    int which;
    GtkEntry *entry = GTK_ENTRY(widget);

    if ( widget == glade_xml_get_widget(editor->gui, "entry-phone1") )
        which = 1;
    else if ( widget == glade_xml_get_widget(editor->gui, "entry-phone2") )
        which = 2;
    else if ( widget == glade_xml_get_widget(editor->gui, "entry-phone3") )
        which = 3;
    else if ( widget == glade_xml_get_widget(editor->gui, "entry-phone4") )
        which = 4;
    else
        return;

    e_contact_set(editor->contact, editor->phone_choice[which - 1], (char*)gtk_entry_get_text(entry));

    widget_changed (widget, editor);
}

static void
email_entry_changed (GtkWidget *widget, EContactEditor *editor)
{
    GtkEntry *entry = GTK_ENTRY(widget);

    e_contact_set (editor->contact, editor->email_choice, (char*)gtk_entry_get_text(entry));

    widget_changed (widget, editor);
}

static gchar *
address_to_text (const EContactAddress *address)
{
    GString *text;
    gchar *str;

    text = g_string_new ("");

    if (address->street && *address->street) {
        g_string_append (text, address->street);
        g_string_append_c (text, '\n');
    }
    if (address->ext && *address->ext) {
        g_string_append (text, address->ext);
        g_string_append_c (text, '\n');
    }
    if (address->po && *address->po) {
        g_string_append (text, address->po);
        g_string_append_c (text, '\n');
    }
    if (address->locality && *address->locality) {
        g_string_append (text, address->locality);
        g_string_append_c (text, '\n');
    }
    if (address->region && *address->region) {
        g_string_append (text, address->region);
        g_string_append_c (text, '\n');
    }
    if (address->code && *address->code) {
        g_string_append (text, address->code);
        g_string_append_c (text, '\n');
    }
    if (address->country && *address->country) {
        g_string_append (text, address->country);
        g_string_append_c (text, '\n');
    }

    str = text->str;
    g_string_free (text, FALSE);
    return str;
}

static EContactAddress *
text_to_address (const gchar *text)
{
    EAddressWestern *western;
    EContactAddress *address;

    g_return_val_if_fail (text != NULL, NULL);

    western = e_address_western_parse (text);
    if (!western)
        return NULL;

    address = g_new0 (EContactAddress, 1);
    address->po = western->po_box;
    address->ext = western->extended;
    address->street = western->street;
    address->locality = western->locality;
    address->region = western->region;
    address->code = western->postal_code;
    address->country = western->country;

    g_free (western);

    return address;
}

static void
address_text_changed (GtkTextBuffer *buffer, EContactEditor *editor)
{
    EContactAddress *address;
    GtkTextIter start_iter, end_iter;
    gchar *text;

    g_print ("editor->address_choice == %d\n", editor->address_choice);

    if (editor->address_choice == -1)
        return;

    gtk_text_buffer_get_start_iter (GTK_TEXT_BUFFER (buffer), &start_iter);
    gtk_text_buffer_get_end_iter (GTK_TEXT_BUFFER (buffer), &end_iter);

    text = gtk_text_buffer_get_text (GTK_TEXT_BUFFER (buffer), &start_iter, &end_iter, FALSE);
    address = text_to_address (text);
    g_free (text);

    if (editor->address_mailing == editor->address_choice || editor->address_mailing == -1) {
        GtkWidget *check;

#if 0
        address->flags |= E_CARD_ADDR_DEFAULT;
#endif

        check = glade_xml_get_widget(editor->gui, "checkbutton-mailingaddress");
        if (check && GTK_IS_CHECK_BUTTON (check)) {
            g_signal_handlers_block_matched (check,
                             G_SIGNAL_MATCH_DATA,
                             0, 0,
                             NULL, NULL, editor);
            gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (check), TRUE);
            g_signal_handlers_unblock_matched (check,
                               G_SIGNAL_MATCH_DATA,
                               0, 0,
                               NULL, NULL, editor);
        }
    }
    
    e_contact_set (editor->contact, editor->address_choice, address);

    g_boxed_free (e_contact_address_get_type (), address);

    widget_changed (NULL, editor);
}


static void
address_mailing_changed (GtkWidget *widget, EContactEditor *editor)
{
#if notyet
    const ECardDeliveryAddress *curr;
    ECardDeliveryAddress *address;
    gboolean mailing_address;

    if (editor->address_choice == -1)
        return;

    mailing_address = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (widget));
    
    /* Mark the current address as the mailing address */
    curr = e_card_simple_get_delivery_address (editor->simple,
                           editor->address_choice);

    address = e_card_delivery_address_copy (curr);

    if (!address)
        address = e_card_delivery_address_new ();

    if (mailing_address)
        address->flags |= E_CARD_ADDR_DEFAULT;
    else
        address->flags &= ~E_CARD_ADDR_DEFAULT;

    e_card_simple_set_delivery_address(editor->simple, editor->address_choice, address);
    e_card_delivery_address_unref (address);

    /* Unset the previous mailing address flag */
    if (mailing_address && editor->address_mailing != -1) {
        const ECardDeliveryAddress *curr;

        curr = e_card_simple_get_delivery_address(editor->simple, 
                              editor->address_mailing);
        address = e_card_delivery_address_copy (curr);
        address->flags &= ~E_CARD_ADDR_DEFAULT;
        e_card_simple_set_delivery_address(editor->simple, 
                           editor->address_mailing,
                           address);
        e_card_delivery_address_unref (address);
    }

    /* Remember the new mailing address */
    if (mailing_address)
        editor->address_mailing = editor->address_choice;
    else
        editor->address_mailing = -1;

    widget_changed (widget, editor);
#endif
}

static void
new_target_cb (EBook *new_book, EBookStatus status, EContactEditor *editor)
{
    editor->load_source_id = 0;
    editor->load_book      = NULL;

    if (status != E_BOOK_ERROR_OK || new_book == NULL) {
        GtkWidget *source_option_menu;

        addressbook_show_load_error_dialog (NULL, e_book_get_source (new_book), status);

        source_option_menu = glade_xml_get_widget (editor->gui, "source-option-menu-source");
        e_source_option_menu_select (E_SOURCE_OPTION_MENU (source_option_menu),
                         e_book_get_source (editor->target_book));

        if (new_book)
            g_object_unref (new_book);
        return;
    }

    g_object_set (editor, "target_book", new_book, NULL);
    g_object_unref (new_book);
    widget_changed (NULL, editor);
}

static void
cancel_load (EContactEditor *editor)
{
    if (editor->load_source_id) {
        addressbook_load_source_cancel (editor->load_source_id);
        editor->load_source_id = 0;

        g_object_unref (editor->load_book);
        editor->load_book = NULL;
    }
}

static void
source_selected (GtkWidget *source_option_menu, ESource *source, EContactEditor *editor)
{
    cancel_load (editor);

    if (e_source_equal (e_book_get_source (editor->target_book), source))
        return;

    if (e_source_equal (e_book_get_source (editor->source_book), source)) {
        g_object_set (editor, "target_book", editor->source_book, NULL);
        return;
    }

    editor->load_book = e_book_new ();
    editor->load_source_id = addressbook_load_source (editor->load_book, source,
                              (EBookCallback) new_target_cb, editor);
}

/* This function tells you whether name_to_style will make sense.  */
static gboolean
style_makes_sense(const EContactName *name, char *company, int style)
{
    switch (style) {
    case 0: /* Fall Through */
    case 1:
        return TRUE;
    case 2:
        if (company && *company)
            return TRUE;
        else
            return FALSE;
    case 3: /* Fall Through */
    case 4:
        if (company && *company && name && ((name->given && *name->given) || (name->family && *name->family)))
            return TRUE;
        else
            return FALSE;
    default:
        return FALSE;
    }
}

static char *
name_to_style(const EContactName *name, char *company, int style)
{
    char *string;
    char *strings[4], **stringptr;
    char *substring;
    switch (style) {
    case 0:
        stringptr = strings;
        if (name) {
            if (name->family && *name->family)
                *(stringptr++) = name->family;
            if (name->given && *name->given)
                *(stringptr++) = name->given;
        }
        *stringptr = NULL;
        string = g_strjoinv(", ", strings);
        break;
    case 1:
        stringptr = strings;
        if (name) {
            if (name->given && *name->given)
                *(stringptr++) = name->given;
            if (name->family && *name->family)
                *(stringptr++) = name->family;
        }
        *stringptr = NULL;
        string = g_strjoinv(" ", strings);
        break;
    case 2:
        string = g_strdup(company);
        break;
    case 3: /* Fall Through */
    case 4:
        stringptr = strings;
        if (name) {
            if (name->family && *name->family)
                *(stringptr++) = name->family;
            if (name->given && *name->given)
                *(stringptr++) = name->given;
        }
        *stringptr = NULL;
        substring = g_strjoinv(", ", strings);
        if (!(company && *company))
            company = "";
        if (style == 3)
            string = g_strdup_printf("%s (%s)", substring, company);
        else
            string = g_strdup_printf("%s (%s)", company, substring);
        g_free(substring);
        break;
    default:
        string = g_strdup("");
    }
    return string;
}

static int
file_as_get_style (EContactEditor *editor)
{
    GtkEntry *file_as = GTK_ENTRY(glade_xml_get_widget(editor->gui, "entry-file-as"));
    char *filestring;
    char *trystring;
    EContactName *name = editor->name;
    int i;
    int style;

    if (!(file_as && GTK_IS_ENTRY(file_as)))
        return -1;

    filestring = g_strdup (gtk_entry_get_text(file_as));

    style = -1;
    for (i = 0; i < 5; i++) {
        trystring = name_to_style(name, editor->company, i);
        if (!strcmp(trystring, filestring)) {
            g_free(trystring);
            g_free(filestring);
            return i;
        }
        g_free(trystring);
    }
    g_free (filestring);
    return -1;
}

static void
file_as_set_style(EContactEditor *editor, int style)
{
    char *string;
    int i;
    GList *strings = NULL;
    GtkEntry *file_as = GTK_ENTRY(glade_xml_get_widget(editor->gui, "entry-file-as"));
    GtkWidget *widget;
        

    if (!(file_as && GTK_IS_ENTRY(file_as)))
        return;

    if (style == -1) {
        string = g_strdup (gtk_entry_get_text(file_as));
        strings = g_list_append(strings, string);
    }

    widget = glade_xml_get_widget(editor->gui, "combo-file-as");

    for (i = 0; i < 5; i++) {
        if (style_makes_sense(editor->name, editor->company, i)) {
            char *u;
            u = name_to_style(editor->name, editor->company, i);
            if (u) strings = g_list_append(strings, u);
        }
    }

    if (widget && GTK_IS_COMBO(widget)) {
        GtkCombo *combo = GTK_COMBO(widget);
        gtk_combo_set_popdown_strings(combo, strings);
        g_list_foreach(strings, (GFunc) g_free, NULL);
        g_list_free(strings);
    }

    if (style != -1) {
        string = name_to_style(editor->name, editor->company, style);
        gtk_entry_set_text(file_as, string);
        g_free(string);
    }
}

static void
name_entry_changed (GtkWidget *widget, EContactEditor *editor)
{
    int style = 0;
    const char *string;

    style = file_as_get_style(editor);

    e_contact_name_free (editor->name);

    string = gtk_entry_get_text (GTK_ENTRY(widget));

    editor->name = e_contact_name_from_string(string);
    
    file_as_set_style(editor, style);

    widget_changed (widget, editor);
}

static void
company_entry_changed (GtkWidget *widget, EContactEditor *editor)
{
    int style = 0;

    style = file_as_get_style(editor);
    
    g_free(editor->company);
    
    editor->company = g_strdup (gtk_entry_get_text(GTK_ENTRY(widget)));
    
    file_as_set_style(editor, style);

    widget_changed (widget, editor);
}

static void
image_chooser_changed (GtkWidget *widget, EContactEditor *editor)
{
    editor->image_set = TRUE;

    widget_changed (widget, editor);
}

static void
field_changed (GtkWidget *widget, EContactEditor *editor)
{
    if (!editor->changed) {
        editor->changed = TRUE;
        command_state_changed (editor);
    }
}

static void
set_entry_changed_signal_phone(EContactEditor *editor, char *id)
{
    GtkWidget *widget = glade_xml_get_widget(editor->gui, id);
    if (widget && GTK_IS_ENTRY(widget))
        g_signal_connect(widget, "changed",
                 G_CALLBACK (phone_entry_changed), editor);
}

static void
set_entry_changed_signal_email(EContactEditor *editor, char *id)
{
    GtkWidget *widget = glade_xml_get_widget(editor->gui, id);
    if (widget && GTK_IS_ENTRY(widget))
        g_signal_connect(widget, "changed",
                 G_CALLBACK (email_entry_changed), editor);
}

static void
widget_changed (GtkWidget *widget, EContactEditor *editor)
{
    if (!editor->target_editable) {
        g_warning ("non-editable contact editor has an editable field in it.");
        return;
    }

    if (!editor->changed) {
        editor->changed = TRUE;
        command_state_changed (editor);
    }
}

static void
set_entry_changed_signal_field(EContactEditor *editor, char *id)
{
    GtkWidget *widget = glade_xml_get_widget(editor->gui, id);
    if (widget && GTK_IS_ENTRY(widget))
        g_signal_connect(widget, "changed",
                 G_CALLBACK (field_changed), editor);
}

static void
set_urlentry_changed_signal_field (EContactEditor *editor, char *id)
{
    GtkWidget *widget = glade_xml_get_widget(editor->gui, id);
    if (widget && E_IS_URL_ENTRY(widget)) {
        GtkWidget *entry = e_url_entry_get_entry (E_URL_ENTRY (widget));
        g_signal_connect (entry, "changed",
                  G_CALLBACK (field_changed), editor);
    }
}

static void
set_entry_changed_signals(EContactEditor *editor)
{
    GtkWidget *widget;
    set_entry_changed_signal_phone(editor, "entry-phone1");
    set_entry_changed_signal_phone(editor, "entry-phone2");
    set_entry_changed_signal_phone(editor, "entry-phone3");
    set_entry_changed_signal_phone(editor, "entry-phone4");

    set_entry_changed_signal_email(editor, "entry-email1");

    widget = glade_xml_get_widget(editor->gui, "text-address");
    if (widget && GTK_IS_TEXT_VIEW(widget)) {
        GtkTextBuffer *buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (widget));
        g_signal_connect(buffer, "changed",
                 G_CALLBACK (address_text_changed), editor);
    }

    widget = glade_xml_get_widget(editor->gui, "entry-fullname");
    if (widget && GTK_IS_ENTRY(widget)) {
        g_signal_connect (widget, "changed",
                  G_CALLBACK (name_entry_changed), editor);
    }

    widget = glade_xml_get_widget(editor->gui, "entry-company");
    if (widget && GTK_IS_ENTRY(widget)) {
        g_signal_connect (widget, "changed",
                  G_CALLBACK (company_entry_changed), editor);
    }

    set_urlentry_changed_signal_field (editor, "entry-web");
    set_urlentry_changed_signal_field (editor, "entry-blog");
    set_urlentry_changed_signal_field (editor, "entry-caluri");
    set_urlentry_changed_signal_field (editor, "entry-fburl");
    set_urlentry_changed_signal_field (editor, "entry-videourl");

    set_entry_changed_signal_field(editor, "entry-categories");
    set_entry_changed_signal_field(editor, "entry-jobtitle");
    set_entry_changed_signal_field(editor, "entry-file-as");
    set_entry_changed_signal_field(editor, "entry-manager");
    set_entry_changed_signal_field(editor, "entry-assistant");
    set_entry_changed_signal_field(editor, "entry-office");
    set_entry_changed_signal_field(editor, "entry-department");
    set_entry_changed_signal_field(editor, "entry-profession");
    set_entry_changed_signal_field(editor, "entry-nickname");
    set_entry_changed_signal_field(editor, "entry-spouse");

    widget = glade_xml_get_widget(editor->gui, "text-comments");
    if (widget && GTK_IS_TEXT_VIEW(widget)) {
        GtkTextBuffer *buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (widget));
        g_signal_connect (buffer, "changed",
                  G_CALLBACK (widget_changed), editor);
    }
    widget = glade_xml_get_widget(editor->gui, "dateedit-birthday");
    if (widget && E_IS_DATE_EDIT(widget)) {
        g_signal_connect (widget, "changed",
                  G_CALLBACK (widget_changed), editor);
    }
    widget = glade_xml_get_widget(editor->gui, "dateedit-anniversary");
    if (widget && E_IS_DATE_EDIT(widget)) {
        g_signal_connect (widget, "changed",
                  G_CALLBACK (widget_changed), editor);
    }

    widget = glade_xml_get_widget (editor->gui, "image-chooser");
    if (widget && E_IS_IMAGE_CHOOSER (widget)) {
        g_signal_connect (widget, "changed",
                  G_CALLBACK (image_chooser_changed), editor);
    }
}

static void
full_name_clicked(GtkWidget *button, EContactEditor *editor)
{
    GtkDialog *dialog = GTK_DIALOG(e_contact_editor_fullname_new(editor->name));
    int result;

    g_object_set (dialog,
              "editable", editor->fullname_editable,
              NULL);
    gtk_widget_show(GTK_WIDGET(dialog));
    result = gtk_dialog_run (dialog);
    gtk_widget_hide (GTK_WIDGET (dialog));

    if (editor->fullname_editable && result == GTK_RESPONSE_OK) {
        EContactName *name;
        GtkWidget *fname_widget;
        int style = 0;

        g_object_get (dialog,
                  "name", &name,
                  NULL);

        style = file_as_get_style(editor);

        fname_widget = glade_xml_get_widget(editor->gui, "entry-fullname");
        if (fname_widget && GTK_IS_ENTRY(fname_widget)) {
            char *full_name = e_contact_name_to_string(name);
            gtk_entry_set_text(GTK_ENTRY(fname_widget), full_name);
            g_free(full_name);
        }

        e_contact_name_free(editor->name);
        editor->name = name;

        file_as_set_style(editor, style);
    }
    gtk_widget_destroy (GTK_WIDGET (dialog));
}

static void
full_addr_clicked(GtkWidget *button, EContactEditor *editor)
{
    GtkDialog *dialog;
    int result;
    EContactAddress *address;

    address = e_contact_get (editor->contact, editor->address_choice);

    dialog = GTK_DIALOG(e_contact_editor_address_new(address));
    g_object_set (dialog,
              "editable", editor->address_editable[editor->address_choice - E_CONTACT_FIRST_ADDRESS_ID],
              NULL);
    gtk_widget_show(GTK_WIDGET(dialog));

    result = gtk_dialog_run (dialog);

    gtk_widget_hide (GTK_WIDGET (dialog));

    if (editor->address_editable[editor->address_choice - E_CONTACT_FIRST_ADDRESS_ID] && result == GTK_RESPONSE_OK) {
        EContactAddress *new_address;
        GtkWidget *address_widget;
        int saved_choice = editor->address_choice;

        editor->address_choice = -1;

        g_object_get (dialog,
                  "address", &new_address,
                  NULL);

        address_widget = glade_xml_get_widget(editor->gui, "text-address");
        if (address_widget && GTK_IS_TEXT_VIEW(address_widget)) {
            GtkTextBuffer *buffer;
            char *string = address_to_text (new_address);

            buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (address_widget));

            gtk_text_buffer_set_text (buffer, string, strlen (string));

            g_free(string);
        }

        editor->address_choice = saved_choice;

        e_contact_set (editor->contact, editor->address_choice, new_address);

        g_boxed_free (e_contact_address_get_type (), new_address);

        widget_changed (NULL, editor);
    }
    gtk_widget_destroy (GTK_WIDGET (dialog));

    g_boxed_free (e_contact_address_get_type (), address);
}

static void
categories_clicked(GtkWidget *button, EContactEditor *editor)
{
    char *categories = NULL;
    GtkDialog *dialog;
    int result;
    GtkWidget *entry = glade_xml_get_widget(editor->gui, "entry-categories");
    ECategoriesMasterList *ecml;
    if (entry && GTK_IS_ENTRY(entry))
        categories = g_strdup (gtk_entry_get_text(GTK_ENTRY(entry)));
    else if (editor->contact)
        categories = e_contact_get (editor->contact, E_CONTACT_CATEGORIES);

    dialog = GTK_DIALOG(e_categories_new(categories));

    if (dialog == NULL) {
        GtkWidget *uh_oh = gtk_message_dialog_new (NULL,
                               0, GTK_MESSAGE_ERROR,
                               GTK_RESPONSE_OK,
                               _("Category editor not available."));
        g_free (categories);
        gtk_widget_show (uh_oh);
        return;
    }

    ecml = e_categories_master_list_wombat_new ();
    g_object_set (dialog,
               "header", _("This contact belongs to these categories:"),
               "ecml", ecml,
               NULL);
    g_object_unref (ecml);
    gtk_widget_show(GTK_WIDGET(dialog));
    result = gtk_dialog_run (dialog);
    g_free (categories);
    if (result == GTK_RESPONSE_OK) {
        g_object_get (dialog,
                  "categories", &categories,
                  NULL);
        if (entry && GTK_IS_ENTRY(entry))
            gtk_entry_set_text(GTK_ENTRY(entry), categories);
        else
            e_contact_set (editor->contact, E_CONTACT_CATEGORIES, categories);

        g_free(categories);
    }
    gtk_widget_destroy(GTK_WIDGET(dialog));
}


typedef struct {
    EContactEditor *ce;
    gboolean should_close;
    gchar *new_id;
} EditorCloseStruct;

static void
contact_moved_cb (EBook *book, EBookStatus status, EditorCloseStruct *ecs)
{
    EContactEditor *ce = ecs->ce;
    gboolean should_close = ecs->should_close;

    gtk_widget_set_sensitive (ce->app, TRUE);
    ce->in_async_call = FALSE;

    e_contact_set (ce->contact, E_CONTACT_UID, ecs->new_id);

    g_signal_emit (ce, contact_editor_signals[CONTACT_DELETED], 0,
               status, ce->contact);

    ce->is_new_contact = FALSE;

    if (should_close) {
        close_dialog (ce);
    }
    else {
        ce->changed = FALSE;

        g_object_ref (ce->target_book);
        g_object_unref (ce->source_book);
        ce->source_book = ce->target_book;

        command_state_changed (ce);
    }

    g_object_unref (ce);
    g_free (ecs->new_id);
    g_free (ecs);
}

static void
contact_added_cb (EBook *book, EBookStatus status, const char *id, EditorCloseStruct *ecs)
{
    EContactEditor *ce = ecs->ce;
    gboolean should_close = ecs->should_close;

    if (ce->source_book != ce->target_book && ce->source_editable &&
        status == E_BOOK_ERROR_OK && ce->is_new_contact == FALSE) {
        ecs->new_id = g_strdup (id);
        e_book_async_remove_contact (ce->source_book, ce->contact,
                         (EBookCallback) contact_moved_cb, ecs);
        return;
    }

    gtk_widget_set_sensitive (ce->app, TRUE);
    ce->in_async_call = FALSE;

    e_contact_set (ce->contact, E_CONTACT_UID, (char *) id);

    g_signal_emit (ce, contact_editor_signals[CONTACT_ADDED], 0,
               status, ce->contact);

    if (status == E_BOOK_ERROR_OK) {
        ce->is_new_contact = FALSE;

        if (should_close) {
            close_dialog (ce);
        }
        else {
            ce->changed = FALSE;
            command_state_changed (ce);
        }
    }

    g_object_unref (ce);
    g_free (ecs);
}

static void
contact_modified_cb (EBook *book, EBookStatus status, EditorCloseStruct *ecs)
{
    EContactEditor *ce = ecs->ce;
    gboolean should_close = ecs->should_close;

    gtk_widget_set_sensitive (ce->app, TRUE);
    ce->in_async_call = FALSE;

    g_signal_emit (ce, contact_editor_signals[CONTACT_MODIFIED], 0,
               status, ce->contact);

    if (status == E_BOOK_ERROR_OK) {
        if (should_close) {
            close_dialog (ce);
        }
        else {
            ce->changed = FALSE;
            command_state_changed (ce);
        }
    }

    g_object_unref (ce);
    g_free (ecs);
}

/* Emits the signal to request saving a contact */
static void
save_contact (EContactEditor *ce, gboolean should_close)
{
    EditorCloseStruct *ecs = g_new(EditorCloseStruct, 1);

    extract_info (ce);
    if (!ce->target_book)
        return;

    ecs->ce = ce;
    g_object_ref (ecs->ce);

    ecs->should_close = should_close;

    gtk_widget_set_sensitive (ce->app, FALSE);
    ce->in_async_call = TRUE;

    if (ce->source_book != ce->target_book) {
        /* Two-step move; add to target, then remove from source */
        eab_merging_book_add_contact (ce->target_book, ce->contact,
                          (EBookIdCallback) contact_added_cb, ecs);
    } else {
        if (ce->is_new_contact)
            eab_merging_book_add_contact (ce->target_book, ce->contact,
                              (EBookIdCallback) contact_added_cb, ecs);
        else
            eab_merging_book_commit_contact (ce->target_book, ce->contact,
                             (EBookCallback) contact_modified_cb, ecs);
    }
}

/* Closes the dialog box and emits the appropriate signals */
static void
close_dialog (EContactEditor *ce)
{
    if (ce->app != NULL) {
        gtk_widget_destroy (ce->app);
        ce->app = NULL;
        g_signal_emit (ce, contact_editor_signals[EDITOR_CLOSED], 0);
    }
}

static gboolean
prompt_to_save_changes (EContactEditor *editor)
{
    if (!editor->changed)
        return TRUE;

    switch (eab_prompt_save_dialog (GTK_WINDOW(editor->app))) {
    case GTK_RESPONSE_YES:
        save_contact (editor, FALSE);
        return TRUE;
    case GTK_RESPONSE_NO:
        return TRUE;
    case GTK_RESPONSE_CANCEL:
    default:
        return FALSE;
    }
}

/* Menu callbacks */

/* File/Save callback */
static void
file_save_cb (GtkWidget *widget, gpointer data)
{
    EContactEditor *ce;

    ce = E_CONTACT_EDITOR (data);
    save_contact (ce, FALSE);
}

/* File/Close callback */
static void
file_close_cb (GtkWidget *widget, gpointer data)
{
    EContactEditor *ce;

    ce = E_CONTACT_EDITOR (data);
    if (!prompt_to_save_changes (ce))
        return;

    close_dialog (ce);
}

static void
file_save_as_cb (GtkWidget *widget, gpointer data)
{
    EContactEditor *ce;
    EContact *contact;

    ce = E_CONTACT_EDITOR (data);

    extract_info (ce);

    contact = ce->contact;
    eab_contact_save(_("Save Contact as VCard"), contact, GTK_WINDOW (ce->app));
}

static void
file_send_as_cb (GtkWidget *widget, gpointer data)
{
    EContactEditor *ce;

    ce = E_CONTACT_EDITOR (data);

    extract_info (ce);

    eab_send_contact(ce->contact, EAB_DISPOSITION_AS_ATTACHMENT);
}

static void
file_send_to_cb (GtkWidget *widget, gpointer data)
{
    EContactEditor *ce;

    ce = E_CONTACT_EDITOR (data);

    extract_info (ce);

    eab_send_contact(ce->contact, EAB_DISPOSITION_AS_TO);
}

gboolean
e_contact_editor_confirm_delete (GtkWindow *parent)
{
    GtkWidget *dialog;
    gint result;

    dialog = gtk_message_dialog_new (parent,
                     0,
                     GTK_MESSAGE_QUESTION,
                     GTK_BUTTONS_NONE,
#if notyet
                     /* XXX we really need to handle the plural case here.. */
                     (plural
                      ? _("Are you sure you want\n"
                          "to delete these contacts?"))
#endif
                      _("Are you sure you want\n"
                        "to delete this contact?"));

    gtk_dialog_add_buttons (GTK_DIALOG (dialog),
                GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
                GTK_STOCK_DELETE, GTK_RESPONSE_ACCEPT,
                NULL);

    result = gtk_dialog_run(GTK_DIALOG (dialog));

    gtk_widget_destroy (dialog);

    return (result == GTK_RESPONSE_ACCEPT);
}

static void
contact_deleted_cb (EBook *book, EBookStatus status, EContactEditor *ce)
{
    gtk_widget_set_sensitive (ce->app, TRUE);
    ce->in_async_call = FALSE;

    g_signal_emit (ce, contact_editor_signals[CONTACT_DELETED], 0,
               status, ce->contact);

    /* always close the dialog after we successfully delete a card */
    if (status == E_BOOK_ERROR_OK)
        close_dialog (ce);
}

static void
delete_cb (GtkWidget *widget, gpointer data)
{
    EContactEditor *ce = E_CONTACT_EDITOR (data);
    EContact *contact = ce->contact;

    g_object_ref(contact);

    if (e_contact_editor_confirm_delete(GTK_WINDOW(ce->app))) {

        extract_info (ce);
        
        if (!ce->is_new_contact && ce->source_book) {
            gtk_widget_set_sensitive (ce->app, FALSE);
            ce->in_async_call = TRUE;

            e_book_async_remove_contact (ce->source_book, contact, (EBookCallback)contact_deleted_cb, ce);
        }
    }

    g_object_unref(contact);
}

/* Emits the signal to request printing a card */
static void
print_cb (BonoboUIComponent *uih, void *data, const char *path)
{
    EContactEditor *ce;

    ce = E_CONTACT_EDITOR (data);

    extract_info (ce);

    gtk_widget_show(e_contact_print_contact_dialog_new(ce->contact));
}

#if 0 /* Envelope printing is disabled for Evolution 1.0. */
/* Emits the signal to request printing a card */
static void
print_envelope_cb (BonoboUIComponent *uih, void *data, const char *path)
{
    EContactEditor *ce;

    ce = E_CONTACT_EDITOR (data);

    extract_info (ce);
    e_card_simple_sync_card (ce->simple);

    gtk_widget_show(e_contact_print_envelope_dialog_new(ce->card));
}
#endif

/* Toolbar/Save and Close callback */
static void
tb_save_and_close_cb (BonoboUIComponent *uih, void *data, const char *path)
{
    EContactEditor *ce;

    ce = E_CONTACT_EDITOR (data);
    save_contact (ce, TRUE);
}

static
BonoboUIVerb verbs [] = {
    BONOBO_UI_UNSAFE_VERB ("ContactEditorSave", file_save_cb),
    BONOBO_UI_UNSAFE_VERB ("ContactEditorSaveAs", file_save_as_cb),
    BONOBO_UI_UNSAFE_VERB ("ContactEditorSaveClose", tb_save_and_close_cb),
    BONOBO_UI_UNSAFE_VERB ("ContactEditorSendAs", file_send_as_cb),
    BONOBO_UI_UNSAFE_VERB ("ContactEditorSendTo", file_send_to_cb),
    BONOBO_UI_UNSAFE_VERB ("ContactEditorDelete", delete_cb),
    BONOBO_UI_UNSAFE_VERB ("ContactEditorPrint", print_cb),
#if 0 /* Envelope printing is disabled for Evolution 1.0. */
    BONOBO_UI_UNSAFE_VERB ("ContactEditorPrintEnvelope", print_envelope_cb),
#endif
    /*  BONOBO_UI_UNSAFE_VERB ("ContactEditorPageSetup", file_page_setup_menu), */
    BONOBO_UI_UNSAFE_VERB ("ContactEditorClose", file_close_cb),
    
    BONOBO_UI_VERB_END
};

EPixmap pixmaps[] = {
    E_PIXMAP ("/commands/ContactEditorSave", "save-16.png"),
    E_PIXMAP ("/commands/ContactEditorSaveClose", "save-16.png"),
    E_PIXMAP ("/commands/ContactEditorSaveAs", "save-as-16.png"),
    E_PIXMAP ("/commands/ContactEditorDelete", "evolution-trash-mini.png"),
    E_PIXMAP ("/commands/ContactEditorPrint", "print.xpm"),
#if 0 /* Envelope printing is disabled for Evolution 1.0. */
    E_PIXMAP ("/commands/ContactEditorPrintEnvelope", "print.xpm"),
#endif
    E_PIXMAP ("/Toolbar/ContactEditorSaveClose", "buttons/save-24.png"),
    E_PIXMAP ("/Toolbar/ContactEditorDelete", "buttons/delete-message.png"),
    E_PIXMAP ("/Toolbar/ContactEditorPrint", "buttons/print.png"),

    E_PIXMAP_END
};

static void
create_ui (EContactEditor *ce)
{
    bonobo_ui_component_add_verb_list_with_data (ce->uic, verbs, ce);

    bonobo_ui_util_set_ui (ce->uic, PREFIX,
                   EVOLUTION_UIDIR "/evolution-contact-editor.xml",
                   "evolution-contact-editor", NULL);

    e_pixmaps_update (ce->uic, pixmaps);
}

/* Callback used when the dialog box is destroyed */
static gint
app_delete_event_cb (GtkWidget *widget, GdkEvent *event, gpointer data)
{
    EContactEditor *ce;

    ce = E_CONTACT_EDITOR (data);

    /* if we're saving, don't allow the dialog to close */
    if (ce->in_async_call)
        return TRUE;

    if (!prompt_to_save_changes (ce))
        return TRUE;

    close_dialog (ce);
    return TRUE;
}

static GList *
add_to_tab_order(GList *list, GladeXML *gui, char *name)
{
    GtkWidget *widget = glade_xml_get_widget(gui, name);
    return g_list_prepend(list, widget);
}

static void
setup_tab_order(GladeXML *gui)
{
    GtkWidget *container;
    GList *list = NULL;

    container = glade_xml_get_widget(gui, "table-contact-editor-general");

    if (container) {
        list = add_to_tab_order(list, gui, "entry-fullname");
        list = add_to_tab_order(list, gui, "entry-jobtitle");
        list = add_to_tab_order(list, gui, "entry-company");
        list = add_to_tab_order(list, gui, "combo-file-as");
        list = add_to_tab_order(list, gui, "entry-phone1");
        list = add_to_tab_order(list, gui, "entry-phone2");
        list = add_to_tab_order(list, gui, "entry-phone3");
        list = add_to_tab_order(list, gui, "entry-phone4");

        list = add_to_tab_order(list, gui, "entry-email1");
        list = add_to_tab_order(list, gui, "alignment-htmlmail");
        list = add_to_tab_order(list, gui, "entry-web");
        list = add_to_tab_order(list, gui, "entry-blog");
        list = add_to_tab_order(list, gui, "button-fulladdr");
        list = add_to_tab_order(list, gui, "text-address");
        list = g_list_reverse(list);
        e_container_change_tab_order(GTK_CONTAINER(container), list);
        g_list_free(list);
    }
}

static void
e_contact_editor_init (EContactEditor *e_contact_editor)
{
    GladeXML *gui;
    GtkWidget *widget;
    GtkWidget *bonobo_win;
    GtkWidget *wants_html;
    BonoboUIContainer *container;
    char *icon_path;

    e_contact_editor->email_info = NULL;
    e_contact_editor->phone_info = NULL;
    e_contact_editor->address_info = NULL;
    e_contact_editor->email_popup = NULL;
    e_contact_editor->phone_popup = NULL;
    e_contact_editor->address_popup = NULL;
    e_contact_editor->email_list = NULL;
    e_contact_editor->phone_list = NULL;
    e_contact_editor->address_list = NULL;
    e_contact_editor->name = e_contact_name_new();
    e_contact_editor->company = g_strdup("");
    
    e_contact_editor->email_choice = E_CONTACT_EMAIL_1;
    e_contact_editor->phone_choice[0] = E_CONTACT_PHONE_BUSINESS;
    e_contact_editor->phone_choice[1] = E_CONTACT_PHONE_HOME;
    e_contact_editor->phone_choice[2] = E_CONTACT_PHONE_BUSINESS_FAX;
    e_contact_editor->phone_choice[3] = E_CONTACT_PHONE_MOBILE;
#if 1
    e_contact_editor->address_choice = E_CONTACT_FIRST_ADDRESS_ID;
    e_contact_editor->address_mailing = -1;
#endif  

    e_contact_editor->contact = NULL;
    e_contact_editor->changed = FALSE;
    e_contact_editor->image_set = FALSE;
    e_contact_editor->in_async_call = FALSE;
    e_contact_editor->source_editable = TRUE;
    e_contact_editor->target_editable = TRUE;

    e_contact_editor->load_source_id = 0;
    e_contact_editor->load_book = NULL;

    gui = glade_xml_new (EVOLUTION_GLADEDIR "/contact-editor.glade", NULL, NULL);
    e_contact_editor->gui = gui;

    setup_tab_order(gui);

    e_contact_editor->app = glade_xml_get_widget (gui, "contact editor");

    connect_arrow_button_signals(e_contact_editor);
    set_entry_changed_signals(e_contact_editor);

    setup_im_treeview(e_contact_editor);

    wants_html = glade_xml_get_widget(e_contact_editor->gui, "checkbutton-htmlmail");
    if (wants_html && GTK_IS_TOGGLE_BUTTON(wants_html))
        g_signal_connect (wants_html, "toggled",
                  G_CALLBACK (wants_html_changed), e_contact_editor);

    widget = glade_xml_get_widget(e_contact_editor->gui, "checkbutton-mailingaddress");
    if (widget && GTK_IS_TOGGLE_BUTTON(widget))
        g_signal_connect (widget, "toggled",
                  G_CALLBACK (address_mailing_changed), e_contact_editor);
    
    widget = glade_xml_get_widget(e_contact_editor->gui, "button-fullname");
    if (widget && GTK_IS_BUTTON(widget))
        g_signal_connect (widget, "clicked",
                  G_CALLBACK (full_name_clicked), e_contact_editor);

    widget = glade_xml_get_widget(e_contact_editor->gui, "button-fulladdr");
    if (widget && GTK_IS_BUTTON(widget))
        g_signal_connect (widget, "clicked",
                  G_CALLBACK (full_addr_clicked), e_contact_editor);

    widget = glade_xml_get_widget(e_contact_editor->gui, "button-categories");
    if (widget && GTK_IS_BUTTON(widget))
        g_signal_connect (widget, "clicked",
                  G_CALLBACK (categories_clicked), e_contact_editor);
    widget = glade_xml_get_widget (e_contact_editor->gui, "source-option-menu-source");
    if (widget && E_IS_SOURCE_OPTION_MENU (widget))
        g_signal_connect (widget, "source_selected",
                  G_CALLBACK (source_selected), e_contact_editor);

    widget = glade_xml_get_widget(e_contact_editor->gui, "button-im-add");
    if (widget && GTK_IS_BUTTON(widget))
        g_signal_connect (widget, "clicked",
                  G_CALLBACK (add_im_clicked), e_contact_editor);

    widget = glade_xml_get_widget(e_contact_editor->gui, "button-im-edit");
    if (widget && GTK_IS_BUTTON(widget))
        g_signal_connect (widget, "clicked",
                  G_CALLBACK (edit_im_clicked), e_contact_editor);

    widget = glade_xml_get_widget(e_contact_editor->gui, "button-im-remove");
    if (widget && GTK_IS_BUTTON(widget))
        g_signal_connect (widget, "clicked",
                  G_CALLBACK (remove_im_clicked), e_contact_editor);


    /* Construct the app */
    bonobo_win = bonobo_window_new ("contact-editor-dialog", _("Contact Editor"));

    /* FIXME: The sucking bit */
    {
        GtkWidget *contents;

        contents = bonobo_dock_get_client_area (gnome_app_get_dock (GNOME_APP(e_contact_editor->app)));

        if (!contents) {
            g_message ("contact_editor_construct(): Could not get contents");
            return;
        }
        g_object_ref (contents);
        gtk_container_remove (GTK_CONTAINER (contents->parent), contents);
        bonobo_window_set_contents (BONOBO_WINDOW (bonobo_win), contents);
        gtk_widget_destroy (e_contact_editor->app);
        e_contact_editor->app = bonobo_win;
    }

    /* Build the menu and toolbar */
    container = bonobo_window_get_ui_container (BONOBO_WINDOW (e_contact_editor->app));

    e_contact_editor->uic = bonobo_ui_component_new_default ();
    if (!e_contact_editor->uic) {
        g_message ("e_contact_editor_init(): eeeeek, could not create the UI handler!");
        return;
    }
    bonobo_ui_component_set_container (e_contact_editor->uic,
                       bonobo_object_corba_objref (BONOBO_OBJECT (container)),
                       NULL);

    create_ui (e_contact_editor);

    widget = glade_xml_get_widget(e_contact_editor->gui, "entry-fullname");
    if (widget)
        gtk_widget_grab_focus (widget);

    /* Connect to the deletion of the dialog */

    g_signal_connect (e_contact_editor->app, "delete_event",
                GTK_SIGNAL_FUNC (app_delete_event_cb), e_contact_editor);

    /* set the icon */
    icon_path = g_build_filename (EVOLUTION_IMAGESDIR, "evolution-contacts-mini.png", NULL);
    gnome_window_icon_set_from_file (GTK_WINDOW (e_contact_editor->app), icon_path);
    g_free (icon_path);
}

void
e_contact_editor_dispose (GObject *object) {
    EContactEditor *e_contact_editor = E_CONTACT_EDITOR(object);

    if (e_contact_editor->writable_fields) {
        g_object_unref(e_contact_editor->writable_fields);
        e_contact_editor->writable_fields = NULL;
    }
    if (e_contact_editor->email_list) {
        g_list_foreach(e_contact_editor->email_list, (GFunc) g_free, NULL);
        g_list_free(e_contact_editor->email_list);
        e_contact_editor->email_list = NULL;
    }
    if (e_contact_editor->email_info) {
        g_free(e_contact_editor->email_info);
        e_contact_editor->email_info = NULL;
    }
    if (e_contact_editor->email_popup) {
        g_object_unref(e_contact_editor->email_popup);
        e_contact_editor->email_popup = NULL;
    }
    
    if (e_contact_editor->phone_list) {
        g_list_foreach(e_contact_editor->phone_list, (GFunc) g_free, NULL);
        g_list_free(e_contact_editor->phone_list);
        e_contact_editor->phone_list = NULL;
    }
    if (e_contact_editor->phone_info) {
        g_free(e_contact_editor->phone_info);
        e_contact_editor->phone_info = NULL;
    }
    if (e_contact_editor->phone_popup) {
        g_object_unref(e_contact_editor->phone_popup);
        e_contact_editor->phone_popup = NULL;
    }
    
    if (e_contact_editor->address_list) {
        g_list_foreach(e_contact_editor->address_list, (GFunc) g_free, NULL);
        g_list_free(e_contact_editor->address_list);
        e_contact_editor->address_list = NULL;
    }
    if (e_contact_editor->address_info) {
        g_free(e_contact_editor->address_info);
        e_contact_editor->address_info = NULL;
    }
    if (e_contact_editor->address_popup) {
        g_object_unref(e_contact_editor->address_popup);
        e_contact_editor->address_popup = NULL;
    }
    
    if (e_contact_editor->contact) {
        g_object_unref(e_contact_editor->contact);
        e_contact_editor->contact = NULL;
    }
    
    if (e_contact_editor->source_book) {
        g_object_unref(e_contact_editor->source_book);
        e_contact_editor->source_book = NULL;
    }

    if (e_contact_editor->target_book) {
        g_object_unref(e_contact_editor->target_book);
        e_contact_editor->target_book = NULL;
    }

    if (e_contact_editor->name) {
        e_contact_name_free(e_contact_editor->name);
        e_contact_editor->name = NULL;
    }

    if (e_contact_editor->company) {
        g_free (e_contact_editor->company);
        e_contact_editor->company = NULL;
    }

    if (e_contact_editor->gui) {
        g_object_unref(e_contact_editor->gui);
        e_contact_editor->gui = NULL;
    }

    cancel_load (e_contact_editor);
}

static void
command_state_changed (EContactEditor *ce)
{
    bonobo_ui_component_set_prop (ce->uic,
                      "/commands/ContactEditorSaveClose",
                      "sensitive",
                      (ce->target_editable && ce->changed) ? "1" : "0", NULL);
    bonobo_ui_component_set_prop (ce->uic,
                      "/commands/ContactEditorSave",
                      "sensitive",
                      (ce->target_editable && ce->changed) ? "1" : "0", NULL);
    bonobo_ui_component_set_prop (ce->uic,
                      "/commands/ContactEditorDelete",
                      "sensitive",
                      (ce->source_editable && !ce->is_new_contact) ? "1" : "0", NULL);
}

static void
supported_fields_cb (EBook *book, EBookStatus status,
             EList *fields, EContactEditor *ce)
{
    if (!g_slist_find (all_contact_editors, ce)) {
        g_warning ("supported_fields_cb called for book that's still around, but contact editor that's been destroyed.");
        return;
    }

    g_object_set (ce,
            "writable_fields", fields,
            NULL);

    e_contact_editor_show (ce);

    command_state_changed (ce);
}

static void
contact_editor_destroy_notify (void *data,
                   GObject *where_the_object_was)
{
    EContactEditor *ce = E_CONTACT_EDITOR (data);

    all_contact_editors = g_slist_remove (all_contact_editors, ce);
}

EContactEditor *
e_contact_editor_new (EBook *book,
              EContact *contact,
              gboolean is_new_contact,
              gboolean editable)
{
    EContactEditor *ce;

    g_return_val_if_fail (E_IS_BOOK (book), NULL);
    g_return_val_if_fail (E_IS_CONTACT (contact), NULL);

    ce = g_object_new (E_TYPE_CONTACT_EDITOR, NULL);

    all_contact_editors = g_slist_prepend (all_contact_editors, ce);
    g_object_weak_ref (G_OBJECT (ce), contact_editor_destroy_notify, ce);

    g_object_ref (ce);
    gtk_object_sink (GTK_OBJECT (ce));

    g_object_set (ce,
              "source_book", book,
              "contact", contact,
              "is_new_contact", is_new_contact,
              "editable", editable,
              NULL);

    if (book)
        e_book_async_get_supported_fields (book, (EBookFieldsCallback)supported_fields_cb, ce);

    return ce;
}

static void
e_contact_editor_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec)
{
    EContactEditor *editor;

    editor = E_CONTACT_EDITOR (object);
    
    switch (prop_id){
    case PROP_SOURCE_BOOK: {
        gboolean writable;
        gboolean changed = FALSE;

        if (editor->source_book)
            g_object_unref(editor->source_book);
        editor->source_book = E_BOOK (g_value_get_object (value));
        g_object_ref (editor->source_book);

        if (!editor->target_book) {
            editor->target_book = editor->source_book;
            g_object_ref (editor->target_book);

            e_book_async_get_supported_fields (editor->target_book,
                               (EBookFieldsCallback) supported_fields_cb, editor);
        }

        writable = e_book_is_writable (editor->source_book);
        if (writable != editor->source_editable) {
            editor->source_editable = writable;
            changed = TRUE;
        }

        writable = e_book_is_writable (editor->target_book);
        if (writable != editor->target_editable) {
            editor->target_editable = writable;
            changed = TRUE;
        }

        if (changed) {
            set_editable (editor);
            command_state_changed (editor);
        }

        /* XXX more here about editable/etc. */
        break;
    }

    case PROP_TARGET_BOOK: {
        gboolean writable;
        gboolean changed = FALSE;

        if (editor->target_book)
            g_object_unref(editor->target_book);
        editor->target_book = E_BOOK (g_value_get_object (value));
        g_object_ref (editor->target_book);

        e_book_async_get_supported_fields (editor->target_book,
                           (EBookFieldsCallback) supported_fields_cb, editor);

        if (!editor->changed && !editor->is_new_contact) {
            editor->changed = TRUE;
            changed = TRUE;
        }

        writable = e_book_is_writable (editor->target_book);
        if (writable != editor->target_editable) {
            editor->target_editable = writable;
            set_editable (editor);
            changed = TRUE;
        }

        if (changed)
            command_state_changed (editor);

        /* If we're trying to load a new target book, cancel that here. */
        cancel_load (editor);

        /* XXX more here about editable/etc. */
        break;
    }

    case PROP_CONTACT:
        if (editor->contact)
            g_object_unref(editor->contact);
        editor->contact = e_contact_duplicate(E_CONTACT(g_value_get_object (value)));
        fill_in_info(editor);
        editor->changed = FALSE;
        break;

    case PROP_IS_NEW_CONTACT:
        editor->is_new_contact = g_value_get_boolean (value) ? TRUE : FALSE;
        break;

    case PROP_EDITABLE: {
        gboolean new_value = g_value_get_boolean (value) ? TRUE : FALSE;
        gboolean changed = (editor->target_editable != new_value);

        editor->target_editable = new_value;

        if (changed) {
            set_editable (editor);
            command_state_changed (editor);
        }
        break;
    }

    case PROP_CHANGED: {
        gboolean new_value = g_value_get_boolean (value) ? TRUE : FALSE;
        gboolean changed = (editor->changed != new_value);

        editor->changed = new_value;

        if (changed)
            command_state_changed (editor);
        break;
    }
    case PROP_WRITABLE_FIELDS:
        if (editor->writable_fields)
            g_object_unref(editor->writable_fields);
        editor->writable_fields = g_value_get_object (value);
        if (editor->writable_fields)
            g_object_ref (editor->writable_fields);
        else
            editor->writable_fields = e_list_new(NULL, NULL, NULL);
        enable_writable_fields (editor);
        break;
    default:
        G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
        break;
    }
}

static void
e_contact_editor_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec)
{
    EContactEditor *e_contact_editor;

    e_contact_editor = E_CONTACT_EDITOR (object);

    switch (prop_id) {
    case PROP_SOURCE_BOOK:
        g_value_set_object (value, e_contact_editor->source_book);
        break;

    case PROP_TARGET_BOOK:
        g_value_set_object (value, e_contact_editor->target_book);
        break;

    case PROP_CONTACT:
        extract_info(e_contact_editor);
        g_value_set_object (value, e_contact_editor->contact);
        break;

    case PROP_IS_NEW_CONTACT:
        g_value_set_boolean (value, e_contact_editor->is_new_contact ? TRUE : FALSE);
        break;

    case PROP_EDITABLE:
        g_value_set_boolean (value, e_contact_editor->target_editable ? TRUE : FALSE);
        break;

    case PROP_CHANGED:
        g_value_set_boolean (value, e_contact_editor->changed ? TRUE : FALSE);
        break;

    case PROP_WRITABLE_FIELDS:
        if (e_contact_editor->writable_fields)
            g_value_set_object (value, e_list_duplicate (e_contact_editor->writable_fields));
        else
            g_value_set_object (value, NULL);
        break;
    default:
        G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
        break;
    }
}

static void
_popup_position(GtkMenu *menu,
        gint *x,
        gint *y,
        gboolean *push_in,
        gpointer data)
{
    GtkWidget *button = GTK_WIDGET(data);
    GtkRequisition request;
    int mh, mw;
    gdk_window_get_origin (button->window, x, y);
    *x += button->allocation.x;
    *y += button->allocation.y;

    gtk_widget_size_request(GTK_WIDGET(menu), &request);

    mh = request.height;
    mw = request.width;

    *x -= mw;
    if (*x < 0)
        *x = 0;
    
    if (*y < 0)
        *y = 0;
    
    if ((*x + mw) > gdk_screen_width ())
        *x = gdk_screen_width () - mw;
    
    if ((*y + mh) > gdk_screen_height ())
        *y = gdk_screen_height () - mh;

    *push_in = FALSE;
}

static gint
_arrow_pressed (GtkWidget *widget, GdkEventButton *button, EContactEditor *editor, GtkWidget *popup, GList **list, GnomeUIInfo **info, gchar *label)
{
    gint menu_item;

    g_signal_stop_emission_by_name (widget, "button_press_event");

    gtk_widget_realize(popup);
    menu_item = gnome_popup_menu_do_popup_modal(popup, _popup_position, widget, button, editor, widget);
    if ( menu_item != -1 ) {
        GtkWidget *label_widget = glade_xml_get_widget(editor->gui, label);
        if (label_widget && GTK_IS_LABEL(label_widget)) {
            g_object_set (label_widget,
                      "label", _(g_list_nth_data(*list, menu_item)),
                      NULL);
        }
    }
    return menu_item;
}

static void
e_contact_editor_build_ui_info(GList *list, GnomeUIInfo **infop)
{
    GnomeUIInfo *info;
    GnomeUIInfo singleton = { GNOME_APP_UI_TOGGLEITEM, NULL, NULL, NULL, NULL, NULL, GNOME_APP_PIXMAP_NONE, 0, 0, 0, NULL };
    GnomeUIInfo end = GNOMEUIINFO_END;
    int length;
    int i;

    info = *infop;

    if ( info )
        g_free(info);
    length = g_list_length( list );
    info = g_new(GnomeUIInfo, length + 2);
    for (i = 0; i < length; i++) {
        info[i] = singleton;
        info[i].label = _(list->data);
        list = list->next;
    }
    info[i] = end;

    *infop = info;
}

static void
e_contact_editor_build_phone_ui (EContactEditor *editor)
{
    if (editor->phone_list == NULL) {
        int i;

        for (i = 0; i < G_N_ELEMENTS (phones); i ++) {
            editor->phone_list = g_list_append(editor->phone_list, g_strdup(e_contact_pretty_name (phones[i])));
        }
    }
    if (editor->phone_info == NULL) {
        e_contact_editor_build_ui_info(editor->phone_list, &editor->phone_info);
        
        if ( editor->phone_popup )
            g_object_unref(editor->phone_popup);
        
        editor->phone_popup = gnome_popup_menu_new(editor->phone_info);
        g_object_ref (editor->phone_popup);
        gtk_object_sink (GTK_OBJECT (editor->phone_popup));
    }
}

static void
e_contact_editor_build_email_ui (EContactEditor *editor)
{
    int i;

    if (editor->email_list == NULL) {
        for (i = 0; i < G_N_ELEMENTS (emails); i++)
            editor->email_list = g_list_append(editor->email_list, g_strdup(e_contact_pretty_name (emails[i])));
    }
    if (editor->email_info == NULL) {
        e_contact_editor_build_ui_info(editor->email_list, &editor->email_info);

        if ( editor->email_popup )
            g_object_unref(editor->email_popup);
        
        editor->email_popup = gnome_popup_menu_new(editor->email_info);
        g_object_ref (editor->email_popup);
        gtk_object_sink (GTK_OBJECT (editor->email_popup));
    }
}

static void
e_contact_editor_build_address_ui (EContactEditor *editor)
{
    int i;

    if (editor->address_list == NULL) {
        static char *info[] = {
            N_("Business"),
            N_("Home"),
            N_("Other")
        };
        
        for (i = 0; i < sizeof(info) / sizeof(info[0]); i++) {
            editor->address_list = g_list_append(editor->address_list, g_strdup(info[i]));
        }
    }
    if (editor->address_info == NULL) {
        e_contact_editor_build_ui_info(editor->address_list, &editor->address_info);

        if ( editor->address_popup )
            g_object_unref(editor->address_popup);
        
        editor->address_popup = gnome_popup_menu_new(editor->address_info);
        g_object_ref (editor->address_popup);
        gtk_object_sink (GTK_OBJECT (editor->address_popup));
    }
}

static void
_phone_arrow_pressed (GtkWidget *widget, GdkEventButton *button, EContactEditor *editor)
{
    int which;
    int i;
    gchar *label;
    gchar *entry;
    int result;
    if ( widget == glade_xml_get_widget(editor->gui, "button-phone1") ) {
        which = 1;
    } else if ( widget == glade_xml_get_widget(editor->gui, "button-phone2") ) {
        which = 2;
    } else if ( widget == glade_xml_get_widget(editor->gui, "button-phone3") ) {
        which = 3;
    } else if ( widget == glade_xml_get_widget(editor->gui, "button-phone4") ) {
        which = 4;
    } else
        return;
    
    label = g_strdup_printf("label-phone%d", which);
    entry = g_strdup_printf("entry-phone%d", which);

    e_contact_editor_build_phone_ui (editor);
    
    for(i = 0; i < G_N_ELEMENTS (phones); i++) {
        char *phone = e_contact_get (editor->contact, phones[i]);
        gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(editor->phone_info[i].widget),
                           phone && *phone);
        g_free (phone);
    }
    
    result = _arrow_pressed (widget, button, editor, editor->phone_popup, &editor->phone_list, &editor->phone_info, label);
    
    if (result != -1) {
        GtkWidget *w = glade_xml_get_widget (editor->gui, entry);
        editor->phone_choice[which - 1] = phones[result];
        set_fields (editor);
        enable_widget (glade_xml_get_widget (editor->gui, label), TRUE);
        enable_widget (w, editor->target_editable);
    }

    g_free(label);
    g_free(entry);
}

static void
_email_arrow_pressed (GtkWidget *widget, GdkEventButton *button, EContactEditor *editor)
{
    int i;
    int result;

    e_contact_editor_build_email_ui (editor);

    for(i = 0; i < G_N_ELEMENTS (emails); i++) {
        char *string = e_contact_get (editor->contact, emails[i]);
        gboolean checked;
        checked = string && *string;
        gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(editor->email_info[i].widget),
                           checked);
        g_free (string);
    }
    
    result = _arrow_pressed (widget, button, editor, editor->email_popup, &editor->email_list, &editor->email_info, "label-email1");
    
    if (result != -1) {
        GtkWidget *entry = glade_xml_get_widget (editor->gui, "entry-email1");
        editor->email_choice = result + E_CONTACT_FIRST_EMAIL_ID;

        set_fields (editor);

        /* make sure the buttons/entry is/are sensitive */
        enable_widget (glade_xml_get_widget (editor->gui, "label-email1"), TRUE);
        enable_widget (entry, editor->target_editable);
        enable_widget (glade_xml_get_widget (editor->gui, "checkbutton-htmlmail"), editor->target_editable);
    }
}

static void
_address_arrow_pressed (GtkWidget *widget, GdkEventButton *button, EContactEditor *editor)
{
    int i;
    int result;

    e_contact_editor_build_address_ui (editor);

    for (i = E_CONTACT_FIRST_ADDRESS_ID; i <= E_CONTACT_LAST_ADDRESS_ID; i++) {
        EContactAddress *address = e_contact_get (editor->contact, i);
        gboolean checked;

        checked = (address != NULL);
        gtk_check_menu_item_set_active (
            GTK_CHECK_MENU_ITEM (editor->address_info [i - E_CONTACT_FIRST_ADDRESS_ID].widget),
            checked);

        if (address)
            g_boxed_free (e_contact_address_get_type (), address);
    }

    result = _arrow_pressed (widget, button, editor, editor->address_popup, &editor->address_list, &editor->address_info, "label-address");

    if (result != -1) {
        set_address_field(editor, result + E_CONTACT_FIRST_ADDRESS_ID);

        /* make sure the buttons/entry is/are sensitive */
        enable_widget (glade_xml_get_widget (editor->gui, "label-address"), TRUE);
        enable_widget (glade_xml_get_widget (editor->gui, "text-address"), editor->address_editable[result]);
        enable_widget (glade_xml_get_widget (editor->gui, "checkbutton-mailingaddress"), editor->address_editable[result]);
    }
}

#if 0
static void
find_address_mailing (EContactEditor *editor)
{
    const EContactAddress *address;
    int i;
    
    editor->address_mailing = -1;
    for (i = E_CONTACT_FIRST_ADDRESS_ID; i <= E_CONTACT_LAST_ADDRESS_ID; i++) {
        address = e_contact_get_const (editor->contact, i);
        if (address && (address->flags & E_CARD_ADDR_DEFAULT)) {
            if (editor->address_mailing == -1) {
                editor->address_mailing = i;
            } else {
                EContactAddress *new;
                
                new = e_card_delivery_address_copy (address);
                new = g_boxed_copy (e_contact_address_get_type (), address);
                new->flags &= ~E_CARD_ADDR_DEFAULT;
                e_card_simple_set_delivery_address(editor->simple, i, new);
                e_card_delivery_address_unref (new);
            }
        }
    }
}
#endif

static void
set_field(EContactEditor *editor, GtkEntry *entry, const char *string)
{
    const char *oldstring = gtk_entry_get_text(entry);
    if (!string)
        string = "";
    if (strcmp(string, oldstring)) {
        g_signal_handlers_block_matched (entry,
                         G_SIGNAL_MATCH_DATA,
                         0, 0, NULL, NULL, editor);
        gtk_entry_set_text(entry, string);
        g_signal_handlers_unblock_matched (entry,
                           G_SIGNAL_MATCH_DATA,
                           0, 0, NULL, NULL, editor);
    }
}

static void
set_phone_field(EContactEditor *editor, GtkWidget *entry, const char *phone_number)
{
    set_field(editor, GTK_ENTRY(entry), phone_number ? phone_number : "");
}

static void
set_source_field (EContactEditor *editor)
{
    GtkWidget *source_menu;
    ESource   *source;

    if (!editor->target_book)
        return;

    source_menu = glade_xml_get_widget (editor->gui, "source-option-menu-source");
    source = e_book_get_source (editor->target_book);

    e_source_option_menu_select (E_SOURCE_OPTION_MENU (source_menu), source);
}

static void
set_fields(EContactEditor *editor)
{
    EContactAddress *address;
    GtkWidget *entry;
    GtkWidget *label_widget;
    int i;

    entry = glade_xml_get_widget(editor->gui, "entry-phone1");
    if (entry && GTK_IS_ENTRY(entry))
        set_phone_field(editor, entry, e_contact_get_const(editor->contact, editor->phone_choice[0]));

    entry = glade_xml_get_widget(editor->gui, "entry-phone2");
    if (entry && GTK_IS_ENTRY(entry))
        set_phone_field(editor, entry, e_contact_get_const(editor->contact, editor->phone_choice[1]));

    entry = glade_xml_get_widget(editor->gui, "entry-phone3");
    if (entry && GTK_IS_ENTRY(entry))
        set_phone_field(editor, entry, e_contact_get_const(editor->contact, editor->phone_choice[2]));

    entry = glade_xml_get_widget(editor->gui, "entry-phone4");
    if (entry && GTK_IS_ENTRY(entry))
        set_phone_field(editor, entry, e_contact_get_const(editor->contact, editor->phone_choice[3]));
    
    entry = glade_xml_get_widget(editor->gui, "entry-email1");
    if (entry && GTK_IS_ENTRY(entry))
        set_field(editor, GTK_ENTRY(entry), e_contact_get_const(editor->contact, editor->email_choice));


    e_contact_editor_build_address_ui (editor);

    /* If address field is selected, try that first. If we don't have that, start
     * from top of field list */
    if (editor->address_choice == -1)
        address = NULL;
    else
        address = e_contact_get (editor->contact, editor->address_choice);

    if (address) {
        i = editor->address_choice;
        g_boxed_free (e_contact_address_get_type (), address);
    } else {
        for (i = E_CONTACT_FIRST_ADDRESS_ID; i <= E_CONTACT_LAST_ADDRESS_ID; i++) {
            address = e_contact_get (editor->contact, i);

            if (address) {
                g_boxed_free (e_contact_address_get_type (), address);
                break;
            }
        }
        if (i > E_CONTACT_LAST_ADDRESS_ID)
            i = E_CONTACT_FIRST_ADDRESS_ID;

        label_widget = glade_xml_get_widget(editor->gui, "label-address");
        if (label_widget && GTK_IS_LABEL(label_widget)) {
            g_object_set (label_widget,
                      "label", _(g_list_nth_data(editor->address_list, i - E_CONTACT_FIRST_ADDRESS_ID)),
                      NULL);
        }
    }

    set_address_field(editor, i);
    set_source_field (editor);
}

static void
add_im_field(EContactEditor *editor,
         EContactField field,
         const char *service,
         const char *desc)
{
    GList *list;
    GList *l;
    GtkTreeIter iter;
    GdkPixbuf *pixbuf;
    GdkPixbuf *scale = NULL;
    char *icon_path;
    char *buf;

    list = e_contact_get_attributes (editor->contact, field);

    buf = g_strdup_printf("im-%s.png", service);
    icon_path = g_concat_dir_and_file(EVOLUTION_IMAGESDIR, buf);
    pixbuf = gdk_pixbuf_new_from_file(icon_path, NULL);
    g_free(icon_path);
    g_free(buf);

    if (pixbuf != NULL)
        scale = gdk_pixbuf_scale_simple(pixbuf, 16, 16, GDK_INTERP_BILINEAR);

    for (l = list; l != NULL; l = l->next) {
        EVCardAttribute *attr = l->data;
        char *account_name;
        char *location;
        char *location_type;

        account_name = e_vcard_attribute_get_value (attr);
        if (!account_name)
            continue;

        if (e_vcard_attribute_has_type (attr, "HOME")) {
            location_type = "HOME";
            location = _("Home");
        }
        else if (e_vcard_attribute_has_type (attr, "WORK")) {
            location_type = "WORK";
            location = _("Work");
        }
        else {
            location_type = NULL;
            location = _("Other");
        }

        gtk_list_store_append(editor->im_model, &iter);

        gtk_list_store_set(editor->im_model, &iter,
                   COLUMN_IM_ICON, scale,
                   COLUMN_IM_SERVICE,  desc,
                   COLUMN_IM_LOCATION, location,
                   COLUMN_IM_LOCATION_TYPE, location_type,
                   COLUMN_IM_SCREENNAME, account_name,
                   COLUMN_IM_SERVICE_FIELD, field,
                   -1);
    }

    if (scale != NULL)
        g_object_unref(G_OBJECT(scale));

    if (pixbuf != NULL)
        g_object_unref(G_OBJECT(pixbuf));
}

static void
set_im_fields(EContactEditor *editor)
{
    gtk_list_store_clear(editor->im_model);

    add_im_field(editor, E_CONTACT_IM_AIM,    "aim",    _("AIM"));
    add_im_field(editor, E_CONTACT_IM_JABBER, "jabber", _("Jabber"));
    add_im_field(editor, E_CONTACT_IM_YAHOO,  "yahoo",  _("Yahoo"));
    add_im_field(editor, E_CONTACT_IM_MSN,    "msn",    _("MSN"));
    add_im_field(editor, E_CONTACT_IM_ICQ,    "icq",    _("ICQ"));
    add_im_field(editor, E_CONTACT_IM_GROUPWISE, "nov", _("GroupWise"));
}

static void
set_address_field(EContactEditor *editor, int result)
{
    GtkWidget *text, *check;
    
    text = glade_xml_get_widget(editor->gui, "text-address");

    if (text && GTK_IS_TEXT_VIEW(text)) {
        GtkTextBuffer *buffer;
        GtkTextIter start_iter, end_iter;
        EContactAddress *address;

        if (result == -1)
            result = editor->address_choice;
        editor->address_choice = -1;

        buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (text));

        gtk_text_buffer_get_start_iter (buffer, &start_iter);
        gtk_text_buffer_get_end_iter (buffer, &end_iter);

        gtk_text_buffer_delete (buffer, &start_iter, &end_iter);

        address = e_contact_get (editor->contact, result);
        if (address) {
            gchar *text;

            text = address_to_text (address);
            gtk_text_buffer_insert (buffer, &start_iter, text, strlen (text));
            g_free (text);
        }

        check = glade_xml_get_widget(editor->gui, "checkbutton-mailingaddress");
        if (check && GTK_IS_CHECK_BUTTON (check)) {
#if 0
            if (address && address->data)
                gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (check), 
                                  address->flags & E_CARD_ADDR_DEFAULT);
            else
                gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (check), FALSE);
#endif
        }
        
        editor->address_choice = result;

        if (address)
            g_boxed_free (e_contact_address_get_type (), address);
    }
}

static struct {
    char *id;
    EContactField field;
} field_mapping [] = {
    { "entry-fullname", E_CONTACT_FULL_NAME },
    { "entry-web", E_CONTACT_HOMEPAGE_URL },
    { "entry-blog", E_CONTACT_BLOG_URL },
    { "entry-company", E_CONTACT_ORG },
    { "entry-department", E_CONTACT_ORG_UNIT },
    { "entry-office", E_CONTACT_OFFICE },
    { "entry-jobtitle", E_CONTACT_TITLE },
    { "entry-profession", E_CONTACT_ROLE },
    { "entry-manager", E_CONTACT_MANAGER },
    { "entry-assistant", E_CONTACT_ASSISTANT },
    { "entry-nickname", E_CONTACT_NICKNAME },
    { "entry-spouse", E_CONTACT_SPOUSE },
    { "text-comments", E_CONTACT_NOTE },
    { "entry-categories", E_CONTACT_CATEGORIES },
    { "entry-caluri", E_CONTACT_CALENDAR_URI },
    { "entry-fburl", E_CONTACT_FREEBUSY_URL },
    { "entry-videourl", E_CONTACT_VIDEO_URL },
};

static void
fill_in_field(EContactEditor *editor, char *id, char *value)
{
    GtkWidget *widget = glade_xml_get_widget(editor->gui, id);

    if (!widget)
        return;

    if (E_IS_URL_ENTRY (widget))
        widget = e_url_entry_get_entry (E_URL_ENTRY (widget));

    if (GTK_IS_TEXT_VIEW (widget)) {
        if (value)
            gtk_text_buffer_set_text (gtk_text_view_get_buffer (GTK_TEXT_VIEW (widget)),
                          value, strlen (value));
    }
    else if (GTK_IS_EDITABLE(widget)) {
        int position = 0;
        GtkEditable *editable = GTK_EDITABLE(widget);
        gtk_editable_delete_text(editable, 0, -1);
        if (value)
            gtk_editable_insert_text(editable, value, strlen(value), &position);
    }
}

static void
disable_widget_foreach (char *key, GtkWidget *widget, gpointer closure)
{
    enable_widget (widget, FALSE);
}

static struct {
    char *widget_name;
    EContactField field_id;
    gboolean desensitize_for_read_only;
} widget_field_mappings[] = {
    { "entry-web", E_CONTACT_HOMEPAGE_URL, TRUE },
    { "accellabel-web", E_CONTACT_HOMEPAGE_URL },

    { "entry-blog", E_CONTACT_BLOG_URL, TRUE },
    { "accellabel-blog", E_CONTACT_BLOG_URL },

    { "entry-jobtitle", E_CONTACT_TITLE, TRUE },
    { "label-jobtitle", E_CONTACT_TITLE },

    { "entry-company", E_CONTACT_ORG, TRUE },
    { "label-company", E_CONTACT_ORG },

    { "combo-file-as", E_CONTACT_FILE_AS, TRUE },
    { "entry-file-as", E_CONTACT_FILE_AS, TRUE },
    { "accellabel-fileas", E_CONTACT_FILE_AS },

    { "label-department", E_CONTACT_ORG_UNIT },
    { "entry-department", E_CONTACT_ORG_UNIT, TRUE },

    { "label-office", E_CONTACT_OFFICE },
    { "entry-office", E_CONTACT_OFFICE, TRUE },

    { "label-profession", E_CONTACT_ROLE },
    { "entry-profession", E_CONTACT_ROLE, TRUE },

    { "label-manager", E_CONTACT_MANAGER },
    { "entry-manager", E_CONTACT_MANAGER, TRUE },

    { "label-assistant", E_CONTACT_ASSISTANT },
    { "entry-assistant", E_CONTACT_ASSISTANT, TRUE },

    { "label-nickname", E_CONTACT_NICKNAME },
    { "entry-nickname", E_CONTACT_NICKNAME, TRUE },

    { "label-spouse", E_CONTACT_SPOUSE },
    { "entry-spouse", E_CONTACT_SPOUSE, TRUE },

    { "label-birthday", E_CONTACT_BIRTH_DATE },
    { "dateedit-birthday", E_CONTACT_BIRTH_DATE, TRUE },

    { "label-anniversary", E_CONTACT_ANNIVERSARY },
    { "dateedit-anniversary", E_CONTACT_ANNIVERSARY, TRUE },

    { "label-comments", E_CONTACT_NOTE },
    { "text-comments", E_CONTACT_NOTE, TRUE },

    { "entry-fullname", E_CONTACT_FULL_NAME, TRUE },

    { "button-categories", E_CONTACT_CATEGORIES, TRUE },
    { "entry-categories", E_CONTACT_CATEGORIES, TRUE },

    { "label-caluri", E_CONTACT_CALENDAR_URI },
    { "entry-caluri", E_CONTACT_CALENDAR_URI, TRUE },

    { "label-fburl", E_CONTACT_FREEBUSY_URL },
    { "entry-fburl", E_CONTACT_FREEBUSY_URL, TRUE },

    { "label-videourl", E_CONTACT_VIDEO_URL },
    { "entry-videourl", E_CONTACT_VIDEO_URL, TRUE }
};
static int num_widget_field_mappings = sizeof(widget_field_mappings) / sizeof (widget_field_mappings[0]);

static void
enable_writable_fields(EContactEditor *editor)
{
    EList *fields = editor->writable_fields;
    EIterator *iter;
    GHashTable *dropdown_hash, *supported_hash;
    int i;
    char *widget_name;

    if (!fields)
        return;

    dropdown_hash = g_hash_table_new (g_str_hash, g_str_equal);
    supported_hash = g_hash_table_new (g_str_hash, g_str_equal);

    /* build our hashtable of the drop down menu items */
    e_contact_editor_build_phone_ui (editor);
    for (i = 0; i < G_N_ELEMENTS (phones); i ++)
        g_hash_table_insert (dropdown_hash,
                     (char*)e_contact_field_name(phones[i]),
                     editor->phone_info[i].widget);
    e_contact_editor_build_email_ui (editor);
    for (i = 0; i < G_N_ELEMENTS (emails); i ++)
        g_hash_table_insert (dropdown_hash,
                     (char*)e_contact_field_name(phones[i]),
                     editor->email_info[i].widget);
    e_contact_editor_build_address_ui (editor);
    for (i = E_CONTACT_FIRST_ADDRESS_ID; i <= E_CONTACT_LAST_ADDRESS_ID; i ++)
        g_hash_table_insert (dropdown_hash,
                     (char*)e_contact_field_name (i),
                     editor->address_info[i - E_CONTACT_FIRST_ADDRESS_ID].widget);

    /* then disable them all */
    g_hash_table_foreach (dropdown_hash, (GHFunc)disable_widget_foreach, NULL);

    /* disable the label widgets for the dropdowns (4 phone, 1
           email and the toggle button, and 1 address and one for
           the full address button */
    for (i = 0; i < 4; i ++) {
        widget_name = g_strdup_printf ("label-phone%d", i+1);
        enable_widget (glade_xml_get_widget (editor->gui, widget_name), FALSE);
        g_free (widget_name);
        widget_name = g_strdup_printf ("entry-phone%d", i+1);
        enable_widget (glade_xml_get_widget (editor->gui, widget_name), FALSE);
        g_free (widget_name);
    }
    enable_widget (glade_xml_get_widget (editor->gui, "label-email1"), FALSE);
    enable_widget (glade_xml_get_widget (editor->gui, "entry-email1"), FALSE);
    enable_widget (glade_xml_get_widget (editor->gui, "checkbutton-htmlmail"), FALSE);
    enable_widget (glade_xml_get_widget (editor->gui, "checkbutton-mailingaddress"), FALSE);
    enable_widget (glade_xml_get_widget (editor->gui, "label-address"), FALSE);
    enable_widget (glade_xml_get_widget (editor->gui, "text-address"), FALSE);

    editor->fullname_editable = FALSE;

    /* enable widgets that map directly from a field to a widget (the drop down items) */
    iter = e_list_get_iterator (fields);
    for (; e_iterator_is_valid (iter); e_iterator_next (iter)) {
        char *field = (char*)e_iterator_get (iter);
        GtkWidget *widget = g_hash_table_lookup (dropdown_hash, field);
        int i;

        if (widget) {
            enable_widget (widget, TRUE);
        }
        else {
            /* if it's not a field that's handled by the
                           dropdown items, add it to the has to be
                           used in the second step */
            g_hash_table_insert (supported_hash, field, field);
        }

        for (i = E_CONTACT_FIRST_ADDRESS_ID; i <= E_CONTACT_LAST_ADDRESS_ID; i ++) {
            if (!strcmp (field, e_contact_field_name (i))) {
                editor->address_editable [i - E_CONTACT_FIRST_ADDRESS_ID] = TRUE;
            }
        }

        /* ugh - this is needed to make sure we don't have a
                   disabled label next to a drop down when the item in
                   the menu (the one reflected in the label) is
                   enabled. */
        if (!strcmp (field, e_contact_field_name (editor->email_choice))) {
            enable_widget (glade_xml_get_widget (editor->gui, "label-email1"), TRUE);
            enable_widget (glade_xml_get_widget (editor->gui, "entry-email1"), editor->target_editable);
            enable_widget (glade_xml_get_widget (editor->gui, "checkbutton-htmlmail"), editor->target_editable);
        }
        else if (!strcmp (field, e_contact_field_name (editor->address_choice))) {
            enable_widget (glade_xml_get_widget (editor->gui, "label-address"), TRUE);
            enable_widget (glade_xml_get_widget (editor->gui, "checkbutton-mailingaddress"), editor->target_editable);
            enable_widget (glade_xml_get_widget (editor->gui, "text-address"), editor->target_editable);
        }
        else for (i = 0; i < 4; i ++) {
            if (!strcmp (field, e_contact_field_name (editor->phone_choice[i]))) {
                widget_name = g_strdup_printf ("label-phone%d", i+1);
                enable_widget (glade_xml_get_widget (editor->gui, widget_name), TRUE);
                g_free (widget_name);
                widget_name = g_strdup_printf ("entry-phone%d", i+1);
                enable_widget (glade_xml_get_widget (editor->gui, widget_name), editor->target_editable);
                g_free (widget_name);
            }
        }
    }

    /* handle the label next to the dropdown widgets */

    for (i = 0; i < num_widget_field_mappings; i ++) {
        gboolean enabled;
        GtkWidget *w;
        const char *field;

        w = glade_xml_get_widget(editor->gui, widget_field_mappings[i].widget_name);
        if (!w) {
            g_warning (_("Could not find widget for a field: `%s'"),
                   widget_field_mappings[i].widget_name);
            continue;
        }
        field = e_contact_field_name (widget_field_mappings[i].field_id);

        enabled = (g_hash_table_lookup (supported_hash, field) != NULL);

        if (widget_field_mappings[i].desensitize_for_read_only && !editor->target_editable) {
            enabled = FALSE;
        }

        enable_widget (w, enabled);
    }

    editor->fullname_editable = (g_hash_table_lookup (supported_hash, "full_name") != NULL);

    g_hash_table_destroy (dropdown_hash);
    g_hash_table_destroy (supported_hash);
}

static void
set_editable (EContactEditor *editor)
{
    int i;
    char *entry;
    /* set the sensitivity of all the non-dropdown entry/texts/dateedits */
    for (i = 0; i < num_widget_field_mappings; i ++) {
        if (widget_field_mappings[i].desensitize_for_read_only) {
            GtkWidget *widget = glade_xml_get_widget(editor->gui, widget_field_mappings[i].widget_name);
            enable_widget (widget, editor->target_editable);
        }
    }

    /* handle the phone dropdown entries */
    for (i = 0; i < 4; i ++) {
        entry = g_strdup_printf ("entry-phone%d", i+1);

        enable_widget (glade_xml_get_widget(editor->gui, entry),
                   editor->target_editable);

        g_free (entry);
    }

    /* handle the email dropdown entry */
    entry = "entry-email1";
    enable_widget (glade_xml_get_widget(editor->gui, entry),
               editor->target_editable);
    enable_widget (glade_xml_get_widget(editor->gui, "checkbutton-htmlmail"),
               editor->target_editable);

    /* handle the address dropdown entry */
    entry = "text-address";
    enable_widget (glade_xml_get_widget(editor->gui, entry),
               editor->target_editable);

    entry = "image-chooser";
    enable_widget (glade_xml_get_widget(editor->gui, entry),
               editor->target_editable);
}

static void
fill_in_info(EContactEditor *editor)
{
    EContact *contact = editor->contact;
    if (contact) {
        char *file_as;
        EContactName *name;
        EContactDate *anniversary;
        EContactDate *bday;
        EContactPhoto *photo;
        int i;
        GtkWidget *widget;
        gboolean wants_html;

        g_object_get (contact,
                  "file_as",          &file_as,
                  "name",             &name,
                  "anniversary",      &anniversary,
                  "birth_date",       &bday,
                  "wants_html",       &wants_html,
                  "photo",            &photo,
                  NULL);

        for (i = 0; i < sizeof(field_mapping) / sizeof(field_mapping[0]); i++) {
            char *string = e_contact_get (contact, field_mapping[i].field);
            fill_in_field(editor, field_mapping[i].id, string);
            g_free (string);
        }

#if 0
        find_address_mailing (editor);
#endif
        
        widget = glade_xml_get_widget(editor->gui, "checkbutton-htmlmail");
        if (widget && GTK_IS_CHECK_BUTTON(widget)) {
            g_object_set (widget,
                      "active", wants_html,
                      NULL);
        }

        /* File as has to come after company and name or else it'll get messed up when setting them. */
        fill_in_field(editor, "entry-file-as", file_as);

        g_free (file_as);
        if (editor->name)
            e_contact_name_free(editor->name);
        editor->name = name;

        widget = glade_xml_get_widget(editor->gui, "dateedit-anniversary");
        if (widget && E_IS_DATE_EDIT(widget)) {
            EDateEdit *dateedit;
            dateedit = E_DATE_EDIT(widget);
            if (anniversary)
                e_date_edit_set_date (dateedit,
                              anniversary->year,
                              anniversary->month,
                              anniversary->day);
            else
                e_date_edit_set_time (dateedit, -1);
        }

        widget = glade_xml_get_widget(editor->gui, "dateedit-birthday");
        if (widget && E_IS_DATE_EDIT(widget)) {
            EDateEdit *dateedit;
            dateedit = E_DATE_EDIT(widget);
            if (bday)
                e_date_edit_set_date (dateedit,
                              bday->year,
                              bday->month,
                              bday->day);
            else
                e_date_edit_set_time (dateedit, -1);
        }

        if (photo) {
            widget = glade_xml_get_widget(editor->gui, "image-chooser");
            if (widget && E_IS_IMAGE_CHOOSER(widget))
                e_image_chooser_set_image_data (E_IMAGE_CHOOSER (widget), photo->data, photo->length);
        }

        e_contact_date_free (anniversary);
        e_contact_date_free (bday);
        e_contact_photo_free (photo);

        set_fields(editor);

        set_im_fields(editor);
    }
}

static void
extract_field(EContactEditor *editor, EContact *contact, char *editable_id, EContactField field)
{
    GtkWidget *widget = glade_xml_get_widget(editor->gui, editable_id);
    char *string = NULL;

    if (!widget)
        return;

    if (E_IS_URL_ENTRY (widget))
        widget = e_url_entry_get_entry (E_URL_ENTRY (widget));

    if (GTK_IS_EDITABLE (widget))
        string = gtk_editable_get_chars(GTK_EDITABLE (widget), 0, -1);
    else if (GTK_IS_TEXT_VIEW (widget)) {
        GtkTextIter start, end;
        GtkTextBuffer *buffer;

        buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (widget));
        gtk_text_buffer_get_start_iter (buffer, &start);
        gtk_text_buffer_get_end_iter (buffer, &end);

        string = gtk_text_buffer_get_text (buffer, &start, &end, TRUE);
    }

    if (string && *string)
        e_contact_set (contact, field, string);
    else
        e_contact_set (contact, field, NULL);

    if (string) g_free(string);
}

static void
extract_info(EContactEditor *editor)
{
    EContact *contact = editor->contact;
    if (contact) {
        EContactDate anniversary;
        EContactDate bday;
        int i;
        GtkWidget *widget;

        widget = glade_xml_get_widget(editor->gui, "entry-file-as");
        if (widget && GTK_IS_EDITABLE(widget)) {
            GtkEditable *editable = GTK_EDITABLE(widget);
            char *string = gtk_editable_get_chars(editable, 0, -1);

            if (string && *string)
                e_contact_set (contact, E_CONTACT_FILE_AS, string);

            g_free(string);
        }

        for (i = 0; i < sizeof(field_mapping) / sizeof(field_mapping[0]); i++) {
            extract_field(editor, contact, field_mapping[i].id, field_mapping[i].field);
        }

        if (editor->name)
            e_contact_set (contact, E_CONTACT_NAME, editor->name);

        widget = glade_xml_get_widget(editor->gui, "dateedit-anniversary");
        if (widget && E_IS_DATE_EDIT(widget)) {
            if (e_date_edit_get_date (E_DATE_EDIT (widget),
                          &anniversary.year,
                          &anniversary.month,
                          &anniversary.day)) {
                /* g_print ("%d %d %d\n", anniversary.year, anniversary.month, anniversary.day); */
                e_contact_set (contact, E_CONTACT_ANNIVERSARY, &anniversary);
            } else
                e_contact_set (contact, E_CONTACT_ANNIVERSARY, NULL);
        }

        widget = glade_xml_get_widget(editor->gui, "dateedit-birthday");
        if (widget && E_IS_DATE_EDIT(widget)) {
            if (e_date_edit_get_date (E_DATE_EDIT (widget),
                          &bday.year,
                          &bday.month,
                          &bday.day)) {
                /* g_print ("%d %d %d\n", bday.year, bday.month, bday.day); */
                e_contact_set (contact, E_CONTACT_BIRTH_DATE, &bday);
            } else
                e_contact_set (contact, E_CONTACT_BIRTH_DATE, NULL);
        }

        widget = glade_xml_get_widget (editor->gui, "image-chooser");
        if (widget && E_IS_IMAGE_CHOOSER (widget)) {
            char *image_data;
            gsize image_data_len;

            if (editor->image_set
                && e_image_chooser_get_image_data (E_IMAGE_CHOOSER (widget),
                                   &image_data,
                                   &image_data_len)) {
                EContactPhoto photo;

                photo.data = image_data;
                photo.length = image_data_len;

                e_contact_set (contact, E_CONTACT_PHOTO, &photo);
                g_free (image_data);
            }
            else {
                e_contact_set (contact, E_CONTACT_PHOTO, NULL);
            }
        }
    }
}

/**
 * e_contact_editor_raise:
 * @config: The %EContactEditor object.
 *
 * Raises the dialog associated with this %EContactEditor object.
 */
void
e_contact_editor_raise (EContactEditor *editor)
{
    /* FIXME: perhaps we should raise at realize time */
    if (GTK_WIDGET (editor->app)->window)
        gdk_window_raise (GTK_WIDGET (editor->app)->window);
}

/**
 * e_contact_editor_show:
 * @ce: The %EContactEditor object.
 *
 * Shows the dialog associated with this %EContactEditor object.
 */
void
e_contact_editor_show (EContactEditor *ce)
{
    gtk_widget_show (ce->app);
}

GtkWidget *
e_contact_editor_create_date(gchar *name,
                 gchar *string1, gchar *string2,
                 gint int1, gint int2);

GtkWidget *
e_contact_editor_create_date(gchar *name,
                 gchar *string1, gchar *string2,
                 gint int1, gint int2)
{
    GtkWidget *widget = e_date_edit_new ();
    e_date_edit_set_allow_no_date_set (E_DATE_EDIT (widget),
                       TRUE);
    e_date_edit_set_show_time (E_DATE_EDIT (widget), FALSE);
    e_date_edit_set_time (E_DATE_EDIT (widget), -1);
    gtk_widget_show (widget);
    return widget;
}

GtkWidget *
e_contact_editor_create_web(gchar *name,
                gchar *string1, gchar *string2,
                gint int1, gint int2);

GtkWidget *
e_contact_editor_create_web(gchar *name,
                gchar *string1, gchar *string2,
                gint int1, gint int2)
{
    GtkWidget *widget = e_url_entry_new ();
    gtk_widget_show (widget);
    return widget;
}

GtkWidget *
e_contact_editor_create_source_option_menu (gchar *name,
                        gchar *string1, gchar *string2,
                        gint int1, gint int2);

GtkWidget *
e_contact_editor_create_source_option_menu (gchar *name,
                        gchar *string1, gchar *string2,
                        gint int1, gint int2)
{
    GtkWidget   *menu;
    GConfClient *gconf_client;
    ESourceList *source_list;

    gconf_client = gconf_client_get_default ();
    source_list = e_source_list_new_for_gconf (gconf_client, "/apps/evolution/addressbook/sources");

    menu = e_source_option_menu_new (source_list);
    g_object_unref (source_list);

    gtk_widget_show (menu);
    return menu;
}

static void
enable_widget (GtkWidget *widget, gboolean enabled)
{
    if (GTK_IS_ENTRY (widget)) {
        gtk_editable_set_editable (GTK_EDITABLE (widget), enabled);
    }
    else if (GTK_IS_TEXT_VIEW (widget)) {
        gtk_text_view_set_editable (GTK_TEXT_VIEW (widget), enabled);
    }
    else if (GTK_IS_COMBO (widget)) {
        gtk_editable_set_editable (GTK_EDITABLE (GTK_COMBO (widget)->entry),
                       enabled);
        gtk_widget_set_sensitive (GTK_COMBO (widget)->button, enabled);
    }
    else if (E_IS_URL_ENTRY (widget)) {
        GtkWidget *e = e_url_entry_get_entry (E_URL_ENTRY (widget));
        gtk_editable_set_editable (GTK_EDITABLE (e), enabled);
    }
    else if (E_IS_DATE_EDIT (widget)) {
        e_date_edit_set_editable (E_DATE_EDIT (widget), enabled);
    }
    else if (E_IS_IMAGE_CHOOSER (widget)) {
        e_image_chooser_set_editable (E_IMAGE_CHOOSER (widget), enabled);
    }
    else
        gtk_widget_set_sensitive (widget, enabled);
}


gboolean
e_contact_editor_request_close_all (void)
{
    GSList *p;
    GSList *pnext;
    gboolean retval;

    retval = TRUE;
    for (p = all_contact_editors; p != NULL; p = pnext) {
        pnext = p->next;

        e_contact_editor_raise (E_CONTACT_EDITOR (p->data));
        if (! prompt_to_save_changes (E_CONTACT_EDITOR (p->data))) {
            retval = FALSE;
            break;
        }

        close_dialog (E_CONTACT_EDITOR (p->data));
    }

    return retval;
}