[seahorse/wip/nielsdg/src-port: 31/31] Port /src to Vala.



commit c290bf53e06eac839987309ba26c75d053a243e2
Author: Niels De Graef <nielsdegraef gmail com>
Date:   Tue Mar 6 21:27:44 2018 +0100

    Port /src to Vala.
    
    Also move Application and SearchProvider to /src, where they belong
    better.

 common/catalog.vala                                |    3 -
 common/config.vapi                                 |   24 +-
 gkr/meson.build                                    |    5 +
 libseahorse/meson.build                            |    9 -
 libseahorse/seahorse-application.c                 |  248 ---
 libseahorse/seahorse-application.h                 |   49 -
 libseahorse/seahorse-search-provider.c             |  589 -------
 libseahorse/seahorse-search-provider.h             |   49 -
 libseahorse/seahorse-widget.c                      |    8 +-
 libseahorse/seahorse-widget.h                      |    2 -
 pkcs11/meson.build                                 |    5 +
 src/application.vala                               |  150 ++
 src/generate-select.vala                           |  140 ++
 src/import-dialog.vala                             |   78 +
 src/key-manager.vala                               |  502 ++++++
 src/main.vala                                      |   36 +
 src/meson.build                                    |   39 +-
 .../org.gnome.ShellSearchProvider2.xml             |    0
 src/seahorse-generate-select.c                     |  298 ----
 src/seahorse-generate-select.h                     |   33 -
 src/seahorse-import-dialog.c                       |  188 ---
 src/seahorse-import-dialog.h                       |   50 -
 src/seahorse-key-manager.c                         |  855 -----------
 src/seahorse-key-manager.h                         |   61 -
 src/seahorse-main.c                                |  100 --
 src/seahorse-sidebar.c                             | 1598 --------------------
 src/seahorse-sidebar.h                             |   61 -
 src/search-provider.vala                           |  264 ++++
 src/sidebar.vala                                   |  915 +++++++++++
 ssh/meson.build                                    |    1 +
 30 files changed, 2144 insertions(+), 4216 deletions(-)
---
diff --git a/common/catalog.vala b/common/catalog.vala
index 394adc0..deef377 100644
--- a/common/catalog.vala
+++ b/common/catalog.vala
@@ -107,8 +107,6 @@ public abstract class Catalog : Gtk.Window {
                this._edit_copy = actions.get_action("edit-export-clipboard");
                this._file_export = actions.get_action("file-export");
                this._ui_manager.insert_action_group (actions, 0);
-
-               Seahorse.Application.get().add_window(this);
        }
 
        public override void dispose() {
@@ -131,7 +129,6 @@ public abstract class Catalog : Gtk.Window {
                        this.get_size(out width, out height);
                        this._settings.set_int("width", width);
                        this._settings.set_int("height", height);
-                       Seahorse.Application.get().remove_window (this);
                }
 
                base.dispose();
diff --git a/common/config.vapi b/common/config.vapi
index 02ccb37..d5cda88 100644
--- a/common/config.vapi
+++ b/common/config.vapi
@@ -4,13 +4,20 @@ namespace Config
        public const string PKGDATADIR;
 
        public const string EXECDIR;
+       public const string LOCALEDIR;
 
        public const string VERSION;
        public const string PACKAGE;
+       public const string PACKAGE_STRING;
        public const string GETTEXT_PACKAGE;
 
        public const string SSH_PATH;
        public const string SSH_KEYGEN_PATH;
+
+       public const string GNUPG;
+       public const int GPG_MAJOR;
+       public const int GPG_MINOR;
+       public const int GPG_MICRO;
 }
 
 /*
@@ -21,10 +28,8 @@ namespace Config
 
 namespace Seahorse {
 
-[CCode (cheader_filename = "libseahorse/seahorse-application.h")]
-namespace Application {
-       public unowned Gtk.Application @get();
-}
+[CCode (cheader_filename = "data/seahorse-resources.h")]
+public void register_resource();
 
 [CCode (cheader_filename = "libseahorse/seahorse-util.h")]
 public static GLib.HashFunc<ulong?> ulong_hash;
@@ -43,10 +48,15 @@ namespace Progress {
        public void show(GLib.Cancellable? cancellable, string title, bool delayed);
 }
 
+[CCode (cheader_filename = "pgp/seahorse-pgp-backend.h")]
+namespace Pgp.Backend {
+       public void initialize();
+}
 }
 
 namespace Egg {
-       namespace TreeMultiDrag {
-               public void add_drag_support(Gtk.TreeView view);
-       }
+[CCode (cheader_filename = "libegg/eggtreemultidnd.h")]
+namespace TreeMultiDrag {
+       public void add_drag_support(Gtk.TreeView view);
+}
 }
diff --git a/gkr/meson.build b/gkr/meson.build
index 10b0e11..2c61028 100644
--- a/gkr/meson.build
+++ b/gkr/meson.build
@@ -23,3 +23,8 @@ gkr_lib = static_library('seahorse-gkr',
   gkr_sources,
   dependencies: gkr_dependencies,
 )
+
+gkr_dep = declare_dependency(
+  link_with: gkr_lib,
+  include_directories: include_directories('.'),
+)
diff --git a/libseahorse/meson.build b/libseahorse/meson.build
index c1a7bbc..b1e8045 100644
--- a/libseahorse/meson.build
+++ b/libseahorse/meson.build
@@ -3,27 +3,18 @@ marshaller = gnome.genmarshal('seahorse-marshal',
   prefix: 'seahorse_marshal',
 )
 
-search_provider_src = gnome.gdbus_codegen('seahorse-shell-search-provider-generated',
-  'org.gnome.ShellSearchProvider2.xml',
-  interface_prefix : 'org.gnome.',
-  namespace : 'Seahorse',
-)
-
 
 libseahorse_sources = [
-  'seahorse-application.c',
   'seahorse-bind.c',
   'seahorse-interaction.c',
   'seahorse-object-list.c',
   'seahorse-object-model.c',
   'seahorse-object-widget.c',
   'seahorse-progress.c',
-  'seahorse-search-provider.c',
   'seahorse-util.c',
   'seahorse-widget.c',
   marshaller,
   resources_src,
-  search_provider_src,
 ]
 
 libseahorse_deps = [
diff --git a/libseahorse/seahorse-widget.c b/libseahorse/seahorse-widget.c
index 6dc7e61..3b1bd2a 100644
--- a/libseahorse/seahorse-widget.c
+++ b/libseahorse/seahorse-widget.c
@@ -135,8 +135,8 @@ seahorse_widget_constructed (GObject *object)
                widgets = g_hash_table_new ((GHashFunc)g_str_hash, (GCompareFunc)g_str_equal);
        g_hash_table_insert (widgets, g_strdup (self->name), self);
 
-       gtk_application_add_window (seahorse_application_get (),
-                                   GTK_WINDOW (seahorse_widget_get_widget (self, self->name)));
+       /* gtk_application_add_window (seahorse_application_get (), */
+       /*                             GTK_WINDOW (seahorse_widget_get_widget (self, self->name))); */
 }
 
 /**
@@ -224,8 +224,8 @@ object_finalize (GObject *gobject)
        }
     }
 
-       gtk_application_remove_window (seahorse_application_get (),
-                                      GTK_WINDOW (seahorse_widget_get_widget (swidget, swidget->name)));
+       /* gtk_application_remove_window (seahorse_application_get (), */
+       /*                                GTK_WINDOW (seahorse_widget_get_widget (swidget, swidget->name))); 
*/
 
     if (seahorse_widget_get_widget (swidget, swidget->name)) {
         gtk_widget_destroy (GTK_WIDGET (seahorse_widget_get_widget (swidget, swidget->name)));
diff --git a/libseahorse/seahorse-widget.h b/libseahorse/seahorse-widget.h
index 068c104..f4ab2c5 100644
--- a/libseahorse/seahorse-widget.h
+++ b/libseahorse/seahorse-widget.h
@@ -24,8 +24,6 @@
 #include <glib.h>
 #include <gtk/gtk.h>
 
-#include "seahorse-application.h"
-
 #define SEAHORSE_TYPE_WIDGET            (seahorse_widget_get_type ())
 #define SEAHORSE_WIDGET(obj)            (G_TYPE_CHECK_INSTANCE_CAST ((obj), SEAHORSE_TYPE_WIDGET, 
SeahorseWidget))
 #define SEAHORSE_WIDGET_CLASS(klass)    (G_TYPE_CHECK_CLASS_CAST ((klass), SEAHORSE_TYPE_WIDGET, 
SeahorseWidgetClass))
diff --git a/pkcs11/meson.build b/pkcs11/meson.build
index d5e6518..760b7bf 100644
--- a/pkcs11/meson.build
+++ b/pkcs11/meson.build
@@ -24,3 +24,8 @@ pkcs11_lib = static_library('seahorse-pkcs11',
   pkcs11_sources,
   dependencies: pkcs11_deps,
 )
+
+pkcs11_dep = declare_dependency(
+  link_with: pkcs11_lib,
+  include_directories: include_directories('.'),
+)
diff --git a/src/application.vala b/src/application.vala
new file mode 100644
index 0000000..76a4656
--- /dev/null
+++ b/src/application.vala
@@ -0,0 +1,150 @@
+/*
+ * Seahorse
+ *
+ * Copyright (C) 2003 Jacob Perkins
+ * Copyright (C) 2005, 2006 Stefan Walter
+ * Copyright (C) 2011 Collabora Ltd.
+ * Copyright (C) 2012 Stefan Walter
+ * Copyright (C) 2018 Niels De Graef
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * 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, see
+ * <http://www.gnu.org/licenses/>.
+ */
+
+extern void seahorse_pkcs11_backend_initialize();
+
+public class Seahorse.Application : Gtk.Application {
+    /* private SearchProvider? search_provider; */
+
+    private const int INACTIVITY_TIMEOUT = 120 * 1000; // Two minutes, in milliseconds
+
+    public Application () {
+        GLib.Object (
+            application_id: "org.gnome.seahorse.Application",
+            flags: ApplicationFlags.HANDLES_COMMAND_LINE
+        );
+
+        /* this.search_provider = new SearchProvider(); */
+    }
+
+    public override void startup() {
+               base.startup();
+
+        // Insert Icons into Stock
+        icons_init ();
+
+        // Initialize the backends
+        Gkr.Backend.initialize();
+        Ssh.Backend.initialize();
+#if WITH_PGP
+        Pgp.Backend.initialize();
+#endif
+#if WITH_PKCS11
+        seahorse_pkcs11_backend_initialize();
+#endif
+    }
+
+    public override void activate() {
+               base.activate();
+
+        KeyManager key_mgr = new Seahorse.KeyManager(this);
+        key_mgr.show();
+    }
+
+    static bool show_version = false;
+    const OptionEntry[] local_options = {
+        { "version", 'v', 0, OptionArg.NONE, out show_version, N_("Version of this application"), null },
+        { null }
+    };
+
+    public override bool local_command_line (ref weak string[] arguments, out int exit_status) {
+        OptionContext context = new OptionContext(N_("- System Settings"));
+        context.set_ignore_unknown_options(true);
+        context.add_main_entries(local_options, Config.GETTEXT_PACKAGE);
+        context.set_translation_domain(Config.GETTEXT_PACKAGE);
+        context.add_group(Gtk.get_option_group (true));
+
+        try {
+            unowned string[] tmp = arguments;
+            context.parse (ref tmp);
+        } catch (Error e) {
+            printerr ("seahorse: %s\n", e.message);
+            exit_status = 1;
+            return true;
+        }
+
+        if (show_version) {
+            print ("%s\n", Config.PACKAGE_STRING);
+#if WITH_PGP
+            print ("GNUPG: %s (%d.%d.%d)\n", Config.GNUPG, Config.GPG_MAJOR, Config.GPG_MINOR, 
Config.GPG_MICRO);
+#endif
+            exit_status = 0;
+            return true;
+        }
+
+        exit_status = 0;
+        /* return base.local_command_line(ref arguments, out exit_status); */
+        return false;
+    }
+
+    static bool no_window = false;
+    const OptionEntry[] options = {
+        { "no-window", 0, 0, OptionArg.NONE, out no_window, N_("Don't display a window"), null },
+        { null }
+    };
+
+    public override int command_line (ApplicationCommandLine command_line) {
+        OptionContext context = new OptionContext(N_("- System Settings"));
+        context.set_ignore_unknown_options (true);
+        context.add_main_entries (options, Config.GETTEXT_PACKAGE);
+        context.set_translation_domain(Config.GETTEXT_PACKAGE);
+
+        string[] arguments = command_line.get_arguments();
+        try {
+            unowned string[] tmp = arguments;
+            context.parse (ref tmp);
+        } catch (Error e) {
+            printerr ("seahorse: %s\n", e.message);
+            return 1;
+        }
+
+        if (no_window) {
+            hold();
+            set_inactivity_timeout(INACTIVITY_TIMEOUT);
+            release();
+            return 0;
+        }
+
+        activate ();
+        return 0;
+    }
+
+    public override bool dbus_register (DBusConnection connection, string object_path) throws Error {
+        if (!base.dbus_register(connection, object_path))
+            return false;
+
+        return true;
+        /* return this.search_provider.dbus_register (connection, object_path); */
+    }
+
+    public override void dbus_unregister (DBusConnection connection, string object_path) {
+        /* if (this.search_provider != null) */
+        /*     this.search_provider.dbus_unregister(connection, object_path); */
+
+        base.dbus_unregister(connection, object_path);
+    }
+
+    public void initialize_search () {
+        /* this.search_provider.initialize(); */
+    }
+}
diff --git a/src/generate-select.vala b/src/generate-select.vala
new file mode 100644
index 0000000..1652c4f
--- /dev/null
+++ b/src/generate-select.vala
@@ -0,0 +1,140 @@
+/*
+ * Seahorse
+ *
+ * Copyright (C) 2008 Stefan Walter
+ *
+ * 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.1 of
+ * the License, or (at your option) any later version.
+ *
+ * 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 this program; if not, see
+ * <http://www.gnu.org/licenses/>.
+ */
+
+public class Seahorse.GenerateSelect : Gtk.Dialog {
+    private Gtk.ListStore store;
+    private Gtk.TreeView view;
+    private List<Gtk.ActionGroup>? action_groups;
+
+    private enum Column {
+        ICON,
+        TEXT,
+        ACTION,
+        N_COLUMNS
+    }
+
+    public GenerateSelect(Gtk.Window? parent) {
+        GLib.Object(
+            transient_for: parent,
+            modal: true
+        );
+        this.store = new Gtk.ListStore(Column.N_COLUMNS, typeof(Icon), typeof(string), typeof(Gtk.Action));
+        this.store.set_default_sort_func(on_list_sort);
+        this.store.set_sort_column_id(Gtk.TREE_SORTABLE_DEFAULT_SORT_COLUMN_ID, Gtk.SortType.ASCENDING);
+
+        this.action_groups = (List<Gtk.ActionGroup>) Registry.object_instances("generator");
+        foreach (Gtk.ActionGroup action_group in this.action_groups) {
+            foreach (weak Gtk.Action action in action_group.list_actions()) {
+                string text = "<span size=\"larger\" weight=\"bold\">%s</span>\n%s"
+                                  .printf(action.label, action.tooltip);
+
+                Icon? icon = action.gicon;
+                if (icon == null) {
+                    if (action.icon_name != null)
+                        icon = new ThemedIcon(action.icon_name);
+                }
+
+                Gtk.TreeIter iter;
+                this.store.append(out iter);
+                this.store.set(iter, Column.TEXT, text,
+                                     Column.ICON, icon,
+                                     Column.ACTION, action,
+                                     -1);
+            }
+        }
+
+        Gtk.Builder builder = new Gtk.Builder();
+        try {
+            string path = "/org/gnome/Seahorse/seahorse-ssh-generate.ui";
+            builder.add_from_resource(path);
+        } catch (GLib.Error err) {
+            critical("%s", err.message);
+        }
+
+        // Setup the dialog
+        set_default_size(-1, 410);
+        get_content_area().pack_start((Gtk.Widget) builder.get_object("generate-select"),
+                                        true, true, 0);
+        add_buttons(Gtk.Stock.CANCEL, Gtk.ResponseType.CANCEL,
+                    _("Continue"), Gtk.ResponseType.OK,
+                    null);
+
+        // Hook it into the view
+        this.view = (Gtk.TreeView) builder.get_object("keytype-tree");
+
+        Gtk.CellRendererPixbuf pixcell = new Gtk.CellRendererPixbuf();
+        pixcell.stock_size = Gtk.IconSize.DND;
+        this.view.insert_column_with_attributes(-1, "", pixcell, "gicon", Column.ICON, null);
+        this.view.insert_column_with_attributes(-1, "", new Gtk.CellRendererText(), "markup", Column.TEXT, 
null);
+        this.view.set_model(this.store);
+
+        // Setup selection, select first item
+        Gtk.TreeSelection selection = this.view.get_selection();
+        selection.set_mode(Gtk.SelectionMode.BROWSE);
+
+        Gtk.TreeIter iter;
+        this.store.get_iter_first(out iter);
+        selection.select_iter(iter);
+
+        this.view.row_activated.connect(on_row_activated);
+        this.view.height_request = 410;
+    }
+
+    private Gtk.Action? get_selected_action() {
+        Gtk.TreeSelection selection = this.view.get_selection();
+
+        Gtk.TreeIter iter;
+        Gtk.TreeModel? model;
+        if (!selection.get_selected(out model, out iter))
+            return null;
+
+        Gtk.Action? action;
+        this.store.get(iter, Column.ACTION, out action, -1);
+        assert (action != null);
+
+        return action;
+    }
+
+    private void on_row_activated(Gtk.TreeView view, Gtk.TreePath path, Gtk.TreeViewColumn col) {
+        Gtk.Action? action = get_selected_action();
+        if (action != null) {
+            Action.activate_with_window(action, null, this.transient_for);
+            destroy();
+        }
+    }
+
+    public override void response(int response)  {
+        Gtk.Action? action = (response == Gtk.ResponseType.OK)? get_selected_action() : null;
+        Gtk.Window? parent = (action != null)? this.transient_for : null;
+
+        if (action != null)
+            Action.activate_with_window(action, null, parent);
+
+        destroy();
+    }
+
+    private int on_list_sort (Gtk.TreeModel? model, Gtk.TreeIter a, Gtk.TreeIter b) {
+        string? a_text = null, b_text = null;
+        model.get(a, Column.TEXT, out a_text, -1);
+        model.get(b, Column.TEXT, out b_text, -1);
+
+        return a_text.collate(b_text);
+    }
+}
diff --git a/src/import-dialog.vala b/src/import-dialog.vala
new file mode 100644
index 0000000..9d228a5
--- /dev/null
+++ b/src/import-dialog.vala
@@ -0,0 +1,78 @@
+/*
+ * Seahorse
+ *
+ * Copyright (C) 2011 Collabora Ltd.
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * 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, see
+ * <http://www.gnu.org/licenses/>.
+ *
+ * Author: Stef Walter <stefw collabora co uk>
+ */
+
+public class Seahorse.ImportDialog : Gtk.Dialog {
+
+    private Gcr.ViewerWidget viewer;
+    private Gcr.ImportButton import;
+
+    public ImportDialog(Gtk.Window? parent) {
+        GLib.Object(transient_for: parent);
+
+        Gtk.Widget button = new Gtk.Button.from_stock(Gtk.Stock.CANCEL);
+        button.show();
+        add_action_widget(button, Gtk.ResponseType.CANCEL);
+
+        this.import = new Gcr.ImportButton(_("Import"));
+        this.import.importing.connect(() => this.viewer.clear_error());
+        this.import.imported.connect(on_import_button_imported);
+        this.import.show();
+        ((Gtk.Box) get_action_area()).pack_start(this.import, false, true, 0);
+
+        Gtk.Frame frame = new Gtk.Frame(_("<b>Data to be imported:</b>"));
+        ((Gtk.Label) frame.label_widget).use_markup = true;
+        ((Gtk.Box) get_content_area()).pack_start(frame, true, true, 0);
+        frame.set_border_width(6);
+        frame.show();
+
+        this.viewer = new Gcr.ViewerWidget();
+        this.viewer.added.connect((v, r, parsed) => this.import.add_parsed(parsed));
+        this.viewer.show();
+
+        frame.add(this.viewer);
+    }
+
+    public void add_uris(string[] uris) {
+        foreach (string uri in uris)
+            this.viewer.load_file(File.new_for_uri(uri));
+    }
+
+    public void add_text(string? display_name, string text) {
+        this.viewer.load_data(display_name, (uint8[]) text);
+    }
+
+    private void on_import_button_imported(GLib.Object importer, Error? error) {
+        if (error == null) {
+            response(Gtk.ResponseType.OK);
+
+            string uri = ((Gcr.Importer) importer).uri;
+            foreach (Backend backend in Backend.get_registered()) {
+                Place? place = backend.lookup_place(uri);
+                if (place != null)
+                    place.load.begin(null);
+            }
+
+        } else {
+            if (!(error is GLib.IOError.CANCELLED))
+                this.viewer.show_error(_("Import failed"), error);
+        }
+    }
+}
diff --git a/src/key-manager.vala b/src/key-manager.vala
new file mode 100644
index 0000000..4fb229e
--- /dev/null
+++ b/src/key-manager.vala
@@ -0,0 +1,502 @@
+/*
+ * Seahorse
+ *
+ * Copyright (C) 2008 Stefan Walter
+ * Copyright (C) 2011 Collabora Ltd.
+ * Copyright (C) 2017 Niels De Graef
+ *
+ * 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.1 of
+ * the License, or (at your option) any later version.
+ *
+ * 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 this program; if not, see
+ * <http://www.gnu.org/licenses/>.
+ */
+
+public class Seahorse.KeyManager : Catalog {
+
+    private Gtk.ActionGroup view_actions;
+    private Gtk.RadioAction show_action;
+    private Gtk.Entry filter_entry;
+    private Predicate pred;
+    private Sidebar sidebar;
+
+    private Gtk.TreeView view;
+    private Gcr.Collection collection; // owned by the sidebar
+    private KeyManagerStore store;
+
+    private GLib.Settings settings;
+    private int sidebar_width;
+    private uint sidebar_width_sig;
+
+    private enum ShowFilter {
+        ANY,
+        PERSONAL,
+        TRUSTED,
+    }
+
+    private enum DndTarget { // Drag 'n Drop target type
+        PLAIN,
+        URIS
+    }
+
+    private const Gtk.ActionEntry[] GENERAL_ACTIONS = {
+        // TRANSLATORS: The "Remote" menu contains key operations on remote systems.
+        { "remote-menu", null, N_("_Remote") },
+        { "new-menu", null, N_("_New") },
+        { "app-quit", Gtk.Stock.QUIT, null, "<control>Q", N_("Close this program"), on_app_quit },
+        { "file-new", Gtk.Stock.NEW, N_("_New…"), "<control>N", N_("Create a new key or item"), on_file_new 
},
+        { "new-object", Gtk.Stock.ADD, N_("_New…"), null, N_("Add a new key or item"), on_file_new },
+        { "file-import", Gtk.Stock.OPEN, N_("_Import…"), "<control>I", N_("Import from a file"), 
on_key_import_file },
+        { "edit-import-clipboard", Gtk.Stock.PASTE, null, "<control>V", N_("Import from the clipboard"), 
on_key_import_clipboard }
+    };
+
+    private const Gtk.ToggleActionEntry[] SIDEBAR_ACTIONS = {
+        { "view-sidebar", null, N_("By _Keyring"), null, N_("Show sidebar listing keyrings"), null, false },
+    };
+
+    private const Gtk.RadioActionEntry[] VIEW_RADIO_ACTIONS = {
+        { "view-personal", null, N_("Show _Personal"), null, N_("Only show personal keys, certificates and 
passwords"), ShowFilter.PERSONAL },
+        { "view-trusted", null, N_("Show _Trusted"), null, N_("Only show trusted keys, certificates and 
passwords"), ShowFilter.TRUSTED },
+        { "view-any", null, N_("Show _Any"), null, N_("Show all keys, certificates and passwords"), 
ShowFilter.ANY },
+    };
+
+    public KeyManager(Application app) {
+        GLib.Object(
+            ui_name: "key-manager",
+            application: app
+        );
+        this.settings = new GLib.Settings("org.gnome.seahorse.manager");
+
+        Gtk.Window window = this.window;
+        window.set_default_geometry(640, 476);
+        window.set_events(Gdk.EventMask.POINTER_MOTION_MASK
+                          | Gdk.EventMask.POINTER_MOTION_HINT_MASK
+                          | Gdk.EventMask.BUTTON_PRESS_MASK
+                          | Gdk.EventMask.BUTTON_RELEASE_MASK);
+        window.set_title(_("Passwords and Keys"));
+
+        this.collection = setup_sidebar();
+
+        // Init key list & selection settings
+        Gtk.Builder builder = get_builder();
+        this.view = (Gtk.TreeView) builder.get_object("key-list");
+        assert (this.view != null);
+
+        Gtk.TreeSelection selection = this.view.get_selection();
+        selection.set_mode(Gtk.SelectionMode.MULTIPLE);
+        selection.changed.connect(on_view_selection_changed);
+        this.view.realize();
+
+        // Add new key store and associate it
+        this.store = new KeyManagerStore(this.collection, this.view, this.pred, this.settings);
+
+        Gtk.ActionGroup actions = new Gtk.ActionGroup("general");
+        actions.set_translation_domain(Config.GETTEXT_PACKAGE);
+        actions.add_actions(GENERAL_ACTIONS, null);
+        include_actions(actions);
+
+        this.view_actions = new Gtk.ActionGroup("view");
+        this.view_actions.set_translation_domain(Config.GETTEXT_PACKAGE);
+        this.view_actions.add_radio_actions(VIEW_RADIO_ACTIONS, -1, () => {
+            this.settings.set_string("item-filter", update_view_filter());
+        });
+        Gtk.RadioAction action = (Gtk.RadioAction) this.view_actions.get_action("view-personal");
+        include_actions(this.view_actions);
+        this.show_action = action;
+
+        // Notify us when settings change
+        this.settings.changed["item-filter"].connect(on_item_filter_changed);
+        on_item_filter_changed(this.settings, "item-filter");
+
+        // first time signals
+        ((Gtk.Button) builder.get_object("import-button")).clicked.connect(on_keymanager_import_button);
+        ((Gtk.Button) builder.get_object("new-button")).clicked.connect(on_keymanager_new_button);
+
+        // Make sure import is only available with clipboard content
+        Gtk.Action import_clipboard_action = actions.get_action("edit-import-clipboard");
+        Gtk.Clipboard clipboard = Gtk.Clipboard.get(Gdk.SELECTION_PRIMARY);
+        clipboard.owner_change.connect((c, e) => update_clipboard_state(c, e, actions));
+        update_clipboard_state(clipboard, null, actions);
+
+        // Flush all updates
+        ensure_updated();
+
+        // Find the toolbar
+        Gtk.Container placeholder = (Gtk.Container) builder.get_object("toolbar-placeholder");
+        if (placeholder != null) {
+            List<weak Gtk.Widget> children = placeholder.get_children();
+            if (children != null && children.data != null) {
+                // The toolbar is the first (and only) element
+                Gtk.Toolbar toolbar = (Gtk.Toolbar) children.data;
+                if (toolbar != null) {
+                    toolbar.get_style_context().add_class("sidebar");
+                    toolbar.reset_style();
+
+                    // Insert a separator to right align the filter
+                    Gtk.SeparatorToolItem sep = new Gtk.SeparatorToolItem();
+                    sep.set_draw(false);
+                    sep.set_expand(true);
+                    sep.show_all();
+                    toolbar.insert(sep, -1);
+
+                    // Insert a filter bar
+                    Gtk.Box box = new Gtk.Box(Gtk.Orientation.HORIZONTAL, 0);
+
+                    this.filter_entry = new Gtk.Entry();
+                    this.filter_entry.set_placeholder_text(_("Filter"));
+                    box.pack_start(this.filter_entry, false, true, 0);
+
+                    box.pack_start(new Gtk.Label(null), false, false, 0);
+                    box.show_all();
+
+                    Gtk.ToolItem item = new Gtk.ToolItem();
+                    item.add(box);
+                    item.show_all();
+                    toolbar.insert(item, -1);
+                }
+            }
+        }
+
+        on_filter_changed(this.filter_entry);
+        this.filter_entry.set_width_chars(30);
+        this.filter_entry.icon_release.connect(() => this.filter_entry.set_text(""));
+
+        // For the filtering
+        this.filter_entry.changed.connect(on_filter_changed);
+        this.view.start_interactive_search.connect(() => {
+            this.filter_entry.grab_focus();
+            return false;
+        });
+
+        // Set focus to the current key list
+        this.view.grab_focus();
+        selection_changed();
+
+        // To avoid flicker
+        show();
+
+        // Setup drops
+        Gtk.drag_dest_set(window, Gtk.DestDefaults.ALL, {}, Gdk.DragAction.COPY);
+        Gtk.TargetList targets = new Gtk.TargetList(null);
+        targets.add_uri_targets(DndTarget.URIS);
+        targets.add_text_targets(DndTarget.PLAIN);
+        Gtk.drag_dest_set_target_list(window, targets);
+
+        window.drag_data_received.connect(on_target_drag_data_received);
+
+        this.view.button_press_event.connect(on_keymanager_key_list_button_pressed);
+        this.view.row_activated.connect(on_keymanager_row_activated);
+        this.view.popup_menu.connect(on_keymanager_key_list_popup_menu);
+
+#if REFACTOR_FIRST
+        // To show first time dialog
+        Timeout.add_seconds(1, on_first_timer);
+#endif
+    }
+
+    ~KeyManager() {
+        if (this.sidebar_width_sig != 0) {
+            Source.remove(this.sidebar_width_sig);
+            this.sidebar_width_sig = 0;
+        }
+    }
+
+    private void on_view_selection_changed(Gtk.TreeSelection selection) {
+        Idle.add(() => {
+            // Fire the signal
+            selection_changed();
+            // Return false so we don't get run again
+            return false;
+        });
+    }
+
+    private void on_keymanager_row_activated(Gtk.TreeView view, Gtk.TreePath? path, Gtk.TreeViewColumn 
column) {
+        if (path == null)
+            return;
+
+        GLib.Object obj = KeyManagerStore.get_object_from_path(view, path);
+        if (obj != null)
+            show_properties(obj);
+    }
+
+    private bool on_keymanager_key_list_button_pressed(Gdk.EventButton event) {
+        if (event.button == 3)
+            show_context_menu(Catalog.MENU_OBJECT, event.button, event.time);
+
+        return false;
+    }
+
+    private bool on_keymanager_key_list_popup_menu() {
+        GLib.List<weak GLib.Object> objects = get_selected_objects();
+        if (objects != null)
+            show_context_menu(Catalog.MENU_OBJECT, 0, Gtk.get_current_event_time());
+        return false;
+    }
+
+    private void on_file_new(Gtk.Action action) {
+        GenerateSelect dialog = new GenerateSelect(this.window);
+        dialog.run();
+        dialog.destroy();
+    }
+
+    private void on_keymanager_new_button(Gtk.Button button) {
+        GenerateSelect dialog = new GenerateSelect(this.window);
+        dialog.run();
+        dialog.destroy();
+    }
+
+#if REFACTOR_FIRST
+    private bool on_first_timer() {
+        // Although not all the keys have completed we'll know whether we have
+        // any or not at this point
+        if (this.collection.get_length() == 0) {
+            Gtk.Widget widget = xxwidget_get_widget (XX_WIDGET (self), "first-time-box");
+            widget.show();
+        }
+
+        return false;
+    }
+#endif
+
+    private void on_filter_changed(Gtk.Editable entry) {
+        string? text = this.filter_entry.get_text();
+        this.store.filter = text;
+
+        if (text == null || text == "") {
+            this.filter_entry.secondary_icon_name = "edit-find-symbolic";
+            this.filter_entry.secondary_icon_activatable = false;
+            this.filter_entry.secondary_icon_sensitive = false;
+        } else {
+            this.filter_entry.secondary_icon_name = "edit-clear-symbolic";
+            this.filter_entry.secondary_icon_activatable = true;
+            this.filter_entry.secondary_icon_sensitive = true;
+        }
+    }
+
+    private void import_files(string[]? uris) {
+        ImportDialog dialog = new ImportDialog(this.window);
+        dialog.add_uris(uris);
+        dialog.run();
+        dialog.destroy();
+    }
+
+    private void import_prompt() {
+        Gtk.FileChooserDialog dialog =
+            new Gtk.FileChooserDialog(_("Import Key"), this.window,
+                                      Gtk.FileChooserAction.OPEN,
+                                      _("_Cancel"), Gtk.ResponseType.CANCEL,
+                                      _("_Open"), Gtk.ResponseType.ACCEPT,
+                                      null);
+
+        dialog.set_default_response(Gtk.ResponseType.ACCEPT);
+        dialog.set_local_only(false);
+
+        // TODO: This should come from libgcr somehow
+        Gtk.FileFilter filter = new Gtk.FileFilter();
+        filter.set_name(_("All key files"));
+        filter.add_mime_type("application/pgp-keys");
+        filter.add_mime_type("application/x-ssh-key");
+        filter.add_mime_type("application/pkcs12");
+        filter.add_mime_type("application/pkcs12+pem");
+        filter.add_mime_type("application/pkcs7-mime");
+        filter.add_mime_type("application/pkcs7-mime+pem");
+        filter.add_mime_type("application/pkcs8");
+        filter.add_mime_type("application/pkcs8+pem");
+        filter.add_mime_type("application/pkix-cert");
+        filter.add_mime_type("application/pkix-cert+pem");
+        filter.add_mime_type("application/pkix-crl");
+        filter.add_mime_type("application/pkix-crl+pem");
+        filter.add_mime_type("application/x-pem-file");
+        filter.add_mime_type("application/x-pem-key");
+        filter.add_mime_type("application/x-pkcs12");
+        filter.add_mime_type("application/x-pkcs7-certificates");
+        filter.add_mime_type("application/x-x509-ca-cert");
+        filter.add_mime_type("application/x-x509-user-cert");
+        filter.add_mime_type("application/pkcs10");
+        filter.add_mime_type("application/pkcs10+pem");
+        filter.add_mime_type("application/x-spkac");
+        filter.add_mime_type("application/x-spkac+base64");
+        filter.add_pattern("*.asc");
+        filter.add_pattern("*.gpg");
+        filter.add_pattern("*.pub");
+        filter.add_pattern("*.pfx");
+        filter.add_pattern("*.cer");
+        filter.add_pattern("*.crt");
+        filter.add_pattern("*.pem");
+        dialog.add_filter(filter);
+        dialog.set_filter(filter);
+
+        filter = new Gtk.FileFilter();
+        filter.set_name(_("All files"));
+        filter.add_pattern("*");
+        dialog.add_filter(filter);
+
+        string? uri = null;
+        if (dialog.run() == Gtk.ResponseType.ACCEPT) {
+            uri = dialog.get_uri();
+        }
+
+        dialog.destroy();
+
+        if (uri != null) {
+            string? uris[2];
+            uris[0] = uri;
+            uris[1] = null;
+            import_files(uris);
+        }
+    }
+
+    private void on_key_import_file(Gtk.Action action) {
+        import_prompt();
+    }
+
+    private void on_keymanager_import_button(Gtk.Button button) {
+        import_prompt();
+    }
+
+    private void import_text(string? display_name, string? text) {
+        ImportDialog dialog = new ImportDialog(this.window);
+        dialog.add_text(display_name, text);
+        dialog.run();
+        dialog.destroy();
+    }
+
+    private void on_target_drag_data_received(Gdk.DragContext context, int x, int y,
+                                              Gtk.SelectionData? selection_data, uint info, uint time) {
+        if (selection_data == null)
+            return;
+
+        if (info == DndTarget.PLAIN) {
+            string? text = selection_data.get_text();
+            import_text(_("Dropped text"), text);
+        } else if (info == DndTarget.URIS) {
+            string[]? uris = selection_data.get_uris();
+            foreach (string uri in uris)
+                uri._strip();
+            import_files(uris);
+        }
+    }
+
+    private void update_clipboard_state(Gtk.Clipboard clipboard, Gdk.Event? event, Gtk.ActionGroup group) {
+        Gtk.Action action = group.get_action("edit-import-clipboard");
+        action.set_sensitive(clipboard.wait_is_text_available());
+    }
+
+    private void on_clipboard_received(Gtk.Clipboard board, string? text) {
+        if (text == null)
+            return;
+
+        assert(this.filter_entry != null);
+        if (this.filter_entry.is_focus)
+            this.filter_entry.paste_clipboard();
+        else
+            if (text != null && text.char_count() > 0)
+                import_text(_("Clipboard text"), text);
+    }
+
+    private void on_key_import_clipboard(Gtk.Action action) {
+        Gdk.Atom atom = Gdk.Atom.intern("CLIPBOARD", false);
+        Gtk.Clipboard clipboard = Gtk.Clipboard.get(atom);
+        clipboard.request_text(on_clipboard_received);
+    }
+
+    private void on_app_quit(Gtk.Action action) {
+        this.application.quit();
+    }
+
+    private string update_view_filter() {
+        string val = "";
+
+        switch (this.show_action.get_current_value()) {
+            case ShowFilter.PERSONAL:
+                this.pred.flags = Seahorse.Flags.PERSONAL;
+                val = "personal";
+                break;
+            case ShowFilter.TRUSTED:
+                this.pred.flags = Seahorse.Flags.TRUSTED;
+                val = "trusted";
+                break;
+            case ShowFilter.ANY:
+                this.pred.flags = 0;
+                val = "";
+                break;
+        }
+
+        this.store.refilter();
+        return val;
+    }
+
+    private void on_item_filter_changed(GLib.Settings settings, string? key) {
+        int radio;
+
+        string? value = settings.get_string(key);
+        if (value == null || value == "")
+            radio = ShowFilter.ANY;
+        else if (value == "personal")
+            radio = ShowFilter.PERSONAL;
+        else if (value == "trusted")
+            radio = ShowFilter.TRUSTED;
+        else
+            radio = -1;
+
+        this.show_action.set_current_value(radio);
+        update_view_filter();
+    }
+
+    public override GLib.List<weak GLib.Object> get_selected_objects() {
+        return KeyManagerStore.get_selected_objects(this.view);
+    }
+
+    public override Place? get_focused_place() {
+        return this.sidebar.get_focused_place();
+    }
+
+    private Gcr.Collection setup_sidebar() {
+        this.sidebar = new Sidebar();
+
+        this.sidebar_width = this.settings.get_int("sidebar-width");
+        Gtk.Builder builder = get_builder();
+
+        Gtk.Paned panes = (Gtk.Paned) builder.get_object("sidebar-panes");
+        panes.position = this.sidebar_width;
+        panes.realize.connect(() =>   { panes.position = this.settings.get_int("sidebar-width"); });
+        panes.unrealize.connect(() => { this.settings.set_int("sidebar-width", panes.position);  });
+
+        panes.get_child1().set_size_request(50, -1);
+        panes.get_child2().set_size_request(150, -1);
+
+        foreach (weak Backend backend in get_backends()) {
+            if (backend.actions != null)
+                include_actions(backend.actions);
+        }
+
+        Gtk.Container area = (Gtk.Container) builder.get_object("sidebar-area");
+        area.add(this.sidebar);
+        this.sidebar.show();
+
+        Gtk.ActionGroup actions = new Gtk.ActionGroup("sidebar");
+        actions.set_translation_domain(Config.GETTEXT_PACKAGE);
+        actions.add_toggle_actions(SIDEBAR_ACTIONS, null);
+        Gtk.Action action = actions.get_action("view-sidebar");
+        this.settings.bind("sidebar-visible", action, "active", SettingsBindFlags.DEFAULT);
+        action.bind_property("active", area, "visible", BindingFlags.SYNC_CREATE);
+        action.bind_property("active", this.sidebar, "combined", BindingFlags.INVERT_BOOLEAN | 
BindingFlags.SYNC_CREATE);
+        include_actions(actions);
+
+        this.settings.bind("keyrings-selected", this.sidebar, "selected-uris", SettingsBindFlags.DEFAULT);
+
+        return this.sidebar.collection;
+    }
+
+    public override List<weak Backend> get_backends() {
+        return this.sidebar.get_backends();
+    }
+}
diff --git a/src/main.vala b/src/main.vala
new file mode 100644
index 0000000..6746b2d
--- /dev/null
+++ b/src/main.vala
@@ -0,0 +1,36 @@
+/*
+ * Seahorse
+ *
+ * Copyright (C) 2003 Jacob Perkins
+ * Copyright (C) 2004,2005 Stefan Walter
+ * Copyright (C) 2017 Niels De Graef
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * 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, see
+ * <http://www.gnu.org/licenses/>.
+ */
+
+public int main (string[] args) {
+    Intl.bindtextdomain(Config.GETTEXT_PACKAGE, Config.LOCALEDIR);
+    Intl.bind_textdomain_codeset(Config.GETTEXT_PACKAGE, "UTF-8");
+    Intl.textdomain(Config.GETTEXT_PACKAGE);
+
+    Seahorse.register_resource();
+
+    Seahorse.Application app = new Seahorse.Application();
+    int status = app.run(args);
+
+    Seahorse.Registry.cleanup();
+    Seahorse.Servers.cleanup();
+
+    return status;
+}
diff --git a/src/meson.build b/src/meson.build
index a4c73a6..e57f932 100644
--- a/src/meson.build
+++ b/src/meson.build
@@ -1,39 +1,54 @@
+search_provider_src = gnome.gdbus_codegen('seahorse-shell-search-provider-generated',
+  'org.gnome.ShellSearchProvider2.xml',
+  interface_prefix : 'org.gnome.',
+  namespace : 'Seahorse',
+)
+
 seahorse_sources = [
-  'seahorse-generate-select.c',
-  'seahorse-import-dialog.c',
-  'seahorse-key-manager.c',
-  'seahorse-main.c',
-  'seahorse-sidebar.c',
-  resources_src,
+  'application.vala',
+  'generate-select.vala',
+  'import-dialog.vala',
+  'key-manager.vala',
+  'main.vala',
+  'search-provider.vala',
+  'sidebar.vala',
+
+  # resources_src,
+  search_provider_src,
 ]
 
 seahorse_dependencies = [
   glib_deps,
   gtk,
-  gcr,
-  config,
+  libsecret,
   common_dep,
   libseahorse_dep,
+  gkr_dep,
+  ssh_dep,
 ]
 
 seahorse_linkedlibs = [
   libeggdatetime_lib,
   libtreemultidnd_lib,
-  gkr_lib,
-  ssh_lib,
+]
+
+seahorse_vala_flags = [
 ]
 
 if with_pgp
-  seahorse_linkedlibs += pgp_lib
+  seahorse_dependencies += pgp_dep
+  seahorse_vala_flags += [ '-D', 'WITH_PGP' ]
 endif
 if with_pkcs11
-  seahorse_linkedlibs += pkcs11_lib
+  seahorse_dependencies += pkcs11_dep
+  seahorse_vala_flags += [ '-D', 'WITH_PKCS11' ]
 endif
 
 
 seahorse_exe = executable('seahorse',
   seahorse_sources,
   dependencies: seahorse_dependencies,
+  vala_args: seahorse_vala_flags,
   link_with: seahorse_linkedlibs,
   install: true,
 )
diff --git a/libseahorse/org.gnome.ShellSearchProvider2.xml b/src/org.gnome.ShellSearchProvider2.xml
similarity index 100%
rename from libseahorse/org.gnome.ShellSearchProvider2.xml
rename to src/org.gnome.ShellSearchProvider2.xml
diff --git a/src/search-provider.vala b/src/search-provider.vala
new file mode 100644
index 0000000..660423f
--- /dev/null
+++ b/src/search-provider.vala
@@ -0,0 +1,264 @@
+/*
+ * Seahorse
+ *
+ * Copyright (C) 2013 Giovanni Campagna <scampa giovanni gmail com>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * 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, see
+ * <http://www.gnu.org/licenses/>.
+ */
+
+//#define SECRET_API_SUBJECT_TO_CHANGE
+
+    /* SeahorseShellSearchProvider2SkeletonClass parent_class; */
+[DBus (name = "org.gnome.Shell.SearchProvider2")]
+public class Seahorse.SearchProvider : GLib.Object {
+    private Gcr.UnionCollection union_collection = new Gcr.UnionCollection();
+
+    private Predicate base_predicate = Predicate();
+    private Gcr.Collection collection;
+    private HashTable<string, weak Object> handles;
+    private int n_loading = 0;
+    private bool loaded = false;
+
+    public SearchProvider() {
+        this.base_predicate.flags = Flags.PERSONAL;
+        this.base_predicate.custom = check_object_type;
+        this.collection = new Collection.for_predicate(this.union_collection, this.base_predicate, null);
+        this.handles = new HashTable<string, weak Object>.full(str_hash, str_equal, free, null);
+    }
+
+    ~SearchProvider() {
+        this.handles.foreach( (__, obj) => {
+            obj.weak_unref(on_object_gone);
+        });
+    }
+
+    private async void load () {
+        // avoid reentering load () from a different request
+        while (!loaded && n_loading >= 0) {
+            var wait = notify["loaded"].connect (() => {
+                load.callback ();
+            });
+
+            yield;
+
+            disconnect (wait);
+            if (this.loaded)
+                return;
+        }
+
+        foreach (Backend backend in Backend.get_registered()) {
+            // XXX won't this give races?
+            this.n_loading++;
+
+            backend.notify["loaded"].connect(on_backend_loaded);
+            backend.added.connect(on_place_added);
+            backend.removed.connect(on_place_removed);
+
+            foreach (GLib.Object place in backend.get_objects())
+                on_place_added(backend, place);
+        }
+    }
+
+    public async string[] GetInitialResultSet(string[] terms) {
+        hold_app();
+
+        if (this.n_loading >= 0)
+            yield load();
+
+        Predicate predicate = Predicate () {
+            custom = object_matches_search,
+            custom_target = terms
+        };
+
+        string?[] results = {};
+        foreach (GLib.Object obj in this.collection.get_objects()) {
+            if (predicate.match(obj)) {
+                string str = "%p".printf(obj);
+
+                if (!(str in this.handles)) {
+                    this.handles.insert(str, (Object) obj);
+                    obj.weak_ref(on_object_gone);
+                }
+                results += str;
+            }
+        }
+
+        release_app ();
+        results += null;
+        return results;
+    }
+
+    public async string[] GetSubsearchResultSet(string[] previous_results, string[] new_terms) {
+        hold_app ();
+
+        Predicate predicate = Predicate() {
+            custom = object_matches_search,
+            custom_target = new_terms
+        };
+
+        string?[] results = {};
+        foreach (string previous_result in previous_results) {
+            GLib.Object? object = this.handles.lookup(previous_result);
+            if (object == null || !this.collection.contains(object))
+                continue; // Bogus value
+
+            if (predicate.match(object))
+                results += previous_result;
+        }
+
+        release_app ();
+        results += null;
+        return results;
+    }
+
+    public async HashTable<string, Variant>[] GetResultMetas(string[] results) {
+        hold_app();
+
+        var metas = new HashTable<string, Variant>[results.length];
+        int good_results = 0;
+        foreach (string result in results) {
+            Seahorse.Object object = this.handles.lookup(result);
+            if (object == null || !(object in this.collection))
+                continue; // Bogus value
+
+            HashTable<string, Variant> meta = new HashTable<string, Variant>(str_hash, str_equal);
+
+            meta["id"] = result;
+            if (object.label != null)
+                meta["name"] = object.label;
+            if (object.icon != null)
+                meta["icon"] = object.icon.serialize();
+
+            string? description = get_description_if_available(object);
+            if (description != null)
+                meta["name"] = Markup.escape_text(description);
+
+            metas[good_results] = meta;
+            good_results++;
+        }
+
+        release_app();
+        return metas[0:good_results];
+    }
+
+    public void ActivateResult(string identifier, string[] terms, uint32 timestamp) {
+        hold_app();
+
+        Object? object = null;
+        identifier.scanf("%p", &object);
+        object = this.handles.lookup(identifier);
+        if (object == null || !(object in this.collection) || !(object is Viewable))
+            return; // Bogus value
+
+        KeyManager key_manager = new KeyManager((Application)GLib.Application.get_default());
+        /* key_manager.show(timestamp); */
+        Viewable.view(object, (Gtk.Window) key_manager);
+
+        release_app ();
+    }
+
+    public void LaunchSearch (string[] terms, uint32 timestamp) {
+        // TODO
+    }
+
+    private static bool object_matches_search (GLib.Object? object, void* terms) {
+        foreach (string term in ((string[]) terms)) {
+            if (!object_contains_filtered_text (object as Object, term))
+                return false;
+        }
+
+        return true;
+    }
+
+    /* Search through row for text */
+    private static bool object_contains_filtered_text (Seahorse.Object object, string? text) {
+        string? name = object.label;
+        if (name != null && (text in name.down()))
+            return true;
+
+        string? description = get_description_if_available (object);
+        if (description != null && (text in description.down()))
+            return true;
+
+        return false;
+    }
+
+    private void on_object_gone(GLib.Object? where_the_object_was) {
+        this.handles.remove("%p".printf(where_the_object_was));
+    }
+
+    // We called before loading, we queue GetInitialResultSet, but
+    // we drop all other calls, because we don't expect to see them
+    // before we reply to GetInitialResultSet
+
+    private void hold_app() {
+        GLib.Application.get_default().hold();
+    }
+
+    private void release_app() {
+        GLib.Application.get_default().release();
+    }
+
+    private void on_place_added (Gcr.Collection places, GLib.Object object) {
+        Place place = (Place) object;
+        if (!this.union_collection.have(place))
+            this.union_collection.add(place);
+    }
+
+    private void on_place_removed (Gcr.Collection places, GLib.Object object) {
+        Place place = (Place) object;
+        if (this.union_collection.have(place))
+            this.union_collection.remove(place);
+    }
+
+    private void on_backend_loaded (GLib.Object? object, ParamSpec pspec) {
+        this.n_loading--;
+        if (this.n_loading == 0)
+            this.loaded = true;
+    }
+
+    private static bool check_object_type (GLib.Object? object, void* custom_target) {
+        if (!(object is Viewable))
+            return false;
+
+        if (object is Secret.Item) {
+            string? schema_name = ((Secret.Item) object).get_schema_name ();
+            if (schema_name != "org.gnome.keyring.Note")
+                return false;
+        }
+
+        return true;
+    }
+
+    /* public bool dbus_register (DBusConnection connection, string? object_path) throws GLib.Error { */
+    /*     return export (connection, object_path, error); */
+    /* } */
+
+    /* public void dbus_unregister (DBusConnection connection, string? object_path) { */
+    /*     if (has_connection(connection)) */
+    /*         unexport_from_connection(connection); */
+    /* } */
+
+    private static string? get_description_if_available (GLib.Object? obj) {
+        if (obj == null)
+            return null;
+
+        if (obj.get_class().find_property("description") == null)
+            return null;
+
+        string? description = null;
+        obj.get("description", out description);
+        return description;
+    }
+}
diff --git a/src/sidebar.vala b/src/sidebar.vala
new file mode 100644
index 0000000..2bbb0a2
--- /dev/null
+++ b/src/sidebar.vala
@@ -0,0 +1,915 @@
+/*
+ * Seahorse
+ *
+ * Copyright (C) 2011 Collabora Ltd.
+ * Copyright (C) 2017 Niels De Graef
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * 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, see
+ * <http://www.gnu.org/licenses/>.
+ */
+
+public class Seahorse.Sidebar : Gtk.ScrolledWindow {
+
+    private const int ACTION_BUTTON_XPAD = 6;
+
+    private Gtk.TreeView tree_view = new Gtk.TreeView();
+
+    private Gtk.ListStore store = new Gtk.ListStore.newv(Column.types());
+    private List<Backend> backends = new List<Backend>();
+    private Gcr.UnionCollection objects = new Gcr.UnionCollection();
+
+    // The selection
+    private HashTable<Gcr.Collection, Gcr.Collection> selection
+         = new HashTable<Gcr.Collection, Gcr.Collection>(direct_hash, direct_equal);
+    private bool updating;
+
+    // A set of chosen uris, used with settings
+    private GenericSet<string?> chosen = new GenericSet<string?>(str_hash, str_equal);
+
+    // Action icons
+    private Gdk.Pixbuf? pixbuf_lock;
+    private Gdk.Pixbuf? pixbuf_unlock;
+    private Gdk.Pixbuf? pixbuf_lock_l;
+    private Gdk.Pixbuf? pixbuf_unlock_l;
+    private Gtk.TreePath? action_highlight_path;
+    private Gtk.CellRendererPixbuf action_cell_renderer;
+    private int action_button_size;
+    private Gtk.AccelGroup accel_group = new Gtk.AccelGroup();
+
+    private uint update_places_sig;
+
+    /**
+     * Collection of objects sidebar represents
+     */
+    public Gcr.Collection collection {
+        get { return this.objects; }
+    }
+
+    /**
+     * The URIs selected by the user
+     */
+    public string[] selected_uris {
+        owned get { return chosen_uris_to_array(); }
+        set { replace_chosen_uris(value); }
+    }
+
+    /**
+     * Collection shows all objects combined
+     */
+    public bool combined {
+        get { return this._combined; }
+        set {
+            if (this._combined != value) {
+                this._combined = value;
+                update_objects_in_collection(false);
+            }
+        }
+    }
+    private bool _combined;
+
+    private enum RowType {
+        BACKEND,
+        PLACE,
+    }
+
+    private enum Column {
+        ROW_TYPE,
+        ICON,
+        LABEL,
+        TOOLTIP,
+        CATEGORY,
+        COLLECTION,
+        URI,
+        N_COLUMNS;
+
+        public static Type[] types() {
+            return {
+                typeof(uint),
+                typeof(Icon),
+                typeof(string),
+                typeof(string),
+                typeof(string),
+                typeof(Gcr.Collection),
+                typeof(string)
+            };
+        }
+    }
+
+    public Sidebar() {
+        get_style_context().add_class("sidebar");
+
+        set_policy(Gtk.PolicyType.NEVER, Gtk.PolicyType.AUTOMATIC);
+        set_hadjustment(null);
+        set_vadjustment(null);
+        set_shadow_type(Gtk.ShadowType.OUT);
+        get_style_context().set_junction_sides(Gtk.JunctionSides.RIGHT | Gtk.JunctionSides.LEFT);
+
+        /* tree view */
+        Gtk.TreeViewColumn col = new Gtk.TreeViewColumn();
+
+        /* initial padding */
+        Gtk.CellRenderer cell = new Gtk.CellRendererText();
+        col.pack_start(cell, false);
+        cell.xpad = 6;
+
+        /* headings */
+        Gtk.CellRendererText headings_cell = new Gtk.CellRendererText();
+        col.pack_start(headings_cell, false);
+        col.set_attributes(headings_cell, "text", Column.LABEL, null);
+        headings_cell.weight = Pango.Weight.BOLD;
+        headings_cell.weight_set = true;
+        headings_cell.ypad = 6;
+        headings_cell.xpad = 0;
+        col.set_cell_data_func(headings_cell, on_cell_renderer_heading_visible);
+
+        /* icon padding */
+        cell = new Gtk.CellRendererText();
+        col.pack_start(cell, false);
+        col.set_cell_data_func(cell, on_padding_cell_renderer);
+
+        /* icon renderer */
+        cell = new Gtk.CellRendererPixbuf();
+        col.pack_start(cell, false);
+        col.set_attributes(cell, "gicon", Column.ICON, null);
+        col.set_cell_data_func(cell, on_cell_renderer_heading_not_visible);
+
+        /* normal text renderer */
+        Gtk.CellRendererText text_cell = new Gtk.CellRendererText();
+        col.pack_start(text_cell, true);
+        text_cell.editable = false;
+        col.set_attributes(text_cell, "text", Column.LABEL, null);
+        col.set_cell_data_func(text_cell, on_cell_renderer_heading_not_visible);
+        text_cell.ellipsize = Pango.EllipsizeMode.END;
+        text_cell.ellipsize_set = true;
+
+        /* lock/unlock icon renderer */
+        this.action_cell_renderer = new Gtk.CellRendererPixbuf();
+        this.action_cell_renderer.mode = Gtk.CellRendererMode.ACTIVATABLE;
+        this.action_cell_renderer.stock_size = Gtk.IconSize.MENU;
+        this.action_cell_renderer.xpad = ACTION_BUTTON_XPAD;
+        this.action_cell_renderer.xalign = 1.0f;
+        col.pack_start(this.action_cell_renderer, false);
+        col.set_cell_data_func(this.action_cell_renderer, on_cell_renderer_action_icon);
+
+        col.set_max_width(24);
+        this.tree_view.append_column(col);
+
+        this.tree_view.set_headers_visible(false);
+        this.tree_view.set_tooltip_column(Column.TOOLTIP);
+        this.tree_view.set_search_column(Column.LABEL);
+        this.tree_view.set_model(this.store);
+        this.tree_view.popup_menu.connect(on_tree_view_popup_menu);
+        this.tree_view.button_press_event.connect(on_tree_view_button_press_event);
+        this.tree_view.motion_notify_event.connect(on_tree_view_motion_notify_event);
+        this.tree_view.button_release_event.connect(on_tree_view_button_release_event);
+        add(this.tree_view);
+        this.tree_view.show();
+
+        Gtk.TreeSelection selection = this.tree_view.get_selection();
+        selection.set_mode(Gtk.SelectionMode.MULTIPLE);
+        selection.set_select_function(on_tree_selection_validate);
+        selection.changed.connect(() => update_objects_for_selection(selection));
+
+        load_backends();
+    }
+
+    ~Sidebar() {
+        foreach (Backend backend in this.backends) {
+            SignalHandler.disconnect_by_func((void*) backend, (void*) on_place_added, this);
+            SignalHandler.disconnect_by_func((void*) backend, (void*) on_place_removed, this);
+            SignalHandler.disconnect_by_func((void*) backend, (void*) on_backend_changed, this);
+
+            foreach (weak GLib.Object obj in backend.get_objects())
+                on_place_removed (backend, (Place) obj);
+        }
+
+        invalidate_sidebar_pixbufs();
+
+        if (this.update_places_sig != 0)
+            Source.remove(this.update_places_sig);
+    }
+
+    private void load_backends() {
+        foreach (Backend backend in Backend.get_registered()) {
+            this.backends.append(backend);
+            backend.added.connect(on_place_added);
+            backend.removed.connect(on_place_removed);
+            backend.notify.connect(on_backend_changed);
+
+            foreach (weak GLib.Object obj in backend.get_objects())
+                on_place_added(backend, obj);
+        }
+
+        this.backends.sort((a, b) => {
+            int ordera = order_from_backend((Backend) a);
+            int orderb = order_from_backend((Backend) b);
+            return ordera - orderb;
+        });
+    }
+
+    private void on_place_added(Gcr.Collection? places, GLib.Object obj) {
+        ((Place) obj).notify.connect(() => update_places_later());
+        update_places_later();
+    }
+
+    private void on_place_removed(Gcr.Collection? places, GLib.Object obj) {
+        SignalHandler.disconnect_by_func((void*) obj, (void*) update_places_later, this);
+        update_places_later();
+    }
+
+    private void on_backend_changed(GLib.Object obj, ParamSpec spec) {
+        update_places_later();
+    }
+
+    private static int order_from_backend (Backend backend) {
+        switch (backend.name) {
+            case "gkr":
+                return 0;
+            case "pgp":
+                return 1;
+            case "pkcs11":
+                return 2;
+            case "ssh":
+                return 3;
+            default:
+                return 10;
+        }
+    }
+
+    private void ensure_sidebar_pixbufs() {
+        if (this.pixbuf_lock != null && this.pixbuf_lock_l != null
+            && this.pixbuf_unlock_l != null && this.pixbuf_unlock != null)
+            return;
+
+        Gtk.IconTheme icon_theme = Gtk.IconTheme.get_default();
+        Gtk.StyleContext style = get_style_context();
+
+        int height;
+        if (!Gtk.icon_size_lookup(Gtk.IconSize.MENU, out this.action_button_size, out height))
+            this.action_button_size = 16;
+
+        // Lock icon
+        Icon icon = new ThemedIcon.with_default_fallbacks("changes-prevent-symbolic");
+        Gtk.IconInfo? icon_info = icon_theme.lookup_by_gicon(icon, this.action_button_size, 
Gtk.IconLookupFlags.GENERIC_FALLBACK);
+        if (icon_info == null)
+            return;
+        try {
+            if (this.pixbuf_lock == null)
+                this.pixbuf_lock = icon_info.load_symbolic_for_context(style, null);
+            if (this.pixbuf_lock_l == null)
+                this.pixbuf_lock_l = create_spotlight_pixbuf(this.pixbuf_lock);
+        } catch (Error e) {
+            debug("Error while looking up lock icon: %s", e.message);
+        }
+
+        // Unlock icon
+        icon = new ThemedIcon.with_default_fallbacks("changes-allow-symbolic");
+        icon_info = icon_theme.lookup_by_gicon(icon, this.action_button_size, 
Gtk.IconLookupFlags.GENERIC_FALLBACK);
+        if (icon_info == null)
+            return;
+        try {
+            if (this.pixbuf_unlock == null)
+                this.pixbuf_unlock = icon_info.load_symbolic_for_context(style, null);
+            if (this.pixbuf_unlock_l == null)
+                this.pixbuf_unlock_l = create_spotlight_pixbuf(this.pixbuf_unlock);
+        } catch (Error e) {
+            debug("Error while looking up unlock icon: %s", e.message);
+        }
+    }
+
+    private Gdk.Pixbuf? create_spotlight_pixbuf (Gdk.Pixbuf? src) {
+        Gdk.Pixbuf? dest = new Gdk.Pixbuf(src.colorspace, src.has_alpha, src.bits_per_sample,
+                                          src.width, src.height);
+
+        bool has_alpha = src.has_alpha;
+        int width = src.width;
+        int height = src.height;
+        int dst_row_stride = dest.rowstride;
+        int src_row_stride = src.rowstride;
+        uint8* target_pixels = dest.pixels;
+        uint8* original_pixels = src.pixels;
+
+        for (int i = 0; i < height; i++) {
+            uint8* pixdest = target_pixels + i * dst_row_stride;
+            uint8* pixsrc = original_pixels + i * src_row_stride;
+            for (int j = 0; j < width; j++) {
+                *pixdest++ = lighten_component (*pixsrc++);
+                *pixdest++ = lighten_component (*pixsrc++);
+                *pixdest++ = lighten_component (*pixsrc++);
+                if (has_alpha) {
+                    *pixdest++ = *pixsrc++;
+                }
+            }
+        }
+        return dest;
+    }
+
+    private uint8 lighten_component(uint8 cur_value) {
+        int new_value = cur_value + 24 + (cur_value >> 3);
+        return (new_value > 255)? (uint8)255 : (uint8)new_value;
+    }
+
+    private void invalidate_sidebar_pixbufs() {
+        this.pixbuf_lock = null;
+        this.pixbuf_unlock = null;
+        this.pixbuf_lock_l = null;
+        this.pixbuf_unlock_l = null;
+    }
+
+    private void next_or_append_row(Gtk.ListStore? store, ref Gtk.TreeIter iter, string? category,
+                                    Gcr.Collection? collection) {
+        // We try to keep the same row in order to preserve checked state
+        // and selections. So if the next row matches the values we want to
+        // set on it, then just keep that row.
+        //
+        // This is complicated by the fact that the first row being inserted
+        // doesn't have a valid iter, and we don't have a standard way to
+        // detect that an iter isn't valid.
+
+        // A marker that tells us the iter is not yet valid
+        if (iter.stamp == int.from_pointer(&iter) && iter.user_data3 == (&iter) &&
+            iter.user_data2 == (&iter) && iter.user_data == (&iter)) {
+            if (!store.get_iter_first(out iter))
+                store.append(out iter);
+            return;
+        }
+
+        if (!store.iter_next(ref iter)) {
+            store.append(out iter);
+            return;
+        }
+
+        for (;;) {
+            string? row_category;
+            Gcr.Collection? row_collection;
+            store.get(iter, Column.CATEGORY, out row_category,
+                            Column.COLLECTION, out row_collection);
+
+            if (row_category == category && row_collection == collection)
+                return;
+
+            if (!store.remove(ref iter)) {
+                store.append(out iter);
+                return;
+            }
+        }
+    }
+
+    private void update_objects_in_collection(bool update_chosen) {
+        if (this.updating) // Updating collection is blocked
+            return;
+
+        bool changed = false;
+        foreach (Backend backend in this.backends) {
+            foreach (weak GLib.Object obj in backend.get_objects()) {
+                Place place = (Place) obj;
+                bool include = this.selection.lookup(place) != null;
+
+                if (update_chosen) {
+                    string? uri = place.uri;
+                    bool have = (uri in this.chosen);
+                    if (include && !have) {
+                        this.chosen.add(uri);
+                        changed = true;
+                    } else if (!include && have) {
+                        this.chosen.remove(uri);
+                        changed = true;
+                    }
+                }
+
+                // Combined overrides and shows all objects
+                if (this.combined)
+                    include = true;
+
+                bool have = this.objects.have(place);
+                if (include && !have)
+                    this.objects.add(place);
+                else if (!include && have)
+                    this.objects.remove(place);
+            }
+        }
+    }
+
+    private void update_objects_for_selection(Gtk.TreeSelection selection) {
+        if (this.updating)
+            return;
+
+        HashTable<Gcr.Collection, Gcr.Collection> selected = new HashTable<Gcr.Collection, 
Gcr.Collection>(direct_hash, direct_equal);
+        selection.selected_foreach((model, path, iter) => {
+            Gcr.Collection? collection = null;
+            model.get(iter, Column.COLLECTION, out collection, -1);
+            if (collection != null)
+                selected.insert(collection, collection);
+        });
+
+        this.selection = selected;
+
+        if (!this.combined)
+            update_objects_in_collection(true);
+    }
+
+    private void update_objects_for_chosen(GenericSet<string?> chosen) {
+        this.updating = true;
+
+        Gtk.TreeSelection selection = this.tree_view.get_selection();
+
+        // Update the display
+        Gtk.TreeIter iter;
+        if (this.store.get_iter_first(out iter)) {
+            do {
+                Gcr.Collection? collection = null;
+                string? uri = null;
+                this.store.get(iter, Column.COLLECTION, out collection,
+                                     Column.URI, out uri, -1);
+
+                if (collection != null && uri != null) {
+                    if (uri in chosen)
+                        selection.select_iter(iter);
+                    else
+                        selection.unselect_iter(iter);
+                }
+            } while (this.store.iter_next(ref iter));
+        }
+
+        this.updating = false;
+        update_objects_for_selection(selection);
+    }
+
+    private void update_places() {
+        Gtk.TreeIter iter = Gtk.TreeIter();
+        iter.stamp = int.from_pointer(&iter); // A marker that tells us the iter is not yet valid
+        iter.user_data3 = iter.user_data2 = iter.user_data = &iter;
+
+        foreach (Backend backend in this.backends)
+            update_backend(backend, ref iter);
+
+        // Update selection
+        update_objects_for_chosen(this.chosen);
+
+        if (this.combined)
+            update_objects_in_collection(false);
+    }
+
+    private void update_backend(Backend? backend, ref Gtk.TreeIter iter) {
+        if (backend.get_objects() == null) // Ignore categories that have nothing
+            return;
+
+        next_or_append_row(this.store, ref iter, backend.name, backend);
+        this.store.set(iter, Column.ROW_TYPE, RowType.BACKEND,
+                             Column.CATEGORY, backend.name,
+                             Column.LABEL, backend.label,
+                             Column.TOOLTIP, backend.description,
+                             Column.COLLECTION, backend);
+
+        foreach (weak GLib.Object obj in backend.get_objects()) {
+            Place place = obj as Place;
+            if (place == null)
+                continue;
+
+            next_or_append_row(this.store, ref iter, backend.name, place);
+            this.store.set(iter, Column.ROW_TYPE, RowType.PLACE,
+                                 Column.CATEGORY, backend.name,
+                                 Column.LABEL, place.label,
+                                 Column.TOOLTIP, place.description,
+                                 Column.ICON, place.icon,
+                                 Column.COLLECTION, place,
+                                 Column.URI, place.uri);
+        }
+    }
+
+    private void update_places_later() {
+        if (this.update_places_sig == 0) {
+            this.update_places_sig = Idle.add(() => {
+                this.update_places_sig = 0;
+                update_places();
+                return false; // don't call again
+            });
+        }
+    }
+
+    private Lockable? lookup_lockable_for_iter(Gtk.TreeModel? model, Gtk.TreeIter? iter) {
+        Gcr.Collection? collection = null;
+        model.get(iter, Column.COLLECTION, out collection, -1);
+
+        return collection as Lockable;
+    }
+
+    private void on_cell_renderer_action_icon(Gtk.CellLayout layout, Gtk.CellRenderer? cell,
+                                              Gtk.TreeModel? model, Gtk.TreeIter? iter) {
+        bool can_lock = false;
+        bool can_unlock = false;
+
+        Lockable? lockable = lookup_lockable_for_iter(model, iter);
+        if (lockable != null) {
+            can_lock = lockable.lockable;
+            can_unlock = lockable.unlockable;
+        }
+
+        if (can_lock || can_unlock) {
+            ensure_sidebar_pixbufs();
+
+            bool highlight = false;
+            if (this.action_highlight_path!= null) {
+                Gtk.TreePath? path = model.get_path(iter);
+                highlight = path.compare(this.action_highlight_path) == 0;
+            }
+
+            Gdk.Pixbuf? pixbuf;
+            if (can_lock)
+                pixbuf = highlight ? this.pixbuf_unlock : this.pixbuf_unlock_l;
+            else
+                pixbuf = highlight ? this.pixbuf_lock : this.pixbuf_lock_l;
+
+            this.action_cell_renderer.visible = true;
+            this.action_cell_renderer.pixbuf = pixbuf;
+        } else {
+            this.action_cell_renderer.visible = false;
+            this.action_cell_renderer.pixbuf = null;
+        }
+    }
+
+    private void on_cell_renderer_heading_visible(Gtk.CellLayout layout, Gtk.CellRenderer? cell,
+                                                  Gtk.TreeModel? model, Gtk.TreeIter? iter) {
+        RowType type;
+        model.get(iter, Column.ROW_TYPE, out type, -1);
+        cell.visible = (type == RowType.BACKEND);
+    }
+
+    private void on_padding_cell_renderer(Gtk.CellLayout layout, Gtk.CellRenderer? cell,
+                                          Gtk.TreeModel? model, Gtk.TreeIter? iter) {
+        RowType type;
+        model.get(iter, Column.ROW_TYPE, out type, -1);
+
+        if (type == RowType.BACKEND) {
+            cell.visible = false;
+            cell.xpad = 0;
+            cell.ypad = 0;
+        } else {
+            cell.visible = true;
+            cell.xpad = 3;
+            cell.ypad = 3;
+        }
+    }
+
+    private void on_cell_renderer_heading_not_visible(Gtk.CellLayout layout, Gtk.CellRenderer? cell,
+                                                      Gtk.TreeModel? model, Gtk.TreeIter? iter) {
+        RowType type;
+        model.get(iter, Column.ROW_TYPE, out type, -1);
+        cell.visible = (type != RowType.BACKEND);
+    }
+
+    private bool on_tree_selection_validate(Gtk.TreeSelection selection, Gtk.TreeModel? model,
+                                            Gtk.TreePath? path, bool path_currently_selected) {
+        Gtk.TreeIter iter;
+        model.get_iter(out iter, path);
+
+        RowType row_type;
+        model.get(iter, Column.ROW_TYPE, out row_type, -1);
+        if (row_type == RowType.BACKEND)
+            return false;
+
+        return true;
+    }
+
+    private void place_lock(Lockable lockable, Gtk.Window? window) {
+        Cancellable cancellable = new Cancellable();
+        TlsInteraction interaction = new Interaction(window);
+
+        lockable.lock.begin(interaction, cancellable, (obj, res) => {
+            try {
+                lockable.lock.end(res);
+            } catch (Error e) {
+                Util.show_error(window, _("Couldn’t lock"), e.message);
+            }
+        });
+    }
+
+    private void on_place_lock(Gtk.MenuItem item, Lockable lockable) {
+        place_lock(lockable, (Gtk.Window) item.get_toplevel());
+    }
+
+    private void place_unlock(Lockable lockable, Gtk.Window? window) {
+        Cancellable cancellable = new Cancellable();
+        TlsInteraction interaction = new Interaction(window);
+
+        lockable.unlock.begin(interaction, cancellable, (obj, res) => {
+            try {
+                lockable.unlock.end(res);
+            } catch (Error e) {
+                Util.show_error(window, _("Couldn’t unlock"), e.message);
+            }
+        });
+    }
+
+    private void on_place_unlock(Gtk.MenuItem item, Lockable lockable) {
+        place_unlock(lockable, (Gtk.Window) item.get_toplevel());
+    }
+
+    private void on_place_delete(Gtk.MenuItem item, Deletable deletable) {
+        Deleter deleter = deletable.create_deleter();
+        if (deleter.prompt((Gtk.Window) item.get_toplevel())) {
+            deleter.delete.begin(null, (obj, res) => {
+                try {
+                    deleter.delete.end(res);
+                } catch (Error e) {
+                    Util.show_error(parent, _("Couldn’t delete"), e.message);
+                }
+            });
+        }
+    }
+
+    private void popup_menu_for_place(Place place, uint button, uint32 activate_time) {
+        Gtk.Menu menu = new Gtk.Menu();
+
+        // First add all the actions from the collection
+        Gtk.ActionGroup actions = place.actions;
+        foreach (weak Gtk.Action action in actions.list_actions()) {
+            action.set_accel_group(this.accel_group);
+            menu.append((Gtk.MenuItem) action.create_menu_item());
+        }
+
+        // Lock and unlock items
+        if (place is Lockable) {
+            Gtk.MenuItem item = new Gtk.MenuItem.with_mnemonic(_("_Lock"));
+            item.activate.connect(() => on_place_lock(item, (Lockable) place));
+            place.bind_property("lockable", item, "visible", BindingFlags.SYNC_CREATE);
+            menu.append(item);
+
+            item = new Gtk.MenuItem.with_mnemonic(_("_Unlock"));
+            item.activate.connect(() => on_place_unlock(item, (Lockable) place));
+            place.bind_property("unlockable", item, "visible", BindingFlags.SYNC_CREATE);
+            menu.append(item);
+        }
+
+        // Delete item
+        if (place is Deletable) {
+            Gtk.ImageMenuItem item = new Gtk.ImageMenuItem.with_mnemonic(_("_Delete"));
+            item.set_image(new Gtk.Image.from_stock(Gtk.Stock.DELETE, Gtk.IconSize.MENU));
+            item.activate.connect(() => on_place_delete(item, (Deletable) place));
+            place.bind_property("deletable", item, "sensitive", BindingFlags.SYNC_CREATE);
+            menu.append(item);
+            item.show();
+        }
+
+        // Properties item
+        if (place is Viewable) {
+            Gtk.ImageMenuItem item = new Gtk.ImageMenuItem.with_mnemonic(_("_Properties"));
+            item.set_image(new Gtk.Image.from_stock(Gtk.Stock.PROPERTIES, Gtk.IconSize.MENU));
+            item.activate.connect(() => Viewable.view(place, (Gtk.Window) item.get_toplevel()));
+            menu.append(item);
+            item.show();
+        }
+
+        bool visible = false;
+        menu.foreach((widget) => {
+            if (widget.visible)
+                visible = true;
+        });
+
+        if (visible) {
+            menu.popup(null, null, null, button, activate_time);
+            menu.attach_to_widget(this, null);
+            menu.show();
+        } else {
+            menu.destroy();
+        }
+    }
+
+    private bool on_tree_view_popup_menu(Gtk.Widget? widget) {
+        Gtk.TreePath? path;
+        this.tree_view.get_cursor(out path, null);
+        if (path == null)
+            return false;
+
+        Gtk.TreeIter iter;
+        if (!this.store.get_iter(out iter, path))
+            return false;
+
+        Gcr.Collection? collection;
+        this.store.get(iter, Column.COLLECTION, out collection, -1);
+
+        if (collection is Place) {
+            popup_menu_for_place((Place) collection, 0, Gtk.get_current_event_time());
+            return true;
+        }
+
+        return false;
+    }
+
+    private void update_action_buttons_take_path(Gtk.TreePath? path) {
+        if (path == this.action_highlight_path) {
+            return;
+        }
+
+        if (path != null && this.action_highlight_path != null &&
+            this.action_highlight_path.compare(path) == 0) {
+            return;
+        }
+
+        Gtk.TreePath? old_path = this.action_highlight_path;
+        this.action_highlight_path = path;
+
+        Gtk.TreeIter? iter = null;
+        if (this.action_highlight_path != null
+              && this.store.get_iter(out iter, this.action_highlight_path))
+            this.store.row_changed(this.action_highlight_path, iter);
+
+        if (old_path != null && this.store.get_iter(out iter, old_path))
+            this.store.row_changed(old_path, iter);
+    }
+
+    private bool over_action_button(int x, int y, out Gtk.TreePath? path) {
+
+        path = null;
+        Gtk.TreeViewColumn column;
+        if (this.tree_view.get_path_at_pos(x, y, out path, out column, null, null)) {
+            Gtk.TreeIter iter;
+            this.store.get_iter(out iter, path);
+
+            int hseparator;
+            this.tree_view.style_get("horizontal-separator", out hseparator, null);
+
+            // Reload cell attributes for this particular row
+            column.cell_set_cell_data(this.store, iter, false, false);
+            int width, x_offset;
+            column.cell_get_position(this.action_cell_renderer, out x_offset, out width);
+
+            // This is kinda weird, but we have to do it to workaround gtk+ expanding
+            // the eject cell renderer (even thought we told it not to) and we then
+            // had to set it right-aligned
+            x_offset += width - hseparator - ACTION_BUTTON_XPAD - this.action_button_size;
+
+            if (x - x_offset >= 0 && x - x_offset <= this.action_button_size)
+                return true;
+        }
+
+        if (path != null)
+            path = null;
+
+        return false;
+    }
+
+    private bool on_tree_view_motion_notify_event(Gtk.Widget? widget, Gdk.EventMotion event) {
+        Gtk.TreePath? path = null;
+        if (over_action_button((int) event.x, (int) event.y, out path)) {
+            update_action_buttons_take_path(path);
+            return true;
+        }
+
+        update_action_buttons_take_path(null);
+        return false;
+    }
+
+    private bool on_tree_view_button_press_event (Gtk.Widget? widget, Gdk.EventButton event) {
+        if (event.button != 3 || event.type != Gdk.EventType.BUTTON_PRESS)
+            return false;
+
+        Gtk.TreePath? path;
+        if (!this.tree_view.get_path_at_pos((int) event.x, (int) event.y, out path, null, null, null))
+            return false;
+
+        this.tree_view.set_cursor(path, null, false);
+        Gtk.TreeIter iter;
+        if (!this.store.get_iter(out iter, path))
+            return false;
+
+        Gcr.Collection? collection;
+        this.store.get(iter, Column.COLLECTION, out collection, -1);
+
+        if (collection is Place)
+            popup_menu_for_place((Place) collection, event.button, event.time);
+
+        return true;
+    }
+
+    private bool on_tree_view_button_release_event (Gtk.Widget? widget, Gdk.EventButton event) {
+        if (event.type != Gdk.EventType.BUTTON_RELEASE)
+            return true;
+
+        Gtk.TreePath? path;
+        if (!over_action_button((int) event.x, (int) event.y, out path))
+            return false;
+
+        Gtk.TreeIter iter;
+        if (!this.store.get_iter(out iter, path))
+            return false;
+
+        Gtk.Window? window = (Gtk.Window) widget.get_toplevel();
+
+        Lockable? lockable = lookup_lockable_for_iter(this.store, iter);
+        if (lockable != null) {
+            if (lockable.lockable)
+                place_lock(lockable, window);
+            else if (lockable.unlockable)
+                place_unlock(lockable, window);
+        }
+
+        return true;
+    }
+
+    public string[] chosen_uris_to_array() {
+        string[] results = {};
+        foreach (string? uri in this.chosen)
+            results += uri;
+
+        results += null;
+
+        return results;
+    }
+
+    public void replace_chosen_uris(string[] uris) {
+        // For quick lookups
+        GenericSet<string?> chosen = new GenericSet<string?>(str_hash, str_equal);
+        foreach (string uri in uris)
+            chosen.add(uri);
+
+        update_objects_for_chosen(chosen);
+        this.chosen = chosen;
+    }
+
+    public List<weak Gcr.Collection>? get_selected_places() {
+        List<weak Gcr.Collection> places = this.objects.elements();
+
+        Gtk.TreePath? path = null;
+        this.tree_view.get_cursor(out path, null);
+        if (path != null) {
+
+            Gtk.TreeIter iter;
+            if (!this.store.get_iter(out iter, path))
+                return null;
+
+            Gcr.Collection? collection;
+            RowType row_type;
+            this.store.get(iter, Column.ROW_TYPE, out row_type,
+                                 Column.COLLECTION, out collection, -1);
+
+            if (collection != null) {
+                if (row_type == RowType.PLACE) {
+                    places.remove(collection);
+                    places.prepend(collection);
+                }
+            }
+        }
+
+        return places;
+    }
+
+    public Place? get_focused_place() {
+        Gtk.TreeIter iter;
+
+        Gtk.TreePath? path = null;
+        this.tree_view.get_cursor(out path, null);
+        if (path != null) {
+            if (!this.store.get_iter(out iter, path))
+                return null;
+
+            Gcr.Collection? collection;
+            RowType row_type;
+            this.store.get(iter, Column.ROW_TYPE, out row_type,
+                                 Column.COLLECTION, out collection, -1);
+
+            if (row_type == RowType.PLACE)
+                return (Place) collection;
+        }
+
+        return null;
+    }
+
+    public List<weak Backend>? get_backends() {
+        Gtk.TreeIter iter;
+
+        List<weak Backend> backends = this.backends.copy();
+        backends.reverse();
+
+        Gtk.TreePath? path = null;
+        this.tree_view.get_cursor(out path, null);
+        if (path != null) {
+            if (!this.store.get_iter(out iter, path))
+                return null;
+
+            Gcr.Collection? collection;
+            RowType row_type;
+            this.store.get(iter, Column.ROW_TYPE, out row_type,
+                                 Column.COLLECTION, out collection, -1);
+
+            if (collection != null) {
+                if (row_type == RowType.BACKEND) {
+                    backends.remove((Backend) collection);
+                    backends.prepend((Backend) collection);
+                }
+            }
+        }
+
+        return backends;
+    }
+}
diff --git a/ssh/meson.build b/ssh/meson.build
index efb0719..f122d23 100644
--- a/ssh/meson.build
+++ b/ssh/meson.build
@@ -18,6 +18,7 @@ ssh_sources = [
 ssh_dependencies = [
   glib_deps,
   gcr,
+  gcr_ui,
   posix,
   gtk,
   common_dep,


[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]