]> spindle.queued.net Git - midori/commitdiff
Refactor bookmarks panel into a separate panel widget
authorChristian Dywan <christian@twotoasts.de>
Sat, 31 Jan 2009 03:09:35 +0000 (04:09 +0100)
committerChristian Dywan <christian@twotoasts.de>
Sat, 31 Jan 2009 03:09:35 +0000 (04:09 +0100)
midori/main.c
midori/midori-browser.c
panels/midori-bookmarks.c [new file with mode: 0644]
panels/midori-bookmarks.h [new file with mode: 0644]
po/POTFILES.in

index d6c29e8166eb80f1fce1b28495943ce9272ae466..96df6faef56bd5391486edf3b9d61a246de65b28 100644 (file)
@@ -16,6 +16,7 @@
 
 #include "midori-addons.h"
 #include "midori-app.h"
+#include "midori-bookmarks.h"
 #include "midori-browser.h"
 #include "midori-console.h"
 #include "midori-extension.h"
@@ -1274,6 +1275,11 @@ midori_app_add_browser_cb (MidoriApp*     app,
 
     panel = katze_object_get_object (browser, "panel");
 
+    /* Bookmarks */
+    addon = g_object_new (MIDORI_TYPE_BOOKMARKS, "app", app, NULL);
+    gtk_widget_show (addon);
+    midori_panel_append_page (MIDORI_PANEL (panel), MIDORI_VIEWABLE (addon));
+
     /* Transfers */
     #if 0
     addon = midori_view_new (net);
index 17b44d07e8d139a90c11a28a882ad7bd52a59444..dd04d30c7d14daafc6b92c36c7bb86c3206171fb 100644 (file)
@@ -1,5 +1,5 @@
 /*
- Copyright (C) 2007-2008 Christian Dywan <christian@twotoasts.de>
+ Copyright (C) 2007-2009 Christian Dywan <christian@twotoasts.de>
  Copyright (C) 2008 Dale Whittaker <dayul@users.sf.net>
 
  This library is free software; you can redistribute it and/or
@@ -53,7 +53,6 @@ struct _MidoriBrowser
     GtkWidget* bookmarkbar;
 
     GtkWidget* panel;
-    GtkWidget* panel_bookmarks;
     GtkWidget* panel_history;
     GtkWidget* notebook;
 
@@ -504,7 +503,8 @@ midori_view_notify_statusbar_text_cb (MidoriView*    view,
     g_free (text);
 }
 
-static void
+/* Private function, used by MidoriBookmarks */
+/* static */ void
 midori_browser_edit_bookmark_dialog_new (MidoriBrowser* browser,
                                          KatzeItem*     bookmark,
                                          gboolean       new_bookmark)
@@ -519,8 +519,6 @@ midori_browser_edit_bookmark_dialog_new (MidoriBrowser* browser,
     GtkWidget* entry_desc;
     GtkWidget* entry_uri;
     GtkWidget* combo_folder;
-    GtkTreeView* treeview;
-    GtkTreeModel* treemodel;
 
     dialog = gtk_dialog_new_with_buttons (
         new_bookmark ? _("New bookmark") : _("Edit bookmark"),
@@ -624,8 +622,6 @@ midori_browser_edit_bookmark_dialog_new (MidoriBrowser* browser,
     {
         gchar* selected;
         KatzeArray* folder;
-        GtkTreeIter iter;
-        GtkTreeIter* piter;
 
         katze_item_set_name (bookmark,
             gtk_entry_get_text (GTK_ENTRY (entry_title)));
@@ -635,43 +631,28 @@ midori_browser_edit_bookmark_dialog_new (MidoriBrowser* browser,
             katze_item_set_uri (bookmark,
                 gtk_entry_get_text (GTK_ENTRY (entry_uri)));
 
-        selected = gtk_combo_box_get_active_text (GTK_COMBO_BOX (combo_folder));
-        if (g_strcmp0 (selected, _("Toplevel folder")))
+        folder = browser->bookmarks;
+        if (new_bookmark)
         {
-            treeview = GTK_TREE_VIEW (browser->panel_bookmarks);
-            treemodel = gtk_tree_view_get_model (treeview);
-            if (gtk_tree_model_iter_children (treemodel, &iter, NULL))
+            selected = gtk_combo_box_get_active_text (GTK_COMBO_BOX (combo_folder));
+            if (g_strcmp0 (selected, _("Toplevel folder")))
             {
-                KatzeItem* item;
+                guint i, n;
 
-                do
+                if ((n = katze_array_get_length (browser->bookmarks)))
+                for (i = 0; i < n; i++)
                 {
-                    gtk_tree_model_get (treemodel, &iter, 0, &item, -1);
-
-                    if (!g_strcmp0 (katze_item_get_name (item), selected))
-                        break;
+                    KatzeItem* item = katze_array_get_nth_item (browser->bookmarks, i);
+                    if (KATZE_IS_ARRAY (item))
+                        if (!g_strcmp0 (katze_item_get_name (item), selected))
+                        {
+                            folder = KATZE_ARRAY (item);
+                            break;
+                        }
                 }
-                while (gtk_tree_model_iter_next (treemodel, &iter));
-
-                folder = KATZE_ARRAY (item);
             }
-            else
-                folder = browser->bookmarks;
-        }
-        else
-            folder = browser->bookmarks;
-        g_free (selected);
-
-        piter = (folder == browser->bookmarks ? NULL : &iter);
-
-        if (new_bookmark)
-        {
+            g_free (selected);
             katze_array_add_item (folder, bookmark);
-            treeview = GTK_TREE_VIEW (browser->panel_bookmarks);
-            treemodel = gtk_tree_view_get_model (treeview);
-            gtk_tree_store_insert_with_values (GTK_TREE_STORE (treemodel),
-                &iter, piter, G_MAXINT, 0, bookmark, -1);
-            g_object_ref (bookmark);
         }
     }
     gtk_widget_destroy (dialog);
@@ -2351,52 +2332,6 @@ _action_search_focus_out (GtkAction*     action,
         gtk_widget_hide (browser->navigationbar);
 }
 
-static void
-midori_panel_bookmarks_row_activated_cb (GtkTreeView*       treeview,
-                                         GtkTreePath*       path,
-                                         GtkTreeViewColumn* column,
-                                         MidoriBrowser*     browser)
-{
-    GtkTreeModel* model;
-    GtkTreeIter iter;
-    KatzeItem* item;
-    const gchar* uri;
-
-    model = gtk_tree_view_get_model (treeview);
-
-    if (gtk_tree_model_get_iter (model, &iter, path))
-    {
-        gtk_tree_model_get (model, &iter, 0, &item, -1);
-        uri = katze_item_get_uri (item);
-        if (uri && *uri)
-            midori_browser_set_current_uri (browser, uri);
-    }
-}
-
-static void
-midori_panel_bookmarks_cursor_or_row_changed_cb (GtkTreeView*   tree_view,
-                                                 MidoriBrowser* browser)
-{
-    GtkTreeModel* model;
-    GtkTreeIter iter;
-    KatzeItem* item;
-    gboolean is_separator;
-
-    if (katze_tree_view_get_selected_iter (tree_view, &model, &iter))
-    {
-        gtk_tree_model_get (model, &iter, 0, &item, -1);
-
-        is_separator = !KATZE_IS_ARRAY (item) && !katze_item_get_uri (item);
-        _action_set_sensitive (browser, "BookmarkEdit", !is_separator);
-        _action_set_sensitive (browser, "BookmarkDelete", TRUE);
-    }
-    else
-    {
-        _action_set_sensitive (browser, "BookmarkEdit", FALSE);
-        _action_set_sensitive (browser, "BookmarkDelete", FALSE);
-    }
-}
-
 static void
 midori_browser_bookmark_popup_item (GtkWidget*     menu,
                                     const gchar*   stock_id,
@@ -2611,54 +2546,6 @@ midori_browser_bookmark_popup (GtkWidget*      widget,
                          event, SOKOKE_MENU_POSITION_CURSOR);
 }
 
-static gboolean
-midori_panel_bookmarks_button_release_event_cb (GtkWidget*      widget,
-                                                GdkEventButton* event,
-                                                MidoriBrowser*  browser)
-{
-    GtkTreeModel* model;
-    GtkTreeIter iter;
-    KatzeItem* item;
-    const gchar* uri;
-    gint n;
-
-    if (event->button != 2 && event->button != 3)
-        return FALSE;
-
-    if (katze_tree_view_get_selected_iter (GTK_TREE_VIEW (widget), &model, &iter))
-    {
-        gtk_tree_model_get (model, &iter, 0, &item, -1);
-        uri = katze_item_get_uri (item);
-        if (event->button == 2)
-        {
-            if (uri && *uri)
-            {
-                n = midori_browser_add_uri (browser, uri);
-                midori_browser_set_current_page (browser, n);
-            }
-        }
-        else
-            midori_browser_bookmark_popup (widget, event, item, FALSE, browser);
-        return TRUE;
-    }
-    return FALSE;
-}
-
-static void
-midori_panel_bookmarks_popup_menu_cb (GtkWidget*     widget,
-                                      MidoriBrowser* browser)
-{
-    GtkTreeModel* model;
-    GtkTreeIter iter;
-    KatzeItem* item;
-
-    if (katze_tree_view_get_selected_iter (GTK_TREE_VIEW (widget), &model, &iter))
-    {
-        gtk_tree_model_get (model, &iter, 0, &item, -1);
-        midori_browser_bookmark_popup (widget, NULL, item, FALSE, browser);
-    }
-}
-
 static void
 midori_panel_history_row_activated_cb (GtkTreeView*       treeview,
                                        GtkTreePath*       path,
@@ -2794,54 +2681,6 @@ midori_panel_history_popup_menu_cb (GtkWidget*     widget,
     }
 }
 
-static void
-midori_browser_bookmarks_item_render_icon_cb (GtkTreeViewColumn* column,
-                                              GtkCellRenderer*   renderer,
-                                              GtkTreeModel*      model,
-                                              GtkTreeIter*       iter,
-                                              GtkWidget*         treeview)
-{
-    KatzeItem* item;
-    GdkPixbuf* pixbuf;
-
-    gtk_tree_model_get (model, iter, 0, &item, -1);
-
-    g_assert (KATZE_IS_ITEM (item));
-
-    /* TODO: Would it be better to not do this on every redraw? */
-    pixbuf = NULL;
-    if (KATZE_IS_ARRAY (item))
-        pixbuf = gtk_widget_render_icon (treeview, GTK_STOCK_DIRECTORY,
-                                         GTK_ICON_SIZE_MENU, NULL);
-    else if (katze_item_get_uri (item))
-        pixbuf = katze_net_load_icon (
-            MIDORI_BROWSER (gtk_widget_get_toplevel (treeview))->net,
-            katze_item_get_uri (item), NULL, treeview, NULL);
-    g_object_set (renderer, "pixbuf", pixbuf, NULL);
-    if (pixbuf)
-        g_object_unref (pixbuf);
-}
-
-static void
-midori_browser_bookmarks_item_render_text_cb (GtkTreeViewColumn* column,
-                                              GtkCellRenderer*   renderer,
-                                              GtkTreeModel*      model,
-                                              GtkTreeIter*       iter,
-                                              GtkWidget*         treeview)
-{
-    KatzeItem* item;
-
-    gtk_tree_model_get (model, iter, 0, &item, -1);
-
-    g_assert (KATZE_IS_ITEM (item));
-
-    if (KATZE_IS_ARRAY (item) || katze_item_get_uri (item))
-        g_object_set (renderer, "markup", NULL,
-                      "text", katze_item_get_name (item), NULL);
-    else
-        g_object_set (renderer, "markup", _("<i>Separator</i>"), NULL);
-}
-
 static void
 midori_browser_menu_bookmarks_item_activate_cb (GtkWidget*     widget,
                                                 MidoriBrowser* browser)
@@ -3255,43 +3094,6 @@ _action_trash_empty_activate (GtkAction*     action,
     _midori_browser_update_actions (browser);
 }
 
-static void
-_action_bookmark_edit_activate (GtkAction*     action,
-                                MidoriBrowser* browser)
-{
-    GtkTreeView* tree_view;
-    GtkTreeModel* model;
-    GtkTreeIter iter;
-    KatzeItem* item;
-
-    tree_view = GTK_TREE_VIEW (browser->panel_bookmarks);
-    if (katze_tree_view_get_selected_iter (tree_view, &model, &iter))
-    {
-        gtk_tree_model_get (model, &iter, 0, &item, -1);
-        if (KATZE_IS_ARRAY (item) || katze_item_get_uri (item))
-            midori_browser_edit_bookmark_dialog_new (browser, item, FALSE);
-    }
-}
-
-static void
-_action_bookmark_delete_activate (GtkAction*     action,
-                                  MidoriBrowser* browser)
-{
-    GtkTreeView* tree_view;
-    GtkTreeModel* model;
-    GtkTreeIter iter;
-    KatzeItem* item;
-    KatzeArray* parent;
-
-    tree_view = GTK_TREE_VIEW (browser->panel_bookmarks);
-    if (katze_tree_view_get_selected_iter (tree_view, &model, &iter))
-    {
-        gtk_tree_model_get (model, &iter, 0, &item, -1);
-        parent = katze_item_get_parent (item);
-        katze_array_remove_item (parent, item);
-    }
-}
-
 static const GtkActionEntry entries[] = {
  { "File", NULL, N_("_File") },
  { "WindowNew", STOCK_WINDOW_NEW,
@@ -3406,12 +3208,6 @@ static const GtkActionEntry entries[] = {
  { "BookmarkAdd", STOCK_BOOKMARK_ADD,
    NULL, "<Ctrl>d",
    N_("Add a new bookmark"), G_CALLBACK (_action_bookmark_add_activate) },
- { "BookmarkEdit", GTK_STOCK_EDIT,
-   NULL, "",
-   N_("Edit the selected bookmark"), G_CALLBACK (_action_bookmark_edit_activate) },
- { "BookmarkDelete", GTK_STOCK_DELETE,
-   NULL, "",
-   N_("Delete the selected bookmark"), G_CALLBACK (_action_bookmark_delete_activate) },
  { "HistoryDelete", GTK_STOCK_DELETE,
    NULL, "",
    N_("Delete the selected history item"), G_CALLBACK (_action_history_delete_activate) },
@@ -3644,11 +3440,6 @@ static const gchar* ui_markup =
   "</menubar>"
   "<toolbar name='toolbar_navigation'>"
   "</toolbar>"
-  "<toolbar name='toolbar_bookmarks'>"
-   "<toolitem action='BookmarkAdd'/>"
-   "<toolitem action='BookmarkEdit'/>"
-   "<toolitem action='BookmarkDelete'/>"
-  "</toolbar>"
   "<toolbar name='toolbar_history'>"
    "<toolitem action='HistoryAddBookmark'/>"
    "<toolitem action='HistoryDelete'/>"
@@ -4166,48 +3957,6 @@ midori_browser_init (MidoriBrowser* browser)
                       G_CALLBACK (midori_panel_close_cb), browser);
     gtk_paned_pack1 (GTK_PANED (hpaned), browser->panel, FALSE, FALSE);
 
-    /* Bookmarks */
-    box = gtk_vbox_new (FALSE, 0);
-    treestore = gtk_tree_store_new (1, KATZE_TYPE_ITEM);
-    treeview = gtk_tree_view_new_with_model (GTK_TREE_MODEL (treestore));
-    gtk_tree_view_set_headers_visible (GTK_TREE_VIEW (treeview), FALSE);
-    column = gtk_tree_view_column_new ();
-    renderer_pixbuf = gtk_cell_renderer_pixbuf_new ();
-    gtk_tree_view_column_pack_start (column, renderer_pixbuf, FALSE);
-    gtk_tree_view_column_set_cell_data_func (column, renderer_pixbuf,
-        (GtkTreeCellDataFunc)midori_browser_bookmarks_item_render_icon_cb,
-        treeview, NULL);
-    renderer_text = gtk_cell_renderer_text_new ();
-    gtk_tree_view_column_pack_start (column, renderer_text, FALSE);
-    gtk_tree_view_column_set_cell_data_func (column, renderer_text,
-        (GtkTreeCellDataFunc)midori_browser_bookmarks_item_render_text_cb,
-        treeview, NULL);
-    gtk_tree_view_append_column (GTK_TREE_VIEW (treeview), column);
-    g_object_unref (treestore);
-    g_object_connect (treeview,
-                      "signal::row-activated",
-                      midori_panel_bookmarks_row_activated_cb, browser,
-                      "signal::cursor-changed",
-                      midori_panel_bookmarks_cursor_or_row_changed_cb, browser,
-                      "signal::columns-changed",
-                      midori_panel_bookmarks_cursor_or_row_changed_cb, browser,
-                      "signal::button-release-event",
-                      midori_panel_bookmarks_button_release_event_cb, browser,
-                      "signal::popup-menu",
-                      midori_panel_bookmarks_popup_menu_cb, browser,
-                      NULL);
-    midori_panel_bookmarks_cursor_or_row_changed_cb (GTK_TREE_VIEW (treeview),
-                                                     browser);
-    gtk_box_pack_start (GTK_BOX (box), treeview, TRUE, TRUE, 0);
-    browser->panel_bookmarks = treeview;
-    gtk_widget_show_all (box);
-    toolbar = gtk_ui_manager_get_widget (ui_manager, "/toolbar_bookmarks");
-    _action_set_sensitive (browser, "BookmarkAdd", FALSE);
-    gtk_toolbar_set_icon_size (GTK_TOOLBAR (toolbar), GTK_ICON_SIZE_MENU);
-    gtk_widget_show_all (toolbar);
-    midori_panel_append_widget (MIDORI_PANEL (browser->panel),
-                              box, STOCK_BOOKMARKS, _("Bookmarks"), toolbar);
-
     /* History */
     box = gtk_vbox_new (FALSE, 0);
     treestore = gtk_tree_store_new (2, KATZE_TYPE_ITEM, G_TYPE_INT64);
@@ -4450,7 +4199,7 @@ midori_browser_toolbar_item_button_press_event_cb (GtkWidget*      toolitem,
     if (event->button == 3)
     {
         midori_browser_toolbar_popup_context_menu_cb (
-            gtk_bin_get_child (GTK_BIN (toolitem)) ?
+            GTK_IS_BIN (toolitem) && gtk_bin_get_child (GTK_BIN (toolitem)) ?
                 gtk_widget_get_parent (toolitem) : toolitem,
             event->x, event->y, event->button, browser);
 
@@ -4676,9 +4425,6 @@ browser_bookmarks_remove_item_cb (KatzeArray*    array,
     GList* children;
     GtkWidget* toolitem;
     KatzeItem* item;
-    GtkTreeModel* model;
-    GtkTreeIter iter;
-    gsize i = 0;
 
     children = gtk_container_get_children (GTK_CONTAINER (browser->bookmarkbar));
     while (children != NULL)
@@ -4689,52 +4435,6 @@ browser_bookmarks_remove_item_cb (KatzeArray*    array,
             gtk_widget_destroy (toolitem);
         children = g_list_next (children);
     }
-
-    model = gtk_tree_view_get_model (GTK_TREE_VIEW (browser->panel_bookmarks));
-    i = 0;
-    item = NULL;
-    while (item != removed_item &&
-        gtk_tree_model_iter_nth_child (model, &iter, NULL, i))
-    {
-        gtk_tree_model_get (model, &iter, 0, &item, -1);
-        if (item == removed_item)
-            midori_browser_model_remove_item (model, item, &iter);
-        i++;
-    }
-}
-
-static void
-bookmarks_model_insert_folder (GtkTreeStore* treestore,
-                               GtkTreeIter*  parent,
-                               KatzeArray*   array)
-{
-    guint n, i;
-    KatzeItem* item;
-    GtkTreeIter iter;
-
-    n = katze_array_get_length (array);
-    for (i = 0; i < n; i++)
-    {
-        item = katze_array_get_nth_item (array, i);
-        g_object_ref (item);
-        gtk_tree_store_insert_with_values (treestore, &iter, parent, n,
-                                           0, item, -1);
-        g_object_ref (item);
-        if (KATZE_IS_ARRAY (item))
-            bookmarks_model_insert_folder (treestore, &iter, KATZE_ARRAY (item));
-    }
-}
-
-static void
-midori_browser_bookmarks_clear_cb (KatzeArray*    bookmarks,
-                                   MidoriBrowser* browser)
-{
-    GtkTreeView* treeview;
-    GtkTreeStore* store;
-
-    treeview = GTK_TREE_VIEW (browser->panel_bookmarks);
-    store = GTK_TREE_STORE (gtk_tree_view_get_model (treeview));
-    gtk_tree_store_clear (store);
 }
 
 static void
@@ -4743,21 +4443,14 @@ midori_browser_set_bookmarks (MidoriBrowser* browser,
 {
     guint i, n;
     KatzeItem* item;
-    GtkTreeModel* treestore;
 
     if (browser->bookmarks == bookmarks)
         return;
 
-    if (browser->bookmarks)
-        g_signal_handlers_disconnect_by_func (browser->bookmarks,
-                                              midori_browser_bookmarks_clear_cb,
-                                              browser);
     if (bookmarks)
         g_object_ref (bookmarks);
     katze_object_assign (browser->bookmarks, bookmarks);
 
-    midori_browser_bookmarks_clear_cb (browser->bookmarks, browser);
-
     g_object_set (_action_by_name (browser, "Bookmarks"), "array",
                   browser->bookmarks, NULL);
 
@@ -4776,14 +4469,6 @@ midori_browser_set_bookmarks (MidoriBrowser* browser,
         G_CALLBACK (browser_bookmarks_add_item_cb), browser);
     g_signal_connect (browser->bookmarks, "remove-item",
         G_CALLBACK (browser_bookmarks_remove_item_cb), browser);
-    g_signal_connect (browser->bookmarks, "clear",
-                      G_CALLBACK (midori_browser_bookmarks_clear_cb), browser);
-
-    treestore = gtk_tree_view_get_model (GTK_TREE_VIEW (browser->panel_bookmarks));
-    bookmarks_model_insert_folder (GTK_TREE_STORE (treestore),
-                                   NULL, browser->bookmarks);
-    midori_panel_bookmarks_cursor_or_row_changed_cb (
-        GTK_TREE_VIEW (browser->panel_bookmarks), browser);
 
     _action_set_sensitive (browser, "BookmarkAdd", TRUE);
 }
diff --git a/panels/midori-bookmarks.c b/panels/midori-bookmarks.c
new file mode 100644 (file)
index 0000000..1f09e2f
--- /dev/null
@@ -0,0 +1,816 @@
+/*
+ Copyright (C) 2008-2009 Christian Dywan <christian@twotoasts.de>
+
+ This library 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.
+
+ See the file COPYING for the full license text.
+*/
+
+#include "midori-bookmarks.h"
+
+#include "midori-app.h"
+#include "midori-browser.h"
+#include "midori-stock.h"
+#include "midori-view.h"
+#include "midori-viewable.h"
+
+#include "sokoke.h"
+
+#include <glib/gi18n.h>
+
+#include <katze/katze.h>
+
+void
+midori_browser_edit_bookmark_dialog_new (MidoriBrowser* browser,
+                                         KatzeItem*     bookmark,
+                                         gboolean       new_bookmark);
+
+struct _MidoriBookmarks
+{
+    GtkVBox parent_instance;
+
+    GtkWidget* toolbar;
+    GtkWidget* edit;
+    GtkWidget* delete;
+    GtkWidget* treeview;
+    MidoriApp* app;
+    KatzeArray* array;
+    KatzeNet* net;
+};
+
+struct _MidoriBookmarksClass
+{
+    GtkVBoxClass parent_class;
+};
+
+static void
+midori_bookmarks_viewable_iface_init (MidoriViewableIface* iface);
+
+G_DEFINE_TYPE_WITH_CODE (MidoriBookmarks, midori_bookmarks, GTK_TYPE_VBOX,
+                         G_IMPLEMENT_INTERFACE (MIDORI_TYPE_VIEWABLE,
+                             midori_bookmarks_viewable_iface_init));
+
+enum
+{
+    PROP_0,
+
+    PROP_APP
+};
+
+static void
+midori_bookmarks_set_property (GObject*      object,
+                               guint         prop_id,
+                               const GValue* value,
+                               GParamSpec*   pspec);
+
+static void
+midori_bookmarks_get_property (GObject*    object,
+                               guint       prop_id,
+                               GValue*     value,
+                               GParamSpec* pspec);
+
+static void
+midori_bookmarks_class_init (MidoriBookmarksClass* class)
+{
+    GObjectClass* gobject_class;
+    GParamFlags flags;
+
+    gobject_class = G_OBJECT_CLASS (class);
+    gobject_class->set_property = midori_bookmarks_set_property;
+    gobject_class->get_property = midori_bookmarks_get_property;
+
+    flags = G_PARAM_READWRITE | G_PARAM_CONSTRUCT;
+
+    g_object_class_install_property (gobject_class,
+                                     PROP_APP,
+                                     g_param_spec_object (
+                                     "app",
+                                     "App",
+                                     "The app",
+                                     MIDORI_TYPE_APP,
+                                     flags));
+}
+
+static const gchar*
+midori_bookmarks_get_label (MidoriViewable* viewable)
+{
+    return _("Bookmarks");
+}
+
+static const gchar*
+midori_bookmarks_get_stock_id (MidoriViewable* viewable)
+{
+    return STOCK_BOOKMARKS;
+}
+
+static void
+midori_bookmarks_add_clicked_cb (GtkWidget* toolitem)
+{
+    GtkWidget* browser = gtk_widget_get_toplevel (toolitem);
+    /* FIXME: Take selected folder into account */
+    midori_browser_edit_bookmark_dialog_new (MIDORI_BROWSER (browser),
+                                             NULL, TRUE);
+}
+
+static void
+midori_bookmarks_edit_clicked_cb (GtkWidget*       toolitem,
+                                  MidoriBookmarks* bookmarks)
+{
+    GtkTreeModel* model;
+    GtkTreeIter iter;
+
+    if (katze_tree_view_get_selected_iter (GTK_TREE_VIEW (bookmarks->treeview),
+                                           &model, &iter))
+    {
+        KatzeItem* item;
+        gboolean is_separator;
+
+        gtk_tree_model_get (model, &iter, 0, &item, -1);
+
+        is_separator = !KATZE_IS_ARRAY (item) && !katze_item_get_uri (item);
+        if (!is_separator)
+        {
+            GtkWidget* browser = gtk_widget_get_toplevel (toolitem);
+            midori_browser_edit_bookmark_dialog_new (MIDORI_BROWSER (browser),
+                                                     item, FALSE);
+        }
+    }
+}
+
+static void
+midori_bookmarks_delete_clicked_cb (GtkWidget*       toolitem,
+                                    MidoriBookmarks* bookmarks)
+{
+    GtkTreeModel* model;
+    GtkTreeIter iter;
+
+    if (katze_tree_view_get_selected_iter (GTK_TREE_VIEW (bookmarks->treeview),
+                                           &model, &iter))
+    {
+        KatzeItem* item;
+        KatzeArray* parent;
+
+        gtk_tree_model_get (model, &iter, 0, &item, -1);
+
+        /* FIXME: Even toplevel items should technically have a parent */
+        g_return_if_fail (katze_item_get_parent (item));
+
+        parent = katze_item_get_parent (item);
+        katze_array_remove_item (parent, item);
+    }
+}
+
+static void
+midori_bookmarks_cursor_or_row_changed_cb (GtkTreeView*     treeview,
+                                           MidoriBookmarks* bookmarks)
+{
+    GtkTreeModel* model;
+    GtkTreeIter iter;
+    KatzeItem* item;
+    gboolean is_separator;
+
+    if (!bookmarks->edit)
+        return;
+
+    if (katze_tree_view_get_selected_iter (treeview, &model, &iter))
+    {
+        gtk_tree_model_get (model, &iter, 0, &item, -1);
+
+        is_separator = !KATZE_IS_ARRAY (item) && !katze_item_get_uri (item);
+        gtk_widget_set_sensitive (bookmarks->edit, !is_separator);
+        gtk_widget_set_sensitive (bookmarks->delete, TRUE);
+    }
+    else
+    {
+        gtk_widget_set_sensitive (bookmarks->edit, FALSE);
+        gtk_widget_set_sensitive (bookmarks->delete, FALSE);
+    }
+}
+
+static GtkWidget*
+midori_bookmarks_get_toolbar (MidoriViewable* viewable)
+{
+    MidoriBookmarks* bookmarks = MIDORI_BOOKMARKS (viewable);
+
+    if (!bookmarks->toolbar)
+    {
+        GtkWidget* toolbar;
+        GtkToolItem* toolitem;
+
+        toolbar = gtk_toolbar_new ();
+        gtk_toolbar_set_style (GTK_TOOLBAR (toolbar), GTK_TOOLBAR_BOTH_HORIZ);
+        gtk_toolbar_set_icon_size (GTK_TOOLBAR (toolbar), GTK_ICON_SIZE_BUTTON);
+        bookmarks->toolbar = toolbar;
+        toolitem = gtk_tool_button_new_from_stock (GTK_STOCK_ADD);
+        gtk_widget_set_tooltip_text (GTK_WIDGET (toolitem),
+                                     _("Add a new bookmark"));
+        gtk_tool_item_set_is_important (toolitem, TRUE);
+        g_signal_connect (toolitem, "clicked",
+            G_CALLBACK (midori_bookmarks_add_clicked_cb), bookmarks);
+        gtk_toolbar_insert (GTK_TOOLBAR (toolbar), toolitem, -1);
+        gtk_widget_show (GTK_WIDGET (toolitem));
+        toolitem = gtk_tool_button_new_from_stock (GTK_STOCK_EDIT);
+        gtk_widget_set_tooltip_text (GTK_WIDGET (toolitem),
+                                     _("Edit the selected bookmark"));
+        g_signal_connect (toolitem, "clicked",
+            G_CALLBACK (midori_bookmarks_edit_clicked_cb), bookmarks);
+        gtk_toolbar_insert (GTK_TOOLBAR (toolbar), toolitem, -1);
+        gtk_widget_show (GTK_WIDGET (toolitem));
+        bookmarks->edit = GTK_WIDGET (toolitem);
+        toolitem = gtk_tool_button_new_from_stock (GTK_STOCK_DELETE);
+        gtk_widget_set_tooltip_text (GTK_WIDGET (toolitem),
+                                     _("Delete the selected bookmark"));
+        g_signal_connect (toolitem, "clicked",
+            G_CALLBACK (midori_bookmarks_delete_clicked_cb), bookmarks);
+        gtk_toolbar_insert (GTK_TOOLBAR (toolbar), toolitem, -1);
+        gtk_widget_show (GTK_WIDGET (toolitem));
+        bookmarks->delete = GTK_WIDGET (toolitem);
+        midori_bookmarks_cursor_or_row_changed_cb (
+            GTK_TREE_VIEW (bookmarks->treeview), bookmarks);
+        g_signal_connect (bookmarks->edit, "destroy",
+            G_CALLBACK (gtk_widget_destroyed), &bookmarks->edit);
+        g_signal_connect (bookmarks->delete, "destroy",
+            G_CALLBACK (gtk_widget_destroyed), &bookmarks->delete);
+    }
+
+    return bookmarks->toolbar;
+}
+
+static void
+midori_bookmarks_viewable_iface_init (MidoriViewableIface* iface)
+{
+    iface->get_stock_id = midori_bookmarks_get_stock_id;
+    iface->get_label = midori_bookmarks_get_label;
+    iface->get_toolbar = midori_bookmarks_get_toolbar;
+}
+
+static void
+midori_bookmarks_add_item_cb (KatzeArray*      array,
+                              KatzeItem*       added_item,
+                              MidoriBookmarks* bookmarks);
+
+static void
+midori_bookmarks_remove_item_cb (KatzeArray*      array,
+                                 KatzeItem*       removed_item,
+                                 MidoriBookmarks* bookmarks);
+
+static void
+midori_bookmarks_clear_cb (KatzeArray*      array,
+                           MidoriBookmarks* bookmarks);
+
+static void
+midori_bookmarks_disconnect_folder (MidoriBookmarks* bookmarks,
+                                    KatzeArray*      array)
+{
+    guint i, n;
+
+    g_assert (KATZE_IS_ARRAY (array));
+
+    g_signal_handlers_disconnect_by_func (array,
+        midori_bookmarks_add_item_cb, bookmarks);
+    g_signal_handlers_disconnect_by_func (array,
+        midori_bookmarks_remove_item_cb, bookmarks);
+    g_signal_handlers_disconnect_by_func (array,
+        midori_bookmarks_clear_cb, bookmarks);
+
+    n = katze_array_get_length (array);
+    for (i = 0; i < n; i++)
+    {
+        KatzeItem* item = katze_array_get_nth_item (bookmarks->array, i);
+        if (KATZE_IS_ARRAY (item))
+            midori_bookmarks_disconnect_folder (bookmarks, KATZE_ARRAY (item));
+        g_object_unref (item);
+    }
+}
+
+static void
+midori_bookmarks_add_item_cb (KatzeArray*      array,
+                              KatzeItem*       added_item,
+                              MidoriBookmarks* bookmarks)
+{
+    GtkTreeModel* model;
+    GtkTreeIter iter;
+    guint i;
+
+    g_assert (KATZE_IS_ARRAY (array));
+    g_assert (KATZE_IS_ITEM (added_item));
+
+    if (KATZE_IS_ARRAY (added_item))
+    {
+        g_signal_connect (added_item, "add-item",
+            G_CALLBACK (midori_bookmarks_add_item_cb), bookmarks);
+        g_signal_connect (added_item, "remove-item",
+            G_CALLBACK (midori_bookmarks_remove_item_cb), bookmarks);
+        g_signal_connect (added_item, "clear",
+            G_CALLBACK (midori_bookmarks_clear_cb), bookmarks);
+    }
+
+    g_object_ref (added_item);
+    model = gtk_tree_view_get_model (GTK_TREE_VIEW (bookmarks->treeview));
+
+    if (array == bookmarks->array)
+    {
+        gtk_tree_store_insert_with_values (GTK_TREE_STORE (model),
+            &iter, NULL, G_MAXINT, 0, added_item, -1);
+        return;
+    }
+
+    i = 0;
+    /* FIXME: Recurse over children of folders, too */
+    while (gtk_tree_model_iter_nth_child (model, &iter, NULL, i))
+    {
+        KatzeItem* item;
+
+        gtk_tree_model_get (model, &iter, 0, &item, -1);
+        if (item == (KatzeItem*)array)
+        {
+            GtkTreeIter child_iter;
+
+            gtk_tree_store_insert_with_values (GTK_TREE_STORE (model),
+                &child_iter, &iter, G_MAXINT, 0, added_item, -1);
+            break;
+        }
+        i++;
+    }
+}
+
+static void
+midori_bookmarks_remove_item_cb (KatzeArray*      array,
+                                 KatzeItem*       removed_item,
+                                 MidoriBookmarks* bookmarks)
+{
+    GtkTreeModel* model;
+    guint i;
+    GtkTreeIter iter;
+
+    g_assert (KATZE_IS_ARRAY (array));
+    g_assert (KATZE_IS_ITEM (removed_item));
+
+    if (KATZE_IS_ARRAY (removed_item))
+        midori_bookmarks_disconnect_folder (bookmarks, KATZE_ARRAY (removed_item));
+
+    model = gtk_tree_view_get_model (GTK_TREE_VIEW (bookmarks->treeview));
+    i = 0;
+    /* FIXME: Recurse over children of folders, too */
+    while (gtk_tree_model_iter_nth_child (model, &iter, NULL, i))
+    {
+        KatzeItem* item;
+
+        gtk_tree_model_get (model, &iter, 0, &item, -1);
+        if (item == removed_item)
+        {
+            gtk_tree_store_remove (GTK_TREE_STORE (model), &iter);
+            g_object_unref (item);
+            break;
+        }
+        i++;
+    }
+    g_object_unref (removed_item);
+}
+
+static void
+midori_bookmarks_clear_cb (KatzeArray*      array,
+                           MidoriBookmarks* bookmarks)
+{
+    GtkTreeView* treeview;
+    GtkTreeStore* store;
+
+    g_assert (KATZE_IS_ARRAY (array));
+
+    /* FIXME: Clear folders */
+    if (array == bookmarks->array)
+    {
+        treeview = GTK_TREE_VIEW (bookmarks->treeview);
+        store = GTK_TREE_STORE (gtk_tree_view_get_model (treeview));
+        gtk_tree_store_clear (store);
+    }
+
+    midori_bookmarks_disconnect_folder (bookmarks, array);
+}
+
+static void
+midori_bookmarks_insert_folder (MidoriBookmarks* bookmarks,
+                                GtkTreeStore*    treestore,
+                                GtkTreeIter*     parent,
+                                KatzeArray*      array)
+{
+    guint n, i;
+    KatzeItem* item;
+    GtkTreeIter iter;
+
+    g_assert (KATZE_IS_ARRAY (array));
+
+    g_signal_connect (array, "add-item",
+        G_CALLBACK (midori_bookmarks_add_item_cb), bookmarks);
+    g_signal_connect (array, "remove-item",
+        G_CALLBACK (midori_bookmarks_remove_item_cb), bookmarks);
+    g_signal_connect (array, "clear",
+            G_CALLBACK (midori_bookmarks_clear_cb), bookmarks);
+
+    n = katze_array_get_length (array);
+    for (i = 0; i < n; i++)
+    {
+        item = katze_array_get_nth_item (array, i);
+        g_object_ref (item);
+        gtk_tree_store_insert_with_values (treestore, &iter, parent, n,
+                                           0, item, -1);
+        if (KATZE_IS_ARRAY (item))
+            midori_bookmarks_insert_folder (bookmarks, treestore,
+                                            &iter, KATZE_ARRAY (item));
+    }
+}
+
+static void
+midori_bookmarks_set_app (MidoriBookmarks* bookmarks,
+                          MidoriApp*       app)
+{
+    GtkTreeModel* model;
+
+    if (bookmarks->array)
+    {
+        midori_bookmarks_disconnect_folder (bookmarks, bookmarks->array);
+        g_object_unref (bookmarks->array);
+        model = gtk_tree_view_get_model (GTK_TREE_VIEW (bookmarks->treeview));
+        gtk_tree_store_clear (GTK_TREE_STORE (model));
+    }
+    katze_assign (bookmarks->app, g_object_ref (app));
+    bookmarks->array = katze_object_get_object (app, "bookmarks");
+    if (bookmarks->array)
+    {
+        /* FIXME: Dereference the app on finalization */
+        model = gtk_tree_view_get_model (GTK_TREE_VIEW (bookmarks->treeview));
+        midori_bookmarks_insert_folder (bookmarks, GTK_TREE_STORE (model),
+                                        NULL, g_object_ref (bookmarks->array));
+    }
+}
+
+static void
+midori_bookmarks_set_property (GObject*      object,
+                               guint         prop_id,
+                               const GValue* value,
+                               GParamSpec*   pspec)
+{
+    MidoriBookmarks* bookmarks = MIDORI_BOOKMARKS (object);
+
+    switch (prop_id)
+    {
+    case PROP_APP:
+        midori_bookmarks_set_app (bookmarks, g_value_get_object (value));
+        break;
+    default:
+        G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+        break;
+    }
+}
+
+static void
+midori_bookmarks_get_property (GObject*    object,
+                               guint       prop_id,
+                               GValue*     value,
+                               GParamSpec* pspec)
+{
+    MidoriBookmarks* bookmarks = MIDORI_BOOKMARKS (object);
+
+    switch (prop_id)
+    {
+    case PROP_APP:
+        g_value_set_object (value, bookmarks->app);
+        break;
+    default:
+        G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+        break;
+    }
+}
+
+static void
+midori_bookmarks_treeview_render_icon_cb (GtkTreeViewColumn* column,
+                                          GtkCellRenderer*   renderer,
+                                          GtkTreeModel*      model,
+                                          GtkTreeIter*       iter,
+                                          GtkWidget*         treeview)
+{
+    KatzeItem* item;
+    GdkPixbuf* pixbuf;
+
+    gtk_tree_model_get (model, iter, 0, &item, -1);
+
+    /* TODO: Would it be better to not do this on every redraw? */
+    pixbuf = NULL;
+    if (KATZE_IS_ARRAY (item))
+        pixbuf = gtk_widget_render_icon (treeview, GTK_STOCK_DIRECTORY,
+                                         GTK_ICON_SIZE_MENU, NULL);
+    else if (katze_item_get_uri (item))
+        pixbuf = katze_net_load_icon (
+            MIDORI_BOOKMARKS (gtk_widget_get_parent (treeview))->net,
+            katze_item_get_uri (item), NULL, treeview, NULL);
+    g_object_set (renderer, "pixbuf", pixbuf, NULL);
+    if (pixbuf)
+        g_object_unref (pixbuf);
+}
+
+static void
+midori_bookmarks_treeview_render_text_cb (GtkTreeViewColumn* column,
+                                          GtkCellRenderer*   renderer,
+                                          GtkTreeModel*      model,
+                                          GtkTreeIter*       iter,
+                                          GtkWidget*         treeview)
+{
+    KatzeItem* item;
+
+    gtk_tree_model_get (model, iter, 0, &item, -1);
+
+    if (KATZE_IS_ARRAY (item) || katze_item_get_uri (item))
+        g_object_set (renderer, "markup", NULL,
+                      "text", katze_item_get_name (item), NULL);
+    else
+        g_object_set (renderer, "markup", _("<i>Separator</i>"), NULL);
+}
+
+static void
+midori_bookmarks_row_activated_cb (GtkTreeView*       treeview,
+                                   GtkTreePath*       path,
+                                   GtkTreeViewColumn* column,
+                                   MidoriBookmarks*   bookmarks)
+{
+    GtkTreeModel* model;
+    GtkTreeIter iter;
+    KatzeItem* item;
+    const gchar* uri;
+
+    model = gtk_tree_view_get_model (treeview);
+
+    if (gtk_tree_model_get_iter (model, &iter, path))
+    {
+        gtk_tree_model_get (model, &iter, 0, &item, -1);
+        uri = katze_item_get_uri (item);
+        if (uri && *uri)
+        {
+            GtkWidget* browser;
+
+            browser = gtk_widget_get_toplevel (GTK_WIDGET (bookmarks));
+            midori_browser_set_current_uri (MIDORI_BROWSER (browser), uri);
+        }
+    }
+}
+
+static void
+midori_bookmarks_popup_item (GtkWidget*       menu,
+                             const gchar*     stock_id,
+                             const gchar*     label,
+                             KatzeItem*       item,
+                             gpointer         callback,
+                             MidoriBookmarks* bookmarks)
+{
+    const gchar* uri;
+    GtkWidget* menuitem;
+
+    uri = katze_item_get_uri (item);
+
+    menuitem = gtk_image_menu_item_new_from_stock (stock_id, NULL);
+    if (label)
+        gtk_label_set_text_with_mnemonic (GTK_LABEL (gtk_bin_get_child (
+        GTK_BIN (menuitem))), label);
+    if (!strcmp (stock_id, GTK_STOCK_EDIT))
+        gtk_widget_set_sensitive (menuitem,
+            KATZE_IS_ARRAY (item) || uri != NULL);
+    else if (strcmp (stock_id, GTK_STOCK_DELETE))
+        gtk_widget_set_sensitive (menuitem, uri != NULL);
+    g_object_set_data (G_OBJECT (menuitem), "KatzeItem", item);
+    g_signal_connect (menuitem, "activate", G_CALLBACK (callback), bookmarks);
+    gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem);
+    gtk_widget_show (menuitem);
+}
+
+static void
+midori_bookmarks_open_activate_cb (GtkWidget*       menuitem,
+                                   MidoriBookmarks* bookmarks)
+{
+    KatzeItem* item;
+    const gchar* uri;
+
+    item = (KatzeItem*)g_object_get_data (G_OBJECT (menuitem), "KatzeItem");
+    uri = katze_item_get_uri (item);
+
+    if (uri && *uri)
+    {
+        GtkWidget* browser = gtk_widget_get_toplevel (GTK_WIDGET (bookmarks));
+        midori_browser_set_current_uri (MIDORI_BROWSER (browser), uri);
+    }
+}
+
+static void
+midori_bookmarks_open_in_tab_activate_cb (GtkWidget*       menuitem,
+                                          MidoriBookmarks* bookmarks)
+{
+    KatzeItem* item;
+    const gchar* uri;
+    guint n;
+
+    item = (KatzeItem*)g_object_get_data (G_OBJECT (menuitem), "KatzeItem");
+    uri = katze_item_get_uri (item);
+
+    if (uri && *uri)
+    {
+        GtkWidget* browser;
+        MidoriWebSettings* settings;
+
+        browser = gtk_widget_get_toplevel (GTK_WIDGET (bookmarks));
+        n = midori_browser_add_item (MIDORI_BROWSER (browser), item);
+        settings = katze_object_get_object (browser, "settings");
+        if (!katze_object_get_boolean (settings, "open-tabs-in-the-background"))
+            midori_browser_set_current_page (MIDORI_BROWSER (browser), n);
+    }
+}
+
+static void
+midori_bookmarks_open_in_window_activate_cb (GtkWidget*       menuitem,
+                                             MidoriBookmarks* bookmarks)
+{
+    KatzeItem* item;
+    const gchar* uri;
+
+    item = (KatzeItem*)g_object_get_data (G_OBJECT (menuitem), "KatzeItem");
+    uri = katze_item_get_uri (item);
+
+    if (uri && *uri)
+    {
+        GtkWidget* browser = gtk_widget_get_toplevel (GTK_WIDGET (bookmarks));
+        g_signal_emit_by_name (browser, "new-window", uri);
+    }
+}
+
+static void
+midori_bookmarks_edit_activate_cb (GtkWidget*       menuitem,
+                                   MidoriBookmarks* bookmarks)
+{
+    KatzeItem* item;
+    gboolean is_separator;
+
+    item = (KatzeItem*)g_object_get_data (G_OBJECT (menuitem), "KatzeItem");
+    is_separator = !KATZE_IS_ARRAY (item) && !katze_item_get_uri (item);
+
+    if (!is_separator)
+    {
+        GtkWidget* browser = gtk_widget_get_toplevel (GTK_WIDGET (bookmarks));
+        midori_browser_edit_bookmark_dialog_new (MIDORI_BROWSER (browser), item, FALSE);
+    }
+}
+
+static void
+midori_bookmarks_delete_activate_cb (GtkWidget*       menuitem,
+                                     MidoriBookmarks* bookmarks)
+{
+    KatzeItem* item;
+    KatzeArray* parent;
+
+    item = (KatzeItem*)g_object_get_data (G_OBJECT (menuitem), "KatzeItem");
+
+    /* FIXME: Even toplevel items should technically have a parent */
+    g_return_if_fail (katze_item_get_parent (item));
+
+    parent = katze_item_get_parent (item);
+    katze_array_remove_item (parent, item);
+}
+
+static void
+midori_bookmarks_popup (GtkWidget*       widget,
+                        GdkEventButton*  event,
+                        KatzeItem*       item,
+                        MidoriBookmarks* bookmarks)
+{
+    GtkWidget* menu;
+    GtkWidget* menuitem;
+
+    menu = gtk_menu_new ();
+    midori_bookmarks_popup_item (menu, GTK_STOCK_OPEN, NULL,
+        item, midori_bookmarks_open_activate_cb, bookmarks);
+    midori_bookmarks_popup_item (menu, STOCK_TAB_NEW, _("Open in New _Tab"),
+        item, midori_bookmarks_open_in_tab_activate_cb, bookmarks);
+    midori_bookmarks_popup_item (menu, STOCK_WINDOW_NEW, _("Open in New _Window"),
+        item, midori_bookmarks_open_in_window_activate_cb, bookmarks);
+    menuitem = gtk_separator_menu_item_new ();
+    gtk_menu_shell_append (GTK_MENU_SHELL (menu), menuitem);
+    gtk_widget_show (menuitem);
+    midori_bookmarks_popup_item (menu, GTK_STOCK_EDIT, NULL,
+        item, midori_bookmarks_edit_activate_cb, bookmarks);
+    midori_bookmarks_popup_item (menu, GTK_STOCK_DELETE, NULL,
+        item, midori_bookmarks_delete_activate_cb, bookmarks);
+
+    sokoke_widget_popup (widget, GTK_MENU (menu),
+                         event, SOKOKE_MENU_POSITION_CURSOR);
+}
+
+static gboolean
+midori_bookmarks_button_release_event_cb (GtkWidget*       widget,
+                                          GdkEventButton*  event,
+                                          MidoriBookmarks* bookmarks)
+{
+    GtkTreeModel* model;
+    GtkTreeIter iter;
+    KatzeItem* item;
+    const gchar* uri;
+    gint n;
+
+    if (event->button != 2 && event->button != 3)
+        return FALSE;
+
+    if (katze_tree_view_get_selected_iter (GTK_TREE_VIEW (widget), &model, &iter))
+    {
+        gtk_tree_model_get (model, &iter, 0, &item, -1);
+        uri = katze_item_get_uri (item);
+        if (event->button == 2)
+        {
+            if (uri && *uri)
+            {
+                GtkWidget* browser = gtk_widget_get_toplevel (widget);
+                n = midori_browser_add_uri (MIDORI_BROWSER (browser), uri);
+                midori_browser_set_current_page (MIDORI_BROWSER (browser), n);
+            }
+        }
+        else
+            midori_bookmarks_popup (widget, event, item, bookmarks);
+        return TRUE;
+    }
+    return FALSE;
+}
+
+static void
+midori_bookmarks_popup_menu_cb (GtkWidget*       widget,
+                                MidoriBookmarks* bookmarks)
+{
+    GtkTreeModel* model;
+    GtkTreeIter iter;
+    KatzeItem* item;
+
+    if (katze_tree_view_get_selected_iter (GTK_TREE_VIEW (widget), &model, &iter))
+    {
+        gtk_tree_model_get (model, &iter, 0, &item, -1);
+        midori_bookmarks_popup (widget, NULL, item, bookmarks);
+    }
+}
+
+static void
+midori_bookmarks_init (MidoriBookmarks* bookmarks)
+{
+    GtkTreeStore* model;
+    GtkWidget* treeview;
+    GtkTreeViewColumn* column;
+    GtkCellRenderer* renderer_pixbuf;
+    GtkCellRenderer* renderer_text;
+
+    bookmarks->net = katze_net_new ();
+    /* FIXME: Dereference the net on finalization */
+
+    /* Create the treeview */
+    model = gtk_tree_store_new (1, KATZE_TYPE_ITEM);
+    treeview = gtk_tree_view_new_with_model (GTK_TREE_MODEL (model));
+    gtk_tree_view_set_headers_visible (GTK_TREE_VIEW (treeview), FALSE);
+    column = gtk_tree_view_column_new ();
+    renderer_pixbuf = gtk_cell_renderer_pixbuf_new ();
+    gtk_tree_view_column_pack_start (column, renderer_pixbuf, FALSE);
+    gtk_tree_view_column_set_cell_data_func (column, renderer_pixbuf,
+        (GtkTreeCellDataFunc)midori_bookmarks_treeview_render_icon_cb,
+        treeview, NULL);
+    renderer_text = gtk_cell_renderer_text_new ();
+    gtk_tree_view_column_pack_start (column, renderer_text, FALSE);
+    gtk_tree_view_column_set_cell_data_func (column, renderer_text,
+        (GtkTreeCellDataFunc)midori_bookmarks_treeview_render_text_cb,
+        treeview, NULL);
+    gtk_tree_view_append_column (GTK_TREE_VIEW (treeview), column);
+    g_object_unref (model);
+    g_object_connect (treeview,
+                      "signal::row-activated",
+                      midori_bookmarks_row_activated_cb, bookmarks,
+                      "signal::cursor-changed",
+                      midori_bookmarks_cursor_or_row_changed_cb, bookmarks,
+                      "signal::columns-changed",
+                      midori_bookmarks_cursor_or_row_changed_cb, bookmarks,
+                      "signal::button-release-event",
+                      midori_bookmarks_button_release_event_cb, bookmarks,
+                      "signal::popup-menu",
+                      midori_bookmarks_popup_menu_cb, bookmarks,
+                      NULL);
+    gtk_widget_show (treeview);
+    gtk_box_pack_start (GTK_BOX (bookmarks), treeview, TRUE, TRUE, 0);
+    bookmarks->treeview = treeview;
+}
+
+/**
+ * midori_bookmarks_new:
+ *
+ * Creates a new empty bookmarks.
+ *
+ * Return value: a new #MidoriBookmarks
+ *
+ * Since: 0.1.3
+ **/
+GtkWidget*
+midori_bookmarks_new (void)
+{
+    MidoriBookmarks* bookmarks = g_object_new (MIDORI_TYPE_BOOKMARKS, NULL);
+
+    return GTK_WIDGET (bookmarks);
+}
diff --git a/panels/midori-bookmarks.h b/panels/midori-bookmarks.h
new file mode 100644 (file)
index 0000000..9679718
--- /dev/null
@@ -0,0 +1,43 @@
+/*
+ Copyright (C) 2009 Christian Dywan <christian@twotoasts.de>
+
+ This library 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.
+
+ See the file COPYING for the full license text.
+*/
+
+#ifndef __MIDORI_BOOKMARKS_H__
+#define __MIDORI_BOOKMARKS_H__
+
+#include <gtk/gtk.h>
+
+G_BEGIN_DECLS
+
+#define MIDORI_TYPE_BOOKMARKS \
+    (midori_bookmarks_get_type ())
+#define MIDORI_BOOKMARKS(obj) \
+    (G_TYPE_CHECK_INSTANCE_CAST ((obj), MIDORI_TYPE_BOOKMARKS, MidoriBookmarks))
+#define MIDORI_BOOKMARKS_CLASS(klass) \
+    (G_TYPE_CHECK_CLASS_CAST ((klass), MIDORI_TYPE_BOOKMARKS, MidoriBookmarksClass))
+#define MIDORI_IS_BOOKMARKS(obj) \
+    (G_TYPE_CHECK_INSTANCE_TYPE ((obj), MIDORI_TYPE_BOOKMARKS))
+#define MIDORI_IS_BOOKMARKS_CLASS(klass) \
+    (G_TYPE_CHECK_CLASS_TYPE ((klass), MIDORI_TYPE_BOOKMARKS))
+#define MIDORI_BOOKMARKS_GET_CLASS(obj) \
+    (G_TYPE_INSTANCE_GET_CLASS ((obj), MIDORI_TYPE_BOOKMARKS, MidoriBookmarksClass))
+
+typedef struct _MidoriBookmarks                MidoriBookmarks;
+typedef struct _MidoriBookmarksClass           MidoriBookmarksClass;
+
+GType
+midori_bookmarks_get_type               (void);
+
+GtkWidget*
+midori_bookmarks_new                    (void);
+
+G_END_DECLS
+
+#endif /* __MIDORI_BOOKMARKS_H__ */
index ccba854cc130131fd29b9ee8d34b9ea501e3a508..c7aed162e40b79bf64de2703dcb35a052b80388c 100644 (file)
@@ -13,6 +13,7 @@ midori/midori-searchaction.c
 midori/sokoke.c
 midori/gjs.c
 panels/midori-addons.c
+panels/midori-bookmarks.c
 panels/midori-console.c
 panels/midori-extensions.c
 panels/midori-plugins.c