aboutsummaryrefslogblamecommitdiffstats
path: root/mail/mail-callbacks.c
blob: f4b2ecca12204a511c793f5bdf3a091e15d49734 (plain) (tree)
1
2
3
4
5
6
7
8
9



                                                                           
            


                                      
  
                                               

                                                                 

                                                                    











                                                                      
 
                    
                   

      
                 
                  
 
                            
 


                               

                                 
                                          
                                               
                                                    
 

                                 

                                    
                                  
                                 
 
                 
                            
                           
                        

                              
                    


                       
                        
                           
                         
                              
                           
                             

                                 
 




                                   
              
 

                                                                                             
 



                                                                                                      
                                                                              




                                       








                                                                                

                                                                                              

                                                                                                 
         

                                                     

                                                                                   

                                  



                                          
                                

                               


                                   
                            
                           
                  

  













































                                                        
                                                           



                              
 
           
                                                        



                         
               
                                  
 
                               





                                                                      
                                                 
                                                                                            

                                                      
                            
         

                                            




                                            
                          
        









                                                                                                       
        

                                                     
        

                                   

                                                                                                



                                   
                                       


                                                                      

                             
        
                    

 
               

                                                                                      
                                 
                           
                     
              



                                                                                               
                            
        
                                       

                                                                                                        
                                                 
                                                                        
                                         

                                                                         
                        
                                                                        

                 
        
                                                 




                                                                                                      
                   


               

                                                      
                                 


                                            
        
                                                                                              
                            




                                                                                                      
        
                   

 
               
                                                                            
 
                                 
                               
                           
        


                                                                                          
                            
        




                                                                               
        






                                                                                  
        





                                                                                                 
        
                   

 
 
                   
                                            
                               
                      


           

                                                                                            
 
                                            
                                       
        

                        
                     
                                                









                                                                                                    
                                         





                                                                                                     
                 
                
                                                                 
                
                                                                      


                                                  
                
                           
                                         

                                                                     



                                                                                                    

                                                                                                 



                                                                                                                  

                 
                                                                          
                                                              
         
        

                                       


                                      
                      

 
                         
                                                                                            
 

                                                    
                                         
                                  

                                  
                            
                           
                          


                                            
        


                                                                                      
        
                                        
                                                              
        

                                            

                                                     
                                                         
                                                                                     

                                              









                                                                                                    


                                 
         
        
                                                           
                             
                                                             
                                                                                         
                        






                                                                                        
                 
                

                                                     
        

                                 
                                                                
                                

                                                                                           
                              

         
                                                        

                                                                                
                                      
         
        


                                                                                                         


                                                                                       
                                                                                   
                                              

                                 
                                                                                  


                                                                                      


                                   
                                                                                                  
                                         
                                              

                 
        
                                  
                                                        
                                                    
                                                              
                                      
         
        
                                                                      
                                                                               












                                                                                                                    
        
                                                                                    

                                                  
        
          
        

                                                 
        


                       







                                                            
                                          

 
    
                                                             
 
                                          
                                  
                               
                                



                              
        
                                                                                    
                          
                            
                
                                                                                                    




                                     




                                          

                     
                                                               

                       
        





                                                                               

                                         
        
                                         
                              

                                    
                           
                                  
                                
                                                
        
                                                             
        
                                                                                
                                     
                                    

 
                         
                                            




                               
                                                                                            
                                                           
 

                                                 
                             
        

                          


                                                                                            
        
                                       
                                 

                                                             

                                                                                                                

                                                                                         



                                                                                                         
         
        
                                 
                                                       








                                                                                            
                          





                                                                                             
         
        
                           


                                                          
         
        
                      
                                                                
        
      
                                       

                                     









                                                
                         
                                        
                                                 
         








                                                                 
                                          



         
                                                                             


                                               

                                     


                               









                                                                                                                  


                                                                                                 
                                       


                                                         
                 
                
                                       
                                          
         







                                                               
                                
                             

                                   


                                                                   

                                    

 
                  
                                                                       
 
                               
                           
                           
        


                                                                              
                            
                                                             
        

                                                                                                   
        





                                                             
        
                       
                                                                                                           
                                                                   

                                                           


                                             




                                                   
                                                       
                                            
                            
                          
        
                                                                               
                       
        


                                                                        
                                                              

                       
        
                         
        



                                                                                            
        




                                                
                                                     
 
                                            
                            
                                 
        


                                                           
                       
        




                                                                             

                       
        
                         
        



                                                                                            
        


                                   
              
                                                                                           
                                                         
 
                                
                          
              

                                                                             








                                                                               
                        


                                                             
                        

                                                                                            
                 




                    
                 
                                                                                                   
 
                                 
                         

              



                            
                 



                                                                                   



                 



                                                                                   


                 

        


                       

                                                                                                               
 
                                
                                 
                        

                                                                          






                                                                                                    


                                                             

                                           
                                       







                                                                                        
                                                                                        







                                                                                                                    
                                       

         

                              

                                                  


                                            

 
                  
                                                                           






                                                                              


                                   






                                                                      

                             
                     
                                                                                               
 
                                                                            
                                                                 
                                                           
                                                 

                                            
                                        
                                      
                                     
                                  
                               
                            
                           
                        
                    

                  

                                            








                                                                                                   
        


                            
                                                                         
        
                                
                                               
                                                                          






                                                                                                    


                                                             

                                           
                                       







                                                                                        
                                                                                        







                                                                                                                    
                                       

         

                              

                                                                                        

                                                 
        


                                                                   
                                                                                   





                                                                      
                      
                                 
                                   
                           
                
                                                                                         

                                                                                      


                                                                                  
                                                         
                                      





                                                                                          
                                                                 
                                              
                         





                                                                
                        




                                                                              
                                                            
                                                                      
                                
                                                              
                                



                                                                               
                         
                 
                
                                                                 
                                                              
                
                                      
                                   
                
                                                                               
                


                                                                         
                
                               

                              


                                                                                                                     

                                                                                                          
                                                             

                                                                            
                                                                                
                                                                      

                                                                                                               
                                 
                         

                 
                                        







                                                                                                         
                        
                                                                         
                 

                                                 

         

                                            





                                                                            
                                                                            
                                                                                                






                                                                   
                                          



                                                                       





                                                                                          
                
                                                                   
                                                                                                                 

                                                                                                 



                                                                      
                      
         
        
                                                 
                                                                    


                                        
                                                      
                                                                       
                      
                                                                 
                                                                              




                                                                                                   

                                                                                                        
                                                        

                                                                             

                                              
                 
         
        

                                                
        

                         
        
                                                                                             
        

                                  
        


                                             

                                                                                    
                         

                                 
                                                                                

                               
                                                                                       




                                                                               


                                                                               
        
                                                   



                        
           
                                                                                            
 
                                          
        

                                                    

 
    
                                                                                  
 
                                            
                               
        
                                                    
                                       
        

                                                                  
                                                                           

                       
        
                                                                

                       
        
                         








                                                                 



                                                                                            
        
                                                        
                                                





                                                       
        
                                                                               
                       
        
                                                                                  






                                                       
                                                                               
                       
        
                                                                                





                                                       
        
                                                                               
                       
        
                                                                              

 
    

                                                               

                                      


                                                             
 
                     
                                                                     
 
                                            
                                 
                               
        

                                                                
                                       




                                                                                                
                                                                                











                                                                                 
                                         
                       
                                 
                



                                                                                                    
                
                                                                                                
                
                                                        
         
        



                        
                                                                                               
 
                                                              
                                  
                             
              
        
                               
                       
        




















                                                                                                          
                 

                                 
         

 

                                                                 
 
                        
        
                                             
                       
        

                                           
        


                                                                     
                                                                                               





                                                                




                                                      





                                                                                                            

                                         
                   
                                                                                 



                                                                
                                                                   

                 


    
                                                        
 
                                                        
                        
        
                                                                               
                       


                                                                     
                                                                   
                                                                                          

 


                                               




                                                                                                
        





                                                     

    
                             
 
                                            
                            








                                                                      
        
                                                             


                       
                                                                                                      
        
                         
        



                                                                                            




                                   


                                                       










                                                                               










                                                                                


                                                 
                                                        
                                            

                                      

                               
                                                                     




                                                                                        
                                                                        










                                                                         





                                                                                    

                                                                        
                                 
                



                                                                                                    







                                                        
                                                                                         















                                                                                          
                                                           













                                                        

                                                                   

 
           


                                           

                              


                                                      


                                                                                                                 




                                                                                      
                                                                                           
                                                                                            

                                                                              
                                                                                        


                                                          
                            


           
                                                             
 
                                                                          
                                                         

                                       
                        
                   


                                             
        
                             
                                              
        



                                               
        
                                                                       
                                                                                                          


                                                                                 
                       
        


                                                           
         
        
                                  
                                                                     

                                 
                                  
                                                                             

                                                               

                                                                             
                                                                            
         
        
                            


    
                                                   
 
                                       


    

                                                                    
                                       



                                                   
 
                                        

 


                                                                    
                                        

 




                                     
        
                                                          











                                                                         

                    

 

                                            

                                          



                                                       
        


                                                            
        


                      



                                                       

                               


                           

                        
        

                                             


                                                                     

                           


                                                                                
                          
        

                                                             
        


                                                                                               
                                                                       
                                          
        


                                                                                                    
                                                       

                                                                                      
        

                                                         

     


                                        

 
    





                                                                                  


                                                       

                        


                                             
                                  


                                                                     


    
                                                                      
 
                                                       


                                             
                       
        






                                                                           












                                                                       
        









                                                                       
        




                                                                            
        



                                                               
                




                                                                          
        






                                                                         


                                             

                                             
                       
        



                                                                    

                                  
                                       




                                                                                                       



                                                                              


    
                                                                            

                                                       


                                             
                       
        




                                                                           

 

                                                       
                                                            
 
                        
              
        
                                             
                         
        
                                                           
                                  

                                                                     
                                         
                                                                                       


                                        
                                       
        
                                      
        
                 

 


                                              


                        
                                             



                                                           

                                                                     
                                         
                              
                
                                                                                       
                



                                                                                   
                 
                



                                                                                      


                                                    
                
                                                                                         
                

                                        
                                       

                                      
        


                 


                                                                        
                                                                                           


    
                                                                          
 
                                                       



                                             

                                                           
                                              

                                
        
                                                  




                                                                            
                                                       

                        

                                             
                       


                                                  
                                       
                                                                                                    
                                                  
                                       
 
 


                                                                             
                                                                                                                       


    





                                                                               




                                                                               







                                 
                                                                             
 


                            
              







                                                                         
        






                                                                                                              
         
 
                               
 
                                  



                                            
    

                                                                             



                                                       
              














                                                                     







                                                                                        

                                                                                    
        

                             
                                       
                





                                                                                                               

         
                                 


    
                                                                                   
 

                                                       
                  







                                                                     
                                                  


                                         
                                
                

                                                                                                  

                                 
                                                                                                    


                                       

                     
                                      


    
                                                                               
 











                                                                     


                                                                                                   



                                       


    

                                                                   





                                                       




                                                                    





                                                       




                                                                      





                                                       

 
           
                                                                                        
 
                                                   
              
        


                             
                                             
                                                    
                                       
                

                                                                                           
                                                                                

                                                           
                
                               
                                         



                                                                            
                         
                        



                                                                                                            
                        



                                                                

               

                                                                  
                          
                      



                                                                                                         


                                                       
                                        



                                                
        
                                         

 










                                                                     
                                                                                                            
                       
        


                                                                   
    


                                                       
        


                                             
                                             

                                                                                       

                       
        
                               

 
           

                                                                                          

              



                                                                              
        

                                                                                   

 
 



                                                       

                        


                                             
                                           

                                                                                  








                                                                     
                                                                                                              
                       
        


                                                                     
 




                                                       



                                             
                                                        

                                  


                                                                                               
                                                                                                       
                                         

                       
        



                                               



                                                       



                                             


                                                    
           



                                                   
                         
                       
                            
        
                                                                                

                            
        


                                                            









                                                                                         

         
                  
                                                                             
                                                                          
                                                                    
                                                            
         






















                                                       
                           
        


                                             








                                                                      


                                                          






                                                                                            



                                               
    





                                                  

                                                  
                                                       

                              
                         
        

                                             
        

                                            

                                                                                
        
                                                                         
                           

                                                                                      
                

                                                                                                                 

                                                                   
                                                                                           
                                                                                            
                                                                              
                    
                                                                                        
                                                          
         


    

                                                    
                                                                             

 



                                                       
        


                                             
                                                                                      


    
                                                       

                                                       
        


                                             
                                                                                                      


    
                                                        

                                                       
        


                                             
                                                                        
                                                                                  


    










                                                           


                                                       
        


                                             
                                                                            
                                          


    
                                                           

                                                       
        


                                             
                                                                            
                                                          

 



                                                            
        


                                             
                                                                            
                                                                                 

 

                                            
 
                                 
        


                                             
                             
                                                                       





                                                                      

 
               
                                   
 
                                 




                                                                                         
                            







                                                                                                         
                   

 
    
                                                                          
 

                                                       


                                             
                                                                                                           
                                       





                                                                                                   
                                                                                          



                                                                                
                                                                                
                

                                                                   

                                                                                          
                        
                                                                         
                                                                                              
                 
                
                                           
                                                                      
         

 




                                                                 
                                                                         

                          
        
                                            
                           
                
                                                                    



                                                                         



                                   

 





                                            
    
                                                                       



                                                       

                            
                                                                      

                       
        
                                   
                                                                 
                                                          
                                                            
                      
        
                                         


                                                                    

                       
        
                                                                                 

                                                                                   
                                                                        

                                                                                                             
                                                     

 

                                                               
    
                                                                                 
 
                        

 
 
              


                                                                                     
                                                                        

                                            

                                                               
                                                 
                                                              

                                               
 



                              
 



                                                                                     
                                                                    
        
                               
                                                                                               

                                                                                      

                                               
                                                        

                                                                    

                                                                                                                        
                                                              
                                                        
                
                              
                                 
         

 







                                                    



                                                                     
        
                                             
                                                                       

                             
                                                                                                                 
        

                                                                          
        


                    
           
                                                   
 
                      
                            
                                         
                                    
                                        
                          
                           
                                 
        
                       
                                                                                                                    
                                                                                            
                                                                                                                              
                
                                                  
                                                       
                              
                                                         

                                       
                        
                                                                  

                               
                

                                                                                     
         
        
                     
                                                            
                                                  
              
                                                          
        

                                                               
                                                                   
        
                                          
                                                                 




                                                                      
        




                                                                                 
                                                       
         
                                                           
                                                       
        

                                                                                                        
                                
        
                                           
        
                                             


                                               
        
                     
                              
                

                                                                                    
                
                                                                  
                
                                 
                                                                                                      
         

                                      

 










                                                                                       
                                                                                               



                                              
                               

                       
        



                                                                                           
        


                                         
        

                                            





                                                             
                                                                              


                                                                                    
                

                                         
                
                                   
                








                                                                                                                              
                                                                                              


                                               






                                            
 


                                                 



                                                       
        
                                            


    

                                                         



                                                       
        
                                           

 

                                                                            
                                          

           
                                                               
 



                                                  

 
    
                                                                                
 
                                
                                                           


                                                                                             
                
                                                         
                
                                                                                    
         

 

                                                                          





                                                                      
                                   






                                                                 
                           

 
    
                                                                            
 




                                                       




                                                                               
                                                      

                                                                                         
         

 
           
                                                                                             
 
                                                  
        



                                             

                              
                                                                                                     
                                                                   
                                     


         


                                                
                                                       
                        
              
        
                                             
                       
        

                                                                     


                                                                                                                                
        
                                            


                                                                                                        
         
                                      


    
                                                
 
                                                       
                                          
        


                                             
                                                                         
                                       







                                                                        


    
                                                                        


                                   
 
    
                                                                        
 
                                      

 


                                                         
                                    




                                                                       
                                
                               

                            
                          

                          
        

                                                           
                                        
                       

                                   


                                               


                                                             

                                                      
                                                               
                                                                                                   
                                       


                                                                                  
                                                                                                        
                                        
                                                     
                                                                                                            
                                         
                                 
                         


                                                                    
                 

                                       

         

                              
                                              
                                                           
                   
                                                                            
        
                                    
 
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/* mail-ops.c: callbacks for the mail toolbar/menus */

/* 
 * Authors: 
 *  Dan Winship <danw@ximian.com>
 *  Peter Williams <peterw@ximian.com>
 *  Jeffrey Stedfast <fejj@ximian.com>
 *
 * Copyright 2000 Ximian, Inc. (www.ximian.com)
 *
 * This program is free software; you can redistribute it and/or 
 * modify it under the terms of version 2 of the GNU General Public 
 * License as published by the Free Software Foundation.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
 * USA
 */


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

#include <time.h>
#include <errno.h>

#include <gtkhtml/gtkhtml.h>

#include <gconf/gconf.h>
#include <gconf/gconf-client.h>

#include <gtk/gtkmessagedialog.h>

#include <libgnomeprint/gnome-print-job.h>
#include <libgnomeprintui/gnome-print-dialog.h>
#include <libgnomeprintui/gnome-print-job-preview.h>

#include <bonobo/bonobo-widget.h>
#include <bonobo/bonobo-socket.h>
#include <gal/e-table/e-table.h>
#include <gal/widgets/e-gui-utils.h>
#include <e-util/e-dialog-utils.h>
#include <filter/filter-editor.h>

#include "mail.h"
#include "message-browser.h"
#include "mail-callbacks.h"
#include "mail-config.h"
#include "mail-accounts.h"
#include "mail-config-druid.h"
#include "mail-mt.h"
#include "mail-tools.h"
#include "mail-ops.h"
#include "mail-local.h"
#include "mail-search.h"
#include "mail-send-recv.h"
#include "mail-vfolder.h"
#include "mail-folder-cache.h"
#include "folder-browser.h"
#include "subscribe-dialog.h"
#include "message-tag-editor.h"
#include "message-tag-followup.h"

#include "Evolution.h"
#include "evolution-storage.h"

#include "evolution-shell-client.h"

#define d(x) x

#define FB_WINDOW(fb) GTK_WINDOW (gtk_widget_get_ancestor (GTK_WIDGET (fb), GTK_TYPE_WINDOW))


/* default is default gtk response
   if again is != NULL, a checkbox "dont show this again" will appear, and the result stored in *again
*/
static gboolean
e_question (GtkWindow *parent, int def, gboolean *again, const char *fmt, ...)
{
    GtkWidget *mbox, *check = NULL;
    va_list ap;
    int button;
    char *str;
    
    va_start (ap, fmt);
    str = g_strdup_vprintf (fmt, ap);
    va_end (ap);
    mbox = gtk_message_dialog_new (parent, GTK_DIALOG_DESTROY_WITH_PARENT,
                       GTK_MESSAGE_QUESTION, GTK_BUTTONS_YES_NO,
                       "%s", str);
    g_free (str);
    gtk_dialog_set_default_response ((GtkDialog *) mbox, def);
    if (again) {
        check = gtk_check_button_new_with_label (_("Don't show this message again."));
        gtk_box_pack_start ((GtkBox *)((GtkDialog *) mbox)->vbox, check, TRUE, TRUE, 10);
        gtk_widget_show (check);
    }
    
    button = gtk_dialog_run ((GtkDialog *) mbox);
    if (again)
        *again = !gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (check));
    gtk_widget_destroy (mbox);
    
    return button == GTK_RESPONSE_YES;
}


struct _composer_callback_data {
    unsigned int ref_count;
    
    CamelFolder *drafts_folder;
    char *drafts_uid;
    
    CamelFolder *folder;
    guint32 flags, set;
    char *uid;
};

static struct _composer_callback_data *
ccd_new (void)
{
    struct _composer_callback_data *ccd;
    
    ccd = g_new (struct _composer_callback_data, 1);
    ccd->ref_count = 1;
    ccd->drafts_folder = NULL;
    ccd->drafts_uid = NULL;
    ccd->folder = NULL;
    ccd->flags = 0;
    ccd->set = 0;
    ccd->uid = NULL;
    
    return ccd;
}

static void
free_ccd (struct _composer_callback_data *ccd)
{
    if (ccd->drafts_folder)
        camel_object_unref (ccd->drafts_folder);
    g_free (ccd->drafts_uid);
    
    if (ccd->folder)
        camel_object_unref (ccd->folder);
    g_free (ccd->uid);
    g_free (ccd);
}

static void
ccd_ref (struct _composer_callback_data *ccd)
{
    ccd->ref_count++;
}

static void
ccd_unref (struct _composer_callback_data *ccd)
{
    ccd->ref_count--;
    if (ccd->ref_count == 0)
        free_ccd (ccd);
}


static void
composer_destroy_cb (gpointer user_data, GObject *deadbeef)
{
    ccd_unref (user_data);
}


static void
druid_destroy_cb (gpointer user_data, GObject *deadbeef)
{
    gtk_main_quit ();
}

static gboolean
configure_mail (FolderBrowser *fb)
{
    MailConfigDruid *druid;
    
    if (e_question (FB_WINDOW (fb), GTK_RESPONSE_YES, NULL,
            _("You have not configured the mail client.\n"
              "You need to do this before you can send,\n"
              "receive or compose mail.\n"
              "Would you like to configure it now?"))) {
        druid = mail_config_druid_new ();
        g_object_weak_ref ((GObject *) druid, (GWeakNotify) druid_destroy_cb, NULL);
        gtk_widget_show ((GtkWidget *) druid);
        gtk_grab_add ((GtkWidget *) druid);
        gtk_main ();
    }
    
    return mail_config_is_configured ();
}

static gboolean
check_send_configuration (FolderBrowser *fb)
{
    EAccount *account;
    
    if (!mail_config_is_configured ()) {
        if (fb == NULL) {
            e_notice (NULL, GTK_MESSAGE_WARNING,
                  _("You need to configure an account\nbefore you can compose mail."));
            return FALSE;
        }
        
        if (!configure_mail (fb))
            return FALSE;
    }
    
    /* Get the default account */
    account = mail_config_get_default_account ();
    
    /* Check for an identity */
    if (!account) {
        e_notice (FB_WINDOW (fb), GTK_MESSAGE_WARNING,
              _("You need to configure an identity\nbefore you can compose mail."));
        return FALSE;
    }
    
    /* Check for a transport */
    if (!account->transport->url) {
        e_notice (FB_WINDOW (fb), GTK_MESSAGE_WARNING,
              _("You need to configure a mail transport\n"
                "before you can compose mail."));
        return FALSE;
    }
    
    return TRUE;
}

static gboolean
ask_confirm_for_unwanted_html_mail (EMsgComposer *composer, EDestination **recipients)
{
    gboolean show_again, res;
    GConfClient *gconf;
    GString *str;
    int i;
    
    gconf = gconf_client_get_default ();
    
    if (!gconf_client_get_bool (gconf, "/apps/evolution/mail/prompts/unwanted_html", NULL))
        return TRUE;
    
    /* FIXME: this wording sucks */
    str = g_string_new (_("You are sending an HTML-formatted message. Please make sure that\n"
                  "the following recipients are willing and able to receive HTML mail:\n"));
    for (i = 0; recipients[i] != NULL; ++i) {
        if (!e_destination_get_html_mail_pref (recipients[i])) {
            const char *name;
            
            name = e_destination_get_textrep (recipients[i]);
            
            g_string_append_printf (str, "     %s\n", name);
        }
    }
    
    g_string_append (str, _("Send anyway?"));
    res = e_question ((GtkWindow *) composer, GTK_RESPONSE_YES, &show_again, "%s", str->str);
    g_string_free (str, TRUE);
    
    gconf_client_set_bool (gconf, "/apps/evolution/mail/prompts/unwanted_html", show_again, NULL);
    
    return res;
}

static gboolean
ask_confirm_for_empty_subject (EMsgComposer *composer)
{
    gboolean show_again, res;
    GConfClient *gconf;
    
    gconf = gconf_client_get_default ();
    
    if (gconf_client_get_bool (gconf, "/apps/evolution/mail/prompts/empty_subject", NULL))
        return TRUE;
    
    res = e_question ((GtkWindow *) composer, GTK_RESPONSE_YES, &show_again,
              _("This message has no subject.\nReally send?"));
    
    gconf_client_set_bool (gconf, "/apps/evolution/mail/prompts/empty_subject", show_again, NULL);
    
    return res;
}

static gboolean
ask_confirm_for_only_bcc (EMsgComposer *composer, gboolean hidden_list_case)
{
    gboolean show_again, res;
    const char *first_text;
    GConfClient *gconf;
    
    gconf = gconf_client_get_default ();
    
    if (!gconf_client_get_bool (gconf, "/apps/evolution/mail/prompts/only_bcc", NULL))
        return TRUE;
    
    /* If the user is mailing a hidden contact list, it is possible for
       them to create a message with only Bcc recipients without really
       realizing it.  To try to avoid being totally confusing, I've changed
       this dialog to provide slightly different text in that case, to
       better explain what the hell is going on. */
    
    if (hidden_list_case) {
        first_text =  _("Since the contact list you are sending to "
                "is configured to hide the list's addresses, "
                "this message will contain only Bcc recipients.");
    } else {
        first_text = _("This message contains only Bcc recipients.");
    }
    
    res = e_question ((GtkWindow *) composer, GTK_RESPONSE_YES, &show_again,
              "%s\n%s", first_text,
              _("It is possible that the mail server may reveal the recipients "
                "by adding an Apparently-To header.\nSend anyway?"));
    
    gconf_client_set_bool (gconf, "/apps/evolution/mail/prompts/only_bcc", show_again, NULL);
    
    return res;
}


struct _send_data {
    struct _composer_callback_data *ccd;
    EMsgComposer *composer;
    gboolean send;
};

static void
composer_send_queued_cb (CamelFolder *folder, CamelMimeMessage *msg, CamelMessageInfo *info,
             int queued, const char *appended_uid, void *data)
{
    struct _composer_callback_data *ccd;
    struct _send_data *send = data;
    
    ccd = send->ccd;
    
    if (queued) {
        if (ccd && ccd->drafts_folder) {
            /* delete the old draft message */
            camel_folder_set_message_flags (ccd->drafts_folder, ccd->drafts_uid,
                            CAMEL_MESSAGE_DELETED | CAMEL_MESSAGE_SEEN,
                            CAMEL_MESSAGE_DELETED | CAMEL_MESSAGE_SEEN);
            camel_object_unref (ccd->drafts_folder);
            ccd->drafts_folder = NULL;
            g_free (ccd->drafts_uid);
            ccd->drafts_uid = NULL;
        }
        
        if (ccd && ccd->folder) {
            /* set any replied flags etc */
            camel_folder_set_message_flags (ccd->folder, ccd->uid, ccd->flags, ccd->set);
            camel_object_unref (ccd->folder);
            ccd->folder = NULL;
            g_free (ccd->uid);
            ccd->uid = NULL;
        }
        
        gtk_widget_destroy (GTK_WIDGET (send->composer));
        
        if (send->send && camel_session_is_online (session)) {
            /* queue a message send */
            mail_send ();
        }
    } else {
        if (!ccd) {
            ccd = ccd_new ();
            
            /* disconnect the previous signal handlers */
            g_signal_handlers_disconnect_matched(send->composer, G_SIGNAL_MATCH_FUNC, 0,
                                 0, NULL, composer_send_cb, NULL);
            g_signal_handlers_disconnect_matched(send->composer, G_SIGNAL_MATCH_FUNC, 0,
                                 0, NULL, composer_save_draft_cb, NULL);
            
            /* reconnect to the signals using a non-NULL ccd for the callback data */
            g_signal_connect (send->composer, "send", G_CALLBACK (composer_send_cb), ccd);
            g_signal_connect (send->composer, "save-draft", G_CALLBACK (composer_save_draft_cb), ccd);
            
            g_object_weak_ref ((GObject *) send->composer, (GWeakNotify) composer_destroy_cb, ccd);
        }
        
        e_msg_composer_set_enable_autosave (send->composer, TRUE);
        gtk_widget_show (GTK_WIDGET (send->composer));
    }
    
    camel_message_info_free (info);
    
    if (send->ccd)
        ccd_unref (send->ccd);
    
    g_free (send);
}

static CamelMimeMessage *
composer_get_message (EMsgComposer *composer, gboolean post, gboolean save_html_object_data)
{
    CamelMimeMessage *message = NULL;
    EDestination **recipients, **recipients_bcc;
    gboolean send_html, confirm_html;
    CamelInternetAddress *cia;
    int hidden = 0, shown = 0;
    int num = 0, num_bcc = 0;
    const char *subject;
    GConfClient *gconf;
    EAccount *account;
    int i;
    
    gconf = gconf_client_get_default ();
    
    /* We should do all of the validity checks based on the composer, and not on
       the created message, as extra interaction may occur when we get the message
       (e.g. to get a passphrase to sign a message) */
    
    /* get the message recipients */
    recipients = e_msg_composer_get_recipients (composer);
    
    cia = camel_internet_address_new ();
    
    /* see which ones are visible/present, etc */
    if (recipients) {
        for (i = 0; recipients[i] != NULL; i++) {
            const char *addr = e_destination_get_address (recipients[i]);
            
            if (addr && addr[0]) {
                camel_address_decode ((CamelAddress *) cia, addr);
                if (camel_address_length ((CamelAddress *) cia) > 0) {
                    camel_address_remove ((CamelAddress *) cia, -1);
                    num++;
                    if (e_destination_is_evolution_list (recipients[i])
                        && !e_destination_list_show_addresses (recipients[i])) {
                        hidden++;
                    } else {
                        shown++;
                    }
                }
            }
        }
    }
    
    recipients_bcc = e_msg_composer_get_bcc (composer);
    if (recipients_bcc) {
        for (i = 0; recipients_bcc[i] != NULL; i++) {
            const char *addr = e_destination_get_address (recipients_bcc[i]);
            
            if (addr && addr[0]) {
                camel_address_decode ((CamelAddress *) cia, addr);
                if (camel_address_length ((CamelAddress *) cia) > 0) {
                    camel_address_remove ((CamelAddress *) cia, -1);
                    num_bcc++;
                }
            }
        }
        
        e_destination_freev (recipients_bcc);
    }
    
    camel_object_unref (cia);
    
    /* I'm sensing a lack of love, er, I mean recipients. */
    if (num == 0 && !post) {
        e_notice ((GtkWindow *) composer, GTK_MESSAGE_WARNING,
              _("You must specify recipients in order to send this message."));
        goto finished;
    }
    
    if (num > 0 && (num == num_bcc || shown == 0)) {
        /* this means that the only recipients are Bcc's */     
        if (!ask_confirm_for_only_bcc (composer, shown == 0))
            goto finished;
    }
    
    send_html = gconf_client_get_bool (gconf, "/apps/evolution/mail/composer/send_html", NULL);
    confirm_html = gconf_client_get_bool (gconf, "/apps/evolution/mail/prompts/unwanted_html", NULL);
    
    /* Only show this warning if our default is to send html.  If it isn't, we've
       manually switched into html mode in the composer and (presumably) had a good
       reason for doing this. */
    if (e_msg_composer_get_send_html (composer) && send_html && confirm_html) {
        gboolean html_problem = FALSE;
        
        if (recipients) {
            for (i = 0; recipients[i] != NULL && !html_problem; i++) {
                if (!e_destination_get_html_mail_pref (recipients[i]))
                    html_problem = TRUE;
            }
        }
        
        if (html_problem) {
            html_problem = !ask_confirm_for_unwanted_html_mail (composer, recipients);
            if (html_problem)
                goto finished;
        }
    }
    
    /* Check for no subject */
    subject = e_msg_composer_get_subject (composer);
    if (subject == NULL || subject[0] == '\0') {
        if (!ask_confirm_for_empty_subject (composer))
            goto finished;
    }
    
    /* actually get the message now, this will sign/encrypt etc */
    message = e_msg_composer_get_message (composer, save_html_object_data);
    if (message == NULL)
        goto finished;
    
    /* Add info about the sending account */
    account = e_msg_composer_get_preferred_account (composer);
    
    if (account) {
        camel_medium_set_header (CAMEL_MEDIUM (message), "X-Evolution-Account", account->name);
        camel_medium_set_header (CAMEL_MEDIUM (message), "X-Evolution-Transport", account->transport->url);
        camel_medium_set_header (CAMEL_MEDIUM (message), "X-Evolution-Fcc", account->sent_folder_uri);
        if (account->id->organization)
            camel_medium_set_header (CAMEL_MEDIUM (message), "Organization", account->id->organization);
    }
    
    /* Get the message recipients and 'touch' them, boosting their use scores */
    if (recipients)
        e_destination_touchv (recipients);
    
 finished:
    
    if (recipients)
        e_destination_freev (recipients);
    
    return message;
}

static void
got_post_folder (char *uri, CamelFolder *folder, void *data)
{
    CamelFolder **fp = data;
    
    *fp = folder;
    
    if (folder)
        camel_object_ref (folder);
}

void
composer_send_cb (EMsgComposer *composer, gpointer user_data)
{
    extern CamelFolder *outbox_folder;
    CamelMimeMessage *message;
    CamelMessageInfo *info;
    struct _send_data *send;
    gboolean post = FALSE;
    CamelFolder *folder;
    XEvolution *xev;
    char *url;
    
    url = e_msg_composer_hdrs_get_post_to ((EMsgComposerHdrs *) composer->hdrs);
    if (url && *url) {
        post = TRUE;
        
        mail_msg_wait (mail_get_folder (url, 0, got_post_folder, &folder, mail_thread_new));
        
        if (!folder) {
            g_free (url);
            return;
        }
    } else {
        folder = outbox_folder;
        camel_object_ref (folder);
    }
    
    g_free (url);
    
    message = composer_get_message (composer, post, FALSE);
    if (!message)
        return;
    
    if (post) {
        /* Remove the X-Evolution* headers if we are in Post-To mode */
        xev = mail_tool_remove_xevolution_headers (message);
        mail_tool_destroy_xevolution (xev);
    }
    
    info = camel_message_info_new ();
    info->flags = CAMEL_MESSAGE_SEEN;
    
    send = g_malloc (sizeof (*send));
    send->ccd = user_data;
    if (send->ccd)
        ccd_ref (send->ccd);
    send->send = !post;
    send->composer = composer;
    g_object_ref (composer);
    gtk_widget_hide (GTK_WIDGET (composer));
    
    e_msg_composer_set_enable_autosave (composer, FALSE);
    
    mail_append_mail (folder, message, info, composer_send_queued_cb, send);
    camel_object_unref (message);
    camel_object_unref (folder);
}

struct _save_draft_info {
    struct _composer_callback_data *ccd;
    EMsgComposer *composer;
    int quit;
};

static void
save_draft_done (CamelFolder *folder, CamelMimeMessage *msg, CamelMessageInfo *info, int ok,
         const char *appended_uid, void *user_data)
{
    struct _save_draft_info *sdi = user_data;
    struct _composer_callback_data *ccd;
    CORBA_Environment ev;
    
    if (!ok)
        goto done;
    CORBA_exception_init (&ev);
    GNOME_GtkHTML_Editor_Engine_runCommand (sdi->composer->editor_engine, "saved", &ev);
    CORBA_exception_free (&ev);
    
    if ((ccd = sdi->ccd) == NULL) {
        ccd = ccd_new ();
        
        /* disconnect the previous signal handlers */
        g_signal_handlers_disconnect_by_func (sdi->composer, G_CALLBACK (composer_send_cb), NULL);
        g_signal_handlers_disconnect_by_func (sdi->composer, G_CALLBACK (composer_save_draft_cb), NULL);
        
        /* reconnect to the signals using a non-NULL ccd for the callback data */
        g_signal_connect (sdi->composer, "send", G_CALLBACK (composer_send_cb), ccd);
        g_signal_connect (sdi->composer, "save-draft", G_CALLBACK (composer_save_draft_cb), ccd);
        
        g_object_weak_ref ((GObject *) sdi->composer, (GWeakNotify) composer_destroy_cb, ccd);
    }
    
    if (ccd->drafts_folder) {
        /* delete the original draft message */
        camel_folder_set_message_flags (ccd->drafts_folder, ccd->drafts_uid,
                        CAMEL_MESSAGE_DELETED | CAMEL_MESSAGE_SEEN,
                        CAMEL_MESSAGE_DELETED | CAMEL_MESSAGE_SEEN);
        camel_object_unref (ccd->drafts_folder);
        ccd->drafts_folder = NULL;
        g_free (ccd->drafts_uid);
        ccd->drafts_uid = NULL;
    }
    
    if (ccd->folder) {
        /* set the replied flags etc */
        camel_folder_set_message_flags (ccd->folder, ccd->uid, ccd->flags, ccd->set);
        camel_object_unref (ccd->folder);
        ccd->folder = NULL;
        g_free (ccd->uid);
        ccd->uid = NULL;
    }
    
    if (appended_uid) {
        camel_object_ref (folder);
        ccd->drafts_folder = folder;
        ccd->drafts_uid = g_strdup (appended_uid);
    }
    
    if (sdi->quit)
        gtk_widget_destroy (GTK_WIDGET (sdi->composer));
    
 done:
    g_object_unref (sdi->composer);
    if (sdi->ccd)
        ccd_unref (sdi->ccd);
    g_free (info);
    g_free (sdi);
}

static void
use_default_drafts_cb (int reply, gpointer data)
{
    extern CamelFolder *drafts_folder;
    CamelFolder **folder = data;
    
    if (reply == 0) {
        *folder = drafts_folder;
        camel_object_ref (drafts_folder);
    }
}

static void
save_draft_folder (char *uri, CamelFolder *folder, gpointer data)
{
    CamelFolder **save = data;
    
    if (folder) {
        *save = folder;
        camel_object_ref (folder);
    }
}

void
composer_save_draft_cb (EMsgComposer *composer, int quit, gpointer user_data)
{
    extern char *default_drafts_folder_uri;
    extern CamelFolder *drafts_folder;
    struct _save_draft_info *sdi;
    CamelFolder *folder = NULL;
    CamelMimeMessage *msg;
    CamelMessageInfo *info;
    EAccount *account;
    
    account = e_msg_composer_get_preferred_account (composer);
    if (account && account->drafts_folder_uri &&
        strcmp (account->drafts_folder_uri, default_drafts_folder_uri) != 0) {
        int id;
        
        id = mail_get_folder (account->drafts_folder_uri, 0, save_draft_folder, &folder, mail_thread_new);
        mail_msg_wait (id);
        
        if (!folder) {
            if (!e_question ((GtkWindow *) composer, GTK_RESPONSE_YES, NULL,
                     _("Unable to open the drafts folder for this account.\n"
                       "Would you like to use the default drafts folder?")))
                return;
            
            folder = drafts_folder;
            camel_object_ref (drafts_folder);
        }
    } else {
        folder = drafts_folder;
        camel_object_ref (folder);
    }
    
    msg = e_msg_composer_get_message_draft (composer);
    
    info = g_new0 (CamelMessageInfo, 1);
    info->flags = CAMEL_MESSAGE_DRAFT | CAMEL_MESSAGE_SEEN;
    
    sdi = g_malloc (sizeof (struct _save_draft_info));
    sdi->composer = composer;
    g_object_ref (composer);
    sdi->ccd = user_data;
    if (sdi->ccd)
        ccd_ref (sdi->ccd);
    sdi->quit = quit;
    
    mail_append_mail (folder, msg, info, save_draft_done, sdi);
    camel_object_unref (folder);
    camel_object_unref (msg);
}

static GtkWidget *
create_msg_composer (EAccount *account, gboolean post, const char *url)
{
    EMsgComposer *composer;
    GConfClient *gconf;
    gboolean send_html;
    
    /* Make sure that we've actually been passed in an account. If one has
     * not been passed in, grab the default account.
     */
    if (account == NULL)
        account = mail_config_get_default_account ();
    
    gconf = gconf_client_get_default ();
    send_html = gconf_client_get_bool (gconf, "/apps/evolution/mail/composer/send_html", NULL);
    
    if (post)
        composer = e_msg_composer_new_post ();
    else if (url)
        composer = e_msg_composer_new_from_url (url);
    else
        composer = e_msg_composer_new ();
    
    if (composer) {
        e_msg_composer_hdrs_set_from_account (E_MSG_COMPOSER_HDRS (composer->hdrs), account->name);
        e_msg_composer_set_send_html (composer, send_html);
        e_msg_composer_unset_changed (composer);
        e_msg_composer_drop_editor_undo (composer);
        return GTK_WIDGET (composer);
    } else
        return NULL;
}

void
compose_msg (GtkWidget *widget, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    struct _composer_callback_data *ccd;
    GtkWidget *composer;
    EAccount *account;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb) || !check_send_configuration (fb))
        return;
    
    /* Figure out which account we want to initially compose from */
    account = mail_config_get_account_by_source_url (fb->uri);
    
    composer = create_msg_composer (account, FALSE, NULL);
    if (!composer)
        return;
    
    ccd = ccd_new ();
    
    g_signal_connect (composer, "send", G_CALLBACK (composer_send_cb), ccd);
    g_signal_connect (composer, "save-draft", G_CALLBACK (composer_save_draft_cb), ccd);
    
    g_object_weak_ref ((GObject *) composer, (GWeakNotify) composer_destroy_cb, ccd);
    
    gtk_widget_show (composer);
}

/* Send according to a mailto (RFC 2368) URL. */
void
send_to_url (const char *url, const char *parent_uri)
{
    struct _composer_callback_data *ccd;
    GtkWidget *composer;
    EAccount *account = NULL;
    
    /* FIXME: no way to get folder browser? Not without
     * big pain in the ass, as far as I can tell */
    if (!check_send_configuration (NULL))
        return;
    
    if (parent_uri)
        account = mail_config_get_account_by_source_url (parent_uri);
    
    composer = create_msg_composer (account, FALSE, url);
    
    if (!composer)
        return;
    
    ccd = ccd_new ();
    
    g_signal_connect (composer, "send", G_CALLBACK (composer_send_cb), ccd);
    g_signal_connect (composer, "save-draft", G_CALLBACK (composer_save_draft_cb), ccd);
    
    g_object_weak_ref ((GObject *) composer, (GWeakNotify) composer_destroy_cb, ccd);
    
    gtk_widget_show (composer);
}   

static GList *
list_add_addresses (GList *list, const CamelInternetAddress *cia, GHashTable *account_hash,
            GHashTable *rcpt_hash, EAccount **me)
{
    const char *name, *addr;
    EAccount *account;
    int i;
    
    for (i = 0; camel_internet_address_get (cia, i, &name, &addr); i++) {
        /* Here I'll check to see if the cc:'d address is the address
           of the sender, and if so, don't add it to the cc: list; this
           is to fix Bugzilla bug #455. */
        account = g_hash_table_lookup (account_hash, addr);
        if (account && me && !*me)
            *me = account;
        
        if (!account && !g_hash_table_lookup (rcpt_hash, addr)) {
            EDestination *dest;
            
            dest = e_destination_new ();
            e_destination_set_name (dest, name);
            e_destination_set_email (dest, addr);
            
            list = g_list_append (list, dest);
            g_hash_table_insert (rcpt_hash, (char *) addr, GINT_TO_POINTER (1));
        }
    }
    
    return list;
}

static EAccount *
guess_me (const CamelInternetAddress *to, const CamelInternetAddress *cc, GHashTable *account_hash)
{
    EAccount *account = NULL;
    const char *addr;
    int i;
    
    /* "optimization" */
    if (!to && !cc)
        return NULL;
    
    if (to) {
        for (i = 0; camel_internet_address_get (to, i, NULL, &addr); i++) {
            account = g_hash_table_lookup (account_hash, addr);
            if (account)
                goto found;
        }
    }
    
    if (cc) {
        for (i = 0; camel_internet_address_get (cc, i, NULL, &addr); i++) {
            account = g_hash_table_lookup (account_hash, addr);
            if (account)
                goto found;
        }
    }
    
 found:
    
    return account;
}

static EAccount *
guess_me_from_accounts (const CamelInternetAddress *to, const CamelInternetAddress *cc, EAccountList *accounts)
{
    EAccount *account, *def;
    GHashTable *account_hash;
    EIterator *iter;
    
    account_hash = g_hash_table_new (g_strcase_hash, g_strcase_equal);
    
    /* add the default account to the hash first */
    if ((def = mail_config_get_default_account ())) {
        if (def->id->address)
            g_hash_table_insert (account_hash, (char *) def->id->address, (void *) def);
    }
    
    iter = e_list_get_iterator ((EList *) accounts);
    while (e_iterator_is_valid (iter)) {
        account = (EAccount *) e_iterator_get (iter);
        
        if (account->id->address) {
            EAccount *acnt;
            
            /* Accounts with identical email addresses that are enabled
             * take precedence over the accounts that aren't. If all
             * accounts with matching email addresses are disabled, then
             * the first one in the list takes precedence. The default
             * account always takes precedence no matter what.
             */
            acnt = g_hash_table_lookup (account_hash, account->id->address);
            if (acnt && acnt != def && !acnt->enabled && account->enabled) {
                g_hash_table_remove (account_hash, acnt->id->address);
                acnt = NULL;
            }
            
            if (!acnt)
                g_hash_table_insert (account_hash, (char *) account->id->address, (void *) account);
        }
        
        e_iterator_next (iter);
    }
    
    g_object_unref (iter);
    
    account = guess_me (to, cc, account_hash);
    
    g_hash_table_destroy (account_hash);
    
    return account;
}

inline static void
mail_ignore (EMsgComposer *composer, const char *name, const char *address)
{
    e_msg_composer_ignore (composer, name && *name ? name : address);
}

static void
mail_ignore_address (EMsgComposer *composer, const CamelInternetAddress *addr)
{
    const char *name, *address;
    int i, max;
    
    max = camel_address_length (CAMEL_ADDRESS (addr));
    for (i = 0; i < max; i++) {
        camel_internet_address_get (addr, i, &name, &address);
        mail_ignore (composer, name, address);
    }
}

#define MAX_SUBJECT_LEN  1024

static EMsgComposer *
mail_generate_reply (CamelFolder *folder, CamelMimeMessage *message, const char *uid, int mode)
{
    const CamelInternetAddress *reply_to, *sender, *to_addrs, *cc_addrs;
    const char *name = NULL, *address = NULL, *source = NULL;
    const char *message_id, *references, *mlist = NULL;
    char *text = NULL, *subject, format[256];
    EAccount *def, *account, *me = NULL;
    EAccountList *accounts = NULL;
    GHashTable *account_hash = NULL;
    CamelMessageInfo *info = NULL;
    GList *to = NULL, *cc = NULL;
    EDestination **tov, **ccv;
    EMsgComposer *composer;
    CamelMimePart *part;
    GConfClient *gconf;
    EIterator *iter;
    time_t date;
    char *url;
    
    gconf = gconf_client_get_default ();
    
    if (mode == REPLY_POST) {
        composer = e_msg_composer_new_post ();
        if (composer != NULL) {
            url = mail_tools_folder_to_url (folder);
            e_msg_composer_hdrs_set_post_to ((EMsgComposerHdrs *) composer->hdrs, url);
            g_free (url);
        }
    } else
        composer = e_msg_composer_new ();
    
    if (!composer)
        return NULL;
    
    e_msg_composer_add_message_attachments (composer, message, TRUE);
    
    /* Set the recipients */
    accounts = mail_config_get_accounts ();
    account_hash = g_hash_table_new (g_strcase_hash, g_strcase_equal);
    
    /* add the default account to the hash first */
    if ((def = mail_config_get_default_account ())) {
        if (def->id->address)
            g_hash_table_insert (account_hash, (char *) def->id->address, (void *) def);
    }
    
    iter = e_list_get_iterator ((EList *) accounts);
    while (e_iterator_is_valid (iter)) {
        account = (EAccount *) e_iterator_get (iter);
        
        if (account->id->address) {
            EAccount *acnt;
            
            /* Accounts with identical email addresses that are enabled
             * take precedence over the accounts that aren't. If all
             * accounts with matching email addresses are disabled, then
             * the first one in the list takes precedence. The default
             * account always takes precedence no matter what.
             */
            acnt = g_hash_table_lookup (account_hash, account->id->address);
            if (acnt && acnt != def && !acnt->enabled && account->enabled) {
                g_hash_table_remove (account_hash, acnt->id->address);
                acnt = NULL;
            }
            
            if (!acnt)
                g_hash_table_insert (account_hash, (char *) account->id->address, (void *) account);
        }
        
        e_iterator_next (iter);
    }
    
    g_object_unref (iter);
    
    to_addrs = camel_mime_message_get_recipients (message, CAMEL_RECIPIENT_TYPE_TO);
    cc_addrs = camel_mime_message_get_recipients (message, CAMEL_RECIPIENT_TYPE_CC);
    mail_ignore_address (composer, to_addrs);
    mail_ignore_address (composer, cc_addrs);
    
    if (mode == REPLY_LIST) {
        /* make sure we can reply to an mlist */
        info = camel_folder_get_message_info (folder, uid);
        if (!(mlist = camel_message_info_mlist (info)) || *mlist == '\0') {
            camel_folder_free_message_info (folder, info);
            mode = REPLY_ALL;
            info = NULL;
        }
    }
    
 determine_recipients:
    if (mode == REPLY_LIST) {
        EDestination *dest;
        int i, max;
        
        /* look through the recipients to find the *real* mailing list address */
        d(printf ("we are looking for the mailing list called: %s\n", mlist));
        
        max = camel_address_length (CAMEL_ADDRESS (to_addrs));
        for (i = 0; i < max; i++) {
            camel_internet_address_get (to_addrs, i, &name, &address);
            if (!strcasecmp (address, mlist))
                break;
        }
        
        if (i == max) {
            max = camel_address_length (CAMEL_ADDRESS (cc_addrs));
            for (i = 0; i < max; i++) {
                camel_internet_address_get (cc_addrs, i, &name, &address);
                if (!strcasecmp (address, mlist))
                    break;
            }
        }
        
        if (address && i != max) {
            dest = e_destination_new ();
            e_destination_set_name (dest, name);
            e_destination_set_email (dest, address);
            
            to = g_list_append (to, dest);
        } else {
            /* mailing list address wasn't found */
            if (strchr (mlist, '@')) {
                /* mlist string has an @, maybe it's valid? */
                dest = e_destination_new ();
                e_destination_set_email (dest, mlist);
                
                to = g_list_append (to, dest);
            } else {
                /* give up and just reply to all recipients? */
                mode = REPLY_ALL;
                camel_folder_free_message_info (folder, info);
                goto determine_recipients;
            }
        }
        
        me = guess_me (to_addrs, cc_addrs, account_hash);
        camel_folder_free_message_info (folder, info);
    } else {
        GHashTable *rcpt_hash;
        EDestination *dest;
        
        rcpt_hash = g_hash_table_new (g_strcase_hash, g_strcase_equal);
        
        reply_to = camel_mime_message_get_reply_to (message);
        if (!reply_to)
            reply_to = camel_mime_message_get_from (message);
        
        if (reply_to) {
            int i;
            
            for (i = 0; camel_internet_address_get (reply_to, i, &name, &address); i++) {
                /* ignore references to the Reply-To address in the To and Cc lists */
                if (address && !(mode == REPLY_ALL && g_hash_table_lookup (account_hash, address))) {
                    /* In the case that we are doing a Reply-To-All, we do not want
                       to include the user's email address because replying to oneself
                       is kinda silly. */
                    dest = e_destination_new ();
                    e_destination_set_name (dest, name);
                    e_destination_set_email (dest, address);
                    to = g_list_append (to, dest);
                    g_hash_table_insert (rcpt_hash, (char *) address, GINT_TO_POINTER (1));
                    mail_ignore (composer, name, address);
                }
            }
        }
        
        if (mode == REPLY_ALL) {
            cc = list_add_addresses (cc, to_addrs, account_hash, rcpt_hash, me ? NULL : &me);
            cc = list_add_addresses (cc, cc_addrs, account_hash, rcpt_hash, me ? NULL : &me);
            
            /* promote the first Cc: address to To: if To: is empty */
            if (to == NULL && cc != NULL) {
                to = cc;
                cc = g_list_remove_link (cc, to);
            }
        } else {
            me = guess_me (to_addrs, cc_addrs, account_hash);
        }
        
        g_hash_table_destroy (rcpt_hash);
    }
    
    g_hash_table_destroy (account_hash);
    
    if (!me) {
        /* default 'me' to the source account... */
        if ((source = camel_mime_message_get_source (message)))
            me = mail_config_get_account_by_source_url (source);
    }
    
    /* set body text here as we want all ignored words to take effect */
    switch (gconf_client_get_int (gconf, "/apps/evolution/mail/format/reply_style", NULL)) {
    case MAIL_CONFIG_REPLY_DO_NOT_QUOTE:
        /* do nothing */
        break;
    case MAIL_CONFIG_REPLY_ATTACH:
        /* attach the original message as an attachment */
        part = mail_tool_make_message_attachment (message);
        e_msg_composer_attach (composer, part);
        camel_object_unref (part);
        break;
    case MAIL_CONFIG_REPLY_QUOTED:
    default:
        /* do what any sane user would want when replying... */
        sender = camel_mime_message_get_from (message);
        if (sender != NULL && camel_address_length (CAMEL_ADDRESS (sender)) > 0) {
            camel_internet_address_get (sender, 0, &name, &address);
        } else {
            name = _("an unknown sender");
        }
        
        date = camel_mime_message_get_date (message, NULL);
        strftime (format, sizeof (format), _("On %a, %Y-%m-%d at %H:%M, %%s wrote:"), localtime (&date));
        text = mail_tool_quote_message (message, format, name && *name ? name : address);
        mail_ignore (composer, name, address);
        if (text) {
            e_msg_composer_set_body_text (composer, text);
            g_free (text);
        }
        break;
    }
    
    /* Set the subject of the new message. */
    subject = (char *) camel_mime_message_get_subject (message);
    if (!subject)
        subject = g_strdup ("");
    else {
        if (!strncasecmp (subject, "Re: ", 4))
            subject = g_strndup (subject, MAX_SUBJECT_LEN);
        else {
            if (strlen (subject) < MAX_SUBJECT_LEN) {
                subject = g_strdup_printf ("Re: %s", subject);
            } else {
                /* We can't use %.*s because it depends on the locale being C/POSIX
                                   or UTF-8 to work correctly in glibc */
                char *sub;
                
                /*subject = g_strdup_printf ("Re: %.*s...", MAX_SUBJECT_LEN, subject);*/
                sub = g_malloc (MAX_SUBJECT_LEN + 8);
                memcpy (sub, "Re: ", 4);
                memcpy (sub + 4, subject, MAX_SUBJECT_LEN);
                memcpy (sub + 4 + MAX_SUBJECT_LEN, "...", 4);
                subject = sub;
            }
        }
    }
    
    tov = e_destination_list_to_vector (to);
    ccv = e_destination_list_to_vector (cc);
    
    g_list_free (to);
    g_list_free (cc);
    
    e_msg_composer_set_headers (composer, me ? me->name : NULL, tov, ccv, NULL, subject);
    
    e_destination_freev (tov);
    e_destination_freev (ccv);
    
    g_free (subject);
    
    /* Add In-Reply-To and References. */
    message_id = camel_medium_get_header (CAMEL_MEDIUM (message), "Message-Id");
    references = camel_medium_get_header (CAMEL_MEDIUM (message), "References");
    if (message_id) {
        char *reply_refs;
        
        e_msg_composer_add_header (composer, "In-Reply-To", message_id);
        
        if (references)
            reply_refs = g_strdup_printf ("%s %s", references, message_id);
        else
            reply_refs = g_strdup (message_id);
        
        e_msg_composer_add_header (composer, "References", reply_refs);
        g_free (reply_refs);
    } else if (references) {
        e_msg_composer_add_header (composer, "References", references);
    }
    
    e_msg_composer_drop_editor_undo (composer);
    
    return composer;
}

static void
requeue_mail_reply (CamelFolder *folder, const char *uid, CamelMimeMessage *msg, void *data)
{
    int mode = GPOINTER_TO_INT (data);
    
    if (msg != NULL)
        mail_reply (folder, msg, uid, mode);
}

void
mail_reply (CamelFolder *folder, CamelMimeMessage *msg, const char *uid, int mode)
{
    struct _composer_callback_data *ccd;
    EMsgComposer *composer;
    
    g_return_if_fail (CAMEL_IS_FOLDER (folder));
    g_return_if_fail (uid != NULL);
    
    if (!msg) {
        mail_get_message (folder, uid, requeue_mail_reply,
                  GINT_TO_POINTER (mode), mail_thread_new);
        return;
    }
    
    composer = mail_generate_reply (folder, msg, uid, mode);
    if (!composer)
        return;
    
    ccd = ccd_new ();
    
    camel_object_ref (folder);
    ccd->folder = folder;
    ccd->uid = g_strdup (uid);
    ccd->flags = CAMEL_MESSAGE_ANSWERED | CAMEL_MESSAGE_SEEN;
    if (mode == REPLY_LIST || mode == REPLY_ALL)
        ccd->flags |= CAMEL_MESSAGE_ANSWERED_ALL;
    ccd->set = ccd->flags;
    
    g_signal_connect (composer, "send", G_CALLBACK (composer_send_cb), ccd);
    g_signal_connect (composer, "save-draft", G_CALLBACK (composer_save_draft_cb), ccd);
    
    g_object_weak_ref ((GObject *) composer, (GWeakNotify) composer_destroy_cb, ccd);
    
    gtk_widget_show (GTK_WIDGET (composer));    
    e_msg_composer_unset_changed (composer);
}

void
reply_to_sender (GtkWidget *widget, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb) || !check_send_configuration (fb))
        return;
    
    mail_reply (fb->folder, NULL, fb->message_list->cursor_uid, REPLY_SENDER);
}

void
reply_to_list (GtkWidget *widget, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb) || !check_send_configuration (fb))
        return;
    
    mail_reply (fb->folder, NULL, fb->message_list->cursor_uid, REPLY_LIST);
}

void
reply_to_all (GtkWidget *widget, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb) || !check_send_configuration (fb))
        return;
    
    mail_reply(fb->folder, NULL, fb->message_list->cursor_uid, REPLY_ALL);
}

void
enumerate_msg (MessageList *ml, const char *uid, gpointer data)
{
    g_return_if_fail (ml != NULL);
    
    g_ptr_array_add ((GPtrArray *) data, g_strdup (uid));
}


static EMsgComposer *
forward_get_composer (CamelMimeMessage *message, const char *subject)
{
    struct _composer_callback_data *ccd;
    EAccount *account = NULL;
    EMsgComposer *composer;
    
    if (message) {
        const CamelInternetAddress *to_addrs, *cc_addrs;
        EAccountList *accounts;
        
        accounts = mail_config_get_accounts ();
        to_addrs = camel_mime_message_get_recipients (message, CAMEL_RECIPIENT_TYPE_TO);
        cc_addrs = camel_mime_message_get_recipients (message, CAMEL_RECIPIENT_TYPE_CC);
        
        account = guess_me_from_accounts (to_addrs, cc_addrs, accounts);
        
        if (!account) {
            const char *source;
            
            source = camel_mime_message_get_source (message);
            account = mail_config_get_account_by_source_url (source);
        }
    }
    
    if (!account)
        account = mail_config_get_default_account ();
    
    composer = e_msg_composer_new ();
    if (composer) {
        ccd = ccd_new ();
        
        g_signal_connect (composer, "send", G_CALLBACK (composer_send_cb), ccd);
        g_signal_connect (composer, "save-draft", G_CALLBACK (composer_save_draft_cb), ccd);
        
        g_object_weak_ref ((GObject *) composer, (GWeakNotify) composer_destroy_cb, ccd);
        
        e_msg_composer_set_headers (composer, account->name, NULL, NULL, NULL, subject);
    } else {
        g_warning ("Could not create composer");
    }
    
    return composer;
}

static void
do_forward_non_attached (CamelFolder *folder, GPtrArray *uids, GPtrArray *messages, void *data)
{
    MailConfigForwardStyle style = GPOINTER_TO_INT (data);
    CamelMimeMessage *message;
    char *subject, *text;
    int i;
    
    if (messages->len == 0)
        return;
    
    for (i = 0; i < messages->len; i++) {
        message = messages->pdata[i];
        subject = mail_tool_generate_forward_subject (message);
        text = mail_tool_forward_message (message, style == MAIL_CONFIG_FORWARD_QUOTED);
        
        if (text) {
            EMsgComposer *composer = forward_get_composer (message, subject);
            if (composer) {
                CamelDataWrapper *wrapper;
                
                e_msg_composer_set_body_text (composer, text);
                
                wrapper = camel_medium_get_content_object (CAMEL_MEDIUM (message));
                if (CAMEL_IS_MULTIPART (wrapper))
                    e_msg_composer_add_message_attachments (composer, message, FALSE);
                
                gtk_widget_show (GTK_WIDGET (composer));
                e_msg_composer_unset_changed (composer);
                e_msg_composer_drop_editor_undo (composer);
            }
            g_free (text);
        }
        
        g_free (subject);
    }
}

static void
forward_message (FolderBrowser *fb, MailConfigForwardStyle style)
{
    GPtrArray *uids;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    if (!check_send_configuration (fb))
        return;
    
    uids = g_ptr_array_new ();
    message_list_foreach (fb->message_list, enumerate_msg, uids);
    
    mail_get_messages (fb->folder, uids, do_forward_non_attached, GINT_TO_POINTER (style));
}

void
forward_inline (GtkWidget *widget, gpointer user_data)
{
    forward_message (user_data, MAIL_CONFIG_FORWARD_INLINE);
}

void
forward_quoted (GtkWidget *widget, gpointer user_data)
{
    forward_message (user_data, MAIL_CONFIG_FORWARD_QUOTED);
}

static void
do_forward_attach (CamelFolder *folder, GPtrArray *messages, CamelMimePart *part, char *subject, void *data)
{
    CamelMimeMessage *message = data;
    
    if (part) {
        EMsgComposer *composer = forward_get_composer (message, subject);
        if (composer) {
            e_msg_composer_attach (composer, part);
            gtk_widget_show (GTK_WIDGET (composer));
            e_msg_composer_unset_changed (composer);
            e_msg_composer_drop_editor_undo (composer);
        }
    }
}

void
forward_attached (GtkWidget *widget, gpointer user_data)
{
    FolderBrowser *fb = (FolderBrowser *) user_data;
    GPtrArray *uids;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb) || !check_send_configuration (fb))
        return;
    
    uids = g_ptr_array_new ();
    message_list_foreach (fb->message_list, enumerate_msg, uids);
    mail_build_attachment (fb->folder, uids, do_forward_attach,
                   uids->len == 1 ? fb->mail_display->current_message : NULL);
}

void
forward (GtkWidget *widget, gpointer user_data)
{
    MailConfigForwardStyle style;
    GConfClient *gconf;
    
    gconf = gconf_client_get_default ();
    style = gconf_client_get_int (gconf, "/apps/evolution/mail/format/forward_style", NULL);
    
    if (style == MAIL_CONFIG_FORWARD_ATTACHED)
        forward_attached (widget, user_data);
    else
        forward_message (user_data, style);
}


void
post_to_url (const char *url)
{
    struct _composer_callback_data *ccd;
    GtkWidget *composer;
    EAccount *account = NULL;
    
    /* FIXME: no way to get folder browser? Not without
     * big pain in the ass, as far as I can tell */
    if (!check_send_configuration (NULL))
        return;
    
    if (url)
        account = mail_config_get_account_by_source_url (url);
    
    composer = create_msg_composer (account, TRUE, NULL);
    if (!composer)
        return;
    
    e_msg_composer_hdrs_set_post_to ((EMsgComposerHdrs *) ((EMsgComposer *) composer)->hdrs, url);
    
    ccd = ccd_new ();
    
    g_signal_connect (composer, "send", G_CALLBACK (composer_send_cb), ccd);
    g_signal_connect (composer, "save-draft", G_CALLBACK (composer_save_draft_cb), ccd);
    
    g_object_weak_ref ((GObject *) composer, (GWeakNotify) composer_destroy_cb, ccd);
    
    gtk_widget_show (composer);
}

void
post_message (GtkWidget *widget, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    char *url;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb) || !check_send_configuration (fb))
        return;
    
    url = mail_tools_folder_to_url (fb->folder);
    post_to_url (url);
    g_free (url);
}

void
post_reply (GtkWidget *widget, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb) || !check_send_configuration (fb))
        return;
    
    mail_reply (fb->folder, NULL, fb->message_list->cursor_uid, REPLY_POST);
}


static EMsgComposer *
redirect_get_composer (CamelMimeMessage *message)
{
    const CamelInternetAddress *to_addrs, *cc_addrs;
    struct _composer_callback_data *ccd;
    EAccountList *accounts = NULL;
    EAccount *account = NULL;
    EMsgComposer *composer;
    
    g_return_val_if_fail (CAMEL_IS_MIME_MESSAGE (message), NULL);
    
    accounts = mail_config_get_accounts ();
    to_addrs = camel_mime_message_get_recipients (message, CAMEL_RECIPIENT_TYPE_TO);
    cc_addrs = camel_mime_message_get_recipients (message, CAMEL_RECIPIENT_TYPE_CC);
    
    account = guess_me_from_accounts (to_addrs, cc_addrs, accounts);
    
    if (!account) {
        const char *source;
        
        source = camel_mime_message_get_source (message);
        account = mail_config_get_account_by_source_url (source);
    }
    
    if (!account)
        account = mail_config_get_default_account ();
    
    /* QMail will refuse to send a message if it finds one of
           it's Delivered-To headers in the message, so remove all
           Delivered-To headers. Fixes bug #23635. */
    while (camel_medium_get_header (CAMEL_MEDIUM (message), "Delivered-To"))
        camel_medium_remove_header (CAMEL_MEDIUM (message), "Delivered-To");
    
    composer = e_msg_composer_new_redirect (message, account->name);
    if (composer) {
        ccd = ccd_new ();
        
        g_signal_connect (composer, "send", G_CALLBACK (composer_send_cb), ccd);
        g_signal_connect (composer, "save-draft", G_CALLBACK (composer_save_draft_cb), ccd);
        
        g_object_weak_ref ((GObject *) composer, (GWeakNotify) composer_destroy_cb, ccd);
    } else {
        g_warning ("Could not create composer");
    }
    
    return composer;
}

static void
do_redirect (CamelFolder *folder, const char *uid, CamelMimeMessage *message, void *data)
{
    EMsgComposer *composer;
    
    if (!message)
        return;
    
    composer = redirect_get_composer (message);
    if (composer) {
        CamelDataWrapper *wrapper;
        
        wrapper = camel_medium_get_content_object (CAMEL_MEDIUM (message));
        if (CAMEL_IS_MULTIPART (wrapper))
            e_msg_composer_add_message_attachments (composer, message, FALSE);
        
        gtk_widget_show (GTK_WIDGET (composer));
        e_msg_composer_unset_changed (composer);
        e_msg_composer_drop_editor_undo (composer);
    }
}

void
redirect (GtkWidget *widget, gpointer user_data)
{
    FolderBrowser *fb = (FolderBrowser *) user_data;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    if (!check_send_configuration (fb))
        return;
    
    mail_get_message (fb->folder, fb->message_list->cursor_uid,
              do_redirect, NULL, mail_thread_new);
}

static void
transfer_msg_done (gboolean ok, void *data)
{
    FolderBrowser *fb = data;
    gboolean hide_deleted;
    GConfClient *gconf;
    int row;
    
    if (ok && !FOLDER_BROWSER_IS_DESTROYED (fb)) {
        gconf = gconf_client_get_default ();
        hide_deleted = !gconf_client_get_bool (gconf, "/apps/evolution/mail/display/show_deleted", NULL);
        
        row = e_tree_row_of_node (fb->message_list->tree,
                      e_tree_get_cursor (fb->message_list->tree));
        
        /* If this is the last message and deleted messages
                   are hidden, select the previous */
        if ((row + 1 == e_tree_row_count (fb->message_list->tree)) && hide_deleted)
            message_list_select (fb->message_list, MESSAGE_LIST_SELECT_PREVIOUS,
                         0, CAMEL_MESSAGE_DELETED, FALSE);
        else
            message_list_select (fb->message_list, MESSAGE_LIST_SELECT_NEXT,
                         0, 0, FALSE);
    }
    
    g_object_unref (fb);
}

static void
transfer_msg (FolderBrowser *fb, gboolean delete_from_source)
{
    static const char *allowed_types[] = { "mail/*", "vtrash", NULL };
    extern EvolutionShellClient *global_shell_client;
    GNOME_Evolution_Folder *folder;
    static char *last_uri = NULL;
    GPtrArray *uids;
    char *desc;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    if (last_uri == NULL)
        last_uri = g_strdup (fb->uri);
    
    if (delete_from_source)
        desc = _("Move message(s) to");
    else
        desc = _("Copy message(s) to");
    
    evolution_shell_client_user_select_folder (global_shell_client,
                           GTK_WINDOW (gtk_widget_get_toplevel (GTK_WIDGET (fb))),
                           desc, last_uri, allowed_types,
                           &folder);
    if (!folder)
        return;
    
    if (strcmp (last_uri, folder->evolutionUri) != 0) {
        g_free (last_uri);
        last_uri = g_strdup (folder->evolutionUri);
    }
    
    uids = g_ptr_array_new ();
    message_list_foreach (fb->message_list, enumerate_msg, uids);
    
    if (delete_from_source) {
        g_object_ref (fb);
        mail_transfer_messages (fb->folder, uids, delete_from_source,
                    folder->physicalUri, 0,
                    transfer_msg_done, fb);
    } else {
        mail_transfer_messages (fb->folder, uids, delete_from_source,
                    folder->physicalUri, 0, NULL, NULL);
    }
    
    CORBA_free (folder);
}

void
move_msg_cb (GtkWidget *widget, gpointer user_data)
{
    transfer_msg (user_data, TRUE);
}

void
move_msg (BonoboUIComponent *uih, void *user_data, const char *path)
{
    transfer_msg (user_data, TRUE);
}

void
copy_msg_cb (GtkWidget *widget, gpointer user_data)
{
    transfer_msg (user_data, FALSE);
}

void
copy_msg (BonoboUIComponent *uih, void *user_data, const char *path)
{
    transfer_msg (user_data, FALSE);
}

/* Copied from e-shell-view.c */
static GtkWidget *
find_socket (GtkContainer *container)
{
        GList *children, *tmp;
    
        children = gtk_container_get_children (container);
        while (children) {
                if (BONOBO_IS_SOCKET (children->data))
                        return children->data;
                else if (GTK_IS_CONTAINER (children->data)) {
                        GtkWidget *socket = find_socket (children->data);
                        if (socket)
                                return socket;
                }
                tmp = children->next;
                g_list_free_1 (children);
                children = tmp;
        }
    
    return NULL;
}

static void
popup_listener_cb (BonoboListener *listener,
           const char *event_name,
           const CORBA_any *any,
           CORBA_Environment *ev,
           gpointer user_data)
{
    char *type = bonobo_event_subtype (event_name);
    
    if (!strcmp (type, "Destroy")) {
        gtk_widget_destroy (GTK_WIDGET (user_data));
    }
    
    g_free (type);
}

void
addrbook_sender (GtkWidget *widget, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    const char *addr_str;
    CamelMessageInfo *info;
    GtkWidget *win;
    GtkWidget *control;
    GtkWidget *socket;
    GPtrArray *uids;
    int i;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    uids = g_ptr_array_new ();
    message_list_foreach (fb->message_list, enumerate_msg, uids);
    if (uids->len != 1)
        goto done;
    
    info = camel_folder_get_message_info (fb->folder, uids->pdata[0]);
    if (info == NULL || (addr_str = camel_message_info_from (info)) == NULL)
        goto done;
    
    win = gtk_window_new (GTK_WINDOW_TOPLEVEL);
    gtk_window_set_title (GTK_WINDOW (win), _("Sender"));
    
    control = bonobo_widget_new_control ("OAFIID:GNOME_Evolution_Addressbook_AddressPopup",
                         CORBA_OBJECT_NIL);
    bonobo_widget_set_property (BONOBO_WIDGET (control),
                    "email", TC_CORBA_string, addr_str,
                    NULL);
    
    bonobo_event_source_client_add_listener (bonobo_widget_get_objref (BONOBO_WIDGET (control)),
                         popup_listener_cb, NULL, NULL, win);
    
    socket = find_socket (GTK_CONTAINER (control));
    
    g_object_weak_ref ((GObject *) socket, (GWeakNotify) gtk_widget_destroy, win);
    
    gtk_container_add (GTK_CONTAINER (win), control);
    gtk_widget_show_all (win);

done:
    for (i = 0; i < uids->len; i++)
        g_free (uids->pdata[i]);
    g_ptr_array_free (uids, TRUE);
}

void
add_sender_to_addrbook (BonoboUIComponent *uih, void *user_data, const char *path)
{
    addrbook_sender (NULL, user_data);
}

void
apply_filters (GtkWidget *widget, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    GPtrArray *uids;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    uids = g_ptr_array_new ();
    message_list_foreach (fb->message_list, enumerate_msg, uids);
    
    mail_filter_on_demand (fb->folder, uids);
}

void
select_all (BonoboUIComponent *uih, void *user_data, const char *path)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    ESelectionModel *etsm;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    if (GTK_WIDGET_HAS_FOCUS (fb->mail_display->html)) {
        gtk_html_select_all (fb->mail_display->html);
    } else {
        etsm = e_tree_get_selection_model (fb->message_list->tree);
        
        e_selection_model_select_all (etsm);
    }
}

/* Thread selection */

typedef struct thread_select_info {
    MessageList *ml;
    GPtrArray *paths;
} thread_select_info_t;

static gboolean
select_node (ETreeModel *tm, ETreePath path, gpointer user_data)
{
    thread_select_info_t *tsi = (thread_select_info_t *) user_data;
    
    g_ptr_array_add (tsi->paths, path);
    return FALSE; /*not done yet*/
}

static void
thread_select_foreach (ETreePath path, gpointer user_data)
{
    thread_select_info_t *tsi = (thread_select_info_t *) user_data;
    ETreeModel *tm = tsi->ml->model;
    ETreePath node;
    
    /* @path part of the initial selection. If it has children,
     * we select them as well. If it doesn't, we select its siblings and
     * their children (ie, the current node must be inside the thread
     * that the user wants to mark.
     */
    
    if (e_tree_model_node_get_first_child (tm, path)) 
        node = path;
    else {
        node = e_tree_model_node_get_parent (tm, path);
        
        /* Let's make an exception: if no parent, then we're about
         * to mark the whole tree. No. */
        if (e_tree_model_node_is_root (tm, node)) 
            node = path;
    }
    
    e_tree_model_node_traverse (tm, node, select_node, tsi);
}

void
select_thread (BonoboUIComponent *uih, void *user_data, const char *path)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    ETreeSelectionModel *selection_model;
    thread_select_info_t tsi;
    int i;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    /* For every selected item, select the thread containing it.
     * We can't alter the selection while iterating through it,
     * so build up a list of paths.
     */
    
    tsi.ml = fb->message_list;
    tsi.paths = g_ptr_array_new ();
    
    e_tree_selected_path_foreach (fb->message_list->tree, thread_select_foreach, &tsi);
    
    selection_model = E_TREE_SELECTION_MODEL (e_tree_get_selection_model (fb->message_list->tree));
    
    for (i = 0; i < tsi.paths->len; i++)
        e_tree_selection_model_add_to_selection (selection_model,
                             tsi.paths->pdata[i]);
    g_ptr_array_free (tsi.paths, TRUE);
}

void
invert_selection (BonoboUIComponent *uih, void *user_data, const char *path)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    ESelectionModel *etsm;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    if (GTK_WIDGET_HAS_FOCUS (fb->message_list)) {
        etsm = e_tree_get_selection_model (fb->message_list->tree);
        
        e_selection_model_invert_selection (etsm);
    }
}

/* flag all selected messages. Return number flagged */
static int
flag_messages (FolderBrowser *fb, guint32 mask, guint32 set)
{
    GPtrArray *uids;
    int i;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return 0;
    
    /* could just use specific callback but i'm lazy */
    uids = g_ptr_array_new ();
    message_list_foreach (fb->message_list, enumerate_msg, uids);
    camel_folder_freeze (fb->folder);
    for (i = 0; i < uids->len; i++) {
        camel_folder_set_message_flags (fb->folder, uids->pdata[i], mask, set);
        g_free (uids->pdata[i]);
    }
    
    camel_folder_thaw (fb->folder);
    
    g_ptr_array_free (uids, TRUE);
    
    return i;
}

static int
toggle_flags (FolderBrowser *fb, guint32 mask)
{
    GPtrArray *uids;
    int i;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return 0;
    
    /* could just use specific callback but i'm lazy */
    uids = g_ptr_array_new ();
    message_list_foreach (fb->message_list, enumerate_msg, uids);
    camel_folder_freeze (fb->folder);
    for (i = 0; i < uids->len; i++) {
        guint32 flags;
        
        flags = ~(camel_folder_get_message_flags (fb->folder, uids->pdata[i]));
        
        /* if we're flagging a message important, always undelete it too */
        if (mask & flags & CAMEL_MESSAGE_FLAGGED) {
            flags &= ~CAMEL_MESSAGE_DELETED;
            mask |= CAMEL_MESSAGE_DELETED;
        }
        
        /* if we're flagging a message deleted, mark it seen. If 
         * we're undeleting it, we also want it to be seen, so always do this.
         */
        if (mask & CAMEL_MESSAGE_DELETED) {
            flags |= CAMEL_MESSAGE_SEEN;
            mask |= CAMEL_MESSAGE_SEEN;
        }
        
        camel_folder_set_message_flags (fb->folder, uids->pdata[i], mask, flags);
        
        g_free (uids->pdata[i]);
    }
    camel_folder_thaw (fb->folder);
    
    g_ptr_array_free (uids, TRUE);
    
    return i;
}

void
mark_as_seen (BonoboUIComponent *uih, void *user_data, const char *path)
{
    flag_messages (FOLDER_BROWSER (user_data), CAMEL_MESSAGE_SEEN, CAMEL_MESSAGE_SEEN);
}

void
mark_as_unseen (BonoboUIComponent *uih, void *user_data, const char *path)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    /* Remove the automatic mark-as-read timer first */
    if (fb->seen_id) {
        g_source_remove (fb->seen_id);
        fb->seen_id = 0;
    }
    
    flag_messages (fb, CAMEL_MESSAGE_SEEN, 0);
}

void
mark_all_as_seen (BonoboUIComponent *uih, void *user_data, const char *path)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    GPtrArray *uids;
    int i;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    uids = camel_folder_get_uids (fb->folder);
    camel_folder_freeze (fb->folder);
    for (i = 0; i < uids->len; i++)
        camel_folder_set_message_flags (fb->folder, uids->pdata[i], CAMEL_MESSAGE_SEEN, ~0);
    camel_folder_free_uids (fb->folder, uids);
    camel_folder_thaw (fb->folder);
}

void
mark_as_important (BonoboUIComponent *uih, void *user_data, const char *path)
{
    flag_messages (FOLDER_BROWSER (user_data), CAMEL_MESSAGE_FLAGGED|CAMEL_MESSAGE_DELETED, CAMEL_MESSAGE_FLAGGED);
}

void
mark_as_unimportant (BonoboUIComponent *uih, void *user_data, const char *path)
{
    flag_messages (FOLDER_BROWSER (user_data), CAMEL_MESSAGE_FLAGGED, 0);
}

void
toggle_as_important (BonoboUIComponent *uih, void *user_data, const char *path)
{
    toggle_flags (FOLDER_BROWSER (user_data), CAMEL_MESSAGE_FLAGGED);
}


struct _tag_editor_data {
    MessageTagEditor *editor;
    FolderBrowser *fb;
    GPtrArray *uids;
};

static void
tag_editor_response(GtkWidget *gd, int button, struct _tag_editor_data *data)
{
    CamelFolder *folder;
    CamelTag *tags, *t;
    GPtrArray *uids;
    int i;

    /*if (FOLDER_BROWSER_IS_DESTROYED (data->fb))
      goto done;*/

    if (button == GTK_RESPONSE_OK
        && (tags = message_tag_editor_get_tag_list (data->editor))) {
        folder = data->fb->folder;
        uids = data->uids;
    
        camel_folder_freeze (folder);
        for (i = 0; i < uids->len; i++) {
            for (t = tags; t; t = t->next)
                camel_folder_set_message_user_tag (folder, uids->pdata[i], t->name, t->value);
        }
        camel_folder_thaw (folder);
        camel_tag_list_free (&tags);
    }

    gtk_widget_destroy(gd);

    g_object_unref (data->fb);
    g_ptr_array_free (data->uids, TRUE);
    g_free (data);
}

void
flag_for_followup (BonoboUIComponent *uih, void *user_data, const char *path)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    struct _tag_editor_data *data;
    GtkWidget *editor;
    GPtrArray *uids;
    int i;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    uids = g_ptr_array_new ();
    message_list_foreach (fb->message_list, enumerate_msg, uids);
    
    editor = (GtkWidget *) message_tag_followup_new ();
    
    data = g_new (struct _tag_editor_data, 1);
    data->editor = MESSAGE_TAG_EDITOR (editor);
    gtk_widget_ref (GTK_WIDGET (fb));
    data->fb = fb;
    data->uids = uids;
    
    for (i = 0; i < uids->len; i++) {
        CamelMessageInfo *info;
        
        info = camel_folder_get_message_info (fb->folder, uids->pdata[i]);
        message_tag_followup_append_message (MESSAGE_TAG_FOLLOWUP (editor),
                             camel_message_info_from (info),
                             camel_message_info_subject (info));
    }

    g_signal_connect(editor, "response", G_CALLBACK(tag_editor_response), data);
    
    /* special-case... */
    if (uids->len == 1) {
        CamelMessageInfo *info;
        
        info = camel_folder_get_message_info (fb->folder, uids->pdata[0]);
        if (info) {
            if (info->user_tags)
                message_tag_editor_set_tag_list (MESSAGE_TAG_EDITOR (editor), info->user_tags);
            camel_folder_free_message_info (fb->folder, info);
        }
    }
    
    gtk_widget_show (editor);
}

void
flag_followup_completed (BonoboUIComponent *uih, void *user_data, const char *path)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    GPtrArray *uids;
    char *now;
    int i;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    uids = g_ptr_array_new ();
    message_list_foreach (fb->message_list, enumerate_msg, uids);
    
    now = header_format_date (time (NULL), 0);
    
    camel_folder_freeze (fb->folder);
    for (i = 0; i < uids->len; i++) {
        const char *tag;
        
        tag = camel_folder_get_message_user_tag (fb->folder, uids->pdata[i], "follow-up");
        if (tag == NULL || *tag == '\0')
            continue;
        
        camel_folder_set_message_user_tag (fb->folder, uids->pdata[i], "completed-on", now);
    }
    camel_folder_thaw (fb->folder);
    
    g_free (now);
    
    g_ptr_array_free (uids, TRUE);
}

void
flag_followup_clear (BonoboUIComponent *uih, void *user_data, const char *path)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    GPtrArray *uids;
    int i;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    uids = g_ptr_array_new ();
    message_list_foreach (fb->message_list, enumerate_msg, uids);
    
    camel_folder_freeze (fb->folder);
    for (i = 0; i < uids->len; i++) {
        camel_folder_set_message_user_tag (fb->folder, uids->pdata[i], "follow-up", "");
        camel_folder_set_message_user_tag (fb->folder, uids->pdata[i], "due-by", "");
        camel_folder_set_message_user_tag (fb->folder, uids->pdata[i], "completed-on", "");
    }
    camel_folder_thaw (fb->folder);
    
    g_ptr_array_free (uids, TRUE);
}

void
zoom_in (BonoboUIComponent *uih, void *user_data, const char *path)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    gtk_html_zoom_in (fb->mail_display->html);
}

void
zoom_out (BonoboUIComponent *uih, void *user_data, const char *path)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    gtk_html_zoom_out (fb->mail_display->html);
}

void
zoom_reset (BonoboUIComponent *uih, void *user_data, const char *path)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    gtk_html_zoom_reset (fb->mail_display->html);
}

static void
do_edit_messages (CamelFolder *folder, GPtrArray *uids, GPtrArray *messages, void *data)
{
    FolderBrowser *fb = (FolderBrowser *) data;
    int i;
    
    if (messages == NULL)
        return;
    
    for (i = 0; i < messages->len; i++) {
        struct _composer_callback_data *ccd;
        EMsgComposer *composer;
        
        camel_medium_remove_header (CAMEL_MEDIUM (messages->pdata[i]), "X-Mailer");
        
        composer = e_msg_composer_new_with_message (messages->pdata[i]);
        e_msg_composer_unset_changed (composer);
        e_msg_composer_drop_editor_undo (composer);
        
        if (composer) {
            ccd = ccd_new ();
            if (folder_browser_is_drafts (fb)) {
                camel_object_ref (folder);
                ccd->drafts_folder = folder;
                ccd->drafts_uid = g_strdup (uids->pdata[i]);
            }
            
            g_signal_connect (composer, "send", G_CALLBACK (composer_send_cb), ccd);
            g_signal_connect (composer, "save-draft", G_CALLBACK (composer_save_draft_cb), ccd);
            
            g_object_weak_ref ((GObject *) composer, (GWeakNotify) composer_destroy_cb, ccd);
            
            gtk_widget_show (GTK_WIDGET (composer));
        }
    }
}

static gboolean
are_you_sure (const char *msg, GPtrArray *uids, FolderBrowser *fb)
{
    GtkWidget *dialog;
    int button, i;

    dialog = gtk_message_dialog_new (FB_WINDOW (fb), GTK_DIALOG_MODAL|GTK_DIALOG_DESTROY_WITH_PARENT,
                     GTK_MESSAGE_QUESTION, GTK_BUTTONS_OK_CANCEL,
                     msg, uids->len);
    button = gtk_dialog_run ((GtkDialog *) dialog);
    gtk_widget_destroy (dialog);
    
    if (button != GTK_RESPONSE_OK) {
        for (i = 0; i < uids->len; i++)
            g_free (uids->pdata[i]);
        g_ptr_array_free (uids, TRUE);
    }
    
    return button == GTK_RESPONSE_OK;
}

static void
edit_msg_internal (FolderBrowser *fb)
{
    GPtrArray *uids;
    
    if (!check_send_configuration (fb))
        return;
    
    uids = g_ptr_array_new ();
    message_list_foreach (fb->message_list, enumerate_msg, uids);
    
    if (uids->len > 10 && !are_you_sure (_("Are you sure you want to edit all %d messages?"), uids, fb))
        return;
    
    mail_get_messages (fb->folder, uids, do_edit_messages, fb);
}

void
edit_msg (GtkWidget *widget, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    if (!folder_browser_is_drafts (fb)) {
        e_notice(FB_WINDOW(fb), GTK_MESSAGE_ERROR,
             _("You may only edit messages saved\nin the Drafts folder."));
        return;
    }
    
    edit_msg_internal (fb);
}

static void
do_resend_messages (CamelFolder *folder, GPtrArray *uids, GPtrArray *messages, void *data)
{
    int i;
    
    for (i = 0; i < messages->len; i++) {
        /* generate a new Message-Id because they need to be unique */
        camel_mime_message_set_message_id (messages->pdata[i], NULL);
    }
    
    /* "Resend" should open up the composer to let the user edit the message */
    do_edit_messages (folder, uids, messages, data);
}


void
resend_msg (GtkWidget *widget, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    GPtrArray *uids;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    if (!folder_browser_is_sent (fb)) {
        e_notice (FB_WINDOW (fb), GTK_MESSAGE_ERROR,
              _("You may only resend messages\nin the Sent folder."));
        return;
    }
    
    if (!check_send_configuration (fb))
        return;
    
    uids = g_ptr_array_new ();
    message_list_foreach (fb->message_list, enumerate_msg, uids);
    
    if (uids->len > 10 && !are_you_sure (_("Are you sure you want to resend all %d messages?"), uids, fb))
        return;
    
    mail_get_messages (fb->folder, uids, do_resend_messages, fb);
}


void
search_msg (GtkWidget *widget, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    GtkWidget *w;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    if (fb->mail_display->current_message == NULL) {
        GtkWidget *dialog;
        
        dialog = gtk_message_dialog_new (FB_WINDOW(fb), GTK_DIALOG_DESTROY_WITH_PARENT,
                         GTK_MESSAGE_WARNING, GTK_BUTTONS_CLOSE,
                         _("No Message Selected"));
        g_signal_connect_swapped (dialog, "response", G_CALLBACK (gtk_widget_destroy), dialog);
        gtk_widget_show (dialog);
        return;
    }
    
    w = mail_search_new (fb->mail_display);
    gtk_widget_show_all (w);
}

void
load_images (GtkWidget *widget, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    mail_display_load_images (fb->mail_display);
}

static void
save_msg_ok (GtkWidget *widget, gpointer user_data)
{
    CamelFolder *folder;
    GPtrArray *uids;
    const char *path;
    struct stat st;
    gboolean ret = TRUE;
    
    path = gtk_file_selection_get_filename (GTK_FILE_SELECTION (user_data));
    if (path[0] == '\0')
        return;
    
    /* make sure we can actually save to it... */
    if (stat (path, &st) != -1 && !S_ISREG (st.st_mode))
        return;

    if (access(path, F_OK) == 0) {
        if (access(path, W_OK) != 0) {
            e_notice(GTK_WINDOW(user_data), GTK_MESSAGE_ERROR,
                 _("Cannot save to `%s'\n %s"), path, g_strerror(errno));
            return;
        }

        ret = e_question(GTK_WINDOW(user_data), GTK_RESPONSE_NO, NULL,
                 _("`%s' already exists.\nOverwrite it?"), path);
    }
    
    if (ret) {
        folder = g_object_get_data ((GObject *) user_data, "folder");
        uids = g_object_steal_data (G_OBJECT (user_data), "uids");
        mail_save_messages (folder, uids, path, NULL, NULL);
        gtk_widget_destroy (GTK_WIDGET (user_data));
    }
}

static void
save_msg_destroy (gpointer user_data)
{
    GPtrArray *uids = user_data;
    
    if (uids) {
        int i;
        
        for (i = 0; i < uids->len; i++)
            g_free (uids->pdata[i]);
        
        g_ptr_array_free (uids, TRUE);
    }
}

void
save_msg (GtkWidget *widget, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    GtkFileSelection *filesel;
    GPtrArray *uids;
    char *title, *path;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    uids = g_ptr_array_new ();
    message_list_foreach (fb->message_list, enumerate_msg, uids);
    
    if (uids->len == 1)
        title = _("Save Message As...");
    else
        title = _("Save Messages As...");
    
    filesel = GTK_FILE_SELECTION (gtk_file_selection_new (title));
    path = g_strdup_printf ("%s/", g_get_home_dir ());
    gtk_file_selection_set_filename (filesel, path);
    g_free (path);
    
    g_object_set_data_full ((GObject *) filesel, "uids", uids, save_msg_destroy);
    g_object_set_data ((GObject *) filesel, "folder", fb->folder);
    
    g_signal_connect (filesel->ok_button, "clicked", G_CALLBACK (save_msg_ok), filesel);
    g_signal_connect_swapped (filesel->cancel_button, "clicked",
                  G_CALLBACK (gtk_widget_destroy), filesel);
    
    gtk_widget_show (GTK_WIDGET (filesel));
}

void
colour_msg (GtkWidget *widget, gpointer user_data)
{
    /* FIXME: implement me? */
}

void
delete_msg (GtkWidget *button, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    gboolean hide_deleted;
    GConfClient *gconf;
    int deleted, row;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    gconf = gconf_client_get_default ();
    
    deleted = flag_messages (fb, CAMEL_MESSAGE_DELETED | CAMEL_MESSAGE_SEEN,
                 CAMEL_MESSAGE_DELETED | CAMEL_MESSAGE_SEEN);
    
    /* Select the next message if we are only deleting one message */
    if (deleted == 1) {
        row = e_tree_row_of_node (fb->message_list->tree,
                      e_tree_get_cursor (fb->message_list->tree));
        
        hide_deleted = !gconf_client_get_bool (gconf, "/apps/evolution/mail/display/show_deleted", NULL);
        
        /* If this is the last message and deleted messages
                   are hidden, select the previous */
        if ((row + 1 == e_tree_row_count (fb->message_list->tree)) && hide_deleted)
            message_list_select (fb->message_list, MESSAGE_LIST_SELECT_PREVIOUS,
                         0, CAMEL_MESSAGE_DELETED, FALSE);
        else
            message_list_select (fb->message_list, MESSAGE_LIST_SELECT_NEXT,
                         0, 0, FALSE);
    }
}

void
undelete_msg (GtkWidget *button, gpointer user_data)
{
    flag_messages (FOLDER_BROWSER (user_data), CAMEL_MESSAGE_DELETED, 0);
}

void
next_msg (GtkWidget *button, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    message_list_select (fb->message_list, MESSAGE_LIST_SELECT_NEXT, 0, 0, FALSE);
}

void
next_unread_msg (GtkWidget *button, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    message_list_select (fb->message_list, MESSAGE_LIST_SELECT_NEXT, 0, CAMEL_MESSAGE_SEEN, TRUE);
}

void
next_flagged_msg (GtkWidget *button, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    message_list_select (fb->message_list, MESSAGE_LIST_SELECT_NEXT,
                 CAMEL_MESSAGE_FLAGGED, CAMEL_MESSAGE_FLAGGED, FALSE);
}

void
next_thread (GtkWidget *button, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    message_list_select_next_thread (fb->message_list);
}

void
previous_msg (GtkWidget *button, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    message_list_select (fb->message_list, MESSAGE_LIST_SELECT_PREVIOUS,
                 0, 0, FALSE);
}

void
previous_unread_msg (GtkWidget *button, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    message_list_select (fb->message_list, MESSAGE_LIST_SELECT_PREVIOUS,
                 0, CAMEL_MESSAGE_SEEN, TRUE);
}

void
previous_flagged_msg (GtkWidget *button, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    message_list_select (fb->message_list, MESSAGE_LIST_SELECT_PREVIOUS,
                 CAMEL_MESSAGE_FLAGGED, CAMEL_MESSAGE_FLAGGED, TRUE);
}

static void
expunged_folder (CamelFolder *f, void *data)
{
    FolderBrowser *fb = data;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    fb->expunging = NULL;
    gtk_widget_set_sensitive (GTK_WIDGET (fb->message_list), TRUE);

    /* FIXME: we should check that the focus hasn't changed in the
     * mean time, otherwise we steal the focus unecessarily.
     * Check :get_toplevel()->focus_widget? */
    if (fb->expunge_mlfocussed)
        gtk_widget_grab_focus((GtkWidget *)fb->message_list);
}

static gboolean
confirm_expunge (FolderBrowser *fb)
{
    gboolean res, show_again;
    GConfClient *gconf;
    
    gconf = gconf_client_get_default ();
    
    if (!gconf_client_get_bool (gconf, "/apps/evolution/mail/prompts/expunge", NULL))
        return TRUE;
    
    res = e_question (FB_WINDOW (fb), GTK_RESPONSE_NO, &show_again,
              _("This operation will permanently erase all messages marked as\n"
                "deleted. If you continue, you will not be able to recover these messages.\n"
                "\nReally erase these messages?"));
    
    gconf_client_set_bool (gconf, "/apps/evolution/mail/prompts/expunge", show_again, NULL);
    
    return res;
}

void
expunge_folder (BonoboUIComponent *uih, void *user_data, const char *path)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    if (fb->folder && (fb->expunging == NULL || fb->folder != fb->expunging) && confirm_expunge (fb)) {
        CamelMessageInfo *info;
        GtkWindow *top;
        GtkWidget *focus;

        /* disable the message list so user can't click on them while we expunge */

        /* nasty hack to find out if some widget inside the message list is focussed ... */
        top = GTK_WINDOW (gtk_widget_get_toplevel((GtkWidget *)fb->message_list));
        focus = top?top->focus_widget:NULL;
        while (focus && focus != (GtkWidget *)fb->message_list)
            focus = focus->parent;
        fb->expunge_mlfocussed = focus == (GtkWidget *)fb->message_list;
        gtk_widget_set_sensitive (GTK_WIDGET (fb->message_list), FALSE);
        
        /* Only blank the mail display if the message being
                   viewed is one of those to be expunged */
        if (fb->loaded_uid) {
            info = camel_folder_get_message_info (fb->folder, fb->loaded_uid);
            
            if (!info || info->flags & CAMEL_MESSAGE_DELETED)
                mail_display_set_message (fb->mail_display, NULL, NULL, NULL);
        }
        
        fb->expunging = fb->folder;
        mail_expunge_folder (fb->folder, expunged_folder, fb);
    }
}

/********************** Begin Filter Editor ********************/

static GtkWidget *filter_editor = NULL;

static void
filter_editor_response (GtkWidget *dialog, int button, FolderBrowser *fb)
{
    FilterContext *fc;
    
    if (button == GTK_RESPONSE_ACCEPT) {
        char *user;
        
        fc = g_object_get_data(G_OBJECT(dialog), "context");
        user = g_strdup_printf ("%s/filters.xml", evolution_dir);
        rule_context_save ((RuleContext *)fc, user);
        g_free (user);
    }

    gtk_widget_destroy(dialog);

    filter_editor = NULL;
}

static const char *filter_source_names[] = {
    "incoming",
    "outgoing",
    NULL,
};

void
filter_edit (BonoboUIComponent *uih, void *user_data, const char *path)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    FilterContext *fc;
    char *user, *system;
    
    if (filter_editor) {
        gdk_window_raise (GTK_WIDGET (filter_editor)->window);
        return;
    }
    
    fc = filter_context_new ();
    user = g_strdup_printf ("%s/filters.xml", evolution_dir);
    system = EVOLUTION_PRIVDATADIR "/filtertypes.xml";
    rule_context_load ((RuleContext *)fc, system, user);
    g_free (user);
    
    if (((RuleContext *)fc)->error) {
        e_notice(FB_WINDOW (fb), GTK_MESSAGE_ERROR,
             _("Error loading filter information:\n%s"),
             ((RuleContext *)fc)->error);
        return;
    }
    
    filter_editor = (GtkWidget *)filter_editor_new (fc, filter_source_names);
    /* FIXME: maybe this needs destroy func? */
    gtk_window_set_transient_for ((GtkWindow *) filter_editor, FB_WINDOW (fb));
    gtk_window_set_title (GTK_WINDOW (filter_editor), _("Filters"));
    g_object_set_data_full ((GObject *) filter_editor, "context", fc, (GtkDestroyNotify) g_object_unref);
    g_signal_connect (filter_editor, "response", G_CALLBACK (filter_editor_response), fb);
    gtk_widget_show (GTK_WIDGET (filter_editor));
}

/********************** End Filter Editor ********************/

void
vfolder_edit_vfolders (BonoboUIComponent *uih, void *user_data, const char *path)
{
    vfolder_edit ();
}


/* static void
header_print_cb (GtkHTML *html, GnomePrintContext *print_context,
         double x, double y, double width, double height, gpointer user_data)
{
    printf ("header_print_cb %f,%f x %f,%f\n", x, y, width, height);

    gnome_print_newpath (print_context);
    gnome_print_setlinewidth (print_context, 12.0);
    gnome_print_setrgbcolor (print_context, 1.0, 0.0, 0.0);
    gnome_print_moveto (print_context, x, y);
    gnome_print_lineto (print_context, x+width, y-height);
    gnome_print_strokepath (print_context);
} */

struct footer_info {
    GnomeFont *local_font;
    gint page_num, pages;
};

static void
footer_print_cb (GtkHTML *html, GnomePrintContext *print_context,
         double x, double y, double width, double height, gpointer user_data)
{
    struct footer_info *info = (struct footer_info *) user_data;
    
    if (info->local_font) {
        char *text = g_strdup_printf (_("Page %d of %d"), info->page_num, info->pages);
        /*gdouble tw = gnome_font_get_width_string (info->local_font, text);*/
        /* FIXME: work out how to measure this */
        gdouble tw = strlen (text) * 8;
        
        gnome_print_gsave       (print_context);
        gnome_print_newpath     (print_context);
        gnome_print_setrgbcolor (print_context, .0, .0, .0);
        gnome_print_moveto      (print_context, x + width - tw, y - gnome_font_get_ascender (info->local_font));
        gnome_print_setfont     (print_context, info->local_font);
        gnome_print_show        (print_context, text);
        gnome_print_grestore    (print_context);
        
        g_free (text);
        info->page_num++;
    }
}

static void
footer_info_free (struct footer_info *info)
{
    if (info->local_font)
        gnome_font_unref (info->local_font);
    g_free (info);
}

static struct footer_info *
footer_info_new (GtkHTML *html, GnomePrintContext *pc, gdouble *line)
{
    struct footer_info *info;
    
    info = g_new (struct footer_info, 1);
    info->local_font = gnome_font_find_closest ("Helvetica", 10.0);
    
    if (info->local_font)
        *line = gnome_font_get_ascender (info->local_font) + gnome_font_get_descender (info->local_font);
    
    info->page_num = 1;
    info->pages = gtk_html_print_get_pages_num (html, pc, 0.0, *line);
    
    return info;
}

static void
do_mail_print (FolderBrowser *fb, gboolean preview)
{
    GtkHTML *html;
    GtkWidget *w = NULL;
    GnomePrintContext *print_context;
    GnomePrintJob *print_master;
    GnomePrintConfig *config = NULL;
    GtkDialog *dialog;
    gdouble line = 0.0;
    struct footer_info *info;
    
    if (!preview) {
        dialog = (GtkDialog *) gnome_print_dialog_new (NULL, _("Print Message"), GNOME_PRINT_DIALOG_COPIES);
        gtk_dialog_set_default_response (dialog, GNOME_PRINT_DIALOG_RESPONSE_PRINT);
        gtk_window_set_transient_for ((GtkWindow *) dialog, (GtkWindow *) gtk_widget_get_toplevel ((GtkWidget *) fb));
        
        switch (gtk_dialog_run (dialog)) {
        case GNOME_PRINT_DIALOG_RESPONSE_PRINT:
            break;
        case GNOME_PRINT_DIALOG_RESPONSE_PREVIEW:
            preview = TRUE;
            break;
        default:
            gtk_widget_destroy ((GtkWidget *) dialog);
            return;
        }
        
        config = gnome_print_dialog_get_config ((GnomePrintDialog *) dialog);
        gtk_widget_destroy ((GtkWidget *)dialog);
    }
    
    if (config) {
        print_master = gnome_print_job_new (config);
        gnome_print_config_unref (config);
    } else
        print_master = gnome_print_job_new (NULL);
    
    /* paper size settings? */
    /*gnome_print_master_set_paper (print_master, paper);*/
    print_context = gnome_print_job_get_context (print_master);
    
    html = GTK_HTML (gtk_html_new ());
    mail_display_initialize_gtkhtml (fb->mail_display, html);
    
    /* Set our 'printing' flag to true and render.  This causes us
       to ignoring any adjustments we made to accomodate the
       user's theme. */
    fb->mail_display->printing = TRUE;
    
    if (!GTK_WIDGET_REALIZED (GTK_WIDGET (html))) {
        /* gtk widgets don't like to be realized outside top level widget
           so we put new html widget into gtk window */
        w = gtk_window_new (GTK_WINDOW_TOPLEVEL);
        gtk_container_add (GTK_CONTAINER (w), GTK_WIDGET (html));
        gtk_widget_realize (GTK_WIDGET (html));
    }
    mail_display_render (fb->mail_display, html, TRUE);
    gtk_html_print_set_master (html, print_master);
    
    info = footer_info_new (html, print_context, &line);
    gtk_html_print_with_header_footer (html, print_context, 0.0, line, NULL, footer_print_cb, info);
    footer_info_free (info);
    
    fb->mail_display->printing = FALSE;
    
    gnome_print_job_close (print_master);
    gtk_widget_destroy (GTK_WIDGET (html));
    if (w)
        gtk_widget_destroy (w);
    
    if (preview){
        GtkWidget *pw;
        
        pw = gnome_print_job_preview_new (print_master, _("Print Preview"));
        gtk_widget_show (pw);
    } else {
        int result = gnome_print_job_print (print_master);
        
        if (result == -1)
            e_notice (FB_WINDOW (fb), GTK_MESSAGE_ERROR, _("Printing of message failed"));
    }

        g_object_unref (print_master);
}

/* This is pretty evil.  FolderBrowser's API should be extended to allow these sorts of
   things to be done in a more natural way. */

/* <evil_code> */

struct blarg_this_sucks {
    FolderBrowser *fb;
    gboolean preview;
};

static void
done_message_selected (CamelFolder *folder, const char *uid, CamelMimeMessage *msg, void *data)
{
    struct blarg_this_sucks *blarg = data;
    FolderBrowser *fb = blarg->fb;
    gboolean preview = blarg->preview;
    CamelMessageInfo *info;
    
    g_free (blarg);
    
    info = camel_folder_get_message_info (fb->folder, uid);
    mail_display_set_message (fb->mail_display, (CamelMedium *) msg, fb->folder, info);
    if (info)
        camel_folder_free_message_info (fb->folder, info);
    
    g_free (fb->loaded_uid);
    fb->loaded_uid = fb->loading_uid;
    fb->loading_uid = NULL;
    
    if (msg)
        do_mail_print (fb, preview);
}

/* Ack!  Most of this is copied from folder-browser.c */
static void
do_mail_fetch_and_print (FolderBrowser *fb, gboolean preview)
{
    if (!fb->preview_shown || fb->mail_display->current_message == NULL) {
        /* If the preview pane is closed, we have to do some
           extra magic to load the message. */
        struct blarg_this_sucks *blarg = g_new (struct blarg_this_sucks, 1);
        
        blarg->fb = fb;
        blarg->preview = preview;
        
        fb->loading_id = 0;
        
        /* if we are loading, then set a pending, but leave the loading, coudl cancel here (?) */
        if (fb->loading_uid) {
            g_free (fb->pending_uid);
            fb->pending_uid = g_strdup (fb->new_uid);
        } else {
            if (fb->new_uid) {
                fb->loading_uid = g_strdup (fb->new_uid);
                mail_get_message (fb->folder, fb->loading_uid, done_message_selected, blarg, mail_thread_new);
            } else {
                mail_display_set_message (fb->mail_display, NULL, NULL, NULL);
                g_free (blarg);
            }
        }
    } else {
        do_mail_print (fb, preview);
    }
}

/* </evil_code> */


void
print_msg (GtkWidget *button, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    do_mail_fetch_and_print (fb, FALSE);
}

void
print_preview_msg (GtkWidget *button, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    do_mail_fetch_and_print (fb, TRUE);
}

/******************** Begin Subscription Dialog ***************************/

static GtkObject *subscribe_dialog = NULL;

static void
subscribe_dialog_destroy (GtkObject *dialog, GObject *deadbeef)
{
    if (subscribe_dialog) {
        g_object_unref (subscribe_dialog);
        subscribe_dialog = NULL;
    }
}

void
manage_subscriptions (BonoboUIComponent *uih, void *user_data, const char *path)
{
    if (!subscribe_dialog) {
        subscribe_dialog = subscribe_dialog_new ();
        
        g_object_weak_ref ((GObject *) SUBSCRIBE_DIALOG (subscribe_dialog)->app,
                   (GWeakNotify) subscribe_dialog_destroy, subscribe_dialog);
        
        subscribe_dialog_show (subscribe_dialog);
    } else {
        gdk_window_raise (SUBSCRIBE_DIALOG (subscribe_dialog)->app->window);
    }
}

/******************** End Subscription Dialog ***************************/

static void
local_configure_done(const char *uri, CamelFolder *folder, void *data)
{
    FolderBrowser *fb = data;

    if (FOLDER_BROWSER_IS_DESTROYED (fb)) {
        g_object_unref(fb);
        return;
    }

    if (folder == NULL)
        folder = fb->folder;

    message_list_set_folder(fb->message_list, folder, FALSE);
    g_object_unref(fb);
}

void
configure_folder (BonoboUIComponent *uih, void *user_data, const char *path)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    if (fb->uri) {
        if (strncmp (fb->uri, "vfolder:", 8) == 0) {
            vfolder_edit_rule (fb->uri);
        } else {
            message_list_set_folder(fb->message_list, NULL, FALSE);
            g_object_ref((GtkObject *)fb);
            mail_local_reconfigure_folder(fb->uri, local_configure_done, fb);
        }
    }
}

static void
do_view_message (CamelFolder *folder, const char *uid, CamelMimeMessage *message, void *data)
{
    FolderBrowser *fb = FOLDER_BROWSER (data);
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    if (message) {
        GtkWidget *mb;
        
        camel_folder_set_message_flags (folder, uid, CAMEL_MESSAGE_SEEN, CAMEL_MESSAGE_SEEN);
        mb = message_browser_new (fb->shell, fb->uri, uid);
        gtk_widget_show (mb);
    }
}

void
view_msg (GtkWidget *widget, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    GPtrArray *uids;
    int i;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    uids = g_ptr_array_new ();
    message_list_foreach (fb->message_list, enumerate_msg, uids);

    if (uids->len > 10 && !are_you_sure (_("Are you sure you want to open all %d messages in separate windows?"), uids, fb))
        return;
    
    /* FIXME: use mail_get_messages() */
    for (i = 0; i < uids->len; i++) {
        mail_get_message (fb->folder, uids->pdata [i], do_view_message, fb, mail_thread_queued);
        g_free (uids->pdata [i]);
    }
    g_ptr_array_free (uids, TRUE);
}

void
open_msg (GtkWidget *widget, gpointer user_data)
{
    FolderBrowser *fb = FOLDER_BROWSER (user_data);
    extern CamelFolder *outbox_folder;
    
    if (FOLDER_BROWSER_IS_DESTROYED (fb))
        return;
    
    if (folder_browser_is_drafts (fb) || fb->folder == outbox_folder)
        edit_msg_internal (fb);
    else
        view_msg (NULL, user_data);
}

void
open_message (BonoboUIComponent *uih, void *user_data, const char *path)
{
    open_msg (NULL, user_data);
}

void
edit_message (BonoboUIComponent *uih, void *user_data, const char *path)
{
        edit_msg (NULL, user_data);
}

void
stop_threads (BonoboUIComponent *uih, void *user_data, const char *path)
{
    camel_operation_cancel (NULL);
}

static void
empty_trash_expunged_cb (CamelFolder *folder, void *data)
{
    camel_object_unref (folder);
}

void
empty_trash (BonoboUIComponent *uih, void *user_data, const char *path)
{
    CamelProvider *provider;
    EAccountList *accounts;
    CamelFolder *vtrash;
    FolderBrowser *fb;
    CamelException ex;
    EAccount *account;
    EIterator *iter;
    
    fb = user_data ? FOLDER_BROWSER (user_data) : NULL;
    
    if (fb && !confirm_expunge (fb))
        return;
    
    camel_exception_init (&ex);
    
    /* expunge all remote stores */
    accounts = mail_config_get_accounts ();
    iter = e_list_get_iterator ((EList *) accounts);
    while (e_iterator_is_valid (iter)) {
        account = (EAccount *) e_iterator_get (iter);
        
        /* make sure this is a valid source */
        if (account->enabled && account->source->url) {
            provider = camel_session_get_provider (session, account->source->url, &ex);
            if (provider) {
                /* make sure this store is a remote store */
                if (provider->flags & CAMEL_PROVIDER_IS_STORAGE &&
                    provider->flags & CAMEL_PROVIDER_IS_REMOTE) {
                    vtrash = mail_tool_get_trash (account->source->url, FALSE, &ex);
                    
                    if (vtrash) {
                        mail_expunge_folder (vtrash, empty_trash_expunged_cb, NULL);
                    }
                }
            }
            
            /* clear the exception for the next round */
            camel_exception_clear (&ex);
        }
        
        e_iterator_next (iter);
    }
    
    g_object_unref (iter);
    
    /* Now empty the local trash folder */
    vtrash = mail_tool_get_trash ("file:/", TRUE, &ex);
    if (vtrash)
        mail_expunge_folder (vtrash, empty_trash_expunged_cb, NULL);
    
    camel_exception_clear (&ex);
}