aboutsummaryrefslogblamecommitdiffstats
path: root/mail/em-vfolder-editor-rule.c
blob: 59c3713ed49c4380b052ad4a9b07ec7a9d1c5f94 (plain) (tree)
1
2
3
4
5
6
7
8
9
  
                                                                


                                                               



                                                                    


                                                                   
                                                                             






                                                        
  








                    
                       
 


                          






                                                

                                      
 
                                                 
                                      
                                                                         
 
                                                 
                                      
                                                                         
 
                                    
                              



               
                    
  
 
                                                                
 
               


                               

           
                                                           
                                                
 
                              

                                             
                                      


                                                                            
 
                                                         
                                                               
         
 

                                                       
 
                                                     
 
 
           
                                                  


                                               
 
                              
                                  

                                                                

                                                            
         
 
                                                                       


           
                                                  


                                             
 
                              
                                  

                                            

                                                                    

                               
 

                                                                       
 
           
                                             
 
                                         
 
                                                           


                                               


                                                    
                                                                               


           
                                              
 
                                                                          

                                                     
                                                                                


           
                                                                   
 


                                            
                                                                              

                                              
                                                                      
                                                                      

                                                              

                                                        



                                                   
                             
                                     
                                  

                             
                                            


                                                 
 
           
                                                       
 
                                                               

 
             
                                                  
 
                                                                 

                             
                                                                       

 
              
                                                              
 

                                                               
                                   

 


                      
                   


                     
                         
                          
                            
                               
                                   
                                        


           
                                         
 

                                    
                                                                  
 



                                                              
                                                                                     


           
                                                  
                                                
 
                             


           

                                                      
 
                                    
 

                                                                 
                         
 


                                                                
 




                                                  
                                                 

                                                                                         

 
           
                                                                    

                                                         




                            

                                             







                                                                      

                                                      




                                                                               

                                                                       








                                                                                     


                                                
 
                                 


                                  
 
                                                                    
                                                        
                                                                           
 
                                                                       
 


                                                                 
                                 
                                            
                                         
 
                                                                          
                                                            
 
                                                                                           
 



                                                                                          
 
                                                                                 
 













                                                                                                     
                                       
                                                                                                   











                                                                                       

                                                                   
 
                                     

         
                                                   
                                                 


           

                                      
 
                                  
                                 
                                    
                          


                                                  
                                                                 
 
                                                    
 
                                         
                              
                                              
                                                  





                                                                         


                                                                              



                                                        
                                 


           

                                         

                                    
                                          

                          

                                                     
               
 
                                                                                  
                                                                     
 
                      


                                                                           
 
                                                                            





















                                                                                                      

                                                                           
 


                                                                         
 





                                                                
                 
         
 











                                                                                          
                                                                                      
                 
                                          
         
 
                             

 
                  

                             
 
                                                 
                              
                                                                                                   
                              

                                  
                                  
                                  
                            
                   
                         
                                    
 
                                                                            
                                    
 
                                          

                      
 








                                                                                          

                                 













                                                                                            

                                 










                                                                                            










                                                                                                                 












                                                                                                                  

                                           









                                                                   
                                                                                     


                                                                             

                                                     

                                                           





                                                                                                 

                                    








                                                                            


                                                                                                 

                                 






                                                                              

                                                     


                                                                                    

                                                        




                                                                                
                                                    
                            
 
                                                                                         
 
                      
                                                                     



                                                               
 
                                                           

                                           



                                                                                      
                                
         
 
                                                                  

                                                                        
                          

                                                         
 
                                      
 
                                                                                               
                          
                                    
                                                               
                                                    
 
                             
 

                                                      
                                    
 
                                                          
 

                      
/*
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2 of the License, or (at your option) version 3.
 *
 * 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with the program; if not, see <http://www.gnu.org/licenses/>
 *
 *
 * Authors:
 *      Not Zed <notzed@lostzed.mmc.com.au>
 *      Jeffrey Stedfast <fejj@ximian.com>
 *
 * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
 *
 */

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

#include <string.h>

#include <gtk/gtk.h>
#include <glib/gi18n.h>

#include <shell/e-shell.h>

#include <e-util/e-util.h>
#include <e-util/e-util-private.h>

#include <libemail-engine/e-mail-folder-utils.h>

#include "em-folder-selector.h"
#include "em-folder-tree.h"
#include "em-utils.h"
#include "em-vfolder-editor-context.h"
#include "em-vfolder-editor-rule.h"

#define EM_VFOLDER_EDITOR_RULE_GET_PRIVATE(obj) \
    (G_TYPE_INSTANCE_GET_PRIVATE \
    ((obj), EM_TYPE_VFOLDER_EDITOR_RULE, EMVFolderEditorRulePrivate))

#define EM_VFOLDER_EDITOR_RULE_GET_PRIVATE(obj) \
    (G_TYPE_INSTANCE_GET_PRIVATE \
    ((obj), EM_TYPE_VFOLDER_EDITOR_RULE, EMVFolderEditorRulePrivate))

struct _EMVFolderEditorRulePrivate {
    EMailSession *session;
};

enum {
    PROP_0,
    PROP_SESSION
};

static GtkWidget *get_widget (EFilterRule *fr, ERuleContext *f);

G_DEFINE_TYPE (
    EMVFolderEditorRule,
    em_vfolder_editor_rule,
    EM_TYPE_VFOLDER_RULE)

static void
vfolder_editor_rule_set_session (EMVFolderEditorRule *rule,
                          EMailSession *session)
{
    if (session == NULL) {
        EShell *shell;
        EShellBackend *shell_backend;
        EMailBackend *backend;

        shell = e_shell_get_default ();
        shell_backend = e_shell_get_backend_by_name (shell, "mail");

        backend = E_MAIL_BACKEND (shell_backend);
        session = e_mail_backend_get_session (backend);
    }

    g_return_if_fail (E_IS_MAIL_SESSION (session));
    g_return_if_fail (rule->priv->session == NULL);

    rule->priv->session = g_object_ref (session);
}

static void
vfolder_editor_rule_set_property (GObject *object,
                           guint property_id,
                           const GValue *value,
                           GParamSpec *pspec)
{
    switch (property_id) {
        case PROP_SESSION:
            vfolder_editor_rule_set_session (
                EM_VFOLDER_EDITOR_RULE (object),
                g_value_get_object (value));
            return;
    }

    G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}

static void
vfolder_editor_rule_get_property (GObject *object,
                           guint property_id,
                           GValue *value,
                           GParamSpec *pspec)
{
    switch (property_id) {
        case PROP_SESSION:
            g_value_set_object (
                value,
                em_vfolder_editor_rule_get_session (
                EM_VFOLDER_EDITOR_RULE (object)));
            return;
    }

    G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}

static void
vfolder_editor_rule_dispose (GObject *object)
{
    EMVFolderEditorRulePrivate *priv;

    priv = EM_VFOLDER_EDITOR_RULE_GET_PRIVATE (object);
    if (priv->session != NULL) {
        g_object_unref (priv->session);
        priv->session = NULL;
    }

    /* Chain up to parent's dispose() method. */
    G_OBJECT_CLASS (em_vfolder_editor_rule_parent_class)->dispose (object);
}

static void
vfolder_editor_rule_finalize (GObject *object)
{
    /* EMVFolderEditorRule *rule = EM_VFOLDER_EDITOR_RULE (object); */

    /* Chain up to parent's finalize() method. */
    G_OBJECT_CLASS (em_vfolder_editor_rule_parent_class)->finalize (object);
}

static void
em_vfolder_editor_rule_class_init (EMVFolderEditorRuleClass *class)
{
    GObjectClass *object_class;
    EFilterRuleClass *filter_rule_class;

    g_type_class_add_private (class, sizeof (EMVFolderEditorRulePrivate));

    object_class = G_OBJECT_CLASS (class);
    object_class->set_property = vfolder_editor_rule_set_property;
    object_class->get_property = vfolder_editor_rule_get_property;
    object_class->dispose = vfolder_editor_rule_dispose;
    object_class->finalize = vfolder_editor_rule_finalize;

    filter_rule_class = E_FILTER_RULE_CLASS (class);
    filter_rule_class->get_widget = get_widget;

    g_object_class_install_property (
        object_class,
        PROP_SESSION,
        g_param_spec_object (
            "session",
            NULL,
            NULL,
            E_TYPE_MAIL_SESSION,
            G_PARAM_READWRITE |
            G_PARAM_CONSTRUCT_ONLY));
}

static void
em_vfolder_editor_rule_init (EMVFolderEditorRule *rule)
{
    rule->priv = EM_VFOLDER_EDITOR_RULE_GET_PRIVATE (rule);
}

EFilterRule *
em_vfolder_editor_rule_new (EMailSession *session)
{
    g_return_val_if_fail (E_IS_MAIL_SESSION (session), NULL);

    return g_object_new (
        EM_TYPE_VFOLDER_EDITOR_RULE, "session", session, NULL);
}

EMailSession *
em_vfolder_editor_rule_get_session (EMVFolderEditorRule *rule)
{
    g_return_val_if_fail (EM_IS_VFOLDER_RULE (rule), NULL);

    return rule->priv->session;
}

enum {
    BUTTON_ADD,
    BUTTON_REMOVE,
    BUTTON_LAST
};

struct _source_data {
    ERuleContext *rc;
    EMVFolderRule *vr;
    GtkListStore *model;
    GtkTreeView *tree_view;
    GtkWidget *source_selector;
    GtkWidget *buttons[BUTTON_LAST];
};

static void
set_sensitive (struct _source_data *data)
{
    GtkTreeSelection *selection;

    selection = gtk_tree_view_get_selection (data->tree_view);

    gtk_widget_set_sensitive (
        GTK_WIDGET (data->buttons[BUTTON_ADD]), TRUE);
    gtk_widget_set_sensitive (
        GTK_WIDGET (data->buttons[BUTTON_REMOVE]),
        selection && gtk_tree_selection_count_selected_rows (selection) > 0);
}

static void
selection_changed_cb (GtkTreeSelection *selection,
                      struct _source_data *data)
{
    set_sensitive (data);
}

static void
select_source_with_changed (GtkWidget *widget,
                            struct _source_data *data)
{
    em_vfolder_rule_with_t with;

    with = gtk_combo_box_get_active (GTK_COMBO_BOX (widget));
    if (with > EM_VFOLDER_RULE_WITH_LOCAL)
        with = 0;

    with = 3 - with;

    gtk_widget_set_sensitive (data->source_selector, !with);

    em_vfolder_rule_set_with (data->vr, with);
}

static void
autoupdate_toggled_cb (GtkToggleButton *toggle,
                       struct _source_data *data)
{
    em_vfolder_rule_set_autoupdate (data->vr, gtk_toggle_button_get_active (toggle));
}

static void
include_subfolders_toggled_cb (GtkCellRendererToggle *cell_renderer,
                               const gchar *path_string,
                               struct _source_data *data)
{
    GtkTreeModel *model;
    GtkTreePath *path;
    GtkTreeIter iter;

    gtk_cell_renderer_toggle_set_active (
        cell_renderer,
        !gtk_cell_renderer_toggle_get_active (cell_renderer));

    model = gtk_tree_view_get_model (data->tree_view);
    path = gtk_tree_path_new_from_string (path_string);

    if (gtk_tree_model_get_iter (model, &iter, path)) {
        gchar *source = NULL;

        gtk_list_store_set (
            GTK_LIST_STORE (model), &iter,
            2, gtk_cell_renderer_toggle_get_active (cell_renderer),
            -1);

        gtk_tree_model_get (model, &iter, 1, &source, -1);
        if (source) {
            em_vfolder_rule_source_set_include_subfolders (
                data->vr, source,
                gtk_cell_renderer_toggle_get_active (cell_renderer));
            g_free (source);
        }
    }

    gtk_tree_path_free (path);
}

static void
vfr_folder_response (EMFolderSelector *selector,
                     gint button,
                     struct _source_data *data)
{
    EMFolderTreeModel *model;
    EMFolderTree *folder_tree;
    CamelSession *session;
    GList *selected_uris;

    folder_tree = em_folder_selector_get_folder_tree (selector);
    model = em_folder_selector_get_model (selector);
    session = CAMEL_SESSION (em_folder_tree_model_get_session (model));

    selected_uris = em_folder_tree_get_selected_uris (folder_tree);

    if (button == GTK_RESPONSE_OK && selected_uris != NULL) {
        GList *uris_iter;
        GHashTable *known_uris;
        GtkTreeIter iter;
        GtkTreeSelection *selection;
        gboolean changed = FALSE;

        selection = gtk_tree_view_get_selection (data->tree_view);
        gtk_tree_selection_unselect_all (selection);

        known_uris = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, NULL);

        if (gtk_tree_model_get_iter_first (GTK_TREE_MODEL (data->model), &iter)) {
            GtkTreeModel *model = GTK_TREE_MODEL (data->model);
            do {
                gchar *known = NULL;

                gtk_tree_model_get (model, &iter, 1, &known, -1);

                if (known)
                    g_hash_table_insert (known_uris, known, GINT_TO_POINTER (1));
            } while (gtk_tree_model_iter_next (model, &iter));
        }

        for (uris_iter = selected_uris; uris_iter != NULL; uris_iter = uris_iter->next) {
            const gchar *uri = uris_iter->data;
            gchar *markup;

            if (!uri || g_hash_table_lookup (known_uris, uri))
                continue;

            g_hash_table_insert (known_uris, g_strdup (uri), GINT_TO_POINTER (1));

            changed = TRUE;
            g_queue_push_tail (em_vfolder_rule_get_sources (data->vr), g_strdup (uri));

            markup = e_mail_folder_uri_to_markup (session, uri, NULL);

            gtk_list_store_append (data->model, &iter);
            gtk_list_store_set (data->model, &iter, 0, markup, 1, uri, -1);
            g_free (markup);

            /* select all newly added folders */
            gtk_tree_selection_select_iter (selection, &iter);
        }

        g_hash_table_destroy (known_uris);
        if (changed)
            em_vfolder_rule_sources_changed (data->vr);

        set_sensitive (data);
    }

    gtk_widget_destroy (GTK_WIDGET (selector));
    g_list_free_full (selected_uris, g_free);
}

static void
source_add (GtkWidget *widget,
            struct _source_data *data)
{
    EMFolderTree *folder_tree;
    EMFolderTreeModel *model;
    GtkTreeSelection *selection;
    GtkWidget *dialog;
    gpointer parent;

    parent = gtk_widget_get_toplevel (widget);
    parent = gtk_widget_is_toplevel (parent) ? parent : NULL;

    model = em_folder_tree_model_get_default ();

    dialog = em_folder_selector_new (
        parent, model,
        EM_FOLDER_SELECTOR_CAN_CREATE,
        _("Add Folder"), NULL, _("_Add"));

    folder_tree = em_folder_selector_get_folder_tree (
        EM_FOLDER_SELECTOR (dialog));

    em_folder_tree_set_excluded (folder_tree, EMFT_EXCLUDE_NOSELECT);

    selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (folder_tree));
    gtk_tree_selection_set_mode (selection, GTK_SELECTION_MULTIPLE);

    g_signal_connect (
        dialog, "response",
        G_CALLBACK (vfr_folder_response), data);

    gtk_widget_show (dialog);
}

static void
source_remove (GtkWidget *widget,
               struct _source_data *data)
{
    GtkTreeSelection *selection;
    const gchar *source, *prev_source;
    GtkTreePath *path;
    GtkTreeIter iter;
    GHashTable *to_remove;
    gint index = 0, first_selected = -1, removed;
    gint n;

    selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (data->tree_view));
    to_remove = g_hash_table_new (g_direct_hash, g_direct_equal);

    source = NULL;
    while ((source = em_vfolder_rule_next_source (data->vr, source))) {
        path = gtk_tree_path_new ();
        gtk_tree_path_append_index (path, index);

        if (gtk_tree_selection_path_is_selected (selection, path)) {
            g_hash_table_insert (to_remove, GINT_TO_POINTER (index), GINT_TO_POINTER (1));

            if (first_selected == -1)
                first_selected = index;
        }

        index++;

        gtk_tree_path_free (path);
    }

    /* do not depend on selection when removing */
    gtk_tree_selection_unselect_all (selection);

    index = 0;
    source = NULL;
    removed = 0;
    prev_source = NULL;
    while ((source = em_vfolder_rule_next_source (data->vr, source))) {
        if (g_hash_table_lookup (to_remove, GINT_TO_POINTER (index + removed))) {
            path = gtk_tree_path_new ();
            gtk_tree_path_append_index (path, index);
            gtk_tree_model_get_iter (
                GTK_TREE_MODEL (data->model), &iter, path);

            em_vfolder_rule_remove_source (data->vr, source);
            gtk_list_store_remove (data->model, &iter);
            gtk_tree_path_free (path);

            /* try again from the previous source */
            removed++;
            source = prev_source;
        } else {
            index++;
            prev_source = source;
        }
    }

    g_hash_table_destroy (to_remove);

    /* now select the next rule */
    n = gtk_tree_model_iter_n_children (
        GTK_TREE_MODEL (data->model), NULL);
    index = first_selected >= n ? n - 1 : first_selected;

    if (index >= 0) {
        path = gtk_tree_path_new ();
        gtk_tree_path_append_index (path, index);
        if (gtk_tree_model_get_iter (GTK_TREE_MODEL (data->model), &iter, path)) {
            gtk_tree_selection_select_iter (selection, &iter);
            gtk_tree_view_set_cursor (data->tree_view, path, NULL, FALSE);
        }
        gtk_tree_path_free (path);
    }

    set_sensitive (data);
}

static GtkWidget *
get_widget (EFilterRule *fr,
            ERuleContext *rc)
{
    EMVFolderRule *vr = (EMVFolderRule *) fr;
    EMailSession *session;
    GtkWidget *widget, *frame, *label, *combobox, *hgrid, *vgrid, *tree_view, *scrolled_window;
    GtkWidget *autoupdate;
    GtkListStore *model;
    GtkCellRenderer *renderer;
    GtkTreeViewColumn *column;
    struct _source_data *data;
    const gchar *source;
    gchar *tmp;
    GtkTreeIter iter;
    GtkTreeSelection *selection;

    widget = E_FILTER_RULE_CLASS (em_vfolder_editor_rule_parent_class)->
        get_widget (fr, rc);

    data = g_malloc0 (sizeof (*data));
    data->rc = rc;
    data->vr = vr;

    frame = gtk_grid_new ();
    gtk_orientable_set_orientation (GTK_ORIENTABLE (frame), GTK_ORIENTATION_VERTICAL);
    gtk_grid_set_row_spacing (GTK_GRID (frame), 6);

    g_object_set_data_full (G_OBJECT (frame), "data", data, g_free);

    tmp = g_strdup_printf ("<b>%s</b>", _("Search Folder Sources"));
    label = gtk_label_new (tmp);
    g_free (tmp);
    g_object_set (
        G_OBJECT (label),
        "use-markup", TRUE,
        "xalign", 0.0,
        NULL);

    gtk_container_add (GTK_CONTAINER (frame), label);

    hgrid = gtk_grid_new ();
    gtk_orientable_set_orientation (GTK_ORIENTABLE (hgrid), GTK_ORIENTATION_HORIZONTAL);
    gtk_container_add (GTK_CONTAINER (frame), hgrid);

    label = gtk_label_new ("    ");
    gtk_container_add (GTK_CONTAINER (hgrid), label);

    vgrid = gtk_grid_new ();
    g_object_set (
        G_OBJECT (vgrid),
        "orientation", GTK_ORIENTATION_VERTICAL,
        "border-width", 6,
        "row-spacing", 6,
        NULL);
    gtk_container_add (GTK_CONTAINER (hgrid), vgrid);

    hgrid = gtk_grid_new ();
    gtk_orientable_set_orientation (GTK_ORIENTABLE (hgrid), GTK_ORIENTATION_HORIZONTAL);
    gtk_grid_set_column_spacing (GTK_GRID (hgrid), 6);
    gtk_container_add (GTK_CONTAINER (vgrid), hgrid);

    autoupdate = gtk_check_button_new_with_mnemonic (_("Automatically update on any _source folder change"));
    gtk_container_add (GTK_CONTAINER (hgrid), autoupdate);

    gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (autoupdate), em_vfolder_rule_get_autoupdate (vr));
    g_signal_connect (autoupdate, "toggled", G_CALLBACK (autoupdate_toggled_cb), data);

    hgrid = gtk_grid_new ();
    gtk_orientable_set_orientation (GTK_ORIENTABLE (hgrid), GTK_ORIENTATION_HORIZONTAL);
    gtk_grid_set_column_spacing (GTK_GRID (hgrid), 6);
    gtk_container_add (GTK_CONTAINER (vgrid), hgrid);

    combobox = gtk_combo_box_text_new ();
    gtk_combo_box_text_append (GTK_COMBO_BOX_TEXT (combobox), NULL, _("All local folders"));
    gtk_combo_box_text_append (GTK_COMBO_BOX_TEXT (combobox), NULL, _("All active remote folders"));
    gtk_combo_box_text_append (GTK_COMBO_BOX_TEXT (combobox), NULL, _("All local and active remote folders"));
    gtk_combo_box_text_append (GTK_COMBO_BOX_TEXT (combobox), NULL, _("Specific folders"));
    gtk_container_add (GTK_CONTAINER (hgrid), combobox);

    hgrid = gtk_grid_new ();
    gtk_orientable_set_orientation (GTK_ORIENTABLE (hgrid), GTK_ORIENTATION_HORIZONTAL);
    gtk_grid_set_column_spacing (GTK_GRID (hgrid), 6);
    gtk_container_add (GTK_CONTAINER (vgrid), hgrid);

    scrolled_window = gtk_scrolled_window_new (NULL, NULL);
    g_object_set (
        G_OBJECT (scrolled_window),
        "hscrollbar-policy", GTK_POLICY_AUTOMATIC,
        "vscrollbar-policy", GTK_POLICY_AUTOMATIC,
        "shadow-type", GTK_SHADOW_IN,
        "halign", GTK_ALIGN_FILL,
        "hexpand", TRUE,
        "valign", GTK_ALIGN_FILL,
        "vexpand", TRUE,
        NULL);
    gtk_container_add (GTK_CONTAINER (hgrid), scrolled_window);

    model = gtk_list_store_new (3, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_BOOLEAN);
    renderer = gtk_cell_renderer_text_new ();
    tree_view = gtk_tree_view_new_with_model (GTK_TREE_MODEL (model));
    gtk_tree_view_set_headers_visible (GTK_TREE_VIEW (tree_view), FALSE);
    gtk_tree_view_insert_column_with_attributes (
        GTK_TREE_VIEW (tree_view),
        -1, "column", renderer, "markup", 0, NULL);

    renderer = gtk_cell_renderer_toggle_new ();
    column = gtk_tree_view_column_new_with_attributes (
        "include subfolders", renderer, "active", 2, NULL);
    g_signal_connect (renderer, "toggled", G_CALLBACK (include_subfolders_toggled_cb), data);

    renderer = gtk_cell_renderer_text_new ();
    g_object_set (
        G_OBJECT (renderer),
        "editable", FALSE,
        "text", _("include subfolders"),
        NULL);
    gtk_tree_view_column_pack_start (column, renderer, TRUE);
    gtk_tree_view_insert_column (GTK_TREE_VIEW (tree_view), column, -1);

    column = gtk_tree_view_get_column (GTK_TREE_VIEW (tree_view), 0);
    gtk_tree_view_column_set_expand (column, TRUE);

    gtk_scrolled_window_add_with_viewport (GTK_SCROLLED_WINDOW (scrolled_window), tree_view);

    vgrid = gtk_grid_new ();
    g_object_set (
        G_OBJECT (vgrid),
        "orientation", GTK_ORIENTATION_VERTICAL,
        "border-width", 6,
        "row-spacing", 6,
        NULL);
    gtk_container_add (GTK_CONTAINER (hgrid), vgrid);

    data->buttons[BUTTON_ADD] = gtk_button_new_from_stock (GTK_STOCK_ADD);
    g_signal_connect (
        data->buttons[BUTTON_ADD], "clicked",
        G_CALLBACK (source_add), data);

    data->buttons[BUTTON_REMOVE] = gtk_button_new_from_stock (GTK_STOCK_REMOVE);
    g_signal_connect (
        data->buttons[BUTTON_REMOVE], "clicked",
        G_CALLBACK (source_remove), data);

    gtk_container_add (GTK_CONTAINER (vgrid), data->buttons[BUTTON_ADD]);
    gtk_container_add (GTK_CONTAINER (vgrid), data->buttons[BUTTON_REMOVE]);

    data->tree_view = GTK_TREE_VIEW (tree_view);
    data->model = model;

    session = em_vfolder_editor_context_get_session (EM_VFOLDER_EDITOR_CONTEXT (rc));

    source = NULL;
    while ((source = em_vfolder_rule_next_source (vr, source))) {
        gchar *markup;

        markup = e_mail_folder_uri_to_markup (
            CAMEL_SESSION (session), source, NULL);

        gtk_list_store_append (data->model, &iter);
        gtk_list_store_set (
            data->model, &iter,
            0, markup,
            1, source,
            2, em_vfolder_rule_source_get_include_subfolders (vr, source),
            -1);
        g_free (markup);
    }

    selection = gtk_tree_view_get_selection (data->tree_view);
    gtk_tree_selection_set_mode (selection, GTK_SELECTION_MULTIPLE);

    g_signal_connect (
        selection, "changed",
        G_CALLBACK (selection_changed_cb), data);

    data->source_selector = hgrid;

    gtk_combo_box_set_active (GTK_COMBO_BOX (combobox), 3 - em_vfolder_rule_get_with (vr));
    g_signal_connect (
        combobox, "changed",
        G_CALLBACK (select_source_with_changed), data);
    select_source_with_changed (combobox, data);

    set_sensitive (data);

    gtk_widget_set_valign (frame, GTK_ALIGN_FILL);
    gtk_widget_set_vexpand (frame, TRUE);
    gtk_widget_show_all (frame);

    gtk_container_add (GTK_CONTAINER (widget), frame);

    return widget;
}